1e6a9e394ce3f3068fce6d8fad6ba79cde9cc639
[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 (IsEmptyStr(bstr("send_button"))) {
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 (!IsEmptyStr(output_data)) {
343                 int len;
344                 len = strlen(output_data);
345                 if (output_data[len-1] == '\n') {
346                         output_data[len-1] = 0;
347                 }
348
349                 /** Output our fun to the other frame. */
350                 wprintf("<img src=\"static/blank.gif\" WIDTH=1 HEIGHT=1\n"
351                         "onLoad=\" \n"
352                 );
353
354                 for (i=0; i<num_tokens(output_data, '\n'); ++i) {
355                         extract_token(buf, output_data, i, '\n', sizeof buf);
356                         extract_token(cl_user, buf, 0, '|', sizeof cl_user);
357                         extract_token(cl_text, buf, 1, '|', sizeof cl_text);
358
359                         if (strcasecmp(cl_text, "NOOP")) {
360
361                                 wprintf("parent.chat_transcript.document.write('");
362         
363                                 if (strcasecmp(cl_user, WC->last_chat_user)) {
364                                         wprintf("<TABLE border=0 WIDTH=100%% "
365                                                 "CELLSPACING=1 CELLPADDING=0 "
366                                                 "BGCOLOR=&quot;#FFFFFF&quot;>"
367                                                 "<TR><TD></TR></TD></TABLE>"
368                                         );
369         
370                                 }
371
372                                 wprintf("<TABLE border=0 WIDTH=100%% "
373                                         "CELLSPACING=0 CELLPADDING=0 "
374                                         "BGCOLOR=&quot;#EEEEEE&quot;>");
375         
376                                 wprintf("<TR><TD>");
377         
378                                 if (!strcasecmp(cl_user, ":")) {
379                                         wprintf("<I>");
380                                 }
381
382                                 if (strcasecmp(cl_user, WC->last_chat_user)) {
383                                         wprintf("<B>");
384         
385                                         if (!strcasecmp(cl_user, WC->wc_fullname)) {
386                                                 wprintf("<FONT COLOR=&quot;#FF0000&quot;>");
387                                         }
388                                         else {
389                                                 wprintf("<FONT COLOR=&quot;#0000FF&quot;>");
390                                         }
391                                         jsescputs(cl_user);
392         
393                                         wprintf("</FONT>: </B>");
394                                 }
395                                 else {
396                                         wprintf("&nbsp;&nbsp;&nbsp;");
397                                 }
398                                 jsescputs(cl_text);
399                                 if (!strcasecmp(cl_user, ":")) {
400                                         wprintf("</I>");
401                                 }
402
403                                 wprintf("</TD></TR></TABLE>");
404                                 wprintf("'); \n");
405
406                                 strcpy(WC->last_chat_user, cl_user);
407                         }
408                 }
409
410                 wprintf("parent.chat_transcript.scrollTo(999999,999999);\">\n");
411         }
412
413         free(output_data);
414
415         wprintf("</BODY></HTML>\n");
416         wDumpContent(0);
417 }
418
419
420 /**
421  * \brief sending side of the chat window
422  */
423 void chat_send(void) {
424         int i;
425         char send_this[SIZ];
426         char buf[SIZ];
427
428         output_headers(0, 0, 0, 0, 0, 0);
429         wprintf("Content-type: text/html; charset=utf-8\n");
430         wprintf("\n");
431         wprintf("<HTML>"
432                 "<BODY onLoad=\"document.chatsendform.send_this.focus();\" >"
433         );
434
435         if (bstr("send_this") != NULL) {
436                 strcpy(send_this, bstr("send_this"));
437         }
438         else {
439                 strcpy(send_this, "");
440         }
441
442         if (!IsEmptyStr(bstr("help_button"))) {
443                 strcpy(send_this, "/help");
444         }
445
446         if (!IsEmptyStr(bstr("list_button"))) {
447                 strcpy(send_this, "/who");
448         }
449
450         if (!IsEmptyStr(bstr("exit_button"))) {
451                 strcpy(send_this, "/quit");
452         }
453
454         if (setup_chat_socket() != 0) {
455                 wprintf(_("An error occurred while setting up the chat socket."));
456                 wprintf("</BODY></HTML>\n");
457                 wDumpContent(0);
458                 return;
459         }
460
461         /** Temporarily swap the serv and chat sockets during chat talk */
462         i = WC->serv_sock;
463         WC->serv_sock = WC->chat_sock;
464         WC->chat_sock = i;
465
466         while (!IsEmptyStr(send_this)) {
467                 if (strlen(send_this) < 67) {
468                         serv_puts(send_this);
469                         strcpy(send_this, "");
470                 }
471                 else {
472                         for (i=55; i<67; ++i) {
473                                 if (send_this[i] == ' ') break;
474                         }
475                         strncpy(buf, send_this, i);
476                         buf[i] = 0;
477                         strcpy(send_this, &send_this[i]);
478                         serv_puts(buf);
479                 }
480         }
481
482         /** Unswap the sockets. */
483         i = WC->serv_sock;
484         WC->serv_sock = WC->chat_sock;
485         WC->chat_sock = i;
486
487         wprintf("<FORM METHOD=\"POST\" action=\"chat_send\" NAME=\"chatsendform\">\n");
488         wprintf("<input type=\"hidden\" name=\"nonce\" value=\"%ld\">\n", WC->nonce);
489         wprintf("<INPUT TYPE=\"text\" SIZE=\"80\" MAXLENGTH=\"%d\" "
490                 "NAME=\"send_this\">\n", SIZ-10);
491         wprintf("<br />");
492         wprintf("<INPUT TYPE=\"submit\" NAME=\"send_button\" VALUE=\"%s\">\n", _("Send"));
493         wprintf("<INPUT TYPE=\"submit\" NAME=\"help_button\" VALUE=\"%s\">\n", _("Help"));
494         wprintf("<INPUT TYPE=\"submit\" NAME=\"list_button\" VALUE=\"%s\">\n", _("List users"));
495         wprintf("<INPUT TYPE=\"submit\" NAME=\"exit_button\" VALUE=\"%s\">\n", _("Exit"));
496         wprintf("</FORM>\n");
497
498         wprintf("</BODY></HTML>\n");
499         wDumpContent(0);
500 }
501
502 /*@}*/