From 06060bb27d61b30d4ed9fb0f66eff1d638bdb6aa Mon Sep 17 00:00:00 2001 From: Vaxry Date: Wed, 18 Dec 2024 15:42:49 +0000 Subject: [PATCH 1/6] core: add nominal size --- include/hyprcursor/shared.h | 1 + libhyprcursor/hyprcursor.cpp | 44 +++++++++++++++++---------- libhyprcursor/internalSharedTypes.hpp | 2 +- libhyprcursor/meta.cpp | 2 ++ libhyprcursor/meta.hpp | 2 +- 5 files changed, 33 insertions(+), 18 deletions(-) diff --git a/include/hyprcursor/shared.h b/include/hyprcursor/shared.h index c3fb9c6..c36456c 100644 --- a/include/hyprcursor/shared.h +++ b/include/hyprcursor/shared.h @@ -54,6 +54,7 @@ struct SCursorRawShapeDataC { char* overridenBy; enum eHyprcursorResizeAlgo resizeAlgo; enum eHyprcursorDataType type; + float nominalSize; }; typedef struct SCursorRawShapeDataC hyprcursor_cursor_raw_shape_data; diff --git a/libhyprcursor/hyprcursor.cpp b/libhyprcursor/hyprcursor.cpp index 544a506..e0926f6 100644 --- a/libhyprcursor/hyprcursor.cpp +++ b/libhyprcursor/hyprcursor.cpp @@ -307,13 +307,15 @@ SCursorImageData** CHyprcursorManager::getShapesC(int& outSize, const char* shap if (REQUESTEDSHAPE != shape->directory && std::find(shape->overrides.begin(), shape->overrides.end(), REQUESTEDSHAPE) == shape->overrides.end()) continue; + const int PIXELSIDE = std::round(info.size / shape->nominalSize); + hotX = shape->hotspotX; hotY = shape->hotspotY; // matched :) bool foundAny = false; for (auto& image : impl->loadedShapes[shape.get()].images) { - if (image->side != info.size) + if (image->side != PIXELSIDE) continue; // found size @@ -333,7 +335,7 @@ SCursorImageData** CHyprcursorManager::getShapesC(int& outSize, const char* shap // find nearest int leader = 13371337; for (auto& image : impl->loadedShapes[shape.get()].images) { - if (std::abs((int)(image->side - info.size)) > std::abs((int)(leader - info.size))) + if (std::abs((int)(image->side - PIXELSIDE)) > std::abs((int)(leader - PIXELSIDE))) continue; leader = image->side; @@ -392,8 +394,9 @@ SCursorRawShapeDataC* CHyprcursorManager::getRawShapeDataC(const char* shape_) { data->overridenBy = nullptr; data->images = nullptr; data->len = 0; - data->hotspotX = 0; - data->hotspotY = 0; + data->hotspotX = 0.f; + data->hotspotY = 0.F; + data->nominalSize = 1.F; data->resizeAlgo = eHyprcursorResizeAlgo::HC_RESIZE_NONE; data->type = eHyprcursorDataType::HC_DATA_PNG; @@ -418,9 +421,10 @@ SCursorRawShapeDataC* CHyprcursorManager::getRawShapeDataC(const char* shape_) { resultingImages.push_back(i.get()); } - data->hotspotX = shape->hotspotX; - data->hotspotY = shape->hotspotY; - data->type = shape->shapeType == SHAPE_PNG ? HC_DATA_PNG : HC_DATA_SVG; + data->hotspotX = shape->hotspotX; + data->hotspotY = shape->hotspotY; + data->nominalSize = shape->nominalSize; + data->type = shape->shapeType == SHAPE_PNG ? HC_DATA_PNG : HC_DATA_SVG; break; } @@ -494,12 +498,16 @@ bool CHyprcursorManager::loadThemeStyle(const SCursorStyleInfo& info) { Debug::log(HC_LOG_TRACE, logFn, "loadThemeStyle: png shape {} has {} frames", shape->directory, FRAMES.size()); + const int PIXELSIDE = std::round(info.size / shape->nominalSize); + + Debug::log(HC_LOG_TRACE, logFn, "loadThemeStyle: png shape has nominal {:.2f}, pixel size will be {}x", shape->nominalSize, PIXELSIDE); + for (auto& f : FRAMES) { auto& newImage = impl->loadedShapes[shape.get()].images.emplace_back(std::make_unique()); newImage->artificial = true; - newImage->side = info.size; - newImage->artificialData = new char[info.size * info.size * 4]; - newImage->cairoSurface = cairo_image_surface_create_for_data((unsigned char*)newImage->artificialData, CAIRO_FORMAT_ARGB32, info.size, info.size, info.size * 4); + newImage->side = PIXELSIDE; + newImage->artificialData = new char[PIXELSIDE * PIXELSIDE * 4]; + newImage->cairoSurface = cairo_image_surface_create_for_data((unsigned char*)newImage->artificialData, CAIRO_FORMAT_ARGB32, PIXELSIDE, PIXELSIDE, PIXELSIDE * 4); newImage->delay = f->delay; const auto PCAIRO = cairo_create(newImage->cairoSurface); @@ -513,12 +521,12 @@ bool CHyprcursorManager::loadThemeStyle(const SCursorStyleInfo& info) { const auto PTN = cairo_pattern_create_for_surface(f->cairoSurface); cairo_pattern_set_extend(PTN, CAIRO_EXTEND_NONE); - const float scale = info.size / (float)f->side; + const float scale = PIXELSIDE / (float)f->side; cairo_scale(PCAIRO, scale, scale); cairo_pattern_set_filter(PTN, shape->resizeAlgo == HC_RESIZE_BILINEAR ? CAIRO_FILTER_GOOD : CAIRO_FILTER_NEAREST); cairo_set_source(PCAIRO, PTN); - cairo_rectangle(PCAIRO, 0, 0, info.size, info.size); + cairo_rectangle(PCAIRO, 0, 0, PIXELSIDE, PIXELSIDE); cairo_fill(PCAIRO); cairo_surface_flush(newImage->cairoSurface); @@ -531,12 +539,16 @@ bool CHyprcursorManager::loadThemeStyle(const SCursorStyleInfo& info) { Debug::log(HC_LOG_TRACE, logFn, "loadThemeStyle: svg shape {} has {} frames", shape->directory, FRAMES.size()); + const int PIXELSIDE = std::round(info.size / shape->nominalSize); + + Debug::log(HC_LOG_TRACE, logFn, "loadThemeStyle: svg shape has nominal {:.2f}, pixel size will be {}x", shape->nominalSize, PIXELSIDE); + for (auto& f : FRAMES) { auto& newImage = impl->loadedShapes[shape.get()].images.emplace_back(std::make_unique()); newImage->artificial = true; - newImage->side = info.size; - newImage->artificialData = new char[info.size * info.size * 4]; - newImage->cairoSurface = cairo_image_surface_create_for_data((unsigned char*)newImage->artificialData, CAIRO_FORMAT_ARGB32, info.size, info.size, info.size * 4); + newImage->side = PIXELSIDE; + newImage->artificialData = new char[PIXELSIDE * PIXELSIDE * 4]; + newImage->cairoSurface = cairo_image_surface_create_for_data((unsigned char*)newImage->artificialData, CAIRO_FORMAT_ARGB32, PIXELSIDE, PIXELSIDE, PIXELSIDE * 4); newImage->delay = f->delay; const auto PCAIRO = cairo_create(newImage->cairoSurface); @@ -584,7 +596,7 @@ void CHyprcursorManager::cursorSurfaceStyleDone(const SCursorStyleInfo& info) { const bool isArtificial = e->artificial; // clean artificial rasters made for this - if (isArtificial && e->side == info.size) + if (isArtificial && e->side == std::round(info.size / shape->nominalSize)) return true; // clean invalid non-svg rasters diff --git a/libhyprcursor/internalSharedTypes.hpp b/libhyprcursor/internalSharedTypes.hpp index 3a032aa..3775d53 100644 --- a/libhyprcursor/internalSharedTypes.hpp +++ b/libhyprcursor/internalSharedTypes.hpp @@ -37,7 +37,7 @@ struct SCursorImage { struct SCursorShape { std::string directory; - float hotspotX = 0, hotspotY = 0; + float hotspotX = 0, hotspotY = 0, nominalSize = 1.F; eHyprcursorResizeAlgo resizeAlgo = HC_RESIZE_NEAREST; std::vector images; std::vector overrides; diff --git a/libhyprcursor/meta.cpp b/libhyprcursor/meta.cpp index 23b2c5b..38ea262 100644 --- a/libhyprcursor/meta.cpp +++ b/libhyprcursor/meta.cpp @@ -142,6 +142,7 @@ std::optional CMeta::parseHL() { meta = std::make_unique(rawdata.c_str(), Hyprlang::SConfigOptions{.pathIsStream = !dataPath}); meta->addConfigValue("hotspot_x", Hyprlang::FLOAT{0.F}); meta->addConfigValue("hotspot_y", Hyprlang::FLOAT{0.F}); + meta->addConfigValue("nominal_size", Hyprlang::FLOAT{1.F}); meta->addConfigValue("resize_algorithm", Hyprlang::STRING{"nearest"}); meta->registerHandler(::parseDefineSize, "define_size", {.allowFlags = false}); meta->registerHandler(::parseOverride, "define_override", {.allowFlags = false}); @@ -164,6 +165,7 @@ std::optional CMeta::parseTOML() { parsedData.hotspotX = MANIFEST["General"]["hotspot_x"].value_or(0.f); parsedData.hotspotY = MANIFEST["General"]["hotspot_y"].value_or(0.f); + parsedData.hotspotY = MANIFEST["General"]["nominal_size"].value_or(1.f); const std::string OVERRIDES = MANIFEST["General"]["define_override"].value_or(""); const std::string SIZES = MANIFEST["General"]["define_size"].value_or(""); diff --git a/libhyprcursor/meta.hpp b/libhyprcursor/meta.hpp index 837d5ee..9f0cb18 100644 --- a/libhyprcursor/meta.hpp +++ b/libhyprcursor/meta.hpp @@ -20,7 +20,7 @@ class CMeta { struct { std::string resizeAlgo; - float hotspotX = 0, hotspotY = 0; + float hotspotX = 0, hotspotY = 0, nominalSize = 1.F; std::vector overrides; std::vector definedSizes; } parsedData; From 29159159f97dcbe95b83d3064f9c20a5bcc3d12f Mon Sep 17 00:00:00 2001 From: Vaxry Date: Wed, 18 Dec 2024 15:48:05 +0000 Subject: [PATCH 2/6] oopse --- libhyprcursor/meta.cpp | 1 + 1 file changed, 1 insertion(+) diff --git a/libhyprcursor/meta.cpp b/libhyprcursor/meta.cpp index 38ea262..a5897ef 100644 --- a/libhyprcursor/meta.cpp +++ b/libhyprcursor/meta.cpp @@ -154,6 +154,7 @@ std::optional CMeta::parseHL() { parsedData.hotspotX = std::any_cast(meta->getConfigValue("hotspot_x")); parsedData.hotspotY = std::any_cast(meta->getConfigValue("hotspot_y")); + parsedData.hotspotY = std::any_cast(meta->getConfigValue("nominal_size")); parsedData.resizeAlgo = std::any_cast(meta->getConfigValue("resize_algorithm")); return {}; From f8de2a65dba528a888468eae344e20e37bd3057c Mon Sep 17 00:00:00 2001 From: Vaxry Date: Wed, 18 Dec 2024 15:52:06 +0000 Subject: [PATCH 3/6] thing --- libhyprcursor/hyprcursor.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/libhyprcursor/hyprcursor.cpp b/libhyprcursor/hyprcursor.cpp index e0926f6..d28c6fa 100644 --- a/libhyprcursor/hyprcursor.cpp +++ b/libhyprcursor/hyprcursor.cpp @@ -455,7 +455,7 @@ bool CHyprcursorManager::loadThemeStyle(const SCursorStyleInfo& info) { if (shape->shapeType == SHAPE_PNG) { for (auto& image : impl->loadedShapes[shape.get()].images) { - if (image->side != info.size) + if (image->side != std::round(info.size / shape->nominalSize)) continue; sizeFound = true; From 3d3e8ad1e6282ea405119397467e310fe156a713 Mon Sep 17 00:00:00 2001 From: Vaxry Date: Wed, 18 Dec 2024 15:57:24 +0000 Subject: [PATCH 4/6] stuf --- libhyprcursor/hyprcursor.cpp | 9 +++++---- libhyprcursor/meta.cpp | 8 ++++---- tests/full_rendering.cpp | 2 +- 3 files changed, 10 insertions(+), 9 deletions(-) diff --git a/libhyprcursor/hyprcursor.cpp b/libhyprcursor/hyprcursor.cpp index d28c6fa..5fb66a5 100644 --- a/libhyprcursor/hyprcursor.cpp +++ b/libhyprcursor/hyprcursor.cpp @@ -778,10 +778,11 @@ std::optional CHyprcursorImplementation::loadTheme() { if (SHAPE->images.empty()) return "meta invalid: no images for shape " + cursor.path().stem().string(); - SHAPE->directory = cursor.path().stem().string(); - SHAPE->hotspotX = meta.parsedData.hotspotX; - SHAPE->hotspotY = meta.parsedData.hotspotY; - SHAPE->resizeAlgo = stringToAlgo(meta.parsedData.resizeAlgo); + SHAPE->directory = cursor.path().stem().string(); + SHAPE->hotspotX = meta.parsedData.hotspotX; + SHAPE->hotspotY = meta.parsedData.hotspotY; + SHAPE->nominalSize = meta.parsedData.nominalSize; + SHAPE->resizeAlgo = stringToAlgo(meta.parsedData.resizeAlgo); zip_discard(zip); } diff --git a/libhyprcursor/meta.cpp b/libhyprcursor/meta.cpp index a5897ef..1ee9bb2 100644 --- a/libhyprcursor/meta.cpp +++ b/libhyprcursor/meta.cpp @@ -152,10 +152,10 @@ std::optional CMeta::parseHL() { return RESULT.getError(); } catch (const char* err) { return "failed parsing meta: " + std::string{err}; } - parsedData.hotspotX = std::any_cast(meta->getConfigValue("hotspot_x")); - parsedData.hotspotY = std::any_cast(meta->getConfigValue("hotspot_y")); - parsedData.hotspotY = std::any_cast(meta->getConfigValue("nominal_size")); - parsedData.resizeAlgo = std::any_cast(meta->getConfigValue("resize_algorithm")); + parsedData.hotspotX = std::any_cast(meta->getConfigValue("hotspot_x")); + parsedData.hotspotY = std::any_cast(meta->getConfigValue("hotspot_y")); + parsedData.nominalSize = std::any_cast(meta->getConfigValue("nominal_size")); + parsedData.resizeAlgo = std::any_cast(meta->getConfigValue("resize_algorithm")); return {}; } diff --git a/tests/full_rendering.cpp b/tests/full_rendering.cpp index b784ea1..728c4ce 100644 --- a/tests/full_rendering.cpp +++ b/tests/full_rendering.cpp @@ -17,7 +17,7 @@ int main(int argc, char** argv) { /* Create a manager. You can optionally pass a logger function. */ - Hyprcursor::CHyprcursorManager mgr(nullptr, logFunction); + Hyprcursor::CHyprcursorManager mgr("HyprBibataModernClassicSVG", logFunction); /* Manager could be invalid if no themes were found, or From 0af712e151478f402559dce946fb16d0605486e9 Mon Sep 17 00:00:00 2001 From: Vaxry Date: Wed, 18 Dec 2024 15:59:18 +0000 Subject: [PATCH 5/6] e --- libhyprcursor/hyprcursor.cpp | 12 +++++++----- 1 file changed, 7 insertions(+), 5 deletions(-) diff --git a/libhyprcursor/hyprcursor.cpp b/libhyprcursor/hyprcursor.cpp index 5fb66a5..42aed3e 100644 --- a/libhyprcursor/hyprcursor.cpp +++ b/libhyprcursor/hyprcursor.cpp @@ -454,8 +454,10 @@ bool CHyprcursorManager::loadThemeStyle(const SCursorStyleInfo& info) { bool sizeFound = false; if (shape->shapeType == SHAPE_PNG) { + const int IDEALSIDE = std::round(info.size / shape->nominalSize); + for (auto& image : impl->loadedShapes[shape.get()].images) { - if (image->side != std::round(info.size / shape->nominalSize)) + if (image->side != IDEALSIDE) continue; sizeFound = true; @@ -469,7 +471,7 @@ bool CHyprcursorManager::loadThemeStyle(const SCursorStyleInfo& info) { SLoadedCursorImage* leader = nullptr; int leaderVal = 1000000; for (auto& image : impl->loadedShapes[shape.get()].images) { - if (image->side < info.size) + if (image->side < IDEALSIDE) continue; if (image->side > leaderVal) @@ -481,7 +483,7 @@ bool CHyprcursorManager::loadThemeStyle(const SCursorStyleInfo& info) { if (!leader) { for (auto& image : impl->loadedShapes[shape.get()].images) { - if (std::abs((int)(image->side - info.size)) > leaderVal) + if (std::abs((int)(image->side - IDEALSIDE)) > leaderVal) continue; leaderVal = image->side; @@ -498,7 +500,7 @@ bool CHyprcursorManager::loadThemeStyle(const SCursorStyleInfo& info) { Debug::log(HC_LOG_TRACE, logFn, "loadThemeStyle: png shape {} has {} frames", shape->directory, FRAMES.size()); - const int PIXELSIDE = std::round(info.size / shape->nominalSize); + const int PIXELSIDE = std::round(leader->side / shape->nominalSize); Debug::log(HC_LOG_TRACE, logFn, "loadThemeStyle: png shape has nominal {:.2f}, pixel size will be {}x", shape->nominalSize, PIXELSIDE); @@ -566,7 +568,7 @@ bool CHyprcursorManager::loadThemeStyle(const SCursorStyleInfo& info) { return false; } - RsvgRectangle rect = {0, 0, (double)info.size, (double)info.size}; + RsvgRectangle rect = {0, 0, (double)PIXELSIDE, (double)PIXELSIDE}; if (!rsvg_handle_render_document(handle, PCAIRO, &rect, &error)) { Debug::log(HC_LOG_ERR, logFn, "Failed rendering svg: {}", error->message); From 3ca41b74c03b8adb42e3df615d5ca8b2072a8305 Mon Sep 17 00:00:00 2001 From: Vaxry Date: Wed, 18 Dec 2024 16:00:12 +0000 Subject: [PATCH 6/6] vaxry moment --- tests/full_rendering.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/full_rendering.cpp b/tests/full_rendering.cpp index 728c4ce..b784ea1 100644 --- a/tests/full_rendering.cpp +++ b/tests/full_rendering.cpp @@ -17,7 +17,7 @@ int main(int argc, char** argv) { /* Create a manager. You can optionally pass a logger function. */ - Hyprcursor::CHyprcursorManager mgr("HyprBibataModernClassicSVG", logFunction); + Hyprcursor::CHyprcursorManager mgr(nullptr, logFunction); /* Manager could be invalid if no themes were found, or