medit/moo/mooedit/moocommanddisplay.c
2007-12-19 14:02:42 -06:00

320 lines
8.4 KiB
C

/*
* moocommanddisplay.c
*
* Copyright (C) 2004-2007 by Yevgen Muntyan <muntyan@math.tamu.edu>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License version 2.1 as published by the Free Software Foundation.
*
* See COPYING file that comes with this distribution.
*/
#define MOOEDIT_COMPILATION
#include "mooedit/moocommanddisplay.h"
#include "mooedit/moocommand-private.h"
#include "mooutils/mooutils-misc.h"
#include <string.h>
typedef struct {
MooCommandData *data;
MooCommandFactory *factory;
gboolean changed;
} Data;
struct _MooCommandDisplay {
MooTreeHelper base;
GtkComboBox *factory_combo;
GtkNotebook *notebook;
Data *data;
int n_factories;
int active;
int original;
};
struct _MooCommandDisplayClass {
MooTreeHelperClass base_class;
};
G_DEFINE_TYPE (MooCommandDisplay, _moo_command_display, MOO_TYPE_TREE_HELPER)
static void combo_changed (MooCommandDisplay *display);
static void
block_combo (MooCommandDisplay *display)
{
g_signal_handlers_block_by_func (display->factory_combo,
(gpointer) combo_changed,
display);
}
static void
unblock_combo (MooCommandDisplay *display)
{
g_signal_handlers_unblock_by_func (display->factory_combo,
(gpointer) combo_changed,
display);
}
static void
combo_changed (MooCommandDisplay *display)
{
GtkWidget *widget;
int index;
MooCommandFactory *factory;
index = gtk_combo_box_get_active (display->factory_combo);
g_return_if_fail (index >= 0);
if (index == display->active)
return;
if (display->active >= 0)
{
widget = gtk_notebook_get_nth_page (display->notebook, display->active);
if (_moo_command_factory_save_data (display->data[display->active].factory, widget,
display->data[display->active].data))
display->data[display->active].changed = TRUE;
}
display->active = index;
factory = display->data[index].factory;
g_return_if_fail (factory != NULL);
if (!display->data[index].data)
display->data[index].data = moo_command_data_new (factory->n_keys);
gtk_notebook_set_current_page (display->notebook, index);
widget = gtk_notebook_get_nth_page (display->notebook, index);
_moo_command_factory_load_data (factory, widget, display->data[index].data);
}
static void
combo_set_active (MooCommandDisplay *display,
int index)
{
block_combo (display);
gtk_combo_box_set_active (display->factory_combo, index);
unblock_combo (display);
if (index < 0)
{
gtk_widget_hide (GTK_WIDGET (display->notebook));
}
else
{
gtk_notebook_set_current_page (display->notebook, index);
gtk_widget_show (GTK_WIDGET (display->notebook));
}
}
static int
combo_find_factory (MooCommandDisplay *display,
MooCommandFactory *factory)
{
int i;
g_return_val_if_fail (!factory || MOO_IS_COMMAND_FACTORY (factory), -1);
if (!factory)
return -1;
for (i = 0; i < display->n_factories; ++i)
if (display->data[i].factory == factory)
return i;
g_return_val_if_reached (-1);
}
void
_moo_command_display_set (MooCommandDisplay *display,
MooCommandFactory *factory,
MooCommandData *data)
{
int index;
GtkWidget *widget;
g_return_if_fail (MOO_IS_COMMAND_DISPLAY (display));
g_return_if_fail (!factory || MOO_IS_COMMAND_FACTORY (factory));
g_return_if_fail (!factory == !data);
for (index = 0; index < display->n_factories; ++index)
{
display->data[index].changed = FALSE;
if (display->data[index].data)
moo_command_data_unref (display->data[index].data);
display->data[index].data = NULL;
}
index = combo_find_factory (display, factory);
display->active = index;
display->original = index;
combo_set_active (display, index);
if (index >= 0)
{
display->data[index].data = moo_command_data_ref (data);
widget = gtk_notebook_get_nth_page (display->notebook, index);
_moo_command_factory_load_data (factory, widget, data);
}
}
gboolean
_moo_command_display_get (MooCommandDisplay *display,
MooCommandFactory **factory_p,
MooCommandData **data_p)
{
GtkWidget *widget;
Data *data;
g_return_val_if_fail (MOO_IS_COMMAND_DISPLAY (display), FALSE);
if (display->active < 0)
return FALSE;
data = &display->data[display->active];
widget = gtk_notebook_get_nth_page (display->notebook, display->active);
if (_moo_command_factory_save_data (data->factory, widget, data->data))
data->changed = TRUE;
if (display->active == display->original && !data->changed)
return FALSE;
*factory_p = data->factory;
*data_p = data->data;
data->changed = FALSE;
display->original = display->active;
return TRUE;
}
static void
init_factory_combo (MooCommandDisplay *display,
GtkComboBox *combo,
GtkNotebook *notebook)
{
GtkListStore *store;
GSList *factories;
GtkCellRenderer *cell;
g_return_if_fail (MOO_IS_COMMAND_DISPLAY (display));
g_return_if_fail (GTK_IS_COMBO_BOX (combo));
g_return_if_fail (GTK_IS_NOTEBOOK (notebook));
display->factory_combo = combo;
display->notebook = notebook;
cell = gtk_cell_renderer_text_new ();
gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (display->factory_combo), cell, TRUE);
gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (display->factory_combo), cell, "text", 0, NULL);
store = gtk_list_store_new (1, G_TYPE_STRING);
factories = moo_command_list_factories ();
display->active = -1;
display->original = -1;
display->n_factories = 0;
display->data = g_new0 (Data, g_slist_length (factories));
while (factories)
{
GtkTreeIter iter;
MooCommandFactory *cmd_factory;
GtkWidget *widget;
cmd_factory = factories->data;
widget = _moo_command_factory_create_widget (cmd_factory);
if (widget)
{
gtk_widget_show (widget);
gtk_notebook_append_page (display->notebook, widget, NULL);
gtk_list_store_append (store, &iter);
gtk_list_store_set (store, &iter, 0, cmd_factory->display_name, -1);
display->data[display->n_factories].factory = cmd_factory;
display->n_factories++;
}
factories = g_slist_delete_link (factories, factories);
}
gtk_combo_box_set_model (display->factory_combo, GTK_TREE_MODEL (store));
g_signal_connect_swapped (display->factory_combo, "changed",
G_CALLBACK (combo_changed), display);
g_object_unref (store);
}
MooCommandDisplay *
_moo_command_display_new (GtkComboBox *factory_combo,
GtkNotebook *notebook,
GtkWidget *treeview,
GtkWidget *new_btn,
GtkWidget *delete_btn,
GtkWidget *up_btn,
GtkWidget *down_btn)
{
MooCommandDisplay *display;
display = g_object_new (MOO_TYPE_COMMAND_DISPLAY, NULL);
_moo_tree_helper_connect (MOO_TREE_HELPER (display), treeview,
new_btn, delete_btn, up_btn, down_btn);
init_factory_combo (display, factory_combo, notebook);
MOO_OBJECT_REF_SINK (display);
return display;
}
static void
moo_command_display_dispose (GObject *object)
{
MooCommandDisplay *display = MOO_COMMAND_DISPLAY (object);
if (display->data)
{
int i;
for (i = 0; i < display->n_factories; ++i)
if (display->data[i].data)
moo_command_data_unref (display->data[i].data);
g_free (display->data);
display->data = NULL;
}
G_OBJECT_CLASS (_moo_command_display_parent_class)->dispose (object);
}
static void
_moo_command_display_init (G_GNUC_UNUSED MooCommandDisplay *display)
{
}
static void
_moo_command_display_class_init (MooCommandDisplayClass *klass)
{
G_OBJECT_CLASS (klass)->dispose = moo_command_display_dispose;
}