#include "ui.h" #include #include #include #include #include "input.h" #include "internal.h" #include "internal/ui.h" namespace kek { UIElement::UIElement(UIValue x, UIValue y) : x(x), y(y) { } UIElement::~UIElement() { UI::unfocusElement(this); } // Returns the element's position relative to its parent in pixels UIPoint UIElement::getPosition() { return UIPoint(uiToScreen(x), uiToScreen(y)); } // Returns the element's origin position on the screen in pixels (not offset by getBounds()) UIPoint UIElement::getScreenOriginPosition() { UIPoint pos = getPosition(); if(parent) { UIPoint parentPos = parent->getScreenPosition(); pos.x += parentPos.x; pos.y += parentPos.y; } return pos; } // Returns the element's position on the screen in pixels (offset by getBounds()) UIPoint UIElement::getScreenPosition() { UIPoint pos = getScreenOriginPosition(); UIBounds bounds = getBounds(); return UIPoint(pos.x + bounds.x, pos.y + bounds.y); } std::vector UIElement::getChildren() { return children; } void UIElement::addChild(UIElement *child) { child->parent = this; children.push_back(child); } void UIElement::removeChild(UIElement *child) { child->parent = nullptr; std::erase(children, child); } int UIElement::uiToScreen(UIValue val) { float px = 0; px += val.pixels; if(parent) px += (int) (val.parentWidth * parent->getBounds().w); px += val.screenWidth * kekData.screenWidth; if(!parent) px += val.parentWidth * kekData.screenWidth; if(parent) px += (int) (val.parentHeight * parent->getBounds().h); px += val.screenHeight * kekData.screenHeight; if(!parent) px += val.parentHeight * kekData.screenHeight; return px; } int UIElement::offsetX(int w, Origin origin) { switch(origin) { case Origin::TOP_LEFT: case Origin::BOTTOM_LEFT: case Origin::LEFT_CENTER: return 0; case Origin::TOP_RIGHT: case Origin::BOTTOM_RIGHT: case Origin::RIGHT_CENTER: return -w; case Origin::TOP_CENTER: case Origin::BOTTOM_CENTER: case Origin::CENTER: return -w / 2; } return 0; } int UIElement::offsetY(int h, Origin origin) { switch(origin) { case Origin::TOP_LEFT: case Origin::TOP_CENTER: case Origin::TOP_RIGHT: return 0; case Origin::BOTTOM_LEFT: case Origin::BOTTOM_CENTER: case Origin::BOTTOM_RIGHT: return -h; case Origin::LEFT_CENTER: case Origin::RIGHT_CENTER: case Origin::CENTER: return -h / 2; } return 0; } UIBounds UIElement::offsetUIBounds(int w, int h, Origin origin) { return UIBounds(offsetX(w, origin), offsetY(h, origin), w, h); } void UIElement::drawAll(UIPoint screenPos, glm::mat4 projection) { if(!visible) return; if(enableClipping) { glEnable(GL_SCISSOR_TEST); UIPoint pos = getScreenPosition(); UIBounds bounds = getBounds(); glScissor(pos.x, kekData.screenHeight - (pos.y + bounds.h), bounds.w, bounds.h); } draw(screenPos, projection); for(UIElement *child : children) { UIPoint pos = child->getPosition(); child->drawAll(UIPoint(screenPos.x + pos.x, screenPos.y + pos.y), projection); } if(enableClipping) { glDisable(GL_SCISSOR_TEST); } } void UIElement::hoverAll(UIPoint pos, UIPoint screenPos) { UIElement *hoveredChild = nullptr; for(UIElement *child : children) { UIPoint childPos = child->getPosition(); int relX = pos.x - childPos.x; int relY = pos.y - childPos.y; UIBounds b = child->getBounds(); if(!b.contains(UIPoint(relX, relY))) continue; hoveredChild = child; } for(UIElement *child : children) { if(child != hoveredChild && child->hovering) child->hoverExitAll(); } if(hoveredChild) { UIPoint childPos = hoveredChild->getPosition(); hoveredChild->hoverAll(UIPoint(pos.x - childPos.x, pos.y - childPos.y), screenPos); } if(!hovering) hoverEnter(pos, screenPos); hovering = true; hover(pos, screenPos); } void UIElement::hoverExitAll() { if(!hovering) return; hovering = false; for(UIElement *child : children) { if(child->hovering) child->hoverExitAll(); } hoverExit(); } void UIElement::clickAll(UIPoint pos, UIPoint screenPos, GLFWMouseButton button) { UIElement *clickedChild = nullptr; for(UIElement *child : children) { if(!child->clickable) continue; UIPoint childPos = child->getPosition(); int relX = pos.x - childPos.x; int relY = pos.y - childPos.y; UIBounds b = child->getBounds(); if(!b.contains(UIPoint(relX, relY))) continue; clickedChild = child; } if(clickedChild != nullptr) { UIPoint childPos = clickedChild->getPosition(); clickedChild->clickAll(UIPoint(pos.x - childPos.x, pos.y - childPos.y), screenPos, button); } else { click(pos, screenPos, button); } } bool UIElement::focusEnterAll(UIPoint pos, UIPoint screenPos) { UIElement *focusedChild = nullptr; for(UIElement *child : children) { UIPoint childPos = child->getPosition(); int relX = pos.x - childPos.x; int relY = pos.y - childPos.y; UIBounds b = child->getBounds(); if(!b.contains(UIPoint(relX, relY))) continue; focusedChild = child; } if(focusedChild) { UIPoint childPos = focusedChild->getPosition(); bool focusable = focusedChild->focusEnterAll(UIPoint(pos.x - childPos.x, pos.y - childPos.y), screenPos); if(focusable) return true; } if(focusable) { UI::focusElement(this); return true; } return false; } UIElement *UIElement::dragEnterAll(UIPoint pos, UIPoint screenPos) { UIBounds bounds = getBounds(); if(!bounds.contains(pos)) return nullptr; // Only used by topmost parent UIElement UIElement *draggedChild = nullptr; for(UIElement *child : children) { UIPoint childPos = child->getPosition(); int relX = pos.x - childPos.x; int relY = pos.y - childPos.y; UIBounds b = child->getBounds(); if(!b.contains(UIPoint(relX, relY))) continue; draggedChild = child; } if(draggedChild) { UIPoint childPos = draggedChild->getPosition(); UIElement *dragged = draggedChild->dragEnterAll(UIPoint(pos.x - childPos.x, pos.y - childPos.y), screenPos); if(dragged) return dragged; } if(draggable) return this; return nullptr; } void UI::init() { kekData.ui = new UIData(); kekData.ui->defaultFont = Font::load(KEK_DEFAULT_FONT); kekData.ui->rectangleShader = new Shader("shader/rectangle/vertex.glsl", "shader/rectangle/fragment.glsl"); } void UI::destroy() { kekData.ui->defaultFont = nullptr; delete kekData.ui->rectangleShader; delete kekData.ui; } void UI::addElement(UIElement *element) { kekData.ui->elements.push_back(element); } void UI::removeElement(UIElement *element) { for(auto it = kekData.ui->elements.begin(); it < kekData.ui->elements.end(); it++) { if(*it == element) { kekData.ui->elements.erase(it); delete element; break; } } } std::vector UI::getElements() { return kekData.ui->elements; } void UI::focusElement(UIElement *element) { if(kekData.ui->focusedElement != element) { if(kekData.ui->focusedElement != nullptr) { unfocusElement(kekData.ui->focusedElement); } kekData.ui->focusedElement = element; if(element != nullptr) { element->focused = true; element->focusEnter(); } } } void UI::unfocusElement(UIElement *element) { if(kekData.ui->focusedElement == nullptr || element != kekData.ui->focusedElement) return; UIElement *oldFocus = kekData.ui->focusedElement; kekData.ui->focusedElement = nullptr; oldFocus->focused = false; oldFocus->focusExit(); } }