16 Commits

117 changed files with 5212 additions and 2691 deletions

View File

@@ -49,6 +49,7 @@ jobs:
sudo apt-get update sudo apt-get update
sudo apt-get install -y cmake build-essential git \ sudo apt-get install -y cmake build-essential git \
libglfw3-dev libfreetype6-dev libfontconfig1-dev \ libglfw3-dev libfreetype6-dev libfontconfig1-dev \
zlib1g-dev libharfbuzz-dev \
libx11-dev libxrandr-dev libxi-dev libgtk-3-dev \ libx11-dev libxrandr-dev libxi-dev libgtk-3-dev \
mesa-common-dev libgl1-mesa-dev libglu1-mesa-dev \ mesa-common-dev libgl1-mesa-dev libglu1-mesa-dev \
libxinerama-dev libxcursor-dev libxkbcommon-dev libxinerama-dev libxcursor-dev libxkbcommon-dev
@@ -83,6 +84,7 @@ jobs:
run: | run: |
dnf install -y cmake gcc gcc-c++ make rpm-build git \ dnf install -y cmake gcc gcc-c++ make rpm-build git \
glfw-devel freetype-devel fontconfig-devel \ glfw-devel freetype-devel fontconfig-devel \
zlib-devel harfbuzz-devel \
libX11-devel libXrandr-devel libXi-devel \ libX11-devel libXrandr-devel libXi-devel \
mesa-libGL-devel mesa-libGLU-devel \ mesa-libGL-devel mesa-libGLU-devel \
libXinerama-devel libXcursor-devel \ libXinerama-devel libXcursor-devel \

2
.gitignore vendored
View File

@@ -3,6 +3,8 @@
.vs .vs
out out
.clangd
build/ build/
build-msvc/ build-msvc/
CMakeCache.txt CMakeCache.txt

27
.vscode/launch.json vendored
View File

@@ -1,7 +1,4 @@
{ {
// Use IntelliSense to learn about possible attributes.
// Hover to view descriptions of existing attributes.
// For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387
"version": "0.2.0", "version": "0.2.0",
"configurations": [ "configurations": [
{ {
@@ -9,11 +6,7 @@
"type": "cppdbg", "type": "cppdbg",
"request": "launch", "request": "launch",
"program": "${command:cmake.launchTargetPath}", "program": "${command:cmake.launchTargetPath}",
"args": [ "args": [],
"--apply",
"--theme",
"dark"
],
"stopAtEntry": false, "stopAtEntry": false,
"cwd": "${workspaceFolder}", "cwd": "${workspaceFolder}",
"environment": [], "environment": [],
@@ -37,16 +30,24 @@
"type": "cppdbg", "type": "cppdbg",
"request": "launch", "request": "launch",
"program": "${command:cmake.launchTargetPath}", "program": "${command:cmake.launchTargetPath}",
"args": [ "args": [],
"--apply",
"--theme",
"dark"
],
"stopAtEntry": false, "stopAtEntry": false,
"cwd": "${workspaceFolder}", "cwd": "${workspaceFolder}",
"environment": [], "environment": [],
"externalConsole": false, "externalConsole": false,
"MIMode": "lldb", "MIMode": "lldb",
"miDebuggerPath": "lldb"
}, },
{
"name": "Debug current target (MSVC)",
"type": "cppvsdbg",
"request": "launch",
"program": "${command:cmake.launchTargetPath}",
"args": [],
"stopAtEntry": false,
"cwd": "${workspaceFolder}",
"environment": [],
"console": "integratedTerminal"
}
] ]
} }

3
.vscode/settnigs.json vendored Normal file
View File

@@ -0,0 +1,3 @@
{
"cmake.useCMakePresets": "always"
}

View File

@@ -1,6 +1,6 @@
# Maintainer: Daniel Dada <dan@binarygoose.dev> # Maintainer: Daniel Dada <dan@binarygoose.dev>
pkgname=clrsync pkgname=clrsync
pkgver=0.1.6 pkgver=1.0.0
pkgrel=1 pkgrel=1
pkgdesc="Color scheme manager" pkgdesc="Color scheme manager"
arch=('x86_64') arch=('x86_64')

View File

@@ -10,6 +10,8 @@ depends=(
glfw glfw
freetype2 freetype2
fontconfig fontconfig
zlib
harfbuzz
mesa mesa
libglvnd libglvnd
libxcursor libxcursor

View File

@@ -1,5 +1,5 @@
cmake_minimum_required(VERSION 3.25) cmake_minimum_required(VERSION 3.25)
project(clrsync VERSION 0.1.6 LANGUAGES CXX) project(clrsync VERSION 1.0.0 LANGUAGES CXX)
include(GNUInstallDirs) include(GNUInstallDirs)
@@ -10,6 +10,9 @@ set(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR})
set(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR}) set(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR})
set(CMAKE_EXPORT_COMPILE_COMMANDS ON) set(CMAKE_EXPORT_COMPILE_COMMANDS ON)
file(WRITE "${CMAKE_SOURCE_DIR}/.clangd"
"CompileFlags:\n CompilationDatabase: ${CMAKE_BINARY_DIR}\n")
option(USE_SYSTEM_GLFW "Use system-installed GLFW instead of fetching it statically" OFF) option(USE_SYSTEM_GLFW "Use system-installed GLFW instead of fetching it statically" OFF)
message(STATUS "USE_SYSTEM_GLFW: ${USE_SYSTEM_GLFW}") message(STATUS "USE_SYSTEM_GLFW: ${USE_SYSTEM_GLFW}")
@@ -57,8 +60,8 @@ endif()
message(STATUS "clrsync version: ${SEMVER}") message(STATUS "clrsync version: ${SEMVER}")
configure_file( configure_file(
${CMAKE_SOURCE_DIR}/src/core/version.hpp.in ${CMAKE_SOURCE_DIR}/src/core/common/version.hpp.in
${CMAKE_SOURCE_DIR}/src/core/version.hpp ${CMAKE_SOURCE_DIR}/src/core/common/version.hpp
@ONLY @ONLY
) )

325
CMakePresets.json Normal file
View File

@@ -0,0 +1,325 @@
{
"version": 6,
"configurePresets": [
{
"name": "base",
"hidden": true
},
{
"name": "debug",
"hidden": true,
"inherits": "base",
"cacheVariables": {
"CMAKE_BUILD_TYPE": "Debug"
}
},
{
"name": "release",
"hidden": true,
"inherits": "base",
"cacheVariables": {
"CMAKE_BUILD_TYPE": "Release"
}
},
{
"name": "linux",
"hidden": true,
"inherits": "base",
"condition": {
"type": "equals",
"lhs": "${hostSystemName}",
"rhs": "Linux"
}
},
{
"name": "windows",
"hidden": true,
"inherits": "base",
"condition": {
"type": "equals",
"lhs": "${hostSystemName}",
"rhs": "Windows"
}
},
{
"name": "macos",
"hidden": true,
"inherits": "base",
"condition": {
"type": "equals",
"lhs": "${hostSystemName}",
"rhs": "Darwin"
}
},
{
"name": "gcc",
"hidden": true,
"cacheVariables": {
"CMAKE_C_COMPILER": "gcc",
"CMAKE_CXX_COMPILER": "g++"
}
},
{
"name": "clang",
"hidden": true,
"cacheVariables": {
"CMAKE_C_COMPILER": "clang",
"CMAKE_CXX_COMPILER": "clang++"
}
},
{
"name": "ninja",
"hidden": true,
"generator": "Ninja"
},
{
"name": "make",
"hidden": true,
"generator": "Unix Makefiles"
},
{
"name": "msvc-ninja",
"displayName": "MSVC (Ninja)",
"generator": "Ninja",
"hidden": true,
"binaryDir": "${sourceDir}/build/windows/msvc",
"cacheVariables": {
"CMAKE_C_COMPILER": "cl",
"CMAKE_CXX_COMPILER": "cl"
}
},
{
"name": "msvc-vs2026",
"displayName": "MSVC (Visual Studio 18 2026)",
"generator": "Visual Studio 18 2026",
"binaryDir": "${sourceDir}/build/windows/msvc/vs2026",
"architecture": {
"value": "x64"
}
},
{
"name": "windows-msvc-ninja-debug",
"inherits": [
"windows",
"msvc-ninja",
"ninja",
"debug"
],
"displayName": "Windows · MSVC · Ninja · Debug",
"binaryDir": "${sourceDir}/build/windows/msvc/ninja/debug"
},
{
"name": "windows-msvc-ninja-release",
"inherits": [
"windows",
"msvc-ninja",
"ninja",
"release"
],
"displayName": "Windows · MSVC · Ninja · Release",
"binaryDir": "${sourceDir}/build/windows/msvc/ninja/release"
},
{
"name": "linux-gcc-ninja-debug",
"inherits": [
"linux",
"gcc",
"ninja",
"debug"
],
"displayName": "Linux · GCC · Ninja · Debug",
"binaryDir": "${sourceDir}/build/linux/gcc/ninja/debug"
},
{
"name": "linux-gcc-ninja-release",
"inherits": [
"linux",
"gcc",
"ninja",
"release"
],
"displayName": "Linux · GCC · Ninja · Release",
"binaryDir": "${sourceDir}/build/linux/gcc/ninja/release"
},
{
"name": "linux-gcc-make-debug",
"inherits": [
"linux",
"gcc",
"make",
"debug"
],
"displayName": "Linux · GCC · Make · Debug",
"binaryDir": "${sourceDir}/build/linux/gcc/make/debug"
},
{
"name": "linux-gcc-make-release",
"inherits": [
"linux",
"gcc",
"make",
"release"
],
"displayName": "Linux · GCC · Make · Release",
"binaryDir": "${sourceDir}/build/linux/gcc/make/release"
},
{
"name": "linux-clang-ninja-debug",
"inherits": [
"linux",
"clang",
"ninja",
"debug"
],
"displayName": "Linux · Clang · Ninja · Debug",
"binaryDir": "${sourceDir}/build/linux/clang/ninja/debug"
},
{
"name": "linux-clang-ninja-release",
"inherits": [
"linux",
"clang",
"ninja",
"release"
],
"displayName": "Linux · Clang · Ninja · Release",
"binaryDir": "${sourceDir}/build/linux/clang/ninja/release"
},
{
"name": "linux-clang-make-debug",
"inherits": [
"linux",
"clang",
"make",
"debug"
],
"displayName": "Linux · Clang · Make · Debug",
"binaryDir": "${sourceDir}/build/linux/clang/make/debug"
},
{
"name": "linux-clang-make-release",
"inherits": [
"linux",
"clang",
"make",
"release"
],
"displayName": "Linux · Clang · Make · Release",
"binaryDir": "${sourceDir}/build/linux/clang/make/release"
},
{
"name": "macos-appleclang-ninja-debug",
"inherits": [
"macos",
"clang",
"ninja",
"debug"
],
"displayName": "macOS · Apple Clang · Ninja · Debug",
"binaryDir": "${sourceDir}/build/macos/appleclang/ninja/debug"
},
{
"name": "macos-appleclang-ninja-release",
"inherits": [
"macos",
"clang",
"ninja",
"release"
],
"displayName": "macOS · Apple Clang · Ninja · Release",
"binaryDir": "${sourceDir}/build/macos/appleclang/ninja/release"
},
{
"name": "macos-appleclang-make-debug",
"inherits": [
"macos",
"clang",
"make",
"debug"
],
"displayName": "macOS · Apple Clang · Make · Debug",
"binaryDir": "${sourceDir}/build/macos/appleclang/make/debug"
},
{
"name": "macos-appleclang-make-release",
"inherits": [
"macos",
"clang",
"make",
"release"
],
"displayName": "macOS · Apple Clang · Make · Release",
"binaryDir": "${sourceDir}/build/macos/appleclang/make/release"
}
],
"buildPresets": [
{
"name": "vs-debug",
"configurePreset": "msvc-vs2026",
"configuration": "Debug"
},
{
"name": "vs-release",
"configurePreset": "msvc-vs2026",
"configuration": "Release"
},
{
"name": "msvc-ninja-debug",
"configurePreset": "windows-msvc-ninja-debug",
"configuration": "Debug"
},
{
"name": "msvc-ninja-release",
"configurePreset": "windows-msvc-ninja-release",
"configuration": "Release"
},
{
"name": "linux-gcc-ninja-debug",
"configurePreset": "linux-gcc-ninja-debug"
},
{
"name": "linux-gcc-ninja-release",
"configurePreset": "linux-gcc-ninja-release"
},
{
"name": "linux-gcc-make-debug",
"configurePreset": "linux-gcc-make-debug"
},
{
"name": "linux-gcc-make-release",
"configurePreset": "linux-gcc-make-release"
},
{
"name": "linux-clang-ninja-debug",
"configurePreset": "linux-clang-ninja-debug"
},
{
"name": "linux-clang-ninja-release",
"configurePreset": "linux-clang-ninja-release"
},
{
"name": "linux-clang-make-debug",
"configurePreset": "linux-clang-make-debug"
},
{
"name": "linux-clang-make-release",
"configurePreset": "linux-clang-make-release"
},
{
"name": "macos-appleclang-ninja-debug",
"configurePreset": "macos-appleclang-ninja-debug"
},
{
"name": "macos-appleclang-ninja-release",
"configurePreset": "macos-appleclang-ninja-release"
},
{
"name": "macos-appleclang-make-debug",
"configurePreset": "macos-appleclang-make-debug"
},
{
"name": "macos-appleclang-make-release",
"configurePreset": "macos-appleclang-make-release"
}
]
}

View File

@@ -1 +1 @@
0.1.6 1.0.0

Binary file not shown.

Before

Width:  |  Height:  |  Size: 370 KiB

After

Width:  |  Height:  |  Size: 64 KiB

View File

@@ -7,6 +7,7 @@ if(WIN32)
freetype freetype
URL https://download.savannah.gnu.org/releases/freetype/freetype-2.14.1.tar.gz URL https://download.savannah.gnu.org/releases/freetype/freetype-2.14.1.tar.gz
) )
FetchContent_MakeAvailable(freetype) FetchContent_MakeAvailable(freetype)
elseif(APPLE) elseif(APPLE)
option(USE_SYSTEM_GLFW ON) option(USE_SYSTEM_GLFW ON)
@@ -15,10 +16,6 @@ elseif(APPLE)
find_package(BZip2 REQUIRED) find_package(BZip2 REQUIRED)
find_package(PNG REQUIRED) find_package(PNG REQUIRED)
find_library(BROTLIDEC_LIBRARY NAMES brotlidec)
find_library(BROTLICOMMON_LIBRARY NAMES brotlicommon)
find_package(PkgConfig QUIET) find_package(PkgConfig QUIET)
if(PkgConfig_FOUND) if(PkgConfig_FOUND)
pkg_check_modules(HARFBUZZ harfbuzz) pkg_check_modules(HARFBUZZ harfbuzz)
@@ -30,10 +27,8 @@ else()
find_package(ZLIB REQUIRED) find_package(ZLIB REQUIRED)
find_package(BZip2 REQUIRED) find_package(BZip2 REQUIRED)
find_package(PNG REQUIRED) find_package(PNG REQUIRED)
find_library(BROTLIDEC_LIBRARY NAMES brotlidec) find_library(BROTLIDEC_LIBRARY NAMES brotlidec)
find_library(BROTLICOMMON_LIBRARY NAMES brotlicommon) find_library(BROTLICOMMON_LIBRARY NAMES brotlicommon)
pkg_check_modules(HARFBUZZ harfbuzz) pkg_check_modules(HARFBUZZ harfbuzz)
pkg_check_modules(WAYLAND_CLIENT wayland-client) pkg_check_modules(WAYLAND_CLIENT wayland-client)
pkg_check_modules(WAYLAND_EGL wayland-egl) pkg_check_modules(WAYLAND_EGL wayland-egl)
@@ -81,7 +76,6 @@ if(BROTLIDEC_LIBRARY AND BROTLICOMMON_LIBRARY)
list(APPEND FREETYPE_EXTRA_LIBS ${BROTLIDEC_LIBRARY} ${BROTLICOMMON_LIBRARY}) list(APPEND FREETYPE_EXTRA_LIBS ${BROTLIDEC_LIBRARY} ${BROTLICOMMON_LIBRARY})
message(STATUS "Found Brotli libraries") message(STATUS "Found Brotli libraries")
endif() endif()
if(HARFBUZZ_FOUND) if(HARFBUZZ_FOUND)
list(APPEND FREETYPE_EXTRA_LIBS ${HARFBUZZ_LIBRARIES}) list(APPEND FREETYPE_EXTRA_LIBS ${HARFBUZZ_LIBRARIES})
message(STATUS "Found HarfBuzz") message(STATUS "Found HarfBuzz")

View File

@@ -27,7 +27,7 @@ set(CPACK_NSIS_CREATE_DESKTOP_LINKS "bin/clrsync_gui.exe;clrsync")
# Debian # Debian
set(CPACK_DEBIAN_PACKAGE_MAINTAINER "Daniel Dada <dan@binarygoose.dev>") set(CPACK_DEBIAN_PACKAGE_MAINTAINER "Daniel Dada <dan@binarygoose.dev>")
set(CPACK_DEBIAN_PACKAGE_DEPENDS "libc6 (>= 2.31), libglfw3, libfreetype6") set(CPACK_DEBIAN_PACKAGE_DEPENDS "libc6 (>= 2.31), libglfw3, libfreetype6, zlib1g, libharfbuzz0b")
set(CPACK_DEBIAN_PACKAGE_SECTION "utils") set(CPACK_DEBIAN_PACKAGE_SECTION "utils")
set(CPACK_DEBIAN_PACKAGE_PRIORITY "optional") set(CPACK_DEBIAN_PACKAGE_PRIORITY "optional")
set(CPACK_DEBIAN_PACKAGE_ARCHITECTURE "amd64") set(CPACK_DEBIAN_PACKAGE_ARCHITECTURE "amd64")
@@ -36,6 +36,6 @@ set(CPACK_DEBIAN_PACKAGE_ARCHITECTURE "amd64")
set(CPACK_RPM_PACKAGE_LICENSE "MIT") set(CPACK_RPM_PACKAGE_LICENSE "MIT")
set(CPACK_RPM_PACKAGE_GROUP "Applications/System") set(CPACK_RPM_PACKAGE_GROUP "Applications/System")
set(CPACK_RPM_PACKAGE_URL "https://github.com/obsqrbtz/clrsync") set(CPACK_RPM_PACKAGE_URL "https://github.com/obsqrbtz/clrsync")
set(CPACK_RPM_PACKAGE_REQUIRES "freetype, glfw, fontconfig") set(CPACK_RPM_PACKAGE_REQUIRES "freetype, glfw, fontconfig, zlib, harfbuzz")
include(CPack) include(CPack)

6
flake.lock generated
View File

