/* * moohtml.c * * Copyright (C) 2004-2010 by Yevgen Muntyan * * This file is part of medit. medit is free software; you can * redistribute it and/or modify it under the terms of the * GNU Lesser General Public License as published by the * Free Software Foundation; either version 2.1 of the License, * or (at your option) any later version. * * You should have received a copy of the GNU Lesser General Public * License along with medit. If not, see . */ #include "config.h" #include "moohtml.h" #include "marshals.h" #include "mooutils/mooutils-misc.h" #include "mooutils/moocompat.h" #include #include #include #include #include #include #define DEFAULT_PAR_SPACING 6 struct _MooHtmlData { GHashTable *root_tags; /* char* -> MooHtmlTag* */ GSList *href_tags; char *title; htmlDocPtr doc; GHashTable *anchors; /* char* -> GtkTextMark* */ char *hover_link; double font_sizes[7]; int par_spacing[7]; double heading_sizes[6]; int heading_spacing[6]; char *heading_faces[6]; char *monospace; GHashTable *font_faces; char *filename; char *basename; char *dirname; gboolean new_line; gboolean space; gboolean button_pressed; gboolean in_drag; GSList *rulers; }; typedef enum { MOO_HTML_FG = 1 << 0, MOO_HTML_BG = 1 << 1, MOO_HTML_BOLD = 1 << 2, MOO_HTML_ITALIC = 1 << 3, MOO_HTML_UNDERLINE = 1 << 4, MOO_HTML_STRIKETHROUGH = 1 << 5, MOO_HTML_LINK = 1 << 6, MOO_HTML_SUB = 1 << 7, MOO_HTML_SUP = 1 << 8, MOO_HTML_LEFT_MARGIN = 1 << 9, MOO_HTML_PRE = 1 << 10, MOO_HTML_MONOSPACE = 1 << 11, MOO_HTML_LARGER = 1 << 12, MOO_HTML_SMALLER = 1 << 13, MOO_HTML_HEADING = 1 << 14, MOO_HTML_FONT_SIZE = 1 << 15, MOO_HTML_FONT_PT_SIZE = 1 << 16, MOO_HTML_FONT_FACE = 1 << 17 } MooHtmlAttrMask; struct _MooHtmlAttr { MooHtmlAttrMask mask; char *fg; char *bg; char *link; int left_margin; guint heading; guint font_size; guint font_pt_size; guint scale; char *font_face; }; static MooHtmlAttr *moo_html_attr_copy (const MooHtmlAttr *src); static void moo_html_attr_free (MooHtmlAttr *attr); static void moo_html_finalize (GObject *object); static void moo_html_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec); static void moo_html_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec); static MooHtmlData *moo_html_get_data (gpointer object); static void moo_html_tag_finalize (GObject *object); static void moo_html_tag_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec); static void moo_html_tag_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec); static void moo_html_size_allocate_cb (GtkWidget *widget, GtkAllocation *allocation); static void moo_html_size_allocate (GtkWidget *widget, GtkAllocation *allocation); static gboolean moo_html_button_press (GtkWidget *widget, GdkEventButton *event); static gboolean moo_html_button_release (GtkWidget *widget, GdkEventButton *event); static gboolean moo_html_motion (GtkWidget *widget, GdkEventMotion *event); static gboolean moo_html_load_url_real (MooHtml *html, const char *url); static void moo_html_clear (GtkTextView *view); static void moo_html_set_doc (GtkTextView *view, htmlDocPtr doc); static void moo_html_load_doc (GtkTextView *view, htmlDocPtr doc); static MooHtmlTag *moo_html_create_tag (GtkTextView *view, const MooHtmlAttr *attr, MooHtmlTag *parent, gboolean force); static void moo_html_create_anchor (GtkTextView *view, GtkTextBuffer *buffer, GtkTextIter *iter, const char *name); static MooHtmlTag *moo_html_get_link_tag (GtkTextIter *iter); static MooHtmlTag *moo_html_get_tag (GtkTextIter *iter); static gboolean moo_html_parse_url (const char *url, char **scheme, char **base, char **anchor); static gboolean moo_html_goto_anchor (GtkTextView *view, const char *anchor); static void moo_html_make_heading_tag (GtkTextView *view, MooHtmlTag *tag, guint heading); static void init_funcs (void); G_DEFINE_TYPE (MooHtml, _moo_html, GTK_TYPE_TEXT_VIEW) G_DEFINE_TYPE (MooHtmlTag, _moo_html_tag, GTK_TYPE_TEXT_TAG) enum { HTML_PROP_0, HTML_PROP_TITLE, HTML_PROP_MARKUP }; enum { LOAD_URL = 0, HOVER_LINK, NUM_HTML_SIGNALS }; enum { TAG_PROP_0, TAG_PROP_HREF }; static guint html_signals[NUM_HTML_SIGNALS]; /************************************************************************/ /* MooHtmlTag */ static void _moo_html_tag_class_init (MooHtmlTagClass *klass) { GObjectClass *gobject_class = G_OBJECT_CLASS (klass); gobject_class->finalize = moo_html_tag_finalize; gobject_class->set_property = moo_html_tag_set_property; gobject_class->get_property = moo_html_tag_get_property; g_object_class_install_property (gobject_class, TAG_PROP_HREF, g_param_spec_string ("href", "href", "href", NULL, (GParamFlags) G_PARAM_READWRITE)); } static void _moo_html_tag_init (MooHtmlTag *tag) { tag->href = NULL; tag->parent = NULL; tag->attr = NULL; } static void moo_html_tag_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { MooHtmlTag *tag = MOO_HTML_TAG (object); switch (prop_id) { case TAG_PROP_HREF: g_free (tag->href); tag->href = g_strdup (g_value_get_string (value)); g_object_notify (object, "href"); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); } } static void moo_html_tag_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { MooHtmlTag *tag = MOO_HTML_TAG (object); switch (prop_id) { case TAG_PROP_HREF: g_value_set_string (value, tag->href); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); } } static void moo_html_tag_finalize (GObject *object) { MooHtmlTag *tag = MOO_HTML_TAG (object); moo_html_attr_free (tag->attr); g_free (tag->href); G_OBJECT_CLASS(_moo_html_tag_parent_class)->finalize (object); } /************************************************************************/ /* MooHtml */ static void _moo_html_class_init (MooHtmlClass *klass) { GObjectClass *gobject_class = G_OBJECT_CLASS (klass); GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass); gobject_class->finalize = moo_html_finalize; gobject_class->set_property = moo_html_set_property; gobject_class->get_property = moo_html_get_property; widget_class->size_allocate = moo_html_size_allocate; widget_class->button_press_event = moo_html_button_press; widget_class->button_release_event = moo_html_button_release; widget_class->motion_notify_event = moo_html_motion; klass->load_url = moo_html_load_url_real; init_funcs (); g_object_class_install_property (gobject_class, HTML_PROP_TITLE, g_param_spec_string ("title", "title", "title", NULL, (GParamFlags) G_PARAM_READWRITE)); g_object_class_install_property (gobject_class, HTML_PROP_MARKUP, g_param_spec_string ("markup", "markup", "markup", NULL, G_PARAM_WRITABLE)); html_signals[LOAD_URL] = g_signal_new ("load-url", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION, G_STRUCT_OFFSET (MooHtmlClass, load_url), g_signal_accumulator_true_handled, NULL, _moo_marshal_BOOLEAN__STRING, G_TYPE_BOOLEAN, 1, G_TYPE_STRING); html_signals[HOVER_LINK] = g_signal_new ("hover-link", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION, G_STRUCT_OFFSET (MooHtmlClass, hover_link), NULL, NULL, _moo_marshal_VOID__STRING, G_TYPE_NONE, 1, G_TYPE_STRING); } static MooHtmlData * moo_html_data_new (void) { MooHtmlData *data = g_new0 (MooHtmlData, 1); data->anchors = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, NULL); data->root_tags = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, NULL); data->href_tags = NULL; data->doc = NULL; data->filename = NULL; data->basename = NULL; data->dirname = NULL; data->font_sizes[0] = PANGO_SCALE_X_SMALL; data->font_sizes[1] = PANGO_SCALE_SMALL; data->font_sizes[2] = PANGO_SCALE_MEDIUM; data->font_sizes[3] = PANGO_SCALE_LARGE; data->font_sizes[4] = PANGO_SCALE_X_LARGE; data->font_sizes[5] = PANGO_SCALE_XX_LARGE; data->font_sizes[6] = PANGO_SCALE_XX_LARGE * PANGO_SCALE_LARGE; data->heading_sizes[0] = PANGO_SCALE_XX_LARGE * PANGO_SCALE_LARGE; data->heading_sizes[1] = PANGO_SCALE_XX_LARGE; data->heading_sizes[2] = PANGO_SCALE_X_LARGE; data->heading_sizes[3] = PANGO_SCALE_LARGE; data->heading_sizes[4] = PANGO_SCALE_MEDIUM; data->heading_sizes[5] = PANGO_SCALE_SMALL; data->monospace = g_strdup ("Monospace"); data->font_faces = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free); return data; } static void moo_html_data_free (MooHtmlData *data) { if (data) { int i; g_hash_table_destroy (data->anchors); g_hash_table_destroy (data->root_tags); g_slist_free (data->href_tags); g_free (data->title); g_free (data->hover_link); if (data->doc) xmlFreeDoc (data->doc); g_hash_table_destroy (data->font_faces); g_free (data->monospace); for (i = 0; i < 6; ++i) g_free (data->heading_faces[i]); g_free (data->filename); g_free (data->basename); g_free (data->dirname); g_free (data); } } static MooHtmlData * moo_html_get_data (gpointer object) { MooHtmlData *data; data = g_object_get_data (object, "moo-html-data"); if (!data) { init_funcs (); data = moo_html_data_new (); g_object_set_data_full (object, "moo-html-data", data, (GDestroyNotify) moo_html_data_free); g_signal_connect (object, "size-allocate", G_CALLBACK (moo_html_size_allocate_cb), NULL); } return data; } static void _moo_html_init (MooHtml *html) { html->data = moo_html_data_new (); g_object_set_data (G_OBJECT (html), "moo-html-data", html->data); g_object_set (html, "cursor-visible", FALSE, "editable", FALSE, "wrap-mode", GTK_WRAP_WORD, "pixels-below-lines", DEFAULT_PAR_SPACING, NULL); } static void moo_html_set_title (GtkTextView *view, const char *title) { MooHtmlData *data = moo_html_get_data (view); g_free (data->title); data->title = g_strdup (title); if (MOO_IS_HTML (view)) g_object_notify (G_OBJECT (view), "title"); } static void moo_html_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { MooHtml *html = MOO_HTML (object); const char *string; switch (prop_id) { case HTML_PROP_TITLE: moo_html_set_title (GTK_TEXT_VIEW (html), g_value_get_string (value)); break; case HTML_PROP_MARKUP: string = g_value_get_string (value); if (!string) string = ""; _moo_html_load_memory (GTK_TEXT_VIEW (html), string, -1, NULL, NULL); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); } } static void moo_html_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { MooHtml *html = MOO_HTML (object); switch (prop_id) { case HTML_PROP_TITLE: g_value_set_string (value, html->data->title); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); } } static void moo_html_finalize (GObject *object) { MooHtml *html = MOO_HTML (object); moo_html_data_free (html->data); G_OBJECT_CLASS (_moo_html_parent_class)->finalize (object); } static gboolean moo_html_load_url (GtkTextView *view, const char *url) { char *scheme, *base, *anchor; gboolean result = FALSE; MooHtmlData *data = moo_html_get_data (view); g_return_val_if_fail (GTK_IS_TEXT_VIEW (view), FALSE); g_return_val_if_fail (url != NULL, FALSE); if (!moo_html_parse_url (url, &scheme, &base, &anchor)) { g_warning ("%s: invalid url '%s'", G_STRLOC, url); return FALSE; } if (!scheme) scheme = g_strdup ("file://"); if (!strcmp (scheme, "mailto:") || !strcmp (scheme, "mailto://")) { result = moo_open_email (base, NULL, NULL); goto out; } if (strcmp (scheme, "file://")) goto out; if (!base || (data->basename && !strcmp (data->basename, base))) { if (anchor) result = moo_html_goto_anchor (view, anchor); else result = TRUE; } else if (!g_path_is_absolute (base)) { if (data->dirname) { char *filename = g_build_filename (data->dirname, base, NULL); result = _moo_html_load_file (view, filename, NULL); if (result && anchor) moo_html_goto_anchor (view, anchor); g_free (filename); } } else { result = _moo_html_load_file (view, base, NULL); if (result && anchor) moo_html_goto_anchor (view, anchor); } out: g_free (scheme); g_free (base); g_free (anchor); return result; } static gboolean moo_html_load_url_real (MooHtml *html, const char *url) { return moo_html_load_url (GTK_TEXT_VIEW (html), url); } static void remove_tag (GtkTextTag *tag, GtkTextTagTable *table) { gtk_text_tag_table_remove (table, tag); } static void moo_html_clear (GtkTextView *view) { GtkTextBuffer *buffer; GtkTextTagTable *table; GtkTextIter start, end; MooHtmlData *data = moo_html_get_data (view); g_hash_table_destroy (data->anchors); data->anchors = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, NULL); buffer = gtk_text_view_get_buffer (view); table = gtk_text_buffer_get_tag_table (buffer); g_slist_foreach (data->href_tags, (GFunc) remove_tag, table); g_slist_free (data->href_tags); data->href_tags = NULL; gtk_text_buffer_get_bounds (buffer, &start, &end); gtk_text_buffer_delete (buffer, &start, &end); data->new_line = TRUE; data->space = TRUE; if (data->doc) xmlFreeDoc (data->doc); data->doc = NULL; g_slist_free (data->rulers); data->rulers = NULL; } void moo_text_view_set_markup (GtkTextView *view, const char *markup) { g_return_if_fail (GTK_IS_TEXT_VIEW (view)); g_return_if_fail (markup != NULL); _moo_html_load_memory (view, markup, -1, NULL, NULL); } gboolean _moo_html_load_memory (GtkTextView *view, const char *buffer, int size, const char *url, const char *encoding) { htmlDocPtr doc; MooHtmlData *data; g_return_val_if_fail (GTK_IS_TEXT_VIEW (view), FALSE); g_return_val_if_fail (buffer != NULL, FALSE); data = moo_html_get_data (view); if (size < 0) size = strlen (buffer); doc = htmlReadMemory (buffer, size, url, encoding, HTML_PARSE_NONET); if (!doc) { g_warning ("%s: htmlReadMemory failed", G_STRLOC); return FALSE; } g_free (data->filename); g_free (data->basename); g_free (data->dirname); data->filename = NULL; data->basename = NULL; data->dirname = NULL; moo_html_set_doc (view, doc); xmlCleanupParser (); return TRUE; } gboolean _moo_html_load_file (GtkTextView *view, const char *file, const char *encoding) { htmlDocPtr doc; MooHtmlData *data = moo_html_get_data (view); g_return_val_if_fail (GTK_IS_TEXT_VIEW (view), FALSE); g_return_val_if_fail (file != NULL, FALSE); doc = htmlReadFile (file, encoding, HTML_PARSE_NONET); if (!doc) return FALSE; g_free (data->filename); g_free (data->basename); g_free (data->dirname); data->filename = g_strdup (file); data->basename = g_path_get_basename (file); data->dirname = g_path_get_dirname (file); moo_html_set_doc (view, doc); xmlCleanupParser (); return TRUE; } static void moo_html_set_doc (GtkTextView *view, htmlDocPtr doc) { MooHtmlData *data; g_return_if_fail (doc != NULL); data = moo_html_get_data (view); g_return_if_fail (doc != data->doc); moo_html_clear (view); data->doc = doc; moo_html_load_doc (view, doc); } static void attr_compose (MooHtmlAttr *dest, const MooHtmlAttr *src) { static MooHtmlAttrMask simple = MOO_HTML_BOLD | MOO_HTML_ITALIC | MOO_HTML_UNDERLINE | MOO_HTML_STRIKETHROUGH | MOO_HTML_MONOSPACE | MOO_HTML_SUB | MOO_HTML_SUP | MOO_HTML_PRE; static MooHtmlAttrMask font_size_mask = MOO_HTML_LARGER | MOO_HTML_SMALLER | MOO_HTML_HEADING | MOO_HTML_FONT_SIZE | MOO_HTML_FONT_PT_SIZE; static MooHtmlAttrMask font_face_mask = MOO_HTML_FONT_FACE | MOO_HTML_MONOSPACE; g_return_if_fail (dest != NULL); if (!src) return; dest->mask |= (src->mask & simple); if (src->mask & MOO_HTML_FG) { dest->mask |= MOO_HTML_FG; dest->fg = src->fg; } if (src->mask & MOO_HTML_BG) { dest->mask |= MOO_HTML_BG; dest->bg = src->bg; } if (src->mask & MOO_HTML_LINK) { dest->mask |= MOO_HTML_LINK; dest->link = src->link; } if (src->mask & MOO_HTML_LEFT_MARGIN) { if (!(dest->mask & MOO_HTML_LEFT_MARGIN)) { dest->mask |= MOO_HTML_LEFT_MARGIN; dest->left_margin = src->left_margin; } else { dest->left_margin += src->left_margin; } } if ((src->mask & font_face_mask) && !(dest->mask & font_face_mask)) { dest->mask |= (src->mask & font_face_mask); dest->font_face = src->font_face; } if (dest->mask & MOO_HTML_HEADING) { g_assert (1 <= dest->heading && dest->heading <= 6); } else if (src->mask & MOO_HTML_HEADING) { dest->mask &= ~font_size_mask; dest->mask |= MOO_HTML_HEADING; dest->heading = src->heading; g_assert (1 <= dest->heading && dest->heading <= 6); } else if (dest->mask & (MOO_HTML_LARGER | MOO_HTML_SMALLER)) { int size = 3; int scale = (dest->mask & MOO_HTML_LARGER) ? (int)dest->scale : -((int)dest->scale); if (src->mask & (MOO_HTML_LARGER | MOO_HTML_SMALLER)) { if (src->mask & MOO_HTML_LARGER) scale += src->scale; else scale -= src->scale; } else if (src->mask & MOO_HTML_FONT_SIZE) { size = src->font_size; } else if (src->mask & MOO_HTML_FONT_PT_SIZE) { /* XXX ??? */ } size += scale; size = CLAMP (size, 1, 7); dest->mask &= ~font_size_mask; dest->mask |= MOO_HTML_FONT_SIZE; dest->font_size = size; } else if (dest->mask & MOO_HTML_FONT_SIZE) { dest->mask &= ~font_size_mask; dest->mask |= MOO_HTML_FONT_SIZE; } else if (dest->mask & MOO_HTML_FONT_PT_SIZE) { dest->mask &= ~font_size_mask; dest->mask |= MOO_HTML_FONT_PT_SIZE; } else { dest->mask &= ~font_size_mask; dest->mask |= (src->mask & font_size_mask); dest->heading = src->heading; dest->font_size = src->font_size; dest->font_pt_size = src->font_pt_size; dest->scale = src->scale; } } static void attr_apply (const MooHtmlAttr *attr, MooHtmlTag *tag, GtkTextView *view) { MooHtmlData *data = moo_html_get_data (view); g_return_if_fail (attr != NULL && tag != NULL); moo_html_attr_free (tag->attr); tag->attr = moo_html_attr_copy (attr); if (attr->mask & MOO_HTML_FG) g_object_set (tag, "foreground", attr->fg, NULL); if (attr->mask & MOO_HTML_BG) g_object_set (tag, "background", attr->bg, NULL); if (attr->mask & MOO_HTML_BOLD) g_object_set (tag, "weight", PANGO_WEIGHT_BOLD, NULL); if (attr->mask & MOO_HTML_ITALIC) g_object_set (tag, "style", PANGO_STYLE_ITALIC, NULL); if (attr->mask & MOO_HTML_UNDERLINE) g_object_set (tag, "underline", PANGO_UNDERLINE_SINGLE, NULL); if (attr->mask & MOO_HTML_STRIKETHROUGH) g_object_set (tag, "strikethrough", TRUE, NULL); if (attr->mask & MOO_HTML_LEFT_MARGIN) g_object_set (tag, "left-margin", attr->left_margin, NULL); if (attr->mask & MOO_HTML_LINK) { g_free (tag->href); tag->href = g_strdup (attr->link); g_object_set (tag, "foreground", "blue", NULL); } if (attr->mask & MOO_HTML_SUP) g_object_set (tag, "rise", 8 * PANGO_SCALE, "size", 8 * PANGO_SCALE, NULL); if (attr->mask & MOO_HTML_SUB) g_object_set (tag, "rise", -8 * PANGO_SCALE, "size", 8 * PANGO_SCALE, NULL); if (attr->mask & MOO_HTML_MONOSPACE) g_object_set (tag, "font", data->monospace, NULL); else if (attr->mask & MOO_HTML_FONT_FACE) g_object_set (tag, "font", attr->font_face, NULL); if (attr->mask & MOO_HTML_HEADING) { moo_html_make_heading_tag (view, tag, attr->heading); } else if (attr->mask & MOO_HTML_LARGER) { double scale; int space; int size = 3 + attr->scale; size = CLAMP (size, 1, 7); scale = data->font_sizes[size - 1]; space = data->par_spacing[size - 1]; g_object_set (tag, "scale", scale, "pixels-below-lines", DEFAULT_PAR_SPACING + space, NULL); } else if (attr->mask & MOO_HTML_SMALLER) { double scale; int space; int size = 3 - attr->scale; size = CLAMP (size, 1, 7); scale = data->font_sizes[size - 1]; space = data->par_spacing[size - 1]; g_object_set (tag, "scale", scale, "pixels-below-lines", DEFAULT_PAR_SPACING + space, NULL); } else if (attr->mask & MOO_HTML_FONT_SIZE) { g_assert (1 <= attr->font_size && attr->font_size <= 7); g_object_set (tag, "scale", data->font_sizes[attr->font_size - 1], "pixels-below-lines", DEFAULT_PAR_SPACING + data->par_spacing[attr->font_size - 1], NULL); } else if (attr->mask & MOO_HTML_FONT_PT_SIZE) { g_object_set (tag, "size-points", (double) attr->font_pt_size, NULL); } } static void moo_html_make_heading_tag (GtkTextView *view, MooHtmlTag *tag, guint heading) { MooHtmlData *data = moo_html_get_data (view); g_assert (1 <= heading && heading <= 6); g_object_set (tag, "pixels-below-lines", DEFAULT_PAR_SPACING + data->heading_spacing[heading - 1], "scale", data->heading_sizes[heading - 1], "weight", PANGO_WEIGHT_BOLD, NULL); if (data->heading_faces[heading - 1]) g_object_set (tag, "family", data->heading_faces[heading - 1], NULL); } static MooHtmlTag* moo_html_create_tag (GtkTextView *view, const MooHtmlAttr *attr, MooHtmlTag *parent, gboolean force) { MooHtmlTag *tag; MooHtmlAttr real_attr; MooHtmlData *data = moo_html_get_data (view); g_return_val_if_fail (attr != NULL, NULL); if (!attr->mask && !force) return parent; if (parent && parent->attr) { real_attr = *parent->attr; attr_compose (&real_attr, attr); } else { real_attr = *attr; } tag = g_object_new (MOO_TYPE_HTML_TAG, (const char*) NULL); gtk_text_tag_table_add (gtk_text_buffer_get_tag_table (gtk_text_view_get_buffer (view)), GTK_TEXT_TAG (tag)); g_object_unref (tag); if (tag->href) data->href_tags = g_slist_prepend (data->href_tags, tag); attr_apply (&real_attr, tag, view); return tag; } static MooHtmlAttr* moo_html_attr_copy (const MooHtmlAttr *src) { MooHtmlAttr *attr; g_return_val_if_fail (src != NULL, NULL); attr = g_new (MooHtmlAttr, 1); *attr = *src; attr->fg = g_strdup (src->fg); attr->bg = g_strdup (src->bg); attr->link = g_strdup (src->link); attr->font_face = g_strdup (src->font_face); return attr; } static void moo_html_attr_free (MooHtmlAttr *attr) { if (attr) { g_free (attr->fg); g_free (attr->bg); g_free (attr->link); g_free (attr->font_face); g_free (attr); } } static void moo_html_create_anchor (GtkTextView *view, GtkTextBuffer *buffer, GtkTextIter *iter, const char *name) { GtkTextMark *mark; char *alt_name; MooHtmlData *data = moo_html_get_data (view); g_return_if_fail (name != NULL && (name[0] != '#' || name[1])); mark = gtk_text_buffer_create_mark (buffer, NULL, iter, TRUE); if (name[0] == '#') alt_name = g_strdup (name + 1); else alt_name = g_strdup_printf ("#%s", name); g_hash_table_insert (data->anchors, g_strdup (name), mark); g_hash_table_insert (data->anchors, alt_name, mark); } static gboolean moo_html_motion (GtkWidget *widget, GdkEventMotion *event) { GtkTextView *textview = GTK_TEXT_VIEW (widget); MooHtml *html = MOO_HTML (widget); GtkTextIter iter; int buf_x, buf_y, x, y, dummy; GdkModifierType state; MooHtmlTag *tag; if (html->data->button_pressed) html->data->in_drag = TRUE; if (event->window != gtk_text_view_get_window (textview, GTK_TEXT_WINDOW_TEXT)) goto out; if (event->is_hint) { gdk_window_get_pointer (event->window, &x, &y, &state); } else { x = event->x; y = event->y; state = event->state; } if (state & (GDK_BUTTON1_MASK | GDK_BUTTON2_MASK | GDK_BUTTON3_MASK)) goto out; gtk_text_view_window_to_buffer_coords (textview, GTK_TEXT_WINDOW_TEXT, x, y, &buf_x, &buf_y); gtk_text_view_get_iter_at_position (textview, &iter, &dummy, buf_x, buf_y); tag = moo_html_get_link_tag (&iter); if (tag) { g_return_val_if_fail (tag->href != NULL, FALSE); if (!html->data->hover_link || strcmp (html->data->hover_link, tag->href)) { GdkCursor *cursor; g_free (html->data->hover_link); html->data->hover_link = g_strdup (tag->href); cursor = gdk_cursor_new (GDK_HAND2); gdk_window_set_cursor (event->window, cursor); gdk_cursor_unref (cursor); g_signal_emit (html, html_signals[HOVER_LINK], 0, tag->href); } } else if (html->data->hover_link) { GdkCursor *cursor; g_free (html->data->hover_link); html->data->hover_link = NULL; cursor = gdk_cursor_new (GDK_XTERM); gdk_window_set_cursor (event->window, cursor); gdk_cursor_unref (cursor); g_signal_emit (html, html_signals[HOVER_LINK], 0, NULL); } out: return GTK_WIDGET_CLASS(_moo_html_parent_class)->motion_notify_event (widget, event); } static MooHtmlTag* moo_html_get_link_tag (GtkTextIter *iter) { MooHtmlTag *tag = moo_html_get_tag (iter); return (tag && tag->href) ? tag : NULL; } static MooHtmlTag* moo_html_get_tag (GtkTextIter *iter) { MooHtmlTag *tag = NULL; GSList *l; GSList *list = gtk_text_iter_get_tags (iter); for (l = list; l != NULL; l = l->next) { if (MOO_IS_HTML_TAG (l->data)) { tag = l->data; break; } } g_slist_free (list); return tag; } static gboolean moo_html_button_press (GtkWidget *widget, GdkEventButton *event) { MooHtml *html = MOO_HTML (widget); html->data->button_pressed = TRUE; html->data->in_drag = FALSE; return GTK_WIDGET_CLASS(_moo_html_parent_class)->button_press_event (widget, event); } static gboolean moo_html_button_release (GtkWidget *widget, GdkEventButton *event) { GtkTextView *textview = GTK_TEXT_VIEW (widget); MooHtml *html = MOO_HTML (widget); GtkTextIter iter; int buf_x, buf_y, dummy; MooHtmlTag *tag; html->data->button_pressed = FALSE; if (html->data->in_drag) { html->data->in_drag = FALSE; goto out; } if (event->window != gtk_text_view_get_window (textview, GTK_TEXT_WINDOW_TEXT)) goto out; gtk_text_view_window_to_buffer_coords (textview, GTK_TEXT_WINDOW_TEXT, event->x, event->y, &buf_x, &buf_y); gtk_text_view_get_iter_at_position (textview, &iter, &dummy, buf_x, buf_y); tag = moo_html_get_link_tag (&iter); if (tag) { gboolean result; g_assert (tag->href != NULL); g_signal_emit (html, html_signals[LOAD_URL], 0, tag->href, &result); } out: return GTK_WIDGET_CLASS(_moo_html_parent_class)->button_release_event (widget, event); } static gboolean moo_html_parse_url (const char *url, char **scheme, char **base, char **anchor) { GRegex *regex; GMatchInfo *match_info; g_return_val_if_fail (url != NULL, FALSE); g_return_val_if_fail (scheme && base && anchor, FALSE); regex = g_regex_new ("^([a-zA-Z]+:(//)?)?([^#]*)(#(.*))?$", 0, 0, NULL); g_return_val_if_fail (regex != NULL, FALSE); if (!g_regex_match (regex, url, 0, &match_info)) { g_match_info_free (match_info); g_regex_unref (regex); return FALSE; } *scheme = g_match_info_fetch (match_info, 1); *base = g_match_info_fetch (match_info, 3); *anchor = g_match_info_fetch (match_info, 5); if (!*scheme || !**scheme) {g_free (*scheme); *scheme = NULL;} if (!*base || !**base) {g_free (*base); *base = NULL;} if (!*anchor || !**anchor) {g_free (*anchor); *anchor = NULL;} g_match_info_free (match_info); g_regex_unref (regex); return TRUE; } static gboolean moo_html_goto_anchor (GtkTextView *view, const char *anchor) { GtkTextMark *mark; MooHtmlData *data = moo_html_get_data (view); g_return_val_if_fail (anchor != NULL, FALSE); mark = g_hash_table_lookup (data->anchors, anchor); if (!mark) { g_warning ("%s: could not find anchor '%s'", G_STRLOC, anchor); return FALSE; } else { gtk_text_view_scroll_to_mark (view, mark, 0.1, TRUE, 0, 0); return TRUE; } } #if 0 void moo_html_set_font (MooHtml *html, const char *string) { PangoFontDescription *font; g_return_if_fail (MOO_IS_HTML (html)); g_return_if_fail (string != NULL); font = pango_font_description_from_string (string); g_return_if_fail (font != NULL); gtk_widget_modify_font (GTK_WIDGET (html), font); pango_font_description_free (font); } #endif static void moo_html_size_allocate_real (GtkWidget *widget, G_GNUC_UNUSED GtkAllocation *allocation) { int border_width, child_width, height; GSList *l; GdkWindow *window; MooHtmlData *data = moo_html_get_data (widget); if (!GTK_WIDGET_REALIZED (widget)) return; window = gtk_text_view_get_window (GTK_TEXT_VIEW (widget), GTK_TEXT_WINDOW_TEXT); g_return_if_fail (window != NULL); gdk_drawable_get_size (window, &child_width, &height); border_width = gtk_container_get_border_width (GTK_CONTAINER (widget)); child_width -= 2 * border_width + 2 * widget->style->xthickness + gtk_text_view_get_left_margin (GTK_TEXT_VIEW (widget)) + gtk_text_view_get_right_margin (GTK_TEXT_VIEW (widget)); child_width = MAX (child_width, 0); for (l = data->rulers; l != NULL; l = l->next) { GtkWidget *ruler = l->data; gtk_widget_set_size_request (ruler, child_width, -1); } } static void moo_html_size_allocate (GtkWidget *widget, GtkAllocation *allocation) { GTK_WIDGET_CLASS(_moo_html_parent_class)->size_allocate (widget, allocation); moo_html_size_allocate_real (widget, allocation); } static void moo_html_size_allocate_cb (GtkWidget *widget, GtkAllocation *allocation) { moo_html_size_allocate_real (widget, allocation); } /***********************************************************************/ /* Loading into text buffer */ #define IS_ELEMENT(node_) (node_ && node_->type == XML_ELEMENT_NODE) #define NODE_NAME_IS_(node_,name_) (node_->name && !strcmp ((char*)node_->name, name_)) #define IS_NAMED_ELM_(node_,name_) (IS_ELEMENT(node_) && NODE_NAME_IS_(node_, name_)) #define IS_HEAD_ELEMENT(node_) IS_NAMED_ELM_(node_, "head") #define IS_BODY_ELEMENT(node_) IS_NAMED_ELM_(node_, "body") #define IS_TITLE_ELEMENT(node_) IS_NAMED_ELM_(node_, "title") #define IS_META_ELEMENT(node_) IS_NAMED_ELM_(node_, "meta") #define IS_LINK_ELEMENT(node_) IS_NAMED_ELM_(node_, "link") #define IS_LI_ELEMENT(node_) IS_NAMED_ELM_(node_, "li") #define IS_IMG_ELEMENT(node_) IS_NAMED_ELM_(node_, "img") #define IS_HEADING_ELEMENT(node_) (IS_ELEMENT(node_) && node_->name && \ node_->name[0] == 'h' && \ ('1' <= node_->name[1] && node_->name[1] <= '6') && \ !node_->name[2]) #define IS_TEXT(node_) (node_ && node_->type == XML_TEXT_NODE) #define IS_COMMENT(node_) (node_ && node_->type == XML_COMMENT_NODE) #define STR_FREE(s__) if (s__) xmlFree (s__) static void moo_html_load_head (GtkTextView *view, xmlNode *node); static void moo_html_load_body (GtkTextView *view, xmlNode *node); static void moo_html_new_line (GtkTextView *view, GtkTextBuffer *buffer, GtkTextIter *iter, MooHtmlTag *tag, gboolean force); static void moo_html_insert_text (GtkTextView *view, GtkTextBuffer *buffer, GtkTextIter *iter, MooHtmlTag *tag, const char *text); static void moo_html_insert_verbatim(GtkTextView *view, GtkTextBuffer *buffer, GtkTextIter *iter, MooHtmlTag *tag, const char *text); static void moo_html_load_doc (GtkTextView *view, htmlDocPtr doc) { xmlNode *root, *node; root = xmlDocGetRootElement (doc); if (!root) { g_message ("moo_html_load_doc: empty document"); return; } for (node = root->children; node != NULL; node = node->next) { if (IS_HEAD_ELEMENT (node)) { moo_html_load_head (view, node); } else if (IS_BODY_ELEMENT (node)) { moo_html_load_body (view, node); } else { g_warning ("%s: unknown node '%s'", G_STRLOC, node->name); } } } static void moo_html_load_head (GtkTextView *view, xmlNode *node) { xmlNode *child; for (child = node->children; child != NULL; child = child->next) { if (IS_TITLE_ELEMENT (child)) { xmlChar *title = xmlNodeGetContent (child); moo_html_set_title (view, (char *) title); STR_FREE (title); } else if (IS_META_ELEMENT (child)) { } else if (IS_LINK_ELEMENT (child)) { } else { g_message ("%s: unknown node '%s'", G_STRLOC, child->name); } } } static void moo_html_new_line (GtkTextView *view, GtkTextBuffer *buffer, GtkTextIter *iter, MooHtmlTag *tag, gboolean force) { MooHtmlData *data = moo_html_get_data (view); if (!data->new_line || force) { if (tag) gtk_text_buffer_insert_with_tags (buffer, iter, "\n", 1, GTK_TEXT_TAG (tag), NULL); else gtk_text_buffer_insert (buffer, iter, "\n", 1); } data->new_line = TRUE; data->space = TRUE; } static const char* str_find_separator (const char *str) { const char *p; for (p = str; *p; ++p) { if (*p == '\n' || *p == '\r' || *p == ' ' || *p == '\t') return p; } return NULL; } static void moo_html_insert_text (GtkTextView *view, GtkTextBuffer *buffer, GtkTextIter *iter, MooHtmlTag *tag, const char *text) { const char *p; MooHtmlData *data = moo_html_get_data (view); if (tag && tag->attr && (tag->attr->mask & MOO_HTML_PRE)) { moo_html_insert_verbatim (view, buffer, iter, tag, text); return; } while (*text) { p = str_find_separator (text); if (p) { if (p != text) { if (tag) { gtk_text_buffer_insert_with_tags (buffer, iter, text, p - text, GTK_TEXT_TAG (tag), NULL); gtk_text_buffer_insert_with_tags (buffer, iter, " ", 1, GTK_TEXT_TAG (tag), NULL); } else { gtk_text_buffer_insert (buffer, iter, text, p - text); gtk_text_buffer_insert (buffer, iter, " ", 1); } data->space = TRUE; data->new_line = FALSE; text = ++p; } else { if (!data->space) { gtk_text_buffer_insert (buffer, iter, " ", 1); data->space = TRUE; } text++; } } else { if (tag) gtk_text_buffer_insert_with_tags (buffer, iter, text, -1, GTK_TEXT_TAG (tag), NULL); else gtk_text_buffer_insert (buffer, iter, text, -1); data->new_line = FALSE; data->space = FALSE; break; } } } static gboolean str_has_trailing_nl (const char *text, int len) { g_assert (len > 0); for (len = len - 1; len >= 0; --len) { if (text[len] == '\n' || text[len] == '\r') return TRUE; else if (text[len] != ' ' && text[len] != '\t') return FALSE; } return FALSE; } static gboolean str_has_trailing_space (const char *text, int len) { g_assert (len > 0); if (text[len-1] == '\n' || text[len-1] == '\r' || text[len-1] == ' ' || text[len-1] == '\t') return TRUE; else return FALSE; } static void moo_html_insert_verbatim (GtkTextView *view, GtkTextBuffer *buffer, GtkTextIter *iter, MooHtmlTag *tag, const char *text) { guint len; MooHtmlData *data = moo_html_get_data (view); g_return_if_fail (text != NULL); if (text[0] == '\n' && data->new_line) text++; len = strlen (text); if (!len) return; if (tag) gtk_text_buffer_insert_with_tags (buffer, iter, text, len, GTK_TEXT_TAG (tag), NULL); else gtk_text_buffer_insert (buffer, iter, text, len); data->new_line = str_has_trailing_nl (text, len); if (data->new_line) data->space = TRUE; else data->space = str_has_trailing_space (text, len); } static void process_elm_body (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *current, GtkTextIter *iter); static void process_text_node (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *node, MooHtmlTag *current, GtkTextIter *iter); static void process_heading_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *current, GtkTextIter *iter); static void process_format_elm (GtkTextView *view, MooHtmlAttr *attr, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *current, GtkTextIter *iter); static void process_img_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *current, GtkTextIter *iter); static void process_p_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter); static void process_a_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter); static void process_pre_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter); static void process_ol_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter); static void process_ul_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter); static void process_font_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter); static void process_cite_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter); static void process_li_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter); static void process_dt_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter); static void process_dl_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter); static void process_dd_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter); static void process_br_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter); static void process_div_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter); static void process_span_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter); static void process_hr_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter); static void process_table_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter); static void process_tr_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter); static void moo_html_load_body (GtkTextView *view, xmlNode *node) { GtkTextIter iter; GtkTextBuffer *buffer; MooHtmlData *data = moo_html_get_data (view); data->new_line = TRUE; data->space = TRUE; buffer = gtk_text_view_get_buffer (view); gtk_text_buffer_get_end_iter (buffer, &iter); process_elm_body (view, buffer, node, NULL, &iter); } typedef struct { MooHtmlAttrMask mask; const char *name; } MaskNamePair; static MooHtmlAttr* get_format_elm_attr (xmlNode *node) { static GHashTable *elms = NULL; MooHtmlAttr *attr; if (!IS_ELEMENT (node)) return NULL; if (!elms) { guint i; static MaskNamePair attrs[] = { { MOO_HTML_BOLD, "strong" }, { MOO_HTML_BOLD, "b" }, { MOO_HTML_ITALIC, "em" }, { MOO_HTML_ITALIC, "i" }, { MOO_HTML_ITALIC, "address" }, { MOO_HTML_UNDERLINE, "ins" }, { MOO_HTML_UNDERLINE, "u" }, { MOO_HTML_STRIKETHROUGH, "del" }, { MOO_HTML_STRIKETHROUGH, "s" }, { MOO_HTML_STRIKETHROUGH, "strike" }, { MOO_HTML_MONOSPACE, "code" }, { MOO_HTML_MONOSPACE, "dfn" }, { MOO_HTML_MONOSPACE, "samp" }, { MOO_HTML_MONOSPACE, "kbd" }, { MOO_HTML_MONOSPACE, "var" }, { MOO_HTML_MONOSPACE, "tt" }, { MOO_HTML_SUB, "sub" }, { MOO_HTML_SUP, "sup" } }; elms = g_hash_table_new (g_str_hash, g_str_equal); for (i = 0; i < G_N_ELEMENTS (attrs); ++i) { attr = g_new0 (MooHtmlAttr, 1); attr->mask = attrs[i].mask; g_hash_table_insert (elms, (char*) attrs[i].name, attr); } attr = g_new0 (MooHtmlAttr, 1); attr->mask = MOO_HTML_LARGER; attr->scale = 1; g_hash_table_insert (elms, (char*) "big", attr); attr = g_new0 (MooHtmlAttr, 1); attr->mask = MOO_HTML_SMALLER; attr->scale = 1; g_hash_table_insert (elms, (char*) "small", attr); } return g_hash_table_lookup (elms, node->name); } typedef void (*ProcessElm) (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter); static GHashTable *proc_elm_funcs__ = NULL; static void add_func__ (const char *static_elm_name, ProcessElm func) { g_hash_table_insert (proc_elm_funcs__, (char*) static_elm_name, func); } static void init_funcs (void) { if (!proc_elm_funcs__) { proc_elm_funcs__ = g_hash_table_new (g_str_hash, g_str_equal); add_func__ ("p", process_p_elm); add_func__ ("a", process_a_elm); add_func__ ("ol", process_ol_elm); add_func__ ("ul", process_ul_elm); add_func__ ("font", process_font_elm); add_func__ ("pre", process_pre_elm); add_func__ ("cite", process_cite_elm); add_func__ ("ul", process_ul_elm); add_func__ ("li", process_li_elm); add_func__ ("hr", process_hr_elm); add_func__ ("img", process_img_elm); add_func__ ("dt", process_dt_elm); add_func__ ("dl", process_dl_elm); add_func__ ("dd", process_dd_elm); add_func__ ("br", process_br_elm); add_func__ ("div", process_div_elm); add_func__ ("span", process_span_elm); } } static ProcessElm get_proc_elm_func (xmlNode *node) { if (IS_ELEMENT (node)) return (ProcessElm) g_hash_table_lookup (proc_elm_funcs__, node->name); else return NULL; } static void process_elm_body (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *current, GtkTextIter *iter) { xmlNode *child; MooHtmlAttr *attr; ProcessElm func; for (child = elm->children; child != NULL; child = child->next) { if (IS_TEXT (child)) process_text_node (view, buffer, child, current, iter); else if (IS_HEADING_ELEMENT (child)) process_heading_elm (view, buffer, child, current, iter); else if ((func = get_proc_elm_func (child))) func (view, buffer, child, current, iter); else if ((attr = get_format_elm_attr (child))) process_format_elm (view, attr, buffer, child, current, iter); else if (IS_NAMED_ELM_ (child, "table")) process_table_elm (view, buffer, child, current, iter); else if (IS_NAMED_ELM_ (child, "tr")) process_tr_elm (view, buffer, child, current, iter); else if (IS_NAMED_ELM_ (child, "td") || IS_NAMED_ELM_ (child, "th") || IS_NAMED_ELM_ (child, "tbody") || IS_NAMED_ELM_ (child, "col")) { process_elm_body (view, buffer, child, current, iter); } else if (IS_ELEMENT (child)) { g_message ("%s: unknown node '%s'", G_STRLOC, child->name); process_elm_body (view, buffer, child, current, iter); } else if (IS_COMMENT (child)) { /* ignore */ } else { g_warning ("%s: unknown node", G_STRLOC); } } } static void process_p_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *current, GtkTextIter *iter) { moo_html_new_line (view, buffer, iter, current, FALSE); process_elm_body (view, buffer, elm, current, iter); moo_html_new_line (view, buffer, iter, current, FALSE); } static void process_heading_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter) { static MooHtmlAttr attr; MooHtmlTag *current; int n; g_return_if_fail (elm->name[0] && elm->name[1]); n = elm->name[1] - '0'; g_return_if_fail (1 <= n && n <= 6); attr.mask = MOO_HTML_HEADING; attr.heading = n; current = moo_html_create_tag (view, &attr, parent, FALSE); moo_html_new_line (view, buffer, iter, current, FALSE); process_elm_body (view, buffer, elm, current, iter); moo_html_new_line (view, buffer, iter, current, FALSE); } static void process_text_node (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *node, MooHtmlTag *current, GtkTextIter *iter) { moo_html_insert_text (view, buffer, iter, current, (char*) node->content); } #define GET_PROP(elm__,prop__) (xmlGetProp (elm__, (const guchar*) prop__)) #define STRCMP(xm__,normal__) (strcmp ((char*) xm__, normal__)) static void process_a_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter) { xmlChar *href, *name; href = GET_PROP (elm, "href"); name = GET_PROP (elm, "name"); if (!name) name = GET_PROP (elm, "id"); if (href) { static MooHtmlAttr attr; MooHtmlTag *current; attr.mask = MOO_HTML_LINK; attr.link = (char*) href; current = moo_html_create_tag (view, &attr, parent, FALSE); process_elm_body (view, buffer, elm, current, iter); } else if (name) { moo_html_create_anchor (view, buffer, iter, (char*) name); } STR_FREE (href); STR_FREE (name); } static gboolean parse_int (const char *str, int *dest) { long num; if (!str) return FALSE; errno = 0; num = strtol (str, NULL, 10); if (errno) return FALSE; #if G_MAXINT != G_MAXLONG if (num < G_MININT || num > G_MAXINT) return FALSE; #endif if (dest) *dest = num; return TRUE; } typedef enum { OL_NUM = 0, OL_LOWER_ALPHA, OL_UPPER_ALPHA, OL_LOWER_ROMAN, OL_UPPER_ROMAN } OLType; static char* make_li_number (int count, OLType type) { g_return_val_if_fail (count > 0, NULL); switch (type) { case OL_UPPER_ROMAN: case OL_LOWER_ROMAN: g_warning ("%s: implement me", G_STRLOC); case OL_NUM: return g_strdup_printf (" %d. ", count); case OL_LOWER_ALPHA: g_return_val_if_fail (count <= 26, NULL); return g_strdup_printf (" %c. ", count - 1 + 'a'); case OL_UPPER_ALPHA: g_return_val_if_fail (count <= 26, NULL); return g_strdup_printf (" %c. ", count - 1 + 'A'); } g_return_val_if_reached (NULL); } static void process_ol_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *current, GtkTextIter *iter) { int count; OLType list_type = OL_NUM; xmlNode *child; xmlChar *start = NULL, *type = NULL; MooHtmlData *data = moo_html_get_data (view); count = 1; start = GET_PROP (elm, "start"); parse_int ((char*) start, &count); if ((type = GET_PROP (elm, "type"))) { if (!STRCMP (type, "1")) list_type = OL_NUM; else if (!STRCMP (type, "a")) list_type = OL_LOWER_ALPHA; else if (!STRCMP (type, "A")) list_type = OL_UPPER_ALPHA; else if (!STRCMP (type, "i")) list_type = OL_LOWER_ROMAN; else if (!STRCMP (type, "I")) list_type = OL_UPPER_ROMAN; else { g_warning ("%s: invalid type attribute '%s'", G_STRLOC, type); } } moo_html_new_line (view, buffer, iter, current, FALSE); for (child = elm->children; child != NULL; child = child->next) { if (IS_LI_ELEMENT (child)) { char *number; gboolean had_new_line; xmlChar *value; value = GET_PROP (child, "value"); parse_int ((char*) value, &count); number = make_li_number (count, list_type); had_new_line = data->new_line; moo_html_insert_verbatim (view, buffer, iter, current, number); data->new_line = had_new_line; process_elm_body (view, buffer, child, current, iter); moo_html_new_line (view, buffer, iter, current, FALSE); count++; g_free (number); STR_FREE (value); } else { g_message ("%s: unknown node '%s'", G_STRLOC, child->name); process_elm_body (view, buffer, child, current, iter); } } STR_FREE (start); STR_FREE (type); } static void process_ul_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *current, GtkTextIter *iter) { xmlNode *child; for (child = elm->children; child != NULL; child = child->next) process_elm_body (view, buffer, child, current, iter); } static void process_li_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *current, GtkTextIter *iter) { gboolean had_new_line; MooHtmlData *data = moo_html_get_data (view); moo_html_new_line (view, buffer, iter, current, FALSE); had_new_line = data->new_line; moo_html_insert_verbatim (view, buffer, iter, current, " * "); data->new_line = had_new_line; process_elm_body (view, buffer, elm, current, iter); moo_html_new_line (view, buffer, iter, current, FALSE); } static void process_pre_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter) { static MooHtmlAttr attr; MooHtmlTag *current; attr.mask = MOO_HTML_MONOSPACE | MOO_HTML_PRE; current = moo_html_create_tag (view, &attr, parent, FALSE); process_elm_body (view, buffer, elm, current, iter); } static void process_font_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter) { static MooHtmlAttr attr; MooHtmlTag *current; xmlChar *size__, *color, *face; int scale = 0; guint size_val; const xmlChar *size; size__ = GET_PROP (elm, "size"); color = GET_PROP (elm, "color"); face = GET_PROP (elm, "face"); attr.mask = 0; attr.font_face = NULL; if (size__) { size = size__; if (size[0] == '+') { scale = 1; size++; } else if (size[0] == '-') { scale = -1; size++; } if (!size[0] || size[0] < '1' || size[0] > '7') { g_warning ("%s: invalid size '%s'", G_STRLOC, size); } else { size_val = size[0] - '1' + 1; if (scale == 1) { attr.mask |= MOO_HTML_LARGER; attr.scale = size_val; } else if (scale == -1) { attr.mask |= MOO_HTML_SMALLER; attr.scale = size_val; } else { attr.mask |= MOO_HTML_FONT_SIZE; attr.font_size = size_val; } } } if (color) { attr.mask |= MOO_HTML_FG; attr.fg = (char*) color; } if (face && face[0]) { attr.mask |= MOO_HTML_FONT_FACE; attr.font_face = (char*) face; } current = moo_html_create_tag (view, &attr, parent, FALSE); process_elm_body (view, buffer, elm, current, iter); STR_FREE (size__); STR_FREE (color); STR_FREE (face); } static void process_cite_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter) { static MooHtmlAttr attr; MooHtmlTag *current; current = moo_html_create_tag (view, &attr, parent, FALSE); process_elm_body (view, buffer, elm, current, iter); } static void process_format_elm (GtkTextView *view, MooHtmlAttr *attr, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter) { MooHtmlTag *current; current = moo_html_create_tag (view, attr, parent, FALSE); process_elm_body (view, buffer, elm, current, iter); } static void process_dt_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter) { moo_html_new_line (view, buffer, iter, parent, FALSE); process_elm_body (view, buffer, elm, parent, iter); moo_html_new_line (view, buffer, iter, parent, FALSE); } static void process_dl_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter) { moo_html_new_line (view, buffer, iter, parent, FALSE); process_elm_body (view, buffer, elm, parent, iter); moo_html_new_line (view, buffer, iter, parent, FALSE); } static void process_dd_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter) { static MooHtmlAttr attr; MooHtmlTag *current; attr.mask = MOO_HTML_LEFT_MARGIN; attr.left_margin = 20; current = moo_html_create_tag (view, &attr, parent, FALSE); moo_html_new_line (view, buffer, iter, current, FALSE); process_elm_body (view, buffer, elm, current, iter); moo_html_new_line (view, buffer, iter, current, FALSE); } static void process_br_elm (GtkTextView *view, GtkTextBuffer *buffer, G_GNUC_UNUSED xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter) { moo_html_new_line (view, buffer, iter, parent, TRUE); } static void process_div_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter) { static MooHtmlAttr attr; MooHtmlTag *current; current = moo_html_create_tag (view, &attr, parent, FALSE); process_elm_body (view, buffer, elm, current, iter); } static void process_span_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter) { static MooHtmlAttr attr; MooHtmlTag *current; current = moo_html_create_tag (view, &attr, parent, FALSE); process_elm_body (view, buffer, elm, current, iter); } static void process_hr_elm (GtkTextView *view, GtkTextBuffer *buffer, G_GNUC_UNUSED xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter) { GtkTextChildAnchor *anchor; GtkWidget *line; MooHtmlData *data = moo_html_get_data (view); line = gtk_hseparator_new (); gtk_widget_show (line); moo_html_new_line (view, buffer, iter, parent, FALSE); anchor = gtk_text_buffer_create_child_anchor (buffer, iter); gtk_text_view_add_child_at_anchor (GTK_TEXT_VIEW (view), line, anchor); data->rulers = g_slist_prepend (data->rulers, line); moo_html_new_line (view, buffer, iter, parent, TRUE); } static void process_img_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *current, GtkTextIter *iter) { xmlChar *src; xmlChar *alt; char *path = NULL; GdkPixbuf *pixbuf; GError *error = NULL; int offset; GtkTextIter before; MooHtmlData *data = moo_html_get_data (view); src = GET_PROP (elm, "src"); alt = GET_PROP (elm, "alt"); g_return_if_fail (src != NULL); if (!data->dirname) goto try_alt; path = g_build_filename (data->dirname, src, NULL); g_return_if_fail (path != NULL); pixbuf = gdk_pixbuf_new_from_file (path, &error); if (!pixbuf) { g_message ("%s: could not load image '%s'", G_STRLOC, path); g_message ("%s: %s", G_STRLOC, error->message); g_error_free (error); goto try_alt; } offset = gtk_text_iter_get_offset (iter); gtk_text_buffer_insert_pixbuf (buffer, iter, pixbuf); gtk_text_buffer_get_iter_at_offset (buffer, &before, offset); if (current) gtk_text_buffer_apply_tag (buffer, GTK_TEXT_TAG (current), &before, iter); g_object_unref (pixbuf); goto out; try_alt: if (alt) { char *text = g_strdup_printf ("[%s]", alt); moo_html_insert_text (view, buffer, iter, current, text); g_free (text); } out: STR_FREE (src); STR_FREE (alt); g_free (path); } static void process_table_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter) { process_elm_body (view, buffer, elm, parent, iter); } static void process_tr_elm (GtkTextView *view, GtkTextBuffer *buffer, xmlNode *elm, MooHtmlTag *parent, GtkTextIter *iter) { moo_html_new_line (view, buffer, iter, parent, FALSE); process_elm_body (view, buffer, elm, parent, iter); moo_html_new_line (view, buffer, iter, parent, FALSE); }