mirror of
https://github.com/MidnightCommander/mc
synced 2025-01-03 10:04:32 +03:00
72ad268a9c
Signed-off-by: Andrew Borodin <aborodin@vmail.ru>
1503 lines
49 KiB
C
1503 lines
49 KiB
C
/*
|
|
Setup loading/saving.
|
|
|
|
Copyright (C) 1994, 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005,
|
|
2006, 2007, 2009, 2010, 2011, 2013
|
|
The Free Software Foundation, Inc.
|
|
|
|
This file is part of the Midnight Commander.
|
|
|
|
The Midnight Commander 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.
|
|
|
|
The Midnight Commander 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/>.
|
|
*/
|
|
|
|
/** \file setup.c
|
|
* \brief Source: setup loading/saving
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
|
|
#include "lib/global.h"
|
|
|
|
#include "lib/tty/tty.h"
|
|
#include "lib/tty/key.h"
|
|
#include "lib/mcconfig.h"
|
|
#include "lib/fileloc.h"
|
|
#include "lib/timefmt.h"
|
|
#include "lib/util.h"
|
|
#include "lib/widget.h"
|
|
|
|
#include "lib/vfs/vfs.h"
|
|
|
|
#ifdef ENABLE_VFS_FTP
|
|
#include "src/vfs/ftpfs/ftpfs.h"
|
|
#endif
|
|
#ifdef ENABLE_VFS_FISH
|
|
#include "src/vfs/fish/fish.h"
|
|
#endif
|
|
|
|
#ifdef HAVE_CHARSET
|
|
#include "lib/charsets.h"
|
|
#endif
|
|
|
|
#include "filemanager/dir.h"
|
|
#include "filemanager/midnight.h"
|
|
#include "filemanager/tree.h" /* xtree_mode */
|
|
#include "filemanager/hotlist.h" /* load/save/done hotlist */
|
|
#include "filemanager/panelize.h" /* load/save/done panelize */
|
|
#include "filemanager/layout.h"
|
|
#include "filemanager/cmd.h"
|
|
|
|
#include "args.h"
|
|
#include "execute.h" /* pause_after_run */
|
|
#include "clipboard.h"
|
|
#include "keybind-defaults.h" /* keybind_lookup_action */
|
|
|
|
#ifdef HAVE_CHARSET
|
|
#include "selcodepage.h"
|
|
#endif
|
|
|
|
#ifdef USE_INTERNAL_EDIT
|
|
#include "src/editor/edit.h"
|
|
#endif
|
|
|
|
#include "src/viewer/mcviewer.h" /* For the externs */
|
|
|
|
#include "setup.h"
|
|
|
|
/*** global variables ****************************************************************************/
|
|
|
|
char *global_profile_name; /* mc.lib */
|
|
|
|
/* Only used at program boot */
|
|
gboolean boot_current_is_left = TRUE;
|
|
|
|
/* If on, default for "No" in delete operations */
|
|
int safe_delete = 0;
|
|
|
|
/* Controls screen clearing before an exec */
|
|
int clear_before_exec = 1;
|
|
|
|
/* Asks for confirmation before deleting a file */
|
|
int confirm_delete = 1;
|
|
/* Asks for confirmation before deleting a hotlist entry */
|
|
int confirm_directory_hotlist_delete = 1;
|
|
/* Asks for confirmation before overwriting a file */
|
|
int confirm_overwrite = 1;
|
|
/* Asks for confirmation before executing a program by pressing enter */
|
|
int confirm_execute = 0;
|
|
/* Asks for confirmation before leaving the program */
|
|
int confirm_exit = 1;
|
|
|
|
/* If true, at startup the user-menu is invoked */
|
|
int auto_menu = 0;
|
|
/* This flag indicates if the pull down menus by default drop down */
|
|
int drop_menus = 0;
|
|
|
|
/* Asks for confirmation when using F3 to view a directory and there
|
|
are tagged files */
|
|
int confirm_view_dir = 0;
|
|
|
|
/* Ask file name before start the editor */
|
|
int editor_ask_filename_before_edit = 0;
|
|
|
|
panel_view_mode_t startup_left_mode;
|
|
panel_view_mode_t startup_right_mode;
|
|
|
|
int setup_copymove_persistent_attr = 1;
|
|
|
|
/* Tab size */
|
|
int option_tab_spacing = DEFAULT_TAB_SPACING;
|
|
|
|
/* Ugly hack to allow panel_save_setup to work as a place holder for */
|
|
/* default panel values */
|
|
int saving_setup;
|
|
|
|
panels_options_t panels_options = {
|
|
.show_mini_info = TRUE,
|
|
.kilobyte_si = FALSE,
|
|
.mix_all_files = FALSE,
|
|
.show_backups = TRUE,
|
|
.show_dot_files = TRUE,
|
|
.fast_reload = FALSE,
|
|
.fast_reload_msg_shown = FALSE,
|
|
.mark_moves_down = TRUE,
|
|
.reverse_files_only = TRUE,
|
|
.auto_save_setup = FALSE,
|
|
.navigate_with_arrows = FALSE,
|
|
.scroll_pages = TRUE,
|
|
.mouse_move_pages = TRUE,
|
|
.filetype_mode = TRUE,
|
|
.permission_mode = FALSE,
|
|
.qsearch_mode = QSEARCH_PANEL_CASE,
|
|
.torben_fj_mode = FALSE
|
|
};
|
|
|
|
int easy_patterns = 1;
|
|
|
|
/* It true saves the setup when quitting */
|
|
int auto_save_setup = 1;
|
|
|
|
/* If true, then the +, - and \ keys have their special meaning only if the
|
|
* command line is emtpy, otherwise they behave like regular letters
|
|
*/
|
|
int only_leading_plus_minus = 1;
|
|
|
|
/* Automatically fills name with current selected item name on mkdir */
|
|
int auto_fill_mkdir_name = 1;
|
|
|
|
/* If set and you don't have subshell support,then C-o will give you a shell */
|
|
int output_starts_shell = 0;
|
|
|
|
/* If set, we execute the file command to check the file type */
|
|
int use_file_to_check_type = 1;
|
|
|
|
int verbose = 1;
|
|
|
|
/*
|
|
* Whether the Midnight Commander tries to provide more
|
|
* information about copy/move sizes and bytes transfered
|
|
* at the expense of some speed
|
|
*/
|
|
int file_op_compute_totals = 1;
|
|
|
|
/* If true use the internal viewer */
|
|
int use_internal_view = 1;
|
|
/* If set, use the builtin editor */
|
|
int use_internal_edit = 1;
|
|
|
|
#ifdef HAVE_CHARSET
|
|
/* Numbers of (file I/O) and (input/display) codepages. -1 if not selected */
|
|
int default_source_codepage = -1;
|
|
char *autodetect_codeset = NULL;
|
|
gboolean is_autodetect_codeset_enabled = FALSE;
|
|
#endif /* !HAVE_CHARSET */
|
|
|
|
#ifdef HAVE_ASPELL
|
|
char *spell_language = NULL;
|
|
#endif
|
|
|
|
/* Value of "other_dir" key in ini file */
|
|
char *saved_other_dir = NULL;
|
|
|
|
/* If set, then print to the given file the last directory we were at */
|
|
char *last_wd_string = NULL;
|
|
|
|
/* Set when main loop should be terminated */
|
|
int quit = 0;
|
|
|
|
/* Set to TRUE to suppress printing the last directory */
|
|
int print_last_revert = FALSE;
|
|
|
|
/* index to record_macro_buf[], -1 if not recording a macro */
|
|
int macro_index = -1;
|
|
|
|
/* macro stuff */
|
|
struct macro_action_t record_macro_buf[MAX_MACRO_LENGTH];
|
|
|
|
GArray *macros_list;
|
|
|
|
/*** file scope macro definitions ****************************************************************/
|
|
|
|
/* In order to use everywhere the same setup for the locale we use defines */
|
|
#define FMTYEAR _("%b %e %Y")
|
|
#define FMTTIME _("%b %e %H:%M")
|
|
|
|
/*** file scope type declarations ****************************************************************/
|
|
|
|
/*** file scope variables ************************************************************************/
|
|
|
|
static char *profile_name = NULL; /* ${XDG_CONFIG_HOME}/mc/ini */
|
|
static char *panels_profile_name = NULL; /* ${XDG_CACHE_HOME}/mc/panels.ini */
|
|
|
|
/* *INDENT-OFF* */
|
|
static const struct
|
|
{
|
|
const char *key;
|
|
int list_type;
|
|
} list_types [] = {
|
|
{ "full", list_full },
|
|
{ "brief", list_brief },
|
|
{ "long", list_long },
|
|
{ "user", list_user },
|
|
{ NULL, 0 }
|
|
};
|
|
|
|
static const struct
|
|
{
|
|
const char *opt_name;
|
|
panel_view_mode_t opt_type;
|
|
} panel_types [] = {
|
|
{ "listing", view_listing },
|
|
{ "quickview", view_quick },
|
|
{ "info", view_info },
|
|
{ "tree", view_tree },
|
|
{ NULL, view_listing }
|
|
};
|
|
|
|
static const struct
|
|
{
|
|
const char *opt_name;
|
|
int *opt_addr;
|
|
} layout [] = {
|
|
{ "message_visible", &mc_global.message_visible },
|
|
{ "keybar_visible", &mc_global.keybar_visible },
|
|
{ "xterm_title", &xterm_title },
|
|
{ "output_lines", &output_lines },
|
|
{ "command_prompt", &command_prompt },
|
|
{ "menubar_visible", &menubar_visible },
|
|
{ "free_space", &free_space },
|
|
{ "horizontal_split", &panels_layout.horizontal_split },
|
|
{ "vertical_equal", &panels_layout.vertical_equal },
|
|
{ "left_panel_size", &panels_layout.left_panel_size },
|
|
{ "horizontal_equal", &panels_layout.horizontal_equal },
|
|
{ "top_panel_size", &panels_layout.top_panel_size },
|
|
{ NULL, NULL }
|
|
};
|
|
|
|
static const struct
|
|
{
|
|
const char *opt_name;
|
|
int *opt_addr;
|
|
} int_options [] = {
|
|
{ "verbose", &verbose },
|
|
{ "pause_after_run", &pause_after_run },
|
|
{ "shell_patterns", &easy_patterns },
|
|
{ "auto_save_setup", &auto_save_setup },
|
|
{ "preallocate_space", &mc_global.vfs.preallocate_space },
|
|
{ "auto_menu", &auto_menu },
|
|
{ "use_internal_view", &use_internal_view },
|
|
{ "use_internal_edit", &use_internal_edit },
|
|
{ "clear_before_exec", &clear_before_exec },
|
|
{ "confirm_delete", &confirm_delete },
|
|
{ "confirm_overwrite", &confirm_overwrite },
|
|
{ "confirm_execute", &confirm_execute },
|
|
{ "confirm_history_cleanup", &mc_global.widget.confirm_history_cleanup },
|
|
{ "confirm_exit", &confirm_exit },
|
|
{ "confirm_directory_hotlist_delete", &confirm_directory_hotlist_delete },
|
|
{ "safe_delete", &safe_delete },
|
|
{ "mouse_repeat_rate", &mou_auto_repeat },
|
|
{ "double_click_speed", &double_click_speed },
|
|
#ifndef HAVE_CHARSET
|
|
{ "eight_bit_clean", &mc_global.eight_bit_clean },
|
|
{ "full_eight_bits", &mc_global.full_eight_bits },
|
|
#endif /* !HAVE_CHARSET */
|
|
{ "use_8th_bit_as_meta", &use_8th_bit_as_meta },
|
|
{ "confirm_view_dir", &confirm_view_dir },
|
|
{ "mouse_move_pages_viewer", &mcview_mouse_move_pages },
|
|
{ "mouse_close_dialog", &mouse_close_dialog},
|
|
{ "fast_refresh", &fast_refresh },
|
|
{ "drop_menus", &drop_menus },
|
|
{ "wrap_mode", &mcview_global_wrap_mode},
|
|
{ "old_esc_mode", &old_esc_mode },
|
|
{ "old_esc_mode_timeout", &old_esc_mode_timeout },
|
|
{ "cd_symlinks", &mc_global.vfs.cd_symlinks },
|
|
{ "show_all_if_ambiguous", &mc_global.widget.show_all_if_ambiguous },
|
|
{ "max_dirt_limit", &mcview_max_dirt_limit },
|
|
{ "use_file_to_guess_type", &use_file_to_check_type },
|
|
{ "alternate_plus_minus", &mc_global.tty.alternate_plus_minus },
|
|
{ "only_leading_plus_minus", &only_leading_plus_minus },
|
|
{ "show_output_starts_shell", &output_starts_shell },
|
|
{ "xtree_mode", &xtree_mode },
|
|
{ "num_history_items_recorded", &num_history_items_recorded },
|
|
{ "file_op_compute_totals", &file_op_compute_totals },
|
|
{ "classic_progressbar", &classic_progressbar},
|
|
#ifdef ENABLE_VFS
|
|
{ "vfs_timeout", &vfs_timeout },
|
|
#ifdef ENABLE_VFS_FTP
|
|
{ "ftpfs_directory_timeout", &ftpfs_directory_timeout },
|
|
{ "use_netrc", &ftpfs_use_netrc },
|
|
{ "ftpfs_retry_seconds", &ftpfs_retry_seconds },
|
|
{ "ftpfs_always_use_proxy", &ftpfs_always_use_proxy },
|
|
{ "ftpfs_use_passive_connections", &ftpfs_use_passive_connections },
|
|
{ "ftpfs_use_passive_connections_over_proxy", &ftpfs_use_passive_connections_over_proxy },
|
|
{ "ftpfs_use_unix_list_options", &ftpfs_use_unix_list_options },
|
|
{ "ftpfs_first_cd_then_ls", &ftpfs_first_cd_then_ls },
|
|
#endif /* ENABLE_VFS_FTP */
|
|
#ifdef ENABLE_VFS_FISH
|
|
{ "fish_directory_timeout", &fish_directory_timeout },
|
|
#endif /* ENABLE_VFS_FISH */
|
|
#endif /* ENABLE_VFS */
|
|
/* option_tab_spacing is used in internal viewer */
|
|
{ "editor_tab_spacing", &option_tab_spacing },
|
|
#ifdef USE_INTERNAL_EDIT
|
|
{ "editor_word_wrap_line_length", &option_word_wrap_line_length },
|
|
{ "editor_fill_tabs_with_spaces", &option_fill_tabs_with_spaces },
|
|
{ "editor_return_does_auto_indent", &option_return_does_auto_indent },
|
|
{ "editor_backspace_through_tabs", &option_backspace_through_tabs },
|
|
{ "editor_fake_half_tabs", &option_fake_half_tabs },
|
|
{ "editor_option_save_mode", &option_save_mode },
|
|
{ "editor_option_save_position", &option_save_position },
|
|
{ "editor_option_auto_para_formatting", &option_auto_para_formatting },
|
|
{ "editor_option_typewriter_wrap", &option_typewriter_wrap },
|
|
{ "editor_edit_confirm_save", &edit_confirm_save },
|
|
{ "editor_syntax_highlighting", &option_syntax_highlighting },
|
|
{ "editor_persistent_selections", &option_persistent_selections },
|
|
{ "editor_cursor_beyond_eol", &option_cursor_beyond_eol },
|
|
{ "editor_cursor_after_inserted_block", &option_cursor_after_inserted_block },
|
|
{ "editor_visible_tabs", &visible_tabs },
|
|
{ "editor_visible_spaces", &visible_tws },
|
|
{ "editor_line_state", &option_line_state },
|
|
{ "editor_simple_statusbar", &simple_statusbar },
|
|
{ "editor_check_new_line", &option_check_nl_at_eof },
|
|
{ "editor_show_right_margin", &show_right_margin },
|
|
{ "editor_group_undo", &option_group_undo },
|
|
#endif /* USE_INTERNAL_EDIT */
|
|
{ "editor_ask_filename_before_edit", &editor_ask_filename_before_edit },
|
|
{ "nice_rotating_dash", &nice_rotating_dash },
|
|
{ "mcview_remember_file_position", &mcview_remember_file_position },
|
|
{ "auto_fill_mkdir_name", &auto_fill_mkdir_name },
|
|
{ "copymove_persistent_attr", &setup_copymove_persistent_attr },
|
|
{ "select_flags", &select_flags },
|
|
{ NULL, NULL }
|
|
};
|
|
|
|
static const struct
|
|
{
|
|
const char *opt_name;
|
|
char **opt_addr;
|
|
const char *opt_defval;
|
|
} str_options[] = {
|
|
#ifdef USE_INTERNAL_EDIT
|
|
{ "editor_backup_extension", &option_backup_ext, "~" },
|
|
{ "editor_filesize_threshold", &option_filesize_threshold, "64M" },
|
|
#endif
|
|
{ "mcview_eof", &mcview_show_eof, "" },
|
|
{ NULL, NULL, NULL }
|
|
};
|
|
|
|
static const struct
|
|
{
|
|
const char *opt_name;
|
|
gboolean *opt_addr;
|
|
} panels_ini_options[] = {
|
|
{ "show_mini_info", &panels_options.show_mini_info },
|
|
{ "kilobyte_si", &panels_options.kilobyte_si },
|
|
{ "mix_all_files", &panels_options.mix_all_files },
|
|
{ "show_backups", &panels_options.show_backups },
|
|
{ "show_dot_files", &panels_options.show_dot_files },
|
|
{ "fast_reload", &panels_options.fast_reload },
|
|
{ "fast_reload_msg_shown", &panels_options.fast_reload_msg_shown },
|
|
{ "mark_moves_down", &panels_options.mark_moves_down },
|
|
{ "reverse_files_only", &panels_options.reverse_files_only },
|
|
{ "auto_save_setup_panels", &panels_options.auto_save_setup },
|
|
{ "navigate_with_arrows", &panels_options.navigate_with_arrows },
|
|
{ "panel_scroll_pages", &panels_options.scroll_pages },
|
|
{ "mouse_move_pages", &panels_options.mouse_move_pages },
|
|
{ "filetype_mode", &panels_options.filetype_mode },
|
|
{ "permission_mode", &panels_options.permission_mode },
|
|
{ "torben_fj_mode", &panels_options.torben_fj_mode },
|
|
{ NULL, NULL }
|
|
};
|
|
/* *INDENT-ON* */
|
|
|
|
/*** file scope functions ************************************************************************/
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/**
|
|
* Get name of config file.
|
|
*
|
|
* @param subdir If not NULL, config is also searched in specified subdir.
|
|
* @param config_file_name If relative, file if searched in standard paths.
|
|
*
|
|
* @return newly allocated string with config name or NULL if file is not found.
|
|
*/
|
|
|
|
static char *
|
|
load_setup_get_full_config_name (const char *subdir, const char *config_file_name)
|
|
{
|
|
/*
|
|
TODO: IMHO, in future, this function shall be placed in mcconfig module.
|
|
*/
|
|
char *lc_basename, *ret;
|
|
char *file_name;
|
|
|
|
if (config_file_name == NULL)
|
|
return NULL;
|
|
|
|
/* check for .keymap suffix */
|
|
if (g_str_has_suffix (config_file_name, ".keymap"))
|
|
file_name = g_strdup (config_file_name);
|
|
else
|
|
file_name = g_strconcat (config_file_name, ".keymap", (char *) NULL);
|
|
|
|
canonicalize_pathname (file_name);
|
|
|
|
if (g_path_is_absolute (file_name))
|
|
return file_name;
|
|
|
|
lc_basename = g_path_get_basename (file_name);
|
|
g_free (file_name);
|
|
|
|
if (lc_basename == NULL)
|
|
return NULL;
|
|
|
|
if (subdir != NULL)
|
|
ret = g_build_filename (mc_config_get_path (), subdir, lc_basename, NULL);
|
|
else
|
|
ret = g_build_filename (mc_config_get_path (), lc_basename, NULL);
|
|
|
|
if (exist_file (ret))
|
|
{
|
|
g_free (lc_basename);
|
|
canonicalize_pathname (ret);
|
|
return ret;
|
|
}
|
|
g_free (ret);
|
|
|
|
if (subdir != NULL)
|
|
ret = g_build_filename (mc_global.sysconfig_dir, subdir, lc_basename, NULL);
|
|
else
|
|
ret = g_build_filename (mc_global.sysconfig_dir, lc_basename, NULL);
|
|
|
|
if (exist_file (ret))
|
|
{
|
|
g_free (lc_basename);
|
|
canonicalize_pathname (ret);
|
|
return ret;
|
|
}
|
|
g_free (ret);
|
|
|
|
if (subdir != NULL)
|
|
ret = g_build_filename (mc_global.share_data_dir, subdir, lc_basename, NULL);
|
|
else
|
|
ret = g_build_filename (mc_global.share_data_dir, lc_basename, NULL);
|
|
|
|
g_free (lc_basename);
|
|
|
|
if (exist_file (ret))
|
|
{
|
|
canonicalize_pathname (ret);
|
|
return ret;
|
|
}
|
|
|
|
g_free (ret);
|
|
return NULL;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
static const char *
|
|
setup__is_cfg_group_must_panel_config (const char *grp)
|
|
{
|
|
return (!strcasecmp ("Dirs", grp) ||
|
|
!strcasecmp ("Temporal:New Right Panel", grp) ||
|
|
!strcasecmp ("Temporal:New Left Panel", grp) ||
|
|
!strcasecmp ("New Left Panel", grp) || !strcasecmp ("New Right Panel", grp))
|
|
? grp : NULL;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
static void
|
|
setup__move_panels_config_into_separate_file (const char *profile)
|
|
{
|
|
mc_config_t *tmp_cfg;
|
|
char **groups, **curr_grp;
|
|
const char *need_grp;
|
|
|
|
if (!exist_file (profile))
|
|
return;
|
|
|
|
tmp_cfg = mc_config_init (profile, FALSE);
|
|
if (!tmp_cfg)
|
|
return;
|
|
|
|
curr_grp = groups = mc_config_get_groups (tmp_cfg, NULL);
|
|
if (!groups)
|
|
{
|
|
mc_config_deinit (tmp_cfg);
|
|
return;
|
|
}
|
|
|
|
while (*curr_grp)
|
|
{
|
|
if (setup__is_cfg_group_must_panel_config (*curr_grp) == NULL)
|
|
mc_config_del_group (tmp_cfg, *curr_grp);
|
|
curr_grp++;
|
|
}
|
|
|
|
mc_config_save_to_file (tmp_cfg, panels_profile_name, NULL);
|
|
mc_config_deinit (tmp_cfg);
|
|
|
|
tmp_cfg = mc_config_init (profile, FALSE);
|
|
if (!tmp_cfg)
|
|
{
|
|
g_strfreev (groups);
|
|
return;
|
|
}
|
|
|
|
curr_grp = groups;
|
|
|
|
while (*curr_grp)
|
|
{
|
|
need_grp = setup__is_cfg_group_must_panel_config (*curr_grp);
|
|
if (need_grp != NULL)
|
|
{
|
|
mc_config_del_group (tmp_cfg, need_grp);
|
|
}
|
|
curr_grp++;
|
|
}
|
|
g_strfreev (groups);
|
|
|
|
mc_config_save_file (tmp_cfg, NULL);
|
|
mc_config_deinit (tmp_cfg);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
/**
|
|
Create new mc_config object from specified ini-file or
|
|
append data to existing mc_config object from ini-file
|
|
*/
|
|
|
|
static void
|
|
load_setup_init_config_from_file (mc_config_t ** config, const char *fname, gboolean read_only)
|
|
{
|
|
/*
|
|
TODO: IMHO, in future, this function shall be placed in mcconfig module.
|
|
*/
|
|
if (exist_file (fname))
|
|
{
|
|
if (*config != NULL)
|
|
mc_config_read_file (*config, fname, read_only, TRUE);
|
|
else
|
|
*config = mc_config_init (fname, read_only);
|
|
}
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
static void
|
|
load_layout (void)
|
|
{
|
|
size_t i;
|
|
int equal_split;
|
|
int first_panel_size;
|
|
|
|
/* legacy options */
|
|
panels_layout.horizontal_split = mc_config_get_int (mc_main_config, CONFIG_APP_SECTION,
|
|
"horizontal_split", 0);
|
|
equal_split = mc_config_get_int (mc_main_config, "Layout", "equal_split", 1);
|
|
first_panel_size = mc_config_get_int (mc_main_config, "Layout", "first_panel_size", 1);
|
|
if (panels_layout.horizontal_split)
|
|
{
|
|
panels_layout.horizontal_equal = equal_split;
|
|
panels_layout.left_panel_size = first_panel_size;
|
|
}
|
|
else
|
|
{
|
|
panels_layout.vertical_equal = equal_split;
|
|
panels_layout.top_panel_size = first_panel_size;
|
|
}
|
|
|
|
/* actual options override legacy ones */
|
|
for (i = 0; layout[i].opt_name != NULL; i++)
|
|
*layout[i].opt_addr = mc_config_get_int (mc_main_config, CONFIG_LAYOUT_SECTION,
|
|
layout[i].opt_name, *layout[i].opt_addr);
|
|
|
|
/* remove legacy options */
|
|
mc_config_del_key (mc_main_config, CONFIG_APP_SECTION, "horizontal_split");
|
|
mc_config_del_key (mc_main_config, "Layout", "equal_split");
|
|
mc_config_del_key (mc_main_config, "Layout", "first_panel_size");
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
static void
|
|
load_keys_from_section (const char *terminal, mc_config_t * cfg)
|
|
{
|
|
char *section_name;
|
|
gchar **profile_keys, **keys;
|
|
gchar **values, **curr_values;
|
|
char *valcopy, *value;
|
|
long key_code;
|
|
gsize len, values_len;
|
|
|
|
if (terminal == NULL)
|
|
return;
|
|
|
|
section_name = g_strconcat ("terminal:", terminal, (char *) NULL);
|
|
profile_keys = keys = mc_config_get_keys (cfg, section_name, &len);
|
|
|
|
while (*profile_keys != NULL)
|
|
{
|
|
/* copy=other causes all keys from [terminal:other] to be loaded. */
|
|
if (g_ascii_strcasecmp (*profile_keys, "copy") == 0)
|
|
{
|
|
valcopy = mc_config_get_string (cfg, section_name, *profile_keys, "");
|
|
load_keys_from_section (valcopy, cfg);
|
|
g_free (valcopy);
|
|
profile_keys++;
|
|
continue;
|
|
}
|
|
|
|
curr_values = values =
|
|
mc_config_get_string_list (cfg, section_name, *profile_keys, &values_len);
|
|
|
|
key_code = lookup_key (*profile_keys, NULL);
|
|
|
|
if (key_code != 0)
|
|
{
|
|
if (curr_values != NULL)
|
|
{
|
|
while (*curr_values != NULL)
|
|
{
|
|
valcopy = convert_controls (*curr_values);
|
|
define_sequence (key_code, valcopy, MCKEY_NOACTION);
|
|
g_free (valcopy);
|
|
curr_values++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
value = mc_config_get_string (cfg, section_name, *profile_keys, "");
|
|
valcopy = convert_controls (value);
|
|
define_sequence (key_code, valcopy, MCKEY_NOACTION);
|
|
g_free (valcopy);
|
|
g_free (value);
|
|
}
|
|
}
|
|
|
|
profile_keys++;
|
|
g_strfreev (values);
|
|
}
|
|
g_strfreev (keys);
|
|
g_free (section_name);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
static void
|
|
load_keymap_from_section (const char *section_name, GArray * keymap, mc_config_t * cfg)
|
|
{
|
|
gchar **profile_keys, **keys;
|
|
gsize len;
|
|
|
|
if (section_name == NULL)
|
|
return;
|
|
|
|
profile_keys = keys = mc_config_get_keys (cfg, section_name, &len);
|
|
|
|
while (*profile_keys != NULL)
|
|
{
|
|
gchar **values, **curr_values;
|
|
|
|
curr_values = values = mc_config_get_string_list (cfg, section_name, *profile_keys, &len);
|
|
|
|
if (curr_values != NULL)
|
|
{
|
|
int action;
|
|
|
|
action = keybind_lookup_action (*profile_keys);
|
|
if (action > 0)
|
|
while (*curr_values != NULL)
|
|
{
|
|
keybind_cmd_bind (keymap, *curr_values, action);
|
|
curr_values++;
|
|
}
|
|
|
|
g_strfreev (values);
|
|
}
|
|
|
|
profile_keys++;
|
|
}
|
|
|
|
g_strfreev (keys);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
static mc_config_t *
|
|
load_setup_get_keymap_profile_config (gboolean load_from_file)
|
|
{
|
|
/*
|
|
TODO: IMHO, in future, this function shall be placed in mcconfig module.
|
|
*/
|
|
mc_config_t *keymap_config;
|
|
char *share_keymap, *sysconfig_keymap;
|
|
char *fname, *fname2;
|
|
|
|
/* 0) Create default keymap */
|
|
keymap_config = create_default_keymap ();
|
|
if (!load_from_file)
|
|
return keymap_config;
|
|
|
|
/* load and merge global keymaps */
|
|
|
|
/* 1) /usr/share/mc (mc_global.share_data_dir) */
|
|
share_keymap = g_build_filename (mc_global.share_data_dir, GLOBAL_KEYMAP_FILE, NULL);
|
|
load_setup_init_config_from_file (&keymap_config, share_keymap, TRUE);
|
|
|
|
/* 2) /etc/mc (mc_global.sysconfig_dir) */
|
|
sysconfig_keymap = g_build_filename (mc_global.sysconfig_dir, GLOBAL_KEYMAP_FILE, NULL);
|
|
load_setup_init_config_from_file (&keymap_config, sysconfig_keymap, TRUE);
|
|
|
|
/* then load and merge one of user-defined keymap */
|
|
|
|
/* 3) --keymap=<keymap> */
|
|
fname = load_setup_get_full_config_name (NULL, mc_args__keymap_file);
|
|
if (fname != NULL && strcmp (fname, sysconfig_keymap) != 0 && strcmp (fname, share_keymap) != 0)
|
|
{
|
|
load_setup_init_config_from_file (&keymap_config, fname, TRUE);
|
|
goto done;
|
|
}
|
|
g_free (fname);
|
|
|
|
/* 4) getenv("MC_KEYMAP") */
|
|
fname = load_setup_get_full_config_name (NULL, g_getenv ("MC_KEYMAP"));
|
|
if (fname != NULL && strcmp (fname, sysconfig_keymap) != 0 && strcmp (fname, share_keymap) != 0)
|
|
{
|
|
load_setup_init_config_from_file (&keymap_config, fname, TRUE);
|
|
goto done;
|
|
}
|
|
g_free (fname);
|
|
|
|
/* 5) main config; [Midnight Commander] -> keymap */
|
|
fname2 = mc_config_get_string (mc_main_config, CONFIG_APP_SECTION, "keymap", NULL);
|
|
if (fname2 != NULL && *fname2 != '\0')
|
|
fname = load_setup_get_full_config_name (NULL, fname2);
|
|
g_free (fname2);
|
|
if (fname != NULL && strcmp (fname, sysconfig_keymap) != 0 && strcmp (fname, share_keymap) != 0)
|
|
{
|
|
load_setup_init_config_from_file (&keymap_config, fname, TRUE);
|
|
goto done;
|
|
}
|
|
g_free (fname);
|
|
|
|
/* 6) ${XDG_CONFIG_HOME}/mc/mc.keymap */
|
|
fname = mc_config_get_full_path (GLOBAL_KEYMAP_FILE);
|
|
load_setup_init_config_from_file (&keymap_config, fname, TRUE);
|
|
|
|
done:
|
|
g_free (fname);
|
|
g_free (sysconfig_keymap);
|
|
g_free (share_keymap);
|
|
|
|
return keymap_config;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
static panel_view_mode_t
|
|
setup__load_panel_state (const char *section)
|
|
{
|
|
char *buffer;
|
|
size_t i;
|
|
panel_view_mode_t mode = view_listing;
|
|
|
|
/* Load the display mode */
|
|
buffer = mc_config_get_string (mc_panels_config, section, "display", "listing");
|
|
|
|
for (i = 0; panel_types[i].opt_name != NULL; i++)
|
|
if (g_ascii_strcasecmp (panel_types[i].opt_name, buffer) == 0)
|
|
{
|
|
mode = panel_types[i].opt_type;
|
|
break;
|
|
}
|
|
|
|
g_free (buffer);
|
|
|
|
return mode;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
static void
|
|
panel_save_type (const char *section, panel_view_mode_t type)
|
|
{
|
|
size_t i;
|
|
|
|
for (i = 0; panel_types[i].opt_name != NULL; i++)
|
|
if (panel_types[i].opt_type == type)
|
|
{
|
|
mc_config_set_string (mc_panels_config, section, "display", panel_types[i].opt_name);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/* save panels.ini */
|
|
static void
|
|
save_panel_types (void)
|
|
{
|
|
panel_view_mode_t type;
|
|
|
|
if (mc_global.mc_run_mode != MC_RUN_FULL)
|
|
return;
|
|
|
|
type = get_display_type (0);
|
|
panel_save_type ("New Left Panel", type);
|
|
if (type == view_listing)
|
|
panel_save_setup (left_panel, left_panel->panel_name);
|
|
type = get_display_type (1);
|
|
panel_save_type ("New Right Panel", type);
|
|
if (type == view_listing)
|
|
panel_save_setup (right_panel, right_panel->panel_name);
|
|
|
|
{
|
|
char *dirs;
|
|
|
|
dirs = get_panel_dir_for (other_panel);
|
|
mc_config_set_string (mc_panels_config, "Dirs", "other_dir", dirs);
|
|
g_free (dirs);
|
|
}
|
|
|
|
if (current_panel != NULL)
|
|
mc_config_set_bool (mc_panels_config, "Dirs", "current_is_left", get_current_index () == 0);
|
|
|
|
if (mc_panels_config->ini_path == NULL)
|
|
mc_panels_config->ini_path = g_strdup (panels_profile_name);
|
|
|
|
mc_config_del_group (mc_panels_config, "Temporal:New Left Panel");
|
|
mc_config_del_group (mc_panels_config, "Temporal:New Right Panel");
|
|
|
|
mc_config_save_file (mc_panels_config, NULL);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
/*** public functions ****************************************************************************/
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
const char *
|
|
setup_init (void)
|
|
{
|
|
if (profile_name == NULL)
|
|
{
|
|
char *profile;
|
|
|
|
profile = mc_config_get_full_path (MC_CONFIG_FILE);
|
|
if (!exist_file (profile))
|
|
{
|
|
char *inifile;
|
|
|
|
inifile = mc_build_filename (mc_global.sysconfig_dir, "mc.ini", NULL);
|
|
if (exist_file (inifile))
|
|
{
|
|
g_free (profile);
|
|
profile = inifile;
|
|
}
|
|
else
|
|
{
|
|
g_free (inifile);
|
|
inifile = mc_build_filename (mc_global.share_data_dir, "mc.ini", NULL);
|
|
if (!exist_file (inifile))
|
|
g_free (inifile);
|
|
else
|
|
{
|
|
g_free (profile);
|
|
profile = inifile;
|
|
}
|
|
}
|
|
}
|
|
|
|
profile_name = profile;
|
|
}
|
|
|
|
return profile_name;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
void
|
|
load_setup (void)
|
|
{
|
|
const char *profile;
|
|
size_t i;
|
|
char *buffer;
|
|
const char *kt;
|
|
|
|
#ifdef HAVE_CHARSET
|
|
load_codepages_list ();
|
|
#endif /* HAVE_CHARSET */
|
|
|
|
profile = setup_init ();
|
|
|
|
/* mc.lib is common for all users, but has priority lower than
|
|
${XDG_CONFIG_HOME}/mc/ini. FIXME: it's only used for keys and treestore now */
|
|
global_profile_name =
|
|
g_build_filename (mc_global.sysconfig_dir, MC_GLOBAL_CONFIG_FILE, (char *) NULL);
|
|
if (!exist_file (global_profile_name))
|
|
{
|
|
g_free (global_profile_name);
|
|
global_profile_name =
|
|
g_build_filename (mc_global.share_data_dir, MC_GLOBAL_CONFIG_FILE, (char *) NULL);
|
|
}
|
|
|
|
panels_profile_name = mc_config_get_full_path (MC_PANELS_FILE);
|
|
|
|
mc_main_config = mc_config_init (profile, FALSE);
|
|
|
|
if (!exist_file (panels_profile_name))
|
|
setup__move_panels_config_into_separate_file (profile);
|
|
|
|
mc_panels_config = mc_config_init (panels_profile_name, FALSE);
|
|
|
|
/* Load integer boolean options */
|
|
for (i = 0; int_options[i].opt_name != NULL; i++)
|
|
*int_options[i].opt_addr =
|
|
mc_config_get_int (mc_main_config, CONFIG_APP_SECTION, int_options[i].opt_name,
|
|
*int_options[i].opt_addr);
|
|
#ifndef USE_INTERNAL_EDIT
|
|
/* reset forced in case of build without internal editor */
|
|
use_internal_edit = 0;
|
|
#endif /* USE_INTERNAL_EDIT */
|
|
|
|
if (option_tab_spacing <= 0)
|
|
option_tab_spacing = DEFAULT_TAB_SPACING;
|
|
|
|
#ifdef USE_INTERNAL_EDIT
|
|
if (option_word_wrap_line_length <= 0)
|
|
option_word_wrap_line_length = DEFAULT_WRAP_LINE_LENGTH;
|
|
#endif /* USE_INTERNAL_EDIT */
|
|
|
|
/* overwrite old_esc_mode_timeout */
|
|
kt = getenv ("KEYBOARD_KEY_TIMEOUT_US");
|
|
if ((kt != NULL) && (kt[0] != '\0'))
|
|
old_esc_mode_timeout = atoi (kt);
|
|
|
|
/* Load string options */
|
|
for (i = 0; str_options[i].opt_name != NULL; i++)
|
|
*str_options[i].opt_addr =
|
|
mc_config_get_string (mc_main_config, CONFIG_APP_SECTION, str_options[i].opt_name,
|
|
str_options[i].opt_defval);
|
|
|
|
load_layout ();
|
|
panels_load_options ();
|
|
load_panelize ();
|
|
|
|
startup_left_mode = setup__load_panel_state ("New Left Panel");
|
|
startup_right_mode = setup__load_panel_state ("New Right Panel");
|
|
|
|
/* At least one of the panels is a listing panel */
|
|
if (startup_left_mode != view_listing && startup_right_mode != view_listing)
|
|
startup_left_mode = view_listing;
|
|
|
|
{
|
|
vfs_path_t *vpath;
|
|
|
|
buffer = mc_config_get_string (mc_panels_config, "Dirs", "other_dir", ".");
|
|
vpath = vfs_path_from_str (buffer);
|
|
if (vfs_file_is_local (vpath))
|
|
saved_other_dir = buffer;
|
|
else
|
|
g_free (buffer);
|
|
vfs_path_free (vpath);
|
|
}
|
|
|
|
boot_current_is_left = mc_config_get_bool (mc_panels_config, "Dirs", "current_is_left", TRUE);
|
|
|
|
/* Load time formats */
|
|
user_recent_timeformat =
|
|
mc_config_get_string (mc_main_config, CONFIG_MISC_SECTION, "timeformat_recent", FMTTIME);
|
|
user_old_timeformat =
|
|
mc_config_get_string (mc_main_config, CONFIG_MISC_SECTION, "timeformat_old", FMTYEAR);
|
|
|
|
#ifdef ENABLE_VFS_FTP
|
|
ftpfs_proxy_host =
|
|
mc_config_get_string (mc_main_config, CONFIG_MISC_SECTION, "ftp_proxy_host", "gate");
|
|
ftpfs_ignore_chattr_errors =
|
|
mc_config_get_bool (mc_main_config, CONFIG_APP_SECTION, "ignore_ftp_chattr_errors", TRUE);
|
|
ftpfs_init_passwd ();
|
|
#endif /* ENABLE_VFS_FTP */
|
|
|
|
/* The default color and the terminal dependent color */
|
|
mc_global.tty.setup_color_string =
|
|
mc_config_get_string (mc_main_config, "Colors", "base_color", "");
|
|
mc_global.tty.term_color_string =
|
|
mc_config_get_string (mc_main_config, "Colors", getenv ("TERM"), "");
|
|
mc_global.tty.color_terminal_string =
|
|
mc_config_get_string (mc_main_config, "Colors", "color_terminals", "");
|
|
|
|
/* Load the directory history */
|
|
/* directory_history_load (); */
|
|
/* Remove the temporal entries */
|
|
|
|
#ifdef HAVE_CHARSET
|
|
if (codepages->len > 1)
|
|
{
|
|
buffer = mc_config_get_string (mc_main_config, CONFIG_MISC_SECTION, "display_codepage", "");
|
|
if (buffer[0] != '\0')
|
|
{
|
|
mc_global.display_codepage = get_codepage_index (buffer);
|
|
cp_display = get_codepage_id (mc_global.display_codepage);
|
|
}
|
|
g_free (buffer);
|
|
buffer = mc_config_get_string (mc_main_config, CONFIG_MISC_SECTION, "source_codepage", "");
|
|
if (buffer[0] != '\0')
|
|
{
|
|
default_source_codepage = get_codepage_index (buffer);
|
|
mc_global.source_codepage = default_source_codepage; /* May be source_codepage doesn't need this */
|
|
cp_source = get_codepage_id (mc_global.source_codepage);
|
|
}
|
|
g_free (buffer);
|
|
}
|
|
|
|
autodetect_codeset =
|
|
mc_config_get_string (mc_main_config, CONFIG_MISC_SECTION, "autodetect_codeset", "");
|
|
if ((autodetect_codeset[0] != '\0') && (strcmp (autodetect_codeset, "off") != 0))
|
|
is_autodetect_codeset_enabled = TRUE;
|
|
|
|
g_free (init_translation_table (mc_global.source_codepage, mc_global.display_codepage));
|
|
buffer = (char *) get_codepage_id (mc_global.display_codepage);
|
|
if (buffer != NULL)
|
|
mc_global.utf8_display = str_isutf8 (buffer);
|
|
#endif /* HAVE_CHARSET */
|
|
|
|
#ifdef HAVE_ASPELL
|
|
spell_language =
|
|
mc_config_get_string (mc_main_config, CONFIG_MISC_SECTION, "spell_language", "en");
|
|
#endif /* HAVE_ASPELL */
|
|
|
|
clipboard_store_path =
|
|
mc_config_get_string (mc_main_config, CONFIG_MISC_SECTION, "clipboard_store", "");
|
|
clipboard_paste_path =
|
|
mc_config_get_string (mc_main_config, CONFIG_MISC_SECTION, "clipboard_paste", "");
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
gboolean
|
|
save_setup (gboolean save_options, gboolean save_panel_options)
|
|
{
|
|
gboolean ret = TRUE;
|
|
|
|
saving_setup = 1;
|
|
|
|
save_hotlist ();
|
|
|
|
if (save_panel_options)
|
|
save_panel_types ();
|
|
|
|
if (save_options)
|
|
{
|
|
char *tmp_profile;
|
|
|
|
save_config ();
|
|
save_layout ();
|
|
panels_save_options ();
|
|
save_panelize ();
|
|
/* directory_history_save (); */
|
|
|
|
#ifdef ENABLE_VFS_FTP
|
|
mc_config_set_string (mc_main_config, CONFIG_MISC_SECTION, "ftpfs_password",
|
|
ftpfs_anonymous_passwd);
|
|
if (ftpfs_proxy_host)
|
|
mc_config_set_string (mc_main_config, CONFIG_MISC_SECTION, "ftp_proxy_host",
|
|
ftpfs_proxy_host);
|
|
#endif /* ENABLE_VFS_FTP */
|
|
|
|
#ifdef HAVE_CHARSET
|
|
mc_config_set_string (mc_main_config, CONFIG_MISC_SECTION, "display_codepage",
|
|
get_codepage_id (mc_global.display_codepage));
|
|
mc_config_set_string (mc_main_config, CONFIG_MISC_SECTION, "source_codepage",
|
|
get_codepage_id (default_source_codepage));
|
|
mc_config_set_string (mc_main_config, CONFIG_MISC_SECTION, "autodetect_codeset",
|
|
autodetect_codeset);
|
|
#endif /* HAVE_CHARSET */
|
|
|
|
#ifdef HAVE_ASPELL
|
|
mc_config_set_string (mc_main_config, CONFIG_MISC_SECTION, "spell_language",
|
|
spell_language);
|
|
#endif /* HAVE_ASPELL */
|
|
|
|
mc_config_set_string (mc_main_config, CONFIG_MISC_SECTION, "clipboard_store",
|
|
clipboard_store_path);
|
|
mc_config_set_string (mc_main_config, CONFIG_MISC_SECTION, "clipboard_paste",
|
|
clipboard_paste_path);
|
|
|
|
tmp_profile = mc_config_get_full_path (MC_CONFIG_FILE);
|
|
ret = mc_config_save_to_file (mc_main_config, tmp_profile, NULL);
|
|
g_free (tmp_profile);
|
|
}
|
|
|
|
saving_setup = 0;
|
|
|
|
return ret;
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
void
|
|
done_setup (void)
|
|
{
|
|
size_t i;
|
|
|
|
g_free (clipboard_store_path);
|
|
g_free (clipboard_paste_path);
|
|
g_free (global_profile_name);
|
|
g_free (mc_global.tty.color_terminal_string);
|
|
g_free (mc_global.tty.term_color_string);
|
|
g_free (mc_global.tty.setup_color_string);
|
|
g_free (profile_name);
|
|
g_free (panels_profile_name);
|
|
mc_config_deinit (mc_main_config);
|
|
mc_config_deinit (mc_panels_config);
|
|
|
|
g_free (user_recent_timeformat);
|
|
g_free (user_old_timeformat);
|
|
|
|
for (i = 0; str_options[i].opt_name != NULL; i++)
|
|
g_free (*str_options[i].opt_addr);
|
|
|
|
done_hotlist ();
|
|
done_panelize ();
|
|
/* directory_history_free (); */
|
|
|
|
#ifdef HAVE_CHARSET
|
|
g_free (autodetect_codeset);
|
|
free_codepages_list ();
|
|
#endif
|
|
|
|
#ifdef HAVE_ASPELL
|
|
g_free (spell_language);
|
|
#endif /* HAVE_ASPELL */
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
void
|
|
save_config (void)
|
|
{
|
|
size_t i;
|
|
|
|
/* Save integer options */
|
|
for (i = 0; int_options[i].opt_name != NULL; i++)
|
|
mc_config_set_int (mc_main_config, CONFIG_APP_SECTION, int_options[i].opt_name,
|
|
*int_options[i].opt_addr);
|
|
|
|
/* Save string options */
|
|
for (i = 0; str_options[i].opt_name != NULL; i++)
|
|
mc_config_set_string (mc_main_config, CONFIG_APP_SECTION, str_options[i].opt_name,
|
|
*str_options[i].opt_addr);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
void
|
|
setup_save_config_show_error (const char *filename, GError ** error)
|
|
{
|
|
if (error != NULL && *error != NULL)
|
|
{
|
|
message (D_ERROR, MSG_ERROR, _("Cannot save file %s:\n%s"), filename, (*error)->message);
|
|
g_error_free (*error);
|
|
*error = NULL;
|
|
}
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
void
|
|
save_layout (void)
|
|
{
|
|
size_t i;
|
|
|
|
/* Save integer options */
|
|
for (i = 0; layout[i].opt_name != NULL; i++)
|
|
mc_config_set_int (mc_main_config, CONFIG_LAYOUT_SECTION, layout[i].opt_name,
|
|
*layout[i].opt_addr);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
void
|
|
load_key_defs (void)
|
|
{
|
|
/*
|
|
* Load keys from mc.lib before ${XDG_CONFIG_HOME}/mc/ini, so that the user
|
|
* definitions override global settings.
|
|
*/
|
|
mc_config_t *mc_global_config;
|
|
|
|
mc_global_config = mc_config_init (global_profile_name, FALSE);
|
|
if (mc_global_config != NULL)
|
|
{
|
|
load_keys_from_section ("general", mc_global_config);
|
|
load_keys_from_section (getenv ("TERM"), mc_global_config);
|
|
mc_config_deinit (mc_global_config);
|
|
}
|
|
|
|
load_keys_from_section ("general", mc_main_config);
|
|
load_keys_from_section (getenv ("TERM"), mc_main_config);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
#ifdef ENABLE_VFS_FTP
|
|
char *
|
|
load_anon_passwd (void)
|
|
{
|
|
char *buffer;
|
|
|
|
buffer = mc_config_get_string (mc_main_config, CONFIG_MISC_SECTION, "ftpfs_password", "");
|
|
|
|
if ((buffer != NULL) && (buffer[0] != '\0'))
|
|
return buffer;
|
|
|
|
g_free (buffer);
|
|
return NULL;
|
|
}
|
|
#endif /* ENABLE_VFS_FTP */
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
void
|
|
load_keymap_defs (gboolean load_from_file)
|
|
{
|
|
/*
|
|
* Load keymap from GLOBAL_KEYMAP_FILE before ${XDG_CONFIG_HOME}/mc/mc.keymap, so that the user
|
|
* definitions override global settings.
|
|
*/
|
|
mc_config_t *mc_global_keymap;
|
|
|
|
mc_global_keymap = load_setup_get_keymap_profile_config (load_from_file);
|
|
|
|
if (mc_global_keymap != NULL)
|
|
{
|
|
main_keymap = g_array_new (TRUE, FALSE, sizeof (global_keymap_t));
|
|
load_keymap_from_section (KEYMAP_SECTION_MAIN, main_keymap, mc_global_keymap);
|
|
main_x_keymap = g_array_new (TRUE, FALSE, sizeof (global_keymap_t));
|
|
load_keymap_from_section (KEYMAP_SECTION_MAIN_EXT, main_x_keymap, mc_global_keymap);
|
|
|
|
panel_keymap = g_array_new (TRUE, FALSE, sizeof (global_keymap_t));
|
|
load_keymap_from_section (KEYMAP_SECTION_PANEL, panel_keymap, mc_global_keymap);
|
|
|
|
dialog_keymap = g_array_new (TRUE, FALSE, sizeof (global_keymap_t));
|
|
load_keymap_from_section (KEYMAP_SECTION_DIALOG, dialog_keymap, mc_global_keymap);
|
|
|
|
input_keymap = g_array_new (TRUE, FALSE, sizeof (global_keymap_t));
|
|
load_keymap_from_section (KEYMAP_SECTION_INPUT, input_keymap, mc_global_keymap);
|
|
|
|
listbox_keymap = g_array_new (TRUE, FALSE, sizeof (global_keymap_t));
|
|
load_keymap_from_section (KEYMAP_SECTION_LISTBOX, listbox_keymap, mc_global_keymap);
|
|
|
|
tree_keymap = g_array_new (TRUE, FALSE, sizeof (global_keymap_t));
|
|
load_keymap_from_section (KEYMAP_SECTION_TREE, tree_keymap, mc_global_keymap);
|
|
|
|
help_keymap = g_array_new (TRUE, FALSE, sizeof (global_keymap_t));
|
|
load_keymap_from_section (KEYMAP_SECTION_HELP, help_keymap, mc_global_keymap);
|
|
|
|
#ifdef USE_INTERNAL_EDIT
|
|
editor_keymap = g_array_new (TRUE, FALSE, sizeof (global_keymap_t));
|
|
load_keymap_from_section (KEYMAP_SECTION_EDITOR, editor_keymap, mc_global_keymap);
|
|
editor_x_keymap = g_array_new (TRUE, FALSE, sizeof (global_keymap_t));
|
|
load_keymap_from_section (KEYMAP_SECTION_EDITOR_EXT, editor_x_keymap, mc_global_keymap);
|
|
#endif
|
|
|
|
viewer_keymap = g_array_new (TRUE, FALSE, sizeof (global_keymap_t));
|
|
load_keymap_from_section (KEYMAP_SECTION_VIEWER, viewer_keymap, mc_global_keymap);
|
|
viewer_hex_keymap = g_array_new (TRUE, FALSE, sizeof (global_keymap_t));
|
|
load_keymap_from_section (KEYMAP_SECTION_VIEWER_HEX, viewer_hex_keymap, mc_global_keymap);
|
|
|
|
#ifdef USE_DIFF_VIEW
|
|
diff_keymap = g_array_new (TRUE, FALSE, sizeof (global_keymap_t));
|
|
load_keymap_from_section (KEYMAP_SECTION_DIFFVIEWER, diff_keymap, mc_global_keymap);
|
|
#endif
|
|
|
|
mc_config_deinit (mc_global_keymap);
|
|
}
|
|
|
|
main_map = (global_keymap_t *) main_keymap->data;
|
|
main_x_map = (global_keymap_t *) main_x_keymap->data;
|
|
panel_map = (global_keymap_t *) panel_keymap->data;
|
|
dialog_map = (global_keymap_t *) dialog_keymap->data;
|
|
input_map = (global_keymap_t *) input_keymap->data;
|
|
listbox_map = (global_keymap_t *) listbox_keymap->data;
|
|
tree_map = (global_keymap_t *) tree_keymap->data;
|
|
help_map = (global_keymap_t *) help_keymap->data;
|
|
#ifdef USE_INTERNAL_EDIT
|
|
editor_map = (global_keymap_t *) editor_keymap->data;
|
|
editor_x_map = (global_keymap_t *) editor_x_keymap->data;
|
|
#endif
|
|
viewer_map = (global_keymap_t *) viewer_keymap->data;
|
|
viewer_hex_map = (global_keymap_t *) viewer_hex_keymap->data;
|
|
#ifdef USE_DIFF_VIEW
|
|
diff_map = (global_keymap_t *) diff_keymap->data;
|
|
#endif
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
void
|
|
free_keymap_defs (void)
|
|
{
|
|
if (main_keymap != NULL)
|
|
g_array_free (main_keymap, TRUE);
|
|
if (main_x_keymap != NULL)
|
|
g_array_free (main_x_keymap, TRUE);
|
|
if (panel_keymap != NULL)
|
|
g_array_free (panel_keymap, TRUE);
|
|
if (dialog_keymap != NULL)
|
|
g_array_free (dialog_keymap, TRUE);
|
|
if (input_keymap != NULL)
|
|
g_array_free (input_keymap, TRUE);
|
|
if (listbox_keymap != NULL)
|
|
g_array_free (listbox_keymap, TRUE);
|
|
if (tree_keymap != NULL)
|
|
g_array_free (tree_keymap, TRUE);
|
|
if (help_keymap != NULL)
|
|
g_array_free (help_keymap, TRUE);
|
|
#ifdef USE_INTERNAL_EDIT
|
|
if (editor_keymap != NULL)
|
|
g_array_free (editor_keymap, TRUE);
|
|
if (editor_x_keymap != NULL)
|
|
g_array_free (editor_x_keymap, TRUE);
|
|
#endif
|
|
if (viewer_keymap != NULL)
|
|
g_array_free (viewer_keymap, TRUE);
|
|
if (viewer_hex_keymap != NULL)
|
|
g_array_free (viewer_hex_keymap, TRUE);
|
|
#ifdef USE_DIFF_VIEW
|
|
if (diff_keymap != NULL)
|
|
g_array_free (diff_keymap, TRUE);
|
|
#endif
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
void
|
|
panel_load_setup (WPanel * panel, const char *section)
|
|
{
|
|
size_t i;
|
|
char *buffer, buffer2[BUF_TINY];
|
|
|
|
panel->sort_info.reverse = mc_config_get_int (mc_panels_config, section, "reverse", 0);
|
|
panel->sort_info.case_sensitive =
|
|
mc_config_get_int (mc_panels_config, section, "case_sensitive",
|
|
OS_SORT_CASE_SENSITIVE_DEFAULT);
|
|
panel->sort_info.exec_first = mc_config_get_int (mc_panels_config, section, "exec_first", 0);
|
|
|
|
/* Load sort order */
|
|
buffer = mc_config_get_string (mc_panels_config, section, "sort_order", "name");
|
|
panel->sort_field = panel_get_field_by_id (buffer);
|
|
if (panel->sort_field == NULL)
|
|
panel->sort_field = panel_get_field_by_id ("name");
|
|
|
|
g_free (buffer);
|
|
|
|
/* Load the listing mode */
|
|
buffer = mc_config_get_string (mc_panels_config, section, "list_mode", "full");
|
|
panel->list_type = list_full;
|
|
for (i = 0; list_types[i].key != NULL; i++)
|
|
if (g_ascii_strcasecmp (list_types[i].key, buffer) == 0)
|
|
{
|
|
panel->list_type = list_types[i].list_type;
|
|
break;
|
|
}
|
|
g_free (buffer);
|
|
|
|
/* User formats */
|
|
g_free (panel->user_format);
|
|
panel->user_format =
|
|
mc_config_get_string (mc_panels_config, section, "user_format", DEFAULT_USER_FORMAT);
|
|
|
|
for (i = 0; i < LIST_TYPES; i++)
|
|
{
|
|
g_free (panel->user_status_format[i]);
|
|
g_snprintf (buffer2, BUF_TINY, "user_status%lld", (long long) i);
|
|
panel->user_status_format[i] =
|
|
mc_config_get_string (mc_panels_config, section, buffer2, DEFAULT_USER_FORMAT);
|
|
}
|
|
|
|
panel->user_mini_status = mc_config_get_int (mc_panels_config, section, "user_mini_status", 0);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
void
|
|
panel_save_setup (struct WPanel *panel, const char *section)
|
|
{
|
|
char buffer[BUF_TINY];
|
|
size_t i;
|
|
|
|
mc_config_set_int (mc_panels_config, section, "reverse", panel->sort_info.reverse);
|
|
mc_config_set_int (mc_panels_config, section, "case_sensitive",
|
|
panel->sort_info.case_sensitive);
|
|
mc_config_set_int (mc_panels_config, section, "exec_first", panel->sort_info.exec_first);
|
|
|
|
mc_config_set_string (mc_panels_config, section, "sort_order", panel->sort_field->id);
|
|
|
|
for (i = 0; list_types[i].key != NULL; i++)
|
|
if (list_types[i].list_type == panel->list_type)
|
|
{
|
|
mc_config_set_string (mc_panels_config, section, "list_mode", list_types[i].key);
|
|
break;
|
|
}
|
|
|
|
mc_config_set_string (mc_panels_config, section, "user_format", panel->user_format);
|
|
|
|
for (i = 0; i < LIST_TYPES; i++)
|
|
{
|
|
g_snprintf (buffer, BUF_TINY, "user_status%lld", (long long) i);
|
|
mc_config_set_string (mc_panels_config, section, buffer, panel->user_status_format[i]);
|
|
}
|
|
|
|
mc_config_set_int (mc_panels_config, section, "user_mini_status", panel->user_mini_status);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/**
|
|
Load panels options from [Panels] section.
|
|
*/
|
|
void
|
|
panels_load_options (void)
|
|
{
|
|
if (mc_config_has_group (mc_main_config, CONFIG_PANELS_SECTION))
|
|
{
|
|
size_t i;
|
|
int qmode;
|
|
|
|
for (i = 0; panels_ini_options[i].opt_name != NULL; i++)
|
|
*panels_ini_options[i].opt_addr =
|
|
mc_config_get_bool (mc_main_config, CONFIG_PANELS_SECTION,
|
|
panels_ini_options[i].opt_name,
|
|
*panels_ini_options[i].opt_addr);
|
|
|
|
qmode = mc_config_get_int (mc_main_config, CONFIG_PANELS_SECTION,
|
|
"quick_search_mode", (int) panels_options.qsearch_mode);
|
|
if (qmode < 0)
|
|
panels_options.qsearch_mode = QSEARCH_CASE_INSENSITIVE;
|
|
else if (qmode >= QSEARCH_NUM)
|
|
panels_options.qsearch_mode = QSEARCH_PANEL_CASE;
|
|
else
|
|
panels_options.qsearch_mode = (qsearch_mode_t) qmode;
|
|
}
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/**
|
|
Save panels options in [Panels] section.
|
|
*/
|
|
void
|
|
panels_save_options (void)
|
|
{
|
|
size_t i;
|
|
|
|
for (i = 0; panels_ini_options[i].opt_name != NULL; i++)
|
|
mc_config_set_bool (mc_main_config, CONFIG_PANELS_SECTION,
|
|
panels_ini_options[i].opt_name, *panels_ini_options[i].opt_addr);
|
|
|
|
mc_config_set_int (mc_main_config, CONFIG_PANELS_SECTION,
|
|
"quick_search_mode", (int) panels_options.qsearch_mode);
|
|
}
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|