/* ncmpc (Ncurses MPD Client) * (c) 2004-2009 The Music Player Daemon Project * Project homepage: http://musicpd.org * This program 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 2 of the License, or * (at your option) any later version. * This program 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, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include "conf.h" #include "config.h" #include "defaults.h" #include "i18n.h" #include "command.h" #include "colors.h" #include "screen_list.h" #include #include #include #include #include #include #include #include #include #include #define MAX_LINE_LENGTH 1024 #define COMMENT_TOKEN '#' /* configuration field names */ #define CONF_ENABLE_COLORS "enable-colors" #define CONF_AUTO_CENTER "auto-center" #define CONF_WIDE_CURSOR "wide-cursor" #define CONF_ENABLE_BELL "enable-bell" #define CONF_KEY_DEFINITION "key" #define CONF_COLOR "color" #define CONF_COLOR_DEFINITION "colordef" #define CONF_LIST_FORMAT "list-format" #define CONF_STATUS_FORMAT "status-format" #define CONF_XTERM_TITLE_FORMAT "xterm-title-format" #define CONF_LIST_WRAP "wrap-around" #define CONF_FIND_WRAP "find-wrap" #define CONF_FIND_SHOW_LAST "find-show-last" #define CONF_AUDIBLE_BELL "audible-bell" #define CONF_VISIBLE_BELL "visible-bell" #define CONF_XTERM_TITLE "set-xterm-title" #define CONF_ENABLE_MOUSE "enable-mouse" #define CONF_CROSSFADE_TIME "crossfade-time" #define CONF_SEARCH_MODE "search-mode" #define CONF_HIDE_CURSOR "hide-cursor" #define CONF_SEEK_TIME "seek-time" #define CONF_SCREEN_LIST "screen-list" #define CONF_TIMEDISPLAY_TYPE "timedisplay-type" #define CONF_HOST "host" #define CONF_PORT "port" #define CONF_PASSWORD "password" #define CONF_LYRICS_TIMEOUT "lyrics-timeout" #define CONF_SHOW_SPLASH "show-splash" #define CONF_SCROLL "scroll" #define CONF_SCROLL_SEP "scroll-sep" #define CONF_VISIBLE_BITRATE "visible-bitrate" #define CONF_WELCOME_SCREEN_LIST "welcome-screen-list" static bool str2bool(char *str) { return strcasecmp(str, "yes") == 0 || strcasecmp(str, "true") == 0 || strcasecmp(str, "on") == 0 || strcasecmp(str, "1") == 0; } static void print_error(const char *msg, const char *input) { fprintf(stderr, "%s: %s ('%s')\n", /* To translators: prefix for error messages */ _("Error"), msg, input); } static int parse_key_value(char *str, char **end) { if (*str == '\'') { if (str[1] == '\'' || str[2] != '\'') { print_error(_("Malformed hotkey definition"), str); return -1; } *end = str + 3; return str[1]; } else { long value = strtol(str, end, 0); if (*end == str) { print_error(_("Malformed hotkey definition"), str); return -1; } return (int)value; } } static int parse_key_definition(char *str) { char buf[MAX_LINE_LENGTH]; char *p; size_t len = strlen(str), i; int j,key; int keys[MAX_COMMAND_KEYS]; command_t cmd; /* get the command name */ i=0; j=0; memset(buf, 0, MAX_LINE_LENGTH); while (i < len && str[i] != '=' && !g_ascii_isspace(str[i])) buf[j++] = str[i++]; if( (cmd=get_key_command_from_name(buf)) == CMD_NONE ) { /* the hotkey configuration contains an unknown command */ print_error(_("Unknown command"), buf); return -1; } /* skip whitespace */ while (i < len && (str[i] == '=' || g_ascii_isspace(str[i]))) i++; /* get the value part */ memset(buf, 0, MAX_LINE_LENGTH); g_strlcpy(buf, str+i, MAX_LINE_LENGTH); if (*buf == 0) { /* the hotkey configuration line is incomplete */ print_error(_("Incomplete hotkey configuration"), str); return -1; } /* parse key values */ i = 0; key = 0; p = buf; memset(keys, 0, sizeof(int)*MAX_COMMAND_KEYS); while (i < MAX_COMMAND_KEYS && *p != 0 && (key = parse_key_value(p, &p)) >= 0) { keys[i++] = key; while (*p==',' || *p==' ' || *p=='\t') p++; } if (key < 0) return -1; return assign_keys(cmd, keys); } static const char * parse_timedisplay_type(const char *str) { if (!strcmp(str,"elapsed") || !strcmp(str,"remaining")) return str; else { /* translators: ncmpc supports displaying the "elapsed" or "remaining" time of a song being played; in this case, the configuration file contained an invalid setting */ print_error(_("Bad time display type"), str); return DEFAULT_TIMEDISPLAY_TYPE; } } #ifdef ENABLE_COLORS static char * separate_value(char *p) { char *value; value = strchr(p, '='); if (value == NULL) { /* an equals sign '=' was expected while parsing a configuration file line */ fprintf(stderr, "%s\n", _("Missing '='")); return NULL; } *value++ = 0; g_strchomp(p); return g_strchug(value); } static int parse_color(char *str) { char *value; value = separate_value(str); if (value == NULL) return -1; return colors_assign(str, value); } /** * Returns the first non-whitespace character after the next comma * character, or the end of the string. This is used to parse comma * separated values. */ static char * after_comma(char *p) { char *comma = strchr(p, ','); if (comma != NULL) { *comma++ = 0; comma = g_strchug(comma); } else comma = p + strlen(p); g_strchomp(p); return comma; } static int parse_color_definition(char *str) { char buf[MAX_LINE_LENGTH]; char *value; short color, rgb[3]; value = separate_value(str); if (value == NULL) return -1; /* get the command name */ color = colors_str2color(str); if (color < 0) { print_error(_("Bad color name"), buf); return -1; } /* parse r,g,b values */ for (unsigned i = 0; i < 3; ++i) { char *next = after_comma(value), *endptr; if (*value == 0) { print_error(_("Incomplete color definition"), str); return -1; } rgb[i] = strtol(value, &endptr, 0); if (endptr == value || *endptr != 0) { print_error(_("Invalid number"), value); return -1; } value = next; } if (*value != 0) { print_error(_("Malformed color definition"), str); return -1; } return colors_define(str, rgb[0], rgb[1], rgb[2]); } #endif static char * get_format(char *str) { gsize len = strlen(str); if (str && str[0]=='\"' && str[len-1] == '\"') { str[len - 1] = '\0'; str++; } return g_strdup(str); } static char ** check_screen_list(char *value) { char **tmp = g_strsplit_set(value, " \t,", 100); char **screen = NULL; int i,j; i=0; j=0; while( tmp && tmp[i] ) { char *name = g_ascii_strdown(tmp[i], -1); if (screen_lookup_name(name) == NULL) { /* an unknown screen name was specified in the configuration file */ print_error(_("Unknown screen name"), name); free(name); } else { screen = g_realloc(screen, (j+2)*sizeof(char *)); screen[j++] = name; screen[j] = NULL; } i++; } g_strfreev(tmp); if( screen == NULL ) return g_strsplit_set(DEFAULT_SCREEN_LIST, " ", 0); return screen; } static bool parse_line(char *line) { size_t len = strlen(line), i = 0, j; char name[MAX_LINE_LENGTH]; char value[MAX_LINE_LENGTH]; bool match_found; /* get the name part */ j = 0; while (i < len && line[i] != '=' && !g_ascii_isspace(line[i])) { name[j++] = line[i++]; } name[j] = '\0'; /* skip '=' and whitespace */ while (i < len && (line[i] == '=' || g_ascii_isspace(line[i]))) i++; /* get the value part */ j = 0; while (i < len) value[j++] = line[i++]; value[j] = '\0'; match_found = true; /* key definition */ if (!strcasecmp(CONF_KEY_DEFINITION, name)) parse_key_definition(value); /* enable colors */ else if(!strcasecmp(CONF_ENABLE_COLORS, name)) #ifdef ENABLE_COLORS options.enable_colors = str2bool(value); #else {} #endif /* auto center */ else if (!strcasecmp(CONF_AUTO_CENTER, name)) options.auto_center = str2bool(value); /* color assignment */ else if (!strcasecmp(CONF_COLOR, name)) #ifdef ENABLE_COLORS parse_color(value); #else {} #endif /* wide cursor */ else if (!strcasecmp(CONF_WIDE_CURSOR, name)) options.wide_cursor = str2bool(value); /* welcome screen list */ else if (!strcasecmp(CONF_WELCOME_SCREEN_LIST, name)) options.welcome_screen_list = str2bool(value); /* visible bitrate */ else if (!strcasecmp(CONF_VISIBLE_BITRATE, name)) options.visible_bitrate = str2bool(value); /* timer display type */ else if (!strcasecmp(CONF_TIMEDISPLAY_TYPE, name)) { g_free(options.timedisplay_type); options.timedisplay_type=g_strdup(parse_timedisplay_type(value)); /* color definition */ } else if (!strcasecmp(CONF_COLOR_DEFINITION, name)) #ifdef ENABLE_COLORS parse_color_definition(value); #else {} #endif /* list format string */ else if (!strcasecmp(CONF_LIST_FORMAT, name)) { g_free(options.list_format); options.list_format = get_format(value); /* status format string */ } else if (!strcasecmp(CONF_STATUS_FORMAT, name)) { g_free(options.status_format); options.status_format = get_format(value); /* xterm title format string */ } else if (!strcasecmp(CONF_XTERM_TITLE_FORMAT, name)) { g_free(options.xterm_title_format); options.xterm_title_format = get_format(value); } else if (!strcasecmp(CONF_LIST_WRAP, name)) options.list_wrap = str2bool(value); else if (!strcasecmp(CONF_FIND_WRAP, name)) options.find_wrap = str2bool(value); else if (!strcasecmp(CONF_FIND_SHOW_LAST,name)) options.find_show_last_pattern = str2bool(value); else if (!strcasecmp(CONF_AUDIBLE_BELL, name)) options.audible_bell = str2bool(value); else if (!strcasecmp(CONF_VISIBLE_BELL, name)) options.visible_bell = str2bool(value); else if (!strcasecmp(CONF_XTERM_TITLE, name)) options.enable_xterm_title = str2bool(value); else if (!strcasecmp(CONF_ENABLE_MOUSE, name)) #ifdef HAVE_GETMOUSE options.enable_mouse = str2bool(value); #else {} #endif else if (!strcasecmp(CONF_CROSSFADE_TIME, name)) options.crossfade_time = atoi(value); else if (!strcasecmp(CONF_SEARCH_MODE, name)) options.search_mode = atoi(value); else if (!strcasecmp(CONF_HIDE_CURSOR, name)) options.hide_cursor = atoi(value); else if (!strcasecmp(CONF_SEEK_TIME, name)) options.seek_time = atoi(value); else if (!strcasecmp(CONF_SCREEN_LIST, name)) { g_strfreev(options.screen_list); options.screen_list = check_screen_list(value); } else if (!strcasecmp(CONF_SHOW_SPLASH, name)) { /* the splash screen was removed */ } else if (!strcasecmp(CONF_HOST, name)) options.host = get_format(value); else if (!strcasecmp(CONF_PORT, name)) options.port = atoi(get_format(value)); else if (!strcasecmp(CONF_PASSWORD, name)) options.password = get_format(value); else if (!strcasecmp(CONF_LYRICS_TIMEOUT, name)) #ifdef ENABLE_LYRICS_SCREEN options.lyrics_timeout = atoi(get_format(value)); #else {} #endif else if (!strcasecmp(CONF_SCROLL, name)) options.scroll = str2bool(value); else if (!strcasecmp(CONF_SCROLL_SEP, name)) { g_free(options.scroll_sep); options.scroll_sep = get_format(value); } else match_found = false; if (!match_found) print_error(_("Unknown configuration parameter"), name); return match_found; } static int read_rc_file(char *filename) { FILE *file; char line[MAX_LINE_LENGTH]; if (filename == NULL) return -1; file = fopen(filename, "r"); if (file == NULL) { perror(filename); return -1; } while (fgets(line, sizeof(line), file) != NULL) { char *p = g_strchug(line); if (*p != 0 && *p != COMMENT_TOKEN) parse_line(g_strchomp(p)); } fclose(file); return 0; } int check_user_conf_dir(void) { int retval; char *directory = g_build_filename(g_get_home_dir(), "." PACKAGE, NULL); if (g_file_test(directory, G_FILE_TEST_IS_DIR)) { g_free(directory); return 0; } retval = mkdir(directory, 0755); g_free(directory); return retval; } char * get_user_key_binding_filename(void) { return g_build_filename(g_get_home_dir(), "." PACKAGE, "keys", NULL); } int read_configuration(void) { char *filename = NULL; /* check for command line configuration file */ if (options.config_file) filename = g_strdup(options.config_file); /* check for user configuration ~/.ncmpc/config */ if (filename == NULL) { filename = g_build_filename(g_get_home_dir(), "." PACKAGE, "config", NULL); if (!g_file_test(filename, G_FILE_TEST_IS_REGULAR)) { g_free(filename); filename = NULL; } } /* check for global configuration SYSCONFDIR/ncmpc/config */ if (filename == NULL) { filename = g_build_filename(SYSCONFDIR, PACKAGE, "config", NULL); if (!g_file_test(filename, G_FILE_TEST_IS_REGULAR)) { g_free(filename); filename = NULL; } } /* load configuration */ if (filename) { read_rc_file(filename); g_free(filename); filename = NULL; } /* check for command line key binding file */ if (options.key_file) filename = g_strdup(options.key_file); /* check for user key bindings ~/.ncmpc/keys */ if (filename == NULL) { filename = get_user_key_binding_filename(); if (!g_file_test(filename, G_FILE_TEST_IS_REGULAR)) { g_free(filename); filename = NULL; } } /* check for global key bindings SYSCONFDIR/ncmpc/keys */ if (filename == NULL) { filename = g_build_filename(SYSCONFDIR, PACKAGE, "keys", NULL); if (!g_file_test(filename, G_FILE_TEST_IS_REGULAR)) { g_free(filename); filename = NULL; } } /* load key bindings */ if (filename) { read_rc_file(filename); g_free(filename); filename = NULL; } return 0; }