scummvm/engines/macventure/gui.h

540 lines
12 KiB
C
Raw Normal View History

2016-06-08 11:02:21 +02: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.
*
*/
#ifndef MACVENTURE_GUI_H
#define MACVENTURE_GUI_H
#include "graphics/macgui/macwindowmanager.h"
2016-06-08 16:07:53 +02:00
#include "graphics/macgui/macwindow.h"
#include "graphics/macgui/macmenu.h"
2016-06-08 11:02:21 +02:00
#include "graphics/font.h"
2016-06-21 13:22:56 +02:00
#include "macventure/container.h"
#include "macventure/image.h"
2016-06-08 11:02:21 +02:00
namespace MacVenture {
using namespace Graphics::MacGUIConstants;
2016-06-09 11:31:19 +02:00
using namespace Graphics::MacWindowConstants;
2016-06-08 16:07:53 +02:00
class MacVentureEngine;
typedef uint32 ObjID;
2016-06-08 11:02:21 +02:00
2016-07-02 17:11:42 +02:00
class Cursor;
2016-07-04 11:45:47 +02:00
class ConsoleText;
class CommandButton;
class ImageAsset;
2016-06-09 11:31:19 +02:00
enum MenuAction {
kMenuActionAbout,
kMenuActionNew,
kMenuActionOpen,
kMenuActionSave,
kMenuActionSaveAs,
kMenuActionQuit,
kMenuActionUndo,
kMenuActionCut,
kMenuActionCopy,
kMenuActionPaste,
kMenuActionClear,
kMenuActionCleanUp,
kMenuActionMessUp,
kMenuActionCommand
};
//} using namespace MacVentureMenuActions;
2016-06-12 20:22:01 +02:00
enum WindowReference {
kNoWindow = 0,
kInventoryStart = 1,
2016-06-12 20:22:01 +02:00
kCommandsWindow = 0x80,
kMainGameWindow = 0x81,
kOutConsoleWindow = 0x82,
kSelfWindow = 0x83,
kExitsWindow = 0x84,
kDiplomaWindow = 0x85
2016-06-12 20:22:01 +02:00
};
enum MVWindowType {
kDocument = 0x00,
kDBox = 0x01,
kPlainDBox = 0x02,
kAltBox = 0x03,
kNoGrowDoc = 0x04,
kMovableDBox = 0x05,
kZoomDoc = 0x08,
kZoomNoGrow = 0x0c,
kRDoc16 = 0x10,
kRDoc4 = 0x12,
kRDoc6 = 0x14,
kRDoc10 = 0x16
};
struct DrawableObject {
ObjID obj;
byte mode;
DrawableObject(ObjID id, byte md) {
obj = id;
mode = md;
}
};
2016-06-12 20:22:01 +02:00
struct WindowData {
Common::Rect bounds;
MVWindowType type;
2016-06-24 21:00:06 +02:00
ObjID objRef;
2016-06-12 20:22:01 +02:00
uint16 visible;
uint16 hasCloseBox;
WindowReference refcon;
uint8 titleLength;
Common::String title;
Common::Array<DrawableObject> children;
bool updateScroll;
2016-06-12 20:22:01 +02:00
};
2016-07-09 19:46:10 +02:00
enum ControlType { // HACK, should correspond exactly with the types of controls (sliders etc)
2016-06-12 20:22:01 +02:00
kControlExitBox = 0,
kControlExamine = 1,
kControlOpen = 2,
kControlClose = 3,
kControlSpeak = 4,
kControlOperate = 5,
kControlGo = 6,
kControlHit = 7,
kControlConsume = 8,
kControlClickToContinue = 9
2016-06-12 20:22:01 +02:00
};
2016-07-05 13:06:51 +02:00
enum ControlAction { // HACK, figure out a way to put it in engine
kNoCommand = 0,
kStartOrResume = 1,
kClose = 2,
kTick = 3,
kActivateObject = 4,
kMoveObject = 5,
kConsume = 6,
kExamine = 7,
kGo = 8,
kHit = 9,
kOpen = 10,
kOperate = 11,
kSpeak = 12,
kBabble = 13,
kTargetName = 14,
kDebugObject = 15,
kClickToContinue = 16
};
2016-06-12 20:22:01 +02:00
struct ControlData {
Common::Rect bounds;
uint16 scrollValue;
uint8 visible;
uint16 scrollMax;
uint16 scrollMin;
uint16 cdef;
2016-07-09 19:46:10 +02:00
ControlAction refcon;
2016-07-05 13:06:51 +02:00
ControlType type;
2016-06-12 20:22:01 +02:00
uint8 titleLength;
char* title;
2016-06-12 22:09:06 +02:00
uint16 border;
2016-06-12 20:22:01 +02:00
};
struct BorderBounds {
uint16 leftOffset;
uint16 topOffset;
uint16 rightOffset;
uint16 bottomOffset;
BorderBounds(uint16 l, uint16 t, uint16 r, uint16 b) : leftOffset(l), topOffset(t), rightOffset(r), bottomOffset(b) {}
};
struct DraggedObj {
ObjID id;
Common::Point pos;
Common::Point mouseOffset;
Common::Point startPos;
WindowReference startWin;
bool hasMoved;
};
2016-07-02 17:11:42 +02:00
enum CursorState {
// HACK, I should define a proper FSM for this
kCursorIdle,
kCursorSingleClick, // Triggered when mouse goes up
kCursorSingleClickAwait, // Triggered when we are in single click and mouse goes down
kCursorSingleClickTrap, // Trap state, for when we are in await, and the timer goes off
kCursorDoubleClick
};
2016-06-12 20:22:01 +02:00
class Gui {
2016-06-08 11:02:21 +02:00
public:
2016-06-08 17:13:02 +02:00
Gui(MacVentureEngine *engine, Common::MacResManager *resman);
2016-06-08 11:02:21 +02:00
~Gui();
void draw();
2016-06-20 08:59:53 +02:00
void drawMenu();
void drawTitle();
2016-06-30 08:41:25 +02:00
void clearControls();
2016-06-08 17:13:02 +02:00
bool processEvent(Common::Event &event);
2016-06-09 11:31:19 +02:00
void handleMenuAction(MenuAction action);
void updateWindow(WindowReference winID, bool containerOpen);
void invertWindowColors(WindowReference winID);
2016-06-24 21:00:06 +02:00
WindowReference createInventoryWindow(ObjID objRef);
bool tryCloseWindow(WindowReference winID);
2016-07-12 11:49:05 +02:00
Common::Point getObjMeasures(ObjID obj);
2016-06-24 21:00:06 +02:00
WindowReference getObjWindow(ObjID objID);
WindowReference findObjWindow(ObjID objID);
// Event processors
2016-06-11 23:57:35 +02:00
bool processCommandEvents(WindowClick click, Common::Event &event);
bool processMainGameEvents(WindowClick click, Common::Event &event);
bool processOutConsoleEvents(WindowClick click, Common::Event &event);
bool processSelfEvents(WindowClick click, Common::Event &event);
bool processExitsEvents(WindowClick click, Common::Event &event);
bool processDiplomaEvents(WindowClick click, Common::Event &event);
2016-06-20 08:59:53 +02:00
bool processInventoryEvents(WindowClick click, Common::Event &event);
2016-06-08 11:02:21 +02:00
2016-07-02 17:11:42 +02:00
void processCursorTick();
2016-06-24 21:00:06 +02:00
//bool processClickObject(ObjID obj, WindowReference win, Common::Event event, bool canDrag);
2016-06-12 22:09:06 +02:00
const WindowData& getWindowData(WindowReference reference);
const Graphics::Font& getCurrentFont();
2016-07-02 17:11:42 +02:00
// Clicks
void handleSingleClick(Common::Point pos);
void handleDoubleClick(Common::Point pos);
// Modifiers
void bringToFront(WindowReference window);
void setWindowTitle(WindowReference winID, Common::String string);
void updateWindowInfo(WindowReference ref, ObjID objID, const Common::Array<ObjID> &children);
void addChild(WindowReference target, ObjID child);
void removeChild(WindowReference target, ObjID child);
2016-07-09 19:46:10 +02:00
void clearExits();
2016-06-30 08:41:25 +02:00
void unselectExits();
void updateExit(ObjID id);
2016-07-04 11:45:47 +02:00
void printText(const Common::String &text);
2016-06-12 22:09:06 +02:00
// Ugly switches
BorderBounds borderBounds(MVWindowType type);
2016-06-12 22:09:06 +02:00
2016-06-08 11:02:21 +02:00
private: // Attributes
2016-06-08 16:07:53 +02:00
MacVentureEngine *_engine;
2016-06-08 17:13:02 +02:00
Common::MacResManager *_resourceManager;
2016-06-08 16:07:53 +02:00
2016-06-08 11:02:21 +02:00
Graphics::ManagedSurface _screen;
Graphics::MacWindowManager _wm;
2016-06-12 22:09:06 +02:00
Common::List<WindowData> *_windowData;
2016-06-30 08:41:25 +02:00
Common::Array<CommandButton> *_controlData;
Common::Array<CommandButton> *_exitsData;
2016-06-12 22:09:06 +02:00
Graphics::MacWindow *_controlsWindow;
Graphics::MacWindow *_mainGameWindow;
2016-06-08 19:02:15 +02:00
Graphics::MacWindow *_outConsoleWindow;
2016-06-12 22:09:06 +02:00
Graphics::MacWindow *_selfWindow;
Graphics::MacWindow *_exitsWindow;
Graphics::MacWindow *_diplomaWindow;
Common::Array<Graphics::MacWindow*> _inventoryWindows;
2016-06-08 16:07:53 +02:00
Graphics::Menu *_menu;
2016-06-21 13:22:56 +02:00
Container *_graphics;
Common::HashMap<ObjID, ImageAsset*> _assets;
2016-06-21 13:22:56 +02:00
Graphics::ManagedSurface _draggedSurface;
DraggedObj _draggedObj;
2016-07-09 19:46:10 +02:00
2016-07-02 17:11:42 +02:00
Cursor *_cursor;
2016-07-04 11:45:47 +02:00
ConsoleText *_consoleText;
2016-06-08 11:02:21 +02:00
private: // Methods
2016-06-12 22:09:06 +02:00
// Initializers
2016-06-08 11:02:21 +02:00
void initGUI();
2016-06-12 22:09:06 +02:00
void initWindows();
2016-06-25 17:38:15 +02:00
void assignObjReferences(); // Mainly guesswork
2016-06-12 22:09:06 +02:00
// Loaders
2016-06-08 17:13:02 +02:00
bool loadMenus();
bool loadWindows();
bool loadControls();
void loadBorder(Graphics::MacWindow * target, Common::String filename, bool active);
2016-06-21 13:22:56 +02:00
void loadGraphics();
2016-06-08 11:02:21 +02:00
2016-06-12 22:09:06 +02:00
// Drawers
2016-06-13 00:36:24 +02:00
void drawWindows();
2016-06-12 22:09:06 +02:00
void drawCommandsWindow();
2016-06-13 00:36:24 +02:00
void drawMainGameWindow();
void drawSelfWindow();
void drawInventories();
void drawExitsWindow();
2016-07-04 11:45:47 +02:00
void drawConsoleWindow();
2016-06-12 20:22:01 +02:00
void drawDraggedObject();
void drawObjectsInWindow(WindowReference target, Graphics::ManagedSurface *surface);
void drawWindowTitle(WindowReference target, Graphics::ManagedSurface *surface);
void moveDraggedObject(Common::Point target);
// Finders
2016-07-02 17:11:42 +02:00
WindowReference findWindowAtPoint(Common::Point point);
Common::Point getWindowSurfacePos(WindowReference reference);
WindowData& findWindowData(WindowReference reference);
Graphics::MacWindow *findWindow(WindowReference reference);
2016-06-25 22:53:11 +02:00
// Utils
bool canBeSelected(ObjID obj, const Common::Event &event, const Common::Rect &clickRect, WindowReference ref);
void checkSelect(const WindowData &data, const Common::Event &event, const Common::Rect &clickRect, WindowReference ref);
2016-06-25 22:53:11 +02:00
bool isRectInsideObject(Common::Rect target, ObjID obj);
void selectDraggable(ObjID child, WindowReference origin, Common::Point startPos);
2016-07-02 17:11:42 +02:00
void handleDragRelease(Common::Point pos, bool shiftPressed, bool isDoubleClick);
Common::Rect calculateClickRect(Common::Point clickPos, Common::Rect windowBounds);
Common::Point localize(Common::Point point, WindowReference origin, WindowReference target);
2016-06-25 22:53:11 +02:00
2016-06-08 11:02:21 +02:00
};
2016-07-02 17:11:42 +02:00
class Cursor {
enum ClickState {
kCursorIdle = 0,
kCursorSC = 1,
kCursorNoTick = 2,
kCursorSCTrans = 3,
kCursorExecSC = 4,
kCursorExecDC = 5,
kCursorStateCount
};
enum CursorInput { // Columns for the FSM transition table
kTickCol = 0,
kButtonDownCol = 1,
kButtonUpCol = 2,
kCursorInputCount
};
ClickState _transitionTable[kCursorStateCount][kCursorInputCount] = {
/* kCursorIdle */ {kCursorIdle, kCursorIdle, kCursorSC },
/* kCursorSC */ {kCursorExecSC, kCursorSCTrans, kCursorExecDC },
/* IgnoreTick */ {kCursorNoTick, kCursorNoTick, kCursorExecSC },
/* SC Transition */ {kCursorNoTick, kCursorNoTick, kCursorExecDC },
/* Exec SC */ {kCursorIdle, kCursorExecSC, kCursorExecSC }, // Trap state
/* Exec DC */ {kCursorIdle, kCursorExecDC, kCursorExecDC } // Trap state
};
public:
Cursor(Gui *gui) {
_gui = gui;
_state = kCursorIdle;
}
~Cursor() {}
void tick() {
2016-07-09 19:46:10 +02:00
executeState();
2016-07-02 17:11:42 +02:00
changeState(kTickCol);
}
bool processEvent(const Common::Event &event) {
if (event.type == Common::EVENT_MOUSEMOVE) {
_pos = event.mouse;
return true;
}
if (event.type == Common::EVENT_LBUTTONDOWN) {
changeState(kButtonDownCol);
return true;
}
if (event.type == Common::EVENT_LBUTTONUP) {
changeState(kButtonUpCol);
return true;
}
return false;
}
Common::Point getPos() {
return _pos;
}
private:
void changeState(CursorInput input) {
2016-07-05 13:06:51 +02:00
debug(4, "Change cursor state: [%d] -> [%d]", _state, _transitionTable[_state][input]);
2016-07-02 17:11:42 +02:00
_state = _transitionTable[_state][input];
}
void executeState() {
if (_state == kCursorExecSC) {
_gui->handleSingleClick(_pos);
changeState(kTickCol);
} else if (_state == kCursorExecDC) {
_gui->handleDoubleClick(_pos);
changeState(kTickCol);
}
}
private:
Gui *_gui;
Common::Point _pos;
ClickState _state;
};
class CommandButton {
enum {
kCommandsLeftPadding = 0,
kCommandsTopPadding = 0
};
public:
2016-06-17 12:44:52 +02:00
CommandButton() {
_gui = nullptr;
}
CommandButton(ControlData data, Gui *g) {
_data = data;
_gui = g;
_selected = false;
}
~CommandButton() {}
2016-06-11 23:57:35 +02:00
void draw(Graphics::ManagedSurface &surface) const {
uint colorFill = _selected ? kColorBlack : kColorWhite;
uint colorText = _selected ? kColorWhite : kColorBlack;
surface.fillRect(_data.bounds, colorFill);
surface.frameRect(_data.bounds, kColorBlack);
2016-06-30 08:41:25 +02:00
if (_data.titleLength > 0) {
const Graphics::Font &font = _gui->getCurrentFont();
Common::String title(_data.title);
font.drawString(
&surface,
title,
_data.bounds.left,
_data.bounds.top,
_data.bounds.right - _data.bounds.left,
colorText,
Graphics::kTextAlignCenter);
}
}
2016-06-11 23:57:35 +02:00
bool isInsideBounds(const Common::Point point) const {
return _data.bounds.contains(point);
}
2016-06-11 23:57:35 +02:00
const ControlData& getData() const {
return _data;
}
void select() {
_selected = true;
}
void unselect() {
_selected = false;
}
bool isSelected() {
return _selected;
}
private:
bool _selected;
ControlData _data;
Gui *_gui;
};
2016-07-04 11:45:47 +02:00
class ConsoleText {
public:
ConsoleText(Gui *gui) {
_gui = gui;
_lines.push_back("");
}
~ConsoleText() {
}
void printLine(const Common::String &str, int maxW) {
Common::StringArray wrappedLines;
int textW = maxW;
const Graphics::Font *font = &_gui->getCurrentFont();
font->wordWrapText(str, textW, wrappedLines);
if (wrappedLines.empty()) // Sometimes we have empty lines
_lines.push_back("");
for (Common::StringArray::const_iterator j = wrappedLines.begin(); j != wrappedLines.end(); ++j)
_lines.push_back(*j);
updateScroll();
}
void renderInto(Graphics::ManagedSurface *target, uint leftOffset) {
target->fillRect(target->getBounds(), kColorWhite);
const Graphics::Font *font = &_gui->getCurrentFont();
// HACK print the last lines visible (no scroll)
uint y = target->h - font->getFontHeight();
for (uint i = _lines.size() - 1; i != 0; i--) {
font->drawString(target, _lines[i], leftOffset, y, font->getStringWidth(_lines[i]), kColorBlack);
y -= font->getFontHeight();
}
}
void updateScroll() {
// TODO implemebt
}
private:
Gui *_gui;
Common::StringArray _lines;
};
2016-06-08 11:02:21 +02:00
} // End of namespace MacVenture
#endif