1
0
mirror of https://github.com/Refactorio/RedMew.git synced 2024-12-12 10:04:40 +02:00

Merge pull request #689 from linaori/toasting

Added a toast feature
This commit is contained in:
Lynn 2019-01-27 12:18:52 +01:00 committed by GitHub
commit 9d73ff8895
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

198
features/gui/toast.lua Normal file
View File

@ -0,0 +1,198 @@
local Event = require 'utils.event'
local Game = require 'utils.game'
local Global = require 'utils.global'
local Gui = require 'utils.gui'
local Color = require 'resources.color_presets'
local type = type
local tonumber = tonumber
local pairs = pairs
local size = table.size
local Public = {}
local memory = {
id = 0,
active_toasts = {},
}
Global.register(memory, function (tbl) memory = tbl end, 'toast')
---Creates a unique ID for a toast message
local function autoincrement()
local id = memory.id + 1
memory.id = id
return id
end
---Toast to a specific player
---@param p number|string|LuaPlayer player index or object
---@param duration number in seconds
local function toast_to(p, duration)
local player
if type(p) == 'string' then
player = Game.get_player_by_index(tonumber(p))
elseif type(p) == 'number' then
player = Game.get_player_by_index(p)
else
player = p
end
if not player or not player.valid then
return nil
end
local frame = player.gui.left.add({type = 'frame', direction = 'vertical', style = 'captionless_frame'})
frame.style.width = 300
local container = frame.add({type = 'flow', direction = 'horizontal'})
container.style.horizontally_stretchable = true
local progressbar = frame.add({type = 'progressbar'})
local style = progressbar.style
style.width = 290
style.height = 3
style.color = Color.grey
progressbar.value = 1 -- it starts full
local id = autoincrement()
local tick = game.tick
if not duration then
duration = 15
end
Gui.set_data(frame, {
is_toast = true,
toast_id = id,
progressbar = progressbar,
start_tick = tick,
end_tick = tick + duration * 60
})
memory.active_toasts[id] = player.name
return container
end
---Attempts to get a toast based on the element, will traverse through parents to find it.
---@param element LuaGuiElement
local function get_toast(element)
if not element or not element.valid then
return nil
end
local data = Gui.get_data(element)
if data and type(data) == 'table' and not data.__self and data.is_toast then
return element, data
end
-- no data, have to check the parent
return get_toast(element.parent)
end
Event.add(defines.events.on_gui_click, function (event)
local element, data = get_toast(event.element)
if not element or not data then
return
end
Gui.destroy(element)
memory.active_toasts[data.toast_id] = nil
end)
Event.on_nth_tick(2, function (event)
local active_toasts = memory.active_toasts
if size(active_toasts) == 0 then
return
end
local tick = event.tick
local players = game.players
for _, player_name in pairs(active_toasts) do
local player = players[player_name]
if player and player.valid then
for _, element in pairs(player.gui.left.children) do
local toast, data = get_toast(element)
if toast and data then
local end_tick = data.end_tick
if tick > end_tick then
Gui.destroy(element)
active_toasts[data.toast_id] = nil
else
local limit = end_tick - data.start_tick
local current = end_tick - tick
data.progressbar.value = current / limit
end
end
end
end
end
end)
---Toast a specific player, template is a callable that receives a LuaGuiElement
---to add contents to and a player as second argument.
---@param player LuaPlayer|number
---@param duration table
---@param template function
function Public.toast_player_template(player, duration, template)
local container = toast_to(player, duration)
if container then
template(container, player)
end
end
---Toast all players of the given force, template is a callable that receives a LuaGuiElement
---to add contents to and a player as second argument.
---@param force LuaForce
---@param duration number
---@param template function
function Public.toast_force_template(force, duration, template)
for _, player in pairs(force.connected_players) do
template(toast_to(player, duration), player)
end
end
---Toast all players, template is a callable that receives a LuaGuiElement
---to add contents to and a player as second argument.
---@param duration number
---@param template function
function Public.toast_all_players_template(duration, template)
for _, player in pairs(game.connected_players) do
template(toast_to(player, duration), player)
end
end
---Toast a message to a specific player
---@param player LuaPlayer|number
---@param duration number
---@param message string
function Public.toast_player(player, duration, message)
Public.toast_player_template(player, duration, function (container)
local label = container.add({type = 'label', caption = message})
label.style.single_line = false
end)
end
---Toast a message to all players of a given force
---@param force LuaForce
---@param duration number
---@param message string
function Public.toast_force(force, duration, message)
for _, player in pairs(force.connected_players) do
Public.toast_player(player, duration, message)
end
end
---Toast a message to all players
---@param duration number
---@param message string
function Public.toast_all_players(duration, message)
for _, player in pairs(game.connected_players) do
Public.toast_player(player, duration, message)
end
end
return Public