mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-23 19:16:21 +00:00
946 lines
22 KiB
C++
946 lines
22 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 3 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, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
#include "common/memstream.h"
|
|
#include "common/system.h"
|
|
#include "graphics/palette.h"
|
|
#include "chewy/chewy.h"
|
|
#include "chewy/events.h"
|
|
#include "chewy/font.h"
|
|
#include "chewy/globals.h"
|
|
#include "chewy/mcga_graphics.h"
|
|
|
|
namespace Chewy {
|
|
|
|
#define VGA_COLOR_TRANS(x) ((x)*255 / 63)
|
|
|
|
McgaGraphics::McgaGraphics() {
|
|
}
|
|
|
|
McgaGraphics::~McgaGraphics() {
|
|
}
|
|
|
|
void McgaGraphics::init() {
|
|
_G(currentScreen) = (byte *)g_screen->getPixels();
|
|
_G(spriteWidth) = 0;
|
|
}
|
|
|
|
void setScummVMPalette(const byte *palette, uint start, uint count) {
|
|
byte tempPal[PALETTE_SIZE];
|
|
byte *dest = &tempPal[0];
|
|
|
|
for (uint i = 0; i < count * 3; ++i, ++palette, ++dest)
|
|
*dest = VGA_COLOR_TRANS(*palette);
|
|
|
|
g_system->getPaletteManager()->setPalette(tempPal, start, count);
|
|
}
|
|
|
|
void McgaGraphics::setPointer(byte *ptr) {
|
|
if (ptr) {
|
|
_G(currentScreen) = ptr;
|
|
} else {
|
|
_G(currentScreen) = (byte *)g_screen->getPixels();
|
|
}
|
|
}
|
|
|
|
void McgaGraphics::setPalette(byte *palette) {
|
|
for (int16 i = 0; i < 768; i++)
|
|
_palTable[i] = palette[i];
|
|
setScummVMPalette(palette, 0, PALETTE_COUNT);
|
|
}
|
|
|
|
void McgaGraphics::raster_col(int16 c, int16 r, int16 g, int16 b) {
|
|
_palTable[c * 3] = r;
|
|
_palTable[c * 3 + 1] = g;
|
|
_palTable[c * 3 + 2] = b;
|
|
setScummVMPalette(&_palTable[c * 3], c, 1);
|
|
}
|
|
|
|
void McgaGraphics::fadeIn(byte *palette) {
|
|
for (int16 j = 63; j >= 0; j--) {
|
|
int16 k = 0;
|
|
for (int16 i = 0; i < 256; i++) {
|
|
int16 r = palette[k];
|
|
int16 g = palette[k + 1];
|
|
int16 b = palette[k + 2];
|
|
int16 r1 = r - j;
|
|
int16 g1 = g - j;
|
|
int16 b1 = b - j;
|
|
if ((r1 > 0) && (r1 <= r))
|
|
_palTable[k] = r1;
|
|
if ((g1 > 0) && (r1 <= r))
|
|
_palTable[k + 1] = g1;
|
|
if ((b1 > 0) && (r1 <= r))
|
|
_palTable[k + 2] = b1;
|
|
k += 3;
|
|
}
|
|
setScummVMPalette(_palTable, 0, PALETTE_COUNT);
|
|
}
|
|
}
|
|
|
|
void McgaGraphics::fadeOut() {
|
|
for (int16 j = 0; j < 64; j++) {
|
|
int16 k = 0;
|
|
for (int16 i = 0; i < 256; i++) {
|
|
int16 r = _palTable[k];
|
|
int16 g = _palTable[k + 1];
|
|
int16 b = _palTable[k + 2];
|
|
if (r > 0) --r;
|
|
if (g > 0) --g;
|
|
if (b > 0) --b;
|
|
_palTable[k] = r;
|
|
_palTable[k + 1] = g;
|
|
_palTable[k + 2] = b;
|
|
k += 3;
|
|
}
|
|
setScummVMPalette(_palTable, 0, PALETTE_COUNT);
|
|
}
|
|
}
|
|
|
|
void McgaGraphics::set_partialpalette(const byte *palette, int16 startCol, int16 nr) {
|
|
int16 k = startCol * 3;
|
|
int16 endcol = startCol + nr;
|
|
for (int16 i = startCol; i < endcol; i++) {
|
|
_palTable[k] = palette[k];
|
|
_palTable[k + 1] = palette[k + 1];
|
|
_palTable[k + 2] = palette[k + 2];
|
|
k += 3;
|
|
}
|
|
setScummVMPalette(_palTable + startCol * 3, startCol, nr);
|
|
}
|
|
|
|
void McgaGraphics::cls() {
|
|
Common::Rect r(0, 0, _G(currentScreen).pitch, _G(currentScreen).h);
|
|
_G(currentScreen).fillRect(r, 0);
|
|
}
|
|
|
|
void McgaGraphics::drawLine(int16 x1, int16 y1, int16 x2, int16 y2, int16 color) {
|
|
_G(currentScreen).drawLine(x1, y1, x2, y2, color);
|
|
}
|
|
|
|
void McgaGraphics::box(int16 x1, int16 y1, int16 x2, int16 y2, int16 color) {
|
|
_G(currentScreen).frameRect(Common::Rect(x1, y1, x2, y2), color);
|
|
}
|
|
|
|
void McgaGraphics::boxFill(int16 x1, int16 y1, int16 x2, int16 y2, int16 color) {
|
|
if (x2 == x1)x2++;
|
|
int16 h = abs(y2 - y1);
|
|
if (h == 0)
|
|
h = 1;
|
|
for (int16 i = 0; i < h; i++)
|
|
drawLine(x1, y1 + i, x2, y1 + i, color);
|
|
}
|
|
|
|
void McgaGraphics::pop_box(int16 x, int16 y, int16 x1, int16 y1, int16 col1, int16 col2, int16 back_col) {
|
|
if (back_col < 255)
|
|
boxFill(x, y, x1, y1, back_col);
|
|
drawLine(x, y1, x1, y1, col2);
|
|
drawLine(x1, y, x1, y1, col2);
|
|
drawLine(x, y, x1, y, col1);
|
|
drawLine(x, y, x, y1 + 1, col1);
|
|
}
|
|
|
|
void McgaGraphics::copyToScreen() {
|
|
byte *destP = (byte *)g_screen->getPixels();
|
|
const byte *ptr = _G(workpage);
|
|
Common::copy(ptr + 4, ptr + 4 + (SCREEN_WIDTH * SCREEN_HEIGHT), destP);
|
|
g_screen->markAllDirty();
|
|
}
|
|
|
|
void McgaGraphics::spriteSave(byte *spritePtr, int16 x, int16 y, int16 width, int16 height) {
|
|
if (width < 4)
|
|
width = 4;
|
|
if (height <= 0)
|
|
height = 1;
|
|
if (x < 0) {
|
|
x = 0;
|
|
width -= (0 - x);
|
|
}
|
|
if ((x + width) > 320 + 1)
|
|
width = 320 - x;
|
|
if (y < 0) {
|
|
y = 0;
|
|
height -= (0 - y);
|
|
}
|
|
if ((y + height) > 200 + 1)
|
|
height = 200 - y;
|
|
if (width < 1)
|
|
width = 0;
|
|
if (height <= 0)
|
|
height = 0;
|
|
|
|
*((int16 *)spritePtr) = width;
|
|
spritePtr += 2;
|
|
*((int16 *)spritePtr) = height;
|
|
spritePtr += 2;
|
|
|
|
int pitch = SCREEN_WIDTH;
|
|
byte *scrP = _G(currentScreen).getPixels() + y * SCREEN_WIDTH + x;
|
|
|
|
if (width >= 1 && height >= 1) {
|
|
for (int row = 0; row < height; ++row) {
|
|
Common::copy(scrP, scrP + width, spritePtr);
|
|
scrP += pitch;
|
|
spritePtr += width;
|
|
}
|
|
}
|
|
}
|
|
|
|
static bool mspr_set_mcga_clip(int x, int y, int pitch, int &width, int &height, const byte *&srcP, byte *&destP) {
|
|
if (y < 0) {
|
|
int yDiff = ABS(0 - y);
|
|
height -= yDiff;
|
|
srcP += yDiff * width;
|
|
y = 0;
|
|
}
|
|
if (height < 1)
|
|
return false;
|
|
|
|
if (x < 0) {
|
|
int xDiff = ABS(0 - x);
|
|
width -= xDiff;
|
|
srcP += xDiff;
|
|
x = 0;
|
|
}
|
|
if (width < 1)
|
|
return false;
|
|
|
|
int x2 = x + width;
|
|
if (x2 > 320) {
|
|
int xDiff = x2 - 320;
|
|
width -= xDiff;
|
|
}
|
|
if (width <= 1)
|
|
return false;
|
|
|
|
int y2 = y + height;
|
|
if (y2 > 200) {
|
|
int yDiff = y2 - 200;
|
|
height -= yDiff;
|
|
}
|
|
if (height < 1)
|
|
return false;
|
|
|
|
destP = _G(currentScreen).getPixels() + pitch * y + x;
|
|
return true;
|
|
}
|
|
|
|
void McgaGraphics::spriteSet(byte *sptr, int16 x, int16 y, int16 scrWidth, uint16 spriteWidth, uint16 spriteHeight) {
|
|
if (!sptr)
|
|
return;
|
|
|
|
byte *destP;
|
|
int width, height;
|
|
|
|
if (spriteWidth == 0 && spriteHeight == 0) {
|
|
width = *((const int16 *)sptr);
|
|
sptr += 2;
|
|
height = *((const int16 *)sptr);
|
|
sptr += 2;
|
|
} else {
|
|
width = spriteWidth;
|
|
height = spriteHeight;
|
|
}
|
|
|
|
const byte *srcP = sptr;
|
|
_G(spriteWidth) = width;
|
|
|
|
if (!(height >= 1 && width >= 4))
|
|
return;
|
|
|
|
int pitch = scrWidth ? scrWidth : SCREEN_WIDTH;
|
|
if (!mspr_set_mcga_clip(x, y, pitch, width, height, srcP, destP))
|
|
return;
|
|
const int destPitchRemainder = pitch - width;
|
|
const int srcPitchRemainder = _G(spriteWidth) - width;
|
|
|
|
for (int row = 0; row < height; ++row,
|
|
srcP += srcPitchRemainder, destP += destPitchRemainder) {
|
|
for (int col = 0; col < width; ++col, ++srcP, ++destP) {
|
|
if (*srcP != 0)
|
|
*destP = *srcP;
|
|
}
|
|
}
|
|
}
|
|
|
|
void McgaGraphics::map_spr2screen(byte *sptr, int16 x, int16 y) {
|
|
const int16 width = ((const int16 *)sptr)[0];
|
|
const int16 height = ((const int16 *)sptr)[1];
|
|
if (width >= 320 || height >= 200) {
|
|
sptr += 4 + y * width + x;
|
|
|
|
for (int row = 0; row < SCREEN_HEIGHT; ++row, sptr += width) {
|
|
Common::copy(
|
|
sptr,
|
|
sptr + SCREEN_WIDTH,
|
|
(byte *)_G(currentScreen).getBasePtr(0, row)
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
int16 McgaGraphics::scanxy(int16 x, int16 y, int16 fcol, int16 bcol, int16 cur_col, int16 scrwidth,
|
|
const char *string, ...) {
|
|
int16 i, j, stelle, stellemax, mode = 0;
|
|
int16 disp_stelle = 0, disp_stellemax = 0, disp_akt = 0;
|
|
int16 ret = 0;
|
|
int16 delay_flag;
|
|
int8 ende = 0, zeichen, zaehler, cursor_z;
|
|
char zstring[81], z1string[81];
|
|
memset(zstring, 0, 81);
|
|
memset(z1string, 0, 81);
|
|
int8 einfuege_cur = 0;
|
|
unsigned char char_anf = 0, char_end = 0;
|
|
uint16 vorzeichen = 0;
|
|
int8 x_mode = 0;
|
|
int8 eing = 0;
|
|
uint16 izahl = 0;
|
|
uint16 *intzahl = nullptr;
|
|
int16 *intzahl1 = nullptr;
|
|
uint32 *longzahl = nullptr;
|
|
char *charstr = nullptr;
|
|
uint32 luzahl;
|
|
va_list parptr;
|
|
va_start(parptr, string);
|
|
|
|
if ((x == -1) || (y == -1)) {
|
|
x = _G(gcurx);
|
|
y = _G(gcury);
|
|
} else {
|
|
_G(gcurx) = x;
|
|
_G(gcury) = y;
|
|
}
|
|
|
|
move(x, y);
|
|
i = 0;
|
|
|
|
ChewyFont *font = _G(fontMgr)->getFont();
|
|
|
|
while (!ende) {
|
|
zeichen = string[i];
|
|
++i;
|
|
|
|
if ((zeichen >= font->getFirst()) && (zeichen <= font->getLast()) && (zeichen != 127)) {
|
|
if (zeichen == '%') {
|
|
zeichen = string[i];
|
|
++i;
|
|
zaehler = 0;
|
|
|
|
if (zeichen == '[') {
|
|
char_anf = string[i];
|
|
++i;
|
|
++i;
|
|
char_end = string[i];
|
|
++i;
|
|
++i;
|
|
zeichen = string[i];
|
|
++i;
|
|
}
|
|
|
|
while ((zeichen >= '0') && (zeichen <= '9')) {
|
|
zaehler *= 10;
|
|
zaehler += (zeichen - 48);
|
|
if (zaehler > 80)
|
|
zaehler = 80;
|
|
zeichen = string[i];
|
|
++i;
|
|
}
|
|
if (_einfuegen == 1) {
|
|
einfuege_cur = 125;
|
|
cursor_z = '_' + einfuege_cur;
|
|
} else {
|
|
einfuege_cur = 0;
|
|
cursor_z = '_';
|
|
}
|
|
|
|
switch (zeichen) {
|
|
case 'd':
|
|
vorzeichen = '-';
|
|
// Fall through
|
|
case 'u':
|
|
if ((char_anf == 0) && (char_end == 0)) {
|
|
char_anf = '0';
|
|
char_end = '9';
|
|
}
|
|
if (!zaehler)
|
|
zaehler = 4;
|
|
if (zeichen == 'd') {
|
|
mode = 4;
|
|
intzahl1 = va_arg(parptr, int16 *);
|
|
luzahl = intzahl1[0];
|
|
} else {
|
|
mode = 1;
|
|
intzahl = va_arg(parptr, uint16 *);
|
|
luzahl = intzahl[0];
|
|
}
|
|
if (luzahl != 0)
|
|
Common::sprintf_s(zstring, "%u", luzahl);
|
|
else {
|
|
zstring[0] = '0';
|
|
zstring[1] = 0 ;
|
|
}
|
|
break;
|
|
case 'l':
|
|
vorzeichen = '-';
|
|
if ((char_anf == 0) && (char_end == 0))
|
|
vorzeichen = '-';
|
|
{
|
|
char_anf = '0';
|
|
char_end = '9';
|
|
}
|
|
mode = 2;
|
|
if (!zaehler)
|
|
zaehler = 9;
|
|
longzahl = va_arg(parptr, uint32 *);
|
|
luzahl = longzahl[0];
|
|
if (luzahl != 0)
|
|
Common::sprintf_s(zstring, "%u", luzahl);
|
|
else {
|
|
zstring[0] = '0';
|
|
zstring[1] = 0 ;
|
|
}
|
|
break;
|
|
case 's':
|
|
if ((char_anf == 0) && (char_end == 0)) {
|
|
char_anf = ' ';
|
|
char_end = 0xdc;
|
|
}
|
|
mode = 3;
|
|
vorzeichen = 0xFF;
|
|
if (!zaehler)
|
|
zaehler = 81;
|
|
charstr = va_arg(parptr, char *);
|
|
Common::strcpy_s(zstring, charstr);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
zeichen = string[i];
|
|
++i;
|
|
if (zeichen == 'x') {
|
|
x_mode = 1;
|
|
zeichen = string[i];
|
|
++i;
|
|
}
|
|
disp_stellemax = 0;
|
|
|
|
while ((zeichen >= '0') && (zeichen <= '9')) {
|
|
disp_stellemax *= 10;
|
|
disp_stellemax += (zeichen - 48);
|
|
if (disp_stellemax > 80)
|
|
disp_stellemax = 80;
|
|
zeichen = string[i];
|
|
++i;
|
|
}
|
|
if (!disp_stellemax)
|
|
disp_stellemax = zaehler;
|
|
|
|
stelle = 0;
|
|
disp_stelle = 0;
|
|
while (zstring[stelle] != 0) {
|
|
++stelle;
|
|
}
|
|
stellemax = stelle;
|
|
if (stelle > disp_stellemax) {
|
|
disp_stelle = stelle - disp_stellemax;
|
|
disp_akt = disp_stellemax;
|
|
} else {
|
|
disp_akt = stelle;
|
|
disp_stelle = 0;
|
|
}
|
|
for (j = stelle; j < j + 1; j++) {
|
|
while ((!kbhit()) || (eing != 0)) {
|
|
EVENTS_UPDATE;
|
|
|
|
eing = 0;
|
|
move(x, y);
|
|
|
|
for (i = disp_stelle; i <= disp_stellemax + disp_stelle; ++i) {
|
|
if (zstring[i] != 0) {
|
|
putz(zstring[i], fcol, bcol, scrwidth);
|
|
_G(gcurx) += _G(fontMgr)->getFont()->getDataWidth();
|
|
} else {
|
|
putz(zstring[i], fcol, bcol, scrwidth);
|
|
_G(gcurx) += _G(fontMgr)->getFont()->getDataWidth();
|
|
break;
|
|
}
|
|
}
|
|
for (delay_flag = 0; (delay_flag < 10) && (!kbhit()); delay_flag++) {
|
|
izahl = devices();
|
|
|
|
if (izahl == Common::KEYCODE_RETURN) {
|
|
eing = 2;
|
|
break;
|
|
} else if (izahl == Common::KEYCODE_ESCAPE) {
|
|
eing = 2;
|
|
break;
|
|
}
|
|
}
|
|
const uint16 fontWidth = _G(fontMgr)->getFont()->getDataWidth();
|
|
plot_scan_cur((x + (disp_akt)*fontWidth), _G(gcury), cur_col, 300, scrwidth, cursor_z);
|
|
for (delay_flag = 0; (delay_flag < 10) && (!kbhit()); delay_flag++) {
|
|
izahl = devices();
|
|
if (izahl == Common::KEYCODE_RETURN) {
|
|
eing = 2;
|
|
break;
|
|
} else if (izahl == Common::KEYCODE_ESCAPE) {
|
|
eing = 2;
|
|
break;
|
|
}
|
|
}
|
|
if (eing == 2)
|
|
break;
|
|
}
|
|
if (eing < 2)
|
|
izahl = (uint16)getch();
|
|
|
|
if (izahl == Common::KEYCODE_RETURN) {
|
|
ret = Common::KEYCODE_RETURN;
|
|
ende = 1;
|
|
break;
|
|
}
|
|
|
|
if (izahl == Common::KEYCODE_ESCAPE) {
|
|
ret = Common::KEYCODE_ESCAPE;
|
|
ende = 2;
|
|
break;
|
|
}
|
|
|
|
if (izahl == Common::KEYCODE_BACKSPACE) {
|
|
eing = 1;
|
|
while (kbhit())
|
|
getch();
|
|
|
|
if (stelle > 0) {
|
|
const uint16 fontWidth = _G(fontMgr)->getFont()->getDataWidth();
|
|
Common::strcpy_s(zstring + stelle - 1, sizeof(zstring) - (stelle - 1), zstring + stelle);
|
|
plot_scan_cur((x + disp_akt * fontWidth), _G(gcury), bcol, bcol, scrwidth, cursor_z);
|
|
--stelle;
|
|
--stellemax;
|
|
if (disp_stelle > 0)
|
|
--disp_stelle;
|
|
else if (disp_akt > 0) {
|
|
--disp_akt;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (izahl == 0) {
|
|
izahl = getch();
|
|
|
|
if (izahl == 75) {
|
|
while (kbhit())
|
|
getch();
|
|
eing = 1;
|
|
if (stelle <= 0) {
|
|
} else {
|
|
if (disp_akt > 0) {
|
|
--disp_akt;
|
|
} else {
|
|
if (disp_stelle > 0)
|
|
--disp_stelle;
|
|
}
|
|
--stelle;
|
|
}
|
|
}
|
|
|
|
if (izahl == 77) {
|
|
eing = 1;
|
|
while (kbhit())
|
|
getch();
|
|
++j;
|
|
if (stelle >= stellemax) {
|
|
stelle = stellemax;
|
|
} else {
|
|
if (disp_akt < disp_stellemax) {
|
|
++disp_akt;
|
|
} else {
|
|
if (disp_stelle <= (zaehler - disp_stellemax)) {
|
|
++disp_stelle;
|
|
}
|
|
}
|
|
++stelle;
|
|
}
|
|
}
|
|
|
|
if (izahl == 71) {
|
|
stelle = 0;
|
|
disp_akt = 0;
|
|
disp_stelle = 0;
|
|
}
|
|
|
|
if (izahl == 79) {
|
|
stelle = stellemax;
|
|
if (stellemax < disp_stellemax) {
|
|
disp_stelle = 0;
|
|
disp_akt = stellemax;
|
|
} else {
|
|
disp_akt = disp_stellemax;
|
|
disp_stelle = stelle - disp_stellemax;
|
|
}
|
|
}
|
|
|
|
if ((izahl == 83) && (stelle < stellemax)) {
|
|
while (kbhit())
|
|
getch();
|
|
eing = 1;
|
|
Common::strcpy_s(zstring + stelle, sizeof(zstring) - stelle, zstring + stelle + 1);
|
|
--stellemax;
|
|
}
|
|
|
|
if (izahl == 82) {
|
|
g_events->delay(200);
|
|
if (!_einfuegen) {
|
|
_einfuegen = 1;
|
|
einfuege_cur = 125;
|
|
cursor_z = '_' + einfuege_cur;
|
|
} else {
|
|
_einfuegen = 0;
|
|
einfuege_cur = 0;
|
|
cursor_z = '_';
|
|
}
|
|
}
|
|
izahl = 0;
|
|
}
|
|
|
|
if (izahl < char_anf || izahl > char_end) {
|
|
if (vorzeichen == izahl && !stelle);
|
|
else {
|
|
if (x_mode == 1) {
|
|
ende = 1;
|
|
ret = izahl;
|
|
break;
|
|
}
|
|
continue;
|
|
}
|
|
}
|
|
if ((stelle == stellemax) && (stellemax >= zaehler)) {
|
|
stellemax = zaehler;
|
|
} else {
|
|
if (disp_akt < disp_stellemax) {
|
|
++disp_akt;
|
|
} else {
|
|
if (disp_stelle <= (zaehler - disp_stellemax)) {
|
|
++disp_stelle;
|
|
disp_akt = disp_stellemax;
|
|
}
|
|
}
|
|
if (!_einfuegen) {
|
|
zstring[stelle] = izahl;
|
|
zstring[stellemax + 1] = 0;
|
|
if (stelle == stellemax) {
|
|
++stelle;
|
|
++stellemax;
|
|
} else {
|
|
++stelle;
|
|
}
|
|
} else {
|
|
Common::strcpy_s(z1string, zstring);
|
|
Common::strcpy_s(zstring + stelle + 1, sizeof(zstring) - (stelle + 1), z1string + stelle);
|
|
zstring[stelle] = izahl;
|
|
zstring[(int)zaehler] = 0;
|
|
if (stellemax < zaehler) {
|
|
++stelle;
|
|
++stellemax;
|
|
} else {
|
|
++stelle;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (x_mode == 1) {
|
|
ende = 1;
|
|
ret = izahl;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
move(x, y);
|
|
for (i = disp_stelle; i <= disp_stellemax + disp_stelle; ++i) {
|
|
if (zstring[i] != 0) {
|
|
putz(zstring[i], fcol, bcol, scrwidth);
|
|
_G(gcurx) += _G(fontMgr)->getFont()->getDataWidth();
|
|
} else {
|
|
putz(zstring[i], fcol, bcol, scrwidth);
|
|
_G(gcurx) += _G(fontMgr)->getFont()->getDataWidth();
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (ende == 1) {
|
|
switch (mode) {
|
|
case 1:
|
|
intzahl[0] = atoi(zstring);
|
|
break;
|
|
case 2:
|
|
longzahl[0] = atol(zstring);
|
|
break;
|
|
case 3:
|
|
Common::strcpy_s(charstr, 81, zstring);
|
|
break;
|
|
case 4:
|
|
intzahl1[0] = atoi(zstring);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
va_end(parptr);
|
|
return ret;
|
|
}
|
|
|
|
void McgaGraphics::plot_scan_cur(int16 x, int16 y, int16 fcol, int16 bcol, int16 scrwidth, char cursor_z) {
|
|
move(x, y);
|
|
putz(cursor_z, fcol, bcol, scrwidth);
|
|
}
|
|
|
|
void McgaGraphics::printxy(int16 x, int16 y, int16 fgCol, int16 bgCol, int16 scrwidth, const char *string) {
|
|
ChewyFont *font = _G(fontMgr)->getFont();
|
|
Graphics::Surface *textSurface = font->getLine(string);
|
|
byte *data = (byte *)textSurface->getPixels();
|
|
|
|
for (int curX = 0; curX < textSurface->pitch; curX++) {
|
|
for (int curY = 0; curY < textSurface->h; curY++) {
|
|
if (curX + x < 320 && curY + y < 200) {
|
|
byte *src = data + (curY * textSurface->pitch) + curX;
|
|
byte *dst = (byte *)_G(currentScreen).getBasePtr(curX + x, curY + y);
|
|
if (*src != 0xFF)
|
|
*dst = fgCol;
|
|
else if (bgCol < 0xFF)
|
|
*dst = bgCol;
|
|
}
|
|
}
|
|
}
|
|
|
|
g_screen->addDirtyRect(Common::Rect(
|
|
x, y, x + textSurface->pitch, y + textSurface->h));
|
|
|
|
textSurface->free();
|
|
delete textSurface;
|
|
}
|
|
|
|
void McgaGraphics::move(int16 x, int16 y) {
|
|
_G(gcurx) = x;
|
|
_G(gcury) = y;
|
|
}
|
|
|
|
int16 McgaGraphics::devices() {
|
|
int16 i = 0;
|
|
// TODO: This is wrong. 'i' should be initialized to "button status" (BX after a call to Int33h/03
|
|
if (i > 1) {
|
|
i = 27;
|
|
} else if (i == 1)
|
|
i = 13;
|
|
else
|
|
i = 0;
|
|
|
|
return i;
|
|
}
|
|
|
|
void McgaGraphics::scale_set(byte *sptr, int16 x, int16 y, int16 xdiff_, int16 ydiff_, int16 scrwidth) {
|
|
if (xdiff_ || ydiff_)
|
|
zoom_set(sptr, x, y, xdiff_, ydiff_, scrwidth);
|
|
else
|
|
spriteSet(sptr, x, y, scrwidth);
|
|
}
|
|
|
|
void McgaGraphics::putz(unsigned char c, int16 fgCol, int16 bgCol, int16 scrWidth) {
|
|
const int16 x = _G(gcurx);
|
|
const int16 y = _G(gcury);
|
|
ChewyFont *font = _G(fontMgr)->getFont();
|
|
Graphics::Surface *textSurface = font->getLine(Common::String(c));
|
|
byte *data = (byte *)textSurface->getPixels();
|
|
|
|
for (int curX = 0; curX < textSurface->pitch; curX++) {
|
|
for (int curY = 0; curY < textSurface->h; curY++) {
|
|
if (curX + x < 320 && curY + y < 200) {
|
|
byte *src = data + (curY * textSurface->pitch) + curX;
|
|
byte *dst = (byte *)_G(currentScreen).getBasePtr(curX + x, curY + y);
|
|
if (*src != 0xFF)
|
|
*dst = fgCol;
|
|
else if (bgCol < 0xFF)
|
|
*dst = bgCol;
|
|
}
|
|
}
|
|
}
|
|
|
|
g_screen->addDirtyRect(Common::Rect(
|
|
x, y, x + textSurface->pitch, y + textSurface->h));
|
|
|
|
textSurface->free();
|
|
delete textSurface;
|
|
}
|
|
|
|
void McgaGraphics::setXVals() {
|
|
if (_zoomSpriteDeltaX2 == 0) {
|
|
_zoomSpriteXVal1 = 0;
|
|
_zoomSpriteXVal2 = 1;
|
|
} else {
|
|
_zoomSpriteXVal1 = _G(spriteWidth) / _zoomSpriteDeltaX2;
|
|
_zoomSpriteXVal2 = 1000 * (_G(spriteWidth) % _zoomSpriteDeltaX2);
|
|
_zoomSpriteXVal2 /= _zoomSpriteDeltaX2;
|
|
}
|
|
}
|
|
|
|
void McgaGraphics::setYVals(int spriteHeight) {
|
|
if (_zoomSpriteDeltaY2 == 0) {
|
|
_zoomSpriteYVal1 = 0;
|
|
_zoomSpriteYVal2 = 1;
|
|
} else {
|
|
_zoomSpriteYVal1 = spriteHeight / _zoomSpriteDeltaY2;
|
|
_zoomSpriteYVal2 = 1000 * (spriteHeight % _zoomSpriteDeltaY2);
|
|
_zoomSpriteYVal2 /= _zoomSpriteDeltaY2;
|
|
}
|
|
}
|
|
|
|
void McgaGraphics::clip(byte *&source, byte *&dest, int16 &x, int16 &y) {
|
|
if (y < 0) {
|
|
int yCount = 0 - y;
|
|
_zoomSpriteDeltaY2 -= yCount;
|
|
|
|
--yCount;
|
|
if (yCount >= 1) {
|
|
for (int yc = 0, countY = _zoomSpriteYVal2; yc < yCount; ++yc) {
|
|
source += _G(spriteWidth) * _zoomSpriteYVal1;
|
|
dest += SCREEN_WIDTH;
|
|
|
|
while (countY > 1000) {
|
|
countY -= 1000;
|
|
source += _G(spriteWidth);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (_zoomSpriteDeltaY2 <= 0) {
|
|
source = nullptr;
|
|
return;
|
|
}
|
|
|
|
if (x < 0) {
|
|
int xCount = 0 - x;
|
|
_zoomSpriteDeltaX2 -= xCount;
|
|
dest += xCount;
|
|
|
|
--xCount;
|
|
if (xCount >= 1) {
|
|
for (int xc = 0, countX = _zoomSpriteXVal2; xc < xCount; ++xc) {
|
|
source += _zoomSpriteXVal1;
|
|
while (countX >= 1000) {
|
|
countX -= 1000;
|
|
++source;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (_zoomSpriteDeltaX2 > 0) {
|
|
int x2 = x + _zoomSpriteDeltaX2;
|
|
if (x2 >= 320) {
|
|
_zoomSpriteDeltaX2 -= x2 - 320;
|
|
}
|
|
|
|
if (_zoomSpriteDeltaY2 > 0) {
|
|
int y2 = y + _zoomSpriteDeltaY2;
|
|
if (y2 >= 200) {
|
|
_zoomSpriteDeltaY2 -= y2 - 200;
|
|
}
|
|
if (_zoomSpriteDeltaY2 <= 0)
|
|
source = nullptr;
|
|
} else {
|
|
source = nullptr;
|
|
}
|
|
} else {
|
|
source = nullptr;
|
|
}
|
|
}
|
|
|
|
void McgaGraphics::zoom_set(byte *source, int16 x, int16 y, int16 xDiff, int16 yDiff, int16 scrWidth) {
|
|
_G(spriteWidth) = ((int16 *)source)[0];
|
|
int spriteHeight = ((int16 *)source)[1];
|
|
source += 4;
|
|
|
|
_zoomSpriteDeltaX2 = _G(spriteWidth) + xDiff;
|
|
_zoomSpriteDeltaY2 = spriteHeight + yDiff;
|
|
|
|
setXVals();
|
|
setYVals(spriteHeight);
|
|
|
|
const int16 screenWidth = scrWidth > 0 ? scrWidth : SCREEN_WIDTH;
|
|
byte *scrP = _G(currentScreen).getPixels() + y * screenWidth + x;
|
|
|
|
clip(source, scrP, x, y);
|
|
|
|
if (source) {
|
|
for (int yc = _zoomSpriteDeltaY2, countY = _zoomSpriteYVal2; yc > 0; --yc) {
|
|
byte *srcLine = source;
|
|
byte *scrLine = scrP;
|
|
|
|
for (int xc = _zoomSpriteDeltaX2, countX = _zoomSpriteXVal2; xc > 0; --xc) {
|
|
if (*source)
|
|
*scrP = *source;
|
|
|
|
++scrP;
|
|
source += _zoomSpriteXVal1;
|
|
countX += _zoomSpriteXVal2;
|
|
while (countX > 1000) {
|
|
countX -= 1000;
|
|
++source;
|
|
}
|
|
}
|
|
|
|
source = srcLine;
|
|
scrP = scrLine + SCREEN_WIDTH;
|
|
|
|
for (int ySkip = 0; ySkip < _zoomSpriteYVal1; ++ySkip) {
|
|
source += _G(spriteWidth);
|
|
}
|
|
|
|
countY += _zoomSpriteYVal2;
|
|
while (countY > 1000) {
|
|
countY -= 1000;
|
|
source += _G(spriteWidth);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
int16 McgaGraphics::findHotspot(const Common::Rect *hotspots) {
|
|
int16 i = 0;
|
|
|
|
do {
|
|
if (hotspots[i].contains(g_events->_mousePos))
|
|
return i;
|
|
i++;
|
|
} while (hotspots[i].left != -1);
|
|
|
|
return -1;
|
|
}
|
|
|
|
} // namespace Chewy
|