mirror of
https://github.com/libretro/scummvm.git
synced 2024-12-14 05:38:56 +00:00
71c170bb13
svn-id: r20813
491 lines
8.7 KiB
C++
491 lines
8.7 KiB
C++
/* ScummVM - Scumm Interpreter
|
|
* Copyright (C) 2006 The ScummVM project
|
|
*
|
|
* cinE Engine is (C) 2004-2005 by CinE Team
|
|
*
|
|
* 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.
|
|
*
|
|
* $URL$
|
|
* $Id$
|
|
*
|
|
*/
|
|
|
|
#include "cine/cine.h"
|
|
|
|
objectStruct objectTable[NUM_MAX_OBJECT];
|
|
u16 globalVars[NUM_MAX_OBJECTDATA];
|
|
overlayHeadElement overlayHead;
|
|
|
|
void unloadAllMasks(void) {
|
|
overlayHeadElement *current = overlayHead.next;
|
|
|
|
while (current) {
|
|
overlayHeadElement *next = current->next;
|
|
|
|
free(current);
|
|
|
|
current = next;
|
|
}
|
|
|
|
resetMessageHead();
|
|
}
|
|
|
|
void resetMessageHead(void) {
|
|
overlayHead.next = NULL;
|
|
overlayHead.previous = NULL;
|
|
}
|
|
|
|
void loadObject(char *pObjectName) {
|
|
u16 numEntry;
|
|
u16 entrySize;
|
|
u16 i;
|
|
u8 *ptr;
|
|
|
|
checkDataDisk(-1);
|
|
|
|
ptr = readBundleFile(findFileInBundle(pObjectName));
|
|
|
|
processPendingUpdates(1);
|
|
|
|
numEntry = *(u16 *) ptr;
|
|
ptr += 2;
|
|
flipU16(&numEntry);
|
|
|
|
entrySize = *(u16 *) ptr;
|
|
ptr += 2;
|
|
flipU16(&entrySize);
|
|
|
|
ASSERT(numEntry <= NUM_MAX_OBJECT);
|
|
ASSERT(entrySize == sizeof(objectStruct)); // carefull, it's directly read to memory
|
|
|
|
for (i = 0; i < numEntry; i++) {
|
|
if (objectTable[i].costume != -2) // flag is keep ?
|
|
{
|
|
memcpy(&objectTable[i], ptr, entrySize);
|
|
|
|
flipU16((u16 *) & objectTable[i].x);
|
|
flipU16((u16 *) & objectTable[i].y);
|
|
flipU16(&objectTable[i].mask);
|
|
flipU16((u16 *) & objectTable[i].frame);
|
|
flipU16((u16 *) & objectTable[i].costume);
|
|
flipU16(&objectTable[i].part);
|
|
}
|
|
|
|
ptr += entrySize;
|
|
}
|
|
|
|
if (!strcmp(pObjectName, "INTRO.OBJ")) {
|
|
for (i = 0; i < 10; i++) {
|
|
objectTable[i].costume = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
s8 removeOverlayElement(u16 objIdx, u16 param) {
|
|
overlayHeadElement *currentHeadPtr = &overlayHead;
|
|
overlayHeadElement *tempHead = currentHeadPtr;
|
|
overlayHeadElement *tempPtr2;
|
|
|
|
currentHeadPtr = tempHead->next;
|
|
|
|
while (currentHeadPtr && (objIdx == currentHeadPtr->objIdx
|
|
|| param == currentHeadPtr->type)) {
|
|
tempHead = currentHeadPtr;
|
|
|
|
currentHeadPtr = tempHead->next;
|
|
}
|
|
|
|
if (!currentHeadPtr) {
|
|
return -1;
|
|
}
|
|
|
|
if (objIdx != currentHeadPtr->objIdx || param != currentHeadPtr->type) {
|
|
return -1;
|
|
}
|
|
|
|
tempHead->next = currentHeadPtr->next;
|
|
tempPtr2 = currentHeadPtr->next;
|
|
|
|
if (!tempPtr2) {
|
|
tempPtr2 = &overlayHead;
|
|
}
|
|
|
|
tempPtr2->previous = currentHeadPtr->previous;
|
|
|
|
free(currentHeadPtr);
|
|
|
|
return (0);
|
|
}
|
|
|
|
s16 freeOverlay(u16 objIdx, u16 param) {
|
|
overlayHeadElement *currentHeadPtr = overlayHead.next;
|
|
overlayHeadElement *tempHead = &overlayHead;
|
|
overlayHeadElement *tempPtr2;
|
|
|
|
while (currentHeadPtr && ((currentHeadPtr->objIdx != objIdx)
|
|
|| (currentHeadPtr->type != param))) {
|
|
tempHead = currentHeadPtr;
|
|
currentHeadPtr = tempHead->next;
|
|
}
|
|
|
|
if (!currentHeadPtr) {
|
|
return -1;
|
|
}
|
|
|
|
if (!((currentHeadPtr->objIdx == objIdx)
|
|
&& (currentHeadPtr->type == param))) {
|
|
return -1;
|
|
}
|
|
|
|
tempHead->next = currentHeadPtr->next;
|
|
tempPtr2 = currentHeadPtr->next;
|
|
|
|
if (!tempPtr2) {
|
|
tempPtr2 = &overlayHead;
|
|
}
|
|
|
|
tempPtr2->previous = currentHeadPtr->previous;
|
|
|
|
//TODO: fix !
|
|
//free(currentHeadPtr);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void loadOverlayElement(u16 objIdx, u16 param) {
|
|
overlayHeadElement *currentHeadPtr = &overlayHead;
|
|
overlayHeadElement *pNewElement;
|
|
|
|
u16 si = objectTable[objIdx].mask;
|
|
|
|
overlayHeadElement *tempHead = currentHeadPtr;
|
|
|
|
currentHeadPtr = tempHead->next;
|
|
|
|
while (currentHeadPtr
|
|
&& (objectTable[currentHeadPtr->objIdx].mask < si)) {
|
|
tempHead = currentHeadPtr;
|
|
|
|
currentHeadPtr = tempHead->next;
|
|
}
|
|
|
|
pNewElement =
|
|
(overlayHeadElement *) malloc(sizeof(overlayHeadElement));
|
|
|
|
ASSERT_PTR(pNewElement);
|
|
|
|
pNewElement->next = tempHead->next;
|
|
tempHead->next = pNewElement;
|
|
|
|
pNewElement->objIdx = objIdx;
|
|
pNewElement->type = param;
|
|
|
|
if (!currentHeadPtr) {
|
|
currentHeadPtr = &overlayHead;
|
|
}
|
|
|
|
pNewElement->previous = currentHeadPtr->previous;
|
|
currentHeadPtr->previous = pNewElement;
|
|
}
|
|
|
|
void setupObject(u8 objIdx, u16 param1, u16 param2, u16 param3, u16 param4) {
|
|
objectTable[objIdx].x = param1;
|
|
objectTable[objIdx].y = param2;
|
|
objectTable[objIdx].mask = param3;
|
|
objectTable[objIdx].frame = param4;
|
|
|
|
if (!removeOverlayElement(objIdx, 0)) {
|
|
loadOverlayElement(objIdx, 0);
|
|
}
|
|
}
|
|
|
|
void subObjectParam(u8 objIdx, u8 paramIdx, s16 newValue) {
|
|
ASSERT(objIdx <= NUM_MAX_OBJECT);
|
|
|
|
paramIdx--;
|
|
|
|
ASSERT(paramIdx >= 0 && paramIdx <= 5);
|
|
|
|
switch (paramIdx) {
|
|
case 0:
|
|
{
|
|
objectTable[objIdx].x -= newValue;
|
|
break;
|
|
}
|
|
case 1:
|
|
{
|
|
objectTable[objIdx].y -= newValue;
|
|
break;
|
|
}
|
|
case 2:
|
|
{
|
|
objectTable[objIdx].mask -= newValue;
|
|
|
|
if (!removeOverlayElement(objIdx, 0)) {
|
|
loadOverlayElement(objIdx, 0);
|
|
}
|
|
break;
|
|
}
|
|
case 3:
|
|
{
|
|
objectTable[objIdx].frame -= newValue;
|
|
break;
|
|
}
|
|
case 4:
|
|
{
|
|
objectTable[objIdx].costume -= newValue;
|
|
break;
|
|
}
|
|
case 5:
|
|
{
|
|
objectTable[objIdx].part -= newValue;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void addObjectParam(u8 objIdx, u8 paramIdx, s16 newValue) {
|
|
ASSERT(objIdx <= NUM_MAX_OBJECT);
|
|
|
|
paramIdx--;
|
|
|
|
ASSERT(paramIdx >= 0 && paramIdx <= 5);
|
|
|
|
switch (paramIdx) {
|
|
case 0:
|
|
{
|
|
objectTable[objIdx].x += newValue;
|
|
break;
|
|
}
|
|
case 1:
|
|
{
|
|
objectTable[objIdx].y += newValue;
|
|
break;
|
|
}
|
|
case 2:
|
|
{
|
|
objectTable[objIdx].mask += newValue;
|
|
|
|
if (!removeOverlayElement(objIdx, 0)) {
|
|
loadOverlayElement(objIdx, 0);
|
|
}
|
|
break;
|
|
}
|
|
case 3:
|
|
{
|
|
objectTable[objIdx].frame += newValue;
|
|
break;
|
|
}
|
|
case 4:
|
|
{
|
|
objectTable[objIdx].costume += newValue;
|
|
break;
|
|
}
|
|
case 5:
|
|
{
|
|
objectTable[objIdx].part += newValue;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void modifyObjectParam(u8 objIdx, u8 paramIdx, s16 newValue) {
|
|
ASSERT(objIdx <= NUM_MAX_OBJECT);
|
|
|
|
paramIdx--;
|
|
|
|
ASSERT(paramIdx >= 0 && paramIdx <= 5);
|
|
|
|
switch (paramIdx) {
|
|
case 0:
|
|
{
|
|
objectTable[objIdx].x = newValue;
|
|
break;
|
|
}
|
|
case 1:
|
|
{
|
|
objectTable[objIdx].y = newValue;
|
|
break;
|
|
}
|
|
case 2:
|
|
{
|
|
objectTable[objIdx].mask = newValue;
|
|
|
|
if (!removeOverlayElement(objIdx, 0)) {
|
|
loadOverlayElement(objIdx, 0);
|
|
}
|
|
break;
|
|
}
|
|
case 3:
|
|
{
|
|
objectTable[objIdx].frame = newValue;
|
|
break;
|
|
}
|
|
case 4:
|
|
{
|
|
if (newValue == -1) {
|
|
objectTable[objIdx].costume = globalVars[0];
|
|
} else {
|
|
objectTable[objIdx].costume = newValue;
|
|
}
|
|
break;
|
|
}
|
|
case 5:
|
|
{
|
|
objectTable[objIdx].part = newValue;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
u8 compareObjectParam(u8 objIdx, u8 param1, s16 param2) {
|
|
u8 compareResult = 0;
|
|
|
|
switch (param1 - 1) {
|
|
case 0:
|
|
{
|
|
if (objectTable[objIdx].x == param2) {
|
|
compareResult |= 1;
|
|
}
|
|
|
|
if (objectTable[objIdx].x > param2) {
|
|
compareResult |= 2;
|
|
}
|
|
|
|
if (objectTable[objIdx].x < param2) {
|
|
compareResult |= 4;
|
|
}
|
|
|
|
break;
|
|
}
|
|
case 1:
|
|
{
|
|
if (objectTable[objIdx].y == param2) {
|
|
compareResult |= 1;
|
|
}
|
|
|
|
if (objectTable[objIdx].y > param2) {
|
|
compareResult |= 2;
|
|
}
|
|
|
|
if (objectTable[objIdx].y < param2) {
|
|
compareResult |= 4;
|
|
}
|
|
|
|
break;
|
|
}
|
|
case 2:
|
|
{
|
|
if (objectTable[objIdx].mask == param2) {
|
|
compareResult |= 1;
|
|
}
|
|
|
|
if (objectTable[objIdx].mask > param2) {
|
|
compareResult |= 2;
|
|
}
|
|
|
|
if (objectTable[objIdx].mask < param2) {
|
|
compareResult |= 4;
|
|
}
|
|
|
|
break;
|
|
}
|
|
case 3:
|
|
{
|
|
if (objectTable[objIdx].frame == param2) {
|
|
compareResult |= 1;
|
|
}
|
|
|
|
if (objectTable[objIdx].frame > param2) {
|
|
compareResult |= 2;
|
|
}
|
|
|
|
if (objectTable[objIdx].frame < param2) {
|
|
compareResult |= 4;
|
|
}
|
|
|
|
break;
|
|
}
|
|
case 4:
|
|
{
|
|
if (objectTable[objIdx].costume == param2) {
|
|
compareResult |= 1;
|
|
}
|
|
|
|
if (objectTable[objIdx].costume > param2) {
|
|
compareResult |= 2;
|
|
}
|
|
|
|
if (objectTable[objIdx].costume < param2) {
|
|
compareResult |= 4;
|
|
}
|
|
|
|
break;
|
|
}
|
|
default:
|
|
{
|
|
printf
|
|
("Unsupported compare type: %d in compareObjectParam\n",
|
|
param1 - 1);
|
|
exit(1);
|
|
}
|
|
}
|
|
|
|
return (compareResult);
|
|
}
|
|
|
|
s16 getObjectParam(u16 objIdx, u16 paramIdx) {
|
|
ASSERT(objIdx <= NUM_MAX_OBJECT);
|
|
|
|
paramIdx--;
|
|
|
|
ASSERT(paramIdx >= 0 && paramIdx <= 5);
|
|
|
|
switch (paramIdx) {
|
|
case 0:
|
|
{
|
|
return objectTable[objIdx].x;
|
|
break;
|
|
}
|
|
case 1:
|
|
{
|
|
return objectTable[objIdx].y;
|
|
break;
|
|
}
|
|
case 2:
|
|
{
|
|
return objectTable[objIdx].mask;
|
|
break;
|
|
}
|
|
case 3:
|
|
{
|
|
return objectTable[objIdx].frame;
|
|
break;
|
|
}
|
|
case 4:
|
|
{
|
|
return objectTable[objIdx].costume;
|
|
break;
|
|
}
|
|
case 5:
|
|
{
|
|
return objectTable[objIdx].part;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|