PIKApp/plug-ins/common/file-pdf-load.c

1825 lines
60 KiB
C

/* PIKA - Photo and Image Kooker Application
* a rebranding of The GNU Image Manipulation Program (created with heckimp)
* A derived work which may be trivial. However, any changes may be (C)2023 by Aldercone Studio
*
* Original copyright, applying to most contents (license remains unchanged):
*
* file-pdf-load.c - PDF file loader
*
* Copyright (C) 2005 Nathan Summers
*
* Some code in render_page_to_surface() borrowed from
* poppler.git/glib/poppler-page.cc.
*
* 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 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
#include "config.h"
#include <string.h>
#include <libpika/pika.h>
#include <libpika/pikaui.h>
#undef GTK_DISABLE_SINGLE_INCLUDES
#include <poppler.h>
#define GTK_DISABLE_SINGLE_INCLUDES
#include "libpika/stdplugins-intl.h"
/**
** the following was formerly part of
** pikaresolutionentry.h and pikaresolutionentry.c,
** moved here because this is the only thing that uses
** it, and it is undesirable to maintain all that api.
** Most unused functions have been removed.
**/
#define PIKA_TYPE_RESOLUTION_ENTRY (pika_resolution_entry_get_type ())
#define PIKA_RESOLUTION_ENTRY(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), PIKA_TYPE_RESOLUTION_ENTRY, PikaResolutionEntry))
#define PIKA_RESOLUTION_ENTRY_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), PIKA_TYPE_RESOLUTION_ENTRY, PikaResolutionEntryClass))
#define PIKA_IS_RESOLUTION_ENTRY(obj) (G_TYPE_CHECK_INSTANCE_TYPE (obj, PIKA_TYPE_RESOLUTION_ENTRY))
#define PIKA_IS_RESOLUTION_ENTRY_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), PIKA_TYPE_RESOLUTION_ENTRY))
#define PIKA_RESOLUTION_ENTRY_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), PIKA_TYPE_RESOLUTION_ENTRY, PikaResolutionEntryClass))
typedef struct _PikaResolutionEntry PikaResolutionEntry;
typedef struct _PikaResolutionEntryClass PikaResolutionEntryClass;
typedef struct _PikaResolutionEntryField PikaResolutionEntryField;
struct _PikaResolutionEntryField
{
PikaResolutionEntry *gre;
PikaResolutionEntryField *corresponding;
gboolean size;
GtkWidget *label;
guint changed_signal;
GtkAdjustment *adjustment;
GtkWidget *spinbutton;
gdouble phy_size;
gdouble value;
gdouble min_value;
gdouble max_value;
gint stop_recursion;
};
struct _PikaResolutionEntry
{
GtkGrid parent_instance;
PikaUnit size_unit;
PikaUnit unit;
GtkWidget *unitmenu;
GtkWidget *chainbutton;
PikaResolutionEntryField width;
PikaResolutionEntryField height;
PikaResolutionEntryField x;
PikaResolutionEntryField y;
};
struct _PikaResolutionEntryClass
{
GtkGridClass parent_class;
void (* value_changed) (PikaResolutionEntry *gse);
void (* refval_changed) (PikaResolutionEntry *gse);
void (* unit_changed) (PikaResolutionEntry *gse);
};
GType pika_resolution_entry_get_type (void) G_GNUC_CONST;
GtkWidget * pika_resolution_entry_new (const gchar *width_label,
gdouble width,
const gchar *height_label,
gdouble height,
PikaUnit size_unit,
const gchar *res_label,
gdouble initial_res,
PikaUnit initial_unit);
GtkWidget * pika_resolution_entry_attach_label (PikaResolutionEntry *gre,
const gchar *text,
gint row,
gint column,
gfloat alignment);
gdouble pika_resolution_entry_get_x_in_dpi (PikaResolutionEntry *gre);
gdouble pika_resolution_entry_get_y_in_dpi (PikaResolutionEntry *gre);
/* signal callback convenience functions */
void pika_resolution_entry_update_x_in_dpi (PikaResolutionEntry *gre,
gpointer data);
void pika_resolution_entry_update_y_in_dpi (PikaResolutionEntry *gre,
gpointer data);
enum
{
WIDTH_CHANGED,
HEIGHT_CHANGED,
X_CHANGED,
Y_CHANGED,
UNIT_CHANGED,
LAST_SIGNAL
};
static void pika_resolution_entry_class_init (PikaResolutionEntryClass *class);
static void pika_resolution_entry_init (PikaResolutionEntry *gre);
static void pika_resolution_entry_update_value (PikaResolutionEntryField *gref,
gdouble value);
static void pika_resolution_entry_value_callback (GtkAdjustment *adjustment,
gpointer data);
static void pika_resolution_entry_update_unit (PikaResolutionEntry *gre,
PikaUnit unit);
static void pika_resolution_entry_unit_callback (GtkWidget *widget,
PikaResolutionEntry *gre);
static void pika_resolution_entry_field_init (PikaResolutionEntry *gre,
PikaResolutionEntryField *gref,
PikaResolutionEntryField *corresponding,
guint changed_signal,
gdouble initial_val,
PikaUnit initial_unit,
gboolean size,
gint spinbutton_width);
static void pika_resolution_entry_format_label (PikaResolutionEntry *gre,
GtkWidget *label,
gdouble size);
/**
** end of pikaresolutionentry stuff
** the actual code can be found at the end of this file
**/
#define LOAD_PROC "file-pdf-load"
#define LOAD_THUMB_PROC "file-pdf-load-thumb"
#define PLUG_IN_BINARY "file-pdf-load"
#define PLUG_IN_ROLE "pika-file-pdf-load"
#define THUMBNAIL_SIZE 128
#define PIKA_PLUGIN_PDF_LOAD_ERROR pika_plugin_pdf_load_error_quark ()
static GQuark
pika_plugin_pdf_load_error_quark (void)
{
return g_quark_from_static_string ("pika-plugin-pdf-load-error-quark");
}
typedef struct
{
gint n_pages;
gint *pages;
} PdfSelectedPages;
typedef struct _Pdf Pdf;
typedef struct _PdfClass PdfClass;
struct _Pdf
{
PikaPlugIn parent_instance;
};
struct _PdfClass
{
PikaPlugInClass parent_class;
};
#define PDF_TYPE (pdf_get_type ())
#define PDF(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), PDF_TYPE, Pdf))
GType pdf_get_type (void) G_GNUC_CONST;
static GList * pdf_query_procedures (PikaPlugIn *plug_in);
static PikaProcedure * pdf_create_procedure (PikaPlugIn *plug_in,
const gchar *name);
static PikaValueArray * pdf_load (PikaProcedure *procedure,
PikaRunMode run_mode,
GFile *file,
PikaMetadata *metadata,
PikaMetadataLoadFlags *flags,
PikaProcedureConfig *config,
gpointer run_data);
static PikaValueArray * pdf_load_thumb (PikaProcedure *procedure,
GFile *file,
gint size,
PikaProcedureConfig *config,
gpointer run_data);
static PikaImage * load_image (PopplerDocument *doc,
GFile *file,
PikaRunMode run_mode,
PikaPageSelectorTarget target,
gdouble resolution,
gboolean antialias,
gboolean white_background,
gboolean reverse_order,
PdfSelectedPages *pages);
static PikaPDBStatusType load_dialog (PopplerDocument *doc,
PdfSelectedPages *pages,
PikaProcedure *procedure,
PikaProcedureConfig *config);
static PopplerDocument * open_document (GFile *file,
const gchar *PDF_password,
PikaRunMode run_mode,
GError **error);
static cairo_surface_t * get_thumb_surface (PopplerDocument *doc,
gint page,
gint preferred_size,
gboolean white_background);
static GdkPixbuf * get_thumb_pixbuf (PopplerDocument *doc,
gint page,
gint preferred_size,
gboolean white_background);
static PikaLayer * layer_from_surface (PikaImage *image,
const gchar *layer_name,
gint position,
cairo_surface_t *surface,
gdouble progress_start,
gdouble progress_scale);
G_DEFINE_TYPE (Pdf, pdf, PIKA_TYPE_PLUG_IN)
PIKA_MAIN (PDF_TYPE)
DEFINE_STD_SET_I18N
static void
pdf_class_init (PdfClass *klass)
{
PikaPlugInClass *plug_in_class = PIKA_PLUG_IN_CLASS (klass);
plug_in_class->query_procedures = pdf_query_procedures;
plug_in_class->create_procedure = pdf_create_procedure;
plug_in_class->set_i18n = STD_SET_I18N;
}
static void
pdf_init (Pdf *pdf)
{
}
static GList *
pdf_query_procedures (PikaPlugIn *plug_in)
{
GList *list = NULL;
list = g_list_append (list, g_strdup (LOAD_THUMB_PROC));
list = g_list_append (list, g_strdup (LOAD_PROC));
return list;
}
static PikaProcedure *
pdf_create_procedure (PikaPlugIn *plug_in,
const gchar *name)
{
PikaProcedure *procedure = NULL;
if (! strcmp (name, LOAD_PROC))
{
procedure = pika_load_procedure_new (plug_in, name,
PIKA_PDB_PROC_TYPE_PLUGIN,
pdf_load, NULL, NULL);
pika_procedure_set_menu_label (procedure, _("Portable Document Format"));
pika_procedure_set_documentation (procedure,
"Load file in PDF format",
"Loads files in Adobe's Portable "
"Document Format. PDF is designed to "
"be easily processed by a variety "
"of different platforms, and is a "
"distant cousin of PostScript.",
name);
pika_procedure_set_attribution (procedure,
"Nathan Summers, Lionel N.",
"Nathan Summers, Lionel N.",
"2005, 2017");
pika_file_procedure_set_mime_types (PIKA_FILE_PROCEDURE (procedure),
"application/pdf");
pika_file_procedure_set_extensions (PIKA_FILE_PROCEDURE (procedure),
"pdf");
pika_file_procedure_set_magics (PIKA_FILE_PROCEDURE (procedure),
"0, string,%PDF-");
pika_load_procedure_set_thumbnail_loader (PIKA_LOAD_PROCEDURE (procedure),
LOAD_THUMB_PROC);
PIKA_PROC_ARG_STRING (procedure, "password",
_("PDF password"),
_("The password to decrypt the encrypted PDF file"),
NULL,
G_PARAM_READWRITE);
PIKA_PROC_ARG_BOOLEAN (procedure, "reverse-order",
_("Load in re_verse order"),
_("Load PDF pages in reverse order"),
FALSE,
G_PARAM_READWRITE);
/* FIXME: this should be a PIKA_PROC_ARG_ENUM of type
* PIKA_TYPE_PAGE_SELECTOR_TARGET but it won't work right now (see FIXME
* comment in libpika/pikagpparams-body.c:116).
PIKA_PROC_ARG_ENUM (procedure, "target",
_("Open pages as"),
_("Number of pages to load (0 for all)"),
PIKA_TYPE_PAGE_SELECTOR_TARGET,
PIKA_PAGE_SELECTOR_TARGET_LAYERS,
G_PARAM_READWRITE);
*/
PIKA_PROC_AUX_ARG_INT (procedure, "target",
_("Open pages as"),
_("Number of pages to load (0 for all)"),
PIKA_PAGE_SELECTOR_TARGET_LAYERS, PIKA_PAGE_SELECTOR_TARGET_IMAGES,
PIKA_PAGE_SELECTOR_TARGET_LAYERS,
G_PARAM_READWRITE);
PIKA_PROC_ARG_INT (procedure, "n-pages",
_("N pages"),
_("Number of pages to load (0 for all)"),
0, G_MAXINT, 0,
G_PARAM_READWRITE);
/* FIXME: shouldn't the whole selector be considered as one argument
* containing properties "target", "n-pages" and "pages" as a single
* object?
* Or actually should we store pages at all? While it makes sense to store
* some settings generally, not sure that the list of page makes sense
* from one PDF document loaded to another (different) one.
*/
PIKA_PROC_ARG_INT32_ARRAY (procedure, "pages",
_("Pages"),
_("The pages to load in the expected order"),
G_PARAM_READWRITE);
PIKA_PROC_ARG_BOOLEAN (procedure, "antialias",
_("Use _Anti-aliasing"),
_("Render texts with anti-aliasing"),
TRUE,
G_PARAM_READWRITE);
PIKA_PROC_ARG_BOOLEAN (procedure, "white-background",
_("_Fill transparent areas with white"),
_("Render all pages as opaque by filling the background in white"),
TRUE,
G_PARAM_READWRITE);
/* FIXME: we will have to think a bit about the most reasonable API. In any
* case, just "resolution" makes no sense without width/height.
*/
PIKA_PROC_AUX_ARG_DOUBLE (procedure, "resolution",
_("Resolution"),
_("Resolution"),
1.0, 5000.0, 300.0,
G_PARAM_READWRITE);
}
else if (! strcmp (name, LOAD_THUMB_PROC))
{
procedure = pika_thumbnail_procedure_new (plug_in, name,
PIKA_PDB_PROC_TYPE_PLUGIN,
pdf_load_thumb, NULL, NULL);
pika_procedure_set_documentation (procedure,
"Loads a preview from a PDF file.",
"Loads a small preview of the first "
"page of the PDF format file. Uses "
"the embedded thumbnail if present.",
name);
pika_procedure_set_attribution (procedure,
"Nathan Summers",
"Nathan Summers",
"2005");
}
return procedure;
}
static PikaValueArray *
pdf_load (PikaProcedure *procedure,
PikaRunMode run_mode,
GFile *file,
PikaMetadata *metadata,
PikaMetadataLoadFlags *flags,
PikaProcedureConfig *config,
gpointer run_data)
{
PikaValueArray *return_vals;
PikaPDBStatusType status = PIKA_PDB_SUCCESS;
PikaImage *image = NULL;
PopplerDocument *doc = NULL;
PdfSelectedPages pages = { 0, NULL };
GError *error = NULL;
gchar *password;
gegl_init (NULL, NULL);
if (run_mode == PIKA_RUN_INTERACTIVE)
pika_ui_init (PLUG_IN_BINARY);
g_object_get (config,
"password", &password,
NULL);
doc = open_document (file,
password,
run_mode, &error);
g_free (password);
if (doc == NULL)
{
status = PIKA_PDB_EXECUTION_ERROR;
}
else if (run_mode == PIKA_RUN_INTERACTIVE)
{
status = load_dialog (doc, &pages, procedure, config);
}
else if (run_mode == PIKA_RUN_NONINTERACTIVE)
{
PopplerPage *test_page = poppler_document_get_page (doc, 0);
if (test_page)
{
gint i;
gint doc_n_pages;
g_object_get (config, "n-pages", &pages.n_pages, NULL);
doc_n_pages = poppler_document_get_n_pages (doc);
/* The number of imported pages may be bigger than
* the number of pages from the original document.
* Indeed it is possible to duplicate some pages
* by setting the same number several times in the
* "pages" argument.
* Not ceiling this value is *not* an error.
*/
if (pages.n_pages <= 0)
{
pages.n_pages = doc_n_pages;
pages.pages = g_new (gint, pages.n_pages);
for (i = 0; i < pages.n_pages; i++)
pages.pages[i] = i;
}
else
{
const gint32 *p;
g_object_get (config, "pages", &p, NULL);
pages.pages = g_new (gint, pages.n_pages);
for (i = 0; i < pages.n_pages; i++)
{
if (p[i] >= doc_n_pages)
{
status = PIKA_PDB_EXECUTION_ERROR;
g_set_error (&error, PIKA_PLUGIN_PDF_LOAD_ERROR, 0,
/* TRANSLATORS: first argument is file name,
* second is out-of-range page number,
* third is number of pages.
* Specify order as in English if needed.
*/
ngettext ("PDF document '%1$s' has %3$d page. Page %2$d is out of range.",
"PDF document '%1$s' has %3$d pages. Page %2$d is out of range.",
doc_n_pages),
pika_file_get_utf8_name (file),
p[i],
doc_n_pages);
break;
}
else
{
pages.pages[i] = p[i];
}
}
}
g_object_unref (test_page);
}
}
if (status == PIKA_PDB_SUCCESS)
{
PikaPageSelectorTarget target;
gboolean reverse_order;
gdouble resolution;
gboolean antialias;
gboolean white_background;
g_object_get (config,
"target", &target,
"reverse-order", &reverse_order,
"resolution", &resolution,
"antialias", &antialias,
"white-background", &white_background,
NULL);
image = load_image (doc,
file,
run_mode,
target,
resolution,
antialias,
white_background,
reverse_order,
&pages);
if (image == NULL)
status = PIKA_PDB_EXECUTION_ERROR;
}
if (doc)
g_object_unref (doc);
g_free (pages.pages);
return_vals = pika_procedure_new_return_values (procedure, status, error);
if (status == PIKA_PDB_SUCCESS)
PIKA_VALUES_SET_IMAGE (return_vals, 1, image);
return return_vals;
}
static PikaValueArray *
pdf_load_thumb (PikaProcedure *procedure,
GFile *file,
gint size,
PikaProcedureConfig *config,
gpointer run_data)
{
PikaValueArray *return_vals;
gdouble width = 0;
gdouble height = 0;
gdouble scale;
PikaImage *image = NULL;
gint num_pages = 0;
PopplerDocument *doc = NULL;
cairo_surface_t *surface = NULL;
GError *error = NULL;
gegl_init (NULL, NULL);
doc = open_document (file,
NULL,
PIKA_RUN_NONINTERACTIVE,
&error);
if (doc)
{
PopplerPage *page = poppler_document_get_page (doc, 0);
if (page)
{
poppler_page_get_size (page, &width, &height);
g_object_unref (page);
}
num_pages = poppler_document_get_n_pages (doc);
surface = get_thumb_surface (doc, 0, size, TRUE);
g_object_unref (doc);
}
if (surface)
{
image = pika_image_new (cairo_image_surface_get_width (surface),
cairo_image_surface_get_height (surface),
PIKA_RGB);
pika_image_undo_disable (image);
layer_from_surface (image, "thumbnail", 0, surface, 0.0, 1.0);
cairo_surface_destroy (surface);
pika_image_undo_enable (image);
pika_image_clean_all (image);
}
/* Thumbnail resolution: 100.0. */
scale = 100.0 / pika_unit_get_factor (PIKA_UNIT_POINT);
width *= scale;
height *= scale;
if (! image)
return pika_procedure_new_return_values (procedure,
PIKA_PDB_EXECUTION_ERROR,
error);
return_vals = pika_procedure_new_return_values (procedure,
PIKA_PDB_SUCCESS,
NULL);
PIKA_VALUES_SET_IMAGE (return_vals, 1, image);
PIKA_VALUES_SET_INT (return_vals, 2, width);
PIKA_VALUES_SET_INT (return_vals, 3, height);
PIKA_VALUES_SET_ENUM (return_vals, 4, PIKA_RGB_IMAGE);
PIKA_VALUES_SET_INT (return_vals, 5, num_pages);
return return_vals;
}
static PopplerDocument *
open_document (GFile *file,
const gchar *PDF_password,
PikaRunMode run_mode,
GError **load_error)
{
PopplerDocument *doc;
GError *error = NULL;
doc = poppler_document_new_from_gfile (file, PDF_password, NULL, &error);
if (run_mode == PIKA_RUN_INTERACTIVE)
{
GtkWidget *label;
label = gtk_label_new (_("PDF is password protected, please input the password:"));
while (error &&
error->domain == POPPLER_ERROR &&
error->code == POPPLER_ERROR_ENCRYPTED)
{
GtkWidget *vbox;
GtkWidget *dialog;
GtkWidget *entry;
gint run;
dialog = pika_dialog_new (_("Encrypted PDF"), PLUG_IN_ROLE,
NULL, 0,
NULL, NULL,
_("_Cancel"), GTK_RESPONSE_CANCEL,
_("_OK"), GTK_RESPONSE_OK,
NULL);
pika_window_set_transient (GTK_WINDOW (dialog));
vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 12);
gtk_container_set_border_width (GTK_CONTAINER (vbox), 12);
gtk_box_pack_start (GTK_BOX (gtk_dialog_get_content_area (GTK_DIALOG (dialog))),
vbox, TRUE, TRUE, 0);
entry = gtk_entry_new ();
gtk_entry_set_visibility (GTK_ENTRY (entry), FALSE);
gtk_entry_set_activates_default (GTK_ENTRY (entry), TRUE);
gtk_container_add (GTK_CONTAINER (vbox), label);
gtk_container_add (GTK_CONTAINER (vbox), entry);
gtk_widget_show_all (dialog);
run = pika_dialog_run (PIKA_DIALOG (dialog));
if (run == GTK_RESPONSE_OK)
{
g_clear_error (&error);
doc = poppler_document_new_from_gfile (file,
gtk_entry_get_text (GTK_ENTRY (entry)),
NULL, &error);
}
label = gtk_label_new (_("Wrong password! Please input the right one:"));
gtk_widget_destroy (dialog);
if (run == GTK_RESPONSE_CANCEL || run == GTK_RESPONSE_DELETE_EVENT)
{
break;
}
}
gtk_widget_destroy (label);
}
/* We can't g_mapped_file_unref(mapped_file) as apparently doc has
* references to data in there. No big deal, this is just a
* short-lived plug-in.
*/
if (! doc)
{
g_set_error (load_error, G_FILE_ERROR, G_FILE_ERROR_FAILED,
_("Could not load '%s': %s"),
pika_file_get_utf8_name (file),
error->message);
g_error_free (error);
return NULL;
}
return doc;
}
static PikaLayer *
layer_from_surface (PikaImage *image,
const gchar *layer_name,
gint position,
cairo_surface_t *surface,
gdouble progress_start,
gdouble progress_scale)
{
PikaLayer *layer;
layer = pika_layer_new_from_surface (image, layer_name, surface,
progress_start,
progress_start + progress_scale);
pika_image_insert_layer (image, layer, NULL, position);
return layer;
}
static cairo_surface_t *
render_page_to_surface (PopplerPage *page,
int width,
int height,
double scale,
gboolean antialias,
gboolean white_background)
{
cairo_surface_t *surface;
cairo_t *cr;
surface = cairo_image_surface_create (CAIRO_FORMAT_ARGB32, width, height);
cr = cairo_create (surface);
cairo_save (cr);
cairo_translate (cr, 0.0, 0.0);
if (scale != 1.0)
cairo_scale (cr, scale, scale);
if (! antialias)
{
cairo_font_options_t *options = cairo_font_options_create ();
cairo_get_font_options (cr, options);
cairo_font_options_set_antialias (options, CAIRO_ANTIALIAS_NONE);
cairo_set_font_options (cr, options);
cairo_font_options_destroy (options);
cairo_set_antialias (cr, CAIRO_ANTIALIAS_NONE);
}
poppler_page_render (page, cr);
cairo_restore (cr);
if (white_background)
{
cairo_set_operator (cr, CAIRO_OPERATOR_DEST_OVER);
cairo_set_source_rgb (cr, 1.0, 1.0, 1.0);
cairo_paint (cr);
}
cairo_destroy (cr);
return surface;
}
#if 0
/* This is currently unused, but we'll have it here in case the military
wants it. */
static GdkPixbuf *
render_page_to_pixbuf (PopplerPage *page,
int width,
int height,
double scale)
{
GdkPixbuf *pixbuf;
cairo_surface_t *surface;
surface = render_page_to_surface (page, width, height, scale);
pixbuf = gdk_pixbuf_get_from_surface (surface, 0, 0,
cairo_image_surface_get_width (surface),
cairo_image_surface_get_height (surface));
cairo_surface_destroy (surface);
return pixbuf;
}
#endif
static PikaImage *
load_image (PopplerDocument *doc,
GFile *file,
PikaRunMode run_mode,
PikaPageSelectorTarget target,
gdouble resolution,
gboolean antialias,
gboolean white_background,
gboolean reverse_order,
PdfSelectedPages *pages)
{
PikaImage *image = NULL;
PikaImage **images = NULL;
gint i;
gdouble scale;
gdouble doc_progress = 0;
gint base_index = 0;
gint sign = 1;
if (reverse_order && pages->n_pages > 0)
{
base_index = pages->n_pages - 1;
sign = -1;
}
if (target == PIKA_PAGE_SELECTOR_TARGET_IMAGES)
images = g_new0 (PikaImage *, pages->n_pages);
pika_progress_init_printf (_("Opening '%s'"),
pika_file_get_utf8_name (file));
scale = resolution / pika_unit_get_factor (PIKA_UNIT_POINT);
/* read the file */
for (i = 0; i < pages->n_pages; i++)
{
PopplerPage *page;
gchar *page_label;
gdouble page_width;
gdouble page_height;
cairo_surface_t *surface;
gint width;
gint height;
gint page_index;
page_index = base_index + sign * i;
page = poppler_document_get_page (doc, pages->pages[page_index]);
poppler_page_get_size (page, &page_width, &page_height);
width = page_width * scale;
height = page_height * scale;
g_object_get (G_OBJECT (page), "label", &page_label, NULL);
if (! image)
{
image = pika_image_new (width, height, PIKA_RGB);
pika_image_undo_disable (image);
pika_image_set_resolution (image, resolution, resolution);
}
surface = render_page_to_surface (page, width, height, scale,
antialias, white_background);
layer_from_surface (image, page_label, 0, surface,
doc_progress, 1.0 / pages->n_pages);
g_free (page_label);
cairo_surface_destroy (surface);
doc_progress = (double) (i + 1) / pages->n_pages;
pika_progress_update (doc_progress);
if (target == PIKA_PAGE_SELECTOR_TARGET_IMAGES)
{
images[i] = image;
pika_image_undo_enable (image);
pika_image_clean_all (image);
image = 0;
}
}
pika_progress_update (1.0);
if (image)
{
pika_image_undo_enable (image);
pika_image_clean_all (image);
}
if (target == PIKA_PAGE_SELECTOR_TARGET_IMAGES)
{
if (run_mode != PIKA_RUN_NONINTERACTIVE)
{
/* Display images in reverse order. The last will be
* displayed by PIKA itself
*/
for (i = pages->n_pages - 1; i > 0; i--)
pika_display_new (images[i]);
}
image = images[0];
g_free (images);
}
return image;
}
static cairo_surface_t *
get_thumb_surface (PopplerDocument *doc,
gint page_num,
gint preferred_size,
gboolean white_background)
{
PopplerPage *page;
cairo_surface_t *surface;
page = poppler_document_get_page (doc, page_num);
if (! page)
return NULL;
surface = poppler_page_get_thumbnail (page);
if (! surface)
{
gdouble width;
gdouble height;
gdouble scale;
poppler_page_get_size (page, &width, &height);
scale = (gdouble) preferred_size / MAX (width, height);
width *= scale;
height *= scale;
surface = render_page_to_surface (page, width, height, scale, TRUE, white_background);
}
g_object_unref (page);
return surface;
}
static GdkPixbuf *
get_thumb_pixbuf (PopplerDocument *doc,
gint page_num,
gint preferred_size,
gboolean white_background)
{
cairo_surface_t *surface;
GdkPixbuf *pixbuf;
surface = get_thumb_surface (doc, page_num, preferred_size, white_background);
pixbuf = gdk_pixbuf_get_from_surface (surface, 0, 0,
cairo_image_surface_get_width (surface),
cairo_image_surface_get_height (surface));
cairo_surface_destroy (surface);
return pixbuf;
}
typedef struct
{
PopplerDocument *document;
PikaPageSelector *selector;
gboolean white_background;
GMutex mutex;
GCond render_thumb;
gboolean stop_thumbnailing;
gboolean render_thumbnails;
} ThreadData;
typedef struct
{
PikaPageSelector *selector;
gint page_no;
GdkPixbuf *pixbuf;
} IdleData;
static gboolean
idle_set_thumbnail (gpointer data)
{
IdleData *idle_data = data;
pika_page_selector_set_page_thumbnail (idle_data->selector,
idle_data->page_no,
idle_data->pixbuf);
g_object_unref (idle_data->pixbuf);
g_free (idle_data);
return FALSE;
}
static gpointer
thumbnail_thread (gpointer data)
{
ThreadData *thread_data = data;
gboolean first_loop = TRUE;
gint n_pages;
gint i;
n_pages = poppler_document_get_n_pages (thread_data->document);
while (TRUE)
{
gboolean white_background;
gboolean stop_thumbnailing;
gboolean render_thumbnails;
g_mutex_lock (&thread_data->mutex);
if (first_loop)
first_loop = FALSE;
else
g_cond_wait (&thread_data->render_thumb, &thread_data->mutex);
stop_thumbnailing = thread_data->stop_thumbnailing;
g_mutex_unlock (&thread_data->mutex);
if (stop_thumbnailing)
break;
g_mutex_lock (&thread_data->mutex);
render_thumbnails = thread_data->render_thumbnails;
white_background = thread_data->white_background;
thread_data->render_thumbnails = FALSE;
g_mutex_unlock (&thread_data->mutex);
/* This handles "spurious wakeup", i.e. cases when g_cond_wait() returned
* even though there was no call asking us to re-render the thumbnails.
* See docs of g_cond_wait().
*/
if (! render_thumbnails)
continue;
for (i = 0; i < n_pages; i++)
{
IdleData *idle_data = g_new0 (IdleData, 1);
gboolean white_background2;
idle_data->selector = thread_data->selector;
idle_data->page_no = i;
/* FIXME get preferred size from somewhere? */
idle_data->pixbuf = get_thumb_pixbuf (thread_data->document, i,
THUMBNAIL_SIZE,
white_background);
g_idle_add (idle_set_thumbnail, idle_data);
g_mutex_lock (&thread_data->mutex);
white_background2 = thread_data->white_background;
stop_thumbnailing = thread_data->stop_thumbnailing;
g_mutex_unlock (&thread_data->mutex);
if (stop_thumbnailing || white_background2 != white_background)
break;
}
if (stop_thumbnailing)
break;
}
return NULL;
}
static void
white_background_toggled (GtkToggleButton *widget,
ThreadData *thread_data)
{
g_mutex_lock (&thread_data->mutex);
thread_data->white_background = gtk_toggle_button_get_active (widget);
thread_data->render_thumbnails = TRUE;
g_cond_signal (&thread_data->render_thumb);
g_mutex_unlock (&thread_data->mutex);
}
static PikaPDBStatusType
load_dialog (PopplerDocument *doc,
PdfSelectedPages *pages,
PikaProcedure *procedure,
PikaProcedureConfig *config)
{
GtkWidget *dialog;
GtkWidget *vbox;
GtkWidget *title;
GtkWidget *selector;
GtkWidget *res_entry;
GtkWidget *white_bg;
ThreadData thread_data;
GThread *thread;
gint i;
gint n_pages;
gdouble width;
gdouble height;
gboolean run;
PikaPageSelectorTarget target;
gdouble resolution;
gboolean white_background;
dialog = pika_procedure_dialog_new (PIKA_PROCEDURE (procedure),
PIKA_PROCEDURE_CONFIG (config),
_("Import from PDF"));
g_object_get (config,
"target", &target,
"resolution", &resolution,
"white-background", &white_background,
NULL);
vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 12);
gtk_container_set_border_width (GTK_CONTAINER (vbox), 12);
gtk_box_pack_start (GTK_BOX (gtk_dialog_get_content_area (GTK_DIALOG (dialog))),
vbox, TRUE, TRUE, 0);
gtk_widget_show (vbox);
/* Title */
title = pika_prop_label_new (G_OBJECT (doc), "title");
gtk_label_set_ellipsize (GTK_LABEL (title), PANGO_ELLIPSIZE_END);
gtk_box_pack_start (GTK_BOX (vbox), title, FALSE, FALSE, 0);
/* Page Selector */
selector = pika_page_selector_new ();
gtk_widget_set_size_request (selector, 380, 360);
gtk_box_pack_start (GTK_BOX (vbox), selector, TRUE, TRUE, 0);
gtk_widget_show (selector);
n_pages = poppler_document_get_n_pages (doc);
if (n_pages <= 0)
{
g_message (_("Error getting number of pages from the given PDF file."));
return PIKA_PDB_EXECUTION_ERROR;
}
pika_page_selector_set_n_pages (PIKA_PAGE_SELECTOR (selector), n_pages);
pika_page_selector_set_target (PIKA_PAGE_SELECTOR (selector),
target);
for (i = 0; i < n_pages; i++)
{
PopplerPage *page;
gchar *label;
page = poppler_document_get_page (doc, i);
g_object_get (G_OBJECT (page), "label", &label, NULL);
pika_page_selector_set_page_label (PIKA_PAGE_SELECTOR (selector), i,
label);
if (i == 0)
poppler_page_get_size (page, &width, &height);
g_object_unref (page);
g_free (label);
}
/* Since selecting none will be equivalent to selecting all, this is
* only useful as a feedback for the default behavior of selecting all
* pages. */
pika_page_selector_select_all (PIKA_PAGE_SELECTOR (selector));
g_signal_connect_swapped (selector, "activate",
G_CALLBACK (gtk_window_activate_default),
dialog);
thread_data.document = doc;
thread_data.selector = PIKA_PAGE_SELECTOR (selector);
thread_data.render_thumbnails = TRUE;
thread_data.stop_thumbnailing = FALSE;
thread_data.white_background = white_background;
g_mutex_init (&thread_data.mutex);
g_cond_init (&thread_data.render_thumb);
thread = g_thread_new ("thumbnailer", thumbnail_thread, &thread_data);
pika_procedure_dialog_fill (PIKA_PROCEDURE_DIALOG (dialog),
"reverse-order",
NULL);
/* Resolution */
res_entry = pika_resolution_entry_new (_("_Width (pixels):"), width,
_("_Height (pixels):"), height,
PIKA_UNIT_POINT,
_("_Resolution:"),
resolution, PIKA_UNIT_INCH);
gtk_box_pack_start (GTK_BOX (gtk_dialog_get_content_area (GTK_DIALOG (dialog))),
res_entry, FALSE, FALSE, 0);
gtk_widget_show (res_entry);
g_signal_connect (res_entry, "x-changed",
G_CALLBACK (pika_resolution_entry_update_x_in_dpi),
&resolution);
white_bg = pika_procedure_dialog_get_widget (PIKA_PROCEDURE_DIALOG (dialog),
"white-background", G_TYPE_NONE);
g_signal_connect (white_bg, "toggled",
G_CALLBACK (white_background_toggled), &thread_data);
pika_procedure_dialog_fill (PIKA_PROCEDURE_DIALOG (dialog),
"antialias",
"white-background",
NULL);
run = pika_procedure_dialog_run (PIKA_PROCEDURE_DIALOG (dialog));
target = pika_page_selector_get_target (PIKA_PAGE_SELECTOR (selector));
g_object_set (config,
"target", target,
"resolution", resolution,
NULL);
pages->pages =
pika_page_selector_get_selected_pages (PIKA_PAGE_SELECTOR (selector),
&pages->n_pages);
/* select all if none selected */
if (pages->n_pages == 0)
{
pika_page_selector_select_all (PIKA_PAGE_SELECTOR (selector));
pages->pages =
pika_page_selector_get_selected_pages (PIKA_PAGE_SELECTOR (selector),
&pages->n_pages);
}
/* cleanup */
g_mutex_lock (&thread_data.mutex);
thread_data.stop_thumbnailing = TRUE;
g_cond_signal (&thread_data.render_thumb);
g_mutex_unlock (&thread_data.mutex);
g_thread_join (thread);
g_mutex_clear (&thread_data.mutex);
g_cond_clear (&thread_data.render_thumb);
/* XXX Unsure why, I get some CRITICALs when destroying the dialog, unless I
* unselect all first.
*/
pika_page_selector_unselect_all (PIKA_PAGE_SELECTOR (selector));
gtk_widget_destroy (dialog);
return run ? PIKA_PDB_SUCCESS : PIKA_PDB_CANCEL;
}
/**
** code for PikaResolutionEntry widget, formerly in libpikawidgets
**/
static guint pika_resolution_entry_signals[LAST_SIGNAL] = { 0 };
static GtkGridClass *parent_class = NULL;
GType
pika_resolution_entry_get_type (void)
{
static GType gre_type = 0;
if (! gre_type)
{
const GTypeInfo gre_info =
{
sizeof (PikaResolutionEntryClass),
(GBaseInitFunc) NULL,
(GBaseFinalizeFunc) NULL,
(GClassInitFunc) pika_resolution_entry_class_init,
NULL, /* class_finalize */
NULL, /* class_data */
sizeof (PikaResolutionEntry),
0, /* n_preallocs */
(GInstanceInitFunc) pika_resolution_entry_init,
};
gre_type = g_type_register_static (GTK_TYPE_GRID,
"PikaResolutionEntry",
&gre_info, 0);
}
return gre_type;
}
static void
pika_resolution_entry_class_init (PikaResolutionEntryClass *klass)
{
parent_class = g_type_class_peek_parent (klass);
pika_resolution_entry_signals[HEIGHT_CHANGED] =
g_signal_new ("height-changed",
G_TYPE_FROM_CLASS (klass),
G_SIGNAL_RUN_FIRST,
G_STRUCT_OFFSET (PikaResolutionEntryClass, value_changed),
NULL, NULL, NULL,
G_TYPE_NONE, 0);
pika_resolution_entry_signals[WIDTH_CHANGED] =
g_signal_new ("width-changed",
G_TYPE_FROM_CLASS (klass),
G_SIGNAL_RUN_FIRST,
G_STRUCT_OFFSET (PikaResolutionEntryClass, value_changed),
NULL, NULL, NULL,
G_TYPE_NONE, 0);
pika_resolution_entry_signals[X_CHANGED] =
g_signal_new ("x-changed",
G_TYPE_FROM_CLASS (klass),
G_SIGNAL_RUN_FIRST,
G_STRUCT_OFFSET (PikaResolutionEntryClass, value_changed),
NULL, NULL, NULL,
G_TYPE_NONE, 0);
pika_resolution_entry_signals[Y_CHANGED] =
g_signal_new ("y-changed",
G_TYPE_FROM_CLASS (klass),
G_SIGNAL_RUN_FIRST,
G_STRUCT_OFFSET (PikaResolutionEntryClass, refval_changed),
NULL, NULL, NULL,
G_TYPE_NONE, 0);
pika_resolution_entry_signals[UNIT_CHANGED] =
g_signal_new ("unit-changed",
G_TYPE_FROM_CLASS (klass),
G_SIGNAL_RUN_FIRST,
G_STRUCT_OFFSET (PikaResolutionEntryClass, unit_changed),
NULL, NULL, NULL,
G_TYPE_NONE, 0);
klass->value_changed = NULL;
klass->refval_changed = NULL;
klass->unit_changed = NULL;
}
static void
pika_resolution_entry_init (PikaResolutionEntry *gre)
{
gre->unitmenu = NULL;
gre->unit = PIKA_UNIT_INCH;
gtk_grid_set_row_spacing (GTK_GRID (gre), 2);
gtk_grid_set_column_spacing (GTK_GRID (gre), 4);
}
static void
pika_resolution_entry_field_init (PikaResolutionEntry *gre,
PikaResolutionEntryField *gref,
PikaResolutionEntryField *corresponding,
guint changed_signal,
gdouble initial_val,
PikaUnit initial_unit,
gboolean size,
gint spinbutton_width)
{
gint digits;
g_return_if_fail (PIKA_IS_RESOLUTION_ENTRY (gre));
gref->gre = gre;
gref->corresponding = corresponding;
gref->changed_signal = pika_resolution_entry_signals[changed_signal];
if (size)
{
gref->value = initial_val /
pika_unit_get_factor (initial_unit) *
corresponding->value *
pika_unit_get_factor (gre->unit);
gref->phy_size = initial_val /
pika_unit_get_factor (initial_unit);
}
else
{
gref->value = initial_val;
}
gref->min_value = PIKA_MIN_RESOLUTION;
gref->max_value = PIKA_MAX_RESOLUTION;
gref->adjustment = NULL;
gref->stop_recursion = 0;
gref->size = size;
if (size)
{
gref->label = g_object_new (GTK_TYPE_LABEL,
"xalign", 0.0,
"yalign", 0.5,
NULL);
pika_label_set_attributes (GTK_LABEL (gref->label),
PANGO_ATTR_STYLE, PANGO_STYLE_ITALIC,
-1);
pika_resolution_entry_format_label (gre, gref->label, gref->phy_size);
}
digits = size ? 0 : MIN (pika_unit_get_digits (initial_unit), 5) + 1;
gref->adjustment = gtk_adjustment_new (gref->value,
gref->min_value,
gref->max_value,
1.0, 10.0, 0.0);
gref->spinbutton = pika_spin_button_new (gref->adjustment,
1.0, digits);
gtk_spin_button_set_numeric (GTK_SPIN_BUTTON (gref->spinbutton), TRUE);
if (spinbutton_width > 0)
{
if (spinbutton_width < 17)
gtk_entry_set_width_chars (GTK_ENTRY (gref->spinbutton),
spinbutton_width);
else
gtk_widget_set_size_request (gref->spinbutton,
spinbutton_width, -1);
}
}
/**
* pika_resolution_entry_new:
* @width_label: Optional label for the width control.
* @width: Width of the item, specified in terms of @size_unit.
* @height_label: Optional label for the height control.
* @height: Height of the item, specified in terms of @size_unit.
* @size_unit: Unit used to specify the width and height.
* @res_label: Optional label for the resolution entry.
* @initial_res: The initial resolution.
* @initial_unit: The initial unit.
*
* Creates a new #PikaResolutionEntry widget.
*
* The #PikaResolutionEntry is derived from #GtkGrid and will have
* an empty border of one cell width on each side plus an empty column left
* of the #PikaUnitMenu to allow the caller to add labels or other widgets.
*
* A #PikaChainButton is displayed if independent is set to %TRUE.
*
* Returns: A pointer to the new #PikaResolutionEntry widget.
**/
GtkWidget *
pika_resolution_entry_new (const gchar *width_label,
gdouble width,
const gchar *height_label,
gdouble height,
PikaUnit size_unit,
const gchar *res_label,
gdouble initial_res,
PikaUnit initial_unit)
{
PikaResolutionEntry *gre;
GtkTreeModel *model;
gre = g_object_new (PIKA_TYPE_RESOLUTION_ENTRY, NULL);
gre->unit = initial_unit;
pika_resolution_entry_field_init (gre, &gre->x,
&gre->width,
X_CHANGED,
initial_res, initial_unit,
FALSE, 0);
gtk_grid_attach (GTK_GRID (gre), gre->x.spinbutton, 1, 3, 1, 1);
g_signal_connect (gre->x.adjustment, "value-changed",
G_CALLBACK (pika_resolution_entry_value_callback),
&gre->x);
gtk_widget_show (gre->x.spinbutton);
gre->unitmenu = pika_unit_combo_box_new ();
model = gtk_combo_box_get_model (GTK_COMBO_BOX (gre->unitmenu));
pika_unit_store_set_has_pixels (PIKA_UNIT_STORE (model), FALSE);
pika_unit_store_set_has_percent (PIKA_UNIT_STORE (model), FALSE);
g_object_set (model,
"short-format", _("pixels/%a"),
"long-format", _("pixels/%a"),
NULL);
pika_unit_combo_box_set_active (PIKA_UNIT_COMBO_BOX (gre->unitmenu),
initial_unit);
gtk_grid_attach (GTK_GRID (gre), gre->unitmenu, 3, 3, 1, 1);
g_signal_connect (gre->unitmenu, "changed",
G_CALLBACK (pika_resolution_entry_unit_callback),
gre);
gtk_widget_show (gre->unitmenu);
pika_resolution_entry_field_init (gre, &gre->width,
&gre->x,
WIDTH_CHANGED,
width, size_unit,
TRUE, 0);
gtk_grid_attach (GTK_GRID (gre), gre->width.spinbutton, 1, 1, 1, 1);
gtk_grid_attach (GTK_GRID (gre), gre->width.label, 3, 1, 1, 1);
g_signal_connect (gre->width.adjustment, "value-changed",
G_CALLBACK (pika_resolution_entry_value_callback),
&gre->width);
gtk_widget_show (gre->width.spinbutton);
gtk_widget_show (gre->width.label);
pika_resolution_entry_field_init (gre, &gre->height, &gre->x,
HEIGHT_CHANGED,
height, size_unit,
TRUE, 0);
gtk_grid_attach (GTK_GRID (gre), gre->height.spinbutton, 1, 2, 1, 1);
gtk_grid_attach (GTK_GRID (gre), gre->height.label, 3, 2, 1, 1);
g_signal_connect (gre->height.adjustment, "value-changed",
G_CALLBACK (pika_resolution_entry_value_callback),
&gre->height);
gtk_widget_show (gre->height.spinbutton);
gtk_widget_show (gre->height.label);
if (width_label)
pika_resolution_entry_attach_label (gre, width_label, 1, 0, 0.0);
if (height_label)
pika_resolution_entry_attach_label (gre, height_label, 2, 0, 0.0);
if (res_label)
pika_resolution_entry_attach_label (gre, res_label, 3, 0, 0.0);
return GTK_WIDGET (gre);
}
/**
* pika_resolution_entry_attach_label:
* @gre: The #PikaResolutionEntry you want to add a label to.
* @text: The text of the label.
* @row: The row where the label will be attached.
* @column: The column where the label will be attached.
* @alignment: The horizontal alignment of the label.
*
* Attaches a #GtkLabel to the #PikaResolutionEntry (which is a #GtkGrid).
*
* Returns: A pointer to the new #GtkLabel widget.
**/
GtkWidget *
pika_resolution_entry_attach_label (PikaResolutionEntry *gre,
const gchar *text,
gint row,
gint column,
gfloat alignment)
{
GtkWidget *label;
g_return_val_if_fail (PIKA_IS_RESOLUTION_ENTRY (gre), NULL);
g_return_val_if_fail (text != NULL, NULL);
label = gtk_label_new_with_mnemonic (text);
if (column == 0)
{
GList *children;
GList *list;
children = gtk_container_get_children (GTK_CONTAINER (gre));
for (list = children; list; list = g_list_next (list))
{
GtkWidget *child = list->data;
gint left_attach;
gint top_attach;
gtk_container_child_get (GTK_CONTAINER (gre), child,
"left-attach", &left_attach,
"top-attach", &top_attach,
NULL);
if (left_attach == 1 && top_attach == row)
{
gtk_label_set_mnemonic_widget (GTK_LABEL (label), child);
break;
}
}
g_list_free (children);
}
gtk_label_set_xalign (GTK_LABEL (label), alignment);
gtk_grid_attach (GTK_GRID (gre), label, column, row, 1, 1);
gtk_widget_show (label);
return label;
}
/**
* pika_resolution_entry_get_x_in_dpi;
* @gre: The #PikaResolutionEntry you want to know the resolution of.
*
* Returns the X resolution of the #PikaResolutionEntry in pixels per inch.
**/
gdouble
pika_resolution_entry_get_x_in_dpi (PikaResolutionEntry *gre)
{
g_return_val_if_fail (PIKA_IS_RESOLUTION_ENTRY (gre), 0);
/* dots_in_one_unit * units_in_one_inch -> dpi */
return gre->x.value * pika_unit_get_factor (gre->unit);
}
/**
* pika_resolution_entry_get_y_in_dpi;
* @gre: The #PikaResolutionEntry you want to know the resolution of.
*
* Returns the Y resolution of the #PikaResolutionEntry in pixels per inch.
**/
gdouble
pika_resolution_entry_get_y_in_dpi (PikaResolutionEntry *gre)
{
g_return_val_if_fail (PIKA_IS_RESOLUTION_ENTRY (gre), 0);
return gre->y.value * pika_unit_get_factor (gre->unit);
}
static void
pika_resolution_entry_update_value (PikaResolutionEntryField *gref,
gdouble value)
{
if (gref->stop_recursion > 0)
return;
gref->value = value;
gref->stop_recursion++;
if (gref->size)
pika_resolution_entry_update_value (gref->corresponding,
gref->value /
gref->phy_size /
pika_unit_get_factor (gref->gre->unit));
else
{
gdouble factor = pika_unit_get_factor (gref->gre->unit);
pika_resolution_entry_update_value (&gref->gre->width,
gref->value *
gref->gre->width.phy_size *
factor);
pika_resolution_entry_update_value (&gref->gre->height,
gref->value *
gref->gre->height.phy_size *
factor);
}
gtk_adjustment_set_value (gref->adjustment, value);
gref->stop_recursion--;
g_signal_emit (gref->gre, gref->changed_signal, 0);
}
static void
pika_resolution_entry_value_callback (GtkAdjustment *adjustment,
gpointer data)
{
PikaResolutionEntryField *gref = (PikaResolutionEntryField *) data;
gdouble new_value;
new_value = gtk_adjustment_get_value (adjustment);
if (gref->value != new_value)
pika_resolution_entry_update_value (gref, new_value);
}
static void
pika_resolution_entry_update_unit (PikaResolutionEntry *gre,
PikaUnit unit)
{
PikaUnit old_unit;
gint digits;
gdouble factor;
old_unit = gre->unit;
gre->unit = unit;
digits = (pika_unit_get_digits (PIKA_UNIT_INCH) -
pika_unit_get_digits (unit));
gtk_spin_button_set_digits (GTK_SPIN_BUTTON (gre->x.spinbutton),
MAX (3 + digits, 3));
factor = pika_unit_get_factor (old_unit) / pika_unit_get_factor (unit);
gre->x.min_value *= factor;
gre->x.max_value *= factor;
gre->x.value *= factor;
gtk_adjustment_set_value (gre->x.adjustment, gre->x.value);
pika_resolution_entry_format_label (gre,
gre->width.label, gre->width.phy_size);
pika_resolution_entry_format_label (gre,
gre->height.label, gre->height.phy_size);
g_signal_emit (gre, pika_resolution_entry_signals[UNIT_CHANGED], 0);
}
static void
pika_resolution_entry_unit_callback (GtkWidget *widget,
PikaResolutionEntry *gre)
{
PikaUnit new_unit;
new_unit = pika_unit_combo_box_get_active (PIKA_UNIT_COMBO_BOX (widget));
if (gre->unit != new_unit)
pika_resolution_entry_update_unit (gre, new_unit);
}
/**
* pika_resolution_entry_update_x_in_dpi:
* @gre: the #PikaResolutionEntry
* @data: a pointer to a gdouble
*
* Convenience function to set a double to the X resolution, suitable
* for use as a signal callback.
*/
void
pika_resolution_entry_update_x_in_dpi (PikaResolutionEntry *gre,
gpointer data)
{
gdouble *val;
g_return_if_fail (gre != NULL);
g_return_if_fail (data != NULL);
g_return_if_fail (PIKA_IS_RESOLUTION_ENTRY (gre));
val = (gdouble *) data;
*val = pika_resolution_entry_get_x_in_dpi (gre);
}
/**
* pika_resolution_entry_update_y_in_dpi:
* @gre: the #PikaResolutionEntry
* @data: a pointer to a gdouble
*
* Convenience function to set a double to the Y resolution, suitable
* for use as a signal callback.
*/
void
pika_resolution_entry_update_y_in_dpi (PikaResolutionEntry *gre,
gpointer data)
{
gdouble *val;
g_return_if_fail (gre != NULL);
g_return_if_fail (data != NULL);
g_return_if_fail (PIKA_IS_RESOLUTION_ENTRY (gre));
val = (gdouble *) data;
*val = pika_resolution_entry_get_y_in_dpi (gre);
}
static void
pika_resolution_entry_format_label (PikaResolutionEntry *gre,
GtkWidget *label,
gdouble size)
{
gchar *format = g_strdup_printf ("%%.%df %%s",
pika_unit_get_digits (gre->unit));
gchar *text = g_strdup_printf (format,
size * pika_unit_get_factor (gre->unit),
pika_unit_get_plural (gre->unit));
g_free (format);
gtk_label_set_text (GTK_LABEL (label), text);
g_free (text);
}