Handle error of mc main loop.
[midnight-commander.git] / src / args.c
blob2a99a477087e23ff1ee50337c1727fa21942a286
1 /*
2 Handle command line arguments.
4 Copyright (C) 2009, 2011
5 The Free Software Foundation, Inc.
7 Written by:
8 Slava Zanko <slavazanko@gmail.com>, 2009.
10 This file is part of the Midnight Commander.
12 The Midnight Commander is free software: you can redistribute it
13 and/or modify it under the terms of the GNU General Public License as
14 published by the Free Software Foundation, either version 3 of the License,
15 or (at your option) any later version.
17 The Midnight Commander is distributed in the hope that it will be useful,
18 but WITHOUT ANY WARRANTY; without even the implied warranty of
19 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 GNU General Public License for more details.
22 You should have received a copy of the GNU General Public License
23 along with this program. If not, see <http://www.gnu.org/licenses/>.
26 #include <config.h>
27 #include <stdlib.h>
28 #include <stdio.h>
30 #include "lib/global.h"
31 #include "lib/tty/tty.h"
32 #include "lib/strutil.h"
33 #include "lib/vfs/vfs.h"
34 #include "lib/util.h" /* x_basename() */
36 #ifdef ENABLE_VFS_SMB
37 #include "src/vfs/smbfs/smbfs.h" /* smbfs_set_debugf() */
38 #endif
40 #include "src/main.h"
41 #include "src/textconf.h"
43 #include "src/args.h"
45 /*** external variables **************************************************************************/
47 /*** global variables ****************************************************************************/
49 /* If true, show version info and exit */
50 gboolean mc_args__show_version = FALSE;
52 /* If true, assume we are running on an xterm terminal */
53 gboolean mc_args__force_xterm = FALSE;
55 gboolean mc_args__nomouse = FALSE;
57 /* Force colors, only used by Slang */
58 gboolean mc_args__force_colors = FALSE;
60 /* Don't load keymap form file and use default one */
61 gboolean mc_args__nokeymap = FALSE;
63 /* Line to start the editor on */
64 int mc_args__edit_start_line = 0;
66 char *mc_args__last_wd_file = NULL;
68 /* when enabled NETCODE, use folowing file as logfile */
69 char *mc_args__netfs_logfile = NULL;
71 /* keymap file */
72 char *mc_args__keymap_file = NULL;
74 /* Debug level */
75 int mc_args__debug_level = 0;
77 /*** file scope macro definitions ****************************************************************/
79 /*** file scope type declarations ****************************************************************/
81 /*** file scope variables ************************************************************************/
83 /* forward declarations */
84 static gboolean parse_mc_e_argument (const gchar * option_name, const gchar * value,
85 gpointer data, GError ** error);
86 static gboolean parse_mc_v_argument (const gchar * option_name, const gchar * value,
87 gpointer data, GError ** error);
89 static GOptionContext *context;
91 static gboolean mc_args__nouse_subshell = FALSE;
92 static gboolean mc_args__show_datadirs = FALSE;
93 static gboolean mc_args__show_datadirs_extended = FALSE;
94 static gboolean mc_args__show_configure_opts = FALSE;
96 static GOptionGroup *main_group;
98 static const GOptionEntry argument_main_table[] = {
99 /* *INDENT-OFF* */
100 /* generic options */
102 "version", 'V', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_NONE,
103 &mc_args__show_version,
104 N_("Displays the current version"),
105 NULL
108 /* options for wrappers */
110 "datadir", 'f', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_NONE,
111 &mc_args__show_datadirs,
112 N_("Print data directory"),
113 NULL
116 /* show extended information about used data directories */
118 "datadir-info", 'F', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_NONE,
119 &mc_args__show_datadirs_extended,
120 N_("Print extended info about used data directories"),
121 NULL
124 /* show configure options */
126 "configure-options", '\0', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_NONE,
127 &mc_args__show_configure_opts,
128 N_("Print configure options"),
129 NULL
133 "printwd", 'P', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_STRING,
134 &mc_args__last_wd_file,
135 N_("Print last working directory to specified file"),
136 "<file>"
139 #ifdef HAVE_SUBSHELL_SUPPORT
141 "subshell", 'U', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_NONE,
142 &mc_global.tty.use_subshell,
143 N_("Enables subshell support (default)"),
144 NULL
148 "nosubshell", 'u', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_NONE,
149 &mc_args__nouse_subshell,
150 N_("Disables subshell support"),
151 NULL
153 #endif
155 /* debug options */
156 #ifdef ENABLE_VFS_FTP
158 "ftplog", 'l', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_STRING,
159 &mc_args__netfs_logfile,
160 N_("Log ftp dialog to specified file"),
161 "<file>"
163 #endif /* ENABLE_VFS_FTP */
164 #ifdef ENABLE_VFS_SMB
166 "debuglevel", 'D', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_INT,
167 &mc_args__debug_level,
168 N_("Set debug level"),
169 "<integer>"
171 #endif /* ENABLE_VFS_SMB */
173 /* single file operations */
175 "view", 'v', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_CALLBACK,
176 parse_mc_v_argument,
177 N_("Launches the file viewer on a file"),
178 "<file>"
182 "edit", 'e', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_CALLBACK,
183 parse_mc_e_argument,
184 N_("Edits one file"),
185 "<file>"},
188 NULL, '\0', 0, 0, NULL, NULL, NULL /* Complete struct initialization */
190 /* *INDENT-ON* */
193 GOptionGroup *terminal_group;
194 #define ARGS_TERM_OPTIONS 0
195 static const GOptionEntry argument_terminal_table[] = {
196 /* *INDENT-OFF* */
197 /* terminal options */
199 "xterm", 'x', ARGS_TERM_OPTIONS, G_OPTION_ARG_NONE,
200 &mc_args__force_xterm,
201 N_("Forces xterm features"),
202 NULL
206 "no-x11", 'X', ARGS_TERM_OPTIONS, G_OPTION_ARG_NONE,
207 &mc_global.tty.disable_x11,
208 N_("Disable X11 support"),
209 NULL
213 "oldmouse", 'g', ARGS_TERM_OPTIONS, G_OPTION_ARG_NONE,
214 &mc_global.tty.old_mouse,
215 N_("Tries to use an old highlight mouse tracking"),
216 NULL
220 "nomouse", 'd', ARGS_TERM_OPTIONS, G_OPTION_ARG_NONE,
221 &mc_args__nomouse,
222 N_("Disable mouse support in text version"),
223 NULL
226 #ifdef HAVE_SLANG
228 "termcap", 't', ARGS_TERM_OPTIONS, G_OPTION_ARG_NONE,
229 &SLtt_Try_Termcap,
230 N_("Tries to use termcap instead of terminfo"),
231 NULL
233 #endif
236 "slow", 's', ARGS_TERM_OPTIONS, G_OPTION_ARG_NONE,
237 &mc_global.tty.slow_terminal,
238 N_("To run on slow terminals"),
239 NULL
243 "stickchars", 'a', ARGS_TERM_OPTIONS, G_OPTION_ARG_NONE,
244 &mc_global.tty.ugly_line_drawing,
245 N_("Use stickchars to draw"),
246 NULL
250 "resetsoft", 'k', ARGS_TERM_OPTIONS, G_OPTION_ARG_NONE,
251 &reset_hp_softkeys,
252 N_("Resets soft keys on HP terminals"),
253 NULL
257 "keymap", 'K', ARGS_TERM_OPTIONS, G_OPTION_ARG_STRING,
258 &mc_args__keymap_file,
259 N_("Load definitions of key bindings from specified file"),
260 "<file>"
264 "nokeymap", '\0', ARGS_TERM_OPTIONS, G_OPTION_ARG_NONE,
265 &mc_args__nokeymap,
266 N_("Don't load definitions of key bindings from file, use defaults"),
267 NULL
271 NULL, '\0', 0, 0, NULL, NULL, NULL /* Complete struct initialization */
273 /* *INDENT-ON* */
276 #undef ARGS_TERM_OPTIONS
278 GOptionGroup *color_group;
279 #define ARGS_COLOR_OPTIONS 0
280 /* #define ARGS_COLOR_OPTIONS G_OPTION_FLAG_IN_MAIN */
281 static const GOptionEntry argument_color_table[] = {
282 /* *INDENT-OFF* */
283 /* color options */
285 "nocolor", 'b', ARGS_COLOR_OPTIONS, G_OPTION_ARG_NONE,
286 &mc_global.tty.disable_colors,
287 N_("Requests to run in black and white"),
288 NULL
292 "color", 'c', ARGS_COLOR_OPTIONS, G_OPTION_ARG_NONE,
293 &mc_args__force_colors,
294 N_("Request to run in color mode"),
295 NULL
299 "colors", 'C', ARGS_COLOR_OPTIONS, G_OPTION_ARG_STRING,
300 &mc_global.tty.command_line_colors,
301 N_("Specifies a color configuration"),
302 "<string>"
306 "skin", 'S', ARGS_COLOR_OPTIONS, G_OPTION_ARG_STRING,
307 &mc_global.skin,
308 N_("Show mc with specified skin"),
309 "<string>"
313 NULL, '\0', 0, 0, NULL, NULL, NULL /* Complete struct initialization */
315 /* *INDENT-ON* */
318 #undef ARGS_COLOR_OPTIONS
320 static gchar *mc_args__loc__colors_string = NULL;
321 static gchar *mc_args__loc__footer_string = NULL;
322 static gchar *mc_args__loc__header_string = NULL;
323 static gchar *mc_args__loc__usage_string = NULL;
325 /*** file scope functions ************************************************************************/
327 /* --------------------------------------------------------------------------------------------- */
328 static void
329 mc_args_clean_temp_help_strings (void)
331 g_free (mc_args__loc__colors_string);
332 mc_args__loc__colors_string = NULL;
334 g_free (mc_args__loc__footer_string);
335 mc_args__loc__footer_string = NULL;
337 g_free (mc_args__loc__header_string);
338 mc_args__loc__header_string = NULL;
340 g_free (mc_args__loc__usage_string);
341 mc_args__loc__usage_string = NULL;
344 /* --------------------------------------------------------------------------------------------- */
346 static GOptionGroup *
347 mc_args_new_color_group (void)
349 /* *INDENT-OFF* */
350 /* FIXME: to preserve translations, lines should be split. */
351 mc_args__loc__colors_string = g_strdup_printf ("%s\n%s",
352 /* TRANSLATORS: don't translate keywords */
353 _("--colors KEYWORD={FORE},{BACK},{ATTR}:KEYWORD2=...\n\n"
354 "{FORE}, {BACK} and {ATTR} can be omitted, and the default will be used\n"
355 "\n Keywords:\n"
356 " Global: errors, disabled, reverse, gauge, header\n"
357 " input, inputmark, inputunchanged, commandlinemark\n"
358 " bbarhotkey, bbarbutton, statusbar\n"
359 " File display: normal, selected, marked, markselect\n"
360 " Dialog boxes: dnormal, dfocus, dhotnormal, dhotfocus, errdhotnormal,\n"
361 " errdhotfocus\n"
362 " Menus: menunormal, menuhot, menusel, menuhotsel, menuinactive\n"
363 " Popup menus: pmenunormal, pmenusel, pmenutitle\n"
364 " Editor: editnormal, editbold, editmarked, editwhitespace,\n"
365 " editlinestate\n"
366 " Viewer: viewbold, viewunderline, viewselected\n"
367 " Help: helpnormal, helpitalic, helpbold, helplink, helpslink\n"),
368 /* TRANSLATORS: don't translate color names and attributes */
369 _("Standard Colors:\n"
370 " black, gray, red, brightred, green, brightgreen, brown,\n"
371 " yellow, blue, brightblue, magenta, brightmagenta, cyan,\n"
372 " brightcyan, lightgray and white\n\n"
373 "Extended colors, when 256 colors are available:\n"
374 " color16 to color255, or rgb000 to rgb555 and gray0 to gray23\n\n"
375 "Attributes:\n"
376 " bold, underline, reverse, blink; append more with '+'\n")
378 /* *INDENT-ON* */
380 return g_option_group_new ("color", mc_args__loc__colors_string,
381 _("Color options"), NULL, NULL);
385 /* --------------------------------------------------------------------------------------------- */
387 static gchar *
388 mc_args_add_usage_info (void)
390 mc_args__loc__usage_string = g_strdup_printf ("[%s] %s\n %s - %s\n",
391 _("+number"),
392 _("[this_dir] [other_panel_dir]"),
393 _("+number"),
395 ("Set initial line number for the internal editor"));
396 return mc_args__loc__usage_string;
399 /* --------------------------------------------------------------------------------------------- */
401 static void
402 mc_args_add_extended_info_to_help (void)
404 mc_args__loc__footer_string = g_strdup_printf ("%s",
406 ("\n"
407 "Please send any bug reports (including the output of `mc -V')\n"
408 "as tickets at www.midnight-commander.org\n"));
409 mc_args__loc__header_string = g_strdup_printf (_("GNU Midnight Commander %s\n"), VERSION);
411 #if GLIB_CHECK_VERSION(2,12,0)
412 g_option_context_set_description (context, mc_args__loc__footer_string);
413 g_option_context_set_summary (context, mc_args__loc__header_string);
414 #endif
417 /* --------------------------------------------------------------------------------------------- */
419 static gboolean
420 mc_setup_by_args (int argc, char *argv[], GError ** error)
422 const char *base;
423 char *tmp;
425 #ifdef ENABLE_VFS_SMB
426 if (mc_args__debug_level != 0)
427 smbfs_set_debug (mc_args__debug_level);
428 #endif /* ENABLE_VFS_SMB */
430 if (mc_args__netfs_logfile != NULL)
432 #ifdef ENABLE_VFS_FTP
433 mc_setctl ("ftp://", VFS_SETCTL_LOGFILE, (void *) mc_args__netfs_logfile);
434 #endif /* ENABLE_VFS_FTP */
435 #ifdef ENABLE_VFS_SMB
436 mc_setctl ("smb://", VFS_SETCTL_LOGFILE, (void *) mc_args__netfs_logfile);
437 #endif /* ENABLE_VFS_SMB */
440 base = x_basename (argv[0]);
441 tmp = (argc > 0) ? argv[1] : NULL;
443 if (strncmp (base, "mce", 3) == 0 || strcmp (base, "vi") == 0)
445 /* mce* or vi is link to mc */
447 mc_run_param0 = g_strdup ("");
448 if (tmp != NULL)
451 * Check for filename:lineno, followed by an optional colon.
452 * This format is used by many programs (especially compilers)
453 * in error messages and warnings. It is supported so that
454 * users can quickly copy and paste file locations.
456 char *end, *p;
458 end = tmp + strlen (tmp);
459 p = end;
461 if (p > tmp && p[-1] == ':')
462 p--;
463 while (p > tmp && g_ascii_isdigit ((gchar) p[-1]))
464 p--;
465 if (tmp < p && p < end && p[-1] == ':')
467 char *fname;
468 struct stat st;
470 fname = g_strndup (tmp, p - 1 - tmp);
472 * Check that the file before the colon actually exists.
473 * If it doesn't exist, revert to the old behavior.
475 if (mc_stat (tmp, &st) == -1 && mc_stat (fname, &st) != -1)
477 mc_run_param0 = fname;
478 mc_args__edit_start_line = atoi (p);
480 else
482 g_free (fname);
483 goto try_plus_filename;
486 else
488 try_plus_filename:
489 if (*tmp == '+' && g_ascii_isdigit ((gchar) tmp[1]))
491 int start_line;
493 start_line = atoi (tmp);
496 * If start_line is zero, position the cursor at the
497 * beginning of the file as other editors (vi, nano)
499 if (start_line == 0)
500 start_line++;
502 if (start_line > 0)
504 char *file;
506 file = (argc > 1) ? argv[2] : NULL;
507 if (file != NULL)
509 tmp = file;
510 mc_args__edit_start_line = start_line;
514 mc_run_param0 = g_strdup (tmp);
517 mc_global.mc_run_mode = MC_RUN_EDITOR;
519 else if (strncmp (base, "mcv", 3) == 0 || strcmp (base, "view") == 0)
521 /* mcv* or view is link to mc */
523 if (tmp != NULL)
524 mc_run_param0 = g_strdup (tmp);
525 else
527 *error = g_error_new (MC_ERROR, 0, "%s\n", _("No arguments given to the viewer."));
528 return FALSE;
530 mc_global.mc_run_mode = MC_RUN_VIEWER;
532 #ifdef USE_DIFF_VIEW
533 else if (strncmp (base, "mcd", 3) == 0 || strcmp (base, "diff") == 0)
535 /* mcd* or diff is link to mc */
537 if (argc < 3)
539 *error = g_error_new (MC_ERROR, 0, "%s\n",
540 _("Two files are required to evoke the diffviewer."));
541 return FALSE;
544 if (tmp != NULL)
546 mc_run_param0 = g_strdup (tmp);
547 tmp = (argc > 1) ? argv[2] : NULL;
548 if (tmp != NULL)
549 mc_run_param1 = g_strdup (tmp);
550 mc_global.mc_run_mode = MC_RUN_DIFFVIEWER;
553 #endif /* USE_DIFF_VIEW */
554 else
556 /* MC is run as mc */
558 switch (mc_global.mc_run_mode)
560 case MC_RUN_EDITOR:
561 case MC_RUN_VIEWER:
562 /* mc_run_param0 is set up in parse_mc_e_argument() and parse_mc_v_argument() */
563 break;
565 case MC_RUN_DIFFVIEWER:
566 /* not implemented yet */
567 break;
569 case MC_RUN_FULL:
570 default:
571 /* sets the current dir and the other dir */
572 if (tmp != NULL)
574 mc_run_param0 = g_strdup (tmp);
575 tmp = (argc > 1) ? argv[2] : NULL;
576 if (tmp != NULL)
577 mc_run_param1 = g_strdup (tmp);
579 mc_global.mc_run_mode = MC_RUN_FULL;
580 break;
584 return TRUE;
587 /* --------------------------------------------------------------------------------------------- */
589 static gboolean
590 mc_args_process (int argc, char *argv[], GError ** error)
592 if (mc_args__show_version)
594 show_version ();
595 return FALSE;
597 if (mc_args__show_datadirs)
599 printf ("%s (%s)\n", mc_global.sysconfig_dir, mc_global.share_data_dir);
600 return FALSE;
603 if (mc_args__show_datadirs_extended)
605 show_datadirs_extended ();
606 return FALSE;
609 if (mc_args__show_configure_opts)
611 show_configure_options ();
612 return FALSE;
615 if (mc_args__force_colors)
616 mc_global.tty.disable_colors = FALSE;
618 #ifdef HAVE_SUBSHELL_SUPPORT
619 if (mc_args__nouse_subshell)
620 mc_global.tty.use_subshell = FALSE;
621 #endif /* HAVE_SUBSHELL_SUPPORT */
623 return mc_setup_by_args (argc, argv, error);
626 /* --------------------------------------------------------------------------------------------- */
628 static gchar *
629 mc_args__convert_help_to_syscharset (const gchar * charset, const gchar * error_message,
630 const gchar * help_str)
632 GString *buffer;
633 GIConv conv;
634 gchar *full_help_str;
636 buffer = g_string_new ("");
637 conv = g_iconv_open (charset, "UTF-8");
638 full_help_str = g_strdup_printf ("%s\n\n%s\n", error_message, help_str);
640 str_convert (conv, full_help_str, buffer);
642 g_free (full_help_str);
643 g_iconv_close (conv);
645 return g_string_free (buffer, FALSE);
648 /* --------------------------------------------------------------------------------------------- */
650 static gboolean
651 parse_mc_e_argument (const gchar * option_name, const gchar * value, gpointer data, GError ** error)
653 (void) option_name;
654 (void) data;
655 (void) error;
657 mc_global.mc_run_mode = MC_RUN_EDITOR;
658 mc_run_param0 = g_strdup (value);
660 return TRUE;
663 /* --------------------------------------------------------------------------------------------- */
665 static gboolean
666 parse_mc_v_argument (const gchar * option_name, const gchar * value, gpointer data, GError ** error)
668 (void) option_name;
669 (void) data;
670 (void) error;
672 mc_global.mc_run_mode = MC_RUN_VIEWER;
673 mc_run_param0 = g_strdup (value);
675 return TRUE;
678 /* --------------------------------------------------------------------------------------------- */
680 /*** public functions ****************************************************************************/
682 /* --------------------------------------------------------------------------------------------- */
684 gboolean
685 mc_args_handle (int argc, char **argv, const char *translation_domain, GError ** error)
687 const gchar *_system_codepage = str_detect_termencoding ();
689 #ifdef ENABLE_NLS
690 if (!str_isutf8 (_system_codepage))
691 bind_textdomain_codeset ("mc", "UTF-8");
692 #endif
694 context = g_option_context_new (mc_args_add_usage_info ());
696 g_option_context_set_ignore_unknown_options (context, FALSE);
698 mc_args_add_extended_info_to_help ();
700 main_group = g_option_group_new ("main", _("Main options"), _("Main options"), NULL, NULL);
702 g_option_group_add_entries (main_group, argument_main_table);
703 g_option_context_set_main_group (context, main_group);
704 g_option_group_set_translation_domain (main_group, translation_domain);
706 terminal_group = g_option_group_new ("terminal", _("Terminal options"),
707 _("Terminal options"), NULL, NULL);
709 g_option_group_add_entries (terminal_group, argument_terminal_table);
710 g_option_context_add_group (context, terminal_group);
711 g_option_group_set_translation_domain (terminal_group, translation_domain);
713 color_group = mc_args_new_color_group ();
715 g_option_group_add_entries (color_group, argument_color_table);
716 g_option_context_add_group (context, color_group);
717 g_option_group_set_translation_domain (color_group, translation_domain);
719 if (!g_option_context_parse (context, &argc, &argv, error))
721 GError *error2 = NULL;
723 if (*error != NULL)
725 gchar *help_str;
727 #if GLIB_CHECK_VERSION(2,14,0)
728 help_str = g_option_context_get_help (context, TRUE, NULL);
729 #else
730 help_str = g_strdup ("");
731 #endif
732 if (str_isutf8 (_system_codepage))
733 error2 = g_error_new ((*error)->domain, (*error)->code, "%s\n\n%s\n",
734 (*error)->message, help_str);
735 else
737 gchar *full_help_str;
739 full_help_str =
740 mc_args__convert_help_to_syscharset (_system_codepage, (*error)->message,
741 help_str);
742 error2 = g_error_new ((*error)->domain, (*error)->code, "%s", full_help_str);
743 g_free (full_help_str);
746 g_free (help_str);
747 g_error_free (*error);
748 *error = error2;
750 else
751 *error = g_error_new (MC_ERROR, 0, "%s\n", _("Arguments parse error!"));
753 g_option_context_free (context);
754 mc_args_clean_temp_help_strings ();
755 return FALSE;
758 g_option_context_free (context);
759 mc_args_clean_temp_help_strings ();
761 #ifdef ENABLE_NLS
762 if (!str_isutf8 (_system_codepage))
763 bind_textdomain_codeset ("mc", _system_codepage);
764 #endif
766 return mc_args_process (argc, argv, error);
769 /* --------------------------------------------------------------------------------------------- */