/* ** 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 "menu.h" #include "vm.h" #include "gstrings.h" #include "printf.h" #include "s_music.h" #include "i_interface.h" #include "base_sbar.h" #include "image.h" #include "s_soundinternal.h" #include "i_time.h" #include "maps.h" #include "types.h" static ZSMap AllServices; static void MarkServices() { ZSMap::Iterator it(AllServices); ZSMap::Pair* pair; while (it.NextPair(pair)) { GC::Mark(pair->Value); } } void InitServices() { PClass* cls = PClass::FindClass("Service"); for (PClass* clss : PClass::AllClasses) { if (clss != cls && cls->IsAncestorOf(clss)) { DObject* obj = clss->CreateNew(); obj->ObjectFlags |= OF_Transient; AllServices.Insert(clss->TypeName, obj); } } GC::AddMarkerFunc(&MarkServices); } void ClearServices() { AllServices.Clear(); } //========================================================================== // // status bar exports // //========================================================================== static double StatusbarToRealCoords(DStatusBarCore* self, double x, double y, double w, double h, double* py, double* pw, double* ph) { self->StatusbarToRealCoords(x, y, w, h); *py = y; *pw = w; *ph = h; return x; } DEFINE_ACTION_FUNCTION_NATIVE(DStatusBarCore, StatusbarToRealCoords, StatusbarToRealCoords) { PARAM_SELF_PROLOGUE(DStatusBarCore); PARAM_FLOAT(x); PARAM_FLOAT(y); PARAM_FLOAT(w); PARAM_FLOAT(h); self->StatusbarToRealCoords(x, y, w, h); if (numret > 0) ret[0].SetFloat(x); if (numret > 1) ret[1].SetFloat(y); if (numret > 2) ret[2].SetFloat(w); if (numret > 3) ret[3].SetFloat(h); return min(4, numret); } void SBar_DrawTexture(DStatusBarCore* self, int texid, double x, double y, int flags, double alpha, double w, double h, double scaleX, double scaleY, int style, int color, int translation, double clipwidth) { if (!twod->HasBegun2D()) ThrowAbortException(X_OTHER, "Attempt to draw to screen outside a draw function"); self->DrawGraphic(FSetTextureID(texid), x, y, flags, alpha, w, h, scaleX, scaleY, ERenderStyle(style), color, translation, clipwidth); } DEFINE_ACTION_FUNCTION_NATIVE(DStatusBarCore, DrawTexture, SBar_DrawTexture) { PARAM_SELF_PROLOGUE(DStatusBarCore); PARAM_INT(texid); PARAM_FLOAT(x); PARAM_FLOAT(y); PARAM_INT(flags); PARAM_FLOAT(alpha); PARAM_FLOAT(w); PARAM_FLOAT(h); PARAM_FLOAT(scaleX); PARAM_FLOAT(scaleY); PARAM_INT(style); PARAM_INT(col); PARAM_INT(trans); PARAM_FLOAT(clipwidth); SBar_DrawTexture(self, texid, x, y, flags, alpha, w, h, scaleX, scaleY, style, col, trans, clipwidth); return 0; } void SBar_DrawImage(DStatusBarCore* self, const FString& texid, double x, double y, int flags, double alpha, double w, double h, double scaleX, double scaleY, int style, int color, int translation, double clipwidth) { if (!twod->HasBegun2D()) ThrowAbortException(X_OTHER, "Attempt to draw to screen outside a draw function"); self->DrawGraphic(TexMan.CheckForTexture(texid.GetChars(), ETextureType::Any), x, y, flags, alpha, w, h, scaleX, scaleY, ERenderStyle(style), color, translation, clipwidth); } DEFINE_ACTION_FUNCTION_NATIVE(DStatusBarCore, DrawImage, SBar_DrawImage) { PARAM_SELF_PROLOGUE(DStatusBarCore); PARAM_STRING(texid); PARAM_FLOAT(x); PARAM_FLOAT(y); PARAM_INT(flags); PARAM_FLOAT(alpha); PARAM_FLOAT(w); PARAM_FLOAT(h); PARAM_FLOAT(scaleX); PARAM_FLOAT(scaleY); PARAM_INT(style); PARAM_INT(col); PARAM_INT(trans); PARAM_FLOAT(clipwidth); SBar_DrawImage(self, texid, x, y, flags, alpha, w, h, scaleX, scaleY, style, col, trans, clipwidth); return 0; } void SBar_DrawImageRotated(DStatusBarCore* self, const FString& texid, double x, double y, int flags, double angle, double alpha, double scaleX, double scaleY, int style, int color, int translation) { if (!twod->HasBegun2D()) ThrowAbortException(X_OTHER, "Attempt to draw to screen outside a draw function"); self->DrawRotated(TexMan.CheckForTexture(texid.GetChars(), ETextureType::Any), x, y, flags, angle, alpha, scaleX, scaleY, color, translation, (ERenderStyle)style); } DEFINE_ACTION_FUNCTION_NATIVE(DStatusBarCore, DrawImageRotated, SBar_DrawImageRotated) { PARAM_SELF_PROLOGUE(DStatusBarCore); PARAM_STRING(texid); PARAM_FLOAT(x); PARAM_FLOAT(y); PARAM_INT(flags); PARAM_FLOAT(angle); PARAM_FLOAT(alpha); PARAM_FLOAT(scaleX); PARAM_FLOAT(scaleY); PARAM_INT(style); PARAM_INT(col); PARAM_INT(trans); SBar_DrawImageRotated(self, texid, x, y, flags, angle, alpha, scaleX, scaleY, style, col, trans); return 0; } void SBar_DrawTextureRotated(DStatusBarCore* self, int texid, double x, double y, int flags, double angle, double alpha, double scaleX, double scaleY, int style, int color, int translation) { if (!twod->HasBegun2D()) ThrowAbortException(X_OTHER, "Attempt to draw to screen outside a draw function"); self->DrawRotated(FSetTextureID(texid), x, y, flags, angle, alpha, scaleX, scaleY, color, translation, (ERenderStyle)style); } DEFINE_ACTION_FUNCTION_NATIVE(DStatusBarCore, DrawTextureRotated, SBar_DrawTextureRotated) { PARAM_SELF_PROLOGUE(DStatusBarCore); PARAM_INT(texid); PARAM_FLOAT(x); PARAM_FLOAT(y); PARAM_INT(flags); PARAM_FLOAT(angle); PARAM_FLOAT(alpha); PARAM_FLOAT(scaleX); PARAM_FLOAT(scaleY); PARAM_INT(style); PARAM_INT(col); PARAM_INT(trans); SBar_DrawTextureRotated(self, texid, x, y, flags, angle, alpha, scaleX, scaleY, style, col, trans); return 0; } void SBar_DrawString(DStatusBarCore* self, DHUDFont* font, const FString& string, double x, double y, int flags, int trans, double alpha, int wrapwidth, int linespacing, double scaleX, double scaleY, int translation, int style); DEFINE_ACTION_FUNCTION_NATIVE(DStatusBarCore, DrawString, SBar_DrawString) { PARAM_SELF_PROLOGUE(DStatusBarCore); PARAM_POINTER_NOT_NULL(font, DHUDFont); PARAM_STRING(string); PARAM_FLOAT(x); PARAM_FLOAT(y); PARAM_INT(flags); PARAM_INT(trans); PARAM_FLOAT(alpha); PARAM_INT(wrapwidth); PARAM_INT(linespacing); PARAM_FLOAT(scaleX); PARAM_FLOAT(scaleY); PARAM_INT(pt); PARAM_INT(style); SBar_DrawString(self, font, string, x, y, flags, trans, alpha, wrapwidth, linespacing, scaleX, scaleY, pt, style); return 0; } static double SBar_TransformRect(DStatusBarCore* self, double x, double y, double w, double h, int flags, double* py, double* pw, double* ph) { self->TransformRect(x, y, w, h, flags); *py = y; *pw = w; *ph = h; return x; } DEFINE_ACTION_FUNCTION_NATIVE(DStatusBarCore, TransformRect, SBar_TransformRect) { PARAM_SELF_PROLOGUE(DStatusBarCore); PARAM_FLOAT(x); PARAM_FLOAT(y); PARAM_FLOAT(w); PARAM_FLOAT(h); PARAM_INT(flags); self->TransformRect(x, y, w, h, flags); if (numret > 0) ret[0].SetFloat(x); if (numret > 1) ret[1].SetFloat(y); if (numret > 2) ret[2].SetFloat(w); if (numret > 3) ret[3].SetFloat(h); return min(4, numret); } static void SBar_Fill(DStatusBarCore* self, int color, double x, double y, double w, double h, int flags) { if (!twod->HasBegun2D()) ThrowAbortException(X_OTHER, "Attempt to draw to screen outside a draw function"); self->Fill(color, x, y, w, h, flags); } DEFINE_ACTION_FUNCTION_NATIVE(DStatusBarCore, Fill, SBar_Fill) { PARAM_SELF_PROLOGUE(DStatusBarCore); PARAM_COLOR(color); PARAM_FLOAT(x); PARAM_FLOAT(y); PARAM_FLOAT(w); PARAM_FLOAT(h); PARAM_INT(flags); SBar_Fill(self, color, x, y, w, h, flags); return 0; } static void SBar_SetClipRect(DStatusBarCore* self, double x, double y, double w, double h, int flags) { self->SetClipRect(x, y, w, h, flags); } DEFINE_ACTION_FUNCTION_NATIVE(DStatusBarCore, SetClipRect, SBar_SetClipRect) { PARAM_SELF_PROLOGUE(DStatusBarCore); PARAM_FLOAT(x); PARAM_FLOAT(y); PARAM_FLOAT(w); PARAM_FLOAT(h); PARAM_INT(flags); self->SetClipRect(x, y, w, h, flags); return 0; } void FormatNumber(int number, int minsize, int maxsize, int flags, const FString& prefix, FString* result); DEFINE_ACTION_FUNCTION_NATIVE(DStatusBarCore, FormatNumber, FormatNumber) { PARAM_PROLOGUE; PARAM_INT(number); PARAM_INT(minsize); PARAM_INT(maxsize); PARAM_INT(flags); PARAM_STRING(prefix); FString fmt; FormatNumber(number, minsize, maxsize, flags, prefix, &fmt); ACTION_RETURN_STRING(fmt); } static void SBar_SetSize(DStatusBarCore* self, int rt, int vw, int vh, int hvw, int hvh) { self->SetSize(rt, vw, vh, hvw, hvh); } DEFINE_ACTION_FUNCTION_NATIVE(DStatusBarCore, SetSize, SBar_SetSize) { PARAM_SELF_PROLOGUE(DStatusBarCore); PARAM_INT(rt); PARAM_INT(vw); PARAM_INT(vh); PARAM_INT(hvw); PARAM_INT(hvh); self->SetSize(rt, vw, vh, hvw, hvh); return 0; } static void SBar_GetHUDScale(DStatusBarCore* self, DVector2* result) { *result = self->GetHUDScale(); } DEFINE_ACTION_FUNCTION_NATIVE(DStatusBarCore, GetHUDScale, SBar_GetHUDScale) { PARAM_SELF_PROLOGUE(DStatusBarCore); ACTION_RETURN_VEC2(self->GetHUDScale()); } static void BeginStatusBar(DStatusBarCore* self, bool fs, int w, int h, int r) { self->BeginStatusBar(w, h, r, fs); } DEFINE_ACTION_FUNCTION_NATIVE(DStatusBarCore, BeginStatusBar, BeginStatusBar) { PARAM_SELF_PROLOGUE(DStatusBarCore); PARAM_BOOL(fs); PARAM_INT(w); PARAM_INT(h); PARAM_INT(r); self->BeginStatusBar(w, h, r, fs); return 0; } static void BeginHUD(DStatusBarCore* self, double a, bool fs, int w, int h) { self->BeginHUD(w, h, a, fs); } DEFINE_ACTION_FUNCTION_NATIVE(DStatusBarCore, BeginHUD, BeginHUD) { PARAM_SELF_PROLOGUE(DStatusBarCore); PARAM_FLOAT(a); PARAM_BOOL(fs); PARAM_INT(w); PARAM_INT(h); self->BeginHUD(w, h, a, fs); return 0; } //===================================================================================== // // // //===================================================================================== DHUDFont* CreateHudFont(FFont* fnt, int spac, int mono, int sx, int sy) { return Create(fnt, spac, EMonospacing(mono), sy, sy); } DEFINE_ACTION_FUNCTION_NATIVE(DHUDFont, Create, CreateHudFont) { PARAM_PROLOGUE; PARAM_POINTER(fnt, FFont); PARAM_INT(spac); PARAM_INT(mono); PARAM_INT(sx); PARAM_INT(sy); ACTION_RETURN_POINTER(CreateHudFont(fnt, spac, mono, sy, sy)); } //========================================================================== // // 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 (TexMan.GetLinkedTexture(lump) == tex) retval = fileSystem.GetFileFullName(lump); } } ACTION_RETURN_STRING(retval); } static int CheckForTexture(const FString& name, int type, int flags) { return TexMan.CheckForTexture(name.GetChars(), static_cast(type), flags).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)); } static int OkForLocalization_(int index, const FString& substitute) { return sysCallbacks.OkForLocalization? sysCallbacks.OkForLocalization(FSetTextureID(index), substitute.GetChars()) : false; } DEFINE_ACTION_FUNCTION_NATIVE(_TexMan, OkForLocalization, OkForLocalization_) { PARAM_PROLOGUE; PARAM_INT(name); PARAM_STRING(subst) ACTION_RETURN_INT(OkForLocalization_(name, subst)); } static int UseGamePalette(int index) { auto tex = TexMan.GameByIndex(index, false); if (!tex) return false; auto image = tex->GetTexture()->GetImage(); return image ? image->UseGamePalette() : false; } DEFINE_ACTION_FUNCTION_NATIVE(_TexMan, UseGamePalette, UseGamePalette) { PARAM_PROLOGUE; PARAM_INT(texid); ACTION_RETURN_INT(UseGamePalette(texid)); } FCanvas* GetTextureCanvas(const FString& texturename); DEFINE_ACTION_FUNCTION(_TexMan, GetCanvas) { PARAM_PROLOGUE; PARAM_STRING(texturename); ACTION_RETURN_POINTER(GetTextureCanvas(texturename)); } //===================================================================================== // // 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, int localize) { const char *txt = (localize && 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); PARAM_BOOL(localize); ACTION_RETURN_INT(StringWidth(self, str, localize)); } static int GetMaxAscender(FFont* font, const FString& str, int localize) { const char* txt = (localize && 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); PARAM_BOOL(localize); ACTION_RETURN_INT(GetMaxAscender(self, str, localize)); } static int CanPrint(FFont *font, const FString &str, int localize) { const char *txt = (localize && 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); PARAM_BOOL(localize); ACTION_RETURN_INT(CanPrint(self, str, localize)); } 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)); } static int GetDefaultKerning(FFont* font) { return font->GetDefaultKerning(); } DEFINE_ACTION_FUNCTION_NATIVE(FFont, GetDefaultKerning, GetDefaultKerning) { PARAM_SELF_STRUCT_PROLOGUE(FFont); ACTION_RETURN_INT(self->GetDefaultKerning()); } static double GetDisplayTopOffset(FFont* font, int c) { auto texc = font->GetChar(c, CR_UNDEFINED, nullptr); return texc ? texc->GetDisplayTopOffset() : 0; } DEFINE_ACTION_FUNCTION_NATIVE(FFont, GetDisplayTopOffset, GetDisplayTopOffset) { PARAM_SELF_STRUCT_PROLOGUE(FFont); PARAM_INT(code); ACTION_RETURN_FLOAT(GetDisplayTopOffset(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.GetChars(), ns, wadnum, exact)); } DEFINE_ACTION_FUNCTION(_Wads, CheckNumForFullName) { PARAM_PROLOGUE; PARAM_STRING(name); ACTION_RETURN_INT(fileSystem.CheckNumForFullName(name.GetChars())); } 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.GetChars(), &startlump, 0 != ns) : -1); } DEFINE_ACTION_FUNCTION(_Wads, FindLumpFullName) { PARAM_PROLOGUE; PARAM_STRING(name); PARAM_INT(startlump); PARAM_BOOL(noext); const bool isLumpValid = startlump >= 0 && startlump < fileSystem.GetNumEntries(); ACTION_RETURN_INT(isLumpValid ? fileSystem.FindLumpFullName(name.GetChars(), &startlump, noext) : -1); } DEFINE_ACTION_FUNCTION(_Wads, GetLumpName) { PARAM_PROLOGUE; PARAM_INT(lump); ACTION_RETURN_STRING(fileSystem.GetFileShortName(lump)); } 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 ? GetStringFromLump(lump, false) : FString()); } DEFINE_ACTION_FUNCTION(_Wads, GetLumpLength) { PARAM_PROLOGUE; PARAM_INT(lump); ACTION_RETURN_INT((int)fileSystem.FileLength(lump)); } //========================================================================== // // 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; if(self->GetFlags() & CVAR_ZS_CUSTOM_CLONE) { auto realCVar = (FBaseCVar*)(self->GetExtraDataPointer()); assert(realCVar->GetFlags() & CVAR_ZS_CUSTOM); v = realCVar->GenericZSCVarCallback(v, CVAR_Int); self->SetGenericRep(v, realCVar->GetRealType()); if(realCVar->GetRealType() == CVAR_String) delete[] v.String; } else { 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; if(self->GetFlags() & CVAR_ZS_CUSTOM_CLONE) { auto realCVar = (FBaseCVar*)(self->GetExtraDataPointer()); assert(realCVar->GetFlags() & CVAR_ZS_CUSTOM); v = realCVar->GenericZSCVarCallback(v, CVAR_Float); self->SetGenericRep(v, realCVar->GetRealType()); if(realCVar->GetRealType() == CVAR_String) delete[] v.String; } else { 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(); if(self->GetFlags() & CVAR_ZS_CUSTOM_CLONE) { auto realCVar = (FBaseCVar*)(self->GetExtraDataPointer()); assert(realCVar->GetFlags() & CVAR_ZS_CUSTOM); v = realCVar->GenericZSCVarCallback(v, CVAR_String); self->SetGenericRep(v, realCVar->GetRealType()); if(realCVar->GetRealType() == CVAR_String) delete[] v.String; } else { 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.GetChars()); 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); PARAM_BOOL(color); auto buffer = C_NameKeys(array->Data(), array->Size(), color); 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.GetChars()); return 0; } DEFINE_ACTION_FUNCTION(FKeyBindings, GetBinding) { PARAM_SELF_STRUCT_PROLOGUE(FKeyBindings); PARAM_INT(key); ACTION_RETURN_STRING(self->GetBinding(key)); } 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.GetChars()); return 0; } // This is only accessible to the special menu item to run CCMDs. DEFINE_ACTION_FUNCTION(DOptionMenuItemCommand, DoCommand) { PARAM_PROLOGUE; PARAM_STRING(cmd); PARAM_BOOL(unsafe); // Only menus are allowed to execute CCMDs. if (DMenu::InMenu == 0) { I_FatalError("Attempt to execute CCMD '%s' outside of menu code", cmd.GetChars()); } UnsafeExecutionScope scope(unsafe); AddCommandString(cmd.GetChars()); 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(_Console, PrintfEx) { PARAM_PROLOGUE; PARAM_INT(printlevel); PARAM_VA_POINTER(va_reginfo) // Get the hidden type information array FString s = FStringFormat(VM_ARGS_NAMES,1); Printf(printlevel,"%s\n", s.GetChars()); return 0; } static void StopAllSounds() { soundEngine->StopAllChannels(); } DEFINE_ACTION_FUNCTION_NATIVE(_System, StopAllSounds, StopAllSounds) { StopAllSounds(); return 0; } static int PlayMusic(const FString& musname, int order, int looped) { return S_ChangeMusic(musname.GetChars(), order, !!looped, true); } DEFINE_ACTION_FUNCTION_NATIVE(_System, PlayMusic, PlayMusic) { PARAM_PROLOGUE; PARAM_STRING(name); PARAM_INT(order); PARAM_BOOL(looped); ACTION_RETURN_BOOL(PlayMusic(name, order, looped)); } static void Mus_Stop() { S_StopMusic(true); } DEFINE_ACTION_FUNCTION_NATIVE(_System, StopMusic, Mus_Stop) { Mus_Stop(); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(_System, SoundEnabled, SoundEnabled) { ACTION_RETURN_INT(SoundEnabled()); } DEFINE_ACTION_FUNCTION_NATIVE(_System, MusicEnabled, MusicEnabled) { ACTION_RETURN_INT(MusicEnabled()); } static double Jit_GetTimeFrac() // cannot use I_GetTimwfrac directly due to default arguments. { return I_GetTimeFrac(); } DEFINE_ACTION_FUNCTION_NATIVE(_System, GetTimeFrac, Jit_GetTimeFrac) { ACTION_RETURN_FLOAT(I_GetTimeFrac()); } 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_FIELD_X(MusPlayingInfo, MusPlayingInfo, handle); DEFINE_GLOBAL_NAMED(PClass::AllClasses, AllClasses) DEFINE_GLOBAL(AllServices) DEFINE_GLOBAL(Bindings) DEFINE_GLOBAL(AutomapBindings) DEFINE_GLOBAL(generic_ui) DEFINE_FIELD(DStatusBarCore, RelTop); DEFINE_FIELD(DStatusBarCore, HorizontalResolution); DEFINE_FIELD(DStatusBarCore, VerticalResolution); DEFINE_FIELD(DStatusBarCore, CompleteBorder); DEFINE_FIELD(DStatusBarCore, Alpha); DEFINE_FIELD(DStatusBarCore, drawOffset); DEFINE_FIELD(DStatusBarCore, drawClip); DEFINE_FIELD(DStatusBarCore, fullscreenOffsets); DEFINE_FIELD(DStatusBarCore, defaultScale); DEFINE_FIELD(DHUDFont, mFont); // // Quaternion void QuatFromAngles(double yaw, double pitch, double roll, DQuaternion* pquat) { *pquat = DQuaternion::FromAngles(DAngle::fromDeg(yaw), DAngle::fromDeg(pitch), DAngle::fromDeg(roll)); } DEFINE_ACTION_FUNCTION_NATIVE(_QuatStruct, FromAngles, QuatFromAngles) { PARAM_PROLOGUE; PARAM_FLOAT(yaw); PARAM_FLOAT(pitch); PARAM_FLOAT(roll); DQuaternion quat; QuatFromAngles(yaw, pitch, roll, &quat); ACTION_RETURN_QUAT(quat); } void QuatAxisAngle(double x, double y, double z, double angleDeg, DQuaternion* pquat) { auto axis = DVector3(x, y, z); auto angle = DAngle::fromDeg(angleDeg); *pquat = DQuaternion::AxisAngle(axis, angle); } DEFINE_ACTION_FUNCTION_NATIVE(_QuatStruct, AxisAngle, QuatAxisAngle) { PARAM_PROLOGUE; PARAM_FLOAT(x); PARAM_FLOAT(y); PARAM_FLOAT(z); PARAM_FLOAT(angle); DQuaternion quat; QuatAxisAngle(x, y, z, angle, &quat); ACTION_RETURN_QUAT(quat); } void QuatNLerp( double ax, double ay, double az, double aw, double bx, double by, double bz, double bw, double t, DQuaternion* pquat ) { auto from = DQuaternion { ax, ay, az, aw }; auto to = DQuaternion { bx, by, bz, bw }; *pquat = DQuaternion::NLerp(from, to, t); } DEFINE_ACTION_FUNCTION_NATIVE(_QuatStruct, NLerp, QuatNLerp) { PARAM_PROLOGUE; PARAM_FLOAT(ax); PARAM_FLOAT(ay); PARAM_FLOAT(az); PARAM_FLOAT(aw); PARAM_FLOAT(bx); PARAM_FLOAT(by); PARAM_FLOAT(bz); PARAM_FLOAT(bw); PARAM_FLOAT(t); DQuaternion quat; QuatNLerp(ax, ay, az, aw, bx, by, bz, bw, t, &quat); ACTION_RETURN_QUAT(quat); } void QuatSLerp( double ax, double ay, double az, double aw, double bx, double by, double bz, double bw, double t, DQuaternion* pquat ) { auto from = DQuaternion { ax, ay, az, aw }; auto to = DQuaternion { bx, by, bz, bw }; *pquat = DQuaternion::SLerp(from, to, t); } DEFINE_ACTION_FUNCTION_NATIVE(_QuatStruct, SLerp, QuatSLerp) { PARAM_PROLOGUE; PARAM_FLOAT(ax); PARAM_FLOAT(ay); PARAM_FLOAT(az); PARAM_FLOAT(aw); PARAM_FLOAT(bx); PARAM_FLOAT(by); PARAM_FLOAT(bz); PARAM_FLOAT(bw); PARAM_FLOAT(t); DQuaternion quat; QuatSLerp(ax, ay, az, aw, bx, by, bz, bw, t, &quat); ACTION_RETURN_QUAT(quat); } void QuatConjugate( double x, double y, double z, double w, DQuaternion* pquat ) { *pquat = DQuaternion(x, y, z, w).Conjugate(); } DEFINE_ACTION_FUNCTION_NATIVE(_QuatStruct, Conjugate, QuatConjugate) { PARAM_SELF_STRUCT_PROLOGUE(DQuaternion); DQuaternion quat; QuatConjugate(self->X, self->Y, self->Z, self->W, &quat); ACTION_RETURN_QUAT(quat); } void QuatInverse( double x, double y, double z, double w, DQuaternion* pquat ) { *pquat = DQuaternion(x, y, z, w).Inverse(); } DEFINE_ACTION_FUNCTION_NATIVE(_QuatStruct, Inverse, QuatInverse) { PARAM_SELF_STRUCT_PROLOGUE(DQuaternion); DQuaternion quat; QuatInverse(self->X, self->Y, self->Z, self->W, &quat); ACTION_RETURN_QUAT(quat); } PFunction * FindFunctionPointer(PClass * cls, int fn_name) { auto fn = dyn_cast(cls->FindSymbol(ENamedName(fn_name), true)); return (fn && (fn->Variants[0].Flags & (VARF_Action | VARF_Virtual)) == 0 ) ? fn : nullptr; } DEFINE_ACTION_FUNCTION_NATIVE(DObject, FindFunction, FindFunctionPointer) { PARAM_PROLOGUE; PARAM_CLASS(cls, DObject); PARAM_NAME(fn); ACTION_RETURN_POINTER(FindFunctionPointer(cls, fn.GetIndex())); } FTranslationID R_FindCustomTranslation(FName name); static int ZFindTranslation(int intname) { return R_FindCustomTranslation(ENamedName(intname)).index(); } static int MakeTransID(int g, int s) { return TRANSLATION(g, s).index(); } DEFINE_ACTION_FUNCTION_NATIVE(_Translation, GetID, ZFindTranslation) { PARAM_PROLOGUE; PARAM_INT(t); ACTION_RETURN_INT(ZFindTranslation(t)); } // same as above for the compiler which needs a class to look this up. DEFINE_ACTION_FUNCTION_NATIVE(DObject, BuiltinFindTranslation, ZFindTranslation) { PARAM_PROLOGUE; PARAM_INT(t); ACTION_RETURN_INT(ZFindTranslation(t)); } DEFINE_ACTION_FUNCTION_NATIVE(_Translation, MakeID, MakeTransID) { PARAM_PROLOGUE; PARAM_INT(g); PARAM_INT(t); ACTION_RETURN_INT(MakeTransID(g, t)); } // DObject-based wrapper around FScanner, for ZScript. class DScriptScanner : public DObject { DECLARE_CLASS(DScriptScanner, DObject) public: FScanner wrapped; }; IMPLEMENT_CLASS(DScriptScanner, false, false); static void OpenLumpNum(DScriptScanner* self, int lump) { return self->wrapped.OpenLumpNum(lump); } static void OpenString(DScriptScanner* self, const FString* name, FString* script) { return self->wrapped.OpenString(name->GetChars(), *script); } static void SavePos(DScriptScanner* self, FScanner::SavedPos *pos) { *pos = self->wrapped.SavePos(); } static void RestorePos(DScriptScanner* self, const FScanner::SavedPos* pos) { return self->wrapped.RestorePos(*pos); } static void GetStringContents(DScriptScanner* self, FString* str) { *str = self->wrapped.String; } static void UnGet(DScriptScanner* self) { return self->wrapped.UnGet(); } static int isText(DScriptScanner* self) { return self->wrapped.isText(); } static int GetMessageLine(DScriptScanner* self) { return self->wrapped.GetMessageLine(); } static void Close(DScriptScanner* self) { return self->wrapped.Close(); } static void SetCMode(DScriptScanner* self, int cmode) { return self->wrapped.SetCMode(cmode); } static void SetNoOctals(DScriptScanner* self, int cmode) { return self->wrapped.SetNoOctals(cmode); } static void SetEscape(DScriptScanner* self, int esc) { return self->wrapped.SetNoOctals(esc); } static void SetNoFatalErrors(DScriptScanner* self, int cmode) { return self->wrapped.SetNoFatalErrors(cmode); } static void AddSymbolUint(DScriptScanner* self, const FString* name, uint32_t value) { return self->wrapped.AddSymbol(name->GetChars(), value); } static void AddSymbolInt(DScriptScanner* self, const FString* name, int32_t value) { return self->wrapped.AddSymbol(name->GetChars(), value); } static void AddSymbolDouble(DScriptScanner* self, const FString* name, double value) { return self->wrapped.AddSymbol(name->GetChars(), value); } static int GetString(DScriptScanner* self) { return self->wrapped.GetString(); } static int GetNumber(DScriptScanner* self, int evaluate) { return self->wrapped.GetNumber(evaluate); } static int GetFloat(DScriptScanner* self, int evaluate) { return self->wrapped.GetFloat(evaluate); } static int CheckValue(DScriptScanner* self, int allowfloat, int evaluate) { return self->wrapped.CheckValue(allowfloat, evaluate); } static int CheckNumber(DScriptScanner* self, int evaluate) { return self->wrapped.CheckNumber(evaluate); } static int CheckBoolToken(DScriptScanner* self) { return self->wrapped.CheckBoolToken(); } static int CheckString(DScriptScanner* self, const FString* name) { return self->wrapped.CheckString(name->GetChars()); } static int CheckFloat(DScriptScanner* self, int evaluate) { return self->wrapped.CheckFloat(evaluate); } static void SetPrependMessage(DScriptScanner* self, const FString* message) { return self->wrapped.SetPrependMessage(*message); } static void SkipToEndOfBlock(DScriptScanner* self) { return self->wrapped.SkipToEndOfBlock(); } static int StartBraces(DScriptScanner* self, FScanner::SavedPos* braceend) { return self->wrapped.StartBraces(braceend); } static int FoundEndBrace(DScriptScanner* self, FScanner::SavedPos* braceend) { return self->wrapped.FoundEndBrace(*braceend); } static void MustGetValue(DScriptScanner* self, int allowfloat, int evaluate) { return self->wrapped.MustGetValue(allowfloat, evaluate); } static void MustGetFloat(DScriptScanner* self, int evaluate) { return self->wrapped.MustGetFloat(evaluate); } static void MustGetNumber(DScriptScanner* self, int evaluate) { return self->wrapped.MustGetNumber(evaluate); } static void MustGetString(DScriptScanner* self) { return self->wrapped.MustGetString(); } static void MustGetStringName(DScriptScanner* self, const FString* name) { return self->wrapped.MustGetStringName(name->GetChars()); } static void MustGetBoolToken(DScriptScanner* self) { return self->wrapped.MustGetBoolToken(); } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, OpenLumpNum, OpenLumpNum) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_INT(lump); OpenLumpNum(self, lump); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, OpenString, OpenString) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_STRING(name); PARAM_STRING_VAL(script); OpenString(self, &name, &script); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, SavePos, SavePos) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_POINTER(pos, FScanner::SavedPos); SavePos(self, pos); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, RestorePos, RestorePos) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_POINTER(pos, FScanner::SavedPos); RestorePos(self, pos); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, GetStringContents, GetStringContents) { PARAM_SELF_PROLOGUE(DScriptScanner); ACTION_RETURN_STRING(self->wrapped.String); } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, UnGet, UnGet) { PARAM_SELF_PROLOGUE(DScriptScanner); UnGet(self); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, isText, isText) { PARAM_SELF_PROLOGUE(DScriptScanner); ACTION_RETURN_BOOL(isText(self)); } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, GetMessageLine, GetMessageLine) { PARAM_SELF_PROLOGUE(DScriptScanner); ACTION_RETURN_INT(GetMessageLine(self)); } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, AddSymbol, AddSymbolInt) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_STRING(name); PARAM_INT(value); AddSymbolInt(self, &name, value); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, AddSymbolUnsigned, AddSymbolUint) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_STRING(name); PARAM_UINT(value); AddSymbolUint(self, &name, value); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, AddSymbolFloat, AddSymbolDouble) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_STRING(name); PARAM_FLOAT(value); AddSymbolDouble(self, &name, value); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, GetString, GetString) { PARAM_SELF_PROLOGUE(DScriptScanner); ACTION_RETURN_BOOL(GetString(self)); } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, GetNumber, GetNumber) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_BOOL(evaluate); ACTION_RETURN_BOOL(GetNumber(self, evaluate)); } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, GetFloat, GetFloat) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_BOOL(evaluate); ACTION_RETURN_BOOL(GetFloat(self, evaluate)); } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, CheckValue, CheckValue) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_BOOL(allowfloat); PARAM_BOOL(evaluate); ACTION_RETURN_BOOL(CheckValue(self, allowfloat, evaluate)); } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, CheckBoolToken, CheckBoolToken) { PARAM_SELF_PROLOGUE(DScriptScanner); ACTION_RETURN_BOOL(CheckBoolToken(self)); } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, CheckNumber, CheckNumber) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_BOOL(evaluate); ACTION_RETURN_BOOL(CheckNumber(self, evaluate)); } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, CheckString, CheckString) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_STRING(name); ACTION_RETURN_BOOL(CheckString(self, &name)); } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, CheckFloat, CheckFloat) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_BOOL(evaluate); ACTION_RETURN_BOOL(CheckFloat(self, evaluate)); } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, SetPrependMessage, SetPrependMessage) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_STRING(message); SetPrependMessage(self, &message); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, SetCMode, SetCMode) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_BOOL(cmode); SetCMode(self, cmode); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, SetNoOctals, SetNoOctals) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_BOOL(cmode); SetNoOctals(self, cmode); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, SetEscape, SetEscape) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_BOOL(esc); SetEscape(self, esc); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, SkipToEndOfBlock, SkipToEndOfBlock) { PARAM_SELF_PROLOGUE(DScriptScanner); SkipToEndOfBlock(self); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, StartBraces, StartBraces) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_POINTER(braceend, FScanner::SavedPos); StartBraces(self, braceend); // the return value of this is useless. return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, FoundEndBrace, FoundEndBrace) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_POINTER(braceend, FScanner::SavedPos); ACTION_RETURN_BOOL(FoundEndBrace(self, braceend)); } DEFINE_ACTION_FUNCTION(DScriptScanner, ScriptError) { PARAM_SELF_PROLOGUE(DScriptScanner); FString s = FStringFormat(VM_ARGS_NAMES); self->wrapped.ScriptError("%s", s.GetChars()); return 0; } DEFINE_ACTION_FUNCTION(DScriptScanner, ScriptMessage) { PARAM_SELF_PROLOGUE(DScriptScanner); FString s = FStringFormat(VM_ARGS_NAMES); self->wrapped.ScriptMessage("%s", s.GetChars()); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, MustGetValue, MustGetValue) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_BOOL(allowfloat); PARAM_BOOL(evaluate); MustGetValue(self, allowfloat, evaluate); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, MustGetNumber, MustGetNumber) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_BOOL(evaluate); MustGetNumber(self, evaluate); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, MustGetFloat, MustGetFloat) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_BOOL(evaluate); MustGetFloat(self, evaluate); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, MustGetString, MustGetString) { PARAM_SELF_PROLOGUE(DScriptScanner); MustGetString(self); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, MustGetStringName, MustGetStringName) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_STRING(name); MustGetStringName(self, &name); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, MustGetBoolToken, MustGetBoolToken) { PARAM_SELF_PROLOGUE(DScriptScanner); MustGetBoolToken(self); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, Close, Close) { PARAM_SELF_PROLOGUE(DScriptScanner); Close(self); return 0; } DEFINE_ACTION_FUNCTION_NATIVE(DScriptScanner, SetNoFatalErrors, SetNoFatalErrors) { PARAM_SELF_PROLOGUE(DScriptScanner); PARAM_BOOL(cmode); SetNoFatalErrors(self, cmode); return 0; } DEFINE_FIELD_NAMED_X(ScriptScanner, DScriptScanner, wrapped.Line, Line); DEFINE_FIELD_NAMED_X(ScriptScanner, DScriptScanner, wrapped.Float, Float); DEFINE_FIELD_NAMED_X(ScriptScanner, DScriptScanner, wrapped.Number, Number); DEFINE_FIELD_NAMED_X(ScriptScanner, DScriptScanner, wrapped.End, End); DEFINE_FIELD_NAMED_X(ScriptScanner, DScriptScanner, wrapped.Crossed, Crossed); DEFINE_FIELD_NAMED_X(ScriptScanner, DScriptScanner, wrapped.ParseError, ParseError);