/* 
 * GChemPaint library
 * fontsel.c 
 *
 * Copyright (C) 2006-2007 Jean Bréfort <jean.brefort@normalesup.org>
 *
 * 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., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301
 * USA
 */

#include "gchempaint-config.h"
#include "fontsel.h"
#include <gtk/gtkbin.h>
#include <gtk/gtkentry.h>
#include <gtk/gtklabel.h>
#include <gtk/gtkscrolledwindow.h>
#include <gtk/gtktable.h>
#include <gtk/gtktreeview.h>
#include <gtk/gtktreeselection.h>
#include <gtk/gtkcellrenderertext.h>
#include <gsf/gsf-impl-utils.h>
#include <glib/gi18n-lib.h>
#include <cstdlib>
#include <map>
#include <string>
#include <cstring>

using namespace std;

struct _GcpFontSel {
	GtkBin base;
	GtkEntry *m_SizeEntry;
	GtkListStore *FamilyList, *FaceList, *SizeList;
	GtkTreeView *FamilyTree, *FacesTree, *SizeTree;
	guint FamilySignal, FaceSignal, SizeSignal;
	GtkTreeSelection *FamilySel, *FaceSel, *SizeSel;
	GtkLabel *Label;
	map<string, PangoFontFamily*> Families;
	map<string, PangoFontFace*> Faces;
	char *FamilyName;
	PangoStyle Style;
	PangoWeight Weight;
	PangoStretch Stretch;
	PangoVariant Variant;
	int Size;
};

typedef struct {
	GtkBinClass base_class;

	/* signals */
	void (*changed) (GcpFontSel *fs);
} GcpFontSelClass;

static void
gcp_font_sel_size_request (GtkWidget      *widget,
			GtkRequisition *requisition)
{
	GtkWidget *w = GTK_WIDGET (gtk_bin_get_child (GTK_BIN (widget)));
	if (w)
    	gtk_widget_size_request (w, requisition);
	else {
		requisition->width = 0;
		requisition->height = 0;
	}
}

static void
gcp_font_sel_size_allocate (GtkWidget     *widget,
			 GtkAllocation *allocation)
{
	GtkWidget *w = GTK_WIDGET (gtk_bin_get_child (GTK_BIN (widget)));
	if (w)
		gtk_widget_size_allocate (GTK_WIDGET (w), allocation);
}

enum {
	FONT_SEL_PROP_0,
	FONT_SEL_PROP_FAMILY,
	FONT_SEL_PROP_STYLE,
	FONT_SEL_PROP_WEIGHT,
	FONT_SEL_PROP_STRETCH,
	FONT_SEL_PROP_VARIANT,
	FONT_SEL_PROP_SIZE
};

enum {
	CHANGED,
	LAST_SIGNAL
};
static gulong gcp_font_sel_signals [LAST_SIGNAL] = { 0, };

static void gcp_font_sel_set_label (GcpFontSel *fs)
{
	PangoFontDescription *fd = pango_font_description_new ();
	pango_font_description_set_family (fd, fs->FamilyName);
	pango_font_description_set_style (fd, fs->Style);
	pango_font_description_set_weight (fd, fs->Weight);
	pango_font_description_set_variant (fd, fs->Variant);
	pango_font_description_set_stretch (fd, fs->Stretch);
	pango_font_description_set_size (fd, fs->Size);
	char *name = pango_font_description_to_string (fd);
	char *markup = g_markup_printf_escaped ("<span font_desc=\"%s\">%s</span>", name, name);
	gtk_label_set_markup (fs->Label, markup);
	g_free (name);
	g_free (markup);
}

