2023-03-27 02:09:28 +00:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2023-03-08 03:06:29 +00:00
|
|
|
# A scene represents a discreet state of gameplay. Things like the main menu,
|
|
|
|
# game over screen, and gameplay.
|
|
|
|
#
|
2023-03-27 02:09:28 +00:00
|
|
|
# Define a new scene by adding one to `app/scenes/` and inheriting from
|
|
|
|
# SceneInstance
|
2023-03-08 03:06:29 +00:00
|
|
|
#
|
|
|
|
# The main `#tick` of the game handles delegating to the current scene based on
|
2023-03-27 03:13:39 +00:00
|
|
|
# the `args.state.scene_stack` value, which contains the current scene, as well
|
|
|
|
# as scenes that can be "popped" back to.
|
2023-03-08 03:06:29 +00:00
|
|
|
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)
|
2023-03-21 02:40:53 +00:00
|
|
|
def switch(args, scene, reset: false, push_or_pop: false)
|
2023-03-27 02:09:28 +00:00
|
|
|
args.state.scene_stack ||= []
|
2023-03-21 02:40:53 +00:00
|
|
|
# if we're here /not/ from push or pop, clear the scene stack
|
|
|
|
args.state.scene_stack.clear unless push_or_pop
|
2023-03-27 06:03:10 +00:00
|
|
|
|
2023-03-27 03:13:39 +00:00
|
|
|
# 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)
|
2023-07-24 01:47:54 +00:00
|
|
|
# puts '---'
|
|
|
|
# puts 'switching to'
|
|
|
|
# puts scene unless scene.is_a?(SceneInstance)
|
|
|
|
# puts SCENES[scene] unless scene.is_a?(SceneInstance)
|
|
|
|
# puts the_scene
|
|
|
|
# puts '---'
|
2023-03-27 03:13:39 +00:00
|
|
|
|
|
|
|
# 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?
|
2023-03-08 03:06:29 +00:00
|
|
|
|
2023-03-27 03:13:39 +00:00
|
|
|
# if we asked to reset the scene, then do so.
|
|
|
|
the_scene.reset(args) if reset
|
2023-03-08 03:06:29 +00:00
|
|
|
|
2023-04-18 05:51:43 +00:00
|
|
|
# raise FinishTick, 'finish tick early'
|
2023-03-21 02:40:53 +00:00
|
|
|
end
|
|
|
|
|
2023-03-27 03:13:39 +00:00
|
|
|
# Change the current scene by pushing it onto the scene stack
|
2023-04-11 06:26:39 +00:00
|
|
|
def push(args, scene, reset: false, reset_on_pop: false)
|
2023-03-21 02:40:53 +00:00
|
|
|
args.state.scene_stack ||= []
|
2023-04-11 06:26:39 +00:00
|
|
|
prev_scene = args.state.scene_stack.last
|
|
|
|
prev_scene.reset_on_pop = reset_on_pop unless prev_scene.nil?
|
2023-03-27 02:09:28 +00:00
|
|
|
the_scene = scene.is_a?(SceneInstance) ? scene : SCENES[scene].new(args)
|
|
|
|
args.state.scene_stack.push(the_scene)
|
2023-03-21 02:40:53 +00:00
|
|
|
|
2023-03-27 02:09:28 +00:00
|
|
|
switch(args, the_scene, reset: reset, push_or_pop: true)
|
2023-03-21 02:40:53 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
# Return to the previous scene on the stack
|
2023-04-11 06:26:39 +00:00
|
|
|
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?
|
2023-07-24 01:47:54 +00:00
|
|
|
# puts reset, scene, scene.reset_on_pop
|
2023-03-27 02:09:28 +00:00
|
|
|
|
2023-04-11 06:26:39 +00:00
|
|
|
switch(args, scene, reset: reset, push_or_pop: true)
|
2023-03-27 02:09:28 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
def default(args)
|
|
|
|
args.state.scene_stack ||= []
|
|
|
|
SCENES[:DEFAULT].new(args)
|
2023-03-08 03:06:29 +00:00
|
|
|
end
|
2023-03-27 06:03:10 +00:00
|
|
|
|
|
|
|
def stack(args)
|
|
|
|
args.state.scene_stack ||= []
|
|
|
|
end
|
2023-03-08 03:06:29 +00:00
|
|
|
end
|
|
|
|
end
|