Flutter Engine
 
Loading...
Searching...
No Matches
fl_settings_portal.cc
Go to the documentation of this file.
1// Copyright 2013 The Flutter Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
6
7#include <gio/gio.h>
8#include <glib.h>
9
10static constexpr char kPortalName[] = "org.freedesktop.portal.Desktop";
11static constexpr char kPortalPath[] = "/org/freedesktop/portal/desktop";
12static constexpr char kPortalSettings[] = "org.freedesktop.portal.Settings";
13
14struct FlSetting {
15 const gchar* ns;
16 const gchar* key;
17 const GVariantType* type;
18};
19
20static constexpr char kXdgAppearance[] = "org.freedesktop.appearance";
21static const FlSetting kColorScheme = {
23 "color-scheme",
24 G_VARIANT_TYPE_UINT32,
25};
26
27static constexpr char kGnomeA11yInterface[] =
28 "org.gnome.desktop.a11y.interface";
29static const FlSetting kHighContrast = {
31 "high-contrast",
32 G_VARIANT_TYPE_BOOLEAN,
33};
34
35static constexpr char kGnomeDesktopInterface[] = "org.gnome.desktop.interface";
36static const FlSetting kClockFormat = {
38 "clock-format",
39 G_VARIANT_TYPE_STRING,
40};
43 "enable-animations",
44 G_VARIANT_TYPE_BOOLEAN,
45};
46static const FlSetting kGtkTheme = {
48 "gtk-theme",
49 G_VARIANT_TYPE_STRING,
50};
53 "text-scaling-factor",
54 G_VARIANT_TYPE_DOUBLE,
55};
56
61
62static constexpr char kClockFormat12Hour[] = "12h";
63static constexpr char kGtkThemeDarkSuffix[] = "-dark";
64
70
73
74 GDBusProxy* dbus_proxy;
75 GVariantDict* values;
76};
77
78static void fl_settings_portal_iface_init(FlSettingsInterface* iface);
79
80G_DEFINE_TYPE_WITH_CODE(FlSettingsPortal,
81 fl_settings_portal,
82 G_TYPE_OBJECT,
83 G_IMPLEMENT_INTERFACE(fl_settings_get_type(),
85
86static gchar* format_key(const FlSetting* setting) {
87 return g_strconcat(setting->ns, "::", setting->key, nullptr);
88}
89
90static gboolean get_value(FlSettingsPortal* portal,
91 const FlSetting* setting,
92 GVariant** value) {
93 g_autofree const gchar* key = format_key(setting);
94 *value = g_variant_dict_lookup_value(portal->values, key, setting->type);
95 return *value != nullptr;
96}
97
98static void set_value(FlSettingsPortal* portal,
99 const FlSetting* setting,
100 GVariant* value) {
101 g_autofree const gchar* key = format_key(setting);
102
103 // ignore redundant changes from multiple XDG desktop portal backends
104 g_autoptr(GVariant) old_value =
105 g_variant_dict_lookup_value(portal->values, key, nullptr);
106 if (old_value != nullptr && value != nullptr &&
107 g_variant_equal(old_value, value)) {
108 return;
109 }
110
111 g_variant_dict_insert_value(portal->values, key, value);
112 fl_settings_emit_changed(FL_SETTINGS(portal));
113}
114
115// Based on
116// https://gitlab.gnome.org/GNOME/Initiatives/-/wikis/Dark-Style-Preference#other
117static gboolean settings_portal_read(GDBusProxy* proxy,
118 const gchar* ns,
119 const gchar* key,
120 GVariant** out) {
121 g_autoptr(GError) error = nullptr;
122 g_autoptr(GVariant) value =
123 g_dbus_proxy_call_sync(proxy, "Read", g_variant_new("(ss)", ns, key),
124 G_DBUS_CALL_FLAGS_NONE, G_MAXINT, nullptr, &error);
125
126 if (error) {
127 if (error->domain == G_DBUS_ERROR &&
128 error->code == G_DBUS_ERROR_SERVICE_UNKNOWN) {
129 g_debug("XDG desktop portal unavailable: %s", error->message);
130 return FALSE;
131 }
132
133 if (error->domain == G_DBUS_ERROR &&
134 error->code == G_DBUS_ERROR_UNKNOWN_METHOD) {
135 g_debug("XDG desktop portal settings unavailable: %s", error->message);
136 return FALSE;
137 }
138
139 g_critical("Failed to read XDG desktop portal settings: %s",
140 error->message);
141 return FALSE;
142 }
143
144 g_autoptr(GVariant) child = nullptr;
145 g_variant_get(value, "(v)", &child);
146 g_variant_get(child, "v", out);
147
148 return TRUE;
149}
150
151static void settings_portal_changed_cb(GDBusProxy* proxy,
152 const char* sender_name,
153 const char* signal_name,
154 GVariant* parameters,
155 gpointer user_data) {
156 FlSettingsPortal* portal = FL_SETTINGS_PORTAL(user_data);
157 if (g_strcmp0(signal_name, "SettingChanged")) {
158 return;
159 }
160
161 FlSetting setting;
162 g_autoptr(GVariant) value = nullptr;
163 g_variant_get(parameters, "(&s&sv)", &setting.ns, &setting.key, &value);
164 set_value(portal, &setting, value);
165}
166
168 FlSettingsPortal* self = FL_SETTINGS_PORTAL(settings);
169
170 FlClockFormat clock_format = FL_CLOCK_FORMAT_24H;
171
172 g_autoptr(GVariant) value = nullptr;
173 if (get_value(self, &kClockFormat, &value)) {
174 const gchar* clock_format_str = g_variant_get_string(value, nullptr);
175 if (g_strcmp0(clock_format_str, kClockFormat12Hour) == 0) {
176 clock_format = FL_CLOCK_FORMAT_12H;
177 }
178 }
179
180 return clock_format;
181}
182
184 FlSettingsPortal* self = FL_SETTINGS_PORTAL(settings);
185
187
188 g_autoptr(GVariant) value = nullptr;
189 if (get_value(self, &kColorScheme, &value)) {
190 if (g_variant_get_uint32(value) == COLOR_SCHEME_PREFER_DARK) {
191 color_scheme = FL_COLOR_SCHEME_DARK;
192 }
193 } else if (get_value(self, &kGtkTheme, &value)) {
194 const gchar* gtk_theme_str = g_variant_get_string(value, nullptr);
195 if (g_str_has_suffix(gtk_theme_str, kGtkThemeDarkSuffix)) {
196 color_scheme = FL_COLOR_SCHEME_DARK;
197 }
198 }
199
200 return color_scheme;
201}
202
203static gboolean fl_settings_portal_get_enable_animations(FlSettings* settings) {
204 FlSettingsPortal* self = FL_SETTINGS_PORTAL(settings);
205
206 gboolean enable_animations = true;
207
208 g_autoptr(GVariant) value = nullptr;
210 enable_animations = g_variant_get_boolean(value);
211 }
212
213 return enable_animations;
214}
215
216static gboolean fl_settings_portal_get_high_contrast(FlSettings* settings) {
217 FlSettingsPortal* self = FL_SETTINGS_PORTAL(settings);
218
219 gboolean high_contrast = false;
220
221 g_autoptr(GVariant) value = nullptr;
223 high_contrast = g_variant_get_boolean(value);
224 }
225
226 return high_contrast;
227}
228
230 FlSettings* settings) {
231 FlSettingsPortal* self = FL_SETTINGS_PORTAL(settings);
232
233 gdouble scaling_factor = 1.0;
234
235 g_autoptr(GVariant) value = nullptr;
237 scaling_factor = g_variant_get_double(value);
238 }
239
240 return scaling_factor;
241}
242
243static void fl_settings_portal_dispose(GObject* object) {
244 FlSettingsPortal* self = FL_SETTINGS_PORTAL(object);
245
246 g_clear_object(&self->dbus_proxy);
247 g_clear_pointer(&self->values, g_variant_dict_unref);
248
249 G_OBJECT_CLASS(fl_settings_portal_parent_class)->dispose(object);
250}
251
252static void fl_settings_portal_class_init(FlSettingsPortalClass* klass) {
253 GObjectClass* object_class = G_OBJECT_CLASS(klass);
254 object_class->dispose = fl_settings_portal_dispose;
255}
256
257static void fl_settings_portal_iface_init(FlSettingsInterface* iface) {
258 iface->get_clock_format = fl_settings_portal_get_clock_format;
259 iface->get_color_scheme = fl_settings_portal_get_color_scheme;
260 iface->get_enable_animations = fl_settings_portal_get_enable_animations;
261 iface->get_high_contrast = fl_settings_portal_get_high_contrast;
262 iface->get_text_scaling_factor = fl_settings_portal_get_text_scaling_factor;
263}
264
265static void fl_settings_portal_init(FlSettingsPortal* self) {}
266
267FlSettingsPortal* fl_settings_portal_new() {
268 g_autoptr(GVariantDict) values = g_variant_dict_new(nullptr);
270}
271
272FlSettingsPortal* fl_settings_portal_new_with_values(GVariantDict* values) {
273 g_return_val_if_fail(values != nullptr, nullptr);
274 FlSettingsPortal* portal =
275 FL_SETTINGS_PORTAL(g_object_new(fl_settings_portal_get_type(), nullptr));
276 portal->values = g_variant_dict_ref(values);
277 return portal;
278}
279
280gboolean fl_settings_portal_start(FlSettingsPortal* self, GError** error) {
281 g_return_val_if_fail(FL_IS_SETTINGS_PORTAL(self), false);
282 g_return_val_if_fail(self->dbus_proxy == nullptr, false);
283
284 self->dbus_proxy = g_dbus_proxy_new_for_bus_sync(
285 G_BUS_TYPE_SESSION, G_DBUS_PROXY_FLAGS_NONE, nullptr, kPortalName,
287
288 if (self->dbus_proxy == nullptr) {
289 return FALSE;
290 }
291
292 for (const FlSetting setting : kAllSettings) {
293 g_autoptr(GVariant) value = nullptr;
294 if (settings_portal_read(self->dbus_proxy, setting.ns, setting.key,
295 &value)) {
296 set_value(self, &setting, value);
297 }
298 }
299
300 g_signal_connect_object(self->dbus_proxy, "g-signal",
301 G_CALLBACK(settings_portal_changed_cb), self,
302 static_cast<GConnectFlags>(0));
303
304 return true;
305}
int32_t value
struct @13 g_debug
g_autoptr(GMutexLocker) locker
return TRUE
const uint8_t uint32_t uint32_t GError ** error
void fl_settings_emit_changed(FlSettings *self)
FlColorScheme
Definition fl_settings.h:33
@ FL_COLOR_SCHEME_DARK
Definition fl_settings.h:35
@ FL_COLOR_SCHEME_LIGHT
Definition fl_settings.h:34
FlClockFormat
Definition fl_settings.h:21
@ FL_CLOCK_FORMAT_12H
Definition fl_settings.h:22
@ FL_CLOCK_FORMAT_24H
Definition fl_settings.h:23
static void fl_settings_portal_class_init(FlSettingsPortalClass *klass)
static gboolean settings_portal_read(GDBusProxy *proxy, const gchar *ns, const gchar *key, GVariant **out)
static constexpr char kGnomeA11yInterface[]
static constexpr char kXdgAppearance[]
static const FlSetting kColorScheme
static gdouble fl_settings_portal_get_text_scaling_factor(FlSettings *settings)
static constexpr char kGnomeDesktopInterface[]
static gboolean get_value(FlSettingsPortal *portal, const FlSetting *setting, GVariant **value)
static void fl_settings_portal_iface_init(FlSettingsInterface *iface)
static void set_value(FlSettingsPortal *portal, const FlSetting *setting, GVariant *value)
static void settings_portal_changed_cb(GDBusProxy *proxy, const char *sender_name, const char *signal_name, GVariant *parameters, gpointer user_data)
static void fl_settings_portal_dispose(GObject *object)
static const FlSetting kAllSettings[]
@ COLOR_SCHEME_DEFAULT
@ COLOR_SCHEME_PREFER_LIGHT
@ COLOR_SCHEME_PREFER_DARK
static constexpr char kPortalPath[]
static const FlSetting kClockFormat
static void fl_settings_portal_init(FlSettingsPortal *self)
static constexpr char kClockFormat12Hour[]
static const FlSetting kGtkTheme
static const FlSetting kTextScalingFactor
static FlColorScheme fl_settings_portal_get_color_scheme(FlSettings *settings)
static gboolean fl_settings_portal_get_high_contrast(FlSettings *settings)
gboolean fl_settings_portal_start(FlSettingsPortal *self, GError **error)
static const FlSetting kEnableAnimations
static const FlSetting kHighContrast
static constexpr char kGtkThemeDarkSuffix[]
FlSettingsPortal * fl_settings_portal_new()
static constexpr char kPortalName[]
static FlClockFormat fl_settings_portal_get_clock_format(FlSettings *settings)
G_DEFINE_TYPE_WITH_CODE(FlSettingsPortal, fl_settings_portal, G_TYPE_OBJECT, G_IMPLEMENT_INTERFACE(fl_settings_get_type(), fl_settings_portal_iface_init)) static gchar *format_key(const FlSetting *setting)
FlSettingsPortal * fl_settings_portal_new_with_values(GVariantDict *values)
static constexpr char kPortalSettings[]
static gboolean fl_settings_portal_get_enable_animations(FlSettings *settings)
const gchar * key
const gchar * ns
const GVariantType * type