mirror of
https://github.com/hyprwm/Hyprland
synced 2024-11-05 17:25:58 +01:00
4a42344e97
* style: apply clang-format * ci: add new clang-format job to CI
264 lines
9.6 KiB
C++
264 lines
9.6 KiB
C++
#include "Events.hpp"
|
|
|
|
#include "../Compositor.hpp"
|
|
#include "../helpers/WLClasses.hpp"
|
|
#include "../managers/input/InputManager.hpp"
|
|
#include "../render/Renderer.hpp"
|
|
|
|
// --------------------------------------------- //
|
|
// _____ ____ _____ _ _ _____ _____ //
|
|
// | __ \ / __ \| __ \| | | | __ \ / ____| //
|
|
// | |__) | | | | |__) | | | | |__) | (___ //
|
|
// | ___/| | | | ___/| | | | ___/ \___ \ //
|
|
// | | | |__| | | | |__| | | ____) | //
|
|
// |_| \____/|_| \____/|_| |_____/ //
|
|
// //
|
|
// --------------------------------------------- //
|
|
|
|
void addPopupGlobalCoords(void* pPopup, int* x, int* y) {
|
|
SXDGPopup* const PPOPUP = (SXDGPopup*)pPopup;
|
|
|
|
auto curPopup = PPOPUP;
|
|
|
|
int px = 0;
|
|
int py = 0;
|
|
|
|
while (true) {
|
|
px += curPopup->popup->current.geometry.x;
|
|
py += curPopup->popup->current.geometry.y;
|
|
|
|
if (curPopup == PPOPUP && PPOPUP->parentWindow) {
|
|
px -= curPopup->popup->base->current.geometry.x;
|
|
py -= curPopup->popup->base->current.geometry.y;
|
|
}
|
|
|
|
if (curPopup->popup && !curPopup->parentPopup && !curPopup->parentWindow) {
|
|
const auto EXTENTSSURFACE = pixman_region32_extents(&curPopup->popup->base->surface->input_region);
|
|
px -= EXTENTSSURFACE->x1;
|
|
py -= EXTENTSSURFACE->y1;
|
|
}
|
|
|
|
if (curPopup->parentPopup) {
|
|
curPopup = curPopup->parentPopup;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
px += PPOPUP->lx;
|
|
py += PPOPUP->ly;
|
|
|
|
*x += px;
|
|
*y += py;
|
|
}
|
|
|
|
void createNewPopup(wlr_xdg_popup* popup, SXDGPopup* pHyprPopup) {
|
|
pHyprPopup->popup = popup;
|
|
|
|
pHyprPopup->hyprListener_destroyPopupXDG.initCallback(&popup->base->events.destroy, &Events::listener_destroyPopupXDG, pHyprPopup, "HyprPopup");
|
|
pHyprPopup->hyprListener_mapPopupXDG.initCallback(&popup->base->surface->events.map, &Events::listener_mapPopupXDG, pHyprPopup, "HyprPopup");
|
|
pHyprPopup->hyprListener_unmapPopupXDG.initCallback(&popup->base->surface->events.unmap, &Events::listener_unmapPopupXDG, pHyprPopup, "HyprPopup");
|
|
pHyprPopup->hyprListener_newPopupFromPopupXDG.initCallback(&popup->base->events.new_popup, &Events::listener_newPopupFromPopupXDG, pHyprPopup, "HyprPopup");
|
|
pHyprPopup->hyprListener_commitPopupXDG.initCallback(&popup->base->surface->events.commit, &Events::listener_commitPopupXDG, pHyprPopup, "HyprPopup");
|
|
pHyprPopup->hyprListener_repositionPopupXDG.initCallback(&popup->events.reposition, &Events::listener_repositionPopupXDG, pHyprPopup, "HyprPopup");
|
|
|
|
const auto PMONITOR = g_pCompositor->m_pLastMonitor;
|
|
|
|
CBox box = {PMONITOR->vecPosition.x - pHyprPopup->lx, PMONITOR->vecPosition.y - pHyprPopup->ly, PMONITOR->vecSize.x, PMONITOR->vecSize.y};
|
|
|
|
wlr_xdg_popup_unconstrain_from_box(popup, box.pWlr());
|
|
|
|
pHyprPopup->monitor = PMONITOR;
|
|
|
|
Debug::log(LOG, "Popup: Unconstrained from lx ly: {:j5}, pHyprPopup lx ly: {:.5f} {:.5f}", PMONITOR->vecPosition, (float)pHyprPopup->lx, (float)pHyprPopup->ly);
|
|
}
|
|
|
|
void Events::listener_newPopup(void* owner, void* data) {
|
|
SLayerSurface* layersurface = (SLayerSurface*)owner;
|
|
|
|
ASSERT(layersurface);
|
|
|
|
Debug::log(LOG, "New layer popup created from surface {:x}", (uintptr_t)layersurface);
|
|
|
|
const auto WLRPOPUP = (wlr_xdg_popup*)data;
|
|
|
|
const auto PNEWPOPUP = g_pCompositor->m_vXDGPopups.emplace_back(std::make_unique<SXDGPopup>()).get();
|
|
|
|
const auto PMONITOR = g_pCompositor->getMonitorFromID(layersurface->monitorID);
|
|
|
|
PNEWPOPUP->popup = WLRPOPUP;
|
|
PNEWPOPUP->lx = layersurface->position.x;
|
|
PNEWPOPUP->ly = layersurface->position.y;
|
|
PNEWPOPUP->monitor = PMONITOR;
|
|
PNEWPOPUP->parentLS = layersurface;
|
|
createNewPopup(WLRPOPUP, PNEWPOPUP);
|
|
}
|
|
|
|
void Events::listener_newPopupXDG(void* owner, void* data) {
|
|
CWindow* PWINDOW = (CWindow*)owner;
|
|
|
|
ASSERT(PWINDOW);
|
|
|
|
if (!PWINDOW->m_bIsMapped)
|
|
return;
|
|
|
|
Debug::log(LOG, "New layer popup created from XDG window {}", PWINDOW);
|
|
|
|
const auto WLRPOPUP = (wlr_xdg_popup*)data;
|
|
|
|
const auto PNEWPOPUP = g_pCompositor->m_vXDGPopups.emplace_back(std::make_unique<SXDGPopup>()).get();
|
|
|
|
const auto PMONITOR = g_pCompositor->getMonitorFromID(PWINDOW->m_iMonitorID);
|
|
|
|
PNEWPOPUP->popup = WLRPOPUP;
|
|
PNEWPOPUP->lx = PWINDOW->m_vRealPosition.goalv().x;
|
|
PNEWPOPUP->ly = PWINDOW->m_vRealPosition.goalv().y;
|
|
PNEWPOPUP->parentWindow = PWINDOW;
|
|
PNEWPOPUP->monitor = PMONITOR;
|
|
createNewPopup(WLRPOPUP, PNEWPOPUP);
|
|
}
|
|
|
|
void Events::listener_newPopupFromPopupXDG(void* owner, void* data) {
|
|
SXDGPopup* PPOPUP = (SXDGPopup*)owner;
|
|
|
|
ASSERT(PPOPUP);
|
|
|
|
if (PPOPUP->parentWindow)
|
|
Debug::log(LOG, "New popup created from XDG Window popup {:x} -> {}", (uintptr_t)PPOPUP, PPOPUP->parentWindow);
|
|
else
|
|
Debug::log(LOG, "New popup created from Non-Window popup {:x}", (uintptr_t)PPOPUP);
|
|
|
|
const auto WLRPOPUP = (wlr_xdg_popup*)data;
|
|
|
|
const auto PNEWPOPUP = g_pCompositor->m_vXDGPopups.emplace_back(std::make_unique<SXDGPopup>()).get();
|
|
|
|
PNEWPOPUP->popup = WLRPOPUP;
|
|
PNEWPOPUP->parentPopup = PPOPUP;
|
|
PNEWPOPUP->lx = PPOPUP->lx;
|
|
PNEWPOPUP->ly = PPOPUP->ly;
|
|
PNEWPOPUP->parentWindow = PPOPUP->parentWindow;
|
|
PNEWPOPUP->monitor = PPOPUP->monitor;
|
|
|
|
createNewPopup(WLRPOPUP, PNEWPOPUP);
|
|
}
|
|
|
|
void Events::listener_mapPopupXDG(void* owner, void* data) {
|
|
SXDGPopup* PPOPUP = (SXDGPopup*)owner;
|
|
|
|
ASSERT(PPOPUP);
|
|
|
|
Debug::log(LOG, "New XDG Popup mapped at {} {}", (int)PPOPUP->lx, (int)PPOPUP->ly);
|
|
|
|
if (PPOPUP->parentWindow)
|
|
PPOPUP->parentWindow->m_lPopupSurfaces.emplace_back(PPOPUP->popup->base->surface);
|
|
else if (PPOPUP->parentLS)
|
|
PPOPUP->parentLS->popupSurfaces.emplace_back(PPOPUP->popup->base->surface);
|
|
|
|
PPOPUP->pSurfaceTree = SubsurfaceTree::createTreeRoot(PPOPUP->popup->base->surface, addPopupGlobalCoords, PPOPUP, PPOPUP->parentWindow);
|
|
|
|
int lx = 0, ly = 0;
|
|
addPopupGlobalCoords(PPOPUP, &lx, &ly);
|
|
|
|
CBox extents;
|
|
wlr_surface_get_extends(PPOPUP->popup->base->surface, extents.pWlr());
|
|
extents.applyFromWlr();
|
|
|
|
g_pHyprRenderer->damageBox(lx - extents.x, ly - extents.y, extents.width + 2, extents.height + 2);
|
|
|
|
if (PPOPUP->monitor) {
|
|
g_pCompositor->setPreferredScaleForSurface(PPOPUP->popup->base->surface, PPOPUP->monitor->scale);
|
|
g_pCompositor->setPreferredTransformForSurface(PPOPUP->popup->base->surface, PPOPUP->monitor->transform);
|
|
}
|
|
|
|
Debug::log(LOG, "XDG Popup got assigned a surfaceTreeNode {:x}", (uintptr_t)PPOPUP->pSurfaceTree);
|
|
}
|
|
|
|
void Events::listener_repositionPopupXDG(void* owner, void* data) {
|
|
SXDGPopup* PPOPUP = (SXDGPopup*)owner;
|
|
|
|
Debug::log(LOG, "XDG Popup {:x} asks for a reposition", (uintptr_t)PPOPUP);
|
|
|
|
int lx = 0, ly = 0;
|
|
addPopupGlobalCoords(PPOPUP, &lx, &ly);
|
|
|
|
CBox extents;
|
|
wlr_surface_get_extends(PPOPUP->popup->base->surface, extents.pWlr());
|
|
extents.applyFromWlr();
|
|
|
|
PPOPUP->lastPos = {lx - extents.x, ly - extents.y};
|
|
PPOPUP->repositionRequested = true;
|
|
|
|
const auto PMONITOR = g_pCompositor->m_pLastMonitor;
|
|
|
|
CBox box = {PMONITOR->vecPosition.x - lx + PPOPUP->popup->current.geometry.x, PMONITOR->vecPosition.y - ly + PPOPUP->popup->current.geometry.y, PMONITOR->vecSize.x,
|
|
PMONITOR->vecSize.y};
|
|
wlr_xdg_popup_unconstrain_from_box(PPOPUP->popup, box.pWlr());
|
|
}
|
|
|
|
void Events::listener_unmapPopupXDG(void* owner, void* data) {
|
|
SXDGPopup* PPOPUP = (SXDGPopup*)owner;
|
|
Debug::log(LOG, "XDG Popup unmapped");
|
|
|
|
ASSERT(PPOPUP);
|
|
|
|
if (PPOPUP->popup->base->surface == g_pCompositor->m_pLastFocus)
|
|
g_pInputManager->releaseAllMouseButtons();
|
|
|
|
SubsurfaceTree::destroySurfaceTree(PPOPUP->pSurfaceTree);
|
|
|
|
int lx = 0, ly = 0;
|
|
addPopupGlobalCoords(PPOPUP, &lx, &ly);
|
|
|
|
CBox extents;
|
|
wlr_surface_get_extends(PPOPUP->popup->base->surface, extents.pWlr());
|
|
extents.applyFromWlr();
|
|
|
|
g_pHyprRenderer->damageBox(lx - extents.x, ly - extents.y, extents.width + 2, extents.height + 2);
|
|
|
|
if (PPOPUP->parentWindow)
|
|
std::erase(PPOPUP->parentWindow->m_lPopupSurfaces, PPOPUP->popup->base->surface);
|
|
else if (PPOPUP->parentLS)
|
|
std::erase(PPOPUP->parentLS->popupSurfaces, PPOPUP->popup->base->surface);
|
|
|
|
PPOPUP->pSurfaceTree = nullptr;
|
|
|
|
g_pInputManager->simulateMouseMovement(); // to focus and return back to an appropriate surface
|
|
}
|
|
|
|
void Events::listener_commitPopupXDG(void* owner, void* data) {
|
|
SXDGPopup* PPOPUP = (SXDGPopup*)owner;
|
|
|
|
if (g_pCompositor->windowValidMapped(PPOPUP->parentWindow)) {
|
|
PPOPUP->lx = PPOPUP->parentWindow->m_vRealPosition.vec().x;
|
|
PPOPUP->ly = PPOPUP->parentWindow->m_vRealPosition.vec().y;
|
|
}
|
|
|
|
int lx = 0, ly = 0;
|
|
addPopupGlobalCoords(PPOPUP, &lx, &ly);
|
|
|
|
CBox extents;
|
|
wlr_surface_get_extends(PPOPUP->popup->base->surface, extents.pWlr());
|
|
extents.applyFromWlr();
|
|
|
|
if (PPOPUP->repositionRequested)
|
|
g_pHyprRenderer->damageBox(PPOPUP->lastPos.x, PPOPUP->lastPos.y, extents.width + 2, extents.height + 2);
|
|
|
|
PPOPUP->repositionRequested = false;
|
|
|
|
g_pHyprRenderer->damageSurface(PPOPUP->popup->base->surface, lx, ly);
|
|
}
|
|
|
|
void Events::listener_destroyPopupXDG(void* owner, void* data) {
|
|
SXDGPopup* PPOPUP = (SXDGPopup*)owner;
|
|
|
|
ASSERT(PPOPUP);
|
|
|
|
Debug::log(LOG, "Destroyed popup XDG {:x}", (uintptr_t)PPOPUP);
|
|
|
|
if (PPOPUP->pSurfaceTree) {
|
|
SubsurfaceTree::destroySurfaceTree(PPOPUP->pSurfaceTree);
|
|
PPOPUP->pSurfaceTree = nullptr;
|
|
}
|
|
|
|
std::erase_if(g_pCompositor->m_vXDGPopups, [&](std::unique_ptr<SXDGPopup>& el) { return el.get() == PPOPUP; });
|
|
}
|