📄 opttypes.c
字号:
/* Option variables types handlers *//* $Id: opttypes.c,v 1.89.2.1 2005/04/05 21:08:40 jonas Exp $ */#ifdef HAVE_CONFIG_H#include "config.h"#endif#include <stdio.h>#include <string.h>#include "elinks.h"#include "bfu/dialog.h"#include "config/options.h"#include "config/opttypes.h"#include "intl/charsets.h"#include "intl/gettext/libintl.h"#include "util/color.h"#include "util/conv.h"#include "util/error.h"#include "util/memory.h"#include "util/string.h"/* Commandline handlers. *//* TAKE CARE! Remember that your _rd handler can be used for commandline * parameters as well - probably, you don't want to be so syntactically * strict, and _ESPECIALLY_ you don't want to move any file pointers ahead, * since you will parse the commandline _TWO TIMES_! Remember! :-) */int commandline = 0;static unsigned char *gen_cmd(struct option *o, unsigned char ***argv, int *argc){ unsigned char *str; int dummy_line = 0; if (!*argc) return gettext("Parameter expected"); /* FIXME!! We will modify argv! (maybe) */ commandline = 1; str = option_types[o->type].read(o, *argv, &dummy_line); commandline = 0; if (str) { /* We ate parameter */ (*argv)++; (*argc)--; if (option_types[o->type].set(o, str)) { mem_free(str); return NULL; } mem_free(str); } return gettext("Read error");}/* If 0 follows, disable option and eat 0. If 1 follows, enable option and * eat 1. If anything else follow, enable option and don't eat anything. */static unsigned char *bool_cmd(struct option *o, unsigned char ***argv, int *argc){ o->value.number = 1; if (!*argc) return NULL; /* Argument is empty or longer than 1 char.. */ if (!(*argv)[0][0] || (*argv)[0][1]) return NULL; switch ((*argv)[0][0]) { case '0': o->value.number = 0; break; case '1': o->value.number = 1; break; default: return NULL; } /* We ate parameter */ (*argv)++; (*argc)--; return NULL;}static unsigned char *exec_cmd(struct option *o, unsigned char ***argv, int *argc){ return o->value.command(o, argv, argc);}/* Wrappers for OPT_ALIAS. *//* Note that they can wrap only to config_options now. I don't think it could be * a problem, but who knows.. however, changing that will be pretty tricky - * possibly changing ptr to structure containing target name and pointer to * options list? --pasky */#define wrap_or_(name_, call_, ret_) \{ \ struct option *real = get_opt_rec(config_options, opt->value.string); \ \ assertm(real, "%s aliased to unknown option %s!", opt->name, opt->value.string); \ if_assert_failed { return ret_; } \ \ if (option_types[real->type].name_) \ return option_types[real->type].call_; \ \ return ret_; \}static unsigned char *redir_cmd(struct option *opt, unsigned char ***argv, int *argc) wrap_or_(cmdline, cmdline(real, argv, argc), NULL);static unsigned char *redir_rd(struct option *opt, unsigned char **file, int *line) wrap_or_(read, read(real, file, line), NULL);static voidredir_wr(struct option *opt, struct string *string){ struct option *real = get_opt_rec(config_options, opt->value.string); assertm(real, "%s aliased to unknown option %s!", opt->name, opt->value.string); if_assert_failed { return; } if (option_types[real->type].write) option_types[real->type].write(real, string);}static intredir_set(struct option *opt, unsigned char *str) wrap_or_(set, set(real, str), 0);static intredir_add(struct option *opt, unsigned char *str) wrap_or_(add, add(real, str), 0);/* Support functions for config file parsing. */static voidadd_optstring_to_string(struct string *s, unsigned char *q, int qlen){ if (!commandline) add_char_to_string(s, '"'); add_quoted_to_string(s, q, qlen); if (!commandline) add_char_to_string(s, '"');}/* Config file handlers. */static unsigned char *num_rd(struct option *opt, unsigned char **file, int *line){ unsigned char *end = *file; long *value = mem_alloc(sizeof(*value)); if (!value) return NULL; /* We don't want to move file if (commandline), but strtolx() second * parameter must not be NULL. */ *value = strtolx(*file, &end); if (!commandline) *file = end; /* Another trap for unwary - we need to check *end, not **file - reason * is left as an exercise to the reader. */ if ((*end != 0 && (commandline || (!isspace(*end) && *end != '#'))) || (*value < opt->min || *value > opt->max)) { mem_free(value); return NULL; } return (unsigned char *) value;}static intnum_set(struct option *opt, unsigned char *str){ opt->value.number = *((long *) str); return 1;}static voidnum_wr(struct option *option, struct string *string){ add_knum_to_string(string, option->value.number);}static unsigned char *str_rd(struct option *opt, unsigned char **file, int *line){ unsigned char *str = *file; struct string str2; if (!init_string(&str2)) return NULL; /* We're getting used in some parser functions in conf.c as well, and * that's w/ opt == NULL; so don't rely on opt to point anywhere. */ if (!commandline) { if (!isquote(*str)) { done_string(&str2); return NULL; } str++; } while (*str && (commandline || !isquote(*str))) { if (*str == '\\') { /* FIXME: This won't work on crlf systems. */ if (str[1] == '\n') { str[1] = ' '; str++; } /* When there's quote char, we will just move on there, * thus we will never test for it in while () condition * and we will treat it just as '"', ignoring the * backslash itself. */ if (isquote(str[1])) str++; /* \\ means \. */ if (str[1] == '\\') str++; } if (*str == '\n') (*line)++; add_char_to_string(&str2, *str); str++; } if (!commandline && !*str) { done_string(&str2); *file = str; return NULL; } str++; /* Skip the quote. */ if (!commandline) *file = str; if (opt && opt->max && str2.length >= opt->max) { done_string(&str2); return NULL; } return str2.source;}static intstr_set(struct option *opt, unsigned char *str){ assert(opt->value.string); safe_strncpy(opt->value.string, str, MAX_STR_LEN); return 1;}static voidstr_wr(struct option *o, struct string *s){ int len = strlen(o->value.string); int_upper_bound(&len, o->max - 1); add_optstring_to_string(s, o->value.string, len);}static voidstr_dup(struct option *opt, struct option *template){ unsigned char *new = mem_alloc(MAX_STR_LEN); if (new) safe_strncpy(new, template->value.string, MAX_STR_LEN); opt->value.string = new;}static intcp_set(struct option *opt, unsigned char *str){ int ret = get_cp_index(str); if (ret < 0) return 0; opt->value.number = ret; return 1;}static voidcp_wr(struct option *o, struct string *s){ unsigned char *mime_name = get_cp_mime_name(o->value.number); add_optstring_to_string(s, mime_name, strlen(mime_name));}static intlang_set(struct option *opt, unsigned char *str){#ifdef ENABLE_NLS opt->value.number = name_to_language(str); set_language(opt->value.number);#endif return 1;}static voidlang_wr(struct option *o, struct string *s){ unsigned char *lang;#ifdef ENABLE_NLS lang = language_to_name(current_language);#else lang = "System";#endif add_optstring_to_string(s, lang, strlen(lang));}static intcolor_set(struct option *opt, unsigned char *str){ return !decode_color(str, strlen(str), &opt->value.color);}static voidcolor_wr(struct option *opt, struct string *str){ color_t color = opt->value.color; unsigned char hexcolor[8]; unsigned char *strcolor = get_color_string(color, hexcolor); add_optstring_to_string(str, strcolor, strlen(strcolor));}static voidtree_dup(struct option *opt, struct option *template){ struct list_head *new = init_options_tree(); struct list_head *tree = template->value.tree; struct option *option; if (!new) return; opt->value.tree = new; foreachback (option, *tree) { struct option *new_opt = copy_option(option); if (!new_opt) continue; add_to_list_end(*new, new_opt); new_opt->root = opt; if (!new_opt->box_item) continue; if (new_opt->name && !strcmp(new_opt->name, "_template_")) new_opt->box_item->visible = get_opt_bool("config.show_template"); if (opt->box_item) { add_to_list(opt->box_item->child, new_opt->box_item); } }}const struct option_type_info option_types[] = { { N_("Boolean"), bool_cmd, num_rd, num_wr, NULL, num_set, NULL, N_("[0|1]") }, { N_("Integer"), gen_cmd, num_rd, num_wr, NULL, num_set, NULL, N_("<num>") }, { N_("Longint"), gen_cmd, num_rd, num_wr, NULL, num_set, NULL, N_("<num>") }, { N_("String"), gen_cmd, str_rd, str_wr, str_dup, str_set, NULL, N_("<str>") }, { N_("Codepage"), gen_cmd, str_rd, cp_wr, NULL, cp_set, NULL, N_("<codepage>") }, { N_("Language"), gen_cmd, str_rd, lang_wr, NULL, lang_set, NULL, N_("<language>") }, { N_("Color"), gen_cmd, str_rd, color_wr, NULL, color_set, NULL, N_("<color|#rrggbb>") }, { N_("Special"), exec_cmd, NULL, NULL, NULL, NULL, NULL, "" }, { N_("Alias"), redir_cmd, redir_rd, redir_wr, NULL, redir_set, redir_add, "" }, /* tree */ { N_("Folder"), NULL, NULL, NULL, tree_dup, NULL, NULL, "" },};unsigned char *get_option_type_name(enum option_type type){ assert(type >= 0 && type < sizeof(option_types)/sizeof(struct option_type_info)); if_assert_failed return ""; return option_types[type].name;}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -