/** * vim: set ts=4 : * ============================================================================= * SourceMod * Copyright (C) 2004-2007 AlliedModders LLC. All rights reserved. * ============================================================================= * * This program is free software; you can redistribute it and/or modify it under * the terms of the GNU General Public License, version 3.0, as published by the * Free Software Foundation. * * This program is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS * FOR A PARTICULAR PURPOSE. See the GNU General Public License for more * details. * * You should have received a copy of the GNU General Public License along with * this program. If not, see . * * As a special exception, AlliedModders LLC gives you permission to link the * code of this program (as well as its derivative works) to "Half-Life 2," the * "Source Engine," the "SourcePawn JIT," and any Game MODs that run on software * by the Valve Corporation. You must obey the GNU General Public License in * all respects for all other code used. Additionally, AlliedModders LLC grants * this exception to all derivative works. AlliedModders LLC defines further * exceptions, found in LICENSE.txt (as of this writing, version JULY-31-2007), * or . * * Version: $Id$ */ #include "ConCmdManager.h" #include "sm_srvcmds.h" #include "AdminCache.h" #include "sm_stringutil.h" #include "PlayerManager.h" #include "Translator.h" #include "HalfLife2.h" #include "ChatTriggers.h" ConCmdManager g_ConCmds; SH_DECL_HOOK0_void(ConCommand, Dispatch, SH_NOATTRIB, false); SH_DECL_HOOK1_void(IServerGameClients, SetCommandClient, SH_NOATTRIB, false, int); struct PlCmdInfo { ConCmdInfo *pInfo; CmdHook *pHook; CmdType type; }; typedef List CmdList; void AddToPlCmdList(CmdList *pList, const PlCmdInfo &info); ConCmdManager::ConCmdManager() : m_Strings(1024) { m_pCmds = sm_trie_create(); m_pCmdGrps = sm_trie_create(); m_CmdClient = 0; } ConCmdManager::~ConCmdManager() { sm_trie_destroy(m_pCmds); sm_trie_destroy(m_pCmdGrps); } void ConCmdManager::OnSourceModAllInitialized() { g_PluginSys.AddPluginsListener(this); g_RootMenu.AddRootConsoleCommand("cmds", "List console commands", this); SH_ADD_HOOK_MEMFUNC(IServerGameClients, SetCommandClient, serverClients, this, &ConCmdManager::SetCommandClient, false); } void ConCmdManager::OnSourceModShutdown() { /* All commands should already be removed by the time we're done */ SH_REMOVE_HOOK_MEMFUNC(IServerGameClients, SetCommandClient, serverClients, this, &ConCmdManager::SetCommandClient, false); g_RootMenu.RemoveRootConsoleCommand("cmds", this); } void ConCmdManager::RemoveConCmds(List &cmdlist, IPluginContext *pContext) { List::iterator iter = cmdlist.begin(); CmdHook *pHook; while (iter != cmdlist.end()) { pHook = (*iter); if (pHook->pf->GetParentContext() == pContext) { delete pHook->pAdmin; delete pHook; iter = cmdlist.erase(iter); } else { iter++; } } } void ConCmdManager::OnPluginDestroyed(IPlugin *plugin) { CmdList *pList; List removed; if (plugin->GetProperty("CommandList", (void **)&pList, true)) { IPluginContext *pContext = plugin->GetBaseContext(); CmdList::iterator iter; /* First pass! * Only bother if there's an actual command list on this plugin... */ for (iter=pList->begin(); iter!=pList->end(); iter++) { PlCmdInfo &cmd = (*iter); ConCmdInfo *pInfo = cmd.pInfo; /* Has this chain already been fully cleaned/removed? */ if (removed.find(pInfo) != removed.end()) { continue; } /* Remove any hooks from us on this command */ RemoveConCmds(pInfo->conhooks, pContext); RemoveConCmds(pInfo->srvhooks, pContext); /* See if there are still hooks */ if (pInfo->srvhooks.size()) { continue; } if (pInfo->conhooks.size()) { continue; } /* Remove the command, it should be safe now */ RemoveConCmd(pInfo); removed.push_back(pInfo); } delete pList; } } void CommandCallback() { g_ConCmds.InternalDispatch(); } void ConCmdManager::SetCommandClient(int client) { m_CmdClient = client + 1; } ResultType ConCmdManager::DispatchClientCommand(int client, const char *cmd, int args, ResultType type) { ConCmdInfo *pInfo; if (sm_trie_retrieve(m_pCmds, cmd, (void **)&pInfo)) { cell_t result = type; cell_t tempres = result; List::iterator iter; CmdHook *pHook; for (iter=pInfo->conhooks.begin(); iter!=pInfo->conhooks.end(); iter++) { pHook = (*iter); if (!pHook->pf->IsRunnable()) { continue; } if (pHook->pAdmin && !CheckAccess(client, cmd, pHook->pAdmin)) { if (result < Pl_Handled) { result = Pl_Handled; } continue; } pHook->pf->PushCell(client); pHook->pf->PushCell(args); if (pHook->pf->Execute(&tempres) == SP_ERROR_NONE) { if (tempres > result) { result = tempres; } if (result == Pl_Stop) { break; } } } type = (ResultType)result; } return type; } void ConCmdManager::InternalDispatch() { /** * Note: Console commands will EITHER go through IServerGameDLL::ClientCommand, * OR this dispatch. They will NEVER go through both. * -- * Whether or not it goes through the callback is determined by FCVAR_GAMEDLL */ char cmd[300]; strncopy(cmd, engine->Cmd_Argv(0), sizeof(cmd)); ConCmdInfo *pInfo; if (!sm_trie_retrieve(m_pCmds, cmd, (void **)&pInfo)) { return; } cell_t result = Pl_Continue; int args = engine->Cmd_Argc() - 1; List::iterator iter; CmdHook *pHook; /* Execute server-only commands if viable */ if (m_CmdClient == 0 && pInfo->srvhooks.size()) { cell_t tempres = result; for (iter=pInfo->srvhooks.begin(); iter!=pInfo->srvhooks.end(); iter++) { pHook = (*iter); if (!pHook->pf->IsRunnable()) { continue; } pHook->pf->PushCell(args); if (pHook->pf->Execute(&tempres) == SP_ERROR_NONE) { if (tempres > result) { result = tempres; } if (result == Pl_Stop) { break; } } } /* Check if there's an early stop */ if (result >= Pl_Stop) { if (!pInfo->sourceMod) { RETURN_META(MRES_SUPERCEDE); } return; } } /* Execute console commands */ if (pInfo->conhooks.size()) { cell_t tempres = result; for (iter=pInfo->conhooks.begin(); iter!=pInfo->conhooks.end(); iter++) { pHook = (*iter); if (!pHook->pf->IsRunnable()) { continue; } if (m_CmdClient && pHook->pAdmin && !CheckAccess(m_CmdClient, cmd, pHook->pAdmin)) { if (result < Pl_Handled) { result = Pl_Handled; } continue; } /* On a listen server, sometimes the server host's client index can be set as 0. * So index 1 is passed to the command callback to correct this potential problem. */ if (m_CmdClient == 0 && !engine->IsDedicatedServer()) { pHook->pf->PushCell(1); } else { pHook->pf->PushCell(m_CmdClient); } pHook->pf->PushCell(args); if (pHook->pf->Execute(&tempres) == SP_ERROR_NONE) { if (tempres > result) { result = tempres; } if (result == Pl_Stop) { break; } } } } if (result >= Pl_Handled) { if (!pInfo->sourceMod) { RETURN_META(MRES_SUPERCEDE); } return; } } bool ConCmdManager::CheckCommandAccess(int client, const char *cmd, FlagBits cmdflags) { if (cmdflags == 0 || client == 0) { return true; } /* If running listen server, then client 1 is the server host and should have 'root' access */ if (client == 1 && !engine->IsDedicatedServer()) { return true; } CPlayer *player = g_Players.GetPlayerByIndex(client); if (!player || player->GetEdict() == NULL || player->IsFakeClient()) { return false; } AdminId adm = player->GetAdminId(); if (adm != INVALID_ADMIN_ID) { FlagBits bits = g_Admins.GetAdminFlags(adm, Access_Effective); /* root knows all, WHOA */ if ((bits & ADMFLAG_ROOT) == ADMFLAG_ROOT) { return true; } /* Check for overrides * :TODO: is it worth optimizing this? */ unsigned int groups = g_Admins.GetAdminGroupCount(adm); GroupId gid; OverrideRule rule; bool override = false; for (unsigned int i=0; ieflags)) { return true; } edict_t *pEdict = engine->PEntityOfEntIndex(client); /* If we got here, the command failed... */ char buffer[128]; if (g_Translator.CoreTrans(client, buffer, sizeof(buffer), "No Access", NULL, NULL) != Trans_Okay) { snprintf(buffer, sizeof(buffer), "You do not have access to this command"); } unsigned int replyto = g_ChatTriggers.GetReplyTo(); if (replyto == SM_REPLY_CONSOLE) { char fullbuffer[192]; snprintf(fullbuffer, sizeof(fullbuffer), "[SM] %s.\n", buffer); engine->ClientPrintf(pEdict, fullbuffer); } else if (replyto == SM_REPLY_CHAT) { char fullbuffer[192]; snprintf(fullbuffer, sizeof(fullbuffer), "[SM] %s.", buffer); g_HL2.TextMsg(client, HUD_PRINTTALK, fullbuffer); } return false; } bool ConCmdManager::AddConsoleCommand(IPluginFunction *pFunction, const char *name, const char *description, int flags) { ConCmdInfo *pInfo = AddOrFindCommand(name, description, flags); if (!pInfo) { return false; } CmdHook *pHook = new CmdHook(); pHook->pf = pFunction; if (description && description[0]) { pHook->helptext.assign(description); } pInfo->conhooks.push_back(pHook); /* Add to the plugin */ CmdList *pList; IPlugin *pPlugin = g_PluginSys.GetPluginByCtx(pFunction->GetParentContext()->GetContext()); if (!pPlugin->GetProperty("CommandList", (void **)&pList)) { pList = new CmdList(); pPlugin->SetProperty("CommandList", pList); } PlCmdInfo info; info.pInfo = pInfo; info.type = Cmd_Console; info.pHook = pHook; AddToPlCmdList(pList, info); return true; } bool ConCmdManager::AddAdminCommand(IPluginFunction *pFunction, const char *name, const char *group, int adminflags, const char *description, int flags) { ConCmdInfo *pInfo = AddOrFindCommand(name, description, flags); if (!pInfo) { return false; } CmdHook *pHook = new CmdHook(); AdminCmdInfo *pAdmin = new AdminCmdInfo(); pHook->pf = pFunction; if (description && description[0]) { pHook->helptext.assign(description); } pHook->pAdmin = pAdmin; void *object; int grpid; if (!sm_trie_retrieve(m_pCmdGrps, group, (void **)&object)) { grpid = m_Strings.AddString(group); sm_trie_insert(m_pCmdGrps, group, (void *)grpid); } else { grpid = (int)object; } pAdmin->cmdGrpId = grpid; pAdmin->flags = adminflags; /* First get the command group override, if any */ bool override = g_Admins.GetCommandOverride(group, Override_CommandGroup, &(pAdmin->eflags)); /* Next get the command override, if any */ if (g_Admins.GetCommandOverride(name, Override_Command, &(pAdmin->eflags))) { override = true; } /* Assign normal flags if there were no overrides */ if (!override) { pAdmin->eflags = pAdmin->flags; } /* Finally, add the hook */ pInfo->conhooks.push_back(pHook); pInfo->admin = *(pHook->pAdmin); pInfo->is_admin_set = true; /* Now add to the plugin */ CmdList *pList; IPlugin *pPlugin = g_PluginSys.GetPluginByCtx(pFunction->GetParentContext()->GetContext()); if (!pPlugin->GetProperty("CommandList", (void **)&pList)) { pList = new CmdList(); pPlugin->SetProperty("CommandList", pList); } PlCmdInfo info; info.pInfo = pInfo; info.type = Cmd_Admin; info.pHook = pHook; AddToPlCmdList(pList, info); return true; } bool ConCmdManager::AddServerCommand(IPluginFunction *pFunction, const char *name, const char *description, int flags) { ConCmdInfo *pInfo = AddOrFindCommand(name, description, flags); if (!pInfo) { return false; } CmdHook *pHook = new CmdHook(); pHook->pf = pFunction; if (description && description[0]) { pHook->helptext.assign(description); } pInfo->srvhooks.push_back(pHook); /* Add to the plugin */ CmdList *pList; IPlugin *pPlugin = g_PluginSys.GetPluginByCtx(pFunction->GetParentContext()->GetContext()); if (!pPlugin->GetProperty("CommandList", (void **)&pList)) { pList = new CmdList(); pPlugin->SetProperty("CommandList", pList); } PlCmdInfo info; info.pInfo = pInfo; info.type = Cmd_Server; info.pHook = pHook; AddToPlCmdList(pList, info); return true; } void AddToPlCmdList(CmdList *pList, const PlCmdInfo &info) { CmdList::iterator iter = pList->begin(); bool inserted = false; const char *orig = NULL; orig = info.pInfo->pCmd->GetName(); /* Insert this into the help list, SORTED alphabetically. */ while (iter != pList->end()) { PlCmdInfo &obj = (*iter); const char *cmd = obj.pInfo->pCmd->GetName(); if (strcmp(orig, cmd) < 0) { pList->insert(iter, info); inserted = true; break; } iter++; } if (!inserted) { pList->push_back(info); } } void ConCmdManager::AddToCmdList(ConCmdInfo *info) { List::iterator iter = m_CmdList.begin(); ConCmdInfo *pInfo; bool inserted = false; const char *orig = NULL; orig = info->pCmd->GetName(); /* Insert this into the help list, SORTED alphabetically. */ while (iter != m_CmdList.end()) { const char *cmd = NULL; pInfo = (*iter); cmd = pInfo->pCmd->GetName(); if (strcmp(orig, cmd) < 0) { m_CmdList.insert(iter, info); inserted = true; break; } iter++; } if (!inserted) { m_CmdList.push_back(info); } } void ConCmdManager::UpdateAdminCmdFlags(const char *cmd, OverrideType type, FlagBits bits) { ConCmdInfo *pInfo; if (type == Override_Command) { if (!sm_trie_retrieve(m_pCmds, cmd, (void **)&pInfo)) { return; } List::iterator iter; CmdHook *pHook; for (iter=pInfo->conhooks.begin(); iter!=pInfo->conhooks.end(); iter++) { pHook = (*iter); if (pHook->pAdmin) { if (bits) { pHook->pAdmin->eflags = bits; } else { pHook->pAdmin->eflags = pHook->pAdmin->flags; } pInfo->admin = *(pHook->pAdmin); } } pInfo->is_admin_set = true; } else if (type == Override_CommandGroup) { void *object; if (!sm_trie_retrieve(m_pCmdGrps, cmd, &object)) { return; } int grpid = (int)object; /* This is bad :( loop through all commands */ List::iterator iter; CmdHook *pHook; for (iter=m_CmdList.begin(); iter!=m_CmdList.end(); iter++) { pInfo = (*iter); for (List::iterator citer=pInfo->conhooks.begin(); citer!=pInfo->conhooks.end(); citer++) { pHook = (*citer); if (pHook->pAdmin && pHook->pAdmin->cmdGrpId == grpid) { if (bits) { pHook->pAdmin->eflags = bits; } else { pHook->pAdmin->eflags = pHook->pAdmin->flags; } pInfo->admin = *(pHook->pAdmin); } } } pInfo->is_admin_set = true; } } void ConCmdManager::RemoveConCmd(ConCmdInfo *info) { /* Remove console-specific information * This should always be true as of right now */ if (info->pCmd) { /* Remove from the trie */ sm_trie_delete(m_pCmds, info->pCmd->GetName()); if (info->sourceMod) { /* Unlink from SourceMM */ g_SMAPI->UnregisterConCmdBase(g_PLAPI, info->pCmd); /* Delete the command's memory */ char *new_help = const_cast(info->pCmd->GetHelpText()); char *new_name = const_cast(info->pCmd->GetName()); delete [] new_help; delete [] new_name; delete info->pCmd; } else { /* Remove the external hook */ SH_REMOVE_HOOK_STATICFUNC(ConCommand, Dispatch, info->pCmd, CommandCallback, false); } } /* Remove from list */ m_CmdList.remove(info); delete info; } bool ConCmdManager::LookForSourceModCommand(const char *cmd) { ConCmdInfo *pInfo; if (!sm_trie_retrieve(m_pCmds, cmd, (void **)&pInfo)) { return false; } return pInfo->sourceMod && (pInfo->conhooks.size() > 0); } bool ConCmdManager::LookForCommandAdminFlags(const char *cmd, FlagBits *pFlags) { ConCmdInfo *pInfo; if (!sm_trie_retrieve(m_pCmds, cmd, (void **)&pInfo)) { return false; } *pFlags = pInfo->admin.eflags; return pInfo->is_admin_set; } ConCmdInfo *ConCmdManager::AddOrFindCommand(const char *name, const char *description, int flags) { ConCmdInfo *pInfo; if (!sm_trie_retrieve(m_pCmds, name, (void **)&pInfo)) { pInfo = new ConCmdInfo(); /* Find the commandopan */ ConCommandBase *pBase = icvar->GetCommands(); ConCommand *pCmd = NULL; while (pBase) { if (strcmp(pBase->GetName(), name) == 0) { /* Don't want to return convar with same name */ if (!pBase->IsCommand()) { return NULL; } pCmd = (ConCommand *)pBase; break; } pBase = const_cast(pBase->GetNext()); } if (!pCmd) { /* Note that we have to duplicate because the source might not be * a static string, and these expect static memory. */ if (!description) { description = ""; } char *new_name = sm_strdup(name); char *new_help = sm_strdup(description); pCmd = new ConCommand(new_name, CommandCallback, new_help, flags); pInfo->sourceMod = true; } else { SH_ADD_HOOK_STATICFUNC(ConCommand, Dispatch, pCmd, CommandCallback, false); } pInfo->pCmd = pCmd; pInfo->is_admin_set = false; sm_trie_insert(m_pCmds, name, pInfo); AddToCmdList(pInfo); } return pInfo; } void ConCmdManager::OnRootConsoleCommand(const char *command, unsigned int argcount) { if (argcount >= 3) { const char *text = engine->Cmd_Argv(2); int id = atoi(text); CPlugin *pPlugin = g_PluginSys.GetPluginByOrder(id); if (!pPlugin) { g_RootMenu.ConsolePrint("[SM] Plugin index %d not found.", id); return; } const sm_plugininfo_t *plinfo = pPlugin->GetPublicInfo(); const char *plname = IS_STR_FILLED(plinfo->name) ? plinfo->name : pPlugin->GetFilename(); CmdList *pList; if (!pPlugin->GetProperty("CommandList", (void **)&pList)) { g_RootMenu.ConsolePrint("[SM] No commands found for: %s", plname); return; } if (!pList->size()) { g_RootMenu.ConsolePrint("[SM] No commands found for: %s", plname); return; } CmdList::iterator iter; const char *type = NULL; const char *name; const char *help; g_RootMenu.ConsolePrint("[SM] Listing %d commands for: %s", pList->size(), plname); g_RootMenu.ConsolePrint(" %-17.16s %-8.7s %s", "[Name]", "[Type]", "[Help]"); for (iter=pList->begin(); iter!=pList->end(); iter++, id++) { PlCmdInfo &cmd = (*iter); if (cmd.type == Cmd_Server) { type = "server"; } else if (cmd.type == Cmd_Console) { type = "console"; } else if (cmd.type == Cmd_Admin) { type = "admin"; } name = cmd.pInfo->pCmd->GetName(); if (cmd.pHook->helptext.size()) { help = cmd.pHook->helptext.c_str(); } else { help = cmd.pInfo->pCmd->GetHelpText(); } g_RootMenu.ConsolePrint(" %-17.16s %-12.11s %s", name, type, help); } return; } g_RootMenu.ConsolePrint("[SM] Usage: sm cmds "); } static int g_yam_state = 0; void _YamState(int state) { if (state == 0) { g_RootMenu.ConsolePrint("Welcome to the SourceMod Text Adventure."); g_RootMenu.ConsolePrint("Type sm_text to see the last message."); g_RootMenu.ConsolePrint("Type sm_text to follow a capital word."); g_RootMenu.ConsolePrint("-------------------------------"); g_RootMenu.ConsolePrint("You are at VALVE HEADQUARTERS."); g_RootMenu.ConsolePrint("To your LEFT is BAILOPAN, rearranging the letters to spell"); g_RootMenu.ConsolePrint("\"A HARD VEAL QUEST\". FORWARD is the entrance to the building."); g_RootMenu.ConsolePrint("To your RIGHT is your last chance to flee in terror."); } else if (state == 1) { g_RootMenu.ConsolePrint("BAILOPAN tells you that his name his pronounced"); g_RootMenu.ConsolePrint("\"bye low pahn,\" not \"bay low pan.\" Do you "); g_RootMenu.ConsolePrint("MOCK him, or NOD quietly?"); } else if (state == 2) { g_RootMenu.ConsolePrint("You enter the Valve building. You hear screams coming from within."); g_RootMenu.ConsolePrint("A grotesque figure lumbers up to greet you; it is Gabe Newell."); g_RootMenu.ConsolePrint("\"Welcome,\" he belches, \"to my lair.\""); g_RootMenu.ConsolePrint("Do you SHAKE Gaben's hand, WALK past him, or OFFER a donut?"); } else if (state == 3) { g_RootMenu.ConsolePrint("You walk into the break room. Alfred \"Alf is from Melmac\" Reynolds and"); g_RootMenu.ConsolePrint("Yahn \"Yeti\" Bernier are discussing something (you overhear "); g_RootMenu.ConsolePrint("the phrase \"and next Steam update, here's what we should break\")."); g_RootMenu.ConsolePrint("Should you DIE in a fire, REPORT a bug, REQUEST a feature, or "); g_RootMenu.ConsolePrint("SPRAY them with butter?"); } g_yam_state = state; } void _IntExt_CallYams() { const char *arg = engine->Cmd_Argv(1); /* should be impossible */ if (!arg || arg[0] == '\0') { _YamState(g_yam_state); } if (g_yam_state == 1) { if (strcasecmp(arg, "mock") == 0) { g_RootMenu.ConsolePrint("You mock BAILOPAN's pronunciation. In a fit of rage, "); g_RootMenu.ConsolePrint("he sticks an INT 3 call into your chest, rendering you broken."); g_RootMenu.ConsolePrint("YOU HAVE DIED. GAME OVER."); g_yam_state = 0; } else if (strcasecmp(arg, "nod") == 0) { g_RootMenu.ConsolePrint("You nod quietly, and then slowly back away into the Valve headquarters."); _YamState(2); } else { g_RootMenu.ConsolePrint("Commands are MOCK and NOD."); } } else if (g_yam_state == 3) { if (strcasecmp(arg, "report") == 0) { g_RootMenu.ConsolePrint("You report a bug to Alfred and Yeti. Immediately, both fall asleep."); g_RootMenu.ConsolePrint("You decay in the break room for two years while they sleep."); g_RootMenu.ConsolePrint("YOU HAVE DIED. GAME OVER."); g_yam_state = 0; } else if (strcasecmp(arg, "request") == 0) { g_RootMenu.ConsolePrint("You request a feature to Alfred and Yeti. They both mutter something"); g_RootMenu.ConsolePrint("about it being implemented \"soon.\" Then, by accident, someone sends"); g_RootMenu.ConsolePrint("a message over \"Friends.\" The entire building catches fire."); g_RootMenu.ConsolePrint("YOU HAVE DIED. GAME OVER."); g_yam_state = 0; } else if (strcasecmp(arg, "die") == 0) { g_RootMenu.ConsolePrint("For no reason, you suddenly catch fire. Alfred and Yeti find this"); g_RootMenu.ConsolePrint("deeply disturbing, and cover your flaming corpse with Episode 2"); g_RootMenu.ConsolePrint("advertisements. Coming soon, with Team Fortress 2, and Portal!"); g_RootMenu.ConsolePrint("YOU HAVE DIED. GAME OVER."); g_yam_state = 0; } else if (strcasecmp(arg, "spray") == 0) { g_RootMenu.ConsolePrint("You spray Alfred and Yeti with butter. Like Jack Thompson to an"); g_RootMenu.ConsolePrint("ambulance, Gabe Newell instantly appears and devours them both."); g_RootMenu.ConsolePrint("You run away, just in time, as Gabe Newell explodes, registering "); g_RootMenu.ConsolePrint("a 5.6 earthquake. Outside, world peace is achieved."); g_RootMenu.ConsolePrint("YOU HAVE WON."); g_yam_state = 0; } } else if (g_yam_state == 2) { if (strcasecmp(arg, "shake") == 0) { g_RootMenu.ConsolePrint("You shake Gaben's hand. It is a terrifying and disgusting experience."); g_RootMenu.ConsolePrint("However, you survive, and continue on."); _YamState(3); } else if (strcasecmp(arg, "offer") == 0) { g_RootMenu.ConsolePrint("You offer Gabe Newell one (1) donut. With a gleam in his eyes, "); g_RootMenu.ConsolePrint("he picks you up and devours you whole."); g_RootMenu.ConsolePrint("YOU HAVE DIED. GAME OVER."); g_yam_state = 0; } else if (strcasecmp(arg, "walk") == 0) { g_RootMenu.ConsolePrint("You walk past Gabe Newell. He can't keep up with your pace!"); _YamState(3); } else { g_RootMenu.ConsolePrint("Commands are SHAKE, OFFER, and WALK."); } } else if (g_yam_state == 0) { if (strcasecmp(arg, "left") == 0) { _YamState(1); } else if (strcasecmp(arg, "right") == 0) { g_RootMenu.ConsolePrint("You run away from the Valve headquarters in sheer terror."); g_RootMenu.ConsolePrint("While running, you smash into an unknown person, who turns out to be your soul mate."); g_RootMenu.ConsolePrint("You marry and raise a family of 3 kids."); g_RootMenu.ConsolePrint("Many years later, you look back, and realize this was your best choice."); g_RootMenu.ConsolePrint("YOU HAVE WON."); g_yam_state = 0; } else if (strcasecmp(arg, "forward") == 0) { _YamState(2); } else if (arg[0] != '\0') { g_RootMenu.ConsolePrint("Commands are FORWARD, LEFT, and RIGHT."); } } } void _IntExt_EnableYams() { static ConCommand *pCmd = NULL; if (!pCmd) { pCmd = new ConCommand("sm_text", _IntExt_CallYams, "Fountain of Yams Adventure Game", FCVAR_GAMEDLL); g_RootMenu.ConsolePrint("Something is now enabled..."); } else { g_RootMenu.ConsolePrint("Something is already enabled..."); } }