blob: 24d63b77aab6210715a35fa9dc92b9e85b4eb1eb [file] [log] [blame]
/*
* Copyright (C) 2008 Carlos Garcia Campos <carlosgc@gnome.org>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2, or (at your option)
* any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA.
*/
#include "config.h"
#include <gtk/gtk.h>
#include "layers.h"
enum
{
LAYERS_TITLE_COLUMN,
LAYERS_VISIBILITY_COLUMN,
LAYERS_ENABLE_COLUMN,
LAYERS_SHOWTOGGLE_COLUMN,
LAYERS_RB_GROUP_COLUMN,
LAYERS_LAYER_COLUMN,
N_COLUMNS
};
typedef struct
{
PopplerDocument *doc;
guint page;
GtkWidget *treeview;
GtkWidget *darea;
cairo_surface_t *surface;
} PgdLayersDemo;
static void pgd_layers_free(PgdLayersDemo *demo)
{
if (!demo) {
return;
}
if (demo->doc) {
g_object_unref(demo->doc);
demo->doc = NULL;
}
if (demo->surface) {
cairo_surface_destroy(demo->surface);
demo->surface = NULL;
}
g_free(demo);
}
static void build_tree(PopplerDocument *document, GtkTreeModel *model, GtkTreeIter *parent, PopplerLayersIter *iter)
{
do {
GtkTreeIter tree_iter;
PopplerLayersIter *child;
PopplerLayer *layer;
gboolean visible;
gchar *markup;
gint rb_group = 0;
layer = poppler_layers_iter_get_layer(iter);
if (layer) {
markup = g_markup_escape_text(poppler_layer_get_title(layer), -1);
visible = poppler_layer_is_visible(layer);
rb_group = poppler_layer_get_radio_button_group_id(layer);
} else {
gchar *title;
title = poppler_layers_iter_get_title(iter);
markup = g_markup_escape_text(title, -1);
g_free(title);
visible = FALSE;
layer = NULL;
}
gtk_tree_store_append(GTK_TREE_STORE(model), &tree_iter, parent);
gtk_tree_store_set(GTK_TREE_STORE(model), &tree_iter, LAYERS_TITLE_COLUMN, markup, LAYERS_VISIBILITY_COLUMN, visible, LAYERS_ENABLE_COLUMN, TRUE, /* FIXME */
LAYERS_SHOWTOGGLE_COLUMN, (layer != NULL), LAYERS_RB_GROUP_COLUMN, rb_group, LAYERS_LAYER_COLUMN, layer, -1);
if (layer) {
g_object_unref(layer);
}
g_free(markup);
child = poppler_layers_iter_get_child(iter);
if (child) {
build_tree(document, model, &tree_iter, child);
}
poppler_layers_iter_free(child);
} while (poppler_layers_iter_next(iter));
}
GtkTreeModel *pgd_layers_create_model(PopplerDocument *document)
{
GtkTreeModel *model;
PopplerLayersIter *iter;
iter = poppler_layers_iter_new(document);
if (iter) {
model = GTK_TREE_MODEL(gtk_tree_store_new(N_COLUMNS, G_TYPE_STRING, G_TYPE_BOOLEAN, G_TYPE_BOOLEAN, G_TYPE_BOOLEAN, G_TYPE_INT, G_TYPE_OBJECT));
build_tree(document, model, NULL, iter);
poppler_layers_iter_free(iter);
} else {
GtkTreeIter tree_iter;
gchar *markup;
model = GTK_TREE_MODEL(gtk_list_store_new(1, G_TYPE_STRING));
gtk_list_store_append(GTK_LIST_STORE(model), &tree_iter);
markup = g_strdup_printf("<span size=\"larger\" style=\"italic\">%s</span>", "The document doesn't contain layers");
gtk_list_store_set(GTK_LIST_STORE(model), &tree_iter, 0, markup, -1);
g_free(markup);
}
return model;
}
static cairo_surface_t *pgd_layers_render_page(PgdLayersDemo *demo)
{
cairo_t *cr;
PopplerPage *page;
gdouble width, height;
cairo_surface_t *surface = NULL;
page = poppler_document_get_page(demo->doc, demo->page);
if (!page) {
return NULL;
}
poppler_page_get_size(page, &width, &height);
gtk_widget_set_size_request(demo->darea, width, height);
surface = cairo_image_surface_create(CAIRO_FORMAT_RGB24, width, height);
cr = cairo_create(surface);
cairo_save(cr);
cairo_set_source_rgb(cr, 1, 1, 1);
cairo_rectangle(cr, 0, 0, width, height);
cairo_fill(cr);
cairo_restore(cr);
cairo_save(cr);
poppler_page_render(page, cr);
cairo_restore(cr);
cairo_destroy(cr);
g_object_unref(page);
return surface;
}
static gboolean pgd_layers_viewer_drawing_area_draw(GtkWidget *area, cairo_t *cr, PgdLayersDemo *demo)
{
if (!demo->surface) {
demo->surface = pgd_layers_render_page(demo);
if (!demo->surface) {
return FALSE;
}
}
cairo_set_source_surface(cr, demo->surface, 0, 0);
cairo_paint(cr);
return TRUE;
}
static gboolean pgd_layers_viewer_redraw(PgdLayersDemo *demo)
{
cairo_surface_destroy(demo->surface);
demo->surface = NULL;
gtk_widget_queue_draw(demo->darea);
return FALSE;
}
static void pgd_layers_viewer_queue_redraw(PgdLayersDemo *demo)
{
g_idle_add((GSourceFunc)pgd_layers_viewer_redraw, demo);
}
static void pgd_layers_page_selector_value_changed(GtkSpinButton *spinbutton, PgdLayersDemo *demo)
{
demo->page = (gint)gtk_spin_button_get_value(spinbutton) - 1;
pgd_layers_viewer_queue_redraw(demo);
}
static GtkWidget *pgd_layers_create_viewer(PgdLayersDemo *demo)
{
GtkWidget *vbox, *hbox;
GtkWidget *label;
GtkWidget *swindow;
GtkWidget *page_selector;
guint n_pages;
gchar *str;
vbox = gtk_box_new(GTK_ORIENTATION_VERTICAL, 6);
hbox = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 6);
label = gtk_label_new("Page:");
gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, TRUE, 0);
gtk_widget_show(label);
n_pages = poppler_document_get_n_pages(demo->doc);
page_selector = gtk_spin_button_new_with_range(1, n_pages, 1);
g_signal_connect(G_OBJECT(page_selector), "value-changed", G_CALLBACK(pgd_layers_page_selector_value_changed), (gpointer)demo);
gtk_box_pack_start(GTK_BOX(hbox), page_selector, FALSE, TRUE, 0);
gtk_widget_show(page_selector);
str = g_strdup_printf("of %d", n_pages);
label = gtk_label_new(str);
gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, TRUE, 0);
gtk_widget_show(label);
g_free(str);
gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, TRUE, 0);
gtk_widget_show(hbox);
demo->darea = gtk_drawing_area_new();
g_signal_connect(G_OBJECT(demo->darea), "draw", G_CALLBACK(pgd_layers_viewer_drawing_area_draw), (gpointer)demo);
swindow = gtk_scrolled_window_new(NULL, NULL);
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(swindow), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
#if GTK_CHECK_VERSION(3, 7, 8)
gtk_container_add(GTK_CONTAINER(swindow), demo->darea);
#else
gtk_scrolled_window_add_with_viewport(GTK_SCROLLED_WINDOW(swindow), demo->darea);
#endif
gtk_widget_show(demo->darea);
gtk_box_pack_start(GTK_BOX(vbox), swindow, TRUE, TRUE, 0);
gtk_widget_show(swindow);
return vbox;
}
static gboolean update_kids(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, GtkTreeIter *parent)
{
if (gtk_tree_store_is_ancestor(GTK_TREE_STORE(model), parent, iter)) {
gboolean visible;
gtk_tree_model_get(model, parent, LAYERS_VISIBILITY_COLUMN, &visible, -1);
gtk_tree_store_set(GTK_TREE_STORE(model), iter, LAYERS_ENABLE_COLUMN, visible, -1);
}
return FALSE;
}
static gboolean clear_rb_group(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gint *rb_group)
{
gint group;
gtk_tree_model_get(model, iter, LAYERS_RB_GROUP_COLUMN, &group, -1);
if (group == *rb_group) {
gtk_tree_store_set(GTK_TREE_STORE(model), iter, LAYERS_VISIBILITY_COLUMN, FALSE, -1);
}
return FALSE;
}
static void pgd_layers_visibility_changed(GtkCellRendererToggle *cell, gchar *path_str, PgdLayersDemo *demo)
{
GtkTreeModel *model;
GtkTreePath *path;
GtkTreeIter iter;
gboolean visible;
PopplerLayer *layer;
model = gtk_tree_view_get_model(GTK_TREE_VIEW(demo->treeview));
path = gtk_tree_path_new_from_string(path_str);
gtk_tree_model_get_iter(model, &iter, path);
gtk_tree_model_get(model, &iter, LAYERS_VISIBILITY_COLUMN, &visible, LAYERS_LAYER_COLUMN, &layer, -1);
visible = !visible;
visible ? poppler_layer_show(layer) : poppler_layer_hide(layer);
if (visible) {
gint rb_group;
rb_group = poppler_layer_get_radio_button_group_id(layer);
if (rb_group) {
gtk_tree_model_foreach(model, (GtkTreeModelForeachFunc)clear_rb_group, &rb_group);
}
}
gtk_tree_store_set(GTK_TREE_STORE(model), &iter, LAYERS_VISIBILITY_COLUMN, visible, -1);
if (poppler_layer_is_parent(layer)) {
gtk_tree_model_foreach(model, (GtkTreeModelForeachFunc)update_kids, &iter);
}
pgd_layers_viewer_queue_redraw(demo);
gtk_tree_path_free(path);
g_object_unref(layer);
}
GtkWidget *pgd_layers_create_widget(PopplerDocument *document)
{
PgdLayersDemo *demo;
GtkWidget *swindow;
GtkWidget *treeview;
GtkTreeModel *model;
GtkCellRenderer *renderer;
GtkWidget *hpaned, *viewer;
demo = g_new0(PgdLayersDemo, 1);
demo->doc = g_object_ref(document);
hpaned = gtk_paned_new(GTK_ORIENTATION_HORIZONTAL);
viewer = pgd_layers_create_viewer(demo);
swindow = gtk_scrolled_window_new(NULL, NULL);
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(swindow), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
model = pgd_layers_create_model(document);
treeview = gtk_tree_view_new_with_model(model);
demo->treeview = treeview;
g_object_unref(model);
renderer = gtk_cell_renderer_text_new();
gtk_tree_view_insert_column_with_attributes(GTK_TREE_VIEW(treeview), 0, "Layer", renderer, "markup", LAYERS_TITLE_COLUMN, NULL);
g_object_set(G_OBJECT(renderer), "ellipsize", PANGO_ELLIPSIZE_END, NULL);
g_object_set(G_OBJECT(gtk_tree_view_get_column(GTK_TREE_VIEW(treeview), 0)), "expand", TRUE, NULL);
if (GTK_IS_TREE_STORE(model)) {
renderer = gtk_cell_renderer_toggle_new();
gtk_tree_view_insert_column_with_attributes(GTK_TREE_VIEW(treeview), 1, "Show/Hide", renderer, "active", LAYERS_VISIBILITY_COLUMN, "activatable", LAYERS_ENABLE_COLUMN, "visible", LAYERS_SHOWTOGGLE_COLUMN, NULL);
g_signal_connect(renderer, "toggled", G_CALLBACK(pgd_layers_visibility_changed), (gpointer)demo);
gtk_tree_view_column_set_clickable(gtk_tree_view_get_column(GTK_TREE_VIEW(treeview), 1), TRUE);
}
gtk_tree_selection_set_mode(gtk_tree_view_get_selection(GTK_TREE_VIEW(treeview)), GTK_SELECTION_NONE);
gtk_container_add(GTK_CONTAINER(swindow), treeview);
gtk_widget_show(treeview);
gtk_paned_add1(GTK_PANED(hpaned), swindow);
gtk_widget_show(swindow);
gtk_paned_add2(GTK_PANED(hpaned), viewer);
gtk_widget_show(viewer);
gtk_paned_set_position(GTK_PANED(hpaned), 150);
g_object_weak_ref(G_OBJECT(hpaned), (GWeakNotify)pgd_layers_free, (gpointer)demo);
return hpaned;
}