forked from a327ex/BYTEPATH
-
Notifications
You must be signed in to change notification settings - Fork 1
/
main.lua
560 lines (488 loc) · 16.4 KB
/
main.lua
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
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
Steam = nil
-- require 'libraries/steamworks'
if type(Steam) == 'boolean' then Steam = nil end
Object = require 'libraries/classic/classic'
Timer = require 'libraries/enhanced_timer/EnhancedTimer'
Input = require 'libraries/boipushy/Input'
fn = require 'libraries/moses/moses'
Camera = require 'libraries/hump/camera'
Physics = require 'libraries/windfield'
Vector = require 'libraries/hump/vector-light'
draft = require('libraries/draft/draft')()
bitser = require 'libraries/bitser/bitser'
Math = require 'libraries/mlib/mlib'
Grid = require 'libraries/grid/grid'
Cam = require 'libraries/STALKER-X/Camera'
Tim = require 'libraries/chrono/Timer'
binser = require 'libraries/binser/binser'
HC = require 'libraries/HC'
ffi = require('ffi')
-- require 'enet'
require 'libraries/sound'
require 'libraries/utf8'
require 'GameObject'
require 'utils'
require 'globals'
require 'tree'
function isModuleAvailable(name)
if package.loaded[name] then
return true
else
for _, searcher in ipairs(package.searchers or package.loaders) do
local loader = searcher(name)
if type(loader) == 'function' then
package.preload[name] = loader
return true
end
end
return false
end
end
USING_GAMERZILLA = true
for i, cmd in ipairs(arg) do
if cmd == "--no-gamerzilla" then
USING_GAMERZILLA = false
end
end
if not isModuleAvailable("luagamerzilla") then
USING_GAMERZILLA = false
end
if USING_GAMERZILLA then
Gamerzilla = require "luagamerzilla"
Gamerzilla.start(false, love.filesystem.getSaveDirectory());
Gamerzilla.setGameFromFile("resources/gamerzilla/bytepath.game", "resources/");
end
function love.load()
time = 0
start_time = os.time()
start_date = os.date("*t")
trailer_mode = true
love.filesystem.setIdentity('BYTEPATH')
love.graphics.setDefaultFilter('nearest', 'nearest')
love.graphics.setLineStyle('rough')
love.graphics.setBackgroundColor(color255To1(background_color))
love.keyboard.setKeyRepeat(false)
loadFonts('resources/fonts')
loadGraphics('resources/graphics')
loadShaders('resources/shaders')
local object_files = {}
recursiveEnumerate('objects', object_files)
requireFiles(object_files)
local room_files = {}
recursiveEnumerate('rooms', room_files)
requireFiles(room_files)
timer = Timer()
input = Input()
camera = Camera()
sound()
--[[
input:bind('f1', function()
print("Before collection: " .. collectgarbage("count")/1024)
collectgarbage()
print("After collection: " .. collectgarbage("count")/1024)
print("Object count: ")
local counts = type_count()
for k, v in pairs(counts) do print(k, v) end
print("-------------------------------------")
end)
]]--
--[[
input:bind('f2', function() gotoRoom('Stage') end)
input:bind('f3', function()
if current_room then
current_room:destroy()
current_room = nil
end
end)
]]--
input:bind('left', 'left')
input:bind('right', 'right')
input:bind('up', 'up')
input:bind('down', 'down')
input:bind('a', 'left')
input:bind('d', 'right')
input:bind('w', 'up')
input:bind('s', 'down')
input:bind('mouse1', 'left_click')
input:bind('wheelup', 'zoom_in')
input:bind('wheeldown', 'zoom_out')
input:bind('return', 'return')
input:bind('backspace', 'backspace')
input:bind('escape', 'escape')
input:bind('dpleft', 'left')
input:bind('dpright', 'right')
input:bind('dpup', 'up')
input:bind('dpdown', 'down')
input:bind('fright', 'up')
input:bind('fdown', 'down')
input:bind('fleft', 'return')
input:bind('select', 'escape')
input:bind('start', 'return')
input:bind('tab', 'tab')
load()
if first_run_ever then resizeFullscreen()
else resize(sx, sy, fullscreen) end
current_room = nil
timer:after(0.5, function() gotoRoom('Console') end)
slow_amount = 1
fps = 60
disable_expensive_shaders = false
pre_disable_expensive_shaders = false
disable_expensive_shaders_time = 0
playRandomSong()
update_times = {}
update_index = 1
draw_times = {}
draw_index = 1
--[[
timer:every(1, function()
local update_sum = 0
for _, time in ipairs(update_times) do update_sum = update_sum + time end
local update_time = update_sum/#update_times
update_times = {}
update_index = 1
local draw_sum = 0
for _, time in ipairs(draw_times) do draw_sum = draw_sum + time end
local draw_time = draw_sum/#draw_times
draw_times = {}
draw_index = 1
print('update: ' .. 1000*update_time, 'draw: ' .. 1000*draw_time, 'total: ' .. 1000*(update_time + draw_time))
end)
]]--
end
function love.update(dt)
local start_time = os.clock()
time = time + dt
timer:update(dt*slow_amount)
camera:update(dt*slow_amount)
soundUpdate(dt*slow_amount)
if current_room then current_room:update(dt*slow_amount) end
-- Disable expensive shaders if FPS remains below 10 for 1 seconds
fps = love.timer.getFPS()
if fps < 10 and not pre_disable_expensive_shaders then
pre_disable_expensive_shaders = true
disable_expensive_shaders_time = love.timer.getTime()
end
if fps > 10 and pre_disable_expensive_shaders then pre_disable_expensive_shaders = false end
if love.timer.getTime() - disable_expensive_shaders_time > 3 and pre_disable_expensive_shaders then
disable_expensive_shaders = true
pre_disable_expensive_shaders = false
end
update_times[update_index] = os.clock() - start_time
update_index = update_index + 1
end
function love.draw()
local start_time = os.clock()
if current_room then current_room:draw() end
if flash_frames then
flash_frames = flash_frames - 1
if flash_frames == -1 then flash_frames = nil end
end
if flash_frames then
love.graphics.setColor(color255To1(background_color))
love.graphics.rectangle('fill', 0, 0, sx*gw, sy*gh)
love.graphics.setColor(color255To1(255, 255, 255))
end
draw_times[draw_index] = os.clock() - start_time
draw_index = draw_index + 1
end
function love.keypressed(key)
if current_room and current_room.keypressed then current_room:keypressed(key) end
end
function love.focus(f)
if not f then
if current_room and current_room:is(Stage) and not current_room.paused then
current_room:pause()
end
end
end
function love.quit()
save()
if USING_GAMERZILLA then Gamerzilla.quit() end
-- Steam already tracks how long each player plays for and since this is the only thing I'm tracking it's unnecessary to run a server for only that info
-- sendDataToServer(binser.serialize({id = id, duration = os.difftime(os.time(), start_time), start_date = start_date, end_date = os.date("*t")}))
end
function resize(x, y, fs)
local y = y or x
fullscreen = fs
love.window.setMode(x*gw, y*gh, {display = display, fullscreen = fs, borderless = fs})
sx, sy = x, y
end
function resizeFullscreen()
fullscreen = true
local w, h = love.window.getDesktopDimensions()
love.window.setMode(w, h, {display = display, fullscreen = true, borderless = true})
sx, sy = w/gw, h/gh
end
function changeToDisplay(n)
display = n
resize(getScaleBasedOnDisplay())
end
function getScaleBasedOnDisplay()
local w, h = love.window.getDesktopDimensions(display)
local sw, sh = math.floor(w/gw), math.floor(h/gh)
if sw == sh then return math.min(sw, sh) - 1
else return math.min(sw, sh) end
end
-- Effects --
function flash(frames)
flash_frames = frames
end
function slow(amount, duration)
slow_amount = amount
timer:tween('slow', duration, _G, {slow_amount = 1}, 'in-out-cubic')
end
function frameStop(duration, object_types)
if current_room then current_room.area:frameStop(duration, object_types) end
end
-- Save/Load --
function save()
local transient_save_data = {}
local permanent_save_data = {}
permanent_save_data.id = id
permanent_save_data.loop = loop
permanent_save_data.sx = sx
permanent_save_data.sy = sy
permanent_save_data.high_score = high_score
permanent_save_data.main_volume = main_volume
permanent_save_data.sfx_volume = sfx_volume
permanent_save_data.music_volume = music_volume
permanent_save_data.screen_shake = screen_shake
permanent_save_data.distortion = distortion
permanent_save_data.glitch = glitch
permanent_save_data.muted = muted
permanent_save_data.fullscreen = fullscreen
permanent_save_data.display = display
permanent_save_data.achievements = achievements
transient_save_data.skill_points = skill_points
transient_save_data.bought_node_indexes = bought_node_indexes
transient_save_data.run = run
transient_save_data.classes = classes
transient_save_data.rank = rank
transient_save_data.device = device
transient_save_data.unlocked_devices = unlocked_devices
transient_save_data.display = display
transient_save_data.keys = keys
transient_save_data.found_keys = found_keys
transient_save_data.max_tree_nodes = max_tree_nodes
transient_save_data.spent_sp = spent_sp
bitser.dumpLoveFile('permanent_save', permanent_save_data)
bitser.dumpLoveFile('transient_save', transient_save_data)
--[[
local data, len = bitser.dumps(save_data)
print('FileWrite: ' .. tostring(Steam.remotestorage.FileWrite('save', data, len)))
]]--
end
function loadAchievementsFromSteam()
print(Steam)
if Steam then
for _, achievement_name in ipairs(achievement_names) do
local steam_achievement_name = achievement_name:upper():gsub(' ', '_')
local b = ffi.new('bool[1]')
Steam.userstats.GetAchievement(steam_achievement_name, b)
achievements[achievement_name] = b[0]
-- print('Steam Achievement Load: ', achievement_name, b[0])
end
end
end
function load()
local loadPermanentVariables = function(save_data)
id = save_data.id
loop = save_data.loop
sx = save_data.sx
sy = save_data.sy
high_score = save_data.high_score
main_volume = save_data.main_volume
sfx_volume = save_data.sfx_volume
music_volume = save_data.music_volume
screen_shake = save_data.screen_shake
distortion = save_data.distortion
glitch = save_data.glitch
muted = save_data.muted
fullscreen = save_data.fullscreen
display = save_data.display
achievements = save_data.achievements
end
local loadTransientVariables = function(save_data)
skill_points = save_data.skill_points
bought_node_indexes = save_data.bought_node_indexes
run = save_data.run
classes = save_data.classes
rank = save_data.rank
device = save_data.device
unlocked_devices = save_data.unlocked_devices
keys = save_data.keys
found_keys = save_data.found_keys
max_tree_nodes = save_data.max_tree_nodes
spent_sp = save_data.spent_sp
end
local localLoad = function()
if love.filesystem.getInfo('permanent_save') then
local save_data = bitser.loadLoveFile('permanent_save')
loadPermanentVariables(save_data)
end
if love.filesystem.getInfo('transient_save') then
local save_data = bitser.loadLoveFile('transient_save')
loadTransientVariables(save_data)
else first_run_ever = true end
end
--[[
if Steam.remotestorage.FileExists('save') then
local file_size = Steam.remotestorage.GetFileSize('save')
local buffer = ffi.new("uint8_t[?]", file_size)
local read_amount = Steam.remotestorage.FileRead('save', buffer, file_size)
print('FileRead: ' .. read_amount)
if read_amount == 0 then
localLoad()
loadAchievementsFromSteam()
else
local save_data = bitser.loads(ffi.string(buffer, file_size))
loadVariables(save_data)
loadAchievementsFromSteam()
end
else
localLoad()
loadAchievementsFromSteam()
end
]]--
localLoad()
loadAchievementsFromSteam()
end
function removeSave()
-- print('FileDelete: ' .. tostring(Steam.remotestorage.FileDelete('save')))
love.filesystem.remove('transient_save')
end
-- Room --
function gotoRoom(room_type, ...)
if current_room and current_room.destroy then current_room:destroy() end
current_room = _G[room_type](...)
end
-- Load --
function recursiveEnumerate(folder, file_list)
local items = love.filesystem.getDirectoryItems(folder)
for _, item in ipairs(items) do
local file = folder .. '/' .. item
if love.filesystem.getInfo(file, "file") then
table.insert(file_list, file)
elseif love.filesystem.getInfo(file, "directory") then
recursiveEnumerate(file, file_list)
end
end
end
function requireFiles(files)
for _, file in ipairs(files) do
local file = file:sub(1, -5)
require(file)
end
end
function loadFonts(path)
fonts = {}
local font_paths = {}
recursiveEnumerate(path, font_paths)
for i = 8, 16, 1 do
for _, font_path in pairs(font_paths) do
local last_forward_slash_index = font_path:find("/[^/]*$")
local font_name = font_path:sub(last_forward_slash_index+1, -5)
local font = love.graphics.newFont(font_path, i)
font:setFilter('nearest', 'nearest')
fonts[font_name .. '_' .. i] = font
end
end
end
function loadGraphics(path)
assets = {}
local asset_paths = {}
recursiveEnumerate(path, asset_paths)
for _, asset_path in pairs(asset_paths) do
local last_forward_slash_index = asset_path:find("/[^/]*$")
local asset_name = asset_path:sub(last_forward_slash_index+1, -5)
local image = love.graphics.newImage(asset_path, {mipmaps = true})
assets[asset_name] = image
end
end
function loadShaders(path)
shaders = {}
local shader_paths = {}
recursiveEnumerate(path, shader_paths)
for _, shader_path in pairs(shader_paths) do
local last_forward_slash_index = shader_path:find("/[^/]*$")
local shader_name = shader_path:sub(last_forward_slash_index+1, -6)
local shader = love.graphics.newShader(shader_path)
shaders[shader_name] = shader
end
end
-- Memory --
function count_all(f)
local seen = {}
local count_table
count_table = function(t)
if seen[t] then return end
f(t)
seen[t] = true
for k,v in pairs(t) do
if type(v) == "table" then
count_table(v)
elseif type(v) == "userdata" then
f(v)
end
end
end
count_table(_G)
end
function type_count()
local counts = {}
local enumerate = function (o)
local t = type_name(o)
counts[t] = (counts[t] or 0) + 1
end
count_all(enumerate)
return counts
end
global_type_table = nil
function type_name(o)
if global_type_table == nil then
global_type_table = {}
for k,v in pairs(_G) do
global_type_table[v] = k
end
global_type_table[0] = "table"
end
return global_type_table[getmetatable(o) or 0] or "Unknown"
end
function love.run()
if love.math then love.math.setRandomSeed(os.time()) end
if love.load then love.load(arg) end
if love.timer then love.timer.step() end
local dt = 0
local fixed_dt = 1/60
local accumulator = 0
while true do
if love.event then
love.event.pump()
for name, a, b, c, d, e, f in love.event.poll() do
if name == 'quit' then
if not love.quit or not love.quit() then
return a
end
end
love.handlers[name](a, b, c, d, e, f)
end
end
if love.timer then
love.timer.step()
dt = love.timer.getDelta()
end
accumulator = accumulator + dt
while accumulator >= fixed_dt do
if love.update then love.update(fixed_dt) end
accumulator = accumulator - fixed_dt
end
if love.graphics and love.graphics.isActive() then
love.graphics.clear(love.graphics.getBackgroundColor())
love.graphics.origin()
if love.draw then love.draw() end
love.graphics.present()
end
if love.timer then love.timer.sleep(0.001) end
end
end