static void select_best_font_face (GcpFontSel *fs)
{
	PangoFontDescription *desc;
	int distance, best;
	PangoStyle Style;
	PangoWeight Weight;
	PangoVariant Variant;
	PangoStretch Stretch;
	map <string, PangoFontFace*>::iterator i, iend = fs->Faces.end ();
	char const *name = NULL, *buf;
	GtkTreeIter iter;

	best = 32000; // This should be enough
	for (i = fs->Faces.begin (); i != iend; i++) {
		desc = pango_font_face_describe ((*i).second);
		// Try to select the best available face
		Style = pango_font_description_get_style (desc);
		Weight = pango_font_description_get_weight (desc);
		Variant = pango_font_description_get_variant (desc);
		Stretch = pango_font_description_get_stretch (desc);
		distance = abs (Weight - fs->Weight)
						+ abs ((Style? Style + 2: 0) - (fs->Style? fs->Style + 2: 0)) * 1000
						+ abs (Variant - fs->Variant) * 10 + abs (Stretch - fs->Stretch);
		if (distance < best) {
			best = distance;
			name = (*i).first.c_str ();
		}
		pango_font_description_free (desc);
	}
	// select the found face
	if (!gtk_tree_model_get_iter_first (GTK_TREE_MODEL (fs->FaceList), &iter))
		return;
	do {
		gtk_tree_model_get (GTK_TREE_MODEL (fs->FaceList), &iter, 0, &buf, -1);
		if (!strcmp (name, buf)) {
			GtkTreePath *path = gtk_tree_model_get_path (GTK_TREE_MODEL (fs->FaceList), &iter);
			gtk_tree_view_set_cursor (fs->FacesTree, path, NULL, FALSE);
			gtk_tree_path_free (path);
			break;
		}
	} while (gtk_tree_model_iter_next (GTK_TREE_MODEL (fs->FaceList), &iter));
}

static void
gcp_font_sel_get_property (GObject *obj, guint param_id,
			GValue *value, GParamSpec *pspec)
{
	GcpFontSel *fs = GCP_FONT_SEL (obj);

	switch (param_id) {
	case FONT_SEL_PROP_FAMILY:
		g_value_set_string (value, fs->FamilyName);
		break;
	case FONT_SEL_PROP_STYLE:
		g_value_set_int (value, fs->Style);
		break;
	case FONT_SEL_PROP_WEIGHT:
		g_value_set_int (value, fs->Weight);
		break;
	case FONT_SEL_PROP_STRETCH:
		g_value_set_int (value, fs->Stretch);
		break;
	case FONT_SEL_PROP_VARIANT:
		g_value_set_int (value, fs->Variant);
		break;
	case FONT_SEL_PROP_SIZE:
		g_value_set_int (value, fs->Size);
		break;
	default: G_OBJECT_WARN_INVALID_PROPERTY_ID (obj, param_id, pspec);
		 return;
	}
}

/* These are what we use as the standard font sizes, for the size list.
 */
static const guint16 font_sizes[] = {
  8, 9, 10, 11, 12, 13, 14, 16, 18, 20, 22, 24, 26, 28,
  32, 36, 40, 48, 56, 64, 72
};

static void gcp_font_sel_set_size_full (GcpFontSel *fs, bool update_list)
{
	char *buf = g_strdup_printf ("%.1f", (double) fs->Size / PANGO_SCALE);
	gtk_entry_set_text (fs->m_SizeEntry, buf);
	g_free (buf);
	if (update_list) {
		GtkTreeIter iter;
		bool found = false;
		g_signal_handler_block (fs->SizeSel, fs->SizeSignal);
		
		gtk_tree_model_get_iter_first (GTK_TREE_MODEL (fs->SizeList), &iter);
		for (unsigned i = 0; i < G_N_ELEMENTS (font_sizes) && !found; i++) {
			if (font_sizes[i] * PANGO_SCALE == fs->Size) {
				GtkTreePath *path = gtk_tree_model_get_path (GTK_TREE_MODEL (fs->SizeList), &iter);
				gtk_tree_view_set_cursor (fs->SizeTree, path, NULL, FALSE);
				gtk_tree_path_free (path);
				found = true;
			}

			gtk_tree_model_iter_next (GTK_TREE_MODEL (fs->SizeList), &iter);
		}
		
		if (!found)
			gtk_tree_selection_unselect_all (fs->SizeSel);
		g_signal_handler_unblock (fs->SizeSel, fs->SizeSignal);
	}
	g_signal_emit (G_OBJECT (fs), gcp_font_sel_signals [CHANGED], 0);
	gcp_font_sel_set_label (fs);
}

