]> www.fi.muni.cz Git - evince.git/blob - shell/main.c
187a794c45d9eb76d195483eb5b6c5935f8e6b64
[evince.git] / shell / main.c
1 /*
2  *  Copyright (C) 2004 Marco Pesenti Gritti
3  *
4  *  This program is free software; you can redistribute it and/or modify
5  *  it under the terms of the GNU General Public License as published by
6  *  the Free Software Foundation; either version 2, or (at your option)
7  *  any later version.
8  *
9  *  This program is distributed in the hope that it will be useful,
10  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
11  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  *  GNU General Public License for more details.
13  *
14  *  You should have received a copy of the GNU General Public License
15  *  along with this program; if not, write to the Free Software
16  *  Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17  *
18  */
19
20 #include "config.h"
21
22 #include <stdlib.h>
23 #include <string.h>
24
25 #include <glib/gstdio.h>
26 #include <glib/gi18n.h>
27 #include <gtk/gtk.h>
28
29 #include "ev-application.h"
30 #include "ev-debug.h"
31 #include "ev-init.h"
32 #include "ev-file-helpers.h"
33 #include "ev-stock-icons.h"
34 #include "ev-metadata.h"
35
36 #ifdef WITH_SMCLIENT
37 #include "eggsmclient.h"
38 #ifdef GDK_WINDOWING_X11
39 #include "eggdesktopfile.h"
40 #endif
41 #endif /* WITH_SMCLIENT */
42
43 #ifdef G_OS_WIN32
44 #ifdef DATADIR
45 #undef DATADIR
46 #endif
47 #include <io.h>
48 #include <conio.h>
49 #if !(_WIN32_WINNT >= 0x0500)
50 #error "_WIN32_WINNT must be defined >= 0x0500"
51 #endif
52 #include <windows.h>
53 #endif
54
55 static gchar   *ev_page_label;
56 static gchar   *ev_find_string;
57 static gint     ev_page_index = 0;
58 static gboolean preview_mode = FALSE;
59 static gboolean fullscreen_mode = FALSE;
60 static gboolean presentation_mode = FALSE;
61 static gboolean unlink_temp_file = FALSE;
62 static gchar   *print_settings;
63 static const char **file_arguments = NULL;
64
65
66 static gboolean
67 option_version_cb (const gchar *option_name,
68                    const gchar *value,
69                    gpointer     data,
70                    GError     **error)
71 {
72   g_print ("%s %s\n", _("GNOME Document Viewer"), VERSION);
73
74   exit (0);
75   return FALSE;
76 }
77
78 static const GOptionEntry goption_options[] =
79 {
80         { "page-label", 'p', 0, G_OPTION_ARG_STRING, &ev_page_label, N_("The page label of the document to display."), N_("PAGE")},
81         { "page-index", 'i', 0, G_OPTION_ARG_INT, &ev_page_index, N_("The page number of the document to display."), N_("NUMBER")},
82         { "fullscreen", 'f', 0, G_OPTION_ARG_NONE, &fullscreen_mode, N_("Run evince in fullscreen mode"), NULL },
83         { "presentation", 's', 0, G_OPTION_ARG_NONE, &presentation_mode, N_("Run evince in presentation mode"), NULL },
84         { "preview", 'w', 0, G_OPTION_ARG_NONE, &preview_mode, N_("Run evince as a previewer"), NULL },
85         { "find", 'l', 0, G_OPTION_ARG_STRING, &ev_find_string, N_("The word or phrase to find in the document"), N_("STRING")},
86         { "unlink-tempfile", 'u', G_OPTION_FLAG_HIDDEN, G_OPTION_ARG_NONE, &unlink_temp_file, NULL, NULL },
87         { "print-settings", 't', G_OPTION_FLAG_HIDDEN, G_OPTION_ARG_FILENAME, &print_settings, NULL, NULL },
88         { "version", 0, G_OPTION_FLAG_NO_ARG | G_OPTION_FLAG_HIDDEN, G_OPTION_ARG_CALLBACK, option_version_cb, NULL, NULL },
89         { G_OPTION_REMAINING, 0, 0, G_OPTION_ARG_FILENAME_ARRAY, &file_arguments, NULL, N_("[FILEā€¦]") },
90         { NULL }
91 };
92
93 static gboolean
94 launch_previewer (void)
95 {
96         GString *cmd_str;
97         gchar   *cmd;
98         gint     argc;
99         gchar  **argv;
100         gboolean retval = FALSE;
101         GError  *error = NULL;
102
103         /* Rebuild the command line, ignoring options
104          * not supported by the previewer and taking only
105          * the first path given
106          */
107         cmd_str = g_string_new ("evince-previewer");
108                 
109         if (print_settings) {
110                 gchar *quoted;
111
112                 quoted = g_shell_quote (print_settings);
113                 g_string_append_printf (cmd_str, " --print-settings %s", quoted);
114                 g_free (quoted);
115         }
116
117         if (unlink_temp_file)
118                 g_string_append (cmd_str, " --unlink-tempfile");
119
120         if (file_arguments) {
121                 gchar *quoted;
122                 
123                 quoted = g_shell_quote (file_arguments[0]);
124                 g_string_append_printf (cmd_str, " %s", quoted);
125                 g_free (quoted);
126         }
127
128         cmd = g_string_free (cmd_str, FALSE);
129         g_shell_parse_argv (cmd, &argc, &argv, &error);
130         g_free (cmd);
131         
132         if (!error) {
133                 retval = gdk_spawn_on_screen (gdk_screen_get_default (),
134                                               NULL, argv, NULL,
135                                               G_SPAWN_SEARCH_PATH,
136                                               NULL, NULL, NULL,
137                                               &error);
138                 g_strfreev (argv);
139         }
140
141         if (error) {
142                 g_warning ("Error launching previewer: %s\n", error->message);
143                 g_error_free (error);
144         }
145
146         return retval;
147 }
148
149 static gchar *
150 get_label_from_filename (const gchar *filename)
151 {
152         GFile   *file;
153         gchar   *label;
154         gboolean exists;
155
156         label = g_strrstr (filename, "#");
157         if (!label)
158                 return NULL;
159
160         /* Filename contains a #, check
161          * whether it's part of the path
162          * or a label
163          */
164         file = g_file_new_for_commandline_arg (filename);
165         exists = g_file_query_exists (file, NULL);
166         g_object_unref (file);
167
168         return exists ? NULL : label;
169 }
170
171 static void
172 load_files (const char **files)
173 {
174         GdkScreen       *screen = gdk_screen_get_default ();
175         EvWindowRunMode  mode = EV_WINDOW_MODE_NORMAL;
176         gint             i;
177         EvLinkDest      *global_dest = NULL;
178
179         if (!files) {
180                 if (!ev_application_has_window (EV_APP))
181                         ev_application_open_window (EV_APP, screen, GDK_CURRENT_TIME);
182                 return;
183         }
184
185         if (ev_page_label)
186                 global_dest = ev_link_dest_new_page_label (ev_page_label);
187         else if (ev_page_index)
188                 global_dest = ev_link_dest_new_page (MAX (0, ev_page_index - 1));
189
190         if (fullscreen_mode)
191                 mode = EV_WINDOW_MODE_FULLSCREEN;
192         else if (presentation_mode)
193                 mode = EV_WINDOW_MODE_PRESENTATION;
194
195         for (i = 0; files[i]; i++) {
196                 const gchar *filename;
197                 gchar       *uri;
198                 gchar       *label;
199                 GFile       *file;
200                 EvLinkDest  *dest = NULL;
201                 const gchar *app_uri;
202
203                 filename = files[i];
204                 label = get_label_from_filename (filename);
205                 if (label) {
206                         *label = 0;
207                         label++;
208                         dest = ev_link_dest_new_page_label (label);
209                 } else if (global_dest) {
210                         dest = g_object_ref (global_dest);
211                 }
212
213                 file = g_file_new_for_commandline_arg (filename);
214                 uri = g_file_get_uri (file);
215                 g_object_unref (file);
216
217                 app_uri = ev_application_get_uri (EV_APP);
218                 if (app_uri && strcmp (app_uri, uri) == 0) {
219                         g_free (uri);
220                         continue;
221                 }
222
223
224
225                 ev_application_open_uri_at_dest (EV_APP, uri, screen, dest,
226                                                  mode, ev_find_string,
227                                                  GDK_CURRENT_TIME);
228
229                 if (dest)
230                         g_object_unref (dest);
231                 g_free (uri);
232         }
233 }
234
235 int
236 main (int argc, char *argv[])
237 {
238         GOptionContext *context;
239         GError         *error = NULL;
240
241 #ifdef WITH_GCONF
242         /* Remove when porting is done */
243         g_setenv ("GSETTINGS_BACKEND", "gconf", FALSE);
244 #endif
245
246 #ifdef G_OS_WIN32
247
248     if (fileno (stdout) != -1 &&
249           _get_osfhandle (fileno (stdout)) != -1)
250         {
251           /* stdout is fine, presumably redirected to a file or pipe */
252         }
253     else
254     {
255           typedef BOOL (* WINAPI AttachConsole_t) (DWORD);
256
257           AttachConsole_t p_AttachConsole =
258             (AttachConsole_t) GetProcAddress (GetModuleHandle ("kernel32.dll"), "AttachConsole");
259
260           if (p_AttachConsole != NULL && p_AttachConsole (ATTACH_PARENT_PROCESS))
261       {
262               freopen ("CONOUT$", "w", stdout);
263               dup2 (fileno (stdout), 1);
264               freopen ("CONOUT$", "w", stderr);
265               dup2 (fileno (stderr), 2);
266
267       }
268         }
269 #endif
270
271         /* Init glib threads asap */
272         if (!g_thread_supported ())
273                 g_thread_init (NULL);
274
275 #ifdef ENABLE_NLS
276         /* Initialize the i18n stuff */
277         bindtextdomain (GETTEXT_PACKAGE, GNOMELOCALEDIR);
278         bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8");
279         textdomain (GETTEXT_PACKAGE);
280 #endif
281
282         context = g_option_context_new (N_("GNOME Document Viewer"));
283         g_option_context_set_translation_domain(context, GETTEXT_PACKAGE);
284         g_option_context_add_main_entries (context, goption_options, GETTEXT_PACKAGE);
285
286 #ifdef WITH_SMCLIENT
287         g_option_context_add_group (context, egg_sm_client_get_option_group ());
288 #endif
289
290         g_option_context_add_group (context, gtk_get_option_group (TRUE));
291
292         if (!g_option_context_parse (context, &argc, &argv, &error)) {
293                 g_printerr ("Cannot parse arguments: %s\n", error->message);
294                 g_error_free (error);
295                 g_option_context_free (context);
296
297                 return 1;
298         }
299         g_option_context_free (context);
300
301         if (preview_mode) {
302                 gboolean retval;
303                 
304                 retval = launch_previewer ();
305                 
306                 return retval ? 0 : 1;
307         }
308
309         if (!ev_init ())
310                 return 1;
311
312         ev_stock_icons_init ();
313
314 #if defined(WITH_SMCLIENT) && defined(GDK_WINDOWING_X11)
315         egg_set_desktop_file (GNOMEDATADIR "/applications/evince.desktop");
316 #else
317         /* Manually set name and icon */
318         g_set_application_name (_("Document Viewer"));
319         gtk_window_set_default_icon_name ("evince");
320 #endif /* WITH_SMCLIENT && GDK_WINDOWING_X11 */
321
322         ev_application_load_session (EV_APP);
323         load_files (file_arguments);
324         if (ev_application_has_window (EV_APP)) {
325                 /* Change directory so we don't prevent unmounting in case the initial cwd
326                  * is on an external device (see bug #575436)
327                  */
328                 g_chdir (g_get_home_dir ());
329
330                 gtk_main ();
331         }
332
333         ev_shutdown ();
334         ev_stock_icons_shutdown ();
335
336         return 0;
337 }