-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathRaindrops.hs
276 lines (225 loc) · 8.27 KB
/
Raindrops.hs
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
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
-- A demo game put together in less than 2 hours to show how easy
-- game programming is in Haskell.
--
-- If you want to use the wiimote, compile the game with the flag
-- -rtsopts
-- and run it with
-- +RTS -V0
-- import Control.Concurrent
import Control.Monad
import Graphics.UI.SDL as SDL
import Graphics.UI.SDL.Image as SDL
import Graphics.UI.SDL.TTF as TTF
import System.CWiid
import System.Random
-- * Game State
data GameState = GameState
{ raindrops :: [(Float, Float)]
, paddle :: (Float, Float)
, lives :: Int
, points :: Int
, lastRaindrop :: Float
, lastFrame :: Float -- Time of last frame
, randomGen :: StdGen
}
deriving Show
level :: GameState -> Int
level gameState = points gameState `div` dropsPerLevel
updatePaddlePos :: GameState -> Float -> GameState
updatePaddlePos gameState newX =
gameState { paddle = (min newX (width - paddleW), height - paddleMargin) }
initialGameState :: StdGen -> GameState
initialGameState gen =
GameState [(width / 2, dropMargin)] (0, height - paddleMargin) maxLives 0 0 0 gen
-- * Runtime environment (resources, devices)
data Env = Env
{ bgImg :: Surface
, dropImg :: Surface
, paddleImg :: Surface
, font :: TTF.Font
, cwiid :: Maybe CWiidWiimote
}
main :: IO ()
main = do
-- Initialise SDL
SDL.init [InitEverything]
ttfOk <- TTF.init
when ttfOk $ do
font <- TTF.tryOpenFont "data/font.ttf" 32
bg <- load "data/background.png"
-- Load with a mask
drop <- load "data/cherry.png"
t <- mapRGB (surfaceGetPixelFormat drop) 0 255 0
setColorKey drop [SrcColorKey, RLEAccel] t
-- Load with a mask
paddle <- load "data/player.png"
t <- mapRGB (surfaceGetPixelFormat paddle) 0 255 0
setColorKey paddle [SrcColorKey, RLEAccel] t
-- Create window, no mouse
SDL.setVideoMode (round width) (round height) 32 [SWSurface]
SDL.setCaption "Raindrops" ""
SDL.showCursor False
-- Create Random number generator
gen <- getStdGen
-- Initialise input devices
wiimote <- initializeWiimote
case font of
Nothing -> return ()
Just ttf -> let env = Env bg drop paddle ttf wiimote
gs = initialGameState gen
in run env gs
-- Game loop
run :: Env -> GameState -> IO ()
run env gameState = do
-- IO: Sense (input)
gameStateP <- calculatePaddlePos env gameState
-- IO: Sense (time)
newTime <- fmap fromIntegral SDL.getTicks
let dt = newTime - lastFrame gameStateP
let gameStateT = gameStateP { lastFrame = newTime }
-- Physics (movement)
let gameStateP = moveForward dt gameStateT
-- Physics (collisions)
let gameStateC = raindropsBottom (raindropsPaddle gameStateP)
-- Logic (new raindrops)
let dtLastRaindrop = newTime - lastRaindrop gameStateC
gameStateN =
if dtLastRaindrop > dropDelay
then let (newX, gen') = randomR (0, round (width - dropW) :: Int)
(randomGen gameStateC)
oldDrops = raindrops gameStateC
in gameStateC { raindrops = (fromIntegral newX, dropMargin) : oldDrops
, lastRaindrop = newTime
, randomGen = gen'
}
else gameStateC
-- Logic (game over)
let gameStateO = if lives gameStateN < 0
then initialGameState (randomGen gameStateN)
else gameStateN
-- IO: Paint
render env gameStateO
-- Loop
run env gameStateO
-- * Physics
moveForward :: Float -> GameState -> GameState
moveForward dt gs = gs { raindrops = movedRaindrops }
where movedRaindrops = map moveRaindrop (raindrops gs)
moveRaindrop (x,y) = (x, y + 0.1 * dt * fromIntegral (level gs + 1))
-- * Collisions
-- ** Collisions with paddle
raindropsPaddle :: GameState -> GameState
raindropsPaddle gs = gs { raindrops = remainingRaindrops
, points = points gs + pts
}
where remainingRaindrops = filter (not.collidesWithPaddle) (raindrops gs)
pts = length (raindrops gs) - length remainingRaindrops
collidesWithPaddle (x,y) = (within x paddleXMin paddleXMax
|| within paddleXMin x (x + dropW))
&& (within y paddleYMin paddleYMax
|| within paddleYMin y (y + dropH))
where paddleXMin = fst (paddle gs)
paddleXMax = fst (paddle gs) + paddleW
paddleYMin = snd (paddle gs)
paddleYMax = snd (paddle gs) + paddleH
within x xMin xMax = x >= xMin && x <= xMax
-- ** Collisions with bottom
raindropsBottom :: GameState -> GameState
raindropsBottom gs = gs { raindrops = remainingRaindrops
, lives = decreasedLives
}
where remainingRaindrops = filter (\(_,y) -> y < height) (raindrops gs)
decreasedLives = lives gs - diffDrops
diffDrops = length (raindrops gs) - length remainingRaindrops
-- * Input sensing
calculatePaddlePos :: Env -> GameState -> IO GameState
calculatePaddlePos env gs = case cwiid env of
Nothing -> calculatePaddlePosSDL env gs
Just wm -> do (x,y) <- senseWiimote wm
return (updatePaddlePos gs x)
-- ** SDL Sensing
calculatePaddlePosSDL :: Env -> GameState -> IO GameState
calculatePaddlePosSDL env gs = do
e <- pollEvent
case e of
NoEvent -> return gs
MouseMotion x y _ _ -> calculatePaddlePosSDL env (updatePaddlePos gs (fromIntegral x))
_ -> calculatePaddlePosSDL env gs
-- * Output drawing
render :: Env -> GameState -> IO ()
render env gameState = do
screen <- getVideoSurface
-- Clear screen
SDL.blitSurface (bgImg env) Nothing
screen (Just (SDL.Rect 0 0 (round width) (round height)))
-- Paint each raindrop
let paintADropAt (x,y) = do
SDL.blitSurface (dropImg env) Nothing
screen (Just (SDL.Rect (round x) (round y) (round dropW) (round dropH)))
mapM_ paintADropAt (raindrops gameState)
-- Paint the paddle
let (x,y) = paddle gameState
SDL.blitSurface (paddleImg env) Nothing
screen (Just (SDL.Rect (round x) (round y) (round paddleW) (round paddleH)))
-- Paint points, lives
let ttf = font env
message <- TTF.renderTextSolid
ttf
("Level " ++ show (level gameState)
++ " / Lives " ++ show (lives gameState))
(SDL.Color 128 128 128)
let w1 = SDL.surfaceGetWidth message
h1 = SDL.surfaceGetHeight message
SDL.blitSurface message Nothing
screen (Just (SDL.Rect 10 10 w1 h1))
-- Present
SDL.flip screen
-- * Game constants
paddleW, paddleH :: Float
paddleW = 126
paddleH = 31
paddleMargin :: Float
paddleMargin = 60
width, height :: Float
width = 800
height = 480
dropW, dropH :: Float
dropW = 70
dropH = 70
dropMargin :: Float
dropMargin = 10
dropsPerLevel :: Int
dropsPerLevel = 20
dropDelay :: Float
dropDelay = 500
maxLives :: Int
maxLives = 10
-- * Wiimote sensing
-- | Initializes the wiimote, optionally returning the sensing function. It
-- returns Nothing if the Wiimote cannot be detected. Users should have a BT
-- device and press 1+2 to connect to it. A message is shown on stdout.
initializeWiimote :: IO (Maybe CWiidWiimote)
initializeWiimote = do
putStrLn "Initializing WiiMote. Please press 1+2 to connect."
wm <- cwiidOpen
case wm of
Nothing -> return ()
Just wm' -> void $ cwiidSetRptMode wm' 15 -- Enable button reception, acc and IR
return wm
senseWiimote :: CWiidWiimote -> IO (Float, Float)
senseWiimote wmdev = do
irs <- cwiidGetIR wmdev
-- Obtain positions of leds 1 and 2 (with a normal wii bar, those
-- will be the ones we use).
let led1 = irs!!0
led2 = irs!!1
-- Calculate mid point between sensor bar leds
let posX = ((cwiidIRSrcPosX led1) + (cwiidIRSrcPosX led2)) `div` 2
posY = ((cwiidIRSrcPosY led1) + (cwiidIRSrcPosY led2)) `div` 2
-- Calculate proportional coordinates
let propX = fromIntegral (1024 - posX) / 1024.0
propY = fromIntegral (max 0 (posY - 384)) / 384.0
-- Calculate game area coordinates
let finX = width * propX
finY = height * propY
return (finX, finY)