mirror of
https://github.com/CHOMPStation2/CHOMPStation2.git
synced 2025-12-10 10:12:45 +00:00
204 lines
4.8 KiB
Plaintext
204 lines
4.8 KiB
Plaintext
/**
|
|
* Copyright (c) 2020 Aleksej Komarov
|
|
* SPDX-License-Identifier: MIT
|
|
*/
|
|
|
|
/datum/tgui_window
|
|
var/id
|
|
var/client/client
|
|
var/pooled
|
|
var/pool_index
|
|
var/status = TGUI_WINDOW_CLOSED
|
|
var/locked = FALSE
|
|
var/datum/tgui/locked_by
|
|
var/fatally_errored = FALSE
|
|
var/message_queue
|
|
|
|
/**
|
|
* public
|
|
*
|
|
* Create a new tgui window.
|
|
*
|
|
* required client /client
|
|
* required id string A unique window identifier.
|
|
*/
|
|
/datum/tgui_window/New(client/client, id, pooled = FALSE)
|
|
src.id = id
|
|
src.client = client
|
|
src.pooled = pooled
|
|
if(pooled)
|
|
client.tgui_windows[id] = src
|
|
src.pool_index = TGUI_WINDOW_INDEX(id)
|
|
|
|
/**
|
|
* public
|
|
*
|
|
* Initializes the window with a fresh page. Puts window into the "loading"
|
|
* state. You can begin sending messages right after initializing. Messages
|
|
* will be put into the queue until the window finishes loading.
|
|
*/
|
|
/datum/tgui_window/proc/initialize()
|
|
log_tgui(client, "[id]/initialize")
|
|
if(!client)
|
|
return
|
|
status = TGUI_WINDOW_LOADING
|
|
fatally_errored = FALSE
|
|
message_queue = null
|
|
// Build window options
|
|
var/options = "file=[id].html;can_minimize=0;auto_format=0;"
|
|
// Remove titlebar and resize handles for a fancy window
|
|
// if(client.prefs.tgui_fancy)
|
|
// options += "titlebar=0;can_resize=0;"
|
|
// else
|
|
options += "titlebar=1;can_resize=1;"
|
|
// Generate page html
|
|
// TODO: Make this static
|
|
var/html = SStgui.basehtml
|
|
html = replacetextEx(html, "\[tgui:windowId]", id)
|
|
// Send required assets
|
|
var/datum/asset/asset
|
|
asset = get_asset_datum(/datum/asset/simple/tgui)
|
|
asset.send(client)
|
|
asset = get_asset_datum(/datum/asset/simple/fontawesome)
|
|
asset.send(client)
|
|
// Open the window
|
|
client << browse(html, "window=[id];[options]")
|
|
// Instruct the client to signal UI when the window is closed.
|
|
winset(client, id, "on-close=\"uiclose [id]\"")
|
|
|
|
/**
|
|
* public
|
|
*
|
|
* Checks if the window is ready to receive data.
|
|
*
|
|
* return bool
|
|
*/
|
|
/datum/tgui_window/proc/is_ready()
|
|
return status == TGUI_WINDOW_READY
|
|
|
|
/**
|
|
* public
|
|
*
|
|
* Checks if the window can be sanely suspended.
|
|
*
|
|
* return bool
|
|
*/
|
|
/datum/tgui_window/proc/can_be_suspended()
|
|
return !fatally_errored \
|
|
&& pooled \
|
|
&& pool_index > 0 \
|
|
&& pool_index <= TGUI_WINDOW_SOFT_LIMIT \
|
|
&& status >= TGUI_WINDOW_READY
|
|
|
|
/**
|
|
* public
|
|
*
|
|
* Acquire the window lock. Pool will not be able to provide this window
|
|
* to other UIs for the duration of the lock.
|
|
*
|
|
* Can be given an optional tgui datum, which will hook its on_message
|
|
* callback into the message stream.
|
|
*
|
|
* optional ui /datum/tgui
|
|
*/
|
|
/datum/tgui_window/proc/acquire_lock(datum/tgui/ui)
|
|
locked = TRUE
|
|
locked_by = ui
|
|
|
|
/**
|
|
* Release the window lock.
|
|
*/
|
|
/datum/tgui_window/proc/release_lock()
|
|
locked = FALSE
|
|
locked_by = null
|
|
|
|
/**
|
|
* public
|
|
*
|
|
* Close the UI.
|
|
*
|
|
* optional can_be_suspended bool
|
|
*/
|
|
/datum/tgui_window/proc/close(can_be_suspended = TRUE)
|
|
if(!client)
|
|
return
|
|
if(can_be_suspended && can_be_suspended())
|
|
log_tgui(client, "[id]/close: suspending")
|
|
status = TGUI_WINDOW_READY
|
|
send_message("suspend")
|
|
return
|
|
log_tgui(client, "[id]/close")
|
|
locked = FALSE
|
|
locked_by = null
|
|
status = TGUI_WINDOW_CLOSED
|
|
message_queue = null
|
|
// Do not close the window to give user some time
|
|
// to read the error message.
|
|
if(!fatally_errored)
|
|
client << browse(null, "window=[id]")
|
|
|
|
/**
|
|
* public
|
|
*
|
|
* Sends a message to tgui window.
|
|
*
|
|
* required type string Message type
|
|
* required payload list Message payload
|
|
* optional force bool Send regardless of the ready status.
|
|
*/
|
|
/datum/tgui_window/proc/send_message(type, list/payload, force)
|
|
if(!client)
|
|
return
|
|
var/message = json_encode(list(
|
|
"type" = type,
|
|
"payload" = payload,
|
|
))
|
|
// Strip #255/improper.
|
|
message = replacetext(message, "\proper", "")
|
|
message = replacetext(message, "\improper", "")
|
|
// Pack for sending via output()
|
|
message = url_encode(message)
|
|
// Place into queue if window is still loading
|
|
if(!force && status == TGUI_WINDOW_LOADING)
|
|
if(!message_queue)
|
|
message_queue = list()
|
|
message_queue += list(message)
|
|
return
|
|
client << output(message, "[id].browser:update")
|
|
|
|
/**
|
|
* private
|
|
*
|
|
* Sends queued messages if the queue wasn't empty.
|
|
*/
|
|
/datum/tgui_window/proc/flush_message_queue()
|
|
if(!client || !message_queue)
|
|
return
|
|
for(var/message in message_queue)
|
|
client << output(message, "[id].browser:update")
|
|
message_queue = null
|
|
|
|
/**
|
|
* private
|
|
*
|
|
* Callback for handling incoming tgui messages.
|
|
*/
|
|
/datum/tgui_window/proc/on_message(type, list/payload, list/href_list)
|
|
switch(type)
|
|
if("ready")
|
|
status = TGUI_WINDOW_READY
|
|
if("log")
|
|
if(href_list["fatal"])
|
|
fatally_errored = TRUE
|
|
// Pass message to UI that requested the lock
|
|
if(locked && locked_by)
|
|
locked_by.on_message(type, payload, href_list)
|
|
flush_message_queue()
|
|
return
|
|
// If not locked, handle these message types
|
|
switch(type)
|
|
if("suspend")
|
|
close(can_be_suspended = TRUE)
|
|
if("close")
|
|
close(can_be_suspended = FALSE)
|