mirror of
https://github.com/Bubberstation/Bubberstation.git
synced 2025-12-10 09:42:29 +00:00
## About The Pull Request Surprisingly never came up until now, anyways this needs to deleted in case the `Execute()` fails because undeleted SQL queries are bad (iirc) ## Why It's Good For The Game No more random errors ## Changelog Unneeded
113 lines
3.7 KiB
Plaintext
113 lines
3.7 KiB
Plaintext
/// Namespace for housing code relating to giving contextual tutorials to users.
|
|
SUBSYSTEM_DEF(tutorials)
|
|
name = "Tutorials"
|
|
flags = SS_NO_FIRE
|
|
|
|
/// A mapping of /datum/tutorial type to their manager singleton.
|
|
/// You probably shouldn't be indexing this directly.
|
|
var/list/datum/tutorial_manager/tutorial_managers = list()
|
|
|
|
VAR_PRIVATE/list/datum/tutorial_manager/tutorial_managers_by_key = list()
|
|
|
|
/datum/controller/subsystem/tutorials/Initialize()
|
|
init_tutorial_managers()
|
|
load_initial_tutorial_completions()
|
|
RegisterSignal(SSdcs, COMSIG_GLOB_CLIENT_CONNECT, PROC_REF(on_client_connect))
|
|
|
|
return SS_INIT_SUCCESS
|
|
|
|
/// Will suggest the passed tutorial type to the user.
|
|
/// Will check that they should actually see it, e.g. hasn't completed it yet, etc.
|
|
/// Then, calls `/datum/tutorial/subtype/perform` with the extra arguments passed in.
|
|
/datum/controller/subsystem/tutorials/proc/suggest_tutorial(mob/user, datum/tutorial/tutorial_type, ...)
|
|
var/datum/tutorial_manager/tutorial_manager = tutorial_managers[tutorial_type]
|
|
if (isnull(tutorial_manager))
|
|
CRASH("[tutorial_type] is not a valid tutorial type")
|
|
|
|
if (!tutorial_manager.should_run(user))
|
|
return
|
|
|
|
INVOKE_ASYNC(tutorial_manager, TYPE_PROC_REF(/datum/tutorial_manager, try_perform), user, args.Copy(3))
|
|
|
|
/datum/controller/subsystem/tutorials/proc/init_tutorial_managers()
|
|
PRIVATE_PROC(TRUE)
|
|
|
|
for (var/datum/tutorial/tutorial_type as anything in subtypesof(/datum/tutorial))
|
|
var/datum/tutorial_manager/tutorial_manager = new /datum/tutorial_manager(tutorial_type)
|
|
tutorial_managers[tutorial_type] = tutorial_manager
|
|
tutorial_managers_by_key[tutorial_manager.get_key()] = tutorial_manager
|
|
|
|
/datum/controller/subsystem/tutorials/proc/load_initial_tutorial_completions()
|
|
PRIVATE_PROC(TRUE)
|
|
set waitfor = FALSE // There's no reason to halt init for this
|
|
|
|
var/list/ckey_options = list()
|
|
var/list/ckeys = list()
|
|
|
|
for (var/client/client as anything in GLOB.clients)
|
|
var/ckey = client?.ckey
|
|
if (!ckey)
|
|
continue // client shenanigans, never trust
|
|
|
|
var/index = ckeys.len + 1
|
|
ckey_options += ":ckey[index]"
|
|
ckeys["ckey[index]"] = ckey
|
|
|
|
if (ckey_options.len == 0)
|
|
return
|
|
|
|
var/datum/db_query/select_all_query = SSdbcore.NewQuery(
|
|
"SELECT ckey, tutorial_key FROM [format_table_name("tutorial_completions")] WHERE ckey in ([ckey_options.Join(", ")])",
|
|
ckeys,
|
|
)
|
|
|
|
if (!select_all_query.Execute())
|
|
qdel(select_all_query)
|
|
return
|
|
|
|
while (select_all_query.NextRow())
|
|
var/ckey = select_all_query.item[1]
|
|
var/tutorial_key = select_all_query.item[2]
|
|
|
|
mark_ckey_completed_tutorial(ckey, tutorial_key)
|
|
|
|
qdel(select_all_query)
|
|
|
|
/datum/controller/subsystem/tutorials/proc/on_client_connect(datum/source, client/client)
|
|
SIGNAL_HANDLER
|
|
|
|
var/ckey = client.ckey
|
|
if (!ckey)
|
|
return
|
|
|
|
INVOKE_ASYNC(src, PROC_REF(check_completed_tutorials_for_ckey), ckey)
|
|
|
|
/datum/controller/subsystem/tutorials/proc/check_completed_tutorials_for_ckey(ckey)
|
|
if (!SSdbcore.IsConnected())
|
|
return
|
|
|
|
var/datum/db_query/select_tutorials_for_ckey = SSdbcore.NewQuery(
|
|
"SELECT tutorial_key FROM [format_table_name("tutorial_completions")] WHERE ckey = :ckey",
|
|
list("ckey" = ckey)
|
|
)
|
|
|
|
if (!select_tutorials_for_ckey.Execute())
|
|
qdel(select_tutorials_for_ckey)
|
|
return
|
|
|
|
while (select_tutorials_for_ckey.NextRow())
|
|
var/tutorial_key = select_tutorials_for_ckey.item[1]
|
|
|
|
mark_ckey_completed_tutorial(ckey, tutorial_key)
|
|
|
|
qdel(select_tutorials_for_ckey)
|
|
|
|
/datum/controller/subsystem/tutorials/proc/mark_ckey_completed_tutorial(ckey, tutorial_key)
|
|
var/datum/tutorial_manager/tutorial_manager = tutorial_managers_by_key[tutorial_key]
|
|
if (isnull(tutorial_manager))
|
|
// Not necessarily a bug.
|
|
// Could be an outdated server or a removed tutorial.
|
|
return
|
|
|
|
tutorial_manager.mark_as_completed(ckey)
|