#include "owl.h" #include "filterproc.h" #include #include #include CALLER_OWN char *owl_function_command(const char *cmdbuff) { owl_function_debugmsg("executing command: %s", cmdbuff); return owl_cmddict_execute(owl_global_get_cmddict(&g), owl_global_get_context(&g), cmdbuff); } CALLER_OWN char *owl_function_command_argv(const char *const *argv, int argc) { return owl_cmddict_execute_argv(owl_global_get_cmddict(&g), owl_global_get_context(&g), argv, argc); } void owl_function_command_norv(const char *cmdbuff) { char *rv; rv=owl_function_command(cmdbuff); g_free(rv); } void owl_function_command_alias(const char *alias_from, const char *alias_to) { owl_cmddict_add_alias(owl_global_get_cmddict(&g), alias_from, alias_to); } const owl_cmd *owl_function_get_cmd(const char *name) { return owl_cmddict_find(owl_global_get_cmddict(&g), name); } void owl_function_show_commands(void) { GPtrArray *l; owl_fmtext fm; owl_fmtext_init_null(&fm); owl_fmtext_append_bold(&fm, "Commands: "); owl_fmtext_append_normal(&fm, "(use 'show command ' for details)\n"); l = owl_cmddict_get_names(owl_global_get_cmddict(&g)); owl_fmtext_append_list(&fm, l, "\n", owl_function_cmd_describe); owl_fmtext_append_normal(&fm, "\n"); owl_function_popless_fmtext(&fm); owl_ptr_array_free(l, g_free); owl_fmtext_cleanup(&fm); } void owl_function_show_view(const char *viewname) { const owl_view *v; owl_fmtext fm; /* we only have the one view right now */ v=owl_global_get_current_view(&g); if (viewname && strcmp(viewname, owl_view_get_name(v))) { owl_function_error("No view named '%s'", viewname); return; } owl_fmtext_init_null(&fm); owl_view_to_fmtext(v, &fm); owl_function_popless_fmtext(&fm); owl_fmtext_cleanup(&fm); } void owl_function_show_styles(void) { GPtrArray *l; owl_fmtext fm; owl_fmtext_init_null(&fm); owl_fmtext_append_bold(&fm, "Styles:\n"); l = owl_global_get_style_names(&g); owl_fmtext_append_list(&fm, l, "\n", owl_function_style_describe); owl_fmtext_append_normal(&fm, "\n"); owl_function_popless_fmtext(&fm); owl_ptr_array_free(l, g_free); owl_fmtext_cleanup(&fm); } CALLER_OWN char *owl_function_style_describe(const char *name) { const char *desc; char *s; const owl_style *style; style = owl_global_get_style_by_name(&g, name); if (style) { desc = owl_style_get_description(style); } else { desc = "???"; } s = g_strdup_printf("%-20s - %s%s", name, 0 == owl_style_validate(style) ? "" : "[INVALID] ", desc); return s; } CALLER_OWN char *owl_function_cmd_describe(const char *name) { const owl_cmd *cmd = owl_cmddict_find(owl_global_get_cmddict(&g), name); if (cmd) return owl_cmd_describe(cmd); else return(NULL); } void owl_function_show_command(const char *name) { owl_function_help_for_command(name); } void owl_function_show_license(void) { char *text = g_strdup_printf( "BarnOwl version %s\n" "Copyright (c) 2006-2011 The BarnOwl Developers. All rights reserved.\n" "Copyright (c) 2004 James Kretchmar. All rights reserved.\n" "\n" "Redistribution and use in source and binary forms, with or without\n" "modification, are permitted provided that the following conditions are\n" "met:\n" "\n" " * Redistributions of source code must retain the above copyright\n" " notice, this list of conditions and the following disclaimer.\n" "\n" " * Redistributions in binary form must reproduce the above copyright\n" " notice, this list of conditions and the following disclaimer in\n" " the documentation and/or other materials provided with the\n" " distribution.\n" "\n" " * Redistributions in any form must be accompanied by information on\n" " how to obtain complete source code for the BarnOwl software and any\n" " accompanying software that uses the BarnOwl software. The source code\n" " must either be included in the distribution or be available for no\n" " more than the cost of distribution plus a nominal fee, and must be\n" " freely redistributable under reasonable conditions. For an\n" " executable file, complete source code means the source code for\n" " all modules it contains. It does not include source code for\n" " modules or files that typically accompany the major components of\n" " the operating system on which the executable file runs.\n" "\n" "THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR\n" "IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED\n" "WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, OR\n" "NON-INFRINGEMENT, ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE\n" "LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR\n" "CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF\n" "SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR\n" "BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,\n" "WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE\n" "OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN\n" "IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.\n", version); owl_function_popless_text(text); g_free(text); } void owl_function_show_quickstart(void) { const char *message = "Move between messages with the arrow keys, and press 'r' to reply.\n" "For more info, press 'h' or visit http://barnowl.mit.edu/\n\n" #ifdef HAVE_LIBZEPHYR "@b(Zephyr:)\n" "To send a message to a user, type ':zwrite @b(username)'. You can also\n" "press 'z' and then type the username. To subscribe to a class, type\n" "':sub @b(class)', and then type ':zwrite -c @b(class)' to send.\n\n" #endif "@b(AIM:)\n" "Log in to AIM with ':aimlogin @b(screenname)'. Use ':aimwrite @b(screenname)',\n" "or 'a' and then the screen name, to send someone a message.\n\n" ; if (owl_perlconfig_is_function("BarnOwl::Hooks::_get_quickstart")) { char *perlquickstart = owl_perlconfig_execute("BarnOwl::Hooks::_get_quickstart()"); if (perlquickstart) { char *result = g_strdup_printf("%s%s", message, perlquickstart); owl_function_adminmsg("BarnOwl Quickstart", result); g_free(result); g_free(perlquickstart); return; } } owl_function_adminmsg("BarnOwl Quickstart", message); } /* Create an admin message, append it to the global list of messages * and redisplay if necessary. */ void owl_function_adminmsg(const char *header, const char *body) { owl_message *m; m=g_new(owl_message, 1); owl_message_create_admin(m, header, body); /* add it to the global list and current view */ owl_messagelist_append_element(owl_global_get_msglist(&g), m); owl_view_consider_message(owl_global_get_current_view(&g), m); /* do followlast if necessary */ if (owl_global_should_followlast(&g)) owl_function_lastmsg(); /* redisplay etc. */ owl_mainwin_redisplay(owl_global_get_mainwin(&g)); } /* Queues outgoing zephyrs; if z sends to n people, queue n messages * (except in case of cc). If there are no recipients queues 1 * message. */ void owl_function_add_outgoing_zephyrs(const owl_zwrite *z) { if (z->cc && owl_zwrite_is_personal(z)) { /* create the message */ owl_message *m = g_new(owl_message, 1); owl_message_create_from_zwrite(m, z, owl_zwrite_get_message(z), 0); owl_global_messagequeue_addmsg(&g, m); } else { int i; for (i = 0; i < owl_zwrite_get_numrecips(z); i++) { owl_message *m; if (!owl_zwrite_recip_is_personal(owl_zwrite_get_recip_n(z, i))) continue; /* create the message */ m = g_new(owl_message, 1); owl_message_create_from_zwrite(m, z, owl_zwrite_get_message(z), i); owl_global_messagequeue_addmsg(&g, m); } } } /* Create an outgoing AIM message, returns a pointer to the created * message or NULL if we're not logged into AIM (and thus unable to * create the message). Does not put it on the global queue. Use * owl_global_messagequeue_addmsg() for that. */ CALLER_OWN owl_message *owl_function_make_outgoing_aim(const char *body, const char *to) { owl_message *m; /* error if we're not logged into aim */ if (!owl_global_is_aimloggedin(&g)) return(NULL); m=g_new(owl_message, 1); owl_message_create_aim(m, owl_global_get_aim_screenname(&g), to, body, OWL_MESSAGE_DIRECTION_OUT, 0); return(m); } /* Create an outgoing loopback message and return a pointer to it. * Does not append it to the global queue, use * owl_global_messagequeue_addmsg() for that. */ CALLER_OWN owl_message *owl_function_make_outgoing_loopback(const char *body) { owl_message *m; /* create the message */ m=g_new(owl_message, 1); owl_message_create_loopback(m, body); owl_message_set_direction_out(m); return(m); } owl_editwin *owl_function_start_edit_win(const char *line) { owl_editwin *e; owl_context *ctx; char *s; /* create and setup the editwin */ e = owl_global_set_typwin_active(&g, OWL_EDITWIN_STYLE_MULTILINE, owl_global_get_msg_history(&g)); owl_editwin_set_dotsend(e); s = g_strdup_printf("----> %s\n", line); owl_editwin_set_locktext(e, s); g_free(s); ctx = owl_editcontext_new(OWL_CTX_EDITMULTI, e, "editmulti", owl_global_deactivate_editcontext, &g); owl_global_push_context_obj(&g, ctx); return e; } static void owl_function_write_setup(const char *noun) { if (!owl_global_get_lockout_ctrld(&g)) owl_function_makemsg("Type your %s below. " "End with ^D or a dot on a line by itself." " ^C will quit.", noun); else owl_function_makemsg("Type your %s below. " "End with a dot on a line by itself. ^C will quit.", noun); } void owl_function_zwrite_setup(owl_zwrite *z) { owl_editwin *e; /* send a ping if necessary */ if (owl_global_is_txping(&g)) { owl_zwrite_send_ping(z); } owl_function_write_setup("zephyr"); e = owl_function_start_edit_win(z->zwriteline); owl_editwin_set_cbdata(e, z, (void (*)(void *))owl_zwrite_delete); owl_editwin_set_callback(e, &owl_callback_zwrite); } void owl_function_aimwrite_setup(const char *to) { owl_editwin *e; /* TODO: We probably actually want an owl_aimwrite object like * owl_zwrite. */ char *line = g_strdup_printf("aimwrite %s", to); owl_function_write_setup("message"); e = owl_function_start_edit_win(line); owl_editwin_set_cbdata(e, g_strdup(to), g_free); owl_editwin_set_callback(e, &owl_callback_aimwrite); g_free(line); } void owl_function_loopwrite_setup(void) { owl_editwin *e; owl_function_write_setup("message"); e = owl_function_start_edit_win("loopwrite"); owl_editwin_set_callback(e, &owl_callback_loopwrite); } void owl_callback_zwrite(owl_editwin *e, bool success) { if (!success) return; owl_zwrite *z = owl_editwin_get_cbdata(e); owl_function_zwrite(z, owl_editwin_get_text(e)); } /* send, log and display outgoing zephyrs. If 'msg' is NULL the * message is expected to be set from the zwrite line itself */ #ifdef HAVE_LIBZEPHYR void owl_function_zwrite(owl_zwrite *z, const char *msg) { int ret; if(strcmp(z->cmd, "zcrypt") == 0) { owl_function_zcrypt(z, msg); return; } /* create the zwrite and send the message */ owl_zwrite_populate_zsig(z); if (msg) { owl_zwrite_set_message(z, msg); } ret = owl_zwrite_send_message(z); if (ret != 0) { owl_function_makemsg("Error sending zephyr: %s", error_message(ret)); return; } owl_function_makemsg("Waiting for ack..."); /* create the outgoing message */ owl_function_add_outgoing_zephyrs(z); } #else void owl_function_zwrite(owl_zwrite *z, const char *msg) { } #endif /* send, log and display outgoing zcrypt zephyrs. If 'msg' is NULL * the message is expected to be set from the zwrite line itself */ void owl_function_zcrypt(owl_zwrite *z, const char *msg) { char *cryptmsg; const char *argv[7]; char *zcrypt; int rv, status; char *old_msg; /* create the zwrite and send the message */ owl_zwrite_populate_zsig(z); if (msg) { owl_zwrite_set_message(z, msg); } old_msg = g_strdup(owl_zwrite_get_message(z)); zcrypt = g_build_filename(owl_get_bindir(), "zcrypt", NULL); argv[0] = zcrypt; argv[1] = "-E"; argv[2] = "-c"; argv[3] = owl_zwrite_get_class(z); argv[4] = "-i"; argv[5] = owl_zwrite_get_instance(z); argv[6] = NULL; rv = call_filter(argv, owl_zwrite_get_message(z), &cryptmsg, &status); g_free(zcrypt); if (rv || status) { g_free(cryptmsg); g_free(old_msg); owl_function_error("Error in zcrypt, possibly no key found. Message not sent."); owl_function_beep(); return; } owl_zwrite_set_message_raw(z, cryptmsg); owl_zwrite_set_opcode(z, "crypt"); owl_zwrite_send_message(z); owl_function_makemsg("Waiting for ack..."); /* Create the outgoing message. Restore the un-crypted message for display. */ owl_zwrite_set_message_raw(z, old_msg); owl_function_add_outgoing_zephyrs(z); /* Clean up. */ g_free(cryptmsg); g_free(old_msg); } void owl_callback_aimwrite(owl_editwin *e, bool success) { if (!success) return; char *to = owl_editwin_get_cbdata(e); owl_function_aimwrite(to, owl_editwin_get_text(e), true); } void owl_function_aimwrite(const char *to, const char *msg, bool unwrap) { int ret; char *format_msg; owl_message *m; /* make a formatted copy of the message */ format_msg = g_strdup(msg); if (unwrap) owl_text_wordunwrap(format_msg); /* send the message */ ret=owl_aim_send_im(to, format_msg); if (!ret) { owl_function_makemsg("AIM message sent."); } else { owl_function_error("Could not send AIM message."); } /* create the outgoing message */ m=owl_function_make_outgoing_aim(msg, to); if (m) { owl_global_messagequeue_addmsg(&g, m); } else { owl_function_error("Could not create outgoing AIM message"); } g_free(format_msg); } void owl_function_send_aimawymsg(const char *to, const char *msg) { int ret; char *format_msg; owl_message *m; /* make a formatted copy of the message */ format_msg=g_strdup(msg); owl_text_wordunwrap(format_msg); /* send the message */ ret=owl_aim_send_awaymsg(to, format_msg); if (!ret) { /* owl_function_makemsg("AIM message sent."); */ } else { owl_function_error("Could not send AIM message."); } /* create the message */ m=owl_function_make_outgoing_aim(msg, to); if (m) { owl_global_messagequeue_addmsg(&g, m); } else { owl_function_error("Could not create AIM message"); } g_free(format_msg); } void owl_callback_loopwrite(owl_editwin *e, bool success) { if (!success) return; owl_function_loopwrite(owl_editwin_get_text(e)); } void owl_function_loopwrite(const char *msg) { owl_message *min, *mout; /* create a message and put it on the message queue. This simulates * an incoming message */ min=g_new(owl_message, 1); mout=owl_function_make_outgoing_loopback(msg); if (owl_global_is_displayoutgoing(&g)) { owl_global_messagequeue_addmsg(&g, mout); } else { owl_message_delete(mout); } owl_message_create_loopback(min, msg); owl_message_set_direction_in(min); owl_global_messagequeue_addmsg(&g, min); /* fake a makemsg */ owl_function_makemsg("loopback message sent"); } /* If filter is non-null, looks for the next message matching * that filter. If skip_deleted, skips any deleted messages. * If last_if_none, will stop at the last message in the view * if no matching messages are found. */ void owl_function_nextmsg_full(const char *filter, int skip_deleted, int last_if_none) { int curmsg, i, viewsize, found; const owl_view *v; const owl_filter *f = NULL; const owl_message *m; v=owl_global_get_current_view(&g); if (filter) { f=owl_global_get_filter(&g, filter); if (!f) { owl_function_error("No %s filter defined", filter); return; } } curmsg=owl_global_get_curmsg(&g); viewsize=owl_view_get_size(v); found=0; /* just check to make sure we're in bounds... */ if (curmsg>viewsize-1) curmsg=viewsize-1; if (curmsg<0) curmsg=0; for (i=curmsg+1; iowl_view_get_size(v)-1) i=owl_view_get_size(v)-1; if (i<0) i=0; if (!found) { owl_function_makemsg("already at last%s message%s%s%s", skip_deleted?" non-deleted":"", filter?" in ":"", filter?filter:"", owl_mainwin_is_curmsg_truncated(owl_global_get_mainwin(&g)) ? ", press Enter to scroll" : ""); /* if (!skip_deleted) owl_function_beep(); */ } if (last_if_none || found) { owl_global_set_curmsg(&g, i); owl_function_calculate_topmsg(OWL_DIRECTION_DOWNWARDS); owl_mainwin_redisplay(owl_global_get_mainwin(&g)); owl_global_set_direction_downwards(&g); } } void owl_function_prevmsg_full(const char *filter, int skip_deleted, int first_if_none) { int curmsg, i, found; const owl_view *v; const owl_filter *f = NULL; const owl_message *m; v=owl_global_get_current_view(&g); if (filter) { f=owl_global_get_filter(&g, filter); if (!f) { owl_function_error("No %s filter defined", filter); return; } } curmsg=owl_global_get_curmsg(&g); found=0; /* just check to make sure we're in bounds... */ if (curmsg<0) curmsg=0; for (i=curmsg-1; i>=0; i--) { m=owl_view_get_element(v, i); if (skip_deleted && owl_message_is_delete(m)) continue; if (f && !owl_filter_message_match(f, m)) continue; found = 1; break; } if (i<0) i=0; if (!found) { owl_function_makemsg("already at first%s message%s%s", skip_deleted?" non-deleted":"", filter?" in ":"", filter?filter:""); /* if (!skip_deleted) owl_function_beep(); */ } if (first_if_none || found) { owl_global_set_curmsg(&g, i); owl_function_calculate_topmsg(OWL_DIRECTION_UPWARDS); owl_mainwin_redisplay(owl_global_get_mainwin(&g)); owl_global_set_direction_upwards(&g); } } void owl_function_nextmsg(void) { owl_function_nextmsg_full(NULL, 0, 1); } void owl_function_prevmsg(void) { owl_function_prevmsg_full(NULL, 0, 1); } void owl_function_nextmsg_notdeleted(void) { owl_function_nextmsg_full(NULL, 1, 1); } void owl_function_prevmsg_notdeleted(void) { owl_function_prevmsg_full(NULL, 1, 1); } void owl_function_delete_and_expunge_message(int n) { owl_messagelist *ml = owl_global_get_msglist(&g); owl_view *v = owl_global_get_current_view(&g); int lastmsgid = owl_function_get_curmsg_id(v); /* delete and expunge the message */ owl_messagelist_delete_and_expunge_element(ml, n); owl_function_redisplay_to_nearest(lastmsgid, v); } void owl_function_delete_and_expunge_cur(bool exclaim_success) { int curmsg; const owl_view *v = owl_global_get_current_view(&g); /* bail if there's no current message */ if (owl_view_get_size(v) < 1) { owl_function_error("No current message to delete"); return; } /* delete the current message */ curmsg = owl_global_get_curmsg(&g); owl_function_delete_and_expunge_message(curmsg); if (exclaim_success) owl_function_makemsg("Message deleted and expunged"); } /* if move_after is 1, moves after the delete */ void owl_function_deletecur(int move_after) { int curmsg; owl_view *v; v=owl_global_get_current_view(&g); /* bail if there's no current message */ if (owl_view_get_size(v) < 1) { owl_function_error("No current message to delete"); return; } /* mark the message for deletion */ curmsg=owl_global_get_curmsg(&g); owl_view_delete_element(v, curmsg); if (move_after) { /* move the poiner in the appropriate direction * to the next undeleted msg */ if (owl_global_get_direction(&g)==OWL_DIRECTION_UPWARDS) { owl_function_prevmsg_notdeleted(); } else { owl_function_nextmsg_notdeleted(); } } } void owl_function_undeletecur(int move_after) { int curmsg; owl_view *v; v=owl_global_get_current_view(&g); if (owl_view_get_size(v) < 1) { owl_function_error("No current message to undelete"); return; } curmsg=owl_global_get_curmsg(&g); owl_view_undelete_element(v, curmsg); if (move_after) { if (owl_global_get_direction(&g)==OWL_DIRECTION_UPWARDS) { if (curmsg>0) { owl_function_prevmsg(); } else { owl_function_nextmsg(); } } else { owl_function_nextmsg(); } } owl_mainwin_redisplay(owl_global_get_mainwin(&g)); } /* returns the current message id, if it exists. Otherwise returns * -1 if we are past the end of the message list, and 0 otherwise. */ int owl_function_get_curmsg_id(const owl_view *v) { int curmsg = owl_global_get_curmsg(&g); const owl_message *m = owl_view_get_element(v, curmsg); if (m) return owl_message_get_id(m); if (curmsg > 0) /* past the end of the message list (probably) */ return -1; return 0; } /* redisplays the view to the nearest message to the id given. * if msgid < 0, redisplay to past the end of the message list */ void owl_function_redisplay_to_nearest(int msgid, owl_view *v) { int curmsg; /* update all views (we only have one right now) */ owl_view_recalculate(v); /* find where the new position should be */ if (msgid < 0) { /* If already at the end, blank the screen and move curmsg * past the end of the messages. */ curmsg = owl_view_get_size(v); owl_global_set_topmsg(&g, curmsg); owl_global_set_curmsg(&g, curmsg); } else { curmsg = owl_view_get_nearest_to_msgid(v, msgid); if (curmsg > owl_view_get_size(v) - 1) curmsg = owl_view_get_size(v) - 1; if (curmsg < 0) curmsg = 0; owl_global_set_curmsg(&g, curmsg); owl_function_calculate_topmsg(OWL_DIRECTION_NONE); } /* if there are no messages set the direction to down in case we * delete everything upwards */ owl_global_set_direction_downwards(&g); owl_mainwin_redisplay(owl_global_get_mainwin(&g)); } void owl_function_expunge(void) { owl_messagelist *ml = owl_global_get_msglist(&g); owl_view *v = owl_global_get_current_view(&g); int lastmsgid = owl_function_get_curmsg_id(v); /* expunge the message list */ owl_messagelist_expunge(ml); owl_function_redisplay_to_nearest(lastmsgid, v); owl_function_makemsg("Messages expunged"); } void owl_function_firstmsg(void) { owl_global_set_curmsg(&g, 0); owl_global_set_topmsg(&g, 0); owl_mainwin_redisplay(owl_global_get_mainwin(&g)); owl_global_set_direction_downwards(&g); } void owl_function_lastmsg(void) { int oldcurmsg, curmsg; const owl_view *v; v=owl_global_get_current_view(&g); oldcurmsg=owl_global_get_curmsg(&g); curmsg=owl_view_get_size(v)-1; if (curmsg<0) curmsg=0; owl_global_set_curmsg(&g, curmsg); if (oldcurmsg < curmsg) { owl_function_calculate_topmsg(OWL_DIRECTION_DOWNWARDS); } else if (curmsg 0) { owl_global_set_rightshift(&g, MAX(shift - 10, 0)); } else { owl_function_beep(); owl_function_makemsg("Already full left"); } } void owl_function_unsuball(void) { if (unsuball()) owl_function_makemsg("Unsubscribed from all messages."); } /* Load zephyr subscriptions from the named 'file' and load zephyr's * default subscriptions as well. An error message is printed if * 'file' can't be opened or if zephyr reports an error in * subscribing. * * If 'file' is NULL, this look for the default filename * $HOME/.zephyr.subs. If the file can not be opened in this case * only, no error message is printed. */ void owl_function_loadsubs(const char *file) { int ret, ret2, ret3; char *path; if (file==NULL) { ret=owl_zephyr_loadsubs(NULL, 0); } else { path = owl_util_makepath(file); ret=owl_zephyr_loadsubs(path, 1); g_free(path); } /* for backwards compatibility for now */ ret2 = owl_zephyr_loaddefaultsubs(); ret3 = owl_zephyr_loadbarnowldefaultsubs(); if (!owl_context_is_interactive(owl_global_get_context(&g))) return; if (ret == 0 && ret2 == 0 && ret3 == 0) { if (!file) { owl_function_makemsg("Subscribed to messages."); } else { owl_function_makemsg("Subscribed to messages from %s", file); } } else if (ret == -1) { owl_function_error("Could not read %s", file ? file : "file"); } else if (ret2 == -1) { owl_function_error("Error subscribing to messages"); } else { owl_function_error("Error subscribing to instanced personals"); } } void owl_function_loadloginsubs(const char *file) { int ret; ret=owl_zephyr_loadloginsubs(file); if (!owl_context_is_interactive(owl_global_get_context(&g))) return; if (ret==0) { } else if (ret==-1) { owl_function_error("Could not open file for login subscriptions."); } else { owl_function_error("Error subscribing to login messages from file."); } } void owl_callback_aimlogin(owl_editwin *e, bool success) { if (!success) return; char *user = owl_editwin_get_cbdata(e); owl_function_aimlogin(user, owl_editwin_get_text(e)); } void owl_function_aimlogin(const char *user, const char *passwd) { int ret; /* clear the buddylist */ owl_buddylist_clear(owl_global_get_buddylist(&g)); /* try to login */ ret=owl_aim_login(user, passwd); if (ret) owl_function_makemsg("Warning: login for %s failed.\n", user); } void owl_function_suspend(void) { endwin(); printf("\n"); kill(getpid(), SIGSTOP); /* resize to reinitialize all the windows when we come back */ owl_command_resize(); } void owl_function_zaway_toggle(void) { if (!owl_global_is_zaway(&g)) { owl_global_set_zaway_msg(&g, owl_global_get_zaway_msg_default(&g)); owl_function_zaway_on(); } else { owl_function_zaway_off(); } } void owl_function_zaway_on(void) { owl_global_set_zaway_on(&g); owl_function_makemsg("zaway set (%s)", owl_global_get_zaway_msg(&g)); } void owl_function_zaway_off(void) { owl_global_set_zaway_off(&g); owl_function_makemsg("zaway off"); } void owl_function_aaway_toggle(void) { if (!owl_global_is_aaway(&g)) { owl_global_set_aaway_msg(&g, owl_global_get_aaway_msg_default(&g)); owl_function_aaway_on(); } else { owl_function_aaway_off(); } } void owl_function_aaway_on(void) { owl_global_set_aaway_on(&g); /* owl_aim_set_awaymsg(owl_global_get_zaway_msg(&g)); */ owl_function_makemsg("AIM away set (%s)", owl_global_get_aaway_msg(&g)); } void owl_function_aaway_off(void) { owl_global_set_aaway_off(&g); /* owl_aim_set_awaymsg(""); */ owl_function_makemsg("AIM away off"); } bool owl_function_is_away(void) { return owl_global_is_zaway(&g) || owl_global_is_aaway(&g) || owl_perlconfig_perl_call_bool("BarnOwl::Hooks::_get_is_away", 0, NULL); } void owl_function_quit(void) { char *ret; /* zlog out if we need to */ if (owl_global_is_havezephyr(&g) && owl_global_is_shutdownlogout(&g)) { owl_zephyr_zlog_out(); } /* execute the commands in shutdown */ ret = owl_perlconfig_execute("BarnOwl::Hooks::_shutdown();"); g_free(ret); /* signal our child process, if any */ if (owl_global_get_newmsgproc_pid(&g)) { kill(owl_global_get_newmsgproc_pid(&g), SIGHUP); } /* Quit AIM */ if (owl_global_is_aimloggedin(&g)) { owl_aim_logout(); } owl_function_debugmsg("Quitting BarnOwl"); owl_select_quit_loop(); } void owl_function_calculate_topmsg(int direction) { int recwinlines, topmsg, curmsg; const owl_view *v; v=owl_global_get_current_view(&g); curmsg=owl_global_get_curmsg(&g); topmsg=owl_global_get_topmsg(&g); recwinlines=owl_global_get_recwin_lines(&g); /* if (owl_view_get_size(v) < 1) { return; } */ switch (owl_global_get_scrollmode(&g)) { case OWL_SCROLLMODE_TOP: topmsg = owl_function_calculate_topmsg_top(direction, v, curmsg, topmsg, recwinlines); break; case OWL_SCROLLMODE_NEARTOP: topmsg = owl_function_calculate_topmsg_neartop(direction, v, curmsg, topmsg, recwinlines); break; case OWL_SCROLLMODE_CENTER: topmsg = owl_function_calculate_topmsg_center(direction, v, curmsg, topmsg, recwinlines); break; case OWL_SCROLLMODE_PAGED: topmsg = owl_function_calculate_topmsg_paged(direction, v, curmsg, topmsg, recwinlines, 0); break; case OWL_SCROLLMODE_PAGEDCENTER: topmsg = owl_function_calculate_topmsg_paged(direction, v, curmsg, topmsg, recwinlines, 1); break; case OWL_SCROLLMODE_NORMAL: default: topmsg = owl_function_calculate_topmsg_normal(direction, v, curmsg, topmsg, recwinlines); } owl_function_debugmsg("Calculated a topmsg of %i", topmsg); owl_global_set_topmsg(&g, topmsg); } /* Returns what the new topmsg should be. * Passed the last direction of movement, * the current view, * the current message number in the view, * the top message currently being displayed, * and the number of lines in the recwin. */ int owl_function_calculate_topmsg_top(int direction, const owl_view *v, int curmsg, int topmsg, int recwinlines) { return(curmsg); } int owl_function_calculate_topmsg_neartop(int direction, const owl_view *v, int curmsg, int topmsg, int recwinlines) { if (curmsg>0 && (owl_message_get_numlines(owl_view_get_element(v, curmsg-1)) < recwinlines/2)) { return(curmsg-1); } else { return(curmsg); } } int owl_function_calculate_topmsg_center(int direction, const owl_view *v, int curmsg, int topmsg, int recwinlines) { int i, last, lines; last = curmsg; lines = 0; for (i=curmsg-1; i>=0; i--) { lines += owl_message_get_numlines(owl_view_get_element(v, i)); if (lines > recwinlines/2) break; last = i; } return(last); } int owl_function_calculate_topmsg_paged(int direction, const owl_view *v, int curmsg, int topmsg, int recwinlines, int center_on_page) { int i, last, lines, savey; /* If we're off the top of the screen, scroll up such that the * curmsg is near the botton of the screen. */ if (curmsg < topmsg) { last = curmsg; lines = 0; for (i=curmsg; i>=0; i--) { lines += owl_message_get_numlines(owl_view_get_element(v, i)); if (lines > recwinlines) break; last = i; } if (center_on_page) { return(owl_function_calculate_topmsg_center(direction, v, curmsg, 0, recwinlines)); } else { return(last); } } /* Find number of lines from top to bottom of curmsg (store in savey) */ savey=0; for (i=topmsg; i<=curmsg; i++) { savey+=owl_message_get_numlines(owl_view_get_element(v, i)); } /* if we're off the bottom of the screen, scroll down */ if (savey > recwinlines) { if (center_on_page) { return(owl_function_calculate_topmsg_center(direction, v, curmsg, 0, recwinlines)); } else { return(curmsg); } } /* else just stay as we are... */ return(topmsg); } int owl_function_calculate_topmsg_normal(int direction, const owl_view *v, int curmsg, int topmsg, int recwinlines) { int savey, i, foo, y; if (curmsg<0) return(topmsg); /* If we're off the top of the screen then center */ if (curmsg recwinlines) { topmsg=curmsg; savey=owl_message_get_numlines(owl_view_get_element(v, curmsg)); direction=OWL_DIRECTION_UPWARDS; } /* If our bottom line is less than 1/4 down the screen then scroll up */ if (direction == OWL_DIRECTION_UPWARDS || direction == OWL_DIRECTION_NONE) { if (savey < (recwinlines / 4)) { y=0; for (i=curmsg; i>=0; i--) { foo=owl_message_get_numlines(owl_view_get_element(v, i)); /* will we run the curmsg off the screen? */ if ((foo+y) >= recwinlines) { i++; if (i>curmsg) i=curmsg; break; } /* have saved 1/2 the screen space? */ y+=foo; if (y > (recwinlines / 2)) break; } if (i<0) i=0; return(i); } } if (direction == OWL_DIRECTION_DOWNWARDS || direction == OWL_DIRECTION_NONE) { /* If curmsg bottom line is more than 3/4 down the screen then scroll down */ if (savey > ((recwinlines * 3)/4)) { y=0; /* count lines from the top until we can save 1/2 the screen size */ for (i=topmsg; i (recwinlines / 2)) break; } if (i==curmsg) { i--; } return(i+1); } } return(topmsg); } void owl_function_resize(void) { owl_global_set_resize_pending(&g); } void G_GNUC_PRINTF(1, 2) owl_function_debugmsg(const char *fmt, ...) { char *tmpbuff; FILE *file; time_t now; va_list ap; va_start(ap, fmt); if (!owl_global_is_debug_fast(&g)) return; file = owl_global_get_debug_file_handle(&g); if (!file) /* XXX should report this */ return; now = time(NULL); tmpbuff = owl_util_format_time(localtime(&now)); fprintf(file, "[%d - %s - %lds]: ", (int) getpid(), tmpbuff, now - owl_global_get_starttime(&g)); g_free(tmpbuff); vfprintf(file, fmt, ap); putc('\n', file); fflush(file); va_end(ap); } void owl_function_beep(void) { if (owl_global_is_bell(&g)) { beep(); } } int owl_function_subscribe(const char *class, const char *inst, const char *recip) { int ret; ret=owl_zephyr_sub(class, inst, recip); if (ret) { owl_function_error("Error subscribing."); } else { owl_function_makemsg("Subscribed."); } return(ret); } void owl_function_unsubscribe(const char *class, const char *inst, const char *recip) { int ret; ret=owl_zephyr_unsub(class, inst, recip); if (ret) { owl_function_error("Error subscribing."); } else { owl_function_makemsg("Unsubscribed."); } } static void _dirty_everything(gpointer data, gpointer user_data) { owl_window *w = data; if (!owl_window_is_realized(w)) return; owl_window_dirty(w); owl_window_children_foreach(w, _dirty_everything, NULL); } void owl_function_full_redisplay(void) { /* Ask every widget to redraw itself. */ _dirty_everything(owl_window_get_screen(), NULL); /* Force ncurses to redisplay everything. */ clearok(stdscr, TRUE); } void owl_function_popless_text(const char *text) { owl_popwin *pw; owl_viewwin *v; if (owl_global_get_popwin(&g) || owl_global_get_viewwin(&g)) { owl_function_error("Popwin already in use."); return; } pw = owl_popwin_new(); owl_global_set_popwin(&g, pw); owl_popwin_up(pw); v = owl_viewwin_new_text(owl_popwin_get_content(pw), text); owl_global_set_viewwin(&g, v); owl_global_push_context(&g, OWL_CTX_POPLESS, v, "popless", NULL); } void owl_function_popless_fmtext(const owl_fmtext *fm) { owl_popwin *pw; owl_viewwin *v; if (owl_global_get_popwin(&g) || owl_global_get_viewwin(&g)) { owl_function_error("Popwin already in use."); return; } pw = owl_popwin_new(); owl_global_set_popwin(&g, pw); owl_popwin_up(pw); v = owl_viewwin_new_fmtext(owl_popwin_get_content(pw), fm); owl_global_set_viewwin(&g, v); owl_global_push_context(&g, OWL_CTX_POPLESS, v, "popless", NULL); } void owl_function_popless_file(const char *filename) { owl_fmtext fm; FILE *file; char *s = NULL; file=fopen(filename, "r"); if (!file) { owl_function_error("Could not open file: %s", filename); return; } owl_fmtext_init_null(&fm); while (owl_getline(&s, file)) owl_fmtext_append_normal(&fm, s); g_free(s); owl_function_popless_fmtext(&fm); owl_fmtext_cleanup(&fm); fclose(file); } void owl_function_about(void) { char *text = g_strdup_printf( "This is BarnOwl version %s.\n\n" "BarnOwl is a fork of the Owl zephyr client, written and\n" "maintained by Alejandro Sedeno and Nelson Elhage at the\n" "Massachusetts Institute of Technology. \n" "\n" "Owl was written by James Kretchmar. The first version, 0.5, was\n" "released in March 2002.\n" "\n" "The name 'owl' was chosen in reference to the owls in the\n" "Harry Potter novels, who are tasked with carrying messages\n" "between Witches and Wizards. The name 'BarnOwl' was chosen\n" "because we feel our owls should live closer to our ponies.\n" "\n" "Copyright (c) 2006-2011 The BarnOwl Developers. All rights reserved.\n" "Copyright (c) 2004 James Kretchmar. All rights reserved.\n" "Copyright 2002 Massachusetts Institute of Technology\n" "\n" "This program is free software. You can redistribute it and/or\n" "modify under the terms of the Sleepycat License. Use the \n" "':show license' command to display the full license\n", version); owl_function_popless_text(text); g_free(text); } void owl_function_info(void) { const owl_message *m; owl_fmtext fm, attrfm; const owl_view *v; char *time; #ifdef HAVE_LIBZEPHYR const ZNotice_t *n; #endif owl_fmtext_init_null(&fm); v=owl_global_get_current_view(&g); m=owl_view_get_element(v, owl_global_get_curmsg(&g)); if (!m || owl_view_get_size(v)==0) { owl_function_error("No message selected\n"); return; } owl_fmtext_append_bold(&fm, "General Information:\n"); owl_fmtext_appendf_normal(&fm, " Msg Id : %i\n", owl_message_get_id(m)); owl_fmtext_append_normal(&fm, " Type : "); owl_fmtext_append_bold(&fm, owl_message_get_type(m)); owl_fmtext_append_normal(&fm, "\n"); if (owl_message_is_direction_in(m)) { owl_fmtext_append_normal(&fm, " Direction : in\n"); } else if (owl_message_is_direction_out(m)) { owl_fmtext_append_normal(&fm, " Direction : out\n"); } else if (owl_message_is_direction_none(m)) { owl_fmtext_append_normal(&fm, " Direction : none\n"); } else { owl_fmtext_append_normal(&fm, " Direction : unknown\n"); } time = owl_message_format_time(m); owl_fmtext_appendf_normal(&fm, " Time : %s\n", time); g_free(time); if (!owl_message_is_type_admin(m)) { owl_fmtext_appendf_normal(&fm, " Sender : %s\n", owl_message_get_sender(m)); owl_fmtext_appendf_normal(&fm, " Recipient : %s\n", owl_message_get_recipient(m)); } if (owl_message_is_type_zephyr(m)) { owl_fmtext_append_bold(&fm, "\nZephyr Specific Information:\n"); owl_fmtext_appendf_normal(&fm, " Class : %s\n", owl_message_get_class(m)); owl_fmtext_appendf_normal(&fm, " Instance : %s\n", owl_message_get_instance(m)); owl_fmtext_appendf_normal(&fm, " Opcode : %s\n", owl_message_get_opcode(m)); #ifdef HAVE_LIBZEPHYR n = owl_message_get_notice(m); if (n != NULL) { char *tmpbuff, *tmpbuff2; int i, fields; const char *f; if (!owl_message_is_pseudo(m)) { owl_fmtext_append_normal(&fm, " Kind : "); if (n->z_kind==UNSAFE) { owl_fmtext_append_normal(&fm, "UNSAFE\n"); } else if (n->z_kind==UNACKED) { owl_fmtext_append_normal(&fm, "UNACKED\n"); } else if (n->z_kind==ACKED) { owl_fmtext_append_normal(&fm, "ACKED\n"); } else if (n->z_kind==HMACK) { owl_fmtext_append_normal(&fm, "HMACK\n"); } else if (n->z_kind==HMCTL) { owl_fmtext_append_normal(&fm, "HMCTL\n"); } else if (n->z_kind==SERVACK) { owl_fmtext_append_normal(&fm, "SERVACK\n"); } else if (n->z_kind==SERVNAK) { owl_fmtext_append_normal(&fm, "SERVNACK\n"); } else if (n->z_kind==CLIENTACK) { owl_fmtext_append_normal(&fm, "CLIENTACK\n"); } else if (n->z_kind==STAT) { owl_fmtext_append_normal(&fm, "STAT\n"); } else { owl_fmtext_append_normal(&fm, "ILLEGAL VALUE\n"); } } owl_fmtext_appendf_normal(&fm, " Host : %s\n", owl_message_get_hostname(m)); if (!owl_message_is_pseudo(m)) { owl_fmtext_append_normal(&fm, "\n"); owl_fmtext_appendf_normal(&fm, " Port : %i\n", ntohs(n->z_port)); owl_fmtext_appendf_normal(&fm, " Charset : %s\n", owl_zephyr_get_charsetstr(n)); owl_fmtext_appendf_normal(&fm, " Auth : %s\n", owl_zephyr_get_authstr(n)); /* FIXME make these more descriptive */ owl_fmtext_appendf_normal(&fm, " Checkd Ath: %i\n", n->z_checked_auth); owl_fmtext_appendf_normal(&fm, " Multi notc: %s\n", n->z_multinotice); owl_fmtext_appendf_normal(&fm, " Num other : %i\n", n->z_num_other_fields); owl_fmtext_appendf_normal(&fm, " Msg Len : %i\n", n->z_message_len); fields=owl_zephyr_get_num_fields(n); owl_fmtext_appendf_normal(&fm, " Fields : %i\n", fields); for (i = 0, f = owl_zephyr_first_raw_field(n); f != NULL; i++, f = owl_zephyr_next_raw_field(n, f)) { tmpbuff = owl_zephyr_field_as_utf8(n, f); tmpbuff2 = owl_text_indent(tmpbuff, 14, false); owl_fmtext_appendf_normal(&fm, " Field %i : %s\n", i + 1, tmpbuff2); g_free(tmpbuff2); g_free(tmpbuff); } tmpbuff = owl_text_indent(n->z_default_format, 14, false); owl_fmtext_appendf_normal(&fm, " Default Fm: %s\n", tmpbuff); g_free(tmpbuff); } } #endif } owl_fmtext_append_bold(&fm, "\nBarnOwl Message Attributes:\n"); owl_message_attributes_tofmtext(m, &attrfm); owl_fmtext_append_fmtext(&fm, &attrfm); owl_function_popless_fmtext(&fm); owl_fmtext_cleanup(&fm); owl_fmtext_cleanup(&attrfm); } /* print the current message in a popup window. * Use the 'default' style regardless of whatever * style the user may be using */ void owl_function_curmsg_to_popwin(void) { const owl_view *v; const owl_message *m; const owl_style *s; owl_fmtext fm; v=owl_global_get_current_view(&g); s=owl_global_get_style_by_name(&g, "default"); m=owl_view_get_element(v, owl_global_get_curmsg(&g)); if (!m || owl_view_get_size(v)==0) { owl_function_error("No current message"); return; } owl_fmtext_init_null(&fm); owl_style_get_formattext(s, &fm, m); owl_function_popless_fmtext(&fm); owl_fmtext_cleanup(&fm); } void owl_function_page_curmsg(int step) { /* scroll down or up within the current message IF the message is truncated */ int offset, curmsg, lines; const owl_view *v; owl_message *m; offset=owl_global_get_curmsg_vert_offset(&g); v=owl_global_get_current_view(&g); curmsg=owl_global_get_curmsg(&g); m=owl_view_get_element(v, curmsg); if (!m || owl_view_get_size(v)==0) return; lines=owl_message_get_numlines(m); if (offset==0) { /* Bail if the curmsg isn't the last one displayed */ if (curmsg != owl_mainwin_get_last_msg(owl_global_get_mainwin(&g))) { owl_function_makemsg("The entire message is already displayed"); return; } /* Bail if we're not truncated */ if (!owl_mainwin_is_curmsg_truncated(owl_global_get_mainwin(&g))) { owl_function_makemsg("The entire message is already displayed"); return; } } /* don't scroll past the last line */ if (step>0) { if (offset+step > lines-1) { owl_global_set_curmsg_vert_offset(&g, lines-1); } else { owl_global_set_curmsg_vert_offset(&g, offset+step); } } /* would we be before the beginning of the message? */ if (step<0) { if (offset+step<0) { owl_global_set_curmsg_vert_offset(&g, 0); } else { owl_global_set_curmsg_vert_offset(&g, offset+step); } } /* redisplay */ owl_mainwin_redisplay(owl_global_get_mainwin(&g)); } void owl_function_resize_typwin(int newsize) { owl_global_set_typwin_lines(&g, newsize); owl_mainpanel_layout_contents(&g.mainpanel); } void owl_function_mainwin_pagedown(void) { int i; i=owl_mainwin_get_last_msg(owl_global_get_mainwin(&g)); if (i<0) return; if (owl_mainwin_is_last_msg_truncated(owl_global_get_mainwin(&g)) && (owl_global_get_curmsg(&g) < i) && (i>0)) { i--; } owl_global_set_curmsg(&g, i); owl_function_nextmsg(); } void owl_function_mainwin_pageup(void) { owl_global_set_curmsg(&g, owl_global_get_topmsg(&g)); owl_function_prevmsg(); } void owl_function_getsubs(void) { char *buff; buff=owl_zephyr_getsubs(); if (buff) { owl_function_popless_text(buff); } else { owl_function_popless_text("Error getting subscriptions"); } g_free(buff); } void owl_function_printallvars(void) { const owl_variable *v; const char *name; char *var; GPtrArray *varnames; int i; GString *str = g_string_new(""); g_string_append_printf(str, "%-20s = %s\n", "VARIABLE", "VALUE"); g_string_append_printf(str, "%-20s %s\n", "--------", "-----"); varnames = owl_variable_dict_get_names(owl_global_get_vardict(&g)); for (i = 0; i < varnames->len; i++) { name = varnames->pdata[i]; if (name && name[0]!='_') { g_string_append_printf(str, "\n%-20s = ", name); v = owl_variable_get_var(owl_global_get_vardict(&g), name); var = owl_variable_get_tostring(v); if (var) { g_string_append(str, var); g_free(var); } else { g_string_append(str, ""); } } } g_string_append(str, "\n"); owl_ptr_array_free(varnames, g_free); owl_function_popless_text(str->str); g_string_free(str, true); } void owl_function_show_variables(void) { const owl_variable *v; GPtrArray *varnames; owl_fmtext fm; int i; const char *varname; owl_fmtext_init_null(&fm); owl_fmtext_append_bold(&fm, "Variables: (use 'show variable ' for details)\n"); varnames = owl_variable_dict_get_names(owl_global_get_vardict(&g)); for (i = 0; i < varnames->len; i++) { varname = varnames->pdata[i]; if (varname && varname[0]!='_') { v = owl_variable_get_var(owl_global_get_vardict(&g), varname); owl_variable_describe(v, &fm); } } owl_ptr_array_free(varnames, g_free); owl_function_popless_fmtext(&fm); owl_fmtext_cleanup(&fm); } void owl_function_show_variable(const char *name) { const owl_variable *v; owl_fmtext fm; owl_fmtext_init_null(&fm); v = owl_variable_get_var(owl_global_get_vardict(&g), name); if (v) owl_variable_get_help(v, &fm); else owl_fmtext_append_normal(&fm, "No such variable...\n"); owl_function_popless_fmtext(&fm); owl_fmtext_cleanup(&fm); } void owl_function_delete_and_expunge_by_id(int id, bool exclaim_success) { const owl_messagelist *ml = owl_global_get_msglist(&g); int msg = owl_messagelist_get_index_by_id(ml, id); if (msg < 0) { owl_function_error("No message with id %d: unable to delete", id); } else { owl_function_delete_and_expunge_message(msg); if (exclaim_success) owl_function_makemsg("Message deleted and expunged"); } } /* note: this applies to global message list, not to view. * If flag is 1, deletes. If flag is 0, undeletes. */ void owl_function_delete_by_id(int id, int flag) { const owl_messagelist *ml; owl_message *m; ml = owl_global_get_msglist(&g); m = owl_messagelist_get_by_id(ml, id); if (m) { if (flag == 1) { owl_message_mark_delete(m); } else if (flag == 0) { owl_message_unmark_delete(m); } owl_mainwin_redisplay(owl_global_get_mainwin(&g)); } else { owl_function_error("No message with id %d: unable to mark for (un)delete",id); } } void owl_function_delete_automsgs(void) { /* mark for deletion all messages in the current view that match the * 'trash' filter */ int i, j, count; owl_message *m; const owl_view *v; const owl_filter *f; /* get the trash filter */ f=owl_global_get_filter(&g, "trash"); if (!f) { owl_function_error("No trash filter defined"); return; } v=owl_global_get_current_view(&g); count=0; j=owl_view_get_size(v); for (i=0; i"); } else { owl_fmtext_append_normal(&fm, buff); } owl_fmtext_append_normal(&fm, "\n"); tmpbuff = owl_util_format_time(localtime(&start)); owl_fmtext_appendf_normal(&fm, " Startup Time: %s\n", tmpbuff); g_free(tmpbuff); up=owl_global_get_runtime(&g); days=up/86400; up-=days*86400; hours=up/3600; up-=hours*3600; minutes=up/60; up-=minutes*60; owl_fmtext_appendf_normal(&fm, " Run Time: %i days %2.2i:%2.2i:%2.2i\n", days, hours, minutes, up); owl_fmtext_append_normal(&fm, "\nProtocol Options:\n"); owl_fmtext_append_normal(&fm, " Zephyr included : "); if (owl_global_is_havezephyr(&g)) { owl_fmtext_append_normal(&fm, "yes\n"); } else { owl_fmtext_append_normal(&fm, "no\n"); } owl_fmtext_append_normal(&fm, " AIM included : yes\n"); owl_fmtext_append_normal(&fm, " Loopback included : yes\n"); owl_fmtext_append_normal(&fm, "\nBuild Options:\n"); owl_fmtext_append_normal(&fm, " Stderr redirection : "); #if OWL_STDERR_REDIR owl_fmtext_append_normal(&fm, "yes\n"); #else owl_fmtext_append_normal(&fm, "no\n"); #endif owl_fmtext_append_normal(&fm, "\nAIM Status:\n"); owl_fmtext_append_normal(&fm, " Logged in: "); if (owl_global_is_aimloggedin(&g)) { owl_fmtext_append_normal(&fm, owl_global_get_aim_screenname(&g)); owl_fmtext_append_normal(&fm, "\n"); } else { owl_fmtext_append_normal(&fm, "(not logged in)\n"); } owl_fmtext_append_normal(&fm, " Processing events: "); if (owl_global_is_doaimevents(&g)) { owl_fmtext_append_normal(&fm, "yes\n"); } else { owl_fmtext_append_normal(&fm, "no\n"); } owl_function_popless_fmtext(&fm); owl_fmtext_cleanup(&fm); } void owl_function_show_term(void) { owl_fmtext fm; owl_fmtext_init_null(&fm); owl_fmtext_appendf_normal(&fm, "Terminal Lines: %i\nTerminal Columns: %i\n", owl_global_get_lines(&g), owl_global_get_cols(&g)); if (has_colors()) { owl_fmtext_append_normal(&fm, "Color: Yes\n"); owl_fmtext_appendf_normal(&fm, "Number of color pairs: %i\n", owl_util_get_colorpairs()); owl_fmtext_appendf_normal(&fm, "Can change colors: %s\n", can_change_color() ? "yes" : "no"); } else { owl_fmtext_append_normal(&fm, "Color: No\n"); } owl_function_popless_fmtext(&fm); owl_fmtext_cleanup(&fm); } /* if type = 0 then normal reply. * if type = 1 then it's a reply to sender * if enter = 0 then allow the command to be edited * if enter = 1 then don't wait for editing */ void owl_function_reply(int type, int enter) { char *buff=NULL; const owl_message *m; const owl_filter *f; if (owl_view_get_size(owl_global_get_current_view(&g))==0) { owl_function_error("No message selected"); } else { char *cmd; m=owl_view_get_element(owl_global_get_current_view(&g), owl_global_get_curmsg(&g)); if (!m) { owl_function_error("No message selected"); return; } /* first check if we catch the reply-lockout filter */ f=owl_global_get_filter(&g, "reply-lockout"); if (f) { if (owl_filter_message_match(f, m)) { owl_function_error("Sorry, replies to this message have been disabled by the reply-lockout filter"); return; } } /* then check if it's a question and just bring up the command prompt */ if (owl_message_is_question(m)) { owl_function_start_command(""); return; } if((type == 0 && (cmd=owl_perlconfig_message_call_method(m, "replycmd", 0, NULL))) || (type == 1 && (cmd=owl_perlconfig_message_call_method(m, "replysendercmd", 0, NULL)))) { buff = cmd; } if(!buff) { owl_function_error("I don't know how to reply to that message."); return; } if (enter) { owl_history *hist = owl_global_get_cmd_history(&g); owl_history_store(hist, buff, false); owl_function_command_norv(buff); } else { owl_function_start_command(buff); } g_free(buff); } } void owl_function_zlocate(int argc, const char *const *argv, int auth) { owl_fmtext fm; char *ptr; char *result; int i; owl_fmtext_init_null(&fm); for (i=0; i' defined as "not filter ". If the * filter 'not-' already exists, do not overwrite it. If * 'filtername' begins with 'not-' and a filter 'filtername' already * exists, then do nothing. If the filter 'filtername' does not * exist, create it and define it as 'not filter ' * * Returns the name of the negated filter, which the caller must free. */ CALLER_OWN char *owl_function_create_negative_filter(const char *filtername) { char *newname; const owl_filter *tmpfilt; const char *argv[5]; owl_function_debugmsg("owl_function_create_negative_filter"); if (!strncmp(filtername, "not-", 4)) { newname=g_strdup(filtername+4); } else { newname=g_strdup_printf("not-%s", filtername); } tmpfilt=owl_global_get_filter(&g, newname); if (!tmpfilt) { argv[0]="filter"; /* anything is fine here */ argv[1]=newname; argv[2]="not"; argv[3]="filter"; argv[4]=filtername; owl_function_create_filter(5, argv); } owl_function_debugmsg("owl_function_create_negative_filter: returning with %s", newname); return(newname); } void owl_function_show_filters(void) { const owl_filter *f; GList *fl; owl_fmtext fm; owl_fmtext_init_null(&fm); owl_fmtext_append_bold(&fm, "Filters:\n"); for (fl = g.filterlist; fl; fl = g_list_next(fl)) { f = fl->data; owl_fmtext_append_normal(&fm, " "); owl_fmtext_append_normal_color(&fm, owl_filter_get_name(f), owl_filter_get_fgcolor(f), owl_filter_get_bgcolor(f)); owl_fmtext_append_normal(&fm, "\n"); } owl_function_popless_fmtext(&fm); owl_fmtext_cleanup(&fm); } void owl_function_show_filter(const char *name) { const owl_filter *f; char *buff, *tmp; f=owl_global_get_filter(&g, name); if (!f) { owl_function_error("There is no filter named %s", name); return; } tmp = owl_filter_print(f); buff = g_strdup_printf("%s: %s", owl_filter_get_name(f), tmp); owl_function_popless_text(buff); g_free(buff); g_free(tmp); } void owl_function_show_zpunts(void) { const owl_filter *f; const GPtrArray *fl; char *tmp; owl_fmtext fm; int i; owl_fmtext_init_null(&fm); fl=owl_global_get_puntlist(&g); owl_fmtext_append_bold(&fm, "Active zpunt filters:\n"); for (i = 0; i < fl->len; i++) { f = fl->pdata[i]; owl_fmtext_appendf_normal(&fm, "[% 2d] ", i+1); tmp = owl_filter_print(f); owl_fmtext_append_normal(&fm, tmp); g_free(tmp); } owl_function_popless_fmtext(&fm); owl_fmtext_cleanup(&fm); } /* Create a filter for a class, instance if one doesn't exist. If * instance is NULL then catch all messgaes in the class. Returns the * name of the filter or null. The caller must free this name. * If 'related' is nonzero, encompass unclasses and .d classes as well. */ CALLER_OWN char *owl_function_classinstfilt(const char *c, const char *i, int related) { owl_filter *f; char *filtname; char *tmpclass, *tmpinstance = NULL; char *class, *instance = NULL; GString *buf; if (related) { class = owl_util_baseclass(c); if (i) { instance = owl_util_baseclass(i); } } else { class = g_strdup(c); if (i) { instance = g_strdup(i); } } /* name for the filter */ if (!instance) { filtname = g_strdup_printf("%sclass-%s", related ? "related-" : "", class); } else { filtname = g_strdup_printf("%sclass-%s-instance-%s", related ? "related-" : "", class, instance); } /* downcase it */ { char *temp = g_utf8_strdown(filtname, -1); if (temp) { g_free(filtname); filtname = temp; } } /* if it already exists then go with it. This lets users override */ if (owl_global_get_filter(&g, filtname)) { goto done; } /* create the new filter */ tmpclass=owl_text_quote(class, OWL_REGEX_QUOTECHARS, OWL_REGEX_QUOTEWITH); if (instance) { tmpinstance=owl_text_quote(instance, OWL_REGEX_QUOTECHARS, OWL_REGEX_QUOTEWITH); } buf = g_string_new(""); owl_string_appendf_quoted(buf, related ? "class ^(un)*%q(\\.d)*$" : "class ^%q$", tmpclass); if (tmpinstance) { owl_string_appendf_quoted(buf, related ? " and ( instance ^(un)*%q(\\.d)*$ )" : " and instance ^%q$", tmpinstance); } g_free(tmpclass); g_free(tmpinstance); f = owl_filter_new_fromstring(filtname, buf->str); g_string_free(buf, true); if (f == NULL) { /* Couldn't make a filter for some reason. Return NULL. */ owl_function_error("Error creating filter '%s'", filtname); g_free(filtname); filtname = NULL; goto done; } /* add it to the global list */ owl_global_add_filter(&g, f); done: g_free(class); g_free(instance); return(filtname); } /* Create a filter for personal zephyrs to or from the specified * zephyr user. Includes login/logout notifications for the user. * The name of the filter will be 'user-'. If a filter already * exists with this name, no new filter will be created. This allows * the configuration to override this function. Returns the name of * the filter, which the caller must free. */ CALLER_OWN char *owl_function_zuserfilt(const char *longuser) { owl_filter *f; char *argbuff, *esclonguser, *shortuser, *filtname; /* name for the filter */ shortuser = short_zuser(longuser); filtname = g_strdup_printf("user-%s", shortuser); g_free(shortuser); /* if it already exists then go with it. This lets users override */ if (owl_global_get_filter(&g, filtname)) { return filtname; } /* create the new-internal filter */ esclonguser = owl_text_quote(longuser, OWL_REGEX_QUOTECHARS, OWL_REGEX_QUOTEWITH); argbuff=owl_string_build_quoted("( type ^zephyr$ and filter personal and " "( ( direction ^in$ and sender ^%q$ ) or ( direction ^out$ and " "recipient ^%q$ ) ) ) or ( ( class ^login$ ) and ( sender ^%q$ ) )", esclonguser, esclonguser, esclonguser); g_free(esclonguser); f = owl_filter_new_fromstring(filtname, argbuff); g_free(argbuff); if (f == NULL) { /* Couldn't make a filter for some reason. Return NULL. */ owl_function_error("Error creating filter '%s'", filtname); g_free(filtname); return NULL; } /* add it to the global list */ owl_global_add_filter(&g, f); return(filtname); } /* Create a filter for AIM IM messages to or from the specified * screenname. The name of the filter will be 'aimuser-'. If a * filter already exists with this name, no new filter will be * created. This allows the configuration to override this function. * Returns the name of the filter, which the caller must free. */ CALLER_OWN char *owl_function_aimuserfilt(const char *user) { owl_filter *f; char *argbuff, *filtname; char *escuser; /* name for the filter */ filtname=g_strdup_printf("aimuser-%s", user); /* if it already exists then go with it. This lets users override */ if (owl_global_get_filter(&g, filtname)) { return filtname; } /* create the new-internal filter */ escuser = owl_text_quote(user, OWL_REGEX_QUOTECHARS, OWL_REGEX_QUOTEWITH); argbuff = g_strdup_printf( "( type ^aim$ and ( ( sender ^%1$s$ and recipient ^%2$s$ ) or " "( sender ^%2$s$ and recipient ^%1$s$ ) ) )", escuser, owl_global_get_aim_screenname_for_filters(&g)); g_free(escuser); f = owl_filter_new_fromstring(filtname, argbuff); g_free(argbuff); if (f == NULL) { owl_function_error("Error creating filter '%s'", filtname); g_free(filtname); return NULL; } /* add it to the global list */ owl_global_add_filter(&g, f); return(filtname); } CALLER_OWN char *owl_function_typefilt(const char *type) { owl_filter *f; char *argbuff, *filtname, *esctype; /* name for the filter */ filtname=g_strdup_printf("type-%s", type); /* if it already exists then go with it. This lets users override */ if (owl_global_get_filter(&g, filtname)) { return filtname; } /* create the new-internal filter */ esctype = owl_text_quote(type, OWL_REGEX_QUOTECHARS, OWL_REGEX_QUOTEWITH); argbuff = owl_string_build_quoted("type ^%q$", esctype); g_free(esctype); f = owl_filter_new_fromstring(filtname, argbuff); g_free(argbuff); if (f == NULL) { owl_function_error("Error creating filter '%s'", filtname); g_free(filtname); return NULL; } /* add it to the global list */ owl_global_add_filter(&g, f); return filtname; } /* If flag is 1, marks for deletion. If flag is 0, * unmarks for deletion. */ void owl_function_delete_curview_msgs(int flag) { const owl_view *v; int i, j; v=owl_global_get_current_view(&g); j=owl_view_get_size(v); for (i=0; istr); g_string_free(buf, true); if (f == NULL) { owl_function_error("Error creating filter '%s'", filtname); g_free(filtname); return NULL; } owl_global_add_filter(&g, f); return filtname; } /* Create a filter based on the current message. Returns the name of * a filter or null. The caller must free this name. * * if the curmsg is a personal zephyr return a filter name * to the zephyr conversation with that user. * If the curmsg is a zephyr class message, instance foo, recip *, * return a filter name to the class, inst. * If the curmsg is a zephyr class message and type==0 then * return a filter name for just the class. * If the curmsg is a zephyr class message and type==1 then * return a filter name for the class and instance. * If the curmsg is a personal AIM message returna filter * name to the AIM conversation with that user */ CALLER_OWN char *owl_function_smartfilter(int type, int invert_related) { const owl_view *v; const owl_message *m; char *filtname = NULL; const char *argv[2], *zperson; int related = owl_global_is_narrow_related(&g) ^ invert_related; v=owl_global_get_current_view(&g); m=owl_view_get_element(v, owl_global_get_curmsg(&g)); if (!m || owl_view_get_size(v)==0) { owl_function_error("No message selected\n"); return(NULL); } /* very simple handling of admin messages for now */ if (owl_message_is_type_admin(m)) { return(owl_function_typefilt("admin")); } /* very simple handling of loopback messages for now */ if (owl_message_is_type_loopback(m)) { return(owl_function_typefilt("loopback")); } /* aim messages */ if (owl_message_is_type_aim(m)) { if (owl_message_is_direction_in(m)) { filtname=owl_function_aimuserfilt(owl_message_get_sender(m)); } else if (owl_message_is_direction_out(m)) { filtname=owl_function_aimuserfilt(owl_message_get_recipient(m)); } return(filtname); } /* narrow personal and login messages to the sender or recip as appropriate */ if (owl_message_is_type_zephyr(m)) { if (owl_message_is_personal(m) || owl_message_is_loginout(m)) { if (owl_message_get_attribute_value(m, "zephyr_ccs") != NULL) { return owl_function_smartfilter_cc(m); } if (owl_message_is_direction_in(m)) { zperson = owl_message_get_sender(m); } else { zperson = owl_message_get_recipient(m); } filtname = owl_function_zuserfilt(zperson); return filtname; } /* narrow class MESSAGE, instance foo, recip * messages to class, inst */ if (!strcasecmp(owl_message_get_class(m), "message")) { filtname=owl_function_classinstfilt(owl_message_get_class(m), owl_message_get_instance(m), related); return(filtname); } /* otherwise narrow to the class */ if (type==0) { filtname=owl_function_classinstfilt(owl_message_get_class(m), NULL, related); } else if (type==1) { filtname=owl_function_classinstfilt(owl_message_get_class(m), owl_message_get_instance(m), related); } return(filtname); } /* pass it off to perl */ argv[0] = type ? "1" : "0"; argv[1] = related ? "1" : "0"; return owl_perlconfig_message_call_method(m, "smartfilter", 2, argv); } void owl_function_smartzpunt(int type) { /* Starts a zpunt command based on the current class,instance pair. * If type=0, uses just class. If type=1, uses instance as well. */ const owl_view *v; const owl_message *m; const char *mclass, *minst; GString *buf; v=owl_global_get_current_view(&g); m=owl_view_get_element(v, owl_global_get_curmsg(&g)); if (!m || owl_view_get_size(v)==0) { owl_function_error("No message selected\n"); return; } /* for now we skip admin messages. */ if (owl_message_is_type_admin(m) || owl_message_is_loginout(m) || !owl_message_is_type_zephyr(m)) { owl_function_error("smartzpunt doesn't support this message type."); return; } mclass = owl_message_get_class(m); minst = owl_message_get_instance(m); if (!mclass || !*mclass || *mclass==' ' || (!strcasecmp(mclass, "message") && !strcasecmp(minst, "personal")) || (type && (!minst || !*minst|| *minst==' '))) { owl_function_error("smartzpunt can't safely do this for <%s,%s>", mclass, minst); } else { buf = g_string_new("start-command zpunt "); owl_string_append_quoted_arg(buf, mclass); if (type) { g_string_append_c(buf, ' '); owl_string_append_quoted_arg(buf, minst); } else { g_string_append(buf, " *"); } owl_function_command_norv(buf->str); g_string_free(buf, true); } } /* Set the color of the current view's filter to * be 'color' */ void owl_function_color_current_filter(const char *fgcolor, const char *bgcolor) { const char *name; name=owl_view_get_filtname(owl_global_get_current_view(&g)); owl_function_color_filter(name, fgcolor, bgcolor); } /* Set the color of the filter 'filter' to be 'color'. If the color * name does not exist, return -1, if the filter does not exist or is * the "all" filter, return -2. Return 0 on success */ int owl_function_color_filter(const char *filtname, const char *fgcolor, const char *bgcolor) { owl_filter *f; f=owl_global_get_filter(&g, filtname); if (!f) { owl_function_error("Unknown filter"); return(-2); } /* don't touch the all filter */ if (!strcmp(filtname, "all")) { owl_function_error("You may not change the 'all' filter."); return(-2); } if (owl_util_string_to_color(fgcolor)==OWL_COLOR_INVALID) { owl_function_error("No color named '%s' avilable.", fgcolor); return(-1); } if (bgcolor != NULL) { if (owl_util_string_to_color(bgcolor)==OWL_COLOR_INVALID) { owl_function_error("No color named '%s' avilable.", bgcolor); return(-1); } owl_filter_set_bgcolor(f, owl_util_string_to_color(bgcolor)); } owl_filter_set_fgcolor(f, owl_util_string_to_color(fgcolor)); owl_mainwin_redisplay(owl_global_get_mainwin(&g)); return(0); } void owl_function_show_colors(void) { owl_fmtext fm; int i; owl_fmtext_init_null(&fm); owl_fmtext_append_normal(&fm,"default: "); owl_fmtext_append_normal_color(&fm, "default\n", OWL_COLOR_DEFAULT, OWL_COLOR_DEFAULT); owl_fmtext_append_normal(&fm,"black: "); owl_fmtext_append_normal_color(&fm, "black\n", OWL_COLOR_BLACK, OWL_COLOR_DEFAULT); owl_fmtext_append_normal(&fm,"red: "); owl_fmtext_append_normal_color(&fm, "red\n", OWL_COLOR_RED, OWL_COLOR_DEFAULT); owl_fmtext_append_normal(&fm,"green: "); owl_fmtext_append_normal_color(&fm, "green\n", OWL_COLOR_GREEN, OWL_COLOR_DEFAULT); owl_fmtext_append_normal(&fm,"yellow: "); owl_fmtext_append_normal_color(&fm, "yellow\n", OWL_COLOR_YELLOW, OWL_COLOR_DEFAULT); owl_fmtext_append_normal(&fm,"blue: "); owl_fmtext_append_normal_color(&fm, "blue\n", OWL_COLOR_BLUE, OWL_COLOR_DEFAULT); owl_fmtext_append_normal(&fm,"magenta: "); owl_fmtext_append_normal_color(&fm, "magenta\n", OWL_COLOR_MAGENTA, OWL_COLOR_DEFAULT); owl_fmtext_append_normal(&fm,"cyan: "); owl_fmtext_append_normal_color(&fm, "cyan\n", OWL_COLOR_CYAN, OWL_COLOR_DEFAULT); owl_fmtext_append_normal(&fm,"white: "); owl_fmtext_append_normal_color(&fm, "white\n", OWL_COLOR_WHITE, OWL_COLOR_DEFAULT); for(i = 8; i < COLORS; ++i) { char* str1 = g_strdup_printf("%4i: ",i); char* str2 = g_strdup_printf("%i\n",i); owl_fmtext_append_normal(&fm,str1); owl_fmtext_append_normal_color(&fm, str2, i, OWL_COLOR_DEFAULT); g_free(str1); g_free(str2); } owl_function_popless_fmtext(&fm); owl_fmtext_cleanup(&fm); } /* add the given class, inst, recip to the punt list for filtering. * if direction==0 then punt * if direction==1 then unpunt */ void owl_function_zpunt(const char *class, const char *inst, const char *recip, int direction) { GPtrArray *argv; char *quoted; argv = g_ptr_array_new(); if (!strcmp(class, "*")) { g_ptr_array_add(argv, g_strdup("class")); g_ptr_array_add(argv, g_strdup(".*")); } else { quoted=owl_text_quote(class, OWL_REGEX_QUOTECHARS, OWL_REGEX_QUOTEWITH); g_ptr_array_add(argv, g_strdup("class")); g_ptr_array_add(argv, g_strdup_printf("^(un)*%s(\\.d)*$", quoted)); g_free(quoted); } if (!strcmp(inst, "*")) { g_ptr_array_add(argv, g_strdup("and")); g_ptr_array_add(argv, g_strdup("instance")); g_ptr_array_add(argv, g_strdup(".*")); } else { quoted=owl_text_quote(inst, OWL_REGEX_QUOTECHARS, OWL_REGEX_QUOTEWITH); g_ptr_array_add(argv, g_strdup("and")); g_ptr_array_add(argv, g_strdup("instance")); g_ptr_array_add(argv, g_strdup_printf("^(un)*%s(\\.d)*$", quoted)); g_free(quoted); } if (!strcmp(recip, "*")) { /* nothing */ } else { if(!strcmp(recip, "%me%")) { recip = owl_zephyr_get_sender(); } quoted=owl_text_quote(recip, OWL_REGEX_QUOTECHARS, OWL_REGEX_QUOTEWITH); g_ptr_array_add(argv, g_strdup("and")); g_ptr_array_add(argv, g_strdup("recipient")); g_ptr_array_add(argv, g_strdup_printf("^%s$", quoted)); g_free(quoted); } owl_function_punt(argv->len, (const char *const*) argv->pdata, direction); owl_ptr_array_free(argv, g_free); } void owl_function_punt(int argc, const char *const *argv, int direction) { owl_filter *f; GPtrArray *fl; int i; fl=owl_global_get_puntlist(&g); /* first, create the filter */ f = owl_filter_new("punt-filter", argc, argv); if (f == NULL) { owl_function_error("Error creating filter for zpunt"); return; } /* Check for an identical filter */ for (i = 0; i < fl->len; i++) { if (owl_filter_equiv(f, fl->pdata[i])) { owl_function_debugmsg("found an equivalent punt filter"); /* if we're punting, then just silently bow out on this duplicate */ if (direction==0) { owl_filter_delete(f); return; } /* if we're unpunting, then remove this filter from the puntlist */ if (direction==1) { owl_filter_delete(g_ptr_array_remove_index(fl, i)); owl_filter_delete(f); return; } } } if (direction == 0) { owl_function_debugmsg("punting"); /* If we're punting, add the filter to the global punt list */ g_ptr_array_add(fl, f); } else if (direction == 1) { owl_function_makemsg("No matching punt filter"); } } void owl_function_show_keymaps(void) { GPtrArray *l; owl_fmtext fm; const owl_keymap *km; const owl_keyhandler *kh; int i; const char *kmname; kh = owl_global_get_keyhandler(&g); owl_fmtext_init_null(&fm); owl_fmtext_append_bold(&fm, "Keymaps: "); owl_fmtext_append_normal(&fm, "(use 'show keymap ' for details)\n"); l = owl_keyhandler_get_keymap_names(kh); owl_fmtext_append_list(&fm, l, "\n", owl_function_keymap_summary); owl_fmtext_append_normal(&fm, "\n"); for (i = 0; i < l->len; i++) { kmname = l->pdata[i]; km = owl_keyhandler_get_keymap(kh, kmname); owl_fmtext_append_bold(&fm, "\n\n----------------------------------------------------------------------------------------------------\n\n"); owl_keymap_get_details(km, &fm, 0); } owl_fmtext_append_normal(&fm, "\n"); owl_function_popless_fmtext(&fm); owl_ptr_array_free(l, g_free); owl_fmtext_cleanup(&fm); } CALLER_OWN char *owl_function_keymap_summary(const char *name) { const owl_keymap *km = owl_keyhandler_get_keymap(owl_global_get_keyhandler(&g), name); if (km) return owl_keymap_summary(km); else return(NULL); } /* TODO: implement for real */ void owl_function_show_keymap(const char *name) { owl_fmtext fm; const owl_keymap *km; owl_fmtext_init_null(&fm); km = owl_keyhandler_get_keymap(owl_global_get_keyhandler(&g), name); if (km) { owl_keymap_get_details(km, &fm, 1); } else { owl_fmtext_append_normal(&fm, "No such keymap...\n"); } owl_function_popless_fmtext(&fm); owl_fmtext_cleanup(&fm); } void owl_function_help_for_command(const char *cmdname) { owl_fmtext fm; owl_fmtext_init_null(&fm); owl_cmd_get_help(owl_global_get_cmddict(&g), cmdname, &fm); owl_function_popless_fmtext(&fm); owl_fmtext_cleanup(&fm); } void owl_function_set_search(const char *string) { owl_regex re; if (string && owl_regex_create_quoted(&re, string) == 0) { owl_global_set_search_re(&g, &re); owl_regex_cleanup(&re); } else { owl_global_set_search_re(&g, NULL); } } void owl_function_search_helper(int consider_current, int direction) { /* move to a message that contains the string. If direction is * OWL_DIRECTION_DOWNWARDS then search fowards, if direction is * OWL_DIRECTION_UPWARDS then search backwards. * * If consider_current is true then it will stay on the * current message if it contains the string. */ const owl_view *v; int viewsize, i, curmsg, start; owl_message *m; v=owl_global_get_current_view(&g); viewsize=owl_view_get_size(v); curmsg=owl_global_get_curmsg(&g); if (viewsize==0) { owl_function_makemsg("No messages present"); return; } if (consider_current) { start=curmsg; } else if (direction==OWL_DIRECTION_DOWNWARDS) { start=curmsg+1; } else { start=curmsg-1; } /* bounds check */ if (start>=viewsize || start<0) { owl_function_makemsg("No further matches found"); return; } for (i=start; i=0;) { m=owl_view_get_element(v, i); if (owl_message_search(m, owl_global_get_search_re(&g))) { owl_global_set_curmsg(&g, i); owl_function_calculate_topmsg(direction); owl_mainwin_redisplay(owl_global_get_mainwin(&g)); if (direction==OWL_DIRECTION_DOWNWARDS) { owl_global_set_direction_downwards(&g); } else { owl_global_set_direction_upwards(&g); } return; } if (direction==OWL_DIRECTION_DOWNWARDS) { i++; } else { i--; } if (owl_global_take_interrupt(&g)) { owl_function_makemsg("Search interrupted!"); owl_mainwin_redisplay(owl_global_get_mainwin(&g)); return; } } owl_mainwin_redisplay(owl_global_get_mainwin(&g)); owl_function_makemsg("No matches found"); } /* strips formatting from ztext and returns the unformatted text. * caller is responsible for freeing. */ CALLER_OWN char *owl_function_ztext_stylestrip(const char *zt) { owl_fmtext fm; char *plaintext; owl_fmtext_init_null(&fm); owl_fmtext_append_ztext(&fm, zt); plaintext = owl_fmtext_print_plain(&fm); owl_fmtext_cleanup(&fm); return(plaintext); } /* Popup a buddylisting. If filename is NULL use the default .anyone */ void owl_function_buddylist(int aim, int zephyr, const char *filename) { int i, j, idle; int interrupted = 0; owl_fmtext fm; const owl_buddylist *bl; const owl_buddy *b; char *timestr; #ifdef HAVE_LIBZEPHYR int x; GPtrArray *anyone; const char *user; char *tmp; ZLocations_t location[200]; int numlocs, ret; #endif owl_fmtext_init_null(&fm); /* AIM first */ if (aim && owl_global_is_aimloggedin(&g)) { bl=owl_global_get_buddylist(&g); owl_fmtext_append_bold(&fm, "AIM users logged in:\n"); /* we're assuming AIM for now */ j=owl_buddylist_get_size(bl); for (i=0; i"); owl_fmtext_append_normal(&fm, "'.\n"); } else { owl_fmtext_append_normal(&fm, " Could not read zephyr buddies from the .anyone file.\n"); } } else { for (i = 0; i < anyone->len; i++) { user = anyone->pdata[i]; ret=ZLocateUser(zstr(user), &numlocs, ZAUTH); if (owl_global_take_interrupt(&g)) { interrupted = 1; owl_function_makemsg("Interrupted!"); break; } if (ret!=ZERR_NONE) { owl_function_error("Error getting location for %s", user); continue; } numlocs=200; ret=ZGetLocations(location, &numlocs); if (ret==0) { for (x=0; x=200) { owl_fmtext_append_normal(&fm, " Too many locations found for this user, truncating.\n"); } } } } owl_ptr_array_free(anyone, g_free); } } #endif if (aim && zephyr) { if (owl_perlconfig_is_function("BarnOwl::Hooks::_get_blist")) { char * perlblist = owl_perlconfig_execute("BarnOwl::Hooks::_get_blist()"); if (perlblist) { owl_fmtext_append_ztext(&fm, perlblist); g_free(perlblist); } } } if(!interrupted) { owl_function_popless_fmtext(&fm); } owl_fmtext_cleanup(&fm); } /* Dump messages in the current view to the file 'filename'. */ void owl_function_dump(const char *filename) { int i, j; owl_message *m; const owl_view *v; FILE *file; char *plaintext; v=owl_global_get_current_view(&g); /* in the future make it ask yes/no */ /* ret=stat(filename, &sbuf); if (!ret) { ret=owl_function_askyesno("File exists, continue? [Y/n]"); if (!ret) return; } */ file=fopen(filename, "w"); if (!file) { owl_function_error("Error opening file"); return; } j=owl_view_get_size(v); for (i=0; i 0) { /* Spawn the child. */ GPid pid; GError *error = NULL; owl_function_debugmsg("About to exec \"%s\" with %d arguments", argv[0], myargc); if (g_spawn_async(NULL, argv, NULL, G_SPAWN_SEARCH_PATH | G_SPAWN_DO_NOT_REAP_CHILD, NULL, NULL, &pid, &error)) { owl_global_set_newmsgproc_pid(&g, pid); owl_function_debugmsg("I'm the parent and I started a new newmsgproc with pid %i", pid); } else { owl_function_debugmsg("Cannot run newmsgproc '%s': %s", owl_global_get_newmsgproc(&g), error->message); g_error_free(error); } } g_strfreev(argv); } } } /* print the xterm escape sequence to raise the window */ void owl_function_xterm_raise(void) { printf("\033[5t"); } /* print the xterm escape sequence to deiconify the window */ void owl_function_xterm_deiconify(void) { printf("\033[1t"); } /* Add the specified command to the startup file. Eventually this * should be clever, and rewriting settings that will obviosly * override earlier settings with 'set' 'bindkey' and 'alias' * commands. For now though we just remove any line that would * duplicate this one and then append this line to the end of * startupfile. */ void owl_function_addstartup(const char *buff) { FILE *file; const char *filename; filename=owl_global_get_startupfile(&g); /* delete earlier copies, if the file exists */ if (g_file_test(filename, G_FILE_TEST_EXISTS)) owl_util_file_deleteline(filename, buff, 1); file=fopen(filename, "a"); if (!file) { owl_function_error("Error opening startupfile for new command"); return; } /* add this line */ fprintf(file, "%s\n", buff); fclose(file); } /* Remove the specified command from the startup file. */ void owl_function_delstartup(const char *buff) { const char *filename; filename=owl_global_get_startupfile(&g); owl_util_file_deleteline(filename, buff, 1); } /* Execute owl commands from the given filename. If the filename * is NULL, use the default owl startup commands file. */ void owl_function_source(const char *filename) { char *path; FILE *file; char *s = NULL; int fail_silent = 0; if (!filename) { fail_silent = 1; path = g_strdup(owl_global_get_startupfile(&g)); } else { path = owl_util_makepath(filename); } file = fopen(path, "r"); g_free(path); if (!file) { if (!fail_silent) { owl_function_error("Error opening file: %s", filename); } return; } while (owl_getline_chomp(&s, file)) { if (s[0] == '\0' || s[0] == '#') continue; owl_function_command_norv(s); } g_free(s); fclose(file); } void owl_function_change_style(owl_view *v, const char *stylename) { const owl_style *s; s=owl_global_get_style_by_name(&g, stylename); if (!s) { owl_function_error("No style named %s", stylename); return; } owl_view_set_style(v, s); owl_messagelist_invalidate_formats(owl_global_get_msglist(&g)); owl_function_calculate_topmsg(OWL_DIRECTION_DOWNWARDS); owl_mainwin_redisplay(owl_global_get_mainwin(&g)); } void owl_function_toggleoneline(void) { owl_view *v; const owl_style *s; v=owl_global_get_current_view(&g); s=owl_view_get_style(v); if (!owl_style_matches_name(s, "oneline")) { owl_function_change_style(v, "oneline"); } else { owl_function_change_style(v, owl_global_get_default_style(&g)); } owl_messagelist_invalidate_formats(owl_global_get_msglist(&g)); owl_function_calculate_topmsg(OWL_DIRECTION_DOWNWARDS); owl_mainwin_redisplay(owl_global_get_mainwin(&g)); } void G_GNUC_PRINTF(1, 2) owl_function_error(const char *fmt, ...) { static int in_error = 0; va_list ap; char *buff; const char *nl; if (++in_error > 2) { /* More than two nested errors, bail immediately. */ in_error--; return; } va_start(ap, fmt); buff = g_strdup_vprintf(fmt, ap); va_end(ap); owl_function_debugmsg("ERROR: %s", buff); owl_function_log_err(buff); nl = strchr(buff, '\n'); /* Showing admin messages triggers a lot of code. If we have a recursive error call, that's the most likely candidate, so suppress the call in that case, to try to avoid infinite looping. */ if(nl && *(nl + 1) && in_error == 1) { /* Multiline error */ owl_function_adminmsg("ERROR", buff); } else { owl_function_makemsg("[Error] %s", buff); } g_free(buff); in_error--; } void owl_function_log_err(const char *string) { char *date; time_t now; char *buff; now = time(NULL); date = owl_util_format_time(localtime(&now)); buff = g_strdup_printf("%s %s", date, string); owl_errqueue_append_err(owl_global_get_errqueue(&g), buff); g_free(buff); g_free(date); } void owl_function_showerrs(void) { owl_fmtext fm; owl_fmtext_init_null(&fm); owl_fmtext_append_normal(&fm, "Errors:\n\n"); owl_errqueue_to_fmtext(owl_global_get_errqueue(&g), &fm); owl_function_popless_fmtext(&fm); } void G_GNUC_PRINTF(1, 2) owl_function_makemsg(const char *fmt, ...) { va_list ap; char *str; va_start(ap, fmt); str = g_strdup_vprintf(fmt, ap); va_end(ap); owl_function_debugmsg("makemsg: %s", str); owl_msgwin_set_text_nocopy(&g.msgwin, str); } /* get locations for everyone in .anyone. If 'notify' is '1' then * send a pseudo login or logout message for everyone not in sync with * the global zephyr buddy list. The list is updated regardless of * the status of 'notify'. */ void owl_function_zephyr_buddy_check(int notify) { #ifdef HAVE_LIBZEPHYR int i; GPtrArray *anyone; GList **zaldlist; GList *zaldptr; ZAsyncLocateData_t *zald; const char *user; if (!owl_global_is_havezephyr(&g)) return; owl_global_set_pseudologin_notify(&g, notify); zaldlist = owl_global_get_zaldlist(&g); /* Clear the existing ZALDs first. */ zaldptr = g_list_first(*zaldlist); while (zaldptr) { ZFreeALD(zaldptr->data); g_free(zaldptr->data); zaldptr = g_list_next(zaldptr); } g_list_free(*zaldlist); *zaldlist = NULL; anyone = owl_zephyr_get_anyone_list(NULL); if (anyone != NULL) { for (i = 0; i < anyone->len; i++) { user = anyone->pdata[i]; zald = g_new(ZAsyncLocateData_t, 1); if (ZRequestLocations(zstr(user), zald, UNACKED, ZAUTH) == ZERR_NONE) { *zaldlist = g_list_append(*zaldlist, zald); } else { g_free(zald); } } owl_ptr_array_free(anyone, g_free); } #endif } void owl_function_aimsearch_results(const char *email, GPtrArray *namelist) { owl_fmtext fm; int i; owl_fmtext_init_null(&fm); owl_fmtext_append_normal(&fm, "AIM screennames associated with "); owl_fmtext_append_normal(&fm, email); owl_fmtext_append_normal(&fm, ":\n"); for (i = 0; i < namelist->len; i++) { owl_fmtext_append_normal(&fm, " "); owl_fmtext_append_normal(&fm, namelist->pdata[i]); owl_fmtext_append_normal(&fm, "\n"); } owl_function_popless_fmtext(&fm); owl_fmtext_cleanup(&fm); } int owl_function_get_color_count(void) { return COLORS; } void _owl_function_mark_message(const owl_message *m) { if (m) { owl_global_set_markedmsgid(&g, owl_message_get_id(m)); owl_mainwin_redisplay(owl_global_get_mainwin(&g)); } } void owl_function_mark_message(void) { const owl_message *m; const owl_view *v; v=owl_global_get_current_view(&g); /* bail if there's no current message */ if (owl_view_get_size(v) < 1) { owl_function_error("No messages to mark"); return; } /* mark the message */ m=owl_view_get_element(v, owl_global_get_curmsg(&g)); _owl_function_mark_message(m); owl_function_makemsg("Mark set"); } void owl_function_swap_cur_marked(void) { int marked_id; const owl_message *m; const owl_view *v; marked_id=owl_global_get_markedmsgid(&g); if (marked_id == -1) { owl_function_error("Mark not set."); return; } v=owl_global_get_current_view(&g); /* bail if there's no current message */ if (owl_view_get_size(v) < 1) { return; } m=owl_view_get_element(v, owl_global_get_curmsg(&g)); _owl_function_mark_message(m); owl_global_set_curmsg(&g, owl_view_get_nearest_to_msgid(v, marked_id)); owl_function_calculate_topmsg(OWL_DIRECTION_NONE); owl_mainwin_redisplay(owl_global_get_mainwin(&g)); owl_global_set_direction_downwards(&g); }