2008-07-23 09:02:47 +00: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.
|
2014-02-18 02:34:25 +01:00
|
|
|
*
|
2008-07-23 09:02:47 +00:00
|
|
|
* 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.
|
2014-02-18 02:34:25 +01:00
|
|
|
*
|
2008-07-23 09:02:47 +00:00
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* Background handling code.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "tinsel/background.h"
|
|
|
|
#include "tinsel/cliprect.h" // object clip rect defs
|
|
|
|
#include "tinsel/graphics.h"
|
|
|
|
#include "tinsel/sched.h" // process sheduler defs
|
|
|
|
#include "tinsel/object.h"
|
|
|
|
#include "tinsel/pid.h" // process identifiers
|
|
|
|
#include "tinsel/tinsel.h"
|
|
|
|
|
|
|
|
namespace Tinsel {
|
|
|
|
|
2010-11-16 09:53:55 +00:00
|
|
|
// FIXME: Avoid non-const global vars
|
|
|
|
|
2008-07-23 09:02:47 +00:00
|
|
|
// current background
|
2012-02-22 23:14:29 +01:00
|
|
|
const BACKGND *g_pCurBgnd = NULL;
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
/**
|
2011-05-25 11:17:11 -04:00
|
|
|
* Called to initialize a background.
|
2008-07-23 09:02:47 +00:00
|
|
|
* @param pBgnd Pointer to data struct for current background
|
|
|
|
*/
|
|
|
|
|
2010-11-16 09:53:55 +00:00
|
|
|
void InitBackground(const BACKGND *pBgnd) {
|
2008-07-23 09:02:47 +00:00
|
|
|
int i; // playfield counter
|
2008-07-23 10:27:24 +00:00
|
|
|
PLAYFIELD *pPlayfield; // pointer to current playfield
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// set current background
|
2012-02-22 23:14:29 +01:00
|
|
|
g_pCurBgnd = pBgnd;
|
2008-07-23 09:02:47 +00:00
|
|
|
|
2011-04-14 14:12:27 +02:00
|
|
|
// init background sky color
|
|
|
|
SetBgndColor(pBgnd->rgbSkyColor);
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// start of playfield array
|
|
|
|
pPlayfield = pBgnd->fieldArray;
|
|
|
|
|
|
|
|
// for each background playfield
|
|
|
|
for (i = 0; i < pBgnd->numPlayfields; i++, pPlayfield++) {
|
|
|
|
// init playfield pos
|
|
|
|
pPlayfield->fieldX = intToFrac(pBgnd->ptInitWorld.x);
|
|
|
|
pPlayfield->fieldY = intToFrac(pBgnd->ptInitWorld.y);
|
|
|
|
|
|
|
|
// no scrolling
|
|
|
|
pPlayfield->fieldXvel = intToFrac(0);
|
|
|
|
pPlayfield->fieldYvel = intToFrac(0);
|
|
|
|
|
|
|
|
// clear playfield display list
|
|
|
|
pPlayfield->pDispList = NULL;
|
|
|
|
|
|
|
|
// clear playfield moved flag
|
|
|
|
pPlayfield->bMoved = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the xy position of the specified playfield in the current background.
|
|
|
|
* @param which Which playfield
|
|
|
|
* @param newXpos New x position
|
|
|
|
* @param newYpos New y position
|
|
|
|
*/
|
|
|
|
|
|
|
|
void PlayfieldSetPos(int which, int newXpos, int newYpos) {
|
2008-07-23 10:27:24 +00:00
|
|
|
PLAYFIELD *pPlayfield; // pointer to relavent playfield
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// make sure there is a background
|
2012-02-22 23:14:29 +01:00
|
|
|
assert(g_pCurBgnd != NULL);
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// make sure the playfield number is in range
|
2012-02-22 23:14:29 +01:00
|
|
|
assert(which >= 0 && which < g_pCurBgnd->numPlayfields);
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// get playfield pointer
|
2012-02-22 23:14:29 +01:00
|
|
|
pPlayfield = g_pCurBgnd->fieldArray + which;
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// set new integer position
|
|
|
|
pPlayfield->fieldX = intToFrac(newXpos);
|
|
|
|
pPlayfield->fieldY = intToFrac(newYpos);
|
|
|
|
|
|
|
|
// set moved flag
|
|
|
|
pPlayfield->bMoved = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the xy position of the specified playfield in the current background.
|
|
|
|
* @param which Which playfield
|
|
|
|
* @param pXpos Returns current x position
|
|
|
|
* @param pYpos Returns current y position
|
|
|
|
*/
|
|
|
|
|
|
|
|
void PlayfieldGetPos(int which, int *pXpos, int *pYpos) {
|
2008-07-23 10:27:24 +00:00
|
|
|
PLAYFIELD *pPlayfield; // pointer to relavent playfield
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// make sure there is a background
|
2012-02-22 23:14:29 +01:00
|
|
|
assert(g_pCurBgnd != NULL);
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// make sure the playfield number is in range
|
2012-02-22 23:14:29 +01:00
|
|
|
assert(which >= 0 && which < g_pCurBgnd->numPlayfields);
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// get playfield pointer
|
2012-02-22 23:14:29 +01:00
|
|
|
pPlayfield = g_pCurBgnd->fieldArray + which;
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// get current integer position
|
|
|
|
*pXpos = fracToInt(pPlayfield->fieldX);
|
|
|
|
*pYpos = fracToInt(pPlayfield->fieldY);
|
|
|
|
}
|
|
|
|
|
2008-12-01 20:35:36 +00:00
|
|
|
/**
|
2011-04-14 14:34:28 +02:00
|
|
|
* Returns the x position of the center of the specified playfield
|
2008-12-01 20:35:36 +00:00
|
|
|
* @param which Which playfield
|
|
|
|
*/
|
|
|
|
|
2011-04-14 14:34:28 +02:00
|
|
|
int PlayfieldGetCenterX(int which) {
|
2008-12-01 20:35:36 +00:00
|
|
|
PLAYFIELD *pPlayfield; // pointer to relavent playfield
|
|
|
|
|
|
|
|
// make sure there is a background
|
2012-02-22 23:14:29 +01:00
|
|
|
assert(g_pCurBgnd != NULL);
|
2008-12-01 20:35:36 +00:00
|
|
|
|
|
|
|
// make sure the playfield number is in range
|
2012-02-22 23:14:29 +01:00
|
|
|
assert(which >= 0 && which < g_pCurBgnd->numPlayfields);
|
2008-12-01 20:35:36 +00:00
|
|
|
|
|
|
|
// get playfield pointer
|
2012-02-22 23:14:29 +01:00
|
|
|
pPlayfield = g_pCurBgnd->fieldArray + which;
|
2008-12-01 20:35:36 +00:00
|
|
|
|
|
|
|
// get current integer position
|
|
|
|
return fracToInt(pPlayfield->fieldX) + SCREEN_WIDTH/2;
|
|
|
|
}
|
|
|
|
|
2008-07-23 09:02:47 +00:00
|
|
|
/**
|
|
|
|
* Returns the display list for the specified playfield.
|
|
|
|
* @param which Which playfield
|
|
|
|
*/
|
|
|
|
|
2011-05-18 14:40:49 +02:00
|
|
|
OBJECT **GetPlayfieldList(int which) {
|
2008-07-23 10:27:24 +00:00
|
|
|
PLAYFIELD *pPlayfield; // pointer to relavent playfield
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// make sure there is a background
|
2012-02-22 23:14:29 +01:00
|
|
|
assert(g_pCurBgnd != NULL);
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// make sure the playfield number is in range
|
2012-02-22 23:14:29 +01:00
|
|
|
assert(which >= 0 && which < g_pCurBgnd->numPlayfields);
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// get playfield pointer
|
2012-02-22 23:14:29 +01:00
|
|
|
pPlayfield = g_pCurBgnd->fieldArray + which;
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// return the display list pointer for this playfield
|
2011-05-18 14:40:49 +02:00
|
|
|
return &pPlayfield->pDispList;
|
2008-07-23 09:02:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Draws all the playfield object lists for the current background.
|
|
|
|
* The playfield velocity is added to the playfield position in order
|
|
|
|
* to scroll each playfield before it is drawn.
|
|
|
|
*/
|
|
|
|
|
2009-11-02 21:54:57 +00:00
|
|
|
void DrawBackgnd() {
|
2008-07-23 09:02:47 +00:00
|
|
|
int i; // playfield counter
|
2008-07-23 10:27:24 +00:00
|
|
|
PLAYFIELD *pPlay; // playfield pointer
|
2008-07-23 09:02:47 +00:00
|
|
|
int prevX, prevY; // save interger part of position
|
|
|
|
Common::Point ptWin; // window top left
|
|
|
|
|
2012-02-22 23:14:29 +01:00
|
|
|
if (g_pCurBgnd == NULL)
|
2008-07-23 09:02:47 +00:00
|
|
|
return; // no current background
|
|
|
|
|
|
|
|
// scroll each background playfield
|
2012-02-22 23:14:29 +01:00
|
|
|
for (i = 0; i < g_pCurBgnd->numPlayfields; i++) {
|
2008-07-23 09:02:47 +00:00
|
|
|
// get pointer to correct playfield
|
2012-02-22 23:14:29 +01:00
|
|
|
pPlay = g_pCurBgnd->fieldArray + i;
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// save integer part of position
|
|
|
|
prevX = fracToInt(pPlay->fieldX);
|
|
|
|
prevY = fracToInt(pPlay->fieldY);
|
|
|
|
|
2008-07-23 10:33:36 +00:00
|
|
|
// update scrolling
|
|
|
|
pPlay->fieldX += pPlay->fieldXvel;
|
|
|
|
pPlay->fieldY += pPlay->fieldYvel;
|
2008-07-23 09:02:47 +00:00
|
|
|
|
2008-07-23 10:33:36 +00:00
|
|
|
// convert fixed point window pos to a int
|
|
|
|
ptWin.x = fracToInt(pPlay->fieldX);
|
|
|
|
ptWin.y = fracToInt(pPlay->fieldY);
|
2008-07-23 09:02:47 +00:00
|
|
|
|
2008-07-23 10:33:36 +00:00
|
|
|
// set the moved flag if the playfield has moved
|
|
|
|
if (prevX != ptWin.x || prevY != ptWin.y)
|
|
|
|
pPlay->bMoved = true;
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// sort the display list for this background - just in case somebody has changed object Z positions
|
2011-05-18 14:40:49 +02:00
|
|
|
SortObjectList(&pPlay->pDispList);
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// generate clipping rects for all objects that have moved etc.
|
2011-05-18 14:40:49 +02:00
|
|
|
FindMovingObjects(&pPlay->pDispList, &ptWin,
|
2008-07-23 10:33:36 +00:00
|
|
|
&pPlay->rcClip, false, pPlay->bMoved);
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// clear playfield moved flag
|
|
|
|
pPlay->bMoved = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// merge the clipping rectangles
|
|
|
|
MergeClipRect();
|
|
|
|
|
|
|
|
// redraw all playfields within the clipping rectangles
|
|
|
|
const RectList &clipRects = GetClipRects();
|
|
|
|
for (RectList::const_iterator r = clipRects.begin(); r != clipRects.end(); ++r) {
|
|
|
|
// clear the clip rectangle on the virtual screen
|
|
|
|
// for each background playfield
|
2012-02-22 23:14:29 +01:00
|
|
|
for (i = 0; i < g_pCurBgnd->numPlayfields; i++) {
|
2008-07-23 09:02:47 +00:00
|
|
|
Common::Rect rcPlayClip; // clip rect for this playfield
|
|
|
|
|
|
|
|
// get pointer to correct playfield
|
2012-02-22 23:14:29 +01:00
|
|
|
pPlay = g_pCurBgnd->fieldArray + i;
|
2008-07-23 09:02:47 +00:00
|
|
|
|
|
|
|
// convert fixed point window pos to a int
|
|
|
|
ptWin.x = fracToInt(pPlay->fieldX);
|
|
|
|
ptWin.y = fracToInt(pPlay->fieldY);
|
|
|
|
|
|
|
|
if (IntersectRectangle(rcPlayClip, pPlay->rcClip, *r))
|
|
|
|
// redraw all objects within this clipping rect
|
2011-05-18 14:40:49 +02:00
|
|
|
UpdateClipRect(&pPlay->pDispList, &ptWin, &rcPlayClip);
|
2008-07-23 09:02:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// transfer any new palettes to the video DAC
|
|
|
|
PalettesToVideoDAC();
|
|
|
|
|
|
|
|
// update the screen within the clipping rectangles
|
|
|
|
for (RectList::const_iterator r = clipRects.begin(); r != clipRects.end(); ++r) {
|
|
|
|
UpdateScreenRect(*r);
|
|
|
|
}
|
|
|
|
|
2009-08-08 09:34:23 +00:00
|
|
|
g_system->updateScreen();
|
|
|
|
|
2008-07-23 09:02:47 +00:00
|
|
|
// delete all the clipping rectangles
|
|
|
|
ResetClipRect();
|
|
|
|
}
|
|
|
|
|
2009-10-04 21:26:33 +00:00
|
|
|
} // End of namespace Tinsel
|