* fix another off by one while retrieving the iterator counter params
[citadel.git] / webcit / roomops.c
blob40c8a3aab830febbbbacd8feaa5b961b97402053
1 /*
2 * $Id$
3 * Lots of different room-related operations.
4 */
6 #include "webcit.h"
7 #include "webserver.h"
8 #define MAX_FLOORS 128
9 char floorlist[MAX_FLOORS][SIZ]; /**< list of our floor names */
11 char *viewdefs[9]; /**< the different kinds of available views */
13 /** See GetFloorListHash and GetRoomListHash for info on these. Basically we pull LFLR/LKRA etc. and set up a room HashList with these keys. */
14 const char FLOOR_PARAM_NAMES[(FLOOR_PARAM_LEN + 1)][15] = {"ID",
15 "NAME",
16 "ROOMS"};
17 const char ROOM_PARAM_NAMES[(ROOM_PARAM_LEN + 1)][20] = {"NAME",
18 "FLAG",
19 "FLOOR",
20 "LISTORDER",
21 "ACL",
22 "CURVIEW",
23 "DEFVIEW",
24 "LASTCHANGE"};
25 /* Because avoiding strlen at run time is a Good Thing(TM) */
26 const int FLOOR_PARAM_NAMELEN[(FLOOR_PARAM_LEN +1)] = {2, 4, 5};
27 const int ROOM_PARAM_NAMELEN[(ROOM_PARAM_LEN +1)] = {4, 4, 5, 9, 3, 7, 7, 8};
29 void display_whok(void);
32 * Initialize the viewdefs with localized strings
34 void initialize_viewdefs(void) {
35 viewdefs[0] = _("Bulletin Board");
36 viewdefs[1] = _("Mail Folder");
37 viewdefs[2] = _("Address Book");
38 viewdefs[3] = _("Calendar");
39 viewdefs[4] = _("Task List");
40 viewdefs[5] = _("Notes List");
41 viewdefs[6] = _("Wiki");
42 viewdefs[7] = _("Calendar List");
43 viewdefs[8] = _("Journal");
47 * Determine which views are allowed as the default for creating a new room.
49 int is_view_allowed_as_default(int which_view)
51 switch(which_view) {
52 case VIEW_BBS: return(1);
53 case VIEW_MAILBOX: return(1);
54 case VIEW_ADDRESSBOOK: return(1);
55 case VIEW_CALENDAR: return(1);
56 case VIEW_TASKS: return(1);
57 case VIEW_NOTES: return(1);
59 #ifdef TECH_PREVIEW
60 case VIEW_WIKI: return(1);
61 #else /* TECH_PREVIEW */
62 case VIEW_WIKI: return(0); /* because it isn't finished yet */
63 #endif /* TECH_PREVIEW */
65 case VIEW_CALBRIEF: return(0);
66 case VIEW_JOURNAL: return(0);
67 default: return(0); /* should never get here */
73 * load the list of floors
75 void load_floorlist(void)
77 int a;
78 char buf[SIZ];
80 for (a = 0; a < MAX_FLOORS; ++a)
81 floorlist[a][0] = 0;
83 serv_puts("LFLR");
84 serv_getln(buf, sizeof buf);
85 if (buf[0] != '1') {
86 strcpy(floorlist[0], "Main Floor");
87 return;
89 while (serv_getln(buf, sizeof buf), strcmp(buf, "000")) {
90 extract_token(floorlist[extract_int(buf, 0)], buf, 1, '|', sizeof floorlist[0]);
96 * Free a session's march list
98 void free_march_list(wcsession *wcf)
100 struct march *mptr;
102 while (wcf->march != NULL) {
103 mptr = wcf->march->next;
104 free(wcf->march);
105 wcf->march = mptr;
113 * remove a room from the march list
115 void remove_march(const StrBuf *aaa)
117 struct march *mptr, *mptr2;
119 if (WC->march == NULL)
120 return;
122 if (!strcasecmp(WC->march->march_name, ChrPtr(aaa))) {
123 mptr = WC->march->next;
124 free(WC->march);
125 WC->march = mptr;
126 return;
128 mptr2 = WC->march;
129 for (mptr = WC->march; mptr != NULL; mptr = mptr->next) {
130 if (!strcasecmp(mptr->march_name, ChrPtr(aaa))) {
131 mptr2->next = mptr->next;
132 free(mptr);
133 mptr = mptr2;
134 } else {
135 mptr2 = mptr;
144 * display rooms in tree structure
146 void room_tree_list(struct roomlisting *rp)
148 char rmname[64];
149 int f;
151 if (rp == NULL) {
152 return;
155 room_tree_list(rp->lnext);
157 strcpy(rmname, rp->rlname);
158 f = rp->rlflags;
160 wprintf("<a href=\"dotgoto&room=");
161 urlescputs(rmname);
162 wprintf("\"");
163 wprintf(">");
164 escputs1(rmname, 1, 1);
165 if ((f & QR_DIRECTORY) && (f & QR_NETWORK))
166 wprintf("}");
167 else if (f & QR_DIRECTORY)
168 wprintf("]");
169 else if (f & QR_NETWORK)
170 wprintf(")");
171 else
172 wprintf("&gt;");
173 wprintf("</a><tt> </tt>\n");
175 room_tree_list(rp->rnext);
176 free(rp);
180 /**
181 * \brief Room ordering stuff (compare first by floor, then by order)
182 * \param r1 first roomlist to compare
183 * \param r2 second roomlist co compare
184 * \return are they the same???
186 int rordercmp(struct roomlisting *r1, struct roomlisting *r2)
188 if ((r1 == NULL) && (r2 == NULL))
189 return (0);
190 if (r1 == NULL)
191 return (-1);
192 if (r2 == NULL)
193 return (1);
194 if (r1->rlfloor < r2->rlfloor)
195 return (-1);
196 if (r1->rlfloor > r2->rlfloor)
197 return (1);
198 if (r1->rlorder < r2->rlorder)
199 return (-1);
200 if (r1->rlorder > r2->rlorder)
201 return (1);
202 return (0);
207 * \brief Common code for all room listings
208 * \param variety what???
210 void listrms(char *variety)
212 char buf[SIZ];
213 int num_rooms = 0;
215 struct roomlisting *rl = NULL;
216 struct roomlisting *rp;
217 struct roomlisting *rs;
219 /** Ask the server for a room list */
220 serv_puts(variety);
221 serv_getln(buf, sizeof buf);
222 if (buf[0] != '1') {
223 wprintf("&nbsp;");
224 return;
227 while (serv_getln(buf, sizeof buf), strcmp(buf, "000")) {
228 ++num_rooms;
229 rp = malloc(sizeof(struct roomlisting));
230 extract_token(rp->rlname, buf, 0, '|', sizeof rp->rlname);
231 rp->rlflags = extract_int(buf, 1);
232 rp->rlfloor = extract_int(buf, 2);
233 rp->rlorder = extract_int(buf, 3);
234 rp->lnext = NULL;
235 rp->rnext = NULL;
237 rs = rl;
238 if (rl == NULL) {
239 rl = rp;
240 } else
241 while (rp != NULL) {
242 if (rordercmp(rp, rs) < 0) {
243 if (rs->lnext == NULL) {
244 rs->lnext = rp;
245 rp = NULL;
246 } else {
247 rs = rs->lnext;
249 } else {
250 if (rs->rnext == NULL) {
251 rs->rnext = rp;
252 rp = NULL;
253 } else {
254 rs = rs->rnext;
260 room_tree_list(rl);
263 * If no rooms were listed, print an nbsp to make the cell
264 * borders show up anyway.
266 if (num_rooms == 0) wprintf("&nbsp;");
271 * \brief list all forgotten rooms
273 void zapped_list(void)
275 WCTemplputParams SubTP;
276 StrBuf *Buf;
278 output_headers(1, 1, 1, 0, 0, 0);
279 memset(&SubTP, 0, sizeof(WCTemplputParams));
280 Buf = NewStrBufPlain(_("Zapped (forgotten) rooms"), -1);
281 SubTP.Filter.ContextType = CTX_STRBUF;
282 SubTP.Context = Buf;
283 DoTemplate(HKEY("beginbox"), NULL, &SubTP);
285 FreeStrBuf(&Buf);
287 listrms("LZRM -1");
289 wprintf("<br /><br />\n");
290 wprintf(_("Click on any room to un-zap it and goto that room.\n"));
291 do_template("endbox", NULL);
292 wDumpContent(1);
297 * \brief read this room's info file (set v to 1 for verbose mode)
299 void readinfo(StrBuf *Target, WCTemplputParams *TP)
301 char buf[256];
302 char briefinfo[128];
303 char fullinfo[8192];
304 int fullinfo_len = 0;
306 serv_puts("RINF");
307 serv_getln(buf, sizeof buf);
308 if (buf[0] == '1') {
310 while (serv_getln(buf, sizeof buf), strcmp(buf, "000")) {
311 if (fullinfo_len < (sizeof fullinfo - sizeof buf)) {
312 strcpy(&fullinfo[fullinfo_len], buf);
313 fullinfo_len += strlen(buf);
317 safestrncpy(briefinfo, fullinfo, sizeof briefinfo);
318 strcpy(&briefinfo[50], "...");
320 wprintf("<div class=\"infos\" "
321 "onclick=\"javascript:Effect.Appear('room_infos', { duration: 0.5 });\" "
322 ">");
323 escputs(briefinfo);
324 wprintf("</div><div id=\"room_infos\" style=\"display:none;\">");
325 wprintf("<img class=\"close_infos\" "
326 "onclick=\"javascript:Effect.Fade('room_infos', { duration: 0.5 });\" "
327 "src=\"static/closewindow.gif\" alt=\"%s\">",
328 _("Close window")
330 escputs(fullinfo);
331 wprintf("</div>");
333 else {
334 wprintf("&nbsp;");
342 * \brief Display room banner icon.
343 * The server doesn't actually
344 * need the room name, but we supply it in order to
345 * keep the browser from using a cached icon from
346 * another room.
348 void embed_room_graphic(StrBuf *Target, WCTemplputParams *TP)
350 char buf[SIZ];
352 serv_puts("OIMG _roompic_");
353 serv_getln(buf, sizeof buf);
354 if (buf[0] == '2') {
355 wprintf("<img height=\"64px\" src=\"image&name=_roompic_&room=");
356 urlescputs(ChrPtr(WC->wc_roomname));
357 wprintf("\">");
358 serv_puts("CLOS");
359 serv_getln(buf, sizeof buf);
361 else if (WC->wc_view == VIEW_ADDRESSBOOK) {
362 wprintf("<img class=\"roompic\" alt=\"\" src=\""
363 "static/viewcontacts_48x.gif"
364 "\">"
367 else if ( (WC->wc_view == VIEW_CALENDAR) || (WC->wc_view == VIEW_CALBRIEF) ) {
368 wprintf("<img class=\"roompic\" alt=\"\" src=\""
369 "static/calarea_48x.gif"
370 "\">"
373 else if (WC->wc_view == VIEW_TASKS) {
374 wprintf("<img class=\"roompic\" alt=\"\" src=\""
375 "static/taskmanag_48x.gif"
376 "\">"
379 else if (WC->wc_view == VIEW_NOTES) {
380 wprintf("<img class=\"roompic\" alt=\"\" src=\""
381 "static/storenotes_48x.gif"
382 "\">"
385 else if (WC->wc_view == VIEW_MAILBOX) {
386 wprintf("<img class=\"roompic\" alt=\"\" src=\""
387 "static/privatemess_48x.gif"
388 "\">"
391 else {
392 wprintf("<img class=\"roompic\" alt=\"\" src=\""
393 "static/chatrooms_48x.gif"
394 "\">"
403 * \brief Display the current view and offer an option to change it
405 void embed_view_o_matic(StrBuf *Target, WCTemplputParams *TP)
407 int i;
409 wprintf("<form name=\"viewomatic\" action=\"changeview\">\n");
410 wprintf("\t<div style=\"display: inline;\">\n\t<input type=\"hidden\" name=\"nonce\" value=\"%d\">\n", WC->nonce);
411 wprintf("<label for=\"view_name\">");
412 wprintf(_("View as:"));
413 wprintf("</label> "
414 "<select name=\"newview\" size=\"1\" "
415 "id=\"view_name\" class=\"selectbox\" "
416 "OnChange=\"location.href=viewomatic.newview.options"
417 "[selectedIndex].value\">\n");
419 for (i=0; i<(sizeof viewdefs / sizeof (char *)); ++i) {
421 * Only offer the views that make sense, given the default
422 * view for the room. For example, don't offer a Calendar
423 * view in a non-Calendar room.
425 if (
426 (i == WC->wc_view)
427 || (i == WC->wc_default_view) /**< default */
428 || ( (i == 0) && (WC->wc_default_view == 1) ) /**< mail or bulletin */
429 || ( (i == 1) && (WC->wc_default_view == 0) ) /**< mail or bulletin */
430 /** || ( (i == 7) && (WC->wc_default_view == 3) ) (calendar list temporarily disabled) */
433 wprintf("<option %s value=\"changeview?view=%d\">",
434 ((i == WC->wc_view) ? "selected" : ""),
435 i );
436 escputs(viewdefs[i]);
437 wprintf("</option>\n");
440 wprintf("</select></div></form>\n");
445 * \brief Display a search box
447 void embed_search_o_matic(StrBuf *Target, WCTemplputParams *TP)
449 wprintf("<form name=\"searchomatic\" action=\"do_search\">\n");
450 wprintf("<div style=\"display: inline;\"><input type=\"hidden\" name=\"nonce\" value=\"%d\">\n", WC->nonce);
451 wprintf("<label for=\"srchquery\">");
452 wprintf(_("Search: "));
453 wprintf("</label><input ");
454 wprintf("%s", serv_info.serv_fulltext_enabled ? "" : "disabled ");
455 wprintf("type=\"text\" name=\"query\" id=\"srchquery\" size=\"15\" maxlength=\"128\" class=\"inputbox\">\n"
457 wprintf("</div></form>\n");
462 * \brief Embed the room banner
464 * \param got The information returned from a GOTO server command
465 * \param navbar_style Determines which navigation buttons to display
469 void embed_room_banner(char *got, int navbar_style) {
470 char buf[256];
471 char buf2[1024];
472 char with_files[256];
473 int file_count=0;
476 * We need to have the information returned by a GOTO server command.
477 * If it isn't supplied, we fake it by issuing our own GOTO.
479 if (got == NULL) {
480 memset(buf, '0', 20);
481 buf[20] = '\0';
482 serv_printf("GOTO %s", ChrPtr(WC->wc_roomname));
483 serv_getln(buf, sizeof buf);
484 got = buf;
487 /** The browser needs some information for its own use */
488 wprintf("<script type=\"text/javascript\"> \n"
489 " room_is_trash = %d; \n"
490 "</script>\n",
491 WC->wc_is_trash
495 * If the user happens to select the "make this my start page" link,
496 * we want it to remember the URL as a "/dotskip" one instead of
497 * a "skip" or "gotonext" or something like that.
499 if (WC->this_page == NULL)
500 WC->this_page = NewStrBuf();
501 StrBufPrintf(WC->this_page,
502 "dotskip&room=%s",
503 ChrPtr(WC->wc_roomname));
505 /** Check for new mail. */
506 WC->new_mail = extract_int(&got[4], 9);
507 WC->wc_view = extract_int(&got[4], 11);
509 /* Is this a directory room and does it contain files and how many? */
510 if ((WC->room_flags & QR_DIRECTORY) && (WC->room_flags & QR_VISDIR))
512 serv_puts("RDIR");
513 serv_getln(buf2, sizeof buf2);
514 if (buf2[0] == '1') while (serv_getln(buf2, sizeof buf2), strcmp(buf2, "000"))
515 file_count++;
516 snprintf (with_files, sizeof with_files,
517 "; <a href=\"do_template?template=files\"> %d %s </a>",
518 file_count,
519 ((file_count>1) || (file_count == 0) ? _("files") : _("file")));
521 else
522 strcpy (with_files, "");
524 svprintf(HKEY("NUMMSGS"), WCS_STRING,
525 _("%d new of %d messages%s"),
526 extract_int(&got[4], 1),
527 extract_int(&got[4], 2),
528 with_files
530 svcallback("ROOMPIC", embed_room_graphic);
531 svcallback("ROOMINFO", readinfo);
532 svcallback("VIEWOMATIC", embed_view_o_matic);
533 svcallback("SEARCHOMATIC", embed_search_o_matic);
534 svcallback("START", offer_start_page);
536 do_template("roombanner", NULL);
537 /* roombanner contains this for mobile */
538 if (navbar_style != navbar_none && !WC->is_mobile) {
540 wprintf("<div id=\"navbar\"><ul>");
542 if (navbar_style == navbar_default) wprintf(
543 "<li class=\"ungoto\">"
544 "<a href=\"ungoto\">"
545 "<img src=\"static/ungoto2_24x.gif\" alt=\"\">"
546 "<span class=\"navbar_link\">%s</span></A>"
547 "</li>\n", _("Ungoto")
550 if ( (navbar_style == navbar_default) && (WC->wc_view == VIEW_BBS) ) {
551 wprintf(
552 "<li class=\"newmess\">"
553 "<a href=\"readnew\">"
554 "<img src=\"static/newmess2_24x.gif\" alt=\"\">"
555 "<span class=\"navbar_link\">%s</span></A>"
556 "</li>\n", _("Read new messages")
560 if (navbar_style == navbar_default) {
561 switch(WC->wc_view) {
562 case VIEW_ADDRESSBOOK:
563 wprintf(
564 "<li class=\"viewcontacts\">"
565 "<a href=\"readfwd\">"
566 "<img src=\"static/viewcontacts_24x.gif\" "
567 "alt=\"\">"
568 "<span class=\"navbar_link\">"
569 "%s"
570 "</span></a></li>\n", _("View contacts")
572 break;
573 case VIEW_CALENDAR:
574 wprintf(
575 "<li class=\"staskday\">"
576 "<a href=\"readfwd?calview=day\">"
577 "<img src=\"static/taskday2_24x.gif\" "
578 "alt=\"\">"
579 "<span class=\"navbar_link\">"
580 "%s"
581 "</span></a></li>\n", _("Day view")
583 wprintf(
584 "<li class=\"monthview\">"
585 "<a href=\"readfwd?calview=month\">"
586 "<img src=\"static/monthview2_24x.gif\" "
587 "alt=\"\">"
588 "<span class=\"navbar_link\">"
589 "%s"
590 "</span></a></li>\n", _("Month view")
592 break;
593 case VIEW_CALBRIEF:
594 wprintf(
595 "<li class=\"monthview\">"
596 "<a href=\"readfwd?calview=month\">"
597 "<img src=\"static/monthview2_24x.gif\" "
598 "alt=\"\">"
599 "<span class=\"navbar_link\">"
600 "%s"
601 "</span></a></li>\n", _("Calendar list")
603 break;
604 case VIEW_TASKS:
605 wprintf(
606 "<li class=\"taskmanag\">"
607 "<a href=\"readfwd\">"
608 "<img src=\"static/taskmanag_24x.gif\" "
609 "alt=\"\">"
610 "<span class=\"navbar_link\">"
611 "%s"
612 "</span></a></li>\n", _("View tasks")
614 break;
615 case VIEW_NOTES:
616 wprintf(
617 "<li class=\"viewnotes\">"
618 "<a href=\"readfwd\">"
619 "<img src=\"static/viewnotes_24x.gif\" "
620 "alt=\"\">"
621 "<span class=\"navbar_link\">"
622 "%s"
623 "</span></a></li>\n", _("View notes")
625 break;
626 case VIEW_MAILBOX:
627 wprintf(
628 "<li class=\"readallmess\">"
629 "<a id=\"m_refresh\" href=\"readfwd\">"
630 "<img src=\"static/readallmess3_24x.gif\" "
631 "alt=\"\">"
632 "<span class=\"navbar_link\">"
633 "%s"
634 "</span></a></li>\n", _("Refresh message list")
636 break;
637 case VIEW_WIKI:
638 wprintf(
639 "<li class=\"readallmess\">"
640 "<a href=\"readfwd\">"
641 "<img src=\"static/readallmess3_24x.gif\" "
642 "alt=\"\">"
643 "<span class=\"navbar_link\">"
644 "%s"
645 "</span></a></li>\n", _("Wiki home")
647 break;
648 default:
649 wprintf(
650 "<li class=\"readallmess\">"
651 "<a href=\"readfwd\">"
652 "<img src=\"static/readallmess3_24x.gif\" "
653 "alt=\"\">"
654 "<span class=\"navbar_link\">"
655 "%s"
656 "</span></a></li>\n", _("Read all messages")
658 break;
662 if (navbar_style == navbar_default) {
663 switch(WC->wc_view) {
664 case VIEW_ADDRESSBOOK:
665 wprintf(
666 "<li class=\"addnewcontact\">"
667 "<a href=\"display_enter\">"
668 "<img src=\"static/addnewcontact_24x.gif\" "
669 "alt=\"\"><span class=\"navbar_link\">"
670 "%s"
671 "</span></a></li>\n", _("Add new contact")
673 break;
674 case VIEW_CALENDAR:
675 case VIEW_CALBRIEF:
676 wprintf("<li class=\"addevent\"><a href=\"display_enter");
677 if (havebstr("year" )) wprintf("?year=%s", bstr("year"));
678 if (havebstr("month")) wprintf("?month=%s", bstr("month"));
679 if (havebstr("day" )) wprintf("?day=%s", bstr("day"));
680 wprintf("\">"
681 "<img src=\"static/addevent_24x.gif\" "
682 "alt=\"\"><span class=\"navbar_link\">"
683 "%s"
684 "</span></a></li>\n", _("Add new event")
686 break;
687 case VIEW_TASKS:
688 wprintf(
689 "<li class=\"newmess\">"
690 "<a href=\"display_enter\">"
691 "<img src=\"static/newmess3_24x.gif\" "
692 "alt=\"\"><span class=\"navbar_link\">"
693 "%s"
694 "</span></a></li>\n", _("Add new task")
696 break;
697 case VIEW_NOTES:
698 wprintf(
699 "<li class=\"enternewnote\">"
700 "<a href=\"add_new_note\">"
701 "<img src=\"static/enternewnote_24x.gif\" "
702 "alt=\"\"><span class=\"navbar_link\">"
703 "%s"
704 "</span></a></li>\n", _("Add new note")
706 break;
707 case VIEW_WIKI:
708 safestrncpy(buf, bstr("page"), sizeof buf);
709 str_wiki_index(buf);
710 wprintf(
711 "<li class=\"newmess\">"
712 "<a href=\"display_enter?wikipage=%s\">"
713 "<img src=\"static/newmess3_24x.gif\" "
714 "alt=\"\"><span class=\"navbar_link\">"
715 "%s"
716 "</span></a></li>\n", buf, _("Edit this page")
718 break;
719 case VIEW_MAILBOX:
720 wprintf(
721 "<li class=\"newmess\">"
722 "<a href=\"display_enter\">"
723 "<img src=\"static/newmess3_24x.gif\" "
724 "alt=\"\"><span class=\"navbar_link\">"
725 "%s"
726 "</span></a></li>\n", _("Write mail")
728 break;
729 default:
730 wprintf(
731 "<li class=\"newmess\">"
732 "<a href=\"display_enter\">"
733 "<img src=\"static/newmess3_24x.gif\" "
734 "alt=\"\"><span class=\"navbar_link\">"
735 "%s"
736 "</span></a></li>\n", _("Enter a message")
738 break;
742 if (navbar_style == navbar_default) wprintf(
743 "<li class=\"skipthisroom\">"
744 "<a href=\"skip\" "
745 "title=\"%s\">"
746 "<img src=\"static/skipthisroom_24x.gif\" alt=\"\">"
747 "<span class=\"navbar_link\">%s</span></a>"
748 "</li>\n",
749 _("Leave all messages marked as unread, go to next room with unread messages"),
750 _("Skip this room")
753 if (navbar_style == navbar_default) wprintf(
754 "<li class=\"markngo\">"
755 "<a href=\"gotonext\" "
756 "title=\"%s\">"
757 "<img src=\"static/markngo_24x.gif\" alt=\"\">"
758 "<span class=\"navbar_link\">%s</span></a>"
759 "</li>\n",
760 _("Mark all messages as read, go to next room with unread messages"),
761 _("Goto next room")
764 wprintf("</ul></div>\n");
771 * back end routine to take the session to a new room
773 long gotoroom(const StrBuf *gname)
775 StrBuf *Buf;
776 static long ls = (-1L);
777 long err = 0;
779 /* store ungoto information */
780 strcpy(WC->ugname, ChrPtr(WC->wc_roomname));
781 WC->uglsn = ls;
782 Buf = NewStrBuf();
783 /** move to the new room */
784 serv_printf("GOTO %s", ChrPtr(gname));
785 StrBuf_ServGetln(Buf);
786 if (GetServerStatus(Buf, &err) != 2) {
787 serv_puts("GOTO _BASEROOM_");
788 StrBuf_ServGetln(Buf);
789 if (GetServerStatus(Buf, &err) != 2) {
790 FreeStrBuf(&Buf);
791 return err;
795 if (WC->wc_roomname == NULL)
796 WC->wc_roomname = NewStrBuf();
797 else
798 FlushStrBuf(WC->wc_roomname);
800 StrBufExtract_token(WC->wc_roomname, Buf, 0, '|');
801 StrBufCutLeft(WC->wc_roomname, 4);
802 WC->room_flags = StrBufExtract_int(Buf, 4, '|');
803 /* highest_msg_read = extract_int(&buf[4],6);
804 maxmsgnum = extract_int(&buf[4],5);
806 WC->is_mailbox = StrBufExtract_int(Buf, 7, '|');
807 ls = StrBufExtract_long(Buf, 6, '|');
808 WC->wc_floor = StrBufExtract_int(Buf, 10, '|');
809 WC->wc_view = StrBufExtract_int(Buf, 11, '|');
810 WC->wc_default_view = StrBufExtract_int(Buf, 12, '|');
811 WC->wc_is_trash = StrBufExtract_int(Buf, 13, '|');
812 WC->room_flags2 = StrBufExtract_int(Buf, 14, '|');
814 if (WC->is_aide)
815 WC->is_room_aide = WC->is_aide;
816 else
817 WC->is_room_aide = (char) StrBufExtract_int(Buf, 8, '|');
819 remove_march(WC->wc_roomname);
820 if (!strcasecmp(ChrPtr(gname), "_BASEROOM_"))
821 remove_march(gname);
822 FreeStrBuf(&Buf);
824 return err;
829 * \brief Locate the room on the march list which we most want to go to.
830 * Each room
831 * is measured given a "weight" of preference based on various factors.
832 * \param desired_floor the room number on the citadel server
833 * \return the roomname
835 char *pop_march(int desired_floor)
837 static char TheRoom[128];
838 int TheFloor = 0;
839 int TheOrder = 32767;
840 int TheWeight = 0;
841 int weight;
842 struct march *mptr = NULL;
844 strcpy(TheRoom, "_BASEROOM_");
845 if (WC->march == NULL)
846 return (TheRoom);
848 for (mptr = WC->march; mptr != NULL; mptr = mptr->next) {
849 weight = 0;
850 if ((strcasecmp(mptr->march_name, "_BASEROOM_")))
851 weight = weight + 10000;
852 if (mptr->march_floor == desired_floor)
853 weight = weight + 5000;
855 weight = weight + ((128 - (mptr->march_floor)) * 128);
856 weight = weight + (128 - (mptr->march_order));
858 if (weight > TheWeight) {
859 TheWeight = weight;
860 strcpy(TheRoom, mptr->march_name);
861 TheFloor = mptr->march_floor;
862 TheOrder = mptr->march_order;
865 return (TheRoom);
871 * Goto next room having unread messages.
873 * We want to skip over rooms that the user has already been to, and take the
874 * user back to the lobby when done. The room we end up in is placed in
875 * newroom - which is set to 0 (the lobby) initially.
876 * We start the search in the current room rather than the beginning to prevent
877 * two or more concurrent users from dragging each other back to the same room.
879 void gotonext(void)
881 char buf[256];
882 struct march *mptr = NULL;
883 struct march *mptr2 = NULL;
884 char room_name[128];
885 StrBuf *next_room;
886 int ELoop = 0;
889 * First check to see if the march-mode list is already allocated.
890 * If it is, pop the first room off the list and go there.
893 if (WC->march == NULL) {
894 serv_puts("LKRN");
895 serv_getln(buf, sizeof buf);
896 if (buf[0] == '1')
897 while (serv_getln(buf, sizeof buf), strcmp(buf, "000")) {
898 if (IsEmptyStr(buf)) {
899 if (ELoop > 10000)
900 return;
901 if (ELoop % 100 == 0)
902 sleeeeeeeeeep(1);
903 ELoop ++;
904 continue;
906 extract_token(room_name, buf, 0, '|', sizeof room_name);
907 if (strcasecmp(room_name, ChrPtr(WC->wc_roomname))) {
908 mptr = (struct march *) malloc(sizeof(struct march));
909 mptr->next = NULL;
910 safestrncpy(mptr->march_name, room_name, sizeof mptr->march_name);
911 mptr->march_floor = extract_int(buf, 2);
912 mptr->march_order = extract_int(buf, 3);
913 if (WC->march == NULL)
914 WC->march = mptr;
915 else
916 mptr2->next = mptr;
917 mptr2 = mptr;
919 buf[0] = '\0';
922 * add _BASEROOM_ to the end of the march list, so the user will end up
923 * in the system base room (usually the Lobby>) at the end of the loop
925 mptr = (struct march *) malloc(sizeof(struct march));
926 mptr->next = NULL;
927 mptr->march_order = 0;
928 mptr->march_floor = 0;
929 strcpy(mptr->march_name, "_BASEROOM_");
930 if (WC->march == NULL) {
931 WC->march = mptr;
932 } else {
933 mptr2 = WC->march;
934 while (mptr2->next != NULL)
935 mptr2 = mptr2->next;
936 mptr2->next = mptr;
939 * ...and remove the room we're currently in, so a <G>oto doesn't make us
940 * walk around in circles
942 remove_march(WC->wc_roomname);
944 if (WC->march != NULL) {
945 next_room = NewStrBufPlain(pop_march(-1), -1);/*TODO: migrate march to strbuf */
946 } else {
947 next_room = NewStrBufPlain(HKEY("_BASEROOM_"));
951 smart_goto(next_room);
952 FreeStrBuf(&next_room);
957 * goto next room
959 void smart_goto(const StrBuf *next_room) {
960 gotoroom(next_room);
961 readloop(readnew);
967 * mark all messages in current room as having been read
969 void slrp_highest(void)
971 char buf[256];
973 serv_puts("SLRP HIGHEST");
974 serv_getln(buf, sizeof buf);
979 * un-goto the previous room
981 void ungoto(void)
983 StrBuf *Buf;
985 if (!strcmp(WC->ugname, "")) {
986 smart_goto(WC->wc_roomname);
987 return;
989 serv_printf("GOTO %s", WC->ugname);
990 Buf = NewStrBuf();
991 StrBuf_ServGetln(Buf);
992 if (GetServerStatus(Buf, NULL) != 2) {
993 smart_goto(WC->wc_roomname);
994 FreeStrBuf(&Buf);
995 return;
997 if (WC->uglsn >= 0L) {
998 serv_printf("SLRP %ld", WC->uglsn);
999 StrBuf_ServGetln(Buf);
1001 FlushStrBuf(Buf);
1002 StrBufAppendBufPlain(Buf, WC->ugname, -1, 0);
1003 strcpy(WC->ugname, "");
1004 smart_goto(Buf);
1005 FreeStrBuf(&Buf);
1008 typedef struct __room_states {
1009 char password[SIZ];
1010 char dirname[SIZ];
1011 char name[SIZ];
1012 int flags;
1013 int floor;
1014 int order;
1015 int view;
1016 int flags2;
1017 } room_states;
1023 * Set/clear/read the "self-service list subscribe" flag for a room
1025 * set newval to 0 to clear, 1 to set, any other value to leave unchanged.
1026 * returns the new value.
1029 int self_service(int newval) {
1030 int current_value = 0;
1031 char buf[SIZ];
1033 char name[SIZ];
1034 char password[SIZ];
1035 char dirname[SIZ];
1036 int flags, floor, order, view, flags2;
1038 serv_puts("GETR");
1039 serv_getln(buf, sizeof buf);
1040 if (buf[0] != '2') return(0);
1042 extract_token(name, &buf[4], 0, '|', sizeof name);
1043 extract_token(password, &buf[4], 1, '|', sizeof password);
1044 extract_token(dirname, &buf[4], 2, '|', sizeof dirname);
1045 flags = extract_int(&buf[4], 3);
1046 floor = extract_int(&buf[4], 4);
1047 order = extract_int(&buf[4], 5);
1048 view = extract_int(&buf[4], 6);
1049 flags2 = extract_int(&buf[4], 7);
1051 if (flags2 & QR2_SELFLIST) {
1052 current_value = 1;
1054 else {
1055 current_value = 0;
1058 if (newval == 1) {
1059 flags2 = flags2 | QR2_SELFLIST;
1061 else if (newval == 0) {
1062 flags2 = flags2 & ~QR2_SELFLIST;
1064 else {
1065 return(current_value);
1068 if (newval != current_value) {
1069 serv_printf("SETR %s|%s|%s|%d|0|%d|%d|%d|%d",
1070 name, password, dirname, flags,
1071 floor, order, view, flags2);
1072 serv_getln(buf, sizeof buf);
1075 return(newval);
1079 int is_selflist(room_states *RoomFlags)
1081 return ((RoomFlags->flags2 & QR2_SELFLIST) != 0);
1084 int is_publiclist(room_states *RoomFlags)
1086 return ((RoomFlags->flags2 & QR2_SMTP_PUBLIC) != 0);
1089 int is_moderatedlist(room_states *RoomFlags)
1091 return ((RoomFlags->flags2 & QR2_MODERATED) != 0);
1095 * Set/clear/read the "self-service list subscribe" flag for a room
1097 * set newval to 0 to clear, 1 to set, any other value to leave unchanged.
1098 * returns the new value.
1101 int get_roomflags(room_states *RoomOps)
1103 char buf[SIZ];
1105 serv_puts("GETR");
1106 serv_getln(buf, sizeof buf);
1107 if (buf[0] != '2') return(0);
1109 extract_token(RoomOps->name, &buf[4], 0, '|', sizeof RoomOps->name);
1110 extract_token(RoomOps->password, &buf[4], 1, '|', sizeof RoomOps->password);
1111 extract_token(RoomOps->dirname, &buf[4], 2, '|', sizeof RoomOps->dirname);
1112 RoomOps->flags = extract_int(&buf[4], 3);
1113 RoomOps->floor = extract_int(&buf[4], 4);
1114 RoomOps->order = extract_int(&buf[4], 5);
1115 RoomOps->view = extract_int(&buf[4], 6);
1116 RoomOps->flags2 = extract_int(&buf[4], 7);
1117 return (1);
1120 int set_roomflags(room_states *RoomOps)
1122 char buf[SIZ];
1124 serv_printf("SETR %s|%s|%s|%d|0|%d|%d|%d|%d",
1125 RoomOps->name,
1126 RoomOps->password,
1127 RoomOps->dirname,
1128 RoomOps->flags,
1129 RoomOps->floor,
1130 RoomOps->order,
1131 RoomOps->view,
1132 RoomOps->flags2);
1133 serv_getln(buf, sizeof buf);
1134 return (1);
1143 * display the form for editing a room
1145 void display_editroom(void)
1147 char buf[SIZ];
1148 char cmd[1024];
1149 char node[256];
1150 char remote_room[128];
1151 char recp[1024];
1152 char er_name[128];
1153 char er_password[10];
1154 char er_dirname[15];
1155 char er_roomaide[26];
1156 unsigned er_flags;
1157 unsigned er_flags2;
1158 int er_floor;
1159 int i, j;
1160 char *tab;
1161 char *shared_with;
1162 char *not_shared_with;
1163 int roompolicy = 0;
1164 int roomvalue = 0;
1165 int floorpolicy = 0;
1166 int floorvalue = 0;
1167 char pop3_host[128];
1168 char pop3_user[32];
1169 int bg = 0;
1171 tab = bstr("tab");
1172 if (IsEmptyStr(tab)) tab = "admin";
1174 load_floorlist();
1175 output_headers(1, 1, 1, 0, 0, 0);
1177 wprintf("<div class=\"fix_scrollbar_bug\">");
1179 wprintf("<br />\n");
1181 /* print the tabbed dialog */
1182 wprintf("<div align=\"center\">");
1183 wprintf("<table id=\"AdminTabs\" border=\"0\" cellspacing=\"0\" cellpadding=\"0\""
1184 "<tr align=\"center\" style=\"cursor:pointer\"><td>&nbsp;</td>"
1187 wprintf("<td class=\"");
1188 if (!strcmp(tab, "admin")) {
1189 wprintf(" tab_cell_label\">");
1190 wprintf(_("Administration"));
1192 else {
1193 wprintf("< tab_cell_edit\"><a href=\"display_editroom&tab=admin\">");
1194 wprintf(_("Administration"));
1195 wprintf("</a>");
1197 wprintf("</td>\n");
1198 wprintf("<td>&nbsp;</td>\n");
1200 if ( (WC->axlevel >= 6) || (WC->is_room_aide) ) {
1202 wprintf("<td class=\"");
1203 if (!strcmp(tab, "config")) {
1204 wprintf(" tab_cell_label\">");
1205 wprintf(_("Configuration"));
1207 else {
1208 wprintf(" tab_cell_edit\"><a href=\"display_editroom&tab=config\">");
1209 wprintf(_("Configuration"));
1210 wprintf("</a>");
1212 wprintf("</td>\n");
1213 wprintf("<td>&nbsp;</td>\n");
1215 wprintf("<td class=\"");
1216 if (!strcmp(tab, "expire")) {
1217 wprintf(" tab_cell_label\">");
1218 wprintf(_("Message expire policy"));
1220 else {
1221 wprintf(" tab_cell_edit\"><a href=\"display_editroom&tab=expire\">");
1222 wprintf(_("Message expire policy"));
1223 wprintf("</a>");
1225 wprintf("</td>\n");
1226 wprintf("<td>&nbsp;</td>\n");
1228 wprintf("<td class=\"");
1229 if (!strcmp(tab, "access")) {
1230 wprintf(" tab_cell_label\">");
1231 wprintf(_("Access controls"));
1233 else {
1234 wprintf(" tab_cell_edit\"><a href=\"display_editroom&tab=access\">");
1235 wprintf(_("Access controls"));
1236 wprintf("</a>");
1238 wprintf("</td>\n");
1239 wprintf("<td>&nbsp;</td>\n");
1241 wprintf("<td class=\"");
1242 if (!strcmp(tab, "sharing")) {
1243 wprintf(" tab_cell_label\">");
1244 wprintf(_("Sharing"));
1246 else {
1247 wprintf(" tab_cell_edit\"><a href=\"display_editroom&tab=sharing\">");
1248 wprintf(_("Sharing"));
1249 wprintf("</a>");
1251 wprintf("</td>\n");
1252 wprintf("<td>&nbsp;</td>\n");
1254 wprintf("<td class=\"");
1255 if (!strcmp(tab, "listserv")) {
1256 wprintf(" tab_cell_label\">");
1257 wprintf(_("Mailing list service"));
1259 else {
1260 wprintf("< tab_cell_edit\"><a href=\"display_editroom&tab=listserv\">");
1261 wprintf(_("Mailing list service"));
1262 wprintf("</a>");
1264 wprintf("</td>\n");
1265 wprintf("<td>&nbsp;</td>\n");
1269 wprintf("<td class=\"");
1270 if (!strcmp(tab, "feeds")) {
1271 wprintf(" tab_cell_label\">");
1272 wprintf(_("Remote retrieval"));
1274 else {
1275 wprintf("< tab_cell_edit\"><a href=\"display_editroom&tab=feeds\">");
1276 wprintf(_("Remote retrieval"));
1277 wprintf("</a>");
1279 wprintf("</td>\n");
1280 wprintf("<td>&nbsp;</td>\n");
1282 wprintf("</tr></table>\n");
1283 wprintf("</div>\n");
1284 /* end tabbed dialog */
1286 wprintf("<script type=\"text/javascript\">"
1287 " Nifty(\"table#AdminTabs td\", \"small transparent top\");"
1288 "</script>"
1291 /* begin content of whatever tab is open now */
1293 if (!strcmp(tab, "admin")) {
1294 wprintf("<div class=\"tabcontent\">");
1295 wprintf("<ul>"
1296 "<li><a href=\"delete_room\" "
1297 "onClick=\"return confirm('");
1298 wprintf(_("Are you sure you want to delete this room?"));
1299 wprintf("');\">\n");
1300 wprintf(_("Delete this room"));
1301 wprintf("</a>\n"
1302 "<li><a href=\"display_editroompic\">\n");
1303 wprintf(_("Set or change the icon for this room's banner"));
1304 wprintf("</a>\n"
1305 "<li><a href=\"display_editinfo\">\n");
1306 wprintf(_("Edit this room's Info file"));
1307 wprintf("</a>\n"
1308 "</ul>");
1309 wprintf("</div>");
1312 if (!strcmp(tab, "config")) {
1313 wprintf("<div class=\"tabcontent\">");
1314 serv_puts("GETR");
1315 serv_getln(buf, sizeof buf);
1317 if (!strncmp(buf, "550", 3)) {
1318 wprintf("<br><br><div align=center>%s</div><br><br>\n",
1319 _("Higher access is required to access this function.")
1322 else if (buf[0] != '2') {
1323 wprintf("<br><br><div align=center>%s</div><br><br>\n", &buf[4]);
1325 else {
1326 extract_token(er_name, &buf[4], 0, '|', sizeof er_name);
1327 extract_token(er_password, &buf[4], 1, '|', sizeof er_password);
1328 extract_token(er_dirname, &buf[4], 2, '|', sizeof er_dirname);
1329 er_flags = extract_int(&buf[4], 3);
1330 er_floor = extract_int(&buf[4], 4);
1331 er_flags2 = extract_int(&buf[4], 7);
1333 wprintf("<form method=\"POST\" action=\"editroom\">\n");
1334 wprintf("<input type=\"hidden\" name=\"nonce\" value=\"%d\">\n", WC->nonce);
1336 wprintf("<ul><li>");
1337 wprintf(_("Name of room: "));
1338 wprintf("<input type=\"text\" NAME=\"er_name\" VALUE=\"%s\" MAXLENGTH=\"%d\">\n",
1339 er_name,
1340 (sizeof(er_name)-1)
1343 wprintf("<li>");
1344 wprintf(_("Resides on floor: "));
1345 wprintf("<select NAME=\"er_floor\" SIZE=\"1\"");
1346 if (er_flags & QR_MAILBOX)
1347 wprintf("disabled >\n");
1348 for (i = 0; i < 128; ++i)
1349 if (!IsEmptyStr(floorlist[i])) {
1350 wprintf("<OPTION ");
1351 if (i == er_floor )
1352 wprintf("SELECTED ");
1353 wprintf("VALUE=\"%d\">", i);
1354 escputs(floorlist[i]);
1355 wprintf("</OPTION>\n");
1357 wprintf("</select>\n");
1359 wprintf("<li>");
1360 wprintf(_("Type of room:"));
1361 wprintf("<ul>\n");
1363 wprintf("<li><input type=\"radio\" NAME=\"type\" VALUE=\"public\" ");
1364 if ((er_flags & (QR_PRIVATE + QR_MAILBOX)) == 0)
1365 wprintf("CHECKED ");
1366 wprintf("OnChange=\""
1367 " if (this.form.type[0].checked == true) { "
1368 " this.form.er_floor.disabled = false; "
1369 " } "
1370 "\"> ");
1371 wprintf(_("Public (automatically appears to everyone)"));
1372 wprintf("\n");
1374 wprintf("<li><input type=\"radio\" NAME=\"type\" VALUE=\"hidden\" ");
1375 if ((er_flags & QR_PRIVATE) &&
1376 (er_flags & QR_GUESSNAME))
1377 wprintf("CHECKED ");
1378 wprintf(" OnChange=\""
1379 " if (this.form.type[1].checked == true) { "
1380 " this.form.er_floor.disabled = false; "
1381 " } "
1382 "\"> ");
1383 wprintf(_("Private - hidden (accessible to anyone who knows its name)"));
1385 wprintf("\n<li><input type=\"radio\" NAME=\"type\" VALUE=\"passworded\" ");
1386 if ((er_flags & QR_PRIVATE) &&
1387 (er_flags & QR_PASSWORDED))
1388 wprintf("CHECKED ");
1389 wprintf(" OnChange=\""
1390 " if (this.form.type[2].checked == true) { "
1391 " this.form.er_floor.disabled = false; "
1392 " } "
1393 "\"> ");
1394 wprintf(_("Private - require password: "));
1395 wprintf("\n<input type=\"text\" NAME=\"er_password\" VALUE=\"%s\" MAXLENGTH=\"9\">\n",
1396 er_password);
1398 wprintf("<li><input type=\"radio\" NAME=\"type\" VALUE=\"invonly\" ");
1399 if ((er_flags & QR_PRIVATE)
1400 && ((er_flags & QR_GUESSNAME) == 0)
1401 && ((er_flags & QR_PASSWORDED) == 0))
1402 wprintf("CHECKED ");
1403 wprintf(" OnChange=\""
1404 " if (this.form.type[3].checked == true) { "
1405 " this.form.er_floor.disabled = false; "
1406 " } "
1407 "\"> ");
1408 wprintf(_("Private - invitation only"));
1410 wprintf("\n<li><input type=\"radio\" NAME=\"type\" VALUE=\"personal\" ");
1411 if (er_flags & QR_MAILBOX)
1412 wprintf("CHECKED ");
1413 wprintf (" OnChange=\""
1414 " if (this.form.type[4].checked == true) { "
1415 " this.form.er_floor.disabled = true; "
1416 " } "
1417 "\"> ");
1418 wprintf(_("Personal (mailbox for you only)"));
1420 wprintf("\n<li><input type=\"checkbox\" NAME=\"bump\" VALUE=\"yes\" ");
1421 wprintf("> ");
1422 wprintf(_("If private, cause current users to forget room"));
1424 wprintf("\n</ul>\n");
1426 wprintf("<li><input type=\"checkbox\" NAME=\"prefonly\" VALUE=\"yes\" ");
1427 if (er_flags & QR_PREFONLY)
1428 wprintf("CHECKED ");
1429 wprintf("> ");
1430 wprintf(_("Preferred users only"));
1432 wprintf("\n<li><input type=\"checkbox\" NAME=\"readonly\" VALUE=\"yes\" ");
1433 if (er_flags & QR_READONLY)
1434 wprintf("CHECKED ");
1435 wprintf("> ");
1436 wprintf(_("Read-only room"));
1438 wprintf("\n<li><input type=\"checkbox\" NAME=\"collabdel\" VALUE=\"yes\" ");
1439 if (er_flags2 & QR2_COLLABDEL)
1440 wprintf("CHECKED ");
1441 wprintf("> ");
1442 wprintf(_("All users allowed to post may also delete messages"));
1444 /** directory stuff */
1445 wprintf("\n<li><input type=\"checkbox\" NAME=\"directory\" VALUE=\"yes\" ");
1446 if (er_flags & QR_DIRECTORY)
1447 wprintf("CHECKED ");
1448 wprintf("> ");
1449 wprintf(_("File directory room"));
1451 wprintf("\n<ul><li>");
1452 wprintf(_("Directory name: "));
1453 wprintf("<input type=\"text\" NAME=\"er_dirname\" VALUE=\"%s\" MAXLENGTH=\"14\">\n",
1454 er_dirname);
1456 wprintf("<li><input type=\"checkbox\" NAME=\"ulallowed\" VALUE=\"yes\" ");
1457 if (er_flags & QR_UPLOAD)
1458 wprintf("CHECKED ");
1459 wprintf("> ");
1460 wprintf(_("Uploading allowed"));
1462 wprintf("\n<li><input type=\"checkbox\" NAME=\"dlallowed\" VALUE=\"yes\" ");
1463 if (er_flags & QR_DOWNLOAD)
1464 wprintf("CHECKED ");
1465 wprintf("> ");
1466 wprintf(_("Downloading allowed"));
1468 wprintf("\n<li><input type=\"checkbox\" NAME=\"visdir\" VALUE=\"yes\" ");
1469 if (er_flags & QR_VISDIR)
1470 wprintf("CHECKED ");
1471 wprintf("> ");
1472 wprintf(_("Visible directory"));
1473 wprintf("</ul>\n");
1475 /** end of directory stuff */
1477 wprintf("<li><input type=\"checkbox\" NAME=\"network\" VALUE=\"yes\" ");
1478 if (er_flags & QR_NETWORK)
1479 wprintf("CHECKED ");
1480 wprintf("> ");
1481 wprintf(_("Network shared room"));
1483 wprintf("\n<li><input type=\"checkbox\" NAME=\"permanent\" VALUE=\"yes\" ");
1484 if (er_flags & QR_PERMANENT)
1485 wprintf("CHECKED ");
1486 wprintf("> ");
1487 wprintf(_("Permanent (does not auto-purge)"));
1489 wprintf("\n<li><input type=\"checkbox\" NAME=\"subjectreq\" VALUE=\"yes\" ");
1490 if (er_flags2 & QR2_SUBJECTREQ)
1491 wprintf("CHECKED ");
1492 wprintf("> ");
1493 wprintf(_("Subject Required (Force users to specify a message subject)"));
1495 /** start of anon options */
1497 wprintf("\n<li>");
1498 wprintf(_("Anonymous messages"));
1499 wprintf("<ul>\n");
1501 wprintf("<li><input type=\"radio\" NAME=\"anon\" VALUE=\"no\" ");
1502 if (((er_flags & QR_ANONONLY) == 0)
1503 && ((er_flags & QR_ANONOPT) == 0))
1504 wprintf("CHECKED ");
1505 wprintf("> ");
1506 wprintf(_("No anonymous messages"));
1508 wprintf("\n<li><input type=\"radio\" NAME=\"anon\" VALUE=\"anononly\" ");
1509 if (er_flags & QR_ANONONLY)
1510 wprintf("CHECKED ");
1511 wprintf("> ");
1512 wprintf(_("All messages are anonymous"));
1514 wprintf("\n<li><input type=\"radio\" NAME=\"anon\" VALUE=\"anon2\" ");
1515 if (er_flags & QR_ANONOPT)
1516 wprintf("CHECKED ");
1517 wprintf("> ");
1518 wprintf(_("Prompt user when entering messages"));
1519 wprintf("</ul>\n");
1521 /* end of anon options */
1523 wprintf("<li>");
1524 wprintf(_("Room aide: "));
1525 serv_puts("GETA");
1526 serv_getln(buf, sizeof buf);
1527 if (buf[0] != '2') {
1528 wprintf("<em>%s</em>\n", &buf[4]);
1529 } else {
1530 extract_token(er_roomaide, &buf[4], 0, '|', sizeof er_roomaide);
1531 wprintf("<input type=\"text\" NAME=\"er_roomaide\" VALUE=\"%s\" MAXLENGTH=\"25\">\n", er_roomaide);
1534 wprintf("</ul><CENTER>\n");
1535 wprintf("<input type=\"hidden\" NAME=\"tab\" VALUE=\"config\">\n"
1536 "<input type=\"submit\" NAME=\"ok_button\" VALUE=\"%s\">"
1537 "&nbsp;"
1538 "<input type=\"submit\" NAME=\"cancel_button\" VALUE=\"%s\">"
1539 "</CENTER>\n",
1540 _("Save changes"),
1541 _("Cancel")
1544 wprintf("</div>");
1548 /* Sharing the room with other Citadel nodes... */
1549 if (!strcmp(tab, "sharing")) {
1550 wprintf("<div class=\"tabcontent\">");
1552 shared_with = strdup("");
1553 not_shared_with = strdup("");
1555 /** Learn the current configuration */
1556 serv_puts("CONF getsys|application/x-citadel-ignet-config");
1557 serv_getln(buf, sizeof buf);
1558 if (buf[0]=='1') while (serv_getln(buf, sizeof buf), strcmp(buf, "000")) {
1559 extract_token(node, buf, 0, '|', sizeof node);
1560 not_shared_with = realloc(not_shared_with,
1561 strlen(not_shared_with) + 32);
1562 strcat(not_shared_with, node);
1563 strcat(not_shared_with, "\n");
1566 serv_puts("GNET");
1567 serv_getln(buf, sizeof buf);
1568 if (buf[0]=='1') while (serv_getln(buf, sizeof buf), strcmp(buf, "000")) {
1569 extract_token(cmd, buf, 0, '|', sizeof cmd);
1570 extract_token(node, buf, 1, '|', sizeof node);
1571 extract_token(remote_room, buf, 2, '|', sizeof remote_room);
1572 if (!strcasecmp(cmd, "ignet_push_share")) {
1573 shared_with = realloc(shared_with,
1574 strlen(shared_with) + 32);
1575 strcat(shared_with, node);
1576 if (!IsEmptyStr(remote_room)) {
1577 strcat(shared_with, "|");
1578 strcat(shared_with, remote_room);
1580 strcat(shared_with, "\n");
1584 for (i=0; i<num_tokens(shared_with, '\n'); ++i) {
1585 extract_token(buf, shared_with, i, '\n', sizeof buf);
1586 extract_token(node, buf, 0, '|', sizeof node);
1587 for (j=0; j<num_tokens(not_shared_with, '\n'); ++j) {
1588 extract_token(cmd, not_shared_with, j, '\n', sizeof cmd);
1589 if (!strcasecmp(node, cmd)) {
1590 remove_token(not_shared_with, j, '\n');
1595 /* Display the stuff */
1596 wprintf("<CENTER><br />"
1597 "<table border=1 cellpadding=5><tr>"
1598 "<td><B><I>");
1599 wprintf(_("Shared with"));
1600 wprintf("</I></B></td>"
1601 "<td><B><I>");
1602 wprintf(_("Not shared with"));
1603 wprintf("</I></B></td></tr>\n"
1604 "<tr><td VALIGN=TOP>\n");
1606 wprintf("<table border=0 cellpadding=5><tr class=\"tab_cell\"><td>");
1607 wprintf(_("Remote node name"));
1608 wprintf("</td><td>");
1609 wprintf(_("Remote room name"));
1610 wprintf("</td><td>");
1611 wprintf(_("Actions"));
1612 wprintf("</td></tr>\n");
1614 for (i=0; i<num_tokens(shared_with, '\n'); ++i) {
1615 extract_token(buf, shared_with, i, '\n', sizeof buf);
1616 extract_token(node, buf, 0, '|', sizeof node);
1617 extract_token(remote_room, buf, 1, '|', sizeof remote_room);
1618 if (!IsEmptyStr(node)) {
1619 wprintf("<form method=\"POST\" action=\"netedit\">");
1620 wprintf("<input type=\"hidden\" name=\"nonce\" value=\"%d\">\n", WC->nonce);
1621 wprintf("<tr><td>%s</td>\n", node);
1623 wprintf("<td>");
1624 if (!IsEmptyStr(remote_room)) {
1625 escputs(remote_room);
1627 wprintf("</td>");
1629 wprintf("<td>");
1631 wprintf("<input type=\"hidden\" NAME=\"line\" "
1632 "VALUE=\"ignet_push_share|");
1633 urlescputs(node);
1634 if (!IsEmptyStr(remote_room)) {
1635 wprintf("|");
1636 urlescputs(remote_room);
1638 wprintf("\">");
1639 wprintf("<input type=\"hidden\" NAME=\"tab\" VALUE=\"sharing\">\n");
1640 wprintf("<input type=\"hidden\" NAME=\"cmd\" VALUE=\"remove\">\n");
1641 wprintf("<input type=\"submit\" "
1642 "NAME=\"unshare_button\" VALUE=\"%s\">", _("Unshare"));
1643 wprintf("</td></tr></form>\n");
1647 wprintf("</table>\n");
1648 wprintf("</td><td VALIGN=TOP>\n");
1649 wprintf("<table border=0 cellpadding=5><tr class=\"tab_cell\"><td>");
1650 wprintf(_("Remote node name"));
1651 wprintf("</td><td>");
1652 wprintf(_("Remote room name"));
1653 wprintf("</td><td>");
1654 wprintf(_("Actions"));
1655 wprintf("</td></tr>\n");
1657 for (i=0; i<num_tokens(not_shared_with, '\n'); ++i) {
1658 extract_token(node, not_shared_with, i, '\n', sizeof node);
1659 if (!IsEmptyStr(node)) {
1660 wprintf("<form method=\"POST\" action=\"netedit\">");
1661 wprintf("<input type=\"hidden\" name=\"nonce\" value=\"%d\">\n", WC->nonce);
1662 wprintf("<tr><td>");
1663 escputs(node);
1664 wprintf("</td><td>"
1665 "<input type=\"INPUT\" "
1666 "NAME=\"suffix\" "
1667 "MAXLENGTH=128>"
1668 "</td><td>");
1669 wprintf("<input type=\"hidden\" "
1670 "NAME=\"line\" "
1671 "VALUE=\"ignet_push_share|");
1672 urlescputs(node);
1673 wprintf("|\">");
1674 wprintf("<input type=\"hidden\" NAME=\"tab\" "
1675 "VALUE=\"sharing\">\n");
1676 wprintf("<input type=\"hidden\" NAME=\"cmd\" "
1677 "VALUE=\"add\">\n");
1678 wprintf("<input type=\"submit\" "
1679 "NAME=\"add_button\" VALUE=\"%s\">", _("Share"));
1680 wprintf("</td></tr></form>\n");
1684 wprintf("</table>\n");
1685 wprintf("</td></tr>"
1686 "</table></CENTER><br />\n"
1687 "<I><B>%s</B><ul><li>", _("Notes:"));
1688 wprintf(_("When sharing a room, "
1689 "it must be shared from both ends. Adding a node to "
1690 "the 'shared' list sends messages out, but in order to"
1691 " receive messages, the other nodes must be configured"
1692 " to send messages out to your system as well. "
1693 "<li>If the remote room name is blank, it is assumed "
1694 "that the room name is identical on the remote node."
1695 "<li>If the remote room name is different, the remote "
1696 "node must also configure the name of the room here."
1697 "</ul></I><br />\n"
1700 wprintf("</div>");
1703 /* Mailing list management */
1704 if (!strcmp(tab, "listserv")) {
1705 room_states RoomFlags;
1706 wprintf("<div class=\"tabcontent\">");
1708 wprintf("<br /><center>"
1709 "<table BORDER=0 WIDTH=100%% CELLPADDING=5>"
1710 "<tr><td VALIGN=TOP>");
1712 wprintf(_("<i>The contents of this room are being "
1713 "mailed <b>as individual messages</b> "
1714 "to the following list recipients:"
1715 "</i><br /><br />\n"));
1717 serv_puts("GNET");
1718 serv_getln(buf, sizeof buf);
1719 if (buf[0]=='1') while (serv_getln(buf, sizeof buf), strcmp(buf, "000")) {
1720 extract_token(cmd, buf, 0, '|', sizeof cmd);
1721 if (!strcasecmp(cmd, "listrecp")) {
1722 extract_token(recp, buf, 1, '|', sizeof recp);
1724 escputs(recp);
1725 wprintf(" <a href=\"netedit&cmd=remove&tab=listserv&line=listrecp|");
1726 urlescputs(recp);
1727 wprintf("\">");
1728 wprintf(_("(remove)"));
1729 wprintf("</A><br />");
1732 wprintf("<br /><form method=\"POST\" action=\"netedit\">\n"
1733 "<input type=\"hidden\" NAME=\"tab\" VALUE=\"listserv\">\n"
1734 "<input type=\"hidden\" NAME=\"prefix\" VALUE=\"listrecp|\">\n");
1735 wprintf("<input type=\"hidden\" name=\"nonce\" value=\"%d\">\n", WC->nonce);
1736 wprintf("<input type=\"text\" id=\"add_as_listrecp\" NAME=\"line\">\n");
1737 wprintf("<input type=\"submit\" NAME=\"add_button\" VALUE=\"%s\">", _("Add"));
1738 wprintf("</form>\n");
1740 wprintf("</td><td VALIGN=TOP>\n");
1742 wprintf(_("<i>The contents of this room are being "
1743 "mailed <b>in digest form</b> "
1744 "to the following list recipients:"
1745 "</i><br /><br />\n"));
1747 serv_puts("GNET");
1748 serv_getln(buf, sizeof buf);
1749 if (buf[0]=='1') while (serv_getln(buf, sizeof buf), strcmp(buf, "000")) {
1750 extract_token(cmd, buf, 0, '|', sizeof cmd);
1751 if (!strcasecmp(cmd, "digestrecp")) {
1752 extract_token(recp, buf, 1, '|', sizeof recp);
1754 escputs(recp);
1755 wprintf(" <a href=\"netedit&cmd=remove&tab=listserv&line="
1756 "digestrecp|");
1757 urlescputs(recp);
1758 wprintf("\">");
1759 wprintf(_("(remove)"));
1760 wprintf("</A><br />");
1763 wprintf("<br /><form method=\"POST\" action=\"netedit\">\n"
1764 "<input type=\"hidden\" NAME=\"tab\" VALUE=\"listserv\">\n"
1765 "<input type=\"hidden\" NAME=\"prefix\" VALUE=\"digestrecp|\">\n");
1766 wprintf("<input type=\"hidden\" name=\"nonce\" value=\"%d\">\n", WC->nonce);
1767 wprintf("<input type=\"text\" id=\"add_as_digestrecp\" NAME=\"line\">\n");
1768 wprintf("<input type=\"submit\" NAME=\"add_button\" VALUE=\"%s\">", _("Add"));
1769 wprintf("</form>\n");
1771 wprintf("</td></tr></table>\n");
1773 /** Pop open an address book -- begin **/
1774 wprintf("<div align=right>"
1775 "<a href=\"javascript:PopOpenAddressBook('add_as_listrecp|%s|add_as_digestrecp|%s');\" "
1776 "title=\"%s\">"
1777 "<img align=middle border=0 width=24 height=24 src=\"static/viewcontacts_24x.gif\">"
1778 "&nbsp;%s</a>"
1779 "</div>",
1780 _("List"),
1781 _("Digest"),
1782 _("Add recipients from Contacts or other address books"),
1783 _("Add recipients from Contacts or other address books")
1785 /* Pop open an address book -- end **/
1787 wprintf("<br />\n<form method=\"GET\" action=\"toggle_self_service\">\n");
1789 get_roomflags (&RoomFlags);
1791 /* Self Service subscription? */
1792 wprintf("<table><tr><td>\n");
1793 wprintf(_("Allow self-service subscribe/unsubscribe requests."));
1794 wprintf("</td><td><input type=\"checkbox\" name=\"QR2_SelfList\" value=\"yes\" %s></td></tr>\n"
1795 " <tr><td colspan=\"2\">\n",
1796 (is_selflist(&RoomFlags))?"checked":"");
1797 wprintf(_("The URL for subscribe/unsubscribe is: "));
1798 wprintf("<TT>%s://%s/listsub</TT></td></tr>\n",
1799 (is_https ? "https" : "http"),
1800 ChrPtr(WC->http_host));
1801 /* Public posting? */
1802 wprintf("<tr><td>");
1803 wprintf(_("Allow non-subscribers to mail to this room."));
1804 wprintf("</td><td><input type=\"checkbox\" name=\"QR2_SubsOnly\" value=\"yes\" %s></td></tr>\n",
1805 (is_publiclist(&RoomFlags))?"checked":"");
1807 /* Moderated List? */
1808 wprintf("<tr><td>");
1809 wprintf(_("Room post publication needs Aide permission."));
1810 wprintf("</td><td><input type=\"checkbox\" name=\"QR2_Moderated\" value=\"yes\" %s></td></tr>\n",
1811 (is_moderatedlist(&RoomFlags))?"checked":"");
1814 wprintf("<tr><td colspan=\"2\" align=\"center\">"
1815 "<input type=\"submit\" NAME=\"add_button\" VALUE=\"%s\"></td></tr>", _("Save changes"));
1816 wprintf("</table></form>");
1819 wprintf("</CENTER>\n");
1820 wprintf("</div>");
1824 /* Configuration of The Dreaded Auto-Purger */
1825 if (!strcmp(tab, "expire")) {
1826 wprintf("<div class=\"tabcontent\">");
1828 serv_puts("GPEX room");
1829 serv_getln(buf, sizeof buf);
1830 if (!strncmp(buf, "550", 3)) {
1831 wprintf("<br><br><div align=center>%s</div><br><br>\n",
1832 _("Higher access is required to access this function.")
1835 else if (buf[0] != '2') {
1836 wprintf("<br><br><div align=center>%s</div><br><br>\n", &buf[4]);
1838 else {
1839 roompolicy = extract_int(&buf[4], 0);
1840 roomvalue = extract_int(&buf[4], 1);
1842 serv_puts("GPEX floor");
1843 serv_getln(buf, sizeof buf);
1844 if (buf[0] == '2') {
1845 floorpolicy = extract_int(&buf[4], 0);
1846 floorvalue = extract_int(&buf[4], 1);
1849 wprintf("<br /><form method=\"POST\" action=\"set_room_policy\">\n");
1850 wprintf("<input type=\"hidden\" name=\"nonce\" value=\"%d\">\n", WC->nonce);
1851 wprintf("<table border=0 cellspacing=5>\n");
1852 wprintf("<tr><td>");
1853 wprintf(_("Message expire policy for this room"));
1854 wprintf("<br />(");
1855 escputs(ChrPtr(WC->wc_roomname));
1856 wprintf(")</td><td>");
1857 wprintf("<input type=\"radio\" NAME=\"roompolicy\" VALUE=\"0\" %s>",
1858 ((roompolicy == 0) ? "CHECKED" : "") );
1859 wprintf(_("Use the default policy for this floor"));
1860 wprintf("<br />\n");
1861 wprintf("<input type=\"radio\" NAME=\"roompolicy\" VALUE=\"1\" %s>",
1862 ((roompolicy == 1) ? "CHECKED" : "") );
1863 wprintf(_("Never automatically expire messages"));
1864 wprintf("<br />\n");
1865 wprintf("<input type=\"radio\" NAME=\"roompolicy\" VALUE=\"2\" %s>",
1866 ((roompolicy == 2) ? "CHECKED" : "") );
1867 wprintf(_("Expire by message count"));
1868 wprintf("<br />\n");
1869 wprintf("<input type=\"radio\" NAME=\"roompolicy\" VALUE=\"3\" %s>",
1870 ((roompolicy == 3) ? "CHECKED" : "") );
1871 wprintf(_("Expire by message age"));
1872 wprintf("<br />");
1873 wprintf(_("Number of messages or days: "));
1874 wprintf("<input type=\"text\" NAME=\"roomvalue\" MAXLENGTH=\"5\" VALUE=\"%d\">", roomvalue);
1875 wprintf("</td></tr>\n");
1877 if (WC->axlevel >= 6) {
1878 wprintf("<tr><td COLSPAN=2><hr /></td></tr>\n");
1879 wprintf("<tr><td>");
1880 wprintf(_("Message expire policy for this floor"));
1881 wprintf("<br />(");
1882 escputs(floorlist[WC->wc_floor]);
1883 wprintf(")</td><td>");
1884 wprintf("<input type=\"radio\" NAME=\"floorpolicy\" VALUE=\"0\" %s>",
1885 ((floorpolicy == 0) ? "CHECKED" : "") );
1886 wprintf(_("Use the system default"));
1887 wprintf("<br />\n");
1888 wprintf("<input type=\"radio\" NAME=\"floorpolicy\" VALUE=\"1\" %s>",
1889 ((floorpolicy == 1) ? "CHECKED" : "") );
1890 wprintf(_("Never automatically expire messages"));
1891 wprintf("<br />\n");
1892 wprintf("<input type=\"radio\" NAME=\"floorpolicy\" VALUE=\"2\" %s>",
1893 ((floorpolicy == 2) ? "CHECKED" : "") );
1894 wprintf(_("Expire by message count"));
1895 wprintf("<br />\n");
1896 wprintf("<input type=\"radio\" NAME=\"floorpolicy\" VALUE=\"3\" %s>",
1897 ((floorpolicy == 3) ? "CHECKED" : "") );
1898 wprintf(_("Expire by message age"));
1899 wprintf("<br />");
1900 wprintf(_("Number of messages or days: "));
1901 wprintf("<input type=\"text\" NAME=\"floorvalue\" MAXLENGTH=\"5\" VALUE=\"%d\">",
1902 floorvalue);
1905 wprintf("<CENTER>\n");
1906 wprintf("<tr><td COLSPAN=2><hr /><CENTER>\n");
1907 wprintf("<input type=\"submit\" NAME=\"ok_button\" VALUE=\"%s\">", _("Save changes"));
1908 wprintf("&nbsp;");
1909 wprintf("<input type=\"submit\" NAME=\"cancel_button\" VALUE=\"%s\">", _("Cancel"));
1910 wprintf("</CENTER></td><tr>\n");
1912 wprintf("</table>\n"
1913 "<input type=\"hidden\" NAME=\"tab\" VALUE=\"expire\">\n"
1914 "</form>\n"
1918 wprintf("</div>");
1921 /* Access controls */
1922 if (!strcmp(tab, "access")) {
1923 wprintf("<div class=\"tabcontent\">");
1924 display_whok();
1925 wprintf("</div>");
1928 /* Fetch messages from remote locations */
1929 if (!strcmp(tab, "feeds")) {
1930 wprintf("<div class=\"tabcontent\">");
1932 wprintf("<i>");
1933 wprintf(_("Retrieve messages from these remote POP3 accounts and store them in this room:"));
1934 wprintf("</i><br />\n");
1936 wprintf("<table class=\"altern\" border=0 cellpadding=5>"
1937 "<tr class=\"even\"><th>");
1938 wprintf(_("Remote host"));
1939 wprintf("</th><th>");
1940 wprintf(_("User name"));
1941 wprintf("</th><th>");
1942 wprintf(_("Password"));
1943 wprintf("</th><th>");
1944 wprintf(_("Keep messages on server?"));
1945 wprintf("</th><th>");
1946 wprintf(_("Interval"));
1947 wprintf("</th><th> </th></tr>");
1949 serv_puts("GNET");
1950 serv_getln(buf, sizeof buf);
1951 bg = 1;
1952 if (buf[0]=='1') while (serv_getln(buf, sizeof buf), strcmp(buf, "000")) {
1953 extract_token(cmd, buf, 0, '|', sizeof cmd);
1954 if (!strcasecmp(cmd, "pop3client")) {
1955 safestrncpy(recp, &buf[11], sizeof recp);
1957 bg = 1 - bg;
1958 wprintf("<tr class=\"%s\">",
1959 (bg ? "even" : "odd")
1962 wprintf("<td>");
1963 extract_token(pop3_host, buf, 1, '|', sizeof pop3_host);
1964 escputs(pop3_host);
1965 wprintf("</td>");
1967 wprintf("<td>");
1968 extract_token(pop3_user, buf, 2, '|', sizeof pop3_user);
1969 escputs(pop3_user);
1970 wprintf("</td>");
1972 wprintf("<td>*****</td>"); /* Don't show the password */
1974 wprintf("<td>%s</td>", extract_int(buf, 4) ? _("Yes") : _("No"));
1976 wprintf("<td>%ld</td>", extract_long(buf, 5)); /* Fetching interval */
1978 wprintf("<td class=\"button_link\">");
1979 wprintf(" <a href=\"netedit&cmd=remove&tab=feeds&line=pop3client|");
1980 urlescputs(recp);
1981 wprintf("\">");
1982 wprintf(_("(remove)"));
1983 wprintf("</a></td>");
1985 wprintf("</tr>");
1989 wprintf("<form method=\"POST\" action=\"netedit\">\n"
1990 "<tr>"
1991 "<input type=\"hidden\" name=\"tab\" value=\"feeds\">"
1992 "<input type=\"hidden\" name=\"prefix\" value=\"pop3client|\">\n");
1993 wprintf("<input type=\"hidden\" name=\"nonce\" value=\"%d\">\n", WC->nonce);
1994 wprintf("<td>");
1995 wprintf("<input type=\"text\" id=\"add_as_pop3host\" NAME=\"line_pop3host\">\n");
1996 wprintf("</td>");
1997 wprintf("<td>");
1998 wprintf("<input type=\"text\" id=\"add_as_pop3user\" NAME=\"line_pop3user\">\n");
1999 wprintf("</td>");
2000 wprintf("<td>");
2001 wprintf("<input type=\"password\" id=\"add_as_pop3pass\" NAME=\"line_pop3pass\">\n");
2002 wprintf("</td>");
2003 wprintf("<td>");
2004 wprintf("<input type=\"checkbox\" id=\"add_as_pop3keep\" NAME=\"line_pop3keep\" VALUE=\"1\">");
2005 wprintf("</td>");
2006 wprintf("<td>");
2007 wprintf("<input type=\"text\" id=\"add_as_pop3int\" NAME=\"line_pop3int\" MAXLENGTH=\"5\">");
2008 wprintf("</td>");
2009 wprintf("<td>");
2010 wprintf("<input type=\"submit\" NAME=\"add_button\" VALUE=\"%s\">", _("Add"));
2011 wprintf("</td></tr>");
2012 wprintf("</form></table>\n");
2014 wprintf("<hr>\n");
2016 wprintf("<i>");
2017 wprintf(_("Fetch the following RSS feeds and store them in this room:"));
2018 wprintf("</i><br />\n");
2020 wprintf("<table class=\"altern\" border=0 cellpadding=5>"
2021 "<tr class=\"even\"><th>");
2022 wprintf("<img src=\"static/rss_16x.png\" width=\"16\" height=\"16\" alt=\" \"> ");
2023 wprintf(_("Feed URL"));
2024 wprintf("</th><th>");
2025 wprintf("</th></tr>");
2027 serv_puts("GNET");
2028 serv_getln(buf, sizeof buf);
2029 bg = 1;
2030 if (buf[0]=='1') while (serv_getln(buf, sizeof buf), strcmp(buf, "000")) {
2031 extract_token(cmd, buf, 0, '|', sizeof cmd);
2032 if (!strcasecmp(cmd, "rssclient")) {
2033 safestrncpy(recp, &buf[10], sizeof recp);
2035 bg = 1 - bg;
2036 wprintf("<tr class=\"%s\">",
2037 (bg ? "even" : "odd")
2040 wprintf("<td>");
2041 extract_token(pop3_host, buf, 1, '|', sizeof pop3_host);
2042 escputs(pop3_host);
2043 wprintf("</td>");
2045 wprintf("<td class=\"button_link\">");
2046 wprintf(" <a href=\"netedit&cmd=remove&tab=feeds&line=rssclient|");
2047 urlescputs(recp);
2048 wprintf("\">");
2049 wprintf(_("(remove)"));
2050 wprintf("</a></td>");
2052 wprintf("</tr>");
2056 wprintf("<form method=\"POST\" action=\"netedit\">\n"
2057 "<tr>"
2058 "<input type=\"hidden\" name=\"tab\" value=\"feeds\">"
2059 "<input type=\"hidden\" name=\"prefix\" value=\"rssclient|\">\n");
2060 wprintf("<input type=\"hidden\" name=\"nonce\" value=\"%d\">\n", WC->nonce);
2061 wprintf("<td>");
2062 wprintf("<input type=\"text\" id=\"add_as_pop3host\" size=\"72\" "
2063 "maxlength=\"256\" name=\"line_pop3host\">\n");
2064 wprintf("</td>");
2065 wprintf("<td>");
2066 wprintf("<input type=\"submit\" name=\"add_button\" value=\"%s\">", _("Add"));
2067 wprintf("</td></tr>");
2068 wprintf("</form></table>\n");
2070 wprintf("</div>");
2074 /* end content of whatever tab is open now */
2075 wprintf("</div>\n");
2077 address_book_popup();
2078 wDumpContent(1);
2083 * Toggle self-service list subscription
2085 void toggle_self_service(void) {
2086 room_states RoomFlags;
2088 get_roomflags (&RoomFlags);
2090 if (yesbstr("QR2_SelfList"))
2091 RoomFlags.flags2 = RoomFlags.flags2 | QR2_SELFLIST;
2092 else
2093 RoomFlags.flags2 = RoomFlags.flags2 & ~QR2_SELFLIST;
2095 if (yesbstr("QR2_SMTP_PUBLIC"))
2096 RoomFlags.flags2 = RoomFlags.flags2 | QR2_SMTP_PUBLIC;
2097 else
2098 RoomFlags.flags2 = RoomFlags.flags2 & ~QR2_SMTP_PUBLIC;
2100 if (yesbstr("QR2_Moderated"))
2101 RoomFlags.flags2 = RoomFlags.flags2 | QR2_MODERATED;
2102 else
2103 RoomFlags.flags2 = RoomFlags.flags2 & ~QR2_MODERATED;
2104 if (yesbstr("QR2_SubsOnly"))
2105 RoomFlags.flags2 = RoomFlags.flags2 | QR2_SMTP_PUBLIC;
2106 else
2107 RoomFlags.flags2 = RoomFlags.flags2 & ~QR2_SMTP_PUBLIC;
2109 set_roomflags (&RoomFlags);
2111 display_editroom();
2117 * save new parameters for a room
2119 void editroom(void)
2121 const StrBuf *Ptr;
2122 StrBuf *Buf;
2123 StrBuf *er_name;
2124 StrBuf *er_password;
2125 StrBuf *er_dirname;
2126 StrBuf *er_roomaide;
2127 int er_floor;
2128 unsigned er_flags;
2129 int er_listingorder;
2130 int er_defaultview;
2131 unsigned er_flags2;
2132 int bump;
2135 if (!havebstr("ok_button")) {
2136 strcpy(WC->ImportantMessage,
2137 _("Cancelled. Changes were not saved."));
2138 display_editroom();
2139 return;
2141 serv_puts("GETR");
2142 Buf = NewStrBuf();
2143 StrBuf_ServGetln(Buf);
2144 if (GetServerStatus(Buf, NULL) != 2) {
2145 StrBufCutLeft(Buf, 4);
2146 strcpy(WC->ImportantMessage, ChrPtr(Buf));
2147 display_editroom();
2148 FreeStrBuf(&Buf);
2149 return;
2152 er_name = NewStrBuf();
2153 er_password = NewStrBuf();
2154 er_dirname = NewStrBuf();
2155 er_roomaide = NewStrBuf();
2157 StrBufCutLeft(Buf, 4);
2158 StrBufExtract_token(er_name, Buf, 0, '|');
2159 StrBufExtract_token(er_password, Buf, 1, '|');
2160 StrBufExtract_token(er_dirname, Buf, 2, '|');
2161 er_flags = StrBufExtract_int(Buf, 3, '|');
2162 er_listingorder = StrBufExtract_int(Buf, 5, '|');
2163 er_defaultview = StrBufExtract_int(Buf, 6, '|');
2164 er_flags2 = StrBufExtract_int(Buf, 7, '|');
2166 er_roomaide = NewStrBufDup(sbstr("er_roomaide"));
2167 if (StrLength(er_roomaide) == 0) {
2168 serv_puts("GETA");
2169 StrBuf_ServGetln(Buf);
2170 if (GetServerStatus(Buf, NULL) != 2) {
2171 FlushStrBuf(er_roomaide);
2172 } else {
2173 StrBufCutLeft(Buf, 4);
2174 StrBufExtract_token(er_roomaide, Buf, 0, '|');
2177 Ptr = sbstr("er_name");
2178 if (StrLength(Ptr) > 0) {
2179 FlushStrBuf(er_name);
2180 StrBufAppendBuf(er_name, Ptr, 0);
2183 Ptr = sbstr("er_password");
2184 if (StrLength(Ptr) > 0) {
2185 FlushStrBuf(er_password);
2186 StrBufAppendBuf(er_password, Ptr, 0);
2190 Ptr = sbstr("er_dirname");
2191 if (StrLength(Ptr) > 0) { /* todo: cut 15 */
2192 FlushStrBuf(er_dirname);
2193 StrBufAppendBuf(er_dirname, Ptr, 0);
2197 Ptr = sbstr("type");
2198 er_flags &= !(QR_PRIVATE | QR_PASSWORDED | QR_GUESSNAME);
2200 if (!strcmp(ChrPtr(Ptr), "invonly")) {
2201 er_flags |= (QR_PRIVATE);
2203 if (!strcmp(ChrPtr(Ptr), "hidden")) {
2204 er_flags |= (QR_PRIVATE | QR_GUESSNAME);
2206 if (!strcmp(ChrPtr(Ptr), "passworded")) {
2207 er_flags |= (QR_PRIVATE | QR_PASSWORDED);
2209 if (!strcmp(ChrPtr(Ptr), "personal")) {
2210 er_flags |= QR_MAILBOX;
2211 } else {
2212 er_flags &= ~QR_MAILBOX;
2215 if (yesbstr("prefonly")) {
2216 er_flags |= QR_PREFONLY;
2217 } else {
2218 er_flags &= ~QR_PREFONLY;
2221 if (yesbstr("readonly")) {
2222 er_flags |= QR_READONLY;
2223 } else {
2224 er_flags &= ~QR_READONLY;
2228 if (yesbstr("collabdel")) {
2229 er_flags2 |= QR2_COLLABDEL;
2230 } else {
2231 er_flags2 &= ~QR2_COLLABDEL;
2234 if (yesbstr("permanent")) {
2235 er_flags |= QR_PERMANENT;
2236 } else {
2237 er_flags &= ~QR_PERMANENT;
2240 if (yesbstr("subjectreq")) {
2241 er_flags2 |= QR2_SUBJECTREQ;
2242 } else {
2243 er_flags2 &= ~QR2_SUBJECTREQ;
2246 if (yesbstr("network")) {
2247 er_flags |= QR_NETWORK;
2248 } else {
2249 er_flags &= ~QR_NETWORK;
2252 if (yesbstr("directory")) {
2253 er_flags |= QR_DIRECTORY;
2254 } else {
2255 er_flags &= ~QR_DIRECTORY;
2258 if (yesbstr("ulallowed")) {
2259 er_flags |= QR_UPLOAD;
2260 } else {
2261 er_flags &= ~QR_UPLOAD;
2264 if (yesbstr("dlallowed")) {
2265 er_flags |= QR_DOWNLOAD;
2266 } else {
2267 er_flags &= ~QR_DOWNLOAD;
2270 if (yesbstr("visdir")) {
2271 er_flags |= QR_VISDIR;
2272 } else {
2273 er_flags &= ~QR_VISDIR;
2276 Ptr = sbstr("anon");
2278 er_flags &= ~(QR_ANONONLY | QR_ANONOPT);
2279 if (!strcmp(ChrPtr(Ptr), "anononly"))
2280 er_flags |= QR_ANONONLY;
2281 if (!strcmp(ChrPtr(Ptr), "anon2"))
2282 er_flags |= QR_ANONOPT;
2284 bump = yesbstr("bump");
2286 er_floor = ibstr("er_floor");
2288 StrBufPrintf(Buf, "SETR %s|%s|%s|%u|%d|%d|%d|%d|%u",
2289 ChrPtr(er_name),
2290 ChrPtr(er_password),
2291 ChrPtr(er_dirname),
2292 er_flags,
2293 bump,
2294 er_floor,
2295 er_listingorder,
2296 er_defaultview,
2297 er_flags2);
2298 serv_putbuf(Buf);
2299 StrBuf_ServGetln(Buf);
2300 if (GetServerStatus(Buf, NULL) != 2) {
2301 strcpy(WC->ImportantMessage, &ChrPtr(Buf)[4]);
2302 display_editroom();
2303 FreeStrBuf(&Buf);
2304 FreeStrBuf(&er_name);
2305 FreeStrBuf(&er_password);
2306 FreeStrBuf(&er_dirname);
2307 FreeStrBuf(&er_roomaide);
2308 return;
2310 gotoroom(er_name);
2312 if (StrLength(er_roomaide) > 0) {
2313 serv_printf("SETA %s", ChrPtr(er_roomaide));
2314 StrBuf_ServGetln(Buf);
2315 if (GetServerStatus(Buf, NULL) != 2) {
2316 strcpy(WC->ImportantMessage, &ChrPtr(Buf)[4]);
2317 display_main_menu();
2318 FreeStrBuf(&Buf);
2319 FreeStrBuf(&er_name);
2320 FreeStrBuf(&er_password);
2321 FreeStrBuf(&er_dirname);
2322 FreeStrBuf(&er_roomaide);
2323 return;
2326 gotoroom(er_name);
2327 strcpy(WC->ImportantMessage, _("Your changes have been saved."));
2328 display_editroom();
2329 FreeStrBuf(&Buf);
2330 FreeStrBuf(&er_name);
2331 FreeStrBuf(&er_password);
2332 FreeStrBuf(&er_dirname);
2333 FreeStrBuf(&er_roomaide);
2334 return;
2339 * Display form for Invite, Kick, and show Who Knows a room
2341 void do_invt_kick(void) {
2342 char buf[SIZ], room[SIZ], username[SIZ];
2344 serv_puts("GETR");
2345 serv_getln(buf, sizeof buf);
2347 if (buf[0] != '2') {
2348 escputs(&buf[4]);
2349 return;
2351 extract_token(room, &buf[4], 0, '|', sizeof room);
2353 strcpy(username, bstr("username"));
2355 if (havebstr("kick_button")) {
2356 sprintf(buf, "KICK %s", username);
2357 serv_puts(buf);
2358 serv_getln(buf, sizeof buf);
2360 if (buf[0] != '2') {
2361 strcpy(WC->ImportantMessage, &buf[4]);
2362 } else {
2363 sprintf(WC->ImportantMessage,
2364 _("<B><I>User %s kicked out of room %s.</I></B>\n"),
2365 username, room);
2369 if (havebstr("invite_button")) {
2370 sprintf(buf, "INVT %s", username);
2371 serv_puts(buf);
2372 serv_getln(buf, sizeof buf);
2374 if (buf[0] != '2') {
2375 strcpy(WC->ImportantMessage, &buf[4]);
2376 } else {
2377 sprintf(WC->ImportantMessage,
2378 _("<B><I>User %s invited to room %s.</I></B>\n"),
2379 username, room);
2383 display_editroom();
2389 * Display form for Invite, Kick, and show Who Knows a room
2391 void display_whok(void)
2393 char buf[SIZ], room[SIZ], username[SIZ];
2395 serv_puts("GETR");
2396 serv_getln(buf, sizeof buf);
2398 if (buf[0] != '2') {
2399 escputs(&buf[4]);
2400 return;
2402 extract_token(room, &buf[4], 0, '|', sizeof room);
2405 wprintf("<table border=0 CELLSPACING=10><tr VALIGN=TOP><td>");
2406 wprintf(_("The users listed below have access to this room. "
2407 "To remove a user from the access list, select the user "
2408 "name from the list and click 'Kick'."));
2409 wprintf("<br /><br />");
2411 wprintf("<CENTER><form method=\"POST\" action=\"do_invt_kick\">\n");
2412 wprintf("<input type=\"hidden\" name=\"nonce\" value=\"%d\">\n", WC->nonce);
2413 wprintf("<input type=\"hidden\" NAME=\"tab\" VALUE=\"access\">\n");
2414 wprintf("<select NAME=\"username\" SIZE=\"10\" style=\"width:100%%\">\n");
2415 serv_puts("WHOK");
2416 serv_getln(buf, sizeof buf);
2417 if (buf[0] == '1') {
2418 while (serv_getln(buf, sizeof buf), strcmp(buf, "000")) {
2419 extract_token(username, buf, 0, '|', sizeof username);
2420 wprintf("<OPTION>");
2421 escputs(username);
2422 wprintf("\n");
2425 wprintf("</select><br />\n");
2427 wprintf("<input type=\"submit\" name=\"kick_button\" value=\"%s\">", _("Kick"));
2428 wprintf("</form></CENTER>\n");
2430 wprintf("</td><td>");
2431 wprintf(_("To grant another user access to this room, enter the "
2432 "user name in the box below and click 'Invite'."));
2433 wprintf("<br /><br />");
2435 wprintf("<CENTER><form method=\"POST\" action=\"do_invt_kick\">\n");
2436 wprintf("<input type=\"hidden\" NAME=\"tab\" VALUE=\"access\">\n");
2437 wprintf("<input type=\"hidden\" name=\"nonce\" value=\"%d\">\n", WC->nonce);
2438 wprintf(_("Invite:"));
2439 wprintf(" ");
2440 wprintf("<input type=\"text\" name=\"username\" id=\"username_id\" style=\"width:100%%\"><br />\n"
2441 "<input type=\"hidden\" name=\"invite_button\" value=\"Invite\">"
2442 "<input type=\"submit\" value=\"%s\">"
2443 "</form></CENTER>\n", _("Invite"));
2444 /* Pop open an address book -- begin **/
2445 wprintf(
2446 "<a href=\"javascript:PopOpenAddressBook('username_id|%s');\" "
2447 "title=\"%s\">"
2448 "<img align=middle border=0 width=24 height=24 src=\"static/viewcontacts_24x.gif\">"
2449 "&nbsp;%s</a>",
2450 _("User"),
2451 _("Users"), _("Users")
2453 /* Pop open an address book -- end **/
2455 wprintf("</td></tr></table>\n");
2456 address_book_popup();
2457 wDumpContent(1);
2463 * display the form for entering a new room
2465 void display_entroom(void)
2467 int i;
2468 char buf[SIZ];
2470 serv_puts("CRE8 0");
2471 serv_getln(buf, sizeof buf);
2473 if (buf[0] != '2') {
2474 strcpy(WC->ImportantMessage, &buf[4]);
2475 display_main_menu();
2476 return;
2479 output_headers(1, 1, 1, 0, 0, 0);
2481 svprintf(HKEY("BOXTITLE"), WCS_STRING, _("Create a new room"));
2482 do_template("beginbox", NULL);
2484 wprintf("<form name=\"create_room_form\" method=\"POST\" action=\"entroom\">\n");
2485 wprintf("<input type=\"hidden\" name=\"nonce\" value=\"%d\">\n", WC->nonce);
2487 wprintf("<table class=\"altern\"> ");
2489 wprintf("<tr class=\"even\"><td>");
2490 wprintf(_("Name of room: "));
2491 wprintf("</td><td>");
2492 wprintf("<input type=\"text\" NAME=\"er_name\" MAXLENGTH=\"127\">\n");
2493 wprintf("</td></tr>");
2495 wprintf("<tr class=\"odd\"><td>");
2496 wprintf(_("Resides on floor: "));
2497 wprintf("</td><td>");
2498 load_floorlist();
2499 wprintf("<select name=\"er_floor\" size=\"1\">\n");
2500 for (i = 0; i < 128; ++i)
2501 if (!IsEmptyStr(floorlist[i])) {
2502 wprintf("<option ");
2503 wprintf("value=\"%d\">", i);
2504 escputs(floorlist[i]);
2505 wprintf("</option>\n");
2507 wprintf("</select>\n");
2508 wprintf("</td></tr>");
2511 * Our clever little snippet of JavaScript automatically selects
2512 * a public room if the view is set to Bulletin Board or wiki, and
2513 * it selects a mailbox room otherwise. The user can override this,
2514 * of course. We also disable the floor selector for mailboxes.
2516 wprintf("<tr class=\"even\"><td>");
2517 wprintf(_("Default view for room: "));
2518 wprintf("</td><td>");
2519 wprintf("<select name=\"er_view\" size=\"1\" OnChange=\""
2520 " if ( (this.form.er_view.value == 0) "
2521 " || (this.form.er_view.value == 6) ) { "
2522 " this.form.type[0].checked=true; "
2523 " this.form.er_floor.disabled = false; "
2524 " } "
2525 " else { "
2526 " this.form.type[4].checked=true; "
2527 " this.form.er_floor.disabled = true; "
2528 " } "
2529 "\">\n");
2530 for (i=0; i<(sizeof viewdefs / sizeof (char *)); ++i) {
2531 if (is_view_allowed_as_default(i)) {
2532 wprintf("<option %s value=\"%d\">",
2533 ((i == 0) ? "selected" : ""), i );
2534 escputs(viewdefs[i]);
2535 wprintf("</option>\n");
2538 wprintf("</select>\n");
2539 wprintf("</td></tr>");
2541 wprintf("<tr class=\"even\"><td>");
2542 wprintf(_("Type of room:"));
2543 wprintf("</td><td>");
2544 wprintf("<ul class=\"adminlist\">\n");
2546 wprintf("<li><input type=\"radio\" NAME=\"type\" VALUE=\"public\" ");
2547 wprintf("CHECKED OnChange=\""
2548 " if (this.form.type[0].checked == true) { "
2549 " this.form.er_floor.disabled = false; "
2550 " } "
2551 "\"> ");
2552 wprintf(_("Public (automatically appears to everyone)"));
2553 wprintf("</li>");
2555 wprintf("\n<li><input type=\"radio\" NAME=\"type\" VALUE=\"hidden\" OnChange=\""
2556 " if (this.form.type[1].checked == true) { "
2557 " this.form.er_floor.disabled = false; "
2558 " } "
2559 "\"> ");
2560 wprintf(_("Private - hidden (accessible to anyone who knows its name)"));
2561 wprintf("</li>");
2563 wprintf("\n<li><input type=\"radio\" NAME=\"type\" VALUE=\"passworded\" OnChange=\""
2564 " if (this.form.type[2].checked == true) { "
2565 " this.form.er_floor.disabled = false; "
2566 " } "
2567 "\"> ");
2568 wprintf(_("Private - require password: "));
2569 wprintf("<input type=\"text\" NAME=\"er_password\" MAXLENGTH=\"9\">\n");
2570 wprintf("</li>");
2572 wprintf("<li><input type=\"radio\" NAME=\"type\" VALUE=\"invonly\" OnChange=\""
2573 " if (this.form.type[3].checked == true) { "
2574 " this.form.er_floor.disabled = false; "
2575 " } "
2576 "\"> ");
2577 wprintf(_("Private - invitation only"));
2578 wprintf("</li>");
2580 wprintf("\n<li><input type=\"radio\" NAME=\"type\" VALUE=\"personal\" "
2581 "OnChange=\""
2582 " if (this.form.type[4].checked == true) { "
2583 " this.form.er_floor.disabled = true; "
2584 " } "
2585 "\"> ");
2586 wprintf(_("Personal (mailbox for you only)"));
2587 wprintf("</li>");
2589 wprintf("\n</ul>\n");
2590 wprintf("</td></tr></table>\n");
2592 wprintf("<div class=\"buttons\">\n");
2593 wprintf("<input type=\"submit\" name=\"ok_button\" value=\"%s\">", _("Create new room"));
2594 wprintf("&nbsp;");
2595 wprintf("<input type=\"submit\" name=\"cancel_button\" value=\"%s\">", _("Cancel"));
2596 wprintf("</div>\n");
2597 wprintf("</form>\n<hr />");
2598 serv_printf("MESG roomaccess");
2599 serv_getln(buf, sizeof buf);
2600 if (buf[0] == '1') {
2601 fmout("LEFT");
2604 do_template("endbox", NULL);
2606 wDumpContent(1);
2613 * support function for entroom() -- sets the default view
2615 void er_set_default_view(int newview) {
2617 char buf[SIZ];
2619 char rm_name[SIZ];
2620 char rm_pass[SIZ];
2621 char rm_dir[SIZ];
2622 int rm_bits1;
2623 int rm_floor;
2624 int rm_listorder;
2625 int rm_bits2;
2627 serv_puts("GETR");
2628 serv_getln(buf, sizeof buf);
2629 if (buf[0] != '2') return;
2631 extract_token(rm_name, &buf[4], 0, '|', sizeof rm_name);
2632 extract_token(rm_pass, &buf[4], 1, '|', sizeof rm_pass);
2633 extract_token(rm_dir, &buf[4], 2, '|', sizeof rm_dir);
2634 rm_bits1 = extract_int(&buf[4], 3);
2635 rm_floor = extract_int(&buf[4], 4);
2636 rm_listorder = extract_int(&buf[4], 5);
2637 rm_bits2 = extract_int(&buf[4], 7);
2639 serv_printf("SETR %s|%s|%s|%d|0|%d|%d|%d|%d",
2640 rm_name, rm_pass, rm_dir, rm_bits1, rm_floor,
2641 rm_listorder, newview, rm_bits2
2643 serv_getln(buf, sizeof buf);
2649 * Create a new room
2651 void entroom(void)
2653 char buf[SIZ];
2654 const StrBuf *er_name;
2655 const StrBuf *er_type;
2656 const StrBuf *er_password;
2657 int er_floor;
2658 int er_num_type;
2659 int er_view;
2661 if (!havebstr("ok_button")) {
2662 strcpy(WC->ImportantMessage,
2663 _("Cancelled. No new room was created."));
2664 display_main_menu();
2665 return;
2667 er_name = sbstr("er_name");
2668 er_type = sbstr("type");
2669 er_password = sbstr("er_password");
2670 er_floor = ibstr("er_floor");
2671 er_view = ibstr("er_view");
2673 er_num_type = 0;
2674 if (!strcmp(ChrPtr(er_type), "hidden"))
2675 er_num_type = 1;
2676 else if (!strcmp(ChrPtr(er_type), "passworded"))
2677 er_num_type = 2;
2678 else if (!strcmp(ChrPtr(er_type), "invonly"))
2679 er_num_type = 3;
2680 else if (!strcmp(ChrPtr(er_type), "personal"))
2681 er_num_type = 4;
2683 serv_printf("CRE8 1|%s|%d|%s|%d|%d|%d",
2684 ChrPtr(er_name),
2685 er_num_type,
2686 ChrPtr(er_password),
2687 er_floor,
2689 er_view);
2691 serv_getln(buf, sizeof buf);
2692 if (buf[0] != '2') {
2693 strcpy(WC->ImportantMessage, &buf[4]);
2694 display_main_menu();
2695 return;
2697 /** TODO: Room created, now udate the left hand icon bar for this user */
2698 burn_folder_cache(0); /* burn the old folder cache */
2701 gotoroom(er_name);
2702 do_change_view(er_view); /* Now go there */
2707 * \brief display the screen to enter a private room
2709 void display_private(char *rname, int req_pass)
2711 WCTemplputParams SubTP;
2712 StrBuf *Buf;
2713 output_headers(1, 1, 1, 0, 0, 0);
2715 Buf = NewStrBufPlain(_("Go to a hidden room"), -1);
2716 memset(&SubTP, 0, sizeof(WCTemplputParams));
2717 SubTP.Filter.ContextType = CTX_STRBUF;
2718 SubTP.Context = Buf;
2719 DoTemplate(HKEY("beginbox"), NULL, &SubTP);
2721 FreeStrBuf(&Buf);
2723 wprintf("<p>");
2724 wprintf(_("If you know the name of a hidden (guess-name) or "
2725 "passworded room, you can enter that room by typing "
2726 "its name below. Once you gain access to a private "
2727 "room, it will appear in your regular room listings "
2728 "so you don't have to keep returning here."));
2729 wprintf("</p>");
2731 wprintf("<form method=\"post\" action=\"goto_private\">\n");
2732 wprintf("<input type=\"hidden\" name=\"nonce\" value=\"%d\">\n", WC->nonce);
2734 wprintf("<table class=\"altern\"> "
2735 "<tr class=\"even\"><td>");
2736 wprintf(_("Enter room name:"));
2737 wprintf("</td><td>"
2738 "<input type=\"text\" name=\"gr_name\" "
2739 "value=\"%s\" maxlength=\"128\">\n", rname);
2741 if (req_pass) {
2742 wprintf("</td></tr><tr class=\"odd\"><td>");
2743 wprintf(_("Enter room password:"));
2744 wprintf("</td><td>");
2745 wprintf("<input type=\"password\" name=\"gr_pass\" maxlength=\"9\">\n");
2747 wprintf("</td></tr></table>\n");
2749 wprintf("<div class=\"buttons\">\n");
2750 wprintf("<input type=\"submit\" name=\"ok_button\" value=\"%s\">"
2751 "&nbsp;"
2752 "<input type=\"submit\" name=\"cancel_button\" value=\"%s\">",
2753 _("Go there"),
2754 _("Cancel")
2756 wprintf("</div></form>\n");
2758 do_template("endbox", NULL);
2760 wDumpContent(1);
2764 * \brief goto a private room
2766 void goto_private(void)
2768 char hold_rm[SIZ];
2769 char buf[SIZ];
2771 if (!havebstr("ok_button")) {
2772 display_main_menu();
2773 return;
2775 strcpy(hold_rm, ChrPtr(WC->wc_roomname));
2776 serv_printf("GOTO %s|%s",
2777 bstr("gr_name"),
2778 bstr("gr_pass"));
2779 serv_getln(buf, sizeof buf);
2781 if (buf[0] == '2') {
2782 smart_goto(sbstr("gr_name"));
2783 return;
2785 if (!strncmp(buf, "540", 3)) {
2786 display_private(bstr("gr_name"), 1);
2787 return;
2789 output_headers(1, 1, 1, 0, 0, 0);
2790 wprintf("%s\n", &buf[4]);
2791 wDumpContent(1);
2792 return;
2797 * \brief display the screen to zap a room
2799 void display_zap(void)
2801 output_headers(1, 1, 2, 0, 0, 0);
2803 wprintf("<div id=\"banner\">\n");
2804 wprintf("<h1>");
2805 wprintf(_("Zap (forget/unsubscribe) the current room"));
2806 wprintf("</h1>\n");
2807 wprintf("</div>\n");
2809 wprintf("<div id=\"content\" class=\"service\">\n");
2811 wprintf(_("If you select this option, <em>%s</em> will "
2812 "disappear from your room list. Is this what you wish "
2813 "to do?<br />\n"), ChrPtr(WC->wc_roomname));
2815 wprintf("<form method=\"POST\" action=\"zap\">\n");
2816 wprintf("<input type=\"hidden\" name=\"nonce\" value=\"%d\">\n", WC->nonce);
2817 wprintf("<input type=\"submit\" NAME=\"ok_button\" VALUE=\"%s\">", _("Zap this room"));
2818 wprintf("&nbsp;");
2819 wprintf("<input type=\"submit\" NAME=\"cancel_button\" VALUE=\"%s\">", _("Cancel"));
2820 wprintf("</form>\n");
2821 wDumpContent(1);
2826 * \brief zap a room
2828 void zap(void)
2830 char buf[SIZ];
2831 StrBuf *final_destination;
2834 * If the forget-room routine fails for any reason, we fall back
2835 * to the current room; otherwise, we go to the Lobby
2837 final_destination = NewStrBufDup(WC->wc_roomname);
2839 if (havebstr("ok_button")) {
2840 serv_printf("GOTO %s", ChrPtr(WC->wc_roomname));
2841 serv_getln(buf, sizeof buf);
2842 if (buf[0] == '2') {
2843 serv_puts("FORG");
2844 serv_getln(buf, sizeof buf);
2845 if (buf[0] == '2') {
2846 FlushStrBuf(final_destination);
2847 StrBufAppendBufPlain(final_destination, HKEY("_BASEROOM_"), 0);
2851 smart_goto(final_destination);
2852 FreeStrBuf(&final_destination);
2858 * \brief Delete the current room
2860 void delete_room(void)
2862 char buf[SIZ];
2865 serv_puts("KILL 1");
2866 serv_getln(buf, sizeof buf);
2867 burn_folder_cache(0); /* Burn the cahce of known rooms to update the icon bar */
2868 if (buf[0] != '2') {
2869 strcpy(WC->ImportantMessage, &buf[4]);
2870 display_main_menu();
2871 return;
2872 } else {
2873 StrBuf *Buf;
2875 Buf = NewStrBufPlain(HKEY("_BASEROOM_"));
2876 smart_goto(Buf);
2877 FreeStrBuf(&Buf);
2884 * \brief Perform changes to a room's network configuration
2886 void netedit(void) {
2887 FILE *fp;
2888 char buf[SIZ];
2889 char line[SIZ];
2890 char cmpa0[SIZ];
2891 char cmpa1[SIZ];
2892 char cmpb0[SIZ];
2893 char cmpb1[SIZ];
2894 int i, num_addrs;
2895 /*/ TODO: do line dynamic! */
2896 if (havebstr("line_pop3host")) {
2897 strcpy(line, bstr("prefix"));
2898 strcat(line, bstr("line_pop3host"));
2899 strcat(line, "|");
2900 strcat(line, bstr("line_pop3user"));
2901 strcat(line, "|");
2902 strcat(line, bstr("line_pop3pass"));
2903 strcat(line, "|");
2904 strcat(line, ibstr("line_pop3keep") ? "1" : "0" );
2905 strcat(line, "|");
2906 sprintf(&line[strlen(line)],"%ld", lbstr("line_pop3int"));
2907 strcat(line, bstr("suffix"));
2909 else if (havebstr("line")) {
2910 strcpy(line, bstr("prefix"));
2911 strcat(line, bstr("line"));
2912 strcat(line, bstr("suffix"));
2914 else {
2915 display_editroom();
2916 return;
2920 fp = tmpfile();
2921 if (fp == NULL) {
2922 display_editroom();
2923 return;
2926 serv_puts("GNET");
2927 serv_getln(buf, sizeof buf);
2928 if (buf[0] != '1') {
2929 fclose(fp);
2930 display_editroom();
2931 return;
2934 /** This loop works for add *or* remove. Spiffy, eh? */
2935 while (serv_getln(buf, sizeof buf), strcmp(buf, "000")) {
2936 extract_token(cmpa0, buf, 0, '|', sizeof cmpa0);
2937 extract_token(cmpa1, buf, 1, '|', sizeof cmpa1);
2938 extract_token(cmpb0, line, 0, '|', sizeof cmpb0);
2939 extract_token(cmpb1, line, 1, '|', sizeof cmpb1);
2940 if ( (strcasecmp(cmpa0, cmpb0))
2941 || (strcasecmp(cmpa1, cmpb1)) ) {
2942 fprintf(fp, "%s\n", buf);
2946 rewind(fp);
2947 serv_puts("SNET");
2948 serv_getln(buf, sizeof buf);
2949 if (buf[0] != '4') {
2950 fclose(fp);
2951 display_editroom();
2952 return;
2955 while (fgets(buf, sizeof buf, fp) != NULL) {
2956 buf[strlen(buf)-1] = 0;
2957 serv_puts(buf);
2960 if (havebstr("add_button")) {
2961 num_addrs = num_tokens(bstr("line"), ',');
2962 if (num_addrs < 2) {
2963 /* just adding one node or address */
2964 serv_puts(line);
2966 else {
2967 /* adding multiple addresses separated by commas */
2968 for (i=0; i<num_addrs; ++i) {
2969 strcpy(line, bstr("prefix"));
2970 extract_token(buf, bstr("line"), i, ',', sizeof buf);
2971 striplt(buf);
2972 strcat(line, buf);
2973 strcat(line, bstr("suffix"));
2974 serv_puts(line);
2979 serv_puts("000");
2980 fclose(fp);
2981 display_editroom();
2987 * \brief Convert a room name to a folder-ish-looking name.
2988 * \param folder the folderish name
2989 * \param room the room name
2990 * \param floor the floor name
2991 * \param is_mailbox is it a mailbox?
2993 void room_to_folder(char *folder, char *room, int floor, int is_mailbox)
2995 int i, len;
2998 * For mailboxes, just do it straight...
3000 if (is_mailbox) {
3001 sprintf(folder, "My folders|%s", room);
3005 * Otherwise, prefix the floor name as a "public folders" moniker
3007 else {
3008 if (floor > MAX_FLOORS) {
3009 wc_backtrace ();
3010 sprintf(folder, "%%%%%%|%s", room);
3012 else {
3013 sprintf(folder, "%s|%s", floorlist[floor], room);
3018 * Replace "\" characters with "|" for pseudo-folder-delimiting
3020 len = strlen (folder);
3021 for (i=0; i<len; ++i) {
3022 if (folder[i] == '\\') folder[i] = '|';
3030 * \brief Back end for change_view()
3031 * \param newview set newview???
3033 void do_change_view(int newview) {
3034 char buf[SIZ];
3036 serv_printf("VIEW %d", newview);
3037 serv_getln(buf, sizeof buf);
3038 WC->wc_view = newview;
3039 smart_goto(WC->wc_roomname);
3045 * \brief Change the view for this room
3047 void change_view(void) {
3048 int view;
3050 view = lbstr("view");
3051 do_change_view(view);
3056 * \brief One big expanded tree list view --- like a folder list
3057 * \param fold the folder to view
3058 * \param max_folders how many folders???
3059 * \param num_floors hom many floors???
3061 void do_folder_view(struct folder *fold, int max_folders, int num_floors) {
3062 char buf[SIZ];
3063 int levels;
3064 int i;
3065 int has_subfolders = 0;
3066 int *parents;
3068 parents = malloc(max_folders * sizeof(int));
3070 /** BEGIN TREE MENU */
3071 wprintf("<div id=\"roomlist_div\">Loading folder list...</div>\n");
3073 /** include NanoTree */
3074 wprintf("<script type=\"text/javascript\" src=\"static/nanotree.js\"></script>\n");
3076 /** initialize NanoTree */
3077 wprintf("<script type=\"text/javascript\"> \n"
3078 " showRootNode = false; \n"
3079 " sortNodes = false; \n"
3080 " dragable = false; \n"
3081 " \n"
3082 " function standardClick(treeNode) { \n"
3083 " } \n"
3084 " \n"
3085 " var closedGif = 'static/folder_closed.gif'; \n"
3086 " var openGif = 'static/folder_open.gif'; \n"
3087 " \n"
3088 " rootNode = new TreeNode(1, 'root node - hide'); \n"
3091 levels = 0;
3092 for (i=0; i<max_folders; ++i) {
3094 has_subfolders = 0;
3095 if ((i+1) < max_folders) {
3096 int len;
3097 len = strlen(fold[i].name);
3098 if ( (!strncasecmp(fold[i].name, fold[i+1].name, len))
3099 && (fold[i+1].name[len] == '|') ) {
3100 has_subfolders = 1;
3104 levels = num_tokens(fold[i].name, '|');
3105 parents[levels] = i;
3107 wprintf("var node%d = new TreeNode(%d, '", i, i);
3109 if (fold[i].selectable) {
3110 wprintf("<a href=\"dotgoto?room=");
3111 urlescputs(fold[i].room);
3112 wprintf("\">");
3115 if (levels == 1) {
3116 wprintf("<span class=\"roomlist_floor\">");
3118 else if (fold[i].hasnewmsgs) {
3119 wprintf("<span class=\"roomlist_new\">");
3121 else {
3122 wprintf("<span class=\"roomlist_old\">");
3124 extract_token(buf, fold[i].name, levels-1, '|', sizeof buf);
3125 escputs(buf);
3126 wprintf("</span>");
3128 wprintf("</a>', ");
3129 if (has_subfolders) {
3130 wprintf("new Array(closedGif, openGif)");
3132 else if (fold[i].view == VIEW_ADDRESSBOOK) {
3133 wprintf("'static/viewcontacts_16x.gif'");
3135 else if (fold[i].view == VIEW_CALENDAR) {
3136 wprintf("'static/calarea_16x.gif'");
3138 else if (fold[i].view == VIEW_CALBRIEF) {
3139 wprintf("'static/calarea_16x.gif'");
3141 else if (fold[i].view == VIEW_TASKS) {
3142 wprintf("'static/taskmanag_16x.gif'");
3144 else if (fold[i].view == VIEW_NOTES) {
3145 wprintf("'static/storenotes_16x.gif'");
3147 else if (fold[i].view == VIEW_MAILBOX) {
3148 wprintf("'static/privatemess_16x.gif'");
3150 else {
3151 wprintf("'static/chatrooms_16x.gif'");
3153 wprintf(", '");
3154 urlescputs(fold[i].name);
3155 wprintf("');\n");
3157 if (levels < 2) {
3158 wprintf("rootNode.addChild(node%d);\n", i);
3160 else {
3161 wprintf("node%d.addChild(node%d);\n", parents[levels-1], i);
3165 wprintf("container = document.getElementById('roomlist_div'); \n"
3166 "showTree(''); \n"
3167 "</script>\n"
3170 free(parents);
3171 /** END TREE MENU */
3175 * \brief Boxes and rooms and lists ... oh my!
3176 * \param fold the folder to view
3177 * \param max_folders how many folders???
3178 * \param num_floors hom many floors???
3180 void do_rooms_view(struct folder *fold, int max_folders, int num_floors) {
3181 char buf[256];
3182 char floor_name[256];
3183 char old_floor_name[256];
3184 int levels, oldlevels;
3185 int i, t;
3186 int num_boxes = 0;
3187 static int columns = 3;
3188 int boxes_per_column = 0;
3189 int current_column = 0;
3190 int nf;
3192 strcpy(floor_name, "");
3193 strcpy(old_floor_name, "");
3195 nf = num_floors;
3196 while (nf % columns != 0) ++nf;
3197 boxes_per_column = (nf / columns);
3198 if (boxes_per_column < 1) boxes_per_column = 1;
3200 /** Outer table (for columnization) */
3201 wprintf("<table BORDER=0 WIDTH=96%% CELLPADDING=5>"
3202 "<tr><td valign=top>");
3204 levels = 0;
3205 oldlevels = 0;
3206 for (i=0; i<max_folders; ++i) {
3208 levels = num_tokens(fold[i].name, '|');
3209 extract_token(floor_name, fold[i].name, 0,
3210 '|', sizeof floor_name);
3212 if ( (strcasecmp(floor_name, old_floor_name))
3213 && (!IsEmptyStr(old_floor_name)) ) {
3214 /* End inner box */
3215 do_template("endbox", NULL);
3216 wprintf("<br>");
3218 ++num_boxes;
3219 if ((num_boxes % boxes_per_column) == 0) {
3220 ++current_column;
3221 if (current_column < columns) {
3222 wprintf("</td><td valign=top>\n");
3226 strcpy(old_floor_name, floor_name);
3228 if (levels == 1) {
3229 StrBuf *Buf;
3230 WCTemplputParams SubTP;
3232 Buf = NewStrBufPlain(floor_name, -1);
3233 memset(&SubTP, 0, sizeof(WCTemplputParams));
3234 SubTP.Filter.ContextType = CTX_STRBUF;
3235 SubTP.Context = Buf;
3236 DoTemplate(HKEY("beginbox"), NULL, &SubTP);
3238 FreeStrBuf(&Buf);
3241 oldlevels = levels;
3243 if (levels > 1) {
3244 wprintf("&nbsp;");
3245 if (levels>2) for (t=0; t<(levels-2); ++t) wprintf("&nbsp;&nbsp;&nbsp;");
3246 if (fold[i].selectable) {
3247 wprintf("<a href=\"dotgoto?room=");
3248 urlescputs(fold[i].room);
3249 wprintf("\">");
3251 else {
3252 wprintf("<i>");
3254 if (fold[i].hasnewmsgs) {
3255 wprintf("<span class=\"roomlist_new\">");
3257 else {
3258 wprintf("<span class=\"roomlist_old\">");
3260 extract_token(buf, fold[i].name, levels-1, '|', sizeof buf);
3261 escputs(buf);
3262 wprintf("</span>");
3263 if (fold[i].selectable) {
3264 wprintf("</A>");
3266 else {
3267 wprintf("</i>");
3269 if (!strcasecmp(fold[i].name, "My Folders|Mail")) {
3270 wprintf(" (INBOX)");
3272 wprintf("<br />\n");
3275 /** End the final inner box */
3276 do_template("endbox", NULL);
3278 wprintf("</td></tr></table>\n");
3282 * \brief print a floor div???
3283 * \param which_floordiv name of the floordiv???
3285 void set_floordiv_expanded(void) {
3286 wcsession *WCC = WC;
3287 StrBuf *FloorDiv;
3289 FloorDiv = NewStrBuf();
3290 StrBufAppendBuf(FloorDiv, WCC->UrlFragment2, 0);
3291 set_preference("floordiv_expanded", FloorDiv, 1);
3292 WCC->floordiv_expanded = FloorDiv;
3296 * \brief view the iconbar
3297 * \param fold the folder to view
3298 * \param max_folders how many folders???
3299 * \param num_floors hom many floors???
3301 void do_iconbar_view(struct folder *fold, int max_folders, int num_floors) {
3302 char buf[256];
3303 char floor_name[256];
3304 char old_floor_name[256];
3305 char floordivtitle[256];
3306 char floordiv_id[32];
3307 int levels, oldlevels;
3308 int i, t;
3309 char *icon = NULL;
3311 strcpy(floor_name, "");
3312 strcpy(old_floor_name, "");
3314 levels = 0;
3315 oldlevels = 0;
3316 for (i=0; i<max_folders; ++i) {
3318 levels = num_tokens(fold[i].name, '|');
3319 extract_token(floor_name, fold[i].name, 0,
3320 '|', sizeof floor_name);
3322 if ( (strcasecmp(floor_name, old_floor_name))
3323 && (!IsEmptyStr(old_floor_name)) ) {
3324 /** End inner box */
3325 wprintf("<br>\n");
3326 wprintf("</div>\n"); /** floordiv */
3328 strcpy(old_floor_name, floor_name);
3330 if (levels == 1) {
3331 /** Begin floor */
3332 stresc(floordivtitle, 256, floor_name, 0, 0);
3333 sprintf(floordiv_id, "floordiv%d", i);
3334 wprintf("<span class=\"ib_roomlist_floor\" "
3335 "onClick=\"expand_floor('%s')\">"
3336 "%s</span><br>\n", floordiv_id, floordivtitle);
3337 wprintf("<div id=\"%s\" style=\"display:%s\">",
3338 floordiv_id,
3339 (!strcasecmp(floordiv_id, ChrPtr(WC->floordiv_expanded)) ? "block" : "none")
3343 oldlevels = levels;
3345 if (levels > 1) {
3346 wprintf("<div id=\"roomdiv%d\">", i);
3347 wprintf("&nbsp;");
3348 if (levels>2) for (t=0; t<(levels-2); ++t) wprintf("&nbsp;");
3350 /** choose the icon */
3351 if (fold[i].view == VIEW_ADDRESSBOOK) {
3352 icon = "viewcontacts_16x.gif" ;
3354 else if (fold[i].view == VIEW_CALENDAR) {
3355 icon = "calarea_16x.gif" ;
3357 else if (fold[i].view == VIEW_CALBRIEF) {
3358 icon = "calarea_16x.gif" ;
3360 else if (fold[i].view == VIEW_TASKS) {
3361 icon = "taskmanag_16x.gif" ;
3363 else if (fold[i].view == VIEW_NOTES) {
3364 icon = "storenotes_16x.gif" ;
3366 else if (fold[i].view == VIEW_MAILBOX) {
3367 icon = "privatemess_16x.gif" ;
3369 else {
3370 icon = "chatrooms_16x.gif" ;
3373 if (fold[i].selectable) {
3374 wprintf("<a href=\"dotgoto?room=");
3375 urlescputs(fold[i].room);
3376 wprintf("\">");
3377 wprintf("<img border=0 src=\"static/%s\" alt=\"\"> ", icon);
3379 else {
3380 wprintf("<i>");
3382 if (fold[i].hasnewmsgs) {
3383 wprintf("<span class=\"ib_roomlist_new\">");
3385 else {
3386 wprintf("<span class=\"ib_roomlist_old\">");
3388 extract_token(buf, fold[i].name, levels-1, '|', sizeof buf);
3389 escputs(buf);
3390 if (!strcasecmp(fold[i].name, "My Folders|Mail")) {
3391 wprintf(" (INBOX)");
3393 wprintf("</span>");
3394 if (fold[i].selectable) {
3395 wprintf("</A>");
3397 else {
3398 wprintf("</i>");
3400 wprintf("<br />");
3401 wprintf("</div>\n"); /** roomdiv */
3404 wprintf("</div>\n"); /** floordiv */
3412 * \brief Burn the cached folder list.
3413 * \param age How old the cahce needs to be before we burn it.
3416 void burn_folder_cache(time_t age)
3418 /** If our cached folder list is very old, burn it. */
3419 if (WC->cache_fold != NULL) {
3420 if ((time(NULL) - WC->cache_timestamp) > age) {
3421 free(WC->cache_fold);
3422 WC->cache_fold = NULL;
3431 * \brief Show the room list.
3432 * (only should get called by
3433 * knrooms() because that's where output_headers() is called from)
3434 * \param viewpref the view preferences???
3437 void list_all_rooms_by_floor(const char *viewpref) {
3438 char buf[SIZ];
3439 int swap = 0;
3440 struct folder *fold = NULL;
3441 struct folder ftmp;
3442 int max_folders = 0;
3443 int alloc_folders = 0;
3444 int *floor_mapping;
3445 int IDMax;
3446 int i, j;
3447 int ShowEmptyFloors;
3448 int ra_flags = 0;
3449 int flags = 0;
3450 int num_floors = 1; /** add an extra one for private folders */
3451 char buf3[SIZ];
3453 /** If our cached folder list is very old, burn it. */
3454 burn_folder_cache(300);
3456 /** Can we do the iconbar roomlist from cache? */
3457 if ((WC->cache_fold != NULL) && (!strcasecmp(viewpref, "iconbar"))) {
3458 do_iconbar_view(WC->cache_fold, WC->cache_max_folders, WC->cache_num_floors);
3459 return;
3462 /** Grab the floor table so we know how to build the list... */
3463 load_floorlist();
3465 /** Start with the mailboxes */
3466 max_folders = 1;
3467 alloc_folders = 1;
3468 fold = malloc(sizeof(struct folder));
3469 memset(fold, 0, sizeof(struct folder));
3470 strcpy(fold[0].name, "My folders");
3471 fold[0].is_mailbox = 1;
3473 /** Then add floors */
3474 serv_puts("LFLR");
3475 serv_getln(buf, sizeof buf);
3476 if (buf[0]=='1') while(serv_getln(buf, sizeof buf), strcmp(buf, "000")) {
3477 if (max_folders >= alloc_folders) {
3478 alloc_folders = max_folders + 100;
3479 fold = realloc(fold,
3480 alloc_folders * sizeof(struct folder));
3482 memset(&fold[max_folders], 0, sizeof(struct folder));
3483 extract_token(fold[max_folders].name, buf, 1, '|', sizeof fold[max_folders].name);
3484 extract_token(buf3, buf, 0, '|', SIZ);
3485 fold[max_folders].floor = atol (buf3);
3486 ++max_folders;
3487 ++num_floors;
3489 IDMax = 0;
3490 for (i=0; i<num_floors; i++)
3491 if (IDMax < fold[i].floor)
3492 IDMax = fold[i].floor;
3493 floor_mapping = malloc (sizeof (int) * (IDMax + 1));
3494 memset (floor_mapping, 0, sizeof (int) * (IDMax + 1));
3495 for (i=0; i<num_floors; i++)
3496 floor_mapping[fold[i].floor]=i;
3498 /** refresh the messages index for this room */
3499 /* TODO serv_puts("GOTO ");
3500 while (serv_getln(buf, sizeof buf), strcmp(buf, "000")); */
3501 /** Now add rooms */
3502 serv_puts("LKRA");
3503 serv_getln(buf, sizeof buf);
3504 if (buf[0]=='1') while(serv_getln(buf, sizeof buf), strcmp(buf, "000")) {
3505 if (max_folders >= alloc_folders) {
3506 alloc_folders = max_folders + 100;
3507 fold = realloc(fold,
3508 alloc_folders * sizeof(struct folder));
3510 memset(&fold[max_folders], 0, sizeof(struct folder));
3511 extract_token(fold[max_folders].room, buf, 0, '|', sizeof fold[max_folders].room);
3512 ra_flags = extract_int(buf, 5);
3513 flags = extract_int(buf, 1);
3514 fold[max_folders].floor = extract_int(buf, 2);
3515 fold[max_folders].hasnewmsgs =
3516 ((ra_flags & UA_HASNEWMSGS) ? 1 : 0 );
3517 if (flags & QR_MAILBOX) {
3518 fold[max_folders].is_mailbox = 1;
3520 fold[max_folders].view = extract_int(buf, 6);
3521 room_to_folder(fold[max_folders].name,
3522 fold[max_folders].room,
3523 fold[max_folders].floor,
3524 fold[max_folders].is_mailbox);
3525 fold[max_folders].selectable = 1;
3526 /* Increase the room count for the associtaed floor */
3527 if (fold[max_folders].is_mailbox) {
3528 fold[0].num_rooms++;
3530 else {
3531 i = floor_mapping[fold[max_folders].floor];
3532 fold[i].num_rooms++;
3534 ++max_folders;
3538 * Remove any floors that don't have rooms
3540 get_pref_yesno("emptyfloors", &ShowEmptyFloors, 0);
3541 if (ShowEmptyFloors)
3543 for (i=0; i<num_floors; i++)
3545 if (fold[i].num_rooms == 0) {
3546 for (j=i; j<max_folders; j++) {
3547 memcpy(&fold[j], &fold[j+1], sizeof(struct folder));
3549 max_folders--;
3550 num_floors--;
3551 i--;
3556 /** Bubble-sort the folder list */
3557 for (i=0; i<max_folders; ++i) {
3558 for (j=0; j<(max_folders-1)-i; ++j) {
3559 if (fold[j].is_mailbox == fold[j+1].is_mailbox) {
3560 swap = strcasecmp(fold[j].name, fold[j+1].name);
3562 else {
3563 if ( (fold[j+1].is_mailbox)
3564 && (!fold[j].is_mailbox)) {
3565 swap = 1;
3567 else {
3568 swap = 0;
3571 if (swap > 0) {
3572 memcpy(&ftmp, &fold[j], sizeof(struct folder));
3573 memcpy(&fold[j], &fold[j+1],
3574 sizeof(struct folder));
3575 memcpy(&fold[j+1], &ftmp,
3576 sizeof(struct folder));
3582 if (!strcasecmp(viewpref, "folders")) {
3583 do_folder_view(fold, max_folders, num_floors);
3585 else if (!strcasecmp(viewpref, "hackish_view")) {
3586 for (i=0; i<max_folders; ++i) {
3587 escputs(fold[i].name);
3588 wprintf("<br />\n");
3591 else if (!strcasecmp(viewpref, "iconbar")) {
3592 do_iconbar_view(fold, max_folders, num_floors);
3594 else {
3595 do_rooms_view(fold, max_folders, num_floors);
3598 /* Don't free the folder list ... cache it for future use! */
3599 if (WC->cache_fold != NULL) {
3600 free(WC->cache_fold);
3602 WC->cache_fold = fold;
3603 WC->cache_max_folders = max_folders;
3604 WC->cache_num_floors = num_floors;
3605 WC->cache_timestamp = time(NULL);
3606 free(floor_mapping);
3611 * \brief Do either a known rooms list or a folders list, depending on the
3612 * user's preference
3614 void knrooms(void)
3616 StrBuf *ListView = NULL;
3618 output_headers(1, 1, 2, 0, 0, 0);
3620 /** Determine whether the user is trying to change views */
3621 if (havebstr("view")) {
3622 ListView = NewStrBufPlain(bstr("view"), -1);
3623 set_preference("roomlistview", ListView, 1);
3625 /** Sanitize the input so its safe */
3626 if(!get_preference("roomlistview", &ListView) ||
3627 ((strcasecmp(ChrPtr(ListView), "folders") != 0) &&
3628 (strcasecmp(ChrPtr(ListView), "table") != 0)))
3630 if (ListView == NULL) {
3631 ListView = NewStrBufPlain("rooms", sizeof("rooms") - 1);
3632 set_preference("roomlistview", ListView, 0);
3634 else {
3635 StrBufPrintf(ListView, "rooms");
3636 save_preferences();
3640 /** title bar */
3641 wprintf("<div id=\"banner\">\n");
3642 wprintf("<div class=\"room_banner\" id=\"room_banner\">");
3643 wprintf("<h1>");
3644 if (!strcasecmp(ChrPtr(ListView), "rooms")) {
3645 wprintf(_("Room list"));
3647 else if (!strcasecmp(ChrPtr(ListView), "folders")) {
3648 wprintf(_("Folder list"));
3650 else if (!strcasecmp(ChrPtr(ListView), "table")) {
3651 wprintf(_("Room list"));
3653 wprintf("</h1></div>\n");
3655 /** offer the ability to switch views */
3656 wprintf("<div id=\"actiondiv\">");
3657 wprintf("<ul class=\"room_actions\">\n");
3658 wprintf("<li class=\"start_page\">");
3659 offer_start_page(NULL, &NoCtx);
3660 wprintf("</li>");
3661 wprintf("<li><form name=\"roomlistomatic\">\n"
3662 "<select name=\"newview\" size=\"1\" "
3663 "OnChange=\"location.href=roomlistomatic.newview.options"
3664 "[selectedIndex].value\">\n");
3666 wprintf("<option %s value=\"knrooms&view=rooms\">"
3667 "View as room list"
3668 "</option>\n",
3669 ( !strcasecmp(ChrPtr(ListView), "rooms") ? "SELECTED" : "" )
3672 wprintf("<option %s value=\"knrooms&view=folders\">"
3673 "View as folder list"
3674 "</option>\n",
3675 ( !strcasecmp(ChrPtr(ListView), "folders") ? "SELECTED" : "" )
3678 wprintf("</select>");
3679 wprintf("</form></li>");
3680 wprintf("</ul></div></div>\n");
3682 wprintf("<div id=\"content\" class=\"service\">\n");
3684 /** Display the room list in the user's preferred format */
3685 list_all_rooms_by_floor(ChrPtr(ListView));
3686 wDumpContent(1);
3692 * \brief Set the message expire policy for this room and/or floor
3694 void set_room_policy(void) {
3695 char buf[SIZ];
3697 if (!havebstr("ok_button")) {
3698 strcpy(WC->ImportantMessage,
3699 _("Cancelled. Changes were not saved."));
3700 display_editroom();
3701 return;
3704 serv_printf("SPEX room|%d|%d", ibstr("roompolicy"), ibstr("roomvalue"));
3705 serv_getln(buf, sizeof buf);
3706 strcpy(WC->ImportantMessage, &buf[4]);
3708 if (WC->axlevel >= 6) {
3709 strcat(WC->ImportantMessage, "<br />\n");
3710 serv_printf("SPEX floor|%d|%d", ibstr("floorpolicy"), ibstr("floorvalue"));
3711 serv_getln(buf, sizeof buf);
3712 strcat(WC->ImportantMessage, &buf[4]);
3715 display_editroom();
3718 HashList *GetFloorListHash(StrBuf *Target, WCTemplputParams *TP) {
3719 /* todo: check context */
3720 const char *Err;
3721 StrBuf *Buf;
3722 StrBuf *Buf2;
3723 HashList *floors;
3724 HashList *floor;
3725 floors = NewHash(1, NULL);
3726 Buf = NewStrBuf();
3727 serv_puts("LFLR"); /* get floors */
3728 StrBufTCP_read_line(Buf, &WC->serv_sock, 0, &Err); /* '100', we hope */
3729 if (ChrPtr(Buf)[0] == '1') while(StrBufTCP_read_line(Buf, &WC->serv_sock, 0, &Err), strcmp(ChrPtr(Buf), "000")) {
3730 int a;
3731 const char *floorNum;
3732 floor = NewHash(1, NULL);
3733 for(a=0; a<FLOOR_PARAM_LEN; a++) {
3734 Buf2 = NewStrBuf();
3735 StrBufExtract_token(Buf2, Buf, a, '|');
3736 if (a==0) {
3737 floorNum = ChrPtr(Buf2); /* hmm, should we copy Buf2 first? */
3739 Put(floor, FPKEY(a), Buf2, NULL);
3741 Put(floors, HKEY(floorNum), floor, NULL);
3743 FreeStrBuf(&Buf);
3744 return floors;
3747 void tmplput_FLOOR_Value(StrBuf *TemplBuffer, WCTemplputParams *TP)
3749 StrBuf *val;
3750 HashList *floor = (HashList *)(TP->Context);
3751 void *value;
3752 GetHash(floor, TKEY(0), &value);
3753 val = (StrBuf *)value;
3754 StrECMAEscAppend(TemplBuffer, val, 0);
3756 HashList *GetRoomListHashLKRA(StrBuf *Target, WCTemplputParams *TP)
3758 serv_puts("LKRA");
3759 return GetRoomListHash(Target, TP);
3761 HashList *GetRoomListHash(StrBuf *Target, WCTemplputParams *TP)
3763 /* TODO: Check context */
3764 HashList *rooms;
3765 HashList *room;
3766 StrBuf *buf;
3767 StrBuf *buf2;
3768 const char *Err;
3769 buf = NewStrBuf();
3770 rooms = NewHash(1, NULL);
3771 StrBufTCP_read_line(buf, &WC->serv_sock, 0, &Err);
3772 if (ChrPtr(buf)[0] == '1') while(StrBufTCP_read_line(buf, &WC->serv_sock, 0, &Err), strcmp(ChrPtr(buf), "000")) {
3773 int i;
3774 const char *rmName;
3775 room = NewHash(1, NULL);
3776 for(i=0; i<ROOM_PARAM_LEN; i++) {
3777 buf2 = NewStrBuf();
3778 StrBufExtract_token(buf2, buf, i, '|');
3779 if (i==0) {
3780 rmName = ChrPtr(buf2);
3782 Put(room, RPKEY(i), buf2, NULL);
3784 Put(rooms, rmName, strlen(rmName), room, NULL);
3786 SortByHashKey(rooms, 1);
3787 /*SortByPayload(rooms, SortRoomsByListOrder); */
3788 FreeStrBuf(&buf);
3789 return rooms;
3791 /** Unused function that orders rooms by the listorder flag */
3792 int SortRoomsByListOrder(const void *room1, const void *room2)
3794 int l1;
3795 int l2;
3796 HashList *r1 = (HashList *)GetSearchPayload(room1);
3797 HashList *r2 = (HashList *)GetSearchPayload(room2);
3798 StrBuf *listOrderBuf1;
3799 StrBuf *listOrderBuf2;
3801 GetHash(r1, RPKEY(3), (void *)&listOrderBuf1);
3802 GetHash(r2, RPKEY(3), (void *)&listOrderBuf2);
3803 l1 = atoi(ChrPtr(listOrderBuf1));
3804 l2 = atoi(ChrPtr(listOrderBuf2));
3805 if (l1 < l2) return -1;
3806 else if (l1 > l2) return +1;
3807 else return 0;
3809 void tmplput_ROOM_Value(StrBuf *TemplBuffer, WCTemplputParams *TP)
3811 void *value;
3812 StrBuf *val;
3813 HashList *room = (HashList *)(TP->Context);
3815 GetHash(room, TKEY(0), &value);
3816 val = (StrBuf *)value;
3817 StrECMAEscAppend(TemplBuffer, val, 0);
3819 void jsonRoomFlr(void) {
3820 /* Send as our own (application/json) content type */
3821 hprintf("HTTP/1.1 200 OK\r\n");
3822 hprintf("Content-type: application/json; charset=utf-8\r\n");
3823 hprintf("Server: %s / %s\r\n", PACKAGE_STRING, ChrPtr(serv_info.serv_software));
3824 hprintf("Connection: close\r\n");
3825 hprintf("Pragma: no-cache\r\nCache-Control: no-store\r\nExpires:-1\r\n");
3826 begin_burst();
3827 DoTemplate(HKEY("json_roomflr"),NULL,&NoCtx);
3828 end_burst();
3830 void tmplput_RoomName(StrBuf *Target, WCTemplputParams *TP)
3832 StrBufAppendTemplate(Target, TP, WC->wc_roomname, 0);
3835 void _gotonext(void) { slrp_highest(); gotonext(); }
3836 void dotskip(void) {smart_goto(sbstr("room"));}
3837 void _display_private(void) { display_private("", 0); }
3838 void dotgoto(void) {
3839 if (WC->wc_view != VIEW_MAILBOX) { /* dotgoto acts like dotskip when we're in a mailbox view */
3840 slrp_highest();
3842 smart_goto(sbstr("room"));
3845 void tmplput_roombanner(StrBuf *Target, WCTemplputParams *TP)
3847 wprintf("<div id=\"banner\">\n");
3848 embed_room_banner(NULL, navbar_default);
3849 wprintf("</div>\n");
3853 void tmplput_ungoto(StrBuf *Target, WCTemplputParams *TP)
3855 wcsession *WCC = WC;
3857 if ((WCC!=NULL) &&
3858 (!IsEmptyStr(WCC->ugname)))
3859 StrBufAppendBufPlain(Target, WCC->ugname, -1, 0);
3863 int ConditionalHaveUngoto(StrBuf *Target, WCTemplputParams *TP)
3865 wcsession *WCC = WC;
3867 return ((WCC!=NULL) &&
3868 (!IsEmptyStr(WCC->ugname)) &&
3869 (strcasecmp(WCC->ugname, ChrPtr(WCC->wc_roomname)) == 0));
3872 int ConditionalRoomHas_QR_PERMANENT(StrBuf *Target, WCTemplputParams *TP)
3874 wcsession *WCC = WC;
3876 return ((WCC!=NULL) &&
3877 ((WCC->room_flags & QR_PERMANENT) != 0));
3880 int ConditionalRoomHas_QR_INUSE(StrBuf *Target, WCTemplputParams *TP)
3882 wcsession *WCC = WC;
3884 return ((WCC!=NULL) &&
3885 ((WCC->room_flags & QR_INUSE) != 0));
3888 int ConditionalRoomHas_QR_PRIVATE(StrBuf *Target, WCTemplputParams *TP)
3890 wcsession *WCC = WC;
3892 return ((WCC!=NULL) &&
3893 ((WCC->room_flags & QR_PRIVATE) != 0));
3896 int ConditionalRoomHas_QR_PASSWORDED(StrBuf *Target, WCTemplputParams *TP)
3898 wcsession *WCC = WC;
3900 return ((WCC!=NULL) &&
3901 ((WCC->room_flags & QR_PASSWORDED) != 0));
3904 int ConditionalRoomHas_QR_GUESSNAME(StrBuf *Target, WCTemplputParams *TP)
3906 wcsession *WCC = WC;
3908 return ((WCC!=NULL) &&
3909 ((WCC->room_flags & QR_GUESSNAME) != 0));
3912 int ConditionalRoomHas_QR_DIRECTORY(StrBuf *Target, WCTemplputParams *TP)
3914 wcsession *WCC = WC;
3916 return ((WCC!=NULL) &&
3917 ((WCC->room_flags & QR_DIRECTORY) != 0));
3920 int ConditionalRoomHas_QR_UPLOAD(StrBuf *Target, WCTemplputParams *TP)
3922 wcsession *WCC = WC;
3924 return ((WCC!=NULL) &&
3925 ((WCC->room_flags & QR_UPLOAD) != 0));
3928 int ConditionalRoomHas_QR_DOWNLOAD(StrBuf *Target, WCTemplputParams *TP)
3930 wcsession *WCC = WC;
3932 return ((WCC!=NULL) &&
3933 ((WCC->room_flags & QR_DOWNLOAD) != 0));
3936 int ConditionalRoomHas_QR_VISDIR(StrBuf *Target, WCTemplputParams *TP)
3938 wcsession *WCC = WC;
3940 return ((WCC!=NULL) &&
3941 ((WCC->room_flags & QR_VISDIR) != 0));
3944 int ConditionalRoomHas_QR_ANONONLY(StrBuf *Target, WCTemplputParams *TP)
3946 wcsession *WCC = WC;
3948 return ((WCC!=NULL) &&
3949 ((WCC->room_flags & QR_ANONONLY) != 0));
3952 int ConditionalRoomHas_QR_ANONOPT(StrBuf *Target, WCTemplputParams *TP)
3954 wcsession *WCC = WC;
3956 return ((WCC!=NULL) &&
3957 ((WCC->room_flags & QR_ANONOPT) != 0));
3960 int ConditionalRoomHas_QR_NETWORK(StrBuf *Target, WCTemplputParams *TP)
3962 wcsession *WCC = WC;
3964 return ((WCC!=NULL) &&
3965 ((WCC->room_flags & QR_NETWORK) != 0));
3968 int ConditionalRoomHas_QR_PREFONLY(StrBuf *Target, WCTemplputParams *TP)
3970 wcsession *WCC = WC;
3972 return ((WCC!=NULL) &&
3973 ((WCC->room_flags & QR_PREFONLY) != 0));
3976 int ConditionalRoomHas_QR_READONLY(StrBuf *Target, WCTemplputParams *TP)
3978 wcsession *WCC = WC;
3980 return ((WCC!=NULL) &&
3981 ((WCC->room_flags & QR_READONLY) != 0));
3984 int ConditionalRoomHas_QR_MAILBOX(StrBuf *Target, WCTemplputParams *TP)
3986 wcsession *WCC = WC;
3988 return ((WCC!=NULL) &&
3989 ((WCC->room_flags & QR_MAILBOX) != 0));
3993 int ConditionalHaveRoomeditRights(StrBuf *Target, WCTemplputParams *TP)
3995 wcsession *WCC = WC;
3997 return ( (WCC!= NULL) &&
3998 ((WCC->axlevel >= 6) ||
3999 (WCC->is_room_aide) ||
4000 (WCC->is_mailbox) ));
4003 void
4004 InitModule_ROOMOPS
4005 (void)
4007 RegisterPreference(HKEY("roomlistview"),
4008 _("Room list view"),
4009 PRF_STRING,
4010 NULL);
4011 RegisterPreference(HKEY("emptyfloors"), _("Show empty floors"), PRF_YESNO, NULL);
4013 RegisterNamespace("ROOMNAME", 0, 1, tmplput_RoomName, 0);
4015 WebcitAddUrlHandler(HKEY("knrooms"), knrooms, 0);
4016 WebcitAddUrlHandler(HKEY("gotonext"), _gotonext, NEED_URL);
4017 WebcitAddUrlHandler(HKEY("skip"), gotonext, NEED_URL);
4018 WebcitAddUrlHandler(HKEY("ungoto"), ungoto, NEED_URL);
4019 WebcitAddUrlHandler(HKEY("dotgoto"), dotgoto, NEED_URL);
4020 WebcitAddUrlHandler(HKEY("dotskip"), dotskip, NEED_URL);
4021 WebcitAddUrlHandler(HKEY("display_private"), _display_private, 0);
4022 WebcitAddUrlHandler(HKEY("goto_private"), goto_private, NEED_URL);
4023 WebcitAddUrlHandler(HKEY("zapped_list"), zapped_list, 0);
4024 WebcitAddUrlHandler(HKEY("display_zap"), display_zap, 0);
4025 WebcitAddUrlHandler(HKEY("zap"), zap, 0);
4026 WebcitAddUrlHandler(HKEY("display_entroom"), display_entroom, 0);
4027 WebcitAddUrlHandler(HKEY("entroom"), entroom, 0);
4028 WebcitAddUrlHandler(HKEY("display_whok"), display_whok, 0);
4029 WebcitAddUrlHandler(HKEY("do_invt_kick"), do_invt_kick, 0);
4030 WebcitAddUrlHandler(HKEY("display_editroom"), display_editroom, 0);
4031 WebcitAddUrlHandler(HKEY("netedit"), netedit, 0);
4032 WebcitAddUrlHandler(HKEY("editroom"), editroom, 0);
4033 WebcitAddUrlHandler(HKEY("delete_room"), delete_room, 0);
4034 WebcitAddUrlHandler(HKEY("set_room_policy"), set_room_policy, 0);
4035 WebcitAddUrlHandler(HKEY("set_floordiv_expanded"), set_floordiv_expanded, NEED_URL|AJAX);
4036 WebcitAddUrlHandler(HKEY("changeview"), change_view, 0);
4037 WebcitAddUrlHandler(HKEY("toggle_self_service"), toggle_self_service, 0);
4038 WebcitAddUrlHandler(HKEY("json_roomflr"), jsonRoomFlr, 0);
4039 RegisterNamespace("ROOMBANNER", 0, 1, tmplput_roombanner, 0);
4041 RegisterConditional(HKEY("COND:ROOM:FLAGS:QR_PERMANENT"), 0, ConditionalRoomHas_QR_PERMANENT, CTX_NONE);
4042 RegisterConditional(HKEY("COND:ROOM:FLAGS:QR_INUSE"), 0, ConditionalRoomHas_QR_INUSE, CTX_NONE);
4043 RegisterConditional(HKEY("COND:ROOM:FLAGS:QR_PRIVATE"), 0, ConditionalRoomHas_QR_PRIVATE, CTX_NONE);
4044 RegisterConditional(HKEY("COND:ROOM:FLAGS:QR_PASSWORDED"), 0, ConditionalRoomHas_QR_PASSWORDED, CTX_NONE);
4045 RegisterConditional(HKEY("COND:ROOM:FLAGS:QR_GUESSNAME"), 0, ConditionalRoomHas_QR_GUESSNAME, CTX_NONE);
4046 RegisterConditional(HKEY("COND:ROOM:FLAGS:QR_DIRECTORY"), 0, ConditionalRoomHas_QR_DIRECTORY, CTX_NONE);
4047 RegisterConditional(HKEY("COND:ROOM:FLAGS:QR_UPLOAD"), 0, ConditionalRoomHas_QR_UPLOAD, CTX_NONE);
4048 RegisterConditional(HKEY("COND:ROOM:FLAGS:QR_DOWNLOAD"), 0, ConditionalRoomHas_QR_DOWNLOAD, CTX_NONE);
4049 RegisterConditional(HKEY("COND:ROOM:FLAGS:QR_VISIDIR"), 0, ConditionalRoomHas_QR_VISDIR, CTX_NONE);
4050 RegisterConditional(HKEY("COND:ROOM:FLAGS:QR_ANONONLY"), 0, ConditionalRoomHas_QR_ANONONLY, CTX_NONE);
4051 RegisterConditional(HKEY("COND:ROOM:FLAGS:QR_ANONOPT"), 0, ConditionalRoomHas_QR_ANONOPT, CTX_NONE);
4052 RegisterConditional(HKEY("COND:ROOM:FLAGS:QR_NETWORK"), 0, ConditionalRoomHas_QR_NETWORK, CTX_NONE);
4053 RegisterConditional(HKEY("COND:ROOM:FLAGS:QR_PREFONLY"), 0, ConditionalRoomHas_QR_PREFONLY, CTX_NONE);
4054 RegisterConditional(HKEY("COND:ROOM:FLAGS:QR_READONLY"), 0, ConditionalRoomHas_QR_READONLY, CTX_NONE);
4055 RegisterConditional(HKEY("COND:ROOM:FLAGS:QR_MAILBOX"), 0, ConditionalRoomHas_QR_MAILBOX, CTX_NONE);
4057 RegisterConditional(HKEY("COND:UNGOTO"), 0, ConditionalHaveUngoto, CTX_NONE);
4058 RegisterConditional(HKEY("COND:ROOM:EDITACCESS"), 0, ConditionalHaveRoomeditRights, CTX_NONE);
4060 RegisterNamespace("ROOM:UNGOTO", 0, 0, tmplput_ungoto, 0);
4061 RegisterIterator("FLOORS", 0, NULL, GetFloorListHash, NULL, DeleteHash, CTX_FLOORS, CTX_NONE, IT_NOFLAG);
4062 RegisterNamespace("FLOOR:INFO", 1, 2, tmplput_FLOOR_Value, CTX_FLOORS);
4063 RegisterIterator("LKRA", 0, NULL, GetRoomListHashLKRA, NULL, NULL, CTX_ROOMS, CTX_NONE, IT_NOFLAG);
4064 RegisterNamespace("ROOM:INFO", 1, 2, tmplput_ROOM_Value, CTX_ROOMS);
4067 /*@}*/