5
5
__ = require 'underscore'
6
6
--inspect = require 'inspect'
8
Player = Animation:extend {
9
image = 'data/player.png',
13
stand = { frames = { 1 }, fps = 1 },
14
walk = { frames = { 2, 3 }, fps = 5 },
15
jump = { frames = { 4 }, fps = 1 }
19
onNew = function (self)
21
self.maxVelocity.y = 400
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
30
-- check existence of properties
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')
38
assert(__.include({'x','y','rotation'}, dir), 'direction should be x, y, or rotation')
43
vel.rotation = vel.rotation or 0
47
if acc[dir] and acc[dir] ~= 0 then
48
vel[dir] = vel[dir] + acc[dir] * elapsed
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
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
64
if vel[dir] ~= 0 then self[dir] = self[dir] + vel[dir] * elapsed end
66
onStartFrame = function (self)
67
-- this is all in startframe so it happens before
68
-- physics calc at beginning of update
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)
75
if (not self.onGround) and (not self.onWall) then
79
self.acceleration.y = 800
82
self.acceleration.y = 0
84
if the.keys:pressed('up') then
85
self.velocity.y = -200
87
elseif the.keys:pressed('down') then
96
if the.keys:pressed('left') then
97
self.velocity.x = -200
98
if self.onGround then self:play('walk') end
99
if self.onWall == 'right' then self.onWall = false end
100
elseif the.keys:pressed('right') then
101
self.velocity.x = 200
102
if self.onGround then self:play('walk') end
103
if self.onWall == 'left' then self.onWall = false end
105
if self.onGround then self:play('stand') end
106
if not self.onWall then
111
if the.keys:justPressed('up') and self.onGround then
112
self.velocity.y = -400
116
update = function (self, elapsed)
117
-- NOTE: this is an override, not a callback
119
self:doPhysics('x', elapsed)
120
self:collide(the.view.map)
122
-- handle X collisions
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
135
self.onGround = false -- right before Y collision callbacks
136
self:doPhysics('y', elapsed)
137
self:collide(the.view.map)
139
-- handle Y collisions
140
for _, col in ipairs(self.collisions) do
141
if self.velocity.y > 0 then
145
col.other:displaceDir(self, 'y')
150
Animation.update(self, elapsed)
152
collide = function (self, ...)
154
Animation.collide(self, ...)
155
-- I could return a true/false value here if I wanted to...
157
onCollide = function (self, other, xOverlap, yOverlap)
158
if other == the.view.map then return end
160
table.insert(self.collisions, {other = other,
162
yOverlap = yOverlap })
166
function Sprite:displaceDir(other, dir)
167
if not self.solid or self == other or not other.solid then return end
168
if STRICT then assert(other:instanceOf(Sprite), 'asked to displace a non-sprite') end
170
if other.sprites then
173
for _, spr in pairs(other.sprites) do
174
self:displace(spr, dir)
179
15
if dir == 'x' then
181
17
elseif dir == 'y' then
187
local negMove = (other[dir] - self[dir]) + other[dim]
188
local posMove = (self[dir] + self[dim]) - other[dir]
190
-- TODO: re-add hinting?
191
if negMove < posMove then
20
if STRICT then error('dir '..dir) end
198
other[dir] = other[dir] + chg
201
25
GameView = View:extend {
202
26
onNew = function (self)
203
27
self:loadLayers('data/map.lua')
204
28
self.focus = the.player
205
29
self:clampTo(self.map)
31
the.recorder = Recorder:new{mousePosInterval = 9999}
32
the.app.meta:add(the.recorder)
33
if the.app.record then
34
the.recorder:startRecording()
35
elseif the.app.playback then
36
local storage = Storage:new{filename = 'record.lua'}
38
--print(inspect(storage.data))
39
the.recorder.record = storage.data
40
the.recorder:startPlaying()
207
43
onUpdate = function (self)
44
--print('drawTook: ', the.drawTook)
209
46
--the.player:collide(self.map)
210
47
--self.map:collide(the.player)
49
-- draw = function (self, x, y)
50
-- View.draw(self, x, y)
52
-- love.graphics.print('FPS:' .. love.timer.getFPS(), 20, 20)
214
56
the.app = App:new {
215
58
onRun = function (self)
216
59
self.view = GameView:new()
217
self.console:watch('onGround', 'the.player.onGround')
218
self.console:watch('onWall', 'the.player.onWall')
61
self.console:watch('onGround', 'the.player.onGround')
62
self.console:watch('onWall', 'the.player.onWall')
63
self.console:watch('updateTook', 'the.updateTook')
64
self.console:watch('drawTook', 'the.drawTook')
65
self.console:watch('recorder state', 'the.recorder.state')
68
--the.profiler = newProfiler('time', 2000)
69
--the.profiler = newProfiler()
70
--the.profiler:start()
220
72
onUpdate = function (self, dt)
221
if the.keys:justPressed('escape') and
222
not self.console.visible then
73
if the.keys:justPressed('escape') then
76
local outfile = io.open( "profile.txt", "w+" )
77
the.profiler:report( outfile )
82
if not love.filesystem.remove('record.lua') then
83
print('could not remove record.lua')
85
local storage = Storage:new{
86
data = the.recorder.record,
87
filename = 'record.lua'
90
--print(inspect(the.recorder.record))
b'\\ No newline at end of file'
96
update = function (self, dt)
97
the.updateStart = love.timer.getMicroTime()
99
if the.updateStart then
100
the.updateTook = love.timer.getMicroTime() - the.updateStart
105
function love.load (arg)
106
opts = getopt(arg, '')
108
the.app.playback = true
109
the.app.record = false
110
elseif opts['r'] then
111
the.app.record = true
b'\\ No newline at end of file'