Annotation of loncom/interface/lonmsg.pm, revision 1.93
1.1 www 1: # The LearningOnline Network with CAPA
1.26 albertel 2: # Routines for messaging
3: #
1.93 ! www 4: # $Id: lonmsg.pm,v 1.92 2004/03/02 02:51:09 www Exp $
1.26 albertel 5: #
6: # Copyright Michigan State University Board of Trustees
7: #
8: # This file is part of the LearningOnline Network with CAPA (LON-CAPA).
9: #
10: # LON-CAPA is free software; you can redistribute it and/or modify
11: # it under the terms of the GNU General Public License as published by
12: # the Free Software Foundation; either version 2 of the License, or
13: # (at your option) any later version.
14: #
15: # LON-CAPA is distributed in the hope that it will be useful,
16: # but WITHOUT ANY WARRANTY; without even the implied warranty of
17: # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18: # GNU General Public License for more details.
19: #
20: # You should have received a copy of the GNU General Public License
21: # along with LON-CAPA; if not, write to the Free Software
22: # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
23: #
24: # /home/httpd/html/adm/gpl.txt
25: #
26: # http://www.lon-capa.org/
1.1 www 27: #
1.75 www 28:
29:
1.1 www 30: package Apache::lonmsg;
31:
1.58 bowersj2 32: =pod
33:
34: =head1 NAME
35:
36: Apache::lonmsg: supports internal messaging
37:
38: =head1 SYNOPSIS
39:
40: lonmsg provides routines for sending messages, receiving messages, and
41: a handler to allow users to read, send, and delete messages.
42:
43: =head1 OVERVIEW
44:
45: =head2 Messaging Overview
46:
47: X<messages>LON-CAPA provides an internal messaging system similar to
48: email, but customized for LON-CAPA's usage. LON-CAPA implements its
49: own messaging system, rather then building on top of email, because of
50: the features LON-CAPA messages can offer that conventional e-mail can
51: not:
52:
53: =over 4
54:
55: =item * B<Critical messages>: A message the recipient B<must>
56: acknowlegde receipt of before they are allowed to continue using the
57: system, preventing a user from claiming they never got a message
58:
59: =item * B<Receipts>: LON-CAPA can reliably send reciepts informing the
60: sender that it has been read; again, useful for preventing students
61: from claiming they did not see a message. (While conventional e-mail
62: has some reciept support, it's sporadic, e-mail client-specific, and
63: generally the receiver can opt to not send one, making it useless in
64: this case.)
65:
66: =item * B<Context>: LON-CAPA knows about the sender, such as where
67: they are in a course. When a student mails an instructor asking for
68: help on the problem, the instructor receives not just the student's
69: question, but all submissions the student has made up to that point,
70: the user's rendering of the problem, and the complete view the student
71: saw of the resource, including discussion up to that point. Finally,
72: the instructor is reading all of this inside of LON-CAPA, not their
73: email program, so they have full access to LON-CAPA's grading
74: interface, or other features they may wish to use in response to the
75: student's query.
76:
77: =back
78:
79: Users can ask LON-CAPA to forward messages to conventional e-mail
80: addresses on their B<PREF> screen, but generally, LON-CAPA messages
81: are much more useful then traditional email can be made to be, even
82: with HTML support.
83:
84: Right now, this document will cover just how to send a message, since
85: it is likely you will not need to programmatically read messages,
86: since lonmsg already implements that functionality.
87:
88: =head1 FUNCTIONS
89:
90: =over 4
91:
92: =cut
93:
1.1 www 94: use strict;
95: use Apache::lonnet();
1.2 www 96: use vars qw($msgcount);
1.47 albertel 97: use HTML::TokeParser();
1.5 www 98: use Apache::Constants qw(:common);
1.47 albertel 99: use Apache::loncommon();
100: use Apache::lontexconvert();
101: use HTML::Entities();
1.53 www 102: use Mail::Send;
1.67 www 103: use Apache::lonlocal;
1.1 www 104:
1.65 www 105: # Querystring component with sorting type
106: my $sqs;
107:
1.1 www 108: # ===================================================================== Package
109:
1.3 www 110: sub packagemsg {
1.51 www 111: my ($subject,$message,$citation,$baseurl,$attachmenturl)=@_;
1.47 albertel 112: $message =&HTML::Entities::encode($message);
113: $citation=&HTML::Entities::encode($citation);
114: $subject =&HTML::Entities::encode($subject);
1.49 albertel 115: #remove machine specification
116: $baseurl =~ s|^http://[^/]+/|/|;
117: $baseurl =&HTML::Entities::encode($baseurl);
1.51 www 118: #remove machine specification
119: $attachmenturl =~ s|^http://[^/]+/|/|;
1.52 www 120: $attachmenturl =&HTML::Entities::encode($attachmenturl);
1.51 www 121:
1.2 www 122: my $now=time;
123: $msgcount++;
1.6 www 124: my $partsubj=$subject;
125: $partsubj=&Apache::lonnet::escape($partsubj);
126: my $msgid=&Apache::lonnet::escape(
127: $now.':'.$partsubj.':'.$ENV{'user.name'}.':'.
128: $ENV{'user.domain'}.':'.$msgcount.':'.$$);
1.49 albertel 129: my $result='<sendername>'.$ENV{'user.name'}.'</sendername>'.
1.1 www 130: '<senderdomain>'.$ENV{'user.domain'}.'</senderdomain>'.
131: '<subject>'.$subject.'</subject>'.
1.67 www 132: '<time>'.&Apache::lonlocal::locallocaltime($now).'</time>'.
1.1 www 133: '<servername>'.$ENV{'SERVER_NAME'}.'</servername>'.
134: '<host>'.$ENV{'HTTP_HOST'}.'</host>'.
135: '<client>'.$ENV{'REMOTE_ADDR'}.'</client>'.
136: '<browsertype>'.$ENV{'browser.type'}.'</browsertype>'.
137: '<browseros>'.$ENV{'browser.os'}.'</browseros>'.
138: '<browserversion>'.$ENV{'browser.version'}.'</browserversion>'.
139: '<browsermathml>'.$ENV{'browser.mathml'}.'</browsermathml>'.
140: '<browserraw>'.$ENV{'HTTP_USER_AGENT'}.'</browserraw>'.
141: '<courseid>'.$ENV{'request.course.id'}.'</courseid>'.
1.85 www 142: '<coursesec>'.$ENV{'request.course.sec'}.'</coursesec>'.
1.1 www 143: '<role>'.$ENV{'request.role'}.'</role>'.
144: '<resource>'.$ENV{'request.filename'}.'</resource>'.
1.2 www 145: '<msgid>'.$msgid.'</msgid>'.
1.49 albertel 146: '<message>'.$message.'</message>';
147: if (defined($citation)) {
148: $result.='<citation>'.$citation.'</citation>';
149: }
150: if (defined($baseurl)) {
151: $result.= '<baseurl>'.$baseurl.'</baseurl>';
152: }
1.51 www 153: if (defined($attachmenturl)) {
1.52 www 154: $result.= '<attachmenturl>'.$attachmenturl.'</attachmenturl>';
1.51 www 155: }
1.49 albertel 156: return $msgid,$result;
1.1 www 157: }
158:
1.2 www 159: # ================================================== Unpack message into a hash
160:
1.3 www 161: sub unpackagemsg {
1.52 www 162: my ($message,$notoken)=@_;
1.2 www 163: my %content=();
164: my $parser=HTML::TokeParser->new(\$message);
165: my $token;
166: while ($token=$parser->get_token) {
167: if ($token->[0] eq 'S') {
168: my $entry=$token->[1];
169: my $value=$parser->get_text('/'.$entry);
170: $content{$entry}=$value;
171: }
172: }
1.52 www 173: if ($content{'attachmenturl'}) {
174: my ($fname,$ft)=($content{'attachmenturl'}=~/\/(\w+)\.(\w+)$/);
175: if ($notoken) {
1.67 www 176: $content{'message'}.='<p>'.&mt('Attachment').': <tt>'.$fname.'.'.$ft.'</tt>';
1.52 www 177: } else {
1.67 www 178: $content{'message'}.='<p>'.&mt('Attachment').': <a href="'.
1.52 www 179: &Apache::lonnet::tokenwrapper($content{'attachmenturl'}).
180: '"><tt>'.$fname.'.'.$ft.'</tt></a>';
181: }
182: }
1.2 www 183: return %content;
184: }
185:
1.6 www 186: # ======================================================= Get info out of msgid
187:
188: sub unpackmsgid {
1.7 www 189: my $msgid=&Apache::lonnet::unescape(shift);
1.6 www 190: my ($sendtime,$shortsubj,$fromname,$fromdomain)=split(/\:/,
1.7 www 191: &Apache::lonnet::unescape($msgid));
1.8 albertel 192: my %status=&Apache::lonnet::get('email_status',[$msgid]);
1.6 www 193: if ($status{$msgid}=~/^error\:/) { $status{$msgid}=''; }
194: unless ($status{$msgid}) { $status{$msgid}='new'; }
195: return ($sendtime,$shortsubj,$fromname,$fromdomain,$status{$msgid});
196: }
197:
1.53 www 198:
199: sub sendemail {
200: my ($to,$subject,$body)=@_;
201: $body=
1.67 www 202: "*** ".&mt('This is an automatic message generated by the LON-CAPA system.')."\n".
203: "*** ".&mt('Please do not reply to this address.')."\n\n".$body;
1.53 www 204: my $msg = new Mail::Send;
205: $msg->to($to);
206: $msg->subject('[LON-CAPA] '.$subject);
1.68 www 207: if (my $fh = $msg->open('smtp',Server => 'localhost')) {
208: print $fh $body;
209: $fh->close;
210: }
1.53 www 211: }
212:
213: # ==================================================== Send notification emails
214:
215: sub sendnotification {
216: my ($to,$touname,$toudom,$subj,$crit)=@_;
217: my $sender=$ENV{'environment.firstname'}.' '.$ENV{'environment.lastname'};
218: my $critical=($crit?' critical':'');
219: my $url='http://'.
220: $Apache::lonnet::hostname{&Apache::lonnet::homeserver($touname,$toudom)}.
1.54 www 221: '/adm/email?username='.$touname.'&domain='.$toudom;
1.53 www 222: my $body=(<<ENDMSG);
223: You received a$critical message from $sender in LON-CAPA. The subject is
224:
225: $subj
226:
227: Use
228:
229: $url
230:
231: to access this message.
232: ENDMSG
233: &sendemail($to,'New'.$critical.' message from '.$sender,$body);
234: }
1.40 www 235: # ============================================================= Check for email
236:
237: sub newmail {
238: if ((time-$ENV{'user.mailcheck.time'})>300) {
239: my %what=&Apache::lonnet::get('email_status',['recnewemail']);
240: &Apache::lonnet::appenv('user.mailcheck.time'=>time);
241: if ($what{'recnewemail'}>0) { return 1; }
242: }
243: return 0;
244: }
245:
1.1 www 246: # =============================== Automated message to the author of a resource
247:
1.58 bowersj2 248: =pod
249:
250: =item * B<author_res_msg($filename, $message)>: Sends message $message to the owner
251: of the resource with the URI $filename.
252:
253: =cut
254:
1.1 www 255: sub author_res_msg {
256: my ($filename,$message)=@_;
1.2 www 257: unless ($message) { return 'empty'; }
1.1 www 258: $filename=&Apache::lonnet::declutter($filename);
1.72 www 259: my ($domain,$author,@dummy)=split(/\//,$filename);
1.1 www 260: my $homeserver=&Apache::lonnet::homeserver($author,$domain);
261: if ($homeserver ne 'no_host') {
262: my $id=unpack("%32C*",$message);
1.2 www 263: my $msgid;
1.72 www 264: ($msgid,$message)=&packagemsg($filename,$message);
1.3 www 265: return &Apache::lonnet::reply('put:'.$domain.':'.$author.
1.72 www 266: ':nohist_res_msgs:'.
267: &Apache::lonnet::escape($filename.'_'.$id).'='.
268: &Apache::lonnet::escape($message),$homeserver);
1.1 www 269: }
1.2 www 270: return 'no_host';
1.73 www 271: }
272:
273: # =========================================== Retrieve author resource messages
274:
275: sub retrieve_author_res_msg {
1.75 www 276: my $url=shift;
1.73 www 277: $url=&Apache::lonnet::declutter($url);
1.80 www 278: my ($domain,$author)=($url=~/^(\w+)\/(\w+)\//);
1.76 www 279: my %errormsgs=&Apache::lonnet::dump('nohist_res_msgs',$domain,$author);
1.73 www 280: my $msgs='';
281: foreach (keys %errormsgs) {
1.80 www 282: if ($_=~/^\Q$url\E\_\d+$/) {
1.73 www 283: my %content=&unpackagemsg($errormsgs{$_});
1.74 www 284: $msgs.='<p><img src="/adm/lonMisc/bomb.gif" /><b>'.
285: $content{'time'}.'</b>: '.$content{'message'}.
286: '<br /></p>';
1.73 www 287: }
288: }
289: return $msgs;
290: }
291:
292:
293: # =============================== Delete all author messages related to one URL
294:
295: sub del_url_author_res_msg {
1.75 www 296: my $url=shift;
1.73 www 297: $url=&Apache::lonnet::declutter($url);
1.77 www 298: my ($domain,$author)=($url=~/^(\w+)\/(\w+)\//);
299: my @delmsgs=();
300: foreach (&Apache::lonnet::getkeys('nohist_res_msgs',$domain,$author)) {
301: if ($_=~/^\Q$url\E\_\d+$/) {
302: push (@delmsgs,$_);
303: }
304: }
305: return &Apache::lonnet::del('nohist_res_msgs',\@delmsgs,$domain,$author);
1.73 www 306: }
307:
308: # ================= Return hash with URLs for which there is a resource message
309:
310: sub all_url_author_res_msg {
311: my ($author,$domain)=@_;
1.75 www 312: my %returnhash=();
1.76 www 313: foreach (&Apache::lonnet::getkeys('nohist_res_msgs',$domain,$author)) {
1.75 www 314: $_=~/^(.+)\_\d+/;
315: $returnhash{$1}=1;
316: }
317: return %returnhash;
1.1 www 318: }
319:
320: # ================================================== Critical message to a user
321:
1.38 www 322: sub user_crit_msg_raw {
1.24 www 323: my ($user,$domain,$subject,$message,$sendback)=@_;
1.2 www 324: # Check if allowed missing
325: my $status='';
326: my $msgid='undefined';
327: unless (($message)&&($user)&&($domain)) { $status='empty'; };
328: my $homeserver=&Apache::lonnet::homeserver($user,$domain);
329: if ($homeserver ne 'no_host') {
1.3 www 330: ($msgid,$message)=&packagemsg($subject,$message);
1.24 www 331: if ($sendback) { $message.='<sendback>true</sendback>'; }
1.4 www 332: $status=&Apache::lonnet::critical(
333: 'put:'.$domain.':'.$user.':critical:'.
334: &Apache::lonnet::escape($msgid).'='.
335: &Apache::lonnet::escape($message),$homeserver);
1.45 www 336: if ($ENV{'request.course.id'}) {
337: &user_normal_msg_raw(
338: $ENV{'course.'.$ENV{'request.course.id'}.'.num'},
339: $ENV{'course.'.$ENV{'request.course.id'}.'.domain'},
340: 'Critical ['.$user.':'.$domain.']',
341: $message);
342: }
1.2 www 343: } else {
344: $status='no_host';
345: }
1.53 www 346: # Notifications
347: my %userenv = &Apache::lonnet::get('environment',['critnotification'],
348: $domain,$user);
349: if ($userenv{'critnotification'}) {
350: &sendnotification($userenv{'critnotification'},$user,$domain,$subject,1);
351: }
352: # Log this
1.2 www 353: &Apache::lonnet::logthis(
1.4 www 354: 'Sending critical email '.$msgid.
1.2 www 355: ', log status: '.
356: &Apache::lonnet::log($ENV{'user.domain'},$ENV{'user.name'},
357: $ENV{'user.home'},
358: 'Sending critical '.$msgid.' to '.$user.' at '.$domain.' with status: '
1.4 www 359: .$status));
1.2 www 360: return $status;
361: }
362:
1.38 www 363: # New routine that respects "forward" and calls old routine
364:
1.58 bowersj2 365: =pod
366:
367: =item * B<user_crit_msg($user, $domain, $subject, $message, $sendback)>: Sends
368: a critical message $message to the $user at $domain. If $sendback is true,
369: a reciept will be sent to the current user when $user recieves the message.
370:
371: =cut
372:
1.38 www 373: sub user_crit_msg {
374: my ($user,$domain,$subject,$message,$sendback)=@_;
375: my $status='';
376: my %userenv = &Apache::lonnet::get('environment',['msgforward'],
377: $domain,$user);
378: my $msgforward=$userenv{'msgforward'};
379: if ($msgforward) {
380: foreach (split(/\,/,$msgforward)) {
381: my ($forwuser,$forwdomain)=split(/\:/,$_);
382: $status.=
383: &user_crit_msg_raw($forwuser,$forwdomain,$subject,$message,
384: $sendback).' ';
385: }
386: } else {
387: $status=&user_crit_msg_raw($user,$domain,$subject,$message,$sendback);
388: }
389: return $status;
390: }
391:
1.2 www 392: # =================================================== Critical message received
393:
394: sub user_crit_received {
1.12 www 395: my $msgid=shift;
396: my %message=&Apache::lonnet::get('critical',[$msgid]);
1.52 www 397: my %contents=&unpackagemsg($message{$msgid},1);
1.24 www 398: my $status='rec: '.($contents{'sendback'}?
1.5 www 399: &user_normal_msg($contents{'sendername'},$contents{'senderdomain'},
1.82 www 400: &mt('Receipt').': '.$ENV{'user.name'}.' '.&mt('at').' '.$ENV{'user.domain'}.', '.$contents{'subject'},
1.67 www 401: &mt('User').' '.$ENV{'user.name'}.' '.&mt('at').' '.$ENV{'user.domain'}.
1.42 www 402: ' acknowledged receipt of message'."\n".' "'.
1.67 www 403: $contents{'subject'}.'"'."\n".&mt('dated').' '.
1.42 www 404: $contents{'time'}.".\n"
405: ):'no msg req');
1.5 www 406: $status.=' trans: '.
1.12 www 407: &Apache::lonnet::put(
408: 'nohist_email',{$contents{'msgid'} => $message{$msgid}});
1.5 www 409: $status.=' del: '.
1.9 albertel 410: &Apache::lonnet::del('critical',[$contents{'msgid'}]);
1.5 www 411: &Apache::lonnet::log($ENV{'user.domain'},$ENV{'user.name'},
412: $ENV{'user.home'},'Received critical message '.
413: $contents{'msgid'}.
414: ', '.$status);
1.12 www 415: return $status;
1.2 www 416: }
417:
418: # ======================================================== Normal communication
419:
1.38 www 420: sub user_normal_msg_raw {
1.51 www 421: my ($user,$domain,$subject,$message,$citation,$baseurl,$attachmenturl)=@_;
1.2 www 422: # Check if allowed missing
423: my $status='';
424: my $msgid='undefined';
425: unless (($message)&&($user)&&($domain)) { $status='empty'; };
426: my $homeserver=&Apache::lonnet::homeserver($user,$domain);
427: if ($homeserver ne 'no_host') {
1.51 www 428: ($msgid,$message)=&packagemsg($subject,$message,$citation,$baseurl,
429: $attachmenturl);
1.4 www 430: $status=&Apache::lonnet::critical(
431: 'put:'.$domain.':'.$user.':nohist_email:'.
432: &Apache::lonnet::escape($msgid).'='.
433: &Apache::lonnet::escape($message),$homeserver);
1.40 www 434: &Apache::lonnet::put
435: ('email_status',{'recnewemail'=>time},$domain,$user);
1.2 www 436: } else {
437: $status='no_host';
1.53 www 438: }
439: # Notifications
440: my %userenv = &Apache::lonnet::get('environment',['notification'],
441: $domain,$user);
442: if ($userenv{'notification'}) {
443: &sendnotification($userenv{'notification'},$user,$domain,$subject,0);
1.2 www 444: }
445: &Apache::lonnet::log($ENV{'user.domain'},$ENV{'user.name'},
446: $ENV{'user.home'},
447: 'Sending '.$msgid.' to '.$user.' at '.$domain.' with status: '.$status);
448: return $status;
449: }
1.38 www 450:
451: # New routine that respects "forward" and calls old routine
452:
1.58 bowersj2 453: =pod
454:
455: =item * B<user_normal_msg($user, $domain, $subject, $message,
456: $citation, $baseurl, $attachmenturl)>: Sends a message to the
457: $user at $domain, with subject $subject and message $message.
458:
459: =cut
460:
1.38 www 461: sub user_normal_msg {
1.52 www 462: my ($user,$domain,$subject,$message,$citation,$baseurl,$attachmenturl)=@_;
1.38 www 463: my $status='';
464: my %userenv = &Apache::lonnet::get('environment',['msgforward'],
465: $domain,$user);
466: my $msgforward=$userenv{'msgforward'};
467: if ($msgforward) {
468: foreach (split(/\,/,$msgforward)) {
469: my ($forwuser,$forwdomain)=split(/\:/,$_);
470: $status.=
471: &user_normal_msg_raw($forwuser,$forwdomain,$subject,$message,
1.52 www 472: $citation,$baseurl,$attachmenturl).' ';
1.38 www 473: }
474: } else {
1.49 albertel 475: $status=&user_normal_msg_raw($user,$domain,$subject,$message,
1.52 www 476: $citation,$baseurl,$attachmenturl);
1.38 www 477: }
478: return $status;
479: }
480:
1.2 www 481:
1.7 www 482: # =============================================================== Status Change
483:
484: sub statuschange {
485: my ($msgid,$newstatus)=@_;
1.8 albertel 486: my %status=&Apache::lonnet::get('email_status',[$msgid]);
1.7 www 487: if ($status{$msgid}=~/^error\:/) { $status{$msgid}=''; }
488: unless ($status{$msgid}) { $status{$msgid}='new'; }
489: unless (($status{$msgid} eq 'replied') ||
490: ($status{$msgid} eq 'forwarded')) {
1.10 albertel 491: &Apache::lonnet::put('email_status',{$msgid => $newstatus});
1.7 www 492: }
1.14 www 493: if (($newstatus eq 'deleted') || ($newstatus eq 'new')) {
494: &Apache::lonnet::put('email_status',{$msgid => $newstatus});
495: }
1.7 www 496: }
1.14 www 497:
1.17 www 498: # ======================================================= Display a course list
499:
500: sub discourse {
501: my $r=shift;
502: my %courselist=&Apache::lonnet::dump(
503: 'classlist',
504: $ENV{'course.'.$ENV{'request.course.id'}.'.domain'},
505: $ENV{'course.'.$ENV{'request.course.id'}.'.num'});
506: my $now=time;
1.67 www 507: my %lt=&Apache::lonlocal::texthash('cfa' => 'Check for All',
508: 'cfs' => 'Check for Section/Group',
509: 'cfn' => 'Check for None');
1.17 www 510: $r->print(<<ENDDISHEADER);
1.92 www 511: <input type="hidden" name="sendmode" value="group" />
1.17 www 512: <script>
513: function checkall() {
514: for (i=0; i<document.forms.compemail.elements.length; i++) {
515: if
516: (document.forms.compemail.elements[i].name.indexOf('send_to_')==0) {
517: document.forms.compemail.elements[i].checked=true;
518: }
519: }
520: }
521:
1.19 www 522: function checksec() {
523: for (i=0; i<document.forms.compemail.elements.length; i++) {
524: if
525: (document.forms.compemail.elements[i].name.indexOf
526: ('send_to_&&&'+document.forms.compemail.chksec.value)==0) {
527: document.forms.compemail.elements[i].checked=true;
528: }
529: }
530: }
531:
1.17 www 532: function uncheckall() {
533: for (i=0; i<document.forms.compemail.elements.length; i++) {
534: if
535: (document.forms.compemail.elements[i].name.indexOf('send_to_')==0) {
536: document.forms.compemail.elements[i].checked=false;
537: }
538: }
539: }
540: </script>
1.92 www 541: <input type="button" onClick="checkall()" value="$lt{'cfa'}" />
542: <input type="button" onClick="checksec()" value="$lt{'cfs'}" />
543: <input type="text" size="5" name=chksec />
544: <input type="button" onClick="uncheckall()" value="$lt{'cfn'}" />
1.17 www 545: <p>
546: ENDDISHEADER
1.61 www 547: my %coursepersonnel=
548: &Apache::lonnet::get_course_adv_roles();
549: foreach my $role (sort keys %coursepersonnel) {
550: foreach (split(/\,/,$coursepersonnel{$role})) {
551: my ($puname,$pudom)=split(/\:/,$_);
552: $r->print(
553: '<br /><input type="checkbox" name="send_to_&&&&&&_'.
554: $puname.':'.$pudom.'" /> '.
555: &Apache::loncommon::plainname($puname,
556: $pudom).' ('.$_.'), <i>'.$role.'</i>');
557: }
558: }
559:
1.28 harris41 560: foreach (sort keys %courselist) {
1.17 www 561: my ($end,$start)=split(/\:/,$courselist{$_});
562: my $active=1;
563: if (($end) && ($now>$end)) { $active=0; }
564: if ($active) {
565: my ($sname,$sdom)=split(/\:/,$_);
566: my %reply=&Apache::lonnet::get('environment',
567: ['firstname','middlename','lastname','generation'],
568: $sdom,$sname);
1.19 www 569: my $section=&Apache::lonnet::usection
570: ($sdom,$sname,$ENV{'request.course.id'});
571: $r->print(
572: '<br><input type=checkbox name="send_to_&&&'.$section.'&&&_'.$_.'"> '.
1.17 www 573: $reply{'firstname'}.' '.
574: $reply{'middlename'}.' '.
575: $reply{'lastname'}.' '.
576: $reply{'generation'}.
1.19 www 577: ' ('.$_.') '.$section);
1.17 www 578: }
1.28 harris41 579: }
1.17 www 580: }
581:
1.13 www 582: # ==================================================== Display Critical Message
1.5 www 583:
1.12 www 584: sub discrit {
585: my $r=shift;
1.67 www 586: my $header = '<h1><font color=red>'.&mt('Critical Messages').'</font></h1>'.
1.30 matthew 587: '<form action=/adm/email method=post>'.
588: '<input type=hidden name=confirm value=true>';
589: my %what=&Apache::lonnet::dump('critical');
590: my $result = '';
591: foreach (sort keys %what) {
592: my %content=&unpackagemsg($what{$_});
593: next if ($content{'senderdomain'} eq '');
594: $content{'message'}=~s/\n/\<br\>/g;
1.67 www 595: $result.='<hr>'.&mt('From').': <b>'.
1.37 www 596: &Apache::loncommon::aboutmewrapper(
597: &Apache::loncommon::plainname($content{'sendername'},$content{'senderdomain'}),$content{'sendername'},$content{'senderdomain'}).'</b> ('.
598: $content{'sendername'}.'@'.
599: $content{'senderdomain'}.') '.$content{'time'}.
1.67 www 600: '<br>'.&mt('Subject').': '.$content{'subject'}.
1.36 www 601: '<br><blockquote>'.
602: &Apache::lontexconvert::msgtexconverted($content{'message'}).
1.84 www 603: '</blockquote><small>'.
604: &mt('You have to confirm that you received this message. After confirmation, this message will be moved to your regular inbox').
605: '</small><br />'.
1.67 www 606: '<input type=submit name="rec_'.$_.'" value="'.&mt('Confirm Receipt').'">'.
1.30 matthew 607: '<input type=submit name="reprec_'.$_.'" '.
1.67 www 608: 'value="'.&mt('Confirm Receipt and Reply').'">';
1.30 matthew 609: }
610: # Check to see if there were any messages.
611: if ($result eq '') {
1.67 www 612: $result = "<h2>".&mt('You have no critical messages.')."</h2>".
613: '<a href="/adm/roles">'.&mt('Select a course').'</a>';
1.30 matthew 614: } else {
615: $r->print($header);
616: }
617: $r->print($result);
618: $r->print('<input type=hidden name="displayedcrit" value="true"></form>');
1.12 www 619: }
620:
1.65 www 621: sub sortedmessages {
622: my @messages = &Apache::lonnet::getkeys('nohist_email');
623: #unpack the varibles and repack into temp for sorting
624: my @temp;
625: foreach (@messages) {
626: my $msgid=&Apache::lonnet::escape($_);
627: my ($sendtime,$shortsubj,$fromname,$fromdomain,$status)=
628: &Apache::lonmsg::unpackmsgid($msgid);
629: my @temp1 = ($sendtime,$shortsubj,$fromname,$fromdomain,$status,
630: $msgid);
631: push @temp ,\@temp1;
632: }
633: #default sort
634: @temp = sort {$a->[0] <=> $b->[0]} @temp;
635: if ($ENV{'form.sortedby'} eq "date"){
636: @temp = sort {$a->[0] <=> $b->[0]} @temp;
637: }
638: if ($ENV{'form.sortedby'} eq "revdate"){
639: @temp = sort {$b->[0] <=> $a->[0]} @temp;
640: }
641: if ($ENV{'form.sortedby'} eq "user"){
642: @temp = sort {lc($a->[2]) cmp lc($b->[2])} @temp;
643: }
644: if ($ENV{'form.sortedby'} eq "revuser"){
645: @temp = sort {lc($b->[2]) cmp lc($a->[2])} @temp;
646: }
647: if ($ENV{'form.sortedby'} eq "domain"){
648: @temp = sort {$a->[3] cmp $b->[3]} @temp;
649: }
650: if ($ENV{'form.sortedby'} eq "revdomain"){
651: @temp = sort {$b->[3] cmp $a->[3]} @temp;
652: }
653: if ($ENV{'form.sortedby'} eq "subject"){
654: @temp = sort {lc($a->[1]) cmp lc($b->[1])} @temp;
655: }
656: if ($ENV{'form.sortedby'} eq "revsubject"){
657: @temp = sort {lc($b->[1]) cmp lc($a->[1])} @temp;
658: }
659: if ($ENV{'form.sortedby'} eq "status"){
660: @temp = sort {$a->[4] cmp $b->[4]} @temp;
661: }
662: if ($ENV{'form.sortedby'} eq "revstatus"){
663: @temp = sort {$b->[4] cmp $a->[4]} @temp;
664: }
665: return @temp;
666: }
667:
1.15 www 668: # ======================================================== Display all messages
669:
1.14 www 670: sub disall {
671: my $r=shift;
1.29 www 672: $r->print(<<ENDDISHEADER);
673: <script>
674: function checkall() {
675: for (i=0; i<document.forms.disall.elements.length; i++) {
676: if
677: (document.forms.disall.elements[i].name.indexOf('delmark_')==0) {
678: document.forms.disall.elements[i].checked=true;
679: }
680: }
681: }
682:
683: function uncheckall() {
684: for (i=0; i<document.forms.disall.elements.length; i++) {
685: if
686: (document.forms.disall.elements[i].name.indexOf('delmark_')==0) {
687: document.forms.disall.elements[i].checked=false;
688: }
689: }
690: }
691: </script>
692: ENDDISHEADER
1.67 www 693: $r->print('<h1>'.&mt('Display All Messages').'</h1><form method=post name=disall '.
1.63 albertel 694: 'action="/adm/email">'.
695: '<table border=2><tr><th colspan=2> </th><th>');
1.62 www 696: if ($ENV{'form.sortedby'} eq "revdate") {
1.67 www 697: $r->print('<a href = "?sortedby=date">'.&mt('Date').'</a></th>');
1.62 www 698: } else {
1.67 www 699: $r->print('<a href = "?sortedby=revdate">'.&mt('Date').'</a></th>');
1.62 www 700: }
701: $r->print('<th>');
702: if ($ENV{'form.sortedby'} eq "revuser") {
1.67 www 703: $r->print('<a href = "?sortedby=user">'.&mt('Username').'</a>');
1.62 www 704: } else {
1.67 www 705: $r->print('<a href = "?sortedby=revuser">'.&mt('Username').'</a>');
1.62 www 706: }
707: $r->print('</th><th>');
708: if ($ENV{'form.sortedby'} eq "revdomain") {
1.67 www 709: $r->print('<a href = "?sortedby=domain">'.&mt('Domain').'</a>');
1.62 www 710: } else {
1.67 www 711: $r->print('<a href = "?sortedby=revdomain">'.&mt('Domain').'</a>');
1.62 www 712: }
713: $r->print('</th><th>');
714: if ($ENV{'form.sortedby'} eq "revsubject") {
1.67 www 715: $r->print('<a href = "?sortedby=subject">'.&mt('Subject').'</a>');
1.62 www 716: } else {
1.67 www 717: $r->print('<a href = "?sortedby=revsubject">'.&mt('Subject').'</a>');
1.62 www 718: }
719: $r->print('</th><th>');
720: if ($ENV{'form.sortedby'} eq "revstatus") {
1.67 www 721: $r->print('<a href = "?sortedby=status">'.&mt('Status').'</th>');
1.62 www 722: } else {
1.67 www 723: $r->print('<a href = "?sortedby=revstatus">'.&mt('Status').'</th>');
1.62 www 724: }
725: $r->print('</tr>');
1.65 www 726: my @temp=sortedmessages();
1.63 albertel 727: foreach (@temp){
1.64 www 728: my ($sendtime,$shortsubj,$fromname,$fromdomain,$status,$origID)= @$_;
1.63 albertel 729: if (($status ne 'deleted') && defined($sendtime) && $sendtime!~/error/) {
1.39 albertel 730: if ($status eq 'new') {
731: $r->print('<tr bgcolor="#FFBB77">');
732: } elsif ($status eq 'read') {
733: $r->print('<tr bgcolor="#BBBB77">');
734: } elsif ($status eq 'replied') {
1.62 www 735: $r->print('<tr bgcolor="#AAAA88">');
1.39 albertel 736: } else {
737: $r->print('<tr bgcolor="#99BBBB">');
738: }
1.65 www 739: $r->print('<td><a href="/adm/email?display='.$origID.$sqs.
1.67 www 740: '">'.&mt('Open').'</a></td><td><a href="/adm/email?markdel='.$origID.$sqs.
1.92 www 741: '">'.&mt('Delete').'</a><input type=checkbox name="delmark_'.$origID.'" /></td>'.
1.66 www 742: '<td>'.&Apache::lonlocal::locallocaltime($sendtime).'</td><td>'.
1.39 albertel 743: $fromname.'</td><td>'.$fromdomain.'</td><td>'.
1.14 www 744: &Apache::lonnet::unescape($shortsubj).'</td><td>'.
745: $status.'</td></tr>');
1.63 albertel 746: }
747: }
748: $r->print('</table><p>'.
1.67 www 749: '<a href="javascript:checkall()">'.&mt('Check All').'</a> '.
750: '<a href="javascript:uncheckall()">'.&mt('Uncheck All').'</a><p>'.
1.65 www 751: '<input type="hidden" name="sortedby" value="'.$ENV{'form.sortedby'}.'" />'.
1.92 www 752: '<input type="submit" name="markeddel" value="'.&mt('Delete Checked').'" />'.
1.25 www 753: '</form></body></html>');
1.14 www 754: }
755:
1.15 www 756: # ============================================================== Compose output
757:
758: sub compout {
1.92 www 759: my ($r,$forwarding,$replying,$broadcast)=@_;
760:
761: if ($broadcast eq 'individual') {
762: &printheader($r,'/adm/email?compose=individual',
763: 'Send a Message');
764: } elsif ($broadcast) {
765: &printheader($r,'/adm/email?compose=group',
766: 'Broadcast Message');
767: } elsif ($forwarding) {
768: &Apache::lonhtmlcommon::add_breadcrumb
769: ({href=>"/adm/email?display=".&Apache::lonnet::escape($forwarding),
770: text=>"Display Message"});
771: &printheader($r,'/adm/email?forward='.&Apache::lonnet::escape($forwarding),
772: 'Forwarding a Message');
773: } elsif ($replying) {
774: &Apache::lonhtmlcommon::add_breadcrumb
775: ({href=>"/adm/email?display=".&Apache::lonnet::escape($replying),
776: text=>"Display Message"});
777: &printheader($r,'/adm/email?replyto='.&Apache::lonnet::escape($replying),
778: 'Replying to a Message');
779: } else {
780: &printheader($r,'/adm/email?compose=upload',
781: 'Distribute from Uploaded File');
782: }
783:
1.89 www 784: my $dispcrit='';
1.15 www 785: my $dissub='';
786: my $dismsg='';
1.67 www 787: my $func=&mt('Send New');
1.69 www 788: my %lt=&Apache::lonlocal::texthash('us' => 'Username',
789: 'do' => 'Domain',
790: 'ad' => 'Additional Recipients',
791: 'sb' => 'Subject',
792: 'ca' => 'Cancel',
793: 'ma' => 'Mail');
794:
795: if (&Apache::lonnet::allowed('srm',$ENV{'request.course.id'})) {
1.35 bowersj2 796: my $crithelp = Apache::loncommon::help_open_topic("Course_Critical_Message");
1.15 www 797: $dispcrit=
1.92 www 798: '<input type="checkbox" name="critmsg" /> '.&mt('Send as critical message').' ' . $crithelp .
1.35 bowersj2 799: '<br>'.
1.92 www 800: '<input type="checkbox" name="sendbck" /> '.&mt('Send as critical message').' ' .
1.67 www 801: &mt('and return receipt') . $crithelp . '<p>';
1.92 www 802: }
803: my %message;
804: my %content;
805: my $defdom=$ENV{'user.domain'};
1.15 www 806: if ($forwarding) {
1.92 www 807: %message=&Apache::lonnet::get('nohist_email',[$forwarding]);
808: %content=&unpackagemsg($message{$forwarding});
809: $dispcrit.='<input type="hidden" name="forwid" value="'.
810: $forwarding.'" />';
811: $func=&mt('Forward');
812:
813: $dissub=&mt('Forwarding').': '.$content{'subject'};
814: $dismsg=&mt('Forwarded message from').' '.
815: $content{'sendername'}.' '.&mt('at').' '.$content{'senderdomain'};
816: }
817: if ($replying) {
818: %message=&Apache::lonnet::get('nohist_email',[$replying]);
819: %content=&unpackagemsg($message{$replying});
820: $dispcrit.='<input type="hidden" name="forwid" value="'.
821: $forwarding.'" />';
822: $func=&mt('Replying to');
823:
824: $dissub=&mt('Reply').': '.$content{'subject'};
825: $dismsg='> '.$content{'message'};
826: $dismsg=~s/\r/\n/g;
827: $dismsg=~s/\f/\n/g;
828: $dismsg=~s/\n+/\n\> /g;
1.15 www 829: }
1.37 www 830: if ($ENV{'form.recdom'}) { $defdom=$ENV{'form.recdom'}; }
1.22 www 831: $r->print(
1.31 matthew 832: '<form action="/adm/email" name="compemail" method="post"'.
833: ' enctype="multipart/form-data">'."\n".
1.92 www 834: '<input type="hidden" name="sendmail" value="on" />'."\n".
1.31 matthew 835: '<table>');
1.22 www 836: unless (($broadcast eq 'group') || ($broadcast eq 'upload')) {
1.92 www 837: if ($replying) {
838: $r->print('<tr><td colspan="2">'.&mt('Replying to').' '.
839: &Apache::loncommon::aboutmewrapper(
840: &Apache::loncommon::plainname($content{'sendername'},$content{'senderdomain'}),$content{'sendername'},$content{'senderdomain'}).' ('.
841: $content{'sendername'}.'@'.
842: $content{'senderdomain'}.')'.
843: '<input type="hidden" name="recuname" value="'.$content{'sendername'}.'" />'.
844: '<input type="hidden" name="recdomain" value="'.$content{'senderdomain'}.'" />'.
845: '</td></tr>');
846: } else {
847: my $domform = &Apache::loncommon::select_dom_form($defdom,'recdomain');
848: my $selectlink=&Apache::loncommon::selectstudent_link
1.46 www 849: ('compemail','recuname','recdomain');
1.92 www 850: $r->print(<<"ENDREC");
1.69 www 851: <tr><td>$lt{'us'}:</td><td><input type="text" size="12" name="recuname" value="$ENV{'form.recname'}"></td><td rowspan="2">$selectlink</td></tr>
852: <tr><td>$lt{'do'}:</td>
1.31 matthew 853: <td>$domform</td></tr>
1.17 www 854: ENDREC
1.92 www 855: }
1.17 www 856: }
1.55 bowersj2 857: my $latexHelp = Apache::loncommon::helpLatexCheatsheet();
1.31 matthew 858: if ($broadcast ne 'upload') {
1.22 www 859: $r->print(<<"ENDCOMP");
1.69 www 860: <tr><td>$lt{'ad'}<br /><tt>username\@domain,username\@domain, ...
1.20 www 861: </tt></td><td>
1.91 www 862: <input type="text" size="50" name="additionalrec" /></td></tr>
863: <tr><td>$lt{'sb'}:</td><td><input type="text" size="50" name="subject" value="$dissub" />
1.15 www 864: </td></tr></table>
1.55 bowersj2 865: $latexHelp
1.92 www 866: <textarea name="message" cols="80" rows="15" wrap="hard">$dismsg
1.69 www 867: </textarea></p><br />
1.15 www 868: $dispcrit
1.69 www 869: <input type="submit" name="send" value="$func $lt{'ma'}" />
870: <input type="submit" name="cancel" value="$lt{'ca'}" />
1.15 www 871: ENDCOMP
1.31 matthew 872: } else { # $broadcast is 'upload'
1.22 www 873: $r->print(<<ENDUPLOAD);
1.91 www 874: <input type="hidden" name="sendmode" value="upload" />
1.86 www 875: <input type="hidden" name="send" value="on" />
1.22 www 876: <h3>Generate messages from a file</h3>
1.31 matthew 877: <p>
1.91 www 878: Subject: <input type="text" size="50" name="subject" />
1.31 matthew 879: </p>
880: <p>General message text<br />
1.91 www 881: <textarea name="message" cols="60" rows="10" wrap="hard">$dismsg
1.31 matthew 882: </textarea></p>
883: <p>
884: The file format for the uploaded portion of the message is:
1.22 www 885: <pre>
886: username1\@domain1: text
887: username2\@domain2: text
1.31 matthew 888: username3\@domain1: text
1.22 www 889: </pre>
1.31 matthew 890: </p>
891: <p>
1.22 www 892: The messages will be assembled from all lines with the respective
1.31 matthew 893: <tt>username\@domain</tt>, and appended to the general message text.</p>
894: <p>
1.91 www 895: <input type="file" name="upfile" size="40" /></p><p>
1.22 www 896: $dispcrit
1.92 www 897: <input type="submit" value="Upload and Send" /></p>
1.22 www 898: ENDUPLOAD
899: }
1.17 www 900: if ($broadcast eq 'group') {
901: &discourse;
902: }
903: $r->print('</form>');
1.15 www 904: }
905:
1.45 www 906: # ---------------------------------------------------- Display all face to face
907:
908: sub disfacetoface {
909: my ($r,$user,$domain)=@_;
910: unless ($ENV{'request.course.id'}) { return; }
911: unless (&Apache::lonnet::allowed('srm',$ENV{'request.course.id'})) {
912: return;
913: }
914: my %records=&Apache::lonnet::dump('nohist_email',
915: $ENV{'course.'.$ENV{'request.course.id'}.'.domain'},
916: $ENV{'course.'.$ENV{'request.course.id'}.'.num'},
917: '%255b'.$user.'%253a'.$domain.'%255d');
918: my $result='';
919: foreach (sort keys %records) {
920: my %content=&unpackagemsg($records{$_});
921: next if ($content{'senderdomain'} eq '');
922: $content{'message'}=~s/\n/\<br\>/g;
923: if ($content{'subject'}=~/^Record/) {
1.69 www 924: $result.='<h3>'.&mt('Record').'</h3>';
1.45 www 925: } else {
1.69 www 926: $result.='<h3>'.&mt('Sent Message').'</h3>';
1.45 www 927: %content=&unpackagemsg($content{'message'});
928: $content{'message'}=
1.92 www 929: '<b>'.&mt('Subject').': '.$content{'subject'}.'</b><br />'.
1.45 www 930: $content{'message'};
931: }
1.69 www 932: $result.=&mt('By').': <b>'.
1.45 www 933: &Apache::loncommon::aboutmewrapper(
934: &Apache::loncommon::plainname($content{'sendername'},$content{'senderdomain'}),$content{'sendername'},$content{'senderdomain'}).'</b> ('.
935: $content{'sendername'}.'@'.
936: $content{'senderdomain'}.') '.$content{'time'}.
1.92 www 937: '<br /><blockquote>'.
1.45 www 938: &Apache::lontexconvert::msgtexconverted($content{'message'}).
939: '</blockquote>';
940: }
941: # Check to see if there were any messages.
942: if ($result eq '') {
1.92 www 943: $r->print("<p><b>".&mt("No notes, face-to-face discussion records, or critical messages in this course.")."</b></p>");
1.45 www 944: } else {
945: $r->print($result);
946: }
947: }
948:
1.44 www 949: # ---------------------------------------------------------------- Face to face
950:
951: sub facetoface {
952: my ($r,$stage)=@_;
953: unless (&Apache::lonnet::allowed('srm',$ENV{'request.course.id'})) {
954: return;
955: }
1.89 www 956: &printheader($r,
957: '/adm/email?recordftf=query',
958: "User Notes, Face-to-Face, Critical Messages");
1.46 www 959: # from query string
1.88 www 960:
1.46 www 961: if ($ENV{'form.recname'}) { $ENV{'form.recuname'}=$ENV{'form.recname'}; }
962: if ($ENV{'form.recdom'}) { $ENV{'form.recdomain'}=$ENV{'form.recdom'}; }
963:
1.44 www 964: my $defdom=$ENV{'user.domain'};
1.46 www 965: # already filled in
1.44 www 966: if ($ENV{'form.recdomain'}) { $defdom=$ENV{'form.recdomain'}; }
1.46 www 967: # generate output
1.44 www 968: my $domform = &Apache::loncommon::select_dom_form($defdom,'recdomain');
1.46 www 969: my $stdbrws = &Apache::loncommon::selectstudent_link
970: ('stdselect','recuname','recdomain');
1.88 www 971: my %lt=&Apache::lonlocal::texthash('user' => 'Username',
972: 'dom' => 'Domain',
973: 'head' => 'User Notes, Records of Face-To-Face Discussions, and Critical Messages in Course',
974: 'subm' => 'Retrieve discussion and message records',
975: 'newr' => 'New Record (record is visible to course faculty and staff)',
976: 'post' => 'Post this Record');
1.44 www 977: $r->print(<<"ENDTREC");
1.88 www 978: <h3>$lt{'head'}</h3>
1.46 www 979: <form method="post" action="/adm/email" name="stdselect">
1.44 www 980: <input type="hidden" name="recordftf" value="retrieve" />
981: <table>
1.88 www 982: <tr><td>$lt{'user'}:</td><td><input type="text" size="12" name="recuname" value="$ENV{'form.recuname'}" /></td>
1.44 www 983: <td rowspan="2">
1.46 www 984: $stdbrws
1.88 www 985: <input type="submit" value="$lt{'subm'}" /></td>
1.44 www 986: </tr>
1.88 www 987: <tr><td>$lt{'dom'}:</td>
1.44 www 988: <td>$domform</td></tr>
989: </table>
990: </form>
991: ENDTREC
992: if (($stage ne 'query') &&
993: ($ENV{'form.recdomain'}) && ($ENV{'form.recuname'})) {
994: chomp($ENV{'form.newrecord'});
995: if ($ENV{'form.newrecord'}) {
1.45 www 996: &user_normal_msg_raw(
997: $ENV{'course.'.$ENV{'request.course.id'}.'.num'},
998: $ENV{'course.'.$ENV{'request.course.id'}.'.domain'},
1.88 www 999: &mt('Record').
1000: ' ['.$ENV{'form.recuname'}.':'.$ENV{'form.recdomain'}.']',
1.45 www 1001: $ENV{'form.newrecord'});
1.44 www 1002: }
1.46 www 1003: $r->print('<h3>'.&Apache::loncommon::plainname($ENV{'form.recuname'},
1004: $ENV{'form.recdomain'}).'</h3>');
1.45 www 1005: &disfacetoface($r,$ENV{'form.recuname'},$ENV{'form.recdomain'});
1.44 www 1006: $r->print(<<ENDRHEAD);
1007: <form method="post" action="/adm/email">
1008: <input name="recdomain" value="$ENV{'form.recdomain'}" type="hidden" />
1009: <input name="recuname" value="$ENV{'form.recuname'}" type="hidden" />
1010: ENDRHEAD
1011: $r->print(<<ENDBFORM);
1.88 www 1012: <hr />$lt{'newr'}<br />
1.44 www 1013: <textarea name="newrecord" cols="80" rows="10" wrap="hard"></textarea>
1.45 www 1014: <br />
1015: <input type="hidden" name="recordftf" value="post" />
1.88 www 1016: <input type="submit" value="$lt{'post'}" />
1.44 www 1017: </form>
1018: ENDBFORM
1019: }
1020: }
1.91 www 1021:
1.90 www 1022: # ----------------------------------------------------------- Display a message
1023:
1024: sub displaymessage {
1025: my ($r,$msgid)=@_;
1026: &statuschange($msgid,'read');
1027: my %message=&Apache::lonnet::get('nohist_email',[$msgid]);
1028: my %content=&unpackagemsg($message{$msgid});
1029: # info to generate "next" and "previous" buttons
1030: my @messages=&sortedmessages();
1031: my $counter=0;
1032: $r->print('<pre>');
1033: my $escmsgid=&Apache::lonnet::escape($msgid);
1034: foreach (@messages) {
1035: if ($_->[5] eq $escmsgid){
1036: last;
1037: }
1038: $counter++;
1039: }
1040: $r->print('</pre>');
1041: my $number_of_messages = scalar(@messages); #subtract 1 for last index
1042: # start output
1.92 www 1043: &printheader($r,'/adm/email?display='.&Apache::lonnet::escape($msgid),'Display a Message','',$content{'baseurl'});
1.90 www 1044: my %courseinfo=&Apache::lonnet::coursedescription($content{'courseid'});
1045: # Functions
1046: $r->print('<table border="2" width="100%"><tr bgcolor="#FFFFAA"><td>'.&mt('Functions').':</td>'.
1047: '<td><a href="/adm/email?replyto='.&Apache::lonnet::escape($msgid).$sqs.
1048: '"><b>'.&mt('Reply').'</b></a></td>'.
1049: '<td><a href="/adm/email?forward='.&Apache::lonnet::escape($msgid).$sqs.
1050: '"><b>'.&mt('Forward').'</b></a></td>'.
1051: '<td><a href="/adm/email?markunread='.&Apache::lonnet::escape($msgid).$sqs.
1052: '"><b>'.&mt('Mark Unread').'</b></a></td>'.
1053: '<td><a href="/adm/email?markdel='.&Apache::lonnet::escape($msgid).$sqs.
1054: '"><b>Delete</b></a></td>'.
1055: '<td><a href="/adm/email?sortedby='.$ENV{'form.sortedby'}.
1056: '"><b>'.&mt('Display all Messages').'</b></a></td>');
1057: if ($counter > 0){
1058: $r->print('<td><a href="/adm/email?display='.$messages[$counter-1]->[5].$sqs.
1059: '"><b>'.&mt('Previous').'</b></a></td>');
1060: }
1061: if ($counter < $number_of_messages - 1){
1062: $r->print('<td><a href="/adm/email?display='.$messages[$counter+1]->[5].$sqs.
1063: '"><b>'.&mt('Next').'</b></a></td>');
1064: }
1065: $r->print('</tr></table>');
1066: $r->print('<br /><b>'.&mt('Subject').':</b> '.$content{'subject'}.
1067: '<br /><b>'.&mt('From').':</b> '.
1068: &Apache::loncommon::aboutmewrapper(
1069: &Apache::loncommon::plainname($content{'sendername'},$content{'senderdomain'}),
1070: $content{'sendername'},$content{'senderdomain'}).' ('.
1071: $content{'sendername'}.' at '.
1072: $content{'senderdomain'}.') '.
1073: ($content{'courseid'}?'<br /><b>'.&mt('Course').':</b> '.$courseinfo{'description'}.
1074: ($content{'coursesec'}?' ('.&mt('Group/Section').': '.$content{'coursesec'}.')':''):'').
1075: '<br /><b>'.&mt('Time').':</b> '.$content{'time'}.
1076: '<p><pre>'.
1077: &Apache::lontexconvert::msgtexconverted($content{'message'},1).
1078: '</pre><hr />'.$content{'citation'}.'</p>');
1079: return;
1080: }
1.44 www 1081:
1.88 www 1082: # ================================================================== The Header
1083:
1084: sub header {
1.90 www 1085: my ($r,$title,$baseurl)=@_;
1.88 www 1086: $r->print('<html><head><title>Communication and Messages</title>');
1087: if ($baseurl) {
1088: $r->print("<base href=\"http://$ENV{'SERVER_NAME'}/$baseurl\" />");
1089: }
1090: $r->print(&Apache::loncommon::studentbrowser_javascript().'</head>'.
1091: &Apache::loncommon::bodytag('Communication and Messages'));
1092: $r->print(&Apache::lonhtmlcommon::breadcrumbs
1.90 www 1093: (undef,($title?$title:'Communication and Messages')));
1.88 www 1094:
1095: }
1096:
1.90 www 1097: # ---------------------------------------------------------------- Print header
1098:
1099: sub printheader {
1100: my ($r,$url,$desc,$title,$baseurl)=@_;
1101: &Apache::lonhtmlcommon::add_breadcrumb
1102: ({href=>$url,
1103: text=>$desc});
1104: &header($r,$title,$baseurl);
1105: }
1106:
1107:
1.13 www 1108: # ===================================================================== Handler
1109:
1.5 www 1110: sub handler {
1111: my $r=shift;
1112:
1113: # ----------------------------------------------------------- Set document type
1.87 www 1114:
1115: &Apache::loncommon::content_type($r,'text/html');
1116: $r->send_http_header;
1117:
1118: return OK if $r->header_only;
1119:
1.6 www 1120: # --------------------------- Get query string for limited number of parameters
1.32 matthew 1121: &Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'},
1122: ['display','replyto','forward','markread','markdel','markunread',
1.44 www 1123: 'sendreply','compose','sendmail','critical','recname','recdom',
1.62 www 1124: 'recordftf','sortedby']);
1.65 www 1125: $sqs='&sortedby='.$ENV{'form.sortedby'};
1.40 www 1126: # ------------------------------------------------------ They checked for email
1.87 www 1127: &Apache::lonnet::put('email_status',{'recnewemail'=>0});
1.88 www 1128:
1129: # ----------------------------------------------------------------- Breadcrumbs
1130:
1131: &Apache::lonhtmlcommon::clear_breadcrumbs();
1132: &Apache::lonhtmlcommon::add_breadcrumb
1133: ({href=>"/adm/communicate",
1134: text=>"Communication/Messages",
1135: faq=>12,bug=>'Communication Tools',});
1136:
1.5 www 1137: # --------------------------------------------------------------- Render Output
1.88 www 1138:
1.87 www 1139: if ($ENV{'form.display'}) {
1.90 www 1140: &displaymessage($r,$ENV{'form.display'});
1.87 www 1141: } elsif ($ENV{'form.replyto'}) {
1.92 www 1142: &compout($r,'',$ENV{'form.replyto'});
1.87 www 1143: } elsif ($ENV{'form.confirm'}) {
1.92 www 1144: &printheader($r,'','Confirmed Receipt');
1.87 www 1145: foreach (keys %ENV) {
1146: if ($_=~/^form\.rec\_(.*)$/) {
1.92 www 1147: $r->print('<b>'.&mt('Confirming Receipt').':</b> '.
1.87 www 1148: &user_crit_received($1).'<br>');
1149: }
1150: if ($_=~/^form\.reprec\_(.*)$/) {
1151: my $msgid=$1;
1.92 www 1152: $r->print('<b>'.&mt('Confirming Receipt').':</b> '.
1.87 www 1153: &user_crit_received($msgid).'<br>');
1.93 ! www 1154: &compout($r,'',$msgid);
1.87 www 1155: }
1156: }
1157: &discrit($r);
1158: } elsif ($ENV{'form.critical'}) {
1.92 www 1159: &printheader($r,'','Displaying Critical Messages');
1.87 www 1160: &discrit($r);
1161: } elsif ($ENV{'form.forward'}) {
1162: &compout($r,$ENV{'form.forward'});
1163: } elsif ($ENV{'form.markdel'}) {
1.92 www 1164: &printheader($r,'','Deleted Message');
1.87 www 1165: &statuschange($ENV{'form.markdel'},'deleted');
1166: &disall($r);
1167: } elsif ($ENV{'form.markeddel'}) {
1168: my $total=0;
1169: foreach (keys %ENV) {
1170: if ($_=~/^form\.delmark_(.*)$/) {
1171: &statuschange(&Apache::lonnet::unescape($1),'deleted');
1172: $total++;
1173: }
1174: }
1.92 www 1175: &printheader($r,'','Deleted Messages');
1.87 www 1176: $r->print('Deleted '.$total.' message(s)<p>');
1177: &disall($r);
1178: } elsif ($ENV{'form.markunread'}) {
1.92 www 1179: &printheader($r,'','Marked Message as Unread');
1.87 www 1180: &statuschange($ENV{'form.markunread'},'new');
1181: &disall($r);
1182: } elsif ($ENV{'form.compose'}) {
1.92 www 1183: &compout($r,'','',$ENV{'form.compose'});
1.87 www 1184: } elsif ($ENV{'form.recordftf'}) {
1185: &facetoface($r,$ENV{'form.recordftf'});
1186: } elsif ($ENV{'form.sendmail'}) {
1187: my $sendstatus='';
1188: if ($ENV{'form.send'}) {
1.92 www 1189: &printheader($r,'','Messages being sent.');
1190: $r->rflush();
1.87 www 1191: my %content=();
1192: undef %content;
1193: if ($ENV{'form.forwid'}) {
1194: my $msgid=$ENV{'form.forwid'};
1195: my %message=&Apache::lonnet::get('nohist_email',[$msgid]);
1196: %content=&unpackagemsg($message{$msgid},1);
1197: &statuschange($msgid,'forwarded');
1198: $ENV{'form.message'}.="\n\n-- Forwarded message --\n\n".
1199: $content{'message'};
1200: }
1201: my %toaddr=();
1202: undef %toaddr;
1203: if ($ENV{'form.sendmode'} eq 'group') {
1204: foreach (keys %ENV) {
1205: if ($_=~/^form\.send\_to\_\&\&\&[^\&]*\&\&\&\_(.+)$/) {
1206: $toaddr{$1}='';
1207: }
1208: }
1209: } elsif ($ENV{'form.sendmode'} eq 'upload') {
1210: foreach (split(/[\n\r\f]+/,$ENV{'form.upfile'})) {
1211: my ($rec,$txt)=split(/\s*\:\s*/,$_);
1212: if ($txt) {
1213: $rec=~s/\@/\:/;
1214: $toaddr{$rec}.=$txt."\n";
1215: }
1216: }
1217: } else {
1218: $toaddr{$ENV{'form.recuname'}.':'.$ENV{'form.recdomain'}}='';
1219: }
1220: if ($ENV{'form.additionalrec'}) {
1221: foreach (split(/\,/,$ENV{'form.additionalrec'})) {
1222: my ($auname,$audom)=split(/\@/,$_);
1223: $toaddr{$auname.':'.$audom}='';
1224: }
1225: }
1.92 www 1226:
1.87 www 1227: foreach (keys %toaddr) {
1228: my ($recuname,$recdomain)=split(/\:/,$_);
1229: my $msgtxt=&Apache::lonfeedback::clear_out_html($ENV{'form.message'});
1.92 www 1230: if ($toaddr{$_}) { $msgtxt.='<hr />'.$toaddr{$_}; }
1231: my $thismsg;
1.87 www 1232: if ((($ENV{'form.critmsg'}) || ($ENV{'form.sendbck'})) &&
1233: (&Apache::lonnet::allowed('srm',$ENV{'request.course.id'}))) {
1.92 www 1234: $r->print(&mt('Sending critical message').' '.$recuname.'@'.$recdomain.': ');
1235: $thismsg=&user_crit_msg($recuname,$recdomain,
1.87 www 1236: &Apache::lonfeedback::clear_out_html($ENV{'form.subject'}),
1237: $msgtxt,
1238: $ENV{'form.sendbck'});
1239: } else {
1.92 www 1240: $r->print(&mt('Sending').' '.$recuname.'@'.$recdomain.': ');
1241: $thismsg=&user_normal_msg($recuname,$recdomain,
1.87 www 1242: &Apache::lonfeedback::clear_out_html($ENV{'form.subject'}),
1243: $msgtxt,
1244: $content{'citation'});
1245: }
1.92 www 1246: $r->print($thismsg.'<br />');
1247: $sendstatus.=' '.$thismsg;
1.87 www 1248: }
1249: }
1250: if ($sendstatus=~/^(\s*(?:ok|con_delayed)\s*)*$/) {
1251: $r->print('<br /><font color="green">'.&mt('Completed.').'</font>');
1252: if ($ENV{'form.displayedcrit'}) {
1253: &discrit($r);
1254: } else {
1255: &disall($r);
1256: }
1257: } else {
1258: $r->print(
1259: '<h2><font color="red">'.&mt('Could not deliver message').'</font></h2>'.
1260: &mt('Please use the browser "Back" button and correct the recipient addresses')
1261: );
1262: }
1263: } else {
1.92 www 1264: &printheader($r,'','Display All Messages');
1.87 www 1265: &disall($r);
1266: }
1267: $r->print('</body></html>');
1268: return OK;
1.5 www 1269: }
1.2 www 1270: # ================================================= Main program, reset counter
1271:
1.27 www 1272: BEGIN {
1.2 www 1273: $msgcount=0;
1.1 www 1274: }
1.58 bowersj2 1275:
1276: =pod
1277:
1278: =back
1279:
1.59 bowersj2 1280: =cut
1281:
1282: 1;
1.1 www 1283:
1284: __END__
1285:
1286:
1287:
1288:
1289:
1290:
1291:
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>