scummvm/engines/ags/engine/ac/string.cpp
Paul Gilbert eb1e337edd AGS: Removed String's "const char*" conversion operator
From upstream 7c426674e7e53bffe9120f8ffc059b77fa03a606
2021-06-13 21:08:59 -07:00

457 lines
17 KiB
C++

/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
*/
//include <cstdio>
#include "ags/engine/ac/string.h"
#include "ags/shared/ac/common.h"
#include "ags/engine/ac/display.h"
#include "ags/shared/ac/game_setup_struct.h"
#include "ags/engine/ac/game_state.h"
#include "ags/engine/ac/global_translation.h"
#include "ags/engine/ac/runtime_defines.h"
#include "ags/engine/ac/dynobj/script_string.h"
#include "ags/shared/font/fonts.h"
#include "ags/engine/debugging/debug_log.h"
#include "ags/engine/script/runtime_script_value.h"
#include "ags/shared/util/string_compat.h"
#include "ags/shared/debugging/out.h"
#include "ags/engine/script/script_api.h"
#include "ags/engine/script/script_runtime.h"
#include "ags/engine/ac/math.h"
#include "ags/globals.h"
namespace AGS3 {
int String_IsNullOrEmpty(const char *thisString) {
if ((thisString == nullptr) || (thisString[0] == 0))
return 1;
return 0;
}
const char *String_Copy(const char *srcString) {
return CreateNewScriptString(srcString);
}
const char *String_Append(const char *thisString, const char *extrabit) {
char *buffer = (char *)malloc(strlen(thisString) + strlen(extrabit) + 1);
strcpy(buffer, thisString);
strcat(buffer, extrabit);
return CreateNewScriptString(buffer, false);
}
const char *String_AppendChar(const char *thisString, char extraOne) {
char *buffer = (char *)malloc(strlen(thisString) + 2);
sprintf(buffer, "%s%c", thisString, extraOne);
return CreateNewScriptString(buffer, false);
}
const char *String_ReplaceCharAt(const char *thisString, int index, char newChar) {
if ((index < 0) || (index >= (int)strlen(thisString)))
quit("!String.ReplaceCharAt: index outside range of string");
char *buffer = (char *)malloc(strlen(thisString) + 1);
strcpy(buffer, thisString);
buffer[index] = newChar;
return CreateNewScriptString(buffer, false);
}
const char *String_Truncate(const char *thisString, int length) {
if (length < 0)
quit("!String.Truncate: invalid length");
if (length >= (int)strlen(thisString)) {
return thisString;
}
char *buffer = (char *)malloc(length + 1);
strncpy(buffer, thisString, length);
buffer[length] = 0;
return CreateNewScriptString(buffer, false);
}
const char *String_Substring(const char *thisString, int index, int length) {
if (length < 0)
quit("!String.Substring: invalid length");
if ((index < 0) || (index > (int)strlen(thisString)))
quit("!String.Substring: invalid index");
char *buffer = (char *)malloc(length + 1);
strncpy(buffer, &thisString[index], length);
buffer[length] = 0;
return CreateNewScriptString(buffer, false);
}
int String_CompareTo(const char *thisString, const char *otherString, bool caseSensitive) {
if (caseSensitive) {
return strcmp(thisString, otherString);
} else {
return ags_stricmp(thisString, otherString);
}
}
int String_StartsWith(const char *thisString, const char *checkForString, bool caseSensitive) {
if (caseSensitive) {
return (strncmp(thisString, checkForString, strlen(checkForString)) == 0) ? 1 : 0;
} else {
return (ags_strnicmp(thisString, checkForString, strlen(checkForString)) == 0) ? 1 : 0;
}
}
int String_EndsWith(const char *thisString, const char *checkForString, bool caseSensitive) {
int checkAtOffset = strlen(thisString) - strlen(checkForString);
if (checkAtOffset < 0) {
return 0;
}
if (caseSensitive) {
return (strcmp(&thisString[checkAtOffset], checkForString) == 0) ? 1 : 0;
} else {
return (ags_stricmp(&thisString[checkAtOffset], checkForString) == 0) ? 1 : 0;
}
}
const char *String_Replace(const char *thisString, const char *lookForText, const char *replaceWithText, bool caseSensitive) {
char resultBuffer[STD_BUFFER_SIZE] = "";
int thisStringLen = (int)strlen(thisString);
int outputSize = 0;
for (int i = 0; i < thisStringLen; i++) {
bool matchHere = false;
if (caseSensitive) {
matchHere = (strncmp(&thisString[i], lookForText, strlen(lookForText)) == 0);
} else {
matchHere = (ags_strnicmp(&thisString[i], lookForText, strlen(lookForText)) == 0);
}
if (matchHere) {
strcpy(&resultBuffer[outputSize], replaceWithText);
outputSize += strlen(replaceWithText);
i += strlen(lookForText) - 1;
} else {
resultBuffer[outputSize] = thisString[i];
outputSize++;
}
}
resultBuffer[outputSize] = 0;
return CreateNewScriptString(resultBuffer, true);
}
const char *String_LowerCase(const char *thisString) {
char *buffer = (char *)malloc(strlen(thisString) + 1);
strcpy(buffer, thisString);
ags_strlwr(buffer);
return CreateNewScriptString(buffer, false);
}
const char *String_UpperCase(const char *thisString) {
char *buffer = (char *)malloc(strlen(thisString) + 1);
strcpy(buffer, thisString);
ags_strupr(buffer);
return CreateNewScriptString(buffer, false);
}
int String_GetChars(const char *texx, int index) {
if ((index < 0) || (index >= (int)strlen(texx)))
return 0;
return texx[index];
}
int StringToInt(const char *stino) {
return atoi(stino);
}
int StrContains(const char *s1, const char *s2) {
VALIDATE_STRING(s1);
VALIDATE_STRING(s2);
char *tempbuf1 = (char *)malloc(strlen(s1) + 1);
char *tempbuf2 = (char *)malloc(strlen(s2) + 1);
strcpy(tempbuf1, s1);
strcpy(tempbuf2, s2);
ags_strlwr(tempbuf1);
ags_strlwr(tempbuf2);
char *offs = strstr(tempbuf1, tempbuf2);
free(tempbuf1);
free(tempbuf2);
if (offs == nullptr)
return -1;
return (offs - tempbuf1);
}
//=============================================================================
const char *CreateNewScriptString(const char *fromText, bool reAllocate) {
return (const char *)CreateNewScriptStringObj(fromText, reAllocate).second;
}
DynObjectRef CreateNewScriptStringObj(const char *fromText, bool reAllocate) {
ScriptString *str;
if (reAllocate) {
str = new ScriptString(fromText);
} else {
str = new ScriptString();
str->text = const_cast<char *>(fromText);
}
void *obj_ptr = str->text;
int32_t handle = ccRegisterManagedObject(obj_ptr, str);
if (handle == 0) {
delete str;
return DynObjectRef(0, nullptr);
}
return DynObjectRef(handle, obj_ptr);
}
size_t break_up_text_into_lines(const char *todis, SplitLines &lines, int wii, int fonnt, size_t max_lines) {
if (fonnt == -1)
fonnt = _GP(play).normal_font;
// char sofar[100];
if (todis[0] == '&') {
while ((todis[0] != ' ') & (todis[0] != 0)) todis++;
if (todis[0] == ' ') todis++;
}
lines.Reset();
_G(longestline) = 0;
// Don't attempt to display anything if the width is tiny
if (wii < 3)
return 0;
int line_length;
split_lines(todis, lines, wii, fonnt, max_lines);
// Right-to-left just means reverse the text then
// write it as normal
if (_GP(game).options[OPT_RIGHTLEFTWRITE])
for (size_t rr = 0; rr < lines.Count(); rr++) {
lines[rr].Reverse();
line_length = wgettextwidth_compensate(lines[rr].GetCStr(), fonnt);
if (line_length > _G(longestline))
_G(longestline) = line_length;
}
else
for (size_t rr = 0; rr < lines.Count(); rr++) {
line_length = wgettextwidth_compensate(lines[rr].GetCStr(), fonnt);
if (line_length > _G(longestline))
_G(longestline) = line_length;
}
return lines.Count();
}
int MAXSTRLEN = MAX_MAXSTRLEN;
void check_strlen(char *ptt) {
MAXSTRLEN = MAX_MAXSTRLEN;
long charstart = (long)&_GP(game).chars[0];
long charend = charstart + sizeof(CharacterInfo) * _GP(game).numcharacters;
if (((long)&ptt[0] >= charstart) && ((long)&ptt[0] <= charend))
MAXSTRLEN = 30;
}
/*void GetLanguageString(int indxx,char*buffr) {
VALIDATE_STRING(buffr);
char*bptr=get_language_text(indxx);
if (bptr==NULL) strcpy(buffr,"[language string error]");
else strncpy(buffr,bptr,199);
buffr[199]=0;
}*/
void my_strncpy(char *dest, const char *src, int len) {
// the normal strncpy pads out the string with zeros up to the
// max length -- we don't want that
if (strlen(src) >= (unsigned)len) {
strncpy(dest, src, len);
dest[len] = 0;
} else
strcpy(dest, src);
}
//=============================================================================
//
// Script API Functions
//
//=============================================================================
// int (const char *thisString)
RuntimeScriptValue Sc_String_IsNullOrEmpty(const RuntimeScriptValue *params, int32_t param_count) {
API_SCALL_INT_POBJ(String_IsNullOrEmpty, const char);
}
// const char* (const char *thisString, const char *extrabit)
RuntimeScriptValue Sc_String_Append(void *self, const RuntimeScriptValue *params, int32_t param_count) {
API_CONST_OBJCALL_OBJ_POBJ(const char, const char, _GP(myScriptStringImpl), String_Append, const char);
}
// const char* (const char *thisString, char extraOne)
RuntimeScriptValue Sc_String_AppendChar(void *self, const RuntimeScriptValue *params, int32_t param_count) {
API_CONST_OBJCALL_OBJ_PINT(const char, const char, _GP(myScriptStringImpl), String_AppendChar);
}
// int (const char *thisString, const char *otherString, bool caseSensitive)
RuntimeScriptValue Sc_String_CompareTo(void *self, const RuntimeScriptValue *params, int32_t param_count) {
API_OBJCALL_INT_POBJ_PBOOL(const char, String_CompareTo, const char);
}
// int (const char *s1, const char *s2)
RuntimeScriptValue Sc_StrContains(void *self, const RuntimeScriptValue *params, int32_t param_count) {
API_OBJCALL_INT_POBJ(const char, StrContains, const char);
}
// const char* (const char *srcString)
RuntimeScriptValue Sc_String_Copy(void *self, const RuntimeScriptValue *params, int32_t param_count) {
API_CONST_OBJCALL_OBJ(const char, const char, _GP(myScriptStringImpl), String_Copy);
}
// int (const char *thisString, const char *checkForString, bool caseSensitive)
RuntimeScriptValue Sc_String_EndsWith(void *self, const RuntimeScriptValue *params, int32_t param_count) {
API_OBJCALL_INT_POBJ_PBOOL(const char, String_EndsWith, const char);
}
// const char* (const char *texx, ...)
RuntimeScriptValue Sc_String_Format(const RuntimeScriptValue *params, int32_t param_count) {
API_SCALL_SCRIPT_SPRINTF(String_Format, 1);
return RuntimeScriptValue().SetDynamicObject(const_cast<char *>(CreateNewScriptString(scsf_buffer)), &_GP(myScriptStringImpl));
}
// const char* (const char *thisString)
RuntimeScriptValue Sc_String_LowerCase(void *self, const RuntimeScriptValue *params, int32_t param_count) {
API_CONST_OBJCALL_OBJ(const char, const char, _GP(myScriptStringImpl), String_LowerCase);
}
// const char* (const char *thisString, const char *lookForText, const char *replaceWithText, bool caseSensitive)
RuntimeScriptValue Sc_String_Replace(void *self, const RuntimeScriptValue *params, int32_t param_count) {
API_CONST_OBJCALL_OBJ_POBJ2_PBOOL(const char, const char, _GP(myScriptStringImpl), String_Replace, const char, const char);
}
// const char* (const char *thisString, int index, char newChar)
RuntimeScriptValue Sc_String_ReplaceCharAt(void *self, const RuntimeScriptValue *params, int32_t param_count) {
API_CONST_OBJCALL_OBJ_PINT2(const char, const char, _GP(myScriptStringImpl), String_ReplaceCharAt);
}
// int (const char *thisString, const char *checkForString, bool caseSensitive)
RuntimeScriptValue Sc_String_StartsWith(void *self, const RuntimeScriptValue *params, int32_t param_count) {
API_OBJCALL_INT_POBJ_PBOOL(const char, String_StartsWith, const char);
}
// const char* (const char *thisString, int index, int length)
RuntimeScriptValue Sc_String_Substring(void *self, const RuntimeScriptValue *params, int32_t param_count) {
API_CONST_OBJCALL_OBJ_PINT2(const char, const char, _GP(myScriptStringImpl), String_Substring);
}
// const char* (const char *thisString, int length)
RuntimeScriptValue Sc_String_Truncate(void *self, const RuntimeScriptValue *params, int32_t param_count) {
API_CONST_OBJCALL_OBJ_PINT(const char, const char, _GP(myScriptStringImpl), String_Truncate);
}
// const char* (const char *thisString)
RuntimeScriptValue Sc_String_UpperCase(void *self, const RuntimeScriptValue *params, int32_t param_count) {
API_CONST_OBJCALL_OBJ(const char, const char, _GP(myScriptStringImpl), String_UpperCase);
}
// FLOAT_RETURN_TYPE (const char *theString);
RuntimeScriptValue Sc_StringToFloat(void *self, const RuntimeScriptValue *params, int32_t param_count) {
API_OBJCALL_FLOAT(const char, StringToFloat);
}
// int (char*stino)
RuntimeScriptValue Sc_StringToInt(void *self, const RuntimeScriptValue *params, int32_t param_count) {
API_OBJCALL_INT(const char, StringToInt);
}
// int (const char *texx, int index)
RuntimeScriptValue Sc_String_GetChars(void *self, const RuntimeScriptValue *params, int32_t param_count) {
API_OBJCALL_INT_PINT(const char, String_GetChars);
}
RuntimeScriptValue Sc_strlen(void *self, const RuntimeScriptValue *params, int32_t param_count) {
ASSERT_SELF(strlen);
return RuntimeScriptValue().SetInt32(strlen((const char *)self));
}
//=============================================================================
//
// Exclusive API for Plugins
//
//=============================================================================
// const char* (const char *texx, ...)
const char *ScPl_String_Format(const char *texx, ...) {
API_PLUGIN_SCRIPT_SPRINTF(texx);
return CreateNewScriptString(scsf_buffer);
}
void RegisterStringAPI() {
ccAddExternalStaticFunction("String::IsNullOrEmpty^1", Sc_String_IsNullOrEmpty);
ccAddExternalObjectFunction("String::Append^1", Sc_String_Append);
ccAddExternalObjectFunction("String::AppendChar^1", Sc_String_AppendChar);
ccAddExternalObjectFunction("String::CompareTo^2", Sc_String_CompareTo);
ccAddExternalObjectFunction("String::Contains^1", Sc_StrContains);
ccAddExternalObjectFunction("String::Copy^0", Sc_String_Copy);
ccAddExternalObjectFunction("String::EndsWith^2", Sc_String_EndsWith);
ccAddExternalStaticFunction("String::Format^101", Sc_String_Format);
ccAddExternalObjectFunction("String::IndexOf^1", Sc_StrContains);
ccAddExternalObjectFunction("String::LowerCase^0", Sc_String_LowerCase);
ccAddExternalObjectFunction("String::Replace^3", Sc_String_Replace);
ccAddExternalObjectFunction("String::ReplaceCharAt^2", Sc_String_ReplaceCharAt);
ccAddExternalObjectFunction("String::StartsWith^2", Sc_String_StartsWith);
ccAddExternalObjectFunction("String::Substring^2", Sc_String_Substring);
ccAddExternalObjectFunction("String::Truncate^1", Sc_String_Truncate);
ccAddExternalObjectFunction("String::UpperCase^0", Sc_String_UpperCase);
ccAddExternalObjectFunction("String::get_AsFloat", Sc_StringToFloat);
ccAddExternalObjectFunction("String::get_AsInt", Sc_StringToInt);
ccAddExternalObjectFunction("String::geti_Chars", Sc_String_GetChars);
ccAddExternalObjectFunction("String::get_Length", Sc_strlen);
/* ----------------------- Registering unsafe exports for plugins -----------------------*/
ccAddExternalFunctionForPlugin("String::IsNullOrEmpty^1", (void *)String_IsNullOrEmpty);
ccAddExternalFunctionForPlugin("String::Append^1", (void *)String_Append);
ccAddExternalFunctionForPlugin("String::AppendChar^1", (void *)String_AppendChar);
ccAddExternalFunctionForPlugin("String::CompareTo^2", (void *)String_CompareTo);
ccAddExternalFunctionForPlugin("String::Contains^1", (void *)StrContains);
ccAddExternalFunctionForPlugin("String::Copy^0", (void *)String_Copy);
ccAddExternalFunctionForPlugin("String::EndsWith^2", (void *)String_EndsWith);
ccAddExternalFunctionForPlugin("String::Format^101", (void *)ScPl_String_Format);
ccAddExternalFunctionForPlugin("String::IndexOf^1", (void *)StrContains);
ccAddExternalFunctionForPlugin("String::LowerCase^0", (void *)String_LowerCase);
ccAddExternalFunctionForPlugin("String::Replace^3", (void *)String_Replace);
ccAddExternalFunctionForPlugin("String::ReplaceCharAt^2", (void *)String_ReplaceCharAt);
ccAddExternalFunctionForPlugin("String::StartsWith^2", (void *)String_StartsWith);
ccAddExternalFunctionForPlugin("String::Substring^2", (void *)String_Substring);
ccAddExternalFunctionForPlugin("String::Truncate^1", (void *)String_Truncate);
ccAddExternalFunctionForPlugin("String::UpperCase^0", (void *)String_UpperCase);
ccAddExternalFunctionForPlugin("String::get_AsFloat", (void *)StringToFloat);
ccAddExternalFunctionForPlugin("String::get_AsInt", (void *)StringToInt);
ccAddExternalFunctionForPlugin("String::geti_Chars", (void *)String_GetChars);
ccAddExternalFunctionForPlugin("String::get_Length", (void *)strlen);
}
} // namespace AGS3