mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-22 01:57:16 +00:00
1120 lines
33 KiB
C++
1120 lines
33 KiB
C++
/* 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 3 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
|
|
* 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.
|
|
*/
|
|
|
|
#include "saga2/saga2.h"
|
|
#include "saga2/fta.h"
|
|
#include "saga2/fontlib.h"
|
|
#include "saga2/speech.h"
|
|
#include "saga2/motion.h"
|
|
#include "saga2/panel.h"
|
|
#include "saga2/grabinfo.h"
|
|
#include "saga2/player.h"
|
|
#include "saga2/annoy.h"
|
|
#include "saga2/cmisc.h"
|
|
#include "saga2/tile.h"
|
|
#include "saga2/tilemode.h"
|
|
|
|
namespace Saga2 {
|
|
|
|
struct TextSpan {
|
|
char *text; // pointer to 1st char of span
|
|
int16 charWidth, // number of characters in span
|
|
pixelWidth; // number of pixels in span
|
|
};
|
|
|
|
//-----------------------------------------------------------------------
|
|
// externs
|
|
|
|
extern StaticPoint16 fineScroll;
|
|
int kludgeHeight = 15;
|
|
extern StaticTilePoint viewCenter; // coordinates of view on map
|
|
|
|
//-----------------------------------------------------------------------
|
|
// constants
|
|
|
|
const int maxWidth = 420;
|
|
const int defaultWidth = 380;
|
|
const int actorHeight = 80; // Assume 80
|
|
|
|
const int lineLeading = 2; // space between lines
|
|
const int outlineWidth = 2; // width of character outline
|
|
const int bulletWidth = 13; // width of bullet symbol
|
|
|
|
//-----------------------------------------------------------------------
|
|
// prototypes
|
|
|
|
int16 buttonWrap(
|
|
TextSpan *lineList, // indicates where line breaks are
|
|
TextSpan *buttonList, // indicates where button breaks are
|
|
int16 &buttonCount, // returns number of buttons
|
|
char *text, // text to wrap
|
|
int16 width, // width of text
|
|
int16 supressText,
|
|
gPort &textPort);
|
|
|
|
//-----------------------------------------------------------------------
|
|
// locals
|
|
|
|
// Temporary: Alarm which determines when speech finishes
|
|
Alarm speechFinished;
|
|
|
|
|
|
// The list of active and non-active speech tasks for all actors
|
|
uint8 speechListBuffer[sizeof(SpeechTaskList)];
|
|
SpeechTaskList &speechList = *((SpeechTaskList *)speechListBuffer);
|
|
|
|
static TextSpan speechLineList[64], // list of speech lines
|
|
speechButtonList[64]; // list of speech buttons
|
|
int16 speechLineCount, // count of speech lines
|
|
speechButtonCount; // count of speech buttons
|
|
|
|
static StaticPoint16 initialSpeechPosition = {0, 0}; // inital coords of speech
|
|
|
|
// Image data for the little "bullet"
|
|
static uint8 BulletData[] = {
|
|
0x00, 0x00, 0x00, 0x18, 0x18, 0x18, 0x00, 0x00, 0x00, // Row 0
|
|
0x00, 0x18, 0x18, 0x18, 0x18, 0x18, 0x18, 0x18, 0x00, // Row 1
|
|
0x00, 0x18, 0x18, 0x4C, 0x4A, 0x4A, 0x18, 0x18, 0x00, // Row 2
|
|
0x18, 0x18, 0x4E, 0x4C, 0x4A, 0x0A, 0x4A, 0x18, 0x18, // Row 3
|
|
0x18, 0x18, 0x50, 0x4E, 0x4C, 0x4A, 0x4A, 0x18, 0x18, // Row 4
|
|
0x18, 0x18, 0x4E, 0x50, 0x50, 0x4E, 0x4E, 0x18, 0x18, // Row 5
|
|
0x00, 0x18, 0x18, 0x4E, 0x50, 0x50, 0x18, 0x18, 0x00, // Row 6
|
|
0x00, 0x18, 0x18, 0x18, 0x18, 0x18, 0x18, 0x18, 0x00, // Row 7
|
|
0x00, 0x00, 0x00, 0x18, 0x18, 0x18, 0x00, 0x00, 0x00, // Row 8
|
|
};
|
|
|
|
static StaticPixelMap BulletImage = {{9, 9}, BulletData};
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Speech button mode override.
|
|
|
|
extern gPanelList *speakButtonControls; // controls for embedded speech button
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Audio resource ID generator
|
|
|
|
static char convBuf[5];
|
|
|
|
inline uint32 extendID(int16 smallID) {
|
|
Common::sprintf_s(convBuf, "%4.4d", smallID);
|
|
return smallID ? MKTAG(convBuf[0] + 'A' - '0', convBuf[1], convBuf[2], convBuf[3]) : 0 ;
|
|
}
|
|
|
|
/* ===================================================================== *
|
|
Speech member functions
|
|
* ===================================================================== */
|
|
|
|
void Speech::read(Common::InSaveFile *in) {
|
|
// Restore the sample count and character count
|
|
_sampleCount = in->readSint16LE();
|
|
_charCount = in->readSint16LE();
|
|
|
|
// Restore the text boundaries
|
|
_bounds.read(in);
|
|
|
|
// Restore the pen color and outline color
|
|
_penColor = in->readUint16LE();
|
|
_outlineColor = in->readUint16LE();
|
|
|
|
// Restore the object ID
|
|
_objID = in->readUint16LE();
|
|
|
|
// Restore the thread ID
|
|
_thread = in->readSint16LE();
|
|
|
|
// Restore the flags
|
|
_speechFlags = in->readSint16LE();
|
|
|
|
debugC(4, kDebugSaveload, "...... sampleCount = %d", _sampleCount);
|
|
debugC(4, kDebugSaveload, "...... charCount = %d", _charCount);
|
|
debugC(4, kDebugSaveload, "...... penColor = %d", _penColor);
|
|
debugC(4, kDebugSaveload, "...... outlineColor = %d", _outlineColor);
|
|
debugC(4, kDebugSaveload, "...... bounds = (%d, %d, %d, %d)",
|
|
_bounds.x, _bounds.y, _bounds.width, _bounds.height);
|
|
debugC(4, kDebugSaveload, "...... objID = %d", _objID);
|
|
debugC(4, kDebugSaveload, "...... thread = %d", _thread);
|
|
debugC(4, kDebugSaveload, "...... speechFlags = %d", _speechFlags);
|
|
|
|
// Restore the sample ID's
|
|
for (int i = 0; i < _sampleCount; i++) {
|
|
_sampleID[i] = in->readUint32BE();
|
|
debugC(4, kDebugSaveload, "...... sampleID[%d] = %d", i, _sampleID[i]);
|
|
}
|
|
|
|
// Restore the text
|
|
in->read(_speechBuffer, _charCount);
|
|
_speechBuffer[_charCount] = '\0';
|
|
debugC(4, kDebugSaveload, "...... _speechBuffer = %s", _speechBuffer);
|
|
|
|
// Requeue the speech if needed
|
|
if (_speechFlags & kSpQueued) {
|
|
// Add to the active list
|
|
speechList.remove(this);
|
|
speechList._list.push_back(this);
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Return the number of bytes needed to archive this SpeechTask
|
|
|
|
int32 Speech::archiveSize() {
|
|
return sizeof(_sampleCount)
|
|
+ sizeof(_charCount)
|
|
+ sizeof(_bounds)
|
|
+ sizeof(_penColor)
|
|
+ sizeof(_outlineColor)
|
|
+ sizeof(_objID)
|
|
+ sizeof(_thread)
|
|
+ sizeof(_speechFlags)
|
|
+ sizeof(uint32) * _sampleCount
|
|
+ sizeof(char) * _charCount;
|
|
}
|
|
|
|
void Speech::write(Common::MemoryWriteStreamDynamic *out) {
|
|
// Store the sample count and character count
|
|
out->writeSint16LE(_sampleCount);
|
|
out->writeSint16LE(_charCount);
|
|
|
|
// Store the text boundaries
|
|
_bounds.write(out);
|
|
|
|
// Store the pen color and outline color
|
|
out->writeUint16LE(_penColor);
|
|
out->writeUint16LE(_outlineColor);
|
|
|
|
// Store the object's ID
|
|
out->writeUint16LE(_objID);
|
|
|
|
// Store the thread ID
|
|
out->writeSint16LE(_thread);
|
|
|
|
// Store the flags. NOTE: Make sure this speech is not stored
|
|
// as being active
|
|
out->writeSint16LE(_speechFlags & ~kSpActive);
|
|
|
|
debugC(4, kDebugSaveload, "...... _sampleCount = %d", _sampleCount);
|
|
debugC(4, kDebugSaveload, "...... _charCount = %d", _charCount);
|
|
debugC(4, kDebugSaveload, "...... _penColor = %d", _penColor);
|
|
debugC(4, kDebugSaveload, "...... _outlineColor = %d", _outlineColor);
|
|
debugC(4, kDebugSaveload, "...... _bounds = (%d, %d, %d, %d)",
|
|
_bounds.x, _bounds.y, _bounds.width, _bounds.height);
|
|
debugC(4, kDebugSaveload, "...... _objID = %d", _objID);
|
|
debugC(4, kDebugSaveload, "...... thread = %d", _thread);
|
|
debugC(4, kDebugSaveload, "...... _speechFlags = %d", _speechFlags);
|
|
|
|
for (int i = 0; i < _sampleCount; i++) {
|
|
out->writeUint32BE(_sampleID[i]);
|
|
debugC(4, kDebugSaveload, "...... _sampleID[%d] = %d", i, _sampleID[i]);
|
|
}
|
|
|
|
// Store the text
|
|
out->write(_speechBuffer, _charCount);
|
|
debugC(4, kDebugSaveload, "...... _speechBuffer = %s", _speechBuffer);
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Append text and sample to existing speech record
|
|
|
|
bool Speech::append(char *text, int32 sampID) {
|
|
int16 len = strlen(text);
|
|
|
|
// Check to see if there's enough room in the character buffer
|
|
if (_charCount + len >= (long)sizeof(_speechBuffer)
|
|
|| _sampleCount >= MAX_SAMPLES) return false;
|
|
|
|
// Copy text to end of text in buffer, including '\0'
|
|
memcpy(&_speechBuffer[_charCount], text, len + 1);
|
|
_charCount += len;
|
|
|
|
// Append sample ID to list of samples.
|
|
// REM: We should translate sample ID's from index to resource
|
|
// number here.
|
|
if (sampID)
|
|
_sampleID[_sampleCount++] = extendID(sampID);
|
|
|
|
return true;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Move speech to active list
|
|
|
|
bool Speech::activate() {
|
|
|
|
// Remove from existing list
|
|
speechList.remove(this);
|
|
|
|
// Add to the active list
|
|
speechList._list.push_back(this);
|
|
|
|
_speechFlags |= kSpQueued;
|
|
|
|
// This routine can't fail
|
|
return true;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Move speech to active list
|
|
|
|
bool Speech::setupActive() {
|
|
int16 x, y;
|
|
int16 buttonNum = 0,
|
|
buttonChars;
|
|
|
|
_speechFlags |= kSpActive;
|
|
|
|
speechFinished.set((_charCount * 4 / 2) + kTicksPerSecond);
|
|
|
|
//Turn Actor Towards Person They Are Talking To
|
|
// MotionTask::turnObject( *obj, GameObject::objectAddress(32794)->getLocation());
|
|
// Actor *a = (Actor *)obj;
|
|
// if(!a->setAction( kActionJumpUp, kAnimateRandom ))
|
|
// throw gError( "Could Not Set Talk Animation");
|
|
|
|
// Set up temp gport for blitting to bitmap
|
|
_textPort.setStyle(kTextStyleThickOutline); // extra Thick Outline
|
|
_textPort.setOutlineColor(_outlineColor); // outline black
|
|
_textPort.setFont(&Amber13Font); // speech font
|
|
_textPort.setColor(_penColor); // color of letters
|
|
_textPort.setMode(kDrawModeMatte); // insure transparency
|
|
|
|
setWidth();
|
|
|
|
// If speech position is off-screen, then skip
|
|
if (!calcPosition(initialSpeechPosition)) return false;
|
|
|
|
if (_sampleCount) {
|
|
GameObject *go = GameObject::objectAddress(_objID);
|
|
Location loc = go->notGetWorldLocation();
|
|
_sampleID[_sampleCount] = 0;
|
|
|
|
//sayVoice(_sampleID);
|
|
|
|
/// EO SEARCH ///
|
|
if (sayVoiceAt(_sampleID, loc))
|
|
_speechFlags |= kSpHasVoice;
|
|
else
|
|
_speechFlags &= ~kSpHasVoice;
|
|
|
|
} else _speechFlags &= ~kSpHasVoice;
|
|
|
|
speechLineCount = buttonWrap(speechLineList,
|
|
speechButtonList,
|
|
speechButtonCount,
|
|
_speechBuffer,
|
|
_bounds.width,
|
|
!g_vm->_speechText && (_speechFlags & kSpHasVoice),
|
|
_textPort);
|
|
|
|
// Compute height of bitmap based on number of lines of text.
|
|
// Include 4 for outline width
|
|
_bounds.height =
|
|
(speechLineCount * (_textPort._font->height + lineLeading))
|
|
+ outlineWidth * 2;
|
|
|
|
// Blit to temp bitmap
|
|
_speechImage._size.x = _bounds.width;
|
|
_speechImage._size.y = _bounds.height;
|
|
_speechImage._data = new uint8[_speechImage.bytes()]();
|
|
_textPort.setMap(&_speechImage);
|
|
|
|
y = outlineWidth; // Plus 2 for Outlines
|
|
buttonChars = speechButtonList[buttonNum].charWidth;
|
|
|
|
for (int i = 0; i < speechLineCount; i++) {
|
|
int16 lineChars = speechLineList[i].charWidth;
|
|
char *lineText = speechLineList[i].text;
|
|
|
|
x = (_bounds.width - speechLineList[i].pixelWidth) / 2
|
|
+ outlineWidth;
|
|
|
|
_textPort.moveTo(x, y);
|
|
|
|
// Draw each button on the line in turn.
|
|
while (lineChars > 0) {
|
|
int16 dChars;
|
|
|
|
// If this is the end of a button
|
|
if (buttonChars <= 0) {
|
|
// Incr to next button
|
|
buttonNum++;
|
|
|
|
// If no more buttons, then go nae ferther...
|
|
// Fer death awaits with nasty, pointy teeth...!
|
|
if (buttonNum > speechButtonCount) break;
|
|
|
|
buttonChars = speechButtonList[buttonNum].charWidth;
|
|
_textPort.setColor(1 + 9);
|
|
|
|
// Blit the little bullet symbol
|
|
lineChars--;
|
|
lineText++;
|
|
buttonChars--;
|
|
|
|
_textPort.bltPixels(
|
|
BulletImage, 0, 0,
|
|
_textPort._penPos.x, _textPort._penPos.y + 1,
|
|
BulletImage.size.x, BulletImage.size.y);
|
|
|
|
_textPort.move(bulletWidth, 0);
|
|
}
|
|
|
|
// Compute how much of this button is on this line.
|
|
dChars = MIN(lineChars, buttonChars);
|
|
|
|
// Draw however much of this button is on this line.
|
|
_textPort.drawText(lineText, dChars);
|
|
|
|
// Move forward by dChars
|
|
lineChars -= dChars;
|
|
buttonChars -= dChars;
|
|
lineText += dChars;
|
|
}
|
|
|
|
y += _textPort._font->height + lineLeading;
|
|
}
|
|
|
|
if (speechButtonCount > 0) {
|
|
// REM: Also set pointer to arrow shape.
|
|
g_vm->_mouseInfo->setIntent(GrabInfo::kIntWalkTo);
|
|
speakButtonControls->enable(true);
|
|
|
|
speechList.SetLock(false);
|
|
} else {
|
|
// If there is a lock flag on this speech, then LockUI()
|
|
speechList.SetLock(_speechFlags & kSpLock);
|
|
}
|
|
|
|
if (!(_speechFlags & kSpNoAnimate) && isActor(_objID)) {
|
|
Actor *a = (Actor *)GameObject::objectAddress(_objID);
|
|
|
|
if (!a->isDead() && !a->isMoving()) MotionTask::talk(*a);
|
|
}
|
|
|
|
// speechFinished.set( ticksPerSecond*2 );
|
|
return (true);
|
|
}
|
|
|
|
//This Function Sets Up Width And Height For A Speech
|
|
|
|
void Speech::setWidth() {
|
|
TextSpan speechLineList_[32], // list of speech lines
|
|
speechButtonList_[32]; // list of speech buttons
|
|
int16 speechLineCount_, // count of speech lines
|
|
speechButtonCount_; // count of speech buttons
|
|
|
|
// How many word-wrapped lines does the speech take up if we word-wrap
|
|
// it to the default line width?
|
|
|
|
speechLineCount_ = buttonWrap(speechLineList_,
|
|
speechButtonList_,
|
|
speechButtonCount_,
|
|
_speechBuffer,
|
|
defaultWidth,
|
|
!g_vm->_speechText && (_speechFlags & kSpHasVoice),
|
|
_textPort);
|
|
|
|
// If it's more than 3 lines, then use the max line width.
|
|
|
|
if (speechLineCount_ > 3) {
|
|
speechLineCount_ = buttonWrap(speechLineList_,
|
|
speechButtonList_,
|
|
speechButtonCount_,
|
|
_speechBuffer,
|
|
maxWidth,
|
|
!g_vm->_speechText && (_speechFlags & kSpHasVoice),
|
|
_textPort);
|
|
}
|
|
|
|
|
|
// The actual width of the _bounds is the widest of the lines.
|
|
|
|
_bounds.width = 0;
|
|
for (int i = 0; i < speechLineCount_; i++) {
|
|
_bounds.width = MAX(_bounds.width, speechLineList_[i].pixelWidth);
|
|
}
|
|
_bounds.width += outlineWidth * 2 + 4; // Some padding just in case.
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Calculate the position of the speech, emanating from the actor.
|
|
|
|
bool Speech::calcPosition(StaticPoint16 &p) {
|
|
GameObject *obj = GameObject::objectAddress(_objID);
|
|
TilePoint tp = obj->getWorldLocation();
|
|
|
|
if (!isVisible(obj)) return false;
|
|
|
|
TileToScreenCoords(tp, p);
|
|
|
|
p.x = clamp(8,
|
|
p.x - _bounds.width / 2,
|
|
8 + maxWidth - _bounds.width);
|
|
|
|
p.y = clamp(kTileRectY + 8,
|
|
p.y - (_bounds.height + actorHeight),
|
|
kTileRectHeight - 50 - _bounds.height);
|
|
|
|
return true;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Draw the text on the back buffer
|
|
|
|
bool Speech::displayText() {
|
|
StaticPoint16 p;
|
|
|
|
// If there are button in the speech, then don't scroll the
|
|
// speech along with the display. Otherwise, calculate the
|
|
// position from the actor.
|
|
if (speechButtonCount > 0)
|
|
p = initialSpeechPosition;
|
|
else if (!calcPosition(p))
|
|
return false;
|
|
|
|
// Blit to the port
|
|
g_vm->_backPort.setMode(kDrawModeMatte);
|
|
g_vm->_backPort.bltPixels(_speechImage,
|
|
0, 0,
|
|
p.x + fineScroll.x,
|
|
p.y + fineScroll.y,
|
|
_bounds.width, _bounds.height);
|
|
|
|
return true;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Dispose of this speech object. If this is the one being displayed,
|
|
// then dealloc the speech image
|
|
|
|
void Speech::dispose() {
|
|
if (speechList.currentActive() == this) {
|
|
// Actor *a = (Actor *)sp->obj;
|
|
// a->animationFlags |= kAnimateFinished;
|
|
// a->setAction( kActionStand, kAnimateRandom );
|
|
|
|
if (!longEnough())
|
|
playVoice(0);
|
|
// Wake up the thread, and return the # of the button
|
|
// that was selected
|
|
wakeUpThread(_thread, _selectedButton);
|
|
|
|
// De-allocate the speech data
|
|
delete[] _speechImage._data;
|
|
_speechImage._data = nullptr;
|
|
|
|
// Clear the number of active buttons
|
|
speechLineCount = speechButtonCount = 0;
|
|
speakButtonControls->enable(false);
|
|
|
|
if (!(_speechFlags & kSpNoAnimate) && isActor(_objID)) {
|
|
Actor *a = (Actor *)GameObject::objectAddress(_objID);
|
|
|
|
if (a->_moveTask)
|
|
a->_moveTask->finishTalking();
|
|
}
|
|
} else wakeUpThread(_thread, 0);
|
|
|
|
GameObject *obj = GameObject::objectAddress(_objID);
|
|
|
|
debugC(1, kDebugTasks, "Speech: Disposing %p for %p (%s) (total = %d)'", (void *)this, (void *)obj, obj->objName(), speechList.speechCount());
|
|
|
|
remove();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Render the speech object at the head of the speech queue.
|
|
|
|
void updateSpeech() {
|
|
Speech *sp;
|
|
|
|
// if there is a speech object
|
|
if ((sp = speechList.currentActive()) != nullptr) {
|
|
// If there is no bitmap, then set one up.
|
|
if (!(sp->_speechFlags & Speech::kSpActive)) {
|
|
sp->setupActive();
|
|
|
|
// If speech failed to set up, then skip it
|
|
if (sp->_speechImage._data == nullptr) {
|
|
sp->dispose();
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Draw the speech bitmap
|
|
sp->displayText();
|
|
|
|
// If this speech has timed-out, then dispose of it.
|
|
|
|
|
|
if (sp->longEnough() &&
|
|
(speechButtonCount == 0 || sp->_selectedButton != 0))
|
|
sp->dispose();
|
|
} else speechList.SetLock(false);
|
|
}
|
|
|
|
bool Speech::longEnough() {
|
|
if (_speechFlags & kSpHasVoice)
|
|
return (!stillDoingVoice(_sampleID));
|
|
else
|
|
return (_selectedButton != 0 || speechFinished.check());
|
|
}
|
|
|
|
// Gets rid of the current speech
|
|
|
|
void Speech::abortSpeech() {
|
|
// Start by displaying first frame straight off, no delay
|
|
speechFinished.set(0);
|
|
if (_speechFlags & kSpHasVoice) {
|
|
PlayVoice(nullptr);
|
|
}
|
|
}
|
|
|
|
void abortSpeech() {
|
|
if (speechList.currentActive()) speechList.currentActive()->abortSpeech();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Delete all speeches relating to a particular actor
|
|
|
|
void deleteSpeech(ObjectID id) { // voice sound sample ID
|
|
Speech *sp;
|
|
|
|
while ((sp = speechList.findSpeech(id)) != nullptr) sp->dispose();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// This routine does a word-wrap on the input text, and also checks for
|
|
// the '@' symbol to see if there are any embedded buttons in the text.
|
|
|
|
int16 buttonWrap(
|
|
TextSpan *lineList, // indicates where line breaks are
|
|
TextSpan *buttonList, // indicates where button breaks are
|
|
int16 &buttonCount, // returns number of buttons
|
|
char *text, // text to wrap
|
|
int16 width, // width of text
|
|
int16 supressText,
|
|
gPort &textPort) {
|
|
int16 i, // loop counter
|
|
line_start, // start of current line
|
|
last_space, // last space encountered
|
|
last_space_pixels = 0, // pixel pos of last space
|
|
charPixels, // pixel length of character
|
|
linePixels, // pixels in current line
|
|
buttonPixels, // pixels in current button
|
|
buttonChars, // char count of current button
|
|
lineCount = 0; // number of lines
|
|
|
|
// If we are not showing the text of the speech, skip over all text
|
|
// until we come to the first button definition.
|
|
if (supressText) {
|
|
while (*text && *text != '@') text++;
|
|
}
|
|
|
|
lineList->text = text; // set ptr to 1st line
|
|
|
|
last_space = -1; // no spaces to word-wrap yet
|
|
line_start = 0; // start index of 1st line
|
|
linePixels = 0; // no pixels counted yet
|
|
|
|
width -= outlineWidth * 2; // compensate for size of outline
|
|
|
|
// For each character in the string, check for word wrap
|
|
|
|
for (i = 0; ; i++) {
|
|
uint8 c = text[i];
|
|
|
|
// REM: Translate from foreign character set if needed...
|
|
// c = TranslationTable[c];
|
|
|
|
// If deliberate end of line
|
|
if (c == '\n' || c == '\r' || c == '\0') {
|
|
lineList->charWidth = i - line_start;
|
|
lineList->pixelWidth = linePixels;
|
|
lineList++;
|
|
lineCount++;
|
|
|
|
line_start = i + 1;
|
|
|
|
if (c == '\0') break;
|
|
|
|
lineList->text = &text[line_start];
|
|
|
|
last_space = -1;
|
|
linePixels = 0;
|
|
continue;
|
|
} else if (c == '@') { // button indicator...
|
|
// Set width of 'bullet' symbol.
|
|
charPixels = bulletWidth;
|
|
|
|
} else { // Any other character
|
|
// if it's a space, save the word wrap position.
|
|
if (c == ' ') {
|
|
last_space = i;
|
|
last_space_pixels = linePixels;
|
|
}
|
|
|
|
// Add to pixel length
|
|
charPixels
|
|
= textPort._font->charKern[c]
|
|
+ textPort._font->charSpace[c];
|
|
}
|
|
|
|
linePixels += charPixels;
|
|
|
|
// If pixel runs off end of line
|
|
if (linePixels > width && last_space > 0) {
|
|
lineList->charWidth = last_space - line_start;
|
|
lineList->pixelWidth = last_space_pixels;
|
|
lineList++;
|
|
lineCount++;
|
|
|
|
line_start = last_space + 1;
|
|
|
|
lineList->text = &text[line_start];
|
|
|
|
last_space = -1;
|
|
linePixels = 0;
|
|
|
|
i = line_start - 1;
|
|
}
|
|
}
|
|
|
|
buttonCount = 0; // assume zero buttons
|
|
buttonPixels = 0; // no pixels counted yet
|
|
buttonChars = 0; // no chars counted yet
|
|
buttonList->text = text; // set ptr to 1st button
|
|
lineList -= lineCount; // reset line list
|
|
|
|
// For each line, look for button markers
|
|
|
|
for (int l = 0; l < lineCount; l++, lineList++) {
|
|
for (i = 0; i < lineList->charWidth; i++) {
|
|
uint8 c = lineList->text[i];
|
|
|
|
// REM: Translate from foreign character set if needed...
|
|
// c = TranslationTable[c];
|
|
|
|
if (c == '@') { // button indicator...
|
|
// A new button
|
|
buttonList->charWidth = buttonChars;
|
|
buttonList->pixelWidth = buttonPixels;
|
|
|
|
buttonPixels = 0;
|
|
buttonChars = 0;
|
|
buttonList++;
|
|
buttonCount++;
|
|
buttonList->text = text; // set ptr to 1st button
|
|
|
|
// Set width of 'bullet' symbol.
|
|
charPixels = bulletWidth;
|
|
|
|
} else { // Any other character
|
|
// Add to pixel length
|
|
charPixels
|
|
= textPort._font->charKern[c]
|
|
+ textPort._font->charSpace[c];
|
|
}
|
|
|
|
buttonPixels += charPixels;
|
|
buttonChars++;
|
|
}
|
|
}
|
|
|
|
// Clean up the final button
|
|
buttonList->charWidth = buttonChars;
|
|
buttonList->pixelWidth = buttonPixels;
|
|
|
|
return lineCount;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Given the original word-wrap info, determines which button (if any)
|
|
// was clicked.
|
|
|
|
int16 pickButton(
|
|
Point16 &pt,
|
|
TextSpan *lineList, // indicates where line breaks are
|
|
int16 numLines, // number of line breaks
|
|
TextSpan *buttonList, // indicates where button breaks are
|
|
int16 buttonCount, // number of buttons
|
|
int16 width,
|
|
gPort textPort) { // width of rectangle
|
|
int16 pickLine,
|
|
pickPixels = 0,
|
|
centerWidth;
|
|
|
|
if (pt.y < 0 // picked off top edge
|
|
|| pt.x < 0 // picked off left edge
|
|
|| buttonCount < 1) // no buttons defined
|
|
return 0;
|
|
|
|
pickLine = pt.y / (textPort._font->height + lineLeading);
|
|
if (pickLine >= numLines) return 0;
|
|
|
|
// Strange algorithm:
|
|
//
|
|
// When we first built these data structures, we took a continuous
|
|
// string of text and broke it up into several lines, and also
|
|
// broke it up into several buttons. Each of these has a count
|
|
// of how many pixels wide it was.
|
|
//
|
|
// Now, consider if we were to lay each of those lines end
|
|
// to end as though we were reconstructing the original non-
|
|
// wrapped text string. And suppose the pick-point were carried
|
|
// along with the line, so that now our 2-d pixel point is
|
|
// now a 1-d pixel offset into the line.
|
|
//
|
|
// We can do this by adding the length of each previous line
|
|
// to the pixel offset, and subtracting the margin space used
|
|
// for centering.
|
|
|
|
for (int i = 0; i < pickLine; i++) {
|
|
pickPixels += lineList[i].pixelWidth;
|
|
}
|
|
|
|
centerWidth = (width - lineList[pickLine].pixelWidth) / 2;
|
|
|
|
// Return 0 if mouse off left or right edge of text.
|
|
if (pt.x < centerWidth || pt.x > width - centerWidth) return 0;
|
|
|
|
pickPixels += pt.x - (width - lineList[pickLine].pixelWidth) / 2;
|
|
|
|
// Now, we lay all the buttons end to end in a similar fashion,
|
|
// and determine which button the pick point fell into, in a
|
|
// simple 1-d comparison.
|
|
|
|
for (int j = 0; j <= buttonCount; j++) {
|
|
pickPixels -= buttonList[j].pixelWidth;
|
|
if (pickPixels < 0) return j;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
bool isVisible(GameObject *obj) {
|
|
|
|
TilePoint tp = obj->getWorldLocation();
|
|
Point16 p, vp;
|
|
TileToScreenCoords(tp, p);
|
|
|
|
//For Determining If Object Is Being Displayed
|
|
//Could We Just Check Display List ???
|
|
int16 distanceX, distanceY;
|
|
int16 viewSizeY = kTileRectHeight;
|
|
int16 viewSizeX = kTileRectWidth;
|
|
|
|
//I Figure This Differently Than In Dispnode
|
|
int16 loadDistX = viewSizeX / 2;
|
|
int16 loadDistY = viewSizeY / 2;
|
|
|
|
TileToScreenCoords(viewCenter, vp);
|
|
|
|
distanceX = ABS(vp.x - p.x);
|
|
distanceY = ABS(vp.y - p.y);
|
|
|
|
if ((distanceY >= loadDistY) ||
|
|
(distanceX >= loadDistX))
|
|
return (false);
|
|
|
|
return (true);
|
|
}
|
|
|
|
/* ===================================================================== *
|
|
SpeechTaskList member functions
|
|
* ===================================================================== */
|
|
|
|
void SpeechTaskList::remove(Speech *p) {
|
|
for (Common::List<Speech *>::iterator it = _list.begin();
|
|
it != _list.end(); ++it) {
|
|
if (p == *it) {
|
|
_list.remove(p);
|
|
break;
|
|
}
|
|
}
|
|
|
|
for (Common::List<Speech *>::iterator it = _inactiveList.begin();
|
|
it != _inactiveList.end(); ++it) {
|
|
if (p == *it) {
|
|
_inactiveList.remove(p);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Initialize the SpeechTaskList
|
|
|
|
SpeechTaskList::SpeechTaskList() {
|
|
_lockFlag = false;
|
|
}
|
|
|
|
SpeechTaskList::SpeechTaskList(Common::InSaveFile *in) {
|
|
int16 count;
|
|
|
|
_lockFlag = false;
|
|
|
|
// Get the speech count
|
|
count = in->readSint16LE();
|
|
debugC(3, kDebugSaveload, "... count = %d", count);
|
|
|
|
// Restore the speeches
|
|
for (int i = 0; i < count; i++) {
|
|
Speech *sp = new Speech;
|
|
assert(sp != nullptr);
|
|
debugC(3, kDebugSaveload, "Loading Speech %d", i++);
|
|
|
|
_inactiveList.push_back(sp);
|
|
sp->read(in);
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Return the number of bytes needed to archive the speech tasks
|
|
|
|
int32 SpeechTaskList::archiveSize() {
|
|
int32 size = 0;
|
|
|
|
size += sizeof(int16); // Speech count
|
|
|
|
for (Common::List<Speech *>::iterator it = _list.begin();
|
|
it != _list.end(); ++it) {
|
|
size += (*it)->archiveSize();
|
|
}
|
|
|
|
for (Common::List<Speech *>::iterator it = _inactiveList.begin();
|
|
it != _inactiveList.end(); ++it) {
|
|
size += (*it)->archiveSize();
|
|
}
|
|
|
|
return size;
|
|
}
|
|
|
|
void SpeechTaskList::write(Common::MemoryWriteStreamDynamic *out) {
|
|
int i = 0;
|
|
int16 count = 0;
|
|
|
|
count += _list.size() + _inactiveList.size();
|
|
|
|
// Store speech count
|
|
out->writeSint16LE(count);
|
|
debugC(3, kDebugSaveload, "... count = %d", count);
|
|
|
|
// Store active speeches
|
|
for (Common::List<Speech *>::iterator it = _list.begin();
|
|
it != _list.end(); ++it) {
|
|
debugC(3, kDebugSaveload, "Saving Speech %d (active)", i++);
|
|
(*it)->write(out);
|
|
}
|
|
|
|
// Store inactive speeches
|
|
for (Common::List<Speech *>::iterator it = _inactiveList.begin();
|
|
it != _inactiveList.end(); ++it) {
|
|
debugC(3, kDebugSaveload, "Saving Speech %d (inactive)", i++);
|
|
(*it)->write(out);
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Cleanup the speech tasks
|
|
|
|
void SpeechTaskList::cleanup() {
|
|
for (Common::List<Speech *>::iterator it = speechList._list.begin();
|
|
it != speechList._list.end(); ++it) {
|
|
delete *it;
|
|
}
|
|
|
|
for (Common::List<Speech *>::iterator it = speechList._inactiveList.begin();
|
|
it != speechList._inactiveList.end(); ++it) {
|
|
delete *it;
|
|
}
|
|
|
|
_list.clear();
|
|
_inactiveList.clear();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Search for a speech task associated with a particular GameObject.
|
|
|
|
Speech *SpeechTaskList::findSpeech(ObjectID id) {
|
|
for (Common::List<Speech *>::iterator it = speechList._inactiveList.begin();
|
|
it != speechList._inactiveList.end(); ++it) {
|
|
if ((*it)->_objID == id)
|
|
return *it;
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Get a new speech task, if there is one available, and initialize it.
|
|
|
|
Speech *SpeechTaskList::newTask(ObjectID id, uint16 flags) {
|
|
Speech *sp;
|
|
GameObject *obj = GameObject::objectAddress(id);
|
|
|
|
// Actors cannot speak if not in the world
|
|
if (obj->world() != currentWorld) return nullptr;
|
|
|
|
if (speechCount() >= MAX_SPEECH_PTRS) {
|
|
warning("Too many speech tasks: > %d", MAX_SPEECH_PTRS);
|
|
return nullptr;
|
|
}
|
|
|
|
sp = new Speech;
|
|
#if DEBUG
|
|
if (sp == NULL) fatal("Ran out of Speech Tasks, Object = %s\n", obj->objName());
|
|
#endif
|
|
if (sp == nullptr) return nullptr;
|
|
|
|
debugC(1, kDebugTasks, "Speech: New Task: %p for %p (%s) (flags = %d) (total = %d)", (void *)sp, (void *)obj, obj->objName(), flags, speechCount());
|
|
|
|
sp->_sampleCount = sp->_charCount = 0;
|
|
sp->_objID = id;
|
|
sp->_speechFlags = flags & (Speech::kSpNoAnimate | Speech::kSpLock);
|
|
sp->_outlineColor = 15 + 9;
|
|
sp->_thread = NoThread;
|
|
sp->_selectedButton = 0;
|
|
|
|
// Set the pen color of the speech based on the actor
|
|
if (isActor(id)) {
|
|
Actor *a = (Actor *)obj;
|
|
|
|
// If actor has color table loaded, then get the speech
|
|
// color for this particular color scheme; else use a
|
|
// default color.
|
|
if (a == getCenterActor()) sp->_penColor = 3 + 9 /* 1 */;
|
|
else if (a->_appearance
|
|
&& a->_appearance->_schemeList) {
|
|
sp->_penColor =
|
|
a->_appearance->_schemeList->_schemes[a->_colorScheme]->speechColor + 9;
|
|
} else sp->_penColor = 4 + 9;
|
|
} else {
|
|
sp->_penColor = 4 + 9;
|
|
}
|
|
|
|
_inactiveList.push_back(sp);
|
|
return sp;
|
|
}
|
|
|
|
void SpeechTaskList::SetLock(int newState) {
|
|
if (newState && _lockFlag == false) {
|
|
noStickyMap();
|
|
LockUI(true);
|
|
_lockFlag = true;
|
|
} else if (_lockFlag && newState == false) {
|
|
LockUI(false);
|
|
_lockFlag = false;
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// When a speech task is finished, call this function to delete it.
|
|
|
|
void Speech::remove() {
|
|
speechList.remove(this);
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// AppFunc for handling clicks on speech
|
|
|
|
int16 pickSpeechButton(Point16 mouse, int16 size, gPort &textPort) {
|
|
Point16 p = mouse - initialSpeechPosition;
|
|
|
|
p.x -= kTileRectX;
|
|
p.y -= kTileRectY;
|
|
|
|
return pickButton(p,
|
|
speechLineList, speechLineCount,
|
|
speechButtonList, speechButtonCount,
|
|
size,
|
|
textPort);
|
|
}
|
|
|
|
APPFUNC(cmdClickSpeech) {
|
|
Speech *sp;
|
|
|
|
switch (ev.eventType) {
|
|
case kEventMouseMove:
|
|
case kEventMouseDrag:
|
|
|
|
g_vm->_mouseInfo->setDoable(Rect16(kTileRectX, kTileRectY, kTileRectWidth, kTileRectHeight).ptInside(ev.mouse));
|
|
break;
|
|
|
|
case kEventMouseDown:
|
|
|
|
if ((sp = speechList.currentActive()) != nullptr) {
|
|
sp->_selectedButton = pickSpeechButton(ev.mouse, sp->_speechImage._size.x, sp->_textPort);
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* ===================================================================== *
|
|
SpeechTask management functions
|
|
* ===================================================================== */
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Initialize the speech task list
|
|
|
|
void initSpeechTasks() {
|
|
// Simply call the SpeechTaskList default constructor
|
|
new (&speechList) SpeechTaskList;
|
|
}
|
|
|
|
void saveSpeechTasks(Common::OutSaveFile *outS) {
|
|
debugC(2, kDebugSaveload, "Saving Speech Tasks");
|
|
|
|
outS->write("SPCH", 4);
|
|
CHUNK_BEGIN;
|
|
speechList.write(out);
|
|
CHUNK_END;
|
|
}
|
|
|
|
void loadSpeechTasks(Common::InSaveFile *in, int32 chunkSize) {
|
|
debugC(2, kDebugSaveload, "Loading Speech Tasks");
|
|
|
|
// If there is no saved data, simply call the default constructor
|
|
if (chunkSize == 0) {
|
|
new (&speechList) SpeechTaskList;
|
|
return;
|
|
}
|
|
|
|
// Reconstruct stackList from archived data
|
|
new (&speechList) SpeechTaskList(in);
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Cleanup the speech task list
|
|
|
|
void cleanupSpeechTasks() {
|
|
// Call speechList's cleanup() function
|
|
speechList.cleanup();
|
|
}
|
|
|
|
} // end of namespace Saga2
|