/zoeplat

To get this branch, use:
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'
20 by Josh C
fairly major overhaul of collision handling to track whether we're on a
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
   },
20 by Josh C
fairly major overhaul of collision handling to track whether we're on a
17
   collisions = {},
18
   onWall = false,
3 by Josh C
jump
19
   onNew = function (self)
20
              self.velocity.y = 0
5 by Josh C
use built-in maxVelocity system
21
              self.maxVelocity.y = 400
3 by Josh C
jump
22
           end,
17 by Josh C
reorganize code - separate X and Y physics so we can collide in each
23
   doPhysics = function (self, dir, elapsed)
24
                  local vel = self.velocity
25
                  local acc = self.acceleration
26
                  local drag = self.drag
27
                  local minVel = self.minVelocity
28
                  local maxVel = self.maxVelocity
29
30
                  -- check existence of properties
31
32
                  if STRICT then
33
                     assert(vel, 'active sprite has no velocity property')
34
                     assert(acc, 'active sprite has no acceleration property')
35
                     assert(drag, 'active sprite has no drag property')
36
                     assert(minVel, 'active sprite has no minVelocity property')
37
                     assert(maxVel, 'active sprite has no maxVelocity property')
20 by Josh C
fairly major overhaul of collision handling to track whether we're on a
38
                     assert(__.include({'x','y','rotation'}, dir), 'direction should be x, y, or rotation')
17 by Josh C
reorganize code - separate X and Y physics so we can collide in each
39
                  end
40
41
                  vel.x = vel.x or 0
42
                  vel.y = vel.y or 0
43
                  vel.rotation = vel.rotation or 0
44
45
                  -- physics
46
47
                  if acc[dir] and acc[dir] ~= 0 then
48
                     vel[dir] = vel[dir] + acc[dir] * elapsed
49
                  else
50
                     if drag[dir] then
51
                        if vel[dir] > 0 then
52
                           vel[dir] = vel[dir] - drag[dir] * elapsed
53
                           if vel[dir] < 0 then vel[dir] = 0 end
54
                        elseif vel[dir] < 0 then
55
                           vel[dir] = vel[dir] + drag[dir] * elapsed
56
                           if vel[dir] > 0 then vel[dir] = 0 end
57
                        end
58
                     end
59
                  end
60
61
                  if minVel[dir] and vel[dir] < minVel[dir] then vel[dir] = minVel[dir] end
62
                  if maxVel[dir] and vel[dir] > maxVel[dir] then vel[dir] = maxVel[dir] end
63
64
                  if vel[dir] ~= 0 then self[dir] = self[dir] + vel[dir] * elapsed end
65
               end,
4 by Josh C
fix jitter caused by focus shift happening in the wrong order. Looks
66
   onStartFrame = function (self)
67
                     -- this is all in startframe so it happens before
68
                     -- physics calc at beginning of update
3 by Josh C
jump
69
12 by Josh C
only jump when you're on the ground
70
                     -- jumping/falling updates could go in EndFrame...
71
                     self.falling = self.velocity.y > 0
72
                     if self.falling then self.jumping = false end
73
                     --print(self.jumping, self.falling)
74
19 by Josh C
climb walls
75
                     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
76
                        self:play('jump')
77
                     end
78
6 by Josh C
whitespace cleanup
79
                     self.acceleration.y = 800
80
19 by Josh C
climb walls
81
                     if self.onWall then
82
                        self.acceleration.y = 0
83
84
                        if the.keys:pressed('up') then
85
                           self.velocity.y = -200
86
                           self:play('stand')
87
                        elseif the.keys:pressed('down') then
88
                           self.velocity.y = 200
89
                           self:play('stand')
90
                        else
91
                           self.velocity.y = 0
92
                           self:play('stand')
93
                        end
94
                     end
95
6 by Josh C
whitespace cleanup
96
                     if the.keys:pressed('left') then
97
                        self.velocity.x = -200
15 by Josh C
more reliable onGround calc
98
                        if self.onGround then self:play('walk') end
19 by Josh C
climb walls
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
21 by Josh C
really easy version of knowing when we reached the top of a wall.
106
                        if not self.onWall then
107
                           self.velocity.x = 0
108
                        end
6 by Josh C
whitespace cleanup
109
                     end
110
15 by Josh C
more reliable onGround calc
111
                     if the.keys:justPressed('up') and self.onGround then
6 by Josh C
whitespace cleanup
112
                        self.velocity.y = -400
12 by Josh C
only jump when you're on the ground
113
                        self.jumping = true
6 by Josh C
whitespace cleanup
114
                     end
115
                  end,
17 by Josh C
reorganize code - separate X and Y physics so we can collide in each
116
   update = function (self, elapsed)
117
               -- NOTE: this is an override, not a callback
118
119
               self:doPhysics('x', elapsed)
