mirror of
https://github.com/ZDoom/qzdoom-gpl.git
synced 2025-01-25 07:51:13 +00:00
226 lines
5.8 KiB
C++
226 lines
5.8 KiB
C++
/*
|
|
** keysections.cpp
|
|
** Custom key bindings
|
|
**
|
|
**---------------------------------------------------------------------------
|
|
** Copyright 1998-2009 Randy Heit
|
|
** Copyright 2010 Christoph Oelckers
|
|
** All rights reserved.
|
|
**
|
|
** Redistribution and use in source and binary forms, with or without
|
|
** modification, are permitted provided that the following conditions
|
|
** are met:
|
|
**
|
|
** 1. Redistributions of source code must retain the above copyright
|
|
** notice, this list of conditions and the following disclaimer.
|
|
** 2. Redistributions in binary form must reproduce the above copyright
|
|
** notice, this list of conditions and the following disclaimer in the
|
|
** documentation and/or other materials provided with the distribution.
|
|
** 3. The name of the author may not be used to endorse or promote products
|
|
** derived from this software without specific prior written permission.
|
|
**
|
|
** THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
** IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
**---------------------------------------------------------------------------
|
|
**
|
|
*/
|
|
|
|
|
|
#include "menu/menu.h"
|
|
#include "g_level.h"
|
|
#include "d_player.h"
|
|
#include "gi.h"
|
|
#include "c_bind.h"
|
|
#include "c_dispatch.h"
|
|
#include "gameconfigfile.h"
|
|
#include "w_wad.h"
|
|
|
|
TArray<FKeySection> KeySections;
|
|
extern TArray<FString> KeyConfWeapons;
|
|
|
|
static void LoadKeys (const char *modname, bool dbl)
|
|
{
|
|
char section[64];
|
|
|
|
mysnprintf (section, countof(section), "%s.%s%sBindings", gameinfo.ConfigName.GetChars(), modname,
|
|
dbl ? ".Double" : ".");
|
|
|
|
FKeyBindings *bindings = dbl? &DoubleBindings : &Bindings;
|
|
if (GameConfig->SetSection (section))
|
|
{
|
|
const char *key, *value;
|
|
while (GameConfig->NextInSection (key, value))
|
|
{
|
|
bindings->DoBind (key, value);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void DoSaveKeys (FConfigFile *config, const char *section, FKeySection *keysection, bool dbl)
|
|
{
|
|
config->SetSection (section, true);
|
|
config->ClearCurrentSection ();
|
|
FKeyBindings *bindings = dbl? &DoubleBindings : &Bindings;
|
|
for (unsigned i = 0; i < keysection->mActions.Size(); ++i)
|
|
{
|
|
bindings->ArchiveBindings (config, keysection->mActions[i].mAction);
|
|
}
|
|
}
|
|
|
|
void M_SaveCustomKeys (FConfigFile *config, char *section, char *subsection, size_t sublen)
|
|
{
|
|
for (unsigned i=0; i<KeySections.Size(); i++)
|
|
{
|
|
mysnprintf (subsection, sublen, "%s.Bindings", KeySections[i].mSection.GetChars());
|
|
DoSaveKeys (config, section, &KeySections[i], false);
|
|
mysnprintf (subsection, sublen, "%s.DoubleBindings", KeySections[i].mSection.GetChars());
|
|
DoSaveKeys (config, section, &KeySections[i], true);
|
|
}
|
|
}
|
|
|
|
static int CurrentKeySection = -1;
|
|
|
|
CCMD (addkeysection)
|
|
{
|
|
if (ParsingKeyConf)
|
|
{
|
|
if (argv.argc() != 3)
|
|
{
|
|
Printf ("Usage: addkeysection <menu section name> <ini name>\n");
|
|
return;
|
|
}
|
|
|
|
// Limit the ini name to 32 chars
|
|
if (strlen (argv[2]) > 32)
|
|
argv[2][32] = 0;
|
|
|
|
for (unsigned i = 0; i < KeySections.Size(); i++)
|
|
{
|
|
if (KeySections[i].mTitle.CompareNoCase(argv[2]) == 0)
|
|
{
|
|
CurrentKeySection = i;
|
|
return;
|
|
}
|
|
}
|
|
|
|
CurrentKeySection = KeySections.Reserve(1);
|
|
KeySections[CurrentKeySection].mTitle = argv[1];
|
|
KeySections[CurrentKeySection].mSection = argv[2];
|
|
// Load bindings for this section from the ini
|
|
LoadKeys (argv[2], 0);
|
|
LoadKeys (argv[2], 1);
|
|
}
|
|
}
|
|
|
|
CCMD (addmenukey)
|
|
{
|
|
if (ParsingKeyConf)
|
|
{
|
|
if (argv.argc() != 3)
|
|
{
|
|
Printf ("Usage: addmenukey <description> <command>\n");
|
|
return;
|
|
}
|
|
if (CurrentKeySection == -1 || CurrentKeySection >= (int)KeySections.Size())
|
|
{
|
|
Printf ("You must use addkeysection first.\n");
|
|
return;
|
|
}
|
|
|
|
FKeySection *sect = &KeySections[CurrentKeySection];
|
|
|
|
FKeyAction *act = §->mActions[sect->mActions.Reserve(1)];
|
|
act->mTitle = argv[1];
|
|
act->mAction = argv[2];
|
|
}
|
|
}
|
|
|
|
//==========================================================================
|
|
//
|
|
// D_LoadWadSettings
|
|
//
|
|
// Parses any loaded KEYCONF lumps. These are restricted console scripts
|
|
// that can only execute the alias, defaultbind, addkeysection,
|
|
// addmenukey, weaponsection, and addslotdefault commands.
|
|
//
|
|
//==========================================================================
|
|
|
|
void D_LoadWadSettings ()
|
|
{
|
|
char cmd[4096];
|
|
int lump, lastlump = 0;
|
|
|
|
ParsingKeyConf = true;
|
|
KeySections.Clear();
|
|
KeyConfWeapons.Clear();
|
|
|
|
while ((lump = Wads.FindLump ("KEYCONF", &lastlump)) != -1)
|
|
{
|
|
FMemLump data = Wads.ReadLump (lump);
|
|
const char *eof = (char *)data.GetMem() + Wads.LumpLength (lump);
|
|
const char *conf = (char *)data.GetMem();
|
|
|
|
while (conf < eof)
|
|
{
|
|
size_t i;
|
|
|
|
// Fetch a line to execute
|
|
for (i = 0; conf + i < eof && conf[i] != '\n'; ++i)
|
|
{
|
|
cmd[i] = conf[i];
|
|
}
|
|
if (i == 0)
|
|
{
|
|
conf++;
|
|
continue;
|
|
}
|
|
cmd[i] = 0;
|
|
conf += i;
|
|
if (*conf == '\n')
|
|
{
|
|
conf++;
|
|
}
|
|
|
|
// Comments begin with //
|
|
char *stop = cmd + i - 1;
|
|
char *comment = cmd;
|
|
int inQuote = 0;
|
|
|
|
if (*stop == '\r')
|
|
*stop-- = 0;
|
|
|
|
while (comment < stop)
|
|
{
|
|
if (*comment == '\"')
|
|
{
|
|
inQuote ^= 1;
|
|
}
|
|
else if (!inQuote && *comment == '/' && *(comment + 1) == '/')
|
|
{
|
|
break;
|
|
}
|
|
comment++;
|
|
}
|
|
if (comment == cmd)
|
|
{ // Comment at line beginning
|
|
continue;
|
|
}
|
|
else if (comment < stop)
|
|
{ // Comment in middle of line
|
|
*comment = 0;
|
|
}
|
|
|
|
AddCommandString (cmd);
|
|
}
|
|
}
|
|
ParsingKeyConf = false;
|
|
}
|
|
|