scummvm/sword2/function.cpp

479 lines
11 KiB
C++
Raw Normal View History

2003-07-28 01:44:38 +00:00
/* Copyright (C) 1994-2003 Revolution Software Ltd
*
* 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
* $Header$
*/
2003-07-28 03:12:49 +00:00
#include "stdafx.h"
#include "driver/driver96.h"
#include "driver/rdwin.h"
2003-07-28 01:44:38 +00:00
#include "build_display.h"
#include "credits.h"
#include "debug.h"
#include "defs.h"
#include "function.h"
#include "interpreter.h"
#include "layers.h" // for 'this_screen' structure
2003-07-28 01:44:38 +00:00
#include "logic.h"
#include "protocol.h"
#include "resman.h"
#include "sound.h"
#include "sword2.h" // for CloseGame()
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
Object_graphic engine_graph; // global for engine
Object_mega engine_mega; // global for engine
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
int32 FN_test_function(int32 *params) {
// param 0 address of a flag
2003-07-28 01:44:38 +00:00
debug(5, " TEST %d %d", params[0], RESULT);
2003-09-17 13:03:13 +00:00
return IR_CONT;
2003-07-28 01:44:38 +00:00
}
2003-09-17 13:03:13 +00:00
int32 FN_test_flags(int32 *params) {
// param 0 value of flag
2003-07-28 01:44:38 +00:00
debug(5, "FN_test_flags %d, %d\n", params[0], params[1]);
2003-09-17 13:03:13 +00:00
return IR_CONT;
}
2003-07-28 01:44:38 +00:00
int32 FN_gosub(int32 *params) {
2003-09-17 13:03:13 +00:00
// hurray, script subroutines
// param 0 id of script
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
LLogic.Logic_up(params[0]);
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
// logic goes up - pc is saved for current level
return IR_GOSUB;
2003-07-28 01:44:38 +00:00
}
int32 FN_new_script(int32 *params) {
2003-09-17 13:03:13 +00:00
// change current script - must be followed by a TERMINATE script
// directive
// param 0 id of script
2003-07-28 01:44:38 +00:00
debug(5, "FN_new_script %d", params[0]);
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
// must clear this
PLAYER_ACTION = 0;
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
LLogic.Logic_replace(params[0]);
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
//drop out no pc save - and around again
return IR_TERMINATE;
2003-07-28 01:44:38 +00:00
}
int32 FN_interact(int32 *params) {
2003-09-17 13:03:13 +00:00
// run targets action on a subroutine
// called by player on his base level 0 idle, for example
// param 0 id of target from which we derive action script
// reference
2003-07-28 01:44:38 +00:00
debug(5, "FN_interact %d", params[0]);
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
// must clear this
PLAYER_ACTION = 0;
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
// 3rd script of clicked on id
LLogic.Logic_up((params[0] * 65536) + 2);
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
// out, up and around again - pc is saved for current level to be
// returned to
return IR_GOSUB;
2003-07-28 01:44:38 +00:00
}
int32 FN_preload(int32 *params) {
2003-09-17 13:03:13 +00:00
// Open & close a resource.
// Forces a resource into memory before it's "officially" opened for
// use. eg. if an anim needs to run on smoothly from another,
// "preloading" gets it into memory in advance to avoid the cacheing
// delay that normally occurs before the first frame.
2003-07-28 01:44:38 +00:00
res_man.Res_open(params[0]); // open resource
res_man.Res_close(params[0]); // close resource
2003-09-17 13:03:13 +00:00
return IR_CONT; // continue script
2003-07-28 01:44:38 +00:00
}
2003-09-17 13:03:13 +00:00
int32 FN_prefetch(int32 *params) {
// Go fetch resource in the background.
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
return IR_CONT;
2003-07-28 01:44:38 +00:00
}
2003-09-17 13:03:13 +00:00
int32 FN_fetch_wait(int32 *params) {
// Fetches a resource in the background but prevents the script from
// continuing until the resource is in memory.
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
return IR_CONT;
2003-07-28 01:44:38 +00:00
}
2003-09-17 13:03:13 +00:00
int32 FN_release(int32 *params) {
// Releases a resource from memory. Used for freeing memory for
// sprites that have just been used and will not be used again.
// Sometimes it is better to kick out a sprite straight away so that
// the memory can be used for more frequent animations.
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
return IR_CONT;
2003-07-28 01:44:38 +00:00
}
int32 FN_random(int32 *params) {
2003-09-17 13:03:13 +00:00
// Generates a random number between 'min' & 'max' inclusive, and
// sticks it in the script flag 'result'
2003-07-28 01:44:38 +00:00
uint32 min = params[0];
uint32 max = params[1];
2003-09-17 13:03:13 +00:00
// return_value = random integer between min and max, inclusive
RESULT = (rand() % (max-min + 1)) + min;
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
// continue script
return IR_CONT;
2003-07-28 01:44:38 +00:00
}
2003-09-17 13:03:13 +00:00
int32 FN_pause(int32 *params) {
2003-07-28 01:44:38 +00:00
// params: 0 pointer to object's logic structure
2003-09-17 13:03:13 +00:00
// 1 number of game-cycles to pause
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
// NB. Pause-value of 0 causes script to continue, 1 causes a 1-cycle
// quit, 2 gives 2 cycles, etc.
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
Object_logic *ob_logic = (Object_logic *) params[0];
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
if (ob_logic->looping == 0) {
// start the pause
2003-07-28 01:44:38 +00:00
ob_logic->looping = 1;
2003-09-17 13:03:13 +00:00
// no. of game cycles
ob_logic->pause = params[1];
2003-07-28 01:44:38 +00:00
}
2003-09-17 13:03:13 +00:00
if (ob_logic->pause) {
// decrement the pause count
ob_logic->pause--;
// drop out of script, but call this again next cycle
return IR_REPEAT;
} else {
2003-07-28 01:44:38 +00:00
ob_logic->looping = 0;
2003-09-17 13:03:13 +00:00
// continue script
return IR_CONT;
2003-07-28 01:44:38 +00:00
}
}
2003-09-17 13:03:13 +00:00
int32 FN_random_pause(int32 *params) {
2003-07-28 01:44:38 +00:00
// params: 0 pointer to object's logic structure
2003-09-17 13:03:13 +00:00
// 1 minimum number of game-cycles to pause
// 2 maximum number of game-cycles to pause
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
Object_logic *ob_logic = (Object_logic *) params[0];
2003-07-28 01:44:38 +00:00
int32 pars[2];
2003-09-17 13:03:13 +00:00
if (ob_logic->looping == 0) {
2003-07-28 01:44:38 +00:00
pars[0] = params[1]; // min
pars[1] = params[2]; // max
FN_random(pars);
2003-09-17 13:03:13 +00:00
// random value between 'min' & 'max' inclusive
pars[1] = RESULT;
2003-07-28 01:44:38 +00:00
}
pars[0] = params[0]; // &logic
return FN_pause(pars);
}
2003-09-17 13:03:13 +00:00
int32 FN_pass_graph(int32 *params) { // Tony28Nov96
// makes an engine local copy of passed graphic_structure and
// mega_structure - run script 4 of an object to request this
// used by FN_turn_to(id) etc
//
// remember, we cannot simply read a compact any longer but instead
// must request it from the object itself
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
//params 0 pointer to a graphic structure (might not need this?)
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
memcpy(&engine_graph, (uint8 *) params[0], sizeof(Object_graphic));
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
// makes no odds
return IR_CONT;
2003-07-28 01:44:38 +00:00
}
int32 FN_pass_mega(int32 *params) {
2003-09-17 13:03:13 +00:00
// makes an engine local copy of passed graphic_structure and
// mega_structure - run script 4 of an object to request this
// used by FN_turn_to(id) etc
//
// remember, we cannot simply read a compact any longer but instead
// must request it from the object itself
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
// params 0 pointer to a mega structure
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
memcpy(&engine_mega, (uint8*) params[0], sizeof(Object_mega));
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
//makes no odds
return IR_CONT;
2003-07-28 01:44:38 +00:00
}
int32 FN_set_value(int32 *params) {
2003-09-17 13:03:13 +00:00
// temp. function!
// used for setting far-referenced megaset resource field in mega
// object, from start script
2003-07-28 01:44:38 +00:00
// params: 0 pointer to object's mega structure
2003-09-17 13:03:13 +00:00
// 1 value to set it to
2003-07-28 01:44:38 +00:00
Object_mega *ob_mega = (Object_mega *) params[0];
2003-07-28 01:44:38 +00:00
ob_mega->megaset_res = params[1];
2003-09-17 13:03:13 +00:00
// continue script
return IR_CONT;
2003-07-28 01:44:38 +00:00
}
2003-09-17 13:03:13 +00:00
2003-07-28 01:44:38 +00:00
#define BLACK 0
#define WHITE 1
2003-09-17 13:03:13 +00:00
#define RED 2
2003-07-28 01:44:38 +00:00
#define GREEN 3
#define BLUE 4
2003-09-17 13:03:13 +00:00
uint8 black[4] = { 0, 0, 0, 0 };
uint8 white[4] = { 255, 255, 255, 0 };
uint8 red[4] = { 255, 0, 0, 0 };
uint8 green[4] = { 0, 255, 0, 0 };
uint8 blue[4] = { 0, 0, 255, 0 };
int32 FN_flash(int32 *params) {
2003-09-17 13:03:13 +00:00
// flash colour 0 (ie. border) - useful during script development
// eg. FN_flash(BLUE) where a text line is missed; RED when some code
// missing, etc
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
// params 0: colour to flash
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
#ifdef _SWORD2_DEBUG
// what colour?
switch (params[0]) {
case WHITE:
BS2_SetPalette(0, 1, white, RDPAL_INSTANT);
break;
case RED:
BS2_SetPalette(0, 1, red, RDPAL_INSTANT);
break;
case GREEN:
BS2_SetPalette(0, 1, green, RDPAL_INSTANT);
break;
case BLUE:
BS2_SetPalette(0, 1, blue, RDPAL_INSTANT);
break;
2003-07-28 01:44:38 +00:00
}
2003-09-17 13:03:13 +00:00
// There used to be a busy-wait loop here, so I don't know how long
// the delay was meant to be. Probably doesn't matter much.
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
ServiceWindows();
g_system->delay_msecs(250);
BS2_SetPalette(0, 1, black, RDPAL_INSTANT);
2003-09-17 13:03:13 +00:00
#endif
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
return IR_CONT;
2003-07-28 01:44:38 +00:00
}
2003-09-17 13:03:13 +00:00
int32 FN_colour(int32 *params) {
2003-09-17 13:03:13 +00:00
// set border colour - useful during script development
// eg. set to colour during a timer situation, then black when timed
// out
2003-07-28 01:44:38 +00:00
// params 0: colour (see defines above)
#ifdef _SWORD2_DEBUG
2003-09-17 13:03:13 +00:00
// what colour?
switch (params[0]) {
case BLACK:
BS2_SetPalette(0, 1, black, RDPAL_INSTANT);
break;
case WHITE:
BS2_SetPalette(0, 1, white, RDPAL_INSTANT);
break;
case RED:
BS2_SetPalette(0, 1, red, RDPAL_INSTANT);
break;
case GREEN:
BS2_SetPalette(0, 1, green, RDPAL_INSTANT);
break;
case BLUE:
BS2_SetPalette(0, 1, blue, RDPAL_INSTANT);
break;
2003-07-28 01:44:38 +00:00
}
2003-09-17 13:03:13 +00:00
#endif
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
return IR_CONT;
2003-07-28 01:44:38 +00:00
}
int32 FN_display_msg(int32 *params) {
2003-09-17 13:03:13 +00:00
// Display a message to the user on the screen.
//
2003-07-28 01:44:38 +00:00
// params 0: Text number of message to be displayed.
2003-09-17 13:03:13 +00:00
uint32 local_text = params[0] & 0xffff;
uint32 text_res = params[0] / SIZE;
2003-07-28 01:44:38 +00:00
// Display message for three seconds.
2003-09-17 13:03:13 +00:00
// +2 to skip the encoded text number in the first 2 chars; 3 is
// duration in seconds
DisplayMsg(FetchTextLine(res_man.Res_open(text_res), local_text) + 2, 3);
2003-07-28 01:44:38 +00:00
res_man.Res_close(text_res);
RemoveMsg();
2003-09-17 13:03:13 +00:00
return IR_CONT;
2003-07-28 01:44:38 +00:00
}
int32 FN_reset_globals(int32 *params) {
2003-09-17 13:03:13 +00:00
// FN_reset_globals is used by the demo - so it can loop back &
// restart itself
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
int32 size;
uint32 *globals;
int j;
size = res_man.Res_fetch_len(1);
size -= sizeof(_standardHeader);
2003-07-28 01:44:38 +00:00
debug(5, "globals size %d", size / 4);
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
globals = (uint32*) ((uint8 *) res_man.Res_open(1) + sizeof(_standardHeader));
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
// blank each global variable
for (j = 0; j < size / 4; j++)
globals[j] = 0;
2003-07-28 01:44:38 +00:00
res_man.Res_close(1);
2003-09-17 13:03:13 +00:00
// all objects but george
res_man.Kill_all_objects(0);
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
// reopen global variables resource & send address to interpreter - it
// won't be moving
// SetGlobalInterpreterVariables((int32 *) (res_man.Res_open(1) + sizeof(_standardHeader)));
// res_man.Res_close(1);
2003-07-28 01:44:38 +00:00
// FOR THE DEMO - FORCE THE SCROLLING TO BE RESET! (James29may97)
// - this is taken from FN_init_background
2003-09-17 13:03:13 +00:00
// switch on scrolling (2 means first time on screen)
this_screen.scroll_flag = 2;
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
return IR_CONT;
}
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
int32 FN_play_credits(int32 *params) {
// params: none
2003-09-17 13:03:13 +00:00
// This function just quits the game if this is the playable demo, ie.
// credits are NOT played in the demo any more!
2003-07-28 01:44:38 +00:00
2003-09-17 13:03:13 +00:00
if (!DEMO) {
uint8 oldPal[1024];
uint8 tmpPal[1024];
int32 music_length;
uint32 safe_looping_music_id;
int32 pars[2];
// FIXME: We need a better method for saving/restoring the
// music state as this one only restarts looping music.
safe_looping_music_id = looping_music_id;
g_sound->MuteFx(1);
g_sound->MuteSpeech(1);
g_sound->StopMusic();
memcpy(oldPal, palCopy, 1024);
memset(tmpPal, 0, 1024);
WaitForFade();
FadeDown(0.75);
WaitForFade();
tmpPal[4] = 255;
tmpPal[5] = 255;
tmpPal[6] = 255;
BS2_SetPalette(0, 256, tmpPal, RDPAL_INSTANT);
// Play the credits music. Is it enough with just one
// repetition of it?
pars[0] = 309;
pars[1] = FX_SPOT;
FN_play_music(pars);
music_length = 1000 * g_sound->MusicTimeRemaining();
debug(0, "Credits music length: ~%d ms", music_length);
CloseMenuImmediately();
while (g_sound->MusicTimeRemaining()) {
EraseBackBuffer();
SetNeedRedraw();
// FIXME: Draw the credits text. The actual text
// messages are stored in credits.clu, and I'm guessing
// that credits.bmp may be the font.
ServiceWindows();
_keyboardEvent ke;
if (ReadKey(&ke) == RD_OK && ke.keycode == 27)
break;
g_system->delay_msecs(30);
2003-09-17 13:03:13 +00:00
}
2003-07-28 01:44:38 +00:00
looping_music_id = safe_looping_music_id;
if (looping_music_id) {
pars[0] = looping_music_id;
pars[1] = FX_LOOP;
FN_play_music(pars);
} else
FN_stop_music(NULL);
BS2_SetPalette(0, 256, oldPal, RDPAL_FADE);
FadeUp(0.75);
ServiceWindows();
Build_display();
WaitForFade();
2003-07-28 01:44:38 +00:00
g_sound->MuteFx(0);
g_sound->MuteSpeech(0);
2003-09-17 13:03:13 +00:00
}
// FIXME: This probably isn't the correct way of shutting down ScummVM
// Anyway, I don't know if we ever call this from the demo.
2003-09-17 13:03:13 +00:00
if (g_sword2->_gameId == GID_SWORD2_DEMO) {
Close_game(); // close engine systems down
2003-07-28 01:44:38 +00:00
CloseAppWindow();
exit(0); // quit the game
}
2003-09-17 13:03:13 +00:00
return IR_CONT;
2003-07-28 01:44:38 +00:00
}