hooks.c
来自「elinks下lynx是最重要的二个文本浏览器, 在linux下非常实用, el」· C语言 代码 · 共 258 行
C
258 行
/* Ruby scripting hooks */#ifdef HAVE_CONFIG_H#include "config.h"#endif#include <ruby.h>#include "elinks.h"#include "cache/cache.h"#include "main/event.h"#include "protocol/uri.h"#include "scripting/ruby/core.h"#include "scripting/ruby/hooks.h"#include "session/location.h"#include "session/session.h"#include "util/string.h"/* The events that will trigger the functions below and what they are expected * to do is explained in doc/events.txt *//* We need to catch and handle errors because, otherwise, Ruby will kill us. */struct erb_protect_info { unsigned char *name; int argc; VALUE *args;};static VALUEdo_erb_protected_method_call(VALUE data){ struct erb_protect_info *info = (struct erb_protect_info *) data; ID method_id; assert(info); method_id = rb_intern(info->name); return rb_funcall3(erb_module, method_id, info->argc, info->args);}static VALUEerb_protected_method_call(unsigned char *name, int argc, VALUE *args, int *error){ struct erb_protect_info info = { name, argc, args }; return rb_protect(do_erb_protected_method_call, (VALUE) &info, error);}static enum evhook_statusscript_hook_goto_url(va_list ap, void *data){ unsigned char **url = va_arg(ap, unsigned char **); struct session *ses = va_arg(ap, struct session *); int error; VALUE args[2]; VALUE result; if (*url == NULL) return EVENT_HOOK_STATUS_NEXT; args[0] = rb_str_new2(*url); if (!ses || !have_location(ses)) { args[1] = Qnil; } else { args[1] = rb_str_new2(struri(cur_loc(ses)->vs.uri)); } result = erb_protected_method_call("goto_url_hook", 2, args, &error); if (error) { erb_report_error(ses, error); return EVENT_HOOK_STATUS_NEXT; } switch (rb_type(result)) { case T_STRING: { unsigned char *new_url; new_url = memacpy(RSTRING(result)->ptr, RSTRING(result)->len); if (new_url) { mem_free_set(url, new_url); } break; } case T_NIL: break; default: alert_ruby_error(ses, "goto_url_hook must return a string or nil"); } return EVENT_HOOK_STATUS_NEXT;}static enum evhook_statusscript_hook_follow_url(va_list ap, void *data){ unsigned char **url = va_arg(ap, unsigned char **); struct session *ses = va_arg(ap, struct session *); int error; VALUE args[1]; VALUE result; evhook_use_params(url && ses); if (*url == NULL) return EVENT_HOOK_STATUS_NEXT; args[0] = rb_str_new2(*url); result = erb_protected_method_call("follow_url_hook", 1, args, &error); if (error) { erb_report_error(ses, error); return EVENT_HOOK_STATUS_NEXT; } switch (rb_type(result)) { case T_STRING: { unsigned char *new_url; new_url = memacpy(RSTRING(result)->ptr, RSTRING(result)->len); if (new_url) { mem_free_set(url, new_url); } break; } case T_NIL: break; default: alert_ruby_error(ses, "follow_url_hook must return a string or nil"); } return EVENT_HOOK_STATUS_NEXT;}static enum evhook_statusscript_hook_pre_format_html(va_list ap, void *data){ struct session *ses = va_arg(ap, struct session *); struct cache_entry *cached = va_arg(ap, struct cache_entry *); struct fragment *fragment = get_cache_fragment(cached); unsigned char *url = struri(cached->uri); int error; VALUE args[2]; VALUE result; evhook_use_params(ses && cached); if (!cached->length || !*fragment->data) return EVENT_HOOK_STATUS_NEXT; args[0] = rb_str_new2(url); /* FIXME: Use html_len */ args[1] = rb_str_new(fragment->data, fragment->length); result = erb_protected_method_call("pre_format_html_hook", 2, args, &error); if (error) { erb_report_error(ses, error); return EVENT_HOOK_STATUS_NEXT; } switch (rb_type(result)) { case T_STRING: { int len = RSTRING(result)->len; add_fragment(cached, 0, RSTRING(result)->ptr, len); normalize_cache_entry(cached, len); break; } case T_NIL: break; default: alert_ruby_error(ses, "pre_format_html_hook must return a string or nil"); } return EVENT_HOOK_STATUS_NEXT;}/* The Ruby function can return: * - "PROXY:PORT" to use the specified proxy * - "" to not use any proxy * - nil to use the default proxies */static enum evhook_statusscript_hook_get_proxy(va_list ap, void *data){ unsigned char **new_proxy_url = va_arg(ap, unsigned char **); unsigned char *url = va_arg(ap, unsigned char *); int error; VALUE args[1]; VALUE result; if (!new_proxy_url || !url) return EVENT_HOOK_STATUS_NEXT; args[0] = rb_str_new2(url); result = erb_protected_method_call("proxy_hook", 1, args, &error); if (error) { erb_report_error(NULL, error); return EVENT_HOOK_STATUS_NEXT; } switch (rb_type(result)) { case T_STRING: { unsigned char *proxy; proxy = memacpy(RSTRING(result)->ptr, RSTRING(result)->len); if (proxy) { mem_free_set(new_proxy_url, proxy); } break; } case T_NIL: break; default: alert_ruby_error(NULL, "proxy_hook must return a string or nil"); } return EVENT_HOOK_STATUS_NEXT;}static enum evhook_statusscript_hook_quit(va_list ap, void *data){ VALUE args[1]; int error; erb_protected_method_call("quit_hook", 0, args, &error); if (error) erb_report_error(NULL, error); return EVENT_HOOK_STATUS_NEXT;}struct event_hook_info ruby_scripting_hooks[] = { { "goto-url", 0, script_hook_goto_url, NULL }, { "follow-url", 0, script_hook_follow_url, NULL }, { "pre-format-html", 0, script_hook_pre_format_html, NULL }, { "get-proxy", 0, script_hook_get_proxy, NULL }, { "quit", 0, script_hook_quit, NULL }, NULL_EVENT_HOOK_INFO,};
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?