#include "borderDeco.hpp" #include #include #include #include #include using namespace Hyprutils::Memory; using namespace Render::GL; #include "BorderppPassElement.hpp" #include "globals.hpp" static size_t borderCount() { return std::clamp(vars.addBorders->value(), 0, static_cast(vars.borderSizes.size())); } CBordersPlusPlus::CBordersPlusPlus(PHLWINDOW pWindow) : IHyprWindowDecoration(pWindow), m_pWindow(pWindow) { m_lastWindowPos = pWindow->m_realPosition->value(); m_lastWindowSize = pWindow->m_realSize->value(); } CBordersPlusPlus::~CBordersPlusPlus() { damageEntire(); } SDecorationPositioningInfo CBordersPlusPlus::getPositioningInfo() { SDecorationPositioningInfo info; info.policy = DECORATION_POSITION_STICKY; info.reserved = true; info.priority = 9990; info.edges = DECORATION_EDGE_BOTTOM | DECORATION_EDGE_LEFT | DECORATION_EDGE_RIGHT | DECORATION_EDGE_TOP; if (m_fLastThickness == 0) { double size = 0; for (size_t i = 0; i < borderCount(); ++i) { size += vars.borderSizes[i]->value(); } info.desiredExtents = {{size, size}, {size, size}}; m_fLastThickness = size; } else info.desiredExtents = {{m_fLastThickness, m_fLastThickness}, {m_fLastThickness, m_fLastThickness}}; return info; } void CBordersPlusPlus::onPositioningReply(const SDecorationPositioningReply& reply) { m_bAssignedGeometry = reply.assignedGeometry; } uint64_t CBordersPlusPlus::getDecorationFlags() { return DECORATION_PART_OF_MAIN_WINDOW; } eDecorationLayer CBordersPlusPlus::getDecorationLayer() { return DECORATION_LAYER_OVER; } std::string CBordersPlusPlus::getDisplayName() { return "Borders++"; } void CBordersPlusPlus::draw(PHLMONITOR pMonitor, const float& a) { if (!validMapped(m_pWindow)) return; const auto PWINDOW = m_pWindow.lock(); if (!PWINDOW->m_ruleApplicator->decorate().valueOrDefault()) return; CBorderPPPassElement::SBorderPPData data; data.deco = this; g_pHyprRenderer->m_renderPass.add(makeUnique(data)); } void CBordersPlusPlus::drawPass(PHLMONITOR pMonitor, const float& a) { const auto PWINDOW = m_pWindow.lock(); static auto PROUNDING = CConfigValue("decoration:rounding"); static auto PBORDERSIZE = CConfigValue("general:border_size"); const auto BORDERS = borderCount(); const auto NATURALROUND = vars.naturalRounding->value(); if (BORDERS < 1) return; if (m_bAssignedGeometry.width < m_seExtents.topLeft.x + 1 || m_bAssignedGeometry.height < m_seExtents.topLeft.y + 1) return; const auto PWORKSPACE = PWINDOW->m_workspace; const auto WORKSPACEOFFSET = PWORKSPACE && !PWINDOW->m_pinned ? PWORKSPACE->m_renderOffset->value() : Vector2D(); auto rounding = PWINDOW->rounding() == 0 ? 0 : (PWINDOW->rounding() + *PBORDERSIZE) * pMonitor->m_scale; const auto ROUNDINGPOWER = PWINDOW->roundingPower(); const auto ORIGINALROUND = rounding == 0 ? 0 : (PWINDOW->rounding() + *PBORDERSIZE) * pMonitor->m_scale; CBox fullBox = m_bAssignedGeometry; fullBox.translate(g_pDecorationPositioner->getEdgeDefinedPoint(DECORATION_EDGE_BOTTOM | DECORATION_EDGE_LEFT | DECORATION_EDGE_RIGHT | DECORATION_EDGE_TOP, m_pWindow.lock())); fullBox.translate(PWINDOW->m_floatingOffset - pMonitor->m_position + WORKSPACEOFFSET); if (fullBox.width < 1 || fullBox.height < 1) return; double fullThickness = 0; for (size_t i = 0; i < BORDERS; ++i) { const int THISBORDERSIZE = vars.borderSizes[i]->value() == -1 ? *PBORDERSIZE : vars.borderSizes[i]->value(); fullThickness += THISBORDERSIZE; } fullBox.expand(-fullThickness).scale(pMonitor->m_scale).round(); for (size_t i = 0; i < BORDERS; ++i) { const int PREVBORDERSIZESCALED = i == 0 ? 0 : (vars.borderSizes[i - 1]->value() == -1 ? *PBORDERSIZE : vars.borderSizes[i - 1]->value()) * pMonitor->m_scale; const int THISBORDERSIZE = vars.borderSizes[i]->value() == -1 ? *PBORDERSIZE : vars.borderSizes[i]->value(); if (i != 0) { rounding += rounding == 0 ? 0 : PREVBORDERSIZESCALED; fullBox.x -= PREVBORDERSIZESCALED; fullBox.y -= PREVBORDERSIZESCALED; fullBox.width += PREVBORDERSIZESCALED * 2; fullBox.height += PREVBORDERSIZESCALED * 2; } if (fullBox.width < 1 || fullBox.height < 1) break; g_pHyprOpenGL->scissor(nullptr); g_pHyprOpenGL->renderBorder(fullBox, CHyprColor{static_cast(vars.borderColors[i]->value())}, {.round = NATURALROUND ? sc(ORIGINALROUND) : sc(rounding), .roundingPower = ROUNDINGPOWER, .borderSize = THISBORDERSIZE, .a = a, .outerRound = NATURALROUND ? sc(ORIGINALROUND) : -1}); } m_seExtents = {{fullThickness, fullThickness}, {fullThickness, fullThickness}}; m_bLastRelativeBox = CBox{0, 0, m_lastWindowSize.x, m_lastWindowSize.y}.addExtents(m_seExtents); if (fullThickness != m_fLastThickness) { m_fLastThickness = fullThickness; g_pDecorationPositioner->repositionDeco(this); } } eDecorationType CBordersPlusPlus::getDecorationType() { return DECORATION_CUSTOM; } void CBordersPlusPlus::updateWindow(PHLWINDOW pWindow) { m_lastWindowPos = pWindow->m_realPosition->value(); m_lastWindowSize = pWindow->m_realSize->value(); damageEntire(); } void CBordersPlusPlus::damageEntire() { CBox dm = m_bLastRelativeBox.copy().translate(m_lastWindowPos).expand(2); g_pHyprRenderer->damageBox(dm); }