1408 lines
34 KiB
C
1408 lines
34 KiB
C
/*
|
|
* Copyright © 2001, 2002 Havoc Pennington
|
|
* Copyright © 2002 Red Hat, Inc.
|
|
* Copyright © 2002 Sun Microsystems
|
|
* Copyright © 2003 Mariano Suarez-Alvarez
|
|
* Copyright © 2008 Christian Persch
|
|
*
|
|
* Mate-terminal 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.
|
|
*
|
|
* Mate-terminal 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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include <errno.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
|
|
#include <glib.h>
|
|
|
|
#include "terminal-options.h"
|
|
#include "terminal-screen.h"
|
|
#include "terminal-app.h"
|
|
#include "terminal-intl.h"
|
|
#include "terminal-util.h"
|
|
|
|
static GOptionContext *get_goption_context (TerminalOptions *options);
|
|
|
|
static InitialTab*
|
|
initial_tab_new (const char *profile,
|
|
gboolean is_id)
|
|
{
|
|
InitialTab *it;
|
|
|
|
it = g_slice_new (InitialTab);
|
|
|
|
it->profile = g_strdup (profile);
|
|
it->profile_is_id = is_id;
|
|
it->exec_argv = NULL;
|
|
it->title = NULL;
|
|
it->working_dir = NULL;
|
|
it->zoom = 1.0;
|
|
it->zoom_set = FALSE;
|
|
it->active = FALSE;
|
|
it->attach_window = FALSE;
|
|
|
|
return it;
|
|
}
|
|
|
|
static void
|
|
initial_tab_free (InitialTab *it)
|
|
{
|
|
g_free (it->profile);
|
|
g_strfreev (it->exec_argv);
|
|
g_free (it->title);
|
|
g_free (it->working_dir);
|
|
g_slice_free (InitialTab, it);
|
|
}
|
|
|
|
static InitialWindow*
|
|
initial_window_new (guint source_tag)
|
|
{
|
|
InitialWindow *iw;
|
|
|
|
iw = g_slice_new0 (InitialWindow);
|
|
iw->source_tag = source_tag;
|
|
|
|
return iw;
|
|
}
|
|
|
|
static void
|
|
initial_window_free (InitialWindow *iw)
|
|
{
|
|
g_list_foreach (iw->tabs, (GFunc) initial_tab_free, NULL);
|
|
g_list_free (iw->tabs);
|
|
g_free (iw->geometry);
|
|
g_free (iw->role);
|
|
g_slice_free (InitialWindow, iw);
|
|
}
|
|
|
|
static void
|
|
apply_defaults (TerminalOptions *options,
|
|
InitialWindow *iw)
|
|
{
|
|
if (options->default_role)
|
|
{
|
|
iw->role = options->default_role;
|
|
options->default_role = NULL;
|
|
}
|
|
|
|
if (iw->geometry == NULL)
|
|
iw->geometry = g_strdup (options->default_geometry);
|
|
|
|
if (options->default_window_menubar_forced)
|
|
{
|
|
iw->force_menubar_state = TRUE;
|
|
iw->menubar_state = options->default_window_menubar_state;
|
|
|
|
options->default_window_menubar_forced = FALSE;
|
|
}
|
|
|
|
iw->start_fullscreen |= options->default_fullscreen;
|
|
iw->start_maximized |= options->default_maximize;
|
|
}
|
|
|
|
static InitialWindow*
|
|
ensure_top_window (TerminalOptions *options)
|
|
{
|
|
InitialWindow *iw;
|
|
|
|
if (options->initial_windows == NULL)
|
|
{
|
|
iw = initial_window_new (0);
|
|
iw->tabs = g_list_append (NULL, initial_tab_new (NULL, FALSE));
|
|
apply_defaults (options, iw);
|
|
|
|
options->initial_windows = g_list_append (options->initial_windows, iw);
|
|
}
|
|
else
|
|
{
|
|
iw = g_list_last (options->initial_windows)->data;
|
|
}
|
|
|
|
g_assert (iw->tabs);
|
|
|
|
return iw;
|
|
}
|
|
|
|
static InitialTab*
|
|
ensure_top_tab (TerminalOptions *options)
|
|
{
|
|
InitialWindow *iw;
|
|
InitialTab *it;
|
|
|
|
iw = ensure_top_window (options);
|
|
|
|
g_assert (iw->tabs);
|
|
|
|
it = g_list_last (iw->tabs)->data;
|
|
|
|
return it;
|
|
}
|
|
|
|
static InitialWindow*
|
|
add_new_window (TerminalOptions *options,
|
|
const char *profile,
|
|
gboolean is_id)
|
|
{
|
|
InitialWindow *iw;
|
|
|
|
iw = initial_window_new (0);
|
|
iw->tabs = g_list_prepend (NULL, initial_tab_new (profile, is_id));
|
|
apply_defaults (options, iw);
|
|
|
|
options->initial_windows = g_list_append (options->initial_windows, iw);
|
|
|
|
return iw;
|
|
}
|
|
|
|
/* handle deprecated command line options */
|
|
static gboolean
|
|
unsupported_option_callback (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
g_printerr (_("Option \"%s\" is no longer supported in this version of mate-terminal;"
|
|
" you might want to create a profile with the desired setting, and use"
|
|
" the new '--profile' option\n"), option_name);
|
|
return TRUE; /* we do not want to bail out here but continue */
|
|
}
|
|
|
|
|
|
static gboolean G_GNUC_NORETURN
|
|
option_version_cb (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
g_print ("%s %s\n", _("MATE Terminal"), VERSION);
|
|
|
|
exit (EXIT_SUCCESS);
|
|
}
|
|
|
|
static gboolean
|
|
option_command_callback (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
TerminalOptions *options = data;
|
|
GError *err = NULL;
|
|
char **exec_argv;
|
|
|
|
if (!g_shell_parse_argv (value, NULL, &exec_argv, &err))
|
|
{
|
|
g_set_error(error,
|
|
G_OPTION_ERROR,
|
|
G_OPTION_ERROR_BAD_VALUE,
|
|
_("Argument to \"%s\" is not a valid command: %s"),
|
|
"--command/-e",
|
|
err->message);
|
|
g_error_free (err);
|
|
return FALSE;
|
|
}
|
|
|
|
if (options->initial_windows)
|
|
{
|
|
InitialTab *it = ensure_top_tab (options);
|
|
|
|
g_strfreev (it->exec_argv);
|
|
it->exec_argv = exec_argv;
|
|
}
|
|
else
|
|
{
|
|
g_strfreev (options->exec_argv);
|
|
options->exec_argv = exec_argv;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
option_profile_cb (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
TerminalOptions *options = data;
|
|
|
|
if (options->initial_windows)
|
|
{
|
|
InitialTab *it = ensure_top_tab (options);
|
|
|
|
g_free (it->profile);
|
|
it->profile = g_strdup (value);
|
|
it->profile_is_id = FALSE;
|
|
}
|
|
else
|
|
{
|
|
g_free (options->default_profile);
|
|
options->default_profile = g_strdup (value);
|
|
options->default_profile_is_id = FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
option_profile_id_cb (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
TerminalOptions *options = data;
|
|
|
|
if (options->initial_windows)
|
|
{
|
|
InitialTab *it = ensure_top_tab (options);
|
|
|
|
g_free (it->profile);
|
|
it->profile = g_strdup (value);
|
|
it->profile_is_id = TRUE;
|
|
}
|
|
else
|
|
{
|
|
g_free (options->default_profile);
|
|
options->default_profile = g_strdup (value);
|
|
options->default_profile_is_id = TRUE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
option_window_callback (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
TerminalOptions *options = data;
|
|
gboolean is_profile_id;
|
|
|
|
is_profile_id = g_str_has_suffix (option_name, "-with-profile-internal-id");
|
|
|
|
add_new_window (options, value, is_profile_id);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
option_tab_callback (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
TerminalOptions *options = data;
|
|
gboolean is_profile_id;
|
|
InitialWindow *iw;
|
|
InitialTab *it;
|
|
|
|
is_profile_id = g_str_has_suffix (option_name, "-with-profile-internal-id");
|
|
|
|
if (options->initial_windows)
|
|
{
|
|
iw = g_list_last (options->initial_windows)->data;
|
|
iw->tabs = g_list_append (iw->tabs, initial_tab_new (value, is_profile_id));
|
|
}
|
|
else
|
|
{
|
|
iw = add_new_window (options, value, is_profile_id);
|
|
it = g_list_last(iw->tabs)->data;
|
|
it->attach_window = TRUE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
option_role_callback (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
TerminalOptions *options = data;
|
|
InitialWindow *iw;
|
|
|
|
if (options->initial_windows)
|
|
{
|
|
iw = g_list_last (options->initial_windows)->data;
|
|
iw->role = g_strdup (value);
|
|
}
|
|
else if (!options->default_role)
|
|
options->default_role = g_strdup (value);
|
|
else
|
|
{
|
|
g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_FAILED,
|
|
"%s", _("Two roles given for one window"));
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
option_show_menubar_callback (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
TerminalOptions *options = data;
|
|
InitialWindow *iw;
|
|
|
|
if (options->initial_windows)
|
|
{
|
|
iw = g_list_last (options->initial_windows)->data;
|
|
if (iw->force_menubar_state && iw->menubar_state == TRUE)
|
|
{
|
|
g_printerr (_("\"%s\" option given twice for the same window\n"),
|
|
"--show-menubar");
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
iw->force_menubar_state = TRUE;
|
|
iw->menubar_state = TRUE;
|
|
}
|
|
else
|
|
{
|
|
options->default_window_menubar_forced = TRUE;
|
|
options->default_window_menubar_state = TRUE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
option_hide_menubar_callback (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
TerminalOptions *options = data;
|
|
InitialWindow *iw;
|
|
|
|
if (options->initial_windows)
|
|
{
|
|
iw = g_list_last (options->initial_windows)->data;
|
|
|
|
if (iw->force_menubar_state && iw->menubar_state == FALSE)
|
|
{
|
|
g_printerr (_("\"%s\" option given twice for the same window\n"),
|
|
"--hide-menubar");
|
|
return TRUE;
|
|
}
|
|
|
|
iw->force_menubar_state = TRUE;
|
|
iw->menubar_state = FALSE;
|
|
}
|
|
else
|
|
{
|
|
options->default_window_menubar_forced = TRUE;
|
|
options->default_window_menubar_state = FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
option_maximize_callback (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
TerminalOptions *options = data;
|
|
InitialWindow *iw;
|
|
|
|
if (options->initial_windows)
|
|
{
|
|
iw = g_list_last (options->initial_windows)->data;
|
|
iw->start_maximized = TRUE;
|
|
}
|
|
else
|
|
options->default_maximize = TRUE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
option_fullscreen_callback (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
TerminalOptions *options = data;
|
|
|
|
if (options->initial_windows)
|
|
{
|
|
InitialWindow *iw;
|
|
|
|
iw = g_list_last (options->initial_windows)->data;
|
|
iw->start_fullscreen = TRUE;
|
|
}
|
|
else
|
|
options->default_fullscreen = TRUE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
option_geometry_callback (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
TerminalOptions *options = data;
|
|
|
|
if (options->initial_windows)
|
|
{
|
|
InitialWindow *iw;
|
|
|
|
iw = g_list_last (options->initial_windows)->data;
|
|
iw->geometry = g_strdup (value);
|
|
}
|
|
else
|
|
options->default_geometry = g_strdup (value);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
option_disable_factory_callback (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
TerminalOptions *options = data;
|
|
|
|
options->use_factory = FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
option_load_save_config_cb (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
TerminalOptions *options = data;
|
|
|
|
if (options->config_file)
|
|
{
|
|
g_set_error_literal (error, TERMINAL_OPTION_ERROR, TERMINAL_OPTION_ERROR_EXCLUSIVE_OPTIONS,
|
|
"Options \"--load-config\" and \"--save-config\" are mutually exclusive");
|
|
return FALSE;
|
|
}
|
|
|
|
options->config_file = terminal_util_resolve_relative_path (options->default_working_dir, value);
|
|
options->load_config = strcmp (option_name, "--load-config") == 0;
|
|
options->save_config = strcmp (option_name, "--save-config") == 0;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
option_title_callback (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
TerminalOptions *options = data;
|
|
|
|
if (options->initial_windows)
|
|
{
|
|
InitialTab *it = ensure_top_tab (options);
|
|
|
|
g_free (it->title);
|
|
it->title = g_strdup (value);
|
|
}
|
|
else
|
|
{
|
|
g_free (options->default_title);
|
|
options->default_title = g_strdup (value);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
option_working_directory_callback (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
TerminalOptions *options = data;
|
|
|
|
if (options->initial_windows)
|
|
{
|
|
InitialTab *it = ensure_top_tab (options);
|
|
|
|
g_free (it->working_dir);
|
|
it->working_dir = g_strdup (value);
|
|
}
|
|
else
|
|
{
|
|
g_free (options->default_working_dir);
|
|
options->default_working_dir = g_strdup (value);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
option_active_callback (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
TerminalOptions *options = data;
|
|
InitialTab *it;
|
|
|
|
it = ensure_top_tab (options);
|
|
it->active = TRUE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
option_zoom_callback (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
TerminalOptions *options = data;
|
|
double zoom;
|
|
char *end;
|
|
|
|
/* Try reading a locale-style double first, in case it was
|
|
* typed by a person, then fall back to ascii_strtod (we
|
|
* always save session in C locale format)
|
|
*/
|
|
end = NULL;
|
|
errno = 0;
|
|
zoom = g_strtod (value, &end);
|
|
if (end == NULL || *end != '\0')
|
|
{
|
|
g_set_error (error,
|
|
G_OPTION_ERROR,
|
|
G_OPTION_ERROR_BAD_VALUE,
|
|
_("\"%s\" is not a valid zoom factor"),
|
|
value);
|
|
return FALSE;
|
|
}
|
|
|
|
if (zoom < (TERMINAL_SCALE_MINIMUM + 1e-6))
|
|
{
|
|
g_printerr (_("Zoom factor \"%g\" is too small, using %g\n"),
|
|
zoom,
|
|
TERMINAL_SCALE_MINIMUM);
|
|
zoom = TERMINAL_SCALE_MINIMUM;
|
|
}
|
|
|
|
if (zoom > (TERMINAL_SCALE_MAXIMUM - 1e-6))
|
|
{
|
|
g_printerr (_("Zoom factor \"%g\" is too large, using %g\n"),
|
|
zoom,
|
|
TERMINAL_SCALE_MAXIMUM);
|
|
zoom = TERMINAL_SCALE_MAXIMUM;
|
|
}
|
|
|
|
if (options->initial_windows)
|
|
{
|
|
InitialTab *it = ensure_top_tab (options);
|
|
it->zoom = zoom;
|
|
it->zoom_set = TRUE;
|
|
}
|
|
else
|
|
options->zoom = zoom;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/* Evaluation of the arguments given to the command line options */
|
|
static gboolean
|
|
digest_options_callback (GOptionContext *context,
|
|
GOptionGroup *group,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
TerminalOptions *options = data;
|
|
InitialTab *it;
|
|
|
|
if (options->execute)
|
|
{
|
|
if (options->exec_argv == NULL)
|
|
{
|
|
g_set_error (error,
|
|
G_OPTION_ERROR,
|
|
G_OPTION_ERROR_BAD_VALUE,
|
|
_("Option \"%s\" requires specifying the command to run"
|
|
" on the rest of the command line"),
|
|
"--execute/-x");
|
|
return FALSE;
|
|
}
|
|
|
|
/* Apply -x/--execute command only to the first tab */
|
|
it = ensure_top_tab (options);
|
|
it->exec_argv = options->exec_argv;
|
|
options->exec_argv = NULL;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
* terminal_options_parse:
|
|
* @working_directory: the default working directory
|
|
* @display_name: the default X display name
|
|
* @startup_id: the startup notification ID
|
|
* @env: the environment as variable=value pairs
|
|
* @remote_arguments: whether the caller is the factory process or not
|
|
* @ignore_unknown_options: whether to ignore unknown options when parsing
|
|
* the arguments
|
|
* @argcp: (inout) address of the argument count. Changed if any arguments were handled
|
|
* @argvp: (inout) address of the argument vector. Any parameters understood by
|
|
* the terminal #GOptionContext are removed
|
|
* @error: a #GError to fill in
|
|
* @...: a %NULL terminated list of extra #GOptionGroup<!-- -->s
|
|
*
|
|
* Parses the argument vector *@argvp.
|
|
*
|
|
* Returns: a new #TerminalOptions containing the windows and tabs to open,
|
|
* or %NULL on error.
|
|
*/
|
|
TerminalOptions *
|
|
terminal_options_parse (const char *working_directory,
|
|
const char *display_name,
|
|
const char *startup_id,
|
|
char **env,
|
|
gboolean remote_arguments,
|
|
gboolean ignore_unknown_options,
|
|
int *argcp,
|
|
char ***argvp,
|
|
GError **error,
|
|
...)
|
|
{
|
|
TerminalOptions *options;
|
|
GOptionContext *context;
|
|
GOptionGroup *extra_group;
|
|
va_list va_args;
|
|
gboolean retval;
|
|
int i;
|
|
char **argv = *argvp;
|
|
|
|
options = g_slice_new0 (TerminalOptions);
|
|
|
|
options->remote_arguments = remote_arguments;
|
|
options->default_window_menubar_forced = FALSE;
|
|
options->default_window_menubar_state = TRUE;
|
|
options->default_fullscreen = FALSE;
|
|
options->default_maximize = FALSE;
|
|
options->execute = FALSE;
|
|
options->use_factory = TRUE;
|
|
options->initial_workspace = -1;
|
|
|
|
options->env = g_strdupv (env);
|
|
options->startup_id = g_strdup (startup_id && startup_id[0] ? startup_id : NULL);
|
|
options->display_name = g_strdup (display_name);
|
|
options->initial_windows = NULL;
|
|
options->default_role = NULL;
|
|
options->default_geometry = NULL;
|
|
options->default_title = NULL;
|
|
options->zoom = 1.0;
|
|
|
|
options->screen_number = -1;
|
|
options->default_working_dir = g_strdup (working_directory);
|
|
|
|
/* The old -x/--execute option is broken, so we need to pre-scan for it. */
|
|
/* We now also support passing the command after the -- switch. */
|
|
options->exec_argv = NULL;
|
|
for (i = 1 ; i < *argcp; ++i)
|
|
{
|
|
gboolean is_execute;
|
|
gboolean is_dashdash;
|
|
int j, last;
|
|
|
|
is_execute = strcmp (argv[i], "-x") == 0 || strcmp (argv[i], "--execute") == 0;
|
|
is_dashdash = strcmp (argv[i], "--") == 0;
|
|
|
|
if (!is_execute && !is_dashdash)
|
|
continue;
|
|
|
|
options->execute = is_execute;
|
|
|
|
/* Skip the switch */
|
|
last = i;
|
|
++i;
|
|
if (i == *argcp)
|
|
break; /* we'll complain about this later for -x/--execute; it's fine for -- */
|
|
|
|
/* Collect the args, and remove them from argv */
|
|
options->exec_argv = g_new0 (char*, *argcp - i + 1);
|
|
for (j = 0; i < *argcp; ++i, ++j)
|
|
options->exec_argv[j] = g_strdup (argv[i]);
|
|
options->exec_argv[j] = NULL;
|
|
|
|
*argcp = last;
|
|
break;
|
|
}
|
|
|
|
context = get_goption_context (options);
|
|
|
|
g_option_context_set_ignore_unknown_options (context, ignore_unknown_options);
|
|
|
|
va_start (va_args, error);
|
|
extra_group = va_arg (va_args, GOptionGroup*);
|
|
while (extra_group != NULL)
|
|
{
|
|
g_option_context_add_group (context, extra_group);
|
|
extra_group = va_arg (va_args, GOptionGroup*);
|
|
}
|
|
va_end (va_args);
|
|
|
|
retval = g_option_context_parse (context, argcp, argvp, error);
|
|
g_option_context_free (context);
|
|
|
|
if (retval)
|
|
return options;
|
|
|
|
terminal_options_free (options);
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* terminal_options_merge_config:
|
|
* @options:
|
|
* @key_file: a #GKeyFile containing to merge the options from
|
|
* @source_tag: a source_tag to use in new #InitialWindow<!-- -->s
|
|
* @error: a #GError to fill in
|
|
*
|
|
* Merges the saved options from @key_file into @options.
|
|
*
|
|
* Returns: %TRUE if @key_file was a valid key file containing a stored
|
|
* terminal configuration, or %FALSE on error
|
|
*/
|
|
gboolean
|
|
terminal_options_merge_config (TerminalOptions *options,
|
|
GKeyFile *key_file,
|
|
guint source_tag,
|
|
GError **error)
|
|
{
|
|
int version, compat_version;
|
|
char **groups;
|
|
guint i;
|
|
gboolean have_error = FALSE;
|
|
GList *initial_windows = NULL;
|
|
|
|
if (!g_key_file_has_group (key_file, TERMINAL_CONFIG_GROUP))
|
|
{
|
|
g_set_error_literal (error, TERMINAL_OPTION_ERROR,
|
|
TERMINAL_OPTION_ERROR_INVALID_CONFIG_FILE,
|
|
_("Not a valid terminal config file."));
|
|
return FALSE;
|
|
}
|
|
|
|
version = g_key_file_get_integer (key_file, TERMINAL_CONFIG_GROUP, TERMINAL_CONFIG_PROP_VERSION, NULL);
|
|
compat_version = g_key_file_get_integer (key_file, TERMINAL_CONFIG_GROUP, TERMINAL_CONFIG_PROP_COMPAT_VERSION, NULL);
|
|
|
|
if (version <= 0 ||
|
|
compat_version <= 0 ||
|
|
compat_version > TERMINAL_CONFIG_COMPAT_VERSION)
|
|
{
|
|
g_set_error_literal (error, TERMINAL_OPTION_ERROR,
|
|
TERMINAL_OPTION_ERROR_INCOMPATIBLE_CONFIG_FILE,
|
|
_("Incompatible terminal config file version."));
|
|
return FALSE;
|
|
}
|
|
|
|
groups = g_key_file_get_string_list (key_file, TERMINAL_CONFIG_GROUP, TERMINAL_CONFIG_PROP_WINDOWS, NULL, error);
|
|
if (!groups)
|
|
return FALSE;
|
|
|
|
for (i = 0; groups[i]; ++i)
|
|
{
|
|
const char *window_group = groups[i];
|
|
char **tab_groups;
|
|
InitialWindow *iw;
|
|
guint j;
|
|
|
|
tab_groups = g_key_file_get_string_list (key_file, window_group, TERMINAL_CONFIG_WINDOW_PROP_TABS, NULL, error);
|
|
if (!tab_groups)
|
|
continue; /* no tabs in this window, skip it */
|
|
|
|
iw = initial_window_new (source_tag);
|
|
initial_windows = g_list_append (initial_windows, iw);
|
|
apply_defaults (options, iw);
|
|
|
|
iw->role = g_key_file_get_string (key_file, window_group, TERMINAL_CONFIG_WINDOW_PROP_ROLE, NULL);
|
|
iw->geometry = g_key_file_get_string (key_file, window_group, TERMINAL_CONFIG_WINDOW_PROP_GEOMETRY, NULL);
|
|
iw->start_fullscreen = g_key_file_get_boolean (key_file, window_group, TERMINAL_CONFIG_WINDOW_PROP_FULLSCREEN, NULL);
|
|
iw->start_maximized = g_key_file_get_boolean (key_file, window_group, TERMINAL_CONFIG_WINDOW_PROP_MAXIMIZED, NULL);
|
|
if (g_key_file_has_key (key_file, window_group, TERMINAL_CONFIG_WINDOW_PROP_MENUBAR_VISIBLE, NULL))
|
|
{
|
|
iw->force_menubar_state = TRUE;
|
|
iw->menubar_state = g_key_file_get_boolean (key_file, window_group, TERMINAL_CONFIG_WINDOW_PROP_MENUBAR_VISIBLE, NULL);
|
|
}
|
|
|
|
for (j = 0; tab_groups[j]; ++j)
|
|
{
|
|
const char *tab_group = tab_groups[j];
|
|
InitialTab *it;
|
|
char *profile;
|
|
|
|
profile = g_key_file_get_string (key_file, tab_group, TERMINAL_CONFIG_TERMINAL_PROP_PROFILE_ID, NULL);
|
|
it = initial_tab_new (profile, TRUE);
|
|
g_free (profile);
|
|
|
|
iw->tabs = g_list_append (iw->tabs, it);
|
|
|
|
/* it->width = g_key_file_get_integer (key_file, tab_group, TERMINAL_CONFIG_TERMINAL_PROP_WIDTH, NULL);
|
|
it->height = g_key_file_get_integer (key_file, tab_group, TERMINAL_CONFIG_TERMINAL_PROP_HEIGHT, NULL);*/
|
|
it->working_dir = terminal_util_key_file_get_string_unescape (key_file, tab_group, TERMINAL_CONFIG_TERMINAL_PROP_WORKING_DIRECTORY, NULL);
|
|
it->title = g_key_file_get_string (key_file, tab_group, TERMINAL_CONFIG_TERMINAL_PROP_TITLE, NULL);
|
|
|
|
if (g_key_file_has_key (key_file, tab_group, TERMINAL_CONFIG_TERMINAL_PROP_COMMAND, NULL) &&
|
|
!(it->exec_argv = terminal_util_key_file_get_argv (key_file, tab_group, TERMINAL_CONFIG_TERMINAL_PROP_COMMAND, NULL, error)))
|
|
{
|
|
have_error = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
g_strfreev (tab_groups);
|
|
|
|
if (have_error)
|
|
break;
|
|
}
|
|
|
|
g_strfreev (groups);
|
|
|
|
if (have_error)
|
|
{
|
|
g_list_foreach (initial_windows, (GFunc) initial_window_free, NULL);
|
|
g_list_free (initial_windows);
|
|
return FALSE;
|
|
}
|
|
|
|
options->initial_windows = g_list_concat (options->initial_windows, initial_windows);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
* terminal_options_ensure_window:
|
|
* @options:
|
|
*
|
|
* Ensure that @options will contain at least one window to open.
|
|
*/
|
|
void
|
|
terminal_options_ensure_window (TerminalOptions *options)
|
|
{
|
|
ensure_top_window (options);
|
|
}
|
|
|
|
/**
|
|
* terminal_options_free:
|
|
* @options:
|
|
*
|
|
* Frees @options.
|
|
*/
|
|
void
|
|
terminal_options_free (TerminalOptions *options)
|
|
{
|
|
g_list_foreach (options->initial_windows, (GFunc) initial_window_free, NULL);
|
|
g_list_free (options->initial_windows);
|
|
|
|
g_strfreev (options->env);
|
|
g_free (options->default_role);
|
|
g_free (options->default_geometry);
|
|
g_free (options->default_working_dir);
|
|
g_free (options->default_title);
|
|
g_free (options->default_profile);
|
|
|
|
g_strfreev (options->exec_argv);
|
|
|
|
g_free (options->display_name);
|
|
g_free (options->startup_id);
|
|
|
|
g_slice_free (TerminalOptions, options);
|
|
}
|
|
|
|
static GOptionContext *
|
|
get_goption_context (TerminalOptions *options)
|
|
{
|
|
const GOptionEntry global_unique_goptions[] =
|
|
{
|
|
{
|
|
"disable-factory",
|
|
0,
|
|
G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_disable_factory_callback,
|
|
N_("Do not register with the activation nameserver, do not re-use an active terminal"),
|
|
NULL
|
|
},
|
|
{
|
|
"load-config",
|
|
0,
|
|
G_OPTION_FLAG_FILENAME,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_load_save_config_cb,
|
|
N_("Load a terminal configuration file"),
|
|
N_("FILE")
|
|
},
|
|
{
|
|
"save-config",
|
|
0,
|
|
G_OPTION_FLAG_FILENAME,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_load_save_config_cb,
|
|
N_("Save the terminal configuration to a file"),
|
|
N_("FILE")
|
|
},
|
|
{ "version", 0, G_OPTION_FLAG_NO_ARG | G_OPTION_FLAG_HIDDEN, G_OPTION_ARG_CALLBACK, option_version_cb, NULL, NULL },
|
|
{ NULL, 0, 0, 0, NULL, NULL, NULL }
|
|
};
|
|
|
|
const GOptionEntry global_multiple_goptions[] =
|
|
{
|
|
{
|
|
"window",
|
|
0,
|
|
G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_window_callback,
|
|
N_("Open a new window containing a tab with the default profile"),
|
|
NULL
|
|
},
|
|
{
|
|
"tab",
|
|
0,
|
|
G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_tab_callback,
|
|
N_("Open a new tab in the last-opened window with the default profile"),
|
|
NULL
|
|
},
|
|
{ NULL, 0, 0, 0, NULL, NULL, NULL }
|
|
};
|
|
|
|
const GOptionEntry window_goptions[] =
|
|
{
|
|
{
|
|
"show-menubar",
|
|
0,
|
|
G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_show_menubar_callback,
|
|
N_("Turn on the menubar"),
|
|
NULL
|
|
},
|
|
{
|
|
"hide-menubar",
|
|
0,
|
|
G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_hide_menubar_callback,
|
|
N_("Turn off the menubar"),
|
|
NULL
|
|
},
|
|
{
|
|
"maximize",
|
|
0,
|
|
G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_maximize_callback,
|
|
N_("Maximize the window"),
|
|
NULL
|
|
},
|
|
{
|
|
"full-screen",
|
|
0,
|
|
G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_fullscreen_callback,
|
|
N_("Full-screen the window"),
|
|
NULL
|
|
},
|
|
{
|
|
"geometry",
|
|
0,
|
|
0,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_geometry_callback,
|
|
N_("Set the window size; for example: 80x24, or 80x24+200+200 (COLSxROWS+X+Y)"),
|
|
N_("GEOMETRY")
|
|
},
|
|
{
|
|
"role",
|
|
0,
|
|
0,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_role_callback,
|
|
N_("Set the window role"),
|
|
N_("ROLE")
|
|
},
|
|
{
|
|
"active",
|
|
0,
|
|
G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_active_callback,
|
|
N_("Set the last specified tab as the active one in its window"),
|
|
NULL
|
|
},
|
|
{ NULL, 0, 0, 0, NULL, NULL, NULL }
|
|
};
|
|
|
|
const GOptionEntry terminal_goptions[] =
|
|
{
|
|
{
|
|
"command",
|
|
'e',
|
|
G_OPTION_FLAG_FILENAME,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_command_callback,
|
|
N_("Execute the argument to this option inside the terminal"),
|
|
NULL
|
|
},
|
|
{
|
|
"profile",
|
|
0,
|
|
0,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_profile_cb,
|
|
N_("Use the given profile instead of the default profile"),
|
|
N_("PROFILE-NAME")
|
|
},
|
|
{
|
|
"title",
|
|
't',
|
|
0,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_title_callback,
|
|
N_("Set the terminal title"),
|
|
N_("TITLE")
|
|
},
|
|
{
|
|
"working-directory",
|
|
0,
|
|
G_OPTION_FLAG_FILENAME,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_working_directory_callback,
|
|
N_("Set the working directory"),
|
|
N_("DIRNAME")
|
|
},
|
|
{
|
|
"zoom",
|
|
0,
|
|
0,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_zoom_callback,
|
|
N_("Set the terminal's zoom factor (1.0 = normal size)"),
|
|
N_("ZOOM")
|
|
},
|
|
{ NULL, 0, 0, 0, NULL, NULL, NULL }
|
|
};
|
|
|
|
const GOptionEntry internal_goptions[] =
|
|
{
|
|
{
|
|
"profile-id",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_profile_id_cb,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"window-with-profile",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_window_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"tab-with-profile",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_tab_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"window-with-profile-internal-id",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_window_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"tab-with-profile-internal-id",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN,
|
|
G_OPTION_ARG_CALLBACK,
|
|
option_tab_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"default-working-directory",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN,
|
|
G_OPTION_ARG_FILENAME,
|
|
&options->default_working_dir,
|
|
NULL, NULL,
|
|
},
|
|
{
|
|
"use-factory",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN,
|
|
G_OPTION_ARG_NONE,
|
|
&options->use_factory,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"startup-id",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN,
|
|
G_OPTION_ARG_STRING,
|
|
&options->startup_id,
|
|
NULL,
|
|
NULL
|
|
},
|
|
/*
|
|
* Crappy old compat args
|
|
*/
|
|
{
|
|
"tclass",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"font",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"nologin",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"login",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"foreground",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"background",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"solid",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"bgscroll",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"bgnoscroll",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"shaded",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"noshaded",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"transparent",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"utmp",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"noutmp",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"wtmp",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"nowtmp",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"lastlog",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"nolastlog",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"icon",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"termname",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{
|
|
"start-factory-server",
|
|
0,
|
|
G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
|
|
G_OPTION_ARG_CALLBACK,
|
|
unsupported_option_callback,
|
|
NULL, NULL
|
|
},
|
|
{ NULL, 0, 0, 0, NULL, NULL, NULL }
|
|
};
|
|
|
|
GOptionContext *context;
|
|
GOptionGroup *group;
|
|
|
|
context = g_option_context_new (NULL);
|
|
g_option_context_set_translation_domain (context, GETTEXT_PACKAGE);
|
|
g_option_context_set_description (context, N_("MATE Terminal Emulator"));
|
|
|
|
group = g_option_group_new ("mate-terminal",
|
|
N_("MATE Terminal Emulator"),
|
|
N_("Show MATE Terminal options"),
|
|
options,
|
|
NULL);
|
|
g_option_group_set_translation_domain (group, GETTEXT_PACKAGE);
|
|
g_option_group_add_entries (group, global_unique_goptions);
|
|
g_option_group_add_entries (group, internal_goptions);
|
|
g_option_group_set_parse_hooks (group, NULL, digest_options_callback);
|
|
g_option_context_set_main_group (context, group);
|
|
|
|
group = g_option_group_new ("terminal",
|
|
N_("Options to open new windows or terminal tabs; more than one of these may be specified:"),
|
|
N_("Show terminal options"),
|
|
options,
|
|
NULL);
|
|
g_option_group_set_translation_domain (group, GETTEXT_PACKAGE);
|
|
g_option_group_add_entries (group, global_multiple_goptions);
|
|
g_option_context_add_group (context, group);
|
|
|
|
group = g_option_group_new ("window-options",
|
|
N_("Window options; if used before the first --window or --tab argument, sets the default for all windows:"),
|
|
N_("Show per-window options"),
|
|
options,
|
|
NULL);
|
|
g_option_group_set_translation_domain (group, GETTEXT_PACKAGE);
|
|
g_option_group_add_entries (group, window_goptions);
|
|
g_option_context_add_group (context, group);
|
|
|
|
group = g_option_group_new ("terminal-options",
|
|
N_("Terminal options; if used before the first --window or --tab argument, sets the default for all terminals:"),
|
|
N_("Show per-terminal options"),
|
|
options,
|
|
NULL);
|
|
g_option_group_set_translation_domain (group, GETTEXT_PACKAGE);
|
|
g_option_group_add_entries (group, terminal_goptions);
|
|
g_option_context_add_group (context, group);
|
|
|
|
return context;
|
|
}
|