@@ -2,11 +2,11 @@
"nodes": { "nodes": {
"nixpkgs": { "nixpkgs": {
"locked": { "locked": {
"lastModified": 1764950072, "lastModified": 1765779637,
"narHash": "sha256-BmPWzogsG2GsXZtlT+MTcAWeDK5hkbGRZTeZNW42fwA=", "narHash": "sha256-KJ2wa/BLSrTqDjbfyNx70ov/HdgNBCBBSQP3BIzKnv4=",
"owner": "NixOS", "owner": "NixOS",
"repo": "nixpkgs", "repo": "nixpkgs",
"rev": "f61125a668a320878494449750330ca58b78c557", "rev": "1306659b587dc277866c7b69eb97e5f07864d8c4",
"type": "github" "type": "github"
}, },
"original": { "original": {

View File

@@ -4,15 +4,15 @@
#include <argparse/argparse.hpp> #include <argparse/argparse.hpp>
#include <core/config/config.hpp> #include "core/common/error.hpp"
#include <core/error.hpp> #include "core/common/utils.hpp"
#include <core/io/toml_file.hpp> #include "core/common/version.hpp"
#include <core/palette/palette_file.hpp> #include "core/config/config.hpp"
#include <core/palette/palette_manager.hpp> #include "core/io/toml_file.hpp"
#include <core/theme/theme_renderer.hpp> #include "core/palette/palette_file.hpp"
#include <core/theme/theme_template.hpp> #include "core/palette/palette_manager.hpp"
#include <core/utils.hpp> #include "core/theme/theme_renderer.hpp"
#include <core/version.hpp> #include "core/theme/theme_template.hpp"
void handle_show_vars() void handle_show_vars()
{ {

View File

@@ -2,8 +2,8 @@ set(CORE_SOURCES
palette/color.cpp palette/color.cpp
io/toml_file.cpp io/toml_file.cpp
config/config.cpp config/config.cpp
utils.cpp common/utils.cpp
version.cpp common/version.cpp
theme/theme_template.cpp theme/theme_template.cpp
) )

266
src/core/common/error.hpp Normal file
View File

@@ -0,0 +1,266 @@
#ifndef CLRSYNC_CORE_ERROR_HPP
#define CLRSYNC_CORE_ERROR_HPP
#include <optional>
#include <string>
#include <variant>
namespace clrsync::core
{
enum class error_code
{
unknown,
file_not_found,
file_open_failed,
file_write_failed,
file_read_failed,
dir_create_failed,
parse_failed,
invalid_format,
config_missing,
config_invalid,
template_not_found,
template_load_failed,
template_apply_failed,
palette_not_found,
palette_load_failed,
init_failed,
invalid_arg,
resource_missing,
};
inline const char *error_code_string(error_code code)
{
switch (code)
{
case error_code::unknown:
return "Unknown error";
case error_code::file_not_found:
return "File not found";
case error_code::file_open_failed:
return "Failed to open file";
case error_code::file_write_failed:
return "Failed to write file";
case error_code::file_read_failed:
return "Failed to read file";
case error_code::dir_create_failed:
return "Failed to create directory";
case error_code::parse_failed:
return "Parse failed";
case error_code::invalid_format:
return "Invalid format";
case error_code::config_missing:
return "Configuration missing";
case error_code::config_invalid:
return "Configuration invalid";
case error_code::template_not_found:
return "Template not found";
case error_code::template_load_failed:
return "Failed to load template";
case error_code::template_apply_failed:
return "Failed to apply template";
case error_code::palette_not_found:
return "Palette not found";
case error_code::palette_load_failed:
return "Failed to load palette";
case error_code::init_failed:
return "Initialization failed";
case error_code::invalid_arg:
return "Invalid argument";
case error_code::resource_missing:
return "Resource missing";
default:
return "Unknown error code";
}
}
struct Error
{
error_code code;
std::string message;
std::string context;
Error(error_code c) : code(c), message(error_code_string(c))
{
}
Error(error_code c, std::string msg) : code(c), message(std::move(msg))
{
}
Error(error_code c, std::string msg, std::string ctx)
: code(c), message(std::move(msg)), context(std::move(ctx))
{
}
std::string description() const
{
if (context.empty())
return message;
return message + " [" + context + "]";
}
};
template <typename T> class [[nodiscard]] Result
{
private:
std::variant<T, Error> m_data;
public:
Result(T value) : m_data(std::move(value))
{
}
Result(Error error) : m_data(std::move(error))
{
}
bool is_ok() const
{
return std::holds_alternative<T>(m_data);
}
bool is_error() const
{
return std::holds_alternative<Error>(m_data);
}
explicit operator bool() const
{
return is_ok();
}
T &value() &
{
return std::get<T>(m_data);
}
const T &value() const &
{
return std::get<T>(m_data);
}
T &&value() &&
{
return std::get<T>(std::move(m_data));
}
const Error &error() const
{
return std::get<Error>(m_data);
}
T value_or(T default_value) const
{
return is_ok() ? std::get<T>(m_data) : std::move(default_value);
}
std::optional<T> ok() const
{
if (is_ok())
return std::get<T>(m_data);
return std::nullopt;
}
std::optional<Error> err() const
{
if (is_error())
return std::get<Error>(m_data);
return std::nullopt;
}
template <typename F> auto map(F &&func) -> Result<decltype(func(std::declval<T>()))>
{
using U = decltype(func(std::declval<T>()));
if (is_ok())
return Result<U>(func(std::get<T>(m_data)));
return Result<U>(std::get<Error>(m_data));
}
template <typename F> auto and_then(F &&func) -> decltype(func(std::declval<T>()))
{
if (is_ok())
return func(std::get<T>(m_data));
using ResultType = decltype(func(std::declval<T>()));
return ResultType(std::get<Error>(m_data));
}
};
template <> class [[nodiscard]] Result<void>
{
private:
std::optional<Error> m_error;
public:
Result() : m_error(std::nullopt)
{
}
Result(Error error) : m_error(std::move(error))
{
}
bool is_ok() const
{
return !m_error.has_value();
}
bool is_error() const
{
return m_error.has_value();
}
explicit operator bool() const
{
return is_ok();
}
const Error &error() const
{
return *m_error;
}
std::optional<Error> err() const
{
return m_error;
}
};
template <typename T> Result<T> Ok(T value)
{
return Result<T>(std::move(value));
}
inline Result<void> Ok()
{
return Result<void>();
}
template <typename T> Result<T> Err(Error error)
{
return Result<T>(std::move(error));
}
template <typename T> Result<T> Err(error_code code)
{
return Result<T>(Error(code));
}
template <typename T> Result<T> Err(error_code code, std::string message)
{
return Result<T>(Error(code, std::move(message)));
}
template <typename T> Result<T> Err(error_code code, std::string message, std::string context)
{
return Result<T>(Error(code, std::move(message), std::move(context)));
}
} // namespace clrsync::core
#endif // CLRSYNC_CORE_ERROR_HPP

View File

@@ -1,6 +1,6 @@
#include "utils.hpp" #include "utils.hpp"
#include <iostream>
#include <filesystem> #include <filesystem>
#include <iostream>
namespace clrsync::core namespace clrsync::core
{ {

View File

@@ -1,10 +1,10 @@
#ifndef CLRSYNC_CORE_UTILS_HPP #ifndef CLRSYNC_CORE_UTILS_HPP
#define CLRSYNC_CORE_UTILS_HPP #define CLRSYNC_CORE_UTILS_HPP
#include <string>
#include <filesystem> #include <filesystem>
#include <string>
#include <core/palette/color_keys.hpp> #include "core/palette/color_keys.hpp"
namespace clrsync::core namespace clrsync::core
{ {

View File

@@ -1,4 +1,4 @@
#include "version.hpp" #include "core/common/version.hpp"
namespace clrsync::core namespace clrsync::core
{ {

View File

@@ -6,7 +6,7 @@
namespace clrsync::core namespace clrsync::core
{ {
const std::string GIT_SEMVER = "0.1.6+git.g1a1747a"; const std::string GIT_SEMVER = "0.1.7+git.gd17776b";
const std::string version_string(); const std::string version_string();
} // namespace clrsync::core } // namespace clrsync::core

View File

@@ -1,8 +1,9 @@
#include "config.hpp" #include "config.hpp"
#include "core/utils.hpp" #include "core/common/error.hpp"
#include "core/error.hpp" #include "core/common/utils.hpp"
#include "core/io/toml_file.hpp"
#include <core/palette/color.hpp> #include "core/palette/color.hpp"
#include <filesystem> #include <filesystem>
#include <fstream> #include <fstream>
@@ -28,7 +29,39 @@ Result<void> config::initialize(std::unique_ptr<clrsync::core::io::file> file)
auto parse_result = m_file->parse(); auto parse_result = m_file->parse();
if (!parse_result) if (!parse_result)
return Err<void>(error_code::config_invalid, parse_result.error().message, parse_result.error().context); return Err<void>(error_code::config_invalid, parse_result.error().message,
parse_result.error().context);
std::filesystem::path config_path = get_user_config_dir() / "config.toml";
std::filesystem::path temp_config_path = get_user_config_dir() / "config-temp.toml";
if (std::filesystem::exists(config_path))
{
std::error_code ec;
auto perms = std::filesystem::status(config_path, ec).permissions();
if (ec || (perms & std::filesystem::perms::owner_write) == std::filesystem::perms::none)
{
m_temp_config_path = temp_config_path.string();
if (std::filesystem::exists(temp_config_path))
{
try
{
auto temp_conf = std::make_unique<clrsync::core::io::toml_file>(temp_config_path.string());
auto temp_parse = temp_conf->parse();
if (temp_parse)
{
m_temp_file = std::move(temp_conf);
}
}
catch (const std::exception &e)
{
std::cerr << "Warning: Failed to load temp config: " << e.what() << std::endl;
}
}
}
}
return Ok(); return Ok();
} }
@@ -39,6 +72,30 @@ std::filesystem::path config::get_user_config_dir()
return home / ".config" / "clrsync"; return home / ".config" / "clrsync";
} }
std::filesystem::path config::get_user_state_dir()
{
std::filesystem::path home = normalize_path("~");
return home / ".local" / "state" / "clrsync";
}
std::filesystem::path config::get_writable_config_path()
{
std::filesystem::path config_path = get_user_config_dir() / "config.toml";
if (std::filesystem::exists(config_path))
{
std::error_code ec;
auto perms = std::filesystem::status(config_path, ec).permissions();
if (ec || (perms & std::filesystem::perms::owner_write) == std::filesystem::perms::none)
{
return get_user_config_dir() / "config-temp.toml";
}
}
return config_path;
}
std::filesystem::path config::get_data_dir() std::filesystem::path config::get_data_dir()
{ {
if (std::filesystem::exists(CLRSYNC_DATADIR)) if (std::filesystem::exists(CLRSYNC_DATADIR))
@@ -134,15 +191,50 @@ void config::copy_default_configs()
} }
} }
Result<void> config::save_config_value(const std::string &section, const std::string &key, const value_type &value)
{
if (!m_temp_config_path.empty())
{
if (!m_temp_file)
{
m_temp_file = std::make_unique<clrsync::core::io::toml_file>(m_temp_config_path);
(void)m_temp_file->parse();
}
m_temp_file->set_value(section, key, value);
return m_temp_file->save_file();
}
m_file->set_value(section, key, value);
return m_file->save_file();
}
const std::string &config::palettes_path() const std::string &config::palettes_path()
{ {
if (m_palettes_dir.empty() && m_file) if (m_palettes_dir.empty() && m_file)
{
if (m_temp_file)
{
auto temp_value = m_temp_file->get_string_value("general", "palettes_path");
if (!temp_value.empty())
{
m_palettes_dir = temp_value;
return m_palettes_dir;
}
}
m_palettes_dir = m_file->get_string_value("general", "palettes_path"); m_palettes_dir = m_file->get_string_value("general", "palettes_path");
}
return m_palettes_dir; return m_palettes_dir;
} }
const std::string config::default_theme() const const std::string config::default_theme() const
{ {
if (m_temp_file)
{
auto temp_value = m_temp_file->get_string_value("general", "default_theme");
if (!temp_value.empty())
return temp_value;
}
if (m_file) if (m_file)
return m_file->get_string_value("general", "default_theme"); return m_file->get_string_value("general", "default_theme");
return {}; return {};
@@ -150,6 +242,12 @@ const std::string config::default_theme() const
const std::string config::font() const const std::string config::font() const
{ {
if (m_temp_file)
{
auto temp_value = m_temp_file->get_string_value("general", "font");
if (!temp_value.empty())
return temp_value;
}
if (m_file) if (m_file)
return m_file->get_string_value("general", "font"); return m_file->get_string_value("general", "font");
return {}; return {};
@@ -157,6 +255,12 @@ const std::string config::font() const
const uint32_t config::font_size() const const uint32_t config::font_size() const
{ {
if (m_temp_file)
{
auto temp_value = m_temp_file->get_uint_value("general", "font_size");
if (temp_value != 0)
return temp_value;
}
if (m_file) if (m_file)
return m_file->get_uint_value("general", "font_size"); return m_file->get_uint_value("general", "font_size");
return 14; return 14;
@@ -167,8 +271,7 @@ Result<void> config::set_default_theme(const std::string &theme)
if (!m_file) if (!m_file)
return Err<void>(error_code::config_missing, "Configuration not initialized"); return Err<void>(error_code::config_missing, "Configuration not initialized");
m_file->set_value("general", "default_theme", theme); return save_config_value("general", "default_theme", theme);
return m_file->save_file();
} }
Result<void> config::set_palettes_path(const std::string &path) Result<void> config::set_palettes_path(const std::string &path)
@@ -176,8 +279,7 @@ Result<void> config::set_palettes_path(const std::string &path)
if (!m_file) if (!m_file)
return Err<void>(error_code::config_missing, "Configuration not initialized"); return Err<void>(error_code::config_missing, "Configuration not initialized");
m_file->set_value("general", "palettes_path", path); return save_config_value("general", "palettes_path", path);
return m_file->save_file();
} }
Result<void> config::set_font(const std::string &font) Result<void> config::set_font(const std::string &font)
@@ -185,16 +287,14 @@ Result<void> config::set_font(const std::string &font)
if (!m_file) if (!m_file)
return Err<void>(error_code::config_missing, "Configuration not initialized"); return Err<void>(error_code::config_missing, "Configuration not initialized");
m_file->set_value("general", "font", font); return save_config_value("general", "font", font);
return m_file->save_file();
} }
Result<void> config::set_font_size(int font_size) Result<void> config::set_font_size(int font_size)
{ {
if (!m_file) if (!m_file)
return Err<void>(error_code::config_missing, "Configuration not initialized"); return Err<void>(error_code::config_missing, "Configuration not initialized");
m_file->set_value("general", "font_size", font_size); return save_config_value("general", "font_size", static_cast<uint32_t>(font_size));
return m_file->save_file();
} }
Result<void> config::update_template(const std::string &key, Result<void> config::update_template(const std::string &key,
@@ -204,11 +304,17 @@ Result<void> config::update_template(const std::string &key,
return Err<void>(error_code::config_missing, "Configuration not initialized"); return Err<void>(error_code::config_missing, "Configuration not initialized");
m_themes[key] = theme_template; m_themes[key] = theme_template;
m_file->set_value("templates." + key, "input_path", theme_template.template_path());
m_file->set_value("templates." + key, "output_path", theme_template.output_path()); auto result1 = save_config_value("templates." + key, "input_path", theme_template.template_path());
m_file->set_value("templates." + key, "enabled", theme_template.enabled()); if (!result1) return result1;
m_file->set_value("templates." + key, "reload_cmd", theme_template.reload_command());
return m_file->save_file(); auto result2 = save_config_value("templates." + key, "output_path", theme_template.output_path());
if (!result2) return result2;
auto result3 = save_config_value("templates." + key, "enabled", theme_template.enabled());
if (!result3) return result3;
return save_config_value("templates." + key, "reload_cmd", theme_template.reload_command());
} }
Result<void> config::remove_template(const std::string &key) Result<void> config::remove_template(const std::string &key)
@@ -223,17 +329,31 @@ Result<void> config::remove_template(const std::string &key)
std::filesystem::path template_file = it->second.template_path(); std::filesystem::path template_file = it->second.template_path();
if (std::filesystem::exists(template_file)) if (std::filesystem::exists(template_file))
{ {
try { try
{
std::filesystem::remove(template_file); std::filesystem::remove(template_file);
} catch (const std::exception& e) { }
return Err<void>(error_code::file_write_failed, "Failed to delete template file", e.what()); catch (const std::exception &e)
{
return Err<void>(error_code::file_write_failed, "Failed to delete template file",
e.what());
} }
} }
m_themes.erase(it); m_themes.erase(it);
m_file->remove_section("templates." + key); if (!m_temp_config_path.empty())
{
if (!m_temp_file)
{
m_temp_file = std::make_unique<clrsync::core::io::toml_file>(m_temp_config_path);
(void)m_temp_file->parse();
}
m_temp_file->remove_section("templates." + key);
return m_temp_file->save_file();
}
m_file->remove_section("templates." + key);
return m_file->save_file(); return m_file->save_file();
} }
@@ -264,14 +384,16 @@ const std::unordered_map<std::string, clrsync::core::theme_template> config::tem
return m_themes; return m_themes;
} }
Result<const clrsync::core::theme_template*> config::template_by_name(const std::string &name) const Result<const clrsync::core::theme_template *> config::template_by_name(
const std::string &name) const
{ {
auto it = m_themes.find(name); auto it = m_themes.find(name);
if (it != m_themes.end()) if (it != m_themes.end())
{ {
return Ok(&it->second); return Ok(&it->second);
} }
return Err<const clrsync::core::theme_template*>(error_code::template_not_found, "Template not found", name); return Err<const clrsync::core::theme_template *>(error_code::template_not_found,
"Template not found", name);
} }
} // namespace clrsync::core } // namespace clrsync::core

View File

@@ -1,9 +1,9 @@
#ifndef CLRSYNC_CORE_CONFIG_HPP #ifndef CLRSYNC_CORE_CONFIG_HPP
#define CLRSYNC_CORE_CONFIG_HPP #define CLRSYNC_CORE_CONFIG_HPP
#include <core/io/file.hpp> #include "core/common/error.hpp"
#include <core/theme/theme_template.hpp> #include "core/io/file.hpp"
#include <core/error.hpp> #include "core/theme/theme_template.hpp"
#include <filesystem> #include <filesystem>
#include <memory> #include <memory>
#include <string> #include <string>
@@ -24,7 +24,8 @@ class config
const std::unordered_map<std::string, clrsync::core::theme_template> templates(); const std::unordered_map<std::string, clrsync::core::theme_template> templates();
Result<const clrsync::core::theme_template *> template_by_name(const std::string &name) const; Result<const clrsync::core::theme_template *> template_by_name(const std::string &name) const;
std::filesystem::path get_user_config_dir(); std::filesystem::path get_user_config_dir();
std::filesystem::path get_user_state_dir();
std::filesystem::path get_writable_config_path();
Result<void> set_default_theme(const std::string &theme); Result<void> set_default_theme(const std::string &theme);
Result<void> set_palettes_path(const std::string &path); Result<void> set_palettes_path(const std::string &path);
@@ -43,7 +44,10 @@ class config
std::string m_palettes_dir{}; std::string m_palettes_dir{};
std::unique_ptr<io::file> m_file; std::unique_ptr<io::file> m_file;
std::unique_ptr<io::file> m_temp_file;
std::string m_temp_config_path;
std::unordered_map<std::string, theme_template> m_themes{}; std::unordered_map<std::string, theme_template> m_themes{};
Result<void> save_config_value(const std::string &section, const std::string &key, const value_type &value);
static void copy_file(const std::filesystem::path &src, const std::filesystem::path &dst); static void copy_file(const std::filesystem::path &src, const std::filesystem::path &dst);
static void copy_dir(const std::filesystem::path &src, const std::filesystem::path &dst); static void copy_dir(const std::filesystem::path &src, const std::filesystem::path &dst);
void copy_default_configs(); void copy_default_configs();

View File

@@ -1,207 +0,0 @@
#ifndef CLRSYNC_CORE_ERROR_HPP
#define CLRSYNC_CORE_ERROR_HPP
#include <string>
#include <variant>
#include <optional>
namespace clrsync::core
{
enum class error_code
{
unknown,
file_not_found,
file_open_failed,
file_write_failed,
file_read_failed,
dir_create_failed,
parse_failed,
invalid_format,
config_missing,
config_invalid,
template_not_found,
template_load_failed,
template_apply_failed,
palette_not_found,
palette_load_failed,
init_failed,
invalid_arg,
resource_missing,
};
inline const char* error_code_string(error_code code)
{
switch (code)
{
case error_code::unknown: return "Unknown error";
case error_code::file_not_found: return "File not found";
case error_code::file_open_failed: return "Failed to open file";
case error_code::file_write_failed: return "Failed to write file";
case error_code::file_read_failed: return "Failed to read file";
case error_code::dir_create_failed: return "Failed to create directory";
case error_code::parse_failed: return "Parse failed";
case error_code::invalid_format: return "Invalid format";
case error_code::config_missing: return "Configuration missing";
case error_code::config_invalid: return "Configuration invalid";
case error_code::template_not_found: return "Template not found";
case error_code::template_load_failed: return "Failed to load template";
case error_code::template_apply_failed: return "Failed to apply template";
case error_code::palette_not_found: return "Palette not found";
case error_code::palette_load_failed: return "Failed to load palette";
case error_code::init_failed: return "Initialization failed";
case error_code::invalid_arg: return "Invalid argument";
case error_code::resource_missing: return "Resource missing";
default: return "Unknown error code";
}
}
struct Error
{
error_code code;
std::string message;
std::string context;
Error(error_code c) : code(c), message(error_code_string(c)) {}
Error(error_code c, std::string msg)
: code(c), message(std::move(msg)) {}
Error(error_code c, std::string msg, std::string ctx)
: code(c), message(std::move(msg)), context(std::move(ctx)) {}
std::string description() const
{
if (context.empty())
return message;
return message + " [" + context + "]";
}
};
template<typename T>
class [[nodiscard]] Result
{
private:
std::variant<T, Error> m_data;
public:
Result(T value) : m_data(std::move(value)) {}
Result(Error error) : m_data(std::move(error)) {}
bool is_ok() const { return std::holds_alternative<T>(m_data); }
bool is_error() const { return std::holds_alternative<Error>(m_data); }
explicit operator bool() const { return is_ok(); }
T& value() & { return std::get<T>(m_data); }
const T& value() const & { return std::get<T>(m_data); }
T&& value() && { return std::get<T>(std::move(m_data)); }
const Error& error() const { return std::get<Error>(m_data); }
T value_or(T default_value) const
{
return is_ok() ? std::get<T>(m_data) : std::move(default_value);
}
std::optional<T> ok() const
{
if (is_ok())
return std::get<T>(m_data);
return std::nullopt;
}
std::optional<Error> err() const
{
if (is_error())
return std::get<Error>(m_data);
return std::nullopt;
}
template<typename F>
auto map(F&& func) -> Result<decltype(func(std::declval<T>()))>
{
using U = decltype(func(std::declval<T>()));
if (is_ok())
return Result<U>(func(std::get<T>(m_data)));
return Result<U>(std::get<Error>(m_data));
}
template<typename F>
auto and_then(F&& func) -> decltype(func(std::declval<T>()))
{
if (is_ok())
return func(std::get<T>(m_data));
using ResultType = decltype(func(std::declval<T>()));
return ResultType(std::get<Error>(m_data));
}
};
template<>
class [[nodiscard]] Result<void>
{
private:
std::optional<Error> m_error;
public:
Result() : m_error(std::nullopt) {}
Result(Error error) : m_error(std::move(error)) {}
bool is_ok() const { return !m_error.has_value(); }
bool is_error() const { return m_error.has_value(); }
explicit operator bool() const { return is_ok(); }
const Error& error() const { return *m_error; }
std::optional<Error> err() const { return m_error; }
};
template<typename T>
Result<T> Ok(T value)
{
return Result<T>(std::move(value));
}
inline Result<void> Ok()
{
return Result<void>();
}
template<typename T>
Result<T> Err(Error error)
{
return Result<T>(std::move(error));
}
template<typename T>
Result<T> Err(error_code code)
{
return Result<T>(Error(code));
}
template<typename T>
Result<T> Err(error_code code, std::string message)
{
return Result<T>(Error(code, std::move(message)));
}
template<typename T>
Result<T> Err(error_code code, std::string message, std::string context)
{
return Result<T>(Error(code, std::move(message), std::move(context)));
}
} // namespace clrsync::core
#endif // CLRSYNC_CORE_ERROR_HPP

View File

@@ -1,10 +1,10 @@
#ifndef CLRSYNC_CORE_IO_FILE_HPP #ifndef CLRSYNC_CORE_IO_FILE_HPP
#define CLRSYNC_CORE_IO_FILE_HPP #define CLRSYNC_CORE_IO_FILE_HPP
#include "core/common/error.hpp"
#include <cstdint> #include <cstdint>
#include <map> #include <map>
#include <string> #include <string>
#include <variant> #include <variant>
#include <core/error.hpp>
using value_type = std::variant<std::string, uint32_t, int, bool>; using value_type = std::variant<std::string, uint32_t, int, bool>;
@@ -16,7 +16,10 @@ class file
file() = default; file() = default;
file(std::string path) {}; file(std::string path) {};
virtual ~file() = default; virtual ~file() = default;
virtual Result<void> parse() { return Ok(); }; virtual Result<void> parse()
{
return Ok();
};
virtual const std::string get_string_value(const std::string &section, virtual const std::string get_string_value(const std::string &section,
const std::string &key) const const std::string &key) const
{ {
@@ -42,7 +45,10 @@ class file
virtual void insert_or_update_value(const std::string &section, const std::string &key, virtual void insert_or_update_value(const std::string &section, const std::string &key,
const value_type &value) {}; const value_type &value) {};
virtual void remove_section(const std::string &section) {}; virtual void remove_section(const std::string &section) {};
virtual Result<void> save_file() { return Ok(); }; virtual Result<void> save_file()
{
return Ok();
};
}; };
} // namespace clrsync::core::io } // namespace clrsync::core::io
#endif #endif

View File

@@ -1,5 +1,5 @@
#include "toml_file.hpp" #include "core/io/toml_file.hpp"
#include "core/utils.hpp" #include "core/common/utils.hpp"
#include <filesystem> #include <filesystem>
#include <fstream> #include <fstream>
#include <vector> #include <vector>
@@ -113,9 +113,12 @@ void toml_file::remove_section(const std::string &section)
Result<void> toml_file::save_file() Result<void> toml_file::save_file()
{ {
try { try
{
std::filesystem::create_directories(std::filesystem::path(m_path).parent_path()); std::filesystem::create_directories(std::filesystem::path(m_path).parent_path());
} catch (const std::exception& e) { }
catch (const std::exception &e)
{
return Err<void>(error_code::dir_create_failed, e.what(), m_path); return Err<void>(error_code::dir_create_failed, e.what(), m_path);
} }

View File

@@ -1,7 +1,7 @@
#ifndef CLRSYNC_CORE_IO_TOML_FILE_HPP #ifndef CLRSYNC_CORE_IO_TOML_FILE_HPP
#define CLRSYNC_CORE_IO_TOML_FILE_HPP #define CLRSYNC_CORE_IO_TOML_FILE_HPP
#include <core/io/file.hpp> #include "core/common/error.hpp"
#include <core/error.hpp> #include "core/io/file.hpp"
#include <string> #include <string>
#include <toml/toml.hpp> #include <toml/toml.hpp>

View File

@@ -111,14 +111,17 @@ void color::from_hex_string(const std::string &str)
if (str.empty() || str[0] != '#') if (str.empty() || str[0] != '#')
throw std::invalid_argument("Invalid hex color format"); throw std::invalid_argument("Invalid hex color format");
if (str.size() == 7) { if (str.size() == 7)
{
uint32_t rgb = static_cast<uint32_t>(std::stoul(str.substr(1), nullptr, 16)); uint32_t rgb = static_cast<uint32_t>(std::stoul(str.substr(1), nullptr, 16));
m_hex = (rgb << 8) | 0xFF; m_hex = (rgb << 8) | 0xFF;
} }
else if (str.size() == 9) { else if (str.size() == 9)
{
m_hex = static_cast<uint32_t>(std::stoul(str.substr(1), nullptr, 16)); m_hex = static_cast<uint32_t>(std::stoul(str.substr(1), nullptr, 16));
} }
else { else
{
throw std::invalid_argument("Invalid hex color format"); throw std::invalid_argument("Invalid hex color format");
} }
} }
@@ -144,23 +147,31 @@ std::string color::format(const std::string& field) const
auto hslv = to_hsl(); auto hslv = to_hsl();
auto hslav = to_hsla(); auto hslav = to_hsla();
if (field == "hex") return to_hex_string(); if (field == "hex")
if (field == "hex_stripped") { return to_hex_string();
if (field == "hex_stripped")
{
auto s = to_hex_string(); auto s = to_hex_string();
return s.substr(1); return s.substr(1);
} }
if (field == "hexa") return to_hex_string_with_alpha(); if (field == "hexa")
if (field == "hexa_stripped") { return to_hex_string_with_alpha();
if (field == "hexa_stripped")
{
auto s = to_hex_string_with_alpha(); auto s = to_hex_string_with_alpha();
return s.substr(1); return s.substr(1);
} }
if (field == "r") return std::to_string(rgb.r); if (field == "r")
if (field == "g") return std::to_string(rgb.g); return std::to_string(rgb.r);
if (field == "b") return std::to_string(rgb.b); if (field == "g")
return std::to_string(rgb.g);
if (field == "b")
return std::to_string(rgb.b);
if (field == "a") { if (field == "a")
{
float af = rgba.a / 255.0f; float af = rgba.a / 255.0f;
return std::format("{:.2f}", af); return std::format("{:.2f}", af);
} }
@@ -169,9 +180,7 @@ std::string color::format(const std::string& field) const
return std::format("rgb({},{},{})", rgb.r, rgb.g, rgb.b); return std::format("rgb({},{},{})", rgb.r, rgb.g, rgb.b);
if (field == "rgba") if (field == "rgba")
return std::format("rgba({},{},{},{:.2f})", return std::format("rgba({},{},{},{:.2f})", rgba.r, rgba.g, rgba.b, rgba.a / 255.0f);
rgba.r, rgba.g, rgba.b,
rgba.a / 255.0f);
if (field == "h") if (field == "h")
return std::format("{:.0f}", hslv.h); return std::format("{:.0f}", hslv.h);
@@ -185,12 +194,10 @@ std::string color::format(const std::string& field) const
return std::format("{:.2f}", hslav.a); return std::format("{:.2f}", hslav.a);
if (field == "hsl") if (field == "hsl")
return std::format("hsl({:.0f},{:.2f},{:.2f})", return std::format("hsl({:.0f},{:.2f},{:.2f})", hslv.h, hslv.s, hslv.l);
hslv.h, hslv.s, hslv.l);
if (field == "hsla") if (field == "hsla")
return std::format("hsla({:.0f},{:.2f},{:.2f},{:.2f})", return std::format("hsla({:.0f},{:.2f},{:.2f},{:.2f})", hslav.h, hslav.s, hslav.l, hslav.a);
hslav.h, hslav.s, hslav.l, hslav.a);
throw std::runtime_error("Unknown color format: " + field); throw std::runtime_error("Unknown color format: " + field);
} }

View File

@@ -3,8 +3,8 @@
#include <cstddef> #include <cstddef>
#include <cstdint> #include <cstdint>
#include <iterator> #include <iterator>
#include <unordered_map>
#include <string> #include <string>
#include <unordered_map>
namespace clrsync::core namespace clrsync::core
{ {

View File

@@ -4,8 +4,8 @@
#include <string> #include <string>
#include <unordered_map> #include <unordered_map>
#include <core/palette/color.hpp> #include "core/palette/color.hpp"
#include <core/palette/color_keys.hpp> #include "core/palette/color_keys.hpp"
namespace clrsync::core namespace clrsync::core
{ {

View File

@@ -5,9 +5,9 @@
#include <cstdint> #include <cstdint>
#include <string> #include <string>
#include <core/io/file.hpp> #include "core/io/file.hpp"
#include <core/palette/color_keys.hpp> #include "core/palette/color_keys.hpp"
#include <core/palette/palette.hpp> #include "core/palette/palette.hpp"
#include <memory> #include <memory>

View File

@@ -1,14 +1,13 @@
#ifndef CLRSYNC_CORE_PALETTE_PALETTE_MANAGER_HPP #ifndef CLRSYNC_CORE_PALETTE_PALETTE_MANAGER_HPP
#define CLRSYNC_CORE_PALETTE_PALETTE_MANAGER_HPP #define CLRSYNC_CORE_PALETTE_PALETTE_MANAGER_HPP
#include "core/utils.hpp" #include "core/common/utils.hpp"
#include <iostream>
#include <string> #include <string>
#include <unordered_map> #include <unordered_map>
#include <core/config/config.hpp> #include "core/config/config.hpp"
#include <core/palette/palette.hpp> #include "core/palette/palette.hpp"
#include <core/palette/palette_file.hpp> #include "core/palette/palette_file.hpp"
#include <filesystem> #include <filesystem>
namespace clrsync::core namespace clrsync::core

View File

@@ -1,17 +1,15 @@
#ifndef CLRSYNC_CORE_THEME_TEMPLATE_MANAGER_HPP #ifndef CLRSYNC_CORE_THEME_TEMPLATE_MANAGER_HPP
#define CLRSYNC_CORE_THEME_TEMPLATE_MANAGER_HPP #define CLRSYNC_CORE_THEME_TEMPLATE_MANAGER_HPP
#include <core/config/config.hpp> #include "core/config/config.hpp"
#include <core/theme/theme_template.hpp> #include "core/theme/theme_template.hpp"
#include <string> #include <string>
#include <unordered_map> #include <unordered_map>
namespace clrsync::core namespace clrsync::core
{ {
template <typename FileType> template <typename FileType> class template_manager
class template_manager
{ {
public: public:
template_manager() = default; template_manager() = default;

View File

@@ -1,9 +1,10 @@
#ifndef CLRSYNC_CORE_THEME_THEME_RENDERER_HPP #ifndef CLRSYNC_CORE_THEME_THEME_RENDERER_HPP
#define CLRSYNC_CORE_THEME_THEME_RENDERER_HPP #define CLRSYNC_CORE_THEME_THEME_RENDERER_HPP
#include <core/config/config.hpp> #include "core/common/error.hpp"
#include <core/palette/palette_manager.hpp> #include "core/config/config.hpp"
#include <core/theme/template_manager.hpp> #include "core/palette/palette_manager.hpp"
#include <core/error.hpp> #include "core/theme/template_manager.hpp"
#include <iostream>
#include <string> #include <string>
namespace clrsync::core namespace clrsync::core
@@ -60,7 +61,8 @@ template <typename FileType> class theme_renderer
int result = std::system(tmpl.reload_command().c_str()); int result = std::system(tmpl.reload_command().c_str());
if (result != 0) if (result != 0)
{ {
std::cerr << "Warning: Command " << tmpl.reload_command() << " failed with code " << result << "\n"; std::cerr << "Warning: Command " << tmpl.reload_command()
<< " failed with code " << result << "\n";
} }
} }
} }

View File

@@ -1,5 +1,5 @@
#include "theme_template.hpp" #include "theme_template.hpp"
#include "core/utils.hpp" #include "core/common/utils.hpp"
#include <filesystem> #include <filesystem>
#include <fstream> #include <fstream>
#include <iostream> #include <iostream>
@@ -47,13 +47,15 @@ Result<void> theme_template::load_template()
{ {
if (!std::filesystem::exists(m_template_path)) if (!std::filesystem::exists(m_template_path))
{ {
return Err<void>(error_code::template_not_found, "Template file is missing", m_template_path); return Err<void>(error_code::template_not_found, "Template file is missing",
m_template_path);
} }
std::ifstream input(m_template_path, std::ios::binary); std::ifstream input(m_template_path, std::ios::binary);
if (!input) if (!input)
{ {
return Err<void>(error_code::template_load_failed, "Failed to open template file", m_template_path); return Err<void>(error_code::template_load_failed, "Failed to open template file",
m_template_path);
} }
m_template_data.assign(std::istreambuf_iterator<char>(input), std::istreambuf_iterator<char>()); m_template_data.assign(std::istreambuf_iterator<char>(input), std::istreambuf_iterator<char>());
@@ -104,13 +106,15 @@ Result<void> theme_template::save_output() const
std::ofstream output(m_output_path, std::ios::binary); std::ofstream output(m_output_path, std::ios::binary);
if (!output) if (!output)
{ {
return Err<void>(error_code::file_write_failed, "Failed to open output file for writing", m_output_path); return Err<void>(error_code::file_write_failed, "Failed to open output file for writing",
m_output_path);
} }
output << m_processed_data; output << m_processed_data;
if (!output) if (!output)
{ {
return Err<void>(error_code::file_write_failed, "Failed to write to output file", m_output_path); return Err<void>(error_code::file_write_failed, "Failed to write to output file",
m_output_path);
} }
return Ok(); return Ok();

View File

@@ -1,8 +1,8 @@
#ifndef clrsync_CORE_IO_THEME_TEMPLATE_HPP #ifndef clrsync_CORE_IO_THEME_TEMPLATE_HPP
#define clrsync_CORE_IO_THEME_TEMPLATE_HPP #define clrsync_CORE_IO_THEME_TEMPLATE_HPP
#include <core/palette/palette.hpp> #include "core/common/error.hpp"
#include <core/error.hpp> #include "core/palette/palette.hpp"
#include <string> #include <string>
namespace clrsync::core namespace clrsync::core

View File

@@ -1,22 +1,44 @@
set(GUI_SOURCES set(GUI_SOURCES
main.cpp main.cpp
color_scheme_editor.cpp views/color_scheme_editor.cpp
color_table_renderer.cpp views/color_table_renderer.cpp
preview_renderer.cpp views/preview_renderer.cpp
theme_applier.cpp controllers/theme_applier.cpp
template_editor.cpp views/template_editor.cpp
palette_controller.cpp controllers/palette_controller.cpp
template_controller.cpp controllers/template_controller.cpp
imgui_helpers.cpp views/about_window.cpp
imgui_helpers.hpp views/settings_window.cpp
about_window.cpp widgets/colors.cpp
settings_window.cpp widgets/dialogs.cpp
font_loader.cpp widgets/palette_selector.cpp
file_browser.cpp widgets/input_dialog.cpp
widgets/action_buttons.cpp
widgets/styled_checkbox.cpp
widgets/autocomplete.cpp
widgets/form_field.cpp
widgets/error_message.cpp
widgets/settings_buttons.cpp
widgets/section_header.cpp
widgets/link_button.cpp
widgets/centered_text.cpp
widgets/validation_message.cpp
widgets/template_controls.cpp
layout/main_layout.cpp
platform/windows/font_loader_windows.cpp
platform/linux/font_loader_linux.cpp
platform/macos/font_loader_macos.cpp
platform/linux/file_browser_linux.cpp
platform/windows/file_browser_windows.cpp
${CMAKE_SOURCE_DIR}/lib/color_text_edit/TextEditor.cpp ${CMAKE_SOURCE_DIR}/lib/color_text_edit/TextEditor.cpp
backend/glfw_opengl.cpp
ui_manager.cpp
)
if(MACOS)
list(APPEND GUI_SOURCES
platform/macos/file_browser_macos.mm
) )
if(APPLE)
list(APPEND GUI_SOURCES file_browser_macos.mm)
endif() endif()
if(WIN32) if(WIN32)

View File

@@ -1,96 +0,0 @@
#include "about_window.hpp"
#include "core/version.hpp"
#include "imgui_helpers.hpp"
#include "imgui.h"
about_window::about_window()
{
}
void about_window::render(const clrsync::core::palette& pal)
{
if (!m_visible)
return;
ImGui::SetNextWindowSize(ImVec2(500, 400), ImGuiCond_FirstUseEver);
if (ImGui::Begin("About clrsync", &m_visible, ImGuiWindowFlags_NoResize))
{
const float window_width = ImGui::GetContentRegionAvail().x;
ImGui::PushFont(ImGui::GetFont());
const char *title = "clrsync";
const float title_size = ImGui::CalcTextSize(title).x;
ImGui::SetCursorPosX((window_width - title_size) * 0.5f);
ImVec4 title_color = palette_utils::get_color(pal, "info", "accent");
ImGui::TextColored(title_color, "%s", title);
ImGui::PopFont();
std::string version = "Version " + clrsync::core::version_string();
const float version_size = ImGui::CalcTextSize(version.c_str()).x;
ImGui::SetCursorPosX((window_width - version_size) * 0.5f);
ImVec4 subtitle_color = palette_utils::get_color(pal, "editor_inactive", "foreground");
ImGui::TextColored(subtitle_color, "%s", version.c_str());
ImGui::Spacing();
ImGui::Separator();
ImGui::Spacing();
ImGui::TextWrapped("A color scheme management tool.");
ImGui::Spacing();
ImGui::Spacing();
ImGui::Separator();
ImGui::Spacing();
ImGui::Text("Links:");
const float button_width = 200.0f;
const float spacing = ImGui::GetStyle().ItemSpacing.x;
const float total_width = 2.0f * button_width + spacing;
ImGui::SetCursorPosX((window_width - total_width) * 0.5f);
if (ImGui::Button("GitHub Repository", ImVec2(button_width, 0)))
{
#ifdef _WIN32
system("start https://github.com/obsqrbtz/clrsync");
#elif __APPLE__
system("open https://github.com/obsqrbtz/clrsync");
#else
system("xdg-open https://github.com/obsqrbtz/clrsync");
#endif
}
ImGui::SameLine();
if (ImGui::Button("Documentation", ImVec2(button_width, 0)))
{
#ifdef _WIN32
system("start https://binarygoose.dev/projects/clrsync/overview/");
#elif __APPLE__
system("open https://binarygoose.dev/projects/clrsync/overview/");
#else
system("xdg-open https://binarygoose.dev/projects/clrsync/overview/");
#endif
}
ImGui::Spacing();
ImGui::Spacing();
ImGui::Separator();
ImGui::Spacing();
ImVec4 license_color = palette_utils::get_color(pal, "editor_inactive", "foreground");
ImGui::TextColored(license_color, "MIT License");
ImGui::TextWrapped(
"Copyright (c) 2025 Daniel Dada\n\n"
"Permission is hereby granted, free of charge, to any person obtaining a copy "
"of this software and associated documentation files (the \"Software\"), to deal "
"in the Software without restriction, including without limitation the rights "
"to use, copy, modify, merge, publish, distribute, sublicense, and/or sell "
"copies of the Software, and to permit persons to whom the Software is "
"furnished to do so, subject to the following conditions:\n\n"
"The above copyright notice and this permission notice shall be included in all "
"copies or substantial portions of the Software.");
}
ImGui::End();
}

View File

@@ -1,21 +0,0 @@
#ifndef CLRSYNC_GUI_ABOUT_WINDOW_HPP
#define CLRSYNC_GUI_ABOUT_WINDOW_HPP
#include "core/palette/palette.hpp"
class about_window
{
public:
about_window();
void render(const clrsync::core::palette& pal);
void render() { render(m_default_palette); }
void show() { m_visible = true; }
void hide() { m_visible = false; }
bool is_visible() const { return m_visible; }
private:
bool m_visible{false};
clrsync::core::palette m_default_palette;
};
#endif // CLRSYNC_GUI_ABOUT_WINDOW_HPP

View File

@@ -0,0 +1,37 @@
#ifndef CLRSYNC_BACKEND_HPP
#define CLRSYNC_BACKEND_HPP
#include <string>
namespace clrsync::gui::backend{
struct window_config{
std::string title = "clrsync";
int width = 1280;
int height = 720;
bool decorated = true;
bool transparent_framebuffer = true;
float clear_color[4] = {0.1f, 0.1f, 0.1f, 0.0f};
};
class backend_interface{
public:
virtual ~backend_interface() = default;
virtual bool initialize(const window_config& config) = 0;
virtual void shutdown() = 0;
virtual bool should_close() const = 0;
virtual void begin_frame() = 0;
virtual void end_frame() = 0;
virtual void* get_native_window() const = 0;
virtual void* get_graphics_context() const = 0;
virtual bool init_imgui_backend() = 0;
virtual void shutdown_imgui_backend() = 0;
virtual void imgui_new_frame() = 0;
virtual void imgui_render_draw_data(void* draw_data) = 0;
};
}
#endif // CLRSYNC_BACKEND_HPP

View File

@@ -0,0 +1,158 @@
#include "gui/backend/glfw_opengl.hpp"
#include <iostream>
#include <string>
#include <GLFW/glfw3.h>
#include <imgui.h>
#include <imgui_impl_glfw.h>
#include <imgui_impl_opengl3.h>
namespace clrsync::gui::backend
{
glfw_opengl_backend::glfw_opengl_backend() = default;
glfw_opengl_backend::~glfw_opengl_backend()
{
glfw_opengl_backend::shutdown();
}
bool glfw_opengl_backend::initialize(const window_config &config)
{
glfwSetErrorCallback([](int error, const char* description) {
std::cerr << "GLFW Error " << error << ": " << description << std::endl;
});
if (!glfwInit())
{
std::cerr << "Failed to initialize GLFW" << std::endl;
return false;
}
glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 3);
#ifdef __APPLE__
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 2);
#else
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 3);
#endif
glfwWindowHint(GLFW_RESIZABLE,GLFW_TRUE);
glfwWindowHint(GLFW_DECORATED, config.decorated ? GLFW_TRUE : GLFW_FALSE);
glfwWindowHint(GLFW_TRANSPARENT_FRAMEBUFFER, config.transparent_framebuffer ? GLFW_TRUE : GLFW_FALSE);
m_window = glfwCreateWindow(config.width, config.height, config.title.c_str(), nullptr, nullptr);
if (!m_window)
{
std::cerr << "Failed to create GLFW window" << std::endl;
glfwTerminate();
return false;
}
glfwMakeContextCurrent(m_window);
glfwSwapInterval(1);
return true;
}
void glfw_opengl_backend::shutdown()
{
if (m_window)
{
glfwDestroyWindow(m_window);
m_window = nullptr;
}
glfwTerminate();
}
bool glfw_opengl_backend::should_close() const
{
return m_window && glfwWindowShouldClose(m_window);
}
void glfw_opengl_backend::begin_frame()
{
glfwPollEvents();
if (m_window)
{
int display_w, display_h;
glfwGetFramebufferSize(m_window, &display_w, &display_h);
glViewport(0, 0, display_w, display_h);
glClearColor(0.1f, 0.1f, 0.1f, 0.0f);
glClear(GL_COLOR_BUFFER_BIT);
}
}
void glfw_opengl_backend::end_frame()
{
if (m_window)
{
glfwSwapBuffers(m_window);
}
}
void *glfw_opengl_backend::get_native_window() const
{
return static_cast<void *>(m_window);
}
void *glfw_opengl_backend::get_graphics_context() const
{
return static_cast<void *>(m_window);
}
std::string glfw_opengl_backend::get_glfw_version() const
{
return glfwGetVersionString();
}
std::string glfw_opengl_backend::get_glfw_platform() const
{
switch (glfwGetPlatform()) {
case GLFW_PLATFORM_WAYLAND: return "Wayland";
case GLFW_PLATFORM_X11: return "X11";
case GLFW_PLATFORM_COCOA: return "Cocoa";
case GLFW_PLATFORM_WIN32: return "Win32";
default: return "Unknown";
}
}
bool glfw_opengl_backend::init_imgui_backend()
{
if (!m_window)
return false;
if (!ImGui_ImplGlfw_InitForOpenGL(m_window, true))
return false;
#ifdef __APPLE__
const char* glsl_version = "#version 150";
#else
const char* glsl_version = "#version 120";
#endif
if (!ImGui_ImplOpenGL3_Init(glsl_version))
{
ImGui_ImplGlfw_Shutdown();
return false;
}
return true;
}
void glfw_opengl_backend::shutdown_imgui_backend()
{
ImGui_ImplOpenGL3_Shutdown();
ImGui_ImplGlfw_Shutdown();
}
void glfw_opengl_backend::imgui_new_frame()
{
ImGui_ImplOpenGL3_NewFrame();
ImGui_ImplGlfw_NewFrame();
}
void glfw_opengl_backend::imgui_render_draw_data(void* draw_data)
{
ImGui_ImplOpenGL3_RenderDrawData(static_cast<ImDrawData*>(draw_data));
}
} // namespace clrsync::gui::backend

View File

@@ -0,0 +1,39 @@
#ifndef CLRSYNC_GLFW_OPENGL_HPP
#define CLRSYNC_GLFW_OPENGL_HPP
#define GL_SILENCE_DEPRECATION
#include <GLFW/glfw3.h>
#include "gui/backend/backend.hpp"
namespace clrsync::gui::backend{
class glfw_opengl_backend : public backend_interface{
public:
glfw_opengl_backend();
~glfw_opengl_backend();
bool initialize(const window_config& config) override;
void shutdown() override;
bool should_close() const override;
void begin_frame() override;
void end_frame() override;
void* get_native_window() const override;
void* get_graphics_context() const override;
std::string get_glfw_version() const;
std::string get_glfw_platform() const;
bool init_imgui_backend() override;
void shutdown_imgui_backend() override;
void imgui_new_frame() override;
void imgui_render_draw_data(void* draw_data) override;
private:
GLFWwindow* m_window = nullptr;
};
}
#endif // CLRSYNC_GLFW_OPENGL_HPP

View File

@@ -1,204 +0,0 @@
#include "color_scheme_editor.hpp"
#include "imgui.h"
#include "imgui_helpers.hpp"
#include "template_editor.hpp"
#include "settings_window.hpp"
#include "theme_applier.hpp"
#include <iostream>
#include <ranges>
color_scheme_editor::color_scheme_editor()
{
const auto &current = m_controller.current_palette();
if (!current.colors().empty())
{
theme_applier::apply_to_imgui(current);
m_preview.apply_palette(current);
}
else
{
std::cout << "WARNING: No palette loaded, skipping theme application\n";
}
}
void color_scheme_editor::notify_palette_changed()
{
if (m_template_editor)
{
m_template_editor->apply_current_palette(m_controller.current_palette());
}
if (m_settings_window)
{
m_settings_window->set_palette(m_controller.current_palette());
}
}
void color_scheme_editor::apply_themes()
{
const auto &current = m_controller.current_palette();
theme_applier::apply_to_imgui(current);
m_preview.apply_palette(current);
notify_palette_changed();
}
void color_scheme_editor::render_controls_and_colors()
{
ImGui::Begin("Color Schemes");
render_controls();
ImGui::Separator();
ImGui::BeginChild("ColorTableContent", ImVec2(0, 0), false);
m_color_table.render(m_controller.current_palette(), m_controller,
[this]() { apply_themes(); });
ImGui::EndChild();
ImGui::End();
}
void color_scheme_editor::render_preview()
{
ImGui::Begin("Color Preview");
m_preview.render(m_controller.current_palette());
ImGui::End();
}
void color_scheme_editor::render_controls()
{
const auto &current = m_controller.current_palette();
const auto &palettes = m_controller.palettes();
ImGui::PushStyleVar(ImGuiStyleVar_ItemSpacing, ImVec2(6, 8));
ImGui::PushStyleVar(ImGuiStyleVar_FramePadding, ImVec2(8, 5));
ImGui::AlignTextToFramePadding();
ImGui::Text("Palette:");
ImGui::SameLine();
ImGui::SetNextItemWidth(200.0f);
if (ImGui::BeginCombo("##scheme", current.name().c_str()))
{
for (const auto &name : palettes | std::views::keys)
{
const bool selected = current.name() == name;
if (ImGui::Selectable(name.c_str(), selected))
{
m_controller.select_palette(name);
apply_themes();
}
if (selected)
ImGui::SetItemDefaultFocus();
}
ImGui::EndCombo();
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Select a color palette to edit");
ImGui::SameLine();
ImGui::SetCursorPosX(ImGui::GetCursorPosX() + 8);
static char new_palette_name_buf[128] = "";
if (ImGui::Button(" + New "))
{
new_palette_name_buf[0] = 0;
ImGui::OpenPopup("New Palette");
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Create a new palette");
if (ImGui::BeginPopupModal("New Palette", nullptr, ImGuiWindowFlags_AlwaysAutoResize))
{
ImGui::Text("Enter a name for the new palette:");
ImGui::Spacing();
ImGui::SetNextItemWidth(250);
ImGui::InputTextWithHint("##new_palette_input", "Palette name...", new_palette_name_buf,
IM_ARRAYSIZE(new_palette_name_buf));
ImGui::Spacing();
ImGui::Separator();
ImGui::Spacing();
bool can_create = strlen(new_palette_name_buf) > 0;
if (!can_create)
ImGui::BeginDisabled();
if (ImGui::Button("Create", ImVec2(120, 0)))
{
m_controller.create_palette(new_palette_name_buf);
m_controller.select_palette(new_palette_name_buf);
apply_themes();
new_palette_name_buf[0] = 0;
ImGui::CloseCurrentPopup();
}
if (!can_create)
ImGui::EndDisabled();
ImGui::SameLine();
if (ImGui::Button("Cancel", ImVec2(120, 0)))
{
new_palette_name_buf[0] = 0;
ImGui::CloseCurrentPopup();
}
ImGui::EndPopup();
}
ImGui::SameLine();
if (ImGui::Button(" Save "))
{
m_controller.save_current_palette();
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Save current palette to file");
ImGui::SameLine();
auto error = palette_utils::get_color(current, "error");
auto error_hover = ImVec4(error.x * 1.1f, error.y * 1.1f, error.z * 1.1f,
error.w);
auto error_active = ImVec4(error.x * 0.8f, error.y * 0.8f, error.z * 0.8f,
error.w);
auto on_error = palette_utils::get_color(current, "on_error");
ImGui::PushStyleColor(ImGuiCol_Button, error);
ImGui::PushStyleColor(ImGuiCol_ButtonHovered, error_hover);
ImGui::PushStyleColor(ImGuiCol_ButtonActive, error_active);
ImGui::PushStyleColor(ImGuiCol_Text, on_error);
if (ImGui::Button(" Delete "))
{
m_show_delete_confirmation = true;
}
ImGui::PopStyleColor(4);
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Delete current palette");
if (m_show_delete_confirmation)
{
ImGui::OpenPopup("Delete Palette?");
m_show_delete_confirmation = false;
}
palette_utils::render_delete_confirmation_popup("Delete Palette?", current.name(), "palette",
current, [this]() {
m_controller.delete_current_palette();
apply_themes();
});
ImGui::SameLine();
ImGui::SetCursorPosX(ImGui::GetCursorPosX() + 16);
if (ImGui::Button(" Apply Theme "))
{
m_controller.apply_current_theme();
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Apply current palette to all enabled templates");
ImGui::PopStyleVar(2);
}

View File

@@ -1,35 +0,0 @@
#ifndef CLRSYNC_GUI_COLOR_SCHEME_EDITOR_HPP
#define CLRSYNC_GUI_COLOR_SCHEME_EDITOR_HPP
#include "palette_controller.hpp"
#include "color_table_renderer.hpp"
#include "preview_renderer.hpp"
class template_editor;
class settings_window;
class color_scheme_editor
{
public:
color_scheme_editor();
void render_controls_and_colors();
void render_preview();
void set_template_editor(template_editor* editor) { m_template_editor = editor; }
void set_settings_window(settings_window* window) { m_settings_window = window; }
const palette_controller& controller() const { return m_controller; }
private:
void render_controls();
void apply_themes();
void notify_palette_changed();
palette_controller m_controller;
color_table_renderer m_color_table;
preview_renderer m_preview;
template_editor* m_template_editor{nullptr};
settings_window* m_settings_window{nullptr};
bool m_show_delete_confirmation{false};
};
#endif // CLRSYNC_GUI_COLOR_SCHEME_EDITOR_HPP

View File

@@ -1,30 +0,0 @@
#ifndef CLRSYNC_GUI_COLOR_TABLE_RENDERER_HPP
#define CLRSYNC_GUI_COLOR_TABLE_RENDERER_HPP
#include "core/palette/palette.hpp"
#include "palette_controller.hpp"
#include <functional>
#include <string>
class color_table_renderer
{
public:
using OnColorChangedCallback = std::function<void()>;
void render(const clrsync::core::palette& palette,
palette_controller& controller,
const OnColorChangedCallback& on_changed);
private:
void render_color_row(const std::string& name,
const clrsync::core::palette& palette,
palette_controller& controller,
const OnColorChangedCallback& on_changed);
bool matches_filter(const std::string& name) const;
char m_filter_text[128] = {0};
bool m_show_only_modified{false};
};
#endif // CLRSYNC_GUI_COLOR_TABLE_RENDERER_HPP

View File

@@ -1,4 +1,4 @@
#include "palette_controller.hpp" #include "gui/controllers/palette_controller.hpp"
#include "core/config/config.hpp" #include "core/config/config.hpp"
#include "core/theme/theme_renderer.hpp" #include "core/theme/theme_renderer.hpp"
@@ -13,9 +13,12 @@ palette_controller::palette_controller()
auto default_theme = clrsync::core::config::instance().default_theme(); auto default_theme = clrsync::core::config::instance().default_theme();
auto it = m_palettes.find(default_theme); auto it = m_palettes.find(default_theme);
if (it != m_palettes.end()) { if (it != m_palettes.end())
{
m_current_palette = it->second; m_current_palette = it->second;
} else { }
else
{
m_current_palette = m_palettes.begin()->second; m_current_palette = m_palettes.begin()->second;
} }
} }
@@ -23,7 +26,8 @@ palette_controller::palette_controller()
void palette_controller::select_palette(const std::string &name) void palette_controller::select_palette(const std::string &name)
{ {
auto it = m_palettes.find(name); auto it = m_palettes.find(name);
if (it != m_palettes.end()) { if (it != m_palettes.end())
{
m_current_palette = it->second; m_current_palette = it->second;
} }
} }

View File

@@ -6,12 +6,19 @@
#include <string> #include <string>
#include <unordered_map> #include <unordered_map>
class palette_controller { class palette_controller
{
public: public:
palette_controller(); palette_controller();
const clrsync::core::palette& current_palette() const { return m_current_palette; } const clrsync::core::palette &current_palette() const
const std::unordered_map<std::string, clrsync::core::palette>& palettes() const { return m_palettes; } {
return m_current_palette;
}
const std::unordered_map<std::string, clrsync::core::palette> &palettes() const
{
return m_palettes;
}
void select_palette(const std::string &name); void select_palette(const std::string &name);
void create_palette(const std::string &name); void create_palette(const std::string &name);

View File

@@ -1,4 +1,4 @@
#include "template_controller.hpp" #include "gui/controllers/template_controller.hpp"
#include "core/config/config.hpp" #include "core/config/config.hpp"
template_controller::template_controller() template_controller::template_controller()
@@ -9,7 +9,8 @@ template_controller::template_controller()
void template_controller::set_template_enabled(const std::string &key, bool enabled) void template_controller::set_template_enabled(const std::string &key, bool enabled)
{ {
auto it = m_templates.find(key); auto it = m_templates.find(key);
if (it != m_templates.end()) { if (it != m_templates.end())
{
it->second.set_enabled(enabled); it->second.set_enabled(enabled);
(void)clrsync::core::config::instance().update_template(key, it->second); (void)clrsync::core::config::instance().update_template(key, it->second);
} }
@@ -18,7 +19,8 @@ void template_controller::set_template_enabled(const std::string& key, bool enab
void template_controller::set_template_input_path(const std::string &key, const std::string &path) void template_controller::set_template_input_path(const std::string &key, const std::string &path)
{ {
auto it = m_templates.find(key); auto it = m_templates.find(key);
if (it != m_templates.end()) { if (it != m_templates.end())
{
it->second.set_template_path(path); it->second.set_template_path(path);
(void)clrsync::core::config::instance().update_template(key, it->second); (void)clrsync::core::config::instance().update_template(key, it->second);
} }
@@ -27,16 +29,19 @@ void template_controller::set_template_input_path(const std::string& key, const
void template_controller::set_template_output_path(const std::string &key, const std::string &path) void template_controller::set_template_output_path(const std::string &key, const std::string &path)
{ {
auto it = m_templates.find(key); auto it = m_templates.find(key);
if (it != m_templates.end()) { if (it != m_templates.end())
{
it->second.set_output_path(path); it->second.set_output_path(path);
(void)clrsync::core::config::instance().update_template(key, it->second); (void)clrsync::core::config::instance().update_template(key, it->second);
} }
} }
void template_controller::set_template_reload_command(const std::string& key, const std::string& cmd) void template_controller::set_template_reload_command(const std::string &key,
const std::string &cmd)
{ {
auto it = m_templates.find(key); auto it = m_templates.find(key);
if (it != m_templates.end()) { if (it != m_templates.end())
{
it->second.set_reload_command(cmd); it->second.set_reload_command(cmd);
(void)clrsync::core::config::instance().update_template(key, it->second); (void)clrsync::core::config::instance().update_template(key, it->second);
} }
@@ -45,7 +50,8 @@ void template_controller::set_template_reload_command(const std::string& key, co
bool template_controller::remove_template(const std::string &key) bool template_controller::remove_template(const std::string &key)
{ {
auto result = clrsync::core::config::instance().remove_template(key); auto result = clrsync::core::config::instance().remove_template(key);
if (result) { if (result)
{
m_templates.erase(key); m_templates.erase(key);
return true; return true;
} }

View File

@@ -1,16 +1,21 @@
#ifndef CLRSYNC_GUI_TEMPLATE_CONTROLLER_HPP #ifndef CLRSYNC_GUI_TEMPLATE_CONTROLLER_HPP
#define CLRSYNC_GUI_TEMPLATE_CONTROLLER_HPP #define CLRSYNC_GUI_TEMPLATE_CONTROLLER_HPP
#include "core/theme/theme_template.hpp"
#include "core/theme/template_manager.hpp"
#include "core/io/toml_file.hpp" #include "core/io/toml_file.hpp"
#include "core/theme/template_manager.hpp"
#include "core/theme/theme_template.hpp"
#include <string> #include <string>
#include <unordered_map> #include <unordered_map>
class template_controller { class template_controller
{
public: public:
template_controller(); template_controller();
[[nodiscard]] const std::unordered_map<std::string, clrsync::core::theme_template>& templates() const { return m_templates; } [[nodiscard]] const std::unordered_map<std::string, clrsync::core::theme_template> &templates()
const
{
return m_templates;
}
void set_template_enabled(const std::string &key, bool enabled); void set_template_enabled(const std::string &key, bool enabled);
void set_template_input_path(const std::string &key, const std::string &path); void set_template_input_path(const std::string &key, const std::string &path);
void set_template_output_path(const std::string &key, const std::string &path); void set_template_output_path(const std::string &key, const std::string &path);

View File

@@ -1,4 +1,4 @@
#include "theme_applier.hpp" #include "gui/controllers/theme_applier.hpp"
#include "imgui.h" #include "imgui.h"
namespace theme_applier namespace theme_applier
@@ -26,27 +26,35 @@ void apply_to_editor(TextEditor& editor, const clrsync::core::palette& current)
palette[int(TextEditor::PaletteIndex::String)] = get_color_u32(current, "editor_string"); palette[int(TextEditor::PaletteIndex::String)] = get_color_u32(current, "editor_string");
palette[int(TextEditor::PaletteIndex::CharLiteral)] = get_color_u32(current, "editor_string"); palette[int(TextEditor::PaletteIndex::CharLiteral)] = get_color_u32(current, "editor_string");
palette[int(TextEditor::PaletteIndex::Punctuation)] = get_color_u32(current, "editor_main"); palette[int(TextEditor::PaletteIndex::Punctuation)] = get_color_u32(current, "editor_main");
palette[int(TextEditor::PaletteIndex::Preprocessor)] = get_color_u32(current, "editor_emphasis"); palette[int(TextEditor::PaletteIndex::Preprocessor)] =
get_color_u32(current, "editor_emphasis");
palette[int(TextEditor::PaletteIndex::Identifier)] = get_color_u32(current, "editor_main"); palette[int(TextEditor::PaletteIndex::Identifier)] = get_color_u32(current, "editor_main");
palette[int(TextEditor::PaletteIndex::KnownIdentifier)] = get_color_u32(current, "editor_link"); palette[int(TextEditor::PaletteIndex::KnownIdentifier)] = get_color_u32(current, "editor_link");
palette[int(TextEditor::PaletteIndex::PreprocIdentifier)] = get_color_u32(current, "editor_link"); palette[int(TextEditor::PaletteIndex::PreprocIdentifier)] =
get_color_u32(current, "editor_link");
palette[int(TextEditor::PaletteIndex::Comment)] = get_color_u32(current, "editor_comment"); palette[int(TextEditor::PaletteIndex::Comment)] = get_color_u32(current, "editor_comment");
palette[int(TextEditor::PaletteIndex::MultiLineComment)] = get_color_u32(current, "editor_comment"); palette[int(TextEditor::PaletteIndex::MultiLineComment)] =
get_color_u32(current, "editor_comment");
palette[int(TextEditor::PaletteIndex::Background)] = get_color_u32(current, "editor_background"); palette[int(TextEditor::PaletteIndex::Background)] =
get_color_u32(current, "editor_background");
palette[int(TextEditor::PaletteIndex::Cursor)] = get_color_u32(current, "cursor"); palette[int(TextEditor::PaletteIndex::Cursor)] = get_color_u32(current, "cursor");
palette[int(TextEditor::PaletteIndex::Selection)] = get_color_u32(current, "editor_selected"); palette[int(TextEditor::PaletteIndex::Selection)] = get_color_u32(current, "editor_selected");
palette[int(TextEditor::PaletteIndex::ErrorMarker)] = get_color_u32(current, "editor_error"); palette[int(TextEditor::PaletteIndex::ErrorMarker)] = get_color_u32(current, "editor_error");
palette[int(TextEditor::PaletteIndex::Breakpoint)] = get_color_u32(current, "editor_error"); palette[int(TextEditor::PaletteIndex::Breakpoint)] = get_color_u32(current, "editor_error");
palette[int(TextEditor::PaletteIndex::LineNumber)] = get_color_u32(current, "editor_line_number"); palette[int(TextEditor::PaletteIndex::LineNumber)] =
get_color_u32(current, "editor_line_number");
palette[int(TextEditor::PaletteIndex::CurrentLineFill)] = get_color_u32(current, "surface_variant"); palette[int(TextEditor::PaletteIndex::CurrentLineFill)] =
palette[int(TextEditor::PaletteIndex::CurrentLineFillInactive)] = get_color_u32(current, "surface"); get_color_u32(current, "surface_variant");
palette[int(TextEditor::PaletteIndex::CurrentLineFillInactive)] =
get_color_u32(current, "surface");
palette[int(TextEditor::PaletteIndex::CurrentLineEdge)] = get_color_u32(current, "border_focused"); palette[int(TextEditor::PaletteIndex::CurrentLineEdge)] =
get_color_u32(current, "border_focused");
editor.SetPalette(palette); editor.SetPalette(palette);
} }
@@ -138,7 +146,8 @@ void apply_to_imgui(const clrsync::core::palette& current)
ImVec4(border.x * 0.7f, border.y * 0.7f, border.z * 0.7f, border.w); ImVec4(border.x * 0.7f, border.y * 0.7f, border.z * 0.7f, border.w);
style.Colors[ImGuiCol_TableRowBg] = ImVec4(0, 0, 0, 0); style.Colors[ImGuiCol_TableRowBg] = ImVec4(0, 0, 0, 0);
style.Colors[ImGuiCol_TableRowBgAlt] = ImVec4(onSurfaceVariant.x, onSurfaceVariant.y, onSurfaceVariant.z, 0.06f); style.Colors[ImGuiCol_TableRowBgAlt] =
ImVec4(onSurfaceVariant.x, onSurfaceVariant.y, onSurfaceVariant.z, 0.06f);
style.Colors[ImGuiCol_Separator] = border; style.Colors[ImGuiCol_Separator] = border;
style.Colors[ImGuiCol_SeparatorHovered] = accent; style.Colors[ImGuiCol_SeparatorHovered] = accent;

View File

@@ -1,13 +1,13 @@
#ifndef CLRSYNC_GUI_THEME_APPLIER_HPP #ifndef CLRSYNC_GUI_THEME_APPLIER_HPP
#define CLRSYNC_GUI_THEME_APPLIER_HPP #define CLRSYNC_GUI_THEME_APPLIER_HPP
#include "core/palette/palette.hpp"
#include "color_text_edit/TextEditor.h" #include "color_text_edit/TextEditor.h"
#include "core/palette/palette.hpp"
namespace theme_applier namespace theme_applier
{ {
void apply_to_imgui(const clrsync::core::palette &pal); void apply_to_imgui(const clrsync::core::palette &pal);
void apply_to_editor(TextEditor &editor, const clrsync::core::palette &pal); void apply_to_editor(TextEditor &editor, const clrsync::core::palette &pal);
} } // namespace theme_applier
#endif // CLRSYNC_GUI_THEME_APPLIER_HPP #endif // CLRSYNC_GUI_THEME_APPLIER_HPP

View File

@@ -1,263 +0,0 @@
#include "file_browser.hpp"
#include <filesystem>
#ifdef _WIN32
#include <windows.h>
#include <commdlg.h>
#include <shlobj.h>
#include <shlwapi.h>
#include <cstring>
namespace file_dialogs {
std::string open_file_dialog(const std::string& title,
const std::string& initial_path,
const std::vector<std::string>& filters) {
OPENFILENAMEA ofn;
char file[MAX_PATH] = "";
std::string filter_str = "All Files (*.*)\0*.*\0";
ZeroMemory(&ofn, sizeof(ofn));
ofn.lStructSize = sizeof(ofn);
ofn.hwndOwner = GetActiveWindow();
ofn.lpstrFile = file;
ofn.nMaxFile = sizeof(file);
ofn.lpstrFilter = filter_str.c_str();
ofn.nFilterIndex = 1;
ofn.lpstrTitle = title.c_str();
ofn.Flags = OFN_PATHMUSTEXIST | OFN_FILEMUSTEXIST | OFN_NOCHANGEDIR;
if (!initial_path.empty() && std::filesystem::exists(initial_path)) {
std::filesystem::path p(initial_path);
if (std::filesystem::is_directory(p)) {
ofn.lpstrInitialDir = initial_path.c_str();
} else {
std::string dir = p.parent_path().string();
std::string name = p.filename().string();
ofn.lpstrInitialDir = dir.c_str();
strncpy(file, name.c_str(), sizeof(file) - 1);
}
}
if (GetOpenFileNameA(&ofn)) {
return std::string(file);
}
return "";
}
std::string save_file_dialog(const std::string& title,
const std::string& initial_path,
const std::vector<std::string>& filters) {
OPENFILENAMEA ofn;
char file[MAX_PATH] = "";
std::string filter_str = "All Files\0*.*\0\0";
ZeroMemory(&ofn, sizeof(ofn));
ofn.lStructSize = sizeof(ofn);
ofn.hwndOwner = GetActiveWindow();
ofn.lpstrFile = file;
ofn.nMaxFile = sizeof(file);
ofn.lpstrFilter = filter_str.c_str();
ofn.nFilterIndex = 1;
ofn.lpstrTitle = title.c_str();
ofn.Flags = OFN_PATHMUSTEXIST | OFN_OVERWRITEPROMPT | OFN_NOCHANGEDIR;
if (!initial_path.empty()) {
std::filesystem::path p(initial_path);
if (std::filesystem::exists(p) && std::filesystem::is_directory(p)) {
ofn.lpstrInitialDir = initial_path.c_str();
} else {
std::string dir = p.parent_path().string();
std::string name = p.filename().string();
if (std::filesystem::exists(dir)) {
ofn.lpstrInitialDir = dir.c_str();
strncpy(file, name.c_str(), sizeof(file) - 1);
}
}
}
if (GetSaveFileNameA(&ofn)) {
return std::string(file);
}
return "";
}
std::string select_folder_dialog(const std::string& title,
const std::string& initial_path) {
IFileOpenDialog *pFileOpen;
HRESULT hr = CoCreateInstance(CLSID_FileOpenDialog, NULL, CLSCTX_ALL,
IID_IFileOpenDialog, reinterpret_cast<void**>(&pFileOpen));
if (SUCCEEDED(hr)) {
DWORD dwFlags;
if (SUCCEEDED(pFileOpen->GetOptions(&dwFlags))) {
pFileOpen->SetOptions(dwFlags | FOS_PICKFOLDERS);
}
std::wstring wtitle(title.begin(), title.end());
pFileOpen->SetTitle(wtitle.c_str());
if (!initial_path.empty() && std::filesystem::exists(initial_path)) {
IShellItem *psi = NULL;
std::wstring winitial(initial_path.begin(), initial_path.end());
hr = SHCreateItemFromParsingName(winitial.c_str(), NULL, IID_IShellItem, (void**)&psi);
if (SUCCEEDED(hr)) {
pFileOpen->SetFolder(psi);
psi->Release();
}
}
hr = pFileOpen->Show(GetActiveWindow());
if (SUCCEEDED(hr)) {
IShellItem *pItem;
hr = pFileOpen->GetResult(&pItem);
if (SUCCEEDED(hr)) {
PWSTR pszFilePath;
hr = pItem->GetDisplayName(SIGDN_FILESYSPATH, &pszFilePath);
if (SUCCEEDED(hr)) {
std::wstring wpath(pszFilePath);
std::string result(wpath.begin(), wpath.end());
CoTaskMemFree(pszFilePath);
pItem->Release();
pFileOpen->Release();
return result;
}
pItem->Release();
}
}
pFileOpen->Release();
}
return "";
}
}
#elif !defined(__APPLE__)
#include <gtk/gtk.h>
namespace file_dialogs {
std::string open_file_dialog(const std::string& title,
const std::string& initial_path,
const std::vector<std::string>& filters) {
if (!gtk_init_check(nullptr, nullptr)) {
return "";
}
GtkFileChooserNative *native = gtk_file_chooser_native_new(
title.c_str(),
nullptr,
GTK_FILE_CHOOSER_ACTION_OPEN,
"_Open",
"_Cancel");
GtkFileChooser *chooser = GTK_FILE_CHOOSER(native);
if (!initial_path.empty()) {
std::filesystem::path p(initial_path);
if (std::filesystem::exists(p)) {
if (std::filesystem::is_directory(p)) {
gtk_file_chooser_set_current_folder(chooser, initial_path.c_str());
} else {
gtk_file_chooser_set_current_folder(chooser, p.parent_path().c_str());
gtk_file_chooser_set_current_name(chooser, p.filename().c_str());
}
}
}
std::string result;
if (gtk_native_dialog_run(GTK_NATIVE_DIALOG(native)) == GTK_RESPONSE_ACCEPT) {
char* filename = gtk_file_chooser_get_filename(chooser);
if (filename) {
result = filename;
g_free(filename);
}
}
g_object_unref(native);
while (gtk_events_pending()) gtk_main_iteration();
return result;
}
std::string save_file_dialog(const std::string& title,
const std::string& initial_path,
const std::vector<std::string>& filters) {
if (!gtk_init_check(nullptr, nullptr)) {
return "";
}
GtkFileChooserNative *native = gtk_file_chooser_native_new(
title.c_str(),
nullptr,
GTK_FILE_CHOOSER_ACTION_SAVE,
"_Save",
"_Cancel");
GtkFileChooser *chooser = GTK_FILE_CHOOSER(native);
gtk_file_chooser_set_do_overwrite_confirmation(chooser, TRUE);
if (!initial_path.empty()) {
std::filesystem::path p(initial_path);
if (std::filesystem::exists(p.parent_path())) {
gtk_file_chooser_set_current_folder(chooser, p.parent_path().c_str());
gtk_file_chooser_set_current_name(chooser, p.filename().c_str());
}
}
std::string result;
if (gtk_native_dialog_run(GTK_NATIVE_DIALOG(native)) == GTK_RESPONSE_ACCEPT) {
char* filename = gtk_file_chooser_get_filename(chooser);
if (filename) {
result = filename;
g_free(filename);
}
}
g_object_unref(native);
while (gtk_events_pending()) gtk_main_iteration();
return result;
}
std::string select_folder_dialog(const std::string& title,
const std::string& initial_path) {
if (!gtk_init_check(nullptr, nullptr)) {
return "";
}
GtkFileChooserNative *native = gtk_file_chooser_native_new(
title.c_str(),
nullptr,
GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER,
"_Select",
"_Cancel");
GtkFileChooser *chooser = GTK_FILE_CHOOSER(native);
if (!initial_path.empty() && std::filesystem::exists(initial_path)) {
gtk_file_chooser_set_current_folder(chooser, initial_path.c_str());
}
std::string result;
if (gtk_native_dialog_run(GTK_NATIVE_DIALOG(native)) == GTK_RESPONSE_ACCEPT) {
char* filename = gtk_file_chooser_get_filename(chooser);
if (filename) {
result = filename;
g_free(filename);
}
}
g_object_unref(native);
while (gtk_events_pending()) gtk_main_iteration();
return result;
}
}
#endif

View File

@@ -1,20 +0,0 @@
#ifndef CLRSYNC_GUI_FILE_BROWSER_HPP
#define CLRSYNC_GUI_FILE_BROWSER_HPP
#include <string>
#include <vector>
namespace file_dialogs {
std::string open_file_dialog(const std::string& title = "Open File",
const std::string& initial_path = "",
const std::vector<std::string>& filters = {});
std::string save_file_dialog(const std::string& title = "Save File",
const std::string& initial_path = "",
const std::vector<std::string>& filters = {});
std::string select_folder_dialog(const std::string& title = "Select Folder",
const std::string& initial_path = "");
}
#endif // CLRSYNC_GUI_FILE_BROWSER_HPP

View File

@@ -1,333 +0,0 @@
#include "font_loader.hpp"
#include "core/config/config.hpp"
#include "imgui_internal.h"
#include <imgui.h>
#include <algorithm>
#if defined(_WIN32)
#include <algorithm>
#include <windows.h>
#include <winreg.h>
static std::string search_registry_for_font(HKEY root_key, const char* subkey, const std::string& font_name_lower, const char* default_font_dir)
{
HKEY hKey;
LONG result = RegOpenKeyExA(root_key, subkey, 0, KEY_READ, &hKey);
if (result != ERROR_SUCCESS)
return {};
char value_name[512];
BYTE value_data[512];
DWORD value_name_size, value_data_size, type;
DWORD index = 0;
std::string found_path;
while (true)
{
value_name_size = sizeof(value_name);
value_data_size = sizeof(value_data);
result = RegEnumValueA(hKey, index++, value_name, &value_name_size, nullptr, &type, value_data, &value_data_size);
if (result != ERROR_SUCCESS)
break;
if (type != REG_SZ)
continue;
std::string reg_font_name = value_name;
std::transform(reg_font_name.begin(), reg_font_name.end(), reg_font_name.begin(), ::tolower);
std::string reg_font_name_clean = reg_font_name;
size_t type_pos = reg_font_name_clean.find(" (");
if (type_pos != std::string::npos)
reg_font_name_clean = reg_font_name_clean.substr(0, type_pos);
if (reg_font_name_clean == font_name_lower)
{
std::string font_file = reinterpret_cast<char*>(value_data);
// If path is not absolute, prepend default font directory
if (font_file.find(":\\") == std::string::npos)
{
found_path = std::string(default_font_dir) + "\\" + font_file;
}
else
{
found_path = font_file;
}
break;
}
}
RegCloseKey(hKey);
return found_path;
}
std::string font_loader::find_font_windows(const char* font_name)
{
std::string font_name_lower = font_name;
std::transform(font_name_lower.begin(), font_name_lower.end(), font_name_lower.begin(), ::tolower);
char windows_dir[MAX_PATH];
GetWindowsDirectoryA(windows_dir, MAX_PATH);
std::string system_fonts_dir = std::string(windows_dir) + "\\Fonts";
// First, try system-wide fonts (HKEY_LOCAL_MACHINE)
std::string path = search_registry_for_font(
HKEY_LOCAL_MACHINE,
"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Fonts",
font_name_lower,
system_fonts_dir.c_str()
);
if (!path.empty())
return path;
// If not found, try per-user fonts (HKEY_CURRENT_USER)
char local_appdata[MAX_PATH];
if (GetEnvironmentVariableA("LOCALAPPDATA", local_appdata, MAX_PATH) > 0)
{
std::string user_fonts_dir = std::string(local_appdata) + "\\Microsoft\\Windows\\Fonts";
path = search_registry_for_font(
HKEY_CURRENT_USER,
"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Fonts",
font_name_lower,
user_fonts_dir.c_str()
);
}
return path;
}
#endif
#if defined(__APPLE__)
#include <CoreText/CoreText.h>
std::vector<unsigned char> font_loader::load_font_macos(const char* font_name)
{
std::vector<unsigned char> out;
CFStringRef cf_name = CFStringCreateWithCString(nullptr, font_name, kCFStringEncodingUTF8);
if (!cf_name)
return out;
CTFontDescriptorRef desc = CTFontDescriptorCreateWithNameAndSize(cf_name, 12);
CFRelease(cf_name);
if (!desc)
return out;
CFURLRef url = (CFURLRef)CTFontDescriptorCopyAttribute(desc, kCTFontURLAttribute);
CFRelease(desc);
if (!url)
return out;
CFDataRef data = nullptr;
Boolean success = CFURLCreateDataAndPropertiesFromResource(kCFAllocatorDefault, url, &data, nullptr, nullptr, nullptr);
CFRelease(url);
if (success && data)
{
CFIndex size = CFDataGetLength(data);
if (size > 100)
{
out.resize(size);
CFDataGetBytes(data, CFRangeMake(0, size), out.data());
}
CFRelease(data);
}
return out;
}
#endif
#if !defined(_WIN32) && !defined(__APPLE__)
#include <fontconfig/fontconfig.h>
std::string font_loader::find_font_linux(const char* font_name)
{
FcInit();
FcPattern* pattern = FcNameParse(reinterpret_cast<const FcChar8*>(font_name));
if (!pattern)
return {};
FcConfigSubstitute(nullptr, pattern, FcMatchPattern);
FcDefaultSubstitute(pattern);
FcResult result;
FcPattern* match = FcFontMatch(nullptr, pattern, &result);
std::string out;
if (match)
{
FcChar8* file = nullptr;
if (FcPatternGetString(match, FC_FILE, 0, &file) == FcResultMatch)
out = reinterpret_cast<const char*>(file);
FcPatternDestroy(match);
}
FcPatternDestroy(pattern);
return out;
}
#endif
std::string font_loader::find_font_path(const char* font_name)
{
#if defined(_WIN32)
return find_font_windows(font_name);
#elif defined(__APPLE__)
(void)font_name;
return {};
#else
return find_font_linux(font_name);
#endif
}
ImFont* font_loader::load_font(const char* font_name, float size_px)
{
#if defined(__APPLE__)
std::vector<unsigned char> buf = load_font_macos(font_name);
if (buf.empty())
return nullptr;
return ImGui::GetIO().Fonts->AddFontFromMemoryTTF(
buf.data(),
static_cast<int>(buf.size()),
size_px
);
#else
std::string path = find_font_path(font_name);
if (path.empty())
return nullptr;
float scale = ImGui::GetIO().DisplayFramebufferScale.y;
return ImGui::GetIO().Fonts->AddFontFromFileTTF(
path.c_str(),
size_px * scale
);
#endif
}
void font_loader::push_default_font()
{
ImGui::PushFont(ImGui::GetDefaultFont(), clrsync::core::config::instance().font_size());
}
void font_loader::pop_font()
{
ImGui::PopFont();
}
std::vector<std::string> font_loader::get_system_fonts()
{
std::vector<std::string> fonts;
#if defined(_WIN32)
auto enumerate_registry_fonts = [&fonts](HKEY root_key, const char* subkey)
{
HKEY hKey;
if (RegOpenKeyExA(root_key, subkey, 0, KEY_READ, &hKey) != ERROR_SUCCESS)
return;
char value_name[512];
DWORD value_name_size;
DWORD index = 0;
while (true)
{
value_name_size = sizeof(value_name);
LONG result = RegEnumValueA(hKey, index++, value_name, &value_name_size, nullptr, nullptr, nullptr, nullptr);
if (result != ERROR_SUCCESS)
break;
std::string font_name = value_name;
size_t pos = font_name.find(" (");
if (pos != std::string::npos)
font_name = font_name.substr(0, pos);
if (std::find(fonts.begin(), fonts.end(), font_name) == fonts.end())
fonts.push_back(font_name);
}
RegCloseKey(hKey);
};
enumerate_registry_fonts(HKEY_LOCAL_MACHINE, "SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Fonts");
enumerate_registry_fonts(HKEY_CURRENT_USER, "SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Fonts");
#elif defined(__APPLE__)
CTFontCollectionRef collection = CTFontCollectionCreateFromAvailableFonts(nullptr);
if (collection)
{
CFArrayRef fontDescriptors = CTFontCollectionCreateMatchingFontDescriptors(collection);
CFRelease(collection);
if (fontDescriptors)
{
CFIndex count = CFArrayGetCount(fontDescriptors);
for (CFIndex i = 0; i < count; i++)
{
CTFontDescriptorRef descriptor = (CTFontDescriptorRef)CFArrayGetValueAtIndex(fontDescriptors, i);
CFStringRef fontName = (CFStringRef)CTFontDescriptorCopyAttribute(descriptor, kCTFontDisplayNameAttribute);
if (fontName)
{
char buffer[256];
if (CFStringGetCString(fontName, buffer, sizeof(buffer), kCFStringEncodingUTF8))
{
std::string font_name = buffer;
if (std::find(fonts.begin(), fonts.end(), font_name) == fonts.end())
fonts.push_back(font_name);
}
CFRelease(fontName);
}
}
CFRelease(fontDescriptors);
}
}
#else
FcInit();
FcPattern* pattern = FcPatternCreate();
FcObjectSet* os = FcObjectSetBuild(FC_FAMILY, nullptr);
FcFontSet* fs = FcFontList(nullptr, pattern, os);
if (fs)
{
for (int i = 0; i < fs->nfont; i++)
{
FcChar8* family = nullptr;
if (FcPatternGetString(fs->fonts[i], FC_FAMILY, 0, &family) == FcResultMatch)
{
std::string font_name = reinterpret_cast<const char*>(family);
if (std::find(fonts.begin(), fonts.end(), font_name) == fonts.end())
fonts.push_back(font_name);
}
}
FcFontSetDestroy(fs);
}
FcObjectSetDestroy(os);
FcPatternDestroy(pattern);
#endif
std::sort(fonts.begin(), fonts.end());
return fonts;
}

View File

@@ -1,266 +0,0 @@
#include <iostream>
#include <string>
#include "GLFW/glfw3.h"
#include "gui/settings_window.hpp"
#include "imgui_impl_glfw.h"
#include "imgui_impl_opengl3.h"
#include "imgui_helpers.hpp"
#include "imgui_internal.h"
GLFWwindow * init_glfw()
{
glfwSetErrorCallback([](int error, const char* description) {
std::cerr << "GLFW Error " << error << ": " << description << std::endl;
});
if (!glfwInit())
{
std::cerr << "Failed to initialize GLFW\n";
return nullptr;
}
glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 3);
#ifdef __APPLE__
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 2);
#else
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 0);
#endif
glfwWindowHint(GLFW_TRANSPARENT_FRAMEBUFFER, GLFW_TRUE);
GLFWwindow* w = glfwCreateWindow(1280, 720, "clrsync", nullptr, nullptr);
if (!w)
{
std::cerr << "Failed to create GLFW window\n";
return nullptr;
}
glfwMakeContextCurrent(w);
glfwSwapInterval(1);
return w;
}
void init_imgui(GLFWwindow* window, const std::string& ini_path)
{
IMGUI_CHECKVERSION();
ImGui::CreateContext();
ImGuiIO& io = ImGui::GetIO();
io.ConfigFlags |= ImGuiConfigFlags_DockingEnable;
io.IniFilename = ini_path.c_str();
ImGui::StyleColorsDark();
ImGui_ImplGlfw_InitForOpenGL(window, true);
#ifdef __APPLE__
ImGui_ImplOpenGL3_Init("#version 150");
#else
ImGui_ImplOpenGL3_Init("#version 120");
#endif
}
void begin_frame()
{
ImGui_ImplOpenGL3_NewFrame();
ImGui_ImplGlfw_NewFrame();
ImGui::NewFrame();
}
void render_menu_bar(about_window* about, settings_window* settings)
{
if (ImGui::BeginMainMenuBar())
{
if (ImGui::BeginMenu("File"))
{
if (ImGui::MenuItem("Settings"))
{
if (settings)
settings->show();
}
ImGui::Separator();
if (ImGui::MenuItem("Exit"))
{
glfwSetWindowShouldClose(glfwGetCurrentContext(), GLFW_TRUE);
}
ImGui::EndMenu();
}
if (ImGui::BeginMenu("Help"))
{
if (ImGui::MenuItem("About"))
{
if (about)
about->show();
}
ImGui::EndMenu();
}
ImGui::EndMainMenuBar();
}
}
void setup_main_dockspace(bool& first_time)
{
const ImGuiViewport* viewport = ImGui::GetMainViewport();
ImGui::SetNextWindowPos(viewport->Pos);
ImGui::SetNextWindowSize(viewport->Size);
ImGui::SetNextWindowViewport(viewport->ID);
constexpr ImGuiWindowFlags flags =
ImGuiWindowFlags_NoTitleBar |
ImGuiWindowFlags_NoCollapse |
ImGuiWindowFlags_NoResize |
ImGuiWindowFlags_NoMove |
ImGuiWindowFlags_NoBringToFrontOnFocus |
ImGuiWindowFlags_NoNavFocus |
ImGuiWindowFlags_MenuBar;
ImGui::PushStyleVar(ImGuiStyleVar_WindowRounding, 0.0f);
ImGui::PushStyleVar(ImGuiStyleVar_WindowBorderSize, 0.0f);
ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2(0,0));
ImGui::Begin("MainDockSpace", nullptr, flags);
ImGui::PopStyleVar(3);
ImGuiID dockspace_id = ImGui::GetID("MainDockSpace");
if (first_time)
{
first_time = false;
ImGui::DockBuilderRemoveNode(dockspace_id);
ImGui::DockBuilderAddNode(dockspace_id, ImGuiDockNodeFlags_DockSpace);
ImGui::DockBuilderSetNodeSize(dockspace_id, viewport->Size);
ImGuiID center, right;
ImGui::DockBuilderSplitNode(dockspace_id, ImGuiDir_Right, 0.5f, &right, &center);
ImGuiDockNode* center_node = ImGui::DockBuilderGetNode(center);
if (center_node)
{
center_node->LocalFlags |= ImGuiDockNodeFlags_CentralNode;
}
ImGui::DockBuilderDockWindow("Color Schemes", right);
ImGui::DockBuilderDockWindow("Color Preview", center);
ImGui::DockBuilderDockWindow("Templates", center);
ImGui::DockBuilderFinish(dockspace_id);
}
ImGui::DockSpace(dockspace_id, ImVec2{0,0}, ImGuiDockNodeFlags_None);
ImGui::End();
}
void end_frame(GLFWwindow* window)
{
ImGui::Render();
int w, h;
glfwGetFramebufferSize(window, &w, &h);
glViewport(0, 0, w, h);
glClearColor(0.1f, 0.1f, 0.1f, 0.f);
glClear(GL_COLOR_BUFFER_BIT);
ImGui_ImplOpenGL3_RenderDrawData(ImGui::GetDrawData());
glfwSwapBuffers(window);
}
void shutdown(GLFWwindow* window)
{
ImGui_ImplOpenGL3_Shutdown();
ImGui_ImplGlfw_Shutdown();
ImGui::DestroyContext();
glfwDestroyWindow(window);
glfwTerminate();
}
namespace palette_utils
{
ImVec4 get_color(const clrsync::core::palette& pal, const std::string& key, const std::string& fallback)
{
auto colors = pal.colors();
if (colors.empty())
return ImVec4(1.0f, 1.0f, 1.0f, 1.0f);
auto it = colors.find(key);
if (it == colors.end() && !fallback.empty())
{
it = colors.find(fallback);
}
if (it != colors.end())
{
const auto& col = it->second;
const uint32_t hex = col.hex();
const float r = ((hex >> 24) & 0xFF) / 255.0f;
const float g = ((hex >> 16) & 0xFF) / 255.0f;
const float b = ((hex >> 8) & 0xFF) / 255.0f;
const float a = (hex & 0xFF) / 255.0f;
return ImVec4(r, g, b, a);
}
return ImVec4(1.0f, 1.0f, 1.0f, 1.0f);
}
uint32_t get_color_u32(const clrsync::core::palette& pal, const std::string& key, const std::string& fallback)
{
auto colors = pal.colors();
if (colors.empty())
return 0xFFFFFFFF;
auto it = colors.find(key);
if (it == colors.end() && !fallback.empty())
{
it = colors.find(fallback);
}
if (it != colors.end())
{
const auto& col = it->second;
const uint32_t hex = col.hex();
const uint32_t r = (hex >> 24) & 0xFF;
const uint32_t g = (hex >> 16) & 0xFF;
const uint32_t b = (hex >> 8) & 0xFF;
const uint32_t a = hex & 0xFF;
return (a << 24) | (b << 16) | (g << 8) | r;
}
return 0xFFFFFFFF;
}
bool render_delete_confirmation_popup(const std::string& popup_title, const std::string& item_name,
const std::string& item_type, const clrsync::core::palette& pal,
const std::function<void()>& on_delete)
{
bool result = false;
if (ImGui::BeginPopupModal(popup_title.c_str(), nullptr, ImGuiWindowFlags_AlwaysAutoResize))
{
ImVec4 warning_color = get_color(pal, "warning", "accent");
ImGui::TextColored(warning_color, "Are you sure you want to delete '%s'?", item_name.c_str());
ImGui::Text("This action cannot be undone.");
ImGui::Spacing();
ImGui::Separator();
ImGui::Spacing();
float button_width = 120.0f;
float total_width = 2.0f * button_width + ImGui::GetStyle().ItemSpacing.x;
float window_width = ImGui::GetContentRegionAvail().x;
ImGui::SetCursorPosX((window_width - total_width) * 0.5f);
if (ImGui::Button("Delete", ImVec2(button_width, 0)))
{
on_delete();
result = true;
ImGui::CloseCurrentPopup();
}
ImGui::SameLine();
if (ImGui::Button("Cancel", ImVec2(button_width, 0)))
{
ImGui::CloseCurrentPopup();
}
ImGui::EndPopup();
}
return result;
}
}

View File

@@ -1,37 +0,0 @@
#ifndef CLRSYNC_IMGUI_HELPERS_HPP
#define CLRSYNC_IMGUI_HELPERS_HPP
#include "gui/about_window.hpp"
#include "core/palette/palette.hpp"
#include "imgui.h"
#include <functional>
#include <string>
struct GLFWwindow;
class settings_window;
GLFWwindow * init_glfw();
void init_imgui(GLFWwindow* window, const std::string& ini_path);
void begin_frame();
void setup_main_dockspace(bool& first_time);
void end_frame(GLFWwindow* window);
void shutdown(GLFWwindow* window);
void render_menu_bar(about_window* about, settings_window* settings);
namespace palette_utils
{
ImVec4 get_color(const clrsync::core::palette& pal, const std::string& key, const std::string& fallback = "");
uint32_t get_color_u32(const clrsync::core::palette& pal, const std::string& key, const std::string& fallback = "");
bool render_delete_confirmation_popup(const std::string& popup_title, const std::string& item_name,
const std::string& item_type, const clrsync::core::palette& pal,
const std::function<void()>& on_delete);
}
namespace imgui_helpers
{
inline ImVec4 get_palette_color(const clrsync::core::palette& pal, const std::string& key, const std::string& fallback = "") {
return palette_utils::get_color(pal, key, fallback);
}
}
#endif // CLRSYNC_IMGUI_HELPERS_HPP

View File

@@ -0,0 +1,117 @@
#include "main_layout.hpp"
#include "imgui.h"
#include "imgui_internal.h"
namespace clrsync::gui::layout
{
void main_layout::render_menu_bar()
{
ImGuiViewport *vp = ImGui::GetMainViewport();
const float bar_height = 30.0f;
ImGui::SetNextWindowPos(vp->Pos);
ImGui::SetNextWindowSize(ImVec2(vp->Size.x, bar_height));
ImGui::SetNextWindowViewport(vp->ID);
ImGuiWindowFlags flags = ImGuiWindowFlags_NoDocking | ImGuiWindowFlags_NoTitleBar |
ImGuiWindowFlags_NoResize | ImGuiWindowFlags_NoMove |
ImGuiWindowFlags_NoScrollbar | ImGuiWindowFlags_NoSavedSettings;
ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2(8, 2));
ImGui::PushStyleVar(ImGuiStyleVar_WindowRounding, 0.0f);
ImGui::PushStyleVar(ImGuiStyleVar_WindowBorderSize, 0.0f);
ImGui::PushStyleVar(ImGuiStyleVar_FramePadding, ImVec2(8, 3));
ImGui::Begin("##TopBar", nullptr, flags);
ImGuiStyle &style = ImGui::GetStyle();
style.FrameBorderSize = 1.0f;
const char *settings_label = "Settings";
const char *about_label = "About";
ImVec2 settings_size = ImGui::CalcTextSize(settings_label);
ImVec2 about_size = ImGui::CalcTextSize(about_label);
float total_width = settings_size.x + style.FramePadding.x * 2.0f + about_size.x +
style.FramePadding.x * 2.0f + style.ItemSpacing.x;
float pos_x = ImGui::GetWindowWidth() - total_width - style.WindowPadding.x;
float button_height = ImGui::GetFrameHeight();
float window_height = ImGui::GetWindowHeight();
float center_y = (window_height - button_height) * 0.5f;
ImGui::SetCursorPos(ImVec2(pos_x, center_y));
if (ImGui::Button(settings_label))
{
m_show_settings = true;
}
ImGui::SameLine();
ImGui::SetCursorPosY(center_y);
if (ImGui::Button(about_label))
{
m_show_about = true;
}
ImGui::End();
ImGui::PopStyleVar(4);
}
void main_layout::setup_dockspace(bool &first_time)
{
const ImGuiViewport *viewport = ImGui::GetMainViewport();
const float topbar_height = 32.0f;
ImGui::SetNextWindowPos(ImVec2(viewport->Pos.x, viewport->Pos.y + topbar_height));
ImGui::SetNextWindowSize(ImVec2(viewport->Size.x, viewport->Size.y - topbar_height));
ImGui::SetNextWindowViewport(viewport->ID);
constexpr ImGuiWindowFlags flags = ImGuiWindowFlags_NoTitleBar | ImGuiWindowFlags_NoCollapse |
ImGuiWindowFlags_NoResize | ImGuiWindowFlags_NoMove |
ImGuiWindowFlags_NoBringToFrontOnFocus |
ImGuiWindowFlags_NoNavFocus;
ImGui::PushStyleVar(ImGuiStyleVar_WindowRounding, 0.0f);
ImGui::PushStyleVar(ImGuiStyleVar_WindowBorderSize, 0.0f);
ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2(0, 0));
ImGui::PushStyleVar(ImGuiStyleVar_FramePadding, ImVec2(8, 3));
ImGui::Begin("MainDockSpace", nullptr, flags);
ImGui::PopStyleVar(4);
ImGuiID dockspace_id = ImGui::GetID("MainDockSpace");
if (first_time)
{
first_time = false;
ImGui::DockBuilderRemoveNode(dockspace_id);
ImGui::DockBuilderAddNode(dockspace_id, ImGuiDockNodeFlags_DockSpace);
ImGui::DockBuilderSetNodeSize(dockspace_id, viewport->Size);
ImGuiID center, right;
ImGui::DockBuilderSplitNode(dockspace_id, ImGuiDir_Right, 0.5f, &right, &center);
ImGuiDockNode *center_node = ImGui::DockBuilderGetNode(center);
if (center_node)
{
center_node->LocalFlags |= ImGuiDockNodeFlags_CentralNode;
}
ImGui::DockBuilderDockWindow("Color Schemes", right);
ImGui::DockBuilderDockWindow("Color Preview", center);
ImGui::DockBuilderDockWindow("Templates", center);
ImGui::DockBuilderFinish(dockspace_id);
}
ImGui::DockSpace(dockspace_id, ImVec2{0, 0}, ImGuiDockNodeFlags_None);
ImGui::End();
}
} // namespace clrsync::gui::layout

View File

@@ -0,0 +1,28 @@
#ifndef CLRSYNC_GUI_LAYOUT_MAIN_LAYOUT_HPP
#define CLRSYNC_GUI_LAYOUT_MAIN_LAYOUT_HPP
namespace clrsync::gui::layout
{
class main_layout
{
public:
void setup_dockspace(bool &first_time);
void render_menu_bar();
bool should_show_about() const { return m_show_about; }
bool should_show_settings() const { return m_show_settings; }
void clear_actions()
{
m_show_about = false;
m_show_settings = false;
}
private:
bool m_show_about = false;
bool m_show_settings = false;
};
} // namespace clrsync::gui::layout
#endif // CLRSYNC_GUI_LAYOUT_MAIN_LAYOUT_HPP

View File

@@ -1,21 +1,18 @@
#include <iostream>
#include <memory> #include <memory>
#include <GLFW/glfw3.h> #include "core/common/error.hpp"
#include <GLFW/glfw3native.h> #include "core/common/utils.hpp"
#include "core/config/config.hpp" #include "core/config/config.hpp"
#include "core/io/toml_file.hpp" #include "core/io/toml_file.hpp"
#include "core/utils.hpp"
#include "core/error.hpp"
#include "color_scheme_editor.hpp"
#include "gui/font_loader.hpp"
#include "gui/settings_window.hpp"
#include "imgui_helpers.hpp"
#include "template_editor.hpp"
#include "about_window.hpp"
#include "gui/backend/glfw_opengl.hpp"
#include "gui/layout/main_layout.hpp"
#include "gui/ui_manager.hpp"
#include "gui/views/about_window.hpp"
#include "gui/views/color_scheme_editor.hpp"
#include "gui/views/settings_window.hpp"
#include "gui/views/template_editor.hpp"
int main(int, char **) int main(int, char **)
{ {
@@ -26,7 +23,9 @@ int main(int, char**)
if (!init_result) if (!init_result)
{ {
std::cerr << "Fatal error: " << init_result.error().description() << std::endl; std::cerr << "Fatal error: " << init_result.error().description() << std::endl;
std::cerr << "Hint: Set CLRSYNC_CONFIG_PATH environment variable or ensure config exists at: " << config_path << std::endl; std::cerr
<< "Hint: Set CLRSYNC_CONFIG_PATH environment variable or ensure config exists at: "
<< config_path << std::endl;
return 1; return 1;
} }
@@ -34,58 +33,76 @@ int main(int, char**)
static std::string ini_path = (base.parent_path() / "layout.ini").string(); static std::string ini_path = (base.parent_path() / "layout.ini").string();
bool first_time = !std::filesystem::exists(ini_path); bool first_time = !std::filesystem::exists(ini_path);
GLFWwindow* window = init_glfw(); auto backend = clrsync::gui::backend::glfw_opengl_backend();
if (!window) return 1; auto window_config = clrsync::gui::backend::window_config();
window_config.title = "clrsync";
window_config.width = 1280;
window_config.height = 720;
window_config.decorated = true;
window_config.transparent_framebuffer = true;
printf("GLFV Version: %s\n", glfwGetVersionString()); if (!backend.initialize(window_config))
{
std::cout << "GLFW runtime platform: "; std::cerr << "Failed to initialize backend." << std::endl;
switch (glfwGetPlatform()) { return 1;
case GLFW_PLATFORM_WAYLAND: std::cout << "Wayland\n"; break;
case GLFW_PLATFORM_X11: std::cout << "X11\n"; break;
case GLFW_PLATFORM_COCOA: std::cout << "Cocoa\n"; break;
case GLFW_PLATFORM_WIN32: std::cout << "Win32\n"; break;
default: std::cout << "Unknown\n";
} }
std::cout << "GLFW Version: " << backend.get_glfw_version() << std::endl;
std::cout << "GLFW runtime platform: " << backend.get_glfw_platform() << std::endl;
init_imgui(window, ini_path); clrsync::gui::ui_manager ui_manager(&backend);
font_loader loader; clrsync::gui::ui_config ui_cfg;
ui_cfg.ini_path = ini_path;
ui_cfg.enable_docking = true;
ui_cfg.enable_keyboard_nav = true;
ImFont* font = if (!ui_manager.initialize(ui_cfg))
loader.load_font(clrsync::core::config::instance().font().c_str(), clrsync::core::config::instance().font_size()); {
std::cerr << "Failed to initialize UI manager." << std::endl;
if (font) return 1;
ImGui::GetIO().FontDefault = font; }
clrsync::gui::layout::main_layout main_layout;
color_scheme_editor colorEditor; color_scheme_editor colorEditor;
template_editor templateEditor; template_editor templateEditor(&ui_manager);
about_window aboutWindow; about_window aboutWindow;
settings_window settingsWindow; settings_window settingsWindow(&ui_manager);
colorEditor.set_template_editor(&templateEditor); colorEditor.set_template_editor(&templateEditor);
colorEditor.set_settings_window(&settingsWindow); colorEditor.set_settings_window(&settingsWindow);
templateEditor.apply_current_palette(colorEditor.controller().current_palette()); templateEditor.apply_current_palette(colorEditor.controller().current_palette());
settingsWindow.set_palette(colorEditor.controller().current_palette()); settingsWindow.set_palette(colorEditor.controller().current_palette());
while (!glfwWindowShouldClose(window)) while (!backend.should_close())
{ {
glfwPollEvents(); backend.begin_frame();
loader.push_default_font();
begin_frame(); ui_manager.push_default_font();
ui_manager.begin_frame();
main_layout.render_menu_bar();
main_layout.setup_dockspace(first_time);
if (main_layout.should_show_about())
aboutWindow.show();
if (main_layout.should_show_settings())
settingsWindow.show();
main_layout.clear_actions();
render_menu_bar(&aboutWindow, &settingsWindow);
setup_main_dockspace(first_time);
templateEditor.render(); templateEditor.render();
colorEditor.render_controls_and_colors(); colorEditor.render_controls_and_colors();
colorEditor.render_preview(); colorEditor.render_preview();
aboutWindow.render(colorEditor.controller().current_palette()); aboutWindow.render(colorEditor.controller().current_palette());
settingsWindow.render(); settingsWindow.render();
loader.pop_font(); ui_manager.pop_font();
end_frame(window);
ui_manager.end_frame();
backend.end_frame();
} }
shutdown(window);
ui_manager.shutdown();
backend.shutdown();
return 0; return 0;
} }

View File

@@ -0,0 +1,21 @@
#ifndef CLRSYNC_GUI_FILE_BROWSER_HPP
#define CLRSYNC_GUI_FILE_BROWSER_HPP
#include <string>
#include <vector>
namespace file_dialogs
{
std::string open_file_dialog(const std::string &title = "Open File",
const std::string &initial_path = "",
const std::vector<std::string> &filters = {});
std::string save_file_dialog(const std::string &title = "Save File",
const std::string &initial_path = "",
const std::vector<std::string> &filters = {});
std::string select_folder_dialog(const std::string &title = "Select Folder",
const std::string &initial_path = "");
} // namespace file_dialogs
#endif // CLRSYNC_GUI_FILE_BROWSER_HPP

View File

@@ -1,9 +1,9 @@
#ifndef CLRSYNC_GUI_SYSTEM_FONT_LOADER_HPP #ifndef CLRSYNC_GUI_SYSTEM_FONT_LOADER_HPP
#define CLRSYNC_GUI_SYSTEM_FONT_LOADER_HPP #define CLRSYNC_GUI_SYSTEM_FONT_LOADER_HPP
#include <imgui.h>
#include <string> #include <string>
#include <vector> #include <vector>
#include <imgui.h>
class font_loader class font_loader
{ {

View File

@@ -0,0 +1,133 @@
#ifdef __linux__
#include "gui/platform/file_browser.hpp"
#include <gtk/gtk.h>
#include <filesystem>
namespace file_dialogs
{
std::string open_file_dialog(const std::string &title, const std::string &initial_path,
const std::vector<std::string> &filters)
{
if (!gtk_init_check(nullptr, nullptr))
{
return "";
}
GtkFileChooserNative *native = gtk_file_chooser_native_new(
title.c_str(), nullptr, GTK_FILE_CHOOSER_ACTION_OPEN, "_Open", "_Cancel");
GtkFileChooser *chooser = GTK_FILE_CHOOSER(native);
if (!initial_path.empty())
{
std::filesystem::path p(initial_path);
if (std::filesystem::exists(p))
{
if (std::filesystem::is_directory(p))
{
gtk_file_chooser_set_current_folder(chooser, initial_path.c_str());
}
else
{
gtk_file_chooser_set_current_folder(chooser, p.parent_path().c_str());
gtk_file_chooser_set_current_name(chooser, p.filename().c_str());
}
}
}
std::string result;
if (gtk_native_dialog_run(GTK_NATIVE_DIALOG(native)) == GTK_RESPONSE_ACCEPT)
{
char *filename = gtk_file_chooser_get_filename(chooser);
if (filename)
{
result = filename;
g_free(filename);
}
}
g_object_unref(native);
while (gtk_events_pending())
gtk_main_iteration();
return result;
}
std::string save_file_dialog(const std::string &title, const std::string &initial_path,
const std::vector<std::string> &filters)
{
if (!gtk_init_check(nullptr, nullptr))
{
return "";
}
GtkFileChooserNative *native = gtk_file_chooser_native_new(
title.c_str(), nullptr, GTK_FILE_CHOOSER_ACTION_SAVE, "_Save", "_Cancel");
GtkFileChooser *chooser = GTK_FILE_CHOOSER(native);
gtk_file_chooser_set_do_overwrite_confirmation(chooser, TRUE);
if (!initial_path.empty())
{
std::filesystem::path p(initial_path);
if (std::filesystem::exists(p.parent_path()))
{
gtk_file_chooser_set_current_folder(chooser, p.parent_path().c_str());
gtk_file_chooser_set_current_name(chooser, p.filename().c_str());
}
}
std::string result;
if (gtk_native_dialog_run(GTK_NATIVE_DIALOG(native)) == GTK_RESPONSE_ACCEPT)
{
char *filename = gtk_file_chooser_get_filename(chooser);
if (filename)
{
result = filename;
g_free(filename);
}
}
g_object_unref(native);
while (gtk_events_pending())
gtk_main_iteration();
return result;
}
std::string select_folder_dialog(const std::string &title, const std::string &initial_path)
{
if (!gtk_init_check(nullptr, nullptr))
{
return "";
}
GtkFileChooserNative *native = gtk_file_chooser_native_new(
title.c_str(), nullptr, GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER, "_Select", "_Cancel");
GtkFileChooser *chooser = GTK_FILE_CHOOSER(native);
if (!initial_path.empty() && std::filesystem::exists(initial_path))
{
gtk_file_chooser_set_current_folder(chooser, initial_path.c_str());
}
std::string result;
if (gtk_native_dialog_run(GTK_NATIVE_DIALOG(native)) == GTK_RESPONSE_ACCEPT)
{
char *filename = gtk_file_chooser_get_filename(chooser);
if (filename)
{
result = filename;
g_free(filename);
}
}
g_object_unref(native);
while (gtk_events_pending())
gtk_main_iteration();
return result;
}
} // namespace file_dialogs
#endif

View File

@@ -0,0 +1,95 @@
#ifdef __linux__
#include "core/config/config.hpp"
#include "gui/platform/font_loader.hpp"
#include "imgui_internal.h"
#include <algorithm>
#include <fontconfig/fontconfig.h>
#include <imgui.h>
std::string font_loader::find_font_linux(const char *font_name)
{
FcInit();
FcPattern *pattern = FcNameParse(reinterpret_cast<const FcChar8 *>(font_name));
if (!pattern)
return {};
FcConfigSubstitute(nullptr, pattern, FcMatchPattern);
FcDefaultSubstitute(pattern);
FcResult result;
FcPattern *match = FcFontMatch(nullptr, pattern, &result);
std::string out;
if (match)
{
FcChar8 *file = nullptr;
if (FcPatternGetString(match, FC_FILE, 0, &file) == FcResultMatch)
out = reinterpret_cast<const char *>(file);
FcPatternDestroy(match);
}
FcPatternDestroy(pattern);
return out;
}
std::string font_loader::find_font_path(const char *font_name)
{
return find_font_linux(font_name);
}
ImFont *font_loader::load_font(const char *font_name, float size_px)
{
std::string path = find_font_path(font_name);
if (path.empty())
return nullptr;
float scale = ImGui::GetIO().DisplayFramebufferScale.y;
return ImGui::GetIO().Fonts->AddFontFromFileTTF(path.c_str(), size_px * scale);
}
void font_loader::push_default_font()
{
ImGui::PushFont(ImGui::GetDefaultFont(), clrsync::core::config::instance().font_size());
}
void font_loader::pop_font()
{
ImGui::PopFont();
}
std::vector<std::string> font_loader::get_system_fonts()
{
std::vector<std::string> fonts;
FcInit();
FcPattern *pattern = FcPatternCreate();
FcObjectSet *os = FcObjectSetBuild(FC_FAMILY, nullptr);
FcFontSet *fs = FcFontList(nullptr, pattern, os);
if (fs)
{
for (int i = 0; i < fs->nfont; i++)
{
FcChar8 *family = nullptr;
if (FcPatternGetString(fs->fonts[i], FC_FAMILY, 0, &family) == FcResultMatch)
{
std::string font_name = reinterpret_cast<const char *>(family);
if (std::find(fonts.begin(), fonts.end(), font_name) == fonts.end())
fonts.push_back(font_name);
}
}
FcFontSetDestroy(fs);
}
FcObjectSetDestroy(os);
FcPatternDestroy(pattern);
std::sort(fonts.begin(), fonts.end());
return fonts;
}
#endif

View File

@@ -1,7 +1,6 @@
#include "file_browser.hpp"
#include <filesystem>
#ifdef __APPLE__ #ifdef __APPLE__
#include "gui/platform/file_browser.hpp"
#include <filesystem>
#include <Cocoa/Cocoa.h> #include <Cocoa/Cocoa.h>
namespace file_dialogs { namespace file_dialogs {
@@ -76,5 +75,4 @@ std::string select_folder_dialog(const std::string& title,
} }
} }
#endif #endif

View File

@@ -0,0 +1,115 @@
#ifdef __APPLE__
#include "core/config/config.hpp"
#include "gui/platform/font_loader.hpp"
#include "imgui_internal.h"
#include <CoreText/CoreText.h>
#include <algorithm>
#include <imgui.h>
std::vector<unsigned char> font_loader::load_font_macos(const char *font_name)
{
std::vector<unsigned char> out;
CFStringRef cf_name = CFStringCreateWithCString(nullptr, font_name, kCFStringEncodingUTF8);
if (!cf_name)
return out;
CTFontDescriptorRef desc = CTFontDescriptorCreateWithNameAndSize(cf_name, 12);
CFRelease(cf_name);
if (!desc)
return out;
CFURLRef url = (CFURLRef)CTFontDescriptorCopyAttribute(desc, kCTFontURLAttribute);
CFRelease(desc);
if (!url)
return out;
CFDataRef data = nullptr;
Boolean success = CFURLCreateDataAndPropertiesFromResource(kCFAllocatorDefault, url, &data,
nullptr, nullptr, nullptr);
CFRelease(url);
if (success && data)
{
CFIndex size = CFDataGetLength(data);
if (size > 100)
{
out.resize(size);
CFDataGetBytes(data, CFRangeMake(0, size), out.data());
}
CFRelease(data);
}
return out;
}
std::string font_loader::find_font_path(const char *font_name)
{
(void)font_name;
return {};
}
ImFont *font_loader::load_font(const char *font_name, float size_px)
{
std::vector<unsigned char> buf = load_font_macos(font_name);
if (buf.empty())
return nullptr;
return ImGui::GetIO().Fonts->AddFontFromMemoryTTF(buf.data(), static_cast<int>(buf.size()),
size_px);
}
void font_loader::push_default_font()
{
ImGui::PushFont(ImGui::GetDefaultFont(), clrsync::core::config::instance().font_size());
}
void font_loader::pop_font()
{
ImGui::PopFont();
}
std::vector<std::string> font_loader::get_system_fonts()
{
std::vector<std::string> fonts;
CTFontCollectionRef collection = CTFontCollectionCreateFromAvailableFonts(nullptr);
if (collection)
{
CFArrayRef fontDescriptors = CTFontCollectionCreateMatchingFontDescriptors(collection);
CFRelease(collection);
if (fontDescriptors)
{
CFIndex count = CFArrayGetCount(fontDescriptors);
for (CFIndex i = 0; i < count; i++)
{
CTFontDescriptorRef descriptor =
(CTFontDescriptorRef)CFArrayGetValueAtIndex(fontDescriptors, i);
CFStringRef fontName = (CFStringRef)CTFontDescriptorCopyAttribute(
descriptor, kCTFontDisplayNameAttribute);
if (fontName)
{
char buffer[256];
if (CFStringGetCString(fontName, buffer, sizeof(buffer), kCFStringEncodingUTF8))
{
std::string font_name = buffer;
if (std::find(fonts.begin(), fonts.end(), font_name) == fonts.end())
fonts.push_back(font_name);
}
CFRelease(fontName);
}
}
CFRelease(fontDescriptors);
}
}
std::sort(fonts.begin(), fonts.end());
return fonts;
}
#endif

View File

@@ -0,0 +1,167 @@
#ifdef _WIN32
// clang-format off
#define WIN32_LEAN_AND_MEAN
#include <windows.h>
#include <commdlg.h>
#include <shlobj.h>
#include <shlwapi.h>
// clang-format on
#include "gui/platform/file_browser.hpp"
#include <filesystem>
namespace file_dialogs
{
std::string open_file_dialog(const std::string &title, const std::string &initial_path,
const std::vector<std::string> &filters)
{
OPENFILENAMEA ofn;
char file[MAX_PATH] = "";
std::string filter_str = "All Files (*.*)\0*.*\0";
ZeroMemory(&ofn, sizeof(ofn));
ofn.lStructSize = sizeof(ofn);
ofn.hwndOwner = GetActiveWindow();
ofn.lpstrFile = file;
ofn.nMaxFile = sizeof(file);
ofn.lpstrFilter = filter_str.c_str();
ofn.nFilterIndex = 1;
ofn.lpstrTitle = title.c_str();
ofn.Flags = OFN_PATHMUSTEXIST | OFN_FILEMUSTEXIST | OFN_NOCHANGEDIR;
if (!initial_path.empty() && std::filesystem::exists(initial_path))
{
std::filesystem::path p(initial_path);
if (std::filesystem::is_directory(p))
{
ofn.lpstrInitialDir = initial_path.c_str();
}
else
{
std::string dir = p.parent_path().string();
std::string name = p.filename().string();
ofn.lpstrInitialDir = dir.c_str();
strncpy(file, name.c_str(), sizeof(file) - 1);
}
}
if (GetOpenFileNameA(&ofn))
{
return std::string(file);
}
return "";
}
std::string save_file_dialog(const std::string &title, const std::string &initial_path,
const std::vector<std::string> &filters)
{
OPENFILENAMEA ofn;
char file[MAX_PATH] = "";
std::string filter_str = "All Files\0*.*\0\0";
ZeroMemory(&ofn, sizeof(ofn));
ofn.lStructSize = sizeof(ofn);
ofn.hwndOwner = GetActiveWindow();
ofn.lpstrFile = file;
ofn.nMaxFile = sizeof(file);
ofn.lpstrFilter = filter_str.c_str();
ofn.nFilterIndex = 1;
ofn.lpstrTitle = title.c_str();
ofn.Flags = OFN_PATHMUSTEXIST | OFN_OVERWRITEPROMPT | OFN_NOCHANGEDIR;
if (!initial_path.empty())
{
std::filesystem::path p(initial_path);
if (std::filesystem::exists(p) && std::filesystem::is_directory(p))
{
ofn.lpstrInitialDir = initial_path.c_str();
}
else
{
std::string dir = p.parent_path().string();
std::string name = p.filename().string();
if (std::filesystem::exists(dir))
{
ofn.lpstrInitialDir = dir.c_str();
strncpy(file, name.c_str(), sizeof(file) - 1);
}
}
}
if (GetSaveFileNameA(&ofn))
{
return std::string(file);
}
return "";
}
std::string select_folder_dialog(const std::string &title, const std::string &initial_path)
{
IFileOpenDialog *pFileOpen;
HRESULT hr = CoCreateInstance(CLSID_FileOpenDialog, NULL, CLSCTX_ALL, IID_IFileOpenDialog,
reinterpret_cast<void **>(&pFileOpen));
if (SUCCEEDED(hr))
{
DWORD dwFlags;
if (SUCCEEDED(pFileOpen->GetOptions(&dwFlags)))
{
pFileOpen->SetOptions(dwFlags | FOS_PICKFOLDERS);
}
std::wstring wtitle(title.begin(), title.end());
pFileOpen->SetTitle(wtitle.c_str());
if (!initial_path.empty() && std::filesystem::exists(initial_path))
{
IShellItem *psi = NULL;
std::wstring winitial(initial_path.begin(), initial_path.end());
hr = SHCreateItemFromParsingName(winitial.c_str(), NULL, IID_IShellItem, (void **)&psi);
if (SUCCEEDED(hr))
{
pFileOpen->SetFolder(psi);
psi->Release();
}
}
hr = pFileOpen->Show(GetActiveWindow());
if (SUCCEEDED(hr))
{
IShellItem *pItem;
hr = pFileOpen->GetResult(&pItem);
if (SUCCEEDED(hr))
{
PWSTR pszFilePath;
hr = pItem->GetDisplayName(SIGDN_FILESYSPATH, &pszFilePath);
if (SUCCEEDED(hr))
{
std::wstring wpath(pszFilePath);
int size = WideCharToMultiByte(CP_UTF8, 0, wpath.c_str(), -1, nullptr, 0,
nullptr, nullptr);
std::string result(size - 1, 0);
WideCharToMultiByte(CP_UTF8, 0, wpath.c_str(), -1, &result[0], size, nullptr,
nullptr);
CoTaskMemFree(pszFilePath);
pItem->Release();
pFileOpen->Release();
return result;
}
pItem->Release();
}
}
pFileOpen->Release();
}
return "";
}
} // namespace file_dialogs
#endif

View File

@@ -0,0 +1,170 @@
#ifdef _WIN32
#include "core/config/config.hpp"
#include "gui/platform/font_loader.hpp"
#include "imgui_internal.h"
#include <algorithm>
#include <imgui.h>
#include <windows.h>
#include <winreg.h>
static std::string search_registry_for_font(HKEY root_key, const char *subkey,
const std::string &font_name_lower,
const char *default_font_dir)
{
HKEY hKey;
LONG result = RegOpenKeyExA(root_key, subkey, 0, KEY_READ, &hKey);
if (result != ERROR_SUCCESS)
return {};
char value_name[512];
BYTE value_data[512];
DWORD value_name_size, value_data_size, type;
DWORD index = 0;
std::string found_path;
while (true)
{
value_name_size = sizeof(value_name);
value_data_size = sizeof(value_data);
result = RegEnumValueA(hKey, index++, value_name, &value_name_size, nullptr, &type,
value_data, &value_data_size);
if (result != ERROR_SUCCESS)
break;
if (type != REG_SZ)
continue;
std::string reg_font_name = value_name;
std::transform(reg_font_name.begin(), reg_font_name.end(), reg_font_name.begin(),
::tolower);
std::string reg_font_name_clean = reg_font_name;
size_t type_pos = reg_font_name_clean.find(" (");
if (type_pos != std::string::npos)
reg_font_name_clean = reg_font_name_clean.substr(0, type_pos);
if (reg_font_name_clean == font_name_lower)
{
std::string font_file = reinterpret_cast<char *>(value_data);
// If path is not absolute, prepend default font directory
if (font_file.find(":\\") == std::string::npos)
{
found_path = std::string(default_font_dir) + "\\" + font_file;
}
else
{
found_path = font_file;
}
break;
}
}
RegCloseKey(hKey);
return found_path;
}
std::string font_loader::find_font_windows(const char *font_name)
{
std::string font_name_lower = font_name;
std::transform(font_name_lower.begin(), font_name_lower.end(), font_name_lower.begin(),
::tolower);
char windows_dir[MAX_PATH];
GetWindowsDirectoryA(windows_dir, MAX_PATH);
std::string system_fonts_dir = std::string(windows_dir) + "\\Fonts";
// First, try system-wide fonts (HKEY_LOCAL_MACHINE)
std::string path = search_registry_for_font(
HKEY_LOCAL_MACHINE, "SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Fonts",
font_name_lower, system_fonts_dir.c_str());
if (!path.empty())
return path;
// If not found, try per-user fonts (HKEY_CURRENT_USER)
char local_appdata[MAX_PATH];
if (GetEnvironmentVariableA("LOCALAPPDATA", local_appdata, MAX_PATH) > 0)
{
std::string user_fonts_dir = std::string(local_appdata) + "\\Microsoft\\Windows\\Fonts";
path = search_registry_for_font(HKEY_CURRENT_USER,
"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Fonts",
font_name_lower, user_fonts_dir.c_str());
}
return path;
}
std::string font_loader::find_font_path(const char *font_name)
{
return find_font_windows(font_name);
}
ImFont *font_loader::load_font(const char *font_name, float size_px)
{
std::string path = find_font_path(font_name);
if (path.empty())
return nullptr;
float scale = ImGui::GetIO().DisplayFramebufferScale.y;
return ImGui::GetIO().Fonts->AddFontFromFileTTF(path.c_str(), size_px * scale);
}
void font_loader::push_default_font()
{
ImGui::PushFont(ImGui::GetDefaultFont(), clrsync::core::config::instance().font_size());
}
void font_loader::pop_font()
{
ImGui::PopFont();
}
std::vector<std::string> font_loader::get_system_fonts()
{
std::vector<std::string> fonts;
auto enumerate_registry_fonts = [&fonts](HKEY root_key, const char *subkey) {
HKEY hKey;
if (RegOpenKeyExA(root_key, subkey, 0, KEY_READ, &hKey) != ERROR_SUCCESS)
return;
char value_name[512];
DWORD value_name_size;
DWORD index = 0;
while (true)
{
value_name_size = sizeof(value_name);
LONG result = RegEnumValueA(hKey, index++, value_name, &value_name_size, nullptr,
nullptr, nullptr, nullptr);
if (result != ERROR_SUCCESS)
break;
std::string font_name = value_name;
size_t pos = font_name.find(" (");
if (pos != std::string::npos)
font_name = font_name.substr(0, pos);
if (std::find(fonts.begin(), fonts.end(), font_name) == fonts.end())
fonts.push_back(font_name);
}
RegCloseKey(hKey);
};
enumerate_registry_fonts(HKEY_LOCAL_MACHINE,
"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Fonts");
enumerate_registry_fonts(HKEY_CURRENT_USER,
"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Fonts");
std::sort(fonts.begin(), fonts.end());
return fonts;
}
#endif

View File

@@ -1,370 +0,0 @@
#include "settings_window.hpp"
#include "core/config/config.hpp"
#include "core/error.hpp"
#include "gui/font_loader.hpp"
#include "gui/imgui_helpers.hpp"
#include "gui/file_browser.hpp"
#include "imgui.h"
#include <cstring>
settings_window::settings_window()
: m_font_size(14), m_selected_font_idx(0), m_settings_changed(false), m_current_tab(0)
{
m_default_theme[0] = '\0';
m_palettes_path[0] = '\0';
m_font[0] = '\0';
font_loader loader;
m_available_fonts = loader.get_system_fonts();
load_settings();
}
void settings_window::render()
{
if (!m_visible)
return;
ImGui::SetNextWindowSize(ImVec2(700, 500), ImGuiCond_FirstUseEver);
ImGui::SetNextWindowPos(ImGui::GetMainViewport()->GetCenter(), ImGuiCond_FirstUseEver, ImVec2(0.5f, 0.5f));
ImGuiWindowFlags window_flags = ImGuiWindowFlags_NoCollapse;
if (ImGui::Begin("Settings", &m_visible, window_flags))
{
if (ImGui::BeginTabBar("SettingsTabs", ImGuiTabBarFlags_None))
{
if (ImGui::BeginTabItem("General"))
{
render_general_tab();
ImGui::EndTabItem();
}
if (ImGui::BeginTabItem("Appearance"))
{
render_appearance_tab();
ImGui::EndTabItem();
}
ImGui::EndTabBar();
}
render_status_messages();
ImGui::Separator();
render_action_buttons();
}
ImGui::End();
}
void settings_window::load_settings()
{
auto& cfg = clrsync::core::config::instance();
std::string default_theme = cfg.default_theme();
strncpy(m_default_theme, default_theme.c_str(), sizeof(m_default_theme) - 1);
m_default_theme[sizeof(m_default_theme) - 1] = '\0';
std::string palettes_path = cfg.palettes_path();
strncpy(m_palettes_path, palettes_path.c_str(), sizeof(m_palettes_path) - 1);
m_palettes_path[sizeof(m_palettes_path) - 1] = '\0';
std::string font = cfg.font();
strncpy(m_font, font.c_str(), sizeof(m_font) - 1);
m_font[sizeof(m_font) - 1] = '\0';
m_selected_font_idx = 0;
for (int i = 0; i < static_cast<int>(m_available_fonts.size()); i++)
{
if (m_available_fonts[i] == font)
{
m_selected_font_idx = i;
break;
}
}
m_font_size = cfg.font_size();
m_error_message.clear();
m_settings_changed = false;
}
void settings_window::apply_settings()
{
auto& cfg = clrsync::core::config::instance();
if (strlen(m_default_theme) == 0)
{
m_error_message = "Default theme cannot be empty";
return;
}
if (strlen(m_palettes_path) == 0)
{
m_error_message = "Palettes path cannot be empty";
return;
}
if (strlen(m_font) == 0)
{
m_error_message = "Font cannot be empty";
return;
}
if (m_font_size < 8 || m_font_size > 48)
{
m_error_message = "Font size must be between 8 and 48";
return;
}
auto result1 = cfg.set_default_theme(m_default_theme);
if (!result1)
{
m_error_message = "Failed to set default theme: " + result1.error().description();
return;
}
auto result2 = cfg.set_palettes_path(m_palettes_path);
if (!result2)
{
m_error_message = "Failed to set palettes path: " + result2.error().description();
return;
}
auto result3 = cfg.set_font(m_font);
if (!result3)
{
m_error_message = "Failed to set font: " + result3.error().description();
return;
}
auto result4 = cfg.set_font_size(m_font_size);
if (!result4)
{
m_error_message = "Failed to set font size: " + result4.error().description();
return;
}
font_loader fn_loader;
auto font = fn_loader.load_font(m_font, m_font_size);
if (font)
ImGui::GetIO().FontDefault = font;
m_error_message.clear();
m_settings_changed = false;
}
void settings_window::render_general_tab()
{
ImGui::Spacing();
auto accent_color = palette_utils::get_color(m_current_palette, "accent");
ImGui::TextColored(accent_color, "Theme Settings");
ImGui::Separator();
ImGui::Spacing();
ImGui::Text("Default Theme:");
ImGui::SameLine();
show_help_marker("The default color scheme to load on startup");
ImGui::SetNextItemWidth(-100.0f);
if (ImGui::InputText("##default_theme", m_default_theme, sizeof(m_default_theme)))
m_settings_changed = true;
ImGui::Spacing();
ImGui::TextColored(accent_color, "Path Settings");
ImGui::Separator();
ImGui::Spacing();
ImGui::Text("Palettes Directory:");
ImGui::SameLine();
show_help_marker("Directory where color palettes are stored\nSupports ~ for home directory");
ImGui::SetNextItemWidth(-120.0f);
if (ImGui::InputText("##palettes_path", m_palettes_path, sizeof(m_palettes_path)))
m_settings_changed = true;
ImGui::SameLine();
if (ImGui::Button("Browse"))
{
std::string selected_path = file_dialogs::select_folder_dialog("Select Palettes Directory", m_palettes_path);
if (!selected_path.empty()) {
strncpy(m_palettes_path, selected_path.c_str(), sizeof(m_palettes_path) - 1);
m_palettes_path[sizeof(m_palettes_path) - 1] = '\0';
m_settings_changed = true;
}
}
}
void settings_window::render_appearance_tab()
{
ImGui::Spacing();
auto accent_color = palette_utils::get_color(m_current_palette, "accent");
ImGui::TextColored(accent_color, "Font Settings");
ImGui::Separator();
ImGui::Spacing();
ImGui::Text("Font Family:");
ImGui::SameLine();
show_help_marker("Select font family for the application interface");
ImGui::SetNextItemWidth(-1.0f);
if (ImGui::BeginCombo("##font", m_font))
{
for (int i = 0; i < static_cast<int>(m_available_fonts.size()); i++)
{
bool is_selected = (i == m_selected_font_idx);
if (ImGui::Selectable(m_available_fonts[i].c_str(), is_selected))
{
m_selected_font_idx = i;
strncpy(m_font, m_available_fonts[i].c_str(), sizeof(m_font) - 1);
m_font[sizeof(m_font) - 1] = '\0';
m_settings_changed = true;
}
if (is_selected)
ImGui::SetItemDefaultFocus();
}
ImGui::EndCombo();
}
ImGui::Spacing();
ImGui::Text("Font Size:");
ImGui::SameLine();
show_help_marker("Font size for the application interface (8-48)");
ImGui::SetNextItemWidth(120.0f);
int old_size = m_font_size;
if (ImGui::SliderInt("##font_size", &m_font_size, 8, 48, "%d px"))
{
if (old_size != m_font_size)
m_settings_changed = true;
}
ImGui::SameLine();
if (ImGui::Button("Reset"))
{
m_font_size = 14;
m_settings_changed = true;
}
}
void settings_window::render_status_messages()
{
if (!m_error_message.empty())
{
ImGui::Spacing();
auto error_bg_color = palette_utils::get_color(m_current_palette, "error");
auto error_text_color = palette_utils::get_color(m_current_palette, "on_error");
ImGui::PushStyleColor(ImGuiCol_ChildBg, error_bg_color);
ImGui::PushStyleColor(ImGuiCol_Border, error_bg_color);
ImGui::PushStyleVar(ImGuiStyleVar_ChildRounding, 4.0f);
ImGui::PushStyleVar(ImGuiStyleVar_ChildBorderSize, 1.0f);
if (ImGui::BeginChild("##error_box", ImVec2(0, 0), ImGuiChildFlags_AutoResizeY | ImGuiChildFlags_Borders))
{
ImGui::PushStyleColor(ImGuiCol_Text, error_text_color);
ImGui::TextWrapped("Error: %s", m_error_message.c_str());
ImGui::PopStyleColor();
ImGui::Spacing();
ImGui::PushStyleColor(ImGuiCol_Button, ImVec4(error_bg_color.x * 0.8f, error_bg_color.y * 0.8f, error_bg_color.z * 0.8f, error_bg_color.w));
ImGui::PushStyleColor(ImGuiCol_ButtonHovered, ImVec4(error_bg_color.x * 0.6f, error_bg_color.y * 0.6f, error_bg_color.z * 0.6f, error_bg_color.w));
ImGui::PushStyleColor(ImGuiCol_Text, error_text_color);
if (ImGui::Button("Dismiss##error"))
m_error_message.clear();
ImGui::PopStyleColor(3);
}
ImGui::EndChild();
ImGui::PopStyleVar(2);
ImGui::PopStyleColor(2);
}
}
void settings_window::render_action_buttons()
{
ImGui::Spacing();
float button_width = 100.0f;
float spacing = ImGui::GetStyle().ItemSpacing.x;
float window_width = ImGui::GetContentRegionAvail().x;
float total_buttons_width = 4 * button_width + 3 * spacing;
float start_pos = (window_width - total_buttons_width) * 0.5f;
if (start_pos > 0)
ImGui::SetCursorPosX(ImGui::GetCursorPosX() + start_pos);
if (ImGui::Button("OK", ImVec2(button_width, 0)))
{
apply_settings();
if (m_error_message.empty())
{
m_visible = false;
m_settings_changed = false;
}
}
ImGui::SameLine();
bool apply_disabled = !m_settings_changed;
if (apply_disabled)
{
ImGui::PushStyleVar(ImGuiStyleVar_Alpha, 0.5f);
}
if (ImGui::Button("Apply", ImVec2(button_width, 0)) && !apply_disabled)
{
apply_settings();
if (m_error_message.empty())
{
m_settings_changed = false;
}
}
if (apply_disabled)
{
ImGui::PopStyleVar();
}
ImGui::SameLine();
if (ImGui::Button("Reset", ImVec2(button_width, 0)))
{
reset_to_defaults();
}
ImGui::SameLine();
if (ImGui::Button("Cancel", ImVec2(button_width, 0)))
{
load_settings();
m_visible = false;
m_error_message.clear();
m_settings_changed = false;
}
}
void settings_window::show_help_marker(const char* desc)
{
ImGui::TextDisabled("(?)");
if (ImGui::BeginItemTooltip())
{
ImGui::PushTextWrapPos(ImGui::GetFontSize() * 35.0f);
ImGui::TextUnformatted(desc);
ImGui::PopTextWrapPos();
ImGui::EndTooltip();
}
}
void settings_window::reset_to_defaults()
{
strncpy(m_default_theme, "dark", sizeof(m_default_theme));
strncpy(m_palettes_path, "~/.config/clrsync/palettes", sizeof(m_palettes_path));
strncpy(m_font, "JetBrains Mono Nerd Font", sizeof(m_font));
m_font_size = 14;
m_error_message.clear();
m_settings_changed = true;
}

View File

@@ -1,50 +0,0 @@
#ifndef CLRSYNC_GUI_SETTINGS_WINDOW_HPP
#define CLRSYNC_GUI_SETTINGS_WINDOW_HPP
#include "core/palette/palette.hpp"
#include <string>
#include <vector>
class settings_window
{
public:
settings_window();
void render();
void show() { m_visible = true; }
void hide() { m_visible = false; }
bool is_visible() const { return m_visible; }
private:
void load_settings();
void save_settings();
void apply_settings();
void render_general_tab();
void render_appearance_tab();
void render_status_messages();
void render_action_buttons();
void show_help_marker(const char* desc);
void reset_to_defaults();
public:
void set_palette(const clrsync::core::palette& palette) {
m_current_palette = palette;
}
bool m_visible{false};
char m_default_theme[128];
char m_palettes_path[512];
char m_font[128];
int m_font_size;
std::vector<std::string> m_available_fonts;
int m_selected_font_idx;
std::string m_error_message;
bool m_settings_changed;
int m_current_tab;
clrsync::core::palette m_current_palette;
};
#endif // CLRSYNC_GUI_SETTINGS_WINDOW_HPP

137
src/gui/ui_manager.cpp Normal file
View File

@@ -0,0 +1,137 @@
#include "gui/ui_manager.hpp"
#include "gui/backend/backend.hpp"
#include "gui/platform/font_loader.hpp"
#include "gui/platform/file_browser.hpp"
#include "core/config/config.hpp"
#include <imgui.h>
namespace clrsync::gui
{
ui_manager::ui_manager(backend::backend_interface* backend)
: m_backend(backend)
{
m_font_loader = new font_loader();
}
ui_manager::~ui_manager()
{
shutdown();
delete m_font_loader;
m_font_loader = nullptr;
}
bool ui_manager::initialize(const ui_config& config)
{
IMGUI_CHECKVERSION();
m_imgui_context = ImGui::CreateContext();
if (!m_imgui_context)
{
return false;
}
ImGuiIO& io = ImGui::GetIO();
if (config.enable_docking)
io.ConfigFlags |= ImGuiConfigFlags_DockingEnable;
if (config.enable_keyboard_nav)
io.ConfigFlags |= ImGuiConfigFlags_NavEnableKeyboard;
if (!config.ini_path.empty())
io.IniFilename = config.ini_path.c_str();
ImGui::StyleColorsDark();
if (!m_backend->init_imgui_backend())
{
ImGui::DestroyContext(static_cast<ImGuiContext*>(m_imgui_context));
m_imgui_context = nullptr;
return false;
}
ImFont *font = m_font_loader->load_font(
clrsync::core::config::instance().font().c_str(),
clrsync::core::config::instance().font_size());
if (font)
io.FontDefault = font;
return true;
}
void ui_manager::shutdown()
{
if (m_imgui_context)
{
m_backend->shutdown_imgui_backend();
ImGui::DestroyContext(static_cast<ImGuiContext*>(m_imgui_context));
m_imgui_context = nullptr;
}
}
void ui_manager::begin_frame()
{
m_backend->imgui_new_frame();
ImGui::NewFrame();
}
void ui_manager::end_frame()
{
ImGui::Render();
m_backend->imgui_render_draw_data(ImGui::GetDrawData());
}
void ui_manager::push_default_font()
{
if (m_font_loader)
m_font_loader->push_default_font();
}
void ui_manager::pop_font()
{
if (m_font_loader)
m_font_loader->pop_font();
}
std::vector<std::string> ui_manager::get_system_fonts() const
{
if (m_font_loader)
return m_font_loader->get_system_fonts();
return {};
}
bool ui_manager::reload_font(const char* font_name, float size)
{
if (!m_font_loader)
return false;
ImFont* font = m_font_loader->load_font(font_name, size);
if (font)
{
ImGui::GetIO().FontDefault = font;
return true;
}
return false;
}
std::string ui_manager::open_file_dialog(const std::string& title,
const std::string& initial_path,
const std::vector<std::string>& filters)
{
return file_dialogs::open_file_dialog(title, initial_path, filters);
}
std::string ui_manager::save_file_dialog(const std::string& title,
const std::string& initial_path,
const std::vector<std::string>& filters)
{
return file_dialogs::save_file_dialog(title, initial_path, filters);
}
std::string ui_manager::select_folder_dialog(const std::string& title,
const std::string& initial_path)
{
return file_dialogs::select_folder_dialog(title, initial_path);
}
}

58
src/gui/ui_manager.hpp Normal file
View File

@@ -0,0 +1,58 @@
#ifndef CLRSYNC_UI_MANAGER_HPP
#define CLRSYNC_UI_MANAGER_HPP
#include <string>
#include <vector>
class font_loader;
struct ImFont;
namespace clrsync::gui::backend
{
class backend_interface;
}
namespace clrsync::gui
{
struct ui_config
{
std::string ini_path = "";
bool enable_docking = true;
bool enable_keyboard_nav = true;
};
class ui_manager
{
public:
explicit ui_manager(backend::backend_interface *backend);
~ui_manager();
bool initialize(const ui_config& config = ui_config());
void shutdown();
void begin_frame();
void end_frame();
void push_default_font();
void pop_font();
std::vector<std::string> get_system_fonts() const;
bool reload_font(const char* font_name, float size);
std::string open_file_dialog(const std::string& title = "Open File",
const std::string& initial_path = "",
const std::vector<std::string>& filters = {});
std::string save_file_dialog(const std::string& title = "Save File",
const std::string& initial_path = "",
const std::vector<std::string>& filters = {});
std::string select_folder_dialog(const std::string& title = "Select Folder",
const std::string& initial_path = "");
private:
backend::backend_interface *m_backend;
void *m_imgui_context = nullptr;
font_loader *m_font_loader = nullptr;
};
}
#endif

View File

@@ -0,0 +1,66 @@
#include "gui/views/about_window.hpp"
#include "core/common/version.hpp"
#include "gui/widgets/centered_text.hpp"
#include "gui/widgets/colors.hpp"
#include "gui/widgets/link_button.hpp"
#include "imgui.h"
about_window::about_window() = default;
void about_window::render(const clrsync::core::palette &pal)
{
if (!m_visible)
return;
ImGui::SetNextWindowSize(ImVec2(500, 400), ImGuiCond_FirstUseEver);
if (ImGui::Begin("About clrsync", &m_visible, ImGuiWindowFlags_NoResize))
{
using namespace clrsync::gui::widgets;
ImVec4 title_color = palette_color(pal, "info", "accent");
centered_text("clrsync", title_color);
ImVec4 subtitle_color = palette_color(pal, "editor_inactive", "foreground");
centered_text("Version " + clrsync::core::version_string(), subtitle_color);
ImGui::Spacing();
ImGui::Separator();
ImGui::Spacing();
ImGui::TextWrapped("A color scheme management tool.");
ImGui::Spacing();
ImGui::Spacing();
ImGui::Separator();
ImGui::Spacing();
constexpr float button_width = 200.0f;
float spacing = ImGui::GetStyle().ItemSpacing.x;
float total_width = 2.0f * button_width + spacing;
centered_buttons(total_width, [button_width]() {
link_button("GitHub Repository", "https://github.com/obsqrbtz/clrsync", button_width);
ImGui::SameLine();
link_button("Documentation", "https://binarygoose.dev/projects/clrsync/overview/", button_width);
});
ImGui::Spacing();
ImGui::Spacing();
ImGui::Separator();
ImGui::Spacing();
ImGui::TextColored(subtitle_color, "MIT License");
ImGui::TextWrapped(
"Copyright (c) 2025 Daniel Dada\n\n"
"Permission is hereby granted, free of charge, to any person obtaining a copy "
"of this software and associated documentation files (the \"Software\"), to deal "
"in the Software without restriction, including without limitation the rights "
"to use, copy, modify, merge, publish, distribute, sublicense, and/or sell "
"copies of the Software, and to permit persons to whom the Software is "
"furnished to do so, subject to the following conditions:\n\n"
"The above copyright notice and this permission notice shall be included in all "
"copies or substantial portions of the Software.");
}
ImGui::End();
}

View File

@@ -0,0 +1,33 @@
#ifndef CLRSYNC_GUI_ABOUT_WINDOW_HPP
#define CLRSYNC_GUI_ABOUT_WINDOW_HPP
#include "core/palette/palette.hpp"
class about_window
{
public:
about_window();
void render(const clrsync::core::palette &pal);
void render()
{
render(m_default_palette);
}
void show()
{
m_visible = true;
}
void hide()
{
m_visible = false;
}
bool is_visible() const
{
return m_visible;
}
private:
bool m_visible{false};
clrsync::core::palette m_default_palette;
};
#endif // CLRSYNC_GUI_ABOUT_WINDOW_HPP

View File

@@ -0,0 +1,151 @@
#include "color_scheme_editor.hpp"
#include "gui/controllers/theme_applier.hpp"
#include "gui/widgets/dialogs.hpp"
#include "gui/widgets/palette_selector.hpp"
#include "gui/widgets/input_dialog.hpp"
#include "gui/widgets/action_buttons.hpp"
#include "imgui.h"
#include "settings_window.hpp"
#include "template_editor.hpp"
#include <iostream>
color_scheme_editor::color_scheme_editor()
{
const auto &current = m_controller.current_palette();
if (!current.colors().empty())
{
theme_applier::apply_to_imgui(current);
m_preview.apply_palette(current);
}
else
{
std::cout << "WARNING: No palette loaded, skipping theme application\n";
}
setup_widgets();
}
void color_scheme_editor::notify_palette_changed()
{
if (m_template_editor)
{
m_template_editor->apply_current_palette(m_controller.current_palette());
}
if (m_settings_window)
{
m_settings_window->set_palette(m_controller.current_palette());
}
}
void color_scheme_editor::apply_themes()
{
const auto &current = m_controller.current_palette();
theme_applier::apply_to_imgui(current);
m_preview.apply_palette(current);
notify_palette_changed();
}
void color_scheme_editor::render_controls_and_colors()
{
ImGui::Begin("Color Schemes");
render_controls();
ImGui::Separator();
ImGui::BeginChild("ColorTableContent", ImVec2(0, 0), false);
m_color_table.render(m_controller.current_palette(), m_controller,
[this]() { apply_themes(); });
ImGui::EndChild();
ImGui::End();
}
void color_scheme_editor::render_preview()
{
ImGui::Begin("Color Preview");
m_preview.render(m_controller.current_palette());
ImGui::End();
}
void color_scheme_editor::render_controls()
{
const auto &current = m_controller.current_palette();
const auto &palettes = m_controller.palettes();
ImGui::PushStyleVar(ImGuiStyleVar_ItemSpacing, ImVec2(6, 8));
ImGui::PushStyleVar(ImGuiStyleVar_FramePadding, ImVec2(8, 5));
ImGui::AlignTextToFramePadding();
ImGui::Text("Palette:");
ImGui::SameLine();
m_palette_selector.render(m_controller, 200.0f);
ImGui::SameLine();
ImGui::SetCursorPosX(ImGui::GetCursorPosX() + 8);
if (ImGui::Button(" + New "))
{
m_new_palette_dialog.open("New Palette", "Enter a name for the new palette:", "Palette name...");
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Create a new palette");
m_new_palette_dialog.render();
ImGui::SameLine();
m_action_buttons.render(current);
if (m_show_delete_confirmation)
{
ImGui::OpenPopup("Delete Palette?");
m_show_delete_confirmation = false;
}
clrsync::gui::widgets::delete_confirmation_dialog("Delete Palette?", current.name(), "palette",
current, [this]() {
m_controller.delete_current_palette();
apply_themes();
});
ImGui::PopStyleVar(2);
}
void color_scheme_editor::setup_widgets()
{
m_palette_selector.set_on_selection_changed([this](const std::string &name) {
m_controller.select_palette(name);
apply_themes();
});
m_new_palette_dialog.set_on_submit([this](const std::string &name) {
m_controller.create_palette(name);
m_controller.select_palette(name);
apply_themes();
});
m_action_buttons.add_button({
" Save ",
"Save current palette to file",
[this]() { m_controller.save_current_palette(); }
});
m_action_buttons.add_button({
" Delete ",
"Delete current palette",
[this]() { m_show_delete_confirmation = true; },
true,
true
});
m_action_buttons.add_button({
" Apply Theme ",
"Apply current palette to all enabled templates",
[this]() { m_controller.apply_current_theme(); }
});
m_action_buttons.set_spacing(16.0f);
}

View File

@@ -0,0 +1,52 @@
#ifndef CLRSYNC_GUI_COLOR_SCHEME_EDITOR_HPP
#define CLRSYNC_GUI_COLOR_SCHEME_EDITOR_HPP
#include "gui/controllers/palette_controller.hpp"
#include "gui/views/color_table_renderer.hpp"
#include "gui/views/preview_renderer.hpp"
#include "gui/widgets/palette_selector.hpp"
#include "gui/widgets/input_dialog.hpp"
#include "gui/widgets/action_buttons.hpp"
class template_editor;
class settings_window;
class color_scheme_editor
{
public:
color_scheme_editor();
void render_controls_and_colors();
void render_preview();
void set_template_editor(template_editor *editor)
{
m_template_editor = editor;
}
void set_settings_window(settings_window *window)
{
m_settings_window = window;
}
const palette_controller &controller() const
{
return m_controller;
}
private:
void render_controls();
void apply_themes();
void notify_palette_changed();
void setup_widgets();
palette_controller m_controller;
color_table_renderer m_color_table;
preview_renderer m_preview;
template_editor *m_template_editor{nullptr};
settings_window *m_settings_window{nullptr};
bool m_show_delete_confirmation{false};
clrsync::gui::widgets::palette_selector m_palette_selector;
clrsync::gui::widgets::input_dialog m_new_palette_dialog;
clrsync::gui::widgets::action_buttons m_action_buttons;
};
#endif // CLRSYNC_GUI_COLOR_SCHEME_EDITOR_HPP

View File

@@ -1,5 +1,5 @@
#include "color_table_renderer.hpp" #include "gui/views/color_table_renderer.hpp"
#include "imgui_helpers.hpp" #include "gui/widgets/colors.hpp"
#include "imgui.h" #include "imgui.h"
#include <algorithm> #include <algorithm>
#include <cctype> #include <cctype>
@@ -36,7 +36,7 @@ void color_table_renderer::render_color_row(const std::string &name,
ImGui::TableSetColumnIndex(0); ImGui::TableSetColumnIndex(0);
const float key_col_width = ImGui::GetContentRegionAvail().x; const float key_col_width = ImGui::GetContentRegionAvail().x;
ImVec4 text_color = palette_utils::get_color(current, "info", "accent"); ImVec4 text_color = clrsync::gui::widgets::palette_color(current, "info", "accent");
ImGui::PushStyleColor(ImGuiCol_Text, text_color); ImGui::PushStyleColor(ImGuiCol_Text, text_color);
const bool copied = ImGui::Selectable(name.c_str(), false, 0, ImVec2(key_col_width, 0.0f)); const bool copied = ImGui::Selectable(name.c_str(), false, 0, ImVec2(key_col_width, 0.0f));
ImGui::PopStyleColor(); ImGui::PopStyleColor();
@@ -107,18 +107,22 @@ void color_table_renderer::render(const clrsync::core::palette& current,
{ {
if (current.colors().empty()) if (current.colors().empty())
{ {
ImVec4 warning_color = palette_utils::get_color(current, "warning", "accent"); ImVec4 warning_color = clrsync::gui::widgets::palette_color(current, "warning", "accent");
ImGui::TextColored(warning_color, "No palette loaded"); ImGui::TextColored(warning_color, "No palette loaded");
return; return;
} }
ImGui::PushStyleVar(ImGuiStyleVar_FramePadding, ImVec2(8, 6)); ImGui::PushStyleVar(ImGuiStyleVar_FramePadding, ImVec2(8, 6));
ImGui::AlignTextToFramePadding();
ImGui::Text("Filter:"); ImGui::Text("Filter:");
ImGui::SameLine(); ImGui::SameLine();
ImGui::SetNextItemWidth(200); ImGui::SetNextItemWidth(200);
bool filter_changed = ImGui::InputTextWithHint("##color_filter", "Search colors...", bool filter_changed =
m_filter_text, sizeof(m_filter_text)); ImGui::InputTextWithHint("##color_filter",
"Search colors...",
m_filter_text,
sizeof(m_filter_text));
if (m_filter_text[0] != '\0') if (m_filter_text[0] != '\0')
{ {
@@ -138,7 +142,8 @@ void color_table_renderer::render(const clrsync::core::palette& current,
ImGui::Separator(); ImGui::Separator();
ImGui::Spacing(); ImGui::Spacing();
auto draw_table = [&](const char *title, const char* id, const std::vector<const char *> &keys) { auto draw_table = [&](const char *title, const char *id,
const std::vector<const char *> &keys) {
bool has_matches = false; bool has_matches = false;
for (auto *k : keys) for (auto *k : keys)
{ {
@@ -152,14 +157,16 @@ void color_table_renderer::render(const clrsync::core::palette& current,
if (!has_matches) if (!has_matches)
return; return;
ImGui::PushStyleColor(ImGuiCol_Text, palette_utils::get_color(current, "accent")); ImGui::PushStyleColor(ImGuiCol_Text, clrsync::gui::widgets::palette_color(current, "accent"));
bool header_open = ImGui::TreeNodeEx(title, ImGuiTreeNodeFlags_DefaultOpen | ImGuiTreeNodeFlags_SpanAvailWidth); bool header_open = ImGui::TreeNodeEx(title, ImGuiTreeNodeFlags_DefaultOpen |
ImGuiTreeNodeFlags_SpanAvailWidth);
ImGui::PopStyleColor(); ImGui::PopStyleColor();
if (header_open) if (header_open)
{ {
if (ImGui::BeginTable(id, 3, if (ImGui::BeginTable(id, 3,
ImGuiTableFlags_Borders | ImGuiTableFlags_RowBg | ImGuiTableFlags_SizingStretchProp)) ImGuiTableFlags_Borders | ImGuiTableFlags_RowBg |
ImGuiTableFlags_SizingStretchProp))
{ {
ImGui::TableSetupColumn("Name", ImGuiTableColumnFlags_WidthFixed, 160.0f); ImGui::TableSetupColumn("Name", ImGuiTableColumnFlags_WidthFixed, 160.0f);
ImGui::TableSetupColumn("HEX", ImGuiTableColumnFlags_WidthFixed, 95.0f); ImGui::TableSetupColumn("HEX", ImGuiTableColumnFlags_WidthFixed, 95.0f);
@@ -177,23 +184,23 @@ void color_table_renderer::render(const clrsync::core::palette& current,
ImGui::Spacing(); ImGui::Spacing();
}; };
draw_table("General UI", "##general_ui", {"background", "on_background", "surface", "on_surface", draw_table("General UI", "##general_ui",
"surface_variant", "on_surface_variant", "foreground", {"background", "on_background", "surface", "on_surface", "surface_variant",
"cursor", "accent"}); "on_surface_variant", "foreground", "cursor", "accent"});
draw_table("Borders", "##borders", {"border_focused", "border"}); draw_table("Borders", "##borders", {"border_focused", "border"});
draw_table("Semantic Colors", "##semantic", {"success", "info", "warning", "error", draw_table(
"on_success", "on_info", "on_warning", "on_error"}); "Semantic Colors", "##semantic",
{"success", "info", "warning", "error", "on_success", "on_info", "on_warning", "on_error"});
draw_table("Editor", "##editor", {"editor_background", "editor_command", "editor_comment", draw_table("Editor", "##editor",
"editor_disabled", "editor_emphasis", "editor_error", {"editor_background", "editor_command", "editor_comment", "editor_disabled",
"editor_inactive", "editor_line_number", "editor_link", "editor_emphasis", "editor_error", "editor_inactive", "editor_line_number",
"editor_main", "editor_selected", "editor_selection_inactive", "editor_link", "editor_main", "editor_selected", "editor_selection_inactive",
"editor_string", "editor_success", "editor_warning"}); "editor_string", "editor_success", "editor_warning"});
draw_table("Terminal (Base16)", "##terminal", {"base00", "base01", "base02", "base03", draw_table("Terminal (Base16)", "##terminal",
"base04", "base05", "base06", "base07", {"base00", "base01", "base02", "base03", "base04", "base05", "base06", "base07",
"base08", "base09", "base0A", "base0B", "base08", "base09", "base0A", "base0B", "base0C", "base0D", "base0E", "base0F"});
"base0C", "base0D", "base0E", "base0F"});
} }

View File

@@ -0,0 +1,27 @@
#ifndef CLRSYNC_GUI_COLOR_TABLE_RENDERER_HPP
#define CLRSYNC_GUI_COLOR_TABLE_RENDERER_HPP
#include "core/palette/palette.hpp"
#include "gui/controllers/palette_controller.hpp"
#include <functional>
#include <string>
class color_table_renderer
{
public:
using OnColorChangedCallback = std::function<void()>;
void render(const clrsync::core::palette &palette, palette_controller &controller,
const OnColorChangedCallback &on_changed);
private:
void render_color_row(const std::string &name, const clrsync::core::palette &palette,
palette_controller &controller, const OnColorChangedCallback &on_changed);
bool matches_filter(const std::string &name) const;
char m_filter_text[128] = {0};
bool m_show_only_modified{false};
};
#endif // CLRSYNC_GUI_COLOR_TABLE_RENDERER_HPP

View File

@@ -1,6 +1,6 @@
#include "preview_renderer.hpp" #include "gui/views/preview_renderer.hpp"
#include "theme_applier.hpp" #include "gui/controllers/theme_applier.hpp"
#include "imgui_helpers.hpp" #include "gui/widgets/colors.hpp"
#include "imgui.h" #include "imgui.h"
#include <algorithm> #include <algorithm>
#include <array> #include <array>
@@ -74,12 +74,8 @@ int main()
static ImVec4 hex_to_imvec4(uint32_t hex) static ImVec4 hex_to_imvec4(uint32_t hex)
{ {
return { return {((hex >> 24) & 0xFF) / 255.0f, ((hex >> 16) & 0xFF) / 255.0f,
((hex >> 24) & 0xFF) / 255.0f, ((hex >> 8) & 0xFF) / 255.0f, (hex & 0xFF) / 255.0f};
((hex >> 16) & 0xFF) / 255.0f,
((hex >> 8) & 0xFF) / 255.0f,
(hex & 0xFF) / 255.0f
};
} }
void preview_renderer::apply_palette(const clrsync::core::palette &palette) void preview_renderer::apply_palette(const clrsync::core::palette &palette)
@@ -218,11 +214,13 @@ void preview_renderer::render_terminal_preview(const clrsync::core::palette& cur
draw_list->AddRect(p0, p1, ImGui::ColorConvertFloat4ToU32(border_col)); draw_list->AddRect(p0, p1, ImGui::ColorConvertFloat4ToU32(border_col));
} }
std::array<ImVec4, 8> bright_colors = {bright_black, bright_red, bright_green, bright_yellow, std::array<ImVec4, 8> bright_colors = {bright_black, bright_red, bright_green,
bright_blue, bright_magenta, bright_cyan, bright_white}; bright_yellow, bright_blue, bright_magenta,
bright_cyan, bright_white};
for (size_t i = 0; i < 8; i++) for (size_t i = 0; i < 8; i++)
{ {
ImVec2 p0 = ImVec2(start_pos.x + i * (box_size + spacing), start_pos.y + box_size + spacing); ImVec2 p0 =
ImVec2(start_pos.x + i * (box_size + spacing), start_pos.y + box_size + spacing);
ImVec2 p1 = ImVec2(p0.x + box_size, p0.y + box_size); ImVec2 p1 = ImVec2(p0.x + box_size, p0.y + box_size);
draw_list->AddRectFilled(p0, p1, ImGui::ColorConvertFloat4ToU32(bright_colors[i])); draw_list->AddRectFilled(p0, p1, ImGui::ColorConvertFloat4ToU32(bright_colors[i]));
draw_list->AddRect(p0, p1, ImGui::ColorConvertFloat4ToU32(border_col)); draw_list->AddRect(p0, p1, ImGui::ColorConvertFloat4ToU32(border_col));
@@ -238,7 +236,7 @@ void preview_renderer::render(const clrsync::core::palette& current)
{ {
if (current.colors().empty()) if (current.colors().empty())
{ {
ImVec4 error_color = palette_utils::get_color(current, "error", "accent"); ImVec4 error_color = clrsync::gui::widgets::palette_color(current, "error", "accent");
ImGui::TextColored(error_color, "Current palette is empty"); ImGui::TextColored(error_color, "Current palette is empty");
return; return;
} }

View File

@@ -1,8 +1,8 @@
#ifndef CLRSYNC_GUI_PREVIEW_RENDERER_HPP #ifndef CLRSYNC_GUI_PREVIEW_RENDERER_HPP
#define CLRSYNC_GUI_PREVIEW_RENDERER_HPP #define CLRSYNC_GUI_PREVIEW_RENDERER_HPP
#include "core/palette/palette.hpp"
#include "color_text_edit/TextEditor.h" #include "color_text_edit/TextEditor.h"
#include "core/palette/palette.hpp"
class preview_renderer class preview_renderer
{ {

View File

@@ -0,0 +1,240 @@
#include "gui/views/settings_window.hpp"
#include "core/common/error.hpp"
#include "core/config/config.hpp"
#include "gui/ui_manager.hpp"
#include "gui/widgets/section_header.hpp"
#include "imgui.h"
settings_window::settings_window(clrsync::gui::ui_manager *ui_mgr) : m_ui_manager(ui_mgr)
{
if (m_ui_manager)
m_available_fonts = m_ui_manager->get_system_fonts();
setup_widgets();
load_settings();
}
void settings_window::setup_widgets()
{
m_form.set_path_browse_callback([this](const std::string &current_path) -> std::string {
if (m_ui_manager)
return m_ui_manager->select_folder_dialog("Select Directory", current_path);
return "";
});
}
void settings_window::render()
{
if (!m_visible)
return;
ImGui::SetNextWindowSize(ImVec2(700, 500), ImGuiCond_FirstUseEver);
ImGui::SetNextWindowPos(ImGui::GetMainViewport()->GetCenter(), ImGuiCond_FirstUseEver,
ImVec2(0.5f, 0.5f));
if (ImGui::Begin("Settings", &m_visible, ImGuiWindowFlags_NoCollapse))
{
if (ImGui::BeginTabBar("SettingsTabs", ImGuiTabBarFlags_None))
{
if (ImGui::BeginTabItem("General"))
{
render_general_tab();
ImGui::EndTabItem();
}
if (ImGui::BeginTabItem("Appearance"))
{
render_appearance_tab();
ImGui::EndTabItem();
}
ImGui::EndTabBar();
}
m_error.render(m_current_palette);
ImGui::Separator();
clrsync::gui::widgets::settings_buttons_callbacks callbacks{
.on_ok =
[this]() {
apply_settings();
if (!m_error.has_error())
{
m_visible = false;
m_settings_changed = false;
}
},
.on_apply =
[this]() {
apply_settings();
if (!m_error.has_error())
m_settings_changed = false;
},
.on_reset = [this]() { reset_to_defaults(); },
.on_cancel =
[this]() {
load_settings();
m_visible = false;
}};
m_buttons.render(callbacks, m_settings_changed);
}
ImGui::End();
}
void settings_window::load_settings()
{
auto &cfg = clrsync::core::config::instance();
m_default_theme = cfg.default_theme();
m_palettes_path = cfg.palettes_path();
m_font = cfg.font();
m_font_size = cfg.font_size();
m_selected_font_idx = 0;
for (int i = 0; i < static_cast<int>(m_available_fonts.size()); i++)
{
if (m_available_fonts[i] == m_font)
{
m_selected_font_idx = i;
break;
}
}
m_error.clear();
m_settings_changed = false;
}
void settings_window::apply_settings()
{
auto &cfg = clrsync::core::config::instance();
if (m_default_theme.empty())
{
m_error.set("Default theme cannot be empty");
return;
}
if (m_palettes_path.empty())
{
m_error.set("Palettes path cannot be empty");
return;
}
if (m_font.empty())
{
m_error.set("Font cannot be empty");
return;
}
if (m_font_size < 8 || m_font_size > 48)
{
m_error.set("Font size must be between 8 and 48");
return;
}
auto result1 = cfg.set_default_theme(m_default_theme);
if (!result1)
{
m_error.set("Failed to set default theme: " + result1.error().description());
return;
}
auto result2 = cfg.set_palettes_path(m_palettes_path);
if (!result2)
{
m_error.set("Failed to set palettes path: " + result2.error().description());
return;
}
auto result3 = cfg.set_font(m_font);
if (!result3)
{
m_error.set("Failed to set font: " + result3.error().description());
return;
}
auto result4 = cfg.set_font_size(m_font_size);
if (!result4)
{
m_error.set("Failed to set font size: " + result4.error().description());
return;
}
if (m_ui_manager && !m_ui_manager->reload_font(m_font.c_str(), m_font_size))
{
m_error.set("Failed to load font: " + m_font);
return;
}
m_error.clear();
m_settings_changed = false;
}
void settings_window::render_general_tab()
{
using namespace clrsync::gui::widgets;
section_header("Theme Settings", m_current_palette);
form_field_config theme_cfg;
theme_cfg.label = "Default Theme";
theme_cfg.label_width = 150.0f;
theme_cfg.tooltip = "The default color scheme to load on startup";
theme_cfg.field_width = -100.0f;
if (m_form.render_text(theme_cfg, m_default_theme))
m_settings_changed = true;
section_header("Path Settings", m_current_palette);
form_field_config path_cfg;
path_cfg.label = "Palettes Directory";
path_cfg.label_width = 150.0f;
path_cfg.tooltip = "Directory where color palettes are stored\nSupports ~ for home directory";
path_cfg.field_width = -1.0f;
path_cfg.type = field_type::path;
if (m_form.render_path(path_cfg, m_palettes_path))
m_settings_changed = true;
}
void settings_window::render_appearance_tab()
{
using namespace clrsync::gui::widgets;
section_header("Font Settings", m_current_palette);
form_field_config font_cfg;
font_cfg.label = "Font Family";
font_cfg.label_width = 150.0f;
font_cfg.tooltip = "Select font family for the application interface";
font_cfg.field_width = -1.0f;
font_cfg.type = field_type::combo;
if (m_form.render_combo(font_cfg, m_available_fonts, m_selected_font_idx, m_font))
m_settings_changed = true;
ImGui::Spacing();
form_field_config size_cfg;
size_cfg.label = "Font Size",
size_cfg.label_width = 150.0f;
size_cfg.tooltip = "Font size for the application interface (8-48)";
size_cfg.field_width = 120.0f;
size_cfg.type = field_type::slider;
size_cfg.min_value = 8.0f;
size_cfg.max_value = 48.0f;
size_cfg.format = "%d px";
size_cfg.show_reset = true;
size_cfg.default_value = 14;
if (m_form.render_slider(size_cfg, m_font_size))
m_settings_changed = true;
}
void settings_window::reset_to_defaults()
{
m_default_theme = "dark";
m_palettes_path = "~/.config/clrsync/palettes";
m_font = "JetBrains Mono Nerd Font";
m_font_size = 14;
m_error.clear();
m_settings_changed = true;
}

View File

@@ -0,0 +1,54 @@
#ifndef CLRSYNC_GUI_SETTINGS_WINDOW_HPP
#define CLRSYNC_GUI_SETTINGS_WINDOW_HPP
#include "core/palette/palette.hpp"
#include "gui/widgets/error_message.hpp"
#include "gui/widgets/form_field.hpp"
#include "gui/widgets/settings_buttons.hpp"
#include <string>
#include <vector>
namespace clrsync::gui
{
class ui_manager;
}
class settings_window
{
public:
settings_window(clrsync::gui::ui_manager* ui_mgr);
void render();
void show() { m_visible = true; }
void hide() { m_visible = false; }
bool is_visible() const { return m_visible; }
void set_palette(const clrsync::core::palette& palette) { m_current_palette = palette; }
private:
void load_settings();
void apply_settings();
void render_general_tab();
void render_appearance_tab();
void reset_to_defaults();
void setup_widgets();
bool m_visible{false};
bool m_settings_changed{false};
std::string m_default_theme;
std::string m_palettes_path;
std::string m_font;
int m_font_size{14};
int m_selected_font_idx{0};
std::vector<std::string> m_available_fonts;
clrsync::core::palette m_current_palette;
clrsync::gui::ui_manager* m_ui_manager;
clrsync::gui::widgets::form_field m_form;
clrsync::gui::widgets::error_message m_error;
clrsync::gui::widgets::settings_buttons m_buttons;
};
#endif // CLRSYNC_GUI_SETTINGS_WINDOW_HPP

View File

@@ -1,28 +1,29 @@
#include "template_editor.hpp" #include "template_editor.hpp"
#include "core/common/utils.hpp"
#include "core/config/config.hpp" #include "core/config/config.hpp"
#include "core/theme/theme_template.hpp"
#include "core/palette/color_keys.hpp" #include "core/palette/color_keys.hpp"
#include "core/utils.hpp" #include "core/theme/theme_template.hpp"
#include "imgui_helpers.hpp" #include "gui/widgets/colors.hpp"
#include "file_browser.hpp" #include "gui/widgets/dialogs.hpp"
#include "gui/ui_manager.hpp"
#include "imgui.h" #include "imgui.h"
#include <algorithm> #include <algorithm>
#include <filesystem> #include <filesystem>
#include <fstream> #include <fstream>
#include <ranges> #include <ranges>
namespace { namespace
{
const std::vector<std::string> COLOR_FORMATS = { const std::vector<std::string> COLOR_FORMATS = {
"hex", "hex_stripped", "hexa", "hexa_stripped", "hex", "hex_stripped", "hexa", "hexa_stripped", "r", "g", "b", "a", "rgb", "rgba", "h", "s",
"r", "g", "b", "a", "l", "hsl", "hsla"};
"rgb", "rgba",
"h", "s", "l",
"hsl", "hsla"
};
} }
template_editor::template_editor() : m_template_name("new_template") template_editor::template_editor(clrsync::gui::ui_manager* ui_mgr)
: m_ui_manager(ui_mgr)
{ {
m_control_state.name = "new_template";
m_autocomplete_bg_color = ImVec4(0.12f, 0.12f, 0.15f, 0.98f); m_autocomplete_bg_color = ImVec4(0.12f, 0.12f, 0.15f, 0.98f);
m_autocomplete_border_color = ImVec4(0.4f, 0.4f, 0.45f, 1.0f); m_autocomplete_border_color = ImVec4(0.4f, 0.4f, 0.45f, 1.0f);
m_autocomplete_selected_color = ImVec4(0.25f, 0.45f, 0.75f, 0.9f); m_autocomplete_selected_color = ImVec4(0.25f, 0.45f, 0.75f, 0.9f);
@@ -50,6 +51,33 @@ template_editor::template_editor() : m_template_name("new_template")
m_editor.SetText("# Enter your template here\n# Use {color_key} for color variables\n# " m_editor.SetText("# Enter your template here\n# Use {color_key} for color variables\n# "
"Examples: {color.hex}, {color.rgb}, {color.r}\n\n"); "Examples: {color.hex}, {color.rgb}, {color.r}\n\n");
m_editor.SetShowWhitespaces(false); m_editor.SetShowWhitespaces(false);
setup_callbacks();
}
void template_editor::setup_callbacks()
{
m_callbacks.on_new = [this]() { new_template(); };
m_callbacks.on_save = [this]() { save_template(); };
m_callbacks.on_delete = [this]() { delete_template(); };
m_callbacks.on_enabled_changed = [this](bool enabled) {
m_template_controller.set_template_enabled(m_control_state.name, enabled);
};
m_callbacks.on_browse_input = [this](const std::string& path) -> std::string {
return m_ui_manager->open_file_dialog("Select Template File", path);
};
m_callbacks.on_browse_output = [this](const std::string& path) -> std::string {
return m_ui_manager->save_file_dialog("Select Output File", path);
};
m_callbacks.on_input_path_changed = [this](const std::string& path) {
m_template_controller.set_template_input_path(m_control_state.name, path);
};
m_callbacks.on_output_path_changed = [this](const std::string& path) {
m_template_controller.set_template_output_path(m_control_state.name, path);
};
m_callbacks.on_reload_command_changed = [this](const std::string& cmd) {
m_template_controller.set_template_reload_command(m_control_state.name, cmd);
};
} }
void template_editor::apply_current_palette(const clrsync::core::palette &pal) void template_editor::apply_current_palette(const clrsync::core::palette &pal)
@@ -59,7 +87,7 @@ void template_editor::apply_current_palette(const clrsync::core::palette &pal)
if (colors.empty()) if (colors.empty())
return; return;
auto get_color_u32 = [&](const std::string &key, const std::string &fallback = "") -> uint32_t { auto get_color_u32 = [&](const std::string &key, const std::string &fallback = "") -> uint32_t {
return palette_utils::get_color_u32(pal, key, fallback); return clrsync::gui::widgets::palette_color_u32(pal, key, fallback);
}; };
auto palette = m_editor.GetPalette(); auto palette = m_editor.GetPalette();
@@ -102,13 +130,14 @@ void template_editor::apply_current_palette(const clrsync::core::palette &pal)
m_editor.SetPalette(palette); m_editor.SetPalette(palette);
m_autocomplete_bg_color = palette_utils::get_color(pal, "surface", "background"); m_autocomplete_bg_color = clrsync::gui::widgets::palette_color(pal, "surface", "background");
m_autocomplete_bg_color.w = 0.98f; m_autocomplete_bg_color.w = 0.98f;
m_autocomplete_border_color = palette_utils::get_color(pal, "border", "surface_variant"); m_autocomplete_border_color = clrsync::gui::widgets::palette_color(pal, "border", "surface_variant");
m_autocomplete_selected_color = palette_utils::get_color(pal, "accent", "surface_variant"); m_autocomplete_selected_color = clrsync::gui::widgets::palette_color(pal, "accent", "surface_variant");
m_autocomplete_text_color = palette_utils::get_color(pal, "on_surface", "foreground"); m_autocomplete_text_color = clrsync::gui::widgets::palette_color(pal, "on_surface", "foreground");
m_autocomplete_selected_text_color = palette_utils::get_color(pal, "on_surface", "foreground"); m_autocomplete_selected_text_color = clrsync::gui::widgets::palette_color(pal, "on_surface", "foreground");
m_autocomplete_dim_text_color = palette_utils::get_color(pal, "on_surface_variant", "editor_inactive"); m_autocomplete_dim_text_color =
clrsync::gui::widgets::palette_color(pal, "on_surface_variant", "editor_inactive");
} }
void template_editor::update_autocomplete_suggestions() void template_editor::update_autocomplete_suggestions()
@@ -148,8 +177,7 @@ void template_editor::update_autocomplete_suggestions()
{ {
bool should_reset_dismissal = false; bool should_reset_dismissal = false;
if (cursor.mLine != m_dismiss_position.mLine || if (cursor.mLine != m_dismiss_position.mLine || brace_pos != m_dismiss_brace_pos ||
brace_pos != m_dismiss_brace_pos ||
abs(cursor.mColumn - m_dismiss_position.mColumn) > 3) abs(cursor.mColumn - m_dismiss_position.mColumn) > 3)
{ {
should_reset_dismissal = true; should_reset_dismissal = true;
@@ -190,8 +218,7 @@ void template_editor::update_autocomplete_suggestions()
{ {
for (const auto &fmt : COLOR_FORMATS) for (const auto &fmt : COLOR_FORMATS)
{ {
if (format_prefix.empty() || if (format_prefix.empty() || fmt.find(format_prefix) == 0 ||
fmt.find(format_prefix) == 0 ||
fmt.find(format_prefix) != std::string::npos) fmt.find(format_prefix) != std::string::npos)
{ {
m_autocomplete_suggestions.push_back(color_key + "." + fmt); m_autocomplete_suggestions.push_back(color_key + "." + fmt);
@@ -204,8 +231,7 @@ void template_editor::update_autocomplete_suggestions()
for (size_t i = 0; i < clrsync::core::NUM_COLOR_KEYS; ++i) for (size_t i = 0; i < clrsync::core::NUM_COLOR_KEYS; ++i)
{ {
std::string key = clrsync::core::COLOR_KEYS[i]; std::string key = clrsync::core::COLOR_KEYS[i];
if (m_autocomplete_prefix.empty() || if (m_autocomplete_prefix.empty() || key.find(m_autocomplete_prefix) == 0 ||
key.find(m_autocomplete_prefix) == 0 ||
key.find(m_autocomplete_prefix) != std::string::npos) key.find(m_autocomplete_prefix) != std::string::npos)
{ {
m_autocomplete_suggestions.push_back(key); m_autocomplete_suggestions.push_back(key);
@@ -241,7 +267,8 @@ void template_editor::render_autocomplete(const ImVec2& editor_pos)
const float line_number_width = 50.0f; const float line_number_width = 50.0f;
ImVec2 popup_pos; ImVec2 popup_pos;
popup_pos.x = editor_pos.x + line_number_width + (m_autocomplete_start_pos.mColumn * char_width); popup_pos.x =
editor_pos.x + line_number_width + (m_autocomplete_start_pos.mColumn * char_width);
popup_pos.y = editor_pos.y + ((cursor.mLine + 1) * line_height); popup_pos.y = editor_pos.y + ((cursor.mLine + 1) * line_height);
ImGui::SetNextWindowPos(popup_pos, ImGuiCond_Always); ImGui::SetNextWindowPos(popup_pos, ImGuiCond_Always);
@@ -249,7 +276,8 @@ void template_editor::render_autocomplete(const ImVec2& editor_pos)
ImGuiWindowFlags flags = ImGuiWindowFlags_NoTitleBar | ImGuiWindowFlags_NoResize | ImGuiWindowFlags flags = ImGuiWindowFlags_NoTitleBar | ImGuiWindowFlags_NoResize |
ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoSavedSettings | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoSavedSettings |
ImGuiWindowFlags_NoFocusOnAppearing | ImGuiWindowFlags_AlwaysAutoResize; ImGuiWindowFlags_NoFocusOnAppearing |
ImGuiWindowFlags_AlwaysAutoResize;
ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2(6, 6)); ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2(6, 6));
ImGui::PushStyleVar(ImGuiStyleVar_WindowRounding, 6.0f); ImGui::PushStyleVar(ImGuiStyleVar_WindowRounding, 6.0f);
@@ -295,8 +323,8 @@ void template_editor::render_autocomplete(const ImVec2& editor_pos)
std::string display_text = " " + suggestion; std::string display_text = " " + suggestion;
if (ImGui::Selectable(display_text.c_str(), is_selected, if (ImGui::Selectable(display_text.c_str(), is_selected, ImGuiSelectableFlags_None,
ImGuiSelectableFlags_None, ImVec2(0, 0))) ImVec2(0, 0)))
{ {
auto start = m_autocomplete_start_pos; auto start = m_autocomplete_start_pos;
auto end = m_editor.GetCursorPosition(); auto end = m_editor.GetCursorPosition();
@@ -361,10 +389,9 @@ void template_editor::render()
m_show_delete_confirmation = false; m_show_delete_confirmation = false;
} }
palette_utils::render_delete_confirmation_popup( clrsync::gui::widgets::delete_confirmation_dialog(
"Delete Template?", m_template_name, "template", m_current_palette, "Delete Template?", m_control_state.name, "template", m_current_palette, [this]() {
[this]() { bool success = m_template_controller.remove_template(m_control_state.name);
bool success = m_template_controller.remove_template(m_template_name);
if (success) if (success)
{ {
new_template(); new_template();
@@ -372,7 +399,7 @@ void template_editor::render()
} }
else else
{ {
m_validation_error = "Failed to delete template"; m_validation.set("Failed to delete template");
} }
}); });
@@ -381,215 +408,29 @@ void template_editor::render()
void template_editor::render_controls() void template_editor::render_controls()
{ {
ImGui::PushStyleVar(ImGuiStyleVar_ItemSpacing, ImVec2(4, 8));
if (ImGui::Button(" + New "))
{
new_template();
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Create a new template");
if (ImGui::IsWindowFocused(ImGuiFocusedFlags_RootAndChildWindows) && if (ImGui::IsWindowFocused(ImGuiFocusedFlags_RootAndChildWindows) &&
ImGui::IsKeyDown(ImGuiKey_LeftCtrl) && ImGui::IsKeyPressed(ImGuiKey_S)) ImGui::IsKeyDown(ImGuiKey_LeftCtrl) && ImGui::IsKeyPressed(ImGuiKey_S))
{ {
save_template(); save_template();
} }
ImGui::SameLine(); m_controls.render(m_control_state, m_callbacks, m_current_palette, m_validation);
if (ImGui::Button(" Save "))
{
save_template();
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Save template (Ctrl+S)");
if (m_is_editing_existing)
{
ImGui::SameLine();
auto error = palette_utils::get_color(m_current_palette, "error");
auto error_hover = ImVec4(error.x * 1.1f, error.y * 1.1f, error.z * 1.1f,
error.w);
auto error_active = ImVec4(error.x * 0.8f, error.y * 0.8f, error.z * 0.8f,
error.w);
auto on_error = palette_utils::get_color(m_current_palette, "on_error");
ImGui::PushStyleColor(ImGuiCol_Button, error);
ImGui::PushStyleColor(ImGuiCol_ButtonHovered, error_hover);
ImGui::PushStyleColor(ImGuiCol_ButtonActive, error_active);
ImGui::PushStyleColor(ImGuiCol_Text, on_error);
if (ImGui::Button(" Delete "))
{
delete_template();
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Delete this template");
ImGui::PopStyleColor(4);
}
ImGui::SameLine();
ImGui::SetCursorPosX(ImGui::GetCursorPosX() + 10);
bool enabled_changed = false;
if (m_enabled)
{
ImVec4 success_color = palette_utils::get_color(m_current_palette, "success", "accent");
ImVec4 success_on_color = palette_utils::get_color(m_current_palette, "on_success", "on_surface");
ImVec4 success_hover = ImVec4(success_color.x * 1.2f, success_color.y * 1.2f, success_color.z * 1.2f, 0.6f);
ImGui::PushStyleColor(ImGuiCol_FrameBg, ImVec4(success_color.x, success_color.y, success_color.z, 0.5f));
ImGui::PushStyleColor(ImGuiCol_FrameBgHovered, success_hover);
ImGui::PushStyleColor(ImGuiCol_CheckMark, success_on_color);
}
else
{
ImVec4 error_color = palette_utils::get_color(m_current_palette, "error", "accent");
ImVec4 error_on_color = palette_utils::get_color(m_current_palette, "on_error", "on_surface");
ImVec4 error_hover = ImVec4(error_color.x * 1.2f, error_color.y * 1.2f, error_color.z * 1.2f, 0.6f);
ImGui::PushStyleColor(ImGuiCol_FrameBg, ImVec4(error_color.x, error_color.y, error_color.z, 0.5f));
ImGui::PushStyleColor(ImGuiCol_FrameBgHovered, error_hover);
ImGui::PushStyleColor(ImGuiCol_CheckMark, error_on_color);
}
enabled_changed = ImGui::Checkbox("Enabled", &m_enabled);
ImGui::PopStyleColor(3);
if (enabled_changed && m_is_editing_existing)
{
m_template_controller.set_template_enabled(m_template_name, m_enabled);
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Enable/disable this template for theme application");
ImGui::PopStyleVar();
ImGui::Spacing();
ImGui::AlignTextToFramePadding();
ImGui::Text("Name:");
ImGui::SameLine(80);
ImGui::SetNextItemWidth(180.0f);
char name_buf[256] = {0};
snprintf(name_buf, sizeof(name_buf), "%s", m_template_name.c_str());
if (ImGui::InputText("##template_name", name_buf, sizeof(name_buf)))
{
m_template_name = name_buf;
if (!m_template_name.empty())
{
m_validation_error = "";
}
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Unique name for this template");
ImGui::AlignTextToFramePadding();
ImGui::Text("Input:");
ImGui::SameLine(80);
ImGui::SetNextItemWidth(-120.0f);
char input_path_buf[512] = {0};
snprintf(input_path_buf, sizeof(input_path_buf), "%s", m_input_path.c_str());
if (ImGui::InputTextWithHint("##input_path", "Path to template file...",
input_path_buf, sizeof(input_path_buf)))
{
m_input_path = input_path_buf;
if (!m_input_path.empty())
{
m_validation_error = "";
}
if (m_is_editing_existing)
{
m_template_controller.set_template_input_path(m_template_name, m_input_path);
}
}
ImGui::SameLine();
if (ImGui::Button("Browse##input"))
{
std::string selected_path = file_dialogs::open_file_dialog("Select Template File", m_input_path);
if (!selected_path.empty()) {
m_input_path = selected_path;
if (m_is_editing_existing) {
m_template_controller.set_template_input_path(m_template_name, m_input_path);
}
m_validation_error = "";
}
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Path where the template source file is stored");
ImGui::AlignTextToFramePadding();
ImGui::Text("Output:");
ImGui::SameLine(80);
ImGui::SetNextItemWidth(-120.0f);
char path_buf[512] = {0};
snprintf(path_buf, sizeof(path_buf), "%s", m_output_path.c_str());
if (ImGui::InputTextWithHint("##output_path", "Path for generated config...",
path_buf, sizeof(path_buf)))
{
m_output_path = path_buf;
if (!m_output_path.empty())
{
m_validation_error = "";
}
if (m_is_editing_existing)
{
m_template_controller.set_template_output_path(m_template_name, m_output_path);
}
}
ImGui::SameLine();
if (ImGui::Button("Browse##output"))
{
std::string selected_path = file_dialogs::save_file_dialog("Select Output File", m_output_path);
if (!selected_path.empty()) {
m_output_path = selected_path;
if (m_is_editing_existing) {
m_template_controller.set_template_output_path(m_template_name, m_output_path);
}
m_validation_error = "";
}
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Path where the processed config will be written");
ImGui::AlignTextToFramePadding();
ImGui::Text("Reload:");
ImGui::SameLine(80);
ImGui::SetNextItemWidth(-FLT_MIN);
char reload_buf[512] = {0};
snprintf(reload_buf, sizeof(reload_buf), "%s", m_reload_command.c_str());
if (ImGui::InputTextWithHint("##reload_cmd", "Command to reload app (optional)...",
reload_buf, sizeof(reload_buf)))
{
m_reload_command = reload_buf;
if (m_is_editing_existing)
{
m_template_controller.set_template_reload_command(m_template_name, m_reload_command);
}
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Shell command to run after applying theme (e.g., 'pkill -USR1 kitty')");
if (!m_validation_error.empty())
{
ImGui::Spacing();
ImVec4 error_color = palette_utils::get_color(m_current_palette, "error", "accent");
ImGui::PushStyleColor(ImGuiCol_Text, error_color);
ImGui::TextWrapped("%s", m_validation_error.c_str());
ImGui::PopStyleColor();
}
} }
void template_editor::render_editor() void template_editor::render_editor()
{ {
ImGui::PushStyleVar(ImGuiStyleVar_ItemSpacing, ImVec2(8, 4)); ImGui::PushStyleVar(ImGuiStyleVar_ItemSpacing, ImVec2(8, 4));
if (!m_is_editing_existing) if (!m_control_state.is_editing_existing)
{ {
ImVec4 success_color = palette_utils::get_color(m_current_palette, "success", "accent"); ImVec4 success_color = clrsync::gui::widgets::palette_color(m_current_palette, "success", "accent");
ImGui::PushStyleColor(ImGuiCol_Text, success_color); ImGui::PushStyleColor(ImGuiCol_Text, success_color);
ImGui::Text(" New Template"); ImGui::Text(" New Template");
ImGui::PopStyleColor(); ImGui::PopStyleColor();
} }
else else
{ {
ImGui::Text(" %s", m_template_name.c_str()); ImGui::Text(" %s", m_control_state.name.c_str());
auto trim_right = [](const std::string &s) -> std::string { auto trim_right = [](const std::string &s) -> std::string {
size_t end = s.find_last_not_of("\r\n"); size_t end = s.find_last_not_of("\r\n");
return (end == std::string::npos) ? "" : s.substr(0, end + 1); return (end == std::string::npos) ? "" : s.substr(0, end + 1);
@@ -602,7 +443,7 @@ void template_editor::render_editor()
if (m_has_unsaved_changes) if (m_has_unsaved_changes)
{ {
ImGui::SameLine(); ImGui::SameLine();
ImVec4 warning_color = palette_utils::get_color(m_current_palette, "warning", "accent"); ImVec4 warning_color = clrsync::gui::widgets::palette_color(m_current_palette, "warning", "accent");
ImGui::PushStyleColor(ImGuiCol_Text, warning_color); ImGui::PushStyleColor(ImGuiCol_Text, warning_color);
ImGui::Text("(unsaved)"); ImGui::Text("(unsaved)");
ImGui::PopStyleColor(); ImGui::PopStyleColor();
@@ -689,9 +530,9 @@ void template_editor::render_template_list()
ImGui::TextDisabled("(%d)", (int)m_template_controller.templates().size()); ImGui::TextDisabled("(%d)", (int)m_template_controller.templates().size());
ImGui::Separator(); ImGui::Separator();
if (!m_is_editing_existing) if (!m_control_state.is_editing_existing)
{ {
ImVec4 success_color = palette_utils::get_color(m_current_palette, "success", "accent"); ImVec4 success_color = clrsync::gui::widgets::palette_color(m_current_palette, "success", "accent");
ImVec4 success_bg = ImVec4(success_color.x, success_color.y, success_color.z, 0.5f); ImVec4 success_bg = ImVec4(success_color.x, success_color.y, success_color.z, 0.5f);
ImGui::PushStyleColor(ImGuiCol_Text, success_color); ImGui::PushStyleColor(ImGuiCol_Text, success_color);
ImGui::PushStyleColor(ImGuiCol_Header, success_bg); ImGui::PushStyleColor(ImGuiCol_Header, success_bg);
@@ -704,11 +545,12 @@ void template_editor::render_template_list()
for (const auto &[key, tmpl] : templates) for (const auto &[key, tmpl] : templates)
{ {
const bool selected = (m_template_name == key && m_is_editing_existing); const bool selected = (m_control_state.name == key && m_control_state.is_editing_existing);
if (!tmpl.enabled()) if (!tmpl.enabled())
{ {
ImVec4 disabled_color = palette_utils::get_color(m_current_palette, "on_surface_variant", "editor_inactive"); ImVec4 disabled_color = clrsync::gui::widgets::palette_color(
m_current_palette, "on_surface_variant", "editor_inactive");
ImGui::PushStyleColor(ImGuiCol_Text, disabled_color); ImGui::PushStyleColor(ImGuiCol_Text, disabled_color);
} }
@@ -788,44 +630,43 @@ bool template_editor::is_valid_path(const std::string &path)
void template_editor::save_template() void template_editor::save_template()
{ {
std::string trimmed_name = m_template_name; std::string trimmed_name = m_control_state.name;
trimmed_name.erase(0, trimmed_name.find_first_not_of(" \t\n\r")); trimmed_name.erase(0, trimmed_name.find_first_not_of(" \t\n\r"));
trimmed_name.erase(trimmed_name.find_last_not_of(" \t\n\r") + 1); trimmed_name.erase(trimmed_name.find_last_not_of(" \t\n\r") + 1);
std::string trimmed_input_path = m_input_path; std::string trimmed_input_path = m_control_state.input_path;
trimmed_input_path.erase(0, trimmed_input_path.find_first_not_of(" \t\n\r")); trimmed_input_path.erase(0, trimmed_input_path.find_first_not_of(" \t\n\r"));
trimmed_input_path.erase(trimmed_input_path.find_last_not_of(" \t\n\r") + 1); trimmed_input_path.erase(trimmed_input_path.find_last_not_of(" \t\n\r") + 1);
std::string trimmed_path = m_output_path; std::string trimmed_path = m_control_state.output_path;
trimmed_path.erase(0, trimmed_path.find_first_not_of(" \t\n\r")); trimmed_path.erase(0, trimmed_path.find_first_not_of(" \t\n\r"));
trimmed_path.erase(trimmed_path.find_last_not_of(" \t\n\r") + 1); trimmed_path.erase(trimmed_path.find_last_not_of(" \t\n\r") + 1);
if (trimmed_name.empty()) if (trimmed_name.empty())
{ {
m_validation_error = "Error: Template name cannot be empty!"; m_validation.set("Error: Template name cannot be empty!");
return; return;
} }
if (trimmed_input_path.empty()) if (trimmed_input_path.empty())
{ {
m_validation_error = "Error: Input path cannot be empty!"; m_validation.set("Error: Input path cannot be empty!");
return; return;
} }
if (trimmed_path.empty()) if (trimmed_path.empty())
{ {
m_validation_error = "Error: Output path cannot be empty!"; m_validation.set("Error: Output path cannot be empty!");
return; return;
} }
if (!is_valid_path(trimmed_path)) if (!is_valid_path(trimmed_path))
{ {
m_validation_error = m_validation.set("Error: Output path is invalid! Must be a valid file path with directory.");
"Error: Output path is invalid! Must be a valid file path with directory.";
return; return;
} }
m_validation_error = ""; m_validation.clear();
auto &cfg = clrsync::core::config::instance(); auto &cfg = clrsync::core::config::instance();
@@ -840,7 +681,7 @@ void template_editor::save_template()
} }
catch (const std::exception &e) catch (const std::exception &e)
{ {
m_validation_error = "Error: Could not create directory for input path"; m_validation.set("Error: Could not create directory for input path");
return; return;
} }
} }
@@ -850,7 +691,7 @@ void template_editor::save_template()
std::ofstream out(template_file); std::ofstream out(template_file);
if (!out.is_open()) if (!out.is_open())
{ {
m_validation_error = "Failed to write template file"; m_validation.set("Failed to write template file");
return; return;
} }
@@ -858,20 +699,20 @@ void template_editor::save_template()
out.close(); out.close();
clrsync::core::theme_template tmpl(trimmed_name, template_file.string(), trimmed_path); clrsync::core::theme_template tmpl(trimmed_name, template_file.string(), trimmed_path);
tmpl.set_reload_command(m_reload_command); tmpl.set_reload_command(m_control_state.reload_command);
tmpl.set_enabled(m_enabled); tmpl.set_enabled(m_control_state.enabled);
auto result = cfg.update_template(trimmed_name, tmpl); auto result = cfg.update_template(trimmed_name, tmpl);
if (!result) if (!result)
{ {
m_validation_error = "Error saving template: " + result.error().description(); m_validation.set("Error saving template: " + result.error().description());
return; return;
} }
m_template_name = trimmed_name; m_control_state.name = trimmed_name;
m_input_path = trimmed_input_path; m_control_state.input_path = trimmed_input_path;
m_output_path = trimmed_path; m_control_state.output_path = trimmed_path;
m_is_editing_existing = true; m_control_state.is_editing_existing = true;
m_saved_content = m_editor.GetText(); m_saved_content = m_editor.GetText();
m_has_unsaved_changes = false; m_has_unsaved_changes = false;
@@ -886,13 +727,13 @@ void template_editor::load_template(const std::string &name)
if (it != templates.end()) if (it != templates.end())
{ {
const auto &tmpl = it->second; const auto &tmpl = it->second;
m_template_name = name; m_control_state.name = name;
m_input_path = tmpl.template_path(); m_control_state.input_path = tmpl.template_path();
m_output_path = tmpl.output_path(); m_control_state.output_path = tmpl.output_path();
m_reload_command = tmpl.reload_command(); m_control_state.reload_command = tmpl.reload_command();
m_enabled = tmpl.enabled(); m_control_state.enabled = tmpl.enabled();
m_is_editing_existing = true; m_control_state.is_editing_existing = true;
m_validation_error = ""; m_validation.clear();
std::ifstream in(tmpl.template_path()); std::ifstream in(tmpl.template_path());
if (in.is_open()) if (in.is_open())
@@ -911,31 +752,31 @@ void template_editor::load_template(const std::string &name)
} }
else else
{ {
m_validation_error = "Error loading template: Failed to open file"; m_validation.set("Error loading template: Failed to open file");
} }
} }
} }
void template_editor::new_template() void template_editor::new_template()
{ {
m_template_name = "new_template"; m_control_state.name = "new_template";
std::string default_content = std::string default_content =
"# Enter your template here\n# Use {color_key} for color variables\n# " "# Enter your template here\n# Use {color_key} for color variables\n# "
"Examples: {color.hex}, {color.rgb}, {color.r}\n\n"; "Examples: {color.hex}, {color.rgb}, {color.r}\n\n";
m_editor.SetText(default_content); m_editor.SetText(default_content);
m_saved_content = default_content; m_saved_content = default_content;
m_input_path = ""; m_control_state.input_path = "";
m_output_path = ""; m_control_state.output_path = "";
m_reload_command = ""; m_control_state.reload_command = "";
m_enabled = true; m_control_state.enabled = true;
m_is_editing_existing = false; m_control_state.is_editing_existing = false;
m_validation_error = ""; m_validation.clear();
m_has_unsaved_changes = false; m_has_unsaved_changes = false;
} }
void template_editor::delete_template() void template_editor::delete_template()
{ {
if (!m_is_editing_existing || m_template_name.empty()) if (!m_control_state.is_editing_existing || m_control_state.name.empty())
return; return;
m_show_delete_confirmation = true; m_show_delete_confirmation = true;

View File

@@ -1,17 +1,24 @@
#ifndef CLRSYNC_GUI_TEMPLATE_EDITOR_HPP #ifndef CLRSYNC_GUI_TEMPLATE_EDITOR_HPP
#define CLRSYNC_GUI_TEMPLATE_EDITOR_HPP #define CLRSYNC_GUI_TEMPLATE_EDITOR_HPP
#include "template_controller.hpp"
#include <core/palette/palette.hpp>
#include "color_text_edit/TextEditor.h" #include "color_text_edit/TextEditor.h"
#include "core/palette/palette.hpp"
#include "gui/controllers/template_controller.hpp"
#include "gui/widgets/template_controls.hpp"
#include "gui/widgets/validation_message.hpp"
#include "imgui.h" #include "imgui.h"
#include <string> #include <string>
#include <vector> #include <vector>
namespace clrsync::gui
{
class ui_manager;
}
class template_editor class template_editor
{ {
public: public:
template_editor(); template_editor(clrsync::gui::ui_manager* ui_mgr);
void render(); void render();
void apply_current_palette(const clrsync::core::palette &pal); void apply_current_palette(const clrsync::core::palette &pal);
@@ -27,22 +34,20 @@ private:
void new_template(); void new_template();
void delete_template(); void delete_template();
void refresh_templates(); void refresh_templates();
void setup_callbacks();
bool is_valid_path(const std::string &path); bool is_valid_path(const std::string &path);
template_controller m_template_controller; template_controller m_template_controller;
TextEditor m_editor; TextEditor m_editor;
std::string m_template_name; clrsync::gui::widgets::template_control_state m_control_state;
std::string m_input_path; clrsync::gui::widgets::template_controls m_controls;
std::string m_output_path; clrsync::gui::widgets::template_control_callbacks m_callbacks;
std::string m_reload_command; clrsync::gui::widgets::validation_message m_validation;
std::string m_validation_error;
std::string m_saved_content;
bool m_has_unsaved_changes = false;
bool m_enabled{true}; std::string m_saved_content;
bool m_is_editing_existing{false}; bool m_has_unsaved_changes{false};
bool m_show_delete_confirmation{false}; bool m_show_delete_confirmation{false};
bool m_show_autocomplete{false}; bool m_show_autocomplete{false};
@@ -62,6 +67,7 @@ private:
ImVec4 m_autocomplete_dim_text_color; ImVec4 m_autocomplete_dim_text_color;
clrsync::core::palette m_current_palette; clrsync::core::palette m_current_palette;
clrsync::gui::ui_manager* m_ui_manager;
}; };
#endif // CLRSYNC_GUI_TEMPLATE_EDITOR_HPP #endif // CLRSYNC_GUI_TEMPLATE_EDITOR_HPP

View File

@@ -0,0 +1,93 @@
#include "action_buttons.hpp"
#include "colors.hpp"
#include "imgui.h"
namespace clrsync::gui::widgets
{
action_buttons::action_buttons() = default;
void action_buttons::add_button(const action_button &button)
{
m_buttons.push_back(button);
}
void action_buttons::clear()
{
m_buttons.clear();
}
void action_buttons::render(const core::palette &theme_palette)
{
if (m_buttons.empty())
return;
if (m_use_separator)
{
ImGui::Separator();
ImGui::Spacing();
}
ImGui::PushStyleVar(ImGuiStyleVar_ItemSpacing, ImVec2(m_spacing, 8));
bool first = true;
for (const auto &button : m_buttons)
{
if (!first)
ImGui::SameLine();
first = false;
int style_colors_pushed = 0;
if (button.use_error_style)
{
auto error = palette_color(theme_palette, "error");
auto error_hover = ImVec4(error.x * 1.1f, error.y * 1.1f, error.z * 1.1f, error.w);
auto error_active = ImVec4(error.x * 0.8f, error.y * 0.8f, error.z * 0.8f, error.w);
auto on_error = palette_color(theme_palette, "on_error");
ImGui::PushStyleColor(ImGuiCol_Button, error);
ImGui::PushStyleColor(ImGuiCol_ButtonHovered, error_hover);
ImGui::PushStyleColor(ImGuiCol_ButtonActive, error_active);
ImGui::PushStyleColor(ImGuiCol_Text, on_error);
style_colors_pushed = 4;
}
else if (button.use_success_style)
{
auto success = palette_color(theme_palette, "success", "accent");
auto success_hover = ImVec4(success.x * 1.1f, success.y * 1.1f, success.z * 1.1f, success.w);
auto success_active = ImVec4(success.x * 0.8f, success.y * 0.8f, success.z * 0.8f, success.w);
auto on_success = palette_color(theme_palette, "on_success", "on_surface");
ImGui::PushStyleColor(ImGuiCol_Button, success);
ImGui::PushStyleColor(ImGuiCol_ButtonHovered, success_hover);
ImGui::PushStyleColor(ImGuiCol_ButtonActive, success_active);
ImGui::PushStyleColor(ImGuiCol_Text, on_success);
style_colors_pushed = 4;
}
bool disabled = !button.enabled;
if (disabled)
ImGui::BeginDisabled();
if (ImGui::Button(button.label.c_str()))
{
if (button.on_click)
{
button.on_click();
}
}
if (disabled)
ImGui::EndDisabled();
if (style_colors_pushed > 0)
ImGui::PopStyleColor(style_colors_pushed);
if (!button.tooltip.empty() && ImGui::IsItemHovered())
{
ImGui::SetTooltip("%s", button.tooltip.c_str());
}
}
ImGui::PopStyleVar();
}
} // namespace clrsync::gui::widgets

View File

@@ -0,0 +1,46 @@
#ifndef CLRSYNC_GUI_WIDGETS_ACTION_BUTTONS_HPP
#define CLRSYNC_GUI_WIDGETS_ACTION_BUTTONS_HPP
#include "core/palette/palette.hpp"
#include <functional>
#include <string>
#include <vector>
namespace clrsync::gui::widgets
{
struct action_button
{
std::string label;
std::string tooltip;
std::function<void()> on_click;
bool enabled = true;
bool use_error_style = false;
bool use_success_style = false;
};
class action_buttons
{
public:
action_buttons();
void add_button(const action_button &button);
void clear();
void render(const core::palette &theme_palette);
void set_spacing(float spacing) { m_spacing = spacing; }
void set_use_separator(bool use) { m_use_separator = use; }
private:
std::vector<action_button> m_buttons;
float m_spacing = 8.0f;
bool m_use_separator = false;
};
} // namespace clrsync::gui::widgets
#endif // CLRSYNC_GUI_WIDGETS_ACTION_BUTTONS_HPP

View File

@@ -0,0 +1,287 @@
#include "autocomplete.hpp"
#include "colors.hpp"
#include "imgui.h"
#include <algorithm>
namespace clrsync::gui::widgets
{
autocomplete_widget::autocomplete_widget()
: m_selected_index(0), m_show_autocomplete(false), m_dismissed(false), m_dismiss_brace_pos(-1)
{
m_bg_color = ImVec4(0.12f, 0.12f, 0.15f, 0.98f);
m_border_color = ImVec4(0.4f, 0.4f, 0.45f, 1.0f);
m_selected_color = ImVec4(0.25f, 0.45f, 0.75f, 0.9f);
m_text_color = ImVec4(0.85f, 0.85f, 0.9f, 1.0f);
m_selected_text_color = ImVec4(1.0f, 1.0f, 1.0f, 1.0f);
m_dim_text_color = ImVec4(0.6f, 0.6f, 0.7f, 1.0f);
}
void autocomplete_widget::update_suggestions(const TextEditor& editor,
const std::vector<std::string>& available_keys,
const std::vector<std::string>& available_formats)
{
m_suggestions.clear();
auto cursor = editor.GetCursorPosition();
std::string line = editor.GetCurrentLineText();
int col = cursor.mColumn;
// Check if inside '{'
int brace_pos = -1;
for (int i = col - 1; i >= 0; --i)
{
if (i < (int)line.length())
{
if (line[i] == '{')
{
brace_pos = i;
break;
}
else if (line[i] == '}' || line[i] == ' ' || line[i] == '\t')
{
break;
}
}
}
if (brace_pos < 0)
{
m_show_autocomplete = false;
m_dismissed = false;
return;
}
if (m_dismissed)
{
bool should_reset_dismissal = false;
if (cursor.mLine != m_dismiss_position.mLine || brace_pos != m_dismiss_brace_pos ||
abs(cursor.mColumn - m_dismiss_position.mColumn) > 3)
{
should_reset_dismissal = true;
}
if (should_reset_dismissal)
{
m_dismissed = false;
}
else
{
m_show_autocomplete = false;
return;
}
}
m_prefix = line.substr(brace_pos + 1, col - brace_pos - 1);
m_start_pos = TextEditor::Coordinates(cursor.mLine, brace_pos + 1);
size_t dot_pos = m_prefix.find('.');
if (dot_pos != std::string::npos)
{
std::string color_key = m_prefix.substr(0, dot_pos);
std::string format_prefix = m_prefix.substr(dot_pos + 1);
bool valid_key = std::find(available_keys.begin(), available_keys.end(), color_key) != available_keys.end();
if (valid_key)
{
for (const auto &fmt : available_formats)
{
if (format_prefix.empty() || fmt.find(format_prefix) == 0 ||
fmt.find(format_prefix) != std::string::npos)
{
m_suggestions.push_back(color_key + "." + fmt);
}
}
}
}
else
{
for (const auto& key : available_keys)
{
if (m_prefix.empty() || key.find(m_prefix) == 0 ||
key.find(m_prefix) != std::string::npos)
{
m_suggestions.push_back(key);
}
}
}
std::sort(m_suggestions.begin(), m_suggestions.end(),
[this](const std::string &a, const std::string &b) {
bool a_prefix = a.find(m_prefix) == 0;
bool b_prefix = b.find(m_prefix) == 0;
if (a_prefix != b_prefix)
return a_prefix;
return a < b;
});
m_show_autocomplete = !m_suggestions.empty();
if (m_show_autocomplete && m_selected_index >= (int)m_suggestions.size())
{
m_selected_index = 0;
}
}
void autocomplete_widget::render(const ImVec2& editor_pos, TextEditor& editor)
{
if (!m_show_autocomplete || m_suggestions.empty())
return;
float line_height = ImGui::GetTextLineHeightWithSpacing();
float char_width = ImGui::GetFontSize() * 0.5f;
auto cursor = editor.GetCursorPosition();
const float line_number_width = 50.0f;
ImVec2 popup_pos;
popup_pos.x = editor_pos.x + line_number_width + (m_start_pos.mColumn * char_width);
popup_pos.y = editor_pos.y + ((cursor.mLine + 1) * line_height);
ImGui::SetNextWindowPos(popup_pos, ImGuiCond_Always);
ImGui::SetNextWindowSize(ImVec2(300, 0));
ImGuiWindowFlags flags = ImGuiWindowFlags_NoTitleBar | ImGuiWindowFlags_NoResize |
ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoSavedSettings |
ImGuiWindowFlags_NoFocusOnAppearing |
ImGuiWindowFlags_AlwaysAutoResize;
ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2(6, 6));
ImGui::PushStyleVar(ImGuiStyleVar_WindowRounding, 6.0f);
ImGui::PushStyleVar(ImGuiStyleVar_ItemSpacing, ImVec2(4, 2));
ImGui::PushStyleColor(ImGuiCol_WindowBg, m_bg_color);
ImGui::PushStyleColor(ImGuiCol_Border, m_border_color);
if (ImGui::Begin("##autocomplete", nullptr, flags))
{
ImGui::PushStyleColor(ImGuiCol_Text, m_dim_text_color);
if (m_prefix.find('.') != std::string::npos)
ImGui::Text("Formats");
else
ImGui::Text("Color Keys");
ImGui::PopStyleColor();
ImGui::Separator();
int max_items = std::min((int)m_suggestions.size(), 8);
for (int i = 0; i < max_items; ++i)
{
const auto &suggestion = m_suggestions[i];
bool is_selected = (i == m_selected_index);
if (is_selected)
{
ImVec4 selected_hover = m_selected_color;
selected_hover.w = std::min(selected_hover.w + 0.1f, 1.0f);
ImGui::PushStyleColor(ImGuiCol_Header, m_selected_color);
ImGui::PushStyleColor(ImGuiCol_HeaderHovered, selected_hover);
ImGui::PushStyleColor(ImGuiCol_Text, m_selected_text_color);
}
else
{
ImVec4 normal_bg = m_bg_color;
normal_bg.w = 0.5f;
ImVec4 hover_bg = m_selected_color;
hover_bg.w = 0.3f;
ImGui::PushStyleColor(ImGuiCol_Header, normal_bg);
ImGui::PushStyleColor(ImGuiCol_HeaderHovered, hover_bg);
ImGui::PushStyleColor(ImGuiCol_Text, m_text_color);
}
std::string display_text = " " + suggestion;
if (ImGui::Selectable(display_text.c_str(), is_selected, ImGuiSelectableFlags_None, ImVec2(0, 0)))
{
auto start = m_start_pos;
auto end = editor.GetCursorPosition();
editor.SetSelection(start, end);
editor.Delete();
editor.InsertText(suggestion + "}");
m_show_autocomplete = false;
m_dismissed = false;
}
ImGui::PopStyleColor(3);
if (is_selected && ImGui::IsWindowAppearing())
{
ImGui::SetScrollHereY();
}
}
if (m_suggestions.size() > 8)
{
ImGui::Separator();
ImGui::PushStyleColor(ImGuiCol_Text, m_dim_text_color);
ImGui::Text(" +%d more", (int)m_suggestions.size() - 8);
ImGui::PopStyleColor();
}
ImGui::Separator();
ImGui::PushStyleColor(ImGuiCol_Text, m_dim_text_color);
ImGui::Text(" Tab/Enter: accept | Esc: dismiss");
ImGui::PopStyleColor();
}
ImGui::End();
ImGui::PopStyleColor(2);
ImGui::PopStyleVar(3);
}
void autocomplete_widget::apply_palette(const core::palette& palette)
{
m_bg_color = palette_color(palette, "surface", "background");
m_bg_color.w = 0.98f;
m_border_color = palette_color(palette, "border", "surface_variant");
m_selected_color = palette_color(palette, "accent", "surface_variant");
m_text_color = palette_color(palette, "on_surface", "foreground");
m_selected_text_color = palette_color(palette, "on_surface", "foreground");
m_dim_text_color = palette_color(palette, "on_surface_variant", "editor_inactive");
}
bool autocomplete_widget::handle_input(TextEditor& editor)
{
if (!m_show_autocomplete || m_suggestions.empty())
return false;
bool handled = false;
if (ImGui::IsKeyPressed(ImGuiKey_DownArrow))
{
m_selected_index = (m_selected_index + 1) % (int)m_suggestions.size();
handled = true;
}
else if (ImGui::IsKeyPressed(ImGuiKey_UpArrow))
{
m_selected_index = (m_selected_index - 1 + (int)m_suggestions.size()) % (int)m_suggestions.size();
handled = true;
}
else if (ImGui::IsKeyPressed(ImGuiKey_Tab) || ImGui::IsKeyPressed(ImGuiKey_Enter))
{
if (m_selected_index >= 0 && m_selected_index < (int)m_suggestions.size())
{
auto start = m_start_pos;
auto end = editor.GetCursorPosition();
editor.SetSelection(start, end);
editor.Delete();
editor.InsertText(m_suggestions[m_selected_index] + "}");
m_show_autocomplete = false;
m_dismissed = false;
}
handled = true;
}
else if (ImGui::IsKeyPressed(ImGuiKey_Escape))
{
m_dismissed = true;
m_dismiss_position = editor.GetCursorPosition();
m_dismiss_brace_pos = -1;
m_show_autocomplete = false;
handled = true;
}
return handled;
}
} // namespace clrsync::gui::widgets

View File

@@ -0,0 +1,53 @@
#ifndef CLRSYNC_GUI_WIDGETS_AUTOCOMPLETE_HPP
#define CLRSYNC_GUI_WIDGETS_AUTOCOMPLETE_HPP
#include "core/palette/palette.hpp"
#include "color_text_edit/TextEditor.h"
#include "imgui.h"
#include <string>
#include <vector>
namespace clrsync::gui::widgets
{
class autocomplete_widget
{
public:
autocomplete_widget();
void update_suggestions(const TextEditor& editor,
const std::vector<std::string>& available_keys,
const std::vector<std::string>& available_formats);
void render(const ImVec2& editor_pos, TextEditor& editor);
void apply_palette(const core::palette& palette);
bool handle_input(TextEditor& editor);
bool is_visible() const { return m_show_autocomplete; }
void dismiss() { m_dismissed = true; m_show_autocomplete = false; }
private:
std::vector<std::string> m_suggestions;
std::string m_prefix;
TextEditor::Coordinates m_start_pos;
int m_selected_index;
bool m_show_autocomplete;
bool m_dismissed;
TextEditor::Coordinates m_dismiss_position;
int m_dismiss_brace_pos;
ImVec4 m_bg_color;
ImVec4 m_border_color;
ImVec4 m_selected_color;
ImVec4 m_text_color;
ImVec4 m_selected_text_color;
ImVec4 m_dim_text_color;
};
} // namespace clrsync::gui::widgets
#endif // CLRSYNC_GUI_WIDGETS_AUTOCOMPLETE_HPP

View File

@@ -0,0 +1,23 @@
#include "centered_text.hpp"
namespace clrsync::gui::widgets
{
void centered_text(const std::string& text, const ImVec4& color)
{
float window_width = ImGui::GetContentRegionAvail().x;
float text_width = ImGui::CalcTextSize(text.c_str()).x;
ImGui::SetCursorPosX((window_width - text_width) * 0.5f);
ImGui::TextColored(color, "%s", text.c_str());
}
void centered_buttons(float total_width, const std::function<void()>& render_buttons)
{
float window_width = ImGui::GetContentRegionAvail().x;
float start_pos = (window_width - total_width) * 0.5f;
if (start_pos > 0)
ImGui::SetCursorPosX(ImGui::GetCursorPosX() + start_pos);
render_buttons();
}
} // namespace clrsync::gui::widgets

View File

@@ -0,0 +1,16 @@
#ifndef CLRSYNC_GUI_WIDGETS_CENTERED_TEXT_HPP
#define CLRSYNC_GUI_WIDGETS_CENTERED_TEXT_HPP
#include "imgui.h"
#include <functional>
#include <string>
namespace clrsync::gui::widgets
{
void centered_text(const std::string& text, const ImVec4& color = ImVec4(1, 1, 1, 1));
void centered_buttons(float total_width, const std::function<void()>& render_buttons);
} // namespace clrsync::gui::widgets
#endif // CLRSYNC_GUI_WIDGETS_CENTERED_TEXT_HPP

View File

@@ -0,0 +1,58 @@
#include "colors.hpp"
namespace clrsync::gui::widgets
{
ImVec4 palette_color(const core::palette &pal, const std::string &key,
const std::string &fallback)
{
auto colors = pal.colors();
if (colors.empty())
return ImVec4(1.0f, 1.0f, 1.0f, 1.0f);
auto it = colors.find(key);
if (it == colors.end() && !fallback.empty())
{
it = colors.find(fallback);
}
if (it != colors.end())
{
const auto &col = it->second;
const uint32_t hex = col.hex();
const float r = ((hex >> 24) & 0xFF) / 255.0f;
const float g = ((hex >> 16) & 0xFF) / 255.0f;
const float b = ((hex >> 8) & 0xFF) / 255.0f;
const float a = (hex & 0xFF) / 255.0f;
return ImVec4(r, g, b, a);
}
return ImVec4(1.0f, 1.0f, 1.0f, 1.0f);
}
uint32_t palette_color_u32(const core::palette &pal, const std::string &key,
const std::string &fallback)
{
auto colors = pal.colors();
if (colors.empty())
return 0xFFFFFFFF;
auto it = colors.find(key);
if (it == colors.end() && !fallback.empty())
{
it = colors.find(fallback);
}
if (it != colors.end())
{
const auto &col = it->second;
const uint32_t hex = col.hex();
const uint32_t r = (hex >> 24) & 0xFF;
const uint32_t g = (hex >> 16) & 0xFF;
const uint32_t b = (hex >> 8) & 0xFF;
const uint32_t a = hex & 0xFF;
return (a << 24) | (b << 16) | (g << 8) | r;
}
return 0xFFFFFFFF;
}
} // namespace clrsync::gui::widgets

View File

@@ -0,0 +1,19 @@
#ifndef CLRSYNC_GUI_WIDGETS_COLORS_HPP
#define CLRSYNC_GUI_WIDGETS_COLORS_HPP
#include "core/palette/palette.hpp"
#include "imgui.h"
#include <string>
namespace clrsync::gui::widgets
{
ImVec4 palette_color(const core::palette &pal, const std::string &key,
const std::string &fallback = "");
uint32_t palette_color_u32(const core::palette &pal, const std::string &key,
const std::string &fallback = "");
} // namespace clrsync::gui::widgets
#endif // CLRSYNC_GUI_WIDGETS_COLORS_HPP

View File

@@ -0,0 +1,46 @@
#include "dialogs.hpp"
#include "colors.hpp"
#include "imgui.h"
namespace clrsync::gui::widgets
{
bool delete_confirmation_dialog(const std::string &popup_title, const std::string &item_name,
const std::string &item_type, const core::palette &theme_palette,
const std::function<void()> &on_delete)
{
bool result = false;
if (ImGui::BeginPopupModal(popup_title.c_str(), nullptr, ImGuiWindowFlags_AlwaysAutoResize))
{
ImVec4 warning_color = palette_color(theme_palette, "warning", "accent");
ImGui::TextColored(warning_color, "Are you sure you want to delete '%s'?",
item_name.c_str());
ImGui::Text("This action cannot be undone.");
ImGui::Spacing();
ImGui::Separator();
ImGui::Spacing();
float button_width = 120.0f;
float total_width = 2.0f * button_width + ImGui::GetStyle().ItemSpacing.x;
float window_width = ImGui::GetContentRegionAvail().x;
ImGui::SetCursorPosX((window_width - total_width) * 0.5f);
if (ImGui::Button("Delete", ImVec2(button_width, 0)))
{
on_delete();
result = true;
ImGui::CloseCurrentPopup();
}
ImGui::SameLine();
if (ImGui::Button("Cancel", ImVec2(button_width, 0)))
{
ImGui::CloseCurrentPopup();
}
ImGui::EndPopup();
}
return result;
}
} // namespace clrsync::gui::widgets

View File

@@ -0,0 +1,17 @@
#ifndef CLRSYNC_GUI_WIDGETS_DIALOGS_HPP
#define CLRSYNC_GUI_WIDGETS_DIALOGS_HPP
#include "core/palette/palette.hpp"
#include <functional>
#include <string>
namespace clrsync::gui::widgets
{
bool delete_confirmation_dialog(const std::string &popup_title, const std::string &item_name,
const std::string &item_type, const core::palette &theme_palette,
const std::function<void()> &on_delete);
} // namespace clrsync::gui::widgets
#endif // CLRSYNC_GUI_WIDGETS_DIALOGS_HPP

View File

@@ -0,0 +1,71 @@
#include "error_message.hpp"
#include "colors.hpp"
#include "imgui.h"
namespace clrsync::gui::widgets
{
void error_message::set(const std::string& message)
{
m_message = message;
}
void error_message::clear()
{
m_message.clear();
}
bool error_message::has_error() const
{
return !m_message.empty();
}
const std::string& error_message::get() const
{
return m_message;
}
void error_message::render(const core::palette& palette)
{
if (m_message.empty())
return;
ImGui::Spacing();
auto error_bg_color = palette_color(palette, "error");
auto error_text_color = palette_color(palette, "on_error");
ImGui::PushStyleColor(ImGuiCol_ChildBg, error_bg_color);
ImGui::PushStyleColor(ImGuiCol_Border, error_bg_color);
ImGui::PushStyleVar(ImGuiStyleVar_ChildRounding, 4.0f);
ImGui::PushStyleVar(ImGuiStyleVar_ChildBorderSize, 1.0f);
if (ImGui::BeginChild("##error_box", ImVec2(0, 0),
ImGuiChildFlags_AutoResizeY | ImGuiChildFlags_Borders))
{
ImGui::PushStyleColor(ImGuiCol_Text, error_text_color);
ImGui::TextWrapped("Error: %s", m_message.c_str());
ImGui::PopStyleColor();
ImGui::Spacing();
ImGui::PushStyleColor(ImGuiCol_Button,
ImVec4(error_bg_color.x * 0.8f, error_bg_color.y * 0.8f,
error_bg_color.z * 0.8f, error_bg_color.w));
ImGui::PushStyleColor(ImGuiCol_ButtonHovered,
ImVec4(error_bg_color.x * 0.6f, error_bg_color.y * 0.6f,
error_bg_color.z * 0.6f, error_bg_color.w));
ImGui::PushStyleColor(ImGuiCol_Text, error_text_color);
if (ImGui::Button("Dismiss##error"))
m_message.clear();
ImGui::PopStyleColor(3);
}
ImGui::EndChild();
ImGui::PopStyleVar(2);
ImGui::PopStyleColor(2);
}
} // namespace clrsync::gui::widgets

View File

@@ -0,0 +1,25 @@
#ifndef CLRSYNC_GUI_WIDGETS_ERROR_MESSAGE_HPP
#define CLRSYNC_GUI_WIDGETS_ERROR_MESSAGE_HPP
#include "core/palette/palette.hpp"
#include <string>
namespace clrsync::gui::widgets
{
class error_message
{
public:
void set(const std::string& message);
void clear();
bool has_error() const;
const std::string& get() const;
void render(const core::palette& palette);
private:
std::string m_message;
};
} // namespace clrsync::gui::widgets
#endif // CLRSYNC_GUI_WIDGETS_ERROR_MESSAGE_HPP

View File

@@ -0,0 +1,246 @@
#include "form_field.hpp"
#include "imgui.h"
#include <cstring>
namespace clrsync::gui::widgets
{
form_field::form_field() = default;
void form_field::render_label(const form_field_config& config)
{
ImGui::AlignTextToFramePadding();
ImGui::Text("%s:", config.label.c_str());
if (config.label_width > 0)
{
ImGui::SameLine(config.label_width);
}
else
{
ImGui::SameLine();
}
}
void form_field::render_tooltip(const form_field_config& config)
{
if (!config.tooltip.empty() && ImGui::IsItemHovered())
{
ImGui::SetTooltip("%s", config.tooltip.c_str());
}
}
bool form_field::render_text(const form_field_config& config, std::string& value)
{
render_label(config);
if (config.field_width > 0)
{
ImGui::SetNextItemWidth(config.field_width);
}
else if (config.field_width < 0)
{
ImGui::SetNextItemWidth(config.field_width);
}
char buffer[512] = {0};
strncpy(buffer, value.c_str(), sizeof(buffer) - 1);
bool changed = false;
std::string id = "##" + config.label;
if (config.type == field_type::text_with_hint)
{
changed = ImGui::InputTextWithHint(id.c_str(), config.hint.c_str(), buffer, sizeof(buffer));
}
else
{
changed = ImGui::InputText(id.c_str(), buffer, sizeof(buffer));
}
if (changed)
{
value = buffer;
}
render_tooltip(config);
return changed;
}
bool form_field::render_number(const form_field_config& config, int& value)
{
render_label(config);
if (config.field_width > 0)
{
ImGui::SetNextItemWidth(config.field_width);
}
else if (config.field_width < 0)
{
ImGui::SetNextItemWidth(config.field_width);
}
std::string id = "##" + config.label;
bool changed = ImGui::InputInt(id.c_str(), &value);
if (value < (int)config.min_value) value = (int)config.min_value;
if (value > (int)config.max_value) value = (int)config.max_value;
render_tooltip(config);
return changed;
}
bool form_field::render_number(const form_field_config& config, float& value)
{
render_label(config);
if (config.field_width > 0)
{
ImGui::SetNextItemWidth(config.field_width);
}
else if (config.field_width < 0)
{
ImGui::SetNextItemWidth(config.field_width);
}
std::string id = "##" + config.label;
bool changed = ImGui::InputFloat(id.c_str(), &value);
if (value < config.min_value) value = config.min_value;
if (value > config.max_value) value = config.max_value;
render_tooltip(config);
return changed;
}
bool form_field::render_path(const form_field_config& config, std::string& value)
{
render_label(config);
float browse_button_width = 80.0f;
float available_width = ImGui::GetContentRegionAvail().x;
float input_width = available_width - browse_button_width - ImGui::GetStyle().ItemSpacing.x;
if (config.field_width > 0)
{
input_width = config.field_width - browse_button_width - ImGui::GetStyle().ItemSpacing.x;
}
ImGui::SetNextItemWidth(input_width);
char buffer[512] = {0};
strncpy(buffer, value.c_str(), sizeof(buffer) - 1);
std::string input_id = "##" + config.label + "_input";
bool changed = false;
if (config.type == field_type::text_with_hint)
{
changed = ImGui::InputTextWithHint(input_id.c_str(), config.hint.c_str(), buffer, sizeof(buffer));
}
else
{
changed = ImGui::InputText(input_id.c_str(), buffer, sizeof(buffer));
}
if (changed)
{
value = buffer;
}
ImGui::SameLine();
std::string button_id = "Browse##" + config.label;
if (ImGui::Button(button_id.c_str()))
{
if (m_path_browse_callback)
{
std::string selected_path = m_path_browse_callback(value);
if (!selected_path.empty())
{
value = selected_path;
changed = true;
}
}
}
render_tooltip(config);
return changed;
}
void form_field::render_readonly(const form_field_config& config, const std::string& value)
{
render_label(config);
ImGui::TextColored(ImVec4(0.7f, 0.7f, 0.7f, 1.0f), "%s", value.c_str());
render_tooltip(config);
}
void form_field::set_path_browse_callback(const std::function<std::string(const std::string&)>& callback)
{
m_path_browse_callback = callback;
}
bool form_field::render_slider(const form_field_config& config, int& value)
{
render_label(config);
if (config.field_width > 0)
ImGui::SetNextItemWidth(config.field_width);
else if (config.field_width < 0)
ImGui::SetNextItemWidth(config.field_width);
std::string id = "##" + config.label;
std::string format = config.format.empty() ? "%d" : config.format.c_str();
int old_value = value;
bool changed = ImGui::SliderInt(id.c_str(), &value, (int)config.min_value, (int)config.max_value, format.c_str());
if (config.show_reset)
{
ImGui::SameLine();
std::string reset_id = "Reset##" + config.label;
if (ImGui::Button(reset_id.c_str()))
{
value = config.default_value;
changed = (old_value != value);
}
}
render_tooltip(config);
return changed;
}
bool form_field::render_combo(const form_field_config& config, const std::vector<std::string>& items, int& selected_idx, std::string& value)
{
render_label(config);
if (config.field_width > 0)
ImGui::SetNextItemWidth(config.field_width);
else if (config.field_width < 0)
ImGui::SetNextItemWidth(config.field_width);
std::string id = "##" + config.label;
bool changed = false;
if (ImGui::BeginCombo(id.c_str(), value.c_str()))
{
for (int i = 0; i < static_cast<int>(items.size()); i++)
{
bool is_selected = (i == selected_idx);
if (ImGui::Selectable(items[i].c_str(), is_selected))
{
selected_idx = i;
value = items[i];
changed = true;
}
if (is_selected)
ImGui::SetItemDefaultFocus();
}
ImGui::EndCombo();
}
render_tooltip(config);
return changed;
}
} // namespace clrsync::gui::widgets

View File

@@ -0,0 +1,74 @@
#ifndef CLRSYNC_GUI_WIDGETS_FORM_FIELD_HPP
#define CLRSYNC_GUI_WIDGETS_FORM_FIELD_HPP
#include <functional>
#include <string>
#include <vector>
namespace clrsync::gui::widgets
{
enum class field_type
{
text,
text_with_hint,
number,
slider,
path,
combo,
readonly_text
};
struct form_field_config
{
std::string label;
std::string tooltip;
float label_width = 80.0f;
float field_width = -1.0f;
bool required = false;
field_type type = field_type::text;
std::string hint;
float min_value = 0.0f;
float max_value = 100.0f;
std::string format;
bool show_reset = false;
int default_value = 0;
};
class form_field
{
public:
form_field();
// Render a text input field
bool render_text(const form_field_config& config, std::string& value);
// Render a number input field
bool render_number(const form_field_config& config, int& value);
bool render_number(const form_field_config& config, float& value);
// Render a slider field
bool render_slider(const form_field_config& config, int& value);
// Render a combo box field
bool render_combo(const form_field_config& config, const std::vector<std::string>& items, int& selected_idx, std::string& value);
// Render a path input field with browse button
bool render_path(const form_field_config& config, std::string& value);
// Render readonly text
void render_readonly(const form_field_config& config, const std::string& value);
// Set callback for path browsing
void set_path_browse_callback(const std::function<std::string(const std::string&)>& callback);
private:
std::function<std::string(const std::string&)> m_path_browse_callback;
void render_label(const form_field_config& config);
void render_tooltip(const form_field_config& config);
};
} // namespace clrsync::gui::widgets
#endif // CLRSYNC_GUI_WIDGETS_FORM_FIELD_HPP

Some files were not shown because too many files have changed in this diff Show More