source: functions.c @ 0e5afa2

release-1.10release-1.4release-1.5release-1.6release-1.7release-1.8release-1.9
Last change on this file since 0e5afa2 was 0e5afa2, checked in by Anders Kaseorg <andersk@mit.edu>, 15 years ago
Use pid_t type for pids. Signed-off-by: Anders Kaseorg <andersk@mit.edu>
  • Property mode set to 100644
File size: 97.9 KB
RevLine 
[7d4fbcd]1#include <stdio.h>
2#include <stdlib.h>
3#include <unistd.h>
4#include <signal.h>
[789462a]5#include <netinet/in.h>
[7d4fbcd]6#include <string.h>
7#include <time.h>
[2adaf1d]8#include <sys/types.h>
9#include <sys/stat.h>
[8f44c6b]10#include <sys/wait.h>
11#include <errno.h>
[d09e5a1]12#include <signal.h>
[7d4fbcd]13#include "owl.h"
14
[e19eb97]15char *owl_function_command(const char *cmdbuff)
[d54838d]16{
[7d4fbcd]17  owl_function_debugmsg("executing command: %s", cmdbuff);
18  return owl_cmddict_execute(owl_global_get_cmddict(&g), 
19                             owl_global_get_context(&g), cmdbuff);
20}
21
[e19eb97]22char *owl_function_command_argv(const char *const *argv, int argc)
[c4ba74d]23{
24  return owl_cmddict_execute_argv(owl_global_get_cmddict(&g),
25                                  owl_global_get_context(&g),
26                                  argv, argc);
27}
28
[e19eb97]29void owl_function_command_norv(const char *cmdbuff)
[d54838d]30{
[7d4fbcd]31  char *rv;
[4b464a4]32  rv=owl_function_command(cmdbuff);
[7d4fbcd]33  if (rv) owl_free(rv);
34}
35
[e19eb97]36void owl_function_command_alias(const char *alias_from, const char *alias_to)
[d54838d]37{
[7d4fbcd]38  owl_cmddict_add_alias(owl_global_get_cmddict(&g), alias_from, alias_to);
39}
40
[0a0fb74]41const owl_cmd *owl_function_get_cmd(const char *name)
[d54838d]42{
[7d4fbcd]43  return owl_cmddict_find(owl_global_get_cmddict(&g), name);
44}
45
[d54838d]46void owl_function_show_commands()
47{
[7d4fbcd]48  owl_list l;
49  owl_fmtext fm;
50
51  owl_fmtext_init_null(&fm);
52  owl_fmtext_append_bold(&fm, "Commands:  ");
53  owl_fmtext_append_normal(&fm, "(use 'show command <name>' for details)\n");
54  owl_cmddict_get_names(owl_global_get_cmddict(&g), &l);
55  owl_fmtext_append_list(&fm, &l, "\n", owl_function_cmd_describe);
56  owl_fmtext_append_normal(&fm, "\n");
57  owl_function_popless_fmtext(&fm);
58  owl_cmddict_namelist_free(&l);
59  owl_fmtext_free(&fm);
60}
61
[e19eb97]62void owl_function_show_view(const char *viewname)
[ef56a67]63{
[9e5c9f3]64  const owl_view *v;
[ef56a67]65  owl_fmtext fm;
66
67  /* we only have the one view right now */
68  v=owl_global_get_current_view(&g);
69  if (viewname && strcmp(viewname, owl_view_get_name(v))) {
[ec6ff52]70    owl_function_error("No view named '%s'", viewname);
[ef56a67]71    return;
72  }
73
74  owl_fmtext_init_null(&fm);
75  owl_view_to_fmtext(v, &fm);
76  owl_function_popless_fmtext(&fm);
77  owl_fmtext_free(&fm);
78}
79
[f1e629d]80void owl_function_show_styles() {
81  owl_list l;
82  owl_fmtext fm;
83
84  owl_fmtext_init_null(&fm);
85  owl_fmtext_append_bold(&fm, "Styles:\n");
86  owl_global_get_style_names(&g, &l);
87  owl_fmtext_append_list(&fm, &l, "\n", owl_function_style_describe);
88  owl_fmtext_append_normal(&fm, "\n");
89  owl_function_popless_fmtext(&fm);
90  owl_list_free_all(&l, owl_free);
91  owl_fmtext_free(&fm);
92}
93
[e19eb97]94char *owl_function_style_describe(const char *name) {
95  const char *desc;
[65b2173]96  char *s;
[1fdab04]97  const owl_style *style;
[f1e629d]98  style = owl_global_get_style_by_name(&g, name);
99  if (style) {
100    desc = owl_style_get_description(style);
101  } else {
102    desc = "???";
103  }
104  s = owl_sprintf("%-20s - %s%s", name, 
105                  0==owl_style_validate(style)?"":"[INVALID] ",
106                  desc);
107  return s;
108}
[ef56a67]109
[e19eb97]110char *owl_function_cmd_describe(const char *name)
[d54838d]111{
[0a0fb74]112  const owl_cmd *cmd = owl_cmddict_find(owl_global_get_cmddict(&g), name);
[7d4fbcd]113  if (cmd) return owl_cmd_describe(cmd);
114  else return(NULL);
115}
116
[e19eb97]117void owl_function_show_command(const char *name)
[d54838d]118{
[7d4fbcd]119  owl_function_help_for_command(name);
120}
121
[debb15d]122void owl_function_show_license()
123{
[e19eb97]124  const char *text;
[debb15d]125
126  text=""
[4228f8b]127    "barnowl version " OWL_VERSION_STRING "\n"
[23e1110]128    "Copyright (c) 2006-2009 The BarnOwl Developers. All rights reserved.\n"
[debb15d]129    "Copyright (c) 2004 James Kretchmar. All rights reserved.\n"
130    "\n"
131    "Redistribution and use in source and binary forms, with or without\n"
132    "modification, are permitted provided that the following conditions are\n"
133    "met:\n"
134    "\n"
135    "   * Redistributions of source code must retain the above copyright\n"
136    "     notice, this list of conditions and the following disclaimer.\n"
137    "\n"
138    "   * Redistributions in binary form must reproduce the above copyright\n"
139    "     notice, this list of conditions and the following disclaimer in\n"
140    "     the documentation and/or other materials provided with the\n"
141    "     distribution.\n"
142    "\n"
143    "   * Redistributions in any form must be accompanied by information on\n"
144    "     how to obtain complete source code for the Owl software and any\n"
145    "     accompanying software that uses the Owl software. The source code\n"
146    "     must either be included in the distribution or be available for no\n"
147    "     more than the cost of distribution plus a nominal fee, and must be\n"
148    "     freely redistributable under reasonable conditions. For an\n"
149    "     executable file, complete source code means the source code for\n"
150    "     all modules it contains. It does not include source code for\n"
151    "     modules or files that typically accompany the major components of\n"
152    "     the operating system on which the executable file runs.\n"
153    "\n"
154    "THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR\n"
155    "IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED\n"
156    "WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, OR\n"
157    "NON-INFRINGEMENT, ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE\n"
158    "LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR\n"
159    "CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF\n"
160    "SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR\n"
161    "BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,\n"
162    "WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE\n"
163    "OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN\n"
164    "IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.\n";
165  owl_function_popless_text(text);
166}
167
[799b60e]168void owl_function_show_quickstart()
169{
[e19eb97]170    const char *message =
[799b60e]171    "Move between messages with the arrow keys, and press 'r' to reply.\n"
172    "For more info, press 'h' or visit http://barnowl.mit.edu/\n\n"
173#ifdef HAVE_LIBZEPHYR
174    "@b(Zephyr:)\n"
175    "To send a message to a user, type ':zwrite @b(username)'. You can also\n"
176    "press 'z' and then type the username. To subscribe to a class, type\n"
177    "':sub @b(class)', and then type ':zwrite -c @b(class)' to send.\n\n"
178#endif
179    "@b(AIM:)\n"
180    "Log in to AIM with ':aimlogin @b(screenname)'. Use ':aimwrite @b(screenname)',\n"
181    "or 'a' and then the screen name, to send someone a message.\n\n"
182    ;
183
184    if (owl_perlconfig_is_function("BarnOwl::Hooks::_get_quickstart")) {
185        char *perlquickstart = owl_perlconfig_execute("BarnOwl::Hooks::_get_quickstart()");
186        if (perlquickstart) {
187            char *result = owl_sprintf("%s%s", message, perlquickstart);
188            owl_function_adminmsg("BarnOwl Quickstart", result);
189            owl_free(result);
190            owl_free(perlquickstart);
191            return;
192        }
193    }
194    owl_function_adminmsg("BarnOwl Quickstart", message);
195}
196
197
[15b34fd]198/* Create an admin message, append it to the global list of messages
199 * and redisplay if necessary.
200 */
[e19eb97]201void owl_function_adminmsg(const char *header, const char *body)
[d54838d]202{
[7d4fbcd]203  owl_message *m;
[4b464a4]204
[7d4fbcd]205  m=owl_malloc(sizeof(owl_message));
[15b34fd]206  owl_message_create_admin(m, header, body);
207 
[8fec514]208  /* add it to the global list and current view */
[7d4fbcd]209  owl_messagelist_append_element(owl_global_get_msglist(&g), m);
210  owl_view_consider_message(owl_global_get_current_view(&g), m);
211
[15b34fd]212  /* do followlast if necessary */
213  if (owl_global_should_followlast(&g)) owl_function_lastmsg_noredisplay();
[7d4fbcd]214
[15b34fd]215  /* redisplay etc. */
[7d4fbcd]216  owl_mainwin_redisplay(owl_global_get_mainwin(&g));
217  if (owl_popwin_is_active(owl_global_get_popwin(&g))) {
218    owl_popwin_refresh(owl_global_get_popwin(&g));
219  }
220  wnoutrefresh(owl_global_get_curs_recwin(&g));
221  owl_global_set_needrefresh(&g);
222}
223
[15b34fd]224/* Create an outgoing zephyr message and return a pointer to it.  Does
[13fe062]225 * not put it on the global queue, use owl_global_messagequeue_addmsg() for
[15b34fd]226 * that.
227 */
[e19eb97]228owl_message *owl_function_make_outgoing_zephyr(const char *body, const char *zwriteline, const char *zsig)
[d09e5a1]229{
230  owl_message *m;
231
232  /* create the message */
233  m=owl_malloc(sizeof(owl_message));
[15b34fd]234  owl_message_create_from_zwriteline(m, zwriteline, body, zsig);
235
236  return(m);
237}
238
239/* Create an outgoing AIM message, returns a pointer to the created
240 * message or NULL if we're not logged into AIM (and thus unable to
241 * create the message).  Does not put it on the global queue.  Use
[13fe062]242 * owl_global_messagequeue_addmsg() for that .
[15b34fd]243 */
[e19eb97]244owl_message *owl_function_make_outgoing_aim(const char *body, const char *to)
[15b34fd]245{
246  owl_message *m;
247
248  /* error if we're not logged into aim */
249  if (!owl_global_is_aimloggedin(&g)) return(NULL);
250 
251  m=owl_malloc(sizeof(owl_message));
[d559df9]252  owl_message_create_aim(m,
253                         owl_global_get_aim_screenname(&g),
254                         to,
255                         body,
256                         OWL_MESSAGE_DIRECTION_OUT,
257                         0);
[15b34fd]258  return(m);
[d09e5a1]259}
260
[15b34fd]261/* Create an outgoing loopback message and return a pointer to it.
262 * Does not append it to the global queue, use
[13fe062]263 * owl_global_messagequeue_addmsg() for that.
[15b34fd]264 */
[e19eb97]265owl_message *owl_function_make_outgoing_loopback(const char *body)
[37eab7f]266{
267  owl_message *m;
268
269  /* create the message */
270  m=owl_malloc(sizeof(owl_message));
271  owl_message_create_loopback(m, body);
272  owl_message_set_direction_out(m);
273
[15b34fd]274  return(m);
[37eab7f]275}
276
[e19eb97]277void owl_function_zwrite_setup(const char *line)
[d54838d]278{
[7d4fbcd]279  owl_editwin *e;
280  char buff[1024];
281  owl_zwrite z;
282  int ret;
283
284  /* check the arguments */
285  ret=owl_zwrite_create_from_line(&z, line);
286  if (ret) {
[d768834]287    owl_function_error("Error in zwrite arguments");
[7d4fbcd]288    owl_zwrite_free(&z);
289    return;
290  }
291
292  /* send a ping if necessary */
293  if (owl_global_is_txping(&g)) {
294    owl_zwrite_send_ping(&z);
295  }
296  owl_zwrite_free(&z);
297
298  /* create and setup the editwin */
299  e=owl_global_get_typwin(&g);
[56330ff]300  owl_editwin_new_style(e, OWL_EDITWIN_STYLE_MULTILINE, owl_global_get_msg_history(&g));
[7d4fbcd]301
[217a43e]302  if (!owl_global_get_lockout_ctrld(&g)) {
[7d4fbcd]303    owl_function_makemsg("Type your zephyr below.  End with ^D or a dot on a line by itself.  ^C will quit.");
304  } else {
305    owl_function_makemsg("Type your zephyr below.  End with a dot on a line by itself.  ^C will quit.");
306  }
307
308  owl_editwin_clear(e);
309  owl_editwin_set_dotsend(e);
310  strcpy(buff, "----> ");
[d81f8d0]311  strncat(buff, line, 1016);
[7d4fbcd]312  strcat(buff, "\n");
313  owl_editwin_set_locktext(e, buff);
314
315  /* make it active */
316  owl_global_set_typwin_active(&g);
[453bd70]317
318  owl_global_set_buffercommand(&g, line);
[1b6b2f3]319  owl_global_set_buffercallback(&g, &owl_callback_zwrite);
[7d4fbcd]320}
321
[e19eb97]322void owl_function_aimwrite_setup(const char *line)
[d09e5a1]323{
324  owl_editwin *e;
325  char buff[1024];
326
327  /* check the arguments */
328
329  /* create and setup the editwin */
330  e=owl_global_get_typwin(&g);
331  owl_editwin_new_style(e, OWL_EDITWIN_STYLE_MULTILINE, owl_global_get_msg_history(&g));
332
333  if (!owl_global_get_lockout_ctrld(&g)) {
334    owl_function_makemsg("Type your message below.  End with ^D or a dot on a line by itself.  ^C will quit.");
335  } else {
336    owl_function_makemsg("Type your message below.  End with a dot on a line by itself.  ^C will quit.");
337  }
338
339  owl_editwin_clear(e);
340  owl_editwin_set_dotsend(e);
341  strcpy(buff, "----> ");
342  strcat(buff, line);
343  strcat(buff, "\n");
344  owl_editwin_set_locktext(e, buff);
345
346  /* make it active */
347  owl_global_set_typwin_active(&g);
[453bd70]348
349  owl_global_set_buffercommand(&g, line);
[1b6b2f3]350  owl_global_set_buffercallback(&g, &owl_callback_aimwrite);
[d09e5a1]351}
352
[37eab7f]353void owl_function_loopwrite_setup()
354{
355  owl_editwin *e;
356
357  /* create and setup the editwin */
358  e=owl_global_get_typwin(&g);
359  owl_editwin_new_style(e, OWL_EDITWIN_STYLE_MULTILINE, owl_global_get_msg_history(&g));
360
361  if (!owl_global_get_lockout_ctrld(&g)) {
362    owl_function_makemsg("Type your message below.  End with ^D or a dot on a line by itself.  ^C will quit.");
363  } else {
364    owl_function_makemsg("Type your message below.  End with a dot on a line by itself.  ^C will quit.");
365  }
366
367  owl_editwin_clear(e);
368  owl_editwin_set_dotsend(e);
369  owl_editwin_set_locktext(e, "----> loopwrite\n");
370
371  /* make it active */
372  owl_global_set_typwin_active(&g);
373
374  owl_global_set_buffercommand(&g, "loopwrite");
[1b6b2f3]375  owl_global_set_buffercallback(&g, &owl_callback_loopwrite);
376}
377
378void owl_callback_zwrite(owl_editwin *e) {
379  owl_function_zwrite(owl_editwin_get_command(e),
380                      owl_editwin_get_text(e));
[37eab7f]381}
382
[9ceee9d]383/* send, log and display an outgoing zephyr.  If 'msg' is NULL
384 * the message is expected to be set from the zwrite line itself
385 */
[e19eb97]386void owl_function_zwrite(const char *line, const char *msg)
[d54838d]387{
[d309eb3]388  owl_zwrite z;
[e19eb97]389  const char *mymsg;
[15b34fd]390  owl_message *m;
[d309eb3]391
[a5fc448]392  if(!strncmp(line, "zcrypt", strlen("zcrypt"))) {
393    owl_function_zcrypt(line, msg);
394    return;
395  }
396
[9ceee9d]397  /* create the zwrite and send the message */
398  owl_zwrite_create_from_line(&z, line);
[3f3ee61]399  owl_zwrite_populate_zsig(&z);
[9ceee9d]400  if (msg) {
401    owl_zwrite_set_message(&z, msg);
[d309eb3]402  }
[9ceee9d]403  owl_zwrite_send_message(&z);
404  owl_function_makemsg("Waiting for ack...");
[d309eb3]405
[15b34fd]406  /* If it's personal */
407  if (owl_zwrite_is_personal(&z)) {
408    /* create the outgoing message */
409    mymsg=owl_zwrite_get_message(&z);
410    m=owl_function_make_outgoing_zephyr(mymsg, line, owl_zwrite_get_zsig(&z));
[d309eb3]411
[3c7d086a]412    if (m) {
[13a3c1db]413      owl_global_messagequeue_addmsg(&g, m);
[15b34fd]414    } else {
[3c7d086a]415      owl_function_error("Could not create outgoing zephyr message");
[9ceee9d]416    }
417  }
[453bd70]418
[9ceee9d]419  /* free the zwrite */
420  owl_zwrite_free(&z);
[d309eb3]421}
422
[9ceee9d]423/* send, log and display an outgoing zcrypt zephyr.  If 'msg' is NULL
[ce7db4d]424 * the message is expected to be set from the zwrite line itself
425 */
[e19eb97]426void owl_function_zcrypt(const char *line, const char *msg)
[d54838d]427{
[7d4fbcd]428  owl_zwrite z;
[e19eb97]429  const char *mymsg;
[9ceee9d]430  char *cryptmsg;
[15b34fd]431  owl_message *m;
[8a54409]432#ifdef OWL_ENABLE_ZCRYPT
433  int ret;
434#endif
[7d4fbcd]435
436  /* create the zwrite and send the message */
437  owl_zwrite_create_from_line(&z, line);
[3f3ee61]438  owl_zwrite_populate_zsig(&z);
[ce7db4d]439  if (msg) {
440    owl_zwrite_set_message(&z, msg);
441  }
442
[9ceee9d]443  mymsg=owl_zwrite_get_message(&z);
444#ifdef OWL_ENABLE_ZCRYPT
[559076c]445  /* Allocate enough space for the crypted message. For each byte of
446   * the message, the encoded cyphertext will have two bytes. Block
447   * size is 8 bytes of input, or 16 bytes of output, so make sure we
448   * have at least one block worth of space allocated. If the message
449   * is empty, no blocks are sent, but we still allocate one
450   * block. The additional 16 bytes also provide space for the null
451   * terminator, as we will never use all of it for cyphertext.
452   */
453  cryptmsg=owl_malloc((strlen(mymsg)*2)+16);
[9ceee9d]454  ret=owl_zcrypt_encrypt(cryptmsg, mymsg, owl_zwrite_get_class(&z), owl_zwrite_get_instance(&z));
455  if (ret) {
[ec6ff52]456    owl_function_error("Error in zcrypt, possibly no key found.  Message not sent.");
[9ceee9d]457    owl_function_beep();
458    owl_free(cryptmsg);
[559076c]459    owl_zwrite_free(&z);
[9ceee9d]460    return;
461  }
462#else
463  cryptmsg=owl_strdup(mymsg);
464#endif
465
466  owl_zwrite_set_message(&z, cryptmsg);
467  owl_zwrite_set_opcode(&z, "crypt");
468  mymsg=cryptmsg;
469   
[ce7db4d]470  owl_zwrite_send_message(&z);
[7d4fbcd]471  owl_function_makemsg("Waiting for ack...");
472
[15b34fd]473  /* If it's personal */
474  if (owl_zwrite_is_personal(&z)) {
475    /* create the outgoing message */
476    mymsg=owl_zwrite_get_message(&z);
477    m=owl_function_make_outgoing_zephyr(mymsg, line, owl_zwrite_get_zsig(&z));
[3c7d086a]478    if (m) {
[13a3c1db]479      owl_global_messagequeue_addmsg(&g, m);
[15b34fd]480    } else {
[3c7d086a]481      owl_function_error("Could not create outgoing zephyr message");
[7d4fbcd]482    }
483  }
484
485  /* free the zwrite */
[9ceee9d]486  owl_free(cryptmsg);
[7d4fbcd]487  owl_zwrite_free(&z);
488}
489
[1b6b2f3]490void owl_callback_aimwrite(owl_editwin *e) {
491  owl_function_aimwrite(owl_editwin_get_command(e),
492                        owl_editwin_get_text(e));
493}
494
[e19eb97]495void owl_function_aimwrite(const char *line, const char *msg)
[d09e5a1]496{
[ec6ff52]497  int ret;
[e19eb97]498  const char *to;
[65b2173]499  char *format_msg;
[15b34fd]500  owl_message *m;
[f82e233]501
[4211f50b]502  to = line + 9;
503
[f82e233]504  /* make a formatted copy of the message */
505  format_msg=owl_strdup(msg);
506  owl_text_wordunwrap(format_msg);
[ec6ff52]507 
[f82e233]508  /* send the message */
509  ret=owl_aim_send_im(to, format_msg);
[ec6ff52]510  if (!ret) {
511    owl_function_makemsg("AIM message sent.");
512  } else {
513    owl_function_error("Could not send AIM message.");
514  }
[d09e5a1]515
[15b34fd]516  /* create the outgoing message */
517  m=owl_function_make_outgoing_aim(msg, to);
518
[3c7d086a]519  if (m) {
[13a3c1db]520    owl_global_messagequeue_addmsg(&g, m);
[15b34fd]521  } else {
[3c7d086a]522    owl_function_error("Could not create outgoing AIM message");
[d09e5a1]523  }
[f82e233]524
525  owl_free(format_msg);
[d09e5a1]526}
527
[e19eb97]528void owl_function_send_aimawymsg(const char *to, const char *msg)
[9854278]529{
530  int ret;
531  char *format_msg;
[15b34fd]532  owl_message *m;
[9854278]533
534  /* make a formatted copy of the message */
535  format_msg=owl_strdup(msg);
536  owl_text_wordunwrap(format_msg);
537 
538  /* send the message */
539  ret=owl_aim_send_awaymsg(to, format_msg);
540  if (!ret) {
541    /* owl_function_makemsg("AIM message sent."); */
542  } else {
543    owl_function_error("Could not send AIM message.");
544  }
545
[15b34fd]546  /* create the message */
547  m=owl_function_make_outgoing_aim(msg, to);
548  if (m) {
[13a3c1db]549    owl_global_messagequeue_addmsg(&g, m);
[15b34fd]550  } else {
551    owl_function_error("Could not create AIM message");
[9854278]552  }
553  owl_free(format_msg);
554}
555
[1b6b2f3]556void owl_callback_loopwrite(owl_editwin *e) {
557  owl_function_loopwrite(owl_editwin_get_text(e));
558}
559
[e19eb97]560void owl_function_loopwrite(const char *msg)
[37eab7f]561{
[15b34fd]562  owl_message *min, *mout;
[37eab7f]563
564  /* create a message and put it on the message queue.  This simulates
565   * an incoming message */
[15b34fd]566  min=owl_malloc(sizeof(owl_message));
[4211f50b]567  mout=owl_function_make_outgoing_loopback(msg);
[13a3c1db]568
[37eab7f]569  if (owl_global_is_displayoutgoing(&g)) {
[13a3c1db]570    owl_global_messagequeue_addmsg(&g, mout);
[15b34fd]571  } else {
572    owl_message_free(mout);
[37eab7f]573  }
574
[13a3c1db]575  owl_message_create_loopback(min, msg);
576  owl_message_set_direction_in(min);
577  owl_global_messagequeue_addmsg(&g, min);
578
[37eab7f]579  /* fake a makemsg */
580  owl_function_makemsg("loopback message sent");
581}
582
[b950088]583/* If filter is non-null, looks for the next message matching
584 * that filter.  If skip_deleted, skips any deleted messages.
585 * If last_if_none, will stop at the last message in the view
586 * if no matching messages are found.  */
[e19eb97]587void owl_function_nextmsg_full(const char *filter, int skip_deleted, int last_if_none)
[d54838d]588{
[b950088]589  int curmsg, i, viewsize, found;
[9e5c9f3]590  const owl_view *v;
[4542047]591  const owl_filter *f = NULL;
[c08c70a]592  const owl_message *m;
[7d4fbcd]593
594  v=owl_global_get_current_view(&g);
[b950088]595
596  if (filter) {
597    f=owl_global_get_filter(&g, filter);
598    if (!f) {
[ec6ff52]599      owl_function_error("No %s filter defined", filter);
[b950088]600      return;
601    }
[7d4fbcd]602  }
603
[b950088]604  curmsg=owl_global_get_curmsg(&g);
605  viewsize=owl_view_get_size(v);
606  found=0;
[7d4fbcd]607
[b950088]608  /* just check to make sure we're in bounds... */
609  if (curmsg>viewsize-1) curmsg=viewsize-1;
610  if (curmsg<0) curmsg=0;
[7d4fbcd]611
[b950088]612  for (i=curmsg+1; i<viewsize; i++) {
613    m=owl_view_get_element(v, i);
614    if (skip_deleted && owl_message_is_delete(m)) continue;
615    if (f && !owl_filter_message_match(f, m)) continue;
616    found = 1;
617    break;
618  }
619
620  if (i>owl_view_get_size(v)-1) i=owl_view_get_size(v)-1;
[5763474]621  if (i<0) i=0;
[b950088]622
623  if (!found) {
[799b60e]624    owl_function_makemsg("already at last%s message%s%s%s",
[b950088]625                         skip_deleted?" non-deleted":"",
[799b60e]626                         filter?" in ":"", filter?filter:"",
627                         owl_mainwin_is_curmsg_truncated(owl_global_get_mainwin(&g)) ?
628                         ", press Enter to scroll" : "");
[5a6e6b9]629    /* if (!skip_deleted) owl_function_beep(); */
[7d4fbcd]630  }
631
[b950088]632  if (last_if_none || found) {
633    owl_global_set_curmsg(&g, i);
634    owl_function_calculate_topmsg(OWL_DIRECTION_DOWNWARDS);
635    owl_mainwin_redisplay(owl_global_get_mainwin(&g));
636    owl_global_set_direction_downwards(&g);
637  }
638}
[7d4fbcd]639
[e19eb97]640void owl_function_prevmsg_full(const char *filter, int skip_deleted, int first_if_none)
[d54838d]641{
[b950088]642  int curmsg, i, viewsize, found;
[9e5c9f3]643  const owl_view *v;
[4542047]644  const owl_filter *f = NULL;
[c08c70a]645  const owl_message *m;
[7d4fbcd]646
647  v=owl_global_get_current_view(&g);
648
[b950088]649  if (filter) {
650    f=owl_global_get_filter(&g, filter);
651    if (!f) {
[ec6ff52]652      owl_function_error("No %s filter defined", filter);
[b950088]653      return;
[7d4fbcd]654    }
[b950088]655  }
[7d4fbcd]656
[b950088]657  curmsg=owl_global_get_curmsg(&g);
658  viewsize=owl_view_get_size(v);
659  found=0;
660
661  /* just check to make sure we're in bounds... */
662  if (curmsg<0) curmsg=0;
663
664  for (i=curmsg-1; i>=0; i--) {
665    m=owl_view_get_element(v, i);
666    if (skip_deleted && owl_message_is_delete(m)) continue;
667    if (f && !owl_filter_message_match(f, m)) continue;
668    found = 1;
669    break;
670  }
671
672  if (i<0) i=0;
673
674  if (!found) {
[f51bc78]675    owl_function_makemsg("already at first%s message%s%s",
[b950088]676                         skip_deleted?" non-deleted":"",
677                         filter?" in ":"", filter?filter:"");
[5a6e6b9]678    /* if (!skip_deleted) owl_function_beep(); */
[b950088]679  }
680
681  if (first_if_none || found) {
682    owl_global_set_curmsg(&g, i);
683    owl_function_calculate_topmsg(OWL_DIRECTION_UPWARDS);
684    owl_mainwin_redisplay(owl_global_get_mainwin(&g));
685    owl_global_set_direction_upwards(&g);
[7d4fbcd]686  }
687}
688
[d54838d]689void owl_function_nextmsg()
690{
[b950088]691  owl_function_nextmsg_full(NULL, 0, 1);
692}
[7d4fbcd]693
[d54838d]694void owl_function_prevmsg()
695{
[b950088]696  owl_function_prevmsg_full(NULL, 0, 1);
697}
[7d4fbcd]698
[d54838d]699void owl_function_nextmsg_notdeleted()
700{
[b950088]701  owl_function_nextmsg_full(NULL, 1, 1);
702}
[7d4fbcd]703
[d54838d]704void owl_function_prevmsg_notdeleted()
705{
[b950088]706  owl_function_prevmsg_full(NULL, 1, 1);
[7d4fbcd]707}
708
[b950088]709/* if move_after is 1, moves after the delete */
[d54838d]710void owl_function_deletecur(int move_after)
711{
[7d4fbcd]712  int curmsg;
713  owl_view *v;
714
715  v=owl_global_get_current_view(&g);
716
717  /* bail if there's no current message */
718  if (owl_view_get_size(v) < 1) {
[ec6ff52]719    owl_function_error("No current message to delete");
[7d4fbcd]720    return;
721  }
722
723  /* mark the message for deletion */
724  curmsg=owl_global_get_curmsg(&g);
725  owl_view_delete_element(v, curmsg);
726
[b950088]727  if (move_after) {
728    /* move the poiner in the appropriate direction
729     * to the next undeleted msg */
730    if (owl_global_get_direction(&g)==OWL_DIRECTION_UPWARDS) {
731      owl_function_prevmsg_notdeleted();
732    } else {
733      owl_function_nextmsg_notdeleted();
734    }
[7d4fbcd]735  }
736}
737
[d54838d]738void owl_function_undeletecur(int move_after)
739{
[7d4fbcd]740  int curmsg;
741  owl_view *v;
742
743  v=owl_global_get_current_view(&g);
744 
745  if (owl_view_get_size(v) < 1) {
[ec6ff52]746    owl_function_error("No current message to undelete");
[7d4fbcd]747    return;
748  }
749  curmsg=owl_global_get_curmsg(&g);
750
751  owl_view_undelete_element(v, curmsg);
752
[b950088]753  if (move_after) {
754    if (owl_global_get_direction(&g)==OWL_DIRECTION_UPWARDS) {
755      if (curmsg>0) {
756        owl_function_prevmsg();
757      } else {
758        owl_function_nextmsg();
759      }
[7d4fbcd]760    } else {
[b950088]761      owl_function_nextmsg();
[7d4fbcd]762    }
763  }
764
765  owl_mainwin_redisplay(owl_global_get_mainwin(&g));
766}
767
[d54838d]768void owl_function_expunge()
769{
[7d4fbcd]770  int curmsg;
[c08c70a]771  const owl_message *m;
[7d4fbcd]772  owl_messagelist *ml;
773  owl_view *v;
[486688f]774  int lastmsgid=0;
[7d4fbcd]775
776  curmsg=owl_global_get_curmsg(&g);
777  v=owl_global_get_current_view(&g);
778  ml=owl_global_get_msglist(&g);
779
780  m=owl_view_get_element(v, curmsg);
[486688f]781  if (m) lastmsgid = owl_message_get_id(m);
[7d4fbcd]782
783  /* expunge the message list */
784  owl_messagelist_expunge(ml);
785
786  /* update all views (we only have one right now) */
787  owl_view_recalculate(v);
788
[486688f]789  /* find where the new position should be
790     (as close as possible to where we last where) */
791  curmsg = owl_view_get_nearest_to_msgid(v, lastmsgid);
792  if (curmsg>owl_view_get_size(v)-1) curmsg = owl_view_get_size(v)-1;
793  if (curmsg<0) curmsg = 0;
794  owl_global_set_curmsg(&g, curmsg);
795  owl_function_calculate_topmsg(OWL_DIRECTION_NONE);
[7d4fbcd]796  /* if there are no messages set the direction to down in case we
797     delete everything upwards */
798  owl_global_set_direction_downwards(&g);
799 
800  owl_function_makemsg("Messages expunged");
801  owl_mainwin_redisplay(owl_global_get_mainwin(&g));
802}
803
[d54838d]804void owl_function_firstmsg()
805{
[7d4fbcd]806  owl_global_set_curmsg(&g, 0);
807  owl_global_set_topmsg(&g, 0);
808  owl_mainwin_redisplay(owl_global_get_mainwin(&g));
809  owl_global_set_direction_downwards(&g);
810}
811
[d54838d]812void owl_function_lastmsg_noredisplay()
813{
[5eeea3b]814  int oldcurmsg, curmsg;
[9e5c9f3]815  const owl_view *v;
[7d4fbcd]816
817  v=owl_global_get_current_view(&g);
[5eeea3b]818  oldcurmsg=owl_global_get_curmsg(&g);
819  curmsg=owl_view_get_size(v)-1; 
[7d4fbcd]820  if (curmsg<0) curmsg=0;
821  owl_global_set_curmsg(&g, curmsg);
[5eeea3b]822  if (oldcurmsg < curmsg) {
823    owl_function_calculate_topmsg(OWL_DIRECTION_DOWNWARDS);
824  } else if (curmsg<owl_view_get_size(v)) {
825    /* If already at the end, blank the screen and move curmsg
826     * past the end of the messages. */
827    owl_global_set_topmsg(&g, curmsg+1);
828    owl_global_set_curmsg(&g, curmsg+1);
829  } 
[6b580b0]830  /* owl_mainwin_redisplay(owl_global_get_mainwin(&g)); */
[7d4fbcd]831  owl_global_set_direction_downwards(&g);
832}
833
[d54838d]834void owl_function_lastmsg()
835{
[7d4fbcd]836  owl_function_lastmsg_noredisplay();
837  owl_mainwin_redisplay(owl_global_get_mainwin(&g)); 
838}
839
[d54838d]840void owl_function_shift_right()
841{
[7d4fbcd]842  owl_global_set_rightshift(&g, owl_global_get_rightshift(&g)+10);
843  owl_mainwin_redisplay(owl_global_get_mainwin(&g));
844  owl_global_set_needrefresh(&g);
845}
846
[d54838d]847void owl_function_shift_left()
848{
[7d4fbcd]849  int shift;
850
851  shift=owl_global_get_rightshift(&g);
852  if (shift>=10) {
853    owl_global_set_rightshift(&g, shift-10);
854    owl_mainwin_redisplay(owl_global_get_mainwin(&g));
855    owl_global_set_needrefresh(&g);
856  } else {
857    owl_function_beep();
[f51bc78]858    owl_function_makemsg("Already full left");
[7d4fbcd]859  }
860}
861
[d54838d]862void owl_function_unsuball()
863{
[7d4fbcd]864  unsuball();
865  owl_function_makemsg("Unsubscribed from all messages.");
866}
867
[95474d7]868
869/* Load zephyr subscriptions from the named 'file' and load zephyr's
870 * default subscriptions as well.  An error message is printed if
871 * 'file' can't be opened or if zephyr reports an error in
872 * subscribing.
873 *
874 * If 'file' is NULL, this look for the default filename
875 * $HOME/.zephyr.subs.  If the file can not be opened in this case
876 * only, no error message is printed.
877 */
[e19eb97]878void owl_function_loadsubs(const char *file)
[d54838d]879{
[4357be8]880  int ret, ret2;
[e19eb97]881  const char *foo;
[65b2173]882  char *path;
[ecd5dc5]883
[95474d7]884  if (file==NULL) {
885    ret=owl_zephyr_loadsubs(NULL, 0);
886  } else {
[27d8d83]887    path = owl_util_makepath(file);
888    ret=owl_zephyr_loadsubs(path, 1);
889    free(path);
[95474d7]890  }
[ecd5dc5]891
[4357be8]892  /* for backwards compatibility for now */
893  ret2=owl_zephyr_loaddefaultsubs();
894
[ecd5dc5]895  if (!owl_context_is_interactive(owl_global_get_context(&g))) return;
[95474d7]896
897  foo=file?file:"file";
898  if (ret==0 && ret2==0) {
899    if (!file) {
900      owl_function_makemsg("Subscribed to messages.");
901    } else {
902      owl_function_makemsg("Subscribed to messages from %s", file);
903    }
[7d4fbcd]904  } else if (ret==-1) {
[95474d7]905    owl_function_error("Could not read %s", foo);
[7d4fbcd]906  } else {
[95474d7]907    owl_function_error("Error subscribing to messages");
[7d4fbcd]908  }
909}
910
[e19eb97]911void owl_function_loadloginsubs(const char *file)
[d54838d]912{
[7933748]913  int ret;
[ecd5dc5]914
[7933748]915  ret=owl_zephyr_loadloginsubs(file);
[ecd5dc5]916
917  if (!owl_context_is_interactive(owl_global_get_context(&g))) return;
[7933748]918  if (ret==0) {
919  } else if (ret==-1) {
[ec6ff52]920    owl_function_error("Could not open file for login subscriptions.");
[7933748]921  } else {
[ec6ff52]922    owl_function_error("Error subscribing to login messages from file.");
[7933748]923  }
924}
925
[1b6b2f3]926void owl_callback_aimlogin(owl_editwin *e) {
927  owl_function_aimlogin(owl_editwin_get_command(e),
[db8b00b]928                        owl_editwin_get_text(e));
[1b6b2f3]929}
930
[e19eb97]931void owl_function_aimlogin(const char *user, const char *passwd) {
[4211f50b]932  int ret;
933
934  /* clear the buddylist */
935  owl_buddylist_clear(owl_global_get_buddylist(&g));
936
937  /* try to login */
938  ret=owl_aim_login(user, passwd);
939  if (ret) owl_function_makemsg("Warning: login for %s failed.\n", user);
940}
941
[d54838d]942void owl_function_suspend()
943{
[7d4fbcd]944  endwin();
945  printf("\n");
946  kill(getpid(), SIGSTOP);
947
948  /* resize to reinitialize all the windows when we come back */
949  owl_command_resize();
950}
951
[d54838d]952void owl_function_zaway_toggle()
953{
[7d4fbcd]954  if (!owl_global_is_zaway(&g)) {
955    owl_global_set_zaway_msg(&g, owl_global_get_zaway_msg_default(&g));
956    owl_function_zaway_on();
957  } else {
958    owl_function_zaway_off();
959  }
960}
961
[d54838d]962void owl_function_zaway_on()
963{
[7d4fbcd]964  owl_global_set_zaway_on(&g);
[4b660cc]965  owl_function_makemsg("zaway set (%s)", owl_global_get_zaway_msg(&g));
[7d4fbcd]966}
967
[d54838d]968void owl_function_zaway_off()
969{
[7d4fbcd]970  owl_global_set_zaway_off(&g);
[4b660cc]971  owl_function_makemsg("zaway off");
972}
973
974void owl_function_aaway_toggle()
975{
976  if (!owl_global_is_aaway(&g)) {
977    owl_global_set_aaway_msg(&g, owl_global_get_aaway_msg_default(&g));
978    owl_function_aaway_on();
979  } else {
980    owl_function_aaway_off();
981  }
982}
983
984void owl_function_aaway_on()
985{
986  owl_global_set_aaway_on(&g);
987  /* owl_aim_set_awaymsg(owl_global_get_zaway_msg(&g)); */
988  owl_function_makemsg("AIM away set (%s)", owl_global_get_aaway_msg(&g));
989}
990
991void owl_function_aaway_off()
992{
993  owl_global_set_aaway_off(&g);
994  /* owl_aim_set_awaymsg(""); */
995  owl_function_makemsg("AIM away off");
[7d4fbcd]996}
997
[d54838d]998void owl_function_quit()
999{
[7d4fbcd]1000  char *ret;
1001 
1002  /* zlog out if we need to */
[7433402]1003  if (owl_global_is_havezephyr(&g) &&
1004      owl_global_is_shutdownlogout(&g)) {
[31e48a3]1005    owl_zephyr_zlog_out();
[7d4fbcd]1006  }
1007
1008  /* execute the commands in shutdown */
[0337203]1009  ret = owl_perlconfig_execute("BarnOwl::Hooks::_shutdown();");
[7d4fbcd]1010  if (ret) owl_free(ret);
1011
[d09e5a1]1012  /* signal our child process, if any */
1013  if (owl_global_get_newmsgproc_pid(&g)) {
1014    kill(owl_global_get_newmsgproc_pid(&g), SIGHUP);
1015  }
1016
[8c46404]1017  /* Quit zephyr */
[09489b89]1018  owl_zephyr_shutdown();
[8c46404]1019 
1020  /* Quit AIM */
1021  if (owl_global_is_aimloggedin(&g)) {
1022    owl_aim_logout();
1023  }
1024
[8232149]1025  /* done with curses */
[7d4fbcd]1026  endwin();
[8232149]1027
1028  /* restore terminal settings */
1029  tcsetattr(0, TCSAFLUSH, owl_global_get_startup_tio(&g));
1030
[7d4fbcd]1031  owl_function_debugmsg("Quitting Owl");
1032  exit(0);
1033}
1034
[d54838d]1035void owl_function_calculate_topmsg(int direction)
1036{
[aa2f33b3]1037  int recwinlines, topmsg, curmsg;
[9e5c9f3]1038  const owl_view *v;
[7d4fbcd]1039
1040  v=owl_global_get_current_view(&g);
[aa2f33b3]1041  curmsg=owl_global_get_curmsg(&g);
1042  topmsg=owl_global_get_topmsg(&g);
[7d4fbcd]1043  recwinlines=owl_global_get_recwin_lines(&g);
1044
[f9c43ae]1045  /*
[7d4fbcd]1046  if (owl_view_get_size(v) < 1) {
1047    return;
1048  }
[f9c43ae]1049  */
[aa2f33b3]1050
1051  switch (owl_global_get_scrollmode(&g)) {
1052  case OWL_SCROLLMODE_TOP:
[f9c43ae]1053    topmsg = owl_function_calculate_topmsg_top(direction, v, curmsg, topmsg, recwinlines);
[aa2f33b3]1054    break;
1055  case OWL_SCROLLMODE_NEARTOP:
[f9c43ae]1056    topmsg = owl_function_calculate_topmsg_neartop(direction, v, curmsg, topmsg, recwinlines);
[aa2f33b3]1057    break;
1058  case OWL_SCROLLMODE_CENTER:
[f9c43ae]1059    topmsg = owl_function_calculate_topmsg_center(direction, v, curmsg, topmsg, recwinlines);
[aa2f33b3]1060    break;
1061  case OWL_SCROLLMODE_PAGED:
[f9c43ae]1062    topmsg = owl_function_calculate_topmsg_paged(direction, v, curmsg, topmsg, recwinlines, 0);
[aa2f33b3]1063    break;
1064  case OWL_SCROLLMODE_PAGEDCENTER:
[f9c43ae]1065    topmsg = owl_function_calculate_topmsg_paged(direction, v, curmsg, topmsg, recwinlines, 1);
[aa2f33b3]1066    break;
1067  case OWL_SCROLLMODE_NORMAL:
1068  default:
[f9c43ae]1069    topmsg = owl_function_calculate_topmsg_normal(direction, v, curmsg, topmsg, recwinlines);
[aa2f33b3]1070  }
[3a2daac]1071  owl_function_debugmsg("Calculated a topmsg of %i", topmsg);
[aa2f33b3]1072  owl_global_set_topmsg(&g, topmsg);
1073}
1074
1075/* Returns what the new topmsg should be. 
1076 * Passed the last direction of movement,
1077 * the current view,
1078 * the current message number in the view,
1079 * the top message currently being displayed,
1080 * and the number of lines in the recwin.
1081 */
[9e5c9f3]1082int owl_function_calculate_topmsg_top(int direction, const owl_view *v, int curmsg, int topmsg, int recwinlines)
[d54838d]1083{
[f9c43ae]1084  return(curmsg);
[aa2f33b3]1085}
1086
[9e5c9f3]1087int owl_function_calculate_topmsg_neartop(int direction, const owl_view *v, int curmsg, int topmsg, int recwinlines)
[d54838d]1088{
[aa2f33b3]1089  if (curmsg>0 
1090      && (owl_message_get_numlines(owl_view_get_element(v, curmsg-1))
1091          <  recwinlines/2)) {
[f9c43ae]1092    return(curmsg-1);
[aa2f33b3]1093  } else {
[f9c43ae]1094    return(curmsg);
[aa2f33b3]1095  }
1096}
1097 
[9e5c9f3]1098int owl_function_calculate_topmsg_center(int direction, const owl_view *v, int curmsg, int topmsg, int recwinlines)
[d54838d]1099{
[aa2f33b3]1100  int i, last, lines;
1101
1102  last = curmsg;
1103  lines = 0;
1104  for (i=curmsg-1; i>=0; i--) {
1105    lines += owl_message_get_numlines(owl_view_get_element(v, i));
1106    if (lines > recwinlines/2) break;
1107    last = i;
1108  }
[f9c43ae]1109  return(last);
[aa2f33b3]1110}
1111 
[9e5c9f3]1112int owl_function_calculate_topmsg_paged(int direction, const owl_view *v, int curmsg, int topmsg, int recwinlines, int center_on_page)
[d54838d]1113{
[aa2f33b3]1114  int i, last, lines, savey;
1115 
1116  /* If we're off the top of the screen, scroll up such that the
1117   * curmsg is near the botton of the screen. */
1118  if (curmsg < topmsg) {
1119    last = curmsg;
1120    lines = 0;
1121    for (i=curmsg; i>=0; i--) {
1122      lines += owl_message_get_numlines(owl_view_get_element(v, i));
1123      if (lines > recwinlines) break;
1124    last = i;
1125    }
1126    if (center_on_page) {
[f9c43ae]1127      return(owl_function_calculate_topmsg_center(direction, v, curmsg, 0, recwinlines));
[aa2f33b3]1128    } else {
[f9c43ae]1129      return(last);
[aa2f33b3]1130    }
1131  }
1132
1133  /* Find number of lines from top to bottom of curmsg (store in savey) */
1134  savey=0;
1135  for (i=topmsg; i<=curmsg; i++) {
1136    savey+=owl_message_get_numlines(owl_view_get_element(v, i));
1137  }
1138
1139  /* if we're off the bottom of the screen, scroll down */
1140  if (savey > recwinlines) {
1141    if (center_on_page) {
[f9c43ae]1142      return(owl_function_calculate_topmsg_center(direction, v, curmsg, 0, recwinlines));
[aa2f33b3]1143    } else {
[f9c43ae]1144      return(curmsg);
[aa2f33b3]1145    }
1146  }
1147
1148  /* else just stay as we are... */
[f9c43ae]1149  return(topmsg);
[aa2f33b3]1150}
1151
[9e5c9f3]1152int owl_function_calculate_topmsg_normal(int direction, const owl_view *v, int curmsg, int topmsg, int recwinlines)
[d54838d]1153{
[801b7ac]1154  int savey, i, foo, y;
[f9c43ae]1155
[88736cb]1156  if (curmsg<0) return(topmsg);
1157   
[f9c43ae]1158  /* If we're off the top of the screen then center */
1159  if (curmsg<topmsg) {
1160    topmsg=owl_function_calculate_topmsg_center(direction, v, curmsg, 0, recwinlines);
1161  }
1162
[801b7ac]1163  /* If curmsg is so far past topmsg that there are more messages than
1164     lines, skip the line counting that follows because we're
1165     certainly off screen.  */
1166  savey=curmsg-topmsg;
1167  if (savey <= recwinlines) {
1168    /* Find number of lines from top to bottom of curmsg (store in savey) */
1169    savey = 0;
1170    for (i=topmsg; i<=curmsg; i++) {
1171      savey+=owl_message_get_numlines(owl_view_get_element(v, i));
1172    }
[7d4fbcd]1173  }
1174
[f9c43ae]1175  /* If we're off the bottom of the screen, set the topmsg to curmsg
1176   * and scroll upwards */
1177  if (savey > recwinlines) {
1178    topmsg=curmsg;
[801b7ac]1179    savey=owl_message_get_numlines(owl_view_get_element(v, curmsg));
[f9c43ae]1180    direction=OWL_DIRECTION_UPWARDS;
[7d4fbcd]1181  }
[f9c43ae]1182 
[7d4fbcd]1183  /* If our bottom line is less than 1/4 down the screen then scroll up */
1184  if (direction == OWL_DIRECTION_UPWARDS || direction == OWL_DIRECTION_NONE) {
1185    if (savey < (recwinlines / 4)) {
1186      y=0;
[801b7ac]1187      for (i=curmsg; i>=0; i--) {
1188        foo=owl_message_get_numlines(owl_view_get_element(v, i));
[7d4fbcd]1189        /* will we run the curmsg off the screen? */
1190        if ((foo+y) >= recwinlines) {
[801b7ac]1191          i++;
1192          if (i>curmsg) i=curmsg;
[7d4fbcd]1193          break;
1194        }
1195        /* have saved 1/2 the screen space? */
1196        y+=foo;
1197        if (y > (recwinlines / 2)) break;
1198      }
[801b7ac]1199      if (i<0) i=0;
1200      return(i);
[7d4fbcd]1201    }
1202  }
1203
1204  if (direction == OWL_DIRECTION_DOWNWARDS || direction == OWL_DIRECTION_NONE) {
1205    /* If curmsg bottom line is more than 3/4 down the screen then scroll down */
1206    if (savey > ((recwinlines * 3)/4)) {
1207      y=0;
1208      /* count lines from the top until we can save 1/2 the screen size */
[801b7ac]1209      for (i=topmsg; i<curmsg; i++) {
1210        y+=owl_message_get_numlines(owl_view_get_element(v, i));
[7d4fbcd]1211        if (y > (recwinlines / 2)) break;
1212      }
[801b7ac]1213      if (i==curmsg) {
1214        i--;
[7d4fbcd]1215      }
[801b7ac]1216      return(i+1);
[7d4fbcd]1217    }
1218  }
[aa2f33b3]1219
[f9c43ae]1220  return(topmsg);
[7d4fbcd]1221}
1222
[d54838d]1223void owl_function_resize()
1224{
[7d4fbcd]1225  owl_global_set_resize_pending(&g);
1226}
1227
[d54838d]1228void owl_function_run_buffercommand()
1229{
[1b6b2f3]1230  owl_editwin_do_callback(owl_global_get_typwin(&g));
[7d4fbcd]1231}
1232
[e19eb97]1233void owl_function_debugmsg(const char *fmt, ...)
[d54838d]1234{
[7d4fbcd]1235  FILE *file;
1236  time_t now;
1237  char buff1[LINE], buff2[LINE];
1238  va_list ap;
1239  va_start(ap, fmt);
1240
1241  if (!owl_global_is_debug_fast(&g)) return;
1242
1243  file=fopen(owl_global_get_debug_file(&g), "a");
1244  if (!file) return;
1245
1246  now=time(NULL);
1247  strcpy(buff1, ctime(&now));
1248  buff1[strlen(buff1)-1]='\0';
1249
1250  owl_global_get_runtime_string(&g, buff2);
1251 
1252  fprintf(file, "[%i -  %s - %s]: ", (int) getpid(), buff1, buff2);
1253  vfprintf(file, fmt, ap);
1254  fprintf(file, "\n");
1255  fclose(file);
1256
1257  va_end(ap);
1258}
1259
[d54838d]1260void owl_function_beep()
1261{
[7d4fbcd]1262  if (owl_global_is_bell(&g)) {
1263    beep();
[b45293f]1264    owl_global_set_needrefresh(&g); /* do we really need this? */
[7d4fbcd]1265  }
1266}
1267
[e19eb97]1268int owl_function_subscribe(const char *class, const char *inst, const char *recip)
[d54838d]1269{
[7d4fbcd]1270  int ret;
1271
1272  ret=owl_zephyr_sub(class, inst, recip);
1273  if (ret) {
[ec6ff52]1274    owl_function_error("Error subscribing.");
[7d4fbcd]1275  } else {
1276    owl_function_makemsg("Subscribed.");
1277  }
[3617286]1278  return(ret);
[7d4fbcd]1279}
1280
[e19eb97]1281void owl_function_unsubscribe(const char *class, const char *inst, const char *recip)
[d54838d]1282{
[7d4fbcd]1283  int ret;
1284
1285  ret=owl_zephyr_unsub(class, inst, recip);
1286  if (ret) {
[ec6ff52]1287    owl_function_error("Error subscribing.");
[7d4fbcd]1288  } else {
1289    owl_function_makemsg("Unsubscribed.");
1290  }
1291}
1292
[d54838d]1293void owl_function_set_cursor(WINDOW *win)
1294{
[7d4fbcd]1295  wnoutrefresh(win);
1296}
1297
[d54838d]1298void owl_function_full_redisplay()
1299{
[7d4fbcd]1300  redrawwin(owl_global_get_curs_recwin(&g));
1301  redrawwin(owl_global_get_curs_sepwin(&g));
[af1920fd]1302  /* Work around curses segfualts with windows off the screen */
[451becb]1303  if (g.lines >= owl_global_get_typwin_lines(&g)+2)
1304      redrawwin(owl_global_get_curs_typwin(&g));
1305  if (g.lines >= 2)
1306      redrawwin(owl_global_get_curs_msgwin(&g));
[7d4fbcd]1307
1308  wnoutrefresh(owl_global_get_curs_recwin(&g));
1309  wnoutrefresh(owl_global_get_curs_sepwin(&g));
1310  wnoutrefresh(owl_global_get_curs_typwin(&g));
1311  wnoutrefresh(owl_global_get_curs_msgwin(&g));
[c2c5c77]1312
1313  if (owl_popwin_is_active(owl_global_get_popwin(&g))) {
1314    owl_popwin_refresh(owl_global_get_popwin(&g));
1315  }
[7d4fbcd]1316 
1317  sepbar("");
1318  owl_function_makemsg("");
1319
1320  owl_global_set_needrefresh(&g);
1321}
1322
[e19eb97]1323void owl_function_popless_text(const char *text)
[d54838d]1324{
[7d4fbcd]1325  owl_popwin *pw;
1326  owl_viewwin *v;
1327
1328  pw=owl_global_get_popwin(&g);
1329  v=owl_global_get_viewwin(&g);
1330
1331  owl_popwin_up(pw);
1332  owl_viewwin_init_text(v, owl_popwin_get_curswin(pw),
1333                        owl_popwin_get_lines(pw), owl_popwin_get_cols(pw),
1334                        text);
1335  owl_popwin_refresh(pw);
1336  owl_viewwin_redisplay(v, 0);
1337  owl_global_set_needrefresh(&g);
1338}
1339
[075ba92]1340void owl_function_popless_fmtext(const owl_fmtext *fm)
[d54838d]1341{
[7d4fbcd]1342  owl_popwin *pw;
1343  owl_viewwin *v;
1344
1345  pw=owl_global_get_popwin(&g);
1346  v=owl_global_get_viewwin(&g);
1347
1348  owl_popwin_up(pw);
1349  owl_viewwin_init_fmtext(v, owl_popwin_get_curswin(pw),
1350                   owl_popwin_get_lines(pw), owl_popwin_get_cols(pw),
1351                   fm);
1352  owl_popwin_refresh(pw);
1353  owl_viewwin_redisplay(v, 0);
1354  owl_global_set_needrefresh(&g);
[f17bff98]1355}
1356
[e19eb97]1357void owl_function_popless_file(const char *filename)
[f17bff98]1358{
1359  owl_fmtext fm;
1360  FILE *file;
1361  char buff[1024];
1362
1363  file=fopen(filename, "r");
1364  if (!file) {
1365    owl_function_error("Could not open file: %s", filename);
1366    return;
1367  }
1368
1369  owl_fmtext_init_null(&fm);
1370  while (fgets(buff, 1024, file)) {
1371    owl_fmtext_append_normal(&fm, buff);
[72836b5]1372    /*    owl_fmtext_append_normal(&fm, "\n"); */
[f17bff98]1373  }
1374
1375  owl_function_popless_fmtext(&fm);
1376  owl_fmtext_free(&fm);
1377  fclose(file);
[7d4fbcd]1378}
1379
[d54838d]1380void owl_function_about()
1381{
[2101a50]1382  owl_function_popless_text(
1383    "This is barnowl version " OWL_VERSION_STRING ".\n\n"
1384    "barnowl is a fork of the Owl zephyr client, written and\n"
1385    "maintained by Alejandro Sedeno and Nelson Elhage at the\n"
1386    "Massachusetts Institute of Technology. \n"
1387    "\n"
1388    "Owl was written by James Kretchmar. The first version, 0.5, was\n"
1389    "released in March 2002.\n"
1390    "\n"
1391    "The name 'owl' was chosen in reference to the owls in the\n"
1392    "Harry Potter novels, who are tasked with carrying messages\n"
1393    "between Witches and Wizards. The name 'barnowl' was chosen\n"
1394    "because we feel our owls should live closer to our ponies.\n"
1395    "\n"
[23e1110]1396    "Copyright (c) 2006-2009 The BarnOwl Developers. All rights reserved.\n"
[2101a50]1397    "Copyright (c) 2004 James Kretchmar. All rights reserved.\n"
1398    "Copyright 2002 Massachusetts Institute of Technology\n"
1399    "\n"
1400    "This program is free software. You can redistribute it and/or\n"
1401    "modify under the terms of the Sleepycat License. Use the \n"
1402    "':show license' command to display the full license\n"
1403  );
[7d4fbcd]1404}
1405
[d54838d]1406void owl_function_info()
1407{
[c08c70a]1408  const owl_message *m;
[5789230]1409  owl_fmtext fm, attrfm;
[9e5c9f3]1410  const owl_view *v;
[09489b89]1411#ifdef HAVE_LIBZEPHYR
[1077891a]1412  const ZNotice_t *n;
[09489b89]1413#endif
[7d4fbcd]1414
[d0d65df]1415  owl_fmtext_init_null(&fm);
1416 
[7d4fbcd]1417  v=owl_global_get_current_view(&g);
[5eeea3b]1418  m=owl_view_get_element(v, owl_global_get_curmsg(&g));
1419  if (!m || owl_view_get_size(v)==0) {
[ec6ff52]1420    owl_function_error("No message selected\n");
[7d4fbcd]1421    return;
1422  }
1423
[5789230]1424  owl_fmtext_append_bold(&fm, "General Information:\n");
[57609b3]1425  owl_fmtext_appendf_normal(&fm, "  Msg Id    : %i\n", owl_message_get_id(m));
[df0d93a]1426
[5789230]1427  owl_fmtext_append_normal(&fm, "  Type      : ");
[37eab7f]1428  owl_fmtext_append_bold(&fm, owl_message_get_type(m));
[df0d93a]1429  owl_fmtext_append_normal(&fm, "\n");
1430
[4b464a4]1431  if (owl_message_is_direction_in(m)) {
[5789230]1432    owl_fmtext_append_normal(&fm, "  Direction : in\n");
[4b464a4]1433  } else if (owl_message_is_direction_out(m)) {
[5789230]1434    owl_fmtext_append_normal(&fm, "  Direction : out\n");
[4b464a4]1435  } else if (owl_message_is_direction_none(m)) {
[5789230]1436    owl_fmtext_append_normal(&fm, "  Direction : none\n");
[4b464a4]1437  } else {
[5789230]1438    owl_fmtext_append_normal(&fm, "  Direction : unknown\n");
[4b464a4]1439  }
[df0d93a]1440
[57609b3]1441  owl_fmtext_appendf_normal(&fm, "  Time      : %s\n", owl_message_get_timestr(m));
[4b464a4]1442
[df0d93a]1443  if (!owl_message_is_type_admin(m)) {
[57609b3]1444    owl_fmtext_appendf_normal(&fm, "  Sender    : %s\n", owl_message_get_sender(m));
1445    owl_fmtext_appendf_normal(&fm, "  Recipient : %s\n", owl_message_get_recipient(m));
[df0d93a]1446  }
[57609b3]1447
[0ff8fb57]1448  if (owl_message_is_type_zephyr(m)) {
[5789230]1449    owl_fmtext_append_bold(&fm, "\nZephyr Specific Information:\n");
[0ff8fb57]1450   
[57609b3]1451    owl_fmtext_appendf_normal(&fm, "  Class     : %s\n", owl_message_get_class(m));
1452    owl_fmtext_appendf_normal(&fm, "  Instance  : %s\n", owl_message_get_instance(m));
1453    owl_fmtext_appendf_normal(&fm, "  Opcode    : %s\n", owl_message_get_opcode(m));
[09489b89]1454#ifdef HAVE_LIBZEPHYR
[d0d65df]1455    if (owl_message_is_direction_in(m)) {
[09489b89]1456      char *ptr, tmpbuff[1024];
1457      int i, j, fields, len;
1458
[d0d65df]1459      n=owl_message_get_notice(m);
[df0d93a]1460
[5a95b69]1461      if (!owl_message_is_pseudo(m)) {
1462        owl_fmtext_append_normal(&fm, "  Kind      : ");
1463        if (n->z_kind==UNSAFE) {
1464          owl_fmtext_append_normal(&fm, "UNSAFE\n");
1465        } else if (n->z_kind==UNACKED) {
1466          owl_fmtext_append_normal(&fm, "UNACKED\n");
1467        } else if (n->z_kind==ACKED) {
1468          owl_fmtext_append_normal(&fm, "ACKED\n");
1469        } else if (n->z_kind==HMACK) {
1470          owl_fmtext_append_normal(&fm, "HMACK\n");
1471        } else if (n->z_kind==HMCTL) {
1472          owl_fmtext_append_normal(&fm, "HMCTL\n");
1473        } else if (n->z_kind==SERVACK) {
1474          owl_fmtext_append_normal(&fm, "SERVACK\n");
1475        } else if (n->z_kind==SERVNAK) {
1476          owl_fmtext_append_normal(&fm, "SERVNACK\n");
1477        } else if (n->z_kind==CLIENTACK) {
1478          owl_fmtext_append_normal(&fm, "CLIENTACK\n");
1479        } else if (n->z_kind==STAT) {
1480          owl_fmtext_append_normal(&fm, "STAT\n");
1481        } else {
1482          owl_fmtext_append_normal(&fm, "ILLEGAL VALUE\n");
1483        }
[d0d65df]1484      }
[57609b3]1485      owl_fmtext_appendf_normal(&fm, "  Host      : %s\n", owl_message_get_hostname(m));
[5a95b69]1486
1487      if (!owl_message_is_pseudo(m)) {
1488        owl_fmtext_append_normal(&fm, "\n");
[57609b3]1489        owl_fmtext_appendf_normal(&fm, "  Port      : %i\n", ntohs(n->z_port));
[f12d199]1490        owl_fmtext_appendf_normal(&fm, "  Auth      : %s\n", owl_zephyr_get_authstr(n));
[57609b3]1491
1492        /* FIXME make these more descriptive */
[f12d199]1493        owl_fmtext_appendf_normal(&fm, "  Checkd Ath: %i\n", n->z_checked_auth);
[57609b3]1494        owl_fmtext_appendf_normal(&fm, "  Multi notc: %s\n", n->z_multinotice);
1495        owl_fmtext_appendf_normal(&fm, "  Num other : %i\n", n->z_num_other_fields);
1496        owl_fmtext_appendf_normal(&fm, "  Msg Len   : %i\n", n->z_message_len);
[5a95b69]1497
1498        fields=owl_zephyr_get_num_fields(n);
[57609b3]1499        owl_fmtext_appendf_normal(&fm, "  Fields    : %i\n", fields);
1500
[5a95b69]1501        for (i=0; i<fields; i++) {
[5376a95]1502          ptr=owl_zephyr_get_field_as_utf8(n, i+1);
[b0430a6]1503          len=strlen(ptr);
[5a95b69]1504          if (len<30) {
1505            strncpy(tmpbuff, ptr, len);
1506            tmpbuff[len]='\0';
1507          } else {
1508            strncpy(tmpbuff, ptr, 30);
1509            tmpbuff[30]='\0';
1510            strcat(tmpbuff, "...");
1511          }
[b0430a6]1512          owl_free(ptr);
[57609b3]1513
[5a95b69]1514          for (j=0; j<strlen(tmpbuff); j++) {
1515            if (tmpbuff[j]=='\n') tmpbuff[j]='~';
1516            if (tmpbuff[j]=='\r') tmpbuff[j]='!';
1517          }
[57609b3]1518
1519          owl_fmtext_appendf_normal(&fm, "  Field %i   : %s\n", i+1, tmpbuff);
[5a95b69]1520        }
[57609b3]1521        owl_fmtext_appendf_normal(&fm, "  Default Fm: %s\n", n->z_default_format);
[d0d65df]1522      }
[57609b3]1523
[7d4fbcd]1524    }
[57609b3]1525#endif
[7d4fbcd]1526  }
[0ff8fb57]1527
[5789230]1528  owl_fmtext_append_bold(&fm, "\nOwl Message Attributes:\n");
1529  owl_message_attributes_tofmtext(m, &attrfm);
1530  owl_fmtext_append_fmtext(&fm, &attrfm);
[d0d65df]1531 
1532  owl_function_popless_fmtext(&fm);
[5789230]1533  owl_fmtext_free(&fm);
1534  owl_fmtext_free(&attrfm);
[7d4fbcd]1535}
1536
[5639bf2]1537/* print the current message in a popup window.
1538 * Use the 'default' style regardless of whatever
1539 * style the user may be using
1540 */
[d54838d]1541void owl_function_curmsg_to_popwin()
1542{
[9e5c9f3]1543  const owl_view *v;
[c08c70a]1544  const owl_message *m;
[1fdab04]1545  const owl_style *s;
[5639bf2]1546  owl_fmtext fm;
[7d4fbcd]1547
[5639bf2]1548  v=owl_global_get_current_view(&g);
1549  s=owl_global_get_style_by_name(&g, "default");
[7d4fbcd]1550
[5eeea3b]1551  m=owl_view_get_element(v, owl_global_get_curmsg(&g));
1552
1553  if (!m || owl_view_get_size(v)==0) {
[ec6ff52]1554    owl_function_error("No current message");
[7d4fbcd]1555    return;
1556  }
1557
[5639bf2]1558  owl_fmtext_init_null(&fm);
1559  owl_style_get_formattext(s, &fm, m);
1560
1561  owl_function_popless_fmtext(&fm);
1562  owl_fmtext_free(&fm);
[7d4fbcd]1563}
1564
[d54838d]1565void owl_function_page_curmsg(int step)
1566{
[7d4fbcd]1567  /* scroll down or up within the current message IF the message is truncated */
1568
1569  int offset, curmsg, lines;
[9e5c9f3]1570  const owl_view *v;
[7d4fbcd]1571  owl_message *m;
1572
1573  offset=owl_global_get_curmsg_vert_offset(&g);
1574  v=owl_global_get_current_view(&g);
1575  curmsg=owl_global_get_curmsg(&g);
1576  m=owl_view_get_element(v, curmsg);
[5eeea3b]1577  if (!m || owl_view_get_size(v)==0) return;
[7d4fbcd]1578  lines=owl_message_get_numlines(m);
1579
1580  if (offset==0) {
1581    /* Bail if the curmsg isn't the last one displayed */
1582    if (curmsg != owl_mainwin_get_last_msg(owl_global_get_mainwin(&g))) {
[f51bc78]1583      owl_function_makemsg("The entire message is already displayed");
[7d4fbcd]1584      return;
1585    }
1586   
1587    /* Bail if we're not truncated */
1588    if (!owl_mainwin_is_curmsg_truncated(owl_global_get_mainwin(&g))) {
[f51bc78]1589      owl_function_makemsg("The entire message is already displayed");
[7d4fbcd]1590      return;
1591    }
1592  }
1593 
1594 
1595  /* don't scroll past the last line */
1596  if (step>0) {
1597    if (offset+step > lines-1) {
1598      owl_global_set_curmsg_vert_offset(&g, lines-1);
1599    } else {
1600      owl_global_set_curmsg_vert_offset(&g, offset+step);
1601    }
1602  }
1603
1604  /* would we be before the beginning of the message? */
1605  if (step<0) {
1606    if (offset+step<0) {
1607      owl_global_set_curmsg_vert_offset(&g, 0);
1608    } else {
1609      owl_global_set_curmsg_vert_offset(&g, offset+step);
1610    }
1611  }
1612 
1613  /* redisplay */
1614  owl_mainwin_redisplay(owl_global_get_mainwin(&g));
1615  owl_global_set_needrefresh(&g);
1616}
1617
[d54838d]1618void owl_function_resize_typwin(int newsize)
1619{
[7d4fbcd]1620  owl_global_set_typwin_lines(&g, newsize);
1621  owl_function_resize();
1622}
1623
[d54838d]1624void owl_function_mainwin_pagedown()
1625{
[7d4fbcd]1626  int i;
1627
1628  i=owl_mainwin_get_last_msg(owl_global_get_mainwin(&g));
1629  if (i<0) return;
[f2e36b5]1630  if (owl_mainwin_is_last_msg_truncated(owl_global_get_mainwin(&g))
1631      && (owl_global_get_curmsg(&g) < i)
1632      && (i>0)) {
1633    i--;
1634  }
[7d4fbcd]1635  owl_global_set_curmsg(&g, i);
1636  owl_function_nextmsg();
1637}
1638
[d54838d]1639void owl_function_mainwin_pageup()
1640{
[7d4fbcd]1641  owl_global_set_curmsg(&g, owl_global_get_topmsg(&g));
1642  owl_function_prevmsg();
1643}
1644
[d54838d]1645void owl_function_getsubs()
1646{
[09489b89]1647  char *buff;
[7d4fbcd]1648
[09489b89]1649  buff=owl_zephyr_getsubs();
[7d4fbcd]1650
[09489b89]1651  if (buff) {
1652    owl_function_popless_text(buff);
1653  } else {
1654    owl_function_popless_text("Error getting subscriptions");
[7d4fbcd]1655  }
[09489b89]1656           
[1c6c4d3]1657  owl_free(buff);
[7d4fbcd]1658}
1659
[d54838d]1660void owl_function_printallvars()
1661{
[e19eb97]1662  const char *name;
[b4c270c]1663  char var[LINE];
[7d4fbcd]1664  owl_list varnames;
[b4c270c]1665  int i, numvarnames;
1666  GString *str   = g_string_new("");
[7d4fbcd]1667
[b4c270c]1668  g_string_append_printf(str, "%-20s = %s\n", "VARIABLE", "VALUE");
1669  g_string_append_printf(str, "%-20s   %s\n",  "--------", "-----");
[7d4fbcd]1670  owl_variable_dict_get_names(owl_global_get_vardict(&g), &varnames);
1671  numvarnames = owl_list_get_size(&varnames);
1672  for (i=0; i<numvarnames; i++) {
1673    name = owl_list_get_element(&varnames, i);
1674    if (name && name[0]!='_') {
[b4c270c]1675      g_string_append_printf(str, "\n%-20s = ", name);
1676      owl_variable_get_tostring(owl_global_get_vardict(&g), name, var, LINE);
1677      g_string_append(str, var);
[7d4fbcd]1678    }
1679  }
[b4c270c]1680  g_string_append(str, "\n");
[7d4fbcd]1681  owl_variable_dict_namelist_free(&varnames);
[b4c270c]1682
1683  owl_function_popless_text(str->str);
1684  g_string_free(str, TRUE);
[7d4fbcd]1685}
1686
[d54838d]1687void owl_function_show_variables()
1688{
[7d4fbcd]1689  owl_list varnames;
1690  owl_fmtext fm; 
1691  int i, numvarnames;
[e19eb97]1692  const char *varname;
[7d4fbcd]1693
1694  owl_fmtext_init_null(&fm);
1695  owl_fmtext_append_bold(&fm, 
1696      "Variables: (use 'show variable <name>' for details)\n");
1697  owl_variable_dict_get_names(owl_global_get_vardict(&g), &varnames);
1698  numvarnames = owl_list_get_size(&varnames);
1699  for (i=0; i<numvarnames; i++) {
1700    varname = owl_list_get_element(&varnames, i);
1701    if (varname && varname[0]!='_') {
[aa2f33b3]1702      owl_variable_describe(owl_global_get_vardict(&g), varname, &fm);
[7d4fbcd]1703    }
1704  }
1705  owl_variable_dict_namelist_free(&varnames);
1706  owl_function_popless_fmtext(&fm);
1707  owl_fmtext_free(&fm);
1708}
1709
[e19eb97]1710void owl_function_show_variable(const char *name)
[d54838d]1711{
[7d4fbcd]1712  owl_fmtext fm; 
1713
1714  owl_fmtext_init_null(&fm);
1715  owl_variable_get_help(owl_global_get_vardict(&g), name, &fm);
1716  owl_function_popless_fmtext(&fm);
1717  owl_fmtext_free(&fm); 
1718}
1719
1720/* note: this applies to global message list, not to view.
1721 * If flag is 1, deletes.  If flag is 0, undeletes. */
[d54838d]1722void owl_function_delete_by_id(int id, int flag)
1723{
[3eb599d]1724  const owl_messagelist *ml;
[7d4fbcd]1725  owl_message *m;
1726  ml = owl_global_get_msglist(&g);
1727  m = owl_messagelist_get_by_id(ml, id);
1728  if (m) {
1729    if (flag == 1) {
1730      owl_message_mark_delete(m);
1731    } else if (flag == 0) {
1732      owl_message_unmark_delete(m);
1733    }
1734    owl_mainwin_redisplay(owl_global_get_mainwin(&g));
1735    owl_global_set_needrefresh(&g);
1736  } else {
[ec6ff52]1737    owl_function_error("No message with id %d: unable to mark for (un)delete",id);
[7d4fbcd]1738  }
1739}
1740
[d54838d]1741void owl_function_delete_automsgs()
1742{
[7d4fbcd]1743  /* mark for deletion all messages in the current view that match the
1744   * 'trash' filter */
1745
1746  int i, j, count;
1747  owl_message *m;
[9e5c9f3]1748  const owl_view *v;
[4542047]1749  const owl_filter *f;
[7d4fbcd]1750
1751  /* get the trash filter */
1752  f=owl_global_get_filter(&g, "trash");
1753  if (!f) {
[ec6ff52]1754    owl_function_error("No trash filter defined");
[7d4fbcd]1755    return;
1756  }
1757
1758  v=owl_global_get_current_view(&g);
1759
1760  count=0;
1761  j=owl_view_get_size(v);
1762  for (i=0; i<j; i++) {
1763    m=owl_view_get_element(v, i);
1764    if (owl_filter_message_match(f, m)) {
1765      count++;
1766      owl_message_mark_delete(m);
1767    }
1768  }
1769  owl_mainwin_redisplay(owl_global_get_mainwin(&g));
[1c6c4d3]1770  owl_function_makemsg("%i messages marked for deletion", count);
[7d4fbcd]1771  owl_global_set_needrefresh(&g);
1772}
1773
[d54838d]1774void owl_function_status()
1775{
[49a8434]1776  char buff[MAXPATHLEN+1];
[7d4fbcd]1777  time_t start;
1778  int up, days, hours, minutes;
[a352335c]1779  owl_fmtext fm;
1780
1781  owl_fmtext_init_null(&fm);
[7d4fbcd]1782
1783  start=owl_global_get_starttime(&g);
1784
[d9b0b972]1785  owl_fmtext_append_normal(&fm, "General Information:\n");
1786
1787  owl_fmtext_append_normal(&fm, "  Version: ");
[a352335c]1788  owl_fmtext_append_normal(&fm, OWL_VERSION_STRING);
1789  owl_fmtext_append_normal(&fm, "\n");
1790
[cdd3959]1791  owl_fmtext_append_normal(&fm, "  Startup Arguments: ");
[a352335c]1792  owl_fmtext_append_normal(&fm, owl_global_get_startupargs(&g));
1793  owl_fmtext_append_normal(&fm, "\n");
[b6a7367]1794
1795  owl_fmtext_append_normal(&fm, "  Current Directory: ");
[49a8434]1796  if(getcwd(buff, MAXPATHLEN) == NULL) {
1797    owl_fmtext_append_normal(&fm, "<Error in getcwd>");
1798  } else {
1799    owl_fmtext_append_normal(&fm, buff);
1800  }
[b6a7367]1801  owl_fmtext_append_normal(&fm, "\n");
1802
[c1d166b]1803  owl_fmtext_appendf_normal(&fm, "  Startup Time: %s", ctime(&start));
[7d4fbcd]1804
1805  up=owl_global_get_runtime(&g);
1806  days=up/86400;
1807  up-=days*86400;
1808  hours=up/3600;
1809  up-=hours*3600;
1810  minutes=up/60;
1811  up-=minutes*60;
[c1d166b]1812  owl_fmtext_appendf_normal(&fm, "  Run Time: %i days %2.2i:%2.2i:%2.2i\n", days, hours, minutes, up);
[7d4fbcd]1813
[d9b0b972]1814  owl_fmtext_append_normal(&fm, "\nProtocol Options:\n");
1815  owl_fmtext_append_normal(&fm, "  Zephyr included    : ");
1816  if (owl_global_is_havezephyr(&g)) {
1817    owl_fmtext_append_normal(&fm, "yes\n");
[7d4fbcd]1818  } else {
[d9b0b972]1819    owl_fmtext_append_normal(&fm, "no\n");
[7d4fbcd]1820  }
[d9b0b972]1821  owl_fmtext_append_normal(&fm, "  AIM included       : yes\n");
1822  owl_fmtext_append_normal(&fm, "  Loopback included  : yes\n");
1823
[8262340]1824
[d9b0b972]1825  owl_fmtext_append_normal(&fm, "\nBuild Options:\n");
1826  owl_fmtext_append_normal(&fm, "  Stderr redirection : ");
1827#if OWL_STDERR_REDIR
1828  owl_fmtext_append_normal(&fm, "yes\n");
1829#else
1830  owl_fmtext_append_normal(&fm, "no\n");
1831#endif
1832 
1833
1834  owl_fmtext_append_normal(&fm, "\nAIM Status:\n");
1835  owl_fmtext_append_normal(&fm, "  Logged in: ");
[a352335c]1836  if (owl_global_is_aimloggedin(&g)) {
1837    owl_fmtext_append_normal(&fm, owl_global_get_aim_screenname(&g));
1838    owl_fmtext_append_normal(&fm, "\n");
1839  } else {
[d9b0b972]1840    owl_fmtext_append_normal(&fm, "(not logged in)\n");
[a352335c]1841  }
[d9b0b972]1842
1843  owl_fmtext_append_normal(&fm, "  Processing events: ");
[a352335c]1844  if (owl_global_is_doaimevents(&g)) {
[d9b0b972]1845    owl_fmtext_append_normal(&fm, "yes\n");
[a352335c]1846  } else {
[d9b0b972]1847    owl_fmtext_append_normal(&fm, "no\n");
[a352335c]1848  }
1849
1850  owl_function_popless_fmtext(&fm);
1851  owl_fmtext_free(&fm);
[7d4fbcd]1852}
1853
[d54838d]1854void owl_function_show_term()
1855{
[7d4fbcd]1856  owl_fmtext fm;
1857
1858  owl_fmtext_init_null(&fm);
[c1d166b]1859  owl_fmtext_appendf_normal(&fm, "Terminal Lines: %i\nTerminal Columns: %i\n",
[7d4fbcd]1860          owl_global_get_lines(&g),
1861          owl_global_get_cols(&g));
1862
1863  if (owl_global_get_hascolors(&g)) {
1864    owl_fmtext_append_normal(&fm, "Color: Yes\n");
[c1d166b]1865    owl_fmtext_appendf_normal(&fm, "Number of color pairs: %i\n", owl_global_get_colorpairs(&g));
1866    owl_fmtext_appendf_normal(&fm, "Can change colors: %s\n", can_change_color() ? "yes" : "no");
[7d4fbcd]1867  } else {
1868    owl_fmtext_append_normal(&fm, "Color: No\n");
1869  }
1870
1871  owl_function_popless_fmtext(&fm);
1872  owl_fmtext_free(&fm);
1873}
1874
[e7cc1c3]1875/* if type = 0 then normal reply.
1876 * if type = 1 then it's a reply to sender
1877 * if enter = 0 then allow the command to be edited
1878 * if enter = 1 then don't wait for editing
1879 */
[d54838d]1880void owl_function_reply(int type, int enter)
1881{
[740d5f7]1882  char *buff=NULL;
[c08c70a]1883  const owl_message *m;
[4542047]1884  const owl_filter *f;
[7d4fbcd]1885 
1886  if (owl_view_get_size(owl_global_get_current_view(&g))==0) {
[ec6ff52]1887    owl_function_error("No message selected");
[7d4fbcd]1888  } else {
[5ebc202]1889    char *cmd;
[7d4fbcd]1890   
1891    m=owl_view_get_element(owl_global_get_current_view(&g), owl_global_get_curmsg(&g));
[5eeea3b]1892    if (!m) {
[ec6ff52]1893      owl_function_error("No message selected");
[5eeea3b]1894      return;
1895    }
1896
[7d4fbcd]1897    /* first check if we catch the reply-lockout filter */
1898    f=owl_global_get_filter(&g, "reply-lockout");
1899    if (f) {
1900      if (owl_filter_message_match(f, m)) {
[ec6ff52]1901        owl_function_error("Sorry, replies to this message have been disabled by the reply-lockout filter");
[7d4fbcd]1902        return;
1903      }
1904    }
[4b464a4]1905
[2c09826]1906    /* then check if it's a question and just bring up the command prompt */
1907    if (owl_message_is_question(m)) {
1908      owl_function_start_command("");
1909      return;
1910    }
1911
[740d5f7]1912    if((type == 0 &&
1913        (cmd=owl_perlconfig_message_call_method(m, "replycmd", 0, NULL))) ||
1914       (type == 1 &&
1915        (cmd=owl_perlconfig_message_call_method(m, "replysendercmd", 0, NULL)))) {
1916      buff = cmd;
[d09e5a1]1917    }
[1b6b2f3]1918
[e0540e4]1919    if(!buff) {
1920        owl_function_error("I don't know how to reply to that message.");
1921        return;
1922    }
[740d5f7]1923
[d09e5a1]1924    if (enter) {
1925      owl_history *hist = owl_global_get_cmd_history(&g);
1926      owl_history_store(hist, buff);
1927      owl_history_reset(hist);
1928      owl_function_command_norv(buff);
1929    } else {
1930      owl_function_start_command(buff);
[7d4fbcd]1931    }
[d09e5a1]1932    owl_free(buff);
[7d4fbcd]1933  }
1934}
1935
[e19eb97]1936void owl_function_zlocate(int argc, const char *const *argv, int auth)
[d54838d]1937{
[2527615]1938  owl_fmtext fm;
1939  char *ptr, buff[LINE];
1940  int i;
1941
1942  owl_fmtext_init_null(&fm);
[7d4fbcd]1943
[2527615]1944  for (i=0; i<argc; i++) {
1945    ptr=long_zuser(argv[i]);
1946    owl_zephyr_zlocate(ptr, buff, auth);
1947    owl_fmtext_append_normal(&fm, buff);
1948    owl_free(ptr);
[7d4fbcd]1949  }
1950
[2527615]1951  owl_function_popless_fmtext(&fm);
1952  owl_fmtext_free(&fm);
[7d4fbcd]1953}
1954
[e19eb97]1955void owl_function_start_command(const char *line)
[d54838d]1956{
[7d4fbcd]1957  owl_editwin *tw;
1958
1959  tw=owl_global_get_typwin(&g);
1960  owl_global_set_typwin_active(&g);
[10b866d]1961  owl_editwin_new_style(tw, OWL_EDITWIN_STYLE_ONELINE, 
1962                        owl_global_get_cmd_history(&g));
1963
[7d4fbcd]1964  owl_editwin_set_locktext(tw, "command: ");
1965  owl_global_set_needrefresh(&g);
1966
[47519e1b]1967  owl_editwin_insert_string(tw, line);
[7d4fbcd]1968  owl_editwin_redisplay(tw, 0);
[cf83b7a]1969
1970  owl_context_set_editline(owl_global_get_context(&g), tw);
1971  owl_function_activate_keymap("editline");
1972}
1973
[e19eb97]1974void owl_function_start_question(const char *line)
[cf83b7a]1975{
1976  owl_editwin *tw;
1977
1978  tw=owl_global_get_typwin(&g);
1979  owl_global_set_typwin_active(&g);
1980  owl_editwin_new_style(tw, OWL_EDITWIN_STYLE_ONELINE, owl_global_get_cmd_history(&g));
1981
1982  owl_editwin_set_locktext(tw, line);
1983  owl_global_set_needrefresh(&g);
1984
1985  owl_editwin_redisplay(tw, 0);
1986
1987  owl_context_set_editresponse(owl_global_get_context(&g), tw);
1988  owl_function_activate_keymap("editresponse");
[7d4fbcd]1989}
1990
[e19eb97]1991void owl_function_start_password(const char *line)
[453bd70]1992{
1993  owl_editwin *tw;
1994
1995  tw=owl_global_get_typwin(&g);
1996  owl_global_set_typwin_active(&g);
1997  owl_editwin_new_style(tw, OWL_EDITWIN_STYLE_ONELINE, owl_global_get_cmd_history(&g));
1998  owl_editwin_set_echochar(tw, '*');
1999
2000  owl_editwin_set_locktext(tw, line);
2001  owl_global_set_needrefresh(&g);
2002
2003  owl_editwin_redisplay(tw, 0);
2004
2005  owl_context_set_editresponse(owl_global_get_context(&g), tw);
2006  owl_function_activate_keymap("editresponse");
2007}
2008
[e19eb97]2009char *owl_function_exec(int argc, const char *const *argv, const char *buff, int type)
[d54838d]2010{
[7d4fbcd]2011  /* if type == 1 display in a popup
2012   * if type == 2 display an admin messages
2013   * if type == 0 return output
2014   * else display in a popup
2015   */
[e19eb97]2016  const char *redirect = " 2>&1 < /dev/null";
[65b2173]2017  char *newbuff;
[7d4fbcd]2018  char *out, buff2[1024];
2019  int size;
2020  FILE *p;
2021
[2a2bb60]2022#if OWL_STDERR_REDIR
2023  redirect = " < /dev/null";
2024#endif
2025
[7d4fbcd]2026  if (argc<2) {
[ec6ff52]2027    owl_function_error("Wrong number of arguments to the exec command");
[7d4fbcd]2028    return NULL;
2029  }
2030
2031  buff = skiptokens(buff, 1);
2032  newbuff = owl_malloc(strlen(buff)+strlen(redirect)+1);
2033  strcpy(newbuff, buff);
2034  strcat(newbuff, redirect);
2035
[afbf668]2036  if (type == 1) {
2037    owl_popexec_new(newbuff);
[7d4fbcd]2038  } else {
[afbf668]2039    p=popen(newbuff, "r");
2040    out=owl_malloc(1024);
2041    size=1024;
2042    strcpy(out, "");
2043    while (fgets(buff2, 1024, p)!=NULL) {
2044      size+=1024;
2045      out=owl_realloc(out, size);
2046      strcat(out, buff2);
2047    }
2048    pclose(p);
2049   
2050    if (type==1) {
2051      owl_function_popless_text(out);
2052    } else if (type==0) {
2053      return out;
2054    } else if (type==2) {
2055      owl_function_adminmsg(buff, out);
2056    } else {
2057      owl_function_popless_text(out);
2058    }
2059    owl_free(out);
[7d4fbcd]2060  }
2061  return NULL;
2062}
2063
[e19eb97]2064char *owl_function_perl(int argc, const char *const *argv, const char *buff, int type)
[d54838d]2065{
[7d4fbcd]2066  /* if type == 1 display in a popup
2067   * if type == 2 display an admin messages
2068   * if type == 0 return output
2069   * else display in a popup
2070   */
2071  char *perlout;
2072
2073  if (argc<2) {
[ec6ff52]2074    owl_function_error("Wrong number of arguments to perl command");
[7d4fbcd]2075    return NULL;
2076  }
2077
2078  /* consume first token (argv[0]) */
2079  buff = skiptokens(buff, 1);
2080
[f1e629d]2081  perlout = owl_perlconfig_execute(buff);
[7d4fbcd]2082  if (perlout) { 
2083    if (type==1) {
2084      owl_function_popless_text(perlout);
2085    } else if (type==2) {
2086      owl_function_adminmsg(buff, perlout);
2087    } else if (type==0) {
2088      return perlout;
2089    } else {
2090      owl_function_popless_text(perlout);
2091    }
2092    owl_free(perlout);
2093  }
2094  return NULL;
2095}
2096
[5e0b690]2097/* Change the filter associated with the current view.
2098 * This also figures out which message in the new filter
2099 * should have the pointer.
2100 */
[e19eb97]2101void owl_function_change_currentview_filter(const char *filtname)
[c3ab155]2102{
2103  owl_view *v;
2104  owl_filter *f;
2105  int curid=-1, newpos, curmsg;
[c08c70a]2106  const owl_message *curm=NULL;
[c3ab155]2107
2108  v=owl_global_get_current_view(&g);
2109
2110  curmsg=owl_global_get_curmsg(&g);
2111  if (curmsg==-1) {
2112    owl_function_debugmsg("Hit the curmsg==-1 case in change_view");
2113  } else {
2114    curm=owl_view_get_element(v, curmsg);
2115    if (curm) {
2116      curid=owl_message_get_id(curm);
2117      owl_view_save_curmsgid(v, curid);
2118    }
2119  }
2120
2121  f=owl_global_get_filter(&g, filtname);
2122  if (!f) {
[ec6ff52]2123    owl_function_error("Unknown filter %s", filtname);
[c3ab155]2124    return;
2125  }
2126
2127  owl_view_new_filter(v, f);
2128
2129  /* Figure out what to set the current message to.
2130   * - If the view we're leaving has messages in it, go to the closest message
2131   *   to the last message pointed to in that view.
2132   * - If the view we're leaving is empty, try to restore the position
2133   *   from the last time we were in the new view.  */
2134  if (curm) {
2135    newpos = owl_view_get_nearest_to_msgid(v, curid);
2136  } else {
2137    newpos = owl_view_get_nearest_to_saved(v);
2138  }
2139
2140  owl_global_set_curmsg(&g, newpos);
2141  owl_function_calculate_topmsg(OWL_DIRECTION_DOWNWARDS);
2142  owl_mainwin_redisplay(owl_global_get_mainwin(&g));
2143  owl_global_set_direction_downwards(&g);
2144}
[7d4fbcd]2145
[5e0b690]2146/* Create a new filter, or replace an existing one
2147 * with a new definition.
2148 */
[e19eb97]2149void owl_function_create_filter(int argc, const char *const *argv)
[d54838d]2150{
[7d4fbcd]2151  owl_filter *f;
[9e5c9f3]2152  const owl_view *v;
[7d4fbcd]2153  int ret, inuse=0;
2154
2155  if (argc < 2) {
[ec6ff52]2156    owl_function_error("Wrong number of arguments to filter command");
[7d4fbcd]2157    return;
2158  }
2159
[3895e23]2160  owl_function_debugmsg("owl_function_create_filter: starting to create filter named %s", argv[1]);
2161
[7d4fbcd]2162  v=owl_global_get_current_view(&g);
2163
2164  /* don't touch the all filter */
2165  if (!strcmp(argv[1], "all")) {
[ec6ff52]2166    owl_function_error("You may not change the 'all' filter.");
[7d4fbcd]2167    return;
2168  }
2169
2170  /* deal with the case of trying change the filter color */
2171  if (argc==4 && !strcmp(argv[2], "-c")) {
2172    f=owl_global_get_filter(&g, argv[1]);
2173    if (!f) {
[ec6ff52]2174      owl_function_error("The filter '%s' does not exist.", argv[1]);
[7d4fbcd]2175      return;
2176    }
[601733d]2177    if (owl_util_string_to_color(argv[3])==OWL_COLOR_INVALID) {
[12c35df]2178      owl_function_error("The color '%s' is not available.", argv[3]);
2179      return;
2180    }
[8fa9562]2181    owl_filter_set_fgcolor(f, owl_util_string_to_color(argv[3]));
2182    owl_global_set_needrefresh(&g);
2183    owl_mainwin_redisplay(owl_global_get_mainwin(&g));
2184    return;
2185  }
2186  if (argc==4 && !strcmp(argv[2], "-b")) {
2187    f=owl_global_get_filter(&g, argv[1]);
2188    if (!f) {
2189      owl_function_error("The filter '%s' does not exist.", argv[1]);
2190      return;
2191    }
[601733d]2192    if (owl_util_string_to_color(argv[3])==OWL_COLOR_INVALID) {
[8fa9562]2193      owl_function_error("The color '%s' is not available.", argv[3]);
2194      return;
2195    }
2196    owl_filter_set_bgcolor(f, owl_util_string_to_color(argv[3]));
[7d4fbcd]2197    owl_global_set_needrefresh(&g);
2198    owl_mainwin_redisplay(owl_global_get_mainwin(&g));
2199    return;
2200  }
2201
2202  /* create the filter and check for errors */
2203  f=owl_malloc(sizeof(owl_filter));
2204  ret=owl_filter_init(f, argv[1], argc-2, argv+2);
2205  if (ret==-1) {
2206    owl_free(f);
[40458b9]2207    owl_function_error("Invalid filter");
[7d4fbcd]2208    return;
2209  }
2210
2211  /* if the named filter is in use by the current view, remember it */
2212  if (!strcmp(owl_view_get_filtname(v), argv[1])) {
2213    inuse=1;
2214  }
2215
2216  /* if the named filter already exists, nuke it */
2217  if (owl_global_get_filter(&g, argv[1])) {
2218    owl_global_remove_filter(&g, argv[1]);
2219  }
2220
2221  /* add the filter */
2222  owl_global_add_filter(&g, f);
2223
2224  /* if it was in use by the current view then update */
2225  if (inuse) {
[3895e23]2226    owl_function_change_currentview_filter(argv[1]);
[7d4fbcd]2227  }
2228  owl_global_set_needrefresh(&g);
2229  owl_mainwin_redisplay(owl_global_get_mainwin(&g));
2230}
2231
[3895e23]2232/* If 'filtername' does not start with 'not-' create a filter named
2233 * 'not-<filtername>' defined as "not filter <filtername>".  If the
2234 * filter 'not-<filtername>' already exists, do not overwrite it.  If
2235 * 'filtername' begins with 'not-' and a filter 'filtername' already
2236 * exists, then do nothing.  If the filter 'filtername' does not
2237 * exist, create it and define it as 'not filter <filtername>'
2238 *
2239 * Returns the name of the negated filter, which the caller must free.
2240 */
[e19eb97]2241char *owl_function_create_negative_filter(const char *filtername)
[3895e23]2242{
2243  char *newname;
[4542047]2244  const owl_filter *tmpfilt;
[e19eb97]2245  const char *argv[5];
[3895e23]2246
2247  owl_function_debugmsg("owl_function_create_negative_filter");
2248 
2249  if (!strncmp(filtername, "not-", 4)) {
2250    newname=owl_strdup(filtername+4);
2251  } else {
2252    newname=owl_sprintf("not-%s", filtername);
2253  }
2254
2255  tmpfilt=owl_global_get_filter(&g, newname);
2256  if (!tmpfilt) {
2257    argv[0]="filter"; /* anything is fine here */
2258    argv[1]=newname;
2259    argv[2]="not";
2260    argv[3]="filter";
2261    argv[4]=filtername;
2262    owl_function_create_filter(5, argv);
2263  }
2264
2265  owl_function_debugmsg("owl_function_create_negative_filter: returning with %s", newname);
2266  return(newname);
2267}
2268
[d54838d]2269void owl_function_show_filters()
2270{
[77bced3]2271  const owl_list *l;
[4542047]2272  const owl_filter *f;
[7d4fbcd]2273  int i, j;
2274  owl_fmtext fm;
2275
2276  owl_fmtext_init_null(&fm);
2277
2278  l=owl_global_get_filterlist(&g);
2279  j=owl_list_get_size(l);
2280
2281  owl_fmtext_append_bold(&fm, "Filters:\n");
2282
2283  for (i=0; i<j; i++) {
2284    f=owl_list_get_element(l, i);
2285    owl_fmtext_append_normal(&fm, "   ");
2286    if (owl_global_get_hascolors(&g)) {
[8fa9562]2287      owl_fmtext_append_normal_color(&fm, owl_filter_get_name(f), owl_filter_get_fgcolor(f), owl_filter_get_bgcolor(f));
[7d4fbcd]2288    } else {
2289      owl_fmtext_append_normal(&fm, owl_filter_get_name(f));
2290    }
2291    owl_fmtext_append_normal(&fm, "\n");
2292  }
2293  owl_function_popless_fmtext(&fm);
2294  owl_fmtext_free(&fm);
2295}
2296
[e19eb97]2297void owl_function_show_filter(const char *name)
[d54838d]2298{
[4542047]2299  const owl_filter *f;
[cdc6ff1]2300  char *buff, *tmp;
[7d4fbcd]2301
2302  f=owl_global_get_filter(&g, name);
2303  if (!f) {
[ec6ff52]2304    owl_function_error("There is no filter named %s", name);
[7d4fbcd]2305    return;
2306  }
[cdc6ff1]2307  tmp = owl_filter_print(f);
2308  buff = owl_sprintf("%s: %s", owl_filter_get_name(f), tmp);
[7d4fbcd]2309  owl_function_popless_text(buff);
[0504f63]2310  owl_free(buff);
[cdc6ff1]2311  owl_free(tmp);
[7d4fbcd]2312}
2313
[d54838d]2314void owl_function_show_zpunts()
2315{
[4542047]2316  const owl_filter *f;
[77bced3]2317  const owl_list *fl;
[7d4fbcd]2318  char buff[5000];
[0504f63]2319  char *tmp;
[7d4fbcd]2320  owl_fmtext fm;
2321  int i, j;
2322
2323  owl_fmtext_init_null(&fm);
2324
2325  fl=owl_global_get_puntlist(&g);
2326  j=owl_list_get_size(fl);
2327  owl_fmtext_append_bold(&fm, "Active zpunt filters:\n");
2328
2329  for (i=0; i<j; i++) {
2330    f=owl_list_get_element(fl, i);
[ce7b824]2331    snprintf(buff, sizeof(buff), "[% 2d] ", i+1);
2332    owl_fmtext_append_normal(&fm, buff);
[0504f63]2333    tmp = owl_filter_print(f);
2334    owl_fmtext_append_normal(&fm, tmp);
2335    owl_free(tmp);
[7d4fbcd]2336  }
2337  owl_function_popless_fmtext(&fm);
2338  owl_fmtext_free(&fm);
2339}
2340
[3abf28b]2341/* Create a filter for a class, instance if one doesn't exist.  If
2342 * instance is NULL then catch all messgaes in the class.  Returns the
2343 * name of the filter, which the caller must free.
2344 */
[e19eb97]2345char *owl_function_classinstfilt(const char *c, const char *i) 
[d54838d]2346{
[7d4fbcd]2347  owl_filter *f;
2348  char *argbuff, *filtname;
[d54838d]2349  char *tmpclass, *tmpinstance = NULL;
[7a20e4c]2350  char *class, *instance = NULL;
[7d4fbcd]2351
[7a20e4c]2352  class = owl_util_baseclass(c);
2353  if(i) {
2354    instance = owl_util_baseclass(i);
2355  }
2356
[7d4fbcd]2357  /* name for the filter */
2358  if (!instance) {
[a0e6082]2359    filtname = owl_sprintf("class-%s", class);
[7d4fbcd]2360  } else {
[a0e6082]2361    filtname = owl_sprintf("class-%s-instance-%s", class, instance);
[7d4fbcd]2362  }
[ed2412d]2363  /* downcase it */
[28ee32b]2364  {
2365    char *temp = g_utf8_strdown(filtname, -1);
2366    if (temp) {
2367      owl_free(filtname);
2368      filtname = temp;
2369    }
2370  }
[4099cf8]2371  /* turn spaces, single quotes, and double quotes into dots */
[e3d9c77]2372  owl_text_tr(filtname, ' ', '.');
[4099cf8]2373  owl_text_tr(filtname, '\'', '.');
2374  owl_text_tr(filtname, '"', '.');
[ed2412d]2375 
[7d4fbcd]2376  /* if it already exists then go with it.  This lets users override */
2377  if (owl_global_get_filter(&g, filtname)) {
[ed2412d]2378    return(filtname);
[7d4fbcd]2379  }
2380
2381  /* create the new filter */
[995eb4b]2382  tmpclass=owl_text_quote(class, OWL_REGEX_QUOTECHARS, OWL_REGEX_QUOTEWITH);
[12c35df]2383  owl_text_tr(tmpclass, ' ', '.');
[4099cf8]2384  owl_text_tr(tmpclass, '\'', '.');
2385  owl_text_tr(tmpclass, '"', '.');
[ed2412d]2386  if (instance) {
[995eb4b]2387    tmpinstance=owl_text_quote(instance, OWL_REGEX_QUOTECHARS, OWL_REGEX_QUOTEWITH);
[12c35df]2388    owl_text_tr(tmpinstance, ' ', '.');
[4099cf8]2389    owl_text_tr(tmpinstance, '\'', '.');
2390    owl_text_tr(tmpinstance, '"', '.');
[ed2412d]2391  }
[a0e6082]2392
2393  argbuff = owl_sprintf("class ^(un)*%s(\\.d)*$", tmpclass);
[d54838d]2394  if (tmpinstance) {
[a0e6082]2395    char *tmp = argbuff;
2396    argbuff = owl_sprintf("%s and ( instance ^(un)*%s(\\.d)*$ )", tmp, tmpinstance);
2397    owl_free(tmp);
[7d4fbcd]2398  }
[ed2412d]2399  owl_free(tmpclass);
[d54838d]2400  if (tmpinstance) owl_free(tmpinstance);
[7d4fbcd]2401
2402  f=owl_malloc(sizeof(owl_filter));
2403  owl_filter_init_fromstring(f, filtname, argbuff);
2404
2405  /* add it to the global list */
2406  owl_global_add_filter(&g, f);
2407
2408  owl_free(argbuff);
[be5aa09]2409  owl_free(class);
2410  if (instance) {
2411    owl_free(instance);
2412  }
[ed2412d]2413  return(filtname);
[7d4fbcd]2414}
2415
[3abf28b]2416/* Create a filter for personal zephyrs to or from the specified
2417 * zephyr user.  Includes login/logout notifications for the user.
2418 * The name of the filter will be 'user-<user>'.  If a filter already
2419 * exists with this name, no new filter will be created.  This allows
2420 * the configuration to override this function.  Returns the name of
2421 * the filter, which the caller must free.
2422 */
[e19eb97]2423char *owl_function_zuserfilt(const char *user)
[d54838d]2424{
[7d4fbcd]2425  owl_filter *f;
[1d12db24]2426  char *argbuff, *longuser, *esclonguser, *shortuser, *filtname;
[7d4fbcd]2427
2428  /* stick the local realm on if it's not there */
[4b464a4]2429  longuser=long_zuser(user);
2430  shortuser=short_zuser(user);
[7d4fbcd]2431
2432  /* name for the filter */
[9a2ae6c]2433  filtname=owl_sprintf("user-%s", shortuser);
[7d4fbcd]2434
2435  /* if it already exists then go with it.  This lets users override */
2436  if (owl_global_get_filter(&g, filtname)) {
[3abf28b]2437    return(owl_strdup(filtname));
[7d4fbcd]2438  }
2439
2440  /* create the new-internal filter */
2441  f=owl_malloc(sizeof(owl_filter));
2442
[1d12db24]2443  esclonguser = owl_text_quote(longuser, OWL_REGEX_QUOTECHARS, OWL_REGEX_QUOTEWITH);
2444
[9a2ae6c]2445  argbuff=owl_sprintf("( type ^zephyr$ and filter personal and "
2446      "( ( direction ^in$ and sender ^%1$s$ ) or ( direction ^out$ and "
2447      "recipient ^%1$s$ ) ) ) or ( ( class ^login$ ) and ( sender ^%1$s$ ) )",
[1d12db24]2448      esclonguser);
[7d4fbcd]2449
2450  owl_filter_init_fromstring(f, filtname, argbuff);
2451
2452  /* add it to the global list */
2453  owl_global_add_filter(&g, f);
2454
2455  /* free stuff */
2456  owl_free(argbuff);
2457  owl_free(longuser);
[1d12db24]2458  owl_free(esclonguser);
[7d4fbcd]2459  owl_free(shortuser);
[7360fab]2460
[ed2412d]2461  return(filtname);
[7d4fbcd]2462}
2463
[3abf28b]2464/* Create a filter for AIM IM messages to or from the specified
2465 * screenname.  The name of the filter will be 'aimuser-<user>'.  If a
2466 * filter already exists with this name, no new filter will be
2467 * created.  This allows the configuration to override this function.
2468 * Returns the name of the filter, which the caller must free.
2469 */
[e19eb97]2470char *owl_function_aimuserfilt(const char *user)
[3abf28b]2471{
2472  owl_filter *f;
2473  char *argbuff, *filtname;
[af9b92e]2474  char *escuser;
[3abf28b]2475
2476  /* name for the filter */
[9a2ae6c]2477  filtname=owl_sprintf("aimuser-%s", user);
[3abf28b]2478
2479  /* if it already exists then go with it.  This lets users override */
2480  if (owl_global_get_filter(&g, filtname)) {
2481    return(owl_strdup(filtname));
2482  }
2483
2484  /* create the new-internal filter */
2485  f=owl_malloc(sizeof(owl_filter));
2486
[af9b92e]2487  escuser = owl_text_quote(user, OWL_REGEX_QUOTECHARS, OWL_REGEX_QUOTEWITH);
2488
[9a2ae6c]2489  argbuff = owl_sprintf(
2490      "( type ^aim$ and ( ( sender ^%1$s$ and recipient ^%2$s$ ) or "
2491      "( sender ^%2$s$ and recipient ^%1$s$ ) ) )",
2492      escuser, owl_global_get_aim_screenname_for_filters(&g));
[3abf28b]2493
2494  owl_filter_init_fromstring(f, filtname, argbuff);
2495
2496  /* add it to the global list */
2497  owl_global_add_filter(&g, f);
2498
2499  /* free stuff */
2500  owl_free(argbuff);
[af9b92e]2501  owl_free(escuser);
[3abf28b]2502
2503  return(filtname);
2504}
2505
[e19eb97]2506char *owl_function_typefilt(const char *type)
[d54838d]2507{
[f73e519]2508  owl_filter *f;
[1d12db24]2509  char *argbuff, *filtname, *esctype;
[f73e519]2510
2511  /* name for the filter */
2512  filtname=owl_sprintf("type-%s", type);
2513
2514  /* if it already exists then go with it.  This lets users override */
2515  if (owl_global_get_filter(&g, filtname)) {
2516    return filtname;
2517  }
2518
2519  /* create the new-internal filter */
2520  f=owl_malloc(sizeof(owl_filter));
2521
[1d12db24]2522  esctype = owl_text_quote(type, OWL_REGEX_QUOTECHARS, OWL_REGEX_QUOTEWITH);
2523
2524  argbuff = owl_sprintf("type ^%s$", esctype);
[f73e519]2525
2526  owl_filter_init_fromstring(f, filtname, argbuff);
2527
2528  /* add it to the global list */
2529  owl_global_add_filter(&g, f);
2530
2531  /* free stuff */
2532  owl_free(argbuff);
[1d12db24]2533  owl_free(esctype);
[f73e519]2534
2535  return filtname;
2536}
2537
[7d4fbcd]2538/* If flag is 1, marks for deletion.  If flag is 0,
2539 * unmarks for deletion. */
[d54838d]2540void owl_function_delete_curview_msgs(int flag)
2541{
[9e5c9f3]2542  const owl_view *v;
[7d4fbcd]2543  int i, j;
2544
2545  v=owl_global_get_current_view(&g);
2546  j=owl_view_get_size(v);
2547  for (i=0; i<j; i++) {
2548    if (flag == 1) {
2549      owl_message_mark_delete(owl_view_get_element(v, i));
2550    } else if (flag == 0) {
2551      owl_message_unmark_delete(owl_view_get_element(v, i));
2552    }
2553  }
2554
2555  owl_function_makemsg("%i messages marked for %sdeletion", j, flag?"":"un");
2556
2557  owl_mainwin_redisplay(owl_global_get_mainwin(&g)); 
2558}
2559
[3abf28b]2560/* Create a filter based on the current message.  Returns the name of
2561 * a filter or null.  The caller must free this name.
2562 *
2563 * if the curmsg is a personal zephyr return a filter name
[e6d989f]2564 *    to the zephyr conversation with that user.
[3abf28b]2565 * If the curmsg is a zephyr class message, instance foo, recip *,
2566 *    return a filter name to the class, inst.
2567 * If the curmsg is a zephyr class message and type==0 then
2568 *    return a filter name for just the class.
2569 * If the curmsg is a zephyr class message and type==1 then
2570 *    return a filter name for the class and instance.
2571 * If the curmsg is a personal AIM message returna  filter
2572 *    name to the AIM conversation with that user
2573 */
[d54838d]2574char *owl_function_smartfilter(int type)
2575{
[9e5c9f3]2576  const owl_view *v;
[c08c70a]2577  const owl_message *m;
[4b464a4]2578  char *zperson, *filtname=NULL;
[e19eb97]2579  const char *argv[1];
[7d4fbcd]2580 
2581  v=owl_global_get_current_view(&g);
2582  m=owl_view_get_element(v, owl_global_get_curmsg(&g));
2583
[5eeea3b]2584  if (!m || owl_view_get_size(v)==0) {
[ec6ff52]2585    owl_function_error("No message selected\n");
[4b464a4]2586    return(NULL);
[7d4fbcd]2587  }
2588
[f73e519]2589  /* very simple handling of admin messages for now */
[4b464a4]2590  if (owl_message_is_type_admin(m)) {
[3abf28b]2591    return(owl_function_typefilt("admin"));
2592  }
2593
[995eb4b]2594  /* very simple handling of loopback messages for now */
2595  if (owl_message_is_type_loopback(m)) {
2596    return(owl_function_typefilt("loopback"));
2597  }
2598
[3abf28b]2599  /* aim messages */
2600  if (owl_message_is_type_aim(m)) {
2601    if (owl_message_is_direction_in(m)) {
2602      filtname=owl_function_aimuserfilt(owl_message_get_sender(m));
2603    } else if (owl_message_is_direction_out(m)) {
2604      filtname=owl_function_aimuserfilt(owl_message_get_recipient(m));
2605    }
2606    return(filtname);
[7d4fbcd]2607  }
2608
[4b464a4]2609  /* narrow personal and login messages to the sender or recip as appropriate */
[25729b2]2610  if (owl_message_is_type_zephyr(m)) {
[0ef0e8f]2611    if (owl_message_is_personal(m) || owl_message_is_loginout(m)) {
[4b464a4]2612      if (owl_message_is_direction_in(m)) {
[25729b2]2613        zperson=short_zuser(owl_message_get_sender(m));
[4b464a4]2614      } else {
[25729b2]2615        zperson=short_zuser(owl_message_get_recipient(m));
[4b464a4]2616      }
[3abf28b]2617      filtname=owl_function_zuserfilt(zperson);
[4b464a4]2618      owl_free(zperson);
2619      return(filtname);
[7d4fbcd]2620    }
2621
[25729b2]2622    /* narrow class MESSAGE, instance foo, recip * messages to class, inst */
[ce74deb]2623    if (!strcasecmp(owl_message_get_class(m), "message")) {
[25729b2]2624      filtname=owl_function_classinstfilt(owl_message_get_class(m), owl_message_get_instance(m));
2625      return(filtname);
2626    }
2627
2628    /* otherwise narrow to the class */
2629    if (type==0) {
2630      filtname=owl_function_classinstfilt(owl_message_get_class(m), NULL);
2631    } else if (type==1) {
2632      filtname=owl_function_classinstfilt(owl_message_get_class(m), owl_message_get_instance(m));
2633    }
[4b464a4]2634    return(filtname);
[7d4fbcd]2635  }
2636
[25729b2]2637  /* pass it off to perl */
2638  if(type) {
2639    argv[0] = "-i";
2640  };
2641  return owl_perlconfig_message_call_method(m, "smartfilter", type ? 1 : 0, argv);
[7d4fbcd]2642}
2643
[d54838d]2644void owl_function_smartzpunt(int type)
2645{
[d36f2cb]2646  /* Starts a zpunt command based on the current class,instance pair.
2647   * If type=0, uses just class.  If type=1, uses instance as well. */
[9e5c9f3]2648  const owl_view *v;
[c08c70a]2649  const owl_message *m;
[e19eb97]2650  const char *cmdprefix, *mclass, *minst;
[65b2173]2651  char *cmd;
[d36f2cb]2652 
2653  v=owl_global_get_current_view(&g);
2654  m=owl_view_get_element(v, owl_global_get_curmsg(&g));
2655
[5eeea3b]2656  if (!m || owl_view_get_size(v)==0) {
[ec6ff52]2657    owl_function_error("No message selected\n");
[d36f2cb]2658    return;
2659  }
2660
2661  /* for now we skip admin messages. */
[4b464a4]2662  if (owl_message_is_type_admin(m)
[5789230]2663      || owl_message_is_loginout(m)
[4b464a4]2664      || !owl_message_is_type_zephyr(m)) {
[ec6ff52]2665    owl_function_error("smartzpunt doesn't support this message type.");
[d36f2cb]2666    return;
2667  }
2668
[cee1f25]2669  mclass = owl_message_get_class(m);
2670  minst = owl_message_get_instance(m);
[d36f2cb]2671  if (!mclass || !*mclass || *mclass==' '
2672      || (!strcasecmp(mclass, "message") && !strcasecmp(minst, "personal"))
2673      || (type && (!minst || !*minst|| *minst==' '))) {
[ec6ff52]2674    owl_function_error("smartzpunt can't safely do this for <%s,%s>",
[d36f2cb]2675                         mclass, minst);
2676  } else {
2677    cmdprefix = "start-command zpunt ";
[cee1f25]2678    cmd = owl_malloc(strlen(cmdprefix)+strlen(mclass)+strlen(minst)+10);
[d36f2cb]2679    strcpy(cmd, cmdprefix);
[cee1f25]2680    strcat(cmd, owl_getquoting(mclass));
[d36f2cb]2681    strcat(cmd, mclass);
[cee1f25]2682    strcat(cmd, owl_getquoting(mclass));
[d36f2cb]2683    if (type) {
2684      strcat(cmd, " ");
[cee1f25]2685      strcat(cmd, owl_getquoting(minst));
[d36f2cb]2686      strcat(cmd, minst);
[cee1f25]2687      strcat(cmd, owl_getquoting(minst));
[d36f2cb]2688    } else {
2689      strcat(cmd, " *");
2690    }
2691    owl_function_command(cmd);
2692    owl_free(cmd);
2693  }
2694}
2695
[5e0b690]2696/* Set the color of the current view's filter to
2697 * be 'color'
2698 */
[e19eb97]2699void owl_function_color_current_filter(const char *fgcolor, const char *bgcolor)
[d54838d]2700{
[e19eb97]2701  const char *name;
[7d4fbcd]2702
2703  name=owl_view_get_filtname(owl_global_get_current_view(&g));
[8fa9562]2704  owl_function_color_filter(name, fgcolor, bgcolor);
[5e0b690]2705}
2706
2707/* Set the color of the filter 'filter' to be 'color'.  If the color
2708 * name does not exist, return -1, if the filter does not exist or is
2709 * the "all" filter, return -2.  Return 0 on success
2710 */
[e19eb97]2711int owl_function_color_filter(const char *filtname, const char *fgcolor, const char *bgcolor)
[5e0b690]2712{
2713  owl_filter *f;
2714
2715  f=owl_global_get_filter(&g, filtname);
[7d4fbcd]2716  if (!f) {
[ec6ff52]2717    owl_function_error("Unknown filter");
[5e0b690]2718    return(-2);
[7d4fbcd]2719  }
2720
2721  /* don't touch the all filter */
[5e0b690]2722  if (!strcmp(filtname, "all")) {
[ec6ff52]2723    owl_function_error("You may not change the 'all' filter.");
[5e0b690]2724    return(-2);
[7d4fbcd]2725  }
2726
[601733d]2727  if (owl_util_string_to_color(fgcolor)==OWL_COLOR_INVALID) {
[8fa9562]2728    owl_function_error("No color named '%s' avilable.", fgcolor);
[5e0b690]2729    return(-1);
[12c35df]2730  }
[8fa9562]2731
2732
2733  if (bgcolor != NULL) {
[601733d]2734    if (owl_util_string_to_color(bgcolor)==OWL_COLOR_INVALID) {
[8fa9562]2735      owl_function_error("No color named '%s' avilable.", bgcolor);
2736      return(-1);
2737    }
2738    owl_filter_set_bgcolor(f, owl_util_string_to_color(bgcolor));
2739  }
2740  owl_filter_set_fgcolor(f, owl_util_string_to_color(fgcolor));
2741 
[7d4fbcd]2742  owl_global_set_needrefresh(&g);
2743  owl_mainwin_redisplay(owl_global_get_mainwin(&g));
[5e0b690]2744  return(0);
[7d4fbcd]2745}
2746
[d54838d]2747void owl_function_show_colors()
2748{
[7d4fbcd]2749  owl_fmtext fm;
[c2c5c77]2750  int i; 
2751 
[7d4fbcd]2752  owl_fmtext_init_null(&fm);
[ca9142e]2753  owl_fmtext_append_normal(&fm, "default: ");
[8fa9562]2754  owl_fmtext_append_normal_color(&fm, "default\n", OWL_COLOR_DEFAULT, OWL_COLOR_DEFAULT);
[ca9142e]2755
2756  owl_fmtext_append_normal(&fm,"red:      ");
[8fa9562]2757  owl_fmtext_append_normal_color(&fm, "red\n", OWL_COLOR_RED, OWL_COLOR_DEFAULT);
[ca9142e]2758
2759  owl_fmtext_append_normal(&fm,"green:    ");
[8fa9562]2760  owl_fmtext_append_normal_color(&fm, "green\n", OWL_COLOR_GREEN, OWL_COLOR_DEFAULT);
[ca9142e]2761
2762  owl_fmtext_append_normal(&fm,"yellow:   ");
[8fa9562]2763  owl_fmtext_append_normal_color(&fm, "yellow\n", OWL_COLOR_YELLOW, OWL_COLOR_DEFAULT);
[ca9142e]2764
2765  owl_fmtext_append_normal(&fm,"blue:     ");
[8fa9562]2766  owl_fmtext_append_normal_color(&fm, "blue\n", OWL_COLOR_BLUE, OWL_COLOR_DEFAULT);
[ca9142e]2767
2768  owl_fmtext_append_normal(&fm,"magenta:  ");
[8fa9562]2769  owl_fmtext_append_normal_color(&fm, "magenta\n", OWL_COLOR_MAGENTA, OWL_COLOR_DEFAULT);
[ca9142e]2770
2771  owl_fmtext_append_normal(&fm,"cyan:     ");
[8fa9562]2772  owl_fmtext_append_normal_color(&fm, "cyan\n", OWL_COLOR_CYAN, OWL_COLOR_DEFAULT);
[ca9142e]2773
2774  owl_fmtext_append_normal(&fm,"white:    ");
[8fa9562]2775  owl_fmtext_append_normal_color(&fm, "white\n", OWL_COLOR_WHITE, OWL_COLOR_DEFAULT);
[7d4fbcd]2776
[c2c5c77]2777  for(i = 8; i < COLORS; ++i) {
2778    char* str1 = owl_sprintf("%4i:     ",i);
2779    char* str2 = owl_sprintf("%i\n",i);
2780    owl_fmtext_append_normal(&fm,str1);
2781    owl_fmtext_append_normal_color(&fm, str2, i, OWL_COLOR_DEFAULT);
2782    owl_free(str1);
2783     owl_free(str2);
2784  }
2785 
[7d4fbcd]2786  owl_function_popless_fmtext(&fm);
2787  owl_fmtext_free(&fm);
2788}
2789
[5bb6c21]2790/* add the given class, inst, recip to the punt list for filtering.
2791 *   if direction==0 then punt
2792 *   if direction==1 then unpunt
2793 */
[e19eb97]2794void owl_function_zpunt(const char *class, const char *inst, const char *recip, int direction)
[d54838d]2795{
[c894c15]2796  char *puntexpr, *classexpr, *instexpr, *recipexpr;
[bc08664]2797  char *quoted;
[7d4fbcd]2798
[5bb6c21]2799  if (!strcmp(class, "*")) {
[c894c15]2800    classexpr = owl_sprintf("class .*");
[5bb6c21]2801  } else {
[bc08664]2802    quoted=owl_text_quote(class, OWL_REGEX_QUOTECHARS, OWL_REGEX_QUOTEWITH);
[cee1f25]2803    owl_text_tr(quoted, ' ', '.');
[4099cf8]2804    owl_text_tr(quoted, '\'', '.');
2805    owl_text_tr(quoted, '"', '.');
[c894c15]2806    classexpr = owl_sprintf("class ^(un)*%s(\\.d)*$", quoted);
[bc08664]2807    owl_free(quoted);
[5bb6c21]2808  }
2809  if (!strcmp(inst, "*")) {
[c894c15]2810    instexpr = owl_sprintf(" and instance .*");
[7d4fbcd]2811  } else {
[bc08664]2812    quoted=owl_text_quote(inst, OWL_REGEX_QUOTECHARS, OWL_REGEX_QUOTEWITH);
[cee1f25]2813    owl_text_tr(quoted, ' ', '.');
[4099cf8]2814    owl_text_tr(quoted, '\'', '.');
2815    owl_text_tr(quoted, '"', '.');
[c894c15]2816    instexpr = owl_sprintf(" and instance ^(un)*%s(\\.d)*$", quoted);
[bc08664]2817    owl_free(quoted);
[7d4fbcd]2818  }
[c894c15]2819  if (!strcmp(recip, "*")) {
2820    recipexpr = owl_sprintf("");
2821  } else {
[e6c59ba]2822    if(!strcmp(recip, "%me%")) {
2823      recip = owl_zephyr_get_sender();
2824    }
[bc08664]2825    quoted=owl_text_quote(recip, OWL_REGEX_QUOTECHARS, OWL_REGEX_QUOTEWITH);
[cee1f25]2826    owl_text_tr(quoted, ' ', '.');
[4099cf8]2827    owl_text_tr(quoted, '\'', '.');
2828    owl_text_tr(quoted, '"', '.');
[7980fb2]2829    recipexpr = owl_sprintf(" and recipient ^%s$", quoted);
[bc08664]2830    owl_free(quoted);
[5bb6c21]2831  }
[ce7b824]2832
[c894c15]2833  puntexpr = owl_sprintf("%s %s %s", classexpr, instexpr, recipexpr);
2834  owl_function_punt(puntexpr, direction);
2835  owl_free(puntexpr);
2836  owl_free(classexpr);
2837  owl_free(instexpr);
2838  owl_free(recipexpr);
[ce7b824]2839}
2840
[e19eb97]2841void owl_function_punt(const char *filter, int direction)
[ce7b824]2842{
2843  owl_filter *f;
2844  owl_list *fl;
2845  int ret, i, j;
2846  fl=owl_global_get_puntlist(&g);
2847
2848  /* first, create the filter */
2849  f=malloc(sizeof(owl_filter));
2850
2851  owl_function_debugmsg("About to filter %s", filter);
2852  ret=owl_filter_init_fromstring(f, "punt-filter", filter);
[7d4fbcd]2853  if (ret) {
[ec6ff52]2854    owl_function_error("Error creating filter for zpunt");
[7d4fbcd]2855    owl_filter_free(f);
2856    return;
2857  }
2858
2859  /* Check for an identical filter */
2860  j=owl_list_get_size(fl);
2861  for (i=0; i<j; i++) {
2862    if (owl_filter_equiv(f, owl_list_get_element(fl, i))) {
[ce7b824]2863      owl_function_debugmsg("found an equivalent punt filter");
[7d4fbcd]2864      /* if we're punting, then just silently bow out on this duplicate */
2865      if (direction==0) {
2866        owl_filter_free(f);
2867        return;
2868      }
2869
2870      /* if we're unpunting, then remove this filter from the puntlist */
2871      if (direction==1) {
2872        owl_filter_free(owl_list_get_element(fl, i));
2873        owl_list_remove_element(fl, i);
[ce7b824]2874        owl_filter_free(f);
[7d4fbcd]2875        return;
2876      }
2877    }
2878  }
2879
[ce7b824]2880  owl_function_debugmsg("punting");
[7d4fbcd]2881  /* If we're punting, add the filter to the global punt list */
2882  if (direction==0) {
2883    owl_list_append_element(fl, f);
2884  }
2885}
2886
[e19eb97]2887void owl_function_activate_keymap(const char *keymap)
[d54838d]2888{
[7d4fbcd]2889  if (!owl_keyhandler_activate(owl_global_get_keyhandler(&g), keymap)) {
[ec6ff52]2890    owl_function_error("Unable to activate keymap '%s'", keymap);
[7d4fbcd]2891  }
2892}
2893
[d54838d]2894void owl_function_show_keymaps()
2895{
[7d4fbcd]2896  owl_list l;
2897  owl_fmtext fm;
[afa200a]2898  const owl_keymap *km;
[12bc46a]2899  const owl_keyhandler *kh;
[1aee7d9]2900  int i, numkm;
[e19eb97]2901  const char *kmname;
[7d4fbcd]2902
[1aee7d9]2903  kh = owl_global_get_keyhandler(&g);
[7d4fbcd]2904  owl_fmtext_init_null(&fm);
2905  owl_fmtext_append_bold(&fm, "Keymaps:   ");
2906  owl_fmtext_append_normal(&fm, "(use 'show keymap <name>' for details)\n");
[1aee7d9]2907  owl_keyhandler_get_keymap_names(kh, &l);
[7d4fbcd]2908  owl_fmtext_append_list(&fm, &l, "\n", owl_function_keymap_summary);
2909  owl_fmtext_append_normal(&fm, "\n");
[1aee7d9]2910
2911  numkm = owl_list_get_size(&l);
2912  for (i=0; i<numkm; i++) {
2913    kmname = owl_list_get_element(&l, i);
2914    km = owl_keyhandler_get_keymap(kh, kmname);
2915    owl_fmtext_append_bold(&fm, "\n\n----------------------------------------------------------------------------------------------------\n\n");
2916    owl_keymap_get_details(km, &fm);   
2917  }
2918  owl_fmtext_append_normal(&fm, "\n");
2919 
[7d4fbcd]2920  owl_function_popless_fmtext(&fm);
2921  owl_keyhandler_keymap_namelist_free(&l);
2922  owl_fmtext_free(&fm);
2923}
2924
[e19eb97]2925char *owl_function_keymap_summary(const char *name)
[d54838d]2926{
[afa200a]2927  const owl_keymap *km
[7d4fbcd]2928    = owl_keyhandler_get_keymap(owl_global_get_keyhandler(&g), name);
2929  if (km) return owl_keymap_summary(km);
2930  else return(NULL);
2931}
2932
2933/* TODO: implement for real */
[e19eb97]2934void owl_function_show_keymap(const char *name)
[d54838d]2935{
[1fd0b25]2936  owl_fmtext fm;
[afa200a]2937  const owl_keymap *km;
[7d4fbcd]2938
2939  owl_fmtext_init_null(&fm);
2940  km = owl_keyhandler_get_keymap(owl_global_get_keyhandler(&g), name);
2941  if (km) {
2942    owl_keymap_get_details(km, &fm);
2943  } else {
2944    owl_fmtext_append_normal(&fm, "No such keymap...\n");
2945  } 
2946  owl_function_popless_fmtext(&fm);
2947  owl_fmtext_free(&fm);
2948}
2949
[e19eb97]2950void owl_function_help_for_command(const char *cmdname)
[d54838d]2951{
[1fd0b25]2952  owl_fmtext fm;
[7d4fbcd]2953
2954  owl_fmtext_init_null(&fm);
2955  owl_cmd_get_help(owl_global_get_cmddict(&g), cmdname, &fm);
2956  owl_function_popless_fmtext(&fm); 
2957  owl_fmtext_free(&fm);
2958}
[1fd0b25]2959
[e19eb97]2960void owl_function_search_start(const char *string, int direction)
[d54838d]2961{
[ab225e0]2962  /* direction is OWL_DIRECTION_DOWNWARDS or OWL_DIRECTION_UPWARDS or
2963   * OWL_DIRECTION_NONE */
[41c9a96]2964  owl_regex re;
2965
2966  if (string && owl_regex_create_quoted(&re, string) == 0) {
2967    owl_global_set_search_re(&g, &re);
2968    owl_regex_free(&re);
2969  } else {
2970    owl_global_set_search_re(&g, NULL);
2971  }
[ab225e0]2972
2973  if (direction == OWL_DIRECTION_NONE)
2974    owl_mainwin_redisplay(owl_global_get_mainwin(&g));
2975  else
2976    owl_function_search_helper(0, direction);
[1fd0b25]2977}
2978
[d54838d]2979void owl_function_search_continue(int direction)
2980{
[1fd0b25]2981  /* direction is OWL_DIRECTION_DOWNWARDS or OWL_DIRECTION_UPWARDS */
2982  owl_function_search_helper(1, direction);
2983}
2984
[d54838d]2985void owl_function_search_helper(int mode, int direction)
2986{
[1fd0b25]2987  /* move to a message that contains the string.  If direction is
2988   * OWL_DIRECTION_DOWNWARDS then search fowards, if direction is
2989   * OWL_DIRECTION_UPWARDS then search backwards.
2990   *
2991   * If mode==0 then it will stay on the current message if it
2992   * contains the string.
2993   */
2994
[9e5c9f3]2995  const owl_view *v;
[1fd0b25]2996  int viewsize, i, curmsg, start;
2997  owl_message *m;
2998
2999  v=owl_global_get_current_view(&g);
3000  viewsize=owl_view_get_size(v);
3001  curmsg=owl_global_get_curmsg(&g);
3002 
3003  if (viewsize==0) {
[ec6ff52]3004    owl_function_error("No messages present");
[1fd0b25]3005    return;
3006  }
3007
3008  if (mode==0) {
3009    start=curmsg;
3010  } else if (direction==OWL_DIRECTION_DOWNWARDS) {
3011    start=curmsg+1;
3012  } else {
3013    start=curmsg-1;
3014  }
3015
3016  /* bounds check */
3017  if (start>=viewsize || start<0) {
[ec6ff52]3018    owl_function_error("No further matches found");
[1fd0b25]3019    return;
3020  }
3021
3022  for (i=start; i<viewsize && i>=0;) {
3023    m=owl_view_get_element(v, i);
[41c9a96]3024    if (owl_message_search(m, owl_global_get_search_re(&g))) {
[1fd0b25]3025      owl_global_set_curmsg(&g, i);
3026      owl_function_calculate_topmsg(direction);
3027      owl_mainwin_redisplay(owl_global_get_mainwin(&g));
3028      if (direction==OWL_DIRECTION_DOWNWARDS) {
3029        owl_global_set_direction_downwards(&g);
3030      <