static void
gcp_font_sel_set_property (GObject *obj, guint param_id,
			GValue const *value, GParamSpec *pspec)
{
	GcpFontSel *fs = GCP_FONT_SEL (obj);

	switch (param_id) {
	case FONT_SEL_PROP_FAMILY:
		if (fs->FamilyName)
			g_free (fs->FamilyName);
		fs->FamilyName = g_strdup (g_value_get_string (value));
		GtkTreeIter iter;
		char const *buf;
		gtk_tree_model_get_iter_first (GTK_TREE_MODEL (fs->FamilyList), &iter);
		do {
			gtk_tree_model_get (GTK_TREE_MODEL (fs->FamilyList), &iter, 0, &buf, -1);
			if (!strcmp (fs->FamilyName, buf)) {
				GtkTreePath *path = gtk_tree_model_get_path (GTK_TREE_MODEL (fs->FamilyList), &iter);
				gtk_tree_view_set_cursor (fs->FamilyTree, path, NULL, FALSE);
				gtk_tree_view_scroll_to_cell (fs->FamilyTree, path, NULL, FALSE, 0., 0.);
				gtk_tree_path_free (path);
				break;
			}
		} while (gtk_tree_model_iter_next (GTK_TREE_MODEL (fs->FamilyList), &iter));
	return;
	case FONT_SEL_PROP_STYLE:
		fs->Style = (PangoStyle) g_value_get_int (value);
		break;
	case FONT_SEL_PROP_WEIGHT:
		fs->Weight = (PangoWeight) g_value_get_int (value);		break;
	case FONT_SEL_PROP_STRETCH:
		fs->Stretch = (PangoStretch) g_value_get_int (value);		break;
	case FONT_SEL_PROP_VARIANT:
		fs->Variant = (PangoVariant) g_value_get_int (value);		break;
	case FONT_SEL_PROP_SIZE:
		fs->Size = g_value_get_int (value);
		gcp_font_sel_set_size_full (fs, true);
		return;
	default: G_OBJECT_WARN_INVALID_PROPERTY_ID (obj, param_id, pspec);
		 return;
	}
	select_best_font_face (fs);
}

static void
gcp_font_sel_class_init (GcpFontSelClass *klass)
{
	GObjectClass *object_class = (GObjectClass*) klass;
	GtkWidgetClass *widget_class = (GtkWidgetClass *) klass;

	object_class->get_property = gcp_font_sel_get_property;
	object_class->set_property = gcp_font_sel_set_property;

	widget_class->size_request = gcp_font_sel_size_request;
	widget_class->size_allocate = gcp_font_sel_size_allocate;
	g_object_class_install_property (object_class, FONT_SEL_PROP_FAMILY,
		g_param_spec_string ("family", _("Family"),
			_("Font family"),
			"Bitstream Vera Sans", (GParamFlags) G_PARAM_READWRITE));
	g_object_class_install_property (object_class, FONT_SEL_PROP_STYLE,
		g_param_spec_int ("style", _("Style"),
			_("The font style (normal, oblique or italic)"),
			PANGO_STYLE_NORMAL, PANGO_STYLE_ITALIC, PANGO_STYLE_NORMAL,
			(GParamFlags) G_PARAM_READWRITE));
	g_object_class_install_property (object_class, FONT_SEL_PROP_WEIGHT,
		g_param_spec_int ("weight", _("Weight"),
			_("The font weight"),
			PANGO_WEIGHT_ULTRALIGHT, PANGO_WEIGHT_HEAVY, PANGO_WEIGHT_NORMAL,
			(GParamFlags) G_PARAM_READWRITE));
	g_object_class_install_property (object_class, FONT_SEL_PROP_VARIANT,
		g_param_spec_int ("variant", _("Variant"),
			_("The font variant"),
			PANGO_VARIANT_NORMAL, PANGO_VARIANT_SMALL_CAPS, PANGO_VARIANT_NORMAL,
			(GParamFlags) G_PARAM_READWRITE));
	g_object_class_install_property (object_class, FONT_SEL_PROP_STRETCH,
		g_param_spec_int ("stretch", _("Stretch"),
			_("The font stretch (condensed, normal or expanded)"),
			PANGO_STRETCH_ULTRA_CONDENSED, PANGO_STRETCH_ULTRA_EXPANDED, PANGO_STRETCH_NORMAL,
			(GParamFlags) G_PARAM_READWRITE));
	g_object_class_install_property (object_class, FONT_SEL_PROP_SIZE,
		g_param_spec_int ("size", _("Size"),
			_("The font size (in pango units)"),
			0, G_MAXINT, 12 * PANGO_SCALE,
			(GParamFlags) G_PARAM_READWRITE));

	gcp_font_sel_signals [CHANGED] = g_signal_new ("changed",
		G_TYPE_FROM_CLASS (klass),
		G_SIGNAL_RUN_LAST,
		G_STRUCT_OFFSET (GcpFontSelClass, changed),
		NULL, NULL,
		g_cclosure_marshal_VOID__VOID,
		G_TYPE_NONE, 0);
}

