scummvm/engines/dreamweb/print.cpp

338 lines
8.6 KiB
C++
Raw Normal View History

/* 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 "dreamweb/dreamweb.h"
namespace DreamGen {
void DreamBase::printBoth(const Frame *charSet, uint16 *x, uint16 y, uint8 c, uint8 nextChar) {
uint16 newX = *x;
uint8 width, height;
printChar(charSet, &newX, y, c, nextChar, &width, &height);
multiDump(*x, y, width, height);
*x = newX;
}
uint8 DreamBase::getNextWord(const Frame *charSet, const uint8 *string, uint8 *totalWidth, uint8 *charCount) {
*totalWidth = 0;
*charCount = 0;
2011-12-08 09:02:51 +00:00
while (true) {
uint8 firstChar = *string;
++string;
++*charCount;
if ((firstChar == ':') || (firstChar == 0)) { //endall
*totalWidth += 6;
return 1;
}
if (firstChar == 32) { //endword
*totalWidth += 6;
return 0;
}
firstChar = engine->modifyChar(firstChar);
if (firstChar != 255) {
uint8 secondChar = *string;
uint8 width = charSet[firstChar - 32 + data.word(kCharshift)].width;
width = kernChars(firstChar, secondChar, width);
*totalWidth += width;
}
}
}
void DreamBase::printChar(const Frame *charSet, uint16* x, uint16 y, uint8 c, uint8 nextChar, uint8 *width, uint8 *height) {
if (c == 255)
return;
uint8 dummyWidth, dummyHeight;
if (width == NULL)
width = &dummyWidth;
if (height == NULL)
height = &dummyHeight;
if (data.byte(kForeignrelease))
y -= 3;
uint16 tmp = c - 32 + data.word(kCharshift);
showFrame(charSet, *x, y, tmp & 0x1ff, (tmp >> 8) & 0xfe, width, height);
if (data.byte(kKerning), 0)
*width = kernChars(c, nextChar, *width);
(*x) += *width;
}
void DreamBase::printChar(const Frame *charSet, uint16 x, uint16 y, uint8 c, uint8 nextChar, uint8 *width, uint8 *height) {
printChar(charSet, &x, y, c, nextChar, width, height);
}
2011-12-08 14:21:42 +00:00
uint8 DreamBase::printSlow(const uint8 *string, uint16 x, uint16 y, uint8 maxWidth, bool centered) {
data.byte(kPointerframe) = 1;
data.byte(kPointermode) = 3;
const Frame* charSet = (const Frame *)getSegment(data.word(kCharset1)).ptr(0, 0);
do {
uint16 offset = x;
uint16 charCount = getNumber(charSet, string, maxWidth, centered, &offset);
do {
2011-08-11 01:19:51 +00:00
uint8 c0 = string[0];
uint8 c1 = string[1];
uint8 c2 = string[2];
c0 = engine->modifyChar(c0);
printBoth(charSet, &offset, y, c0, c1);
if ((c1 == 0) || (c1 == ':')) {
return 0;
}
if (charCount != 1) {
c1 = engine->modifyChar(c1);
data.word(kCharshift) = 91;
uint16 offset2 = offset;
printBoth(charSet, &offset2, y, c1, c2);
data.word(kCharshift) = 0;
for (int i=0; i<2; ++i) {
uint16 mouseState = waitFrames();
if (_quitRequested)
return 0;
2011-08-25 06:03:27 +00:00
if (mouseState == 0)
continue;
2011-08-25 06:03:27 +00:00
if (mouseState != data.word(kOldbutton)) {
return 1;
}
}
}
2011-08-11 01:19:51 +00:00
++string;
--charCount;
} while (charCount);
y += 10;
} while (true);
}
uint8 DreamBase::printDirect(const uint8* string, uint16 x, uint16 y, uint8 maxWidth, bool centered) {
return printDirect(&string, x, &y, maxWidth, centered);
}
uint8 DreamBase::printDirect(const uint8** string, uint16 x, uint16 *y, uint8 maxWidth, bool centered) {
data.word(kLastxpos) = x;
const Frame *charSet = engine->currentCharset();
while (true) {
uint16 offset = x;
uint8 charCount = getNumber(charSet, *string, maxWidth, centered, &offset);
uint16 i = offset;
do {
uint8 c = (*string)[0];
uint8 nextChar = (*string)[1];
++(*string);
if ((c == 0) || (c == ':')) {
return c;
}
c = engine->modifyChar(c);
uint8 width, height;
printChar(charSet, &i, *y, c, nextChar, &width, &height);
data.word(kLastxpos) = i;
--charCount;
2011-12-08 09:02:51 +00:00
} while (charCount);
*y += data.word(kLinespacing);
}
}
uint8 DreamBase::getNumber(const Frame *charSet, const uint8 *string, uint16 maxWidth, bool centered, uint16* offset) {
uint8 totalWidth = 0;
uint8 charCount = 0;
while (true) {
uint8 wordTotalWidth, wordCharCount;
uint8 done = getNextWord(charSet, string, &wordTotalWidth, &wordCharCount);
string += wordCharCount;
2011-12-04 23:37:19 +00:00
uint16 tmp = totalWidth + wordTotalWidth - 10;
if (done == 1) { //endoftext
2011-12-04 23:37:19 +00:00
if (tmp < maxWidth) {
totalWidth += wordTotalWidth;
charCount += wordCharCount;
}
if (centered) {
2011-12-04 23:37:19 +00:00
tmp = (maxWidth & 0xfe) + 2 + 20 - totalWidth;
tmp /= 2;
} else {
2011-12-04 23:37:19 +00:00
tmp = 0;
}
2011-12-04 23:37:19 +00:00
*offset += tmp;
return charCount;
}
2011-12-04 23:37:19 +00:00
if (tmp >= maxWidth) { //gotoverend
if (centered) {
2011-12-04 23:37:19 +00:00
tmp = (maxWidth & 0xfe) - totalWidth + 20;
tmp /= 2;
} else {
2011-12-04 23:37:19 +00:00
tmp = 0;
}
2011-12-04 23:37:19 +00:00
*offset += tmp;
return charCount;
}
totalWidth += wordTotalWidth;
charCount += wordCharCount;
}
}
uint8 DreamBase::kernChars(uint8 firstChar, uint8 secondChar, uint8 width) {
2011-12-04 13:16:30 +00:00
if ((firstChar == 'a') || (firstChar == 'u')) {
if ((secondChar == 'n') || (secondChar == 't') || (secondChar == 'r') || (secondChar == 'i') || (secondChar == 'l'))
return width-1;
}
return width;
}
2011-12-08 14:21:42 +00:00
uint16 DreamBase::waitFrames() {
readMouse();
showPointer();
vSync();
dumpPointer();
delPointer();
2011-08-25 06:03:27 +00:00
return data.word(kMousebutton);
}
2011-12-08 14:21:42 +00:00
const char *DreamBase::monPrint(const char *string) {
2011-11-18 17:17:16 +00:00
data.byte(kKerning) = 1;
uint16 x = data.word(kMonadx);
Frame *charset = engine->tempCharset();
2011-11-18 17:17:16 +00:00
const char *iterator = string;
2011-11-19 18:17:04 +00:00
bool done = false;
while (!done) {
uint16 count = getNumber(charset, (const uint8 *)iterator, 166, false, &x);
2011-11-19 18:17:04 +00:00
do {
2011-11-18 17:17:16 +00:00
char c = *iterator++;
if (c == ':')
break;
2011-11-19 18:17:04 +00:00
if ((c == 0) || (c == '"') || (c == '=')) {
done = true;
break;
}
2011-11-18 17:17:16 +00:00
if (c == '%') {
data.byte(kLasttrigger) = *iterator;
iterator += 2;
2011-11-19 18:17:04 +00:00
done = true;
break;
2011-11-18 17:17:16 +00:00
}
c = engine->modifyChar(c);
printChar(charset, &x, data.word(kMonady), c, 0, NULL, NULL);
2011-11-18 17:17:16 +00:00
data.word(kCurslocx) = x;
data.word(kCurslocy) = data.word(kMonady);
data.word(kMaintimer) = 1;
printCurs();
vSync();
lockMon();
delCurs();
2011-11-19 18:17:04 +00:00
} while (--count);
2011-11-18 17:17:16 +00:00
x = data.word(kMonadx);
scrollMonitor();
2011-11-18 17:17:16 +00:00
data.word(kCurslocx) = data.word(kMonadx);
}
2011-11-19 18:17:04 +00:00
2011-11-18 17:17:16 +00:00
data.byte(kKerning) = 0;
return iterator;
}
void DreamBase::rollEndCreditsGameWon() {
playChannel0(16, 255);
data.byte(kVolume) = 7;
data.byte(kVolumeto) = 0;
data.byte(kVolumedirection) = (byte)-1;
multiGet(mapStore(), 75, 20, 160, 160);
const uint8 *string = getTextInFile1(3);
const int linespacing = data.word(kLinespacing);
for (int i = 0; i < 254; ++i) {
// Output the text, initially with an offset of 10 pixels,
// then move it up one pixel until we shifted it by a complete
// line of text.
for (int j = 0; j < linespacing; ++j) {
vSync();
multiPut(mapStore(), 75, 20, 160, 160);
vSync();
// Output up to 18 lines of text
uint16 y = 10 - j;
const uint8 *tmp_str = string;
for (int k = 0; k < 18; ++k) {
DreamBase::printDirect(&tmp_str, 75, &y, 160 + 1, true);
y += linespacing;
}
vSync();
multiDump(75, 20, 160, 160);
}
// Skip to the next text line
byte c;
do {
c = *string++;
} while (c != ':' && c != 0);
}
hangOn(100);
panelToMap();
fadeScreenUpHalf();
}
void DreamBase::rollEndCreditsGameLost() {
multiGet(mapStore(), 25, 20, 160, 160);
const uint8 *string = getTextInFile1(49);
const int linespacing = data.word(kLinespacing);
for (int i = 0; i < 80; ++i) {
// Output the text, initially with an offset of 10 pixels,
// then move it up one pixel until we shifted it by a complete
// line of text.
for (int j = 0; j < linespacing; ++j) {
vSync();
multiPut(mapStore(), 25, 20, 160, 160);
vSync();
// Output up to 18 lines of text
uint16 y = 10 - j;
const uint8 *tmp_str = string;
for (int k = 0; k < 18; ++k) {
DreamBase::printDirect(&tmp_str, 25, &y, 160 + 1, true);
y += linespacing;
}
vSync();
multiDump(25, 20, 160, 160);
if (data.byte(kLasthardkey) == 1)
return;
}
// Skip to the next text line
byte c;
do {
c = *string++;
} while (c != ':' && c != 0);
if (data.byte(kLasthardkey) == 1)
return;
}
hangOne(120);
}
} // End of namespace DreamGen