/mob var/next_emote var/next_emote_refresh var/last_emote_summary /mob/proc/get_available_emotes() return GLOB.default_mob_emotes.Copy() /mob/proc/can_emote(var/emote_type) return (stat == CONSCIOUS) #define EMOTE_REFRESH_SPAM_COOLDOWN (5 SECONDS) /mob/living/can_emote(var/emote_type) return (..() && !(silent && emote_type == AUDIBLE_MESSAGE)) /mob/proc/emote(var/act, var/m_type, var/message) set waitfor = FALSE // s-s-snowflake if(src.stat == DEAD && act != "deathgasp") return if(is_paralyzed()) return if(usr == src) //client-called emote if (client && (client.prefs.muted & MUTE_IC)) to_chat(src, span_warning("You cannot send IC messages (muted).")) return if(world.time < next_emote) to_chat(src, span_warning("You cannot use another emote yet.")) return if(forced_psay) pme(message) return if(autowhisper) return me_verb_subtle(message) if(act == "help") if(world.time >= next_emote_refresh) var/list/usable_emotes = list() next_emote_refresh = world.time + EMOTE_REFRESH_SPAM_COOLDOWN for(var/emote in get_available_emotes()) var/datum/decl/emote/emote_datum = GLOB.decls_repository.get_decl(emote) if(emote_datum.mob_can_use(src)) usable_emotes[emote_datum.key] = emote_datum last_emote_summary = english_list(sortAssoc(usable_emotes)) to_chat(src,span_bold("Usable emotes:") + " [last_emote_summary].") return if(!can_emote(m_type)) to_chat(src, span_warning("You cannot currently [m_type == AUDIBLE_MESSAGE ? "audibly" : "visually"] emote!")) return if(act == "me") return custom_emote(m_type, message) if(act == "custom") if(!message) message = sanitize_or_reflect(tgui_input_text(src,"Choose an emote to display.", encode = FALSE), src) //VOREStation Edit - Reflect too long messages, within reason if(!message) return if (!m_type) if(tgui_alert(src, "Is this an audible emote?", "Emote", list("Yes", "No")) != "Yes") m_type = VISIBLE_MESSAGE else m_type = AUDIBLE_MESSAGE return custom_emote(m_type, message) var/splitpoint = findtext(act, " ") if(splitpoint > 0) var/tempstr = act act = copytext(tempstr,1,splitpoint) message = copytext(tempstr,splitpoint+1,0) //VOREStation Add - NIF soulcatcher shortcuts if(act == "nsay") return nsay(message) if(act == "nme") return nme(message) //VOREStation Add End var/datum/decl/emote/use_emote = get_emote_by_key(act) if(!istype(use_emote)) to_chat(src, span_warning("Unknown emote '[act]'. Type " + span_bold("say *help") + " for a list of usable emotes. ([act] [message])")) // Add full message in the event you used * instead of ! or something like that return if(!use_emote.mob_can_use(src)) to_chat(src, span_warning("You cannot use the emote '[act]'. Type say *help for a list of usable emotes.")) return if(m_type != use_emote.message_type && use_emote.conscious && stat != CONSCIOUS) return next_emote = world.time + use_emote.emote_delay if(use_emote.message_type == AUDIBLE_MESSAGE && is_muzzled()) var/muffle_message = use_emote.emote_message_muffled || "makes a muffled sound." audible_message(span_bold("\The [src]") + " [muffle_message]", runemessage = "[muffle_message]") return use_emote.do_emote(src, message) for (var/obj/item/implant/I in src) if (I.implanted) I.trigger(act, src) #undef EMOTE_REFRESH_SPAM_COOLDOWN /mob/proc/format_emote(var/emoter = null, var/message = null) var/pretext var/subtext var/nametext var/end_char var/start_char var/name_anchor if(!message || !emoter) return message = html_decode(message) name_anchor = findtext(message, "*") if(name_anchor > 0) // User supplied emote with visible_emote token (default ^) pretext = copytext(message, 1, name_anchor) subtext = copytext(message, name_anchor + 1, length(message) + 1) else // No token. Just the emote as usual. subtext = message // Oh shit, we got this far! Let's see... did the user attempt to use more than one token? if(findtext(subtext, "*")) // abort abort! to_chat(emoter, span_warning("You may use only one \"["*"]\" symbol in your emote.")) to_chat(emoter, span_warning(message)) return if(pretext) // Add a space at the end if we didn't already supply one. end_char = copytext(pretext, length(pretext), length(pretext) + 1) if(end_char != " ") pretext += " " // Grab the last character of the emote message. end_char = copytext(subtext, length(subtext), length(subtext) + 1) if(!(end_char in list(".", "?", "!", "\"", "-", "~"))) // gotta include ~ for all you fucking weebs // No punctuation supplied. Tack a period on the end. subtext += "." // Add a space to the subtext, unless it begins with an apostrophe or comma. if(subtext != ".") // First, let's get rid of any existing space, to account for sloppy emoters ("X, ^ , Y") subtext = trim_left(subtext) start_char = copytext(subtext, 1, 2) if(start_char != "," && start_char != "'") subtext = " " + subtext pretext = capitalize(html_encode(pretext)) nametext = html_encode(nametext) subtext = html_encode(subtext) // Store the player's name in a nice bold, naturalement nametext = span_bold("[emoter]") return list("pretext" = pretext, "nametext" = nametext, "subtext" = subtext) // Specific mob type exceptions below. /mob/living/silicon/ai/emote(var/act, var/type, var/message) var/obj/machinery/hologram/holopad/T = src.holo if(T && T.masters[src]) //Is the AI using a holopad? src.holopad_emote(message) else //Emote normally, then. ..()