/* * Copyright (C) 2007 Carlos Garcia Campos * * 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 #include #include #include #include "utils.h" void pgd_table_add_property_with_custom_widget(GtkGrid *table, const gchar *markup, GtkWidget *widget, gint *row) { GtkWidget *label; label = gtk_label_new(NULL); g_object_set(G_OBJECT(label), "xalign", 0.0, "yalign", 0.0, NULL); gtk_label_set_markup(GTK_LABEL(label), markup); gtk_grid_attach(GTK_GRID(table), label, 0, *row, 1, 1); gtk_widget_show(label); gtk_grid_attach(GTK_GRID(table), widget, 1, *row, 1, 1); gtk_widget_set_hexpand(widget, TRUE); gtk_widget_show(widget); *row += 1; } void pgd_table_add_property_with_value_widget(GtkGrid *table, const gchar *markup, GtkWidget **value_widget, const gchar *value, gint *row) { GtkWidget *label; *value_widget = label = gtk_label_new(value); g_object_set(G_OBJECT(label), "xalign", 0.0, "selectable", TRUE, "ellipsize", PANGO_ELLIPSIZE_END, NULL); pgd_table_add_property_with_custom_widget(table, markup, label, row); } void pgd_table_add_property(GtkGrid *table, const gchar *markup, const gchar *value, gint *row) { GtkWidget *label; pgd_table_add_property_with_value_widget(table, markup, &label, value, row); } GtkWidget *pgd_action_view_new(PopplerDocument *document) { GtkWidget *frame, *label; frame = gtk_frame_new(NULL); gtk_frame_set_shadow_type(GTK_FRAME(frame), GTK_SHADOW_NONE); label = gtk_label_new(NULL); gtk_label_set_markup(GTK_LABEL(label), "Action Properties"); gtk_frame_set_label_widget(GTK_FRAME(frame), label); gtk_widget_show(label); g_object_set_data(G_OBJECT(frame), "document", document); return frame; } static void pgd_action_view_add_destination(GtkWidget *action_view, GtkGrid *table, PopplerDest *dest, gboolean remote, gint *row) { PopplerDocument *document; GEnumValue *enum_value; gchar *str; pgd_table_add_property(table, "Type:", "Destination", row); enum_value = g_enum_get_value((GEnumClass *)g_type_class_ref(POPPLER_TYPE_DEST_TYPE), dest->type); pgd_table_add_property(table, "Destination Type:", enum_value->value_name, row); document = g_object_get_data(G_OBJECT(action_view), "document"); if (dest->type != POPPLER_DEST_NAMED) { str = NULL; if (document && !remote) { PopplerPage *poppler_page; gchar *page_label; poppler_page = poppler_document_get_page(document, MAX(0, dest->page_num - 1)); g_object_get(G_OBJECT(poppler_page), "label", &page_label, NULL); if (page_label) { str = g_strdup_printf("%d (%s)", dest->page_num, page_label); g_free(page_label); } } if (!str) str = g_strdup_printf("%d", dest->page_num); pgd_table_add_property(table, "Page:", str, row); g_free(str); str = g_strdup_printf("%.2f", dest->left); pgd_table_add_property(table, "Left:", str, row); g_free(str); str = g_strdup_printf("%.2f", dest->right); pgd_table_add_property(table, "Right:", str, row); g_free(str); str = g_strdup_printf("%.2f", dest->top); pgd_table_add_property(table, "Top:", str, row); g_free(str); str = g_strdup_printf("%.2f", dest->bottom); pgd_table_add_property(table, "Bottom:", str, row); g_free(str); str = g_strdup_printf("%.2f", dest->zoom); pgd_table_add_property(table, "Zoom:", str, row); g_free(str); } else { if (document && !remote) { PopplerDest *new_dest; new_dest = poppler_document_find_dest(document, dest->named_dest); if (new_dest) { GtkWidget *new_table; gint new_row = 0; new_table = gtk_grid_new(); gtk_widget_set_margin_top(new_table, 5); gtk_widget_set_margin_bottom(new_table, 5); #if GTK_CHECK_VERSION(3, 12, 0) gtk_widget_set_margin_start(new_table, 12); gtk_widget_set_margin_end(new_table, 5); #else gtk_widget_set_margin_left(new_table, 12); gtk_widget_set_margin_right(new_table, 5); #endif gtk_grid_set_column_spacing(GTK_GRID(new_table), 6); gtk_grid_set_row_spacing(GTK_GRID(new_table), 6); pgd_action_view_add_destination(action_view, GTK_GRID(new_table), new_dest, FALSE, &new_row); poppler_dest_free(new_dest); gtk_grid_attach(GTK_GRID(table), new_table, 0, *row, 1, 1); gtk_widget_show(new_table); *row += 1; } } } } static const gchar *get_movie_op(PopplerActionMovieOperation op) { switch (op) { case POPPLER_ACTION_MOVIE_PLAY: return "Play"; case POPPLER_ACTION_MOVIE_PAUSE: return "Pause"; case POPPLER_ACTION_MOVIE_RESUME: return "Resume"; case POPPLER_ACTION_MOVIE_STOP: return "Stop"; } return NULL; } static void free_tmp_file(GFile *tmp_file) { g_file_delete(tmp_file, NULL, NULL); g_object_unref(tmp_file); } static gboolean save_helper(const gchar *buf, gsize count, gpointer data, GError **error) { gint fd = GPOINTER_TO_INT(data); return write(fd, buf, count) == count; } static void pgd_action_view_play_rendition(GtkWidget *button, PopplerMedia *media) { GFile *file = NULL; gchar *uri; if (poppler_media_is_embedded(media)) { gint fd; gchar *tmp_filename = NULL; fd = g_file_open_tmp(NULL, &tmp_filename, NULL); if (fd != -1) { if (poppler_media_save_to_callback(media, save_helper, GINT_TO_POINTER(fd), NULL)) { file = g_file_new_for_path(tmp_filename); g_object_set_data_full(G_OBJECT(media), "tmp-file", g_object_ref(file), (GDestroyNotify)free_tmp_file); } else { g_free(tmp_filename); } close(fd); } else if (tmp_filename) { g_free(tmp_filename); } } else { const gchar *filename; filename = poppler_media_get_filename(media); if (g_path_is_absolute(filename)) { file = g_file_new_for_path(filename); } else if (g_strrstr(filename, "://")) { file = g_file_new_for_uri(filename); } else { gchar *cwd; gchar *path; // FIXME: relative to doc uri, not cwd cwd = g_get_current_dir(); path = g_build_filename(cwd, filename, NULL); g_free(cwd); file = g_file_new_for_path(path); g_free(path); } } if (file) { uri = g_file_get_uri(file); g_object_unref(file); if (uri) { #if GTK_CHECK_VERSION(3, 22, 0) GtkWidget *toplevel; toplevel = gtk_widget_get_toplevel(button); gtk_show_uri_on_window(gtk_widget_is_toplevel(toplevel) ? GTK_WINDOW(toplevel) : NULL, uri, GDK_CURRENT_TIME, NULL); #else gtk_show_uri(gtk_widget_get_screen(button), uri, GDK_CURRENT_TIME, NULL); #endif g_free(uri); } } } static void pgd_action_view_do_action_layer(GtkWidget *button, GList *state_list) { GList *l, *m; for (l = state_list; l; l = g_list_next(l)) { PopplerActionLayer *action_layer = (PopplerActionLayer *)l->data; for (m = action_layer->layers; m; m = g_list_next(m)) { PopplerLayer *layer = (PopplerLayer *)m->data; switch (action_layer->action) { case POPPLER_ACTION_LAYER_ON: poppler_layer_show(layer); break; case POPPLER_ACTION_LAYER_OFF: poppler_layer_hide(layer); break; case POPPLER_ACTION_LAYER_TOGGLE: if (poppler_layer_is_visible(layer)) poppler_layer_hide(layer); else poppler_layer_show(layer); break; } } } } void pgd_action_view_set_action(GtkWidget *action_view, PopplerAction *action) { GtkWidget *table; gint row = 0; table = gtk_bin_get_child(GTK_BIN(action_view)); if (table) { gtk_container_remove(GTK_CONTAINER(action_view), table); } if (!action) return; table = gtk_grid_new(); gtk_widget_set_margin_top(table, 5); gtk_widget_set_margin_bottom(table, 5); #if GTK_CHECK_VERSION(3, 12, 0) gtk_widget_set_margin_start(table, 12); gtk_widget_set_margin_end(table, 5); #else gtk_widget_set_margin_left(table, 12); gtk_widget_set_margin_right(table, 5); #endif gtk_grid_set_column_spacing(GTK_GRID(table), 6); gtk_grid_set_row_spacing(GTK_GRID(table), 6); pgd_table_add_property(GTK_GRID(table), "Title:", action->any.title, &row); switch (action->type) { case POPPLER_ACTION_UNKNOWN: pgd_table_add_property(GTK_GRID(table), "Type:", "Unknown", &row); break; case POPPLER_ACTION_NONE: pgd_table_add_property(GTK_GRID(table), "Type:", "None", &row); break; case POPPLER_ACTION_GOTO_DEST: pgd_action_view_add_destination(action_view, GTK_GRID(table), action->goto_dest.dest, FALSE, &row); break; case POPPLER_ACTION_GOTO_REMOTE: pgd_table_add_property(GTK_GRID(table), "Type:", "Remote Destination", &row); pgd_table_add_property(GTK_GRID(table), "Filename:", action->goto_remote.file_name, &row); pgd_action_view_add_destination(action_view, GTK_GRID(table), action->goto_remote.dest, TRUE, &row); break; case POPPLER_ACTION_LAUNCH: pgd_table_add_property(GTK_GRID(table), "Type:", "Launch", &row); pgd_table_add_property(GTK_GRID(table), "Filename:", action->launch.file_name, &row); pgd_table_add_property(GTK_GRID(table), "Params:", action->launch.params, &row); break; case POPPLER_ACTION_URI: pgd_table_add_property(GTK_GRID(table), "Type:", "External URI", &row); pgd_table_add_property(GTK_GRID(table), "URI", action->uri.uri, &row); break; case POPPLER_ACTION_NAMED: pgd_table_add_property(GTK_GRID(table), "Type:", "Named Action", &row); pgd_table_add_property(GTK_GRID(table), "Name:", action->named.named_dest, &row); break; case POPPLER_ACTION_MOVIE: { GtkWidget *movie_view = pgd_movie_view_new(); pgd_table_add_property(GTK_GRID(table), "Type:", "Movie", &row); pgd_table_add_property(GTK_GRID(table), "Operation:", get_movie_op(action->movie.operation), &row); pgd_movie_view_set_movie(movie_view, action->movie.movie); pgd_table_add_property_with_custom_widget(GTK_GRID(table), "Movie:", movie_view, &row); } break; case POPPLER_ACTION_RENDITION: { gchar *text; pgd_table_add_property(GTK_GRID(table), "Type:", "Rendition", &row); text = g_strdup_printf("%d", action->rendition.op); pgd_table_add_property(GTK_GRID(table), "Operation:", text, &row); g_free(text); if (action->rendition.media) { gboolean embedded = poppler_media_is_embedded(action->rendition.media); GtkWidget *button; pgd_table_add_property(GTK_GRID(table), "Embedded:", embedded ? "Yes" : "No", &row); if (embedded) { const gchar *mime_type = poppler_media_get_mime_type(action->rendition.media); pgd_table_add_property(GTK_GRID(table), "Mime type:", mime_type ? mime_type : "", &row); } else { pgd_table_add_property(GTK_GRID(table), "Filename:", poppler_media_get_filename(action->rendition.media), &row); } button = gtk_button_new_with_mnemonic("_Play"); g_signal_connect(button, "clicked", G_CALLBACK(pgd_action_view_play_rendition), action->rendition.media); pgd_table_add_property_with_custom_widget(GTK_GRID(table), NULL, button, &row); gtk_widget_show(button); } } break; case POPPLER_ACTION_OCG_STATE: { GList *l; GtkWidget *button; pgd_table_add_property(GTK_GRID(table), "Type:", "OCGState", &row); for (l = action->ocg_state.state_list; l; l = g_list_next(l)) { PopplerActionLayer *action_layer = (PopplerActionLayer *)l->data; gchar *text = NULL; gint n_layers = g_list_length(action_layer->layers); switch (action_layer->action) { case POPPLER_ACTION_LAYER_ON: text = g_strdup_printf("%d layers On", n_layers); break; case POPPLER_ACTION_LAYER_OFF: text = g_strdup_printf("%d layers Off", n_layers); break; case POPPLER_ACTION_LAYER_TOGGLE: text = g_strdup_printf("%d layers Toggle", n_layers); break; } pgd_table_add_property(GTK_GRID(table), "Action:", text, &row); g_free(text); } button = gtk_button_new_with_label("Do action"); g_signal_connect(button, "clicked", G_CALLBACK(pgd_action_view_do_action_layer), action->ocg_state.state_list); pgd_table_add_property_with_custom_widget(GTK_GRID(table), NULL, button, &row); gtk_widget_show(button); } break; case POPPLER_ACTION_JAVASCRIPT: { GtkTextBuffer *buffer; GtkWidget *textview; GtkWidget *swindow; pgd_table_add_property(GTK_GRID(table), "Type:", "JavaScript", &row); buffer = gtk_text_buffer_new(NULL); if (action->javascript.script) gtk_text_buffer_set_text(buffer, action->javascript.script, -1); textview = gtk_text_view_new_with_buffer(buffer); gtk_text_view_set_editable(GTK_TEXT_VIEW(textview), FALSE); g_object_unref(buffer); swindow = gtk_scrolled_window_new(NULL, NULL); gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(swindow), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); gtk_container_add(GTK_CONTAINER(swindow), textview); gtk_widget_show(textview); pgd_table_add_property_with_custom_widget(GTK_GRID(table), NULL, swindow, &row); gtk_widget_show(swindow); } break; case POPPLER_ACTION_RESET_FORM: pgd_table_add_property(GTK_GRID(table), "Type:", "ResetForm", &row); break; default: g_assert_not_reached(); } gtk_container_add(GTK_CONTAINER(action_view), table); gtk_widget_show(table); } gchar *pgd_format_date(time_t utime) { GDateTime *dt = NULL; gchar *s = NULL; if (utime == 0) return NULL; dt = g_date_time_new_from_unix_local(utime); if (dt == NULL) return NULL; s = g_date_time_format(dt, "%c"); g_date_time_unref(dt); return s; } GtkWidget *pgd_movie_view_new(void) { GtkWidget *frame, *label; frame = gtk_frame_new(NULL); gtk_frame_set_shadow_type(GTK_FRAME(frame), GTK_SHADOW_NONE); label = gtk_label_new(NULL); gtk_label_set_markup(GTK_LABEL(label), "Movie Properties"); gtk_frame_set_label_widget(GTK_FRAME(frame), label); gtk_widget_show(label); return frame; } static void pgd_movie_view_play_movie(GtkWidget *button, PopplerMovie *movie) { const gchar *filename; GFile *file; gchar *uri; filename = poppler_movie_get_filename(movie); if (g_path_is_absolute(filename)) { file = g_file_new_for_path(filename); } else if (g_strrstr(filename, "://")) { file = g_file_new_for_uri(filename); } else { gchar *cwd; gchar *path; // FIXME: relative to doc uri, not cwd cwd = g_get_current_dir(); path = g_build_filename(cwd, filename, NULL); g_free(cwd); file = g_file_new_for_path(path); g_free(path); } uri = g_file_get_uri(file); g_object_unref(file); if (uri) { #if GTK_CHECK_VERSION(3, 22, 0) GtkWidget *toplevel; toplevel = gtk_widget_get_toplevel(button); gtk_show_uri_on_window(gtk_widget_is_toplevel(toplevel) ? GTK_WINDOW(toplevel) : NULL, uri, GDK_CURRENT_TIME, NULL); #else gtk_show_uri(gtk_widget_get_screen(button), uri, GDK_CURRENT_TIME, NULL); #endif g_free(uri); } } void pgd_movie_view_set_movie(GtkWidget *movie_view, PopplerMovie *movie) { GtkWidget *table; GtkWidget *button; GEnumValue *enum_value; gint width, height; gint row = 0; table = gtk_bin_get_child(GTK_BIN(movie_view)); if (table) { gtk_container_remove(GTK_CONTAINER(movie_view), table); } if (!movie) return; table = gtk_grid_new(); gtk_widget_set_margin_top(table, 5); gtk_widget_set_margin_bottom(table, 5); #if GTK_CHECK_VERSION(3, 12, 0) gtk_widget_set_margin_start(table, 12); gtk_widget_set_margin_end(table, 5); #else gtk_widget_set_margin_left(table, 12); gtk_widget_set_margin_right(table, 5); #endif gtk_grid_set_column_spacing(GTK_GRID(table), 6); gtk_grid_set_row_spacing(GTK_GRID(table), 6); pgd_table_add_property(GTK_GRID(table), "Filename:", poppler_movie_get_filename(movie), &row); pgd_table_add_property(GTK_GRID(table), "Need Poster:", poppler_movie_need_poster(movie) ? "Yes" : "No", &row); pgd_table_add_property(GTK_GRID(table), "Show Controls:", poppler_movie_show_controls(movie) ? "Yes" : "No", &row); enum_value = g_enum_get_value((GEnumClass *)g_type_class_ref(POPPLER_TYPE_MOVIE_PLAY_MODE), poppler_movie_get_play_mode(movie)); pgd_table_add_property(GTK_GRID(table), "Play Mode:", enum_value->value_name, &row); pgd_table_add_property(GTK_GRID(table), "Synchronous Play:", poppler_movie_is_synchronous(movie) ? "Yes" : "No", &row); pgd_table_add_property(GTK_GRID(table), "Volume:", g_strdup_printf("%g", poppler_movie_get_volume(movie)), &row); pgd_table_add_property(GTK_GRID(table), "Rate:", g_strdup_printf("%g", poppler_movie_get_rate(movie)), &row); pgd_table_add_property(GTK_GRID(table), "Start:", g_strdup_printf("%g s", poppler_movie_get_start(movie) / 1e9), &row); pgd_table_add_property(GTK_GRID(table), "Duration:", g_strdup_printf("%g s", poppler_movie_get_duration(movie) / 1e9), &row); pgd_table_add_property(GTK_GRID(table), "Rotation Angle:", g_strdup_printf("%u", poppler_movie_get_rotation_angle(movie)), &row); poppler_movie_get_aspect(movie, &width, &height); pgd_table_add_property(GTK_GRID(table), "Aspect:", g_strdup_printf("%dx%d", width, height), &row); button = gtk_button_new_with_mnemonic("_Play"); g_signal_connect(button, "clicked", G_CALLBACK(pgd_movie_view_play_movie), movie); pgd_table_add_property_with_custom_widget(GTK_GRID(table), NULL, button, &row); gtk_widget_show(button); gtk_container_add(GTK_CONTAINER(movie_view), table); gtk_widget_show(table); } GdkPixbuf *pgd_pixbuf_new_for_color(PopplerColor *poppler_color) { GdkPixbuf *pixbuf; gint num, x; guchar *pixels; if (!poppler_color) return NULL; pixbuf = gdk_pixbuf_new(GDK_COLORSPACE_RGB, FALSE, 8, 64, 16); pixels = gdk_pixbuf_get_pixels(pixbuf); num = gdk_pixbuf_get_width(pixbuf) * gdk_pixbuf_get_height(pixbuf); for (x = 0; x < num; x++) { pixels[0] = poppler_color->red; pixels[1] = poppler_color->green; pixels[2] = poppler_color->blue; pixels += 3; } return pixbuf; }