2016-02-07 16:22:16 -05: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.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2016-02-25 08:48:21 -05:00
|
|
|
#include "titanic/core/tree_item.h"
|
|
|
|
#include "titanic/core/dont_save_file_item.h"
|
|
|
|
#include "titanic/core/file_item.h"
|
2016-03-12 22:03:01 -05:00
|
|
|
#include "titanic/core/game_object.h"
|
|
|
|
#include "titanic/core/game_object_desc_item.h"
|
|
|
|
#include "titanic/core/link_item.h"
|
2016-04-14 20:02:52 -04:00
|
|
|
#include "titanic/core/mail_man.h"
|
2016-03-08 21:13:24 -05:00
|
|
|
#include "titanic/core/named_item.h"
|
2016-03-12 22:03:01 -05:00
|
|
|
#include "titanic/core/node_item.h"
|
2016-03-17 23:58:24 -04:00
|
|
|
#include "titanic/core/project_item.h"
|
2016-03-12 22:03:01 -05:00
|
|
|
#include "titanic/core/view_item.h"
|
2016-03-13 15:07:27 -04:00
|
|
|
#include "titanic/core/room_item.h"
|
2016-03-24 21:02:25 -04:00
|
|
|
#include "titanic/pet_control/pet_control.h"
|
|
|
|
#include "titanic/game_manager.h"
|
2016-06-25 17:16:07 -04:00
|
|
|
#include "titanic/game/placeholder/place_holder_item.h"
|
2016-02-07 16:22:16 -05:00
|
|
|
|
|
|
|
namespace Titanic {
|
|
|
|
|
2016-06-28 21:18:13 -04:00
|
|
|
EMPTY_MESSAGE_MAP(CTreeItem, CMessageTarget)
|
2016-04-06 22:17:26 -04:00
|
|
|
|
2016-02-18 23:07:01 -05:00
|
|
|
CTreeItem::CTreeItem() : _parent(nullptr), _firstChild(nullptr),
|
|
|
|
_nextSibling(nullptr), _priorSibling(nullptr), _field14(0) {
|
|
|
|
}
|
|
|
|
|
2016-03-13 16:25:25 -04:00
|
|
|
void CTreeItem::dump(int indent) {
|
|
|
|
CString line = dumpItem(indent);
|
|
|
|
debug("%s", line.c_str());
|
|
|
|
|
|
|
|
CTreeItem *item = getFirstChild();
|
|
|
|
while (item) {
|
|
|
|
item->dump(indent + 1);
|
|
|
|
|
|
|
|
item = item->getNextSibling();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CString CTreeItem::dumpItem(int indent) const {
|
|
|
|
CString result;
|
|
|
|
for (int idx = 0; idx < indent; ++idx)
|
|
|
|
result += '\t';
|
|
|
|
result += getType()->_className;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-06-29 19:53:16 -04:00
|
|
|
void CTreeItem::save(SimpleFile *file, int indent) {
|
2016-03-19 09:29:11 -04:00
|
|
|
file->writeNumberLine(0, indent);
|
|
|
|
CMessageTarget::save(file, indent);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CTreeItem::load(SimpleFile *file) {
|
|
|
|
file->readNumber();
|
|
|
|
CMessageTarget::load(file);
|
|
|
|
}
|
|
|
|
|
2016-03-12 22:03:01 -05:00
|
|
|
bool CTreeItem::isFileItem() const {
|
|
|
|
return isInstanceOf(CFileItem::_type);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CTreeItem::isRoomItem() const {
|
|
|
|
return isInstanceOf(CRoomItem::_type);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CTreeItem::isNodeItem() const {
|
|
|
|
return isInstanceOf(CNodeItem::_type);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CTreeItem::isViewItem() const {
|
|
|
|
return isInstanceOf(CViewItem::_type);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CTreeItem::isLinkItem() const {
|
|
|
|
return isInstanceOf(CLinkItem::_type);
|
|
|
|
}
|
|
|
|
|
2016-06-25 17:16:07 -04:00
|
|
|
bool CTreeItem::isPlaceHolderItem() const {
|
|
|
|
return isInstanceOf(CPlaceHolderItem::_type);
|
|
|
|
}
|
|
|
|
|
2016-03-12 22:03:01 -05:00
|
|
|
bool CTreeItem::isNamedItem() const {
|
|
|
|
return isInstanceOf(CNamedItem::_type);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CTreeItem::isGameObject() const {
|
|
|
|
return isInstanceOf(CGameObject::_type);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CTreeItem::isGameObjectDescItem() const {
|
|
|
|
return isInstanceOf(CGameObjectDescItem::_type);
|
|
|
|
}
|
|
|
|
|
2016-03-08 20:33:57 -05:00
|
|
|
CGameManager *CTreeItem::getGameManager() const {
|
2016-02-23 20:59:57 -05:00
|
|
|
return _parent ? _parent->getGameManager() : nullptr;
|
|
|
|
}
|
|
|
|
|
2016-03-17 23:58:24 -04:00
|
|
|
CProjectItem *CTreeItem::getRoot() const {
|
2016-02-23 20:59:57 -05:00
|
|
|
CTreeItem *parent = getParent();
|
|
|
|
|
|
|
|
if (parent) {
|
|
|
|
do {
|
|
|
|
parent = parent->getParent();
|
|
|
|
} while (parent->getParent());
|
|
|
|
}
|
|
|
|
|
2016-03-17 23:58:24 -04:00
|
|
|
return dynamic_cast<CProjectItem *>(parent);
|
2016-02-23 20:59:57 -05:00
|
|
|
}
|
|
|
|
|
2016-02-18 23:07:01 -05:00
|
|
|
CTreeItem *CTreeItem::getLastSibling() {
|
|
|
|
CTreeItem *item = this;
|
|
|
|
while (item->getNextSibling())
|
|
|
|
item = item->getNextSibling();
|
|
|
|
|
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
2016-03-08 20:33:57 -05:00
|
|
|
CTreeItem *CTreeItem::getLastChild() const {
|
2016-02-18 23:07:01 -05:00
|
|
|
if (!_firstChild)
|
|
|
|
return nullptr;
|
|
|
|
return _firstChild->getLastSibling();
|
|
|
|
}
|
|
|
|
|
2016-03-08 20:33:57 -05:00
|
|
|
CTreeItem *CTreeItem::scan(CTreeItem *item) const {
|
2016-03-06 22:57:45 -05:00
|
|
|
if (_firstChild)
|
|
|
|
return _firstChild;
|
|
|
|
|
2016-03-08 20:33:57 -05:00
|
|
|
const CTreeItem *treeItem = this;
|
2016-03-06 22:57:45 -05:00
|
|
|
while (treeItem != item) {
|
|
|
|
if (treeItem->_nextSibling)
|
|
|
|
return treeItem->_nextSibling;
|
|
|
|
|
|
|
|
treeItem = treeItem->_parent;
|
|
|
|
if (!treeItem)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2016-03-12 19:23:00 -05:00
|
|
|
CTreeItem *CTreeItem::findChildInstanceOf(ClassDef *classDef) const {
|
|
|
|
for (CTreeItem *treeItem = _firstChild; treeItem; treeItem = treeItem->getNextSibling()) {
|
|
|
|
if (treeItem->isInstanceOf(classDef))
|
|
|
|
return treeItem;
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
CTreeItem *CTreeItem::findNextInstanceOf(ClassDef *classDef, CTreeItem *startItem) const {
|
|
|
|
CTreeItem *treeItem = startItem ? startItem->getNextSibling() : getFirstChild();
|
|
|
|
|
|
|
|
for (; treeItem; treeItem = treeItem->getNextSibling()) {
|
|
|
|
if (treeItem->isInstanceOf(classDef))
|
|
|
|
return treeItem;
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2016-02-19 19:45:17 -05:00
|
|
|
void CTreeItem::addUnder(CTreeItem *newParent) {
|
|
|
|
if (newParent->_firstChild)
|
2016-03-13 18:00:30 -04:00
|
|
|
addSibling(newParent->_firstChild->getLastSibling());
|
2016-02-19 19:45:17 -05:00
|
|
|
else
|
|
|
|
setParent(newParent);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CTreeItem::setParent(CTreeItem *newParent) {
|
|
|
|
_parent = newParent;
|
|
|
|
_priorSibling = nullptr;
|
|
|
|
_nextSibling = newParent->_firstChild;
|
|
|
|
|
|
|
|
if (newParent->_firstChild)
|
|
|
|
newParent->_firstChild->_priorSibling = this;
|
|
|
|
newParent->_firstChild = this;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CTreeItem::addSibling(CTreeItem *item) {
|
2016-03-29 22:21:55 -04:00
|
|
|
_priorSibling = item;
|
2016-02-19 19:45:17 -05:00
|
|
|
_nextSibling = item->_nextSibling;
|
|
|
|
_parent = item->_parent;
|
|
|
|
|
|
|
|
if (item->_nextSibling)
|
|
|
|
item->_nextSibling->_priorSibling = this;
|
|
|
|
item->_nextSibling = this;
|
|
|
|
}
|
|
|
|
|
2016-04-15 08:01:39 -04:00
|
|
|
void CTreeItem::moveUnder(CTreeItem *newParent) {
|
|
|
|
if (newParent) {
|
|
|
|
detach();
|
|
|
|
addUnder(newParent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-23 20:59:57 -05:00
|
|
|
void CTreeItem::destroyAll() {
|
|
|
|
destroyOthers();
|
|
|
|
detach();
|
|
|
|
delete this;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CTreeItem::destroyOthers() {
|
|
|
|
if (!_firstChild)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
CTreeItem *item = this, *child, *nextSibling;
|
|
|
|
int total = 0;
|
|
|
|
|
|
|
|
do {
|
|
|
|
child = item->_firstChild;
|
|
|
|
nextSibling = item->_nextSibling;
|
|
|
|
|
|
|
|
if (child)
|
|
|
|
total += child->destroyOthers();
|
|
|
|
child->detach();
|
|
|
|
delete child;
|
|
|
|
++total;
|
|
|
|
} while ((item = nextSibling) != nullptr);
|
|
|
|
|
|
|
|
return total;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CTreeItem::detach() {
|
|
|
|
// Delink this item from any prior and/or next siblings
|
|
|
|
if (_priorSibling)
|
|
|
|
_priorSibling->_nextSibling = _nextSibling;
|
|
|
|
if (_nextSibling)
|
|
|
|
_nextSibling->_priorSibling = _priorSibling;
|
|
|
|
|
|
|
|
if (_parent && _parent->_firstChild == this)
|
|
|
|
_parent->_firstChild = _nextSibling;
|
|
|
|
|
|
|
|
_priorSibling = _nextSibling = _parent = nullptr;
|
|
|
|
}
|
2016-02-19 19:45:17 -05:00
|
|
|
|
2016-03-08 21:13:24 -05:00
|
|
|
CNamedItem *CTreeItem::findByName(const CString &name, int maxLen) {
|
|
|
|
CString nameLower = name;
|
|
|
|
nameLower.toLowercase();
|
|
|
|
|
2016-03-17 22:29:16 -04:00
|
|
|
for (CTreeItem *treeItem = this; treeItem; treeItem = treeItem->scan(this)) {
|
2016-03-08 21:13:24 -05:00
|
|
|
CString nodeName = treeItem->getName();
|
|
|
|
nodeName.toLowercase();
|
|
|
|
|
|
|
|
if (maxLen) {
|
|
|
|
if (nodeName.left(maxLen).compareTo(nameLower))
|
|
|
|
return dynamic_cast<CNamedItem *>(treeItem);
|
|
|
|
} else {
|
2016-03-17 22:29:16 -04:00
|
|
|
if (!nodeName.compareTo(nameLower))
|
2016-03-08 21:13:24 -05:00
|
|
|
return dynamic_cast<CNamedItem *>(treeItem);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2016-02-07 16:22:16 -05:00
|
|
|
} // End of namespace Titanic
|