static void on_select_family (GtkTreeSelection *selection, GcpFontSel *fs)
{
	GtkTreeModel *model;
	GtkTreeIter iter, selected;
	char const *name;
	if (!gtk_tree_selection_get_selected (selection, &model, &iter))
		return;
	gtk_tree_model_get (model, &iter, 0, &fs->FamilyName, -1);
	PangoFontFamily *family = fs->Families[fs->FamilyName];
	PangoFontFace **faces;
	int i, besti, nb;
	g_signal_handler_block (fs->FaceSel, fs->FaceSignal);
	pango_font_family_list_faces (family, &faces, &nb);
	gtk_list_store_clear (fs->FaceList);
	map<string, PangoFontFace*>::iterator j, jend = fs->Faces.end ();
	for (j = fs->Faces.begin (); j != jend; j++) {
		g_object_unref ((*j).second);
	}
	fs->Faces.clear ();
	PangoFontDescription *desc;
	int distance, best;
	PangoStyle Style;
	PangoWeight Weight;
	PangoVariant Variant;
	PangoStretch Stretch;

	best = 32000; // This should be enough
	for (i = 0; i < nb; i++) {
		name = pango_font_face_get_face_name (faces[i]);
		desc = pango_font_face_describe (faces[i]);
		fs->Faces[name] = (PangoFontFace*) g_object_ref (faces[i]);
		gtk_list_store_append (fs->FaceList, &iter);
		gtk_list_store_set (fs->FaceList, &iter,
				  0, name,
				  -1);
		// Try to select the best available face
		Style = pango_font_description_get_style (desc);
		Weight = pango_font_description_get_weight (desc);
		Variant = pango_font_description_get_variant (desc);
		Stretch = pango_font_description_get_stretch (desc);
		distance = abs (Weight - fs->Weight)
						+ abs ((Style? Style + 2: 0) - (fs->Style? fs->Style + 2: 0)) * 1000
						+ abs (Variant - fs->Variant) * 10 + abs (Stretch - fs->Stretch);
		if (distance < best) {
			best = distance;
			selected = iter;
			besti = i;
		}
		// TODO: write this code
		pango_font_description_free (desc);
	}
	g_signal_handler_unblock (fs->FaceSel, fs->FaceSignal);
	GtkTreePath *path = gtk_tree_model_get_path (GTK_TREE_MODEL (fs->FaceList), &selected);
	if (path) {
		gtk_tree_selection_select_path (GTK_TREE_SELECTION (fs->FaceSel), path);
		gtk_tree_path_free (path);
	}
}

static void on_select_face (GtkTreeSelection *selection, GcpFontSel *fs)
{
	GtkTreeModel *model;
	GtkTreeIter iter;
	char const *name;
	if (!gtk_tree_selection_get_selected (selection, &model, &iter))
		return;
	gtk_tree_model_get (model, &iter, 0, &name, -1);
	PangoFontFace *face = fs->Faces[name];
	PangoFontDescription *desc = pango_font_face_describe (face);
	fs->Style = pango_font_description_get_style (desc);
	fs->Weight = pango_font_description_get_weight (desc);
	fs->Variant = pango_font_description_get_variant (desc);
	fs->Stretch = pango_font_description_get_stretch (desc);
	pango_font_description_free (desc);
	g_signal_emit (G_OBJECT (fs), gcp_font_sel_signals [CHANGED], 0);
	gcp_font_sel_set_label (fs);
}

