hyprland-plugins/hyprexpo/overview.cpp

461 lines
16 KiB
C++
Raw Normal View History

2024-04-03 20:14:34 +02:00
#include "overview.hpp"
#include <any>
#define private public
#include <hyprland/src/render/Renderer.hpp>
#include <hyprland/src/Compositor.hpp>
#include <hyprland/src/config/ConfigValue.hpp>
2025-01-17 19:20:10 +01:00
#include <hyprland/src/managers/AnimationManager.hpp>
#include <hyprland/src/managers/input/InputManager.hpp>
2024-04-03 20:14:34 +02:00
#undef private
2024-12-22 17:33:07 +01:00
#include "OverviewPassElement.hpp"
2024-04-03 20:14:34 +02:00
2025-01-07 19:30:33 +01:00
static void damageMonitor(WP<Hyprutils::Animation::CBaseAnimatedVariable> thisptr) {
2024-04-03 20:14:34 +02:00
g_pOverview->damage();
}
2025-01-07 19:30:33 +01:00
static void removeOverview(WP<Hyprutils::Animation::CBaseAnimatedVariable> thisptr) {
2024-04-03 20:14:34 +02:00
g_pOverview.reset();
}
COverview::~COverview() {
2024-04-03 22:21:55 +02:00
g_pHyprRenderer->makeEGLCurrent();
images.clear(); // otherwise we get a vram leak
2024-04-03 20:14:34 +02:00
g_pInputManager->unsetCursorImage();
2024-10-21 20:17:02 +02:00
g_pHyprOpenGL->markBlurDirtyForMonitor(pMonitor.lock());
2024-04-03 20:14:34 +02:00
}
COverview::COverview(PHLWORKSPACE startedOn_, bool swipe_) : startedOn(startedOn_), swipe(swipe_) {
2024-10-21 20:17:02 +02:00
const auto PMONITOR = g_pCompositor->m_pLastMonitor.lock();
2024-04-03 20:14:34 +02:00
pMonitor = PMONITOR;
static auto* const* PCOLUMNS = (Hyprlang::INT* const*)HyprlandAPI::getConfigValue(PHANDLE, "plugin:hyprexpo:columns")->getDataStaticPtr();
static auto* const* PGAPS = (Hyprlang::INT* const*)HyprlandAPI::getConfigValue(PHANDLE, "plugin:hyprexpo:gap_size")->getDataStaticPtr();
static auto* const* PCOL = (Hyprlang::INT* const*)HyprlandAPI::getConfigValue(PHANDLE, "plugin:hyprexpo:bg_col")->getDataStaticPtr();
static auto const* PMETHOD = (Hyprlang::STRING const*)HyprlandAPI::getConfigValue(PHANDLE, "plugin:hyprexpo:workspace_method")->getDataStaticPtr();
SIDE_LENGTH = **PCOLUMNS;
GAP_WIDTH = **PGAPS;
BG_COLOR = **PCOL;
// process the method
2024-12-04 15:58:09 +01:00
bool methodCenter = true;
2024-04-03 20:14:34 +02:00
int methodStartID = pMonitor->activeWorkspaceID();
CVarList method{*PMETHOD, 0, 's', true};
if (method.size() < 2)
Debug::log(ERR, "[he] invalid workspace_method");
else {
2024-12-04 15:58:09 +01:00
methodCenter = method[0] == "center";
2024-06-23 20:46:19 +02:00
methodStartID = getWorkspaceIDNameFromString(method[1]).id;
2024-04-03 20:14:34 +02:00
if (methodStartID == WORKSPACE_INVALID)
methodStartID = pMonitor->activeWorkspaceID();
}
images.resize(SIDE_LENGTH * SIDE_LENGTH);
if (methodCenter) {
int currentID = methodStartID;
int firstID = currentID;
int backtracked = 0;
for (size_t i = 1; i < images.size() / 2; ++i) {
2024-06-23 20:46:19 +02:00
currentID = getWorkspaceIDNameFromString("r-" + std::to_string(i)).id;
2024-04-03 20:14:34 +02:00
if (currentID >= firstID)
break;
backtracked++;
firstID = currentID;
}
for (size_t i = 0; i < SIDE_LENGTH * SIDE_LENGTH; ++i) {
2024-12-04 15:58:09 +01:00
auto& image = images[i];
2024-04-03 20:14:34 +02:00
currentID =
2024-06-23 20:46:19 +02:00
getWorkspaceIDNameFromString("r" + ((int64_t)i - backtracked < 0 ? std::to_string((int64_t)i - backtracked) : "+" + std::to_string((int64_t)i - backtracked))).id;
2024-04-03 20:14:34 +02:00
image.workspaceID = currentID;
}
} else {
int currentID = methodStartID;
images[0].workspaceID = currentID;
auto PWORKSPACESTART = g_pCompositor->getWorkspaceByID(currentID);
if (!PWORKSPACESTART)
2024-10-28 15:25:34 +01:00
PWORKSPACESTART = CWorkspace::create(currentID, pMonitor.lock(), std::to_string(currentID));
pMonitor->activeWorkspace = PWORKSPACESTART;
2024-04-03 20:14:34 +02:00
for (size_t i = 1; i < SIDE_LENGTH * SIDE_LENGTH; ++i) {
2024-12-04 15:58:09 +01:00
auto& image = images[i];
2024-06-23 20:46:19 +02:00
currentID = getWorkspaceIDNameFromString("r+" + std::to_string(i)).id;
2024-04-03 20:14:34 +02:00
image.workspaceID = currentID;
}
pMonitor->activeWorkspace = startedOn;
2024-04-03 20:14:34 +02:00
}
g_pHyprRenderer->makeEGLCurrent();
timespec now;
clock_gettime(CLOCK_MONOTONIC, &now);
2024-12-27 11:36:51 +01:00
Vector2D tileSize = pMonitor->vecSize / SIDE_LENGTH;
Vector2D tileRenderSize = (pMonitor->vecSize - Vector2D{GAP_WIDTH * pMonitor->scale, GAP_WIDTH * pMonitor->scale} * (SIDE_LENGTH - 1)) / SIDE_LENGTH;
CBox monbox{0, 0, tileSize.x * 2, tileSize.y * 2};
if (!ENABLE_LOWRES)
monbox = {{0, 0}, pMonitor->vecPixelSize};
2024-04-03 20:14:34 +02:00
int currentid = 0;
PHLWORKSPACE openSpecial = PMONITOR->activeSpecialWorkspace;
if (openSpecial)
PMONITOR->activeSpecialWorkspace.reset();
g_pHyprRenderer->m_bBlockSurfaceFeedback = true;
startedOn->m_bVisible = false;
for (size_t i = 0; i < SIDE_LENGTH * SIDE_LENGTH; ++i) {
COverview::SWorkspaceImage& image = images[i];
2024-07-21 13:40:40 +02:00
image.fb.alloc(monbox.w, monbox.h, PMONITOR->output->state->state().drmFormat);
2024-04-03 20:14:34 +02:00
CRegion fakeDamage{0, 0, INT16_MAX, INT16_MAX};
g_pHyprRenderer->beginRender(PMONITOR, fakeDamage, RENDER_MODE_FULL_FAKE, nullptr, &image.fb);
2024-12-04 15:58:09 +01:00
g_pHyprOpenGL->clear(CHyprColor{0, 0, 0, 1.0});
2024-04-03 20:14:34 +02:00
const auto PWORKSPACE = g_pCompositor->getWorkspaceByID(image.workspaceID);
if (PWORKSPACE == startedOn)
currentid = i;
if (PWORKSPACE) {
image.pWorkspace = PWORKSPACE;
PMONITOR->activeWorkspace = PWORKSPACE;
PWORKSPACE->startAnim(true, true, true);
PWORKSPACE->m_bVisible = true;
if (PWORKSPACE == startedOn)
PMONITOR->activeSpecialWorkspace = openSpecial;
g_pHyprRenderer->renderWorkspace(PMONITOR, PWORKSPACE, &now, monbox);
PWORKSPACE->m_bVisible = false;
PWORKSPACE->startAnim(false, false, true);
if (PWORKSPACE == startedOn)
PMONITOR->activeSpecialWorkspace.reset();
} else
g_pHyprRenderer->renderWorkspace(PMONITOR, PWORKSPACE, &now, monbox);
image.box = {(i % SIDE_LENGTH) * tileRenderSize.x + (i % SIDE_LENGTH) * GAP_WIDTH, (i / SIDE_LENGTH) * tileRenderSize.y + (i / SIDE_LENGTH) * GAP_WIDTH, tileRenderSize.x,
tileRenderSize.y};
g_pHyprOpenGL->m_RenderData.blockScreenShader = true;
2024-04-03 20:14:34 +02:00
g_pHyprRenderer->endRender();
}
g_pHyprRenderer->m_bBlockSurfaceFeedback = false;
PMONITOR->activeSpecialWorkspace = openSpecial;
PMONITOR->activeWorkspace = startedOn;
startedOn->m_bVisible = true;
startedOn->startAnim(true, true, true);
// zoom on the current workspace.
const auto& TILE = images[std::clamp(currentid, 0, SIDE_LENGTH * SIDE_LENGTH)];
2025-01-07 19:30:33 +01:00
g_pAnimationManager->createAnimation(pMonitor->vecSize * pMonitor->vecSize / tileSize, size, g_pConfigManager->getAnimationPropertyConfig("windowsMove"), AVARDAMAGE_NONE);
g_pAnimationManager->createAnimation((-((pMonitor->vecSize / (double)SIDE_LENGTH) * Vector2D{currentid % SIDE_LENGTH, currentid / SIDE_LENGTH}) * pMonitor->scale) *
(pMonitor->vecSize / tileSize),
pos, g_pConfigManager->getAnimationPropertyConfig("windowsMove"), AVARDAMAGE_NONE);
2024-04-03 20:14:34 +02:00
2025-01-07 19:30:33 +01:00
size->setUpdateCallback(damageMonitor);
pos->setUpdateCallback(damageMonitor);
2024-04-03 20:14:34 +02:00
if (!swipe) {
2025-01-07 19:30:33 +01:00
*size = pMonitor->vecSize;
*pos = {0, 0};
2024-04-03 20:14:34 +02:00
2025-01-07 19:30:33 +01:00
size->setCallbackOnEnd([this](auto) { redrawAll(true); });
2024-04-03 20:14:34 +02:00
}
openedID = currentid;
g_pInputManager->setCursorImageUntilUnset("left_ptr");
lastMousePosLocal = g_pInputManager->getMouseCoordsInternal() - pMonitor->vecPosition;
auto onCursorMove = [this](void* self, SCallbackInfo& info, std::any param) {
2024-04-03 20:14:34 +02:00
if (closing)
return;
info.cancelled = true;
lastMousePosLocal = g_pInputManager->getMouseCoordsInternal() - pMonitor->vecPosition;
};
2024-04-03 20:14:34 +02:00
auto onCursorSelect = [this](void* self, SCallbackInfo& info, std::any param) {
2024-04-03 20:14:34 +02:00
if (closing)
return;
info.cancelled = true;
// get tile x,y
int x = lastMousePosLocal.x / pMonitor->vecSize.x * SIDE_LENGTH;
int y = lastMousePosLocal.y / pMonitor->vecSize.y * SIDE_LENGTH;
closeOnID = x + y * SIDE_LENGTH;
close();
};
mouseMoveHook = g_pHookSystem->hookDynamic("mouseMove", onCursorMove);
touchMoveHook = g_pHookSystem->hookDynamic("touchMove", onCursorMove);
mouseButtonHook = g_pHookSystem->hookDynamic("mouseButton", onCursorSelect);
touchDownHook = g_pHookSystem->hookDynamic("touchDown", onCursorSelect);
2024-04-03 20:14:34 +02:00
}
void COverview::redrawID(int id, bool forcelowres) {
if (pMonitor->activeWorkspace != startedOn && !closing) {
// likely user changed.
onWorkspaceChange();
}
2024-04-03 20:14:34 +02:00
blockOverviewRendering = true;
g_pHyprRenderer->makeEGLCurrent();
id = std::clamp(id, 0, SIDE_LENGTH * SIDE_LENGTH);
Vector2D tileSize = pMonitor->vecSize / SIDE_LENGTH;
Vector2D tileRenderSize = (pMonitor->vecSize - Vector2D{GAP_WIDTH, GAP_WIDTH} * (SIDE_LENGTH - 1)) / SIDE_LENGTH;
CBox monbox{0, 0, tileSize.x * 2, tileSize.y * 2};
2025-01-07 19:30:33 +01:00
if (!forcelowres && (size->value() != pMonitor->vecSize || closing))
2024-04-03 20:14:34 +02:00
monbox = {{0, 0}, pMonitor->vecPixelSize};
if (!ENABLE_LOWRES)
monbox = {{0, 0}, pMonitor->vecPixelSize};
timespec now;
clock_gettime(CLOCK_MONOTONIC, &now);
auto& image = images[id];
if (image.fb.m_vSize != monbox.size()) {
image.fb.release();
2024-07-21 13:40:40 +02:00
image.fb.alloc(monbox.w, monbox.h, pMonitor->output->state->state().drmFormat);
2024-04-03 20:14:34 +02:00
}
CRegion fakeDamage{0, 0, INT16_MAX, INT16_MAX};
2024-10-21 20:17:02 +02:00
g_pHyprRenderer->beginRender(pMonitor.lock(), fakeDamage, RENDER_MODE_FULL_FAKE, nullptr, &image.fb);
2024-04-03 20:14:34 +02:00
2024-12-04 15:58:09 +01:00
g_pHyprOpenGL->clear(CHyprColor{0, 0, 0, 1.0});
2024-04-03 20:14:34 +02:00
const auto PWORKSPACE = image.pWorkspace;
PHLWORKSPACE openSpecial = pMonitor->activeSpecialWorkspace;
if (openSpecial)
pMonitor->activeSpecialWorkspace.reset();
startedOn->m_bVisible = false;
if (PWORKSPACE) {
pMonitor->activeWorkspace = PWORKSPACE;
PWORKSPACE->startAnim(true, true, true);
PWORKSPACE->m_bVisible = true;
if (PWORKSPACE == startedOn)
pMonitor->activeSpecialWorkspace = openSpecial;
2024-10-21 20:17:02 +02:00
g_pHyprRenderer->renderWorkspace(pMonitor.lock(), PWORKSPACE, &now, monbox);
2024-04-03 20:14:34 +02:00
PWORKSPACE->m_bVisible = false;
PWORKSPACE->startAnim(false, false, true);
if (PWORKSPACE == startedOn)
pMonitor->activeSpecialWorkspace.reset();
} else
2024-10-21 20:17:02 +02:00
g_pHyprRenderer->renderWorkspace(pMonitor.lock(), PWORKSPACE, &now, monbox);
2024-04-03 20:14:34 +02:00
g_pHyprOpenGL->m_RenderData.blockScreenShader = true;
2024-04-03 20:14:34 +02:00
g_pHyprRenderer->endRender();
pMonitor->activeSpecialWorkspace = openSpecial;
pMonitor->activeWorkspace = startedOn;
startedOn->m_bVisible = true;
startedOn->startAnim(true, true, true);
blockOverviewRendering = false;
}
void COverview::redrawAll(bool forcelowres) {
for (size_t i = 0; i < SIDE_LENGTH * SIDE_LENGTH; ++i) {
redrawID(i, forcelowres);
}
}
void COverview::damage() {
blockDamageReporting = true;
2024-10-21 20:17:02 +02:00
g_pHyprRenderer->damageMonitor(pMonitor.lock());
2024-04-03 20:14:34 +02:00
blockDamageReporting = false;
}
void COverview::onDamageReported() {
damageDirty = true;
2025-01-07 19:30:33 +01:00
Vector2D SIZE = size->value();
2024-04-03 20:14:34 +02:00
Vector2D tileSize = (SIZE / SIDE_LENGTH);
Vector2D tileRenderSize = (SIZE - Vector2D{GAP_WIDTH, GAP_WIDTH} * (SIDE_LENGTH - 1)) / SIDE_LENGTH;
const auto& TILE = images[std::clamp(openedID, 0, SIDE_LENGTH * SIDE_LENGTH)];
CBox texbox = CBox{(openedID % SIDE_LENGTH) * tileRenderSize.x + (openedID % SIDE_LENGTH) * GAP_WIDTH,
(openedID / SIDE_LENGTH) * tileRenderSize.y + (openedID / SIDE_LENGTH) * GAP_WIDTH, tileRenderSize.x, tileRenderSize.y}
.translate(pMonitor->vecPosition);
damage();
blockDamageReporting = true;
g_pHyprRenderer->damageBox(&texbox);
blockDamageReporting = false;
2024-10-21 20:17:02 +02:00
g_pCompositor->scheduleFrameForMonitor(pMonitor.lock());
2024-04-03 20:14:34 +02:00
}
void COverview::close() {
if (closing)
return;
const int ID = closeOnID == -1 ? openedID : closeOnID;
const auto& TILE = images[std::clamp(ID, 0, SIDE_LENGTH * SIDE_LENGTH)];
Vector2D tileSize = (pMonitor->vecSize / SIDE_LENGTH);
2025-01-07 19:30:33 +01:00
*size = pMonitor->vecSize * pMonitor->vecSize / tileSize;
*pos = (-((pMonitor->vecSize / (double)SIDE_LENGTH) * Vector2D{ID % SIDE_LENGTH, ID / SIDE_LENGTH}) * pMonitor->scale) * (pMonitor->vecSize / tileSize);
2024-04-03 20:14:34 +02:00
2025-01-07 19:30:33 +01:00
size->setCallbackOnEnd(removeOverview);
2024-04-03 20:14:34 +02:00
closing = true;
redrawAll();
if (TILE.workspaceID != pMonitor->activeWorkspaceID()) {
pMonitor->setSpecialWorkspace(0);
const auto NEWIDWS = g_pCompositor->getWorkspaceByID(TILE.workspaceID);
const auto OLDWS = pMonitor->activeWorkspace;
2024-04-03 20:14:34 +02:00
if (!NEWIDWS)
g_pKeybindManager->changeworkspace(std::to_string(TILE.workspaceID));
else
g_pKeybindManager->changeworkspace(NEWIDWS->getConfigName());
pMonitor->activeWorkspace->startAnim(true, true, true);
OLDWS->startAnim(false, false, true);
2024-04-03 20:14:34 +02:00
startedOn = pMonitor->activeWorkspace;
}
}
void COverview::onPreRender() {
if (damageDirty) {
damageDirty = false;
2024-04-03 22:19:26 +02:00
redrawID(closing ? (closeOnID == -1 ? openedID : closeOnID) : openedID);
2024-04-03 20:14:34 +02:00
}
}
void COverview::onWorkspaceChange() {
if (valid(startedOn))
startedOn->startAnim(false, false, true);
else
startedOn = pMonitor->activeWorkspace;
2024-04-03 20:14:34 +02:00
for (size_t i = 0; i < SIDE_LENGTH * SIDE_LENGTH; ++i) {
if (images[i].workspaceID != pMonitor->activeWorkspaceID())
continue;
openedID = i;
break;
}
closeOnID = openedID;
close();
2024-04-03 20:14:34 +02:00
}
void COverview::render() {
2024-12-22 17:33:07 +01:00
g_pHyprRenderer->m_sRenderPass.add(makeShared<COverviewPassElement>());
}
2024-04-03 20:14:34 +02:00
2024-12-22 17:33:07 +01:00
void COverview::fullRender() {
2025-01-07 19:30:33 +01:00
const auto GAPSIZE = (closing ? (1.0 - size->getPercent()) : size->getPercent()) * GAP_WIDTH;
2024-04-03 20:14:34 +02:00
if (pMonitor->activeWorkspace != startedOn && !closing) {
2024-04-03 20:14:34 +02:00
// likely user changed.
onWorkspaceChange();
}
2025-01-07 19:30:33 +01:00
Vector2D SIZE = size->value();
2024-04-03 20:14:34 +02:00
Vector2D tileSize = (SIZE / SIDE_LENGTH);
Vector2D tileRenderSize = (SIZE - Vector2D{GAPSIZE, GAPSIZE} * (SIDE_LENGTH - 1)) / SIDE_LENGTH;
2024-04-04 00:27:43 +02:00
g_pHyprOpenGL->clear(BG_COLOR.stripA());
2024-04-03 20:14:34 +02:00
for (size_t y = 0; y < SIDE_LENGTH; ++y) {
for (size_t x = 0; x < SIDE_LENGTH; ++x) {
CBox texbox = {x * tileRenderSize.x + x * GAPSIZE, y * tileRenderSize.y + y * GAPSIZE, tileRenderSize.x, tileRenderSize.y};
2025-01-07 19:30:33 +01:00
texbox.scale(pMonitor->scale).translate(pos->value());
2024-04-03 20:14:34 +02:00
texbox.round();
CRegion damage{0, 0, INT16_MAX, INT16_MAX};
g_pHyprOpenGL->renderTextureInternalWithDamage(images[x + y * SIDE_LENGTH].fb.getTexture(), &texbox, 1.0, damage);
2024-04-03 20:14:34 +02:00
}
}
}
static float lerp(const float& from, const float& to, const float perc) {
return (to - from) * perc + from;
}
static Vector2D lerp(const Vector2D& from, const Vector2D& to, const float perc) {
return Vector2D{lerp(from.x, to.x, perc), lerp(from.y, to.y, perc)};
}
void COverview::onSwipeUpdate(double delta) {
2024-04-03 22:11:05 +02:00
if (swipeWasCommenced)
return;
2024-04-03 20:14:34 +02:00
2024-04-03 22:11:05 +02:00
static auto* const* PDISTANCE = (Hyprlang::INT* const*)HyprlandAPI::getConfigValue(PHANDLE, "plugin:hyprexpo:gesture_distance")->getDataStaticPtr();
const float PERC = 1.0 - std::clamp(delta / (double)**PDISTANCE, 0.0, 1.0);
2024-04-03 20:14:34 +02:00
2024-04-03 22:11:05 +02:00
Vector2D tileSize = (pMonitor->vecSize / SIDE_LENGTH);
const auto SIZEMAX = pMonitor->vecSize * pMonitor->vecSize / tileSize;
const auto POSMAX =
2024-04-03 20:14:34 +02:00
(-((pMonitor->vecSize / (double)SIDE_LENGTH) * Vector2D{openedID % SIDE_LENGTH, openedID / SIDE_LENGTH}) * pMonitor->scale) * (pMonitor->vecSize / tileSize);
const auto SIZEMIN = pMonitor->vecSize;
const auto POSMIN = Vector2D{0, 0};
2025-01-07 19:30:33 +01:00
size->setValueAndWarp(lerp(SIZEMIN, SIZEMAX, PERC));
pos->setValueAndWarp(lerp(POSMIN, POSMAX, PERC));
2024-04-03 20:14:34 +02:00
}
void COverview::onSwipeEnd() {
const auto SIZEMIN = pMonitor->vecSize;
const auto SIZEMAX = pMonitor->vecSize * pMonitor->vecSize / (pMonitor->vecSize / SIDE_LENGTH);
2025-01-07 19:30:33 +01:00
const auto PERC = (size->value() - SIZEMIN).x / (SIZEMAX - SIZEMIN).x;
if (PERC > 0.5) {
close();
return;
}
2025-01-07 19:30:33 +01:00
*size = pMonitor->vecSize;
*pos = {0, 0};
2024-04-03 20:14:34 +02:00
2025-01-07 19:30:33 +01:00
size->setCallbackOnEnd([this](WP<Hyprutils::Animation::CBaseAnimatedVariable> thisptr) { redrawAll(true); });
2024-04-03 22:11:05 +02:00
swipeWasCommenced = true;
}