chore: structured src/gui, run clang-format

This commit is contained in:
2025-12-18 13:23:50 +03:00
parent 613c2c80f5
commit 7641846600
55 changed files with 1647 additions and 1399 deletions

Binary file not shown.

Before

Width:  |  Height:  |  Size: 370 KiB

After

Width:  |  Height:  |  Size: 64 KiB

View File

@@ -1,6 +1,6 @@
#include "config.hpp" #include "config.hpp"
#include "core/utils.hpp"
#include "core/error.hpp" #include "core/error.hpp"
#include "core/utils.hpp"
#include <core/palette/color.hpp> #include <core/palette/color.hpp>
#include <filesystem> #include <filesystem>
@@ -28,7 +28,8 @@ Result<void> config::initialize(std::unique_ptr<clrsync::core::io::file> file)
auto parse_result = m_file->parse(); auto parse_result = m_file->parse();
if (!parse_result) if (!parse_result)
return Err<void>(error_code::config_invalid, parse_result.error().message, parse_result.error().context); return Err<void>(error_code::config_invalid, parse_result.error().message,
parse_result.error().context);
return Ok(); return Ok();
} }
@@ -223,10 +224,14 @@ Result<void> config::remove_template(const std::string &key)
std::filesystem::path template_file = it->second.template_path(); std::filesystem::path template_file = it->second.template_path();
if (std::filesystem::exists(template_file)) if (std::filesystem::exists(template_file))
{ {
try { try
{
std::filesystem::remove(template_file); std::filesystem::remove(template_file);
} catch (const std::exception& e) { }
return Err<void>(error_code::file_write_failed, "Failed to delete template file", e.what()); catch (const std::exception &e)
{
return Err<void>(error_code::file_write_failed, "Failed to delete template file",
e.what());
} }
} }
@@ -264,14 +269,16 @@ const std::unordered_map<std::string, clrsync::core::theme_template> config::tem
return m_themes; return m_themes;
} }
Result<const clrsync::core::theme_template*> config::template_by_name(const std::string &name) const Result<const clrsync::core::theme_template *> config::template_by_name(
const std::string &name) const
{ {
auto it = m_themes.find(name); auto it = m_themes.find(name);
if (it != m_themes.end()) if (it != m_themes.end())
{ {
return Ok(&it->second); return Ok(&it->second);
} }
return Err<const clrsync::core::theme_template*>(error_code::template_not_found, "Template not found", name); return Err<const clrsync::core::theme_template *>(error_code::template_not_found,
"Template not found", name);
} }
} // namespace clrsync::core } // namespace clrsync::core

View File

@@ -1,9 +1,9 @@
#ifndef CLRSYNC_CORE_CONFIG_HPP #ifndef CLRSYNC_CORE_CONFIG_HPP
#define CLRSYNC_CORE_CONFIG_HPP #define CLRSYNC_CORE_CONFIG_HPP
#include <core/error.hpp>
#include <core/io/file.hpp> #include <core/io/file.hpp>
#include <core/theme/theme_template.hpp> #include <core/theme/theme_template.hpp>
#include <core/error.hpp>
#include <filesystem> #include <filesystem>
#include <memory> #include <memory>
#include <string> #include <string>
@@ -25,7 +25,6 @@ class config
Result<const clrsync::core::theme_template *> template_by_name(const std::string &name) const; Result<const clrsync::core::theme_template *> template_by_name(const std::string &name) const;
std::filesystem::path get_user_config_dir(); std::filesystem::path get_user_config_dir();
Result<void> set_default_theme(const std::string &theme); Result<void> set_default_theme(const std::string &theme);
Result<void> set_palettes_path(const std::string &path); Result<void> set_palettes_path(const std::string &path);
Result<void> set_font(const std::string &font); Result<void> set_font(const std::string &font);

View File

