2D Dynamic Water Effect for iOS

I’ve been wanting to make a cool dynamic 2D water effect for a while now. There are a lot of code examples out there for other platforms: Android, HTML 5 and JavaScript, but i was unable to find any example for iOS. So I spent the afternoon putting this together, download the Xcode project from GitHub here.

Here’s the main SKScene, based off the Xcode SpriteKit template.

The basic premise is that you have an array of objects laid out along the x-axis. They keep track of their own yPos, speed and Y destination for easing. When we increase their speed they oscillate up and down. The interesting part of the code, and the part that makes the cool outward ripple effect, is that before we move the current object in the array up or down we also use it’s speed to affect the two adjacent objects on either side of it in the array, but with a slight dampening. So the object’s force is not only applied to the up and down acceleration, but radiates outward as well. To get a more watery feel play around with the spread, dampening and tension variables.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
@interface GameScene ()
 
// SPRING STUFF
@property (nonatomic, assign) CGFloat tension;
@property (nonatomic, assign) CGFloat dampening;
@property (nonatomic, assign) CGFloat spread;
 
// waves
@property (nonatomic, assign) CGFloat wavesNum;
@property (nonatomic, strong) NSMutableArray *waves;
@property (nonatomic, strong) NSDate *lastWave;
@property (nonatomic, assign) CGFloat waveDelay;
 
@property (nonatomic, strong) WaveNode *waveNode;
@property (nonatomic, strong) SKShapeNode *yourline;
@property (nonatomic, assign) CGSize screenSize;
@end
 
@implementation GameScene
 
-(void)didMoveToView:(SKView *)view {
     
    self.backgroundColor = [SKColor blackColor];
     
    self.screenSize = [[UIScreen mainScreen] bounds].size;
     
    self.wavesNum = 100;
    self.waves = [NSMutableArray array];
    self.lastWave = [NSDate date];
    self.waveDelay = 500;
     
    self.tension = .025; //.2;//
    self.dampening = .02;//.025,
    self.spread = .3;//.25;
     
     
    for(NSInteger i = 0; i < self.wavesNum; i++){
        CGFloat x =  ceil(self.screenSize.width/self.wavesNum)*i;
        CGFloat y = self.screenSize.height - (self.screenSize.height/3)*2;
        CGPoint point = CGPointMake(x, y);
        WaveObject *waveObject = [[WaveObject alloc] init];
        waveObject.pos = point;
        waveObject.height = self.screenSize.height - y;
        waveObject.targetHeight = self.screenSize.height - y;
        waveObject.speed = 0;
        [self.waves addObject:waveObject];
    }
     
    self.waveNode = [WaveNode node];
    self.waveNode.position = CGPointMake(self.screenSize.width/2., self.screenSize.height/2.);
    [self addChild:self.waveNode];
     
    self.yourline = [SKShapeNode node];
    CGMutablePathRef pathToDraw = CGPathCreateMutable();
    CGPathMoveToPoint(pathToDraw, NULL, 100.0, 100.0);
    CGPathAddLineToPoint(pathToDraw, NULL, 50.0, 50.0);
    self.yourline.path = pathToDraw;
    [self.yourline setStrokeColor:[UIColor whiteColor]];
    [self addChild:self.yourline];
}
 
-(void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {
    /* Called when a touch begins */
     
    for (UITouch *touch in touches) {
        CGPoint location = [touch locationInNode:self];
         
        NSInteger num = floorf(location.x / (self.screenSize.width/self.wavesNum));
        WaveObject *waveObject = self.waves[num];
        waveObject.speed -= location.y;
    }
}
 
-(void)update:(CFTimeInterval)currentTime
{
  // randomly make waves
//    if ([self randomValueBetween:0 andValue:100] < 3) {
//        self.lastWave = [NSDate date];
//        self.waveDelay = floor([self randomValueBetween:0 andValue:1000]);
//        NSInteger num = floorf([self randomValueBetween:0 andValue:self.wavesNum]);
//        WaveObject *waveObject = self.waves[num];
//        waveObject.speed -= 20 + [self randomValueBetween:0 andValue:80];
//    }
    
     
    for (NSInteger i = 0; i < self.wavesNum; i++) {
        WaveObject *waveObject = self.waves[i];
        CGFloat heightDiff = waveObject.targetHeight - waveObject.height;
        waveObject.speed += self.tension * heightDiff - waveObject.speed * self.dampening;
        waveObject.height += waveObject.speed;
    }
     
    NSMutableArray *lDeltas = [NSMutableArray array];
    NSMutableArray *rDeltas = [NSMutableArray array];
     
    [lDeltas addObject:[NSNumber numberWithFloat:0]];
     
    for (NSInteger i = 0; i < self.wavesNum; i++) {
        if (i > 0){
            WaveObject *waveObject = [self.waves objectAtIndex:i];
            WaveObject *previousWaveObject = [self.waves objectAtIndex:i-1];
             
            CGFloat delta = self.spread * (waveObject.height - previousWaveObject.height);
            previousWaveObject.speed += delta;
             
            [lDeltas addObject:[NSNumber numberWithFloat:delta]];
        }
         
        if (i < self.wavesNum - 1){
            WaveObject *waveObject = [self.waves objectAtIndex:i];
            WaveObject *nextWaveObject = [self.waves objectAtIndex:i+1];
             
            CGFloat delta = self.spread * (waveObject.height - nextWaveObject.height);
             
            [rDeltas addObject:[NSNumber numberWithFloat:delta]];
            nextWaveObject.speed += delta;
        }
    }
     
    [rDeltas addObject:[NSNumber numberWithFloat:0]];
     
    CGMutablePathRef path = CGPathCreateMutable();
     
    for (NSInteger i = 0; i < self.wavesNum; i++) {
         
        if (i > 0){
            WaveObject *previousWaveObject = [self.waves objectAtIndex:i-1];
            previousWaveObject.height += [[lDeltas objectAtIndex:i] floatValue];
        }
         
        if (i < self.waves.count - 1){
            WaveObject *nextWaveObject = [self.waves objectAtIndex:i+1];
            nextWaveObject.height += [[rDeltas objectAtIndex:i] floatValue];
        }
         
        WaveObject *waveObject = [self.waves objectAtIndex:i];
        waveObject.pos = CGPointMake(waveObject.pos.x, self.screenSize.height - waveObject.height);
         
        if (i == 0) {
            WaveObject *waveObjectFirst = [self.waves objectAtIndex:0];
            CGPathMoveToPoint(path, NULL, waveObjectFirst.pos.x, waveObjectFirst.pos.y);
        }
         
        if (i< self.wavesNum-1) {
            WaveObject *waveObjectNew = [self.waves objectAtIndex:i];
            CGPathAddLineToPoint(path, NULL, waveObjectNew.pos.x, waveObjectNew.pos.y);
        }
    }
     
     
    self.yourline.path = path;
}
 
- (float)randomValueBetween:(float)low andValue:(float)high {
    return (((float) arc4random() / 0xFFFFFFFFu) * (high - low)) + low;
}

Leave a Reply

Your email address will not be published. Required fields are marked *