/* 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 . * */ #include "ultima/shared/engine/messages.h" #include "ultima/shared/early/ultima_early.h" #include "ultima/shared/early/game_base.h" #include "ultima/shared/gfx/screen.h" #include "ultima/shared/early/game.h" namespace Ultima { namespace Shared { CMessage::CMessage() : BaseObject() { } bool CMessage::execute(TreeItem *target, const ClassDef *classDef, int flags) { // If no target was specified, then there's nothing to do if (!target) return false; bool result = false; TreeItem *item = target; TreeItem *nextItem = nullptr; do { if (flags & MSGFLAG_SCAN) nextItem = item->scan(target); if (!classDef || item->isInstanceOf(*classDef)) { bool handled = perform(item); if (handled) { result = true; if (flags & MSGFLAG_BREAK_IF_HANDLED) return true; } } item = nextItem; } while (nextItem); return result; } bool CMessage::execute(const Common::String &target, const ClassDef *classDef, int flags) { // Scan for the target by name GameBase *game = g_vm->_game; for (TreeItem *treeItem = game; treeItem; treeItem = treeItem->scan(game)) { if (!treeItem->getName().compareToIgnoreCase(target)) return execute(treeItem, classDef, flags); } return false; } const MSGMAP_ENTRY *CMessage::findMapEntry(const TreeItem *treeItem, const ClassDef &classDef) { // Iterate through the class and any parent classes for (const MSGMAP *msgMap = treeItem->getMessageMap(); msgMap->pFnGetBaseMap; msgMap = msgMap->pFnGetBaseMap()) { // Iterate through the map entries for this class for (const MSGMAP_ENTRY *entry = msgMap->lpEntries; entry->_classDef != nullptr; ++entry) { // Check if the class or any of it's ancesotrs is handled by this entry for (ClassDef def = (*entry->_classDef)(); def.hasParent(); def = def.parent()) { if (def == classDef) return entry; } } } return nullptr; } bool CMessage::perform(TreeItem *treeItem) { const MSGMAP_ENTRY *entry = findMapEntry(treeItem, getType()); return entry && (*treeItem.*(entry->_fn))(this); } bool CMessage::supports(const TreeItem *treeItem, const ClassDef &classDef) { return findMapEntry(treeItem, classDef) != nullptr; } bool CMessage::isMouseMsg() const { return dynamic_cast(this) != nullptr; } bool CMessage::isButtonDownMsg() const { return dynamic_cast(this) != nullptr; } bool CMessage::isButtonUpMsg() const { return dynamic_cast(this) != nullptr; } bool CMessage::isMouseMoveMsg() const { return dynamic_cast(this) != nullptr; } bool CMessage::isDoubleClickMsg() const { return dynamic_cast(this) != nullptr; } } // End of namespace Shared } // End of namespace Ultima