2011-12-17 18:59:59 +00:00
|
|
|
/*
|
|
|
|
* CIntObject.h, part of VCMI engine
|
|
|
|
*
|
|
|
|
* Authors: listed in file AUTHORS in main folder
|
|
|
|
*
|
|
|
|
* License: GNU General Public License v2.0 or later
|
|
|
|
* Full text of license available in license.txt file, in main folder
|
|
|
|
*
|
|
|
|
*/
|
2014-05-23 20:46:54 +04:00
|
|
|
#pragma once
|
|
|
|
|
2014-08-09 15:14:31 +03:00
|
|
|
#include <SDL_events.h>
|
2014-05-23 20:46:54 +04:00
|
|
|
#include "Geometries.h"
|
|
|
|
#include "../Graphics.h"
|
|
|
|
|
|
|
|
struct SDL_Surface;
|
|
|
|
class CGuiHandler;
|
2016-10-16 04:23:38 +03:00
|
|
|
class CPicture;
|
2011-12-17 18:59:59 +00:00
|
|
|
|
2014-07-15 10:14:49 +03:00
|
|
|
struct SDL_KeyboardEvent;
|
|
|
|
|
2011-12-17 18:59:59 +00:00
|
|
|
using boost::logic::tribool;
|
|
|
|
|
2011-12-22 13:05:19 +00:00
|
|
|
// Defines a activate/deactive method
|
|
|
|
class IActivatable
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
virtual void activate()=0;
|
|
|
|
virtual void deactivate()=0;
|
2017-07-18 00:04:00 +03:00
|
|
|
virtual ~IActivatable(){};
|
2011-12-22 13:05:19 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class IUpdateable
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
virtual void update()=0;
|
2017-07-18 00:04:00 +03:00
|
|
|
virtual ~IUpdateable(){};
|
2011-12-22 13:05:19 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// Defines a show method
|
|
|
|
class IShowable
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
virtual void redraw()=0;
|
|
|
|
virtual void show(SDL_Surface * to) = 0;
|
|
|
|
virtual void showAll(SDL_Surface * to)
|
|
|
|
{
|
|
|
|
show(to);
|
|
|
|
}
|
2017-07-18 00:04:00 +03:00
|
|
|
virtual ~IShowable(){};
|
2011-12-22 13:05:19 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class IShowActivatable : public IShowable, public IActivatable
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
//redraw parent flag - this int may be semi-transparent and require redraw of parent window
|
2012-06-15 17:08:19 +00:00
|
|
|
enum {BLOCK_ADV_HOTKEYS = 2, REDRAW_PARENT=8};
|
2011-12-22 13:05:19 +00:00
|
|
|
int type; //bin flags using etype
|
|
|
|
IShowActivatable();
|
2017-07-18 00:04:00 +03:00
|
|
|
virtual ~IShowActivatable(){};
|
2011-12-22 13:05:19 +00:00
|
|
|
};
|
|
|
|
|
2017-06-07 20:16:18 +02:00
|
|
|
enum class EIntObjMouseBtnType { LEFT, MIDDLE, RIGHT };
|
2012-10-06 20:35:04 +00:00
|
|
|
//typedef ui16 ActivityFlag;
|
|
|
|
|
2011-12-17 18:59:59 +00:00
|
|
|
// Base UI element
|
|
|
|
class CIntObject : public IShowActivatable //interface object
|
|
|
|
{
|
2012-06-02 15:16:54 +00:00
|
|
|
ui16 used;//change via addUsed() or delUsed
|
2011-12-17 18:59:59 +00:00
|
|
|
|
2012-06-02 15:16:54 +00:00
|
|
|
//time handling
|
|
|
|
int toNextTick;
|
|
|
|
int timerDelay;
|
2012-10-06 20:35:04 +00:00
|
|
|
|
2017-06-07 20:16:18 +02:00
|
|
|
std::map<EIntObjMouseBtnType, bool> currentMouseState;
|
|
|
|
|
2012-06-02 15:16:54 +00:00
|
|
|
void onTimer(int timePassed);
|
|
|
|
|
|
|
|
//non-const versions of fields to allow changing them in CIntObject
|
|
|
|
CIntObject *parent_m; //parent object
|
|
|
|
ui16 active_m;
|
2018-04-07 14:34:11 +03:00
|
|
|
|
2012-10-06 20:35:04 +00:00
|
|
|
protected:
|
|
|
|
//activate or deactivate specific action (LCLICK, RCLICK...)
|
|
|
|
void activate(ui16 what);
|
|
|
|
void deactivate(ui16 what);
|
2012-06-02 15:16:54 +00:00
|
|
|
|
|
|
|
public:
|
|
|
|
/*
|
|
|
|
* Functions and fields that supposed to be private but are not for now.
|
|
|
|
* Don't use them unless you really know what they are for
|
|
|
|
*/
|
|
|
|
std::vector<CIntObject *> children;
|
2015-01-13 20:57:41 +01:00
|
|
|
|
2012-06-02 15:16:54 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Public interface
|
|
|
|
*/
|
|
|
|
|
|
|
|
/// read-only parent access. May not be a "clean" solution but allows some compatibility
|
|
|
|
CIntObject * const & parent;
|
|
|
|
|
|
|
|
/// position of object on the screen. Please do not modify this anywhere but in constructor - use moveBy\moveTo instead
|
|
|
|
/*const*/ Rect pos;
|
|
|
|
|
|
|
|
CIntObject(int used=0, Point offset=Point());
|
2017-07-18 00:04:00 +03:00
|
|
|
virtual ~CIntObject();
|
2012-06-02 15:16:54 +00:00
|
|
|
|
2017-06-07 20:16:18 +02:00
|
|
|
void updateMouseState(EIntObjMouseBtnType btn, bool state) { currentMouseState[btn] = state; }
|
|
|
|
bool mouseState(EIntObjMouseBtnType btn) const { return currentMouseState.count(btn) ? currentMouseState.at(btn) : false; }
|
2012-06-02 15:16:54 +00:00
|
|
|
|
2017-06-07 20:16:18 +02:00
|
|
|
virtual void click(EIntObjMouseBtnType btn, tribool down, bool previousState);
|
|
|
|
virtual void clickLeft(tribool down, bool previousState) {}
|
|
|
|
virtual void clickRight(tribool down, bool previousState) {}
|
|
|
|
virtual void clickMiddle(tribool down, bool previousState) {}
|
2012-06-02 15:16:54 +00:00
|
|
|
|
|
|
|
//hover handling
|
|
|
|
/*const*/ bool hovered; //for determining if object is hovered
|
|
|
|
virtual void hover (bool on){}
|
|
|
|
|
|
|
|
//keyboard handling
|
|
|
|
bool captureAllKeys; //if true, only this object should get info about pressed keys
|
|
|
|
virtual void keyPressed(const SDL_KeyboardEvent & key){}
|
|
|
|
virtual bool captureThisEvent(const SDL_KeyboardEvent & key); //allows refining captureAllKeys against specific events (eg. don't capture ENTER)
|
2015-01-13 20:57:41 +01:00
|
|
|
|
2014-05-23 20:46:54 +04:00
|
|
|
virtual void textInputed(const SDL_TextInputEvent & event){};
|
|
|
|
virtual void textEdited(const SDL_TextEditingEvent & event){};
|
2012-06-02 15:16:54 +00:00
|
|
|
|
|
|
|
//mouse movement handling
|
|
|
|
bool strongInterest; //if true - report all mouse movements, if not - only when hovered
|
|
|
|
virtual void mouseMoved (const SDL_MouseMotionEvent & sEvent){}
|
2011-12-17 18:59:59 +00:00
|
|
|
|
|
|
|
//time handling
|
2012-06-02 15:16:54 +00:00
|
|
|
void setTimer(int msToTrigger);//set timer delay and activate timer if needed.
|
|
|
|
virtual void tick(){}
|
2011-12-17 18:59:59 +00:00
|
|
|
|
|
|
|
//mouse wheel
|
2012-06-02 15:16:54 +00:00
|
|
|
virtual void wheelScrolled(bool down, bool in){}
|
2011-12-17 18:59:59 +00:00
|
|
|
|
|
|
|
//double click
|
2012-06-02 15:16:54 +00:00
|
|
|
virtual void onDoubleClick(){}
|
2011-12-17 18:59:59 +00:00
|
|
|
|
2022-09-18 22:45:30 -07:00
|
|
|
// These are the arguments that can be used to determine what kind of input the CIntObject will receive
|
2017-06-07 20:16:18 +02:00
|
|
|
enum {LCLICK=1, RCLICK=2, HOVER=4, MOVE=8, KEYBOARD=16, TIME=32, GENERAL=64, WHEEL=128, DOUBLECLICK=256, TEXTINPUT=512, MCLICK=1024, ALL=0xffff};
|
2012-06-02 15:16:54 +00:00
|
|
|
const ui16 & active;
|
|
|
|
void addUsedEvents(ui16 newActions);
|
|
|
|
void removeUsedEvents(ui16 newActions);
|
2011-12-17 18:59:59 +00:00
|
|
|
|
|
|
|
enum {ACTIVATE=1, DEACTIVATE=2, UPDATE=4, SHOWALL=8, DISPOSE=16, SHARE_POS=32};
|
|
|
|
ui8 defActions; //which calls will be tried to be redirected to children
|
2012-06-02 15:16:54 +00:00
|
|
|
ui8 recActions; //which calls we allow to receive from parent
|
2011-12-17 18:59:59 +00:00
|
|
|
|
|
|
|
void disable(); //deactivates if needed, blocks all automatic activity, allows only disposal
|
2012-06-02 15:16:54 +00:00
|
|
|
void enable(); //activates if needed, all activity enabled (Warning: may not be symetric with disable if recActions was limited!)
|
2012-01-19 14:33:22 +00:00
|
|
|
|
|
|
|
// activate or deactivate object. Inactive object won't receive any input events (keyboard\mouse)
|
|
|
|
// usually used automatically by parent
|
2015-10-12 16:47:10 +03:00
|
|
|
void activate() override;
|
|
|
|
void deactivate() override;
|
2012-01-19 14:33:22 +00:00
|
|
|
|
|
|
|
//called each frame to update screen
|
2015-10-12 16:47:10 +03:00
|
|
|
void show(SDL_Surface * to) override;
|
2012-01-19 14:33:22 +00:00
|
|
|
//called on complete redraw only
|
2015-10-12 16:47:10 +03:00
|
|
|
void showAll(SDL_Surface * to) override;
|
2012-06-02 15:16:54 +00:00
|
|
|
//request complete redraw of this object
|
2015-10-12 16:47:10 +03:00
|
|
|
void redraw() override;
|
2011-12-17 18:59:59 +00:00
|
|
|
|
|
|
|
bool isItInLoc(const SDL_Rect &rect, int x, int y);
|
2011-12-22 13:05:19 +00:00
|
|
|
bool isItInLoc(const SDL_Rect &rect, const Point &p);
|
|
|
|
const Rect & center(const Rect &r, bool propagate = true); //sets pos so that r will be in the center of screen, assigns sizes of r to pos, returns new position
|
|
|
|
const Rect & center(const Point &p, bool propagate = true); //moves object so that point p will be in its center
|
|
|
|
const Rect & center(bool propagate = true); //centers when pos.w and pos.h are set, returns new position
|
2011-12-17 18:59:59 +00:00
|
|
|
void fitToScreen(int borderWidth, bool propagate = true); //moves window to fit into screen
|
2011-12-22 13:05:19 +00:00
|
|
|
void moveBy(const Point &p, bool propagate = true);
|
2012-01-19 14:33:22 +00:00
|
|
|
void moveTo(const Point &p, bool propagate = true);//move this to new position, coordinates are absolute (0,0 is topleft screen corner)
|
2011-12-17 18:59:59 +00:00
|
|
|
|
|
|
|
void addChild(CIntObject *child, bool adjustPosition = false);
|
|
|
|
void removeChild(CIntObject *child, bool adjustPosition = false);
|
2012-06-02 15:16:54 +00:00
|
|
|
//delChild - not needed, use normal "delete child" instead
|
|
|
|
//delChildNull - not needed, use "vstd::clear_pointer(child)" instead
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Functions that should be used only by specific GUI elements. Don't use them unless you really know why they are here
|
|
|
|
*/
|
2018-07-25 01:36:48 +03:00
|
|
|
|
2012-06-02 15:16:54 +00:00
|
|
|
//functions for printing text. Use CLabel where possible instead
|
|
|
|
void printAtLoc(const std::string & text, int x, int y, EFonts font, SDL_Color color, SDL_Surface * dst);
|
|
|
|
void printAtMiddleLoc(const std::string & text, int x, int y, EFonts font, SDL_Color color, SDL_Surface * dst);
|
|
|
|
void printAtMiddleLoc(const std::string & text, const Point &p, EFonts font, SDL_Color color, SDL_Surface * dst);
|
|
|
|
void printAtMiddleWBLoc(const std::string & text, int x, int y, EFonts font, int charsPerLine, SDL_Color color, SDL_Surface * dst);
|
2011-12-17 18:59:59 +00:00
|
|
|
|
2012-06-02 15:16:54 +00:00
|
|
|
//image blitting. If possible use CPicture or CAnimImage instead
|
|
|
|
void blitAtLoc(SDL_Surface * src, int x, int y, SDL_Surface * dst);
|
|
|
|
void blitAtLoc(SDL_Surface * src, const Point &p, SDL_Surface * dst);
|
2011-12-17 18:59:59 +00:00
|
|
|
|
2012-06-02 15:16:54 +00:00
|
|
|
friend class CGuiHandler;
|
2011-12-22 13:05:19 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/// Class for binding keys to left mouse button clicks
|
|
|
|
/// Classes wanting use it should have it as one of their base classes
|
|
|
|
class CKeyShortcut : public virtual CIntObject
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
std::set<int> assignedKeys;
|
2014-08-03 14:16:19 +03:00
|
|
|
CKeyShortcut();
|
|
|
|
CKeyShortcut(int key);
|
|
|
|
CKeyShortcut(std::set<int> Keys);
|
2015-10-12 16:47:10 +03:00
|
|
|
virtual void keyPressed(const SDL_KeyboardEvent & key) override; //call-in
|
2012-06-02 15:16:54 +00:00
|
|
|
};
|
2018-07-25 01:36:48 +03:00
|
|
|
|
|
|
|
class WindowBase : public CIntObject
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
WindowBase(int used_ = 0, Point pos_ = Point());
|
|
|
|
protected:
|
2020-10-05 16:27:04 -07:00
|
|
|
void close();
|
2018-07-25 01:36:48 +03:00
|
|
|
};
|
2022-11-18 17:54:10 +02:00
|
|
|
|
|
|
|
class IStatusBar
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
virtual ~IStatusBar();
|
2022-12-12 18:23:41 +02:00
|
|
|
|
|
|
|
/// set current text for the status bar
|
|
|
|
virtual void write(const std::string & text) = 0;
|
|
|
|
|
|
|
|
/// remove any current text from the status bar
|
2022-11-18 17:54:10 +02:00
|
|
|
virtual void clear() = 0;
|
2022-12-12 18:23:41 +02:00
|
|
|
|
|
|
|
/// remove text from status bar if current text matches tested text
|
|
|
|
virtual void clearIfMatching(const std::string & testedText) = 0;
|
|
|
|
|
|
|
|
/// enables mode for entering text instead of showing hover text
|
|
|
|
virtual void setEnteringMode(bool on) = 0;
|
|
|
|
|
|
|
|
/// overrides hover text from controls with text entered into in-game console (for chat/cheats)
|
|
|
|
virtual void setEnteredText(const std::string & text) = 0;
|
|
|
|
|
2022-11-18 17:54:10 +02:00
|
|
|
};
|