David Robillard
8635f0e897
git-svn-id: svn://localhost/ardour2/branches/3.0@3857 d708f5d6-7413-0410-9779-e7cbd77b26cf
483 lines
13 KiB
C
483 lines
13 KiB
C
/* Clearlooks theme engine
|
|
* Copyright (C) 2005 Richard Stellingwerff
|
|
* Copyright (C) 2007 Benjamin Berg
|
|
* Copyright (C) 2007 Andrea Cimitan
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with this library; if not, write to the
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*
|
|
* Written by Owen Taylor <otaylor@redhat.com>
|
|
* and by Alexander Larsson <alexl@redhat.com>
|
|
* Modified by Richard Stellingwerff <remenic@gmail.com>
|
|
* Modified by Kulyk Nazar <schamane@myeburg.net>
|
|
*/
|
|
|
|
#include <string.h>
|
|
#include <widget-information.h>
|
|
#include "clearlooks_style.h"
|
|
#include "clearlooks_rc_style.h"
|
|
|
|
#include "animation.h"
|
|
|
|
#ifdef HAVE_ANIMATION
|
|
static void clearlooks_rc_style_finalize (GObject *object);
|
|
#endif
|
|
static GtkStyle *clearlooks_rc_style_create_style (GtkRcStyle *rc_style);
|
|
static guint clearlooks_rc_style_parse (GtkRcStyle *rc_style,
|
|
GtkSettings *settings,
|
|
GScanner *scanner);
|
|
static void clearlooks_rc_style_merge (GtkRcStyle *dest,
|
|
GtkRcStyle *src);
|
|
|
|
enum
|
|
{
|
|
TOKEN_FOCUSCOLOR = G_TOKEN_LAST + 1,
|
|
TOKEN_SCROLLBARCOLOR,
|
|
TOKEN_COLORIZESCROLLBAR,
|
|
TOKEN_CONTRAST,
|
|
TOKEN_SUNKENMENU,
|
|
TOKEN_PROGRESSBARSTYLE,
|
|
TOKEN_RELIEFSTYLE,
|
|
TOKEN_MENUBARSTYLE,
|
|
TOKEN_TOOLBARSTYLE,
|
|
TOKEN_MENUITEMSTYLE,
|
|
TOKEN_LISTVIEWITEMSTYLE,
|
|
TOKEN_ANIMATION,
|
|
TOKEN_STYLE,
|
|
TOKEN_RADIUS,
|
|
TOKEN_HINT,
|
|
|
|
TOKEN_CLASSIC,
|
|
TOKEN_GLOSSY,
|
|
TOKEN_INVERTED,
|
|
TOKEN_GUMMY,
|
|
|
|
TOKEN_TRUE,
|
|
TOKEN_FALSE,
|
|
|
|
TOKEN_LAST
|
|
};
|
|
|
|
static gchar* clearlooks_rc_symbols =
|
|
"focus_color\0"
|
|
"scrollbar_color\0"
|
|
"colorize_scrollbar\0"
|
|
"contrast\0"
|
|
"sunkenmenubar\0"
|
|
"progressbarstyle\0"
|
|
"reliefstyle\0"
|
|
"menubarstyle\0"
|
|
"toolbarstyle\0"
|
|
"menuitemstyle\0"
|
|
"listviewitemstyle\0"
|
|
"animation\0"
|
|
"style\0"
|
|
"radius\0"
|
|
"hint\0"
|
|
|
|
"CLASSIC\0"
|
|
"GLOSSY\0"
|
|
"INVERTED\0"
|
|
"GUMMY\0"
|
|
|
|
"TRUE\0"
|
|
"FALSE\0";
|
|
|
|
G_DEFINE_DYNAMIC_TYPE (ClearlooksRcStyle, clearlooks_rc_style, GTK_TYPE_RC_STYLE)
|
|
|
|
void
|
|
clearlooks_rc_style_register_types (GTypeModule *module)
|
|
{
|
|
clearlooks_rc_style_register_type (module);
|
|
}
|
|
|
|
static void
|
|
clearlooks_rc_style_init (ClearlooksRcStyle *clearlooks_rc)
|
|
{
|
|
clearlooks_rc->style = CL_STYLE_CLASSIC;
|
|
|
|
clearlooks_rc->flags = 0;
|
|
|
|
clearlooks_rc->contrast = 1.0;
|
|
clearlooks_rc->reliefstyle = 0;
|
|
clearlooks_rc->menubarstyle = 0;
|
|
clearlooks_rc->toolbarstyle = 0;
|
|
clearlooks_rc->animation = FALSE;
|
|
clearlooks_rc->colorize_scrollbar = FALSE;
|
|
clearlooks_rc->radius = 3.0;
|
|
clearlooks_rc->hint = 0;
|
|
}
|
|
|
|
#ifdef HAVE_ANIMATION
|
|
static void
|
|
clearlooks_rc_style_finalize (GObject *object)
|
|
{
|
|
/* cleanup all the animation stuff */
|
|
clearlooks_animation_cleanup ();
|
|
|
|
if (G_OBJECT_CLASS (clearlooks_rc_style_parent_class)->finalize != NULL)
|
|
G_OBJECT_CLASS (clearlooks_rc_style_parent_class)->finalize (object);
|
|
}
|
|
#endif
|
|
|
|
|
|
static void
|
|
clearlooks_rc_style_class_init (ClearlooksRcStyleClass *klass)
|
|
{
|
|
GtkRcStyleClass *rc_style_class = GTK_RC_STYLE_CLASS (klass);
|
|
#ifdef HAVE_ANIMATION
|
|
GObjectClass *g_object_class = G_OBJECT_CLASS (klass);
|
|
#endif
|
|
|
|
rc_style_class->parse = clearlooks_rc_style_parse;
|
|
rc_style_class->create_style = clearlooks_rc_style_create_style;
|
|
rc_style_class->merge = clearlooks_rc_style_merge;
|
|
|
|
#ifdef HAVE_ANIMATION
|
|
g_object_class->finalize = clearlooks_rc_style_finalize;
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
clearlooks_rc_style_class_finalize (ClearlooksRcStyleClass *klass)
|
|
{
|
|
}
|
|
|
|
static guint
|
|
clearlooks_gtk2_rc_parse_boolean (GtkSettings *settings,
|
|
GScanner *scanner,
|
|
gboolean *retval)
|
|
{
|
|
guint token;
|
|
token = g_scanner_get_next_token(scanner);
|
|
|
|
token = g_scanner_get_next_token(scanner);
|
|
if (token != G_TOKEN_EQUAL_SIGN)
|
|
return G_TOKEN_EQUAL_SIGN;
|
|
|
|
token = g_scanner_get_next_token(scanner);
|
|
if (token == TOKEN_TRUE)
|
|
*retval = TRUE;
|
|
else if (token == TOKEN_FALSE)
|
|
*retval = FALSE;
|
|
else
|
|
return TOKEN_TRUE;
|
|
|
|
return G_TOKEN_NONE;
|
|
}
|
|
|
|
static guint
|
|
clearlooks_gtk2_rc_parse_color(GtkSettings *settings,
|
|
GScanner *scanner,
|
|
GtkRcStyle *style,
|
|
GdkColor *color)
|
|
{
|
|
guint token;
|
|
|
|
/* Skip 'blah_color' */
|
|
token = g_scanner_get_next_token(scanner);
|
|
|
|
token = g_scanner_get_next_token(scanner);
|
|
if (token != G_TOKEN_EQUAL_SIGN)
|
|
return G_TOKEN_EQUAL_SIGN;
|
|
|
|
return gtk_rc_parse_color_full (scanner, style, color);
|
|
}
|
|
|
|
static guint
|
|
clearlooks_gtk2_rc_parse_double (GtkSettings *settings,
|
|
GScanner *scanner,
|
|
gdouble *val)
|
|
{
|
|
guint token;
|
|
|
|
/* Skip 'blah' */
|
|
token = g_scanner_get_next_token(scanner);
|
|
|
|
token = g_scanner_get_next_token(scanner);
|
|
if (token != G_TOKEN_EQUAL_SIGN)
|
|
return G_TOKEN_EQUAL_SIGN;
|
|
|
|
token = g_scanner_get_next_token(scanner);
|
|
if (token != G_TOKEN_FLOAT)
|
|
return G_TOKEN_FLOAT;
|
|
|
|
*val = scanner->value.v_float;
|
|
|
|
return G_TOKEN_NONE;
|
|
}
|
|
|
|
static guint
|
|
clearlooks_gtk2_rc_parse_int (GtkSettings *settings,
|
|
GScanner *scanner,
|
|
guint8 *progressbarstyle)
|
|
{
|
|
guint token;
|
|
|
|
/* Skip option name */
|
|
token = g_scanner_get_next_token(scanner);
|
|
|
|
token = g_scanner_get_next_token(scanner);
|
|
if (token != G_TOKEN_EQUAL_SIGN)
|
|
return G_TOKEN_EQUAL_SIGN;
|
|
|
|
token = g_scanner_get_next_token(scanner);
|
|
if (token != G_TOKEN_INT)
|
|
return G_TOKEN_INT;
|
|
|
|
*progressbarstyle = scanner->value.v_int;
|
|
|
|
return G_TOKEN_NONE;
|
|
}
|
|
|
|
static guint
|
|
clearlooks_gtk2_rc_parse_style (GtkSettings *settings,
|
|
GScanner *scanner,
|
|
ClearlooksStyles *style)
|
|
{
|
|
guint token;
|
|
|
|
g_assert (CL_NUM_STYLES == CL_STYLE_GUMMY + 1); /* so that people don't forget ;-) */
|
|
|
|
/* Skip 'style' */
|
|
token = g_scanner_get_next_token (scanner);
|
|
|
|
token = g_scanner_get_next_token (scanner);
|
|
if (token != G_TOKEN_EQUAL_SIGN)
|
|
return G_TOKEN_EQUAL_SIGN;
|
|
|
|
token = g_scanner_get_next_token (scanner);
|
|
|
|
switch (token)
|
|
{
|
|
case TOKEN_CLASSIC:
|
|
*style = CL_STYLE_CLASSIC;
|
|
break;
|
|
case TOKEN_GLOSSY:
|
|
*style = CL_STYLE_GLOSSY;
|
|
break;
|
|
case TOKEN_INVERTED:
|
|
*style = CL_STYLE_INVERTED;
|
|
break;
|
|
case TOKEN_GUMMY:
|
|
*style = CL_STYLE_GUMMY;
|
|
break;
|
|
default:
|
|
return TOKEN_CLASSIC;
|
|
}
|
|
|
|
return G_TOKEN_NONE;
|
|
}
|
|
|
|
static guint
|
|
clearlooks_gtk2_rc_parse_dummy (GtkSettings *settings,
|
|
GScanner *scanner,
|
|
gchar *name)
|
|
{
|
|
guint token;
|
|
|
|
/* Skip option */
|
|
token = g_scanner_get_next_token (scanner);
|
|
|
|
/* print a warning. Isn't there a way to get the string from the scanner? */
|
|
g_scanner_warn (scanner, "Clearlooks configuration option \"%s\" is not supported and will be ignored.", name);
|
|
|
|
/* equal sign */
|
|
token = g_scanner_get_next_token (scanner);
|
|
if (token != G_TOKEN_EQUAL_SIGN)
|
|
return G_TOKEN_EQUAL_SIGN;
|
|
|
|
/* eat whatever comes next */
|
|
token = g_scanner_get_next_token (scanner);
|
|
|
|
return G_TOKEN_NONE;
|
|
}
|
|
|
|
static guint
|
|
clearlooks_rc_style_parse (GtkRcStyle *rc_style,
|
|
GtkSettings *settings,
|
|
GScanner *scanner)
|
|
{
|
|
static GQuark scope_id = 0;
|
|
ClearlooksRcStyle *clearlooks_style = CLEARLOOKS_RC_STYLE (rc_style);
|
|
|
|
guint old_scope;
|
|
guint token;
|
|
|
|
/* Set up a new scope in this scanner. */
|
|
|
|
if (!scope_id)
|
|
scope_id = g_quark_from_string("clearlooks_theme_engine");
|
|
|
|
/* If we bail out due to errors, we *don't* reset the scope, so the
|
|
* error messaging code can make sense of our tokens.
|
|
*/
|
|
old_scope = g_scanner_set_scope(scanner, scope_id);
|
|
|
|
/* Now check if we already added our symbols to this scope
|
|
* (in some previous call to clearlooks_rc_style_parse for the
|
|
* same scanner.
|
|
*/
|
|
if (!g_scanner_lookup_symbol(scanner, clearlooks_rc_symbols)) {
|
|
gchar *current_symbol = clearlooks_rc_symbols;
|
|
gint i = G_TOKEN_LAST + 1;
|
|
|
|
/* Add our symbols */
|
|
while ((current_symbol[0] != '\0') && (i < TOKEN_LAST)) {
|
|
g_scanner_scope_add_symbol(scanner, scope_id, current_symbol, GINT_TO_POINTER (i));
|
|
|
|
current_symbol += strlen(current_symbol) + 1;
|
|
i++;
|
|
}
|
|
g_assert (i == TOKEN_LAST && current_symbol[0] == '\0');
|
|
}
|
|
|
|
/* We're ready to go, now parse the top level */
|
|
|
|
token = g_scanner_peek_next_token(scanner);
|
|
while (token != G_TOKEN_RIGHT_CURLY)
|
|
{
|
|
switch (token)
|
|
{
|
|
case TOKEN_FOCUSCOLOR:
|
|
token = clearlooks_gtk2_rc_parse_color (settings, scanner, rc_style, &clearlooks_style->focus_color);
|
|
clearlooks_style->flags |= CL_FLAG_FOCUS_COLOR;
|
|
break;
|
|
case TOKEN_SCROLLBARCOLOR:
|
|
token = clearlooks_gtk2_rc_parse_color (settings, scanner, rc_style, &clearlooks_style->scrollbar_color);
|
|
clearlooks_style->flags |= CL_FLAG_SCROLLBAR_COLOR;
|
|
break;
|
|
case TOKEN_COLORIZESCROLLBAR:
|
|
token = clearlooks_gtk2_rc_parse_boolean (settings, scanner, &clearlooks_style->colorize_scrollbar);
|
|
clearlooks_style->flags |= CL_FLAG_COLORIZE_SCROLLBAR;
|
|
break;
|
|
case TOKEN_CONTRAST:
|
|
token = clearlooks_gtk2_rc_parse_double (settings, scanner, &clearlooks_style->contrast);
|
|
clearlooks_style->flags |= CL_FLAG_CONTRAST;
|
|
break;
|
|
case TOKEN_RELIEFSTYLE:
|
|
token = clearlooks_gtk2_rc_parse_int (settings, scanner, &clearlooks_style->reliefstyle);
|
|
clearlooks_style->flags |= CL_FLAG_RELIEFSTYLE;
|
|
break;
|
|
case TOKEN_MENUBARSTYLE:
|
|
token = clearlooks_gtk2_rc_parse_int (settings, scanner, &clearlooks_style->menubarstyle);
|
|
clearlooks_style->flags |= CL_FLAG_MENUBARSTYLE;
|
|
break;
|
|
case TOKEN_TOOLBARSTYLE:
|
|
token = clearlooks_gtk2_rc_parse_int (settings, scanner, &clearlooks_style->toolbarstyle);
|
|
clearlooks_style->flags |= CL_FLAG_TOOLBARSTYLE;
|
|
break;
|
|
case TOKEN_ANIMATION:
|
|
token = clearlooks_gtk2_rc_parse_boolean (settings, scanner, &clearlooks_style->animation);
|
|
clearlooks_style->flags |= CL_FLAG_ANIMATION;
|
|
break;
|
|
case TOKEN_STYLE:
|
|
token = clearlooks_gtk2_rc_parse_style (settings, scanner, &clearlooks_style->style);
|
|
clearlooks_style->flags |= CL_FLAG_STYLE;
|
|
break;
|
|
case TOKEN_RADIUS:
|
|
token = clearlooks_gtk2_rc_parse_double (settings, scanner, &clearlooks_style->radius);
|
|
clearlooks_style->flags |= CL_FLAG_RADIUS;
|
|
break;
|
|
case TOKEN_HINT:
|
|
token = ge_rc_parse_hint (scanner, &clearlooks_style->hint);
|
|
clearlooks_style->flags |= CL_FLAG_HINT;
|
|
break;
|
|
|
|
/* stuff to ignore */
|
|
case TOKEN_SUNKENMENU:
|
|
token = clearlooks_gtk2_rc_parse_dummy (settings, scanner, "sunkenmenu");
|
|
break;
|
|
case TOKEN_PROGRESSBARSTYLE:
|
|
token = clearlooks_gtk2_rc_parse_dummy (settings, scanner, "progressbarstyle");
|
|
break;
|
|
case TOKEN_MENUITEMSTYLE:
|
|
token = clearlooks_gtk2_rc_parse_dummy (settings, scanner, "menuitemstyle");
|
|
break;
|
|
case TOKEN_LISTVIEWITEMSTYLE:
|
|
token = clearlooks_gtk2_rc_parse_dummy (settings, scanner, "listviewitemstyle");
|
|
break;
|
|
|
|
default:
|
|
g_scanner_get_next_token(scanner);
|
|
token = G_TOKEN_RIGHT_CURLY;
|
|
break;
|
|
}
|
|
|
|
if (token != G_TOKEN_NONE)
|
|
return token;
|
|
|
|
token = g_scanner_peek_next_token(scanner);
|
|
}
|
|
|
|
g_scanner_get_next_token(scanner);
|
|
|
|
g_scanner_set_scope(scanner, old_scope);
|
|
|
|
return G_TOKEN_NONE;
|
|
}
|
|
|
|
static void
|
|
clearlooks_rc_style_merge (GtkRcStyle *dest,
|
|
GtkRcStyle *src)
|
|
{
|
|
ClearlooksRcStyle *dest_w, *src_w;
|
|
ClearlooksRcFlags flags;
|
|
|
|
GTK_RC_STYLE_CLASS (clearlooks_rc_style_parent_class)->merge (dest, src);
|
|
|
|
if (!CLEARLOOKS_IS_RC_STYLE (src))
|
|
return;
|
|
|
|
src_w = CLEARLOOKS_RC_STYLE (src);
|
|
dest_w = CLEARLOOKS_RC_STYLE (dest);
|
|
|
|
flags = (~dest_w->flags) & src_w->flags;
|
|
|
|
if (flags & CL_FLAG_STYLE)
|
|
dest_w->style = src_w->style;
|
|
if (flags & CL_FLAG_CONTRAST)
|
|
dest_w->contrast = src_w->contrast;
|
|
if (flags & CL_FLAG_RELIEFSTYLE)
|
|
dest_w->reliefstyle = src_w->reliefstyle;
|
|
if (flags & CL_FLAG_MENUBARSTYLE)
|
|
dest_w->menubarstyle = src_w->menubarstyle;
|
|
if (flags & CL_FLAG_TOOLBARSTYLE)
|
|
dest_w->toolbarstyle = src_w->toolbarstyle;
|
|
if (flags & CL_FLAG_FOCUS_COLOR)
|
|
dest_w->focus_color = src_w->focus_color;
|
|
if (flags & CL_FLAG_SCROLLBAR_COLOR)
|
|
dest_w->scrollbar_color = src_w->scrollbar_color;
|
|
if (flags & CL_FLAG_COLORIZE_SCROLLBAR)
|
|
dest_w->colorize_scrollbar = src_w->colorize_scrollbar;
|
|
if (flags & CL_FLAG_ANIMATION)
|
|
dest_w->animation = src_w->animation;
|
|
if (flags & CL_FLAG_RADIUS)
|
|
dest_w->radius = src_w->radius;
|
|
if (flags & CL_FLAG_HINT)
|
|
dest_w->hint = src_w->hint;
|
|
|
|
dest_w->flags |= src_w->flags;
|
|
}
|
|
|
|
|
|
/* Create an empty style suitable to this RC style
|
|
*/
|
|
static GtkStyle *
|
|
clearlooks_rc_style_create_style (GtkRcStyle *rc_style)
|
|
{
|
|
return GTK_STYLE (g_object_new (CLEARLOOKS_TYPE_STYLE, NULL));
|
|
}
|