Project

General

Profile

Download (9.58 KB) Statistics
| Branch: | Revision:

library / src / main / java / org / distorted / library / effect / VertexEffectDeform.java @ 041b6dee

1
///////////////////////////////////////////////////////////////////////////////////////////////////
2
// Copyright 2017 Leszek Koltunski                                                               //
3
//                                                                                               //
4
// This file is part of Distorted.                                                               //
5
//                                                                                               //
6
// Distorted is free software: you can redistribute it and/or modify                             //
7
// it under the terms of the GNU General Public License as published by                          //
8
// the Free Software Foundation, either version 2 of the License, or                             //
9
// (at your option) any later version.                                                           //
10
//                                                                                               //
11
// Distorted is distributed in the hope that it will be useful,                                  //
12
// but WITHOUT ANY WARRANTY; without even the implied warranty of                                //
13
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the                                 //
14
// GNU General Public License for more details.                                                  //
15
//                                                                                               //
16
// You should have received a copy of the GNU General Public License                             //
17
// along with Distorted.  If not, see <http://www.gnu.org/licenses/>.                            //
18
///////////////////////////////////////////////////////////////////////////////////////////////////
19

    
20
package org.distorted.library.effect;
21

    
22
import org.distorted.library.type.Data3D;
23
import org.distorted.library.type.Data4D;
24
import org.distorted.library.type.Static4D;
25

    
26
///////////////////////////////////////////////////////////////////////////////////////////////////
27

    
28
public class VertexEffectDeform extends VertexEffect
29
  {
30
  private Data3D mVector, mCenter;
31
  private Data4D mRegion;
32

    
33
///////////////////////////////////////////////////////////////////////////////////////////////////
34
/**
35
 * Deform the shape of the whole Object with a (possibly changing in time) vector of force applied to
36
 * a (possibly changing in time) point on the Object.
37
 *
38
 * @param vector Vector of force that deforms the shape of the whole Object.
39
 * @param center 3-dimensional Data that, at any given time, returns the Center of the Effect.
40
 * @param region Region that masks the Effect.
41
 */
42
  public VertexEffectDeform(Data3D vector, Data3D center, Data4D region)
43
    {
44
    super(EffectName.DEFORM);
45
    mVector = vector;
46
    mCenter = center;
47
    mRegion = (region==null ? new Static4D(0,0,Float.MAX_VALUE, Float.MAX_VALUE) : region);
48
    }
49

    
50
///////////////////////////////////////////////////////////////////////////////////////////////////
51
/**
52
 * Deform the shape of the whole Object with a (possibly changing in time) vector of force applied to
53
 * a (possibly changing in time) point on the Object.
54
 *
55
 * @param vector Vector of force that deforms the shape of the whole Object.
56
 * @param center 3-dimensional Data that, at any given time, returns the Center of the Effect.
57
 */
58
  public VertexEffectDeform(Data3D vector, Data3D center)
59
    {
60
    super(EffectName.DEFORM);
61
    mVector = vector;
62
    mCenter = center;
63
    mRegion = new Static4D(0,0,Float.MAX_VALUE, Float.MAX_VALUE);
64
    }
65

    
66
///////////////////////////////////////////////////////////////////////////////////////////////////
67

    
68
  public boolean compute(float[] uniforms, int index, long currentDuration, long step )
69
    {
70
    mCenter.get(uniforms,index+5,currentDuration,step);
71
    mRegion.get(uniforms,index+8,currentDuration,step);
72
    boolean ret = mVector.get(uniforms,index,currentDuration,step);
73

    
74
    uniforms[index+9] =-uniforms[index+9];
75

    
76
    return ret;
77
    }
78

    
79
///////////////////////////////////////////////////////////////////////////////////////////////////
80
// Deform the whole shape of the Object by force V. Algorithm is as follows:
81
//
82
// Suppose we apply force (Vx,Vy) at point (Cx,Cy) (i.e. the center of the effect). Then, first of all,
83
// divide the rectangle into 4 smaller rectangles along the 1 horizontal + 1 vertical lines that pass
84
// through (Cx,Cy). Now suppose we have already understood the following case:
85
//
86
// A vertical (0,Vy) force applied to a rectangle (WxH) in size, at center which is the top-left corner
87
// of the rectangle.  (*)
88
//
89
// If we understand (*), then we understand everything, because in order to compute the movement of the
90
// whole rectangle we can apply (*) 8 times: for each one of the 4 sub-rectangles, apply (*) twice,
91
// once for the vertical component of the force vector, the second time for the horizontal one.
92
//
93
// Let's then compute (*):
94
// 1) the top-left point will move by exactly (0,Vy)
95
// 2) we arbitrarily decide that the top-right point will move by (|Vy|/(|Vy|+A*W))*Vy, where A is some
96
//    arbitrary constant (const float A below). The F(V,W) = (|Vy|/(|Vy|+A*W)) comes from the following:
97
//    a) we want F(V,0) = 1
98
//    b) we want lim V->inf (F) = 1
99
//    c) we actually want F() to only depend on W/V, which we have here.
100
// 3) then the top edge of the rectangle will move along the line Vy*G(x), where G(x) = (1 - (A*W/(|Vy|+A*W))*(x/W)^2)
101
// 4) Now we decide that the left edge of the rectangle will move along Vy*H(y), where H(y) = (1 - |y|/(|Vy|+C*|y|))
102
//    where C is again an arbitrary constant. Again, H(y) comes from the requirement that no matter how
103
//    strong we push the left edge of the rectangle up or down, it can never 'go over itself', but its
104
//    length will approach 0 if squeezed very hard.
105
// 5) The last point we need to compute is the left-right motion of the top-right corner (i.e. if we push
106
//    the top-left corner up very hard, we want to have the top-right corner not only move up, but also to
107
//    the left at least a little bit).
108
//    We arbitrarily decide that, in addition to moving up-down by Vy*F(V,W), the corner will also move
109
//    left-right by I(V,W) = B*W*F(V,W), where B is again an arbitrary constant.
110
// 6) combining 3), 4) and 5) together, we arrive at a movement of an arbitrary point (x,y) away from the
111
//    top-left corner:
112
//    X(x,y) = -B*x * (|Vy|/(|Vy|+A*W)) * (1-(y/H)^2)                               (**)
113
//    Y(x,y) = Vy * (1 - |y|/(|Vy|+C*|y|)) * (1 - (A*W/(|Vy|+A*W))*(x/W)^2)         (**)
114
//
115
// We notice that formulas (**) have been construed so that it is possible to continously mirror them
116
// left-right and up-down (i.e. apply not only to the 'bottom-right' rectangle of the 4 subrectangles
117
// but to all 4 of them!).
118
//
119
// Constants:
120
// a) A : valid values: (0,infinity). 'Bendiness' if the surface - the higher A is, the more the surface
121
//        bends. A<=0 destroys the system.
122
// b) B : valid values: <-1,1>. The amount side edges get 'sucked' inwards when we pull the middle of the
123
//        top edge up. B=0 --> not at all, B=1: a looot. B=-0.5: the edges will actually be pushed outwards
124
//        quite a bit. One can also set it to <-1 or >1, but it will look a bit ridiculous.
125
// c) C : valid values: <1,infinity). The derivative of the H(y) function at 0, i.e. the rate of 'squeeze'
126
//        surface gets along the force line. C=1: our point gets pulled very closely to points above it
127
//        even when we apply only small vertical force to it. The higher C is, the more 'uniform' movement
128
//        along the force line is.
129
//        0<=C<1 looks completely ridiculous and C<0 destroys the system.
130

    
131
  public static void enable()
132
    {
133
    addEffect( EffectName.DEFORM,
134

    
135
        "const vec2 ONE = vec2(1.0,1.0);  \n"
136
      + "const float A = 0.5; \n"
137
      + "const float B = 0.2; \n"
138
      + "const float C = 5.0; \n"
139

    
140
      + "vec2 center = vUniforms[effect+1].yz; \n"
141
      + "vec2 ps     = center-v.xy; \n"
142
      + "vec2 aPS    = abs(ps); \n"
143
      + "vec2 maxps  = u_objD.xy + abs(center); \n"
144
      + "float d     = degree_region(vUniforms[effect+2],ps); \n"
145
      + "vec3 force  = vUniforms[effect].xyz * d; \n"
146
      + "vec2 aForce = abs(force.xy); \n"
147
      + "float denom = dot(ps+(1.0-d)*force.xy,ps); \n"
148
      + "float one_over_denom = 1.0/(denom-0.001*(sign(denom)-1.0)); \n"
149
      + "vec2 Aw = A*maxps; \n"
150
      + "vec2 quot = ps / maxps; \n"
151
      + "quot = quot*quot; \n"                                                 // ( (x/W)^2 , (y/H)^2 ) where x,y are distances from V to center
152

    
153
      + "float denomV = 1.0 / (aForce.y + Aw.x); \n"
154
      + "float denomH = 1.0 / (aForce.x + Aw.y); \n"
155

    
156
      + "vec2 vertCorr= ONE - aPS / ( aForce+C*aPS + (ONE-sign(aForce)) ); \n" // avoid division by 0 when force and PS both are 0
157

    
158
      + "float mvXvert = -B * ps.x * aForce.y * (1.0-quot.y) * denomV; \n"     // impact the vertical   component of the force vector has on horizontal movement
159
      + "float mvYhorz = -B * ps.y * aForce.x * (1.0-quot.x) * denomH; \n"     // impact the horizontal component of the force vector has on vertical   movement
160
      + "float mvYvert =  force.y * (1.0-quot.x*Aw.x*denomV) * vertCorr.y; \n" // impact the vertical   component of the force vector has on vertical   movement
161
      + "float mvXhorz = -force.x * (1.0-quot.y*Aw.y*denomH) * vertCorr.x; \n" // impact the horizontal component of the force vector has on horizontal movement
162

    
163
      + "v.x -= (mvXvert+mvXhorz); \n"
164
      + "v.y -= (mvYvert+mvYhorz); \n"
165

    
166
      + "v.z += force.z*d*d*(3.0*d*d -8.0*d +6.0); \n"                         // thick bubble
167
      + "float b = -(12.0*force.z*d*(1.0-d)*(1.0-d)*(1.0-d))*one_over_denom; \n"
168

    
169
      + "n.xy += n.z*b*ps;"
170
      );
171
    }
172
  }
(20-20/25)