mirror of
https://github.com/libretro/scummvm.git
synced 2025-02-12 06:00:48 +00:00
170 lines
4.7 KiB
C++
170 lines
4.7 KiB
C++
/* Residual - Virtual machine to run LucasArts' 3D adventure games
|
|
*
|
|
* Residual is the legal property of its developers, whose names
|
|
* are too numerous to list here. Please refer to the AUTHORS
|
|
* file distributed with this source distribution.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
* This library 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
|
|
* Lesser General Public License for more details.
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*
|
|
* $URL$
|
|
* $Id$
|
|
*
|
|
*/
|
|
|
|
#include "common/sys.h"
|
|
#include "common/debug.h"
|
|
|
|
#include "engine/textsplit.h"
|
|
|
|
#include <cstdio>
|
|
#include <cstring>
|
|
#include <cctype>
|
|
#include <cstdarg>
|
|
|
|
// FIXME: Replace this with a proper parser (this is just too dodgy :)
|
|
int residual_vsscanf(const char *str, int field_count, const char *format, va_list ap) {
|
|
unsigned int f01 = va_arg(ap, long);
|
|
unsigned int f02 = va_arg(ap, long);
|
|
unsigned int f03 = va_arg(ap, long);
|
|
unsigned int f04 = va_arg(ap, long);
|
|
unsigned int f05 = va_arg(ap, long);
|
|
unsigned int f06 = va_arg(ap, long);
|
|
unsigned int f07 = va_arg(ap, long);
|
|
unsigned int f08 = va_arg(ap, long);
|
|
unsigned int f09 = va_arg(ap, long);
|
|
unsigned int f10 = va_arg(ap, long);
|
|
unsigned int f11 = va_arg(ap, long);
|
|
unsigned int f12 = va_arg(ap, long);
|
|
unsigned int f13 = va_arg(ap, long);
|
|
unsigned int f14 = va_arg(ap, long);
|
|
unsigned int f15 = va_arg(ap, long);
|
|
unsigned int f16 = va_arg(ap, long);
|
|
unsigned int f17 = va_arg(ap, long);
|
|
unsigned int f18 = va_arg(ap, long);
|
|
unsigned int f19 = va_arg(ap, long);
|
|
unsigned int f20 = va_arg(ap, long);
|
|
|
|
if (field_count > 20)
|
|
error("Too many fields requested of residual_vsscanf (%d)", field_count);
|
|
|
|
return sscanf(str, format, f01, f02, f03, f04, f05, f06, f07, f08, f09, f10,
|
|
f11, f12, f13, f14, f15, f16, f17, f18, f19, f20);
|
|
}
|
|
|
|
TextSplitter::TextSplitter(const char *data, int len) {
|
|
char *line, *tmpData;
|
|
int i;
|
|
|
|
tmpData = new char[len+1];
|
|
std::memcpy(tmpData, data, len);
|
|
tmpData[len] = '\0';
|
|
// Find out how many lines of text there are
|
|
_numLines = _lineIndex = 0;
|
|
line = (char *) tmpData;
|
|
while (line != NULL) {
|
|
line = std::strchr(line, '\n');
|
|
if (line != NULL) {
|
|
_numLines++;
|
|
line++;
|
|
}
|
|
}
|
|
// Allocate an array of the lines
|
|
_lines = new TextLines[_numLines];
|
|
line = (char *) tmpData;
|
|
for (i=0;i<_numLines;i++) {
|
|
char *lastLine = line;
|
|
|
|
line = std::strchr(lastLine, '\n');
|
|
_lines[i].setData(lastLine, line-lastLine);
|
|
line++;
|
|
}
|
|
delete[] tmpData;
|
|
_currLine = NULL;
|
|
processLine();
|
|
}
|
|
|
|
bool TextSplitter::checkString(const char *needle) {
|
|
// checkString also needs to check for extremely optional
|
|
// components like "object_art" which can be missing entirely
|
|
if (currentLine() == NULL)
|
|
return false;
|
|
else if (std::strstr(currentLine(), needle))
|
|
return true;
|
|
else
|
|
return false;
|
|
}
|
|
|
|
void TextSplitter::expectString(const char *expected) {
|
|
if (_currLine == NULL)
|
|
error("Expected `%s', got EOF\n", expected);
|
|
if (std::strcmp(currentLine(), expected) != 0)
|
|
error("Expected `%s', got `%s'\n", expected, currentLine());
|
|
nextLine();
|
|
}
|
|
|
|
void TextSplitter::scanString(const char *fmt, int field_count, ...) {
|
|
if (_currLine == NULL)
|
|
error("Expected line of format `%s', got EOF\n", fmt);
|
|
|
|
std::va_list va;
|
|
|
|
va_start(va, field_count);
|
|
|
|
#ifdef WIN32
|
|
if (residual_vsscanf(currentLine(), field_count, fmt, va) < field_count)
|
|
#else
|
|
if (vsscanf(currentLine(), fmt, va) < field_count)
|
|
#endif
|
|
error("Expected line of format `%s', got `%s'\n", fmt, currentLine());
|
|
va_end(va);
|
|
|
|
nextLine();
|
|
}
|
|
|
|
void TextSplitter::processLine() {
|
|
if (eof())
|
|
return;
|
|
|
|
_currLine = _lines[_lineIndex++].getData();
|
|
|
|
// Cut off comments
|
|
char *comment_start = std::strchr(_currLine, '#');
|
|
if (comment_start != NULL)
|
|
*comment_start = '\0';
|
|
|
|
// Cut off trailing whitespace (including '\r')
|
|
char *strend = std::strchr(_currLine, '\0');
|
|
while (strend > _currLine && std::isspace(strend[-1]))
|
|
strend--;
|
|
*strend = '\0';
|
|
|
|
// Skip blank lines
|
|
if (*_currLine == '\0')
|
|
nextLine();
|
|
|
|
// Convert to lower case
|
|
if (!eof())
|
|
for (char *s = _currLine; *s != '\0'; s++)
|
|
*s = std::tolower(*s);
|
|
}
|
|
|
|
void TextSplitter::TextLines::setData(char *data, int length) {
|
|
int _lineLength = length;
|
|
|
|
_lineData = new char[_lineLength];
|
|
std::memcpy(_lineData, data, _lineLength);
|
|
_lineData[_lineLength-1] = 0;
|
|
}
|