mirror of
				https://github.com/luanti-org/luanti.git
				synced 2025-10-25 13:45:23 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			940 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			940 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright (C) 2002-2012 Nikolaus Gebhardt
 | |
| // This file is part of the "Irrlicht Engine".
 | |
| // For conditions of distribution and use, see copyright notice in irrlicht.h
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include "IReferenceCounted.h"
 | |
| #include "rect.h"
 | |
| #include "irrString.h"
 | |
| #include "IEventReceiver.h"
 | |
| #include "EGUIElementTypes.h"
 | |
| #include "EGUIAlignment.h"
 | |
| #include "IGUIEnvironment.h"
 | |
| #include <cassert>
 | |
| #include <list>
 | |
| #include <vector>
 | |
| 
 | |
| namespace irr
 | |
| {
 | |
| namespace gui
 | |
| {
 | |
| //! Base class of all GUI elements.
 | |
| class IGUIElement : virtual public IReferenceCounted, public IEventReceiver
 | |
| {
 | |
| public:
 | |
| 	//! Constructor
 | |
| 	IGUIElement(EGUI_ELEMENT_TYPE type, IGUIEnvironment *environment, IGUIElement *parent,
 | |
| 			s32 id, const core::rect<s32> &rectangle) :
 | |
| 			Parent(0),
 | |
| 			RelativeRect(rectangle), AbsoluteRect(rectangle),
 | |
| 			AbsoluteClippingRect(rectangle), DesiredRect(rectangle),
 | |
| 			MaxSize(0, 0), MinSize(1, 1), IsVisible(true), IsEnabled(true),
 | |
| 			IsSubElement(false), NoClip(false), ID(id), IsTabStop(false), TabOrder(-1), IsTabGroup(false),
 | |
| 			AlignLeft(EGUIA_UPPERLEFT), AlignRight(EGUIA_UPPERLEFT), AlignTop(EGUIA_UPPERLEFT), AlignBottom(EGUIA_UPPERLEFT),
 | |
| 			Environment(environment), Type(type)
 | |
| 	{
 | |
| #ifdef _DEBUG
 | |
| 		setDebugName("IGUIElement");
 | |
| #endif
 | |
| 
 | |
| 		// if we were given a parent to attach to
 | |
| 		if (parent) {
 | |
| 			parent->addChildToEnd(this);
 | |
| 			recalculateAbsolutePosition(true);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	//! Destructor
 | |
| 	virtual ~IGUIElement()
 | |
| 	{
 | |
| 		for (auto child : Children) {
 | |
| 			child->Parent = nullptr;
 | |
| 			child->drop();
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	//! Returns parent of this element.
 | |
| 	IGUIElement *getParent() const
 | |
| 	{
 | |
| 		return Parent;
 | |
| 	}
 | |
| 
 | |
| 	//! Returns the relative rectangle of this element.
 | |
| 	core::rect<s32> getRelativePosition() const
 | |
| 	{
 | |
| 		return RelativeRect;
 | |
| 	}
 | |
| 
 | |
| 	//! Sets the relative rectangle of this element.
 | |
| 	/** \param r The absolute position to set */
 | |
| 	void setRelativePosition(const core::rect<s32> &r)
 | |
| 	{
 | |
| 		if (Parent) {
 | |
| 			const core::rect<s32> &r2 = Parent->getAbsolutePosition();
 | |
| 
 | |
| 			core::dimension2df d((f32)(r2.getSize().Width), (f32)(r2.getSize().Height));
 | |
| 
 | |
| 			if (AlignLeft == EGUIA_SCALE)
 | |
| 				ScaleRect.UpperLeftCorner.X = (f32)r.UpperLeftCorner.X / d.Width;
 | |
| 			if (AlignRight == EGUIA_SCALE)
 | |
| 				ScaleRect.LowerRightCorner.X = (f32)r.LowerRightCorner.X / d.Width;
 | |
| 			if (AlignTop == EGUIA_SCALE)
 | |
| 				ScaleRect.UpperLeftCorner.Y = (f32)r.UpperLeftCorner.Y / d.Height;
 | |
| 			if (AlignBottom == EGUIA_SCALE)
 | |
| 				ScaleRect.LowerRightCorner.Y = (f32)r.LowerRightCorner.Y / d.Height;
 | |
| 		}
 | |
| 
 | |
| 		DesiredRect = r;
 | |
| 		updateAbsolutePosition();
 | |
| 	}
 | |
| 
 | |
| 	//! Sets the relative rectangle of this element, maintaining its current width and height
 | |
| 	/** \param position The new relative position to set. Width and height will not be changed. */
 | |
| 	void setRelativePosition(const core::position2di &position)
 | |
| 	{
 | |
| 		const core::dimension2di mySize = RelativeRect.getSize();
 | |
| 		const core::rect<s32> rectangle(position.X, position.Y,
 | |
| 				position.X + mySize.Width, position.Y + mySize.Height);
 | |
| 		setRelativePosition(rectangle);
 | |
| 	}
 | |
| 
 | |
| 	//! Sets the relative rectangle of this element as a proportion of its parent's area.
 | |
| 	/** \note This method used to be 'void setRelativePosition(const core::rect<f32>& r)'
 | |
| 	\param r  The rectangle to set, interpreted as a proportion of the parent's area.
 | |
| 	Meaningful values are in the range [0...1], unless you intend this element to spill
 | |
| 	outside its parent. */
 | |
| 	void setRelativePositionProportional(const core::rect<f32> &r)
 | |
| 	{
 | |
| 		if (!Parent)
 | |
| 			return;
 | |
| 
 | |
| 		const core::dimension2di &d = Parent->getAbsolutePosition().getSize();
 | |
| 
 | |
| 		DesiredRect = core::rect<s32>(
 | |
| 				core::floor32((f32)d.Width * r.UpperLeftCorner.X),
 | |
| 				core::floor32((f32)d.Height * r.UpperLeftCorner.Y),
 | |
| 				core::floor32((f32)d.Width * r.LowerRightCorner.X),
 | |
| 				core::floor32((f32)d.Height * r.LowerRightCorner.Y));
 | |
| 
 | |
| 		ScaleRect = r;
 | |
| 
 | |
| 		updateAbsolutePosition();
 | |
| 	}
 | |
| 
 | |
| 	//! Gets the absolute rectangle of this element
 | |
| 	core::rect<s32> getAbsolutePosition() const
 | |
| 	{
 | |
| 		return AbsoluteRect;
 | |
| 	}
 | |
| 
 | |
| 	//! Returns the visible area of the element.
 | |
| 	core::rect<s32> getAbsoluteClippingRect() const
 | |
| 	{
 | |
| 		return AbsoluteClippingRect;
 | |
| 	}
 | |
| 
 | |
| 	//! Sets whether the element will ignore its parent's clipping rectangle
 | |
| 	/** \param noClip If true, the element will not be clipped by its parent's clipping rectangle. */
 | |
| 	void setNotClipped(bool noClip)
 | |
| 	{
 | |
| 		NoClip = noClip;
 | |
| 		updateAbsolutePosition();
 | |
| 	}
 | |
| 
 | |
| 	//! Gets whether the element will ignore its parent's clipping rectangle
 | |
| 	/** \return true if the element is not clipped by its parent's clipping rectangle. */
 | |
| 	bool isNotClipped() const
 | |
| 	{
 | |
| 		return NoClip;
 | |
| 	}
 | |
| 
 | |
| 	//! Sets the maximum size allowed for this element
 | |
| 	/** If set to 0,0, there is no maximum size */
 | |
| 	void setMaxSize(core::dimension2du size)
 | |
| 	{
 | |
| 		MaxSize = size;
 | |
| 		updateAbsolutePosition();
 | |
| 	}
 | |
| 
 | |
| 	//! Sets the minimum size allowed for this element
 | |
| 	void setMinSize(core::dimension2du size)
 | |
| 	{
 | |
| 		MinSize = size;
 | |
| 		if (MinSize.Width < 1)
 | |
| 			MinSize.Width = 1;
 | |
| 		if (MinSize.Height < 1)
 | |
| 			MinSize.Height = 1;
 | |
| 		updateAbsolutePosition();
 | |
| 	}
 | |
| 
 | |
| 	//! The alignment defines how the borders of this element will be positioned when the parent element is resized.
 | |
| 	void setAlignment(EGUI_ALIGNMENT left, EGUI_ALIGNMENT right, EGUI_ALIGNMENT top, EGUI_ALIGNMENT bottom)
 | |
| 	{
 | |
| 		AlignLeft = left;
 | |
| 		AlignRight = right;
 | |
| 		AlignTop = top;
 | |
| 		AlignBottom = bottom;
 | |
| 
 | |
| 		if (Parent) {
 | |
| 			core::rect<s32> r(Parent->getAbsolutePosition());
 | |
| 
 | |
| 			core::dimension2df d((f32)r.getSize().Width, (f32)r.getSize().Height);
 | |
| 
 | |
| 			if (AlignLeft == EGUIA_SCALE)
 | |
| 				ScaleRect.UpperLeftCorner.X = (f32)DesiredRect.UpperLeftCorner.X / d.Width;
 | |
| 			if (AlignRight == EGUIA_SCALE)
 | |
| 				ScaleRect.LowerRightCorner.X = (f32)DesiredRect.LowerRightCorner.X / d.Width;
 | |
| 			if (AlignTop == EGUIA_SCALE)
 | |
| 				ScaleRect.UpperLeftCorner.Y = (f32)DesiredRect.UpperLeftCorner.Y / d.Height;
 | |
| 			if (AlignBottom == EGUIA_SCALE)
 | |
| 				ScaleRect.LowerRightCorner.Y = (f32)DesiredRect.LowerRightCorner.Y / d.Height;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	//! How left element border is aligned when parent is resized
 | |
| 	EGUI_ALIGNMENT getAlignLeft() const
 | |
| 	{
 | |
| 		return AlignLeft;
 | |
| 	}
 | |
| 
 | |
| 	//! How right element border is aligned when parent is resized
 | |
| 	EGUI_ALIGNMENT getAlignRight() const
 | |
| 	{
 | |
| 		return AlignRight;
 | |
| 	}
 | |
| 
 | |
| 	//! How top element border is aligned when parent is resized
 | |
| 	EGUI_ALIGNMENT getAlignTop() const
 | |
| 	{
 | |
| 		return AlignTop;
 | |
| 	}
 | |
| 
 | |
| 	//! How bottom element border is aligned when parent is resized
 | |
| 	EGUI_ALIGNMENT getAlignBottom() const
 | |
| 	{
 | |
| 		return AlignBottom;
 | |
| 	}
 | |
| 
 | |
| 	//! Updates the absolute position.
 | |
| 	virtual void updateAbsolutePosition()
 | |
| 	{
 | |
| 		recalculateAbsolutePosition(false);
 | |
| 
 | |
| 		// update all children
 | |
| 		for (auto child : Children) {
 | |
| 			child->updateAbsolutePosition();
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	//! Returns the topmost GUI element at the specific position.
 | |
| 	/**
 | |
| 	This will check this GUI element and all of its descendants, so it
 | |
| 	may return this GUI element.  To check all GUI elements, call this
 | |
| 	function on device->getGUIEnvironment()->getRootGUIElement(). Note
 | |
| 	that the root element is the size of the screen, so doing so (with
 | |
| 	an on-screen point) will always return the root element if no other
 | |
| 	element is above it at that point.
 | |
| 	\param point: The point at which to find a GUI element.
 | |
| 	\return The topmost GUI element at that point, or 0 if there are
 | |
| 	no candidate elements at this point.
 | |
| 	*/
 | |
| 	virtual IGUIElement *getElementFromPoint(const core::position2d<s32> &point)
 | |
| 	{
 | |
| 		IGUIElement *target = 0;
 | |
| 
 | |
| 		if (isVisible()) {
 | |
| 			// we have to search from back to front, because later children
 | |
| 			// might be drawn over the top of earlier ones.
 | |
| 			auto it = Children.rbegin();
 | |
| 			auto ie = Children.rend();
 | |
| 			while (it != ie) {
 | |
| 				target = (*it)->getElementFromPoint(point);
 | |
| 				if (target)
 | |
| 					return target;
 | |
| 
 | |
| 				++it;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if (isVisible() && isPointInside(point))
 | |
| 			target = this;
 | |
| 
 | |
| 		return target;
 | |
| 	}
 | |
| 
 | |
| 	//! Returns true if a point is within this element.
 | |
| 	/** Elements with a shape other than a rectangle should override this method */
 | |
| 	virtual bool isPointInside(const core::position2d<s32> &point) const
 | |
| 	{
 | |
| 		return AbsoluteClippingRect.isPointInside(point);
 | |
| 	}
 | |
| 
 | |
| 	//! Adds a GUI element as new child of this element.
 | |
| 	virtual void addChild(IGUIElement *child)
 | |
| 	{
 | |
| 		if (child && child != this) {
 | |
| 			addChildToEnd(child);
 | |
| 			child->updateAbsolutePosition();
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	//! Removes a child.
 | |
| 	virtual void removeChild(IGUIElement *child)
 | |
| 	{
 | |
| 		assert(child->Parent == this);
 | |
| 		Children.erase(child->ParentPos);
 | |
| 		child->Parent = nullptr;
 | |
| 		child->drop();
 | |
| 	}
 | |
| 
 | |
| 	//! Removes all children.
 | |
| 	virtual void removeAllChildren()
 | |
| 	{
 | |
| 		while (!Children.empty()) {
 | |
| 			auto child = Children.back();
 | |
| 			child->remove();
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	//! Removes this element from its parent.
 | |
| 	virtual void remove()
 | |
| 	{
 | |
| 		if (Parent)
 | |
| 			Parent->removeChild(this);
 | |
| 	}
 | |
| 
 | |
| 	//! Draws the element and its children.
 | |
| 	virtual void draw()
 | |
| 	{
 | |
| 		if (isVisible()) {
 | |
| 			for (auto child : Children)
 | |
| 				child->draw();
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	//! animate the element and its children.
 | |
| 	virtual void OnPostRender(u32 timeMs)
 | |
| 	{
 | |
| 		if (isVisible()) {
 | |
| 			for (auto child : Children)
 | |
| 				child->OnPostRender(timeMs);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	//! Moves this element.
 | |
| 	virtual void move(core::position2d<s32> absoluteMovement)
 | |
| 	{
 | |
| 		setRelativePosition(DesiredRect + absoluteMovement);
 | |
| 	}
 | |
| 
 | |
| 	//! Returns true if element is visible.
 | |
| 	virtual bool isVisible() const
 | |
| 	{
 | |
| 		return IsVisible;
 | |
| 	}
 | |
| 
 | |
| 	//! Check whether the element is truly visible, taking into accounts its parents' visibility
 | |
| 	/** \return true if the element and all its parents are visible,
 | |
| 	false if this or any parent element is invisible. */
 | |
| 	virtual bool isTrulyVisible() const
 | |
| 	{
 | |
| 		if (!IsVisible)
 | |
| 			return false;
 | |
| 
 | |
| 		if (!Parent)
 | |
| 			return true;
 | |
| 
 | |
| 		return Parent->isTrulyVisible();
 | |
| 	}
 | |
| 
 | |
| 	//! Sets the visible state of this element.
 | |
| 	virtual void setVisible(bool visible)
 | |
| 	{
 | |
| 		IsVisible = visible;
 | |
| 	}
 | |
| 
 | |
| 	//! Returns true if this element was created as part of its parent control
 | |
| 	virtual bool isSubElement() const
 | |
| 	{
 | |
| 		return IsSubElement;
 | |
| 	}
 | |
| 
 | |
| 	//! Sets whether this control was created as part of its parent.
 | |
| 	/** For example, it is true when a scrollbar is part of a listbox.
 | |
| 	SubElements are not saved to disk when calling guiEnvironment->saveGUI() */
 | |
| 	virtual void setSubElement(bool subElement)
 | |
| 	{
 | |
| 		IsSubElement = subElement;
 | |
| 	}
 | |
| 
 | |
| 	//! If set to true, the focus will visit this element when using the tab key to cycle through elements.
 | |
| 	/** If this element is a tab group (see isTabGroup/setTabGroup) then
 | |
| 	ctrl+tab will be used instead. */
 | |
| 	void setTabStop(bool enable)
 | |
| 	{
 | |
| 		IsTabStop = enable;
 | |
| 	}
 | |
| 
 | |
| 	//! Returns true if this element can be focused by navigating with the tab key
 | |
| 	bool isTabStop() const
 | |
| 	{
 | |
| 		return IsTabStop;
 | |
| 	}
 | |
| 
 | |
| 	//! Sets the priority of focus when using the tab key to navigate between a group of elements.
 | |
| 	/** See setTabGroup, isTabGroup and getTabGroup for information on tab groups.
 | |
| 	Elements with a lower number are focused first */
 | |
| 	void setTabOrder(s32 index)
 | |
| 	{
 | |
| 		// negative = autonumber
 | |
| 		if (index < 0) {
 | |
| 			TabOrder = 0;
 | |
| 			IGUIElement *el = getTabGroup();
 | |
| 			while (IsTabGroup && el && el->Parent)
 | |
| 				el = el->Parent;
 | |
| 
 | |
| 			IGUIElement *first = 0, *closest = 0;
 | |
| 			if (el) {
 | |
| 				// find the highest element number
 | |
| 				el->getNextElement(-1, true, IsTabGroup, first, closest, true, true);
 | |
| 				if (first) {
 | |
| 					TabOrder = first->getTabOrder() + 1;
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 		} else
 | |
| 			TabOrder = index;
 | |
| 	}
 | |
| 
 | |
| 	//! Returns the number in the tab order sequence
 | |
| 	s32 getTabOrder() const
 | |
| 	{
 | |
| 		return TabOrder;
 | |
| 	}
 | |
| 
 | |
| 	//! Sets whether this element is a container for a group of elements which can be navigated using the tab key.
 | |
| 	/** For example, windows are tab groups.
 | |
| 	Groups can be navigated using ctrl+tab, providing isTabStop is true. */
 | |
| 	void setTabGroup(bool isGroup)
 | |
| 	{
 | |
| 		IsTabGroup = isGroup;
 | |
| 	}
 | |
| 
 | |
| 	//! Returns true if this element is a tab group.
 | |
| 	bool isTabGroup() const
 | |
| 	{
 | |
| 		return IsTabGroup;
 | |
| 	}
 | |
| 
 | |
| 	//! Returns the container element which holds all elements in this element's tab group.
 | |
| 	IGUIElement *getTabGroup()
 | |
| 	{
 | |
| 		IGUIElement *ret = this;
 | |
| 
 | |
| 		while (ret && !ret->isTabGroup())
 | |
| 			ret = ret->getParent();
 | |
| 
 | |
| 		return ret;
 | |
| 	}
 | |
| 
 | |
| 	//! Returns true if element is enabled
 | |
| 	/** Currently elements do _not_ care about parent-states.
 | |
| 		So if you want to affect children you have to enable/disable them all.
 | |
| 		The only exception to this are sub-elements which also check their parent.
 | |
| 	*/
 | |
| 	virtual bool isEnabled() const
 | |
| 	{
 | |
| 		if (isSubElement() && IsEnabled && getParent())
 | |
| 			return getParent()->isEnabled();
 | |
| 
 | |
| 		return IsEnabled;
 | |
| 	}
 | |
| 
 | |
| 	//! Sets the enabled state of this element.
 | |
| 	virtual void setEnabled(bool enabled)
 | |
| 	{
 | |
| 		IsEnabled = enabled;
 | |
| 	}
 | |
| 
 | |
| 	//! Sets the new caption of this element.
 | |
| 	virtual void setText(const wchar_t *text)
 | |
| 	{
 | |
| 		Text = text;
 | |
| 	}
 | |
| 
 | |
| 	//! Returns caption of this element.
 | |
| 	virtual const wchar_t *getText() const
 | |
| 	{
 | |
| 		return Text.c_str();
 | |
| 	}
 | |
| 
 | |
| 	//! Sets the new caption of this element.
 | |
| 	virtual void setToolTipText(const wchar_t *text)
 | |
| 	{
 | |
| 		ToolTipText = text;
 | |
| 	}
 | |
| 
 | |
| 	//! Returns caption of this element.
 | |
| 	virtual const core::stringw &getToolTipText() const
 | |
| 	{
 | |
| 		return ToolTipText;
 | |
| 	}
 | |
| 
 | |
| 	//! Returns id. Can be used to identify the element.
 | |
| 	virtual s32 getID() const
 | |
| 	{
 | |
| 		return ID;
 | |
| 	}
 | |
| 
 | |
| 	//! Sets the id of this element
 | |
| 	virtual void setID(s32 id)
 | |
| 	{
 | |
| 		ID = id;
 | |
| 	}
 | |
| 
 | |
| 	//! Called if an event happened.
 | |
| 	bool OnEvent(const SEvent &event) override
 | |
| 	{
 | |
| 		return Parent ? Parent->OnEvent(event) : false;
 | |
| 	}
 | |
| 
 | |
| 	//! Brings a child to front
 | |
| 	/** \return True if successful, false if not. */
 | |
| 	virtual bool bringToFront(IGUIElement *child)
 | |
| 	{
 | |
| 		if (child->Parent != this)
 | |
| 			return false;
 | |
| 		if (std::next(child->ParentPos) == Children.end()) // already there
 | |
| 			return true;
 | |
| 		Children.erase(child->ParentPos);
 | |
| 		child->ParentPos = Children.insert(Children.end(), child);
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	//! Moves a child to the back, so it's siblings are drawn on top of it
 | |
| 	/** \return True if successful, false if not. */
 | |
| 	virtual bool sendToBack(IGUIElement *child)
 | |
| 	{
 | |
| 		if (child->Parent != this)
 | |
| 			return false;
 | |
| 		if (child->ParentPos == Children.begin()) // already there
 | |
| 			return true;
 | |
| 		Children.erase(child->ParentPos);
 | |
| 		child->ParentPos = Children.insert(Children.begin(), child);
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	//! Returns list with children of this element
 | |
| 	virtual const std::list<IGUIElement *> &getChildren() const
 | |
| 	{
 | |
| 		return Children;
 | |
| 	}
 | |
| 
 | |
| 	//! Finds the first element with the given id.
 | |
| 	/** \param id: Id to search for.
 | |
| 	\param searchchildren: Set this to true, if also children of this
 | |
| 	element may contain the element with the searched id and they
 | |
| 	should be searched too.
 | |
| 	\return Returns the first element with the given id. If no element
 | |
| 	with this id was found, 0 is returned. */
 | |
| 	virtual IGUIElement *getElementFromId(s32 id, bool searchchildren = false) const
 | |
| 	{
 | |
| 		IGUIElement *e = 0;
 | |
| 
 | |
| 		for (auto child : Children) {
 | |
| 			if (child->getID() == id)
 | |
| 				return child;
 | |
| 
 | |
| 			if (searchchildren)
 | |
| 				e = child->getElementFromId(id, true);
 | |
| 
 | |
| 			if (e)
 | |
| 				return e;
 | |
| 		}
 | |
| 
 | |
| 		return e;
 | |
| 	}
 | |
| 
 | |
| 	//! returns true if the given element is a child of this one.
 | |
| 	//! \param child: The child element to check
 | |
| 	bool isMyChild(IGUIElement *child) const
 | |
| 	{
 | |
| 		if (!child)
 | |
| 			return false;
 | |
| 		do {
 | |
| 			if (child->Parent)
 | |
| 				child = child->Parent;
 | |
| 
 | |
| 		} while (child->Parent && child != this);
 | |
| 
 | |
| 		return child == this;
 | |
| 	}
 | |
| 
 | |
| 	//! searches elements to find the closest next element to tab to
 | |
| 	/** \param startOrder: The TabOrder of the current element, -1 if none
 | |
| 	\param reverse: true if searching for a lower number
 | |
| 	\param group: true if searching for a higher one
 | |
| 	\param first: element with the highest/lowest known tab order depending on search direction
 | |
| 	\param closest: the closest match, depending on tab order and direction
 | |
| 	\param includeInvisible: includes invisible elements in the search (default=false)
 | |
| 	\param includeDisabled: includes disabled elements in the search (default=false)
 | |
| 	\return true if successfully found an element, false to continue searching/fail */
 | |
| 	bool getNextElement(s32 startOrder, bool reverse, bool group,
 | |
| 			IGUIElement *&first, IGUIElement *&closest, bool includeInvisible = false,
 | |
| 			bool includeDisabled = false) const
 | |
| 	{
 | |
| 		// we'll stop searching if we find this number
 | |
| 		s32 wanted = startOrder + (reverse ? -1 : 1);
 | |
| 		if (wanted == -2)
 | |
| 			wanted = 1073741824; // maximum s32
 | |
| 
 | |
| 		auto it = Children.begin();
 | |
| 
 | |
| 		s32 closestOrder, currentOrder;
 | |
| 
 | |
| 		while (it != Children.end()) {
 | |
| 			// ignore invisible elements and their children
 | |
| 			if (((*it)->isVisible() || includeInvisible) &&
 | |
| 					(group == true || (*it)->isTabGroup() == false)) {
 | |
| 				// ignore disabled, but children are checked (disabled is currently per element ignoring parent states)
 | |
| 				if ((*it)->isEnabled() || includeDisabled) {
 | |
| 					// only check tab stops and those with the same group status
 | |
| 					if ((*it)->isTabStop() && ((*it)->isTabGroup() == group)) {
 | |
| 						currentOrder = (*it)->getTabOrder();
 | |
| 
 | |
| 						// is this what we're looking for?
 | |
| 						if (currentOrder == wanted) {
 | |
| 							closest = *it;
 | |
| 							return true;
 | |
| 						}
 | |
| 
 | |
| 						// is it closer than the current closest?
 | |
| 						if (closest) {
 | |
| 							closestOrder = closest->getTabOrder();
 | |
| 							if ((reverse && currentOrder > closestOrder && currentOrder < startOrder) || (!reverse && currentOrder < closestOrder && currentOrder > startOrder)) {
 | |
| 								closest = *it;
 | |
| 							}
 | |
| 						} else if ((reverse && currentOrder < startOrder) || (!reverse && currentOrder > startOrder)) {
 | |
| 							closest = *it;
 | |
| 						}
 | |
| 
 | |
| 						// is it before the current first?
 | |
| 						if (first) {
 | |
| 							closestOrder = first->getTabOrder();
 | |
| 
 | |
| 							if ((reverse && closestOrder < currentOrder) || (!reverse && closestOrder > currentOrder)) {
 | |
| 								first = *it;
 | |
| 							}
 | |
| 						} else {
 | |
| 							first = *it;
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 				// search within children
 | |
| 				if ((*it)->getNextElement(startOrder, reverse, group, first, closest, includeInvisible, includeDisabled)) {
 | |
| 					return true;
 | |
| 				}
 | |
| 			}
 | |
| 			++it;
 | |
| 		}
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	//! Returns the type of the gui element.
 | |
| 	/** This is needed for the .NET wrapper but will be used
 | |
| 	later for serializing and deserializing.
 | |
| 	If you wrote your own GUIElements, you need to set the type for your element as first parameter
 | |
| 	in the constructor of IGUIElement. For own (=unknown) elements, simply use EGUIET_ELEMENT as type */
 | |
| 	EGUI_ELEMENT_TYPE getType() const
 | |
| 	{
 | |
| 		return Type;
 | |
| 	}
 | |
| 
 | |
| 	//! Returns true if the gui element supports the given type.
 | |
| 	/** This is mostly used to check if you can cast a gui element to the class that goes with the type.
 | |
| 	Most gui elements will only support their own type, but if you derive your own classes from interfaces
 | |
| 	you can overload this function and add a check for the type of the base-class additionally.
 | |
| 	This allows for checks comparable to the dynamic_cast of c++ with enabled rtti.
 | |
| 	Note that you can't do that by calling BaseClass::hasType(type), but you have to do an explicit
 | |
| 	comparison check, because otherwise the base class usually just checks for the member variable
 | |
| 	Type which contains the type of your derived class.
 | |
| 	*/
 | |
| 	virtual bool hasType(EGUI_ELEMENT_TYPE type) const
 | |
| 	{
 | |
| 		return type == Type;
 | |
| 	}
 | |
| 
 | |
| 	//! Returns the type name of the gui element.
 | |
| 	/** This is needed serializing elements. */
 | |
| 	virtual const c8 *getTypeName() const
 | |
| 	{
 | |
| 		return GUIElementTypeNames[Type];
 | |
| 	}
 | |
| 
 | |
| 	//! Returns the name of the element.
 | |
| 	/** \return Name as character string. */
 | |
| 	virtual const c8 *getName() const
 | |
| 	{
 | |
| 		return Name.c_str();
 | |
| 	}
 | |
| 
 | |
| 	//! Sets the name of the element.
 | |
| 	/** \param name New name of the gui element. */
 | |
| 	virtual void setName(const c8 *name)
 | |
| 	{
 | |
| 		Name = name;
 | |
| 	}
 | |
| 
 | |
| 	//! Sets the name of the element.
 | |
| 	/** \param name New name of the gui element. */
 | |
| 	virtual void setName(const core::stringc &name)
 | |
| 	{
 | |
| 		Name = name;
 | |
| 	}
 | |
| 
 | |
| 	//! Returns whether the element takes input from the IME
 | |
| 	virtual bool acceptsIME()
 | |
| 	{
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| protected:
 | |
| 	// not virtual because needed in constructor
 | |
| 	void addChildToEnd(IGUIElement *child)
 | |
| 	{
 | |
| 		if (child) {
 | |
| 			child->grab();   // prevent destruction when removed
 | |
| 			child->remove(); // remove from old parent
 | |
| 			child->LastParentRect = getAbsolutePosition();
 | |
| 			child->Parent = this;
 | |
| 			child->ParentPos = Children.insert(Children.end(), child);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| #ifndef NDEBUG
 | |
| 	template <typename Iterator>
 | |
| 	static size_t _fastSetChecksum(Iterator begin, Iterator end)
 | |
| 	{
 | |
| 		std::hash<typename Iterator::value_type> hasher;
 | |
| 		size_t checksum = 0;
 | |
| 		for (Iterator it = begin; it != end; ++it) {
 | |
| 			size_t h = hasher(*it);
 | |
| 			checksum ^= 966073049 + (h * 3432918353) + ((h >> 16) * 461845907);
 | |
| 		}
 | |
| 		return checksum;
 | |
| 	}
 | |
| #endif
 | |
| 
 | |
| 	// Reorder children [from, to) to the order given by `neworder`
 | |
| 	void reorderChildren(
 | |
| 			std::list<IGUIElement *>::iterator from,
 | |
| 			std::list<IGUIElement *>::iterator to,
 | |
| 			const std::vector<IGUIElement *> &neworder)
 | |
| 	{
 | |
| 		assert(_fastSetChecksum(from, to) == _fastSetChecksum(neworder.begin(), neworder.end()));
 | |
| 		for (auto e : neworder) {
 | |
| 			*from = e;
 | |
| 			e->ParentPos = from;
 | |
| 			++from;
 | |
| 		}
 | |
| 		assert(from == to);
 | |
| 	}
 | |
| 
 | |
| 	// not virtual because needed in constructor
 | |
| 	void recalculateAbsolutePosition(bool recursive)
 | |
| 	{
 | |
| 		core::rect<s32> parentAbsolute(0, 0, 0, 0);
 | |
| 		core::rect<s32> parentAbsoluteClip;
 | |
| 		f32 fw = 0.f, fh = 0.f;
 | |
| 
 | |
| 		if (Parent) {
 | |
| 			parentAbsolute = Parent->AbsoluteRect;
 | |
| 
 | |
| 			if (NoClip) {
 | |
| 				IGUIElement *p = this;
 | |
| 				while (p->Parent)
 | |
| 					p = p->Parent;
 | |
| 				parentAbsoluteClip = p->AbsoluteClippingRect;
 | |
| 			} else
 | |
| 				parentAbsoluteClip = Parent->AbsoluteClippingRect;
 | |
| 		}
 | |
| 
 | |
| 		const s32 diffx = parentAbsolute.getWidth() - LastParentRect.getWidth();
 | |
| 		const s32 diffy = parentAbsolute.getHeight() - LastParentRect.getHeight();
 | |
| 
 | |
| 		if (AlignLeft == EGUIA_SCALE || AlignRight == EGUIA_SCALE)
 | |
| 			fw = (f32)parentAbsolute.getWidth();
 | |
| 
 | |
| 		if (AlignTop == EGUIA_SCALE || AlignBottom == EGUIA_SCALE)
 | |
| 			fh = (f32)parentAbsolute.getHeight();
 | |
| 
 | |
| 		switch (AlignLeft) {
 | |
| 		case EGUIA_UPPERLEFT:
 | |
| 			break;
 | |
| 		case EGUIA_LOWERRIGHT:
 | |
| 			DesiredRect.UpperLeftCorner.X += diffx;
 | |
| 			break;
 | |
| 		case EGUIA_CENTER:
 | |
| 			DesiredRect.UpperLeftCorner.X += diffx / 2;
 | |
| 			break;
 | |
| 		case EGUIA_SCALE:
 | |
| 			DesiredRect.UpperLeftCorner.X = core::round32(ScaleRect.UpperLeftCorner.X * fw);
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		switch (AlignRight) {
 | |
| 		case EGUIA_UPPERLEFT:
 | |
| 			break;
 | |
| 		case EGUIA_LOWERRIGHT:
 | |
| 			DesiredRect.LowerRightCorner.X += diffx;
 | |
| 			break;
 | |
| 		case EGUIA_CENTER:
 | |
| 			DesiredRect.LowerRightCorner.X += diffx / 2;
 | |
| 			break;
 | |
| 		case EGUIA_SCALE:
 | |
| 			DesiredRect.LowerRightCorner.X = core::round32(ScaleRect.LowerRightCorner.X * fw);
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		switch (AlignTop) {
 | |
| 		case EGUIA_UPPERLEFT:
 | |
| 			break;
 | |
| 		case EGUIA_LOWERRIGHT:
 | |
| 			DesiredRect.UpperLeftCorner.Y += diffy;
 | |
| 			break;
 | |
| 		case EGUIA_CENTER:
 | |
| 			DesiredRect.UpperLeftCorner.Y += diffy / 2;
 | |
| 			break;
 | |
| 		case EGUIA_SCALE:
 | |
| 			DesiredRect.UpperLeftCorner.Y = core::round32(ScaleRect.UpperLeftCorner.Y * fh);
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		switch (AlignBottom) {
 | |
| 		case EGUIA_UPPERLEFT:
 | |
| 			break;
 | |
| 		case EGUIA_LOWERRIGHT:
 | |
| 			DesiredRect.LowerRightCorner.Y += diffy;
 | |
| 			break;
 | |
| 		case EGUIA_CENTER:
 | |
| 			DesiredRect.LowerRightCorner.Y += diffy / 2;
 | |
| 			break;
 | |
| 		case EGUIA_SCALE:
 | |
| 			DesiredRect.LowerRightCorner.Y = core::round32(ScaleRect.LowerRightCorner.Y * fh);
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		RelativeRect = DesiredRect;
 | |
| 
 | |
| 		const s32 w = RelativeRect.getWidth();
 | |
| 		const s32 h = RelativeRect.getHeight();
 | |
| 
 | |
| 		// make sure the desired rectangle is allowed
 | |
| 		if (w < (s32)MinSize.Width)
 | |
| 			RelativeRect.LowerRightCorner.X = RelativeRect.UpperLeftCorner.X + MinSize.Width;
 | |
| 		if (h < (s32)MinSize.Height)
 | |
| 			RelativeRect.LowerRightCorner.Y = RelativeRect.UpperLeftCorner.Y + MinSize.Height;
 | |
| 		if (MaxSize.Width && w > (s32)MaxSize.Width)
 | |
| 			RelativeRect.LowerRightCorner.X = RelativeRect.UpperLeftCorner.X + MaxSize.Width;
 | |
| 		if (MaxSize.Height && h > (s32)MaxSize.Height)
 | |
| 			RelativeRect.LowerRightCorner.Y = RelativeRect.UpperLeftCorner.Y + MaxSize.Height;
 | |
| 
 | |
| 		RelativeRect.repair();
 | |
| 
 | |
| 		AbsoluteRect = RelativeRect + parentAbsolute.UpperLeftCorner;
 | |
| 
 | |
| 		if (!Parent)
 | |
| 			parentAbsoluteClip = AbsoluteRect;
 | |
| 
 | |
| 		AbsoluteClippingRect = AbsoluteRect;
 | |
| 		AbsoluteClippingRect.clipAgainst(parentAbsoluteClip);
 | |
| 
 | |
| 		LastParentRect = parentAbsolute;
 | |
| 
 | |
| 		if (recursive) {
 | |
| 			// update all children
 | |
| 			for (auto child : Children) {
 | |
| 				child->recalculateAbsolutePosition(recursive);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| protected:
 | |
| 	//! List of all children of this element
 | |
| 	std::list<IGUIElement *> Children;
 | |
| 
 | |
| 	//! Pointer to the parent
 | |
| 	IGUIElement *Parent;
 | |
| 
 | |
| 	//! Our position in the parent list. Only valid when Parent != nullptr
 | |
| 	std::list<IGUIElement *>::iterator ParentPos;
 | |
| 
 | |
| 	//! relative rect of element
 | |
| 	core::rect<s32> RelativeRect;
 | |
| 
 | |
| 	//! absolute rect of element
 | |
| 	core::rect<s32> AbsoluteRect;
 | |
| 
 | |
| 	//! absolute clipping rect of element
 | |
| 	core::rect<s32> AbsoluteClippingRect;
 | |
| 
 | |
| 	//! the rectangle the element would prefer to be,
 | |
| 	//! if it was not constrained by parent or max/min size
 | |
| 	core::rect<s32> DesiredRect;
 | |
| 
 | |
| 	//! for calculating the difference when resizing parent
 | |
| 	core::rect<s32> LastParentRect;
 | |
| 
 | |
| 	//! relative scale of the element inside its parent
 | |
| 	core::rect<f32> ScaleRect;
 | |
| 
 | |
| 	//! maximum and minimum size of the element
 | |
| 	core::dimension2du MaxSize, MinSize;
 | |
| 
 | |
| 	//! is visible?
 | |
| 	bool IsVisible;
 | |
| 
 | |
| 	//! is enabled?
 | |
| 	bool IsEnabled;
 | |
| 
 | |
| 	//! is a part of a larger whole and should not be serialized?
 | |
| 	bool IsSubElement;
 | |
| 
 | |
| 	//! does this element ignore its parent's clipping rectangle?
 | |
| 	bool NoClip;
 | |
| 
 | |
| 	//! caption
 | |
| 	core::stringw Text;
 | |
| 
 | |
| 	//! tooltip
 | |
| 	core::stringw ToolTipText;
 | |
| 
 | |
| 	//! users can set this for identifying the element by string
 | |
| 	core::stringc Name;
 | |
| 
 | |
| 	//! users can set this for identifying the element by integer
 | |
| 	s32 ID;
 | |
| 
 | |
| 	//! tab stop like in windows
 | |
| 	bool IsTabStop;
 | |
| 
 | |
| 	//! tab order
 | |
| 	s32 TabOrder;
 | |
| 
 | |
| 	//! tab groups are containers like windows, use ctrl+tab to navigate
 | |
| 	bool IsTabGroup;
 | |
| 
 | |
| 	//! tells the element how to act when its parent is resized
 | |
| 	EGUI_ALIGNMENT AlignLeft, AlignRight, AlignTop, AlignBottom;
 | |
| 
 | |
| 	//! GUI Environment
 | |
| 	IGUIEnvironment *Environment;
 | |
| 
 | |
| 	//! type of element
 | |
| 	EGUI_ELEMENT_TYPE Type;
 | |
| };
 | |
| 
 | |
| } // end namespace gui
 | |
| } // end namespace irr
 |