]> www.fi.muni.cz Git - evince.git/blob - properties/ev-properties-view.c
6382af4a7ef5a24831d73355ca6e92e994f7e771
[evince.git] / properties / ev-properties-view.c
1 /* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8; c-indent-level: 8 -*- */
2 /* this file is part of evince, a gnome document viewer
3  *
4  *  Copyright (C) 2005 Red Hat, Inc
5  *
6  * Evince is free software; you can redistribute it and/or modify it
7  * under the terms of the GNU General Public License as published by
8  * the Free Software Foundation; either version 2 of the License, or
9  * (at your option) any later version.
10  *
11  * Evince is distributed in the hope that it will be useful, but
12  * WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program; if not, write to the Free Software
18  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA.
19  */
20
21 #include "config.h"
22
23 #include <string.h>
24 #include <sys/time.h>
25 #include <time.h>
26
27 #ifdef HAVE__NL_MEASUREMENT_MEASUREMENT
28 #include <langinfo.h>
29 #endif
30
31 #include <glib/gi18n.h>
32 #include <gtk/gtk.h>
33
34 #include "ev-properties-view.h"
35
36 typedef enum {
37         TITLE_PROPERTY,
38         URI_PROPERTY,
39         SUBJECT_PROPERTY,
40         AUTHOR_PROPERTY,
41         KEYWORDS_PROPERTY,
42         PRODUCER_PROPERTY,
43         CREATOR_PROPERTY,
44         CREATION_DATE_PROPERTY,
45         MOD_DATE_PROPERTY,
46         N_PAGES_PROPERTY,
47         LINEARIZED_PROPERTY,
48         FORMAT_PROPERTY,
49         SECURITY_PROPERTY,
50         PAPER_SIZE_PROPERTY
51 } Property;
52
53 typedef struct {
54         Property property;
55         const char *label;
56 } PropertyInfo;
57
58 static const PropertyInfo properties_info[] = {
59         { TITLE_PROPERTY,         N_("Title") },
60         { URI_PROPERTY,           N_("Location") },
61         { SUBJECT_PROPERTY,       N_("Subject") },
62         { AUTHOR_PROPERTY,        N_("Author") },
63         { KEYWORDS_PROPERTY,      N_("Keywords") },
64         { PRODUCER_PROPERTY,      N_("Producer") },
65         { CREATOR_PROPERTY,       N_("Creator") },
66         { CREATION_DATE_PROPERTY, N_("Created") },
67         { MOD_DATE_PROPERTY,      N_("Modified") },
68         { N_PAGES_PROPERTY,       N_("Number of Pages") },
69         { LINEARIZED_PROPERTY,    N_("Optimized") },
70         { FORMAT_PROPERTY,        N_("Format") },
71         { SECURITY_PROPERTY,      N_("Security") },
72         { PAPER_SIZE_PROPERTY,    N_("Paper Size") }
73 };
74
75 struct _EvPropertiesView {
76         GtkVBox base_instance;
77
78         GtkWidget *table;
79         gchar     *uri;
80 };
81
82 struct _EvPropertiesViewClass {
83         GtkVBoxClass base_class;
84 };
85
86 G_DEFINE_TYPE (EvPropertiesView, ev_properties_view, GTK_TYPE_VBOX)
87
88 static void
89 ev_properties_view_dispose (GObject *object)
90 {
91         EvPropertiesView *properties = EV_PROPERTIES_VIEW (object);
92         
93         if (properties->uri) {
94                 g_free (properties->uri);
95                 properties->uri = NULL;
96         }
97         
98         G_OBJECT_CLASS (ev_properties_view_parent_class)->dispose (object);
99 }
100
101 static void
102 ev_properties_view_class_init (EvPropertiesViewClass *properties_class)
103 {
104         GObjectClass *g_object_class = G_OBJECT_CLASS (properties_class);
105
106         g_object_class->dispose = ev_properties_view_dispose;
107 }
108
109 /* Returns a locale specific date and time representation */
110 static char *
111 ev_properties_view_format_date (GTime utime)
112 {
113         time_t time = (time_t) utime;
114         struct tm t;
115         char s[256];
116         const char *fmt_hack = "%c";
117         size_t len;
118
119         if (time == 0 || !localtime_r (&time, &t)) return NULL;
120
121         len = strftime (s, sizeof (s), fmt_hack, &t);
122         if (len == 0 || s[0] == '\0') return NULL;
123
124         return g_locale_to_utf8 (s, -1, NULL, NULL, NULL);
125 }
126
127 /* This is cut out of gconvert.c from glib (and mildly modified).  Not all
128    backends give valid UTF-8 for properties, so we make sure that is.
129  */
130 static gchar *
131 make_valid_utf8 (const gchar *name)
132 {
133   GString *string;
134   const gchar *remainder, *invalid;
135   gint remaining_bytes, valid_bytes;
136   
137   string = NULL;
138   remainder = name;
139   remaining_bytes = strlen (name);
140   
141   while (remaining_bytes != 0) 
142     {
143       if (g_utf8_validate (remainder, remaining_bytes, &invalid)) 
144         break;
145       valid_bytes = invalid - remainder;
146     
147       if (string == NULL) 
148         string = g_string_sized_new (remaining_bytes);
149
150       g_string_append_len (string, remainder, valid_bytes);
151       g_string_append_c (string, '?');
152       
153       remaining_bytes -= valid_bytes + 1;
154       remainder = invalid + 1;
155     }
156   
157   if (string == NULL)
158     return g_strdup (name);
159   
160   g_string_append (string, remainder);
161
162   g_assert (g_utf8_validate (string->str, -1, NULL));
163   
164   return g_string_free (string, FALSE);
165 }
166
167 static void
168 set_property (GtkTable    *table,
169               Property     property,
170               const gchar *text,
171               gint        *row)
172 {
173         GtkWidget *label;
174         gchar     *markup;
175         gchar     *valid_text;
176
177         label = gtk_label_new (NULL);
178         g_object_set (G_OBJECT (label), "xalign", 0.0, NULL);
179         markup = g_strdup_printf ("<b>%s:</b>", properties_info[property].label);
180         gtk_label_set_markup (GTK_LABEL (label), markup);
181         g_free (markup);
182         
183         gtk_table_attach (table, label, 0, 1, *row, *row + 1,
184                           GTK_FILL, GTK_FILL, 0, 0);
185         gtk_widget_show (label);
186
187         label = gtk_label_new (NULL);
188         g_object_set (G_OBJECT (label),
189                       "xalign", 0.0,
190                       "width_chars", 25,
191                       "selectable", TRUE,
192                       "ellipsize", PANGO_ELLIPSIZE_END,
193                       NULL);
194         
195
196         if (text == NULL || text[0] == '\000') {
197                 markup = g_markup_printf_escaped ("<i>%s</i>", _("None"));
198                 gtk_label_set_markup (GTK_LABEL (label), markup);
199                 g_free (markup);
200         } else {
201                 valid_text = make_valid_utf8 (text ? text : "");
202                 gtk_label_set_text (GTK_LABEL (label), valid_text);
203                 g_free (valid_text);
204         }
205
206         gtk_table_attach (table, label, 1, 2, *row, *row + 1,
207                           GTK_FILL | GTK_EXPAND, GTK_FILL, 0, 0);
208         gtk_widget_show (label);
209
210         *row += 1;
211 }
212
213 static GtkUnit
214 get_default_user_units (void)
215 {
216         /* Translate to the default units to use for presenting
217          * lengths to the user. Translate to default:inch if you
218          * want inches, otherwise translate to default:mm.
219          * Do *not* translate it to "predefinito:mm", if it
220          * it isn't default:mm or default:inch it will not work
221          */
222         gchar *e = _("default:mm");
223
224 #ifdef HAVE__NL_MEASUREMENT_MEASUREMENT
225         gchar *imperial = NULL;
226         
227         imperial = nl_langinfo (_NL_MEASUREMENT_MEASUREMENT);
228         if (imperial && imperial[0] == 2)
229                 return GTK_UNIT_INCH;  /* imperial */
230         if (imperial && imperial[0] == 1)
231                 return GTK_UNIT_MM;  /* metric */
232 #endif
233
234         if (strcmp (e, "default:mm") == 0)
235                 return GTK_UNIT_MM;
236         if (strcmp (e, "default:inch") == 0)
237                 return GTK_UNIT_INCH;
238         
239         g_warning ("Whoever translated default:mm did so wrongly.\n");
240                                 
241         return GTK_UNIT_MM;
242 }
243
244 static gdouble
245 get_tolerance (gdouble size)
246 {
247         if (size < 150.0f)
248                 return 1.5f;
249         else if (size >= 150.0f && size <= 600.0f)
250                 return 2.0f;
251         else
252                 return 3.0f;
253 }
254
255 static char *
256 ev_regular_paper_size (const EvDocumentInfo *info)
257 {
258         GList *paper_sizes, *l;
259         gchar *exact_size;
260         gchar *str = NULL;
261         GtkUnit units;
262
263         units = get_default_user_units ();
264
265         if (units == GTK_UNIT_MM) {
266                 exact_size = g_strdup_printf(_("%.0f x %.0f mm"),
267                                              info->paper_width,
268                                              info->paper_height);
269         } else {
270                 exact_size = g_strdup_printf (_("%.2f x %.2f inch"),
271                                               info->paper_width  / 25.4f,
272                                               info->paper_height / 25.4f);
273         }
274
275         paper_sizes = gtk_paper_size_get_paper_sizes (FALSE);
276         
277         for (l = paper_sizes; l && l->data; l = g_list_next (l)) {
278                 GtkPaperSize *size = (GtkPaperSize *) l->data;
279                 gdouble paper_width;
280                 gdouble paper_height;
281                 gdouble width_tolerance;
282                 gdouble height_tolerance;
283
284                 paper_width = gtk_paper_size_get_width (size, GTK_UNIT_MM);
285                 paper_height = gtk_paper_size_get_height (size, GTK_UNIT_MM);
286
287                 width_tolerance = get_tolerance (paper_width);
288                 height_tolerance = get_tolerance (paper_height);
289
290                 if (ABS (info->paper_height - paper_height) <= height_tolerance &&
291                     ABS (info->paper_width  - paper_width) <= width_tolerance) {
292                         /* Note to translators: first placeholder is the paper name (eg.
293                          * A4), second placeholder is the paper size (eg. 297x210 mm) */
294                         str = g_strdup_printf (_("%s, Portrait (%s)"),
295                                                gtk_paper_size_get_display_name (size),
296                                                exact_size);
297                 } else if (ABS (info->paper_width  - paper_height) <= height_tolerance &&
298                            ABS (info->paper_height - paper_width) <= width_tolerance) {
299                         /* Note to translators: first placeholder is the paper name (eg.
300                          * A4), second placeholder is the paper size (eg. 297x210 mm) */
301                         str = g_strdup_printf ( _("%s, Landscape (%s)"),
302                                                 gtk_paper_size_get_display_name (size),
303                                                 exact_size);
304                 }
305         }
306
307         g_list_foreach (paper_sizes, (GFunc) gtk_paper_size_free, NULL);
308         g_list_free (paper_sizes);
309
310         if (str != NULL) {
311                 g_free (exact_size);
312                 return str;
313         }
314         
315         return exact_size;
316 }
317
318 void
319 ev_properties_view_set_info (EvPropertiesView *properties, const EvDocumentInfo *info)
320 {
321         GtkWidget *table;
322         gchar     *text;
323         gint       row = 0;
324
325         table = properties->table;
326
327         if (info->fields_mask & EV_DOCUMENT_INFO_TITLE) {
328                 set_property (GTK_TABLE (table), TITLE_PROPERTY, info->title, &row);
329         }
330         set_property (GTK_TABLE (table), URI_PROPERTY, properties->uri, &row);
331         if (info->fields_mask & EV_DOCUMENT_INFO_SUBJECT) {
332                 set_property (GTK_TABLE (table), SUBJECT_PROPERTY, info->subject, &row);
333         }
334         if (info->fields_mask & EV_DOCUMENT_INFO_AUTHOR) {
335                 set_property (GTK_TABLE (table), AUTHOR_PROPERTY, info->author, &row);
336         }
337         if (info->fields_mask & EV_DOCUMENT_INFO_KEYWORDS) {
338                 set_property (GTK_TABLE (table), KEYWORDS_PROPERTY, info->keywords, &row);
339         }
340         if (info->fields_mask & EV_DOCUMENT_INFO_PRODUCER) {
341                 set_property (GTK_TABLE (table), PRODUCER_PROPERTY, info->producer, &row);
342         }
343         if (info->fields_mask & EV_DOCUMENT_INFO_CREATOR) {
344                 set_property (GTK_TABLE (table), CREATOR_PROPERTY, info->creator, &row);
345         }
346         if (info->fields_mask & EV_DOCUMENT_INFO_CREATION_DATE) {
347                 text = ev_properties_view_format_date (info->creation_date);
348                 set_property (GTK_TABLE (table), CREATION_DATE_PROPERTY, text, &row);
349                 g_free (text);
350         }
351         if (info->fields_mask & EV_DOCUMENT_INFO_MOD_DATE) {
352                 text = ev_properties_view_format_date (info->modified_date);
353                 set_property (GTK_TABLE (table), MOD_DATE_PROPERTY, text, &row);
354                 g_free (text);
355         }
356         if (info->fields_mask & EV_DOCUMENT_INFO_FORMAT) {
357                 set_property (GTK_TABLE (table), FORMAT_PROPERTY, info->format, &row);
358         }
359         if (info->fields_mask & EV_DOCUMENT_INFO_N_PAGES) {
360                 text = g_strdup_printf ("%d", info->n_pages);
361                 set_property (GTK_TABLE (table), N_PAGES_PROPERTY, text, &row);
362                 g_free (text);
363         }
364         if (info->fields_mask & EV_DOCUMENT_INFO_LINEARIZED) {
365                 set_property (GTK_TABLE (table), LINEARIZED_PROPERTY, info->linearized, &row);
366         }
367         if (info->fields_mask & EV_DOCUMENT_INFO_SECURITY) {
368                 set_property (GTK_TABLE (table), SECURITY_PROPERTY, info->security, &row);
369         }
370         if (info->fields_mask & EV_DOCUMENT_INFO_PAPER_SIZE) {
371                 text = ev_regular_paper_size (info);
372                 set_property (GTK_TABLE (table), PAPER_SIZE_PROPERTY, text, &row);
373                 g_free (text);
374         }
375 }
376
377 static void
378 ev_properties_view_init (EvPropertiesView *properties)
379 {
380         properties->table = gtk_table_new (13, 2, FALSE);
381         gtk_table_set_col_spacings (GTK_TABLE (properties->table), 12);
382         gtk_table_set_row_spacings (GTK_TABLE (properties->table), 6);
383         gtk_container_set_border_width (GTK_CONTAINER (properties->table), 12);
384         gtk_box_pack_start (GTK_BOX (properties), properties->table, 
385                             TRUE, TRUE, 0);
386         gtk_widget_show (properties->table);
387 }
388
389 void
390 ev_properties_view_register_type (GTypeModule *module)
391 {
392         ev_properties_view_get_type ();
393 }
394
395 GtkWidget *
396 ev_properties_view_new (const gchar *uri)
397 {
398         EvPropertiesView *properties;
399
400         properties = g_object_new (EV_TYPE_PROPERTIES, NULL);
401         properties->uri = g_strdup (uri);
402
403         return GTK_WIDGET (properties);
404 }