@@ -1,9 +1,9 @@
#ifndef CLRSYNC_CORE_ERROR_HPP #ifndef CLRSYNC_CORE_ERROR_HPP
#define CLRSYNC_CORE_ERROR_HPP #define CLRSYNC_CORE_ERROR_HPP
#include <optional>
#include <string> #include <string>
#include <variant> #include <variant>
#include <optional>
namespace clrsync::core namespace clrsync::core
{ {
@@ -40,25 +40,44 @@ inline const char* error_code_string(error_code code)
{ {
switch (code) switch (code)
{ {
case error_code::unknown: return "Unknown error"; case error_code::unknown:
case error_code::file_not_found: return "File not found"; return "Unknown error";
case error_code::file_open_failed: return "Failed to open file"; case error_code::file_not_found:
case error_code::file_write_failed: return "Failed to write file"; return "File not found";
case error_code::file_read_failed: return "Failed to read file"; case error_code::file_open_failed:
case error_code::dir_create_failed: return "Failed to create directory"; return "Failed to open file";
case error_code::parse_failed: return "Parse failed"; case error_code::file_write_failed:
case error_code::invalid_format: return "Invalid format"; return "Failed to write file";
case error_code::config_missing: return "Configuration missing"; case error_code::file_read_failed:
case error_code::config_invalid: return "Configuration invalid"; return "Failed to read file";
case error_code::template_not_found: return "Template not found"; case error_code::dir_create_failed:
case error_code::template_load_failed: return "Failed to load template"; return "Failed to create directory";
case error_code::template_apply_failed: return "Failed to apply template"; case error_code::parse_failed:
case error_code::palette_not_found: return "Palette not found"; return "Parse failed";
case error_code::palette_load_failed: return "Failed to load palette"; case error_code::invalid_format:
case error_code::init_failed: return "Initialization failed"; return "Invalid format";
case error_code::invalid_arg: return "Invalid argument"; case error_code::config_missing:
case error_code::resource_missing: return "Resource missing"; return "Configuration missing";
default: return "Unknown error code"; case error_code::config_invalid:
return "Configuration invalid";
case error_code::template_not_found:
return "Template not found";
case error_code::template_load_failed:
return "Failed to load template";
case error_code::template_apply_failed:
return "Failed to apply template";
case error_code::palette_not_found:
return "Palette not found";
case error_code::palette_load_failed:
return "Failed to load palette";
case error_code::init_failed:
return "Initialization failed";
case error_code::invalid_arg:
return "Invalid argument";
case error_code::resource_missing:
return "Resource missing";
default:
return "Unknown error code";
} }
} }
@@ -68,13 +87,18 @@ struct Error
std::string message; std::string message;
std::string context; std::string context;
Error(error_code c) : code(c), message(error_code_string(c)) {} Error(error_code c) : code(c), message(error_code_string(c))
{
}
Error(error_code c, std::string msg) Error(error_code c, std::string msg) : code(c), message(std::move(msg))
: code(c), message(std::move(msg)) {} {
}
Error(error_code c, std::string msg, std::string ctx) Error(error_code c, std::string msg, std::string ctx)
: code(c), message(std::move(msg)), context(std::move(ctx)) {} : code(c), message(std::move(msg)), context(std::move(ctx))
{
}
std::string description() const std::string description() const
{ {
@@ -84,28 +108,52 @@ struct Error
} }
}; };
template<typename T> template <typename T> class [[nodiscard]] Result
class [[nodiscard]] Result
{ {
private: private:
std::variant<T, Error> m_data; std::variant<T, Error> m_data;
public: public:
Result(T value) : m_data(std::move(value)) {} Result(T value) : m_data(std::move(value))
{
}
Result(Error error) : m_data(std::move(error)) {} Result(Error error) : m_data(std::move(error))
{
}
bool is_ok() const { return std::holds_alternative<T>(m_data); } bool is_ok() const
{
return std::holds_alternative<T>(m_data);
}
bool is_error() const { return std::holds_alternative<Error>(m_data); } bool is_error() const
{
return std::holds_alternative<Error>(m_data);
}
explicit operator bool() const { return is_ok(); } explicit operator bool() const
{
return is_ok();
}
T& value() & { return std::get<T>(m_data); } T &value() &
const T& value() const & { return std::get<T>(m_data); } {
T&& value() && { return std::get<T>(std::move(m_data)); } return std::get<T>(m_data);
}
const T &value() const &
{
return std::get<T>(m_data);
}
T &&value() &&
{
return std::get<T>(std::move(m_data));
}
const Error& error() const { return std::get<Error>(m_data); } const Error &error() const
{
return std::get<Error>(m_data);
}
T value_or(T default_value) const T value_or(T default_value) const
{ {
@@ -126,8 +174,7 @@ public:
return std::nullopt; return std::nullopt;
} }
template<typename F> template <typename F> auto map(F &&func) -> Result<decltype(func(std::declval<T>()))>
auto map(F&& func) -> Result<decltype(func(std::declval<T>()))>
{ {
using U = decltype(func(std::declval<T>())); using U = decltype(func(std::declval<T>()));
if (is_ok()) if (is_ok())
@@ -135,8 +182,7 @@ public:
return Result<U>(std::get<Error>(m_data)); return Result<U>(std::get<Error>(m_data));
} }
template<typename F> template <typename F> auto and_then(F &&func) -> decltype(func(std::declval<T>()))
auto and_then(F&& func) -> decltype(func(std::declval<T>()))
{ {
if (is_ok()) if (is_ok())
return func(std::get<T>(m_data)); return func(std::get<T>(m_data));
@@ -145,30 +191,47 @@ public:
} }
}; };
template<> template <> class [[nodiscard]] Result<void>
class [[nodiscard]] Result<void>
{ {
private: private:
std::optional<Error> m_error; std::optional<Error> m_error;
public: public:
Result() : m_error(std::nullopt) {} Result() : m_error(std::nullopt)
{
}
Result(Error error) : m_error(std::move(error)) {} Result(Error error) : m_error(std::move(error))
{
}
bool is_ok() const { return !m_error.has_value(); } bool is_ok() const
{
return !m_error.has_value();
}
bool is_error() const { return m_error.has_value(); } bool is_error() const
{
return m_error.has_value();
}
explicit operator bool() const { return is_ok(); } explicit operator bool() const
{
return is_ok();
}
const Error& error() const { return *m_error; } const Error &error() const
{
return *m_error;
}
std::optional<Error> err() const { return m_error; } std::optional<Error> err() const
{
return m_error;
}
}; };
template<typename T> template <typename T> Result<T> Ok(T value)
Result<T> Ok(T value)
{ {
return Result<T>(std::move(value)); return Result<T>(std::move(value));
} }
@@ -178,26 +241,22 @@ inline Result<void> Ok()
return Result<void>(); return Result<void>();
} }
template<typename T> template <typename T> Result<T> Err(Error error)
Result<T> Err(Error error)
{ {
return Result<T>(std::move(error)); return Result<T>(std::move(error));
} }
template<typename T> template <typename T> Result<T> Err(error_code code)
Result<T> Err(error_code code)
{ {
return Result<T>(Error(code)); return Result<T>(Error(code));
} }
template<typename T> template <typename T> Result<T> Err(error_code code, std::string message)
Result<T> Err(error_code code, std::string message)
{ {
return Result<T>(Error(code, std::move(message))); return Result<T>(Error(code, std::move(message)));
} }
template<typename T> template <typename T> Result<T> Err(error_code code, std::string message, std::string context)
Result<T> Err(error_code code, std::string message, std::string context)
{ {
return Result<T>(Error(code, std::move(message), std::move(context))); return Result<T>(Error(code, std::move(message), std::move(context)));
} }

View File

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

View File

