Nearly all <FORM> blocks now contain a hidden input
[citadel.git] / webcit / paging.c
1 /*
2  * $Id$
3  */
4 /**
5  * \defgroup PageFunc Functions which implement the chat and paging facilities.
6  * \ingroup ClientPower
7  */
8 /*@{*/
9 #include "webcit.h"
10
11 /**
12  * \brief display the form for paging (x-messaging) another user
13  */
14 void display_page(void)
15 {
16         char recp[SIZ];
17
18         strcpy(recp, bstr("recp"));
19
20         output_headers(1, 1, 2, 0, 0, 0);
21         wprintf("<div id=\"banner\">\n"
22                 "<TABLE class=\"paging_banner\"><TR><TD>"
23                 "<SPAN CLASS=\"titlebar\">");
24         wprintf(_("Send instant message"));
25         wprintf("</SPAN>"
26                 "</TD></TR></TABLE>\n"
27                 "</div>\n<div id=\"content\">\n"
28         );
29                                                                                                                              
30         wprintf("<div class=\"fix_scrollbar_bug\">"
31                 "<table class=\"paging_background\"><tr><td>\n");
32
33         wprintf(_("Send an instant message to: "));
34         escputs(recp);
35         wprintf("<br>\n");
36
37         wprintf("<FORM METHOD=\"POST\" action=\"page_user\">\n");
38         wprintf("<input type=\"hidden\" name=\"nonce\" value=\"%ld\">\n", WC->nonce);
39
40         wprintf("<TABLE border=0 width=100%%><TR><TD>\n");
41
42         wprintf("<INPUT TYPE=\"hidden\" NAME=\"recp\" VALUE=\"");
43         escputs(recp);
44         wprintf("\">\n");
45
46         wprintf(_("Enter message text:"));
47         wprintf("<br />");
48
49         wprintf("<TEXTAREA NAME=\"msgtext\" wrap=soft ROWS=5 COLS=40 "
50                 "WIDTH=40></TEXTAREA>\n");
51
52         wprintf("</TD></TR></TABLE><br />\n");
53
54         wprintf("<INPUT TYPE=\"submit\" NAME=\"send_button\" VALUE=\"%s\">", _("Send message"));
55         wprintf("<br /><a href=\"javascript:window.close();\"%s</A>\n", _("Cancel"));
56
57         wprintf("</FORM></CENTER>\n");
58         wprintf("</td></tr></table></div>\n");
59         wDumpContent(1);
60 }
61
62 /**
63  * \brief page another user
64  */
65 void page_user(void)
66 {
67         char recp[256];
68         char buf[256];
69
70         safestrncpy(recp, bstr("recp"), sizeof recp);
71
72         if (strlen(bstr("send_button")) == 0) {
73                 safestrncpy(WC->ImportantMessage,
74                         _("Message was not sent."),
75                         sizeof WC->ImportantMessage
76                 );
77         } else {
78                 serv_printf("SEXP %s|-", recp);
79                 serv_getln(buf, sizeof buf);
80
81                 if (buf[0] == '4') {
82                         text_to_server(bstr("msgtext"));
83                         serv_puts("000");
84                         stresc(buf, recp, 0, 0);
85                         snprintf(WC->ImportantMessage,
86                                 sizeof WC->ImportantMessage,
87                                 "%s%s.",
88                                 _("Message has been sent to "),
89                                 buf
90                         );
91                 }
92                 else {
93                         safestrncpy(WC->ImportantMessage, &buf[4], sizeof WC->ImportantMessage);
94                 }
95         }
96
97         who();
98 }
99
100
101
102 /**
103  * \brief multiuser chat
104  */
105 void do_chat(void)
106 {
107         char buf[SIZ];
108
109         /** First, check to make sure we're still allowed in this room. */
110         serv_printf("GOTO %s", WC->wc_roomname);
111         serv_getln(buf, sizeof buf);
112         if (buf[0] != '2') {
113                 smart_goto("_BASEROOM_");
114                 return;
115         }
116
117         /**
118          * If the chat socket is still open from a previous chat,
119          * close it -- because it might be stale or in the wrong room.
120          */
121         if (WC->chat_sock < 0) {
122                 close(WC->chat_sock);
123                 WC->chat_sock = (-1);
124         }
125
126         /**
127          * WebCit Chat works by having transmit, receive, and refresh
128          * frames.  Load the frameset.  (This isn't AJAX but the headers
129          * output by begin_ajax_response() happen to be the ones we need.)
130          */
131         begin_ajax_response();
132         do_template("chatframeset");
133         end_ajax_response();
134         return;
135 }
136
137
138 /**
139  * \brief display page popup
140  * If there are instant messages waiting, and we notice that we haven't checked them in
141  * a while, it probably means that we need to open the instant messenger window.
142  */
143 void page_popup(void)
144 {
145         char buf[SIZ];
146
147         /** JavaScript function to alert the user that popups are probably blocked */
148         wprintf("<script type=\"text/javascript\">      "
149                 "function PopUpFailed() {       "
150                 " alert(\"%s\");        "
151                 "}      "
152                 "</script>\n",
153                 _("You have one or more instant messages waiting, but the Citadel Instant Messenger "
154                   "window failed to open.  This is probably because you have a popup blocker "
155                   "installed.  Please configure your popup blocker to allow popups from this site "
156                   "if you wish to receive instant messages.")
157         );
158
159         /** First, do the check as part of our page load. */
160         serv_puts("NOOP");
161         serv_getln(buf, sizeof buf);
162         if (buf[3] == '*') {
163                 if ((time(NULL) - WC->last_pager_check) > 60) {
164                         wprintf("<script type=\"text/javascript\">"
165                                 " var oWin = window.open('static/instant_messenger.html', "
166                                 " 'CTDL_MESSENGER', 'width=700,height=400');    "
167                                 " if (oWin==null || typeof(oWin)==\"undefined\") {      "
168                                 "  PopUpFailed();       "
169                                 " }     "
170                                 "</script>"
171                         );      
172                 }
173         }
174
175         /** Then schedule it to happen again a minute from now if the user is idle. */
176         wprintf("<script type=\"text/javascript\">      "
177                 " function HandleSslp(sslg_xmlresponse) {       "
178                 "  sslg_response = sslg_xmlresponse.responseText.substr(0, 1);  "
179                 "  if (sslg_response == 'Y') {  "
180                 "   var oWin = window.open('static/instant_messenger.html', 'CTDL_MESSENGER',   "
181                 "    'width=700,height=400');   "
182                 "   if (oWin==null || typeof(oWin)==\"undefined\") {    "
183                 "    PopUpFailed();     "
184                 "   }   "
185                 "  }    "
186                 " }     "
187                 " function CheckPager() {       "
188                 "  new Ajax.Request('sslg', { method: 'get', parameters: CtdlRandomString(),    "
189                 "   onSuccess: HandleSslp } );  "
190                 " }     "
191                 " new PeriodicalExecuter(CheckPager, 30);       "
192                 "</script>      "
193         );
194 }
195
196
197
198 /**
199  * \brief Support function for chat
200  * make sure the chat socket is connected
201  * and in chat mode.
202  */
203 int setup_chat_socket(void) {
204         char buf[SIZ];
205         int i;
206         int good_chatmode = 0;
207
208         if (WC->chat_sock < 0) {
209
210                 if (!strcasecmp(ctdlhost, "uds")) {
211                         /** unix domain socket */
212                         sprintf(buf, "%s/citadel.socket", ctdlport);
213                         WC->chat_sock = uds_connectsock(buf);
214                 }
215                 else {
216                         /** tcp socket */
217                         WC->chat_sock = tcp_connectsock(ctdlhost, ctdlport);
218                 }
219
220                 if (WC->chat_sock < 0) {
221                         return(errno);
222                 }
223
224                 /** Temporarily swap the serv and chat sockets during chat talk */
225                 i = WC->serv_sock;
226                 WC->serv_sock = WC->chat_sock;
227                 WC->chat_sock = i;
228
229                 serv_getln(buf, sizeof buf);
230                 if (buf[0] == '2') {
231                         serv_printf("USER %s", WC->wc_username);
232                         serv_getln(buf, sizeof buf);
233                         if (buf[0] == '3') {
234                                 serv_printf("PASS %s", WC->wc_password);
235                                 serv_getln(buf, sizeof buf);
236                                 if (buf[0] == '2') {
237                                         serv_printf("GOTO %s", WC->wc_roomname);
238                                         serv_getln(buf, sizeof buf);
239                                         if (buf[0] == '2') {
240                                                 serv_puts("CHAT");
241                                                 serv_getln(buf, sizeof buf);
242                                                 if (buf[0] == '8') {
243                                                         good_chatmode = 1;
244                                                 }
245                                         }
246                                 }
247                         }
248                 }
249
250                 /** Unswap the sockets. */
251                 i = WC->serv_sock;
252                 WC->serv_sock = WC->chat_sock;
253                 WC->chat_sock = i;
254
255                 if (!good_chatmode) close(WC->serv_sock);
256
257         }
258         return(0);
259 }
260
261
262
263 /**
264  * \brief Receiving side of the chat window.  
265  * This is implemented in a
266  * tiny hidden IFRAME that just does JavaScript writes to
267  * other frames whenever it refreshes and finds new data.
268  */
269 void chat_recv(void) {
270         int i;
271         struct pollfd pf;
272         int got_data = 0;
273         int end_chat_now = 0;
274         char buf[SIZ];
275         char cl_user[SIZ];
276         char cl_text[SIZ];
277         char *output_data = NULL;
278
279         output_headers(0, 0, 0, 0, 0, 0);
280
281         wprintf("Content-type: text/html; charset=utf-8\n");
282         wprintf("\n");
283         wprintf("<html>\n"
284                 "<head>\n"
285                 "<meta http-equiv=\"refresh\" content=\"3\" />\n"
286                 "</head>\n"
287
288                 "<body bgcolor=\"#FFFFFF\">\n"
289         );
290
291         if (setup_chat_socket() != 0) {
292                 wprintf(_("An error occurred while setting up the chat socket."));
293                 wprintf("</BODY></HTML>\n");
294                 wDumpContent(0);
295                 return;
296         }
297
298         /**
299          * See if there is any chat data waiting.
300          */
301         output_data = strdup("");
302         do {
303                 got_data = 0;
304                 pf.fd = WC->chat_sock;
305                 pf.events = POLLIN;
306                 pf.revents = 0;
307                 if (poll(&pf, 1, 1) > 0) if (pf.revents & POLLIN) {
308                         ++got_data;
309
310                         /** Temporarily swap the serv and chat sockets during chat talk */
311                         i = WC->serv_sock;
312                         WC->serv_sock = WC->chat_sock;
313                         WC->chat_sock = i;
314         
315                         serv_getln(buf, sizeof buf);
316
317                         if (!strcmp(buf, "000")) {
318                                 strcpy(buf, ":|");
319                                 strcat(buf, _("Now exiting chat mode."));
320                                 end_chat_now = 1;
321                         }
322                         
323                         /** Unswap the sockets. */
324                         i = WC->serv_sock;
325                         WC->serv_sock = WC->chat_sock;
326                         WC->chat_sock = i;
327
328                         /** Append our output data */
329                         output_data = realloc(output_data, strlen(output_data) + strlen(buf) + 4);
330                         strcat(output_data, buf);
331                         strcat(output_data, "\n");
332                 }
333
334         } while ( (got_data) && (!end_chat_now) );
335
336         if (end_chat_now) {
337                 close(WC->chat_sock);
338                 WC->chat_sock = (-1);
339                 wprintf("<img src=\"static/blank.gif\" onLoad=\"parent.window.close();\">\n");
340         }
341
342         if (strlen(output_data) > 0) {
343
344                 if (output_data[strlen(output_data)-1] == '\n') {
345                         output_data[strlen(output_data)-1] = 0;
346                 }
347
348                 /** Output our fun to the other frame. */
349                 wprintf("<img src=\"static/blank.gif\" WIDTH=1 HEIGHT=1\n"
350                         "onLoad=\" \n"
351                 );
352
353                 for (i=0; i<num_tokens(output_data, '\n'); ++i) {
354                         extract_token(buf, output_data, i, '\n', sizeof buf);
355                         extract_token(cl_user, buf, 0, '|', sizeof cl_user);
356                         extract_token(cl_text, buf, 1, '|', sizeof cl_text);
357
358                         if (strcasecmp(cl_text, "NOOP")) {
359
360                                 wprintf("parent.chat_transcript.document.write('");
361         
362                                 if (strcasecmp(cl_user, WC->last_chat_user)) {
363                                         wprintf("<TABLE border=0 WIDTH=100%% "
364                                                 "CELLSPACING=1 CELLPADDING=0 "
365                                                 "BGCOLOR=&quot;#FFFFFF&quot;>"
366                                                 "<TR><TD></TR></TD></TABLE>"
367                                         );
368         
369                                 }
370
371                                 wprintf("<TABLE border=0 WIDTH=100%% "
372                                         "CELLSPACING=0 CELLPADDING=0 "
373                                         "BGCOLOR=&quot;#EEEEEE&quot;>");
374         
375                                 wprintf("<TR><TD>");
376         
377                                 if (!strcasecmp(cl_user, ":")) {
378                                         wprintf("<I>");
379                                 }
380
381                                 if (strcasecmp(cl_user, WC->last_chat_user)) {
382                                         wprintf("<B>");
383         
384                                         if (!strcasecmp(cl_user, WC->wc_fullname)) {
385                                                 wprintf("<FONT COLOR=&quot;#FF0000&quot;>");
386                                         }
387                                         else {
388                                                 wprintf("<FONT COLOR=&quot;#0000FF&quot;>");
389                                         }
390                                         jsescputs(cl_user);
391         
392                                         wprintf("</FONT>: </B>");
393                                 }
394                                 else {
395                                         wprintf("&nbsp;&nbsp;&nbsp;");
396                                 }
397                                 jsescputs(cl_text);
398                                 if (!strcasecmp(cl_user, ":")) {
399                                         wprintf("</I>");
400                                 }
401
402                                 wprintf("</TD></TR></TABLE>");
403                                 wprintf("'); \n");
404
405                                 strcpy(WC->last_chat_user, cl_user);
406                         }
407                 }
408
409                 wprintf("parent.chat_transcript.scrollTo(999999,999999);\">\n");
410         }
411
412         free(output_data);
413
414         wprintf("</BODY></HTML>\n");
415         wDumpContent(0);
416 }
417
418
419 /**
420  * \brief sending side of the chat window
421  */
422 void chat_send(void) {
423         int i;
424         char send_this[SIZ];
425         char buf[SIZ];
426
427         output_headers(0, 0, 0, 0, 0, 0);
428         wprintf("Content-type: text/html; charset=utf-8\n");
429         wprintf("\n");
430         wprintf("<HTML>"
431                 "<BODY onLoad=\"document.chatsendform.send_this.focus();\" >"
432         );
433
434         if (bstr("send_this") != NULL) {
435                 strcpy(send_this, bstr("send_this"));
436         }
437         else {
438                 strcpy(send_this, "");
439         }
440
441         if (strlen(bstr("help_button")) > 0) {
442                 strcpy(send_this, "/help");
443         }
444
445         if (strlen(bstr("list_button")) > 0) {
446                 strcpy(send_this, "/who");
447         }
448
449         if (strlen(bstr("exit_button")) > 0) {
450                 strcpy(send_this, "/quit");
451         }
452
453         if (setup_chat_socket() != 0) {
454                 wprintf(_("An error occurred while setting up the chat socket."));
455                 wprintf("</BODY></HTML>\n");
456                 wDumpContent(0);
457                 return;
458         }
459
460         /** Temporarily swap the serv and chat sockets during chat talk */
461         i = WC->serv_sock;
462         WC->serv_sock = WC->chat_sock;
463         WC->chat_sock = i;
464
465         while (strlen(send_this) > 0) {
466                 if (strlen(send_this) < 67) {
467                         serv_puts(send_this);
468                         strcpy(send_this, "");
469                 }
470                 else {
471                         for (i=55; i<67; ++i) {
472                                 if (send_this[i] == ' ') break;
473                         }
474                         strncpy(buf, send_this, i);
475                         buf[i] = 0;
476                         strcpy(send_this, &send_this[i]);
477                         serv_puts(buf);
478                 }
479         }
480
481         /** Unswap the sockets. */
482         i = WC->serv_sock;
483         WC->serv_sock = WC->chat_sock;
484         WC->chat_sock = i;
485
486         wprintf("<FORM METHOD=\"POST\" action=\"chat_send\" NAME=\"chatsendform\">\n");
487         wprintf("<input type=\"hidden\" name=\"nonce\" value=\"%ld\">\n", WC->nonce);
488         wprintf("<INPUT TYPE=\"text\" SIZE=\"80\" MAXLENGTH=\"%d\" "
489                 "NAME=\"send_this\">\n", SIZ-10);
490         wprintf("<br />");
491         wprintf("<INPUT TYPE=\"submit\" NAME=\"send_button\" VALUE=\"%s\">\n", _("Send"));
492         wprintf("<INPUT TYPE=\"submit\" NAME=\"help_button\" VALUE=\"%s\">\n", _("Help"));
493         wprintf("<INPUT TYPE=\"submit\" NAME=\"list_button\" VALUE=\"%s\">\n", _("List users"));
494         wprintf("<INPUT TYPE=\"submit\" NAME=\"exit_button\" VALUE=\"%s\">\n", _("Exit"));
495         wprintf("</FORM>\n");
496
497         wprintf("</BODY></HTML>\n");
498         wDumpContent(0);
499 }
500
501 /*@}*/