diff options
author | 3gg <3gg@shellblade.net> | 2025-01-04 16:26:25 -0800 |
---|---|---|
committer | 3gg <3gg@shellblade.net> | 2025-01-04 16:26:25 -0800 |
commit | d20e822e806afe67c8e255a645061638b75d3546 (patch) | |
tree | 10129ce9bcf31e75dfa257f56a62046390e68a4e /Demos | |
parent | 548e40add0fd3ebea18510f7c16a464680aed63c (diff) |
Add balls demo.
Diffstat (limited to 'Demos')
-rw-r--r-- | Demos/Balls/Main.hs | 177 |
1 files changed, 177 insertions, 0 deletions
diff --git a/Demos/Balls/Main.hs b/Demos/Balls/Main.hs new file mode 100644 index 0000000..2e759bc --- /dev/null +++ b/Demos/Balls/Main.hs | |||
@@ -0,0 +1,177 @@ | |||
1 | {-# LANGUAGE MultiParamTypeClasses #-} | ||
2 | --{-# LANGUAGE NoImplicitPrelude #-} | ||
3 | {-# LANGUAGE FlexibleContexts #-} | ||
4 | {-# LANGUAGE TypeSynonymInstances #-} | ||
5 | |||
6 | module Main where | ||
7 | |||
8 | import Spear.App | ||
9 | import Spear.Game | ||
10 | import Spear.Math.AABB | ||
11 | import qualified Spear.Math.Matrix3 as Matrix3 | ||
12 | import qualified Spear.Math.Matrix4 as Matrix4 | ||
13 | import Spear.Math.Spatial | ||
14 | import Spear.Math.Spatial2 | ||
15 | import Spear.Math.Vector | ||
16 | import Spear.Physics.Collision | ||
17 | --import Spear.Prelude | ||
18 | import Spear.Render.Core.Pipeline | ||
19 | import Spear.Render.Core.State | ||
20 | import Spear.Render.Immediate | ||
21 | import Spear.Sound.Sound | ||
22 | import Spear.Sound.State | ||
23 | import Spear.Window | ||
24 | |||
25 | import Control.Monad (when) | ||
26 | import Spear.Math.Vector (Vector3) | ||
27 | |||
28 | |||
29 | ballSize = 0.01 | ||
30 | numBalls = 1000 | ||
31 | |||
32 | data Ball = Ball | ||
33 | { ballPosition :: {-# UNPACK #-} !Vector2 | ||
34 | , ballVelocity :: {-# UNPACK #-} !Vector2 | ||
35 | } | ||
36 | |||
37 | instance Positional Ball Vector2 where | ||
38 | setPosition p ball = ball { ballPosition = p } | ||
39 | position = ballPosition | ||
40 | translate v ball = ball { ballPosition = v + ballPosition ball } | ||
41 | |||
42 | instance Bounded2 Ball where | ||
43 | boundingVolume ball = aabb2Volume $ translate (ballPosition ball) (AABB2 (-size) size) | ||
44 | where size = vec2 s s | ||
45 | s = ballSize / (2::Float) | ||
46 | |||
47 | data World = World | ||
48 | { viewProjection :: Matrix4.Matrix4 | ||
49 | , balls :: [Ball] | ||
50 | } | ||
51 | |||
52 | type GameState = AppState World | ||
53 | |||
54 | |||
55 | options = defaultAppOptions { title = "Balls" } | ||
56 | |||
57 | app = App options initGame endGame step render resize | ||
58 | |||
59 | |||
60 | main :: IO () | ||
61 | main = runApp app | ||
62 | |||
63 | initGame :: Game AppContext World | ||
64 | initGame = | ||
65 | let | ||
66 | world = zipWith Ball positions velocities | ||
67 | positions = (+vec2 0.5 0.5) . makePosition <$> numbers | ||
68 | makePosition i = radius * vec2 (sin (f*i)) (cos (f*i)) | ||
69 | velocities = makeVelocity <$> numbers | ||
70 | makeVelocity i = scale speed $ vec2 (sin (f*i)) (cos (f*i)) | ||
71 | numbers = [1..numBalls] | ||
72 | f = 2*pi / numBalls | ||
73 | radius = 0.05 | ||
74 | speed = 0.4 | ||
75 | in | ||
76 | return $ World Matrix4.id world | ||
77 | |||
78 | endGame :: Game GameState () | ||
79 | endGame = return () | ||
80 | |||
81 | |||
82 | step :: Elapsed -> Dt -> [InputEvent] -> Game GameState Bool | ||
83 | step elapsed dt inputEvents = do | ||
84 | modifyGameState $ \world -> world | ||
85 | { balls = moveBalls dt $ balls world | ||
86 | } | ||
87 | return (not $ exitRequested inputEvents) | ||
88 | |||
89 | exitRequested = elem (KeyDown KEY_ESC) | ||
90 | |||
91 | moveBalls :: Elapsed -> [Ball] -> [Ball] | ||
92 | moveBalls dt = (bounceBall dt . moveBall dt <$>) | ||
93 | |||
94 | moveBall :: Elapsed -> Ball -> Ball | ||
95 | moveBall dt ball = translate (scale (realToFrac dt) $ ballVelocity ball) ball | ||
96 | |||
97 | bounceBall :: Elapsed -> Ball -> Ball | ||
98 | bounceBall dt ball = | ||
99 | let | ||
100 | (AABB2Volume (AABB2 pmin pmax)) = boundingVolume ball | ||
101 | sideCollision = x pmin < 0 || x pmax > 1 | ||
102 | backCollision = y pmin < 0 || y pmax > 1 | ||
103 | flipX v@(Vector2 x y) = if sideCollision then vec2 (-x) y else v | ||
104 | flipY v@(Vector2 x y) = if backCollision then vec2 x (-y) else v | ||
105 | velocity = ballVelocity ball | ||
106 | velocity' | ||
107 | = flipX | ||
108 | . flipY | ||
109 | $ velocity | ||
110 | collision = velocity' /= velocity | ||
111 | -- Apply offset when collision occurs to avoid sticky collisions. | ||
112 | delta = if collision then 1 else 0 | ||
113 | dt' = realToFrac dt | ||
114 | in | ||
115 | ball | ||
116 | { ballPosition = ballPosition ball + scale (delta * dt') velocity' | ||
117 | , ballVelocity = velocity' | ||
118 | } | ||
119 | |||
120 | |||
121 | render :: Game GameState () | ||
122 | render = do | ||
123 | gameState <- getGameState | ||
124 | siblingGame $ do | ||
125 | immStart | ||
126 | immSetViewProjectionMatrix (viewProjection gameState) | ||
127 | -- Clear the background to a different colour than the playable area to make | ||
128 | -- the latter distinguishable. | ||
129 | setClearColour (0.2, 0.2, 0.2, 0.0) | ||
130 | clearBuffers [ColourBuffer] | ||
131 | render' $ balls gameState | ||
132 | immEnd | ||
133 | |||
134 | render' :: [Ball] -> Game ImmRenderState () | ||
135 | render' balls = do | ||
136 | immLoadIdentity | ||
137 | renderBackground | ||
138 | -- Draw objects. | ||
139 | immSetColour (vec4 1.0 1.0 1.0 1.0) | ||
140 | mapM_ renderBall balls | ||
141 | |||
142 | renderBackground :: Game ImmRenderState () | ||
143 | renderBackground = | ||
144 | let pmin = 0 :: Float | ||
145 | pmax = 1 :: Float | ||
146 | in do | ||
147 | immSetColour (vec4 0.0 0.25 0.41 1.0) | ||
148 | immDrawQuads2d [ | ||
149 | (vec2 pmin pmin | ||
150 | ,vec2 pmax pmin | ||
151 | ,vec2 pmax pmax | ||
152 | ,vec2 pmin pmax)] | ||
153 | |||
154 | renderBall :: Ball -> Game ImmRenderState () | ||
155 | renderBall ball = | ||
156 | let (AABB2Volume (AABB2 (Vector2 xmin ymin) (Vector2 xmax ymax))) = boundingVolume ball | ||
157 | in | ||
158 | immDrawQuads2d [ | ||
159 | (vec2 xmin ymin | ||
160 | ,vec2 xmax ymin | ||
161 | ,vec2 xmax ymax | ||
162 | ,vec2 xmin ymax)] | ||
163 | |||
164 | |||
165 | resize :: WindowEvent -> Game GameState () | ||
166 | resize (ResizeEvent w h) = | ||
167 | let r = fromIntegral w / fromIntegral h | ||
168 | pad = if r > 1 then (r-1) / 2 else (1/r - 1) / 2 | ||
169 | left = if r > 1 then -pad else 0 | ||
170 | right = if r > 1 then 1 + pad else 1 | ||
171 | bottom = if r > 1 then 0 else -pad | ||
172 | top = if r > 1 then 1 else 1 + pad | ||
173 | in do | ||
174 | setViewport 0 0 w h | ||
175 | modifyGameState $ \pong -> pong { | ||
176 | viewProjection = Matrix4.ortho left right bottom top (-1) 1 | ||
177 | } | ||