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