22 Commits

Author SHA1 Message Date
d852d58948 tested and fixed new templates 2026-01-12 20:31:02 +03:00
41939f4df4 fixed hyprland and kvantum configs 2026-01-12 19:18:25 +03:00
a813b7f6c9 bump version 2026-01-12 14:45:10 +03:00
9803f4948b fix: run reload command detached 2026-01-12 14:39:00 +03:00
c17960d1e6 untested: added gtk, hyprland, qtct and kvantum templates 2026-01-12 14:30:40 +03:00
19291f35ee added telegram template 2026-01-12 14:00:47 +03:00
4d61ed3194 added firefox template 2026-01-12 13:13:08 +03:00
d722499e80 fixed borders in vscode template 2026-01-12 11:04:23 +03:00
e3cd9cd362 added vscode template 2026-01-12 10:50:51 +03:00
e256dcad2e added some palettes and templates 2026-01-03 03:55:36 +03:00
5b0599a958 changed default kitty config location 2025-12-21 23:39:31 +03:00
afa7275e37 Update README.md 2025-12-20 00:23:34 +03:00
fc5663839e Update README.md 2025-12-20 00:11:15 +03:00
997e7c3eae add pkgbuild 2025-12-19 23:36:21 +03:00
4229db457c merged dev 2025-12-19 23:32:17 +03:00
d17776b8e4 fix: minor ui fixes 2025-12-19 20:04:30 +03:00
8112096647 chore :refactored remaining views 2025-12-19 17:37:42 +03:00
4ada2c44ed chore: refactor 2025-12-19 17:22:23 +03:00
82998d688c build: write .clangd with proper compile_commands path on configure 2025-12-19 10:33:14 +03:00
6ac9c03ec4 fixed linux deps 2025-12-19 10:11:13 +03:00
2a433483d7 Merge branch 'master' into dev 2025-12-19 09:56:18 +03:00
ece7c84371 added cmake presets 2025-12-19 09:55:29 +03:00
81 changed files with 12877 additions and 1080 deletions

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.7 pkgver=1.0.2
pkgrel=1 pkgrel=1
pkgdesc="Color scheme manager" pkgdesc="Color scheme manager"
arch=('x86_64') arch=('x86_64')

View File

@@ -1,6 +1,6 @@
# Maintainer: Daniel Dada <dan@binarygoose.dev> # Maintainer: Daniel Dada <dan@binarygoose.dev>
pkgname=clrsync-git pkgname=clrsync-git
pkgver=r22.d8baae2 pkgver=r107.4229db4
pkgrel=1 pkgrel=1
pkgdesc="Color scheme manager (git version)" pkgdesc="Color scheme manager (git version)"
arch=('x86_64') arch=('x86_64')

View File

@@ -1,5 +1,5 @@
cmake_minimum_required(VERSION 3.25) cmake_minimum_required(VERSION 3.25)
project(clrsync VERSION 0.1.7 LANGUAGES CXX) project(clrsync VERSION 1.0.2 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}")

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

