Mercurial > minori
changeset 351:c844f8bb87ce
gui/theme: add xsettings backend
this also adds newly-necessary endianness methods in core/endian.h
which just so happen to be constexpr as well
author | Paper <paper@paper.us.eu.org> |
---|---|
date | Sun, 14 Jul 2024 23:23:56 -0400 (6 months ago) |
parents | daa03aa2262d |
children | a0e96f50bcce |
files | CMakeLists.txt include/core/endian.h include/sys/glib/dark_theme.h include/sys/x11/dark_theme.h include/sys/x11/settings.h src/gui/theme.cc src/sys/glib/dark_theme.cc src/sys/x11/dark_theme.cc src/sys/x11/settings.cc |
diffstat | 9 files changed, 596 insertions(+), 21 deletions(-) [+] |
line wrap: on
line diff
--- a/CMakeLists.txt Sun Jul 14 19:12:40 2024 -0400 +++ b/CMakeLists.txt Sun Jul 14 23:23:56 2024 -0400 @@ -238,7 +238,15 @@ list(APPEND SRC_FILES src/sys/glib/dark_theme.cc) list(APPEND INCLUDE ${GLIB_INCLUDE_DIRS}) list(APPEND LIBRARIES ${GLIB_LINK_LIBRARIES}) - list(APPEND DEFINES GLIB) + list(APPEND DEFINES GLIB) # XXX rename HAVE_GLIB or something + endif() + + pkg_check_modules(XCB xcb) + if (XCB_FOUND) + list(APPEND SRC_FILES src/sys/x11/settings.cc src/sys/x11/dark_theme.cc) + list(APPEND INCLUDE ${XCB_INCLUDE_DIRS}) + list(APPEND LIBRARIES ${XCB_LINK_LIBRARIES}) + list(APPEND DEFINES HAVE_XCB) endif() endif() endif()
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/include/core/endian.h Sun Jul 14 23:23:56 2024 -0400 @@ -0,0 +1,121 @@ +#ifndef MINORI_CORE_ENDIAN_H_ +#define MINORI_CORE_ENDIAN_H_ + +/* definition of endian-related stuff. primarily used for*/ + +#include <cstdint> +#include <type_traits> + +class Endian { +private: + static constexpr uint32_t uint32_ = 0x01020304; + static constexpr uint8_t magic_ = static_cast<const uint8_t&>(uint32_); + + /* check for compiler builtins for byteswapping */ +#ifdef __has_builtin +# if __has_builtin(__builtin_bswap16) +# define COMPILER_BUILTIN_BSWAP16(x) __builtin_bswap16(x) +# endif +# if __has_builtin(__builtin_bswap32) +# define COMPILER_BUILTIN_BSWAP32(x) __builtin_bswap32(x) +# endif +# if __has_builtin(__builtin_bswap64) +# define COMPILER_BUILTIN_BSWAP64(x) __builtin_bswap64(x) +# endif +#endif + + static constexpr uint16_t byteswap_16(uint16_t x) { +#ifdef COMPILER_BUILTIN_BSWAP16 + return COMPILER_BUILTIN_BSWAP16(x); +#else + return ( + ((x & UINT16_C(0x00FF)) << 8) + | ((x & UINT16_C(0xFF00)) >> 8) + ); +#endif + } + + static constexpr uint32_t byteswap_32(uint32_t x) { +#ifdef COMPILER_BUILTIN_BSWAP32 + return COMPILER_BUILTIN_BSWAP32(x); +#else + return ( + ((x & UINT32_C(0x000000FF)) << 24) + | ((x & UINT32_C(0x0000FF00)) << 8) + | ((x & UINT32_C(0x00FF0000)) >> 8) + | ((x & UINT32_C(0xFF000000)) >> 24) + ); +#endif + } + + static constexpr uint64_t byteswap_64(uint64_t x) { +#ifdef COMPILER_BUILTIN_BSWAP64 + return COMPILER_BUILTIN_BSWAP64(x); +#else + return ( + ((x & UINT64_C(0x00000000000000FF)) << 56) + | ((x & UINT64_C(0x000000000000FF00)) << 40) + | ((x & UINT64_C(0x0000000000FF0000)) << 24) + | ((x & UINT64_C(0x00000000FF000000)) << 8) + | ((x & UINT64_C(0x000000FF00000000)) >> 8) + | ((x & UINT64_C(0x0000FF0000000000)) >> 24) + | ((x & UINT64_C(0x00FF000000000000)) >> 40) + | ((x & UINT64_C(0xFF00000000000000)) >> 56) + ); +#endif + } + +#ifdef COMPILER_BUILTIN_BSWAP16 +# undef COMPILER_BUILTIN_BSWAP16 +#endif +#ifdef COMPILER_BUILTIN_BSWAP32 +# undef COMPILER_BUILTIN_BSWAP32 +#endif +#ifdef COMPILER_BUILTIN_BSWAP64 +# undef COMPILER_BUILTIN_BSWAP64 +#endif +public: + static constexpr bool little = magic_ == 0x04; + static constexpr bool big = magic_ == 0x01; + static_assert(little || big, "unsupported endianness"); + + template<typename T> + static constexpr T byteswap(T x) { + static_assert(std::is_integral<T>::value); + static_assert(std::is_unsigned<T>::value); + + if constexpr (std::is_same<T, uint8_t>::value) { + return x; + } else if constexpr (std::is_same<T, uint16_t>::value) { + return byteswap_16(x); + } else if constexpr (std::is_same<T, uint32_t>::value) { + return byteswap_32(x); + } else if constexpr (std::is_same<T, uint64_t>::value) { + return byteswap_64(x); + } else { + static_assert(false, "byteswapping with unknown integer type"); + } + } + + template<typename T> + static constexpr T byteswap_little_to_host(T x) { + if constexpr (little) { + return x; + } else if constexpr (big) { + return byteswap(x); + } + } + + template<typename T> + static constexpr T byteswap_big_to_host(T x) { + if constexpr (big) { + return x; + } else if constexpr (little) { + return byteswap(x); + } + } +private: + Endian() = delete; +}; + +#endif /* MINORI_CORE_ENDIAN_H_ */
--- a/include/sys/glib/dark_theme.h Sun Jul 14 19:12:40 2024 -0400 +++ b/include/sys/glib/dark_theme.h Sun Jul 14 23:23:56 2024 -0400 @@ -1,8 +1,11 @@ #ifndef MINORI_SYS_GLIB_DARK_THEME_H_ #define MINORI_SYS_GLIB_DARK_THEME_H_ +#include <string_view> + namespace glib { +bool IsGTKThemeDark(const std::string_view str); bool IsInDarkTheme(); }
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/include/sys/x11/dark_theme.h Sun Jul 14 23:23:56 2024 -0400 @@ -0,0 +1,10 @@ +#ifndef MINORI_SYS_X11_DARK_THEME_H_ +#define MINORI_SYS_X11_DARK_THEME_H_ + +namespace x11 { + +bool IsInDarkTheme(); + +} + +#endif /* MINORI_SYS_X11_DARK_THEME_H_ */ \ No newline at end of file
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/include/sys/x11/settings.h Sun Jul 14 23:23:56 2024 -0400 @@ -0,0 +1,40 @@ +#ifndef MINORI_SYS_X11_SETTINGS_H_ +#define MINORI_SYS_X11_SETTINGS_H_ + +#include <cstdint> +#include <string> +#include <vector> + +namespace x11 { + +/* stores the item, type, etc */ +struct SettingsItem { + enum Type { + TypeInt = 0, + TypeStr = 1, + TypeRgba = 2, + }; + + /* could technically be a union */ + struct Data { + std::uint32_t integer; + std::string string; + struct { + std::uint16_t red, green, blue, alpha; + } rgba; + }; + + bool VerifyType(); + + std::uint8_t type; /* one of Type */ + std::string name; /* name of the item */ + std::uint32_t serial; /* last-changed serial */ + Data data; /* type-specific data */ +}; + +bool GetSettings(std::vector<SettingsItem>& settings); +bool FindSetting(const std::string& name, SettingsItem& setting); + +} + +#endif /* MINORI_SYS_X11_SETTINGS_H_ */ \ No newline at end of file
--- a/src/gui/theme.cc Sun Jul 14 19:12:40 2024 -0400 +++ b/src/gui/theme.cc Sun Jul 14 23:23:56 2024 -0400 @@ -16,6 +16,9 @@ # ifdef GLIB # include "sys/glib/dark_theme.h" # endif +# ifdef HAVE_XCB +# include "sys/x11/dark_theme.h" +# endif #endif /* Weird quirks of this implementation: @@ -45,11 +48,18 @@ if (win32::DarkThemeAvailable()) return win32::IsInDarkTheme(); #else -# ifdef GLIB - return glib::IsInDarkTheme(); +# ifdef HAVE_XCB + if (x11::IsInDarkTheme()) + return true; # endif +# ifdef GLIB + if (glib::IsInDarkTheme()) + return true; +# endif + break; #endif - default: break; + default: + break; } return (theme == Theme::Dark); }
--- a/src/sys/glib/dark_theme.cc Sun Jul 14 19:12:40 2024 -0400 +++ b/src/sys/glib/dark_theme.cc Sun Jul 14 23:23:56 2024 -0400 @@ -5,6 +5,7 @@ #include <string_view> #include <memory> #include <array> +#include <iostream> namespace glib { @@ -33,6 +34,29 @@ template<typename T> using GMallocPtr = std::unique_ptr<T, g_malloc_del<T>>; +/* not really "glib" but GNOME-related enough */ +bool IsGTKThemeDark(const std::string_view str) { + /* if that doesn't exist, use the GTK theme and check for some known + * suffixes. if one is found, return + * + * XXX probably better to use case folding here */ + static constexpr std::array<std::string_view, 3> suffixes = { + "-dark", /* Adwaita-dark */ + "-Dark", /* Arc-Dark */ + "-Darker", /* Arc-Darker */ + }; + + for (const auto& suffix : suffixes) { + if (str.size() < suffix.size()) + continue; + + if (std::equal(str.data() + str.size() - suffix.length(), str.data() + str.size(), suffix.begin(), suffix.end())) + return true; + } + + return false; +} + bool IsInDarkTheme() { GObjectPtr<GSettings> settings(::g_settings_new("org.gnome.desktop.interface")); if (!settings) @@ -50,22 +74,13 @@ if (!str) return false; - bool success = !std::strncmp(str, size, "prefer-dark"); + bool success = !std::strncmp(str, "prefer-dark", size); if (success) return true; } { - /* if that doesn't exist, use the GTK theme and check for some known - * suffixes. if one is found, return - * - * XXX probably better to use case folding here */ - static constexpr std::array<std::string_view, 3> suffixes = { - "-dark", /* Adwaita-dark */ - "-Dark", /* Arc-Dark */ - "-Darker", /* Arc-Darker */ - }; GVariantPtr<GVariant> gtk_theme(::g_settings_get_value(settings.get(), "gtk-theme")); if (!gtk_theme) @@ -76,13 +91,8 @@ if (!str) return false; - for (const auto& suffix : suffixes) { - if (size < suffix.size()) - continue; - - if (std::equal(str + size - suffix.length(), str + size, suffix.begin(), suffix.end())) - return true; - } + if (IsGTKThemeDark({str, size})) + return true; } /* welp, we tried */
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/sys/x11/dark_theme.cc Sun Jul 14 23:23:56 2024 -0400 @@ -0,0 +1,17 @@ +#include "sys/x11/dark_theme.h" +#include "sys/x11/settings.h" +#include "sys/glib/dark_theme.h" /* glib::IsGTKThemeDark */ + +#include <iostream> + +namespace x11 { + +bool IsInDarkTheme() { + SettingsItem setting; + if (!FindSetting(u8"Net/ThemeName", setting)) + return false; + + return glib::IsGTKThemeDark(setting.data.string); +} + +} // namespace glib
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/sys/x11/settings.cc Sun Jul 14 23:23:56 2024 -0400 @@ -0,0 +1,356 @@ +#include "sys/x11/settings.h" +#include "core/endian.h" + +#include <cstring> +#include <cstdint> +#include <climits> +#include <string_view> +#include <memory> +#include <array> +#include <optional> +#include <iostream> +#include <map> + +#include <xcb/xcb.h> + +#include "fmt/core.h" + +namespace x11 { + +bool SettingsItem::VerifyType() { + switch (type) { + case SettingsItem::TypeInt: + case SettingsItem::TypeStr: + case SettingsItem::TypeRgba: + return true; + default: + return false; + } +} + +/* -------------------------------------------------------------------------- */ +/* xsettings parser */ + +static constexpr std::size_t GetPadding(std::size_t length, std::size_t increment) { + /* ripped from xsettingsd */ + return (increment - (length % increment)) % increment; +} + +class Parser { +public: + Parser(std::uint8_t *bytes, std::size_t size); + + std::vector<SettingsItem> ParseAllItems(void); + std::optional<SettingsItem> ParseNextItem(void); + + std::uint32_t GetTotalItems(void); + +private: + /* byte order values */ + enum { + LSBFirst = 0, + MSBFirst = 1, + }; + + template<typename T> + bool ReadData(T& ret) { + if (offset_ + sizeof(T) >= size_) + return false; + + ret = *reinterpret_cast<T*>(bytes_ + offset_); + Advance(sizeof(T)); + return true; + } + + template<typename T> + bool ReadInt(T& ret) { + static_assert(std::is_integral<T>::value); + + if (!ReadData<T>(ret)) + return false; + + switch (byte_order_) { + case LSBFirst: + ret = Endian::byteswap_little_to_host(ret); + break; + case MSBFirst: + ret = Endian::byteswap_big_to_host(ret); + break; + default: + /* can't know for sure. punt */ + return false; + } + + return true; + } + + bool ReadString(std::string& str, std::size_t size); + bool Advance(std::size_t amount); + + /* raw data */ + std::uint8_t *bytes_ = nullptr; + std::size_t offset_ = 0; + std::size_t size_ = 0; + + /* parsed in the constructor */ + std::uint8_t byte_order_ = 0; /* unused */ + std::uint32_t serial_ = 0; + std::uint32_t total_items_ = 0; +}; + +std::uint32_t Parser::GetTotalItems(void) { + return total_items_; +} + +bool Parser::ReadString(std::string& str, std::size_t size) { + if (offset_ + size >= size_) + return false; + + str.assign(reinterpret_cast<const char *>(bytes_ + offset_), size); + Advance(size); + return true; +} + +bool Parser::Advance(std::size_t amount) { + if (offset_ + amount >= size_) + return false; + + offset_ += amount; + return true; +} + +Parser::Parser(std::uint8_t *bytes, std::size_t size) { + bytes_ = bytes; + size_ = size; + + /* unused for now... don't know what the values are! :) + * assuming host byte order */ + if (!ReadData<std::uint8_t>(byte_order_)) + return; + + Advance(3); + + if (!ReadData<std::uint32_t>(serial_)) + return; + + if (!ReadData<std::uint32_t>(total_items_)) + return; +} + +std::optional<SettingsItem> Parser::ParseNextItem(void) { + SettingsItem item; + + /* read one byte */ + if (!ReadInt<std::uint8_t>(item.type)) + return std::nullopt; + + if (!item.VerifyType()) + return std::nullopt; + + /* skip padding */ + if (!Advance(1)) + return std::nullopt; + + /* parse the name */ + std::uint16_t name_size; + if (!ReadInt<std::uint16_t>(name_size)) + return std::nullopt; + + if (!ReadString(item.name, name_size)) + return std::nullopt; + + /* padding */ + if (!Advance(GetPadding(name_size, 4))) + return std::nullopt; + + if (!ReadInt<std::uint32_t>(item.serial)) + return std::nullopt; + + switch (item.type) { + case SettingsItem::TypeInt: { + if (!ReadInt<std::uint32_t>(item.data.integer)) + return std::nullopt; + + break; + } + case SettingsItem::TypeStr: { + std::uint32_t size; + if (!ReadInt<std::uint32_t>(size)) + return std::nullopt; + + if (!ReadString(item.data.string, size)) + return std::nullopt; + + /* padding */ + if (!Advance(GetPadding(size, 4))) + return std::nullopt; + + break; + } + case SettingsItem::TypeRgba: { + /* The order here is important!! */ + if (!ReadInt<std::uint16_t>(item.data.rgba.red) + || !ReadInt<std::uint16_t>(item.data.rgba.blue) + || !ReadInt<std::uint16_t>(item.data.rgba.green) + || !ReadInt<std::uint16_t>(item.data.rgba.alpha)) + return std::nullopt; + + break; + } + default: + /* can't do anything now, can we? */ + return std::nullopt; + } + + return item; +} + +std::vector<SettingsItem> Parser::ParseAllItems(void) { + offset_ = 0; + + std::uint32_t i; + std::vector<SettingsItem> items; + + for (i = 0; i < total_items_; i++) { + std::optional<SettingsItem> item = ParseNextItem(); + if (!item) + break; + + items.push_back(item.value()); + } + + return items; +} + +/* ------------------------------------------------------------------------- */ +/* real X11 code */ + +template<typename T> +struct MallocDestructor { + void operator()(T *t) const { std::free(t); }; +}; + +struct XcbConnectionDestructor { + void operator()(xcb_connection_t *conn) const { ::xcb_disconnect(conn); }; +}; + +template<typename T> +using MallocPtr = std::unique_ptr<T, MallocDestructor<T>>; + +using XcbConnectionPtr = std::unique_ptr<xcb_connection_t, XcbConnectionDestructor>; + +/* RAII is nice */ +struct XcbGrabber { + XcbGrabber(::xcb_connection_t *conn) { ::xcb_grab_server(conn); conn_ = conn; } + ~XcbGrabber() { ::xcb_ungrab_server(conn_); } + +private: + ::xcb_connection_t *conn_; +}; + +static ::xcb_window_t GetSelectionOwner(::xcb_connection_t *conn, ::xcb_atom_t selection) { + ::xcb_window_t owner = XCB_NONE; + MallocPtr<::xcb_get_selection_owner_reply_t> reply(::xcb_get_selection_owner_reply(conn, ::xcb_get_selection_owner(conn, selection), nullptr)); + + if (reply) + owner = reply->owner; + + return owner; +} + +static bool GetRawSettingsData(std::vector<uint8_t>& bytes) { + int screen; + + XcbConnectionPtr conn(::xcb_connect(nullptr, &screen)); + if (::xcb_connection_has_error(conn.get())) + return false; + + /* get our needed atoms, available as atoms[Atom] */ + enum Atom { + XSETTINGS_SCREEN, /* _XSETTINGS_S[N] */ + XSETTINGS_SETTINGS, /* _XSETTINGS_SETTINGS */ + }; + + std::map<Atom, ::xcb_atom_t> atoms; + { + std::map<Atom, std::string> names = { + {XSETTINGS_SCREEN, fmt::format("_XSETTINGS_S{}", screen)}, + {XSETTINGS_SETTINGS, "_XSETTINGS_SETTINGS"}, + }; + + std::map<Atom, ::xcb_intern_atom_cookie_t> atom_cookies; + for (const auto& name : names) + atom_cookies[name.first] = ::xcb_intern_atom(conn.get(), false, name.second.size(), name.second.data()); + + for (const auto& cookie : atom_cookies) { + MallocPtr<::xcb_intern_atom_reply_t> reply(::xcb_intern_atom_reply(conn.get(), cookie.second, nullptr)); + if (!reply || reply->atom == XCB_NONE) + return false; + + atoms[cookie.first] = reply->atom; + } + } + + MallocPtr<xcb_get_property_reply_t> reply; + { + /* grab the X server as *required* by xsettings docs */ + const XcbGrabber grabber(conn.get()); + + ::xcb_window_t win = GetSelectionOwner(conn.get(), atoms[XSETTINGS_SCREEN]); + if (win == XCB_NONE) + return false; + + reply.reset(::xcb_get_property_reply(conn.get(), ::xcb_get_property(conn.get(), 0, win, atoms[XSETTINGS_SETTINGS], XCB_ATOM_ANY, 0L, UINT_MAX), nullptr)); + }; + if (!reply) + return false; + + uint8_t *data = reinterpret_cast<uint8_t *>(xcb_get_property_value(reply.get())); + int size = xcb_get_property_value_length(reply.get()); + if (size < 0) + return false; + + bytes.assign(data, data + size); + + return true; +} + +/* ------------------------------------------------------------------------- */ +/* now for the actual all-important public API stringing all this together */ + +bool GetSettings(std::vector<SettingsItem>& settings) { + std::vector<std::uint8_t> xsettings_raw; + if (!GetRawSettingsData(xsettings_raw)) + return false; + + Parser parser(xsettings_raw.data(), xsettings_raw.size()); + settings = parser.ParseAllItems(); + + return true; +} + +bool FindSetting(const std::string& name, SettingsItem& setting) { + std::vector<std::uint8_t> xsettings_raw; + if (!GetRawSettingsData(xsettings_raw)) + return false; + + Parser parser(xsettings_raw.data(), xsettings_raw.size()); + + std::uint32_t total = parser.GetTotalItems(); + + for (; total; total--) { + std::optional<SettingsItem> opt_item = parser.ParseNextItem(); + if (!opt_item) + return false; + + SettingsItem& item = opt_item.value(); + if (item.name == name) { + setting = item; + return true; + } + } + + return false; +} + +} // namespace x11