/* ** 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" #include "s_music.h" #include "i_interface.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 CheckForTexture(const FString& name, int type, int flags) { // ForceLookup is intentionally blocked here, this flag is for internal use only. return TexMan.CheckForTexture(name, static_cast(type), (flags & ~FTextureManager::TEXMAN_ForceLookup)).GetIndex(); } DEFINE_ACTION_FUNCTION_NATIVE(_TexMan, CheckForTexture, CheckForTexture) { PARAM_PROLOGUE; PARAM_STRING(name); PARAM_INT(type); PARAM_INT(flags); ACTION_RETURN_INT(CheckForTexture(name, type, flags)); } //========================================================================== // // // //========================================================================== 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())); } static int GetGlyphHeight(FFont* fnt, int code) { auto glyph = fnt->GetChar(code, CR_UNTRANSLATED, nullptr); return glyph ? (int)glyph->GetDisplayHeight() : 0; } DEFINE_ACTION_FUNCTION_NATIVE(FFont, GetGlyphHeight, GetGlyphHeight) { PARAM_SELF_STRUCT_PROLOGUE(FFont); PARAM_INT(code); ACTION_RETURN_INT(GetGlyphHeight(self, code)); } //========================================================================== // // 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, NameAllKeys) { PARAM_PROLOGUE; PARAM_POINTER(array, TArray); auto buffer = C_NameKeys(array->Data(), array->Size(), true); 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, GetAllKeysForCommand) { PARAM_SELF_STRUCT_PROLOGUE(FKeyBindings); PARAM_POINTER(array, TArray); PARAM_STRING(cmd); *array = self->GetKeysForCommand(cmd); return 0; } 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_GLOBAL_NAMED(mus_playing, musplaying); DEFINE_FIELD_X(MusPlayingInfo, MusPlayingInfo, name); DEFINE_FIELD_X(MusPlayingInfo, MusPlayingInfo, baseorder); DEFINE_FIELD_X(MusPlayingInfo, MusPlayingInfo, loop); DEFINE_GLOBAL_NAMED(PClass::AllClasses, AllClasses) DEFINE_GLOBAL(Bindings) DEFINE_GLOBAL(AutomapBindings) DEFINE_GLOBAL(generic_ui)