char *viewdefs[] = {
"Messages",
"Summary",
- "Address Book"
+ "Address Book",
+ "Calendar",
+ "Tasks"
};
char floorlist[128][SIZ];
/*
- * list all rooms by floor
+ * list all rooms by floor (only should get called from knrooms() because
+ * that's where output_headers() is called from)
*/
void list_all_rooms_by_floor(void)
{
int a;
char buf[SIZ];
- load_floorlist();
-
- output_headers(1);
-
wprintf("<TABLE width=100%% border><TR><TH>Floor</TH>");
wprintf("<TH>Rooms with new messages</TH>");
wprintf("<TH>Rooms with no new messages</TH></TR>\n");
serv_puts("RINF");
serv_gets(buf);
if (buf[0] == '1') {
- wprintf("<FONT SIZE=-1>");
fmout(NULL);
- wprintf("</FONT>");
}
}
serv_puts("OIMG _roompic_");
serv_gets(buf);
if (buf[0] == '2') {
- wprintf("<TD>");
+ wprintf("<TD BGCOLOR=444455>");
wprintf("<IMG SRC=\"/image&name=_roompic_&room=");
urlescputs(WC->wc_roomname);
wprintf("\"></TD>");
*/
void embed_newmail_button(void) {
if ( (WC->new_mail > WC->remember_new_mail) && (WC->new_mail>0) ) {
- wprintf("<TD VALIGN=TOP>"
+ wprintf(
"<A HREF=\"/dotgoto?room=_MAIL_\">"
"<IMG SRC=\"/static/mail.gif\" border=0 "
"ALT=\"You have new mail\">"
- "<BR><BLINK>%d</BLINK>", WC->new_mail);
- wprintf("<FONT SIZE=-2> new mail messages</FONT></A></TD>");
+ "<BR><FONT SIZE=2 COLOR=FFFFFF>"
+ "%d new mail</FONT></A>", WC->new_mail);
WC->remember_new_mail = WC->new_mail;
}
}
got = fakegot;
}
+ /* If the user happens to select the "make this my start page" link,
+ * we want it to remember the URL as a "/dotskip" one instead of
+ * a "skip" or "gotonext" or something like that.
+ */
+ snprintf(WC->this_page, sizeof(WC->this_page), "/dotskip&room=%s",
+ WC->wc_roomname);
+
/* Check for new mail. */
WC->new_mail = extract_int(&got[4], 9);
WC->wc_view = extract_int(&got[4], 11);
svcallback("ROOMINFO", readinfo);
svcallback("YOUHAVEMAIL", embed_newmail_button);
svcallback("VIEWOMATIC", embed_view_o_matic);
+ svcallback("START", offer_start_page);
- do_template("roombanner.html");
+ do_template("roombanner");
clear_local_substs();
}
"<META HTTP-EQUIV=\"expired\" CONTENT=\"28-May-1971 18:10:00 GMT\">\n"
"<meta name=\"MSSmartTagsPreventParsing\" content=\"TRUE\">\n"
"</HEAD>\n");
- do_template("background.html");
+ do_template("background");
}
if (display_name != 2) {
/* store ungoto information */
smart_goto(buf);
}
+
+
+
+
+/*
+ * Set/clear/read the "self-service list subscribe" flag for a room
+ *
+ * Set 'newval' to 0 to clear, 1 to set, any other value to leave unchanged.
+ * Always returns the new value.
+ */
+
+int self_service(int newval) {
+ int current_value = 0;
+ char buf[SIZ];
+
+ char name[SIZ];
+ char password[SIZ];
+ char dirname[SIZ];
+ int flags, floor, order, view, flags2;
+
+ serv_puts("GETR");
+ serv_gets(buf);
+ if (buf[0] != '2') return(0);
+
+ extract(name, &buf[4], 0);
+ extract(password, &buf[4], 1);
+ extract(dirname, &buf[4], 2);
+ flags = extract_int(&buf[4], 3);
+ floor = extract_int(&buf[4], 4);
+ order = extract_int(&buf[4], 5);
+ view = extract_int(&buf[4], 6);
+ flags2 = extract_int(&buf[4], 7);
+
+ if (flags2 & QR2_SELFLIST) {
+ current_value = 1;
+ }
+ else {
+ current_value = 0;
+ }
+
+ if (newval == 1) {
+ flags2 = flags2 | QR2_SELFLIST;
+ }
+ else if (newval == 0) {
+ flags2 = flags2 & ~QR2_SELFLIST;
+ }
+ else {
+ return(current_value);
+ }
+
+ if (newval != current_value) {
+ serv_printf("SETR %s|%s|%s|%d|0|%d|%d|%d|%d",
+ name, password, dirname, flags,
+ floor, order, view, flags2);
+ serv_gets(buf);
+ }
+
+ return(newval);
+
+}
+
+
+
+
+
+
/*
* display the form for editing a room
*/
output_headers(1);
/* print the tabbed dialog */
- wprintf("<TABLE border=0 width=100%%><TR BGCOLOR=FFFFFF><TD> </TD>");
+ wprintf("<TABLE border=0 cellspacing=0 cellpadding=0 width=100%%>"
+ "<TR ALIGN=CENTER BGCOLOR=FFFFFF>"
+ "<TD> </TD>\n");
if (!strcmp(tab, "admin")) {
wprintf("<TD BGCOLOR=000077><FONT SIZE=+1 COLOR=\"FFFFFF\"><B>");
}
else {
- wprintf("<TD><A HREF=\"/display_editroom&tab=admin\">");
+ wprintf("<TD BGCOLOR=AAAAAA><A HREF=\"/display_editroom&tab=admin\">");
}
wprintf("Room administration");
if (!strcmp(tab, "admin")) {
wprintf("</A></TD>\n");
}
- wprintf("<TD> </TD>\n");
+ wprintf("<TD> </TD>\n");
if (!strcmp(tab, "config")) {
wprintf("<TD BGCOLOR=000077><FONT SIZE=+1 COLOR=\"FFFFFF\"><B>");
}
else {
- wprintf("<TD><A HREF=\"/display_editroom&tab=config\">");
+ wprintf("<TD BGCOLOR=AAAAAA><A HREF=\"/display_editroom&tab=config\">");
}
wprintf("Room configuration");
if (!strcmp(tab, "config")) {
wprintf("</A></TD>\n");
}
- wprintf("<TD> </TD>\n");
+ wprintf("<TD> </TD>\n");
if (!strcmp(tab, "sharing")) {
wprintf("<TD BGCOLOR=000077><FONT SIZE=+1 COLOR=\"FFFFFF\"><B>");
}
else {
- wprintf("<TD><A HREF=\"/display_editroom&tab=sharing\">");
+ wprintf("<TD BGCOLOR=AAAAAA><A HREF=\"/display_editroom&tab=sharing\">");
}
wprintf("Sharing");
if (!strcmp(tab, "sharing")) {
wprintf("</A></TD>\n");
}
- wprintf("<TD> </TD>\n");
+ wprintf("<TD> </TD>\n");
if (!strcmp(tab, "listserv")) {
wprintf("<TD BGCOLOR=000077><FONT SIZE=+1 COLOR=\"FFFFFF\"><B>");
}
else {
- wprintf("<TD><A HREF=\"/display_editroom&tab=listserv\">");
+ wprintf("<TD BGCOLOR=AAAAAA><A HREF=\"/display_editroom&tab=listserv\">");
}
wprintf("Mailing list service");
if (!strcmp(tab, "listserv")) {
wprintf("</A></TD>\n");
}
- wprintf("<TD> </TD></TR></TABLE>\n");
+ wprintf("<TD> </TD></TR>"
+ "<TR><TD BGCOLOR=000077 COLSPAN=9 HEIGHT=5> </TD></TR>"
+ "</TABLE>\n");
/* end tabbed dialog */
}
}
- wprintf("</TD></TR></TABLE><BR>\n"
+ wprintf("</TD></TR>"
+ "</TABLE><BR>\n"
"<I><B>Reminder:</B> When sharing a room, "
"it must be shared from both ends. Adding a node to "
"the 'shared' list sends messages out, but in order to"
/* Mailing list management */
if (!strcmp(tab, "listserv")) {
- wprintf("<BR><center><i>The contents of this room are being "
- "mailed to the following list recipients:"
+ wprintf("<BR><center>"
+ "<TABLE BORDER=0 WIDTH=100%% CELLPADDING=5>"
+ "<TR><TD VALIGN=TOP>");
+
+ wprintf("<i>The contents of this room are being "
+ "mailed <b>as individual messages</b> "
+ "to the following list recipients:"
"</i><br><br>\n");
serv_puts("GNET");
"<INPUT TYPE=\"hidden\" NAME=\"prefix\" VALUE=\"listrecp|\">\n");
wprintf("<INPUT TYPE=\"text\" NAME=\"line\">\n");
wprintf("<INPUT TYPE=\"submit\" NAME=\"cmd\" VALUE=\"Add\">");
- wprintf("</FORM><BR></CENTER>\n");
+ wprintf("</FORM>\n");
+
+ wprintf("</TD><TD VALIGN=TOP>\n");
+
+ wprintf("<i>The contents of this room are being "
+ "mailed <b>in digest form</b> "
+ "to the following list recipients:"
+ "</i><br><br>\n");
+
+ serv_puts("GNET");
+ serv_gets(buf);
+ if (buf[0]=='1') while (serv_gets(buf), strcmp(buf, "000")) {
+ extract(cmd, buf, 0);
+ if (!strcasecmp(cmd, "digestrecp")) {
+ extract(recp, buf, 1);
+
+ escputs(recp);
+ wprintf(" <A HREF=\"/netedit&cmd=remove&line="
+ "digestrecp|");
+ urlescputs(recp);
+ wprintf("&tab=listserv\">(remove)</A><BR>");
+
+ }
+ }
+ wprintf("<BR><FORM METHOD=\"POST\" ACTION=\"/netedit\">\n"
+ "<INPUT TYPE=\"hidden\" NAME=\"tab\" VALUE=\"listserv\">\n"
+ "<INPUT TYPE=\"hidden\" NAME=\"prefix\" VALUE=\"digestrecp|\">\n");
+ wprintf("<INPUT TYPE=\"text\" NAME=\"line\">\n");
+ wprintf("<INPUT TYPE=\"submit\" NAME=\"cmd\" VALUE=\"Add\">");
+ wprintf("</FORM>\n");
+
+ wprintf("</TD></TR></TABLE><HR>\n");
+
+ if (self_service(999) == 1) {
+ wprintf("This room is configured to allow "
+ "self-service subscribe/unsubscribe requests."
+ " <A HREF=\"/toggle_self_service?newval=0&"
+ "tab=listserv\">"
+ "Click to disable.</A><BR>\n"
+ "The URL for subscribe/unsubscribe is: "
+ "<TT>http://%s/listsub</TT><BR>\n",
+ WC->http_host
+ );
+ }
+ else {
+ wprintf("This room is <i>not</i> configured to allow "
+ "self-service subscribe/unsubscribe requests."
+ " <A HREF=\"/toggle_self_service?newval=1&"
+ "tab=listserv\">"
+ "Click to enable.</A><BR>\n"
+ );
+ }
+
+
+ wprintf("</CENTER>\n");
}
wDumpContent(1);
}
+/*
+ * Toggle self-service list subscription
+ */
+void toggle_self_service(void) {
+ int newval = 0;
+
+ newval = atoi(bstr("newval"));
+ self_service(newval);
+ display_editroom();
+}
+
+
+
/*
* save new parameters for a room
*/
wprintf("so you don't have to keep returning here.\n");
wprintf("<BR><BR>");
- wprintf("<FORM METHOD=\"POST\" ACTION=\"/goto_private\">\n");
+ wprintf("<FORM METHOD=\"GET\" ACTION=\"/goto_private\">\n");
wprintf("<TABLE border><TR><TD>");
wprintf("Enter room name:</TD><TD>");
wprintf("disappear from your room list. Is this what you wish ");
wprintf("to do?<BR>\n");
- wprintf("<FORM METHOD=\"POST\" ACTION=\"/zap\">\n");
+ wprintf("<FORM METHOD=\"GET\" ACTION=\"/zap\">\n");
wprintf("<INPUT TYPE=\"submit\" NAME=\"sc\" VALUE=\"OK\">");
wprintf("<INPUT TYPE=\"submit\" NAME=\"sc\" VALUE=\"Cancel\">");
wprintf("</FORM>\n");
wprintf("</FONT></TD></TR></TABLE>\n");
wprintf("<CENTER>");
- wprintf("<FORM METHOD=\"POST\" ACTION=\"/delete_room\">\n");
+ wprintf("<FORM METHOD=\"GET\" ACTION=\"/delete_room\">\n");
wprintf("Are you sure you want to delete <FONT SIZE=+1>");
escputs(WC->wc_roomname);
+/*
+ * Convert a room name to a folder-ish-looking name.
+ */
+void room_to_folder(char *folder, char *room, int floor, int is_mailbox)
+{
+ int i;
+
+ /*
+ * For mailboxes, just do it straight...
+ */
+ if (is_mailbox) {
+ sprintf(folder, "My folders|%s", room);
+ }
+
+ /*
+ * Otherwise, prefix the floor name as a "public folders" moniker
+ */
+ else {
+ sprintf(folder, "%s|%s", floorlist[floor], room);
+ }
+
+ /*
+ * Replace "/" characters with "|" for pseudo-folder-delimiting
+ */
+ for (i=0; i<strlen(folder); ++i) {
+ if (folder[i] == '/') folder[i] = '|';
+ }
+}
+
+
+
+
+
+
+
/*
* Change the view for this room
*/
/*
- * Show the room list in "folders" format
+ * Show the room list in "folders" format. (only should get called by
+ * knrooms() because that's where output_headers() is called from)
*/
void folders(void) {
char buf[SIZ];
+ int levels, oldlevels;
+ int swap = 0;
+
struct folder {
- char name[SIZ];
char room[SIZ];
+ char name[SIZ];
int hasnewmsgs;
int is_mailbox;
+ int selectable;
};
struct folder *fold = NULL;
struct folder ftmp;
int max_folders = 0;
int alloc_folders = 0;
- int i, j;
+ int i, j, k, t;
int p;
int flags;
+ int floor;
+ int nests = 0;
+
+ /* Start with the mailboxes */
+ max_folders = 1;
+ alloc_folders = 1;
+ fold = malloc(sizeof(struct folder));
+ memset(fold, 0, sizeof(struct folder));
+ strcpy(fold[0].name, "My folders");
+ fold[0].is_mailbox = 1;
+
+ /* Then add floors */
+ serv_puts("LFLR");
+ serv_gets(buf);
+ if (buf[0]=='1') while(serv_gets(buf), strcmp(buf, "000")) {
+ if (max_folders >= alloc_folders) {
+ alloc_folders = max_folders + 100;
+ fold = realloc(fold,
+ alloc_folders * sizeof(struct folder));
+ }
+ memset(&fold[max_folders], 0, sizeof(struct folder));
+ extract(fold[max_folders].name, buf, 1);
+ ++max_folders;
+ }
- load_floorlist();
-
- output_headers(1);
- wprintf("<TABLE WIDTH=100%% BORDER=0 BGCOLOR=000077><TR><TD>");
- wprintf("<FONT SIZE=+1 COLOR=\"FFFFFF\"");
- wprintf("<B>Folder list</B>\n");
- wprintf("</TD></TR></TABLE><BR>\n");
-
+ /* Now add rooms */
for (p = 0; p < 2; ++p) {
if (p == 0) serv_puts("LKRN");
else if (p == 1) serv_puts("LKRO");
- else serv_puts("NOOP");
serv_gets(buf);
if (buf[0]=='1') while(serv_gets(buf), strcmp(buf, "000")) {
- ++max_folders;
- if (max_folders > alloc_folders) {
+ if (max_folders >= alloc_folders) {
alloc_folders = max_folders + 100;
fold = realloc(fold,
- max_folders * sizeof(struct folder));
+ alloc_folders * sizeof(struct folder));
}
- extract(fold[max_folders-1].name, buf, 0);
- extract(fold[max_folders-1].room, buf, 0);
- if (p == 0) fold[max_folders-1].hasnewmsgs = 1;
+ memset(&fold[max_folders], 0, sizeof(struct folder));
+ extract(fold[max_folders].room, buf, 0);
+ if (p == 0) fold[max_folders].hasnewmsgs = 1;
flags = extract_int(buf, 1);
+ floor = extract_int(buf, 2);
if (flags & QR_MAILBOX) {
- fold[max_folders-1].is_mailbox = 1;
+ fold[max_folders].is_mailbox = 1;
}
+ room_to_folder(fold[max_folders].name,
+ fold[max_folders].room,
+ floor,
+ fold[max_folders].is_mailbox);
+ fold[max_folders].selectable = 1;
+ ++max_folders;
}
}
/* Bubble-sort the folder list */
for (i=0; i<max_folders; ++i) {
- for (j=0; j<max_folders-i-1; ++j) {
- if (strcasecmp(fold[j].name, fold[j+1].name) > 0) {
+ for (j=0; j<(max_folders-1)-i; ++j) {
+ if (fold[j].is_mailbox == fold[j+1].is_mailbox) {
+ swap = strcasecmp(fold[j].name, fold[j+1].name);
+ }
+ else {
+ if ( (fold[j+1].is_mailbox)
+ && (!fold[j].is_mailbox)) {
+ swap = 1;
+ }
+ else {
+ swap = 0;
+ }
+ }
+ if (swap > 0) {
memcpy(&ftmp, &fold[j], sizeof(struct folder));
memcpy(&fold[j], &fold[j+1],
sizeof(struct folder));
}
/* Output */
+ nests = 0;
+ levels = 0;
+ oldlevels = 0;
for (i=0; i<max_folders; ++i) {
+
+ levels = num_tokens(fold[i].name, '|');
+ if (levels > oldlevels) {
+ for (k=0; k<(levels-oldlevels); ++k) {
+ ++nests;
+ }
+ }
+ if (levels < oldlevels) {
+ for (k=0; k<(oldlevels-levels); ++k) {
+ --nests;
+ }
+ }
+ oldlevels = levels;
+
+ for (t=0; t<nests; ++t) wprintf(" ");
+ if (fold[i].selectable) {
+ wprintf("<A HREF=\"/dotgoto?room=");
+ urlescputs(fold[i].room);
+ wprintf("\">");
+ }
+ else {
+ wprintf("<i>");
+ }
if (fold[i].hasnewmsgs) wprintf("<B>");
- escputs(fold[i].name);
+ extract(buf, fold[i].name, levels-1);
+ escputs(buf);
if (fold[i].hasnewmsgs) wprintf("</B>");
+ if (fold[i].selectable) {
+ wprintf("</A>");
+ }
+ else {
+ wprintf("</i>");
+ }
+ if (!strcasecmp(fold[i].name, "My Folders|Mail")) {
+ wprintf(" (INBOX)");
+ }
wprintf("<BR>\n");
}
+ while (nests-- > 0) ;;
free(fold);
-
wDumpContent(1);
}
+
+
+/* Do either a known rooms list or a folders list, depending on the
+ * user's preference
+ */
+void knrooms() {
+ char listviewpref[SIZ];
+
+ output_headers(3);
+ load_floorlist();
+
+ /* Determine whether the user is trying to change views */
+ if (bstr("view") != NULL) {
+ if (strlen(bstr("view")) > 0) {
+ set_preference("roomlistview", bstr("view"));
+ }
+ }
+
+ get_preference("roomlistview", listviewpref);
+
+ if (strcasecmp(listviewpref, "folders")) {
+ strcpy(listviewpref, "rooms");
+ }
+
+ /* title bar */
+ wprintf("<TABLE WIDTH=100%% BORDER=0 BGCOLOR=000077><TR><TD>"
+ "<FONT SIZE=+1 COLOR=\"FFFFFF\"<B>"
+ );
+ if (!strcasecmp(listviewpref, "rooms")) {
+ wprintf("Room list");
+ }
+ if (!strcasecmp(listviewpref, "folders")) {
+ wprintf("Folder list");
+ }
+ wprintf("</B></TD>\n");
+
+
+ /* offer the ability to switch views */
+ wprintf("<TD><FORM NAME=\"roomlistomatic\">\n"
+ "<SELECT NAME=\"newview\" SIZE=\"1\" "
+ "OnChange=\"location.href=roomlistomatic.newview.options"
+ "[selectedIndex].value\">\n");
+
+ wprintf("<OPTION %s VALUE=\"/knrooms&view=rooms\">"
+ "View as room list"
+ "</OPTION>\n",
+ ( !strcasecmp(listviewpref, "rooms") ? "SELECTED" : "" )
+ );
+
+ wprintf("<OPTION %s VALUE=\"/knrooms&view=folders\">"
+ "View as folder list"
+ "</OPTION>\n",
+ ( !strcasecmp(listviewpref, "folders") ? "SELECTED" : "" )
+ );
+
+ wprintf("</SELECT></FORM></TD><TD>\n");
+ offer_start_page();
+ wprintf("</TD></TR></TABLE><BR>\n");
+
+ /* Display the room list in the user's preferred format */
+ if (!strcasecmp(listviewpref, "folders")) {
+ folders();
+ }
+ else {
+ list_all_rooms_by_floor();
+ }
+}