cube-tube/app/util/scene.rb
2023-04-11 16:26:39 +10:00

73 lines
2.5 KiB
Ruby

# frozen_string_literal: true
# A scene represents a discreet state of gameplay. Things like the main menu,
# game over screen, and gameplay.
#
# Define a new scene by adding one to `app/scenes/` and inheriting from
# SceneInstance
#
# The main `#tick` of the game handles delegating to the current scene based on
# the `args.state.scene_stack` value, which contains the current scene, as well
# as scenes that can be "popped" back to.
module Scene
class << self
# Change the current scene, and optionally reset the scene that's begin
# changed to so any data is cleared out
# ex:
# Scene.switch(args, :gameplay)
def switch(args, scene, reset: false, push_or_pop: false)
args.state.scene_stack ||= []
# if we're here /not/ from push or pop, clear the scene stack
args.state.scene_stack.clear unless push_or_pop
# if `scene` is not a `SceneInstance`, it's probably a symbol representing
# the scene we're switching to, so go get it.
the_scene = scene.is_a?(SceneInstance) ? scene : SCENES[scene].new(args)
puts '---'
puts 'switching to'
puts scene unless scene.is_a?(SceneInstance)
puts SCENES[scene] unless scene.is_a?(SceneInstance)
puts the_scene
puts '---'
# if the stack is empty (e.g. we just cleared it), then push this scene
args.state.scene_stack.push(the_scene) if args.state.scene_stack.empty?
# if we asked to reset the scene, then do so.
the_scene.reset(args) if reset
raise FinishTick, 'finish tick early'
end
# Change the current scene by pushing it onto the scene stack
def push(args, scene, reset: false, reset_on_pop: false)
args.state.scene_stack ||= []
prev_scene = args.state.scene_stack.last
prev_scene.reset_on_pop = reset_on_pop unless prev_scene.nil?
the_scene = scene.is_a?(SceneInstance) ? scene : SCENES[scene].new(args)
args.state.scene_stack.push(the_scene)
switch(args, the_scene, reset: reset, push_or_pop: true)
end
# Return to the previous scene on the stack
def pop(args, reset: nil)
args.state.scene_stack&.pop
scene = args.state.scene_stack.last
scene = scene._?(default(args))
reset = scene.reset_on_pop if reset.nil?
puts reset, scene, scene.reset_on_pop
switch(args, scene, reset: reset, push_or_pop: true)
end
def default(args)
args.state.scene_stack ||= []
SCENES[:DEFAULT].new(args)
end
def stack(args)
args.state.scene_stack ||= []
end
end
end