diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index ea5f328d1..96dd25e7f 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -1163,7 +1163,7 @@ set (PCH_SOURCES common/scripting/vm/vmexec.cpp common/scripting/vm/vmframe.cpp common/scripting/interface/stringformat.cpp - #common/scripting/interface/exports.cpp + common/scripting/interface/vmnatives.cpp common/scripting/frontend/ast.cpp common/scripting/frontend/zcc_compile.cpp common/scripting/frontend/zcc_parser.cpp diff --git a/src/common/scripting/interface/vmnatives.cpp b/src/common/scripting/interface/vmnatives.cpp new file mode 100644 index 000000000..03348124a --- /dev/null +++ b/src/common/scripting/interface/vmnatives.cpp @@ -0,0 +1,620 @@ +/* +** vmnatives.cpp +** +** VM exports for engine backend classes +** +**--------------------------------------------------------------------------- +** Copyright 2005-2020 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 "texturemanager.h" +#include "filesystem.h" +#include "c_console.h" +#include "c_cvars.h" +#include "c_bind.h" +#include "c_dispatch.h" +#include "templates.h" +#include "menu.h" +#include "vm.h" +#include "gstrings.h" +#include "printf.h" + + +//========================================================================== +// +// texture manager exports +// +//========================================================================== + +DEFINE_ACTION_FUNCTION(_TexMan, GetName) +{ + PARAM_PROLOGUE; + PARAM_INT(texid); + auto tex = TexMan.GameByIndex(texid); + FString retval; + + if (tex != nullptr) + { + if (tex->GetName().IsNotEmpty()) retval = tex->GetName(); + else + { + // Textures for full path names do not have their own name, they merely link to the source lump. + auto lump = tex->GetSourceLump(); + if (fileSystem.GetLinkedTexture(lump) == tex) + retval = fileSystem.GetFileFullName(lump); + } + } + ACTION_RETURN_STRING(retval); +} + +//========================================================================== +// +// +// +//========================================================================== + +static int GetTextureSize(int texid, int* py) +{ + auto tex = TexMan.GameByIndex(texid); + int x, y; + if (tex != nullptr) + { + x = int(0.5 + tex->GetDisplayWidth()); + y = int(0.5 + tex->GetDisplayHeight()); + } + else x = y = -1; + if (py) *py = y; + return x; +} + +DEFINE_ACTION_FUNCTION_NATIVE(_TexMan, GetSize, GetTextureSize) +{ + PARAM_PROLOGUE; + PARAM_INT(texid); + int x, y; + x = GetTextureSize(texid, &y); + if (numret > 0) ret[0].SetInt(x); + if (numret > 1) ret[1].SetInt(y); + return MIN(numret, 2); +} + +//========================================================================== +// +// +// +//========================================================================== +static void GetScaledSize(int texid, DVector2* pvec) +{ + auto tex = TexMan.GameByIndex(texid); + double x, y; + if (tex != nullptr) + { + x = tex->GetDisplayWidth(); + y = tex->GetDisplayHeight(); + } + else x = y = -1; + if (pvec) + { + pvec->X = x; + pvec->Y = y; + } +} + +DEFINE_ACTION_FUNCTION_NATIVE(_TexMan, GetScaledSize, GetScaledSize) +{ + PARAM_PROLOGUE; + PARAM_INT(texid); + DVector2 vec; + GetScaledSize(texid, &vec); + ACTION_RETURN_VEC2(vec); +} + +//========================================================================== +// +// +// +//========================================================================== +static void GetScaledOffset(int texid, DVector2* pvec) +{ + auto tex = TexMan.GameByIndex(texid); + double x, y; + if (tex != nullptr) + { + x = tex->GetDisplayLeftOffset(); + y = tex->GetDisplayTopOffset(); + } + else x = y = -1; + if (pvec) + { + pvec->X = x; + pvec->Y = y; + } +} + +DEFINE_ACTION_FUNCTION_NATIVE(_TexMan, GetScaledOffset, GetScaledOffset) +{ + PARAM_PROLOGUE; + PARAM_INT(texid); + DVector2 vec; + GetScaledOffset(texid, &vec); + ACTION_RETURN_VEC2(vec); +} + +//========================================================================== +// +// +// +//========================================================================== + +static int CheckRealHeight(int texid) +{ + auto tex = TexMan.GameByIndex(texid); + if (tex != nullptr) return tex->CheckRealHeight(); + else return -1; +} + +DEFINE_ACTION_FUNCTION_NATIVE(_TexMan, CheckRealHeight, CheckRealHeight) +{ + PARAM_PROLOGUE; + PARAM_INT(texid); + ACTION_RETURN_INT(CheckRealHeight(texid)); +} + +bool OkForLocalization(FTextureID texnum, const char* substitute); + +static int OkForLocalization_(int index, const FString& substitute) +{ + return OkForLocalization(FSetTextureID(index), substitute); +} + +DEFINE_ACTION_FUNCTION_NATIVE(_TexMan, OkForLocalization, OkForLocalization_) +{ + PARAM_PROLOGUE; + PARAM_INT(name); + PARAM_STRING(subst) + ACTION_RETURN_INT(OkForLocalization_(name, subst)); +} + + +//===================================================================================== +// +// FFont exports +// +//===================================================================================== + +static FFont *GetFont(int name) +{ + return V_GetFont(FName(ENamedName(name)).GetChars()); +} + +DEFINE_ACTION_FUNCTION_NATIVE(FFont, GetFont, GetFont) +{ + PARAM_PROLOGUE; + PARAM_INT(name); + ACTION_RETURN_POINTER(GetFont(name)); +} + +static FFont *FindFont(int name) +{ + return FFont::FindFont(FName(ENamedName(name))); +} + +DEFINE_ACTION_FUNCTION_NATIVE(FFont, FindFont, FindFont) +{ + PARAM_PROLOGUE; + PARAM_NAME(name); + ACTION_RETURN_POINTER(FFont::FindFont(name)); +} + +static int GetCharWidth(FFont *font, int code) +{ + return font->GetCharWidth(code); +} + +DEFINE_ACTION_FUNCTION_NATIVE(FFont, GetCharWidth, GetCharWidth) +{ + PARAM_SELF_STRUCT_PROLOGUE(FFont); + PARAM_INT(code); + ACTION_RETURN_INT(self->GetCharWidth(code)); +} + +static int GetHeight(FFont *font) +{ + return font->GetHeight(); +} + +DEFINE_ACTION_FUNCTION_NATIVE(FFont, GetHeight, GetHeight) +{ + PARAM_SELF_STRUCT_PROLOGUE(FFont); + ACTION_RETURN_INT(self->GetHeight()); +} + +static int GetDisplacement(FFont* font) +{ + return font->GetDisplacement(); +} + +DEFINE_ACTION_FUNCTION_NATIVE(FFont, GetDisplacement, GetDisplacement) +{ + PARAM_SELF_STRUCT_PROLOGUE(FFont); + ACTION_RETURN_INT(self->GetDisplacement()); +} + +double GetBottomAlignOffset(FFont *font, int c); +DEFINE_ACTION_FUNCTION_NATIVE(FFont, GetBottomAlignOffset, GetBottomAlignOffset) +{ + PARAM_SELF_STRUCT_PROLOGUE(FFont); + PARAM_INT(code); + ACTION_RETURN_FLOAT(GetBottomAlignOffset(self, code)); +} + +static int StringWidth(FFont *font, const FString &str) +{ + const char *txt = str[0] == '$' ? GStrings(&str[1]) : str.GetChars(); + return font->StringWidth(txt); +} + +DEFINE_ACTION_FUNCTION_NATIVE(FFont, StringWidth, StringWidth) +{ + PARAM_SELF_STRUCT_PROLOGUE(FFont); + PARAM_STRING(str); + ACTION_RETURN_INT(StringWidth(self, str)); +} + +static int GetMaxAscender(FFont* font, const FString& str) +{ + const char* txt = str[0] == '$' ? GStrings(&str[1]) : str.GetChars(); + return font->GetMaxAscender(txt); +} + +DEFINE_ACTION_FUNCTION_NATIVE(FFont, GetMaxAscender, GetMaxAscender) +{ + PARAM_SELF_STRUCT_PROLOGUE(FFont); + PARAM_STRING(str); + ACTION_RETURN_INT(GetMaxAscender(self, str)); +} + +static int CanPrint(FFont *font, const FString &str) +{ + const char *txt = str[0] == '$' ? GStrings(&str[1]) : str.GetChars(); + return font->CanPrint(txt); +} + +DEFINE_ACTION_FUNCTION_NATIVE(FFont, CanPrint, CanPrint) +{ + PARAM_SELF_STRUCT_PROLOGUE(FFont); + PARAM_STRING(str); + ACTION_RETURN_INT(CanPrint(self, str)); +} + +static int FindFontColor(int name) +{ + return V_FindFontColor(ENamedName(name)); +} + +DEFINE_ACTION_FUNCTION_NATIVE(FFont, FindFontColor, FindFontColor) +{ + PARAM_PROLOGUE; + PARAM_NAME(code); + ACTION_RETURN_INT((int)V_FindFontColor(code)); +} + +static void GetCursor(FFont *font, FString *result) +{ + *result = font->GetCursor(); +} + +DEFINE_ACTION_FUNCTION_NATIVE(FFont, GetCursor, GetCursor) +{ + PARAM_SELF_STRUCT_PROLOGUE(FFont); + ACTION_RETURN_STRING(FString(self->GetCursor())); +} + +//========================================================================== +// +// file system +// +//========================================================================== + +DEFINE_ACTION_FUNCTION(_Wads, GetNumLumps) +{ + PARAM_PROLOGUE; + ACTION_RETURN_INT(fileSystem.GetNumEntries()); +} + +DEFINE_ACTION_FUNCTION(_Wads, CheckNumForName) +{ + PARAM_PROLOGUE; + PARAM_STRING(name); + PARAM_INT(ns); + PARAM_INT(wadnum); + PARAM_BOOL(exact); + ACTION_RETURN_INT(fileSystem.CheckNumForName(name, ns, wadnum, exact)); +} + +DEFINE_ACTION_FUNCTION(_Wads, CheckNumForFullName) +{ + PARAM_PROLOGUE; + PARAM_STRING(name); + ACTION_RETURN_INT(fileSystem.CheckNumForFullName(name)); +} + +DEFINE_ACTION_FUNCTION(_Wads, FindLump) +{ + PARAM_PROLOGUE; + PARAM_STRING(name); + PARAM_INT(startlump); + PARAM_INT(ns); + const bool isLumpValid = startlump >= 0 && startlump < fileSystem.GetNumEntries(); + ACTION_RETURN_INT(isLumpValid ? fileSystem.FindLump(name, &startlump, 0 != ns) : -1); +} + +DEFINE_ACTION_FUNCTION(_Wads, GetLumpName) +{ + PARAM_PROLOGUE; + PARAM_INT(lump); + FString lumpname; + fileSystem.GetFileShortName(lumpname, lump); + ACTION_RETURN_STRING(lumpname); +} + +DEFINE_ACTION_FUNCTION(_Wads, GetLumpFullName) +{ + PARAM_PROLOGUE; + PARAM_INT(lump); + ACTION_RETURN_STRING(fileSystem.GetFileFullName(lump)); +} + +DEFINE_ACTION_FUNCTION(_Wads, GetLumpNamespace) +{ + PARAM_PROLOGUE; + PARAM_INT(lump); + ACTION_RETURN_INT(fileSystem.GetFileNamespace(lump)); +} + +DEFINE_ACTION_FUNCTION(_Wads, ReadLump) +{ + PARAM_PROLOGUE; + PARAM_INT(lump); + const bool isLumpValid = lump >= 0 && lump < fileSystem.GetNumEntries(); + ACTION_RETURN_STRING(isLumpValid ? fileSystem.ReadFile(lump).GetString() : FString()); +} + +//========================================================================== +// +// CVARs +// +//========================================================================== + +DEFINE_ACTION_FUNCTION(_CVar, GetInt) +{ + PARAM_SELF_STRUCT_PROLOGUE(FBaseCVar); + auto v = self->GetGenericRep(CVAR_Int); + ACTION_RETURN_INT(v.Int); +} + +DEFINE_ACTION_FUNCTION(_CVar, GetFloat) +{ + PARAM_SELF_STRUCT_PROLOGUE(FBaseCVar); + auto v = self->GetGenericRep(CVAR_Float); + ACTION_RETURN_FLOAT(v.Float); +} + +DEFINE_ACTION_FUNCTION(_CVar, GetString) +{ + PARAM_SELF_STRUCT_PROLOGUE(FBaseCVar); + auto v = self->GetGenericRep(CVAR_String); + ACTION_RETURN_STRING(v.String); +} + +DEFINE_ACTION_FUNCTION(_CVar, SetInt) +{ + // Only menus are allowed to change CVARs. + PARAM_SELF_STRUCT_PROLOGUE(FBaseCVar); + if (!(self->GetFlags() & CVAR_MOD)) + { + // Only menus are allowed to change non-mod CVARs. + if (DMenu::InMenu == 0) + { + ThrowAbortException(X_OTHER, "Attempt to change CVAR '%s' outside of menu code", self->GetName()); + } + } + PARAM_INT(val); + UCVarValue v; + v.Int = val; + self->SetGenericRep(v, CVAR_Int); + return 0; +} + +DEFINE_ACTION_FUNCTION(_CVar, SetFloat) +{ + PARAM_SELF_STRUCT_PROLOGUE(FBaseCVar); + if (!(self->GetFlags() & CVAR_MOD)) + { + // Only menus are allowed to change non-mod CVARs. + if (DMenu::InMenu == 0) + { + ThrowAbortException(X_OTHER, "Attempt to change CVAR '%s' outside of menu code", self->GetName()); + } + } + PARAM_FLOAT(val); + UCVarValue v; + v.Float = (float)val; + self->SetGenericRep(v, CVAR_Float); + return 0; +} + +DEFINE_ACTION_FUNCTION(_CVar, SetString) +{ + // Only menus are allowed to change CVARs. + PARAM_SELF_STRUCT_PROLOGUE(FBaseCVar); + if (!(self->GetFlags() & CVAR_MOD)) + { + // Only menus are allowed to change non-mod CVARs. + if (DMenu::InMenu == 0) + { + ThrowAbortException(X_OTHER, "Attempt to change CVAR '%s' outside of menu code", self->GetName()); + } + } + PARAM_STRING(val); + UCVarValue v; + v.String = val.GetChars(); + self->SetGenericRep(v, CVAR_String); + return 0; +} + +DEFINE_ACTION_FUNCTION(_CVar, GetRealType) +{ + PARAM_SELF_STRUCT_PROLOGUE(FBaseCVar); + ACTION_RETURN_INT(self->GetRealType()); +} + +DEFINE_ACTION_FUNCTION(_CVar, ResetToDefault) +{ + PARAM_SELF_STRUCT_PROLOGUE(FBaseCVar); + if (!(self->GetFlags() & CVAR_MOD)) + { + // Only menus are allowed to change non-mod CVARs. + if (DMenu::InMenu == 0) + { + ThrowAbortException(X_OTHER, "Attempt to change CVAR '%s' outside of menu code", self->GetName()); + } + } + + self->ResetToDefault(); + return 0; +} + +DEFINE_ACTION_FUNCTION(_CVar, FindCVar) +{ + PARAM_PROLOGUE; + PARAM_NAME(name); + ACTION_RETURN_POINTER(FindCVar(name.GetChars(), nullptr)); +} + +//============================================================================= +// +// +// +//============================================================================= + +DEFINE_ACTION_FUNCTION(FKeyBindings, SetBind) +{ + PARAM_SELF_STRUCT_PROLOGUE(FKeyBindings); + PARAM_INT(k); + PARAM_STRING(cmd); + + // Only menus are allowed to change bindings. + if (DMenu::InMenu == 0) + { + I_FatalError("Attempt to change key bindings outside of menu code to '%s'", cmd.GetChars()); + } + + + self->SetBind(k, cmd); + return 0; +} + +DEFINE_ACTION_FUNCTION(FKeyBindings, NameKeys) +{ + PARAM_PROLOGUE; + PARAM_INT(k1); + PARAM_INT(k2); + char buffer[120]; + C_NameKeys(buffer, k1, k2); + ACTION_RETURN_STRING(buffer); +} + +DEFINE_ACTION_FUNCTION(FKeyBindings, GetKeysForCommand) +{ + PARAM_SELF_STRUCT_PROLOGUE(FKeyBindings); + PARAM_STRING(cmd); + int k1, k2; + self->GetKeysForCommand(cmd.GetChars(), &k1, &k2); + if (numret > 0) ret[0].SetInt(k1); + if (numret > 1) ret[1].SetInt(k2); + return MIN(numret, 2); +} + +DEFINE_ACTION_FUNCTION(FKeyBindings, UnbindACommand) +{ + PARAM_SELF_STRUCT_PROLOGUE(FKeyBindings); + PARAM_STRING(cmd); + + // Only menus are allowed to change bindings. + if (DMenu::InMenu == 0) + { + I_FatalError("Attempt to unbind key bindings for '%s' outside of menu code", cmd.GetChars()); + } + + self->UnbindACommand(cmd); + return 0; +} + +// This is only accessible to the special menu item to run CCMDs. +DEFINE_ACTION_FUNCTION(DOptionMenuItemCommand, DoCommand) +{ + if (CurrentMenu == nullptr) return 0; + PARAM_PROLOGUE; + PARAM_STRING(cmd); + PARAM_BOOL(unsafe); + UnsafeExecutionScope scope(unsafe); + C_DoCommand(cmd); + return 0; +} + +DEFINE_ACTION_FUNCTION(_Console, MidPrint) +{ + PARAM_PROLOGUE; + PARAM_POINTER(fnt, FFont); + PARAM_STRING(text); + PARAM_BOOL(bold); + + const char* txt = text[0] == '$' ? GStrings(&text[1]) : text.GetChars(); + C_MidPrint(fnt, txt, bold); + return 0; +} + +DEFINE_ACTION_FUNCTION(_Console, HideConsole) +{ + C_HideConsole(); + return 0; +} + +DEFINE_ACTION_FUNCTION(_Console, Printf) +{ + PARAM_PROLOGUE; + PARAM_VA_POINTER(va_reginfo) // Get the hidden type information array + + FString s = FStringFormat(VM_ARGS_NAMES); + Printf("%s\n", s.GetChars()); + return 0; +} diff --git a/src/common/textures/texturemanager.h b/src/common/textures/texturemanager.h index 90ad35209..d9da9d20d 100644 --- a/src/common/textures/texturemanager.h +++ b/src/common/textures/texturemanager.h @@ -3,6 +3,7 @@ #include #include "tarray.h" #include "textureid.h" +#include "textures.h" #include "basics.h" #include "texmanip.h" #include "name.h" diff --git a/src/scripting/vmthunks.cpp b/src/scripting/vmthunks.cpp index 9af53b2b4..6b8343f0a 100644 --- a/src/scripting/vmthunks.cpp +++ b/src/scripting/vmthunks.cpp @@ -1621,162 +1621,6 @@ DEFINE_ACTION_FUNCTION_NATIVE(_TexMan, CheckForTexture, CheckForTexture) ACTION_RETURN_INT(CheckForTexture(name, type, flags)); } -//========================================================================== -// -// -// -//========================================================================== - -DEFINE_ACTION_FUNCTION(_TexMan, GetName) -{ - PARAM_PROLOGUE; - PARAM_INT(texid); - auto tex = TexMan.GameByIndex(texid); - FString retval; - - if (tex != nullptr) - { - if (tex->GetName().IsNotEmpty()) retval = tex->GetName(); - else - { - // Textures for full path names do not have their own name, they merely link to the source lump. - auto lump = tex->GetSourceLump(); - if (fileSystem.GetLinkedTexture(lump) == tex) - retval = fileSystem.GetFileFullName(lump); - } - } - ACTION_RETURN_STRING(retval); -} - -//========================================================================== -// -// -// -//========================================================================== - -static int GetTextureSize(int texid, int* py) -{ - auto tex = TexMan.GameByIndex(texid); - int x, y; - if (tex != nullptr) - { - x = int(0.5 + tex->GetDisplayWidth()); - y = int(0.5 + tex->GetDisplayHeight()); - } - else x = y = -1; - if (py) *py = y; - return x; -} - -DEFINE_ACTION_FUNCTION_NATIVE(_TexMan, GetSize, GetTextureSize) -{ - PARAM_PROLOGUE; - PARAM_INT(texid); - int x, y; - x = GetTextureSize(texid, &y); - if (numret > 0) ret[0].SetInt(x); - if (numret > 1) ret[1].SetInt(y); - return MIN(numret, 2); -} - -//========================================================================== -// -// -// -//========================================================================== -static void GetScaledSize(int texid, DVector2* pvec) -{ - auto tex = TexMan.GameByIndex(texid); - double x, y; - if (tex != nullptr) - { - x = tex->GetDisplayWidth(); - y = tex->GetDisplayHeight(); - } - else x = y = -1; - if (pvec) - { - pvec->X = x; - pvec->Y = y; - } -} - -DEFINE_ACTION_FUNCTION_NATIVE(_TexMan, GetScaledSize, GetScaledSize) -{ - PARAM_PROLOGUE; - PARAM_INT(texid); - DVector2 vec; - GetScaledSize(texid, &vec); - ACTION_RETURN_VEC2(vec); -} - -//========================================================================== -// -// -// -//========================================================================== -static void GetScaledOffset(int texid, DVector2* pvec) -{ - auto tex = TexMan.GameByIndex(texid); - double x, y; - if (tex != nullptr) - { - x = tex->GetDisplayLeftOffset(); - y = tex->GetDisplayTopOffset(); - } - else x = y = -1; - if (pvec) - { - pvec->X = x; - pvec->Y = y; - } -} - -DEFINE_ACTION_FUNCTION_NATIVE(_TexMan, GetScaledOffset, GetScaledOffset) -{ - PARAM_PROLOGUE; - PARAM_INT(texid); - DVector2 vec; - GetScaledOffset(texid, &vec); - ACTION_RETURN_VEC2(vec); -} - -//========================================================================== -// -// -// -//========================================================================== - -static int CheckRealHeight(int texid) -{ - auto tex = TexMan.GameByIndex(texid); - if (tex != nullptr) return tex->CheckRealHeight(); - else return -1; -} - -DEFINE_ACTION_FUNCTION_NATIVE(_TexMan, CheckRealHeight, CheckRealHeight) -{ - PARAM_PROLOGUE; - PARAM_INT(texid); - ACTION_RETURN_INT(CheckRealHeight(texid)); -} - -bool OkForLocalization(FTextureID texnum, const char* substitute); - -static int OkForLocalization_(int index, const FString& substitute) -{ - return OkForLocalization(FSetTextureID(index), substitute); -} - -DEFINE_ACTION_FUNCTION_NATIVE(_TexMan, OkForLocalization, OkForLocalization_) -{ - PARAM_PROLOGUE; - PARAM_INT(name); - PARAM_STRING(subst) - ACTION_RETURN_INT(OkForLocalization_(name, subst)); -} - - //===================================================================================== @@ -1909,140 +1753,6 @@ DEFINE_ACTION_FUNCTION_NATIVE(_Secplane, PointToDist, PointToDist) ACTION_RETURN_FLOAT(self->PointToDist(DVector2(x, y), z)); } -//===================================================================================== -// -// FFont exports -// -//===================================================================================== - -static FFont *GetFont(int name) -{ - return V_GetFont(FName(ENamedName(name)).GetChars()); -} - -DEFINE_ACTION_FUNCTION_NATIVE(FFont, GetFont, GetFont) -{ - PARAM_PROLOGUE; - PARAM_INT(name); - ACTION_RETURN_POINTER(GetFont(name)); -} - -static FFont *FindFont(int name) -{ - return FFont::FindFont(FName(ENamedName(name))); -} - -DEFINE_ACTION_FUNCTION_NATIVE(FFont, FindFont, FindFont) -{ - PARAM_PROLOGUE; - PARAM_NAME(name); - ACTION_RETURN_POINTER(FFont::FindFont(name)); -} - -static int GetCharWidth(FFont *font, int code) -{ - return font->GetCharWidth(code); -} - -DEFINE_ACTION_FUNCTION_NATIVE(FFont, GetCharWidth, GetCharWidth) -{ - PARAM_SELF_STRUCT_PROLOGUE(FFont); - PARAM_INT(code); - ACTION_RETURN_INT(self->GetCharWidth(code)); -} - -static int GetHeight(FFont *font) -{ - return font->GetHeight(); -} - -DEFINE_ACTION_FUNCTION_NATIVE(FFont, GetHeight, GetHeight) -{ - PARAM_SELF_STRUCT_PROLOGUE(FFont); - ACTION_RETURN_INT(self->GetHeight()); -} - -static int GetDisplacement(FFont* font) -{ - return font->GetDisplacement(); -} - -DEFINE_ACTION_FUNCTION_NATIVE(FFont, GetDisplacement, GetDisplacement) -{ - PARAM_SELF_STRUCT_PROLOGUE(FFont); - ACTION_RETURN_INT(self->GetDisplacement()); -} - -double GetBottomAlignOffset(FFont *font, int c); -DEFINE_ACTION_FUNCTION_NATIVE(FFont, GetBottomAlignOffset, GetBottomAlignOffset) -{ - PARAM_SELF_STRUCT_PROLOGUE(FFont); - PARAM_INT(code); - ACTION_RETURN_FLOAT(GetBottomAlignOffset(self, code)); -} - -static int StringWidth(FFont *font, const FString &str) -{ - const char *txt = str[0] == '$' ? GStrings(&str[1]) : str.GetChars(); - return font->StringWidth(txt); -} - -DEFINE_ACTION_FUNCTION_NATIVE(FFont, StringWidth, StringWidth) -{ - PARAM_SELF_STRUCT_PROLOGUE(FFont); - PARAM_STRING(str); - ACTION_RETURN_INT(StringWidth(self, str)); -} - -static int GetMaxAscender(FFont* font, const FString& str) -{ - const char* txt = str[0] == '$' ? GStrings(&str[1]) : str.GetChars(); - return font->GetMaxAscender(txt); -} - -DEFINE_ACTION_FUNCTION_NATIVE(FFont, GetMaxAscender, GetMaxAscender) -{ - PARAM_SELF_STRUCT_PROLOGUE(FFont); - PARAM_STRING(str); - ACTION_RETURN_INT(GetMaxAscender(self, str)); -} - -static int CanPrint(FFont *font, const FString &str) -{ - const char *txt = str[0] == '$' ? GStrings(&str[1]) : str.GetChars(); - return font->CanPrint(txt); -} - -DEFINE_ACTION_FUNCTION_NATIVE(FFont, CanPrint, CanPrint) -{ - PARAM_SELF_STRUCT_PROLOGUE(FFont); - PARAM_STRING(str); - ACTION_RETURN_INT(CanPrint(self, str)); -} - -static int FindFontColor(int name) -{ - return V_FindFontColor(ENamedName(name)); -} - -DEFINE_ACTION_FUNCTION_NATIVE(FFont, FindFontColor, FindFontColor) -{ - PARAM_PROLOGUE; - PARAM_NAME(code); - ACTION_RETURN_INT((int)V_FindFontColor(code)); -} - -static void GetCursor(FFont *font, FString *result) -{ - *result = font->GetCursor(); -} - -DEFINE_ACTION_FUNCTION_NATIVE(FFont, GetCursor, GetCursor) -{ - PARAM_SELF_STRUCT_PROLOGUE(FFont); - ACTION_RETURN_STRING(FString(self->GetCursor())); -} - //===================================================================================== // // WeaponSlots exports @@ -3000,188 +2710,6 @@ DEFINE_ACTION_FUNCTION_NATIVE(_AltHUD, GetLatency, Net_GetLatency) return numret; } -//========================================================================== -// -// file system -// -//========================================================================== - -DEFINE_ACTION_FUNCTION(_Wads, GetNumLumps) -{ - PARAM_PROLOGUE; - ACTION_RETURN_INT(fileSystem.GetNumEntries()); -} - -DEFINE_ACTION_FUNCTION(_Wads, CheckNumForName) -{ - PARAM_PROLOGUE; - PARAM_STRING(name); - PARAM_INT(ns); - PARAM_INT(wadnum); - PARAM_BOOL(exact); - ACTION_RETURN_INT(fileSystem.CheckNumForName(name, ns, wadnum, exact)); -} - -DEFINE_ACTION_FUNCTION(_Wads, CheckNumForFullName) -{ - PARAM_PROLOGUE; - PARAM_STRING(name); - ACTION_RETURN_INT(fileSystem.CheckNumForFullName(name)); -} - -DEFINE_ACTION_FUNCTION(_Wads, FindLump) -{ - PARAM_PROLOGUE; - PARAM_STRING(name); - PARAM_INT(startlump); - PARAM_INT(ns); - const bool isLumpValid = startlump >= 0 && startlump < fileSystem.GetNumEntries(); - ACTION_RETURN_INT(isLumpValid ? fileSystem.FindLump(name, &startlump, 0 != ns) : -1); -} - -DEFINE_ACTION_FUNCTION(_Wads, GetLumpName) -{ - PARAM_PROLOGUE; - PARAM_INT(lump); - FString lumpname; - fileSystem.GetFileShortName(lumpname, lump); - ACTION_RETURN_STRING(lumpname); -} - -DEFINE_ACTION_FUNCTION(_Wads, GetLumpFullName) -{ - PARAM_PROLOGUE; - PARAM_INT(lump); - ACTION_RETURN_STRING(fileSystem.GetFileFullName(lump)); -} - -DEFINE_ACTION_FUNCTION(_Wads, GetLumpNamespace) -{ - PARAM_PROLOGUE; - PARAM_INT(lump); - ACTION_RETURN_INT(fileSystem.GetFileNamespace(lump)); -} - -DEFINE_ACTION_FUNCTION(_Wads, ReadLump) -{ - PARAM_PROLOGUE; - PARAM_INT(lump); - const bool isLumpValid = lump >= 0 && lump < fileSystem.GetNumEntries(); - ACTION_RETURN_STRING(isLumpValid ? fileSystem.ReadFile(lump).GetString() : FString()); -} - -//========================================================================== -// -// CVARs -// -//========================================================================== - -DEFINE_ACTION_FUNCTION(_CVar, GetInt) -{ - PARAM_SELF_STRUCT_PROLOGUE(FBaseCVar); - auto v = self->GetGenericRep(CVAR_Int); - ACTION_RETURN_INT(v.Int); -} - -DEFINE_ACTION_FUNCTION(_CVar, GetFloat) -{ - PARAM_SELF_STRUCT_PROLOGUE(FBaseCVar); - auto v = self->GetGenericRep(CVAR_Float); - ACTION_RETURN_FLOAT(v.Float); -} - -DEFINE_ACTION_FUNCTION(_CVar, GetString) -{ - PARAM_SELF_STRUCT_PROLOGUE(FBaseCVar); - auto v = self->GetGenericRep(CVAR_String); - ACTION_RETURN_STRING(v.String); -} - -DEFINE_ACTION_FUNCTION(_CVar, SetInt) -{ - // Only menus are allowed to change CVARs. - PARAM_SELF_STRUCT_PROLOGUE(FBaseCVar); - if (!(self->GetFlags() & CVAR_MOD)) - { - // Only menus are allowed to change non-mod CVARs. - if (DMenu::InMenu == 0) - { - ThrowAbortException(X_OTHER, "Attempt to change CVAR '%s' outside of menu code", self->GetName()); - } - } - PARAM_INT(val); - UCVarValue v; - v.Int = val; - self->SetGenericRep(v, CVAR_Int); - return 0; -} - -DEFINE_ACTION_FUNCTION(_CVar, SetFloat) -{ - PARAM_SELF_STRUCT_PROLOGUE(FBaseCVar); - if (!(self->GetFlags() & CVAR_MOD)) - { - // Only menus are allowed to change non-mod CVARs. - if (DMenu::InMenu == 0) - { - ThrowAbortException(X_OTHER, "Attempt to change CVAR '%s' outside of menu code", self->GetName()); - } - } - PARAM_FLOAT(val); - UCVarValue v; - v.Float = (float)val; - self->SetGenericRep(v, CVAR_Float); - return 0; -} - -DEFINE_ACTION_FUNCTION(_CVar, SetString) -{ - // Only menus are allowed to change CVARs. - PARAM_SELF_STRUCT_PROLOGUE(FBaseCVar); - if (!(self->GetFlags() & CVAR_MOD)) - { - // Only menus are allowed to change non-mod CVARs. - if (DMenu::InMenu == 0) - { - ThrowAbortException(X_OTHER, "Attempt to change CVAR '%s' outside of menu code", self->GetName()); - } - } - PARAM_STRING(val); - UCVarValue v; - v.String = val.GetChars(); - self->SetGenericRep(v, CVAR_String); - return 0; -} - -DEFINE_ACTION_FUNCTION(_CVar, GetRealType) -{ - PARAM_SELF_STRUCT_PROLOGUE(FBaseCVar); - ACTION_RETURN_INT(self->GetRealType()); -} - -DEFINE_ACTION_FUNCTION(_CVar, ResetToDefault) -{ - PARAM_SELF_STRUCT_PROLOGUE(FBaseCVar); - if (!(self->GetFlags() & CVAR_MOD)) - { - // Only menus are allowed to change non-mod CVARs. - if (DMenu::InMenu == 0) - { - ThrowAbortException(X_OTHER, "Attempt to change CVAR '%s' outside of menu code", self->GetName()); - } - } - - self->ResetToDefault(); - return 0; -} - -DEFINE_ACTION_FUNCTION(_CVar, FindCVar) -{ - PARAM_PROLOGUE; - PARAM_NAME(name); - ACTION_RETURN_POINTER(FindCVar(name.GetChars(), nullptr)); -} - DEFINE_ACTION_FUNCTION(_CVar, GetCVar) { PARAM_PROLOGUE; @@ -3190,105 +2718,6 @@ DEFINE_ACTION_FUNCTION(_CVar, GetCVar) ACTION_RETURN_POINTER(GetCVar(plyr ? int(plyr - players) : -1, name.GetChars())); } -//============================================================================= -// -// -// -//============================================================================= - -DEFINE_ACTION_FUNCTION(FKeyBindings, SetBind) -{ - PARAM_SELF_STRUCT_PROLOGUE(FKeyBindings); - PARAM_INT(k); - PARAM_STRING(cmd); - - // Only menus are allowed to change bindings. - if (DMenu::InMenu == 0) - { - I_FatalError("Attempt to change key bindings outside of menu code to '%s'", cmd.GetChars()); - } - - - self->SetBind(k, cmd); - return 0; -} - -DEFINE_ACTION_FUNCTION(FKeyBindings, NameKeys) -{ - PARAM_PROLOGUE; - PARAM_INT(k1); - PARAM_INT(k2); - char buffer[120]; - C_NameKeys(buffer, k1, k2); - ACTION_RETURN_STRING(buffer); -} - -DEFINE_ACTION_FUNCTION(FKeyBindings, GetKeysForCommand) -{ - PARAM_SELF_STRUCT_PROLOGUE(FKeyBindings); - PARAM_STRING(cmd); - int k1, k2; - self->GetKeysForCommand(cmd.GetChars(), &k1, &k2); - if (numret > 0) ret[0].SetInt(k1); - if (numret > 1) ret[1].SetInt(k2); - return MIN(numret, 2); -} - -DEFINE_ACTION_FUNCTION(FKeyBindings, UnbindACommand) -{ - PARAM_SELF_STRUCT_PROLOGUE(FKeyBindings); - PARAM_STRING(cmd); - - // Only menus are allowed to change bindings. - if (DMenu::InMenu == 0) - { - I_FatalError("Attempt to unbind key bindings for '%s' outside of menu code", cmd.GetChars()); - } - - self->UnbindACommand(cmd); - return 0; -} - -// This is only accessible to the special menu item to run CCMDs. -DEFINE_ACTION_FUNCTION(DOptionMenuItemCommand, DoCommand) -{ - if (CurrentMenu == nullptr) return 0; - PARAM_PROLOGUE; - PARAM_STRING(cmd); - PARAM_BOOL(unsafe); - UnsafeExecutionScope scope(unsafe); - C_DoCommand(cmd); - return 0; -} - -DEFINE_ACTION_FUNCTION(_Console, MidPrint) -{ - PARAM_PROLOGUE; - PARAM_POINTER(fnt, FFont); - PARAM_STRING(text); - PARAM_BOOL(bold); - - const char* txt = text[0] == '$' ? GStrings(&text[1]) : text.GetChars(); - C_MidPrint(fnt, txt, bold); - return 0; -} - -DEFINE_ACTION_FUNCTION(_Console, HideConsole) -{ - C_HideConsole(); - return 0; -} - -DEFINE_ACTION_FUNCTION(_Console, Printf) -{ - PARAM_PROLOGUE; - PARAM_VA_POINTER(va_reginfo) // Get the hidden type information array - - FString s = FStringFormat(VM_ARGS_NAMES); - Printf("%s\n", s.GetChars()); - return 0; -} - DEFINE_ACTION_FUNCTION(DObject, S_ChangeMusic) {