mirror of
https://github.com/britzl/monarch.git
synced 2025-06-27 18:37:46 +02:00
The transition module will now properly handle if the same transition is started more than once before completed. In such a case only one animation will be played but all senders will be notified when completed.
162 lines
4.8 KiB
Lua
162 lines
4.8 KiB
Lua
local monarch = require "monarch.monarch"
|
|
|
|
local M = {}
|
|
|
|
local WIDTH = tonumber(sys.get_config("display.width"))
|
|
local HEIGHT = tonumber(sys.get_config("display.height"))
|
|
|
|
local LEFT = vmath.vector3(-WIDTH * 2, 0, 0)
|
|
local RIGHT = vmath.vector3(WIDTH * 2, 0, 0)
|
|
local TOP = vmath.vector3(0, HEIGHT * 2, 0)
|
|
local BOTTOM = vmath.vector3(0, - HEIGHT * 2, 0)
|
|
|
|
local ZERO_SCALE = vmath.vector3(0, 0, 1)
|
|
|
|
function M.instant(node, to, easing, duration, delay, cb)
|
|
cb()
|
|
end
|
|
|
|
local function slide_in(direction, node, to, easing, duration, delay, cb)
|
|
local from = to + direction
|
|
gui.set_position(node, from)
|
|
gui.animate(node, gui.PROP_POSITION, to, easing, duration, delay, cb)
|
|
end
|
|
|
|
function M.slide_in_left(node, to, easing, duration, delay, cb)
|
|
return slide_in(LEFT, node, to.pos, easing, duration, delay, cb)
|
|
end
|
|
|
|
function M.slide_in_right(node, to, easing, duration, delay, cb)
|
|
slide_in(RIGHT, node, to.pos, easing, duration, delay, cb)
|
|
end
|
|
|
|
function M.slide_in_top(node, to, easing, duration, delay, cb)
|
|
slide_in(TOP, node, to.pos, easing, duration, delay, cb)
|
|
end
|
|
|
|
function M.slide_in_bottom(node, to, easing, duration, delay, cb)
|
|
slide_in(BOTTOM, node, to.pos, easing, duration, delay, cb)
|
|
end
|
|
|
|
|
|
local function slide_out(direction, node, from, easing, duration, delay, cb)
|
|
local to = from + direction
|
|
gui.set_position(node, from)
|
|
gui.animate(node, gui.PROP_POSITION, to, easing, duration, delay, cb)
|
|
end
|
|
|
|
function M.slide_out_left(node, from, easing, duration, delay, cb)
|
|
slide_out(LEFT, node, from.pos, easing, duration, delay, cb)
|
|
end
|
|
|
|
function M.slide_out_right(node, from, easing, duration, delay, cb)
|
|
slide_out(RIGHT, node, from.pos, easing, duration, delay, cb)
|
|
end
|
|
|
|
function M.slide_out_top(node, from, easing, duration, delay, cb)
|
|
slide_out(TOP, node, from.pos, easing, duration, delay, cb)
|
|
end
|
|
|
|
function M.slide_out_bottom(node, from, easing, duration, delay, cb)
|
|
slide_out(BOTTOM, node, from.pos, easing, duration, delay, cb)
|
|
end
|
|
|
|
function M.scale_in(node, to, easing, duration, delay, cb)
|
|
gui.set_scale(node, ZERO_SCALE)
|
|
gui.animate(node, gui.PROP_SCALE, to.scale, easing, duration, delay, cb)
|
|
end
|
|
|
|
function M.scale_out(node, from, easing, duration, delay, cb)
|
|
gui.set_scale(node, from.scale)
|
|
gui.animate(node, gui.PROP_SCALE, ZERO_SCALE, easing, duration, delay, cb)
|
|
end
|
|
|
|
--- Create a transition for a node
|
|
-- @return Transition instance
|
|
function M.create(node)
|
|
assert(node, "You must provide a node")
|
|
|
|
local instance = {}
|
|
|
|
local transitions = {}
|
|
|
|
local initial_data = {}
|
|
initial_data.pos = gui.get_position(node)
|
|
initial_data.scale = gui.get_scale(node)
|
|
|
|
local function create_transition(fn, easing, duration, delay)
|
|
return {
|
|
fn = fn,
|
|
easing = easing,
|
|
duration = duration,
|
|
delay = delay,
|
|
in_progress = false,
|
|
urls = {},
|
|
}
|
|
end
|
|
|
|
local function start_transition(transition, url)
|
|
table.insert(transition.urls, url)
|
|
if not transition.in_progress then
|
|
transition.in_progress = true
|
|
transition.fn(node, initial_data, transition.easing, transition.duration, transition.delay or 0, function()
|
|
transition.in_progress = false
|
|
while #transition.urls > 0 do
|
|
local url = table.remove(transition.urls)
|
|
msg.post(url, monarch.TRANSITION.DONE)
|
|
end
|
|
end)
|
|
end
|
|
end
|
|
|
|
-- Forward on_message calls here
|
|
function instance.handle(message_id, message, sender)
|
|
local transition = transitions[message_id]
|
|
if transition then
|
|
start_transition(transition, sender)
|
|
end
|
|
end
|
|
|
|
-- Specify the transition function when this node is transitioned
|
|
-- to
|
|
-- @param fn Transition function (see slide_in_left and other above)
|
|
-- @param easing Easing function to use
|
|
-- @param duration Transition duration
|
|
-- @param delay Transition delay
|
|
function instance.show_in(fn, easing, duration, delay)
|
|
transitions[monarch.TRANSITION.SHOW_IN] = create_transition(fn, easing, duration, delay)
|
|
return instance
|
|
end
|
|
|
|
-- Specify the transition function when this node is transitioned
|
|
-- from when showing another screen
|
|
function instance.show_out(fn, easing, duration, delay)
|
|
transitions[monarch.TRANSITION.SHOW_OUT] = create_transition(fn, easing, duration, delay)
|
|
return instance
|
|
end
|
|
|
|
--- Specify the transition function when this node is transitioned
|
|
-- to when navigating back in the screen stack
|
|
function instance.back_in(fn, easing, duration, delay)
|
|
transitions[monarch.TRANSITION.BACK_IN] = create_transition(fn, easing, duration, delay)
|
|
return instance
|
|
end
|
|
|
|
--- Specify the transition function when this node is transitioned
|
|
-- from when navigating back in the screen stack
|
|
function instance.back_out(fn, easing, duration, delay)
|
|
transitions[monarch.TRANSITION.BACK_OUT] = create_transition(fn, easing, duration, delay)
|
|
return instance
|
|
end
|
|
|
|
-- set default transitions (instant)
|
|
instance.show_in(M.instant)
|
|
instance.show_out(M.instant)
|
|
instance.back_in(M.instant)
|
|
instance.back_out(M.instant)
|
|
|
|
return instance
|
|
end
|
|
|
|
return M
|