scummvm/engines/lab/graphics.cpp

1007 lines
33 KiB
C++
Raw Normal View History

2014-10-06 14:50:05 +02:00
/* 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.
*
*/
/*
* This code is based on Labyrinth of Time code with assistance of
*
* Copyright (c) 1993 Terra Nova Development
* Copyright (c) 2004 The Wyrmkeep Entertainment Co.
*
*/
2014-12-27 14:18:40 +01:00
#include "lab/lab.h"
2015-12-01 20:10:42 +01:00
#include "lab/anim.h"
2014-10-06 14:50:05 +02:00
#include "lab/parsetypes.h"
#include "lab/image.h"
2014-10-06 14:50:05 +02:00
#include "lab/labfun.h"
#include "lab/parsefun.h"
#include "lab/text.h"
#include "lab/resource.h"
#include "lab/graphics.h"
2014-10-06 14:50:05 +02:00
namespace Lab {
2014-12-26 00:32:42 +01:00
BitMap bit1, bit2, *DispBitMap = &bit1, *DrawBitMap = &bit1;
2014-10-06 14:50:05 +02:00
DisplayMan::DisplayMan(LabEngine *vm) : _vm(vm) {
_longWinInFront = false;
_lastMessageLong = false;
_screenBytesPerPage = 65536;
_curapen = 0;
}
2015-12-02 11:58:53 +01:00
/*****************************************************************************/
/* Scales the x co-ordinates to that of the new display. In the room parser */
/* file, co-ordinates are set up on a 360x336 display. */
/*****************************************************************************/
uint16 DisplayMan::scaleX(uint16 x) {
if (_vm->_isHiRes)
2015-12-02 11:58:53 +01:00
return (uint16)((x * 16) / 9);
else
return (uint16)((x * 8) / 9);
}
/*****************************************************************************/
/* Scales the y co-ordinates to that of the new display. In the room parser */
/* file, co-ordinates are set up on a 368x336 display. */
/*****************************************************************************/
uint16 DisplayMan::scaleY(uint16 y) {
if (_vm->_isHiRes)
2015-12-02 11:58:53 +01:00
return (y + (y / 14));
else
return ((y * 10) / 24);
}
/*****************************************************************************/
/* Scales the VGA cords to SVGA if necessary; otherwise, returns VGA cords. */
/*****************************************************************************/
int16 DisplayMan::VGAScaleX(int16 x) {
if (_vm->_isHiRes)
2015-12-02 11:58:53 +01:00
return (x * 2);
else
return x;
}
/*****************************************************************************/
/* Scales the VGA cords to SVGA if necessary; otherwise, returns VGA cords. */
/*****************************************************************************/
int16 DisplayMan::VGAScaleY(int16 y) {
if (_vm->_isHiRes)
2015-12-02 11:58:53 +01:00
return ((y * 12) / 5);
else
return y;
}
uint16 DisplayMan::SVGACord(uint16 cord) {
if (_vm->_isHiRes)
2015-12-02 11:58:53 +01:00
return cord;
else
return 0;
}
2014-10-06 14:50:05 +02:00
/*---------------------------------------------------------------------------*/
/*------ From readPict.c. Reads in pictures and animations from disk. ------*/
/*---------------------------------------------------------------------------*/
/*****************************************************************************/
/* Reads in a picture into the dest bitmap. */
/*****************************************************************************/
bool DisplayMan::readPict(const char *filename, bool playOnce) {
_vm->_anim->stopDiff();
2014-10-06 14:50:05 +02:00
byte **file = _vm->_music->newOpen(filename);
2014-10-06 14:50:05 +02:00
if (file == NULL) {
if ((filename[0] == 'p') || (filename[0] == 'P'))
blackScreen();
return false;
}
DispBitMap->_bytesPerRow = _vm->_screenWidth;
DispBitMap->_rows = _vm->_screenHeight;
DispBitMap->_flags = BITMAPF_VIDEO;
2014-10-06 14:50:05 +02:00
_vm->_anim->readDiff(playOnce);
2014-10-06 14:50:05 +02:00
return true;
}
/*****************************************************************************/
/* Reads in a picture into buffer memory. */
/*****************************************************************************/
byte *DisplayMan::readPictToMem(const char *filename, uint16 x, uint16 y) {
2015-12-01 11:18:11 +01:00
byte *mem;
2014-10-06 14:50:05 +02:00
_vm->_anim->stopDiff();
2014-10-06 14:50:05 +02:00
2015-12-01 11:18:11 +01:00
allocFile((void **)&mem, (int32)x * (int32)y, "Bitmap");
byte *curMem = mem;
2014-10-06 14:50:05 +02:00
byte **file = _vm->_music->newOpen(filename);
2014-10-06 14:50:05 +02:00
if (file == NULL)
return NULL;
DispBitMap->_bytesPerRow = x;
2015-12-03 14:30:08 +02:00
DispBitMap->_rows = y;
DispBitMap->_flags = BITMAPF_NONE;
DispBitMap->_planes[0] = curMem;
DispBitMap->_planes[1] = DispBitMap->_planes[0] + 0x10000;
DispBitMap->_planes[2] = DispBitMap->_planes[1] + 0x10000;
DispBitMap->_planes[3] = DispBitMap->_planes[2] + 0x10000;
DispBitMap->_planes[4] = DispBitMap->_planes[3] + 0x10000;
2014-10-06 14:50:05 +02:00
_vm->_anim->readDiff(true);
2014-10-06 14:50:05 +02:00
2015-12-01 11:18:11 +01:00
return mem;
2014-10-06 14:50:05 +02:00
}
2015-12-03 14:30:08 +02:00
/*****************************************************************************/
/* Reads in a music file. Ignores any graphics. */
/*****************************************************************************/
bool readMusic(const char *filename, bool waitTillFinished) {
Common::File *file = g_lab->_resource->openDataFile(filename, MKTAG('D', 'I', 'F', 'F'));
g_lab->_music->updateMusic();
if (!g_lab->_music->_doNotFilestopSoundEffect)
g_lab->_music->stopSoundEffect();
if (!file)
return false;
g_lab->_anim->_doBlack = false;
g_lab->_anim->readSound(waitTillFinished, file);
return true;
}
2014-10-06 14:50:05 +02:00
/*---------------------------------------------------------------------------*/
/*------------ Does all the text rendering to the message boxes. ------------*/
/*---------------------------------------------------------------------------*/
bool DoNotDrawMessage = false;
/*----- The flowText routines -----*/
/******************************************************************************/
/* Extracts the first word from a string. */
/******************************************************************************/
2015-12-01 10:35:31 +01:00
static void getWord(char *wordBuffer, const char *mainBuffer, uint16 *wordWidth) {
2014-10-06 14:50:05 +02:00
uint16 width = 0;
2015-12-01 10:35:31 +01:00
while ((mainBuffer[width] != ' ') && mainBuffer[width] &&
(mainBuffer[width] != '\n')) {
wordBuffer[width] = mainBuffer[width];
2014-10-06 14:50:05 +02:00
width++;
}
2015-12-01 10:35:31 +01:00
wordBuffer[width] = 0;
2014-10-06 14:50:05 +02:00
2015-12-01 10:35:31 +01:00
*wordWidth = width;
2014-10-06 14:50:05 +02:00
}
/******************************************************************************/
/* Gets a line of text for flowText; makes sure that its length is less than */
/* or equal to the maximum width. */
/******************************************************************************/
2015-12-01 10:35:31 +01:00
static void getLine(TextFont *tf, char *lineBuffer, const char **mainBuffer, uint16 lineWidth) {
uint16 curWidth = 0, wordWidth;
char wordBuffer[100];
2014-10-06 14:50:05 +02:00
bool doit = true;
2015-12-01 10:35:31 +01:00
lineWidth += textLength(tf, " ", 1);
2014-10-06 14:50:05 +02:00
2015-12-01 10:35:31 +01:00
lineBuffer[0] = 0;
2014-10-06 14:50:05 +02:00
2015-12-01 10:35:31 +01:00
while ((*mainBuffer)[0] && doit) {
getWord(wordBuffer, *mainBuffer, &wordWidth);
strcat(wordBuffer, " ");
2014-10-06 14:50:05 +02:00
2015-12-01 10:35:31 +01:00
if ((curWidth + textLength(tf, wordBuffer, wordWidth + 1)) <= lineWidth) {
strcat(lineBuffer, wordBuffer);
(*mainBuffer) += wordWidth;
2014-10-06 14:50:05 +02:00
2015-12-01 10:35:31 +01:00
if ((*mainBuffer)[0] == '\n')
2014-10-06 14:50:05 +02:00
doit = false;
2015-12-01 10:35:31 +01:00
if ((*mainBuffer)[0])
(*mainBuffer)++;
2014-10-06 14:50:05 +02:00
2015-12-01 10:35:31 +01:00
curWidth = textLength(tf, lineBuffer, strlen(lineBuffer));
2014-10-06 14:50:05 +02:00
} else
doit = false;
}
}
/******************************************************************************/
/* Dumps a chunk of text to an arbitrary box; flows it within that box and */
/* optionally centers it. Returns the number of characters that were */
/* processed. */
/* */
/* Note: Every individual word MUST be int16 enough to fit on a line, and */
/* each line less than 255 characters. */
/******************************************************************************/
uint32 DisplayMan::flowText(void *font, /* the TextAttr pointer */
int16 spacing, /* How much vertical spacing between the lines */
byte pencolor, /* pen number to use for text */
byte backpen, /* the background color */
2014-10-06 14:50:05 +02:00
bool fillback, /* Whether to fill the background */
bool centerh, /* Whether to center the text horizontally */
bool centerv, /* Whether to center the text vertically */
bool output, /* Whether to output any text */
uint16 x1, /* Cords */
uint16 y1, uint16 x2, uint16 y2, const char *str) { /* The text itself */
2015-12-03 11:01:50 +01:00
TextFont *_msgFont = (TextFont *)font;
2014-10-06 14:50:05 +02:00
char linebuffer[256];
const char *temp;
uint16 numlines, actlines, fontheight, width;
uint16 x, y;
if (fillback) {
setAPen(backpen);
rectFill(x1, y1, x2, y2);
2014-10-06 14:50:05 +02:00
}
if (str == NULL)
return 0L;
setAPen(pencolor);
2014-10-06 14:50:05 +02:00
2015-12-03 11:01:50 +01:00
fontheight = textHeight(_msgFont) + spacing;
2014-10-06 14:50:05 +02:00
numlines = (y2 - y1 + 1) / fontheight;
width = x2 - x1 + 1;
y = y1;
if (centerv && output) {
temp = str;
actlines = 0;
while (temp[0]) {
2015-12-03 11:01:50 +01:00
getLine(_msgFont, linebuffer, &temp, width);
2014-10-06 14:50:05 +02:00
actlines++;
}
if (actlines <= numlines)
y += ((y2 - y1 + 1) - (actlines * fontheight)) / 2;
}
temp = str;
while (numlines && str[0]) {
2015-12-03 11:01:50 +01:00
getLine(_msgFont, linebuffer, &str, width);
2014-10-06 14:50:05 +02:00
x = x1;
if (centerh)
2015-12-03 11:01:50 +01:00
x += (width - textLength(_msgFont, linebuffer, strlen(linebuffer))) / 2;
2014-10-06 14:50:05 +02:00
if (output)
2015-12-03 11:01:50 +01:00
text(_msgFont, x, y, pencolor, linebuffer, strlen(linebuffer));
2014-10-06 14:50:05 +02:00
numlines--;
y += fontheight;
}
return (str - temp);
}
/******************************************************************************/
/* Calls flowText, but flows it to memory. Same restrictions as flowText. */
/******************************************************************************/
uint32 DisplayMan::flowTextToMem(Image *destIm, void *font, /* the TextAttr pointer */
int16 spacing, /* How much vertical spacing between the lines */
byte pencolor, /* pen number to use for text */
byte backpen, /* the background color */
2014-10-06 14:50:05 +02:00
bool fillback, /* Whether to fill the background */
bool centerh, /* Whether to center the text horizontally */
bool centerv, /* Whether to center the text vertically */
bool output, /* Whether to output any text */
uint16 x1, /* Cords */
uint16 y1, uint16 x2, uint16 y2, const char *str) { /* The text itself */
uint32 res, vgabyte = _screenBytesPerPage;
byte *tmp = _vm->_currentDisplayBuffer;
2014-10-06 14:50:05 +02:00
_vm->_currentDisplayBuffer = destIm->_imageData;
_screenBytesPerPage = (uint32)destIm->_width * (int32)destIm->_height;
2014-10-06 14:50:05 +02:00
res = _vm->_graphics->flowText(font, spacing, pencolor, backpen, fillback, centerh, centerv, output, x1, y1, x2, y2, str);
2014-10-06 14:50:05 +02:00
_screenBytesPerPage = vgabyte;
_vm->_currentDisplayBuffer = tmp;
2014-10-06 14:50:05 +02:00
return res;
}
/*----- The control panel stuff -----*/
void DisplayMan::createBox(uint16 y2) {
setAPen(7); /* Message box area */
rectFill(VGAScaleX(4), VGAScaleY(154), VGAScaleX(315), VGAScaleY(y2 - 2));
2015-11-24 23:59:30 +01:00
setAPen(0); /* Box around message area */
drawHLine(VGAScaleX(2), VGAScaleY(152), VGAScaleX(317));
drawVLine(VGAScaleX(317), VGAScaleY(152), VGAScaleY(y2));
drawHLine(VGAScaleX(2), VGAScaleY(y2), VGAScaleX(317));
drawVLine(VGAScaleX(2), VGAScaleY(152), VGAScaleY(y2));
2014-10-06 14:50:05 +02:00
}
int32 DisplayMan::longDrawMessage(const char *str) {
2015-12-01 10:35:31 +01:00
char newText[512];
2014-10-06 14:50:05 +02:00
if (str == NULL)
return 0;
2014-10-06 14:50:05 +02:00
_vm->_event->attachGadgetList(NULL);
_vm->_event->mouseHide();
2015-12-01 10:35:31 +01:00
strcpy(newText, str);
2014-10-06 14:50:05 +02:00
2015-12-03 11:01:50 +01:00
if (!_longWinInFront) {
_longWinInFront = true;
2015-12-01 11:18:11 +01:00
setAPen(3); /* Clear Area */
rectFill(0, VGAScaleY(149) + SVGACord(2), VGAScaleX(319), VGAScaleY(199));
2014-10-06 14:50:05 +02:00
}
createBox(198);
_vm->_event->mouseShow();
return flowText(_vm->_msgFont, 0, 1, 7, false, true, true, true, VGAScaleX(6), VGAScaleY(155), VGAScaleX(313), VGAScaleY(195), str);
2014-10-06 14:50:05 +02:00
}
void LabEngine::drawStaticMessage(byte index) {
_graphics->drawMessage(_resource->getStaticText((StaticText)index).c_str());
}
2014-10-06 14:50:05 +02:00
/******************************************************************************/
/* Draws a message to the message box. */
/******************************************************************************/
void DisplayMan::drawMessage(const char *str) {
2014-10-06 14:50:05 +02:00
if (DoNotDrawMessage) {
DoNotDrawMessage = false;
return;
}
if (str) {
if ((textLength(_vm->_msgFont, str, strlen(str)) > VGAScaleX(306))) {
2014-10-06 14:50:05 +02:00
longDrawMessage(str);
2015-12-01 10:35:31 +01:00
_lastMessageLong = true;
2014-10-06 14:50:05 +02:00
} else {
2015-12-03 11:01:50 +01:00
if (_longWinInFront) {
_longWinInFront = false;
2015-12-01 11:18:11 +01:00
drawPanel();
2014-10-06 14:50:05 +02:00
}
_vm->_event->mouseHide();
2014-10-06 14:50:05 +02:00
createBox(168);
text(_vm->_msgFont, VGAScaleX(7), VGAScaleY(155) + SVGACord(2), 1, str, strlen(str));
_vm->_event->mouseShow();
2015-12-01 10:35:31 +01:00
_lastMessageLong = false;
2014-10-06 14:50:05 +02:00
}
}
}
/*---------------------------------------------------------------------------*/
/*--------------------------- All the wipe stuff. ---------------------------*/
/*---------------------------------------------------------------------------*/
#define TRANSWIPE 1
#define SCROLLWIPE 2
#define SCROLLBLACK 3
#define SCROLLBOUNCE 4
#define TRANSPORTER 5
#define READFIRSTFRAME 6
#define READNEXTFRAME 7
/*****************************************************************************/
/* Scrolls the display to black. */
/*****************************************************************************/
void DisplayMan::doScrollBlack() {
2014-10-06 14:50:05 +02:00
byte *mem, *tempmem;
2015-12-01 10:35:31 +01:00
Image im;
2014-10-06 14:50:05 +02:00
uint32 size, copysize;
2015-12-01 10:35:31 +01:00
uint32 *baseAddr;
2014-10-06 14:50:05 +02:00
_vm->_event->mouseHide();
2015-12-01 11:18:11 +01:00
uint16 width = VGAScaleX(320);
uint16 height = VGAScaleY(149) + SVGACord(2);
2014-10-06 14:50:05 +02:00
2015-12-01 10:35:31 +01:00
allocFile((void **)&mem, (int32)width * (int32)height, "Temp Mem");
2014-12-27 01:28:50 +01:00
2015-12-02 11:49:01 +01:00
im._width = width;
im._height = height;
im._imageData = mem;
_vm->_music->updateMusic();
im.readScreenImage(0, 0);
_vm->_music->updateMusic();
2014-10-06 14:50:05 +02:00
baseAddr = (uint32 *)_vm->getCurrentDrawingBuffer();
2014-10-06 14:50:05 +02:00
2015-12-01 11:18:11 +01:00
uint16 by = VGAScaleX(4);
uint16 nheight = height;
2014-10-06 14:50:05 +02:00
while (nheight) {
_vm->_music->updateMusic();
2014-10-06 14:50:05 +02:00
if (!_vm->_isHiRes)
_vm->waitTOF();
2014-10-06 14:50:05 +02:00
baseAddr = (uint32 *)_vm->getCurrentDrawingBuffer();
2014-10-06 14:50:05 +02:00
if (by > nheight)
by = nheight;
mem += by * width;
nheight -= by;
2015-12-01 10:35:31 +01:00
size = (int32)nheight * (int32)width;
2014-10-06 14:50:05 +02:00
tempmem = mem;
while (size) {
2015-12-01 11:18:11 +01:00
if (size > _screenBytesPerPage)
copysize = _screenBytesPerPage;
2014-10-06 14:50:05 +02:00
else
copysize = size;
size -= copysize;
2015-12-01 10:35:31 +01:00
memcpy(baseAddr, tempmem, copysize);
2014-10-06 14:50:05 +02:00
tempmem += copysize;
}
2015-12-01 11:18:11 +01:00
setAPen(0);
rectFill(0, nheight, width - 1, nheight + by - 1);
2014-10-06 14:50:05 +02:00
2015-12-01 11:18:11 +01:00
screenUpdate();
2014-10-06 14:50:05 +02:00
if (!_vm->_isHiRes) {
2014-10-06 14:50:05 +02:00
if (nheight <= (height / 8))
by = 1;
else if (nheight <= (height / 4))
by = 2;
else if (nheight <= (height / 2))
by = 3;
}
}
freeAllStolenMem();
_vm->_event->mouseShow();
2014-10-06 14:50:05 +02:00
}
void DisplayMan::copyPage(uint16 width, uint16 height, uint16 nheight, uint16 startline, byte *mem) {
2015-12-01 10:35:31 +01:00
uint32 size, offSet, copysize;
uint16 curPage;
uint32 *baseAddr;
2014-10-06 14:50:05 +02:00
baseAddr = (uint32 *)_vm->getCurrentDrawingBuffer();
2014-10-06 14:50:05 +02:00
2015-12-01 10:35:31 +01:00
size = (int32)(height - nheight) * (int32)width;
2014-10-06 14:50:05 +02:00
mem += startline * width;
curPage = ((int32)nheight * (int32)width) / _vm->_graphics->_screenBytesPerPage;
offSet = ((int32)nheight * (int32)width) - (curPage * _vm->_graphics->_screenBytesPerPage);
2014-10-06 14:50:05 +02:00
while (size) {
if (size > (_vm->_graphics->_screenBytesPerPage - offSet))
copysize = _vm->_graphics->_screenBytesPerPage - offSet;
2014-10-06 14:50:05 +02:00
else
copysize = size;
size -= copysize;
2015-12-01 10:35:31 +01:00
memcpy(baseAddr + (offSet >> 2), mem, copysize);
2014-10-06 14:50:05 +02:00
mem += copysize;
2015-12-01 10:35:31 +01:00
curPage++;
offSet = 0;
2014-10-06 14:50:05 +02:00
}
}
/*****************************************************************************/
/* Scrolls the display to a new picture from a black screen. */
/*****************************************************************************/
void DisplayMan::doScrollWipe(char *filename) {
2015-12-01 11:18:11 +01:00
uint16 startline = 0, onrow = 0;
2014-10-06 14:50:05 +02:00
_vm->_event->mouseHide();
2015-12-01 11:18:11 +01:00
uint16 width = VGAScaleX(320);
uint16 height = VGAScaleY(149) + SVGACord(2);
2014-10-06 14:50:05 +02:00
while (_vm->_music->isSoundEffectActive()) {
_vm->_music->updateMusic();
_vm->waitTOF();
2014-10-06 14:50:05 +02:00
}
_vm->_anim->_isBM = true;
2014-10-06 14:50:05 +02:00
readPict(filename, true);
_vm->setPalette(_vm->_anim->_diffPalette, 256);
_vm->_anim->_isBM = false;
byte *mem = _vm->_anim->_rawDiffBM._planes[0];
2014-10-06 14:50:05 +02:00
_vm->_music->updateMusic();
2015-12-01 11:18:11 +01:00
uint16 by = VGAScaleX(3);
uint16 nheight = height;
2014-10-06 14:50:05 +02:00
while (onrow < _vm->_anim->_headerdata._height) {
_vm->_music->updateMusic();
2014-10-06 14:50:05 +02:00
if ((by > nheight) && nheight)
by = nheight;
if ((startline + by) > (_vm->_anim->_headerdata._height - height - 1))
2014-10-06 14:50:05 +02:00
break;
if (nheight)
nheight -= by;
copyPage(width, height, nheight, startline, mem);
2015-12-01 11:18:11 +01:00
screenUpdate();
2014-10-06 14:50:05 +02:00
if (!nheight)
startline += by;
onrow += by;
if (nheight <= (height / 4))
by = VGAScaleX(5);
else if (nheight <= (height / 3))
by = VGAScaleX(4);
else if (nheight <= (height / 2))
by = VGAScaleX(3);
}
_vm->_event->mouseShow();
2014-10-06 14:50:05 +02:00
}
/*****************************************************************************/
/* Does the scroll bounce. Assumes bitmap already in memory. */
/*****************************************************************************/
void DisplayMan::doScrollBounce() {
2014-12-27 14:18:40 +01:00
const uint16 *newby, *newby1;
const uint16 newbyd[5] = {5, 4, 3, 2, 1}, newby1d[8] = {3, 3, 2, 2, 2, 1, 1, 1};
const uint16 newbyw[5] = {10, 8, 6, 4, 2}, newby1w[8] = {6, 6, 4, 4, 4, 2, 2, 2};
if (_vm->getPlatform() != Common::kPlatformWindows) {
2014-12-27 14:18:40 +01:00
newby = newbyd;
newby1 = newby1d;
} else {
newby = newbyw;
newby1 = newby1w;
}
_vm->_event->mouseHide();
2014-12-27 14:18:40 +01:00
int width = VGAScaleX(320);
int height = VGAScaleY(149) + SVGACord(2);
byte *mem = _vm->_anim->_rawDiffBM._planes[0];
2014-10-06 14:50:05 +02:00
_vm->_music->updateMusic();
int startline = _vm->_anim->_headerdata._height - height - 1;
2014-10-06 14:50:05 +02:00
for (int i = 0; i < 5; i++) {
_vm->_music->updateMusic();
startline -= newby[i];
2014-10-06 14:50:05 +02:00
copyPage(width, height, 0, startline, mem);
2015-12-01 11:18:11 +01:00
screenUpdate();
_vm->waitTOF();
2014-10-06 14:50:05 +02:00
}
for (int i = 8; i > 0; i--) {
_vm->_music->updateMusic();
startline += newby1[i - 1];
2014-10-06 14:50:05 +02:00
copyPage(width, height, 0, startline, mem);
2015-12-01 11:18:11 +01:00
screenUpdate();
_vm->waitTOF();
2014-10-06 14:50:05 +02:00
}
_vm->_event->mouseShow();
2014-10-06 14:50:05 +02:00
}
/*****************************************************************************/
/* Does the transporter wipe. */
/*****************************************************************************/
void DisplayMan::doTransWipe(CloseDataPtr *cPtr, char *filename) {
2015-12-01 10:35:31 +01:00
uint16 lastY, curY, linesdone = 0, lineslast;
Image imSource, imDest;
2014-10-06 14:50:05 +02:00
if (_vm->_isHiRes) {
2014-10-06 14:50:05 +02:00
lineslast = 3;
2015-12-01 10:35:31 +01:00
lastY = 358;
2014-10-06 14:50:05 +02:00
} else {
lineslast = 1;
2015-12-01 10:35:31 +01:00
lastY = 148;
2014-10-06 14:50:05 +02:00
}
for (uint16 i = 0; i < 2; i++) {
2015-12-01 10:35:31 +01:00
curY = i * 2;
2014-10-06 14:50:05 +02:00
2015-12-01 10:35:31 +01:00
while (curY < lastY) {
2014-10-06 14:50:05 +02:00
if (linesdone >= lineslast) {
_vm->_music->updateMusic();
_vm->waitTOF();
2014-10-06 14:50:05 +02:00
linesdone = 0;
}
_vm->overlayRect(0, 0, curY, _vm->_screenWidth - 1, curY + 1);
2015-12-01 10:35:31 +01:00
curY += 4;
2014-10-06 14:50:05 +02:00
linesdone++;
}
}
2015-12-01 11:18:11 +01:00
setAPen(0);
2014-10-06 14:50:05 +02:00
for (uint16 i = 0; i < 2; i++) {
2015-12-01 10:35:31 +01:00
curY = i * 2;
2014-10-06 14:50:05 +02:00
2015-12-01 10:35:31 +01:00
while (curY <= lastY) {
2014-10-06 14:50:05 +02:00
if (linesdone >= lineslast) {
_vm->_music->updateMusic();
_vm->waitTOF();
2014-10-06 14:50:05 +02:00
linesdone = 0;
}
rectFill(0, curY, _vm->_screenWidth - 1, curY + 1);
2015-12-01 10:35:31 +01:00
curY += 4;
2014-10-06 14:50:05 +02:00
linesdone++;
}
}
if (filename == NULL)
_vm->_curFileName = getPictName(cPtr);
2014-10-06 14:50:05 +02:00
else if (filename[0] > ' ')
_vm->_curFileName = filename;
2014-10-06 14:50:05 +02:00
else
_vm->_curFileName = getPictName(cPtr);
2014-10-06 14:50:05 +02:00
byte *BitMapMem = readPictToMem(_vm->_curFileName, _vm->_screenWidth, lastY + 5);
_vm->setPalette(_vm->_anim->_diffPalette, 256);
2014-10-06 14:50:05 +02:00
if (BitMapMem) {
imSource._width = _vm->_screenWidth;
2015-12-02 11:49:01 +01:00
imSource._height = lastY;
imSource._imageData = BitMapMem;
2014-10-06 14:50:05 +02:00
imDest._width = _vm->_screenWidth;
imDest._height = _vm->_screenHeight;
imDest._imageData = _vm->getCurrentDrawingBuffer();
2014-10-06 14:50:05 +02:00
for (uint16 i = 0; i < 2; i++) {
2015-12-01 10:35:31 +01:00
curY = i * 2;
2014-10-06 14:50:05 +02:00
2015-12-01 10:35:31 +01:00
while (curY < lastY) {
2014-10-06 14:50:05 +02:00
if (linesdone >= lineslast) {
_vm->_music->updateMusic();
_vm->waitTOF();
2014-10-06 14:50:05 +02:00
linesdone = 0;
}
imDest._imageData = _vm->getCurrentDrawingBuffer();
2014-10-06 14:50:05 +02:00
imSource.bltBitMap(0, curY, &imDest, 0, curY, _vm->_screenWidth, 2);
_vm->overlayRect(0, 0, curY, _vm->_screenWidth - 1, curY + 1);
2015-12-01 10:35:31 +01:00
curY += 4;
2014-10-06 14:50:05 +02:00
linesdone++;
}
}
for (uint16 i = 0; i < 2; i++) {
2015-12-01 10:35:31 +01:00
curY = i * 2;
2014-10-06 14:50:05 +02:00
2015-12-01 10:35:31 +01:00
while (curY <= lastY) {
2014-10-06 14:50:05 +02:00
if (linesdone >= lineslast) {
_vm->_music->updateMusic();
_vm->waitTOF();
2014-10-06 14:50:05 +02:00
linesdone = 0;
}
imDest._imageData = _vm->getCurrentDrawingBuffer();
2014-10-06 14:50:05 +02:00
2015-12-01 10:35:31 +01:00
if (curY == lastY)
imSource.bltBitMap(0, curY, &imDest, 0, curY, _vm->_screenWidth, 1);
2014-10-06 14:50:05 +02:00
else
imSource.bltBitMap(0, curY, &imDest, 0, curY, _vm->_screenWidth, 2);
2014-10-06 14:50:05 +02:00
2015-12-01 10:35:31 +01:00
curY += 4;
2014-10-06 14:50:05 +02:00
linesdone++;
}
}
}
}
/*****************************************************************************/
/* Does a certain number of pre-programmed wipes. */
/*****************************************************************************/
void DisplayMan::doWipe(uint16 wipeType, CloseDataPtr *cPtr, char *filename) {
2015-12-01 10:35:31 +01:00
if ((wipeType == TRANSWIPE) || (wipeType == TRANSPORTER))
doTransWipe(cPtr, filename);
else if (wipeType == SCROLLWIPE)
2014-10-06 14:50:05 +02:00
doScrollWipe(filename);
2015-12-01 10:35:31 +01:00
else if (wipeType == SCROLLBLACK)
2014-10-06 14:50:05 +02:00
doScrollBlack();
2015-12-01 10:35:31 +01:00
else if (wipeType == SCROLLBOUNCE)
2014-10-06 14:50:05 +02:00
doScrollBounce();
2015-12-01 10:35:31 +01:00
else if (wipeType == READFIRSTFRAME)
2014-10-06 14:50:05 +02:00
readPict(filename, false);
2015-12-01 10:35:31 +01:00
else if (wipeType == READNEXTFRAME)
_vm->_anim->diffNextFrame();
2015-12-01 20:10:42 +01:00
}
/*****************************************************************************/
/* Changes the front screen to black. */
/*****************************************************************************/
void DisplayMan::blackScreen() {
2015-12-03 14:22:04 +02:00
byte pal[256 * 3];
memset(pal, 0, 248 * 3);
_vm->writeColorRegs(pal, 8, 248);
2015-12-01 20:10:42 +01:00
g_system->delayMillis(32);
}
/*****************************************************************************/
/* Changes the front screen to white. */
/*****************************************************************************/
void DisplayMan::whiteScreen() {
2015-12-03 14:22:04 +02:00
byte pal[256 * 3];
memset(pal, 255, 248 * 3);
_vm->writeColorRegs(pal, 8, 248);
2015-12-01 20:10:42 +01:00
}
/*****************************************************************************/
/* Changes the entire screen to black. */
/*****************************************************************************/
void DisplayMan::blackAllScreen() {
2015-12-03 14:22:04 +02:00
byte pal[256 * 3];
memset(pal, 0, 256 * 3);
_vm->writeColorRegs(pal, 0, 256);
2015-12-01 20:10:42 +01:00
g_system->delayMillis(32);
2014-10-06 14:50:05 +02:00
}
/******************************************************************************/
/* Draws the control panel display. */
/******************************************************************************/
void DisplayMan::drawPanel() {
_vm->_event->mouseHide();
setAPen(3); /* Clear Area */
rectFill(0, VGAScaleY(149) + SVGACord(2), VGAScaleX(319), VGAScaleY(199));
setAPen(0); /* First Line */
drawHLine(0, VGAScaleY(149) + SVGACord(2), VGAScaleX(319));
setAPen(5); /* Second Line */
drawHLine(0, VGAScaleY(149) + 1 + SVGACord(2), VGAScaleX(319));
/* Gadget Separators */
setAPen(0);
drawHLine(0, VGAScaleY(170), VGAScaleX(319)); /* First black line to separate buttons */
if (!_vm->_alternate) {
setAPen(4);
drawHLine(0, VGAScaleY(170) + 1, VGAScaleX(319)); /* The horizontal lines under the black one */
drawGadgetList(_vm->_moveGadgetList);
} else {
if (_vm->getPlatform() != Common::kPlatformWindows) {
drawVLine(VGAScaleX(124), VGAScaleY(170) + 1, VGAScaleY(199)); /* Vertical Black lines */
drawVLine(VGAScaleX(194), VGAScaleY(170) + 1, VGAScaleY(199));
} else {
drawVLine(VGAScaleX(90), VGAScaleY(170) + 1, VGAScaleY(199)); /* Vertical Black lines */
drawVLine(VGAScaleX(160), VGAScaleY(170) + 1, VGAScaleY(199));
drawVLine(VGAScaleX(230), VGAScaleY(170) + 1, VGAScaleY(199));
}
setAPen(4);
drawHLine(0, VGAScaleY(170) + 1, VGAScaleX(122)); /* The horizontal lines under the black one */
drawHLine(VGAScaleX(126), VGAScaleY(170) + 1, VGAScaleX(192));
drawHLine(VGAScaleX(196), VGAScaleY(170) + 1, VGAScaleX(319));
drawVLine(VGAScaleX(1), VGAScaleY(170) + 2, VGAScaleY(198)); /* The vertical high light lines */
if (_vm->getPlatform() != Common::kPlatformWindows) {
drawVLine(VGAScaleX(126), VGAScaleY(170) + 2, VGAScaleY(198));
drawVLine(VGAScaleX(196), VGAScaleY(170) + 2, VGAScaleY(198));
} else {
drawVLine(VGAScaleX(92), VGAScaleY(170) + 2, VGAScaleY(198));
drawVLine(VGAScaleX(162), VGAScaleY(170) + 2, VGAScaleY(198));
drawVLine(VGAScaleX(232), VGAScaleY(170) + 2, VGAScaleY(198));
}
drawGadgetList(_vm->_invGadgetList);
}
_vm->_event->mouseShow();
}
/******************************************************************************/
/* Sets up the Labyrinth screens, and opens up the initial windows. */
/******************************************************************************/
bool DisplayMan::setUpScreens() {
if (!createScreen(_vm->_isHiRes))
return false;
Common::File *controlFile = _vm->_resource->openDataFile("P:Control");
for (uint16 i = 0; i < 20; i++)
_vm->_moveImages[i] = new Image(controlFile);
delete controlFile;
/* Creates the gadgets for the movement control panel */
uint16 y = VGAScaleY(173) - SVGACord(2);
if (_vm->getPlatform() == Common::kPlatformWindows) {
_vm->_moveGadgetList = createButton(1, y, 0, 't', _vm->_moveImages[0], _vm->_moveImages[1]);
Gadget *curGadget = _vm->_moveGadgetList;
curGadget->NextGadget = createButton(33, y, 1, 'm', _vm->_moveImages[2], _vm->_moveImages[3]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(65, y, 2, 'o', _vm->_moveImages[4], _vm->_moveImages[5]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(97, y, 3, 'c', _vm->_moveImages[6], _vm->_moveImages[7]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(129, y, 4, 'l', _vm->_moveImages[8], _vm->_moveImages[9]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(161, y, 5, 'i', _vm->_moveImages[12], _vm->_moveImages[13]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(193, y, 6, VKEY_LTARROW, _vm->_moveImages[14], _vm->_moveImages[15]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(225, y, 7, VKEY_UPARROW, _vm->_moveImages[16], _vm->_moveImages[17]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(257, y, 8, VKEY_RTARROW, _vm->_moveImages[18], _vm->_moveImages[19]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(289, y, 9, 'p', _vm->_moveImages[10], _vm->_moveImages[11]);
} else {
_vm->_moveGadgetList = createButton(1, y, 0, 0, _vm->_moveImages[0], _vm->_moveImages[1]);
Gadget *curGadget = _vm->_moveGadgetList;
curGadget->NextGadget = createButton(33, y, 1, 0, _vm->_moveImages[2], _vm->_moveImages[3]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(65, y, 2, 0, _vm->_moveImages[4], _vm->_moveImages[5]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(97, y, 3, 0, _vm->_moveImages[6], _vm->_moveImages[7]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(129, y, 4, 0, _vm->_moveImages[8], _vm->_moveImages[9]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(161, y, 5, 0, _vm->_moveImages[12], _vm->_moveImages[13]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(193, y, 6, 0, _vm->_moveImages[14], _vm->_moveImages[15]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(225, y, 7, 0, _vm->_moveImages[16], _vm->_moveImages[17]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(257, y, 8, 0, _vm->_moveImages[18], _vm->_moveImages[19]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(289, y, 9, 0, _vm->_moveImages[10], _vm->_moveImages[11]);
}
Common::File *invFile = _vm->_resource->openDataFile("P:Inv");
if (_vm->getPlatform() == Common::kPlatformWindows) {
for (uint16 imgIdx = 0; imgIdx < 10; imgIdx++)
_vm->_invImages[imgIdx] = new Image(invFile);
_vm->_invGadgetList = createButton(24, y, 0, 'm', _vm->_invImages[0], _vm->_invImages[1]);
Gadget *curGadget = _vm->_invGadgetList;
curGadget->NextGadget = createButton(56, y, 1, 'g', _vm->_invImages[2], _vm->_invImages[3]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(94, y, 2, 'u', _vm->_invImages[4], _vm->_invImages[5]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(126, y, 3, 'l', _vm->_moveImages[8], _vm->_moveImages[9]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(164, y, 4, VKEY_LTARROW, _vm->_moveImages[14], _vm->_moveImages[15]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(196, y, 5, VKEY_RTARROW, _vm->_moveImages[18], _vm->_moveImages[19]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(234, y, 6, 'b', _vm->_invImages[6], _vm->_invImages[7]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(266, y, 7, 'f', _vm->_invImages[8], _vm->_invImages[9]);
curGadget = curGadget->NextGadget;
} else {
for (uint16 imgIdx = 0; imgIdx < 6; imgIdx++)
_vm->_invImages[imgIdx] = new Image(invFile);
_vm->_invGadgetList = createButton(58, y, 0, 0, _vm->_invImages[0], _vm->_invImages[1]);
Gadget *curGadget = _vm->_invGadgetList;
curGadget->NextGadget = createButton(90, y, 1, 0, _vm->_invImages[2], _vm->_invImages[3]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(128, y, 2, 0, _vm->_invImages[4], _vm->_invImages[5]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(160, y, 3, 0, _vm->_moveImages[8], _vm->_moveImages[9]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(198, y, 4, 0, _vm->_moveImages[14], _vm->_moveImages[15]);
curGadget = curGadget->NextGadget;
curGadget->NextGadget = createButton(230, y, 5, 0, _vm->_moveImages[18], _vm->_moveImages[19]);
curGadget = curGadget->NextGadget;
}
delete invFile;
return true;
}
/*****************************************************************************/
/* Sets the pen number to use on all the drawing operations. */
/*****************************************************************************/
void DisplayMan::setAPen(byte pennum) {
_curapen = pennum;
}
/*****************************************************************************/
/* Fills in a rectangle. */
/*****************************************************************************/
void DisplayMan::rectFill(uint16 x1, uint16 y1, uint16 x2, uint16 y2) {
int w = x2 - x1 + 1;
int h = y2 - y1 + 1;
if (x1 + w > _vm->_screenWidth)
w = _vm->_screenWidth - x1;
if (y1 + h > _vm->_screenHeight)
h = _vm->_screenHeight - y1;
if ((w > 0) && (h > 0)) {
char *d = (char *)_vm->getCurrentDrawingBuffer() + y1 * _vm->_screenWidth + x1;
while (h-- > 0) {
char *dd = d;
int ww = w;
while (ww-- > 0) {
*dd++ = _curapen;
}
d += _vm->_screenWidth;
}
}
}
/*****************************************************************************/
/* Draws a horizontal line. */
/*****************************************************************************/
void DisplayMan::drawVLine(uint16 x, uint16 y1, uint16 y2) {
rectFill(x, y1, x, y2);
}
/*****************************************************************************/
/* Draws a vertical line. */
/*****************************************************************************/
void DisplayMan::drawHLine(uint16 x1, uint16 y, uint16 x2) {
rectFill(x1, y, x2, y);
}
void DisplayMan::screenUpdate() {
g_system->copyRectToScreen(_vm->_displayBuffer, _vm->_screenWidth, 0, 0, _vm->_screenWidth, _vm->_screenHeight);
g_system->updateScreen();
_vm->_event->processInput();
}
/*****************************************************************************/
/* Sets up either a low-res or a high-res 256 color screen. */
/*****************************************************************************/
bool DisplayMan::createScreen(bool hiRes) {
if (hiRes) {
_vm->_screenWidth = 640;
_vm->_screenHeight = 480;
} else {
_vm->_screenWidth = 320;
_vm->_screenHeight = 200;
}
_screenBytesPerPage = _vm->_screenWidth * _vm->_screenHeight;
_vm->_displayBuffer = new byte[_screenBytesPerPage]; // FIXME: Memory leak!
return true;
}
2014-10-06 14:50:05 +02:00
} // End of namespace Lab