@@ -3,8 +3,6 @@
# clrsync # clrsync
**Notice:** This application is not yet released and is subject to change.
A theme management tool for synchronizing color schemes across multiple applications. clrsync allows to define color palettes once and apply them consistently to all configurable applications. A theme management tool for synchronizing color schemes across multiple applications. clrsync allows to define color palettes once and apply them consistently to all configurable applications.
![Preview](assets/screenshot.png) ![Preview](assets/screenshot.png)
@@ -34,6 +32,7 @@ A theme management tool for synchronizing color schemes across multiple applicat
- [Usage](#usage) - [Usage](#usage)
- [CLI](#cli) - [CLI](#cli)
- [GUI](#gui) - [GUI](#gui)
- [Extras](#extras)
- [Acknowledgments](#acknowledgments) - [Acknowledgments](#acknowledgments)
## Features ## Features
@@ -47,6 +46,14 @@ A theme management tool for synchronizing color schemes across multiple applicat
### Linux ### Linux
#### Arch Linux
Install the package from AUR using any helper or install manually
```shell
yay -S clrsync-git
```
#### Ubuntu #### Ubuntu
1. Download the latest .deb from the [releases page](https://github.com/obsqrbtz/clrsync/releases) 1. Download the latest .deb from the [releases page](https://github.com/obsqrbtz/clrsync/releases)
@@ -457,6 +464,10 @@ The GUI provides:
- **Template Editor**: Edit template files - **Template Editor**: Edit template files
- **Live Preview**: See changes in real-time - **Live Preview**: See changes in real-time
## Extras
You may find some pre-configured color schemes and templates in [extra](extra) directory of this repository.
## Acknowledgments ## Acknowledgments
- **[matugen](https://github.com/InioX/matugen)** - A material you color generation tool - **[matugen](https://github.com/InioX/matugen)** - A material you color generation tool
@@ -465,4 +476,3 @@ The GUI provides:
- **[toml++](https://github.com/marzer/tomlplusplus)** - Header-only TOML config file parser and serializer for C++17 - **[toml++](https://github.com/marzer/tomlplusplus)** - Header-only TOML config file parser and serializer for C++17
- **[argparse](https://github.com/p-ranav/argparse)** - Argument Parser for Modern C++ - **[argparse](https://github.com/p-ranav/argparse)** - Argument Parser for Modern C++
- **[ImGuiColorTextEdit](https://github.com/BalazsJako/ImGuiColorTextEdit)** - Syntax highlighting text editor for ImGui - **[ImGuiColorTextEdit](https://github.com/BalazsJako/ImGuiColorTextEdit)** - Syntax highlighting text editor for ImGui
- **cursed** by **[pyratebeard](https://pyratebeard.net)** - Color scheme

View File

@@ -1 +1 @@
0.1.7 1.0.2

View File

@@ -8,12 +8,6 @@ if(WIN32)
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
) )
set(FT_DISABLE_ZLIB FALSE CACHE BOOL "" FORCE)
set(FT_DISABLE_BZIP2 TRUE CACHE BOOL "" FORCE)
set(FT_DISABLE_PNG TRUE CACHE BOOL "" FORCE)
set(FT_DISABLE_HARFBUZZ FALSE CACHE BOOL "" FORCE)
set(FT_DISABLE_BROTLI TRUE CACHE BOOL "" FORCE)
FetchContent_MakeAvailable(freetype) FetchContent_MakeAvailable(freetype)
elseif(APPLE) elseif(APPLE)
option(USE_SYSTEM_GLFW ON) option(USE_SYSTEM_GLFW ON)
@@ -33,6 +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(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)
@@ -76,6 +72,10 @@ else()
endif() endif()
set(FREETYPE_EXTRA_LIBS "") set(FREETYPE_EXTRA_LIBS "")
if(BROTLIDEC_LIBRARY AND BROTLICOMMON_LIBRARY)
list(APPEND FREETYPE_EXTRA_LIBS ${BROTLIDEC_LIBRARY} ${BROTLICOMMON_LIBRARY})
message(STATUS "Found Brotli libraries")
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

@@ -12,7 +12,7 @@ output_path = '~/.config/clrsync/formats-demo'
[templates.kitty] [templates.kitty]
enabled = true enabled = true
input_path = '~/.config/clrsync/templates/kitty.conf' input_path = '~/.config/clrsync/templates/kitty.conf'
output_path = '~/.config/kitty/kitty_test.conf' output_path = '~/.config/kitty/clrsync.conf'
reload_cmd = 'pkill -SIGUSR1 kitty' reload_cmd = 'pkill -SIGUSR1 kitty'
[templates.nvim] [templates.nvim]

308
extra/README.md Normal file
View File

@@ -0,0 +1,308 @@
# Extras
This directory contains additional palettes and pre-configured templates for various applications.
## Navigation
- [Palettes](#palettes)
- [Cursed](#cursed)
- [Cursed Light](#cursed-light)
- [Nord](#nord)
- [Pre-configured Templates](#pre-configured-templates)
- [Terminal Emulators](#terminal-emulators)
- [Kitty](#kitty)
- [Alacritty](#alacritty)
- [Ghostty](#ghostty)
- [Text Editors](#text-editors)
- [Neovim](#neovim)
- [Visual Studio Code](#visual-studio-code)
- [Browsers](#browsers)
- [Firefox](#firefox)
- [Applications](#applications)
- [Telegram](#telegram)
- [Desktop Themes](#desktop-themes)
- [GTK](#gtk)
- [Qt Applications](#qt-applications)
- [Window Managers](#window-managers)
- [Hyprland](#hyprland)
## Palettes
### Cursed
A dark color scheme inspired by the `cursed` theme by [pyratebeard](https://pyratebeard.net).
<details>
<summary>Preview</summary>
![cursed](img/cursed.png)
</details>
### Cursed Light
A light variant of the `cursed` color scheme.
<details>
<summary>Preview</summary>
![cursed-light](img/cursed-light.png)
</details>
### Nord
A color scheme based on the `Nord` palette.
<details>
<summary>Preview</summary>
![nord](img/nord.png)
</details>
## Pre-configured Templates
### Terminal Emulators
#### Kitty
1. Download the [template file](templates/kitty.conf)
2. Configure the template in `~/.config/clrsync/config.toml`:
```toml
[templates.kitty]
enabled = true
input_path = '~/.config/clrsync/templates/kitty.conf'
output_path = '~/.config/kitty/clrsync.conf'
reload_cmd = 'pkill -SIGUSR1 kitty'
```
3. Import the generated color scheme in `~/.config/kitty/kitty.conf`:
```conf
include clrsync.conf
```
#### Alacritty
1. Download the [template file](templates/alacritty.toml)
2. Configure the template in `~/.config/clrsync/config.toml`:
```toml
[templates.alacritty]
enabled = true
input_path = '~/.config/clrsync/templates/alacritty.toml'
output_path = '~/.config/alacritty/clrsync.toml'
reload_cmd = ''
```
3. Import the generated color scheme in `~/.config/alacritty/alacritty.toml`:
```toml
[general]
import = ["clrsync.toml"]
```
#### Ghostty
1. Download the [template file](templates/ghostty)
2. Configure the template in `~/.config/clrsync/config.toml`:
```toml
[templates.ghostty]
enabled = true
input_path = '~/.config/clrsync/templates/ghostty'
output_path = '~/.config/ghostty/themes/clrsync'
reload_cmd = 'pkill -SIGUSR2 ghostty'
```
3. Set the generated color scheme in `~/.config/ghostty/config`:
```conf
theme = "clrsync"
```
### Text Editors
#### Neovim
1. Download the [template file](templates/nvim.lua)
2. Configure the template in `~/.config/clrsync/config.toml`:
```toml
[templates.nvim]
enabled = true
input_path = '~/.config/clrsync/templates/nvim.lua'
output_path = '~/.config/nvim/colors/clrsync.lua'
reload_cmd = ''
```
3. Set the colorscheme in your Neovim config:
```lua
vim.cmd.colorscheme 'clrsync'
```
#### Visual Studio Code
1. Install the [clrsync VS Code theme](https://marketplace.visualstudio.com/items?itemName=obsqrbtz.clrsync)
2. Download the [template file](templates/code.json)
3. Configure the template in `~/.config/clrsync/config.toml`:
```toml
[templates.vscode]
enabled = true
input_path = '~/.config/clrsync/templates/code.json'
output_path = '~/.vscode/extensions/obsqrbtz.clrsync-1.0.2/themes/clrsync-color-theme.json'
reload_cmd = ''
```
4. Set the `clrsync` color scheme in VS Code
### Browsers
#### Firefox
1. Go to `about:config` and set `toolkit.legacyUserProfileCustomizations.stylesheets` to `true`
2. Go to `about:support` and find your profile directory
3. Create a `chrome` directory in your profile
4. Download the [template file](templates/userChrome.css)
5. Configure clrsync to output to `<profile>/chrome/userChrome.css`:
```toml
[templates.firefox]
enabled = true
input_path = '~/.config/clrsync/templates/userChrome.css'
output_path = '<profile directory>/chrome/userChrome.css'
reload_cmd = ''
```
### Applications
#### Telegram
1. Download the [template file](templates/telegram.tdesktop-theme)
2. Configure the template in `~/.config/clrsync/config.toml`:
```toml
[templates.telegram]
enabled = true
input_path = '~/.config/clrsync/templates/telegram.tdesktop-theme'
output_path = '~/clrsync.tdesktop-theme'
reload_cmd = ''
```
3. Apply the palette with clrsync and send the generated `clrsync.tdesktop-theme` file to yourself in Telegram
4. Click on the theme file in the Telegram dialog to apply it
### Desktop Themes
#### GTK
1. Download the [template file](templates/gtk.css)
2. Configure the template in `~/.config/clrsync/config.toml`:
```toml
[templates.gtk3]
enabled = true
input_path = '~/.config/clrsync/templates/gtk.css'
output_path = '~/.config/gtk-3.0/colors.css'
reload_cmd = ''
[templates.gtk4]
enabled = true
input_path = '~/.config/clrsync/templates/gtk.css'
output_path = '~/.config/gtk-4.0/colors.css'
reload_cmd = ''
```
3. Import the color scheme at the top of `~/.config/gtk-3.0/gtk.css`, `~/.config/gtk-4.0/gtk.css`, and `~/.config/gtk-4.0/gtk-dark.css`:
```css
@import 'colors.css';
```
#### Qt Applications
1. Download the templates:
- Kvantum: [kvantum.kvconfig](templates/kvantum/kvantum.kvconfig) and [kvantum.svg](templates/kvantum/kvantum.svg)
- Qt5ct/Qt6ct: [qtct.conf](templates/qtct.conf)
2. Configure the templates in `~/.config/clrsync/config.toml`:
```toml
[templates.kvantum]
enabled = true
input_path = '~/.config/clrsync/templates/kvantum/kvantum.kvconfig'
output_path = '~/.config/Kvantum/clrsync/clrsync.kvconfig'
reload_cmd = ''
[templates.kvantum-svg]
enabled = true
input_path = '~/.config/clrsync/templates/kvantum/kvantum.svg'
output_path = '~/.config/Kvantum/clrsync/clrsync.svg'
reload_cmd = ''
[templates.qt5ct]
enabled = true
input_path = '~/.config/clrsync/templates/qtct.conf'
output_path = '~/.config/qt5ct/colors/clrsync.conf'
reload_cmd = ''
[templates.qt6ct]
enabled = true
input_path = '~/.config/clrsync/templates/qtct.conf'
output_path = '~/.config/qt6ct/colors/clrsync.conf'
reload_cmd = ''
```
3. Set the theme in `~/.config/Kvantum/kvantum.kvconfig`:
```conf
[General]
theme=clrsync
```
4. Set the theme in `~/.config/qt5ct/qt5ct.conf` and `~/.config/qt6ct/qt6ct.conf`:
```conf
[Appearance]
color_scheme_path=$HOME/.config/qt5ct/colors/clrsync.conf
custom_palette=true
```
### Window Managers
#### Hyprland
1. Download the [template file](templates/hyprland.conf)
2. Configure the template in `~/.config/clrsync/config.toml`:
```toml
[templates.hyprland]
enabled = true
input_path = '~/.config/clrsync/templates/hyprland.conf'
output_path = '~/.config/hypr/hyprland/clrsync.conf'
reload_cmd = ''
```
3. Source the color theme in your Hyprland config:
```conf
source=~/.config/hypr/hyprland/clrsync.conf
```

BIN
extra/img/cursed-light.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 243 KiB

BIN
extra/img/cursed.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 235 KiB

BIN
extra/img/nord.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 243 KiB

View File

@@ -1,5 +1,5 @@
[colors] [colors]
accent = '#5E81ACFF' accent = '#A1CDFAFF'
background = '#2E3440FF' background = '#2E3440FF'
base00 = '#2E3440FF' base00 = '#2E3440FF'
base01 = '#BF616AFF' base01 = '#BF616AFF'

View File

@@ -0,0 +1,69 @@
[colors.primary]
background = '{background}'
foreground = '{foreground}'
[colors.cursor]
text = '{background}'
cursor = '{cursor}'
[colors.vi_mode_cursor]
text = '{background}'
cursor = '{cursor}'
[colors.selection]
text = 'CellForeground'
background = '{editor_selected}'
[colors.search.matches]
foreground = '{background}'
background = '{base0A}'
[colors.search.focused_match]
foreground = '{background}'
background = '{accent}'
[colors.footer_bar]
foreground = '{foreground}'
background = '{surface}'
[colors.hints.start]
foreground = '{background}'
background = '{warning}'
[colors.hints.end]
foreground = '{background}'
background = '{surface_variant}'
[colors.line_indicator]
foreground = 'None'
background = 'None'
[colors.normal]
black = '{base00}'
red = '{base08}'
green = '{base01}'
yellow = '{base02}'
blue = '{base06}'
magenta = '{base05}'
cyan = '{base0E}'
white = '{base07}'
[colors.bright]
black = '{border_focused}'
red = '{editor_error}'
green = '{editor_success}'
yellow = '{base0A}'
blue = '{editor_link}'
magenta = '{base0D}'
cyan = '{editor_comment}'
white = '{base0F}'
[colors.dim]
black = '{border}'
red = '{base0C}'
green = '{base01}'
yellow = '{editor_string}'
blue = '{base06}'
magenta = '{base05}'
cyan = '{base0E}'
white = '{base07}'

2094
extra/templates/code.json Normal file

File diff suppressed because it is too large Load Diff

26
extra/templates/ghostty Normal file
View File

@@ -0,0 +1,26 @@
background = {background}
foreground = {foreground}
cursor-color = {cursor}
cursor-text = {foreground}
selection-background = {editor_selected}
selection-foreground = {foreground}
palette = 0={base00}
palette = 1={base08}
palette = 2={base01}
palette = 3={base02}
palette = 4={base06}
palette = 5={base05}
palette = 6={base0E}
palette = 7={base07}
palette = 8={border_focused}
palette = 9={editor_error}
palette = 10={editor_success}
palette = 11={base0A}
palette = 12={editor_link}
palette = 13={base0D}
palette = 14={editor_comment}
palette = 15={base0F}

88
extra/templates/gtk.css Normal file
View File

@@ -0,0 +1,88 @@
@define-color accent_color {accent};
@define-color accent_bg_color {accent};
@define-color accent_fg_color {on_surface};
@define-color destructive_color {error};
@define-color destructive_bg_color {error};
@define-color destructive_fg_color {on_error};
@define-color success_color {success};
@define-color success_bg_color {success};
@define-color success_fg_color {on_success};
@define-color warning_color {warning};
@define-color warning_bg_color {warning};
@define-color warning_fg_color {on_warning};
@define-color error_color {error};
@define-color error_bg_color {error};
@define-color error_fg_color {on_error};
@define-color window_bg_color {background};
@define-color window_fg_color {on_background};
@define-color view_bg_color {background};
@define-color view_fg_color {foreground};
@define-color headerbar_bg_color {surface};
@define-color headerbar_fg_color {on_surface};
@define-color headerbar_border_color {border};
@define-color headerbar_backdrop_color @window_bg_color;
@define-color headerbar_shade_color {border};
@define-color sidebar_bg_color {surface_variant};
@define-color sidebar_fg_color {on_surface};
@define-color sidebar_border_color {border};
@define-color sidebar_backdrop_color @window_bg_color;
@define-color sidebar_shade_color {border};
@define-color card_bg_color {surface};
@define-color card_fg_color {on_surface};
@define-color card_shade_color {border};
@define-color dialog_bg_color {surface};
@define-color dialog_fg_color {on_surface};
@define-color popover_bg_color {surface_variant};
@define-color popover_fg_color {on_surface_variant};
@define-color shade_color {border};
@define-color scrollbar_outline_color {border};
@define-color blue_1 {base0C};
@define-color blue_2 {base04};
@define-color blue_3 {base04};
@define-color blue_4 {base04};
@define-color blue_5 {base04};
@define-color green_1 {success};
@define-color green_2 {success};
@define-color green_3 {success};
@define-color green_4 {base02};
@define-color green_5 {base02};
@define-color yellow_1 {base0B};
@define-color yellow_2 {base03};
@define-color yellow_3 {base03};
@define-color yellow_4 {base03};
@define-color yellow_5 {base03};
@define-color orange_1 {warning};
@define-color orange_2 {warning};
@define-color orange_3 {warning};
@define-color orange_4 {warning};
@define-color orange_5 {warning};
@define-color red_1 {error};
@define-color red_2 {error};
@define-color red_3 {error};
@define-color red_4 {base01};
@define-color red_5 {base01};
@define-color purple_1 {base0D};
@define-color purple_2 {base05};
@define-color purple_3 {base05};
@define-color purple_4 {base05};
@define-color purple_5 {base05};
@define-color brown_1 {base0E};
@define-color brown_2 {base0E};
@define-color brown_3 {base0E};
@define-color brown_4 {base0E};
@define-color brown_5 {base0E};
@define-color light_1 {on_background};
@define-color light_2 {on_surface};
@define-color light_3 {on_surface_variant};
@define-color light_4 {border};
@define-color light_5 {border};
@define-color dark_1 {border};
@define-color dark_2 {surface_variant};
@define-color dark_3 {surface};
@define-color dark_4 {background};
@define-color dark_5 {base00};
scale trough highlight {
background: {accent}
}

View File

@@ -0,0 +1,25 @@
$primary = rgb({accent.hex_stripped})
$surface = rgb({surface.hex_stripped})
$secondary = rgb({base04.hex_stripped})
$error = rgb({error.hex_stripped})
$tertiary = rgb({base06.hex_stripped})
$surface_lowest = rgb({background.hex_stripped})
general {
col.active_border = $primary
col.inactive_border = $surface
}
group {
col.border_active = $secondary
col.border_inactive = $surface
col.border_locked_active = $error
col.border_locked_inactive = $surface
groupbar {
col.active = $secondary
col.inactive = $surface
col.locked_active = $error
col.locked_inactive = $surface
}
}

View File

@@ -0,0 +1,32 @@
cursor {cursor}
cursor_text_color {background}
foreground {foreground}
background {background}
selection_foreground {on_surface}
selection_background {surface}
url_color {accent}
color0 {base00}
color8 {base08}
color1 {base01}
color9 {base09}
color2 {base02}
color10 {base0A}
color3 {base03}
color11 {base0B}
color4 {base04}
color12 {base0C}
color5 {base05}
color13 {base0D}
color6 {base06}
color14 {base0E}
color7 {base07}
color15 {base0F}

View File

@@ -0,0 +1,571 @@
[%General]
author=Matugen, based on MaterialADW by Vince Liuice
comment=A Material You theme generated by Matugen
x11drag=none
alt_mnemonic=true
left_tabs=false
attach_active_tab=false
mirror_doc_tabs=true
group_toolbar_buttons=false
toolbar_item_spacing=0
toolbar_interior_spacing=2
spread_progressbar=true
composite=true
menu_shadow_depth=6
spread_menuitems=false
tooltip_shadow_depth=7
splitter_width=1
scroll_width=9
scroll_arrows=false
scroll_min_extent=60
slider_width=2
slider_handle_width=23
slider_handle_length=22
tickless_slider_handle_size=22
center_toolbar_handle=true
check_size=24
textless_progressbar=false
progressbar_thickness=2
menubar_mouse_tracking=true
toolbutton_style=1
double_click=false
translucent_windows=false
blurring=false
popup_blurring=false
vertical_spin_indicators=false
spin_button_width=24
fill_rubberband=false
merge_menubar_with_toolbar=true
small_icon_size=16
large_icon_size=32
button_icon_size=16
toolbar_icon_size=16
combo_as_lineedit=true
animate_states=true
button_contents_shift=false
combo_menu=true
hide_combo_checkboxes=true
combo_focus_rect=false
groupbox_top_label=true
inline_spin_indicators=true
joined_inactive_tabs=false
layout_spacing=3
layout_margin=3
scrollbar_in_view=true
transient_scrollbar=true
transient_groove=false
submenu_overlap=0
tooltip_delay=0
tree_branch_line=false
no_window_pattern=false
opaque=kaffeine,kmplayer,subtitlecomposer,kdenlive,vlc,smplayer,smplayer2,avidemux,avidemux2_qt4,avidemux3_qt4,avidemux3_qt5,kamoso,QtCreator,VirtualBox,VirtualBoxVM,trojita,dragon,digikam,lyx
reduce_window_opacity=0
respect_DE=true
scrollable_menu=false
submenu_delay=150
no_inactiveness=false
reduce_menu_opacity=0
click_behavior=2
contrast=1.00
dialog_button_layout=0
intensity=1.00
saturation=1.00
shadowless_popup=false
drag_from_buttons=false
menu_blur_radius=0
tooltip_blur_radius=0
[GeneralColors]
window.color={background}
base.color={surface}
alt.base.color={surface}
button.color={surface_variant}
light.color={surface}
mid.light.color={surface_variant}
dark.color={surface_variant}
mid.color={surface_variant}
highlight.color={accent}
inactive.highlight.color={accent}
text.color={on_background}
window.text.color={on_background}
button.text.color={on_surface_variant}
disabled.text.color={editor_disabled}
tooltip.text.color={on_surface}
highlight.text.color={on_surface}
link.color={base04}
link.visited.color={base05}
progress.indicator.text.color={on_surface}
[Hacks]
transparent_ktitle_label=true
transparent_dolphin_view=true
transparent_pcmanfm_sidepane=true
blur_translucent=false
transparent_menutitle=true
respect_darkness=true
kcapacitybar_as_progressbar=true
force_size_grip=true
iconless_pushbutton=false
iconless_menu=false
disabled_icon_opacity=100
lxqtmainmenu_iconsize=16
normal_default_pushbutton=true
single_top_toolbar=true
tint_on_mouseover=0
transparent_pcmanfm_view=true
no_selection_tint=true
transparent_arrow_button=true
middle_click_scroll=false
opaque_colors=false
kinetic_scrolling=false
scroll_jump_workaround=true
centered_forms=false
noninteger_translucency=false
style_vertical_toolbars=false
blur_only_active_window=true
[PanelButtonCommand]
frame=true
frame.element=button
frame.top=6
frame.bottom=6
frame.left=6
frame.right=6
interior=true
interior.element=button
indicator.size=8
text.normal.color={on_surface}
text.focus.color={on_surface}
text.press.color={on_surface}
text.toggle.color={on_surface}
text.shadow=0
text.margin=4
text.iconspacing=4
indicator.element=arrow
frame.expansion=0
[PanelButtonTool]
inherits=PanelButtonCommand
text.normal.color={on_surface}
text.focus.color={on_surface}
text.press.color={on_surface}
text.toggle.color={on_surface}
text.bold=false
indicator.element=arrow
indicator.size=8
frame.expansion=0
[ToolbarButton]
frame=true
frame.element=tbutton
interior.element=tbutton
frame.top=16
frame.bottom=16
frame.left=16
frame.right=16
indicator.element=tarrow
text.normal.color={on_surface}
text.focus.color={on_surface}
text.press.color={on_surface}
text.toggle.color={on_surface}
text.bold=false
frame.expansion=32
[Dock]
inherits=PanelButtonCommand
interior.element=dock
frame.element=dock
frame.top=1
frame.bottom=1
frame.left=1
frame.right=1
text.normal.color={on_surface}
[DockTitle]
inherits=PanelButtonCommand
frame=false
interior=false
text.normal.color={on_surface}
text.focus.color={on_surface}
text.bold=false
[IndicatorSpinBox]
inherits=PanelButtonCommand
frame=true
interior=true
frame.top=2
frame.bottom=2
frame.left=2
frame.right=2
indicator.element=spin
indicator.size=8
text.normal.color={on_surface}
text.margin.top=2
text.margin.bottom=2
text.margin.left=2
text.margin.right=2
[RadioButton]
inherits=PanelButtonCommand
frame=false
interior.element=radio
text.normal.color={on_surface}
text.focus.color={on_surface}
min_width=+0.3font
min_height=+0.3font
[CheckBox]
inherits=PanelButtonCommand
frame=false
interior.element=checkbox
text.normal.color={on_surface}
text.focus.color={on_surface}
min_width=+0.3font
min_height=+0.3font
[Focus]
inherits=PanelButtonCommand
frame=true
frame.element=focus
frame.top=2
frame.bottom=2
frame.left=2
frame.right=2
frame.patternsize=14
[GenericFrame]
inherits=PanelButtonCommand
frame=true
interior=false
frame.element=common
interior.element=common
frame.top=1
frame.bottom=1
frame.left=1
frame.right=1
[LineEdit]
inherits=PanelButtonCommand
frame.element=lineedit
interior.element=lineedit
frame.top=6
frame.bottom=6
frame.left=6
frame.right=6
text.margin.top=2
text.margin.bottom=2
text.margin.left=2
text.margin.right=2
[ToolbarLineEdit]
frame.element=lineedit
interior.element=lineedit
[DropDownButton]
inherits=PanelButtonCommand
indicator.element=arrow-down
[IndicatorArrow]
indicator.element=arrow
indicator.size=8
[ToolboxTab]
inherits=PanelButtonCommand
text.normal.color={on_surface}
text.press.color={on_surface}
text.focus.color={on_surface}
[Tab]
inherits=PanelButtonCommand
interior.element=tab
text.margin.left=8
text.margin.right=8
text.margin.top=0
text.margin.bottom=0
frame.element=tab
indicator.element=tab
indicator.size=22
frame.top=8
frame.bottom=8
frame.left=8
frame.right=8
text.normal.color={on_surface}
text.focus.color={on_surface}
text.press.color={on_surface}
text.toggle.color={on_surface}
frame.expansion=0
text.bold=false
[TabFrame]
inherits=PanelButtonCommand
frame.element=tabframe
interior.element=tabframe
frame.top=6
frame.bottom=6
frame.left=6
frame.right=6
[TreeExpander]
inherits=PanelButtonCommand
indicator.size=8
indicator.element=tree
[HeaderSection]
inherits=PanelButtonCommand
interior.element=header
frame.element=header
frame.top=0
frame.bottom=1
frame.left=1
frame.right=1
text.normal.color={on_surface}
text.focus.color={on_surface}
text.press.color={on_surface}
text.toggle.color={on_surface}
frame.expansion=0
[SizeGrip]
indicator.element=resize-grip
[Toolbar]
inherits=PanelButtonCommand
indicator.element=toolbar
indicator.size=5
text.margin=0
interior.element=menubar
frame.element=menubar
text.normal.color={on_surface}
text.focus.color={on_surface}
text.press.color={on_surface}
text.toggle.color={on_surface}
frame.left=6
frame.right=6
frame.top=0
frame.bottom=1
frame.expansion=0
[Slider]
inherits=PanelButtonCommand
frame.element=slider
focusFrame=true
interior.element=slider
frame.top=3
frame.bottom=3
frame.left=3
frame.right=3
[SliderCursor]
inherits=PanelButtonCommand
frame=false
interior.element=slidercursor
[Progressbar]
inherits=PanelButtonCommand
frame.element=progress
interior.element=progress
text.margin=0
text.normal.color={on_surface}
text.focus.color={on_surface}
text.press.color={on_surface}
text.toggle.color={on_surface}
text.bold=false
frame.expansion=8
[ProgressbarContents]
inherits=PanelButtonCommand
frame=true
frame.element=progress-pattern
interior.element=progress-pattern
[ItemView]
inherits=PanelButtonCommand
text.margin=0
frame.element=itemview
interior.element=itemview
frame.top=4
frame.bottom=4
frame.left=4
frame.right=4
text.margin.top=0
text.margin.bottom=0
text.margin.left=8
text.margin.right=8
text.normal.color={on_surface}
text.focus.color={on_surface}
text.press.color={on_surface}
text.toggle.color={on_surface}
min_width=+0.3font
min_height=+0.3font
frame.expansion=0
[Splitter]
interior.element=splitter
frame=false
indicator.size=0
[Scrollbar]
inherits=PanelButtonCommand
indicator.element=arrow
indicator.size=12
[ScrollbarSlider]
inherits=PanelButtonCommand
frame.element=scrollbarslider
interior=false
frame.left=5
frame.right=5
frame.top=5
frame.bottom=5
indicator.element=grip
indicator.size=12
[ScrollbarGroove]
inherits=PanelButtonCommand
interior=false
frame=false
[Menu]
inherits=PanelButtonCommand
frame.top=10
frame.bottom=10
frame.left=10
frame.right=10
frame.element=menu
interior.element=menu
text.normal.color={on_surface}
text.shadow=false
frame.expansion=0
text.bold=false
[MenuItem]
inherits=PanelButtonCommand
frame=true
frame.element=menuitem
interior.element=menuitem
indicator.element=menuitem
text.normal.color={on_surface}
text.focus.color={on_surface}
text.margin.top=0
text.margin.bottom=0
text.margin.left=6
text.margin.right=6
frame.top=4
frame.bottom=4
frame.left=4
frame.right=4
text.bold=false
frame.expansion=0
[MenuBar]
inherits=PanelButtonCommand
frame.element=menubar
interior.element=menubar
frame.bottom=0
text.normal.color={on_surface}
text.focus.color={on_surface}
text.press.color={on_surface}
text.toggle.color={on_surface}
frame.expansion=0
text.bold=false
[MenuBarItem]
inherits=PanelButtonCommand
interior=true
interior.element=menubaritem
frame.element=menubaritem
frame.top=2
frame.bottom=2
frame.left=2
frame.right=2
text.margin.left=4
text.margin.right=4
text.margin.top=0
text.margin.bottom=0
text.normal.color={on_surface}
text.focus.color={on_surface}
text.press.color={on_surface}
text.toggle.color={on_surface}
text.bold=false
min_width=+0.3font
min_height=+0.3font
frame.expansion=0
[TitleBar]
inherits=PanelButtonCommand
frame=false
text.margin.top=2
text.margin.bottom=2
text.margin.left=2
text.margin.right=2
interior.element=titlebar
indicator.size=16
indicator.element=mdi
text.normal.color={on_surface}
text.focus.color={on_surface}
text.bold=false
text.italic=true
frame.expansion=0
[ComboBox]
inherits=PanelButtonCommand
frame.element=combo
interior.element=combo
frame.top=6
frame.bottom=6
frame.left=6
frame.right=6
text.margin.top=2
text.margin.bottom=2
text.margin.left=2
text.margin.right=2
text.focus.color={on_surface}
text.press.color={on_surface}
text.toggle.color={on_surface}
[GroupBox]
inherits=GenericFrame
frame=false
text.shadow=0
text.margin=0
text.normal.color={on_surface}
text.focus.color={on_surface}
text.bold=false
frame.expansion=0
[TabBarFrame]
inherits=GenericFrame
frame=false
frame.element=tabBarFrame
interior=false
frame.top=0
frame.bottom=0
frame.left=0
frame.right=0
[ToolTip]
inherits=GenericFrame
frame.top=6
frame.bottom=6
frame.left=6
frame.right=6
interior=true
text.shadow=0
text.margin=6
interior.element=tooltip
frame.element=tooltip
frame.expansion=6
[StatusBar]
inherits=GenericFrame
frame=false
interior=false
[Window]
interior=true
interior.element=window
frame=true
frame.element=window
frame.bottom=10
frame.top=10
text.disabled.color={editor_disabled}

File diff suppressed because it is too large Load Diff

After

Width:  |  Height:  |  Size: 170 KiB

93
extra/templates/nvim.lua Normal file
View File

@@ -0,0 +1,93 @@
vim.cmd("highlight clear")
vim.cmd("syntax reset")
vim.g.colors_name = "clrsync"
local palette = {
-- Editor colors
Default = "{editor_main.hex}",
Keyword = "{editor_command.hex}",
Number = "{editor_warning.hex}",
String = "{editor_string.hex}",
CharLiteral = "{editor_string.hex}",
Punctuation = "{editor_main.hex}",
Preprocessor = "{editor_emphasis.hex}",
Identifier = "{editor_main.hex}",
KnownIdentifier = "{editor_link.hex}",
PreprocIdentifier = "{editor_link.hex}",
Comment = "{editor_comment.hex}",
MultiLineComment = "{editor_comment.hex}",
Background = "{editor_background.hex}",
Cursor = "{cursor.hex}",
Selection = "{editor_selected.hex}",
ErrorMarker = "{editor_error.hex}",
Breakpoint = "{editor_error.hex}",
LineNumber = "{editor_line_number.hex}",
CurrentLineFill = "{surface_variant.hex}",
CurrentLineFillInactive = "{surface.hex}",
CurrentLineEdge = "{border_focused.hex}",
-- Semantic colors
Success = "{success.hex}",
Warning = "{warning.hex}",
Error = "{error.hex}",
Info = "{info.hex}",
}
-- Helper function to set highlights in Neovim
local function set_hl(group, opts)
vim.api.nvim_set_hl(0, group, opts)
end
vim.o.winborder = "rounded"
-- Basic editor highlights using the mapped palette
set_hl("Normal", { fg = palette.Default, bg = palette.Background })
set_hl("CursorLine", { bg = palette.CurrentLineFill })
set_hl("Visual", { bg = palette.Selection })
set_hl("LineNr", { fg = palette.LineNumber })
set_hl("CursorLineNr", { fg = palette.Keyword })
-- Syntax highlights
set_hl("Comment", { fg = palette.Comment, italic = true })
set_hl("Constant", { fg = palette.Number })
set_hl("String", { fg = palette.String })
set_hl("Character", { fg = palette.CharLiteral })
set_hl("Identifier", { fg = palette.Identifier })
set_hl("Function", { fg = palette.Keyword })
set_hl("Statement", { fg = palette.Keyword })
set_hl("PreProc", { fg = palette.Preprocessor })
set_hl("Type", { fg = palette.Keyword })
set_hl("Special", { fg = palette.PreprocIdentifier })
set_hl("Underlined", { fg = palette.KnownIdentifier })
set_hl("Error", { fg = palette.ErrorMarker, bg = palette.Background })
set_hl("Todo", { fg = palette.Default, bg = palette.Keyword })
-- Floating windows
set_hl("NormalFloat", { bg = palette.Background })
set_hl("FloatBorder", { fg = palette.CurrentLineEdge, bg = palette.Background })
-- Completion menu
set_hl("Pmenu", { bg = palette.Background })
set_hl("PmenuSel", { bg = palette.Keyword, fg = palette.Background })
-- Git and diagnostic highlights
set_hl("DiffAdd", { fg = palette.Success, bg = palette.Background })
set_hl("DiffChange", { fg = palette.Keyword, bg = palette.Background })
set_hl("DiffDelete", { fg = palette.ErrorMarker, bg = palette.Background })
set_hl("DiagnosticError", { fg = palette.Error })
set_hl("DiagnosticWarn", { fg = palette.Warning })
set_hl("DiagnosticInfo", { fg = palette.Info })
set_hl("DiagnosticHint", { fg = palette.PreprocIdentifier })
-- Treesitter links
set_hl("@comment", { link = "Comment" })
set_hl("@string", { fg = palette.String })
set_hl("@function", { fg = palette.Keyword })
set_hl("@variable", { fg = palette.Identifier })
set_hl("@keyword", { fg = palette.Keyword })
set_hl("@type", { fg = palette.Preprocessor })

View File

@@ -0,0 +1,4 @@
[ColorScheme]
active_colors={on_background}, {surface}, #ffffff, #cacaca, #9f9f9f, #b8b8b8, {on_background}, #ffffff, {on_surface}, {background}, {background}, {border}, {accent}, {on_surface}, {base04}, {accent}, {surface}, {surface}, {surface}, {on_surface}, {base04}
disabled_colors={editor_disabled}, {surface}, #ffffff, #cacaca, #9f9f9f, #b8b8b8, {editor_disabled}, #ffffff, {editor_disabled}, {background}, {background}, {border}, {surface_variant}, {on_surface_variant}, {base04}, {accent}, {surface}, {surface}, {surface}, {editor_disabled}, {base04}
inactive_colors={on_background}, {surface}, #ffffff, #cacaca, #9f9f9f, #b8b8b8, {on_background}, #ffffff, {on_surface}, {background}, {background}, {border}, {surface_variant}, {on_surface_variant}, {base04}, {accent}, {surface}, {surface}, {surface}, {on_surface}, {base04}

View File

@@ -0,0 +1,213 @@
// Template mostly stolen from noctalia-shell
// https://github.com/noctalia-dev/noctalia-shell/blob/main/Assets/MatugenTemplates/telegram.tdesktop-theme
COLOR_GRAY: {border};
COLOR_DARK: {surface_variant};
windowBg: {background}; // Main background
windowFg: {on_background}; // Main text
windowBgOver: {surface_variant}; // Generic background on hover
windowBgRipple: {surface_variant}; // Ripple effect
windowFgOver: {on_surface_variant}; // Text on hover
windowSubTextFg: {foreground}60; // Minor text
windowSubTextFgOver: {foreground}80; // Minor text on hover
dialogsBg: {background}; // Sidebar background
dialogsBgActive: {surface}; // Active chat background
dialogsBgOver: {surface_variant}; // Hover background
dialogsRippleBg: {surface_variant}; // Ripple effect
dialogsRippleBgActive: {surface_variant}; // Ripple effect for active chat
dialogsNameFg: {foreground}; // Chat name
dialogsNameFgActive: {foreground}; // Chat name for active chat
dialogsNameFgOver: {foreground}; // Chat name on hover
dialogsTextFg: {foreground}; // Message preview
dialogsTextFgActive: {foreground}; // Message preview for active chat
dialogsTextFgOver: {foreground}; // Message preview on hover
dialogsTextFgService: {foreground}; // Service text (group sender name)
dialogsTextFgServiceActive: {foreground}; // Service text for active chat
dialogsTextFgServiceOver: {foreground}; // Service text on hover
dialogsDateFg: {foreground}60; // Date text
dialogsDateFgActive: {foreground}60; // Date text for active chat
dialogsDateFgOver: {foreground}60; // Date text on hover
dialogsDraftFg: {foreground}; // Draft label
dialogsDraftFgActive: {foreground}; // Draft label for active chat
dialogsDraftFgOver: {foreground}; // Draft label on hover
dialogsVerifiedIconBg: {accent}; // Verified badge background
dialogsVerifiedIconBgActive: {accent}; // Verified badge for active chat
dialogsVerifiedIconBgOver: {accent}; // Verified badge on hover
dialogsVerifiedIconFg: {background}; // Verified icon
dialogsVerifiedIconFgActive: {background}; // Verified icon for active chat
dialogsVerifiedIconFgOver: {background}; // Verified icon on hover
dialogsSendingIconFg: {foreground}; // Sending icon (clock)
dialogsSendingIconFgActive: {foreground}; // Sending icon for active chat
dialogsSendingIconFgOver: {foreground}; // Sending icon on hover
dialogsSentIconFg: {accent}; // Sent icon (tick)
dialogsSentIconFgActive: {accent}; // Sent icon for active chat
dialogsSentIconFgOver: {accent}; // Sent icon on hover
dialogsUnreadBg: {accent}; // Unread badge background
dialogsUnreadBgActive: {accent}; // Unread badge for active chat
dialogsUnreadBgOver: {accent}; // Unread badge on hover
dialogsUnreadBgMuted: {foreground}; // Muted unread badge
dialogsUnreadBgMutedActive: {foreground}; // Muted unread badge for active chat
dialogsUnreadBgMutedOver: {foreground}; // Muted unread badge on hover
dialogsUnreadFg: {background}; // Unread badge text
dialogsUnreadFgActive: {background}; // Unread badge text for active chat
dialogsUnreadFgOver: {background}; // Unread badge text on hover
dialogsChatIconFg: {foreground}; // Group/channel icon
dialogsChatIconFgActive: {foreground}; // Group/channel icon for active chat
dialogsChatIconFgOver: {foreground}; // Group/channel icon on hover
dialogsOnlineBadgeFg: {foreground}; // Online status
dialogsOnlineBadgeFgActive: {foreground}; // Online status for active chat
dialogsForwardBg: {surface}; // Forwarding panel background
dialogsForwardFg: {foreground}; // Forwarding panel text
dialogsMenuIconFg: {foreground}; // Main menu icon
dialogsMenuIconFgOver: {foreground}60; // Main menu icon on hover
windowBoldFg: {on_background}; // Bold text
windowBoldFgOver: {on_surface_variant}; // Bold text on hover
windowBgActive: {surface}; // Active items background
windowFgActive: {foreground}; // Active items text
windowActiveTextFg: {accent}; // Active items text
windowShadowFg: {border}; // Window shadow
windowShadowFgFallback: {border}; // Fallback for shadow
historyOutIconFg: {accent};
historyIconFgInverted: {on_surface};
msgServiceBg: {surface}80;
msgServiceFg: {foreground};
msgOutBg: {surface};
msgOutBgSelected: {surface_variant};
msgOutServiceFg: {on_surface};
msgOutDateFg: {foreground}80;
historySentIconFg: {foreground};
msgOutDateFgSelected: {foreground};
msgInBg: {surface_variant};
msgInBgSelected: {surface};
msgDateImgFg: {on_surface};
shadowFg: {border}; // General shadow
slideFadeOutBg: {background};
slideFadeOutShadowFg: {border};
imageBg: {surface};
imageBgTransparent: {surface};
activeButtonBg: {accent}; // Active button background
activeButtonBgOver: {surface_variant}; // Active button hover background
activeButtonBgRipple: {on_surface_variant}; // Active button ripple
activeButtonFg: {on_background}; // Active button text
activeButtonFgOver: {on_surface_variant}; // Active button hover text
activeButtonSecondaryFg: {on_background}; // Active button secondary text
activeButtonSecondaryFgOver: {on_surface_variant}; // Active button secondary hover text
activeLineFg: {accent};
dialogsBgActive: {surface};
lightButtonBg: {surface}; // Light button background
lightButtonBgOver: {surface_variant}; // Light button hover background
lightButtonBgRipple: {accent}; // Light button ripple
lightButtonFg: {on_surface}; // Light button text
lightButtonFgOver: {on_surface_variant}; // Light button hover text
attentionButtonFg: {error};
attentionButtonFgOver: {error};
attentionButtonBgOver: {surface_variant};
attentionButtonBgRipple: {on_surface};
outlineButtonBg: {surface}; // Outline button background
outlineButtonBgOver: {surface_variant}; // Outline button hover background
outlineButtonOutlineFg: {accent}; // Outline button color
outlineButtonBgRipple: {accent}; // Outline button ripple
menuBg: {surface};
menuBgOver: {surface_variant};
menuBgRipple: {accent};
menuIconFg: {on_surface};
menuIconFgOver: {on_surface_variant};
menuSubmenuArrowFg: {border};
menuFgDisabled: {border};
menuSeparatorFg: {border};
scrollBarBg: {accent}40; // Scroll bar background (40% opacity)
scrollBarBgOver: {accent}60; // Scroll bar hover background (60% opacity)
scrollBg: {surface_variant}40; // Scroll bar track (40% opacity)
scrollBgOver: {surface_variant}60; // Scroll bar track on hover (60% opacity)
smallCloseIconFg: {border};
smallCloseIconFgOver: {on_surface_variant};
radialFg: {accent};
radialBg: {surface};
placeholderFg: {border}; // Placeholder text
placeholderFgActive: {accent}; // Active placeholder text
inputBorderFg: {border}; // Input border
filterInputBorderFg: {border}; // Search input border
filterInputInactiveBg: {surface}; // Inactive search input background
checkboxFg: {accent}; // Checkbox color
// Filters sidebar (left side bar with folder filters)
sideBarBg: {surface}; // Filters sidebar background
sideBarBgActive: {surface_variant}; // Filters sidebar active filter background
sideBarBgRipple: {background}; // Filters sidebar ripple effect
sideBarTextFg: {foreground}; // Filters sidebar text
sideBarTextFgActive: {foreground}; // Filters sidebar active filter text
sideBarIconFg: {foreground}; // Filters sidebar icon
sideBarIconFgActive: {foreground}; // Filters sidebar active filter icon
sideBarBadgeBg: {accent}; // Filters sidebar badge background
sideBarBadgeBgMuted: {foreground}60; // Filters sidebar muted badge background
titleBg: {surface}; // Window title background
titleShadow: {border};
titleButtonFg: {on_surface}; // Title button color
titleButtonBgOver: {surface_variant}; // Title button hover background
titleButtonFgOver: {on_surface_variant}; // Title button hover color
titleButtonCloseBgOver: {error};
titleButtonCloseFgOver: {on_error};
titleFgActive: {on_surface}; // Active title text
titleFg: {on_surface}; // Inactive title text
trayCounterBg: {error}; // Tray counter background
trayCounterBgMute: {border}; // Muted tray counter background
trayCounterFg: {on_error}; // Tray counter text
trayCounterBgMacInvert: {error}; // Mac tray counter
trayCounterFgMacInvert: {on_error}; // Mac tray counter text
layerBg: {surface}99; // Layer background (60% opacity)
cancelIconFg: {error}; // Cancel icon
cancelIconFgOver: {error}; // Cancel icon on hover
boxBg: {surface}; // Box background
boxTextFg: {on_surface}; // Box text
boxTextFgGood: {accent}; // Box good text
boxTextFgError: {error}; // Box error text
boxTitleFg: {on_surface}; // Box title text
boxSearchBg: {surface}; // Box search field background
boxSearchCancelIconFg: {error}; // Box search cancel icon
boxSearchCancelIconFgOver: {error}; // Box search cancel icon on hover
contactsBg: {surface}; // Contacts background
contactsBgOver: {surface_variant}; // Contacts background on hover
contactsNameFg: {on_surface}; // Contact name
contactsStatusFg: {border}; // Contact status
contactsStatusFgOver: {on_surface_variant}; // Contact status on hover
contactsStatusFgOnline: {accent}; // Online contact status
photoCropFadeBg: {surface}cc; // Photo crop fade background
photoCropPointFg: {accent}; // Photo crop points
chat_inBubbleSelected: {surface_variant}; // inbox selected chat background
chat_outBubbleSelected: {surface_variant}; // outbox selected chat background

View File

@@ -0,0 +1,109 @@
/* clrsync Firefox Theme */
/* Place in: <Firefox Profile>/chrome/userChrome.css */
/* Enable: about:config -> toolkit.legacyUserProfileCustomizations.stylesheets = true */
:root {
--lwt-accent-color: {background} !important;
--lwt-text-color: {foreground} !important;
--toolbar-bgcolor: {surface} !important;
--toolbar-color: {on_surface} !important;
--toolbarbutton-hover-background: {surface_variant} !important;
--toolbarbutton-active-background: {border_focused} !important;
--toolbarbutton-icon-fill: {on_surface} !important;
--tab-selected-bgcolor: {surface} !important;
--tab-selected-textcolor: {foreground} !important;
--tab-loading-fill: {accent} !important;
--lwt-tab-text: {foreground} !important;
--tab-line-color: {accent} !important;
--urlbar-box-bgcolor: {surface_variant} !important;
--urlbar-box-text-color: {on_surface} !important;
--urlbar-box-hover-bgcolor: {surface} !important;
--urlbar-box-focus-bgcolor: {surface} !important;
--urlbar-popup-url-color: {editor_link} !important;
--arrowpanel-background: {surface} !important;
--arrowpanel-color: {on_surface} !important;
--arrowpanel-border-color: {border} !important;
--panel-separator-color: {border} !important;
--sidebar-background-color: {background} !important;
--sidebar-text-color: {foreground} !important;
--sidebar-border-color: {border} !important;
--toolbar-field-focus-background-color: {surface} !important;
--toolbar-field-focus-color: {on_surface} !important;
--toolbar-field-focus-border-color: {accent} !important;
}
#TabsToolbar {
background-color: {background} !important;
}
.tab-background[selected="true"] {
background-color: {surface} !important;
}
.tabbrowser-tab:hover > .tab-stack > .tab-background:not([selected="true"]) {
background-color: {surface_variant} !important;
}
.tab-line {
background-color: {accent} !important;
}
#nav-bar {
background-color: {surface} !important;
}
#urlbar-background {
background-color: {surface_variant} !important;
border-color: {border} !important;
}
#urlbar[focused="true"] > #urlbar-background {
background-color: {surface} !important;
border-color: {accent} !important;
}
#urlbar-input {
color: {on_surface} !important;
}
#PersonalToolbar {
background-color: {surface} !important;
}
menupopup, panel {
--panel-background: {surface} !important;
--panel-color: {on_surface} !important;
}
menupopup {
background-color: {surface} !important;
color: {on_surface} !important;
}
menuitem:hover, menu:hover {
background-color: {surface_variant} !important;
}
#sidebar-box {
background-color: {background} !important;
}
#sidebar-header {
background-color: {surface} !important;
border-bottom-color: {border} !important;
}
findbar {
background-color: {surface} !important;
color: {on_surface} !important;
}
* {
scrollbar-color: {surface_variant} {background} !important;
}

View File

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

View File

@@ -198,7 +198,7 @@ Result<void> config::save_config_value(const std::string &section, const std::st
if (!m_temp_file) if (!m_temp_file)
{ {
m_temp_file = std::make_unique<clrsync::core::io::toml_file>(m_temp_config_path); m_temp_file = std::make_unique<clrsync::core::io::toml_file>(m_temp_config_path);
m_temp_file->parse(); (void)m_temp_file->parse();
} }
m_temp_file->set_value(section, key, value); m_temp_file->set_value(section, key, value);
@@ -347,7 +347,7 @@ Result<void> config::remove_template(const std::string &key)
if (!m_temp_file) if (!m_temp_file)
{ {
m_temp_file = std::make_unique<clrsync::core::io::toml_file>(m_temp_config_path); m_temp_file = std::make_unique<clrsync::core::io::toml_file>(m_temp_config_path);
m_temp_file->parse(); (void)m_temp_file->parse();
} }
m_temp_file->remove_section("templates." + key); m_temp_file->remove_section("templates." + key);
return m_temp_file->save_file(); return m_temp_file->save_file();

View File

@@ -6,6 +6,7 @@
#include "core/theme/template_manager.hpp" #include "core/theme/template_manager.hpp"
#include <iostream> #include <iostream>
#include <string> #include <string>
#include <thread>
namespace clrsync::core namespace clrsync::core
{ {
@@ -58,12 +59,15 @@ template <typename FileType> class theme_renderer
if (!tmpl.reload_command().empty()) if (!tmpl.reload_command().empty())
{ {
int result = std::system(tmpl.reload_command().c_str()); std::string cmd = tmpl.reload_command();
std::thread([cmd]() {
int result = std::system(cmd.c_str());
if (result != 0) if (result != 0)
{ {
std::cerr << "Warning: Command " << tmpl.reload_command() std::cerr << "Warning: Reload command '" << cmd
<< " failed with code " << result << "\n"; << "' failed with code " << result << "\n";
} }
}).detach();
} }
} }
return Ok(); return Ok();

View File

@@ -7,19 +7,30 @@ set(GUI_SOURCES
views/template_editor.cpp views/template_editor.cpp
controllers/palette_controller.cpp controllers/palette_controller.cpp
controllers/template_controller.cpp controllers/template_controller.cpp
helpers/imgui_helpers.cpp
helpers/imgui_helpers.hpp
views/about_window.cpp views/about_window.cpp
views/settings_window.cpp views/settings_window.cpp
widgets/colors.cpp
widgets/dialogs.cpp
widgets/palette_selector.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/windows/font_loader_windows.cpp
platform/linux/font_loader_linux.cpp platform/linux/font_loader_linux.cpp
platform/macos/font_loader_macos.cpp platform/macos/font_loader_macos.cpp
platform/linux/file_browser_linux.cpp platform/linux/file_browser_linux.cpp
platform/windows/file_browser_windows.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
platform/linux/font_loader_linux.cpp
platform/macos/font_loader_macos.cpp
platform/windows/font_loader_windows.cpp
backend/glfw_opengl.cpp backend/glfw_opengl.cpp
ui_manager.cpp ui_manager.cpp
) )

View File

@@ -9,7 +9,6 @@ namespace clrsync::gui::backend{
std::string title = "clrsync"; std::string title = "clrsync";
int width = 1280; int width = 1280;
int height = 720; int height = 720;
bool resizable = true;
bool decorated = true; bool decorated = true;
bool transparent_framebuffer = true; bool transparent_framebuffer = true;
float clear_color[4] = {0.1f, 0.1f, 0.1f, 0.0f}; float clear_color[4] = {0.1f, 0.1f, 0.1f, 0.0f};

View File

@@ -1,5 +1,6 @@
#include "gui/backend/glfw_opengl.hpp" #include "gui/backend/glfw_opengl.hpp"
#include <iostream> #include <iostream>
#include <string>
#include <GLFW/glfw3.h> #include <GLFW/glfw3.h>
#include <imgui.h> #include <imgui.h>
#include <imgui_impl_glfw.h> #include <imgui_impl_glfw.h>
@@ -33,7 +34,7 @@ bool glfw_opengl_backend::initialize(const window_config &config)
#else #else
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 3); glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 3);
#endif #endif
glfwWindowHint(GLFW_RESIZABLE, config.resizable ? GLFW_TRUE : GLFW_FALSE); glfwWindowHint(GLFW_RESIZABLE,GLFW_TRUE);
glfwWindowHint(GLFW_DECORATED, config.decorated ? GLFW_TRUE : GLFW_FALSE); glfwWindowHint(GLFW_DECORATED, config.decorated ? GLFW_TRUE : GLFW_FALSE);
glfwWindowHint(GLFW_TRANSPARENT_FRAMEBUFFER, config.transparent_framebuffer ? GLFW_TRUE : GLFW_FALSE); glfwWindowHint(GLFW_TRANSPARENT_FRAMEBUFFER, config.transparent_framebuffer ? GLFW_TRUE : GLFW_FALSE);
@@ -98,6 +99,22 @@ void *glfw_opengl_backend::get_graphics_context() const
return static_cast<void *>(m_window); 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() bool glfw_opengl_backend::init_imgui_backend()
{ {
if (!m_window) if (!m_window)

View File

@@ -23,6 +23,9 @@ namespace clrsync::gui::backend{
void* get_native_window() const override; void* get_native_window() const override;
void* get_graphics_context() 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; bool init_imgui_backend() override;
void shutdown_imgui_backend() override; void shutdown_imgui_backend() override;
void imgui_new_frame() override; void imgui_new_frame() override;

View File

@@ -1,266 +0,0 @@
#include <iostream>
#include <string>
#include "GLFW/glfw3.h"
#include "gui/views/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;
}
} // namespace palette_utils

View File

@@ -1,42 +0,0 @@
#ifndef CLRSYNC_IMGUI_HELPERS_HPP
#define CLRSYNC_IMGUI_HELPERS_HPP
#include "core/palette/palette.hpp"
#include "gui/views/about_window.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 palette_utils
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);
}
} // namespace imgui_helpers
#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,17 +1,13 @@
#include <iostream> #include <iostream>
#include <memory> #include <memory>
#include <GLFW/glfw3.h>
#include <GLFW/glfw3native.h>
#include "core/common/error.hpp" #include "core/common/error.hpp"
#include "core/common/utils.hpp" #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 "gui/backend/glfw_opengl.hpp" #include "gui/backend/glfw_opengl.hpp"
#include "gui/helpers/imgui_helpers.hpp" #include "gui/layout/main_layout.hpp"
#include "gui/platform/font_loader.hpp"
#include "gui/ui_manager.hpp" #include "gui/ui_manager.hpp"
#include "gui/views/about_window.hpp" #include "gui/views/about_window.hpp"
#include "gui/views/color_scheme_editor.hpp" #include "gui/views/color_scheme_editor.hpp"
@@ -37,13 +33,12 @@ 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);
printf("GLFW Version: %s\n", glfwGetVersionString());
auto backend = clrsync::gui::backend::glfw_opengl_backend(); auto backend = clrsync::gui::backend::glfw_opengl_backend();
auto window_config = clrsync::gui::backend::window_config(); auto window_config = clrsync::gui::backend::window_config();
window_config.title = "clrsync"; window_config.title = "clrsync";
window_config.width = 1280; window_config.width = 1280;
window_config.height = 720; window_config.height = 720;
window_config.decorated = true;
window_config.transparent_framebuffer = true; window_config.transparent_framebuffer = true;
if (!backend.initialize(window_config)) if (!backend.initialize(window_config))
@@ -52,14 +47,8 @@ int main(int, char **)
return 1; return 1;
} }
std::cout << "GLFW runtime platform: "; std::cout << "GLFW Version: " << backend.get_glfw_version() << std::endl;
switch (glfwGetPlatform()) { std::cout << "GLFW runtime platform: " << backend.get_glfw_platform() << std::endl;
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";
}
clrsync::gui::ui_manager ui_manager(&backend); clrsync::gui::ui_manager ui_manager(&backend);
@@ -74,18 +63,11 @@ int main(int, char **)
return 1; return 1;
} }
font_loader loader; clrsync::gui::layout::main_layout main_layout;
ImFont *font = loader.load_font(clrsync::core::config::instance().font().c_str(),
clrsync::core::config::instance().font_size());
if (font)
ImGui::GetIO().FontDefault = font;
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);
@@ -96,18 +78,25 @@ int main(int, char **)
{ {
backend.begin_frame(); backend.begin_frame();
loader.push_default_font(); ui_manager.push_default_font();
ui_manager.begin_frame(); ui_manager.begin_frame();
render_menu_bar(&aboutWindow, &settingsWindow); main_layout.render_menu_bar();
setup_main_dockspace(first_time); 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();
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();
ui_manager.end_frame(); ui_manager.end_frame();
backend.end_frame(); backend.end_frame();

View File

@@ -1,5 +1,8 @@
#include "gui/ui_manager.hpp" #include "gui/ui_manager.hpp"
#include "gui/backend/backend.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> #include <imgui.h>
@@ -8,11 +11,14 @@ namespace clrsync::gui
ui_manager::ui_manager(backend::backend_interface* backend) ui_manager::ui_manager(backend::backend_interface* backend)
: m_backend(backend) : m_backend(backend)
{ {
m_font_loader = new font_loader();
} }
ui_manager::~ui_manager() ui_manager::~ui_manager()
{ {
shutdown(); shutdown();
delete m_font_loader;
m_font_loader = nullptr;
} }
bool ui_manager::initialize(const ui_config& config) bool ui_manager::initialize(const ui_config& config)
@@ -44,6 +50,13 @@ bool ui_manager::initialize(const ui_config& config)
return false; 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; return true;
} }
@@ -68,4 +81,57 @@ void ui_manager::end_frame()
ImGui::Render(); ImGui::Render();
m_backend->imgui_render_draw_data(ImGui::GetDrawData()); 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);
}
} }

View File

@@ -1,6 +1,10 @@
#ifndef CLRSYNC_UI_MANAGER_HPP #ifndef CLRSYNC_UI_MANAGER_HPP
#define CLRSYNC_UI_MANAGER_HPP #define CLRSYNC_UI_MANAGER_HPP
#include <string> #include <string>
#include <vector>
class font_loader;
struct ImFont;
namespace clrsync::gui::backend namespace clrsync::gui::backend
{ {
@@ -29,9 +33,25 @@ public:
void begin_frame(); void begin_frame();
void end_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: private:
backend::backend_interface *m_backend; backend::backend_interface *m_backend;
void *m_imgui_context = nullptr; void *m_imgui_context = nullptr;
font_loader *m_font_loader = nullptr;
}; };
} }

View File

@@ -1,11 +1,11 @@
#include "gui/views/about_window.hpp" #include "gui/views/about_window.hpp"
#include "core/common/version.hpp" #include "core/common/version.hpp"
#include "gui/helpers/imgui_helpers.hpp" #include "gui/widgets/centered_text.hpp"
#include "gui/widgets/colors.hpp"
#include "gui/widgets/link_button.hpp"
#include "imgui.h" #include "imgui.h"
about_window::about_window() about_window::about_window() = default;
{
}
void about_window::render(const clrsync::core::palette &pal) void about_window::render(const clrsync::core::palette &pal)
{ {
@@ -16,21 +16,13 @@ void about_window::render(const clrsync::core::palette &pal)
if (ImGui::Begin("About clrsync", &m_visible, ImGuiWindowFlags_NoResize)) if (ImGui::Begin("About clrsync", &m_visible, ImGuiWindowFlags_NoResize))
{ {
const float window_width = ImGui::GetContentRegionAvail().x; using namespace clrsync::gui::widgets;
ImGui::PushFont(ImGui::GetFont()); ImVec4 title_color = palette_color(pal, "info", "accent");
const char *title = "clrsync"; centered_text("clrsync", title_color);
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(); ImVec4 subtitle_color = palette_color(pal, "editor_inactive", "foreground");
const float version_size = ImGui::CalcTextSize(version.c_str()).x; centered_text("Version " + clrsync::core::version_string(), subtitle_color);
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::Spacing();
ImGui::Separator(); ImGui::Separator();
@@ -40,47 +32,25 @@ void about_window::render(const clrsync::core::palette &pal)
ImGui::Spacing(); ImGui::Spacing();
ImGui::Spacing(); ImGui::Spacing();
ImGui::Separator(); ImGui::Separator();
ImGui::Spacing(); ImGui::Spacing();
ImGui::Text("Links:"); constexpr float button_width = 200.0f;
float spacing = ImGui::GetStyle().ItemSpacing.x;
float total_width = 2.0f * button_width + spacing;
const float button_width = 200.0f; centered_buttons(total_width, [button_width]() {
const float spacing = ImGui::GetStyle().ItemSpacing.x; link_button("GitHub Repository", "https://github.com/obsqrbtz/clrsync", button_width);
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(); ImGui::SameLine();
link_button("Documentation", "https://binarygoose.dev/projects/clrsync/overview/", button_width);
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::Spacing(); ImGui::Spacing();
ImGui::Separator(); ImGui::Separator();
ImGui::Spacing(); ImGui::Spacing();
ImVec4 license_color = palette_utils::get_color(pal, "editor_inactive", "foreground"); ImGui::TextColored(subtitle_color, "MIT License");
ImGui::TextColored(license_color, "MIT License");
ImGui::TextWrapped( ImGui::TextWrapped(
"Copyright (c) 2025 Daniel Dada\n\n" "Copyright (c) 2025 Daniel Dada\n\n"
"Permission is hereby granted, free of charge, to any person obtaining a copy " "Permission is hereby granted, free of charge, to any person obtaining a copy "

View File

@@ -1,11 +1,13 @@
#include "color_scheme_editor.hpp" #include "color_scheme_editor.hpp"
#include "gui/controllers/theme_applier.hpp" #include "gui/controllers/theme_applier.hpp"
#include "gui/helpers/imgui_helpers.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 "imgui.h"
#include "settings_window.hpp" #include "settings_window.hpp"
#include "template_editor.hpp" #include "template_editor.hpp"
#include <iostream> #include <iostream>
#include <ranges>
color_scheme_editor::color_scheme_editor() color_scheme_editor::color_scheme_editor()
{ {
@@ -20,6 +22,8 @@ color_scheme_editor::color_scheme_editor()
{ {
std::cout << "WARNING: No palette loaded, skipping theme application\n"; std::cout << "WARNING: No palette loaded, skipping theme application\n";
} }
setup_widgets();
} }
void color_scheme_editor::notify_palette_changed() void color_scheme_editor::notify_palette_changed()
@@ -78,102 +82,22 @@ void color_scheme_editor::render_controls()
ImGui::Text("Palette:"); ImGui::Text("Palette:");
ImGui::SameLine(); ImGui::SameLine();
ImGui::SetNextItemWidth(200.0f); m_palette_selector.render(m_controller, 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::SameLine();
ImGui::SetCursorPosX(ImGui::GetCursorPosX() + 8); ImGui::SetCursorPosX(ImGui::GetCursorPosX() + 8);
static char new_palette_name_buf[128] = "";
if (ImGui::Button(" + New ")) if (ImGui::Button(" + New "))
{ {
new_palette_name_buf[0] = 0; m_new_palette_dialog.open("New Palette", "Enter a name for the new palette:", "Palette name...");
ImGui::OpenPopup("New Palette");
} }
if (ImGui::IsItemHovered()) if (ImGui::IsItemHovered())
ImGui::SetTooltip("Create a new palette"); ImGui::SetTooltip("Create a new palette");
if (ImGui::BeginPopupModal("New Palette", nullptr, ImGuiWindowFlags_AlwaysAutoResize)) m_new_palette_dialog.render();
{
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(); ImGui::SameLine();
m_action_buttons.render(current);
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) if (m_show_delete_confirmation)
{ {
@@ -181,21 +105,47 @@ void color_scheme_editor::render_controls()
m_show_delete_confirmation = false; m_show_delete_confirmation = false;
} }
palette_utils::render_delete_confirmation_popup("Delete Palette?", current.name(), "palette", clrsync::gui::widgets::delete_confirmation_dialog("Delete Palette?", current.name(), "palette",
current, [this]() { current, [this]() {
m_controller.delete_current_palette(); m_controller.delete_current_palette();
apply_themes(); 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); 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

@@ -4,6 +4,9 @@
#include "gui/controllers/palette_controller.hpp" #include "gui/controllers/palette_controller.hpp"
#include "gui/views/color_table_renderer.hpp" #include "gui/views/color_table_renderer.hpp"
#include "gui/views/preview_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 template_editor;
class settings_window; class settings_window;
@@ -32,6 +35,7 @@ class color_scheme_editor
void render_controls(); void render_controls();
void apply_themes(); void apply_themes();
void notify_palette_changed(); void notify_palette_changed();
void setup_widgets();
palette_controller m_controller; palette_controller m_controller;
color_table_renderer m_color_table; color_table_renderer m_color_table;
@@ -39,6 +43,10 @@ class color_scheme_editor
template_editor *m_template_editor{nullptr}; template_editor *m_template_editor{nullptr};
settings_window *m_settings_window{nullptr}; settings_window *m_settings_window{nullptr};
bool m_show_delete_confirmation{false}; 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 #endif // CLRSYNC_GUI_COLOR_SCHEME_EDITOR_HPP

View File

@@ -1,5 +1,5 @@
#include "gui/views/color_table_renderer.hpp" #include "gui/views/color_table_renderer.hpp"
#include "gui/helpers/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')
{ {
@@ -153,7 +157,7 @@ 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 | bool header_open = ImGui::TreeNodeEx(title, ImGuiTreeNodeFlags_DefaultOpen |
ImGuiTreeNodeFlags_SpanAvailWidth); ImGuiTreeNodeFlags_SpanAvailWidth);
ImGui::PopStyleColor(); ImGui::PopStyleColor();

View File

@@ -1,6 +1,6 @@
#include "gui/views/preview_renderer.hpp" #include "gui/views/preview_renderer.hpp"
#include "gui/controllers/theme_applier.hpp" #include "gui/controllers/theme_applier.hpp"
#include "gui/helpers/imgui_helpers.hpp" #include "gui/widgets/colors.hpp"
#include "imgui.h" #include "imgui.h"
#include <algorithm> #include <algorithm>
#include <array> #include <array>
@@ -236,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,25 +1,28 @@
#include "gui/views/settings_window.hpp" #include "gui/views/settings_window.hpp"
#include "core/common/error.hpp" #include "core/common/error.hpp"
#include "core/config/config.hpp" #include "core/config/config.hpp"
#include "gui/helpers/imgui_helpers.hpp" #include "gui/ui_manager.hpp"
#include "gui/platform/file_browser.hpp" #include "gui/widgets/section_header.hpp"
#include "gui/platform/font_loader.hpp"
#include "imgui.h" #include "imgui.h"
#include <cstring>
settings_window::settings_window() settings_window::settings_window(clrsync::gui::ui_manager *ui_mgr) : m_ui_manager(ui_mgr)
: m_font_size(14), m_selected_font_idx(0), m_settings_changed(false), m_current_tab(0)
{ {
m_default_theme[0] = '\0'; if (m_ui_manager)
m_palettes_path[0] = '\0'; m_available_fonts = m_ui_manager->get_system_fonts();
m_font[0] = '\0';
font_loader loader;
m_available_fonts = loader.get_system_fonts();
setup_widgets();
load_settings(); 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() void settings_window::render()
{ {
if (!m_visible) if (!m_visible)
@@ -29,8 +32,7 @@ void settings_window::render()
ImGui::SetNextWindowPos(ImGui::GetMainViewport()->GetCenter(), ImGuiCond_FirstUseEver, ImGui::SetNextWindowPos(ImGui::GetMainViewport()->GetCenter(), ImGuiCond_FirstUseEver,
ImVec2(0.5f, 0.5f)); ImVec2(0.5f, 0.5f));
ImGuiWindowFlags window_flags = ImGuiWindowFlags_NoCollapse; if (ImGui::Begin("Settings", &m_visible, ImGuiWindowFlags_NoCollapse))
if (ImGui::Begin("Settings", &m_visible, window_flags))
{ {
if (ImGui::BeginTabBar("SettingsTabs", ImGuiTabBarFlags_None)) if (ImGui::BeginTabBar("SettingsTabs", ImGuiTabBarFlags_None))
{ {
@@ -49,10 +51,33 @@ void settings_window::render()
ImGui::EndTabBar(); ImGui::EndTabBar();
} }
render_status_messages(); m_error.render(m_current_palette);
ImGui::Separator(); ImGui::Separator();
render_action_buttons();
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(); ImGui::End();
} }
@@ -61,31 +86,22 @@ void settings_window::load_settings()
{ {
auto &cfg = clrsync::core::config::instance(); auto &cfg = clrsync::core::config::instance();
std::string default_theme = cfg.default_theme(); m_default_theme = cfg.default_theme();
strncpy(m_default_theme, default_theme.c_str(), sizeof(m_default_theme) - 1); m_palettes_path = cfg.palettes_path();
m_default_theme[sizeof(m_default_theme) - 1] = '\0'; m_font = cfg.font();
m_font_size = cfg.font_size();
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; m_selected_font_idx = 0;
for (int i = 0; i < static_cast<int>(m_available_fonts.size()); i++) for (int i = 0; i < static_cast<int>(m_available_fonts.size()); i++)
{ {
if (m_available_fonts[i] == font) if (m_available_fonts[i] == m_font)
{ {
m_selected_font_idx = i; m_selected_font_idx = i;
break; break;
} }
} }
m_font_size = cfg.font_size(); m_error.clear();
m_error_message.clear();
m_settings_changed = false; m_settings_changed = false;
} }
@@ -93,286 +109,132 @@ void settings_window::apply_settings()
{ {
auto &cfg = clrsync::core::config::instance(); auto &cfg = clrsync::core::config::instance();
if (strlen(m_default_theme) == 0) if (m_default_theme.empty())
{ {
m_error_message = "Default theme cannot be empty"; m_error.set("Default theme cannot be empty");
return; return;
} }
if (strlen(m_palettes_path) == 0) if (m_palettes_path.empty())
{ {
m_error_message = "Palettes path cannot be empty"; m_error.set("Palettes path cannot be empty");
return; return;
} }
if (strlen(m_font) == 0) if (m_font.empty())
{ {
m_error_message = "Font cannot be empty"; m_error.set("Font cannot be empty");
return; return;
} }
if (m_font_size < 8 || m_font_size > 48) if (m_font_size < 8 || m_font_size > 48)
{ {
m_error_message = "Font size must be between 8 and 48"; m_error.set("Font size must be between 8 and 48");
return; return;
} }
auto result1 = cfg.set_default_theme(m_default_theme); auto result1 = cfg.set_default_theme(m_default_theme);
if (!result1) if (!result1)
{ {
m_error_message = "Failed to set default theme: " + result1.error().description(); m_error.set("Failed to set default theme: " + result1.error().description());
return; return;
} }
auto result2 = cfg.set_palettes_path(m_palettes_path); auto result2 = cfg.set_palettes_path(m_palettes_path);
if (!result2) if (!result2)
{ {
m_error_message = "Failed to set palettes path: " + result2.error().description(); m_error.set("Failed to set palettes path: " + result2.error().description());
return; return;
} }
auto result3 = cfg.set_font(m_font); auto result3 = cfg.set_font(m_font);
if (!result3) if (!result3)
{ {
m_error_message = "Failed to set font: " + result3.error().description(); m_error.set("Failed to set font: " + result3.error().description());
return; return;
} }
auto result4 = cfg.set_font_size(m_font_size); auto result4 = cfg.set_font_size(m_font_size);
if (!result4) if (!result4)
{ {
m_error_message = "Failed to set font size: " + result4.error().description(); m_error.set("Failed to set font size: " + result4.error().description());
return; return;
} }
font_loader fn_loader; if (m_ui_manager && !m_ui_manager->reload_font(m_font.c_str(), m_font_size))
auto font = fn_loader.load_font(m_font, m_font_size); {
if (font) m_error.set("Failed to load font: " + m_font);
ImGui::GetIO().FontDefault = font; return;
}
m_error_message.clear(); m_error.clear();
m_settings_changed = false; m_settings_changed = false;
} }
void settings_window::render_general_tab() void settings_window::render_general_tab()
{ {
ImGui::Spacing(); using namespace clrsync::gui::widgets;
auto accent_color = palette_utils::get_color(m_current_palette, "accent"); section_header("Theme Settings", m_current_palette);
ImGui::TextColored(accent_color, "Theme Settings");
ImGui::Separator();
ImGui::Spacing();
ImGui::Text("Default Theme:"); form_field_config theme_cfg;
ImGui::SameLine(); theme_cfg.label = "Default Theme";
show_help_marker("The default color scheme to load on startup"); theme_cfg.label_width = 150.0f;
ImGui::SetNextItemWidth(-100.0f); theme_cfg.tooltip = "The default color scheme to load on startup";
if (ImGui::InputText("##default_theme", m_default_theme, sizeof(m_default_theme))) theme_cfg.field_width = -100.0f;
if (m_form.render_text(theme_cfg, m_default_theme))
m_settings_changed = true; m_settings_changed = true;
ImGui::Spacing(); section_header("Path Settings", m_current_palette);
ImGui::TextColored(accent_color, "Path Settings"); form_field_config path_cfg;
ImGui::Separator(); path_cfg.label = "Palettes Directory";
ImGui::Spacing(); path_cfg.label_width = 150.0f;
path_cfg.tooltip = "Directory where color palettes are stored\nSupports ~ for home directory";
ImGui::Text("Palettes Directory:"); path_cfg.field_width = -1.0f;
ImGui::SameLine(); path_cfg.type = field_type::path;
show_help_marker("Directory where color palettes are stored\nSupports ~ for home directory"); if (m_form.render_path(path_cfg, m_palettes_path))
ImGui::SetNextItemWidth(-120.0f);
if (ImGui::InputText("##palettes_path", m_palettes_path, sizeof(m_palettes_path)))
m_settings_changed = true; 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() void settings_window::render_appearance_tab()
{ {
ImGui::Spacing(); using namespace clrsync::gui::widgets;
auto accent_color = palette_utils::get_color(m_current_palette, "accent"); section_header("Font Settings", m_current_palette);
ImGui::TextColored(accent_color, "Font Settings");
ImGui::Separator();
ImGui::Spacing();
ImGui::Text("Font Family:"); form_field_config font_cfg;
ImGui::SameLine(); font_cfg.label = "Font Family";
show_help_marker("Select font family for the application interface"); font_cfg.label_width = 150.0f;
ImGui::SetNextItemWidth(-1.0f); font_cfg.tooltip = "Select font family for the application interface";
if (ImGui::BeginCombo("##font", m_font)) font_cfg.field_width = -1.0f;
{ font_cfg.type = field_type::combo;
for (int i = 0; i < static_cast<int>(m_available_fonts.size()); i++) if (m_form.render_combo(font_cfg, m_available_fonts, m_selected_font_idx, m_font))
{
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; m_settings_changed = true;
}
if (is_selected)
ImGui::SetItemDefaultFocus();
}
ImGui::EndCombo();
}
ImGui::Spacing(); ImGui::Spacing();
ImGui::Text("Font Size:"); form_field_config size_cfg;
ImGui::SameLine(); size_cfg.label = "Font Size",
show_help_marker("Font size for the application interface (8-48)"); size_cfg.label_width = 150.0f;
ImGui::SetNextItemWidth(120.0f); size_cfg.tooltip = "Font size for the application interface (8-48)";
int old_size = m_font_size; size_cfg.field_width = 120.0f;
if (ImGui::SliderInt("##font_size", &m_font_size, 8, 48, "%d px")) size_cfg.type = field_type::slider;
{ size_cfg.min_value = 8.0f;
if (old_size != m_font_size) 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; 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() void settings_window::reset_to_defaults()
{ {
strncpy(m_default_theme, "dark", sizeof(m_default_theme)); m_default_theme = "dark";
strncpy(m_palettes_path, "~/.config/clrsync/palettes", sizeof(m_palettes_path)); m_palettes_path = "~/.config/clrsync/palettes";
strncpy(m_font, "JetBrains Mono Nerd Font", sizeof(m_font)); m_font = "JetBrains Mono Nerd Font";
m_font_size = 14; m_font_size = 14;
m_error_message.clear(); m_error.clear();
m_settings_changed = true; m_settings_changed = true;
} }

View File

@@ -2,59 +2,53 @@
#define CLRSYNC_GUI_SETTINGS_WINDOW_HPP #define CLRSYNC_GUI_SETTINGS_WINDOW_HPP
#include "core/palette/palette.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 <string>
#include <vector> #include <vector>
namespace clrsync::gui
{
class ui_manager;
}
class settings_window class settings_window
{ {
public: public:
settings_window(); settings_window(clrsync::gui::ui_manager* ui_mgr);
void render(); void render();
void show() void show() { m_visible = true; }
{ void hide() { m_visible = false; }
m_visible = true; bool is_visible() const { return m_visible; }
}
void hide() void set_palette(const clrsync::core::palette& palette) { m_current_palette = palette; }
{
m_visible = false;
}
bool is_visible() const
{
return m_visible;
}
private: private:
void load_settings(); void load_settings();
void save_settings();
void apply_settings(); void apply_settings();
void render_general_tab(); void render_general_tab();
void render_appearance_tab(); void render_appearance_tab();
void render_status_messages();
void render_action_buttons();
void show_help_marker(const char *desc);
void reset_to_defaults(); void reset_to_defaults();
void setup_widgets();
public:
void set_palette(const clrsync::core::palette &palette)
{
m_current_palette = palette;
}
bool m_visible{false}; bool m_visible{false};
bool m_settings_changed{false};
char m_default_theme[128]; std::string m_default_theme;
char m_palettes_path[512]; std::string m_palettes_path;
char m_font[128]; std::string m_font;
int m_font_size; int m_font_size{14};
int m_selected_font_idx{0};
std::vector<std::string> m_available_fonts; 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; 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 #endif // CLRSYNC_GUI_SETTINGS_WINDOW_HPP

View File

@@ -3,8 +3,9 @@
#include "core/config/config.hpp" #include "core/config/config.hpp"
#include "core/palette/color_keys.hpp" #include "core/palette/color_keys.hpp"
#include "core/theme/theme_template.hpp" #include "core/theme/theme_template.hpp"
#include "gui/helpers/imgui_helpers.hpp" #include "gui/widgets/colors.hpp"
#include "gui/platform/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>
@@ -18,8 +19,11 @@ const std::vector<std::string> COLOR_FORMATS = {
"l", "hsl", "hsla"}; "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);
@@ -47,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)
@@ -56,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();
@@ -99,14 +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 = m_autocomplete_dim_text_color =
palette_utils::get_color(pal, "on_surface_variant", "editor_inactive"); clrsync::gui::widgets::palette_color(pal, "on_surface_variant", "editor_inactive");
} }
void template_editor::update_autocomplete_suggestions() void template_editor::update_autocomplete_suggestions()
@@ -358,9 +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, [this]() { "Delete Template?", m_control_state.name, "template", m_current_palette, [this]() {
bool success = m_template_controller.remove_template(m_template_name); bool success = m_template_controller.remove_template(m_control_state.name);
if (success) if (success)
{ {
new_template(); new_template();
@@ -368,7 +399,7 @@ void template_editor::render()
} }
else else
{ {
m_validation_error = "Failed to delete template"; m_validation.set("Failed to delete template");
} }
}); });
@@ -377,225 +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);
@@ -608,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();
@@ -695,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);
@@ -710,11 +545,11 @@ 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( ImVec4 disabled_color = clrsync::gui::widgets::palette_color(
m_current_palette, "on_surface_variant", "editor_inactive"); m_current_palette, "on_surface_variant", "editor_inactive");
ImGui::PushStyleColor(ImGuiCol_Text, disabled_color); ImGui::PushStyleColor(ImGuiCol_Text, disabled_color);
} }
@@ -795,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();
@@ -847,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;
} }
} }
@@ -857,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;
} }
@@ -865,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;
@@ -893,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())
@@ -918,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

@@ -4,14 +4,21 @@
#include "color_text_edit/TextEditor.h" #include "color_text_edit/TextEditor.h"
#include "core/palette/palette.hpp" #include "core/palette/palette.hpp"
#include "gui/controllers/template_controller.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 @@ class template_editor
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 @@ class template_editor
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

View File

@@ -0,0 +1,111 @@
#include "input_dialog.hpp"
#include "imgui.h"
#include <cstring>
namespace clrsync::gui::widgets
{
input_dialog::input_dialog() : m_input_buffer{0}, m_is_open(false), m_focus_input(false) {}
void input_dialog::open(const std::string &title, const std::string &prompt, const std::string &hint)
{
m_title = title;
m_prompt = prompt;
m_hint = hint;
m_input_buffer[0] = 0;
m_is_open = true;
m_focus_input = true;
ImGui::OpenPopup(m_title.c_str());
}
bool input_dialog::render()
{
bool submitted = false;
if (!m_is_open)
return false;
if (ImGui::BeginPopupModal(m_title.c_str(), &m_is_open, ImGuiWindowFlags_AlwaysAutoResize))
{
ImGui::Text("%s", m_prompt.c_str());
ImGui::Spacing();
ImGui::SetNextItemWidth(250);
if (m_focus_input)
{
ImGui::SetKeyboardFocusHere();
m_focus_input = false;
}
bool enter_pressed = ImGui::InputTextWithHint("##input", m_hint.c_str(), m_input_buffer,
IM_ARRAYSIZE(m_input_buffer),
ImGuiInputTextFlags_EnterReturnsTrue);
ImGui::Spacing();
ImGui::Separator();
ImGui::Spacing();
bool can_submit = strlen(m_input_buffer) > 0;
bool submit_clicked = false;
if (!can_submit)
ImGui::BeginDisabled();
if (ImGui::Button("OK", ImVec2(120, 0)) || (enter_pressed && can_submit))
{
if (m_on_submit)
{
m_on_submit(m_input_buffer);
}
submitted = true;
submit_clicked = true;
m_is_open = false;
ImGui::CloseCurrentPopup();
}
if (!can_submit)
ImGui::EndDisabled();
ImGui::SameLine();
if (ImGui::Button("Cancel", ImVec2(120, 0)))
{
if (m_on_cancel)
{
m_on_cancel();
}
m_is_open = false;
ImGui::CloseCurrentPopup();
}
if (ImGui::IsKeyPressed(ImGuiKey_Escape))
{
if (m_on_cancel)
{
m_on_cancel();
}
m_is_open = false;
ImGui::CloseCurrentPopup();
}
ImGui::EndPopup();
}
else
{
m_is_open = false;
}
return submitted;
}
void input_dialog::set_on_submit(const std::function<void(const std::string &)> &callback)
{
m_on_submit = callback;
}
void input_dialog::set_on_cancel(const std::function<void()> &callback)
{
m_on_cancel = callback;
}
} // namespace clrsync::gui::widgets

View File

@@ -0,0 +1,38 @@
#ifndef CLRSYNC_GUI_WIDGETS_INPUT_DIALOG_HPP
#define CLRSYNC_GUI_WIDGETS_INPUT_DIALOG_HPP
#include <functional>
#include <string>
namespace clrsync::gui::widgets
{
class input_dialog
{
public:
input_dialog();
void open(const std::string &title, const std::string &prompt, const std::string &hint = "");
bool render();
void set_on_submit(const std::function<void(const std::string &)> &callback);
void set_on_cancel(const std::function<void()> &callback);
bool is_open() const { return m_is_open; }
private:
std::string m_title;
std::string m_prompt;
std::string m_hint;
char m_input_buffer[256];
bool m_is_open;
bool m_focus_input;
std::function<void(const std::string &)> m_on_submit;
std::function<void()> m_on_cancel;
};
} // namespace clrsync::gui::widgets
#endif // CLRSYNC_GUI_WIDGETS_INPUT_DIALOG_HPP

View File

@@ -0,0 +1,28 @@
#include "link_button.hpp"
#include "imgui.h"
#include <cstdlib>
namespace clrsync::gui::widgets
{
void open_url(const std::string& url)
{
#ifdef _WIN32
std::string cmd = "start " + url;
#elif __APPLE__
std::string cmd = "open " + url;
#else
std::string cmd = "xdg-open " + url;
#endif
std::system(cmd.c_str());
}
bool link_button(const std::string& label, const std::string& url, float width)
{
bool clicked = ImGui::Button(label.c_str(), ImVec2(width, 0));
if (clicked)
open_url(url);
return clicked;
}
} // namespace clrsync::gui::widgets

View File

@@ -0,0 +1,14 @@
#ifndef CLRSYNC_GUI_WIDGETS_LINK_BUTTON_HPP
#define CLRSYNC_GUI_WIDGETS_LINK_BUTTON_HPP
#include <string>
namespace clrsync::gui::widgets
{
void open_url(const std::string& url);
bool link_button(const std::string& label, const std::string& url, float width = 0.0f);
} // namespace clrsync::gui::widgets
#endif // CLRSYNC_GUI_WIDGETS_LINK_BUTTON_HPP

View File

@@ -0,0 +1,46 @@
#include "palette_selector.hpp"
#include "imgui.h"
#include <ranges>
namespace clrsync::gui::widgets
{
palette_selector::palette_selector() = default;
bool palette_selector::render(const palette_controller &controller, float width)
{
const auto &current = controller.current_palette();
const auto &palettes = controller.palettes();
bool selection_changed = false;
ImGui::SetNextItemWidth(width);
if (ImGui::BeginCombo("##palette", 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))
{
if (m_on_selection_changed)
{
m_on_selection_changed(name);
}
selection_changed = true;
}
if (selected)
ImGui::SetItemDefaultFocus();
}
ImGui::EndCombo();
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Select a color palette to edit");
return selection_changed;
}
void palette_selector::set_on_selection_changed(const std::function<void(const std::string &)> &callback)
{
m_on_selection_changed = callback;
}
} // namespace clrsync::gui::widgets

View File

@@ -0,0 +1,26 @@
#ifndef CLRSYNC_GUI_WIDGETS_PALETTE_SELECTOR_HPP
#define CLRSYNC_GUI_WIDGETS_PALETTE_SELECTOR_HPP
#include "gui/controllers/palette_controller.hpp"
#include <functional>
#include <string>
namespace clrsync::gui::widgets
{
class palette_selector
{
public:
palette_selector();
bool render(const palette_controller &controller, float width = 200.0f);
void set_on_selection_changed(const std::function<void(const std::string &)> &callback);
private:
std::function<void(const std::string &)> m_on_selection_changed;
};
} // namespace clrsync::gui::widgets
#endif // CLRSYNC_GUI_WIDGETS_PALETTE_SELECTOR_HPP

View File

@@ -0,0 +1,17 @@
#include "section_header.hpp"
#include "colors.hpp"
#include "imgui.h"
namespace clrsync::gui::widgets
{
void section_header(const std::string& title, const core::palette& palette)
{
ImGui::Spacing();
auto accent_color = palette_color(palette, "accent");
ImGui::TextColored(accent_color, "%s", title.c_str());
ImGui::Separator();
ImGui::Spacing();
}
} // namespace clrsync::gui::widgets

View File

@@ -0,0 +1,14 @@
#ifndef CLRSYNC_GUI_WIDGETS_SECTION_HEADER_HPP
#define CLRSYNC_GUI_WIDGETS_SECTION_HEADER_HPP
#include "core/palette/palette.hpp"
#include <string>
namespace clrsync::gui::widgets
{
void section_header(const std::string& title, const core::palette& palette);
} // namespace clrsync::gui::widgets
#endif // CLRSYNC_GUI_WIDGETS_SECTION_HEADER_HPP

View File

@@ -0,0 +1,56 @@
#include "settings_buttons.hpp"
#include "imgui.h"
namespace clrsync::gui::widgets
{
void settings_buttons::render(const settings_buttons_callbacks& callbacks, bool apply_enabled)
{
ImGui::Spacing();
float spacing = ImGui::GetStyle().ItemSpacing.x;
float window_width = ImGui::GetContentRegionAvail().x;
float total_buttons_width = 4 * m_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(m_button_width, 0)))
{
if (callbacks.on_ok)
callbacks.on_ok();
}
ImGui::SameLine();
if (!apply_enabled)
ImGui::PushStyleVar(ImGuiStyleVar_Alpha, 0.5f);
if (ImGui::Button("Apply", ImVec2(m_button_width, 0)) && apply_enabled)
{
if (callbacks.on_apply)
callbacks.on_apply();
}
if (!apply_enabled)
ImGui::PopStyleVar();
ImGui::SameLine();
if (ImGui::Button("Reset", ImVec2(m_button_width, 0)))
{
if (callbacks.on_reset)
callbacks.on_reset();
}
ImGui::SameLine();
if (ImGui::Button("Cancel", ImVec2(m_button_width, 0)))
{
if (callbacks.on_cancel)
callbacks.on_cancel();
}
}
} // namespace clrsync::gui::widgets

View File

@@ -0,0 +1,30 @@
#ifndef CLRSYNC_GUI_WIDGETS_SETTINGS_BUTTONS_HPP
#define CLRSYNC_GUI_WIDGETS_SETTINGS_BUTTONS_HPP
#include <functional>
namespace clrsync::gui::widgets
{
struct settings_buttons_callbacks
{
std::function<void()> on_ok;
std::function<void()> on_apply;
std::function<void()> on_reset;
std::function<void()> on_cancel;
};
class settings_buttons
{
public:
void render(const settings_buttons_callbacks& callbacks, bool apply_enabled);
void set_button_width(float width) { m_button_width = width; }
private:
float m_button_width = 100.0f;
};
} // namespace clrsync::gui::widgets
#endif // CLRSYNC_GUI_WIDGETS_SETTINGS_BUTTONS_HPP

View File

@@ -0,0 +1,95 @@
#include "styled_checkbox.hpp"
#include "colors.hpp"
#include "imgui.h"
namespace clrsync::gui::widgets
{
styled_checkbox::styled_checkbox() = default;
bool styled_checkbox::render(const std::string &label, bool *value, const core::palette &theme_palette,
checkbox_style style)
{
ImVec4 bg_color, hover_color, check_color;
switch (style)
{
case checkbox_style::success:
if (*value)
{
bg_color = palette_color(theme_palette, "success", "accent");
bg_color.w = 0.5f;
hover_color = ImVec4(bg_color.x * 1.2f, bg_color.y * 1.2f, bg_color.z * 1.2f, 0.6f);
check_color = palette_color(theme_palette, "on_success", "on_surface");
}
else
{
bg_color = palette_color(theme_palette, "surface", "background");
hover_color = palette_color(theme_palette, "surface_variant", "surface");
check_color = palette_color(theme_palette, "on_surface", "foreground");
}
break;
case checkbox_style::error:
if (*value)
{
bg_color = palette_color(theme_palette, "error", "accent");
bg_color.w = 0.5f;
hover_color = ImVec4(bg_color.x * 1.2f, bg_color.y * 1.2f, bg_color.z * 1.2f, 0.6f);
check_color = palette_color(theme_palette, "on_error", "on_surface");
}
else
{
bg_color = palette_color(theme_palette, "error", "accent");
bg_color.w = 0.2f;
hover_color = ImVec4(bg_color.x, bg_color.y, bg_color.z, 0.3f);
check_color = palette_color(theme_palette, "on_error", "on_surface");
}
break;
case checkbox_style::warning:
if (*value)
{
bg_color = palette_color(theme_palette, "warning", "accent");
bg_color.w = 0.5f;
hover_color = ImVec4(bg_color.x * 1.2f, bg_color.y * 1.2f, bg_color.z * 1.2f, 0.6f);
check_color = palette_color(theme_palette, "on_warning", "on_surface");
}
else
{
bg_color = palette_color(theme_palette, "surface", "background");
hover_color = palette_color(theme_palette, "surface_variant", "surface");
check_color = palette_color(theme_palette, "on_surface", "foreground");
}
break;
case checkbox_style::normal:
default:
bg_color = palette_color(theme_palette, "surface", "background");
hover_color = palette_color(theme_palette, "surface_variant", "surface");
check_color = palette_color(theme_palette, "accent", "foreground");
break;
}
ImGui::PushStyleColor(ImGuiCol_FrameBg, bg_color);
ImGui::PushStyleColor(ImGuiCol_FrameBgHovered, hover_color);
ImGui::PushStyleColor(ImGuiCol_CheckMark, check_color);
bool changed = ImGui::Checkbox(label.c_str(), value);
ImGui::PopStyleColor(3);
if (changed && m_on_changed)
{
m_on_changed(*value);
}
if (!m_tooltip.empty() && ImGui::IsItemHovered())
{
ImGui::SetTooltip("%s", m_tooltip.c_str());
}
return changed;
}
} // namespace clrsync::gui::widgets

View File

@@ -0,0 +1,38 @@
#ifndef CLRSYNC_GUI_WIDGETS_STYLED_CHECKBOX_HPP
#define CLRSYNC_GUI_WIDGETS_STYLED_CHECKBOX_HPP
#include "core/palette/palette.hpp"
#include <functional>
#include <string>
namespace clrsync::gui::widgets
{
enum class checkbox_style
{
normal,
success,
error,
warning
};
class styled_checkbox
{
public:
styled_checkbox();
bool render(const std::string &label, bool *value, const core::palette &theme_palette,
checkbox_style style = checkbox_style::normal);
void set_tooltip(const std::string &tooltip) { m_tooltip = tooltip; }
void set_on_changed(const std::function<void(bool)> &callback) { m_on_changed = callback; }
private:
std::string m_tooltip;
std::function<void(bool)> m_on_changed;
};
} // namespace clrsync::gui::widgets
#endif // CLRSYNC_GUI_WIDGETS_STYLED_CHECKBOX_HPP

View File

@@ -0,0 +1,181 @@
#include "template_controls.hpp"
#include "colors.hpp"
#include "styled_checkbox.hpp"
#include "imgui.h"
namespace clrsync::gui::widgets
{
template_controls::template_controls() = default;
void template_controls::render(template_control_state& state,
const template_control_callbacks& callbacks,
const core::palette& palette,
validation_message& validation)
{
ImGui::PushStyleVar(ImGuiStyleVar_ItemSpacing, ImVec2(4, 8));
render_action_buttons(state, callbacks, palette);
ImGui::PopStyleVar();
ImGui::Spacing();
render_fields(state, callbacks);
validation.render(palette);
}
void template_controls::render_action_buttons(template_control_state& state,
const template_control_callbacks& callbacks,
const core::palette& palette)
{
if (ImGui::Button(" + New "))
{
if (callbacks.on_new)
callbacks.on_new();
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Create a new template");
ImGui::SameLine();
if (ImGui::Button(" Save "))
{
if (callbacks.on_save)
callbacks.on_save();
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Save template (Ctrl+S)");
if (state.is_editing_existing)
{
ImGui::SameLine();
auto error = palette_color(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(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 "))
{
if (callbacks.on_delete)
callbacks.on_delete();
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Delete this template");
ImGui::PopStyleColor(4);
}
ImGui::SameLine();
ImGui::SetCursorPosX(ImGui::GetCursorPosX() + 10);
bool old_enabled = state.enabled;
styled_checkbox checkbox;
checkbox.render("Enabled", &state.enabled, palette,
state.enabled ? checkbox_style::success : checkbox_style::error);
if (old_enabled != state.enabled && state.is_editing_existing)
{
if (callbacks.on_enabled_changed)
callbacks.on_enabled_changed(state.enabled);
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Enable/disable this template for theme application");
}
void template_controls::render_fields(template_control_state& state,
const template_control_callbacks& callbacks)
{
ImGui::AlignTextToFramePadding();
ImGui::Text("Name:");
ImGui::SameLine(80);
ImGui::SetNextItemWidth(180.0f);
char name_buf[256] = {0};
snprintf(name_buf, sizeof(name_buf), "%s", state.name.c_str());
if (ImGui::InputText("##template_name", name_buf, sizeof(name_buf)))
{
state.name = name_buf;
}
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", state.input_path.c_str());
if (ImGui::InputTextWithHint("##input_path", "Path to template file...", input_path_buf,
sizeof(input_path_buf)))
{
state.input_path = input_path_buf;
if (state.is_editing_existing && callbacks.on_input_path_changed)
callbacks.on_input_path_changed(state.input_path);
}
ImGui::SameLine();
if (ImGui::Button("Browse##input"))
{
if (callbacks.on_browse_input)
{
std::string selected = callbacks.on_browse_input(state.input_path);
if (!selected.empty())
{
state.input_path = selected;
if (state.is_editing_existing && callbacks.on_input_path_changed)
callbacks.on_input_path_changed(state.input_path);
}
}
}
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", state.output_path.c_str());
if (ImGui::InputTextWithHint("##output_path", "Path for generated config...", path_buf,
sizeof(path_buf)))
{
state.output_path = path_buf;
if (state.is_editing_existing && callbacks.on_output_path_changed)
callbacks.on_output_path_changed(state.output_path);
}
ImGui::SameLine();
if (ImGui::Button("Browse##output"))
{
if (callbacks.on_browse_output)
{
std::string selected = callbacks.on_browse_output(state.output_path);
if (!selected.empty())
{
state.output_path = selected;
if (state.is_editing_existing && callbacks.on_output_path_changed)
callbacks.on_output_path_changed(state.output_path);
}
}
}
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", state.reload_command.c_str());
if (ImGui::InputTextWithHint("##reload_cmd", "Command to reload app (optional)...", reload_buf,
sizeof(reload_buf)))
{
state.reload_command = reload_buf;
if (state.is_editing_existing && callbacks.on_reload_command_changed)
callbacks.on_reload_command_changed(state.reload_command);
}
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Shell command to run after applying theme (e.g., 'pkill -USR1 kitty')");
}
} // namespace clrsync::gui::widgets

View File

@@ -0,0 +1,59 @@
#ifndef CLRSYNC_GUI_WIDGETS_TEMPLATE_CONTROLS_HPP
#define CLRSYNC_GUI_WIDGETS_TEMPLATE_CONTROLS_HPP
#include "core/palette/palette.hpp"
#include "gui/widgets/form_field.hpp"
#include "gui/widgets/validation_message.hpp"
#include <functional>
#include <string>
namespace clrsync::gui::widgets
{
struct template_control_state
{
std::string name;
std::string input_path;
std::string output_path;
std::string reload_command;
bool enabled{true};
bool is_editing_existing{false};
};
struct template_control_callbacks
{
std::function<void()> on_new;
std::function<void()> on_save;
std::function<void()> on_delete;
std::function<void(bool)> on_enabled_changed;
std::function<std::string(const std::string&)> on_browse_input;
std::function<std::string(const std::string&)> on_browse_output;
std::function<void(const std::string&)> on_input_path_changed;
std::function<void(const std::string&)> on_output_path_changed;
std::function<void(const std::string&)> on_reload_command_changed;
};
class template_controls
{
public:
template_controls();
void render(template_control_state& state,
const template_control_callbacks& callbacks,
const core::palette& palette,
validation_message& validation);
private:
void render_action_buttons(template_control_state& state,
const template_control_callbacks& callbacks,
const core::palette& palette);
void render_fields(template_control_state& state,
const template_control_callbacks& callbacks);
form_field m_form;
};
} // namespace clrsync::gui::widgets
#endif // CLRSYNC_GUI_WIDGETS_TEMPLATE_CONTROLS_HPP

View File

@@ -0,0 +1,40 @@
#include "validation_message.hpp"
#include "colors.hpp"
#include "imgui.h"
namespace clrsync::gui::widgets
{
void validation_message::set(const std::string& message)
{
m_message = message;
}
void validation_message::clear()
{
m_message.clear();
}
bool validation_message::has_error() const
{
return !m_message.empty();
}
const std::string& validation_message::get() const
{
return m_message;
}
void validation_message::render(const core::palette& palette)
{
if (m_message.empty())
return;
ImGui::Spacing();
ImVec4 error_color = palette_color(palette, "error", "accent");
ImGui::PushStyleColor(ImGuiCol_Text, error_color);
ImGui::TextWrapped("%s", m_message.c_str());
ImGui::PopStyleColor();
}
} // namespace clrsync::gui::widgets

View File

@@ -0,0 +1,25 @@
#ifndef CLRSYNC_GUI_WIDGETS_VALIDATION_MESSAGE_HPP
#define CLRSYNC_GUI_WIDGETS_VALIDATION_MESSAGE_HPP
#include "core/palette/palette.hpp"
#include <string>
namespace clrsync::gui::widgets
{
class validation_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_VALIDATION_MESSAGE_HPP