osb/source/windowing/StarPaneManager.cpp
2024-09-11 19:52:01 +10:00

419 lines
14 KiB
C++

#include "StarPaneManager.hpp"
#include "StarGameTypes.hpp"
#include "StarJsonExtra.hpp"
#include "StarAssets.hpp"
#include "StarRoot.hpp"
namespace Star {
EnumMap<PaneLayer> const PaneLayerNames{
{PaneLayer::Tooltip, "Tooltip"},
{PaneLayer::ModalWindow, "ModalWindow"},
{PaneLayer::Window, "Window"},
{PaneLayer::Hud, "Hud"},
{PaneLayer::World, "World"}
};
PaneManager::PaneManager()
: m_context(GuiContext::singletonPtr()), m_prevInterfaceScale(1) {
auto assets = Root::singleton().assets();
m_tooltipMouseoverRadius = assets->json("/panes.config:tooltipMouseoverRadius").toFloat();
m_tooltipMouseOffset = jsonToVec2I(assets->json("/panes.config:tooltipMouseoverOffset"));
m_tooltipShowTimer = GameTimer(assets->json("/panes.config:tooltipMouseoverTime").toFloat());
}
void PaneManager::displayPane(PaneLayer paneLayer, PanePtr const& pane, DismissCallback onDismiss) {
if (!m_displayedPanes[paneLayer].insertFront(pane, std::move(onDismiss)).second)
throw GuiException("Pane displayed twice in PaneManager::displayPane");
if (!pane->hasDisplayed() && pane->anchor() == PaneAnchor::None)
pane->setPosition(Vec2I((windowSize() - pane->size()) / 2) + pane->centerOffset()); // center it
pane->displayed();
}
bool PaneManager::isDisplayed(PanePtr const& pane) const {
for (auto const& layerPair : m_displayedPanes) {
if (layerPair.second.contains(pane))
return true;
}
return false;
}
void PaneManager::dismissPane(PanePtr const& pane) {
if (!dismiss(pane))
throw GuiException("No such pane in PaneManager::dismissPane");
}
void PaneManager::dismissAllPanes(Set<PaneLayer> const& paneLayers) {
for (auto const& paneLayer : paneLayers) {
for (auto const& panePair : copy(m_displayedPanes[paneLayer]))
dismiss(panePair.first);
}
}
void PaneManager::dismissAllPanes() {
for (auto layerPair : copy(m_displayedPanes)) {
for (auto const& panePair : layerPair.second)
dismiss(panePair.first);
}
}
PanePtr PaneManager::topPane(Set<PaneLayer> const& paneLayers) const {
for (auto const& layerPair : m_displayedPanes) {
if (paneLayers.contains(layerPair.first) && !layerPair.second.empty())
return layerPair.second.firstKey();
}
return {};
}
PanePtr PaneManager::topPane() const {
for (auto const& layerPair : m_displayedPanes) {
if (!layerPair.second.empty())
return layerPair.second.firstKey();
}
return {};
}
void PaneManager::bringToTop(PanePtr const& pane) {
for (auto& layerPair : m_displayedPanes) {
if (layerPair.second.contains(pane)) {
layerPair.second.toFront(pane);
return;
}
}
throw GuiException("Pane was not displayed in PaneManager::bringToTop");
}
void PaneManager::bringPaneAdjacent(PanePtr const& anchor, PanePtr const& adjacent, int gap) {
Vec2I centerAdjacent = anchor->position() + (anchor->size() / 2) - (adjacent->size() / 2);
centerAdjacent = centerAdjacent.piecewiseClamp(Vec2I(), windowSize() - adjacent->size()); // keeps pane inside window
if (anchor->position()[0] + anchor->size()[0] + gap + adjacent->size()[0] <= windowSize()[0])
adjacent->setPosition(Vec2I(anchor->position()[0] + anchor->size()[0] + gap, centerAdjacent[1])); // place to the right
else if (anchor->position()[0] - gap - adjacent->size()[0] >= 0)
adjacent->setPosition(Vec2I(anchor->position()[0] - gap - adjacent->size()[0], centerAdjacent[1])); // place to the left
else if (anchor->position()[1] + anchor->size()[1] + gap + adjacent->size()[1] <= windowSize()[1])
adjacent->setPosition(Vec2I(centerAdjacent[0], anchor->position()[1] + anchor->size()[1] + gap)); // place above
else if (anchor->position()[1] - gap - adjacent->size()[1] >= 0)
adjacent->setPosition(Vec2I(centerAdjacent[0], anchor->position()[1] - gap - adjacent->size()[1])); // place below
else
adjacent->setPosition(centerAdjacent);
bringToTop(adjacent);
}
PanePtr PaneManager::getPaneAt(Set<PaneLayer> const& paneLayers, Vec2I const& position) const {
for (auto const& layerPair : m_displayedPanes) {
if (!paneLayers.contains(layerPair.first))
continue;
for (auto const& panePair : layerPair.second) {
if (panePair.first->inWindow(position) && panePair.first->active())
return panePair.first;
}
}
return {};
}
PanePtr PaneManager::getPaneAt(Vec2I const& position) const {
for (auto const& layerPair : m_displayedPanes) {
for (auto const& panePair : layerPair.second) {
if (panePair.first != m_activeTooltip
&& panePair.first->inWindow(position)
&& panePair.first->active())
return panePair.first;
}
}
return {};
}
void PaneManager::setBackgroundWidget(WidgetPtr bg) {
m_backgroundWidget = bg;
}
void PaneManager::dismissWhere(function<bool(PanePtr const&)> func) {
if (!func)
return;
for (auto& layerPair : m_displayedPanes) {
eraseWhere(layerPair.second, [&](auto& panePair) {
if (func(panePair.first)) {
panePair.first->dismissed();
if (panePair.second)
panePair.second(panePair.first);
return true;
}
return false;
});
}
}
PanePtr PaneManager::keyboardCapturedPane() const {
for (auto const& layerPair : m_displayedPanes) {
for (auto const& panePair : layerPair.second) {
if (panePair.first->active() && panePair.first->keyboardCaptured() != KeyboardCaptureMode::None)
return panePair.first;
}
}
return {};
}
bool PaneManager::keyboardCapturedForTextInput() const {
if (auto pane = keyboardCapturedPane())
return pane->keyboardCaptured() == KeyboardCaptureMode::TextInput;
return false;
}
bool PaneManager::sendInputEvent(InputEvent const& event) {
if (event.is<MouseMoveEvent>()) {
m_tooltipLastMousePos = *m_context->mousePosition(event);
for (auto const& layerPair : m_displayedPanes) {
for (auto const& panePair : layerPair.second) {
if (panePair.first->dragActive()) {
panePair.first->drag(*m_context->mousePosition(event));
return true;
}
}
}
}
if (event.is<MouseButtonDownEvent>()) {
m_tooltipShowTimer.reset();
if (m_activeTooltip) {
dismiss(m_activeTooltip);
m_activeTooltip.reset();
m_tooltipParentPane.reset();
m_tooltipShowTimer.reset();
}
}
if (event.is<MouseButtonUpEvent>()) {
for (auto const& layerPair : m_displayedPanes) {
for (auto const& panePair : layerPair.second) {
if (panePair.first->dragActive()) {
panePair.first->setDragActive(false, {});
return true;
}
}
}
}
// The gui close event can only be intercepted by a pane that has captured
// the keyboard otherwise it will always be used to close first before being
// a normal event. This is so a window can control its own closing if it
// really needs to (like the keybindings window).
if (event.is<KeyDownEvent>() && m_context->actions(event).contains(InterfaceAction::GuiClose)) {
if (auto top = topPane({PaneLayer::ModalWindow, PaneLayer::Window})) {
dismiss(top);
return true;
}
}
// If there is a pane that has captured the keyboard, keyboard events will
// ONLY be sent to it.
auto keyCapturePane = keyboardCapturedPane();
if (keyCapturePane && (event.is<KeyDownEvent>() || event.is<KeyUpEvent>() || event.is<TextInputEvent>()))
return keyCapturePane->sendEvent(event);
bool foundModal = false;
for (auto& layerPair : m_displayedPanes) {
for (auto const& panePair : copy(layerPair.second)) {
if (panePair.first->sendEvent(event)) {
if (event.is<MouseButtonDownEvent>())
layerPair.second.toFront(panePair.first);
return true;
}
// If any modal windows are shown, Only the first modal window should
// have a chance to consume the input event and all other panes below it
// including different layers should ignore it.
if (layerPair.first == PaneLayer::ModalWindow) {
foundModal = true;
break;
}
}
if (foundModal)
break;
}
return false;
}
void PaneManager::render() {
if (m_backgroundWidget) {
auto size = m_backgroundWidget->size();
m_backgroundWidget->setPosition(Vec2I((windowSize()[0] - size[0]) / 2, (windowSize()[1] - size[1]) / 2));
m_backgroundWidget->render(RectI(Vec2I(), windowSize()));
}
for (auto const& layerPair : reverseIterate(m_displayedPanes)) {
for (auto const& panePair : reverseIterate(layerPair.second)) {
if (panePair.first->active()) {
if (m_prevInterfaceScale != m_context->interfaceScale())
panePair.first->setPosition(
calculateNewInterfacePosition(panePair.first, (float)m_context->interfaceScale() / m_prevInterfaceScale));
panePair.first->setDrawingOffset(calculatePaneOffset(panePair.first));
panePair.first->render(RectI(Vec2I(), windowSize()));
}
}
}
m_context->resetInterfaceScissorRect();
m_prevInterfaceScale = m_context->interfaceScale();
}
void PaneManager::update(float dt) {
auto newTooltipParentPane = getPaneAt(m_tooltipLastMousePos);
bool updateTooltip = m_tooltipShowTimer.tick(dt) || (m_activeTooltip && (
vmag(m_tooltipInitialPosition - m_tooltipLastMousePos) > m_tooltipMouseoverRadius
|| m_tooltipParentPane != newTooltipParentPane
|| !m_tooltipParentPane->inWindow(m_tooltipLastMousePos)));
if (updateTooltip) {
if (m_activeTooltip) {
dismiss(m_activeTooltip);
m_activeTooltip.reset();
m_tooltipParentPane.reset();
}
m_tooltipShowTimer.reset();
if (newTooltipParentPane) {
if (auto tooltip = newTooltipParentPane->createTooltip(m_tooltipLastMousePos)) {
m_activeTooltip = std::move(tooltip);
m_tooltipParentPane = std::move(newTooltipParentPane);
m_tooltipInitialPosition = m_tooltipLastMousePos;
displayPane(PaneLayer::Tooltip, m_activeTooltip);
}
}
}
if (m_activeTooltip) {
Vec2I offsetDirection = Vec2I::filled(1);
Vec2I offsetAdjust = Vec2I();
if (m_tooltipLastMousePos[0] + m_tooltipMouseOffset[0] + m_activeTooltip->size()[0] > (int)m_context->windowWidth() / m_context->interfaceScale()) {
offsetDirection[0] = -1;
offsetAdjust[0] = -m_activeTooltip->size()[0];
}
if (m_tooltipLastMousePos[1] + m_tooltipMouseOffset[1] - m_activeTooltip->size()[1] < 0)
offsetDirection[1] = -1;
else
offsetAdjust[1] = -m_activeTooltip->size()[1];
m_activeTooltip->setPosition(m_tooltipLastMousePos + (offsetAdjust + m_tooltipMouseOffset.piecewiseMultiply(offsetDirection)));
}
for (auto const& layerPair : m_displayedPanes) {
for (auto const& panePair : copy(layerPair.second)) {
if (panePair.first->isDismissed())
dismiss(panePair.first);
}
}
for (auto const& layerPair : reverseIterate(m_displayedPanes)) {
for (auto const& panePair : reverseIterate(layerPair.second)) {
panePair.first->tick(dt);
if (panePair.first->active())
panePair.first->update(dt);
}
}
}
Vec2I PaneManager::windowSize() const {
return Vec2I(m_context->windowInterfaceSize());
}
Vec2I PaneManager::calculatePaneOffset(PanePtr const& pane) const {
Vec2I size = pane->size();
switch (pane->anchor()) {
case PaneAnchor::None:
return pane->anchorOffset();
case PaneAnchor::BottomLeft:
return pane->anchorOffset();
case PaneAnchor::BottomRight:
return pane->anchorOffset() + Vec2I{windowSize()[0] - size[0], 0};
case PaneAnchor::TopLeft:
return pane->anchorOffset() + Vec2I{0, windowSize()[1] - size[1]};
case PaneAnchor::TopRight:
return pane->anchorOffset() + (windowSize() - size);
case PaneAnchor::CenterTop:
return pane->anchorOffset() + Vec2I{(windowSize()[0] - size[0]) / 2, windowSize()[1] - size[1]};
case PaneAnchor::CenterBottom:
return pane->anchorOffset() + Vec2I{(windowSize()[0] - size[0]) / 2, 0};
case PaneAnchor::CenterLeft:
return pane->anchorOffset() + Vec2I{0, (windowSize()[1] - size[1]) / 2};
case PaneAnchor::CenterRight:
return pane->anchorOffset() + Vec2I{windowSize()[0] - size[0], (windowSize()[1] - size[1]) / 2};
case PaneAnchor::Center:
return pane->anchorOffset() + ((windowSize() - size) / 2);
default:
return pane->anchorOffset();
}
}
Vec2I PaneManager::calculateNewInterfacePosition(PanePtr const& pane, float interfaceScaleRatio) const {
Vec2F position(pane->relativePosition());
Vec2F size(pane->size());
Mat3F scale;
switch (pane->anchor()) {
case PaneAnchor::None:
scale = Mat3F::scaling(interfaceScaleRatio, Vec2F(windowSize()) / 2);
break;
case PaneAnchor::BottomLeft:
scale = Mat3F::scaling(interfaceScaleRatio);
break;
case PaneAnchor::BottomRight:
scale = Mat3F::scaling(interfaceScaleRatio, {size[0], 0});
break;
case PaneAnchor::TopLeft:
scale = Mat3F::scaling(interfaceScaleRatio, {0, size[1]});
break;
case PaneAnchor::TopRight:
scale = Mat3F::scaling(interfaceScaleRatio, size);
break;
case PaneAnchor::CenterTop:
scale = Mat3F::scaling(interfaceScaleRatio, {size[0] / 2, size[1]});
break;
case PaneAnchor::CenterBottom:
scale = Mat3F::scaling(interfaceScaleRatio, {size[0] / 2, 0});
break;
case PaneAnchor::CenterLeft:
scale = Mat3F::scaling(interfaceScaleRatio, {0, size[1] / 2});
break;
case PaneAnchor::CenterRight:
scale = Mat3F::scaling(interfaceScaleRatio, {size[0], size[1] / 2});
break;
case PaneAnchor::Center:
scale = Mat3F::scaling(interfaceScaleRatio, size / 2);
break;
default:
scale = Mat3F::scaling(interfaceScaleRatio, Vec2F(windowSize()) / 2);
}
return Vec2I::round((scale * Vec3F(position, 0)).vec2());
}
bool PaneManager::dismiss(PanePtr const& pane) {
bool dismissed = false;
for (auto& layerPair : m_displayedPanes) {
if (auto panePair = layerPair.second.maybeTake(pane)) {
dismissed = true;
panePair->first->dismissed();
if (panePair->second)
panePair->second(pane);
}
}
return dismissed;
}
}