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; } |