1
+ class FruitGame {
2
+ constructor ( ) {
3
+ this . canvas = document . getElementById ( 'game-canvas' ) ;
4
+ this . ctx = this . canvas . getContext ( '2d' ) ;
5
+ this . scoreElement = document . getElementById ( 'score-value' ) ;
6
+ this . score = 0 ;
7
+ this . fruits = [ ] ;
8
+ this . slices = [ ] ;
9
+ this . lastTime = 0 ;
10
+ this . fruitInterval = 1000 ;
11
+ this . lastFruitTime = 0 ;
12
+ this . gameState = 'playing' ; // 'playing', 'gameover'
13
+ this . lives = 3 ;
14
+
15
+ this . fruitTypes = [ 'apple' , 'orange' , 'watermelon' , 'pineapple' , 'bomb' ] ;
16
+ this . fruitImages = { } ;
17
+ this . loadImages ( ) ;
18
+
19
+ this . sliceSound = new Audio ( this . sliceSoundData ) ;
20
+ this . bombSound = new Audio ( this . bombSoundData ) ;
21
+ this . backgroundMusic = new Audio ( this . backgroundMusicData ) ;
22
+ this . backgroundMusic . loop = true ;
23
+
24
+ this . resizeCanvas ( ) ;
25
+ window . addEventListener ( 'resize' , ( ) => this . resizeCanvas ( ) ) ;
26
+ this . canvas . addEventListener ( 'touchstart' , ( e ) => this . handleTouch ( e ) ) ;
27
+ this . canvas . addEventListener ( 'touchmove' , ( e ) => this . handleTouch ( e ) ) ;
28
+
29
+ this . backgroundMusic . play ( ) ;
30
+ requestAnimationFrame ( ( time ) => this . gameLoop ( time ) ) ;
31
+ }
32
+
33
+ loadImages ( ) {
34
+ const imageData = {
35
+ apple : '' ,
36
+ orange : '' ,
37
+ watermelon : '' ,
38
+ pineapple : '' ,
39
+ bomb : ''
40
+ } ;
41
+ this . fruitImages = imageData ;
42
+ }
43
+
44
+ resizeCanvas ( ) {
45
+ this . canvas . width = window . innerWidth ;
46
+ this . canvas . height = window . innerHeight ;
47
+ }
48
+
49
+ handleTouch ( e ) {
50
+ e . preventDefault ( ) ;
51
+ const touch = e . touches [ 0 ] ;
52
+ const rect = this . canvas . getBoundingClientRect ( ) ;
53
+ const x = touch . clientX - rect . left ;
54
+ const y = touch . clientY - rect . top ;
55
+ this . checkFruitCut ( x , y ) ;
56
+ this . slices . push ( { x, y, time : Date . now ( ) } ) ;
57
+ }
58
+
59
+ checkFruitCut ( x , y ) {
60
+ this . fruits . forEach ( ( fruit , index ) => {
61
+ if ( this . isPointInFruit ( x , y , fruit ) ) {
62
+ if ( fruit . type === 'bomb' ) {
63
+ this . bombSound . play ( ) ;
64
+ this . lives -- ;
65
+ if ( this . lives <= 0 ) {
66
+ this . gameState = 'gameover' ;
67
+ }
68
+ } else {
69
+ this . sliceSound . play ( ) ;
70
+ this . fruits . splice ( index , 1 ) ;
71
+ this . score += fruit . points ;
72
+ this . scoreElement . textContent = this . score ;
73
+ this . createFruitSlices ( fruit ) ;
74
+ }
75
+ }
76
+ } ) ;
77
+ }
78
+
79
+ isPointInFruit ( x , y , fruit ) {
80
+ const dx = x - fruit . x ;
81
+ const dy = y - fruit . y ;
82
+ return dx * dx + dy * dy <= fruit . radius * fruit . radius ;
83
+ }
84
+
85
+ createFruitSlices ( fruit ) {
86
+ // Create two halves of the fruit
87
+ for ( let i = 0 ; i < 2 ; i ++ ) {
88
+ this . fruits . push ( {
89
+ ...fruit ,
90
+ isSlice : true ,
91
+ rotationSpeed : ( Math . random ( ) - 0.5 ) * 0.2 ,
92
+ speedX : ( Math . random ( ) - 0.5 ) * 5
93
+ } ) ;
94
+ }
95
+ }
96
+
97
+ spawnFruit ( ) {
98
+ const type = this . fruitTypes [ Math . floor ( Math . random ( ) * this . fruitTypes . length ) ] ;
99
+ const fruit = {
100
+ x : Math . random ( ) * this . canvas . width ,
101
+ y : this . canvas . height ,
102
+ radius : 30 ,
103
+ speedY : - 10 - Math . random ( ) * 5 ,
104
+ rotation : 0 ,
105
+ rotationSpeed : ( Math . random ( ) - 0.5 ) * 0.2 ,
106
+ type : type ,
107
+ points : type === 'bomb' ? 0 : 10
108
+ } ;
109
+ this . fruits . push ( fruit ) ;
110
+ }
111
+
112
+ updateFruits ( deltaTime ) {
113
+ this . fruits . forEach ( ( fruit , index ) => {
114
+ fruit . y += fruit . speedY * deltaTime / 16 ;
115
+ fruit . x += ( fruit . speedX || 0 ) * deltaTime / 16 ;
116
+ fruit . rotation += fruit . rotationSpeed * deltaTime / 16 ;
117
+ fruit . speedY += 0.2 ; // gravity
118
+ if ( fruit . y - fruit . radius > this . canvas . height ) {
119
+ if ( ! fruit . isSlice && fruit . type !== 'bomb' ) {
120
+ this . lives -- ;
121
+ if ( this . lives <= 0 ) {
122
+ this . gameState = 'gameover' ;
123
+ }
124
+ }
125
+ this . fruits . splice ( index , 1 ) ;
126
+ }
127
+ } ) ;
128
+ }
129
+
130
+ drawFruits ( ) {
131
+ this . fruits . forEach ( fruit => {
132
+ this . ctx . save ( ) ;
133
+ this . ctx . translate ( fruit . x , fruit . y ) ;
134
+ this . ctx . rotate ( fruit . rotation ) ;
135
+ this . ctx . drawImage ( this . fruitImages [ fruit . type ] , - fruit . radius , - fruit . radius , fruit . radius * 2 , fruit . radius * 2 ) ;
136
+ this . ctx . restore ( ) ;
137
+ } ) ;
138
+ }
139
+
140
+ drawSlices ( ) {
141
+ this . ctx . strokeStyle = 'white' ;
142
+ this . ctx . lineWidth = 3 ;
143
+ this . ctx . beginPath ( ) ;
144
+ this . slices . forEach ( ( slice , index ) => {
145
+ if ( index === 0 ) {
146
+ this . ctx . moveTo ( slice . x , slice . y ) ;
147
+ } else {
148
+ this . ctx . lineTo ( slice . x , slice . y ) ;
149
+ }
150
+ if ( Date . now ( ) - slice . time > 100 ) {
151
+ this . slices . splice ( index , 1 ) ;
152
+ }
153
+ } ) ;
154
+ this . ctx . stroke ( ) ;
155
+ }
156
+
157
+ drawLives ( ) {
158
+ this . ctx . fillStyle = 'red' ;
159
+ for ( let i = 0 ; i < this . lives ; i ++ ) {
160
+ this . ctx . fillRect ( 10 + i * 30 , 40 , 20 , 20 ) ;
161
+ }
162
+ }
163
+
164
+ gameLoop ( currentTime ) {
165
+ const deltaTime = currentTime - this . lastTime ;
166
+ this . lastTime = currentTime ;
167
+
168
+ this . ctx . clearRect ( 0 , 0 , this . canvas . width , this . canvas . height ) ;
169
+
170
+ if ( this . gameState === 'playing' ) {
171
+ if ( currentTime - this . lastFruitTime > this . fruitInterval ) {
172
+ this . spawnFruit ( ) ;
173
+ this . lastFruitTime = currentTime ;
174
+ }
175
+
176
+ this . updateFruits ( deltaTime ) ;
177
+ this . drawFruits ( ) ;
178
+ this . drawSlices ( ) ;
179
+ this . drawLives ( ) ;
180
+ } else if ( this . gameState === 'gameover' ) {
181
+ this . ctx . fillStyle = 'black' ;
182
+ this . ctx . font = '48px Arial' ;
183
+ this . ctx . fillText ( 'Game Over' , this . canvas . width / 2 - 100 , this . canvas . height / 2 ) ;
184
+ this . ctx . fillText ( `Score: ${ this . score } ` , this . canvas . width / 2 - 70 , this . canvas . height / 2 + 60 ) ;
185
+ }
186
+
187
+ requestAnimationFrame ( ( time ) => this . gameLoop ( time ) ) ;
188
+ }
189
+ }
190
+
191
+ window . onload = ( ) => {
192
+ new FruitGame ( ) ;
193
+ } ;
0 commit comments