@@ -113,9 +113,12 @@ void toml_file::remove_section(const std::string &section)
Result<void> toml_file::save_file() Result<void> toml_file::save_file()
{ {
try { try
{
std::filesystem::create_directories(std::filesystem::path(m_path).parent_path()); std::filesystem::create_directories(std::filesystem::path(m_path).parent_path());
} catch (const std::exception& e) { }
catch (const std::exception &e)
{
return Err<void>(error_code::dir_create_failed, e.what(), m_path); return Err<void>(error_code::dir_create_failed, e.what(), m_path);
} }

View File

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

View File

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

View File

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

View File

@@ -6,12 +6,10 @@
#include <string> #include <string>
#include <unordered_map> #include <unordered_map>
namespace clrsync::core namespace clrsync::core
{ {
template <typename FileType> template <typename FileType> class template_manager
class template_manager
{ {
public: public:
template_manager() = default; template_manager() = default;

View File

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

View File

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

View File

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

View File

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

View File

@@ -1,8 +1,8 @@
#ifndef CLRSYNC_CORE_UTILS_HPP #ifndef CLRSYNC_CORE_UTILS_HPP
#define CLRSYNC_CORE_UTILS_HPP #define CLRSYNC_CORE_UTILS_HPP
#include <string>
#include <filesystem> #include <filesystem>
#include <string>
#include <core/palette/color_keys.hpp> #include <core/palette/color_keys.hpp>

View File

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

View File

@@ -1,23 +1,27 @@
set(GUI_SOURCES set(GUI_SOURCES
main.cpp main.cpp
color_scheme_editor.cpp views/color_scheme_editor.cpp
color_table_renderer.cpp views/color_table_renderer.cpp
preview_renderer.cpp views/preview_renderer.cpp
theme_applier.cpp controllers/theme_applier.cpp
template_editor.cpp views/template_editor.cpp
palette_controller.cpp controllers/palette_controller.cpp
template_controller.cpp controllers/template_controller.cpp
imgui_helpers.cpp helpers/imgui_helpers.cpp
imgui_helpers.hpp helpers/imgui_helpers.hpp
about_window.cpp views/about_window.cpp
settings_window.cpp views/settings_window.cpp
font_loader.cpp platform/windows/font_loader_windows.cpp
file_browser.cpp platform/linux/font_loader_linux.cpp
platform/macos/font_loader_macos.cpp
platform/linux/file_browser_linux.cpp
platform/windows/file_browser_windows.cpp
platform/macos/file_browser_macos.mm
${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
) )
if(APPLE)
list(APPEND GUI_SOURCES file_browser_macos.mm)
endif()
if(WIN32) if(WIN32)
add_executable(clrsync_gui WIN32 ${GUI_SOURCES}) add_executable(clrsync_gui WIN32 ${GUI_SOURCES})

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

@@ -2,7 +2,7 @@
#include <string> #include <string>
#include "GLFW/glfw3.h" #include "GLFW/glfw3.h"
#include "gui/settings_window.hpp" #include "gui/views/settings_window.hpp"
#include "imgui_impl_glfw.h" #include "imgui_impl_glfw.h"
#include "imgui_impl_opengl3.h" #include "imgui_impl_opengl3.h"
@@ -106,14 +106,10 @@ void setup_main_dockspace(bool& first_time)
ImGui::SetNextWindowSize(viewport->Size); ImGui::SetNextWindowSize(viewport->Size);
ImGui::SetNextWindowViewport(viewport->ID); ImGui::SetNextWindowViewport(viewport->ID);
constexpr ImGuiWindowFlags flags = constexpr ImGuiWindowFlags flags = ImGuiWindowFlags_NoTitleBar | ImGuiWindowFlags_NoCollapse |
ImGuiWindowFlags_NoTitleBar | ImGuiWindowFlags_NoResize | ImGuiWindowFlags_NoMove |
ImGuiWindowFlags_NoCollapse |
ImGuiWindowFlags_NoResize |
ImGuiWindowFlags_NoMove |
ImGuiWindowFlags_NoBringToFrontOnFocus | ImGuiWindowFlags_NoBringToFrontOnFocus |
ImGuiWindowFlags_NoNavFocus | ImGuiWindowFlags_NoNavFocus | ImGuiWindowFlags_MenuBar;
ImGuiWindowFlags_MenuBar;
ImGui::PushStyleVar(ImGuiStyleVar_WindowRounding, 0.0f); ImGui::PushStyleVar(ImGuiStyleVar_WindowRounding, 0.0f);
ImGui::PushStyleVar(ImGuiStyleVar_WindowBorderSize, 0.0f); ImGui::PushStyleVar(ImGuiStyleVar_WindowBorderSize, 0.0f);
@@ -176,7 +172,8 @@ void shutdown(GLFWwindow* window)
namespace palette_utils namespace palette_utils
{ {
ImVec4 get_color(const clrsync::core::palette& pal, const std::string& key, const std::string& fallback) ImVec4 get_color(const clrsync::core::palette &pal, const std::string &key,
const std::string &fallback)
{ {
auto colors = pal.colors(); auto colors = pal.colors();
if (colors.empty()) if (colors.empty())
@@ -201,7 +198,8 @@ ImVec4 get_color(const clrsync::core::palette& pal, const std::string& key, cons
return ImVec4(1.0f, 1.0f, 1.0f, 1.0f); 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) uint32_t get_color_u32(const clrsync::core::palette &pal, const std::string &key,
const std::string &fallback)
{ {
auto colors = pal.colors(); auto colors = pal.colors();
if (colors.empty()) if (colors.empty())
@@ -227,14 +225,16 @@ uint32_t get_color_u32(const clrsync::core::palette& pal, const std::string& key
} }
bool render_delete_confirmation_popup(const std::string &popup_title, const std::string &item_name, 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::string &item_type,
const clrsync::core::palette &pal,
const std::function<void()> &on_delete) const std::function<void()> &on_delete)
{ {
bool result = false; bool result = false;
if (ImGui::BeginPopupModal(popup_title.c_str(), nullptr, ImGuiWindowFlags_AlwaysAutoResize)) if (ImGui::BeginPopupModal(popup_title.c_str(), nullptr, ImGuiWindowFlags_AlwaysAutoResize))
{ {
ImVec4 warning_color = get_color(pal, "warning", "accent"); 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::TextColored(warning_color, "Are you sure you want to delete '%s'?",
item_name.c_str());
ImGui::Text("This action cannot be undone."); ImGui::Text("This action cannot be undone.");
ImGui::Spacing(); ImGui::Spacing();
@@ -263,4 +263,4 @@ bool render_delete_confirmation_popup(const std::string& popup_title, const std:
return result; return result;
} }
} } // namespace palette_utils

View File

@@ -0,0 +1,42 @@
#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

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

View File

@@ -3,19 +3,17 @@
#include <GLFW/glfw3.h> #include <GLFW/glfw3.h>
#include <GLFW/glfw3native.h> #include <GLFW/glfw3native.h>
#include "core/config/config.hpp" #include "core/config/config.hpp"
#include "core/error.hpp"
#include "core/io/toml_file.hpp" #include "core/io/toml_file.hpp"
#include "core/utils.hpp" #include "core/utils.hpp"
#include "core/error.hpp"
#include "color_scheme_editor.hpp"
#include "gui/font_loader.hpp"
#include "gui/settings_window.hpp"
#include "imgui_helpers.hpp"
#include "template_editor.hpp"
#include "about_window.hpp"
#include "gui/helpers/imgui_helpers.hpp"
#include "gui/platform/font_loader.hpp"
#include "gui/views/about_window.hpp"
#include "gui/views/color_scheme_editor.hpp"
#include "gui/views/settings_window.hpp"
#include "gui/views/template_editor.hpp"
int main(int, char **) int main(int, char **)
{ {
@@ -26,7 +24,9 @@ int main(int, char**)
if (!init_result) if (!init_result)
{ {
std::cerr << "Fatal error: " << init_result.error().description() << std::endl; std::cerr << "Fatal error: " << init_result.error().description() << std::endl;
std::cerr << "Hint: Set CLRSYNC_CONFIG_PATH environment variable or ensure config exists at: " << config_path << std::endl; std::cerr
<< "Hint: Set CLRSYNC_CONFIG_PATH environment variable or ensure config exists at: "
<< config_path << std::endl;
return 1; return 1;
} }
@@ -35,26 +35,36 @@ int main(int, char**)
bool first_time = !std::filesystem::exists(ini_path); bool first_time = !std::filesystem::exists(ini_path);
GLFWwindow *window = init_glfw(); GLFWwindow *window = init_glfw();
if (!window) return 1; if (!window)
return 1;
printf("GLFV Version: %s\n", glfwGetVersionString()); printf("GLFV Version: %s\n", glfwGetVersionString());
std::cout << "GLFW runtime platform: "; std::cout << "GLFW runtime platform: ";
switch (glfwGetPlatform()) { switch (glfwGetPlatform())
case GLFW_PLATFORM_WAYLAND: std::cout << "Wayland\n"; break; {
case GLFW_PLATFORM_X11: std::cout << "X11\n"; break; case GLFW_PLATFORM_WAYLAND:
case GLFW_PLATFORM_COCOA: std::cout << "Cocoa\n"; break; std::cout << "Wayland\n";
case GLFW_PLATFORM_WIN32: std::cout << "Win32\n"; break; break;
default: std::cout << "Unknown\n"; 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";
} }
init_imgui(window, ini_path); init_imgui(window, ini_path);
font_loader loader; font_loader loader;
ImFont* font = ImFont *font = loader.load_font(clrsync::core::config::instance().font().c_str(),
loader.load_font(clrsync::core::config::instance().font().c_str(), clrsync::core::config::instance().font_size()); clrsync::core::config::instance().font_size());
if (font) if (font)
ImGui::GetIO().FontDefault = font; ImGui::GetIO().FontDefault = font;

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

@@ -1,6 +1,6 @@
#include "about_window.hpp" #include "about_window.hpp"
#include "core/version.hpp" #include "core/version.hpp"
#include "imgui_helpers.hpp" #include "gui/helpers/imgui_helpers.hpp"
#include "imgui.h" #include "imgui.h"
about_window::about_window() about_window::about_window()

View File

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

View File

@@ -1,13 +1,12 @@
#include "color_scheme_editor.hpp" #include "color_scheme_editor.hpp"
#include "gui/controllers/theme_applier.hpp"
#include "gui/helpers/imgui_helpers.hpp"
#include "imgui.h" #include "imgui.h"
#include "imgui_helpers.hpp"
#include "template_editor.hpp"
#include "settings_window.hpp" #include "settings_window.hpp"
#include "theme_applier.hpp" #include "template_editor.hpp"
#include <iostream> #include <iostream>
#include <ranges> #include <ranges>
color_scheme_editor::color_scheme_editor() color_scheme_editor::color_scheme_editor()
{ {
const auto &current = m_controller.current_palette(); const auto &current = m_controller.current_palette();
@@ -161,10 +160,8 @@ void color_scheme_editor::render_controls()
ImGui::SameLine(); ImGui::SameLine();
auto error = palette_utils::get_color(current, "error"); auto error = palette_utils::get_color(current, "error");
auto error_hover = ImVec4(error.x * 1.1f, error.y * 1.1f, error.z * 1.1f, auto error_hover = ImVec4(error.x * 1.1f, error.y * 1.1f, error.z * 1.1f, error.w);
error.w); auto error_active = ImVec4(error.x * 0.8f, error.y * 0.8f, error.z * 0.8f, 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"); auto on_error = palette_utils::get_color(current, "on_error");
ImGui::PushStyleColor(ImGuiCol_Button, error); ImGui::PushStyleColor(ImGuiCol_Button, error);
ImGui::PushStyleColor(ImGuiCol_ButtonHovered, error_hover); ImGui::PushStyleColor(ImGuiCol_ButtonHovered, error_hover);

View File

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

View File

@@ -1,5 +1,5 @@
#include "color_table_renderer.hpp" #include "gui/views/color_table_renderer.hpp"
#include "imgui_helpers.hpp" #include "gui/helpers/imgui_helpers.hpp"
#include "imgui.h" #include "imgui.h"
#include <algorithm> #include <algorithm>
#include <cctype> #include <cctype>
@@ -138,7 +138,8 @@ void color_table_renderer::render(const clrsync::core::palette& current,
ImGui::Separator(); ImGui::Separator();
ImGui::Spacing(); ImGui::Spacing();
auto draw_table = [&](const char *title, const char* id, const std::vector<const char *> &keys) { auto draw_table = [&](const char *title, const char *id,
const std::vector<const char *> &keys) {
bool has_matches = false; bool has_matches = false;
for (auto *k : keys) for (auto *k : keys)
{ {
@@ -153,13 +154,15 @@ void color_table_renderer::render(const clrsync::core::palette& current,
return; return;
ImGui::PushStyleColor(ImGuiCol_Text, palette_utils::get_color(current, "accent")); ImGui::PushStyleColor(ImGuiCol_Text, palette_utils::get_color(current, "accent"));
bool header_open = ImGui::TreeNodeEx(title, ImGuiTreeNodeFlags_DefaultOpen | ImGuiTreeNodeFlags_SpanAvailWidth); bool header_open = ImGui::TreeNodeEx(title, ImGuiTreeNodeFlags_DefaultOpen |
ImGuiTreeNodeFlags_SpanAvailWidth);
ImGui::PopStyleColor(); ImGui::PopStyleColor();
if (header_open) if (header_open)
{ {
if (ImGui::BeginTable(id, 3, if (ImGui::BeginTable(id, 3,
ImGuiTableFlags_Borders | ImGuiTableFlags_RowBg | ImGuiTableFlags_SizingStretchProp)) ImGuiTableFlags_Borders | ImGuiTableFlags_RowBg |
ImGuiTableFlags_SizingStretchProp))
{ {
ImGui::TableSetupColumn("Name", ImGuiTableColumnFlags_WidthFixed, 160.0f); ImGui::TableSetupColumn("Name", ImGuiTableColumnFlags_WidthFixed, 160.0f);
ImGui::TableSetupColumn("HEX", ImGuiTableColumnFlags_WidthFixed, 95.0f); ImGui::TableSetupColumn("HEX", ImGuiTableColumnFlags_WidthFixed, 95.0f);
@@ -177,23 +180,23 @@ void color_table_renderer::render(const clrsync::core::palette& current,
ImGui::Spacing(); ImGui::Spacing();
}; };
draw_table("General UI", "##general_ui", {"background", "on_background", "surface", "on_surface", draw_table("General UI", "##general_ui",
"surface_variant", "on_surface_variant", "foreground", {"background", "on_background", "surface", "on_surface", "surface_variant",
"cursor", "accent"}); "on_surface_variant", "foreground", "cursor", "accent"});
draw_table("Borders", "##borders", {"border_focused", "border"}); draw_table("Borders", "##borders", {"border_focused", "border"});
draw_table("Semantic Colors", "##semantic", {"success", "info", "warning", "error", draw_table(
"on_success", "on_info", "on_warning", "on_error"}); "Semantic Colors", "##semantic",
{"success", "info", "warning", "error", "on_success", "on_info", "on_warning", "on_error"});
draw_table("Editor", "##editor", {"editor_background", "editor_command", "editor_comment", draw_table("Editor", "##editor",
"editor_disabled", "editor_emphasis", "editor_error", {"editor_background", "editor_command", "editor_comment", "editor_disabled",
"editor_inactive", "editor_line_number", "editor_link", "editor_emphasis", "editor_error", "editor_inactive", "editor_line_number",
"editor_main", "editor_selected", "editor_selection_inactive", "editor_link", "editor_main", "editor_selected", "editor_selection_inactive",
"editor_string", "editor_success", "editor_warning"}); "editor_string", "editor_success", "editor_warning"});
draw_table("Terminal (Base16)", "##terminal", {"base00", "base01", "base02", "base03", draw_table("Terminal (Base16)", "##terminal",
"base04", "base05", "base06", "base07", {"base00", "base01", "base02", "base03", "base04", "base05", "base06", "base07",
"base08", "base09", "base0A", "base0B", "base08", "base09", "base0A", "base0B", "base0C", "base0D", "base0E", "base0F"});
"base0C", "base0D", "base0E", "base0F"});
} }

View File

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

View File

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

View File

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

View File

@@ -1,9 +1,9 @@
#include "settings_window.hpp" #include "gui/views/settings_window.hpp"
#include "core/config/config.hpp" #include "core/config/config.hpp"
#include "core/error.hpp" #include "core/error.hpp"
#include "gui/font_loader.hpp" #include "gui/helpers/imgui_helpers.hpp"
#include "gui/imgui_helpers.hpp" #include "gui/platform/file_browser.hpp"
#include "gui/file_browser.hpp" #include "gui/platform/font_loader.hpp"
#include "imgui.h" #include "imgui.h"
#include <cstring> #include <cstring>
@@ -26,7 +26,8 @@ void settings_window::render()
return; return;
ImGui::SetNextWindowSize(ImVec2(700, 500), ImGuiCond_FirstUseEver); ImGui::SetNextWindowSize(ImVec2(700, 500), ImGuiCond_FirstUseEver);
ImGui::SetNextWindowPos(ImGui::GetMainViewport()->GetCenter(), ImGuiCond_FirstUseEver, ImVec2(0.5f, 0.5f)); ImGui::SetNextWindowPos(ImGui::GetMainViewport()->GetCenter(), ImGuiCond_FirstUseEver,
ImVec2(0.5f, 0.5f));
ImGuiWindowFlags window_flags = ImGuiWindowFlags_NoCollapse; ImGuiWindowFlags window_flags = ImGuiWindowFlags_NoCollapse;
if (ImGui::Begin("Settings", &m_visible, window_flags)) if (ImGui::Begin("Settings", &m_visible, window_flags))
@@ -184,8 +185,10 @@ void settings_window::render_general_tab()
ImGui::SameLine(); ImGui::SameLine();
if (ImGui::Button("Browse")) if (ImGui::Button("Browse"))
{ {
std::string selected_path = file_dialogs::select_folder_dialog("Select Palettes Directory", m_palettes_path); std::string selected_path =
if (!selected_path.empty()) { 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); strncpy(m_palettes_path, selected_path.c_str(), sizeof(m_palettes_path) - 1);
m_palettes_path[sizeof(m_palettes_path) - 1] = '\0'; m_palettes_path[sizeof(m_palettes_path) - 1] = '\0';
m_settings_changed = true; m_settings_changed = true;
@@ -259,7 +262,8 @@ void settings_window::render_status_messages()
ImGui::PushStyleVar(ImGuiStyleVar_ChildRounding, 4.0f); ImGui::PushStyleVar(ImGuiStyleVar_ChildRounding, 4.0f);
ImGui::PushStyleVar(ImGuiStyleVar_ChildBorderSize, 1.0f); ImGui::PushStyleVar(ImGuiStyleVar_ChildBorderSize, 1.0f);
if (ImGui::BeginChild("##error_box", ImVec2(0, 0), ImGuiChildFlags_AutoResizeY | ImGuiChildFlags_Borders)) if (ImGui::BeginChild("##error_box", ImVec2(0, 0),
ImGuiChildFlags_AutoResizeY | ImGuiChildFlags_Borders))
{ {
ImGui::PushStyleColor(ImGuiCol_Text, error_text_color); ImGui::PushStyleColor(ImGuiCol_Text, error_text_color);
ImGui::TextWrapped("Error: %s", m_error_message.c_str()); ImGui::TextWrapped("Error: %s", m_error_message.c_str());
@@ -267,8 +271,12 @@ void settings_window::render_status_messages()
ImGui::Spacing(); 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_Button,
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)); 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); ImGui::PushStyleColor(ImGuiCol_Text, error_text_color);
if (ImGui::Button("Dismiss##error")) if (ImGui::Button("Dismiss##error"))

View File

@@ -10,9 +10,18 @@ class settings_window
public: public:
settings_window(); settings_window();
void render(); void render();
void show() { m_visible = true; } void show()
void hide() { m_visible = false; } {
bool is_visible() const { return m_visible; } m_visible = true;
}
void hide()
{
m_visible = false;
}
bool is_visible() const
{
return m_visible;
}
private: private:
void load_settings(); void load_settings();
@@ -26,7 +35,8 @@ private:
void reset_to_defaults(); void reset_to_defaults();
public: public:
void set_palette(const clrsync::core::palette& palette) { void set_palette(const clrsync::core::palette &palette)
{
m_current_palette = palette; m_current_palette = palette;
} }

View File

@@ -1,24 +1,21 @@
#include "template_editor.hpp" #include "template_editor.hpp"
#include "core/config/config.hpp" #include "core/config/config.hpp"
#include "core/theme/theme_template.hpp"
#include "core/palette/color_keys.hpp" #include "core/palette/color_keys.hpp"
#include "core/theme/theme_template.hpp"
#include "core/utils.hpp" #include "core/utils.hpp"
#include "imgui_helpers.hpp" #include "gui/helpers/imgui_helpers.hpp"
#include "file_browser.hpp" #include "gui/platform/file_browser.hpp"
#include "imgui.h" #include "imgui.h"
#include <algorithm> #include <algorithm>
#include <filesystem> #include <filesystem>
#include <fstream> #include <fstream>
#include <ranges> #include <ranges>
namespace { namespace
{
const std::vector<std::string> COLOR_FORMATS = { const std::vector<std::string> COLOR_FORMATS = {
"hex", "hex_stripped", "hexa", "hexa_stripped", "hex", "hex_stripped", "hexa", "hexa_stripped", "r", "g", "b", "a", "rgb", "rgba", "h", "s",
"r", "g", "b", "a", "l", "hsl", "hsla"};
"rgb", "rgba",
"h", "s", "l",
"hsl", "hsla"
};
} }
template_editor::template_editor() : m_template_name("new_template") template_editor::template_editor() : m_template_name("new_template")
@@ -108,7 +105,8 @@ void template_editor::apply_current_palette(const clrsync::core::palette &pal)
m_autocomplete_selected_color = palette_utils::get_color(pal, "accent", "surface_variant"); m_autocomplete_selected_color = palette_utils::get_color(pal, "accent", "surface_variant");
m_autocomplete_text_color = palette_utils::get_color(pal, "on_surface", "foreground"); m_autocomplete_text_color = palette_utils::get_color(pal, "on_surface", "foreground");
m_autocomplete_selected_text_color = palette_utils::get_color(pal, "on_surface", "foreground"); m_autocomplete_selected_text_color = palette_utils::get_color(pal, "on_surface", "foreground");
m_autocomplete_dim_text_color = palette_utils::get_color(pal, "on_surface_variant", "editor_inactive"); m_autocomplete_dim_text_color =
palette_utils::get_color(pal, "on_surface_variant", "editor_inactive");
} }
void template_editor::update_autocomplete_suggestions() void template_editor::update_autocomplete_suggestions()
@@ -148,8 +146,7 @@ void template_editor::update_autocomplete_suggestions()
{ {
bool should_reset_dismissal = false; bool should_reset_dismissal = false;
if (cursor.mLine != m_dismiss_position.mLine || if (cursor.mLine != m_dismiss_position.mLine || brace_pos != m_dismiss_brace_pos ||
brace_pos != m_dismiss_brace_pos ||
abs(cursor.mColumn - m_dismiss_position.mColumn) > 3) abs(cursor.mColumn - m_dismiss_position.mColumn) > 3)
{ {
should_reset_dismissal = true; should_reset_dismissal = true;
@@ -190,8 +187,7 @@ void template_editor::update_autocomplete_suggestions()
{ {
for (const auto &fmt : COLOR_FORMATS) for (const auto &fmt : COLOR_FORMATS)
{ {
if (format_prefix.empty() || if (format_prefix.empty() || fmt.find(format_prefix) == 0 ||
fmt.find(format_prefix) == 0 ||
fmt.find(format_prefix) != std::string::npos) fmt.find(format_prefix) != std::string::npos)
{ {
m_autocomplete_suggestions.push_back(color_key + "." + fmt); m_autocomplete_suggestions.push_back(color_key + "." + fmt);
@@ -204,8 +200,7 @@ void template_editor::update_autocomplete_suggestions()
for (size_t i = 0; i < clrsync::core::NUM_COLOR_KEYS; ++i) for (size_t i = 0; i < clrsync::core::NUM_COLOR_KEYS; ++i)
{ {
std::string key = clrsync::core::COLOR_KEYS[i]; std::string key = clrsync::core::COLOR_KEYS[i];
if (m_autocomplete_prefix.empty() || if (m_autocomplete_prefix.empty() || key.find(m_autocomplete_prefix) == 0 ||
key.find(m_autocomplete_prefix) == 0 ||
key.find(m_autocomplete_prefix) != std::string::npos) key.find(m_autocomplete_prefix) != std::string::npos)
{ {
m_autocomplete_suggestions.push_back(key); m_autocomplete_suggestions.push_back(key);
@@ -241,7 +236,8 @@ void template_editor::render_autocomplete(const ImVec2& editor_pos)
const float line_number_width = 50.0f; const float line_number_width = 50.0f;
ImVec2 popup_pos; ImVec2 popup_pos;
popup_pos.x = editor_pos.x + line_number_width + (m_autocomplete_start_pos.mColumn * char_width); popup_pos.x =
editor_pos.x + line_number_width + (m_autocomplete_start_pos.mColumn * char_width);
popup_pos.y = editor_pos.y + ((cursor.mLine + 1) * line_height); popup_pos.y = editor_pos.y + ((cursor.mLine + 1) * line_height);
ImGui::SetNextWindowPos(popup_pos, ImGuiCond_Always); ImGui::SetNextWindowPos(popup_pos, ImGuiCond_Always);
@@ -249,7 +245,8 @@ void template_editor::render_autocomplete(const ImVec2& editor_pos)
ImGuiWindowFlags flags = ImGuiWindowFlags_NoTitleBar | ImGuiWindowFlags_NoResize | ImGuiWindowFlags flags = ImGuiWindowFlags_NoTitleBar | ImGuiWindowFlags_NoResize |
ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoSavedSettings | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoSavedSettings |
ImGuiWindowFlags_NoFocusOnAppearing | ImGuiWindowFlags_AlwaysAutoResize; ImGuiWindowFlags_NoFocusOnAppearing |
ImGuiWindowFlags_AlwaysAutoResize;
ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2(6, 6)); ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2(6, 6));
ImGui::PushStyleVar(ImGuiStyleVar_WindowRounding, 6.0f); ImGui::PushStyleVar(ImGuiStyleVar_WindowRounding, 6.0f);
@@ -295,8 +292,8 @@ void template_editor::render_autocomplete(const ImVec2& editor_pos)
std::string display_text = " " + suggestion; std::string display_text = " " + suggestion;
if (ImGui::Selectable(display_text.c_str(), is_selected, if (ImGui::Selectable(display_text.c_str(), is_selected, ImGuiSelectableFlags_None,
ImGuiSelectableFlags_None, ImVec2(0, 0))) ImVec2(0, 0)))
{ {
auto start = m_autocomplete_start_pos; auto start = m_autocomplete_start_pos;
auto end = m_editor.GetCursorPosition(); auto end = m_editor.GetCursorPosition();
@@ -362,8 +359,7 @@ void template_editor::render()
} }
palette_utils::render_delete_confirmation_popup( palette_utils::render_delete_confirmation_popup(
"Delete Template?", m_template_name, "template", m_current_palette, "Delete Template?", m_template_name, "template", m_current_palette, [this]() {
[this]() {
bool success = m_template_controller.remove_template(m_template_name); bool success = m_template_controller.remove_template(m_template_name);
if (success) if (success)
{ {
@@ -408,10 +404,8 @@ void template_editor::render_controls()
{ {
ImGui::SameLine(); ImGui::SameLine();
auto error = palette_utils::get_color(m_current_palette, "error"); 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, auto error_hover = ImVec4(error.x * 1.1f, error.y * 1.1f, error.z * 1.1f, error.w);
error.w); auto error_active = ImVec4(error.x * 0.8f, error.y * 0.8f, error.z * 0.8f, 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"); auto on_error = palette_utils::get_color(m_current_palette, "on_error");
ImGui::PushStyleColor(ImGuiCol_Button, error); ImGui::PushStyleColor(ImGuiCol_Button, error);
ImGui::PushStyleColor(ImGuiCol_ButtonHovered, error_hover); ImGui::PushStyleColor(ImGuiCol_ButtonHovered, error_hover);
@@ -433,18 +427,24 @@ void template_editor::render_controls()
if (m_enabled) if (m_enabled)
{ {
ImVec4 success_color = palette_utils::get_color(m_current_palette, "success", "accent"); 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_on_color =
ImVec4 success_hover = ImVec4(success_color.x * 1.2f, success_color.y * 1.2f, success_color.z * 1.2f, 0.6f); palette_utils::get_color(m_current_palette, "on_success", "on_surface");
ImGui::PushStyleColor(ImGuiCol_FrameBg, ImVec4(success_color.x, success_color.y, success_color.z, 0.5f)); 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_FrameBgHovered, success_hover);
ImGui::PushStyleColor(ImGuiCol_CheckMark, success_on_color); ImGui::PushStyleColor(ImGuiCol_CheckMark, success_on_color);
} }
else else
{ {
ImVec4 error_color = palette_utils::get_color(m_current_palette, "error", "accent"); 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_on_color =
ImVec4 error_hover = ImVec4(error_color.x * 1.2f, error_color.y * 1.2f, error_color.z * 1.2f, 0.6f); palette_utils::get_color(m_current_palette, "on_error", "on_surface");
ImGui::PushStyleColor(ImGuiCol_FrameBg, ImVec4(error_color.x, error_color.y, error_color.z, 0.5f)); 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_FrameBgHovered, error_hover);
ImGui::PushStyleColor(ImGuiCol_CheckMark, error_on_color); ImGui::PushStyleColor(ImGuiCol_CheckMark, error_on_color);
} }
@@ -486,8 +486,8 @@ void template_editor::render_controls()
ImGui::SetNextItemWidth(-120.0f); ImGui::SetNextItemWidth(-120.0f);
char input_path_buf[512] = {0}; char input_path_buf[512] = {0};
snprintf(input_path_buf, sizeof(input_path_buf), "%s", m_input_path.c_str()); snprintf(input_path_buf, sizeof(input_path_buf), "%s", m_input_path.c_str());
if (ImGui::InputTextWithHint("##input_path", "Path to template file...", if (ImGui::InputTextWithHint("##input_path", "Path to template file...", input_path_buf,
input_path_buf, sizeof(input_path_buf))) sizeof(input_path_buf)))
{ {
m_input_path = input_path_buf; m_input_path = input_path_buf;
if (!m_input_path.empty()) if (!m_input_path.empty())
@@ -502,10 +502,13 @@ void template_editor::render_controls()
ImGui::SameLine(); ImGui::SameLine();
if (ImGui::Button("Browse##input")) if (ImGui::Button("Browse##input"))
{ {
std::string selected_path = file_dialogs::open_file_dialog("Select Template File", m_input_path); std::string selected_path =
if (!selected_path.empty()) { file_dialogs::open_file_dialog("Select Template File", m_input_path);
if (!selected_path.empty())
{
m_input_path = selected_path; m_input_path = selected_path;
if (m_is_editing_existing) { if (m_is_editing_existing)
{
m_template_controller.set_template_input_path(m_template_name, m_input_path); m_template_controller.set_template_input_path(m_template_name, m_input_path);
} }
m_validation_error = ""; m_validation_error = "";
@@ -520,8 +523,8 @@ void template_editor::render_controls()
ImGui::SetNextItemWidth(-120.0f); ImGui::SetNextItemWidth(-120.0f);
char path_buf[512] = {0}; char path_buf[512] = {0};
snprintf(path_buf, sizeof(path_buf), "%s", m_output_path.c_str()); snprintf(path_buf, sizeof(path_buf), "%s", m_output_path.c_str());
if (ImGui::InputTextWithHint("##output_path", "Path for generated config...", if (ImGui::InputTextWithHint("##output_path", "Path for generated config...", path_buf,
path_buf, sizeof(path_buf))) sizeof(path_buf)))
{ {
m_output_path = path_buf; m_output_path = path_buf;
if (!m_output_path.empty()) if (!m_output_path.empty())
@@ -536,10 +539,13 @@ void template_editor::render_controls()
ImGui::SameLine(); ImGui::SameLine();
if (ImGui::Button("Browse##output")) if (ImGui::Button("Browse##output"))
{ {
std::string selected_path = file_dialogs::save_file_dialog("Select Output File", m_output_path); std::string selected_path =
if (!selected_path.empty()) { file_dialogs::save_file_dialog("Select Output File", m_output_path);
if (!selected_path.empty())
{
m_output_path = selected_path; m_output_path = selected_path;
if (m_is_editing_existing) { if (m_is_editing_existing)
{
m_template_controller.set_template_output_path(m_template_name, m_output_path); m_template_controller.set_template_output_path(m_template_name, m_output_path);
} }
m_validation_error = ""; m_validation_error = "";
@@ -554,8 +560,8 @@ void template_editor::render_controls()
ImGui::SetNextItemWidth(-FLT_MIN); ImGui::SetNextItemWidth(-FLT_MIN);
char reload_buf[512] = {0}; char reload_buf[512] = {0};
snprintf(reload_buf, sizeof(reload_buf), "%s", m_reload_command.c_str()); snprintf(reload_buf, sizeof(reload_buf), "%s", m_reload_command.c_str());
if (ImGui::InputTextWithHint("##reload_cmd", "Command to reload app (optional)...", if (ImGui::InputTextWithHint("##reload_cmd", "Command to reload app (optional)...", reload_buf,
reload_buf, sizeof(reload_buf))) sizeof(reload_buf)))
{ {
m_reload_command = reload_buf; m_reload_command = reload_buf;
if (m_is_editing_existing) if (m_is_editing_existing)
@@ -708,7 +714,8 @@ void template_editor::render_template_list()
if (!tmpl.enabled()) if (!tmpl.enabled())
{ {
ImVec4 disabled_color = palette_utils::get_color(m_current_palette, "on_surface_variant", "editor_inactive"); ImVec4 disabled_color = palette_utils::get_color(
m_current_palette, "on_surface_variant", "editor_inactive");
ImGui::PushStyleColor(ImGuiCol_Text, disabled_color); ImGui::PushStyleColor(ImGuiCol_Text, disabled_color);
} }

View File

@@ -1,10 +1,10 @@
#ifndef CLRSYNC_GUI_TEMPLATE_EDITOR_HPP #ifndef CLRSYNC_GUI_TEMPLATE_EDITOR_HPP
#define CLRSYNC_GUI_TEMPLATE_EDITOR_HPP #define CLRSYNC_GUI_TEMPLATE_EDITOR_HPP
#include "template_controller.hpp"
#include <core/palette/palette.hpp>
#include "color_text_edit/TextEditor.h" #include "color_text_edit/TextEditor.h"
#include "gui/controllers/template_controller.hpp"
#include "imgui.h" #include "imgui.h"
#include <core/palette/palette.hpp>
#include <string> #include <string>
#include <vector> #include <vector>