bzr branch
http://9ix.org/bzr/zoeplat
2
by Josh C
basic tiles, map, player, movement |
1 |
STRICT = true |
2 |
DEBUG = true |
|
3 |
||
4 |
require 'zoetrope' |
|
17
by Josh C
reorganize code - separate X and Y physics so we can collide in each |
5 |
_ = require 'underscore' |
12
by Josh C
only jump when you're on the ground |
6 |
--inspect = require 'inspect' |
2
by Josh C
basic tiles, map, player, movement |
7 |
|
10
by Josh C
make player an animation |
8 |
Player = Animation:extend { |
2
by Josh C
basic tiles, map, player, movement |
9 |
image = 'data/player.png', |
10
by Josh C
make player an animation |
10 |
height = 32, |
11 |
width = 32, |
|
12 |
sequences = { |
|
13 |
stand = { frames = { 1 }, fps = 1 }, |
|
14 |
walk = { frames = { 2, 3 }, fps = 5 }, |
|
15 |
jump = { frames = { 4 }, fps = 1 } |
|
16 |
}, |
|
3
by Josh C
jump |
17 |
onNew = function (self) |
18 |
self.velocity.y = 0 |
|
5
by Josh C
use built-in maxVelocity system |
19 |
self.maxVelocity.y = 400 |
3
by Josh C
jump |
20 |
end, |
17
by Josh C
reorganize code - separate X and Y physics so we can collide in each |
21 |
doPhysics = function (self, dir, elapsed) |
22 |
local vel = self.velocity |
|
23 |
local acc = self.acceleration |
|
24 |
local drag = self.drag |
|
25 |
local minVel = self.minVelocity |
|
26 |
local maxVel = self.maxVelocity |
|
27 |
||
28 |
-- check existence of properties |
|
29 |
||
30 |
if STRICT then |
|
31 |
assert(vel, 'active sprite has no velocity property') |
|
32 |
assert(acc, 'active sprite has no acceleration property') |
|
33 |
assert(drag, 'active sprite has no drag property') |
|
34 |
assert(minVel, 'active sprite has no minVelocity property') |
|
35 |
assert(maxVel, 'active sprite has no maxVelocity property') |
|
36 |
assert(_.include({'x','y','rotation'}, dir), 'direction should be x, y, or rotation') |
|
37 |
end |
|
38 |
||
39 |
vel.x = vel.x or 0 |
|
40 |
vel.y = vel.y or 0 |
|
41 |
vel.rotation = vel.rotation or 0 |
|
42 |
||
43 |
-- physics |
|
44 |
||
45 |
if acc[dir] and acc[dir] ~= 0 then |
|
46 |
vel[dir] = vel[dir] + acc[dir] * elapsed |
|
47 |
else |
|
48 |
if drag[dir] then |
|
49 |
if vel[dir] > 0 then |
|
50 |
vel[dir] = vel[dir] - drag[dir] * elapsed |
|
51 |
if vel[dir] < 0 then vel[dir] = 0 end |
|
52 |
elseif vel[dir] < 0 then |
|
53 |
vel[dir] = vel[dir] + drag[dir] * elapsed |
|
54 |
if vel[dir] > 0 then vel[dir] = 0 end |
|
55 |
end |
|
56 |
end |
|
57 |
end |
|
58 |
||
59 |
if minVel[dir] and vel[dir] < minVel[dir] then vel[dir] = minVel[dir] end |
|
60 |
if maxVel[dir] and vel[dir] > maxVel[dir] then vel[dir] = maxVel[dir] end |
|
61 |
||
62 |
if vel[dir] ~= 0 then self[dir] = self[dir] + vel[dir] * elapsed end |
|
63 |
end, |
|
4
by Josh C
fix jitter caused by focus shift happening in the wrong order. Looks |
64 |
onStartFrame = function (self) |
65 |
-- this is all in startframe so it happens before |
|
66 |
-- physics calc at beginning of update |
|
3
by Josh C
jump |
67 |
|
12
by Josh C
only jump when you're on the ground |
68 |
-- jumping/falling updates could go in EndFrame... |
69 |
self.falling = self.velocity.y > 0 |
|
70 |
if self.falling then self.jumping = false end |
|
71 |
--print(self.jumping, self.falling) |
|
72 |
||
19
by Josh C
climb walls |
73 |
if (not self.onGround) and (not self.onWall) then |
13
by Josh C
reapply jump animation after Y collision. (there's a frame of no |
74 |
self:play('jump') |
75 |
end |
|
76 |
||
6
by Josh C
whitespace cleanup |
77 |
self.velocity.x = 0 |
78 |
self.acceleration.y = 800 |
|
79 |
||
19
by Josh C
climb walls |
80 |
if self.onWall then |
81 |
self.acceleration.y = 0 |
|
82 |
||
83 |
if the.keys:pressed('up') then |
|
84 |
self.velocity.y = -200 |
|
85 |
self:play('stand') |
|
86 |
elseif the.keys:pressed('down') then |
|
87 |
self.velocity.y = 200 |
|
88 |
self:play('stand') |
|
89 |
else |
|
90 |
self.velocity.y = 0 |
|
91 |
self:play('stand') |
|
92 |
end |
|
93 |
end |
|
94 |
||
6
by Josh C
whitespace cleanup |
95 |
if the.keys:pressed('left') then |
96 |
self.velocity.x = -200 |
|
15
by Josh C
more reliable onGround calc |
97 |
if self.onGround then self:play('walk') end |
19
by Josh C
climb walls |
98 |
if self.onWall == 'right' then self.onWall = false end |
99 |
if self.onWall == 'right' then self.onWall = false end |
|
6
by Josh C
whitespace cleanup |
100 |
elseif the.keys:pressed('right') then |
101 |
self.velocity.x = 200 |
|
15
by Josh C
more reliable onGround calc |
102 |
if self.onGround then self:play('walk') end |
19
by Josh C
climb walls |
103 |
if self.onWall == 'left' then self.onWall = false end |
10
by Josh C
make player an animation |
104 |
else |
15
by Josh C
more reliable onGround calc |
105 |
if self.onGround then self:play('stand') end |
6
by Josh C
whitespace cleanup |
106 |
end |
107 |
||
15
by Josh C
more reliable onGround calc |
108 |
if the.keys:justPressed('up') and self.onGround then |
6
by Josh C
whitespace cleanup |
109 |
self.velocity.y = -400 |
12
by Josh C
only jump when you're on the ground |
110 |
self.jumping = true |
6
by Josh C
whitespace cleanup |
111 |
end |
112 |
end, |
|
17
by Josh C
reorganize code - separate X and Y physics so we can collide in each |
113 |
update = function (self, elapsed) |
114 |
-- NOTE: this is an override, not a callback |
|
115 |
||
116 |
self:doPhysics('x', elapsed) |
|
117 |
||
118 |
self.onGround = false -- right before Y collision callbacks |
|
119 |
self:doPhysics('y', elapsed) |
|
120 |
||
121 |
self:collide(the.view.map) |
|
18
by Josh C
call Animation.update so we actually get animations |
122 |
|
123 |
Animation.update(self, elapsed) |
|
17
by Josh C
reorganize code - separate X and Y physics so we can collide in each |
124 |
end, |
8
by Josh C
some basic collision (and workarounds) |
125 |
onCollide = function (self, other, xOverlap, yOverlap) |
126 |
-- seriously, why does this even fire? |
|
127 |
if other == the.view.map then return end |
|
128 |
||
129 |
--print(string.format('col s{x=%i y=%i w=%i h=%i} %s', self.x, self.y, self.width, self.height, tostring(other))) |
|
130 |
--print('vel.x:'..self.velocity.x.." vel.y:"..self.velocity.y) |
|
131 |
||
132 |
-- assumption: any other collision is with a solid map tile |
|
133 |
if yOverlap > xOverlap then |
|
134 |
other:displace(self) |
|
16
by Josh C
try to track X collisions. break out Sprite's physics in prep for |
135 |
|
136 |
if self.velocity.x > 0 then |
|
137 |
self.onWall = 'right' |
|
138 |
elseif self.velocity.x < 0 then |
|
139 |
self.onWall = 'left' |
|
140 |
else |
|
17
by Josh C
reorganize code - separate X and Y physics so we can collide in each |
141 |
print 'x ??' |
16
by Josh C
try to track X collisions. break out Sprite's physics in prep for |
142 |
end |
8
by Josh C
some basic collision (and workarounds) |
143 |
elseif xOverlap > yOverlap then |
144 |
-- check if we've moved since collisions were generated |
|
145 |
local xov, yov = self:overlap(other.x, other.y, |
|
146 |
other.width, other.height) |
|
147 |
if xov ~= 0 and yov ~= 0 then |
|
15
by Josh C
more reliable onGround calc |
148 |
--print('y collision') |
149 |
if self.velocity.y > 0 then |
|
150 |
self.onGround = true |
|
151 |
end |
|
152 |
||
8
by Josh C
some basic collision (and workarounds) |
153 |
self.velocity.y = 0 |
154 |
other:displace(self) |
|
12
by Josh C
only jump when you're on the ground |
155 |
self.jumping = false |
8
by Josh C
some basic collision (and workarounds) |
156 |
end |
157 |
else |
|
17
by Josh C
reorganize code - separate X and Y physics so we can collide in each |
158 |
print('xy ??') |
8
by Josh C
some basic collision (and workarounds) |
159 |
end |
160 |
||
12
by Josh C
only jump when you're on the ground |
161 |
end, |
2
by Josh C
basic tiles, map, player, movement |
162 |
} |
163 |
||
164 |
GameView = View:extend { |
|
165 |
onNew = function (self) |
|
166 |
self:loadLayers('data/map.lua') |
|
167 |
self.focus = the.player |
|
168 |
self:clampTo(self.map) |
|
8
by Josh C
some basic collision (and workarounds) |
169 |
end, |
170 |
onUpdate = function (self) |
|
171 |
--print('tick') |
|
17
by Josh C
reorganize code - separate X and Y physics so we can collide in each |
172 |
--the.player:collide(self.map) |
8
by Josh C
some basic collision (and workarounds) |
173 |
--self.map:collide(the.player) |
174 |
end |
|
2
by Josh C
basic tiles, map, player, movement |
175 |
} |
176 |
||
177 |
the.app = App:new { |
|
178 |
onRun = function (self) |
|
179 |
self.view = GameView:new() |
|
12
by Josh C
only jump when you're on the ground |
180 |
--print(inspect(_(the.app):keys())) |
15
by Josh C
more reliable onGround calc |
181 |
self.console:watch('onGround', 'the.player.onGround') |
16
by Josh C
try to track X collisions. break out Sprite's physics in prep for |
182 |
self.console:watch('onWall', 'the.player.onWall') |
2
by Josh C
basic tiles, map, player, movement |
183 |
end, |
184 |
onUpdate = function (self, dt) |
|
14
by Josh C
don't quit w/ escape if debug console is up |
185 |
if the.keys:justPressed('escape') and |
186 |
not self.console.visible then |
|
17
by Josh C
reorganize code - separate X and Y physics so we can collide in each |
187 |
self.quit() |
12
by Josh C
only jump when you're on the ground |
188 |
end |
2
by Josh C
basic tiles, map, player, movement |
189 |
end |
190 |
} |