tcl.c
来自「Linux下的多协议即时通讯程序源代码」· C语言 代码 · 共 554 行 · 第 1/2 页
C
554 行
/** * @file tcl.c Purple Tcl plugin bindings * * purple * * Copyright (C) 2003 Ethan Blanton <eblanton@cs.purdue.edu> * * 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */#include "config.h"#include <tcl.h>#ifdef HAVE_TK#include <tk.h>#endif#include <stdio.h>#include <sys/types.h>#include <sys/stat.h>#include <unistd.h>#include <string.h>#include "tcl_glib.h"#include "tcl_purple.h"#include "internal.h"#include "connection.h"#include "plugin.h"#include "signals.h"#include "debug.h"#include "util.h"#include "version.h"struct tcl_plugin_data { PurplePlugin *plugin; Tcl_Interp *interp;};PurpleStringref *PurpleTclRefAccount;PurpleStringref *PurpleTclRefConnection;PurpleStringref *PurpleTclRefConversation;PurpleStringref *PurpleTclRefPointer;PurpleStringref *PurpleTclRefPlugin;PurpleStringref *PurpleTclRefPresence;PurpleStringref *PurpleTclRefStatus;PurpleStringref *PurpleTclRefStatusAttr;PurpleStringref *PurpleTclRefStatusType;PurpleStringref *PurpleTclRefXfer;static GHashTable *tcl_plugins = NULL;PurplePlugin *_tcl_plugin;static gboolean tcl_loaded = FALSE;PurplePlugin *tcl_interp_get_plugin(Tcl_Interp *interp){ struct tcl_plugin_data *data; if (tcl_plugins == NULL) return NULL; data = g_hash_table_lookup(tcl_plugins, (gpointer)interp); return data != NULL ? data->plugin : NULL;}static int tcl_init_interp(Tcl_Interp *interp){ char *rcfile; char init[] = "namespace eval ::purple {\n" " namespace export account buddy connection conversation\n" " namespace export core debug notify prefs send_im\n" " namespace export signal unload\n" " namespace eval _callback { }\n" "\n" " proc conv_send { account who text } {\n" " set gc [purple::account connection $account]\n" " set convo [purple::conversation new $account $who]\n" " set myalias [purple::account alias $account]\n" "\n" " if {![string length $myalias]} {\n" " set myalias [purple::account username $account]\n" " }\n" "\n" " purple::send_im $gc $who $text\n" " purple::conversation write $convo send $myalias $text\n" " }\n" "}\n" "\n" "proc bgerror { message } {\n" " global errorInfo\n" " purple::notify -error \"Tcl Error\" \"Tcl Error: $message\" \"$errorInfo\"\n" "}\n"; if (Tcl_EvalEx(interp, init, -1, TCL_EVAL_GLOBAL) != TCL_OK) { return 1; } Tcl_SetVar(interp, "argc", "0", TCL_GLOBAL_ONLY); Tcl_SetVar(interp, "argv0", "purple", TCL_GLOBAL_ONLY); Tcl_SetVar(interp, "tcl_interactive", "0", TCL_GLOBAL_ONLY); rcfile = g_strdup_printf("%s" G_DIR_SEPARATOR_S "tclrc", purple_user_dir()); Tcl_SetVar(interp, "tcl_rcFileName", rcfile, TCL_GLOBAL_ONLY); g_free(rcfile); Tcl_SetVar(interp, "::purple::version", VERSION, TCL_GLOBAL_ONLY); Tcl_SetVar(interp, "::purple::user_dir", purple_user_dir(), TCL_GLOBAL_ONLY);#ifdef HAVE_TK Tcl_SetVar(interp, "::purple::tk_available", "1", TCL_GLOBAL_ONLY);#else Tcl_SetVar(interp, "::purple::tk_available", "0", TCL_GLOBAL_ONLY);#endif /* HAVE_TK */ Tcl_CreateObjCommand(interp, "::purple::account", tcl_cmd_account, (ClientData)NULL, NULL); Tcl_CreateObjCommand(interp, "::purple::buddy", tcl_cmd_buddy, (ClientData)NULL, NULL); Tcl_CreateObjCommand(interp, "::purple::cmd", tcl_cmd_cmd, (ClientData)NULL, NULL); Tcl_CreateObjCommand(interp, "::purple::connection", tcl_cmd_connection, (ClientData)NULL, NULL); Tcl_CreateObjCommand(interp, "::purple::conversation", tcl_cmd_conversation, (ClientData)NULL, NULL); Tcl_CreateObjCommand(interp, "::purple::core", tcl_cmd_core, (ClientData)NULL, NULL); Tcl_CreateObjCommand(interp, "::purple::debug", tcl_cmd_debug, (ClientData)NULL, NULL); Tcl_CreateObjCommand(interp, "::purple::notify", tcl_cmd_notify, (ClientData)NULL, NULL); Tcl_CreateObjCommand(interp, "::purple::plugins", tcl_cmd_plugins, (ClientData)NULL, NULL); Tcl_CreateObjCommand(interp, "::purple::prefs", tcl_cmd_prefs, (ClientData)NULL, NULL); Tcl_CreateObjCommand(interp, "::purple::presence", tcl_cmd_presence, (ClientData)NULL, NULL); Tcl_CreateObjCommand(interp, "::purple::send_im", tcl_cmd_send_im, (ClientData)NULL, NULL); Tcl_CreateObjCommand(interp, "::purple::savedstatus", tcl_cmd_savedstatus, (ClientData)NULL, NULL); Tcl_CreateObjCommand(interp, "::purple::signal", tcl_cmd_signal, (ClientData)NULL, NULL); Tcl_CreateObjCommand(interp, "::purple::status", tcl_cmd_status, (ClientData)NULL, NULL); Tcl_CreateObjCommand(interp, "::purple::status_attr", tcl_cmd_status_attr, (ClientData)NULL, NULL); Tcl_CreateObjCommand(interp, "::purple::status_type", tcl_cmd_status_type, (ClientData)NULL, NULL); Tcl_CreateObjCommand(interp, "::purple::unload", tcl_cmd_unload, (ClientData)NULL, NULL); return 0;}static Tcl_Interp *tcl_create_interp(){ Tcl_Interp *interp; interp = Tcl_CreateInterp(); if (Tcl_Init(interp) == TCL_ERROR) { Tcl_DeleteInterp(interp); return NULL; } if (tcl_init_interp(interp)) { Tcl_DeleteInterp(interp); return NULL; } Tcl_StaticPackage(interp, "purple", tcl_init_interp, NULL); return interp;}static gboolean tcl_probe_plugin(PurplePlugin *plugin){ PurplePluginInfo *info; Tcl_Interp *interp; Tcl_Parse parse; Tcl_Obj *result, **listitems; struct stat st; FILE *fp; char *buf, *cur; const char *next; int len, found = 0, err = 0, nelems; gboolean status = FALSE; if ((fp = g_fopen(plugin->path, "r")) == NULL) return FALSE; if (fstat(fileno(fp), &st)) { fclose(fp); return FALSE; } len = st.st_size; buf = g_malloc(len + 1); cur = buf; while (fgets(cur, GPOINTER_TO_INT(buf) - (buf - cur), fp)) { cur += strlen(cur); if (feof(fp)) break; } if (ferror(fp)) { purple_debug(PURPLE_DEBUG_ERROR, "tcl", "error reading %s (%s)\n", plugin->path, strerror(errno)); g_free(buf); fclose(fp); return FALSE; } fclose(fp); if ((interp = tcl_create_interp()) == NULL) { return FALSE; } next = buf; do { if (Tcl_ParseCommand(interp, next, len, 0, &parse) == TCL_ERROR) { purple_debug(PURPLE_DEBUG_ERROR, "tcl", "parse error in %s: %s\n", plugin->path, Tcl_GetString(Tcl_GetObjResult(interp))); err = 1; break; } if (parse.tokenPtr[0].type == TCL_TOKEN_SIMPLE_WORD && !strncmp(parse.tokenPtr[0].start, "proc", parse.tokenPtr[0].size)) { if (!strncmp(parse.tokenPtr[2].start, "plugin_init", parse.tokenPtr[2].size)) { if (Tcl_EvalEx(interp, parse.commandStart, parse.commandSize, TCL_EVAL_GLOBAL) != TCL_OK) { Tcl_FreeParse(&parse); break; } found = 1; /* We'll continue parsing the file, just in case */ } } len -= (parse.commandStart + parse.commandSize) - next; next = parse.commandStart + parse.commandSize; Tcl_FreeParse(&parse); } while (len); if (found && !err) { if (Tcl_EvalEx(interp, "plugin_init", -1, TCL_EVAL_GLOBAL) == TCL_OK) { result = Tcl_GetObjResult(interp); if (Tcl_ListObjGetElements(interp, result, &nelems, &listitems) == TCL_OK) { if ((nelems == 6) || (nelems == 7)) { info = g_new0(PurplePluginInfo, 1); info->magic = PURPLE_PLUGIN_MAGIC; info->major_version = PURPLE_MAJOR_VERSION; info->minor_version = PURPLE_MINOR_VERSION; info->type = PURPLE_PLUGIN_STANDARD; info->dependencies = g_list_append(info->dependencies, "core-tcl"); info->name = g_strdup(Tcl_GetString(listitems[0])); info->version = g_strdup(Tcl_GetString(listitems[1])); info->summary = g_strdup(Tcl_GetString(listitems[2])); info->description = g_strdup(Tcl_GetString(listitems[3])); info->author = g_strdup(Tcl_GetString(listitems[4])); info->homepage = g_strdup(Tcl_GetString(listitems[5])); if (nelems == 6) info->id = g_strdup_printf("tcl-%s", Tcl_GetString(listitems[0])); else if (nelems == 7) info->id = g_strdup_printf("tcl-%s", Tcl_GetString(listitems[6])); plugin->info = info; if (purple_plugin_register(plugin)) status = TRUE; } } } } Tcl_DeleteInterp(interp); g_free(buf); return status;}static gboolean tcl_load_plugin(PurplePlugin *plugin){ struct tcl_plugin_data *data;
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?