Files
2026-02-02 04:50:13 +01:00

451 lines
13 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
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
*/
#include "common/events.h"
#include "common/stream.h"
#include "director/director.h"
#include "director/cast.h"
#include "director/movie.h"
#include "director/castmember/castmember.h"
#include "director/lingo/lingo-the.h"
#include "director/util.h"
namespace Director {
void EditInfo::read(Common::ReadStreamEndian *stream) {
rect = Movie::readRect(*stream);
selStart = stream->readUint32();
selEnd = stream->readUint32();
version = stream->readByte();
rulerFlag = stream->readByte();
// We're ignoring 2 bytes here
valid = true;
debugC(3, kDebugLoading, " EditInfo: rect: [%s], selStart: %d, selEnd: %d, version: %d, rulerFlag: %d", rect.toString().c_str(), selStart, selEnd, version, rulerFlag);
}
void EditInfo::write(Common::WriteStream *stream) {
Movie::writeRect(stream, rect);
stream->writeUint32BE(selStart);
stream->writeUint32BE(selEnd);
stream->writeByte(version);
stream->writeByte(rulerFlag);
}
CastMember::CastMember(Cast *cast, uint16 castId, Common::SeekableReadStreamEndian &stream) : Object<CastMember>("CastMember") {
_type = kCastTypeNull;
_cast = cast;
_castId = castId;
_hilite = false;
_purgePriority = 3;
_size = stream.size();
_flags1 = 0;
_loaded = false;
_modified = true;
_isChanged = false;
_needsReload = false;
_objType = kCastMemberObj;
_widget = nullptr;
_erase = false;
_index = -1;
}
CastMember::CastMember(Cast *cast, uint16 castId) : Object<CastMember>("CastMember") {
_type = kCastTypeNull;
_cast = cast;
_castId = castId;
_hilite = false;
_purgePriority = 3;
_size = 0;
_flags1 = 0;
_loaded = false;
_modified = true;
_isChanged = false;
_objType = kCastMemberObj;
_widget = nullptr;
_erase = false;
_index = -1;
}
CastMember *CastMember::duplicate(Cast *cast, uint16 castId) {
warning("CastMember::duplicate(): unsupported cast type %s", castType2str(_type));
return nullptr;
}
void CastMember::setModified(bool modified) {
_modified = modified;
if (modified)
_isChanged = true;
}
Common::Rect CastMember::getBbox() {
Common::Rect result(_initialRect);
Common::Point offset = getRegistrationOffset();
result.moveTo(-offset.x, -offset.y);
return result;
}
Common::Rect CastMember::getBbox(int16 currentWidth, int16 currentHeight) {
Common::Rect result(currentWidth, currentHeight);
Common::Point offset = getRegistrationOffset(currentWidth, currentHeight);
result.moveTo(-offset.x, -offset.y);
return result;
}
bool CastMember::hasProp(const Common::String &propName) {
Common::String fieldName = Common::String::format("%d%s", kTheCast, propName.c_str());
return g_lingo->_theEntityFields.contains(fieldName) && hasField(g_lingo->_theEntityFields[fieldName]->field);
}
Datum CastMember::getProp(const Common::String &propName) {
Common::String fieldName = Common::String::format("%d%s", kTheCast, propName.c_str());
if (g_lingo->_theEntityFields.contains(fieldName)) {
return getField(g_lingo->_theEntityFields[fieldName]->field);
}
warning("CastMember::getProp: unknown property '%s'", propName.c_str());
return Datum();
}
void CastMember::setProp(const Common::String &propName, const Datum &value, bool force) {
Common::String fieldName = Common::String::format("%d%s", kTheCast, propName.c_str());
if (g_lingo->_theEntityFields.contains(fieldName)) {
setField(g_lingo->_theEntityFields[fieldName]->field, value);
return;
}
warning("CastMember::setProp: unknown property '%s'", propName.c_str());
}
bool CastMember::hasField(int field) {
switch (field) {
case kTheBackColor:
case kTheCastLibNum:
case kTheCastType:
case kTheFileName:
case kTheForeColor:
case kTheHeight:
case kTheHilite:
case kTheLoaded:
case kTheMediaReady:
case kTheModified:
case kTheMemberNum:
case kTheName:
case kTheNumber:
case kTheRect:
case kThePreLoad:
case kThePurgePriority:
case kTheScriptText:
case kTheSize:
case kTheType:
case kTheWidth:
return true;
default:
break;
}
return false;
}
Datum CastMember::getField(int field) {
Datum d;
CastMemberInfo *castInfo = _cast->getCastMemberInfo(_castId);
if (!castInfo)
warning("CastMember::getField(): CastMember info for %d not found", _castId);
switch (field) {
case kTheBackColor:
d = (int)getBackColor();
break;
case kTheCastLibNum:
d = _cast->_castLibID;
break;
case kTheCastType:
case kTheType:
d = Common::String(castType2str(_type));
if (g_director->getVersion() >= 500 && _type == kCastText) {
// D5 changes this from "text" to "field"
d = Common::String("field");
}
d.type = SYMBOL;
break;
case kTheFileName:
if (castInfo)
d = Datum(castInfo->directory + g_director->_dirSeparator + castInfo->fileName);
break;
case kTheForeColor:
d = (int)getForeColor();
break;
case kTheHeight:
d = _cast->getCastMemberInitialRect(_castId).height();
break;
case kTheHilite:
d = (int)_hilite;
break;
case kTheLoaded:
d = 1; // Not loaded handled in Lingo::getTheCast
break;
case kTheMediaReady:
d = 1; // Media is always downloaded from internet in ScummVM
break;
case kTheModified:
d = (int)_isChanged;
break;
case kTheName:
if (castInfo)
d = Datum(castInfo->name);
break;
case kTheMemberNum:
d = _castId;
break;
case kTheNumber:
if (g_director->getVersion() >= 500) {
d = CastMemberID(_castId, _cast->_castLibID).toMultiplex();
} else {
d = _castId;
}
break;
case kTheRect:
// not sure get the initial rect would be fine to castmember
d = Datum(_cast->getCastMember(_castId)->_initialRect);
break;
/*
ScummVM does not do preloading so we will always return false here.
simpsonscartoonstudio checks this flag and if set to true does an updateStage
in a loop which causes performance issues loading cartoons.
*/
case kThePreLoad:
d = 0;
break;
case kThePurgePriority:
d = _purgePriority;
break;
case kTheScriptText:
if (castInfo)
d = Datum(castInfo->script);
break;
case kTheSize:
d = (int)_size;
break;
case kTheWidth:
d = _cast->getCastMemberInitialRect(_castId).width();
break;
default:
warning("CastMember::getField(): Unprocessed getting field \"%s\" of cast %d", g_lingo->field2str(field), _castId);
//TODO find out about String fields
}
return d;
}
void CastMember::setField(int field, const Datum &d) {
CastMemberInfo *castInfo = _cast->getCastMemberInfo(_castId);
switch (field) {
case kTheBackColor:
_cast->getCastMember(_castId)->setBackColor(d.asInt());
return;
case kTheCastType:
case kTheType:
warning("BUILDBOT: CastMember::setField(): Attempt to set read-only field %s of cast %d", g_lingo->entity2str(field), _castId);
return;
case kTheFileName:
if (!castInfo) {
warning("CastMember::setField(): CastMember info for %d not found", _castId);
return;
} else {
Common::String rawPath = d.asString();
Common::String filename = getFileName(rawPath);
castInfo->fileName = filename;
castInfo->directory = rawPath.substr(0, MAX((uint)0, rawPath.size() - filename.size() - 1));
_needsReload = true;
_modified = true;
}
return;
case kTheForeColor:
_cast->getCastMember(_castId)->setForeColor(d.asInt());
return;
case kTheHeight:
warning("BUILDBOT: CastMember::setField(): Attempt to set read-only field \"%s\" of cast %d", g_lingo->field2str(field), _castId);
return;
case kTheHilite:
_hilite = (bool)d.asInt();
_modified = true;
return;
case kTheName:
if (!castInfo) {
warning("CastMember::setField(): CastMember info for %d not found", _castId);
return;
}
castInfo->name = d.asString();
_cast->rebuildCastNameCache();
return;
case kTheRect:
warning("CastMember::setField(): Attempt to set read-only field \"%s\" of cast %d", g_lingo->field2str(field), _castId);
return;
/*
ScummVM does not do preloading so we will make this a no-op.
*/
case kThePreLoad:
return;
case kThePurgePriority:
_purgePriority = CLIP<int>(d.asInt(), 0, 3);
return;
case kTheScriptText:
if (!castInfo) {
warning("CastMember::setField(): CastMember info for %d not found", _castId);
return;
}
_cast->_lingoArchive->replaceCode(*d.u.s, kCastScript, _castId);
castInfo->script = d.asString();
return;
case kTheWidth:
warning("BUILDBOT: CastMember::setField(): Attempt to set read-only field \"%s\" of cast %d", g_lingo->field2str(field), _castId);
return;
default:
warning("CastMember::setField(): Unprocessed setting field \"%s\" of cast %d", g_lingo->field2str(field), _castId);
}
}
CastMemberInfo *CastMember::getInfo() {
return _cast->getCastMemberInfo(_castId);
}
void CastMember::load() {
if (_loaded)
return;
_loaded = true;
}
void CastMember::unload() {
if (!_loaded)
return;
_loaded = false;
}
// Cast members have two types of "information", one is _data_ and the other is _info_
// _data_ is the actual "data" (e.g. the pixel image data of a bitmap, sound data of a sound cast member)
// Whereas _info_ is metadata (size, name, flags, etc.)
// Some cast members have their _data_ as well as _info_ in this very 'CASt' resource, e.g. TextCastMember
// Whereas some other have their _info_ in a 'CASt' resource and _data_ in a dedicated resource (e.g. PaletteCastMember has 'CLUT' resource)
uint32 CastMember::writeCAStResource(Common::SeekableWriteStream *writeStream) {
uint32 castResourceSize = getCastResourceSize();
writeStream->writeUint32LE(MKTAG('C', 'A', 'S', 't'));
writeStream->writeUint32LE(castResourceSize); // this is excluding the 'CASt' header and the size itself (- 8 bytes)
uint32 castDataToWrite = getCastDataSize();
uint32 castInfoToWrite = getCastInfoSize();
if (_cast->_version >= kFileVer400 && _cast->_version < kFileVer500) {
writeStream->writeUint16BE(castDataToWrite);
writeStream->writeUint32BE(castInfoToWrite);
writeStream->writeByte((uint8)_type);
castDataToWrite -= 1;
if (_flags1 != 0xFF) { // In case of TextCastMember, this should be true
writeStream->writeByte(_flags1);
castDataToWrite -= 1;
}
// For cast members with dedicated resources for data, the castDataToWrite is zero
if (castDataToWrite) {
writeCastData(writeStream);
}
if (castInfoToWrite) {
_cast->writeCastInfo(writeStream, _castId);
}
} else if (_cast->_version >= kFileVer500 && _cast->_version < kFileVer600) {
writeStream->writeUint32BE((uint32)_type);
writeStream->writeUint32BE(castInfoToWrite);
writeStream->writeUint32BE(castDataToWrite);
if (castInfoToWrite) {
_cast->writeCastInfo(writeStream, _castId);
}
// For cast members with dedicated resources for data, the castDataToWrite is zero
if (castDataToWrite) {
writeCastData(writeStream);
}
}
return 0;
}
// This is the data that is inside the 'CASt' resource
// These functions (getCastDataSize() and writeCastData() default implementations, are not supposed to be called
// If the data is modified in the cast member, we implement a custom getCastDataSize() and writeCastData() for that member
// If it is not modified, then we write it as it is from the original source in the overridden
// writeCAStResource(Common::MemoryWriteStream, uint32, uint32) function which doesn't call these default functions
uint32 CastMember::getCastDataSize() {
warning("CastMember::getDataSize(): Defualt implementation of 'CASt' resource data size");
return _castDataSize;
}
void CastMember::writeCastData(Common::SeekableWriteStream *writeStream) {
warning("CastMember::getDataSize(): Defualt implementation of 'CASt' resource data");
if (_cast->_version >= kFileVer400 && _cast->_version < kFileVer500) {
if (_flags1 != 0xFF) {
writeStream->write(0, _castDataSize - 2);
} else {
writeStream->write(0, _castDataSize - 1);
}
} else {
writeStream->write(0, _castDataSize);
}
}
// This is the info that is inside the 'CASt' resource
uint32 CastMember::getCastInfoSize() {
return _cast->getCastInfoSize(_castId);
}
// getCastResourceSize only returns the size of the resource without the header
uint32 CastMember::getCastResourceSize() {
uint32 headerSize = 0;
if (_cast->_version >= kFileVer400 && _cast->_version < kFileVer500) {
// Header size for director version 4
headerSize = 7; // see Cast::loadCastData() for director version 4
if (_flags1 != 0xFF) {
headerSize += 1;
}
} else if (_cast->_version >= kFileVer500 && _cast->_version < kFileVer600) {
// Header size for director version 5
headerSize = 12; // See Cast::loadCastData() for director version 5
}
return headerSize + getCastInfoSize() + getCastDataSize();
}
} // End of namespace Director