2010-11-20 10:37:14 +00:00
|
|
|
/* (c) Magnus Auvinen. See licence.txt in the root of the distribution for more information. */
|
|
|
|
/* If you are missing that file, acquire a complete release at teeworlds.com. */
|
2010-05-29 07:25:38 +00:00
|
|
|
#ifndef GAME_CLIENT_UI_H
|
|
|
|
#define GAME_CLIENT_UI_H
|
2007-08-22 07:52:33 +00:00
|
|
|
|
2020-10-12 10:29:47 +00:00
|
|
|
#include <base/system.h>
|
|
|
|
#include <engine/textrender.h>
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
|
|
|
|
2009-10-27 14:38:53 +00:00
|
|
|
class CUIRect
|
2008-01-12 12:27:55 +00:00
|
|
|
{
|
2009-10-27 14:38:53 +00:00
|
|
|
// TODO: Refactor: Redo UI scaling
|
2010-12-14 00:20:47 +00:00
|
|
|
float Scale() const;
|
2020-09-26 19:41:58 +00:00
|
|
|
|
2009-10-27 14:38:53 +00:00
|
|
|
public:
|
2011-04-13 18:37:12 +00:00
|
|
|
float x, y, w, h;
|
|
|
|
|
2020-08-18 10:50:25 +00:00
|
|
|
/**
|
|
|
|
* Splits 2 CUIRect inside *this* CUIRect horizontally. You can pass null pointers.
|
|
|
|
*
|
|
|
|
* @param pTop This rect will end up taking the top half of this CUIRect
|
|
|
|
* @param pBottom This rect will end up taking the bottom half of this CUIRect
|
|
|
|
*/
|
2011-03-26 15:19:37 +00:00
|
|
|
void HSplitMid(CUIRect *pTop, CUIRect *pBottom) const;
|
2020-08-18 10:50:25 +00:00
|
|
|
/**
|
|
|
|
* Splits 2 CUIRect inside *this* CUIRect.
|
|
|
|
*
|
|
|
|
* The cut parameter determines the height of the top rect, so it allows more customization than HSplitMid.
|
|
|
|
*
|
|
|
|
* This method doesn't check if Cut is bigger than *this* rect height.
|
|
|
|
*
|
|
|
|
* @param Cut The height of the pTop rect.
|
|
|
|
* @param pTop The rect that ends up at the top with a height equal to Cut.
|
|
|
|
* @param pBottom The rect that ends up at the bottom with a height equal to *this* rect minus the Cut.
|
|
|
|
*/
|
2009-10-27 14:38:53 +00:00
|
|
|
void HSplitTop(float Cut, CUIRect *pTop, CUIRect *pBottom) const;
|
2020-08-18 10:50:25 +00:00
|
|
|
/**
|
|
|
|
* Splits 2 CUIRect inside *this* CUIRect.
|
|
|
|
*
|
|
|
|
* The cut parameter determines the height of the bottom rect, so it allows more customization than HSplitMid.
|
|
|
|
*
|
|
|
|
* This method doesn't check if Cut is bigger than *this* rect height.
|
|
|
|
*
|
|
|
|
* @param Cut The height of the pBottom rect.
|
|
|
|
* @param pTop The rect that ends up at the top with a height equal to *this* CUIRect height minus Cut.
|
|
|
|
* @param pBottom The rect that ends up at the bottom with a height equal to Cut.
|
|
|
|
*/
|
2009-10-27 14:38:53 +00:00
|
|
|
void HSplitBottom(float Cut, CUIRect *pTop, CUIRect *pBottom) const;
|
2020-08-18 10:50:25 +00:00
|
|
|
/**
|
|
|
|
* Splits 2 CUIRect inside *this* CUIRect vertically. You can pass null pointers.
|
|
|
|
*
|
|
|
|
* @param pLeft This rect will take up the left half of *this* CUIRect.
|
|
|
|
* @param pRight This rect will take up the right half of *this* CUIRect.
|
|
|
|
*/
|
2009-10-27 14:38:53 +00:00
|
|
|
void VSplitMid(CUIRect *pLeft, CUIRect *pRight) const;
|
2020-08-18 10:50:25 +00:00
|
|
|
/**
|
|
|
|
* Splits 2 CUIRect inside *this* CUIRect.
|
|
|
|
*
|
|
|
|
* The cut parameter determines the width of the left rect, so it allows more customization than VSplitMid.
|
|
|
|
*
|
|
|
|
* This method doesn't check if Cut is bigger than *this* rect width.
|
|
|
|
*
|
|
|
|
* @param Cut The width of the pLeft rect.
|
|
|
|
* @param pLeft The rect that ends up at the left with a width equal to Cut.
|
|
|
|
* @param pRight The rect that ends up at the right with a width equal to *this* rect minus the Cut.
|
|
|
|
*/
|
2009-10-27 14:38:53 +00:00
|
|
|
void VSplitLeft(float Cut, CUIRect *pLeft, CUIRect *pRight) const;
|
2020-08-18 10:50:25 +00:00
|
|
|
/**
|
|
|
|
* Splits 2 CUIRect inside *this* CUIRect.
|
|
|
|
*
|
|
|
|
* The cut parameter determines the width of the right rect, so it allows more customization than VSplitMid.
|
|
|
|
*
|
|
|
|
* This method doesn't check if Cut is bigger than *this* rect width.
|
|
|
|
*
|
|
|
|
* @param Cut The width of the pRight rect.
|
|
|
|
* @param pLeft The rect that ends up at the left with a width equal to *this* CUIRect width minus Cut.
|
|
|
|
* @param pRight The rect that ends up at the right with a width equal to Cut.
|
|
|
|
*/
|
2009-10-27 14:38:53 +00:00
|
|
|
void VSplitRight(float Cut, CUIRect *pLeft, CUIRect *pRight) const;
|
|
|
|
|
2020-08-18 10:50:25 +00:00
|
|
|
/**
|
|
|
|
* Places pOtherRect inside *this* CUIRect with Cut as the margin.
|
|
|
|
*
|
|
|
|
* @param Cut The margin.
|
|
|
|
* @param pOtherRect The CUIRect to place inside *this* CUIRect.
|
|
|
|
*/
|
2009-10-27 14:38:53 +00:00
|
|
|
void Margin(float Cut, CUIRect *pOtherRect) const;
|
2020-08-18 10:50:25 +00:00
|
|
|
/**
|
|
|
|
* Places pOtherRect inside *this* CUIRect applying Cut as the margin only on the vertical axis.
|
|
|
|
*
|
|
|
|
* @param Cut The margin.
|
|
|
|
* @param pOtherRect The CUIRect to place inside *this* CUIRect
|
|
|
|
*/
|
2009-10-27 14:38:53 +00:00
|
|
|
void VMargin(float Cut, CUIRect *pOtherRect) const;
|
2020-08-18 10:50:25 +00:00
|
|
|
/**
|
|
|
|
* Places pOtherRect inside *this* CUIRect applying Cut as the margin only on the horizontal axis.
|
|
|
|
*
|
|
|
|
* @param Cut The margin.
|
|
|
|
* @param pOtherRect The CUIRect to place inside *this* CUIRect
|
|
|
|
*/
|
2009-10-27 14:38:53 +00:00
|
|
|
void HMargin(float Cut, CUIRect *pOtherRect) const;
|
|
|
|
};
|
2008-01-12 12:27:55 +00:00
|
|
|
|
2020-10-12 10:29:47 +00:00
|
|
|
class CUI;
|
|
|
|
|
|
|
|
class CUIElement
|
|
|
|
{
|
|
|
|
friend class CUI;
|
|
|
|
|
|
|
|
CUIElement(CUI *pUI) { Init(pUI); }
|
|
|
|
|
|
|
|
public:
|
|
|
|
struct SUIElementRect
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
int m_UIRectQuadContainer;
|
|
|
|
int m_UITextContainer;
|
|
|
|
|
|
|
|
float m_X;
|
|
|
|
float m_Y;
|
|
|
|
float m_Width;
|
|
|
|
float m_Height;
|
|
|
|
|
|
|
|
std::string m_Text;
|
|
|
|
|
|
|
|
CTextCursor m_Cursor;
|
|
|
|
|
2020-11-08 18:41:16 +00:00
|
|
|
STextRenderColor m_TextColor;
|
|
|
|
STextRenderColor m_TextOutlineColor;
|
|
|
|
|
2020-10-12 10:29:47 +00:00
|
|
|
SUIElementRect() :
|
|
|
|
m_UIRectQuadContainer(-1), m_UITextContainer(-1), m_X(-1), m_Y(-1), m_Width(-1), m_Height(-1)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
protected:
|
|
|
|
std::vector<SUIElementRect> m_UIRects;
|
|
|
|
|
|
|
|
// used for marquees or other user implemented things
|
|
|
|
int64 m_ElementTime;
|
|
|
|
|
|
|
|
public:
|
|
|
|
CUIElement() = default;
|
|
|
|
|
|
|
|
void Init(CUI *pUI);
|
|
|
|
|
|
|
|
SUIElementRect *Get(size_t Index)
|
|
|
|
{
|
|
|
|
return &m_UIRects[Index];
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t Size()
|
|
|
|
{
|
|
|
|
return m_UIRects.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Clear() { m_UIRects.clear(); }
|
|
|
|
|
|
|
|
void Add(SUIElementRect &ElRect)
|
|
|
|
{
|
|
|
|
m_UIRects.push_back(ElRect);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2009-10-27 14:38:53 +00:00
|
|
|
class CUI
|
2008-01-12 12:27:55 +00:00
|
|
|
{
|
2009-10-27 14:38:53 +00:00
|
|
|
const void *m_pHotItem;
|
|
|
|
const void *m_pActiveItem;
|
|
|
|
const void *m_pLastActiveItem;
|
|
|
|
const void *m_pBecommingHotItem;
|
2010-05-29 07:25:38 +00:00
|
|
|
float m_MouseX, m_MouseY; // in gui space
|
|
|
|
float m_MouseWorldX, m_MouseWorldY; // in world space
|
2009-10-27 14:38:53 +00:00
|
|
|
unsigned m_MouseButtons;
|
|
|
|
unsigned m_LastMouseButtons;
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2009-10-27 14:38:53 +00:00
|
|
|
CUIRect m_Screen;
|
|
|
|
class IGraphics *m_pGraphics;
|
2010-05-29 07:25:38 +00:00
|
|
|
class ITextRender *m_pTextRender;
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2020-10-12 10:29:47 +00:00
|
|
|
std::vector<CUIElement *> m_OwnUIElements; // ui elements maintained by CUI class
|
|
|
|
std::vector<CUIElement *> m_UIElements;
|
|
|
|
|
2009-10-27 14:38:53 +00:00
|
|
|
public:
|
|
|
|
// TODO: Refactor: Fill this in
|
2020-09-26 19:41:58 +00:00
|
|
|
void SetGraphics(class IGraphics *pGraphics, class ITextRender *pTextRender)
|
|
|
|
{
|
|
|
|
m_pGraphics = pGraphics;
|
|
|
|
m_pTextRender = pTextRender;
|
|
|
|
}
|
2009-10-27 14:38:53 +00:00
|
|
|
class IGraphics *Graphics() { return m_pGraphics; }
|
2010-05-29 07:25:38 +00:00
|
|
|
class ITextRender *TextRender() { return m_pTextRender; }
|
2008-01-12 12:27:55 +00:00
|
|
|
|
2009-10-27 14:38:53 +00:00
|
|
|
CUI();
|
2020-10-12 10:29:47 +00:00
|
|
|
~CUI();
|
|
|
|
|
|
|
|
void ResetUIElement(CUIElement &UIElement);
|
|
|
|
|
|
|
|
CUIElement *GetNewUIElement();
|
|
|
|
|
|
|
|
void AddUIElement(CUIElement *pElement);
|
|
|
|
void OnElementsReset();
|
|
|
|
void OnWindowResize();
|
|
|
|
void OnLanguageChange();
|
2007-05-22 15:03:32 +00:00
|
|
|
|
2009-10-27 14:38:53 +00:00
|
|
|
enum
|
|
|
|
{
|
2020-09-26 19:41:58 +00:00
|
|
|
CORNER_TL = 1,
|
|
|
|
CORNER_TR = 2,
|
|
|
|
CORNER_BL = 4,
|
|
|
|
CORNER_BR = 8,
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
CORNER_T = CORNER_TL | CORNER_TR,
|
|
|
|
CORNER_B = CORNER_BL | CORNER_BR,
|
|
|
|
CORNER_R = CORNER_TR | CORNER_BR,
|
|
|
|
CORNER_L = CORNER_TL | CORNER_BL,
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
CORNER_ALL = CORNER_T | CORNER_B
|
2009-10-27 14:38:53 +00:00
|
|
|
};
|
2007-05-22 15:03:32 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
int Update(float mx, float my, float Mwx, float Mwy, int m_Buttons);
|
2007-05-22 15:03:32 +00:00
|
|
|
|
2009-10-27 14:38:53 +00:00
|
|
|
float MouseX() const { return m_MouseX; }
|
|
|
|
float MouseY() const { return m_MouseY; }
|
|
|
|
float MouseWorldX() const { return m_MouseWorldX; }
|
|
|
|
float MouseWorldY() const { return m_MouseWorldY; }
|
2020-09-26 19:41:58 +00:00
|
|
|
int MouseButton(int Index) const { return (m_MouseButtons >> Index) & 1; }
|
|
|
|
int MouseButtonClicked(int Index) { return MouseButton(Index) && !((m_LastMouseButtons >> Index) & 1); }
|
2008-01-12 12:27:55 +00:00
|
|
|
|
2009-10-27 14:38:53 +00:00
|
|
|
void SetHotItem(const void *pID) { m_pBecommingHotItem = pID; }
|
2020-09-26 19:41:58 +00:00
|
|
|
void SetActiveItem(const void *pID)
|
|
|
|
{
|
|
|
|
m_pActiveItem = pID;
|
|
|
|
if(pID)
|
|
|
|
m_pLastActiveItem = pID;
|
|
|
|
}
|
2009-10-27 14:38:53 +00:00
|
|
|
void ClearLastActiveItem() { m_pLastActiveItem = 0; }
|
|
|
|
const void *HotItem() const { return m_pHotItem; }
|
|
|
|
const void *NextHotItem() const { return m_pBecommingHotItem; }
|
|
|
|
const void *ActiveItem() const { return m_pActiveItem; }
|
|
|
|
const void *LastActiveItem() const { return m_pLastActiveItem; }
|
2008-01-12 12:27:55 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
int MouseInside(const CUIRect *pRect);
|
2011-07-02 22:36:07 +00:00
|
|
|
void ConvertMouseMove(float *x, float *y);
|
2008-01-12 12:27:55 +00:00
|
|
|
|
2009-10-27 14:38:53 +00:00
|
|
|
CUIRect *Screen();
|
2012-01-10 22:13:19 +00:00
|
|
|
float PixelSize();
|
2010-05-29 07:25:38 +00:00
|
|
|
void ClipEnable(const CUIRect *pRect);
|
2009-10-27 14:38:53 +00:00
|
|
|
void ClipDisable();
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2009-10-27 14:38:53 +00:00
|
|
|
// TODO: Refactor: Redo UI scaling
|
|
|
|
void SetScale(float s);
|
2010-12-14 00:20:47 +00:00
|
|
|
float Scale();
|
2007-05-22 15:03:32 +00:00
|
|
|
|
2020-10-12 10:29:47 +00:00
|
|
|
int DoButtonLogic(const void *pID, int Checked, const CUIRect *pRect);
|
2009-10-27 14:38:53 +00:00
|
|
|
int DoButtonLogic(const void *pID, const char *pText /* TODO: Refactor: Remove */, int Checked, const CUIRect *pRect);
|
2015-08-17 12:10:08 +00:00
|
|
|
int DoPickerLogic(const void *pID, const CUIRect *pRect, float *pX, float *pY);
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2009-10-27 14:38:53 +00:00
|
|
|
// TODO: Refactor: Remove this?
|
2020-10-06 10:25:10 +00:00
|
|
|
void DoLabel(const CUIRect *pRect, const char *pText, float Size, int Align, int MaxWidth = -1, int AlignVertically = 1);
|
|
|
|
void DoLabelScaled(const CUIRect *pRect, const char *pText, float Size, int Align, int MaxWidth = -1, int AlignVertically = 1);
|
2020-10-12 10:29:47 +00:00
|
|
|
|
|
|
|
void DoLabel(CUIElement::SUIElementRect &RectEl, const CUIRect *pRect, const char *pText, float Size, int Align, int MaxWidth = -1, int AlignVertically = 1, bool StopAtEnd = false, int StrLen = -1, class CTextCursor *pReadCursor = NULL);
|
|
|
|
void DoLabelStreamed(CUIElement::SUIElementRect &RectEl, const CUIRect *pRect, const char *pText, float Size, int Align, int MaxWidth = -1, int AlignVertically = 1, bool StopAtEnd = false, int StrLen = -1, class CTextCursor *pReadCursor = NULL);
|
2009-10-27 14:38:53 +00:00
|
|
|
};
|
2007-08-22 07:52:33 +00:00
|
|
|
|
2007-05-22 15:03:32 +00:00
|
|
|
#endif
|