13
0
livetrax/libs/clearlooks/widget-information.c
David Robillard 8635f0e897 Update to clearlooks from gtk-engines-2.16.0
git-svn-id: svn://localhost/ardour2/branches/3.0@3857 d708f5d6-7413-0410-9779-e7cbd77b26cf
2008-10-05 00:09:32 +00:00

488 lines
11 KiB
C

#include <gtk/gtk.h>
#include "general-support.h"
#include "widget-information.h"
#include "config.h"
#include <math.h>
#include <string.h>
static gchar ge_widget_hints[] =
"treeview\0"
"treeview-header\0"
"statusbar\0"
"comboboxentry\0"
"spinbutton\0"
"scale\0"
"vscale\0"
"hscale\0"
"scrollbar\0"
"vscrollbar\0"
"hscrollbar\0"
"progressbar\0"
"menubar\0";
gboolean
ge_check_hint (GEHint hint,
GQuark style_hint,
GtkWidget *widget)
{
static GEHint quark_hint_lookup[GE_HINT_COUNT] = {0};
g_assert ((hint >= 0) && (hint < GE_HINT_COUNT));
/* Initilize lookup table */
if (G_UNLIKELY (quark_hint_lookup[0] == 0))
{
guint i = 0;
gchar *cur_hint_str = ge_widget_hints;
while ((i < GE_HINT_COUNT) && cur_hint_str[0])
{
/* Can't use _from_static_string as the engine may be unloaded. */
quark_hint_lookup[i] = g_quark_from_string (cur_hint_str);
cur_hint_str += strlen(cur_hint_str) + 1;
i++;
}
g_assert (i == GE_HINT_COUNT && cur_hint_str[0] == '\0');
}
if (quark_hint_lookup[hint] == style_hint)
return TRUE;
/* Try to decide based on other hints, eg. hscale is also a scale. */
if (hint == GE_HINT_SCALE)
if (ge_check_hint (GE_HINT_VSCALE, style_hint, widget) ||
ge_check_hint (GE_HINT_HSCALE, style_hint, widget))
return TRUE;
if (hint == GE_HINT_SCROLLBAR)
if (ge_check_hint (GE_HINT_VSCROLLBAR, style_hint, widget) ||
ge_check_hint (GE_HINT_HSCROLLBAR, style_hint, widget))
return TRUE;
if (hint == GE_HINT_TREEVIEW)
if (ge_check_hint (GE_HINT_TREEVIEW_HEADER, style_hint, widget))
return TRUE;
/* These may be caused by applications so we never want to disable them.
* TODO: This does not catch the case where the theme uses appears-as-list
* and the application turns it off again. Though this case
* is even less likely. */
switch (hint) {
case GE_HINT_COMBOBOX_ENTRY:
if (widget && ge_object_is_a (G_OBJECT (widget), "GtkComboBox"))
{
gboolean appears_as_list = FALSE;
gtk_widget_style_get (widget, "appears-as-list", &appears_as_list, NULL);
if (appears_as_list)
return TRUE;
}
break;
default:
break;
}
#ifdef ENABLE_WIDGET_CHECKS
/* If a style_hint *was* set, and nothing matched, just give up right away.
* A theme shall either support it fully, or not at all. */
if (style_hint != 0)
return FALSE;
/* No widget? Just give up. Nothing we can do. */
if (widget == NULL)
return FALSE;
/* Try to do something based on the passed in widget pointer. */
switch (hint) {
case GE_HINT_TREEVIEW:
if (widget->parent && (ge_object_is_a (G_OBJECT (widget->parent), "GtkTreeView")))
return TRUE;
break;
case GE_HINT_TREEVIEW_HEADER:
if (ge_object_is_a (G_OBJECT (widget), "GtkButton") && widget->parent &&
(ge_object_is_a (G_OBJECT (widget->parent), "GtkTreeView") || ge_object_is_a (G_OBJECT (widget->parent), "GtkCList") ||
ge_object_is_a (G_OBJECT (widget->parent), "GtkCTree")))
return TRUE;
if (widget->parent && ge_object_is_a (G_OBJECT (widget->parent), "ETreeView"))
return TRUE;
break;
case GE_HINT_COMBOBOX_ENTRY:
if (ge_is_in_combo_box (widget))
return TRUE;
break;
case GE_HINT_SPINBUTTON:
if (ge_object_is_a (G_OBJECT (widget), "GtkSpinButton"))
return TRUE;
break;
case GE_HINT_STATUSBAR:
if (widget->parent && ge_object_is_a (G_OBJECT (widget), "GtkStatusbar"))
return TRUE;
break;
case GE_HINT_SCALE:
if (ge_object_is_a (G_OBJECT (widget), "GtkScale"))
return TRUE;
break;
case GE_HINT_HSCALE:
if (ge_object_is_a (G_OBJECT (widget), "GtkHScale"))
return TRUE;
break;
case GE_HINT_VSCALE:
if (ge_object_is_a (G_OBJECT (widget), "GtkVScale"))
return TRUE;
break;
case GE_HINT_SCROLLBAR:
if (ge_object_is_a (G_OBJECT (widget), "GtkScrollbar"))
return TRUE;
break;
case GE_HINT_HSCROLLBAR:
if (ge_object_is_a (G_OBJECT (widget), "GtkHScrollbar"))
return TRUE;
break;
case GE_HINT_VSCROLLBAR:
if (ge_object_is_a (G_OBJECT (widget), "GtkVScrollbar"))
return TRUE;
break;
case GE_HINT_PROGRESSBAR:
if (ge_object_is_a (G_OBJECT (widget), "GtkProgressBar"))
return TRUE;
break;
case GE_HINT_MENUBAR:
if (ge_object_is_a (G_OBJECT (widget), "GtkMenuBar") ||
ge_object_is_a (G_OBJECT (widget->parent), "GtkMenuBar"))
return TRUE;
break;
default:
break;
}
#endif
return FALSE;
}
/* Widget Type Lookups/Macros
Based on/modified from functions in
Smooth-Engine.
*/
gboolean
ge_object_is_a (const GObject * object, const gchar * type_name)
{
gboolean result = FALSE;
if ((object))
{
GType tmp = g_type_from_name (type_name);
if (tmp)
result = g_type_check_instance_is_a ((GTypeInstance *) object, tmp);
}
return result;
}
gboolean
ge_is_combo_box_entry (GtkWidget * widget)
{
gboolean result = FALSE;
if ((widget) && (widget->parent))
{
if (GE_IS_COMBO_BOX_ENTRY (widget->parent))
result = TRUE;
else
result = ge_is_combo_box_entry (widget->parent);
}
return result;
}
static gboolean
ge_combo_box_is_using_list (GtkWidget * widget)
{
gboolean result = FALSE;
if (GE_IS_COMBO_BOX (widget))
gtk_widget_style_get (widget, "appears-as-list", &result, NULL);
return result;
}
gboolean
ge_is_combo_box (GtkWidget * widget, gboolean as_list)
{
gboolean result = FALSE;
if ((widget) && (widget->parent))
{
if (GE_IS_COMBO_BOX (widget->parent))
{
if (as_list)
result = (ge_combo_box_is_using_list(widget->parent));
else
result = (!ge_combo_box_is_using_list(widget->parent));
}
else
result = ge_is_combo_box (widget->parent, as_list);
}
return result;
}
gboolean
ge_is_combo (GtkWidget * widget)
{
gboolean result = FALSE;
if ((widget) && (widget->parent))
{
if (GE_IS_COMBO (widget->parent))
result = TRUE;
else
result = ge_is_combo (widget->parent);
}
return result;
}
gboolean
ge_is_in_combo_box (GtkWidget * widget)
{
return ((ge_is_combo (widget) || ge_is_combo_box (widget, TRUE) || ge_is_combo_box_entry (widget)));
}
gboolean
ge_is_toolbar_item (GtkWidget * widget)
{
gboolean result = FALSE;
if ((widget) && (widget->parent)) {
if ((GE_IS_BONOBO_TOOLBAR (widget->parent))
|| (GE_IS_BONOBO_DOCK_ITEM (widget->parent))
|| (GE_IS_EGG_TOOLBAR (widget->parent))
|| (GE_IS_TOOLBAR (widget->parent))
|| (GE_IS_HANDLE_BOX (widget->parent)))
result = TRUE;
else
result = ge_is_toolbar_item (widget->parent);
}
return result;
}
gboolean
ge_is_panel_widget_item (GtkWidget * widget)
{
gboolean result = FALSE;
if ((widget) && (widget->parent))
{
if (GE_IS_PANEL_WIDGET (widget->parent))
result = TRUE;
else
result = ge_is_panel_widget_item (widget->parent);
}
return result;
}
gboolean
ge_is_bonobo_dock_item (GtkWidget * widget)
{
gboolean result = FALSE;
if ((widget))
{
if (GE_IS_BONOBO_DOCK_ITEM(widget) || GE_IS_BONOBO_DOCK_ITEM (widget->parent))
result = TRUE;
else if (GE_IS_BOX(widget) || GE_IS_BOX(widget->parent))
{
GtkContainer *box = GE_IS_BOX(widget)?GTK_CONTAINER(widget):GTK_CONTAINER(widget->parent);
GList *children = NULL, *child = NULL;
children = gtk_container_get_children(box);
for (child = g_list_first(children); child; child = g_list_next(child))
{
if (GE_IS_BONOBO_DOCK_ITEM_GRIP(child->data))
{
result = TRUE;
child = NULL;
}
}
if (children)
g_list_free(children);
}
}
return result;
}
static GtkWidget *
ge_find_combo_box_entry_widget (GtkWidget * widget)
{
GtkWidget *result = NULL;
if (widget)
{
if (GE_IS_COMBO_BOX_ENTRY (widget))
result = widget;
else
result = ge_find_combo_box_entry_widget (widget->parent);
}
return result;
}
static GtkWidget *
ge_find_combo_box_widget (GtkWidget * widget, gboolean as_list)
{
GtkWidget *result = NULL;
if (widget)
{
if (GE_IS_COMBO_BOX (widget))
{
if (as_list)
result = (ge_combo_box_is_using_list(widget))?widget:NULL;
else
result = (!ge_combo_box_is_using_list(widget))?widget:NULL;
}
else
result = ge_find_combo_box_widget (widget->parent, as_list);
}
return result;
}
static GtkWidget *
ge_find_combo_widget (GtkWidget * widget)
{
GtkWidget *result = NULL;
if (widget)
{
if (GE_IS_COMBO (widget))
result = widget;
else
result = ge_find_combo_widget(widget->parent);
}
return result;
}
GtkWidget*
ge_find_combo_box_widget_parent (GtkWidget * widget)
{
GtkWidget *result = NULL;
if (!result)
result = ge_find_combo_widget(widget);
if (!result)
result = ge_find_combo_box_widget(widget, TRUE);
if (!result)
result = ge_find_combo_box_entry_widget(widget);
return result;
}
/***********************************************
* option_menu_get_props -
*
* Find Option Menu Size and Spacing
*
* Taken from Smooth
***********************************************/
void
ge_option_menu_get_props (GtkWidget * widget,
GtkRequisition * indicator_size,
GtkBorder * indicator_spacing)
{
GtkRequisition default_size = { 9, 5 };
GtkBorder default_spacing = { 7, 5, 2, 2 };
GtkRequisition *tmp_size = NULL;
GtkBorder *tmp_spacing = NULL;
if ((widget) && GE_IS_OPTION_MENU(widget))
gtk_widget_style_get (widget,
"indicator_size", &tmp_size,
"indicator_spacing", &tmp_spacing, NULL);
if (tmp_size)
{
*indicator_size = *tmp_size;
gtk_requisition_free (tmp_size);
}
else
*indicator_size = default_size;
if (tmp_spacing)
{
*indicator_spacing = *tmp_spacing;
gtk_border_free (tmp_spacing);
}
else
*indicator_spacing = default_spacing;
}
void
ge_button_get_default_border (GtkWidget *widget,
GtkBorder *border)
{
GtkBorder default_border = {1, 1, 1, 1};
GtkBorder *tmp_border = NULL;
if (widget && GE_IS_BUTTON (widget))
gtk_widget_style_get (widget, "default-border", &tmp_border, NULL);
if (tmp_border)
{
*border = *tmp_border;
gtk_border_free (tmp_border);
}
else
{
*border = default_border;
}
}
gboolean
ge_widget_is_ltr (GtkWidget *widget)
{
GtkTextDirection dir = GTK_TEXT_DIR_NONE;
if (GE_IS_WIDGET (widget))
dir = gtk_widget_get_direction (widget);
if (dir == GTK_TEXT_DIR_NONE)
dir = gtk_widget_get_default_direction ();
if (dir == GTK_TEXT_DIR_RTL)
return FALSE;
else
return TRUE;
}
guint
ge_rc_parse_hint (GScanner *scanner,
GQuark *quark)
{
guint token;
/* Skip 'hint' */
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_STRING)
return G_TOKEN_STRING;
*quark = g_quark_from_string (scanner->value.v_string);
return G_TOKEN_NONE;
}