2021-05-17 18:47:39 +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.
|
|
|
|
*
|
2021-12-26 17:47:58 +00:00
|
|
|
* 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.
|
2021-05-17 18:47:39 +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.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* aint32 with this program; if not, write to the Free Software
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* Based on the original sources
|
|
|
|
* Faery Tale II -- The Halls of the Dead
|
|
|
|
* (c) 1993-1996 The Wyrmkeep Entertainment Co.
|
|
|
|
*/
|
|
|
|
|
2021-06-23 12:41:01 +00:00
|
|
|
#include "saga2/saga2.h"
|
2021-06-01 21:47:48 +00:00
|
|
|
#include "saga2/objproto.h"
|
2021-05-17 18:47:39 +00:00
|
|
|
#include "saga2/spellbuk.h"
|
2021-05-31 22:55:54 +00:00
|
|
|
#include "saga2/spelshow.h"
|
2021-06-01 22:16:29 +00:00
|
|
|
#include "saga2/spellio.h"
|
|
|
|
#include "saga2/hresmgr.h"
|
2021-05-17 18:47:39 +00:00
|
|
|
|
|
|
|
namespace Saga2 {
|
|
|
|
|
|
|
|
//-------------------------------------------------------------------
|
|
|
|
// The initialization for spells is done in this module:
|
|
|
|
// - Display Effects are defined (ball spell, bolt spell, etc)
|
|
|
|
// - The spell definitions are loaded from the resource file
|
|
|
|
// - The spell internal effect definitions are loaded from disk
|
|
|
|
// and attached to the appropriate spell
|
|
|
|
// - Spell Color maps are loaded from the resource file
|
|
|
|
// - This file is also home to most of the global variables
|
|
|
|
// which relate to spell casting
|
|
|
|
//
|
|
|
|
|
|
|
|
/* ===================================================================== *
|
|
|
|
Constants
|
|
|
|
* ===================================================================== */
|
|
|
|
|
2021-06-02 15:31:17 +00:00
|
|
|
const uint32 spellSpriteID = MKTAG('S', 'P', 'F', 'X');
|
2021-05-17 18:47:39 +00:00
|
|
|
|
2022-10-29 12:39:34 +00:00
|
|
|
const int32 maxSpells = kTotalSpellBookPages;
|
|
|
|
const int32 maxSpellPrototypes = kTotalSpellBookPages;
|
2021-05-17 18:47:39 +00:00
|
|
|
const int32 maxEffectPrototypes = 16;
|
|
|
|
|
|
|
|
const int32 maxSpellColorMaps = 32;
|
|
|
|
|
|
|
|
/* const */ // For some reason, MVC can't handle constant static classes.
|
|
|
|
// ( Note: It would be better to use a SIN/COS table anyway than
|
|
|
|
// a table of points, since that allows more than 24 directions ).
|
2021-07-01 17:31:36 +00:00
|
|
|
StaticTilePoint WallVectors[8] = {
|
|
|
|
{2, 0, 0}, {1, 1, 0}, {0, 2, 0},
|
|
|
|
{-1, 2, 0}, {-2, 0, 0}, {-1, -1, 0},
|
|
|
|
{0, -2, 0}, {1, -1, 0}
|
2021-05-17 18:47:39 +00:00
|
|
|
};
|
|
|
|
|
2021-07-01 17:31:36 +00:00
|
|
|
StaticTilePoint FireballVectors[24] = {
|
|
|
|
{4, 0, 0}, {4, 1, 0}, {3, 2, 0},
|
|
|
|
{3, 3, 0}, {2, 3, 0}, {1, 4, 0},
|
|
|
|
{0, 4, 0}, {-1, 4, 0}, {-2, 3, 0},
|
|
|
|
{-3, 3, 0}, {-3, 2, 0}, {-4, 1, 0},
|
|
|
|
{-4, 0, 0}, {-4, -1, 0}, {-3, -2, 0},
|
|
|
|
{-3, -3, 0}, {-2, -3, 0}, {-1, -4, 0},
|
|
|
|
{0, -4, 0}, {1, -4, 0}, {2, -3, 0},
|
|
|
|
{3, -3, 0}, {3, -2, 0}, {4, -1, 0}
|
2021-05-17 18:47:39 +00:00
|
|
|
};
|
|
|
|
|
2021-07-01 17:31:36 +00:00
|
|
|
StaticTilePoint SquareSpellVectors[32] = {
|
|
|
|
{4, 0, 0}, {4, 1, 0}, {4, 2, 0}, {4, 3, 0},
|
|
|
|
{4, 4, 0}, {3, 4, 0}, {2, 4, 0}, {1, 4, 0},
|
|
|
|
{0, 4, 0}, {-1, 4, 0}, {-2, 4, 0}, {-3, 4, 0},
|
|
|
|
{-4, 4, 0}, {-4, 3, 0}, {-4, 2, 0}, {-4, 1, 0},
|
|
|
|
{-4, 0, 0}, {-4, -1, 0}, {-4, -2, 0}, {-4, -3, 0},
|
|
|
|
{-4, -4, 0}, {-3, -4, 0}, {-2, -4, 0}, {-1, -4, 0},
|
|
|
|
{0, -4, 0}, {1, -4, 0}, {2, -4, 0}, {3, -4, 0},
|
|
|
|
{4, -4, 0}, {4, -3, 0}, {4, -2, 0}, {4, -1, 0}
|
2021-05-17 18:47:39 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/* ===================================================================== *
|
|
|
|
Imports
|
|
|
|
* ===================================================================== */
|
|
|
|
|
|
|
|
extern hResContext *spriteRes; // sprite resource handle
|
|
|
|
extern hResContext *schemeRes; // sprite resource handle
|
|
|
|
extern ColorTable identityColors;
|
|
|
|
|
|
|
|
/* ===================================================================== *
|
|
|
|
Global data
|
|
|
|
* ===================================================================== */
|
|
|
|
|
2021-06-17 09:32:27 +00:00
|
|
|
SpriteSet *spellSprites; // longsword test sprites
|
2021-07-18 08:38:49 +00:00
|
|
|
SpellStuff *spellBook;
|
2021-05-17 18:47:39 +00:00
|
|
|
|
|
|
|
ColorTable spellColorMaps[maxSpellColorMaps];
|
2021-06-17 09:32:27 +00:00
|
|
|
ColorSchemeList *spellSchemes;
|
2021-05-17 18:47:39 +00:00
|
|
|
|
|
|
|
int32 loadedColorMaps;
|
|
|
|
|
|
|
|
/* ===================================================================== *
|
|
|
|
prototypes
|
|
|
|
* ===================================================================== */
|
|
|
|
|
2021-09-11 09:13:35 +00:00
|
|
|
static void defineEffects();
|
|
|
|
static void loadMagicData();
|
2021-05-17 18:47:39 +00:00
|
|
|
|
|
|
|
/* ===================================================================== *
|
|
|
|
code
|
|
|
|
* ===================================================================== */
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------
|
|
|
|
// InitMagic called from main startup code
|
|
|
|
|
2021-09-11 09:13:35 +00:00
|
|
|
void initMagic() {
|
2021-07-04 07:34:33 +00:00
|
|
|
g_vm->_edpList = new EffectDisplayPrototypeList(maxEffectPrototypes);
|
|
|
|
g_vm->_sdpList = new SpellDisplayPrototypeList(maxSpellPrototypes);
|
|
|
|
|
2021-07-18 08:38:49 +00:00
|
|
|
spellBook = new SpellStuff[maxSpells]();
|
|
|
|
|
2021-05-17 18:47:39 +00:00
|
|
|
defineEffects();
|
|
|
|
loadMagicData();
|
|
|
|
|
2021-06-17 09:32:27 +00:00
|
|
|
const int colorSchemeSize = 44;
|
|
|
|
Common::SeekableReadStream *stream;
|
|
|
|
|
|
|
|
stream = loadResourceToStream(spriteRes, spellSpriteID, "spell sprites");
|
|
|
|
spellSprites = new SpriteSet(stream);
|
2021-05-27 18:05:08 +00:00
|
|
|
assert(spellSprites);
|
2021-06-17 09:32:27 +00:00
|
|
|
delete stream;
|
|
|
|
|
|
|
|
loadedColorMaps = schemeRes->size(spellSpriteID) / colorSchemeSize;
|
|
|
|
|
|
|
|
stream = loadResourceToStream(schemeRes, spellSpriteID, "scheme list");
|
|
|
|
spellSchemes = new ColorSchemeList(loadedColorMaps, stream);
|
2021-05-27 18:05:08 +00:00
|
|
|
assert(spellSchemes);
|
2021-06-17 09:32:27 +00:00
|
|
|
delete stream;
|
2021-05-17 18:47:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-09-11 09:13:35 +00:00
|
|
|
void cleanupMagic() {
|
2021-07-16 17:48:22 +00:00
|
|
|
g_vm->_activeSpells->cleanup();
|
2021-05-17 18:47:39 +00:00
|
|
|
for (int i = 0; i < maxSpells; i++) {
|
|
|
|
spellBook[i].killEffects();
|
|
|
|
}
|
2021-08-20 16:21:47 +00:00
|
|
|
delete[] spellBook;
|
2021-07-18 08:38:49 +00:00
|
|
|
|
2021-07-04 07:34:33 +00:00
|
|
|
g_vm->_sdpList->cleanup();
|
|
|
|
g_vm->_edpList->cleanup();
|
|
|
|
|
|
|
|
delete g_vm->_sdpList;
|
|
|
|
delete g_vm->_edpList;
|
2021-05-17 18:47:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* ===================================================================== *
|
|
|
|
Effect data
|
|
|
|
* ===================================================================== */
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------
|
|
|
|
// the macros make things more legible than the entire call
|
|
|
|
|
|
|
|
// Set up a display effect shape
|
2021-07-04 07:34:33 +00:00
|
|
|
#define ADD_EFFECT( n, p, s, f, h, b, i ) ( g_vm->_edpList->add( new EffectDisplayPrototype(n,&p,&s,&f,&h,&b,&i )))
|
2021-05-17 18:47:39 +00:00
|
|
|
// Chain another effect when done
|
2021-07-04 07:34:33 +00:00
|
|
|
#define SECOND_EFFECT( e, n, p, s, f, h, b, i ) ( g_vm->_edpList->append( new EffectDisplayPrototype(n,&p,&s,&f,&h,&b,&i ),e))
|
2021-05-17 18:47:39 +00:00
|
|
|
|
|
|
|
//-----------------------------------------------------------------------
|
|
|
|
|
2021-09-11 09:13:35 +00:00
|
|
|
static void defineEffects() {
|
2021-05-17 18:47:39 +00:00
|
|
|
int16 i;
|
|
|
|
ADD_EFFECT(1, invisibleSpellPos, invisibleSprites, invisibleSpellSta, ShortTillThere, ThinTillThere, invisibleSpellInit);
|
|
|
|
ADD_EFFECT(1, auraSpellPos, auraSprites, auraSpellSta, ShortTillThere, ThinTillThere, auraSpellInit);
|
|
|
|
ADD_EFFECT(1, projectileSpellPos, projectileSprites, projectileSpellSta, StaticHeight, StaticBreadth, projectileSpellInit);
|
|
|
|
ADD_EFFECT(12, exchangeSpellPos, exchangeSprites, exchangeSpellSta, StaticHeight, StaticBreadth, exchangeSpellInit);
|
|
|
|
ADD_EFFECT(36, boltSpellPos, boltSprites, boltSpellSta, StaticHeight, StaticBreadth, boltSpellInit);
|
|
|
|
ADD_EFFECT(24, coneSpellPos, coneSprites, coneSpellSta, GrowLinear, BulkLinear, coneSpellInit);
|
|
|
|
i = ADD_EFFECT(1, projectileSpellPos, projectileSprites, projectileSpellSta, StaticHeight, StaticBreadth, projectileSpellInit);
|
|
|
|
SECOND_EFFECT(i, 24, ballSpellPos, ballSprites, ballSpellSta, ShortTillThere, ThinTillThere, ballSpellInit);
|
|
|
|
i = ADD_EFFECT(1, projectileSpellPos, projectileSprites, projectileSpellSta, StaticHeight, StaticBreadth, projectileSpellInit);
|
|
|
|
SECOND_EFFECT(i, 32, squareSpellPos, squareSprites, squareSpellSta, StaticHeight, StaticBreadth, squareSpellInit);
|
|
|
|
ADD_EFFECT(24, waveSpellPos, waveSprites, waveSpellSta, GrowLinear, BulkLinear, waveSpellInit);
|
|
|
|
i = ADD_EFFECT(1, projectileSpellPos, projectileSprites, projectileSpellSta, StaticHeight, StaticBreadth, projectileSpellInit);
|
|
|
|
SECOND_EFFECT(i, 24, stormSpellPos, stormSprites, stormSpellSta, ShortTillThere, ThinTillThere, stormSpellInit);
|
|
|
|
i = ADD_EFFECT(1, projectileSpellPos, projectileSprites, projectileSpellSta, StaticHeight, StaticBreadth, projectileSpellInit);
|
|
|
|
SECOND_EFFECT(i, 1, glowSpellPos, auraSprites, auraSpellSta, ShortTillThere, ThinTillThere, glowSpellInit);
|
|
|
|
ADD_EFFECT(1, glowSpellPos, auraSprites, auraSpellSta, ShortTillThere, ThinTillThere, glowSpellInit);
|
|
|
|
ADD_EFFECT(20, beamSpellPos, beamSprites, beamSpellSta, StaticHeight, StaticBreadth, beamSpellInit);
|
|
|
|
ADD_EFFECT(8, wallSpellPos, wallSprites, wallSpellSta, StaticHeight, StaticBreadth, wallSpellInit);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------
|
2021-07-04 07:34:33 +00:00
|
|
|
#define ADD_SHOW( e, a, b, c, d, f, g, m, i, s, n ) ( g_vm->_sdpList->add( new SpellDisplayPrototype( e, a, b, c, d, f, g, m, i, s, n )))
|
2021-05-17 18:47:39 +00:00
|
|
|
|
|
|
|
//-----------------------------------------------------------------------
|
|
|
|
// loadMagicData : reads magic related data from the resource file
|
|
|
|
|
2021-09-11 09:13:35 +00:00
|
|
|
static void loadMagicData() {
|
2021-05-17 18:47:39 +00:00
|
|
|
int16 i;
|
|
|
|
hResContext *spellRes;
|
|
|
|
|
|
|
|
// Get spell definitions
|
|
|
|
spellRes = auxResFile->newContext(
|
2021-06-02 15:31:17 +00:00
|
|
|
MKTAG('S', 'P', 'E', 'L'),
|
2021-05-17 18:47:39 +00:00
|
|
|
"spell resources");
|
2021-11-13 21:40:38 +00:00
|
|
|
if (spellRes == nullptr || !spellRes->_valid)
|
2021-05-17 18:47:39 +00:00
|
|
|
error("Error accessing spell resource group.\n");
|
|
|
|
i = 1;
|
2022-10-29 12:41:52 +00:00
|
|
|
ADD_SHOW(keAreaInvisible, 0, 0, 0, 0, kDiFlagInc, kEcFlagNone, 30, MKTAG('S', 'T', 'A', 0), 23, 24);
|
2021-05-17 18:47:39 +00:00
|
|
|
|
2022-10-29 12:33:52 +00:00
|
|
|
spellBook[0].setManaType(ksManaIDSkill);
|
2021-05-17 18:47:39 +00:00
|
|
|
|
|
|
|
while (spellRes->size(
|
2021-06-02 15:31:17 +00:00
|
|
|
MKTAG('I', 'N', 'F', i)) > 0) {
|
2021-05-17 18:47:39 +00:00
|
|
|
ResourceSpellItem *rsi =
|
|
|
|
(ResourceSpellItem *)LoadResource(
|
|
|
|
spellRes,
|
2021-06-02 15:31:17 +00:00
|
|
|
MKTAG('I', 'N', 'F', i),
|
2021-05-17 18:47:39 +00:00
|
|
|
"spell");
|
|
|
|
|
2021-11-13 21:40:38 +00:00
|
|
|
if (rsi == nullptr)
|
2021-05-17 18:47:39 +00:00
|
|
|
error("Unable to load data for spell %d", i);
|
|
|
|
|
|
|
|
spellBook[rsi->spell].setupFromResource(rsi);
|
2021-07-04 07:34:33 +00:00
|
|
|
g_vm->_sdpList->add(new SpellDisplayPrototype(rsi));
|
2021-05-17 18:47:39 +00:00
|
|
|
|
2021-06-23 09:40:36 +00:00
|
|
|
free(rsi);
|
2021-05-17 18:47:39 +00:00
|
|
|
i++;
|
|
|
|
}
|
2021-05-27 18:05:08 +00:00
|
|
|
assert(i > 1);
|
2021-05-17 18:47:39 +00:00
|
|
|
|
|
|
|
// get spell effects
|
|
|
|
i = 0;
|
|
|
|
while (spellRes->size(
|
2021-06-02 15:31:17 +00:00
|
|
|
MKTAG('E', 'F', 'F', i)) > 0) {
|
2021-05-17 18:47:39 +00:00
|
|
|
ResourceSpellEffect *rse =
|
|
|
|
(ResourceSpellEffect *)LoadResource(
|
|
|
|
spellRes,
|
2021-06-02 15:31:17 +00:00
|
|
|
MKTAG('E', 'F', 'F', i),
|
2021-05-17 18:47:39 +00:00
|
|
|
"spell effect");
|
|
|
|
|
2021-11-13 21:40:38 +00:00
|
|
|
if (rse == nullptr)
|
2021-05-17 18:47:39 +00:00
|
|
|
error("Unable to load effects for spell %d", i);
|
|
|
|
|
|
|
|
if (rse->spell)
|
|
|
|
spellBook[rse->spell].addEffect(rse);
|
|
|
|
|
2021-06-23 09:40:36 +00:00
|
|
|
free(rse);
|
2021-05-17 18:47:39 +00:00
|
|
|
i++;
|
|
|
|
}
|
2021-05-27 18:05:08 +00:00
|
|
|
assert(i > 1);
|
2021-05-17 18:47:39 +00:00
|
|
|
|
|
|
|
|
|
|
|
// get spell color maps
|
|
|
|
memcpy(spellColorMaps[0], identityColors, sizeof(ColorTable));
|
|
|
|
auxResFile->disposeContext(spellRes);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // end of namespace Saga2
|