your-own-drum/main.lua

428 lines
9.9 KiB
Lua
Raw Permalink Normal View History

2023-01-13 13:11:30 +00:00
local love = love
2023-01-14 18:47:46 +00:00
local step = 1.0 / 120
2023-01-13 18:28:58 +00:00
local sitelenpona
local text
2023-01-13 20:05:19 +00:00
local marble
local wave
2023-01-15 13:39:58 +00:00
local particles
local audio
local recorder
local DetectCollision
2023-01-15 13:39:58 +00:00
local OnImpact
local OnVictory
local BeatScore
local Draw
local Update
local ExtrapolateBeatScore
local _ExtrapolateBeatScore
local mouseControl
local scores
2023-01-13 18:28:58 +00:00
local state
state = {
Reset = function()
2023-01-18 22:51:03 +00:00
state.tick = 0
2023-01-13 18:28:58 +00:00
state.beat = {}
state.currentBeat = 1
2023-01-13 20:05:19 +00:00
state.timeToSimulate = 0.0
state.longestStreak = 0
state.currentStreak = 0
2023-01-20 13:53:24 +00:00
state.lastCollisionTick = 0
state.lastCollisionX = 0
state.lastCollisionY = 0
2023-01-13 18:28:58 +00:00
end,
2023-01-13 15:38:14 +00:00
2023-01-13 20:05:19 +00:00
timeToSimulate = 0.0,
lastBeatScore = 0.0,
2023-01-13 16:43:12 +00:00
currentBeat = 1,
longestStreak = 0,
currentStreak = 0,
2023-01-13 15:38:14 +00:00
beat = {
t = nil,
mu = nil,
},
}
local function UpdateWindowTransform( w, h )
local d = math.min( w, h )
local r = love.graphics.get
local tf = love.math.newTransform()
local size = 3
tf:translate( w / 2, h / 2)
tf:scale( d / size, -d / size )
transform = tf
end
OnImpact = function( impact )
if not impact then return end
local score = BeatScore( impact.t )
--DEBUG
state.lastBeatScore = score
2023-01-20 13:53:24 +00:00
state.lastCollisionTick = state.tick
local pass = false
if score > 1.0 then
pass = true
state.currentStreak = state.currentStreak + 1
if state.currentStreak > state.longestStreak then
state.longestStreak = state.currentStreak
end
state.currentBeat = state.currentBeat + 1
if state.currentBeat >= 119 then
return OnVictory()
end
else
state.currentStreak = 0
end
scores.OnImpact( state.tick, pass )
local x, y = impact.r * math.cos(impact.th), impact.r * math.sin(impact.th)
particles:setPosition( x, y )
particles:setEmissionArea( "normal", 0.1, 0.2, impact.th, true )
particles:emit( 50 * score * score )
state.lastCollisionX, state.lastCollisionY = transform:transformPoint( 1.2 * x, 1.2 * y )
audio.OnImpact( impact, score, pass, state.currentBeat )
marble.OnImpact( impact, state.currentBeat )
wave.OnImpact( impact, state.currentBeat )
end
local _OnImpact = OnImpact
2023-01-20 22:38:06 +00:00
local function NewGame( demoName )
2023-01-13 20:05:19 +00:00
love.graphics.setBackgroundColor( 245 / 255, 169 / 255, 184 / 255 ) --Trans pink.
OnImpact = _OnImpact
ExtrapolateBeatScore = _ExtrapolateBeatScore
love.draw = Draw
love.update = Update
2023-01-20 22:38:06 +00:00
do --particle shit
particles:reset()
particles:setSizes( 0.0007, 0.0001, 0.0003 )
2023-01-20 22:38:06 +00:00
--particles:setSizes( 0.0007, 0.0001, 0.0003 )
particles:setSizeVariation( 1 )
particles:setRadialAcceleration( 0, 0.5 )
particles:setSpeed( 0.2, 1 )
particles:setLinearDamping( 1, 10 )
particles:setRelativeRotation( true )
particles:setEmitterLifetime( -1 )
particles:setParticleLifetime( 0, 5 )
particles:setSpread( 0.05 )
particles:setColors(
1, 1, 1, 1,
91 / 255, 206 / 255, 250 / 255, 1,
245 / 255, 169 / 255, 184 / 255, 1,
1,1,1,0,
245 / 255, 169 / 255, 184 / 255, 1,
1,1,1,0,
245 / 255, 169 / 255, 184 / 255, 1,
1,1,1,0)
end
mouseControl.Reset()
state.Reset()
marble.Reset()
wave.Reset()
text.Reset()
audio.Reset()
2023-01-18 22:51:03 +00:00
recorder.Reset()
scores.Reset()
2023-01-18 22:51:03 +00:00
2023-01-20 22:38:06 +00:00
state.isDemo = demoName
if demoName then
recorder.Load( demoName )
2023-01-18 22:51:03 +00:00
end
end
function love.load()
love.mouse.setRelativeMode( true )
UpdateWindowTransform( love.graphics.getDimensions() )
2023-01-15 13:39:58 +00:00
2023-01-15 13:39:58 +00:00
do--particle system setup
particles = love.graphics.newParticleSystem(
love.graphics.newImage( "prideflag.png" ),
2023-01-20 22:38:06 +00:00
2048)
2023-01-15 13:39:58 +00:00
2023-01-20 22:38:06 +00:00
2023-01-15 13:39:58 +00:00
end
loadGame = assert( require "loadgame" )
options = assert( require( "options" ) )
mouseControl = assert( require "mousecontrols" )
2023-01-13 18:28:58 +00:00
sitelenpona = assert( require "sitelenpona" )
text = assert( require "text" )
2023-01-13 20:05:19 +00:00
marble = assert( require "marble" )
wave = assert( require "wave" )
DetectCollision = assert ( require "collision" )
2023-01-15 13:39:58 +00:00
audio = assert( require "audio" )
recorder = assert( require "recorder" )
scores = assert( require "scores" )
return NewGame()
2023-01-13 15:38:14 +00:00
2023-01-13 14:27:17 +00:00
end
ExtrapolateBeatScore = function( )
2023-01-20 22:38:06 +00:00
local t = state.tick / 120.0
2023-01-15 13:39:58 +00:00
local beat = state.beat
if not beat.t then return 2.0 end
if not beat.mu then return 2.0 end
if beat.mu < 0.001 then return 2.0 end
local pow = 1 / math.max( beat.mu, 0.2 )
t = 1.05 * math.sin( 0.5 * math.pi * ( t - beat.t ) / beat.mu )
return math.pow( t * t, pow )
2023-01-15 13:39:58 +00:00
end
_ExtrapolateBeatScore = ExtrapolateBeatScore
2023-01-13 18:28:58 +00:00
BeatScore = function( t )
2023-01-13 16:43:12 +00:00
local beat = state.beat
--Base case 1: first tap.
2023-01-13 15:38:14 +00:00
if not beat.t then
beat.t = t
2023-01-13 16:43:12 +00:00
return 2.0
2023-01-13 15:38:14 +00:00
end
2023-01-13 14:27:17 +00:00
2023-01-13 16:43:12 +00:00
local dt = t - beat.t
beat.t = t
2023-01-13 14:27:17 +00:00
2023-01-13 16:43:12 +00:00
--Base case 2: second tap.
if not beat.mu then
beat.mu = dt
return 2.0
end
2023-01-13 14:27:17 +00:00
local pow = 1 / math.max( beat.mu, 0.1 )
local score = 1.1 * math.sin( 0.5 * math.pi * dt / beat.mu )
score = math.pow( score * score, pow )
2023-01-13 15:38:14 +00:00
2023-01-15 13:39:58 +00:00
--General case: update average beat length.
--Debounce: max BPM 200
if dt > 60.0 / 200.0 then
2023-01-13 15:38:14 +00:00
local WEIGHT = 0.75 --High number makes the last beat more significant.
2023-01-15 13:39:58 +00:00
local mu = ( 1.0 - WEIGHT ) * beat.mu + WEIGHT * dt
beat.mu = mu
2023-01-13 15:38:14 +00:00
else
score = 0
2023-01-15 13:39:58 +00:00
end
2023-01-13 15:38:14 +00:00
2023-01-15 13:39:58 +00:00
return score
2023-01-13 16:43:12 +00:00
end
2023-01-13 15:38:14 +00:00
2023-01-15 13:39:58 +00:00
OnVictory = function()
2023-01-20 13:53:24 +00:00
if state.isDemo then
state.isDemo = false
else
scores.Save()
recorder.Save( scores.Get(), state.tick)
2023-01-20 13:53:24 +00:00
end
2023-01-18 22:51:03 +00:00
particles:setParticleLifetime( 0, 30 )
particles:setSizes( 0.001, 0.0001, 0.0005 )
particles:setColors(
245 / 255, 169 / 255, 184 / 255, 1,
1, 1, 1, 1,
245 / 255, 169 / 255, 184 / 255, 1,
1, 1, 1, 0,
245 / 255, 169 / 255, 184 / 255, 1,
1, 1, 1, 0
)
particles:setPosition( 0, 0 )
particles:setEmissionArea( "normal", 0.5, 0.5, 0, true )
particles:emit( 500 )
particles:setEmissionArea( "normal", 0.01, 0.01, 0, true )
love.graphics.setCanvas( marble.Canvas() )
Draw()
love.graphics.setCanvas()
2023-01-20 13:53:24 +00:00
local totalTime = state.tick / 120.0
2023-01-15 13:39:58 +00:00
OnImpact = function() end
love.update = function( dt )
while dt > step do
marble.Integrate( step )
wave.Integrate( step )
marble.Update()
wave.Update()
dt = dt - step
end
local marblePos = marble.Current()
particles:emit( 1 )
particles:update( dt )
particles:moveTo( marblePos.x, marblePos.y )
end
love.draw = function()
love.graphics.push( "transform" )
love.graphics.applyTransform( transform )
love.graphics.setColor( 1, 1, 1, 1 )
love.graphics.draw( particles )
love.graphics.pop()
text.Draw( 119 )
scores.RenderHighScores()
2023-01-18 22:51:03 +00:00
2023-01-16 15:09:22 +00:00
local score = state.longestStreak * state.longestStreak / totalTime
marble.Draw()
end
marble.OnVictory()
love.graphics.setBackgroundColor( 91 / 255, 206 / 255, 250 / 255 ) --Trans blue.
2023-01-15 13:39:58 +00:00
end
2023-01-13 15:38:14 +00:00
Draw = function()
2023-01-13 21:34:01 +00:00
2023-01-15 13:39:58 +00:00
local score = ExtrapolateBeatScore()
love.graphics.push( "transform" )
love.graphics.applyTransform( transform )
2023-01-15 13:39:58 +00:00
love.graphics.setColor( 1.0, 1.0, 1.0, 1.0 )
love.graphics.draw(particles, 0, 0)
wave.Draw( score )
2023-01-15 13:39:58 +00:00
love.graphics.pop()
sitelenpona.Draw( text.words[state.currentBeat] )
2023-01-13 20:05:19 +00:00
marble.Draw()
text.Draw( state.currentBeat )
2023-01-20 13:53:24 +00:00
love.graphics.setColor( 0.3, 0.3, 0.3, 1.0 - score )
local s = "+"..state.currentStreak..(state.currentStreak > 50 and "!" or "")
love.graphics.print( s, state.lastCollisionX, state.lastCollisionY, 0, 0.5 - 0.1 * score, 0.5 - 0.1 * score)
2023-01-13 16:43:12 +00:00
end
Update = function( dt )
2023-01-15 13:39:58 +00:00
audio.Update( ExtrapolateBeatScore(), state.currentBeat )
2023-01-15 13:39:58 +00:00
particles:update( dt )
2023-01-13 20:05:19 +00:00
dt = dt + state.timeToSimulate
2023-01-15 13:39:58 +00:00
--Physics tick.
2023-01-13 20:05:19 +00:00
while dt > step do
2023-01-18 22:51:03 +00:00
state.tick = state.tick + 1
if mouseControl.isActive then
mouseControl.update()
marble.SetAcceleration( mouseControl.getAcceleration() )
end
2023-01-18 22:51:03 +00:00
if state.isDemo then
local ddx, ddy = recorder.NextTick()
if ddx then marble.SetAcceleration( ddx, ddy )
else state.isDemo = false end
else
recorder.Update( marble.GetAcceleration() ) --For savegames.
end
2023-01-15 13:39:58 +00:00
marble.Integrate( step )
wave.Integrate( step )
2023-01-15 13:39:58 +00:00
OnImpact(
DetectCollision(
marble.Current(), marble.Next(),
wave.Current(), wave.Next() ))
2023-01-15 13:39:58 +00:00
marble.Update()
wave.Update()
2023-01-15 13:39:58 +00:00
2023-01-13 20:05:19 +00:00
dt = dt - step
2023-01-13 18:28:58 +00:00
end
state.timeToSimulate = dt
2023-01-13 18:28:58 +00:00
end
_Update = Update
2023-01-13 14:27:17 +00:00
local function OptionsMenu()
if not options then options = require( "options" ) end
return options.EnterMenu()
end
2023-01-20 22:38:06 +00:00
local function LoadGame() -- Load game screen.
local Click, Press = assert(loadGame.OnClick), assert(loadGame.KeyPress)
loadGame.LoadGameMenu()
2023-01-20 22:38:06 +00:00
love.mousepressed = function( x, y, button, istouch, presses)
local demoName = Click( x, y, button, istouch, presses )
if demoName then return NewGame( demoName ) end
end
love.keypressed = function( key, code, isRepeat)
local demoName = Press( key, code, isRepeat )
2023-01-20 22:38:06 +00:00
if demoName then return NewGame( demoName ) end
end
2023-01-20 22:38:06 +00:00
end
2023-01-13 16:43:12 +00:00
function love.keypressed( key, code, isRepeat )
if code == "escape" then return love.event.quit() end
if code == "space" then return NewGame() end
2023-01-20 22:38:06 +00:00
if code == "return" then return LoadGame() end --Play demo.
if code == "o" then return OptionsMenu() end
2023-01-18 22:51:03 +00:00
if state.isDemo then return end
mouseControl.isActive = false
return marble.OnKey(
love.keyboard.isScancodeDown( options.up.value ),
love.keyboard.isScancodeDown( options.left.value ),
love.keyboard.isScancodeDown( options.down.value ),
love.keyboard.isScancodeDown( options.right.value )
)
2023-01-13 18:28:58 +00:00
end
function love.keyreleased( key, code )
return marble.OnKey(
love.keyboard.isScancodeDown( options.up.value),
love.keyboard.isScancodeDown( options.left.value),
love.keyboard.isScancodeDown( options.down.value),
love.keyboard.isScancodeDown( options.right.value) )
end
function love.resize( w, h )
UpdateWindowTransform( w, h )
if marble then marble.Resize() end
end
--TODO: make this feel better.
function love.mousemoved( x, y, dx, dy, number, istouch )
mouseControl.isActive = true
dx, dy = mouseControl.mousemoved( dx, dy )
if dx then return marble.SetAcceleration( dx, dy ) end
end
function love.mousepressed( x, y, button, istouch, presses )
return NewGame()
2023-01-13 16:43:12 +00:00
end