2 * BBS View renderer module for WebCit
4 * Note: we briefly had a dynamic UI for this. I thought it was cool, but
5 * it was not received well by the user community. If you want to play
6 * with it, go get commit dcf99fe61379b78436c387ea3f89ebfd4ffaf635 of
7 * bbsview_renderer.c and have fun.
9 * Copyright (c) 1996-2011 by the citadel.org team
11 * This program is open source software. You can redistribute it and/or
12 * modify it under the terms of the GNU General Public License as
13 * published by the Free Software Foundation; either version 3 of the
14 * License, or (at your option) any later version.
16 * This program is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 * GNU General Public License for more details.
21 * You should have received a copy of the GNU General Public License
22 * along with this program; if not, write to the Free Software
23 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
29 #include "webserver.h"
33 * Data which gets passed around between the various functions in this module
37 long *msgs; /* Array of msgnums for messages we are displaying */
38 int num_msgs; /* Number of msgnums stored in 'msgs' */
39 long lastseen; /* The number of the last seen message in this room */
40 int alloc_msgs; /* Currently allocated size of array */
41 int requested_page; /* Which page number did the user request? */
42 int num_pages; /* Total number of pages in this room */
43 long start_reading_at; /* Start reading at the page containing this message */
48 * Attempt to determine the closest thing to the "last seen message number" using the
49 * results of the GTSN command
51 long bbsview_get_last_seen(void)
56 serv_getln(buf, sizeof buf);
61 comma_pos = strchr(buf, ','); /* kill first comma and everything to its right */
66 colon_pos = strchr(buf, ':'); /* kill first colon and everything to its left */
68 strcpy(buf, ++colon_pos);
78 * Entry point for message read operations.
80 int bbsview_GetParamsGetServerCall(SharedMessageStatus *Stat,
88 struct bbsview *BBS = malloc(sizeof(struct bbsview));
89 memset(BBS, 0, sizeof(struct bbsview));
92 Stat->startmsg = (-1); /* not used here */
93 Stat->sortit = 1; /* not used here */
94 Stat->num_displayed = DEFAULT_MAXMSGS; /* not used here */
95 BBS->requested_page = 0;
96 BBS->lastseen = bbsview_get_last_seen();
97 BBS->start_reading_at = 0;
99 /* By default, the requested page is the first one. */
100 if (havebstr("start_reading_at")) {
101 BBS->start_reading_at = lbstr("start_reading_at");
102 BBS->requested_page = (-4);
105 /* However, if we are asked to start with a specific message number, make sure
106 * we start on the page containing that message
109 /* Or, if a specific page was requested, make sure we go there */
110 else if (havebstr("page")) {
111 BBS->requested_page = ibstr("page");
114 /* Otherwise, if this is a "read new" operation, make sure we start on the page
115 * containing the first new message
117 else if (oper == 3) {
118 BBS->requested_page = (-3);
121 if (havebstr("maxmsgs")) {
122 Stat->maxmsgs = ibstr("maxmsgs");
124 if (Stat->maxmsgs == 0) Stat->maxmsgs = DEFAULT_MAXMSGS;
126 /* perform a "read all" call to fetch the message list -- we'll cut it down later */
127 rlid[2].cmd(cmd, len);
134 * This function is called for every message in the list.
136 int bbsview_LoadMsgFromServer(SharedMessageStatus *Stat,
138 message_summary* Msg,
142 struct bbsview *BBS = (struct bbsview *) *ViewSpecific;
144 if (BBS->alloc_msgs == 0) {
145 BBS->alloc_msgs = 1000;
146 BBS->msgs = malloc(BBS->alloc_msgs * sizeof(long));
147 memset(BBS->msgs, 0, (BBS->alloc_msgs * sizeof(long)) );
150 /* Check our buffer size */
151 if (BBS->num_msgs >= BBS->alloc_msgs) {
152 BBS->alloc_msgs *= 2;
153 BBS->msgs = realloc(BBS->msgs, (BBS->alloc_msgs * sizeof(long)));
154 memset(&BBS->msgs[BBS->num_msgs], 0, ((BBS->alloc_msgs - BBS->num_msgs) * sizeof(long)) );
157 BBS->msgs[BBS->num_msgs++] = Msg->msgnum;
163 int bbsview_sortfunc(const void *s1, const void *s2) {
170 if (l1 > l2) return(+1);
171 if (l1 < l2) return(-1);
176 int bbsview_RenderView_or_Tail(SharedMessageStatus *Stat,
180 struct bbsview *BBS = (struct bbsview *) *ViewSpecific;
186 int go_to_the_very_end = 0;
188 if (Stat->nummsgs > 0) {
189 syslog(9, "sorting %d messages\n", BBS->num_msgs);
190 qsort(BBS->msgs, (size_t)(BBS->num_msgs), sizeof(long), bbsview_sortfunc);
193 if ((BBS->num_msgs % Stat->maxmsgs) == 0) {
194 BBS->num_pages = BBS->num_msgs / Stat->maxmsgs;
197 BBS->num_pages = (BBS->num_msgs / Stat->maxmsgs) + 1;
200 /* If the requested page number is -4,
201 * it means "whichever page on which msg#xxxxx starts"
202 * Change to the page number which contains that message.
204 if (BBS->requested_page == (-4)) {
205 if (BBS->num_msgs == 0) {
206 BBS->requested_page = 0;
209 for (i=0; i<BBS->num_msgs; ++i) {
211 (BBS->msgs[i] >= BBS->start_reading_at)
212 && (BBS->requested_page == (-4))
214 BBS->requested_page = (i / Stat->maxmsgs) ;
220 /* If the requested page number is -3,
221 * it means "whichever page on which new messages start"
222 * Change that to an actual page number now.
224 if (BBS->requested_page == (-3)) {
225 if (BBS->num_msgs == 0) {
227 * The room is empty; just start at the top and leave it there.
229 BBS->requested_page = 0;
233 && (BBS->lastseen <= BBS->msgs[0])
236 * All messages are new; this is probably the user's first visit to the room,
237 * so start at the last page instead of showing ancient history.
239 BBS->requested_page = BBS->num_pages - 1;
240 go_to_the_very_end = 1;
244 * Some messages are old and some are new. Go to the start of new messages.
246 for (i=0; i<BBS->num_msgs; ++i) {
248 (BBS->msgs[i] > BBS->lastseen)
249 && ( (i == 0) || (BBS->msgs[i-1] <= BBS->lastseen) )
251 BBS->requested_page = (i / Stat->maxmsgs) ;
257 /* Still set to -3 ? If so, that probably means that there are no new messages,
258 * so we'll go to the *end* of the final page.
260 if (BBS->requested_page == (-3)) {
261 if (BBS->num_msgs == 0) {
262 BBS->requested_page = 0;
265 BBS->requested_page = BBS->num_pages - 1;
269 /* keep the requested page within bounds */
270 if (BBS->requested_page < 0) BBS->requested_page = 0;
271 if (BBS->requested_page >= BBS->num_pages) BBS->requested_page = BBS->num_pages - 1;
273 start_index = BBS->requested_page * Stat->maxmsgs;
274 if (start_index < 0) start_index = 0;
275 end_index = start_index + Stat->maxmsgs - 1;
277 for (seq = 0; seq < 3; ++seq) { /* cheap & sleazy way of rendering the page numbers twice */
279 if ( (seq == 1) && (Stat->nummsgs > 0)) {
280 /* display the selected range of messages */
282 for (i=start_index; (i<=end_index && i<BBS->num_msgs); ++i) {
284 (BBS->msgs[i] > BBS->lastseen)
285 && ( (i == 0) || (BBS->msgs[i-1] <= BBS->lastseen) )
287 /* new messages start here */
288 do_template("start_of_new_msgs");
289 if (!go_to_the_very_end) {
290 StrBufAppendPrintf(WC->trailing_javascript, "location.href=\"#newmsgs\";\n");
293 if (BBS->msgs[i] > 0L) {
294 read_message(WC->WBuf, HKEY("view_message"), BBS->msgs[i], NULL, &Mime);
297 (i == (BBS->num_msgs - 1))
298 && (BBS->msgs[i] <= BBS->lastseen)
300 /* no new messages */
301 do_template("no_new_msgs");
302 if (!go_to_the_very_end) {
303 StrBufAppendPrintf(WC->trailing_javascript, "location.href=\"#nonewmsgs\";\n");
309 else if ( (seq == 0) || (seq == 2) ) {
312 /* Display the selecto-bar with the page numbers */
314 wc_printf("<div class=\"moreprompt\">");
316 wc_printf("<a name=\"end_of_msgs\">");
318 wc_printf(_("Go to page: "));
324 last = BBS->num_pages - 1;
326 for (i=0; i<=last; ++i) {
331 || (i == BBS->requested_page)
333 ((BBS->requested_page - i) < RANGE)
334 && ((BBS->requested_page - i) > (0 - RANGE))
340 && (last - BBS->requested_page > RANGE)
342 wc_printf("... ");
344 if (i == BBS->requested_page) {
348 wc_printf("<a href=\"readfwd?go=");
349 urlescputs(ChrPtr(WC->CurRoom.name));
350 wc_printf("?start_reading_at=%ld\">",
351 BBS->msgs[i*Stat->maxmsgs]
353 /* wc_printf("?page=%d\">", i); */
354 wc_printf("<span class=\"moreprompt_link\">");
358 && (BBS->requested_page > (RANGE + 1))
360 wc_printf(_("First"));
364 && (last - BBS->requested_page > RANGE)
366 wc_printf(_("Last"));
369 wc_printf("%d", i + 1); /* change to one-based for display */
371 if (i == BBS->requested_page) {
375 wc_printf("</span>");
380 && (BBS->requested_page > (RANGE + 1))
382 wc_printf(" ...");
389 wc_printf("</div>\n");
393 if (go_to_the_very_end) {
394 StrBufAppendPrintf(WC->trailing_javascript, "location.href=\"#end_of_msgs\";\n");
400 int bbsview_Cleanup(void **ViewSpecific)
402 struct bbsview *BBS = (struct bbsview *) *ViewSpecific;
404 if (BBS->alloc_msgs != 0) {
415 InitModule_BBSVIEWRENDERERS
418 RegisterReadLoopHandlerset(
420 bbsview_GetParamsGetServerCall,
424 bbsview_LoadMsgFromServer,
425 bbsview_RenderView_or_Tail,