2418 lines
68 KiB
C
2418 lines
68 KiB
C
/*
|
|
* moohtml.c
|
|
*
|
|
* Copyright (C) 2004-2010 by Yevgen Muntyan <emuntyan@users.sourceforge.net>
|
|
*
|
|
* 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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include "moohtml.h"
|
|
#include "marshals.h"
|
|
#include "mooutils/mooutils-misc.h"
|
|
#include "mooutils/moocompat.h"
|
|
#include <gdk/gdkkeysyms.h>
|
|
#include <gtk/gtk.h>
|
|
#include <glib.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
|
|
#include <libxml/HTMLtree.h>
|
|
|
|
#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 ("invalid url '%s'", 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 ("htmlReadMemory failed");
|
|
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 ("could not find anchor '%s'",
|
|
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 ("unknown node '%s'", 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 ("unknown node '%s'", 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 ("unknown node '%s'", child->name);
|
|
process_elm_body (view, buffer, child, current, iter);
|
|
}
|
|
else if (IS_COMMENT (child))
|
|
{
|
|
/* ignore */
|
|
}
|
|
else
|
|
{
|
|
g_warning ("unknown node");
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
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 ("implement me");
|
|
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 ("invalid type attribute '%s'",
|
|
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 ("unknown node '%s'", 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 ("invalid size '%s'", 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 ("could not load image '%s': %s", path, moo_error_message (error));
|
|
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);
|
|
}
|