7641c6f952
2002-10-03 Havoc Pennington <hp@pobox.com> Button-reordering patch. Has all the code except actually installing a gconf schema and reading the gconf key in prefs.c. metacity-theme-viewer displays the button layouts for testing themes. * src/preview-widget.c (meta_preview_size_request): make up a width/height if no child widget * src/prefs.c (meta_prefs_get_button_layout): new function * src/frames.c: get the button layout from prefs and use it when drawing * src/theme.c (meta_frame_layout_calc_geometry): enhance to be able to lay out buttons in different arrangements (button_rect): draw the new button background rectangles (meta_theme_draw_frame): require a button layout argument (meta_theme_calc_geometry): pass in the button layout * src/preview-widget.h: mod to handle button layouts * src/theme-viewer.c: mod to handle button layouts
1238 lines
30 KiB
C
1238 lines
30 KiB
C
/* Metacity preferences */
|
|
|
|
/*
|
|
* Copyright (C) 2001 Havoc Pennington, Copyright (C) 2002 Red Hat Inc.
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License as
|
|
* published by the Free Software Foundation; either version 2 of the
|
|
* License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful, but
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
|
* 02111-1307, USA.
|
|
*/
|
|
|
|
#include <config.h>
|
|
#include "prefs.h"
|
|
#include "ui.h"
|
|
#include "util.h"
|
|
#include <gconf/gconf-client.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
|
|
/* If you add a key, it needs updating in init() and in the gconf
|
|
* notify listener and of course in the .schemas file
|
|
*/
|
|
#define KEY_FOCUS_MODE "/apps/metacity/general/focus_mode"
|
|
#define KEY_AUTO_RAISE "/apps/metacity/general/auto_raise"
|
|
#define KEY_AUTO_RAISE_DELAY "/apps/metacity/general/auto_raise_delay"
|
|
#define KEY_THEME "/apps/metacity/general/theme"
|
|
#define KEY_USE_SYSTEM_FONT "/apps/metacity/general/titlebar_uses_system_font"
|
|
#define KEY_TITLEBAR_FONT "/apps/metacity/general/titlebar_font"
|
|
#define KEY_NUM_WORKSPACES "/apps/metacity/general/num_workspaces"
|
|
#define KEY_APPLICATION_BASED "/apps/metacity/general/application_based"
|
|
#define KEY_DISABLE_WORKAROUNDS "/apps/metacity/general/disable_workarounds"
|
|
|
|
#define KEY_COMMAND_PREFIX "/apps/metacity/keybinding_commands/command_"
|
|
#define KEY_SCREEN_BINDINGS_PREFIX "/apps/metacity/global_keybindings"
|
|
#define KEY_WINDOW_BINDINGS_PREFIX "/apps/metacity/window_keybindings"
|
|
|
|
static GConfClient *default_client = NULL;
|
|
static GList *listeners = NULL;
|
|
static GList *changes = NULL;
|
|
static guint changed_idle;
|
|
static gboolean use_system_font = TRUE;
|
|
static PangoFontDescription *titlebar_font = NULL;
|
|
static MetaFocusMode focus_mode = META_FOCUS_MODE_CLICK;
|
|
static char* current_theme = NULL;
|
|
static int num_workspaces = 4;
|
|
static gboolean application_based = FALSE;
|
|
static gboolean disable_workarounds = FALSE;
|
|
static gboolean auto_raise = FALSE;
|
|
static gboolean auto_raise_delay = 500;
|
|
#define NUM_COMMANDS 12
|
|
static char *commands[NUM_COMMANDS] = { NULL, NULL, NULL, NULL, NULL, NULL,
|
|
NULL, NULL, NULL, NULL, NULL, NULL };
|
|
|
|
static gboolean update_use_system_font (gboolean value);
|
|
static gboolean update_titlebar_font (const char *value);
|
|
static gboolean update_focus_mode (const char *value);
|
|
static gboolean update_theme (const char *value);
|
|
static gboolean update_num_workspaces (int value);
|
|
static gboolean update_application_based (gboolean value);
|
|
static gboolean update_disable_workarounds (gboolean value);
|
|
static gboolean update_auto_raise (gboolean value);
|
|
static gboolean update_auto_raise_delay (int value);
|
|
static gboolean update_window_binding (const char *name,
|
|
const char *value);
|
|
static gboolean update_screen_binding (const char *name,
|
|
const char *value);
|
|
static void init_bindings (void);
|
|
static gboolean update_binding (MetaKeyPref *binding,
|
|
const char *value);
|
|
static gboolean update_command (const char *name,
|
|
const char *value);
|
|
static void init_commands (void);
|
|
|
|
static void queue_changed (MetaPreference pref);
|
|
static void change_notify (GConfClient *client,
|
|
guint cnxn_id,
|
|
GConfEntry *entry,
|
|
gpointer user_data);
|
|
|
|
|
|
|
|
|
|
typedef struct
|
|
{
|
|
MetaPrefsChangedFunc func;
|
|
gpointer data;
|
|
} MetaPrefsListener;
|
|
|
|
void
|
|
meta_prefs_add_listener (MetaPrefsChangedFunc func,
|
|
gpointer data)
|
|
{
|
|
MetaPrefsListener *l;
|
|
|
|
l = g_new (MetaPrefsListener, 1);
|
|
l->func = func;
|
|
l->data = data;
|
|
|
|
listeners = g_list_prepend (listeners, l);
|
|
}
|
|
|
|
void
|
|
meta_prefs_remove_listener (MetaPrefsChangedFunc func,
|
|
gpointer data)
|
|
{
|
|
GList *tmp;
|
|
|
|
tmp = listeners;
|
|
while (tmp != NULL)
|
|
{
|
|
MetaPrefsListener *l = tmp->data;
|
|
|
|
if (l->func == func &&
|
|
l->data == data)
|
|
{
|
|
g_free (l);
|
|
listeners = g_list_delete_link (listeners, tmp);
|
|
|
|
return;
|
|
}
|
|
|
|
tmp = tmp->next;
|
|
}
|
|
|
|
meta_bug ("Did not find listener to remove\n");
|
|
}
|
|
|
|
static void
|
|
emit_changed (MetaPreference pref)
|
|
{
|
|
GList *tmp;
|
|
GList *copy;
|
|
|
|
meta_verbose ("Notifying listeners that pref %s changed\n",
|
|
meta_preference_to_string (pref));
|
|
|
|
copy = g_list_copy (listeners);
|
|
|
|
tmp = copy;
|
|
while (tmp != NULL)
|
|
{
|
|
MetaPrefsListener *l = tmp->data;
|
|
|
|
(* l->func) (pref, l->data);
|
|
|
|
tmp = tmp->next;
|
|
}
|
|
|
|
g_list_free (copy);
|
|
}
|
|
|
|
static gboolean
|
|
changed_idle_handler (gpointer data)
|
|
{
|
|
GList *tmp;
|
|
GList *copy;
|
|
|
|
changed_idle = 0;
|
|
|
|
copy = g_list_copy (changes); /* reentrancy paranoia */
|
|
|
|
g_list_free (changes);
|
|
changes = NULL;
|
|
|
|
tmp = copy;
|
|
while (tmp != NULL)
|
|
{
|
|
MetaPreference pref = GPOINTER_TO_INT (tmp->data);
|
|
|
|
emit_changed (pref);
|
|
|
|
tmp = tmp->next;
|
|
}
|
|
|
|
g_list_free (copy);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
queue_changed (MetaPreference pref)
|
|
{
|
|
meta_verbose ("Queueing change of pref %s\n",
|
|
meta_preference_to_string (pref));
|
|
|
|
if (g_list_find (changes, GINT_TO_POINTER (pref)) == NULL)
|
|
changes = g_list_prepend (changes, GINT_TO_POINTER (pref));
|
|
else
|
|
meta_verbose ("Change of pref %s was already pending\n",
|
|
meta_preference_to_string (pref));
|
|
|
|
/* add idle at priority below the gconf notify idle */
|
|
if (changed_idle == 0)
|
|
changed_idle = g_idle_add_full (META_PRIORITY_PREFS_NOTIFY,
|
|
changed_idle_handler, NULL, NULL);
|
|
}
|
|
|
|
static void
|
|
cleanup_error (GError **error)
|
|
{
|
|
if (*error)
|
|
{
|
|
meta_warning ("%s\n", (*error)->message);
|
|
|
|
g_error_free (*error);
|
|
*error = NULL;
|
|
}
|
|
}
|
|
|
|
void
|
|
meta_prefs_init (void)
|
|
{
|
|
GError *err = NULL;
|
|
char *str_val;
|
|
int int_val;
|
|
gboolean bool_val;
|
|
|
|
if (default_client != NULL)
|
|
return;
|
|
|
|
/* returns a reference which we hold forever */
|
|
default_client = gconf_client_get_default ();
|
|
|
|
gconf_client_add_dir (default_client, "/apps/metacity",
|
|
GCONF_CLIENT_PRELOAD_RECURSIVE,
|
|
&err);
|
|
cleanup_error (&err);
|
|
|
|
str_val = gconf_client_get_string (default_client, KEY_FOCUS_MODE,
|
|
&err);
|
|
cleanup_error (&err);
|
|
update_focus_mode (str_val);
|
|
g_free (str_val);
|
|
|
|
bool_val = gconf_client_get_bool (default_client, KEY_AUTO_RAISE,
|
|
&err);
|
|
cleanup_error (&err);
|
|
update_auto_raise (bool_val);
|
|
|
|
int_val = gconf_client_get_int (default_client, KEY_AUTO_RAISE_DELAY,
|
|
&err);
|
|
cleanup_error (&err);
|
|
update_auto_raise_delay (int_val);
|
|
|
|
|
|
str_val = gconf_client_get_string (default_client, KEY_THEME,
|
|
&err);
|
|
cleanup_error (&err);
|
|
update_theme (str_val);
|
|
g_free (str_val);
|
|
|
|
/* If the keys aren't set in the database, we use essentially
|
|
* bogus values instead of any kind of default. This is
|
|
* just lazy. But they keys ought to be set, anyhow.
|
|
*/
|
|
|
|
bool_val = gconf_client_get_bool (default_client, KEY_USE_SYSTEM_FONT,
|
|
&err);
|
|
cleanup_error (&err);
|
|
update_use_system_font (bool_val);
|
|
|
|
str_val = gconf_client_get_string (default_client, KEY_TITLEBAR_FONT,
|
|
&err);
|
|
cleanup_error (&err);
|
|
update_titlebar_font (str_val);
|
|
g_free (str_val);
|
|
|
|
int_val = gconf_client_get_int (default_client, KEY_NUM_WORKSPACES,
|
|
&err);
|
|
cleanup_error (&err);
|
|
update_num_workspaces (int_val);
|
|
|
|
bool_val = gconf_client_get_bool (default_client, KEY_APPLICATION_BASED,
|
|
&err);
|
|
cleanup_error (&err);
|
|
update_application_based (bool_val);
|
|
|
|
bool_val = gconf_client_get_bool (default_client, KEY_DISABLE_WORKAROUNDS,
|
|
&err);
|
|
cleanup_error (&err);
|
|
update_disable_workarounds (bool_val);
|
|
|
|
/* Load keybindings prefs */
|
|
init_bindings ();
|
|
|
|
/* commands */
|
|
init_commands ();
|
|
|
|
gconf_client_notify_add (default_client, "/apps/metacity",
|
|
change_notify,
|
|
NULL,
|
|
NULL,
|
|
&err);
|
|
cleanup_error (&err);
|
|
}
|
|
|
|
/* from eel */
|
|
static gboolean
|
|
str_has_prefix (const char *haystack, const char *needle)
|
|
{
|
|
const char *h, *n;
|
|
|
|
/* Eat one character at a time. */
|
|
h = haystack == NULL ? "" : haystack;
|
|
n = needle == NULL ? "" : needle;
|
|
do
|
|
{
|
|
if (*n == '\0')
|
|
return TRUE;
|
|
if (*h == '\0')
|
|
return FALSE;
|
|
}
|
|
while (*h++ == *n++);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
change_notify (GConfClient *client,
|
|
guint cnxn_id,
|
|
GConfEntry *entry,
|
|
gpointer user_data)
|
|
{
|
|
const char *key;
|
|
GConfValue *value;
|
|
|
|
key = gconf_entry_get_key (entry);
|
|
value = gconf_entry_get_value (entry);
|
|
|
|
if (strcmp (key, KEY_FOCUS_MODE) == 0)
|
|
{
|
|
const char *str;
|
|
|
|
if (value && value->type != GCONF_VALUE_STRING)
|
|
{
|
|
meta_warning (_("GConf key \"%s\" is set to an invalid type\n"),
|
|
KEY_FOCUS_MODE);
|
|
goto out;
|
|
}
|
|
|
|
str = value ? gconf_value_get_string (value) : NULL;
|
|
|
|
if (update_focus_mode (str))
|
|
queue_changed (META_PREF_FOCUS_MODE);
|
|
}
|
|
if (strcmp (key, KEY_THEME) == 0)
|
|
{
|
|
const char *str;
|
|
|
|
if (value && value->type != GCONF_VALUE_STRING)
|
|
{
|
|
meta_warning (_("GConf key \"%s\" is set to an invalid type\n"),
|
|
KEY_THEME);
|
|
goto out;
|
|
}
|
|
|
|
str = value ? gconf_value_get_string (value) : NULL;
|
|
|
|
if (update_theme (str))
|
|
queue_changed (META_PREF_THEME);
|
|
}
|
|
else if (strcmp (key, KEY_TITLEBAR_FONT) == 0)
|
|
{
|
|
const char *str;
|
|
|
|
if (value && value->type != GCONF_VALUE_STRING)
|
|
{
|
|
meta_warning (_("GConf key \"%s\" is set to an invalid type\n"),
|
|
KEY_TITLEBAR_FONT);
|
|
goto out;
|
|
}
|
|
|
|
str = value ? gconf_value_get_string (value) : NULL;
|
|
|
|
if (update_titlebar_font (str))
|
|
queue_changed (META_PREF_TITLEBAR_FONT);
|
|
}
|
|
else if (strcmp (key, KEY_USE_SYSTEM_FONT) == 0)
|
|
{
|
|
gboolean b;
|
|
|
|
if (value && value->type != GCONF_VALUE_BOOL)
|
|
{
|
|
meta_warning (_("GConf key \"%s\" is set to an invalid type\n"),
|
|
KEY_USE_SYSTEM_FONT);
|
|
goto out;
|
|
}
|
|
|
|
b = value ? gconf_value_get_bool (value) : TRUE;
|
|
|
|
/* There's no external pref for this, it just affects whether
|
|
* get_titlebar_font returns NULL, so that's what we queue
|
|
* the change on
|
|
*/
|
|
if (update_use_system_font (b))
|
|
queue_changed (META_PREF_TITLEBAR_FONT);
|
|
}
|
|
else if (strcmp (key, KEY_NUM_WORKSPACES) == 0)
|
|
{
|
|
int d;
|
|
|
|
if (value && value->type != GCONF_VALUE_INT)
|
|
{
|
|
meta_warning (_("GConf key \"%s\" is set to an invalid type\n"),
|
|
KEY_NUM_WORKSPACES);
|
|
goto out;
|
|
}
|
|
|
|
d = value ? gconf_value_get_int (value) : num_workspaces;
|
|
|
|
if (update_num_workspaces (d))
|
|
queue_changed (META_PREF_NUM_WORKSPACES);
|
|
}
|
|
else if (strcmp (key, KEY_APPLICATION_BASED) == 0)
|
|
{
|
|
gboolean b;
|
|
|
|
if (value && value->type != GCONF_VALUE_BOOL)
|
|
{
|
|
meta_warning (_("GConf key \"%s\" is set to an invalid type\n"),
|
|
KEY_APPLICATION_BASED);
|
|
goto out;
|
|
}
|
|
|
|
b = value ? gconf_value_get_bool (value) : application_based;
|
|
|
|
if (update_application_based (b))
|
|
queue_changed (META_PREF_APPLICATION_BASED);
|
|
}
|
|
else if (strcmp (key, KEY_DISABLE_WORKAROUNDS) == 0)
|
|
{
|
|
gboolean b;
|
|
|
|
if (value && value->type != GCONF_VALUE_BOOL)
|
|
{
|
|
meta_warning (_("GConf key \"%s\" is set to an invalid type\n"),
|
|
KEY_APPLICATION_BASED);
|
|
goto out;
|
|
}
|
|
|
|
b = value ? gconf_value_get_bool (value) : disable_workarounds;
|
|
|
|
if (update_disable_workarounds (b))
|
|
queue_changed (META_PREF_DISABLE_WORKAROUNDS);
|
|
}
|
|
else if (str_has_prefix (key, KEY_WINDOW_BINDINGS_PREFIX))
|
|
{
|
|
const char *str;
|
|
|
|
if (value && value->type != GCONF_VALUE_STRING)
|
|
{
|
|
meta_warning (_("GConf key \"%s\" is set to an invalid type\n"),
|
|
key);
|
|
goto out;
|
|
}
|
|
|
|
str = value ? gconf_value_get_string (value) : NULL;
|
|
|
|
if (update_window_binding (key, str))
|
|
queue_changed (META_PREF_WINDOW_KEYBINDINGS);
|
|
}
|
|
else if (str_has_prefix (key, KEY_SCREEN_BINDINGS_PREFIX))
|
|
{
|
|
const char *str;
|
|
|
|
if (value && value->type != GCONF_VALUE_STRING)
|
|
{
|
|
meta_warning (_("GConf key \"%s\" is set to an invalid type\n"),
|
|
key);
|
|
goto out;
|
|
}
|
|
|
|
str = value ? gconf_value_get_string (value) : NULL;
|
|
|
|
if (update_screen_binding (key, str))
|
|
queue_changed (META_PREF_SCREEN_KEYBINDINGS);
|
|
}
|
|
else if (strcmp (key, KEY_AUTO_RAISE) == 0)
|
|
{
|
|
gboolean b;
|
|
|
|
if (value && value->type != GCONF_VALUE_BOOL)
|
|
{
|
|
meta_warning (_("GConf key \"%s\" is set to an invalid type\n"),
|
|
KEY_AUTO_RAISE);
|
|
goto out;
|
|
}
|
|
|
|
b = value ? gconf_value_get_bool (value) : auto_raise;
|
|
|
|
if (update_auto_raise (b))
|
|
queue_changed (META_PREF_AUTO_RAISE);
|
|
}
|
|
else if (strcmp (key, KEY_AUTO_RAISE_DELAY) == 0)
|
|
{
|
|
int d;
|
|
|
|
if (value && value->type != GCONF_VALUE_INT)
|
|
{
|
|
meta_warning (_("GConf key \"%s\" is set to an invalid type\n"),
|
|
KEY_AUTO_RAISE_DELAY);
|
|
goto out;
|
|
}
|
|
|
|
d = value ? gconf_value_get_int (value) : 0;
|
|
|
|
if (update_auto_raise_delay (d))
|
|
queue_changed (META_PREF_AUTO_RAISE_DELAY);
|
|
|
|
}
|
|
else if (str_has_prefix (key, KEY_COMMAND_PREFIX))
|
|
{
|
|
const char *str;
|
|
|
|
if (value && value->type != GCONF_VALUE_STRING)
|
|
{
|
|
meta_warning (_("GConf key \"%s\" is set to an invalid type\n"),
|
|
key);
|
|
goto out;
|
|
}
|
|
|
|
str = value ? gconf_value_get_string (value) : NULL;
|
|
|
|
if (update_command (key, str))
|
|
queue_changed (META_PREF_COMMANDS);
|
|
}
|
|
else
|
|
{
|
|
meta_verbose ("Key %s doesn't mean anything to Metacity\n",
|
|
key);
|
|
}
|
|
|
|
out:
|
|
/* nothing */
|
|
return; /* AIX compiler wants something after a label like out: */
|
|
}
|
|
|
|
static gboolean
|
|
update_focus_mode (const char *value)
|
|
{
|
|
MetaFocusMode old_mode = focus_mode;
|
|
|
|
if (value != NULL)
|
|
{
|
|
if (g_ascii_strcasecmp (value, "click") == 0)
|
|
focus_mode = META_FOCUS_MODE_CLICK;
|
|
else if (g_ascii_strcasecmp (value, "sloppy") == 0)
|
|
focus_mode = META_FOCUS_MODE_SLOPPY;
|
|
else if (g_ascii_strcasecmp (value, "mouse") == 0)
|
|
focus_mode = META_FOCUS_MODE_MOUSE;
|
|
else
|
|
meta_warning (_("GConf key '%s' is set to an invalid value"),
|
|
KEY_FOCUS_MODE);
|
|
}
|
|
|
|
return (old_mode != focus_mode);
|
|
}
|
|
|
|
static gboolean
|
|
update_theme (const char *value)
|
|
{
|
|
char *old_theme;
|
|
gboolean changed;
|
|
|
|
old_theme = current_theme;
|
|
|
|
if (value != NULL && *value)
|
|
{
|
|
current_theme = g_strdup (value);
|
|
}
|
|
|
|
changed = TRUE;
|
|
if ((old_theme && current_theme &&
|
|
strcmp (old_theme, current_theme) == 0) ||
|
|
(old_theme == NULL && current_theme == NULL))
|
|
changed = FALSE;
|
|
|
|
if (old_theme != current_theme)
|
|
g_free (old_theme);
|
|
|
|
if (current_theme == NULL)
|
|
{
|
|
/* Fallback crackrock */
|
|
current_theme = g_strdup ("Atlanta");
|
|
changed = TRUE;
|
|
}
|
|
|
|
return changed;
|
|
}
|
|
|
|
MetaFocusMode
|
|
meta_prefs_get_focus_mode (void)
|
|
{
|
|
return focus_mode;
|
|
}
|
|
|
|
const char*
|
|
meta_prefs_get_theme (void)
|
|
{
|
|
return current_theme;
|
|
}
|
|
|
|
static gboolean
|
|
update_use_system_font (gboolean value)
|
|
{
|
|
gboolean old = use_system_font;
|
|
|
|
use_system_font = value;
|
|
|
|
return old != value;
|
|
}
|
|
|
|
static gboolean
|
|
update_titlebar_font (const char *value)
|
|
{
|
|
PangoFontDescription *new_desc;
|
|
|
|
new_desc = NULL;
|
|
|
|
if (value)
|
|
{
|
|
new_desc = pango_font_description_from_string (value);
|
|
if (new_desc == NULL)
|
|
meta_warning (_("Could not parse font description \"%s\" from GConf key %s\n"),
|
|
value, KEY_TITLEBAR_FONT);
|
|
}
|
|
|
|
if (new_desc && titlebar_font &&
|
|
pango_font_description_equal (new_desc, titlebar_font))
|
|
{
|
|
pango_font_description_free (new_desc);
|
|
return FALSE;
|
|
}
|
|
else
|
|
{
|
|
if (titlebar_font)
|
|
pango_font_description_free (titlebar_font);
|
|
|
|
titlebar_font = new_desc;
|
|
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
const PangoFontDescription*
|
|
meta_prefs_get_titlebar_font (void)
|
|
{
|
|
if (use_system_font)
|
|
return NULL;
|
|
else
|
|
return titlebar_font;
|
|
}
|
|
|
|
#define MAX_REASONABLE_WORKSPACES 32
|
|
|
|
static gboolean
|
|
update_num_workspaces (int value)
|
|
{
|
|
int old = num_workspaces;
|
|
|
|
if (value < 1 || value > MAX_REASONABLE_WORKSPACES)
|
|
{
|
|
meta_warning (_("%d stored in GConf key %s is not a reasonable number of workspaces, current maximum is %d\n"),
|
|
value, KEY_NUM_WORKSPACES, MAX_REASONABLE_WORKSPACES);
|
|
if (value < 1)
|
|
value = 1;
|
|
else if (value > MAX_REASONABLE_WORKSPACES)
|
|
value = MAX_REASONABLE_WORKSPACES;
|
|
}
|
|
|
|
num_workspaces = value;
|
|
|
|
return old != num_workspaces;
|
|
}
|
|
|
|
int
|
|
meta_prefs_get_num_workspaces (void)
|
|
{
|
|
return num_workspaces;
|
|
}
|
|
|
|
static gboolean
|
|
update_application_based (gboolean value)
|
|
{
|
|
gboolean old = application_based;
|
|
|
|
application_based = value;
|
|
|
|
return old != application_based;
|
|
}
|
|
|
|
gboolean
|
|
meta_prefs_get_application_based (void)
|
|
{
|
|
return application_based;
|
|
}
|
|
|
|
static gboolean
|
|
update_disable_workarounds (gboolean value)
|
|
{
|
|
gboolean old = disable_workarounds;
|
|
|
|
disable_workarounds = value;
|
|
|
|
{
|
|
static gboolean first_disable = TRUE;
|
|
|
|
if (disable_workarounds && first_disable)
|
|
{
|
|
first_disable = FALSE;
|
|
|
|
meta_warning (_("Workarounds for broken applications disabled. Some applications may not behave properly.\n"));
|
|
}
|
|
}
|
|
|
|
return old != disable_workarounds;
|
|
}
|
|
|
|
gboolean
|
|
meta_prefs_get_disable_workarounds (void)
|
|
{
|
|
return disable_workarounds;
|
|
}
|
|
|
|
static gboolean
|
|
update_auto_raise (gboolean value)
|
|
{
|
|
gboolean old = auto_raise;
|
|
|
|
auto_raise = value;
|
|
|
|
return old != auto_raise;
|
|
}
|
|
|
|
#define MAX_REASONABLE_AUTO_RAISE_DELAY 10000
|
|
|
|
static gboolean
|
|
update_auto_raise_delay (int value)
|
|
{
|
|
int old = auto_raise_delay;
|
|
|
|
if (value < 0 || value > MAX_REASONABLE_AUTO_RAISE_DELAY)
|
|
{
|
|
meta_warning (_("%d stored in GConf key %s is out of range 0 to %d\n"),
|
|
value, KEY_AUTO_RAISE_DELAY,
|
|
MAX_REASONABLE_AUTO_RAISE_DELAY);
|
|
value = 0;
|
|
}
|
|
|
|
auto_raise_delay = value;
|
|
|
|
return old != auto_raise_delay;
|
|
}
|
|
|
|
const char*
|
|
meta_preference_to_string (MetaPreference pref)
|
|
{
|
|
switch (pref)
|
|
{
|
|
case META_PREF_FOCUS_MODE:
|
|
return "FOCUS_MODE";
|
|
|
|
case META_PREF_THEME:
|
|
return "THEME";
|
|
|
|
case META_PREF_TITLEBAR_FONT:
|
|
return "TITLEBAR_FONT";
|
|
|
|
case META_PREF_NUM_WORKSPACES:
|
|
return "NUM_WORKSPACES";
|
|
|
|
case META_PREF_APPLICATION_BASED:
|
|
return "APPLICATION_BASED";
|
|
|
|
case META_PREF_SCREEN_KEYBINDINGS:
|
|
return "SCREEN_KEYBINDINGS";
|
|
|
|
case META_PREF_WINDOW_KEYBINDINGS:
|
|
return "WINDOW_KEYBINDINGS";
|
|
|
|
case META_PREF_DISABLE_WORKAROUNDS:
|
|
return "DISABLE_WORKAROUNDS";
|
|
|
|
case META_PREF_AUTO_RAISE:
|
|
return "AUTO_RAISE";
|
|
|
|
case META_PREF_AUTO_RAISE_DELAY:
|
|
return "AUTO_RAISE_DELAY";
|
|
|
|
case META_PREF_COMMANDS:
|
|
return "COMMANDS";
|
|
|
|
case META_PREF_BUTTON_LAYOUT:
|
|
return "BUTTON_LAYOUT";
|
|
break;
|
|
}
|
|
|
|
return "(unknown)";
|
|
}
|
|
|
|
void
|
|
meta_prefs_set_num_workspaces (int n_workspaces)
|
|
{
|
|
GError *err;
|
|
|
|
if (default_client == NULL)
|
|
return;
|
|
|
|
if (n_workspaces < 1)
|
|
n_workspaces = 1;
|
|
if (n_workspaces > MAX_REASONABLE_WORKSPACES)
|
|
n_workspaces = MAX_REASONABLE_WORKSPACES;
|
|
|
|
err = NULL;
|
|
gconf_client_set_int (default_client,
|
|
KEY_NUM_WORKSPACES,
|
|
n_workspaces,
|
|
&err);
|
|
|
|
if (err)
|
|
{
|
|
meta_warning (_("Error setting number of workspaces to %d: %s\n"),
|
|
num_workspaces,
|
|
err->message);
|
|
g_error_free (err);
|
|
}
|
|
}
|
|
|
|
/* Indexes must correspond to MetaKeybindingAction */
|
|
static MetaKeyPref screen_bindings[] = {
|
|
{ META_KEYBINDING_WORKSPACE_1, 0, 0 },
|
|
{ META_KEYBINDING_WORKSPACE_2, 0, 0 },
|
|
{ META_KEYBINDING_WORKSPACE_3, 0, 0 },
|
|
{ META_KEYBINDING_WORKSPACE_4, 0, 0 },
|
|
{ META_KEYBINDING_WORKSPACE_5, 0, 0 },
|
|
{ META_KEYBINDING_WORKSPACE_6, 0, 0 },
|
|
{ META_KEYBINDING_WORKSPACE_7, 0, 0 },
|
|
{ META_KEYBINDING_WORKSPACE_8, 0, 0 },
|
|
{ META_KEYBINDING_WORKSPACE_9, 0, 0 },
|
|
{ META_KEYBINDING_WORKSPACE_10, 0, 0 },
|
|
{ META_KEYBINDING_WORKSPACE_11, 0, 0 },
|
|
{ META_KEYBINDING_WORKSPACE_12, 0, 0 },
|
|
{ META_KEYBINDING_WORKSPACE_LEFT, 0, 0 },
|
|
{ META_KEYBINDING_WORKSPACE_RIGHT, 0, 0 },
|
|
{ META_KEYBINDING_WORKSPACE_UP, 0, 0 },
|
|
{ META_KEYBINDING_WORKSPACE_DOWN, 0, 0 },
|
|
{ META_KEYBINDING_SWITCH_WINDOWS, 0, 0 },
|
|
{ META_KEYBINDING_SWITCH_PANELS, 0, 0 },
|
|
{ META_KEYBINDING_CYCLE_WINDOWS, 0, 0 },
|
|
{ META_KEYBINDING_CYCLE_PANELS, 0, 0 },
|
|
{ META_KEYBINDING_SHOW_DESKTOP, 0, 0 },
|
|
{ META_KEYBINDING_COMMAND_1, 0, 0 },
|
|
{ META_KEYBINDING_COMMAND_2, 0, 0 },
|
|
{ META_KEYBINDING_COMMAND_3, 0, 0 },
|
|
{ META_KEYBINDING_COMMAND_4, 0, 0 },
|
|
{ META_KEYBINDING_COMMAND_5, 0, 0 },
|
|
{ META_KEYBINDING_COMMAND_6, 0, 0 },
|
|
{ META_KEYBINDING_COMMAND_7, 0, 0 },
|
|
{ META_KEYBINDING_COMMAND_8, 0, 0 },
|
|
{ META_KEYBINDING_COMMAND_9, 0, 0 },
|
|
{ META_KEYBINDING_COMMAND_10, 0, 0 },
|
|
{ META_KEYBINDING_COMMAND_11, 0, 0 },
|
|
{ META_KEYBINDING_COMMAND_12, 0, 0 },
|
|
{ NULL, 0, 0 }
|
|
};
|
|
|
|
static MetaKeyPref window_bindings[] = {
|
|
{ META_KEYBINDING_WINDOW_MENU, 0, 0 },
|
|
{ META_KEYBINDING_TOGGLE_FULLSCREEN, 0, 0 },
|
|
{ META_KEYBINDING_TOGGLE_MAXIMIZE, 0, 0 },
|
|
{ META_KEYBINDING_MAXIMIZE, 0, 0 },
|
|
{ META_KEYBINDING_UNMAXIMIZE, 0, 0 },
|
|
{ META_KEYBINDING_TOGGLE_SHADE, 0, 0 },
|
|
{ META_KEYBINDING_MINIMIZE, 0, 0 },
|
|
{ META_KEYBINDING_CLOSE, 0, 0 },
|
|
{ META_KEYBINDING_BEGIN_MOVE, 0, 0 },
|
|
{ META_KEYBINDING_BEGIN_RESIZE, 0, 0 },
|
|
{ META_KEYBINDING_TOGGLE_STICKY, 0, 0 },
|
|
{ META_KEYBINDING_MOVE_WORKSPACE_1, 0, 0 },
|
|
{ META_KEYBINDING_MOVE_WORKSPACE_2, 0, 0 },
|
|
{ META_KEYBINDING_MOVE_WORKSPACE_3, 0, 0 },
|
|
{ META_KEYBINDING_MOVE_WORKSPACE_4, 0, 0 },
|
|
{ META_KEYBINDING_MOVE_WORKSPACE_5, 0, 0 },
|
|
{ META_KEYBINDING_MOVE_WORKSPACE_6, 0, 0 },
|
|
{ META_KEYBINDING_MOVE_WORKSPACE_7, 0, 0 },
|
|
{ META_KEYBINDING_MOVE_WORKSPACE_8, 0, 0 },
|
|
{ META_KEYBINDING_MOVE_WORKSPACE_9, 0, 0 },
|
|
{ META_KEYBINDING_MOVE_WORKSPACE_10, 0, 0 },
|
|
{ META_KEYBINDING_MOVE_WORKSPACE_11, 0, 0 },
|
|
{ META_KEYBINDING_MOVE_WORKSPACE_12, 0, 0 },
|
|
{ META_KEYBINDING_MOVE_WORKSPACE_LEFT, 0, 0 },
|
|
{ META_KEYBINDING_MOVE_WORKSPACE_RIGHT, 0, 0 },
|
|
{ META_KEYBINDING_MOVE_WORKSPACE_UP, 0, 0 },
|
|
{ META_KEYBINDING_MOVE_WORKSPACE_DOWN, 0, 0 },
|
|
{ META_KEYBINDING_RAISE_OR_LOWER, 0, 0 },
|
|
{ META_KEYBINDING_RAISE, 0, 0 },
|
|
{ META_KEYBINDING_LOWER, 0, 0 },
|
|
{ NULL, 0, 0 }
|
|
};
|
|
|
|
static void
|
|
init_bindings (void)
|
|
{
|
|
int i;
|
|
GError *err;
|
|
|
|
i = 0;
|
|
while (window_bindings[i].name)
|
|
{
|
|
char *str_val;
|
|
char *key;
|
|
|
|
key = g_strconcat (KEY_WINDOW_BINDINGS_PREFIX, "/",
|
|
window_bindings[i].name, NULL);
|
|
|
|
err = NULL;
|
|
str_val = gconf_client_get_string (default_client, key, &err);
|
|
cleanup_error (&err);
|
|
|
|
update_binding (&window_bindings[i], str_val);
|
|
|
|
g_free (str_val);
|
|
g_free (key);
|
|
|
|
++i;
|
|
}
|
|
|
|
i = 0;
|
|
while (screen_bindings[i].name)
|
|
{
|
|
char *str_val;
|
|
char *key;
|
|
|
|
key = g_strconcat (KEY_SCREEN_BINDINGS_PREFIX, "/",
|
|
screen_bindings[i].name, NULL);
|
|
|
|
err = NULL;
|
|
str_val = gconf_client_get_string (default_client, key, &err);
|
|
cleanup_error (&err);
|
|
|
|
update_binding (&screen_bindings[i], str_val);
|
|
|
|
g_free (str_val);
|
|
g_free (key);
|
|
|
|
++i;
|
|
}
|
|
}
|
|
|
|
static void
|
|
init_commands (void)
|
|
{
|
|
int i;
|
|
GError *err;
|
|
|
|
i = 0;
|
|
while (i < NUM_COMMANDS)
|
|
{
|
|
char *str_val;
|
|
char *key;
|
|
|
|
key = meta_prefs_get_gconf_key_for_command (i);
|
|
|
|
err = NULL;
|
|
str_val = gconf_client_get_string (default_client, key, &err);
|
|
cleanup_error (&err);
|
|
|
|
update_command (key, str_val);
|
|
|
|
g_free (str_val);
|
|
g_free (key);
|
|
|
|
++i;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
update_binding (MetaKeyPref *binding,
|
|
const char *value)
|
|
{
|
|
unsigned int keysym;
|
|
MetaVirtualModifier mods;
|
|
gboolean changed;
|
|
|
|
meta_topic (META_DEBUG_KEYBINDINGS,
|
|
"Binding \"%s\" has new gconf value \"%s\"\n",
|
|
binding->name, value ? value : "none");
|
|
|
|
keysym = 0;
|
|
mods = 0;
|
|
if (value)
|
|
{
|
|
if (!meta_ui_parse_accelerator (value, &keysym, &mods))
|
|
{
|
|
meta_topic (META_DEBUG_KEYBINDINGS,
|
|
"Failed to parse new gconf value\n");
|
|
meta_warning (_("\"%s\" found in configuration database is not a valid value for keybinding \"%s\""),
|
|
value, binding->name);
|
|
}
|
|
}
|
|
|
|
changed = FALSE;
|
|
if (keysym != binding->keysym ||
|
|
mods != binding->modifiers)
|
|
{
|
|
changed = TRUE;
|
|
|
|
binding->keysym = keysym;
|
|
binding->modifiers = mods;
|
|
|
|
meta_topic (META_DEBUG_KEYBINDINGS,
|
|
"New keybinding for \"%s\" is keysym = 0x%x mods = 0x%x\n",
|
|
binding->name, binding->keysym, binding->modifiers);
|
|
}
|
|
else
|
|
{
|
|
meta_topic (META_DEBUG_KEYBINDINGS,
|
|
"Keybinding for \"%s\" is unchanged\n", binding->name);
|
|
}
|
|
|
|
return changed;
|
|
}
|
|
|
|
static const gchar*
|
|
relative_key (const gchar* key)
|
|
{
|
|
const gchar* end;
|
|
|
|
end = strrchr (key, '/');
|
|
|
|
++end;
|
|
|
|
return end;
|
|
}
|
|
|
|
/* Return value is TRUE if a preference changed and we need to
|
|
* notify
|
|
*/
|
|
static gboolean
|
|
find_and_update_binding (MetaKeyPref *bindings,
|
|
const char *name,
|
|
const char *value)
|
|
{
|
|
const char *key;
|
|
int i;
|
|
|
|
if (*name == '/')
|
|
key = relative_key (name);
|
|
else
|
|
key = name;
|
|
|
|
i = 0;
|
|
while (bindings[i].name &&
|
|
strcmp (key, bindings[i].name) != 0)
|
|
++i;
|
|
|
|
if (bindings[i].name)
|
|
return update_binding (&bindings[i], value);
|
|
else
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
update_window_binding (const char *name,
|
|
const char *value)
|
|
{
|
|
return find_and_update_binding (window_bindings, name, value);
|
|
}
|
|
|
|
static gboolean
|
|
update_screen_binding (const char *name,
|
|
const char *value)
|
|
{
|
|
return find_and_update_binding (screen_bindings, name, value);
|
|
}
|
|
|
|
static gboolean
|
|
update_command (const char *name,
|
|
const char *value)
|
|
{
|
|
char *p;
|
|
int i;
|
|
|
|
p = strrchr (name, '_');
|
|
if (p == NULL)
|
|
{
|
|
meta_topic (META_DEBUG_KEYBINDINGS,
|
|
"Command %s has no underscore?\n", name);
|
|
return FALSE;
|
|
}
|
|
|
|
++p;
|
|
|
|
if (!g_ascii_isdigit (*p))
|
|
{
|
|
meta_topic (META_DEBUG_KEYBINDINGS,
|
|
"Command %s doesn't end in number?\n", name);
|
|
return FALSE;
|
|
}
|
|
|
|
i = atoi (p);
|
|
i -= 1; /* count from 0 not 1 */
|
|
|
|
if (i >= NUM_COMMANDS)
|
|
{
|
|
meta_topic (META_DEBUG_KEYBINDINGS,
|
|
"Command %d is too highly numbered, ignoring\n", i);
|
|
return FALSE;
|
|
}
|
|
|
|
g_free (commands[i]);
|
|
commands[i] = g_strdup (value);
|
|
|
|
meta_topic (META_DEBUG_KEYBINDINGS,
|
|
"Updated command %d to \"%s\"\n",
|
|
i, commands[i] ? commands[i] : "none");
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
const char*
|
|
meta_prefs_get_command (int i)
|
|
{
|
|
g_return_val_if_fail (i >= 0 && i < NUM_COMMANDS, NULL);
|
|
|
|
return commands[i];
|
|
}
|
|
|
|
char*
|
|
meta_prefs_get_gconf_key_for_command (int i)
|
|
{
|
|
char *key;
|
|
|
|
key = g_strdup_printf (KEY_COMMAND_PREFIX"%d", i + 1);
|
|
|
|
return key;
|
|
}
|
|
|
|
void
|
|
meta_prefs_get_button_layout (MetaButtonLayout *button_layout)
|
|
{
|
|
/* FIXME */
|
|
int i;
|
|
|
|
i = 0;
|
|
while (i < MAX_BUTTONS_PER_CORNER)
|
|
{
|
|
button_layout->left_buttons[i] = META_BUTTON_FUNCTION_LAST;
|
|
button_layout->right_buttons[i] = META_BUTTON_FUNCTION_LAST;
|
|
++i;
|
|
}
|
|
|
|
button_layout->left_buttons[0] = META_BUTTON_FUNCTION_MENU;
|
|
|
|
button_layout->right_buttons[0] = META_BUTTON_FUNCTION_MINIMIZE;
|
|
button_layout->right_buttons[1] = META_BUTTON_FUNCTION_MAXIMIZE;
|
|
button_layout->right_buttons[2] = META_BUTTON_FUNCTION_CLOSE;
|
|
}
|
|
|
|
void
|
|
meta_prefs_get_screen_bindings (const MetaKeyPref **bindings,
|
|
int *n_bindings)
|
|
{
|
|
|
|
*bindings = screen_bindings;
|
|
*n_bindings = (int) G_N_ELEMENTS (screen_bindings) - 1;
|
|
}
|
|
|
|
void
|
|
meta_prefs_get_window_bindings (const MetaKeyPref **bindings,
|
|
int *n_bindings)
|
|
{
|
|
*bindings = window_bindings;
|
|
*n_bindings = (int) G_N_ELEMENTS (window_bindings) - 1;
|
|
}
|
|
|
|
gboolean
|
|
meta_prefs_get_auto_raise ()
|
|
{
|
|
return auto_raise;
|
|
}
|
|
|
|
int
|
|
meta_prefs_get_auto_raise_delay ()
|
|
{
|
|
return auto_raise_delay;
|
|
}
|
|
|
|
MetaKeyBindingAction
|
|
meta_prefs_get_keybinding_action (const char *name)
|
|
{
|
|
int i;
|
|
|
|
i = G_N_ELEMENTS (screen_bindings) - 2; /* -2 for dummy entry at end */
|
|
while (i >= 0)
|
|
{
|
|
if (strcmp (screen_bindings[i].name, name) == 0)
|
|
return (MetaKeyBindingAction) i;
|
|
|
|
--i;
|
|
}
|
|
|
|
return META_KEYBINDING_ACTION_NONE;
|
|
}
|
|
|
|
void
|
|
meta_prefs_get_window_binding (const char *name,
|
|
unsigned int *keysym,
|
|
MetaVirtualModifier *modifiers)
|
|
{
|
|
int i;
|
|
|
|
i = G_N_ELEMENTS (window_bindings) - 2; /* -2 for dummy entry at end */
|
|
while (i >= 0)
|
|
{
|
|
if (strcmp (window_bindings[i].name, name) == 0)
|
|
{
|
|
*keysym = window_bindings[i].keysym;
|
|
*modifiers = window_bindings[i].modifiers;
|
|
return;
|
|
}
|
|
|
|
--i;
|
|
}
|
|
|
|
g_assert_not_reached ();
|
|
}
|