static void on_select_size (GtkTreeSelection *selection, GcpFontSel *fs)
{
	GtkTreeModel *model;
	GtkTreeIter iter;
	gtk_tree_selection_get_selected (selection, &model, &iter);
	gtk_tree_model_get (model, &iter, 0, &fs->Size, -1);
	fs->Size *= PANGO_SCALE;
	gcp_font_sel_set_size_full (fs, false);
}

static void on_size_activate (GtkEntry *entry, GcpFontSel *fs)
{
	char const *text = gtk_entry_get_text (fs->m_SizeEntry);
	fs->Size = (int) (MAX (0.1, atof (text) * PANGO_SCALE + 0.5));
	gcp_font_sel_set_size_full (fs, true);
}

static bool on_size_focus_out (GtkEntry *entry, GdkEventFocus *event, GcpFontSel *fs)
{
	on_size_activate (entry, fs);
	return true;
}

static void
gcp_font_sel_init (GcpFontSel *fs)
{
	int i, nb;
	PangoFontFamily **families;
	GtkWidget *sc, *w = gtk_table_new (3, 4, FALSE);
	g_object_set (G_OBJECT (w), "border-width", 6, NULL);
	fs->Families = map<string, PangoFontFamily*>();
	fs->Faces = map<string, PangoFontFace*>();
	GtkTable *table = GTK_TABLE (w);
	gtk_table_set_col_spacings (table, 12);
	gtk_container_add (GTK_CONTAINER (fs), GTK_WIDGET (w));
	w = gtk_label_new ("");
	fs->Label = GTK_LABEL (w);
	gtk_table_attach (table, w, 0, 3, 3, 4,
			(GtkAttachOptions) (GTK_EXPAND | GTK_FILL),
			(GtkAttachOptions) (GTK_EXPAND | GTK_FILL), 0, 6);
	// Initialize faces list
	fs->FaceList = gtk_list_store_new (1, G_TYPE_STRING);
	gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (fs->FaceList), 0, GTK_SORT_ASCENDING);
	fs->FacesTree = GTK_TREE_VIEW (gtk_tree_view_new_with_model (GTK_TREE_MODEL (fs->FaceList)));
	gtk_tree_view_set_headers_visible (fs->FacesTree, false);
	sc = gtk_scrolled_window_new (NULL, NULL);
	gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (sc), GTK_POLICY_NEVER, GTK_POLICY_ALWAYS);
	gtk_scrolled_window_add_with_viewport (GTK_SCROLLED_WINDOW (sc), GTK_WIDGET (fs->FacesTree));
	gtk_table_attach (table, sc, 1, 2, 1, 3,
			(GtkAttachOptions) (GTK_EXPAND | GTK_FILL),
			(GtkAttachOptions) (GTK_EXPAND | GTK_FILL), 0, 0);
	GtkCellRenderer *renderer = gtk_cell_renderer_text_new ();
	GtkTreeViewColumn *column = gtk_tree_view_column_new_with_attributes (NULL, renderer, "text", 0, NULL);
	gtk_tree_view_append_column (fs->FacesTree, column);
	GtkTreeSelection *selection = gtk_tree_view_get_selection (fs->FacesTree);
	fs->FaceSel = selection;
	fs->FaceSignal = g_signal_connect (fs->FaceSel, "changed", G_CALLBACK (on_select_face), fs);
	// Initialize sizes list
	fs->SizeList = gtk_list_store_new (1, G_TYPE_INT);
	w = gtk_tree_view_new_with_model (GTK_TREE_MODEL (fs->SizeList));
	fs->SizeTree =  GTK_TREE_VIEW (w);
	gtk_tree_view_set_headers_visible (GTK_TREE_VIEW (w), false);
	renderer = gtk_cell_renderer_text_new ();
	column = gtk_tree_view_column_new_with_attributes (NULL, renderer, "text", 0, NULL);
	gtk_tree_view_append_column (GTK_TREE_VIEW (w), column);
	GtkTreeIter iter;
	for (i = 0; i < (int) G_N_ELEMENTS (font_sizes); i++) {
		gtk_list_store_append (fs->SizeList, &iter);
		gtk_list_store_set (fs->SizeList, &iter,
				  0, font_sizes[i],
				  -1);
	}
	selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (w));
	fs->SizeSel = selection;
	fs->SizeSignal = g_signal_connect (fs->SizeSel, "changed", G_CALLBACK (on_select_size), fs);
	sc = gtk_scrolled_window_new (NULL, NULL);
	gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (sc), GTK_POLICY_NEVER, GTK_POLICY_ALWAYS);
	gtk_scrolled_window_add_with_viewport (GTK_SCROLLED_WINDOW (sc), w);
	gtk_table_attach (table, sc, 2, 3, 2, 3,
			(GtkAttachOptions) (GTK_FILL),
			(GtkAttachOptions) (GTK_EXPAND | GTK_FILL), 0, 0);
	w = gtk_entry_new ();
	fs->m_SizeEntry = GTK_ENTRY (w);
	g_signal_connect (w, "activate", G_CALLBACK (on_size_activate), fs);
	g_signal_connect_after (w, "focus_out_event", G_CALLBACK (on_size_focus_out), fs);
	gcp_font_sel_set_size_full (fs, true);
	gtk_table_attach (table, w, 2, 3, 1, 2,
			(GtkAttachOptions) (0),
			(GtkAttachOptions) 0, 0, 0);
	PangoContext *pc = gtk_widget_get_pango_context (w);
	PangoLayout *pl = pango_layout_new (pc);
	pango_layout_set_text (pl, "0000000", -1);
	PangoRectangle rect;
	pango_layout_get_extents (pl, NULL, &rect);
	g_object_unref (G_OBJECT (pl));
	gtk_widget_set_size_request (sc, -1, rect.height / PANGO_SCALE * 12);
	gtk_widget_set_size_request (w, rect.width / PANGO_SCALE, -1);
	pango_context_list_families (pc, &families, &nb);
	fs->FamilyList = gtk_list_store_new (1, G_TYPE_STRING);
	gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (fs->FamilyList), 0, GTK_SORT_ASCENDING);
	fs->FamilyTree =  GTK_TREE_VIEW (gtk_tree_view_new_with_model (GTK_TREE_MODEL (fs->FamilyList)));
	gtk_tree_view_set_headers_visible (fs->FamilyTree, false);
	renderer = gtk_cell_renderer_text_new ();
	column = gtk_tree_view_column_new_with_attributes (NULL, renderer, "text", 0, NULL);
	gtk_tree_view_append_column (fs->FamilyTree, column);
	string name;
	for (i = 0; i < nb; i++) {
		PangoFontFace **faces;
		int *sizes, n;
		pango_font_family_list_faces (families[i], &faces, &n);
		if (n <= 0)
			continue;
		pango_font_face_list_sizes (faces[0], &sizes, &n);
		if (n > 0) // Do not use bitmap fonts
			continue;
		name = pango_font_family_get_name (families[i]);
		fs->Families[name] = (PangoFontFamily*) g_object_ref (families[i]);
		gtk_list_store_append (fs->FamilyList, &iter);
		gtk_list_store_set (fs->FamilyList, &iter,
				  0, name.c_str (),
				  -1);
	}
	fs->FamilySel = gtk_tree_view_get_selection (fs->FamilyTree);
	gtk_tree_selection_set_mode (fs->FamilySel, GTK_SELECTION_BROWSE);
	fs->FamilySignal = g_signal_connect (G_OBJECT (fs->FamilySel), "changed", G_CALLBACK (on_select_family), fs);
	sc = gtk_scrolled_window_new (NULL, NULL);
	gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (sc), GTK_POLICY_NEVER, GTK_POLICY_ALWAYS);
	gtk_scrolled_window_add_with_viewport (GTK_SCROLLED_WINDOW (sc), GTK_WIDGET (fs->FamilyTree));
	gtk_table_attach (table, sc, 0, 1, 1, 3,
			(GtkAttachOptions) (GTK_EXPAND | GTK_FILL),
			(GtkAttachOptions) (GTK_EXPAND | GTK_FILL), 0, 0);
}

GSF_CLASS (GcpFontSel, gcp_font_sel,
	   gcp_font_sel_class_init, gcp_font_sel_init,
	   GTK_TYPE_BIN)


syntax highlighted by Code2HTML, v. 0.9.1