20 by Josh C
fairly major overhaul of collision handling to track whether we're on a
120
               self:collide(the.view.map)
121
122
               -- handle X collisions
21 by Josh C
really easy version of knowing when we reached the top of a wall.
123
               self.onWall = false
20 by Josh C
fairly major overhaul of collision handling to track whether we're on a
124
               for _, col in ipairs(self.collisions) do
125
                  col.other:displaceDir(self, 'x')
126
                  if self.velocity.x > 0 then
127
                     self.onWall = 'right'
128
                  elseif self.velocity.x < 0 then
129
                     self.onWall = 'left'
130
                  else
131
                     print 'x ??'
132
                  end
133
               end
17 by Josh C
reorganize code - separate X and Y physics so we can collide in each
134
135
               self.onGround = false -- right before Y collision callbacks
136
               self:doPhysics('y', elapsed)
137
               self:collide(the.view.map)
18 by Josh C
call Animation.update so we actually get animations
138
20 by Josh C
fairly major overhaul of collision handling to track whether we're on a
139
               -- handle Y collisions
140
               for _, col in ipairs(self.collisions) do
141
                  if self.velocity.y > 0 then
142
                     self.onGround = true
143
                  end
144
145
                  col.other:displaceDir(self, 'y')
146
                  self.velocity.y = 0
147
                  self.jumping = false
148
               end
149
18 by Josh C
call Animation.update so we actually get animations
150
               Animation.update(self, elapsed)
17 by Josh C
reorganize code - separate X and Y physics so we can collide in each
151
            end,
20 by Josh C
fairly major overhaul of collision handling to track whether we're on a
152
   collide = function (self, ...)
153
                self.collisions = {}
154
                Animation.collide(self, ...)
155
                -- I could return a true/false value here if I wanted to...
156
             end,
8 by Josh C
some basic collision (and workarounds)
157
   onCollide = function (self, other, xOverlap, yOverlap)
158
                  if other == the.view.map then return end
159
20 by Josh C
fairly major overhaul of collision handling to track whether we're on a
160
                  table.insert(self.collisions, {other = other,
161
                                                 xOverlap = xOverlap,
162
                                                 yOverlap = yOverlap })
163
               end
2 by Josh C
basic tiles, map, player, movement
164
}
165
20 by Josh C
fairly major overhaul of collision handling to track whether we're on a
166
-- displace on a specific axis (monkey patch Sprite)
167
function Sprite:displaceDir(other, dir)
168
   if not self.solid or self == other or not other.solid then return end
169
   if STRICT then assert(other:instanceOf(Sprite), 'asked to displace a non-sprite') end
170
171
   if other.sprites then
172
      -- handle groups
173
174
      for _, spr in pairs(other.sprites) do
175
         self:displace(spr, dir)
176
      end
177
   else
178
      -- handle sprites
179
      local dim
180
      if dir == 'x' then
181
         dim = 'width'
182
      elseif dir == 'y' then
183
         dim = 'height'
184
      else
185
         print 'dir ??'
186
      end
187
188
      local negMove = (other[dir] - self[dir]) + other[dim]
189
      local posMove = (self[dir] + self[dim]) - other[dir]
190
191
      -- TODO: re-add hinting?
192
      if negMove < posMove then
193
         chg = - negMove
194
      else
195
         chg = posMove
196
      end
197
   end
198
199
   other[dir] = other[dir] + chg
200
end
201
2 by Josh C
basic tiles, map, player, movement
202
GameView = View:extend {
203
   onNew = function (self)
204
              self:loadLayers('data/map.lua')
205
              self.focus = the.player
206
              self:clampTo(self.map)
8 by Josh C
some basic collision (and workarounds)
207
           end,
208
   onUpdate = function (self)
209
                 --print('tick')
17 by Josh C
reorganize code - separate X and Y physics so we can collide in each
210
                 --the.player:collide(self.map)
8 by Josh C
some basic collision (and workarounds)
211
                 --self.map:collide(the.player)
212
              end
2 by Josh C
basic tiles, map, player, movement
213
}
214
215
the.app = App:new {
216
   onRun = function (self)
217
              self.view = GameView:new()
15 by Josh C
more reliable onGround calc
218
              self.console:watch('onGround', 'the.player.onGround')
16 by Josh C
try to track X collisions. break out Sprite's physics in prep for
219
              self.console:watch('onWall', 'the.player.onWall')
2 by Josh C
basic tiles, map, player, movement
220
           end,
221
   onUpdate = function (self, dt)
14 by Josh C
don't quit w/ escape if debug console is up
222
                 if the.keys:justPressed('escape') and 
223
                   not self.console.visible then
17 by Josh C
reorganize code - separate X and Y physics so we can collide in each
224
                    self.quit()
12 by Josh C
only jump when you're on the ground
225
                 end
2 by Josh C
basic tiles, map, player, movement
226
              end
227
}