1: # The LearningOnline Network with CAPA
2: # The LON-CAPA group sort handler
3: # Allows for sorting prior to import into RAT.
4: #
5: # $Id: groupsort.pm,v 1.35 2005/06/10 15:17:35 www Exp $
6: #
7: # Copyright Michigan State University Board of Trustees
8: #
9: # This file is part of the LearningOnline Network with CAPA (LON-CAPA).
10: #
11: # LON-CAPA is free software; you can redistribute it and/or modify
12: # it under the terms of the GNU General Public License as published by
13: # the Free Software Foundation; either version 2 of the License, or
14: # (at your option) any later version.
15: #
16: # LON-CAPA 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.
20: #
21: # You should have received a copy of the GNU General Public License
22: # along with LON-CAPA; if not, write to the Free Software
23: # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
24: #
25: # /home/httpd/html/adm/gpl.txt
26: #
27: # http://www.lon-capa.org/
28: #
29: ###
30:
31: package Apache::groupsort;
32:
33: use strict;
34:
35: use Apache::Constants qw(:common);
36: use GDBM_File;
37: use Apache::loncommon;
38: use Apache::lonlocal;
39: use Apache::lonnet;
40:
41: my $iconpath; # variable to be accessible to multiple subroutines
42: my %hash; # variable to tie to user specific database
43:
44:
45: sub readfromdb {
46: my ($r,$shash,$thash)=@_;
47:
48: my $diropendb;
49: # ------------------------------ which file do we open? Easy if explictly given
50: if ($env{'form.catalogmode'} eq 'groupsearch') {
51: $diropendb =
52: "/home/httpd/perl/tmp/$env{'user.domain'}_$env{'user.name'}_searchcat.db";
53: } elsif ($env{'form.catalogmode'} eq 'groupimport') {
54: $diropendb =
55: "/home/httpd/perl/tmp/$env{'user.domain'}_$env{'user.name'}_indexer.db";
56: } elsif ($env{'form.catalogmode'} eq 'groupsec') {
57: $diropendb =
58: "/home/httpd/perl/tmp/$env{'user.domain'}_$env{'user.name'}_groupsec.db";
59: } else {
60: # --------------------- not explicitly given, choose the one most recently used
61: my @dbfn;
62: my @dbst;
63:
64: $dbfn[0] =
65: "/home/httpd/perl/tmp/$env{'user.domain'}_$env{'user.name'}_searchcat.db";
66: $dbst[0]=-1;
67: if (-e $dbfn[0]) {
68: $dbst[0]=(stat($dbfn[0]))[9];
69: }
70: $dbfn[1] =
71: "/home/httpd/perl/tmp/$env{'user.domain'}_$env{'user.name'}_indexer.db";
72: $dbst[1]=-1;
73: if (-e $dbfn[1]) {
74: $dbst[1]=(stat($dbfn[1]))[9];
75: }
76: $dbfn[2] =
77: "/home/httpd/perl/tmp/$env{'user.domain'}_$env{'user.name'}_groupsec.db";
78: $dbst[2]=-1;
79: if (-e $dbfn[2]) {
80: $dbst[2]=(stat($dbfn[2]))[9];
81: }
82: # Expand here for more modes
83: # ....
84:
85: # Okay, find most recent existing
86:
87: my $newest=0;
88: $diropendb='';
89: for (my $i=0; $i<=$#dbfn; $i++) {
90: if ($dbst[$i]>$newest) {
91: $newest=$dbst[$i];
92: $diropendb=$dbfn[$i];
93: }
94: }
95:
96: }
97: # ----------------------------- diropendb is now the filename of the db to open
98: if (tie(%hash,'GDBM_File',$diropendb,&GDBM_WRCREAT(),0640)) {
99: my $acts = $env{'form.acts'};
100: my @Acts = split(/b/,$acts);
101: my %ahash;
102: my %achash;
103: my $ac = 0;
104: foreach (@Acts) {
105: my ($state,$ref) = split(/a/);
106: $ahash{$ref} = $state;
107: $achash{$ref} = $ac;
108: $ac++;
109: }
110: foreach (sort {$achash{$a} <=> $achash{$b}} (keys %ahash)) {
111: my $key = $_;
112: if ($ahash{$key} eq '1') {
113: # my $keyz=join("<br />",keys %hash);
114: # print "<br />$key<br />$keyz".$hash{'pre_'.$key.'_link'}."<br />\n";
115: $hash{'store_'.$hash{'pre_'.$key.'_link'}} =
116: $hash{'pre_'.$key.'_title'};
117: $hash{'storectr_'.$hash{'pre_'.$key.'_link'}} =
118: $hash{'storectr'}+0;
119: $hash{'storectr'}++;
120: }
121: if ($ahash{$key} eq '0') {
122: if ($hash{'store_'.$hash{'pre_'.$key.'_link'}}) {
123: delete $hash{'store_'.$hash{'pre_'.$key.'_link'}};
124: }
125: }
126: }
127: foreach (keys %hash) {
128: if ($_ =~ /^store_/) {
129: my $key = $_;
130: $key =~ s/^store_//;
131: $$shash{$key} = $hash{'storectr_'.$key};
132: if (&Apache::lonnet::gettitle($key) eq '') {
133: $$thash{$key} = $hash{'store_'.$key};
134: } else {
135: $$thash{$key} = &Apache::lonnet::gettitle($key);
136: }
137: }
138: }
139: if ($env{'form.oldval'}) {
140: my $newctr = 0;
141: my %chash;
142: foreach (sort {$$shash{$a} <=> $$shash{$b}} (keys %{$shash})) {
143: my $key = $_;
144: $newctr++;
145: $$shash{$key} = $newctr;
146: $hash{'storectr_'.$key} = $newctr;
147: $chash{$newctr} = $key;
148: }
149: my $oldval = $env{'form.oldval'};
150: my $newval = $env{'form.newval'};
151: if ($oldval != $newval) {
152: # when newval==0, then push down and delete
153: if ($newval!=0) {
154: $$shash{$chash{$oldval}} = $newval;
155: $hash{'storectr_'.$chash{$oldval}} = $newval;
156: } else {
157: $$shash{$chash{$oldval}} = $newctr;
158: $hash{'storectr_'.$chash{$oldval}} = $newctr;
159: }
160: if ($newval==0) { # push down
161: my $newval2=$newctr;
162: for my $idx ($oldval..($newval2-1)) {
163: $$shash{$chash{$idx+1}} = $idx;
164: $hash{'storectr_'.$chash{$idx+1}} = $idx;
165: }
166: delete $$shash{$chash{$oldval}};
167: delete $hash{'storectr_'.$chash{$oldval}};
168: delete $hash{'store_'.$chash{$oldval}};
169: } elsif ($oldval < $newval) { # push down
170: for my $idx ($oldval..($newval-1)) {
171: $$shash{$chash{$idx+1}} = $idx;
172: $hash{'storectr_'.$chash{$idx+1}} = $idx;
173: }
174: } elsif ($oldval > $newval) { # push up
175: for my $idx (reverse($newval..($oldval-1))) {
176: $$shash{$chash{$idx}} = $idx+1;
177: $hash{'storectr_'.$chash{$idx}} = $idx+1;
178: }
179: }
180: }
181: }
182: } else {
183: $r->print('Unable to tie hash to db file');
184: }
185: untie %hash;
186: return ($shash,$thash);
187: }
188:
189:
190:
191: sub cleanup {
192: if (tied(%hash)){
193: &Apache::lonnet::logthis('Cleanup groupsort: hash');
194: unless (untie(%hash)) {
195: &Apache::lonnet::logthis('Failed cleanup groupsort: hash');
196: }
197: }
198: }
199:
200: # -------------------------------------------------------------- Read from file
201:
202: sub readfromfile {
203: my ($r,$shash,$thash,$nhash)=@_;
204: my $cont=&Apache::lonnet::getfile
205: (&Apache::lonnet::filelocation('',$env{'form.readfile'}));
206: if ($cont==-1) {
207: $r->print('Unable to read file: '.
208: &Apache::lonnet::filelocation('',$env{'form.readfile'}));
209: } else {
210: my $parser = HTML::TokeParser->new(\$cont);
211: my $token;
212: my $n=1;
213: while ($token = $parser->get_token) {
214: if ($token->[0] eq 'S') {
215: if ($token->[1] eq 'resource') {
216: if ($env{'form.recover'}) {
217: if ($token->[2]->{'type'} ne 'zombie') { next; }
218: } else {
219: if ($token->[2]->{'type'} eq 'zombie') { next; }
220: }
221:
222: my $url=$token->[2]->{'src'};
223: my $name=$token->[2]->{'title'};
224: $name=~s/ \[\((\d+)\,(\w+)\,(\w+)\)\]$//;
225: if ($1) {
226: $$nhash{$url}='<br />'.&mt('Removed by ').
227: &Apache::loncommon::plainname($2,$3).', '.
228: &Apache::lonlocal::locallocaltime($1);
229: }
230: $$thash{$url}=$name;
231: $$shash{$url}=$n;
232: $n++;
233: }
234: }
235: }
236: }
237: return ($shash,$thash);
238: }
239:
240: # ---------------------------------------------------------------- Main Handler
241: sub handler {
242: my $r = shift;
243:
244: &Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'},
245: ['acts','catalogmode','mode','readfile','recover']);
246: # color scheme
247: my $fileclr = '#ffffe6';
248: my $titleclr = '#ddffff';
249:
250: &Apache::loncommon::content_type($r,'text/html');
251: $r->send_http_header;
252: return OK if $r->header_only;
253:
254: # finish_import looks different for graphical or "simple" RAT
255: my $finishimport='';
256: if ($env{'form.mode'} eq 'simple' || $env{'form.mode'} eq '') {
257: $finishimport=(<<ENDSMP);
258: function finish_import() {
259: opener.document.forms.simpleedit.importdetail.value='';
260: for (var num=0; num<document.forms.groupsort.fnum.value; num++) {
261: opener.document.forms.simpleedit.importdetail.value+='&'+
262: escape(eval("document.forms.groupsort.title"+num+".value"))+'='+
263: escape(eval("document.forms.groupsort.filelink"+num+".value"));
264: }
265: opener.document.forms.simpleedit.submit();
266: self.close();
267: }
268: ENDSMP
269: } else {
270: $finishimport=(<<ENDADV);
271: function finish_import() {
272: var linkflag=false;
273: for (var num=0; num<document.forms.groupsort.fnum.value; num++) {
274: insertRowInLastRow();
275: placeResourceInLastRow(
276: eval("document.forms.groupsort.title"+num+".value"),
277: eval("document.forms.groupsort.filelink"+num+".value"),
278: linkflag
279: );
280: linkflag=true;
281: }
282: opener.editmode=0;
283: opener.notclear=0;
284: opener.linkmode=0;
285: opener.draw();
286: self.close();
287: }
288: ENDADV
289: }
290:
291: # output start of web page
292: my $html=&Apache::lonxml::xmlbegin();
293: $r->print(<<END);
294: $html
295: <head>
296: <title>The LearningOnline Network With CAPA Group Sorter</title>
297: <script language='javascript'>
298: function insertRowInLastRow() {
299: opener.insertrow(opener.maxrow);
300: opener.addobj(opener.maxrow,'e&2');
301: }
302: function placeResourceInLastRow (title,url,linkflag) {
303: opener.mostrecent=opener.newresource(opener.maxrow,2,opener.escape(title),
304: opener.escape(url),'false','normal');
305: opener.save();
306: if (linkflag) {
307: opener.joinres(opener.linkmode,opener.mostrecent,0);
308: }
309: opener.linkmode=opener.mostrecent;
310: }
311: $finishimport
312: function selectchange(val) {
313: var newval=0+eval("document.forms.groupsort.alt"+val+".selectedIndex");
314: orderchange(val,newval);
315: }
316: function move(val,newval) {
317: orderchange(val,newval);
318: }
319: function orderchange(val,newval) {
320: document.forms.groupsort.oldval.value=val;
321: document.forms.groupsort.newval.value=newval;
322: document.forms.groupsort.submit();
323: }
324: </script>
325: </head>
326: END
327: # read pertinent machine configuration
328: my $domain = $r->dir_config('lonDefDomain');
329: $iconpath = $r->dir_config('lonIconsURL') . "/";
330:
331: my %shash; # sort order (key is resource location, value is sort order)
332: my %thash; # title (key is resource location, value is title)
333: my %nhash; # notes (key is resource location);
334:
335: if ($env{'form.readfile'}) {
336: &readfromfile($r,\%shash,\%thash,\%nhash);
337: } else {
338: &readfromdb($r,\%shash,\%thash);
339: }
340:
341: my $ctr = 0;
342: my $clen = scalar(keys %shash);
343: if ($clen > 1) {
344: my %lt=&Apache::lonlocal::texthash(
345: 'fin'=> 'Finalize order of resources',
346: 'gb' => 'Go Back',
347: 'ns' => 'New Search',
348: 'fi' => 'Finish Import',
349: 'ca' => 'Cancel',
350: 'co' => 'Change Order',
351: 'ti' => 'Title',
352: 'pa' => 'Path'
353: );
354: $r->print(&Apache::loncommon::bodytag('Sort Imported Resources'));
355: $r->print(<<END);
356: <b><font color="#888888">$lt{'fin'}</font></b>
357: <form method='post' action='/adm/groupsort' name='groupsort'
358: enctype='application/x-www-form-urlencoded'>
359: <input type="hidden" name="fnum" value="$clen" />
360: <input type="hidden" name="oldval" value="" />
361: <input type="hidden" name="newval" value="" />
362: <input type="hidden" name="mode" value="$env{'form.mode'}" />
363: END
364:
365: # --- Expand here if "GO BACK" button desired
366: if ($env{'form.catalogmode'} eq 'groupimport') {
367: my $resurl = &Apache::loncommon::lastresurl();
368: $r->print(<<END);
369: <input type="button" name="alter" value="$lt{'gb'}"
370: onClick="window.location='$resurl?catalogmode=groupimport'" />
371: END
372: }
373: if ($env{'form.catalogmode'} eq 'groupsearch') {
374: $r->print(<<END);
375: <input type="button" name="alter" value="$lt{'ns'}"
376: onClick="window.location='/adm/searchcat?catalogmode=groupsearch&cleargroupsort=1'" />
377: END
378: }
379: # ---
380:
381: $r->print(<<END);
382: <input type="button" name="alter" value="$lt{'fi'}"
383: onClick="finish_import()" />
384: <input type="button" name="alter" value="$lt{'ca'}" onClick="self.close()" />
385: END
386: $r->print("<table border='0'><tr><td bgcolor='#eeeeee'>");
387: $r->print("<table border=0><tr>\n");
388: $r->print("<td colspan='2' bgcolor='$titleclr'><b>$lt{'co'}</b></td>\n");
389: $r->print("<td colspan='2' bgcolor='$titleclr'><b>$lt{'ti'}</b></td>\n");
390: $r->print("<td bgcolor='$titleclr'><b>$lt{'pa'}</b></td></tr>\n");
391: } else {
392: $r->print(<<END);
393: <body>
394: <form method='post' action='/adm/groupsort' name='groupsort'
395: enctype='application/x-www-form-urlencoded'>
396: <input type="hidden" name="fnum" value="$clen" />
397: <input type="hidden" name="oldval" value="" />
398: <input type="hidden" name="newval" value="" />
399: <input type="hidden" name="mode" value="$env{'form.mode'}" />
400: END
401: }
402: foreach (sort {$shash{$a}<=>$shash{$b}} (keys %shash)) {
403: my $key=$_;
404: $ctr++;
405: my $iconname=&Apache::loncommon::icon($key);
406: if ($clen > 1) {
407: $r->print("<tr><td bgcolor='$fileclr'>");
408: $r->print(&movers($clen,$ctr));
409: }
410: $r->print(&hidden($ctr-1,$thash{$key},$key));
411: if ($clen > 1) {
412: $r->print("</td><td bgcolor='$fileclr'>");
413: $r->print(&select_box($clen,$ctr));
414: $r->print("</td><td bgcolor='$fileclr'>");
415: $r->print("<img src='$iconname' />");
416: $r->print("</td><td bgcolor='$fileclr'>");
417: $r->print("$thash{$key}</td><td bgcolor='$fileclr'>\n");
418: $r->print("$key</td></tr>\n");
419: }
420: }
421: if ($clen > 1) {
422: $r->print("</table></td></tr></table></form>");
423: } else {
424: $r->print(<<END);
425: <script type="text/javascript">
426: finish_import();
427: </script>
428: END
429: }
430: $r->print(<<END);
431: </body>
432: </html>
433: END
434:
435: return OK;
436: }
437:
438: # --------------------------------------- Hidden values (returns scalar string)
439: sub hidden {
440: my ($sel,$title,$filelink) = @_;
441: my $string = '<input type="hidden" name="title'.$sel.'" value="'.$title.
442: '" />';
443: $string .= '<input type="hidden" name="filelink'.$sel.'" value="'.
444: $filelink.'" />';
445: return $string;
446: }
447:
448: # --------------------------------------- Moving arrows (returns scalar string)
449: sub movers {
450: my ($total,$sel) = @_;
451: my $dsel = $sel-1;
452: my $usel = $sel+1;
453: $usel = 1 if $usel > $total;
454: $dsel = $total if $dsel < 1;
455: my $string;
456: $string = (<<END);
457: <table border='0' cellspacing='0' cellpadding='0'>
458: <tr><td><a href='javascript:move($sel,$dsel)'>
459: <img src="${iconpath}move_up.gif" alt='UP' border='0' /></a></td></tr>
460: <tr><td><a href='javascript:move($sel,$usel)'>
461: <img src="${iconpath}move_down.gif" alt='DOWN' border='0' /></a></td></tr>
462: </table>
463: END
464: return $string;
465: }
466:
467: # ------------------------------------------ Select box (returns scalar string)
468: sub select_box {
469: my ($total,$sel) = @_;
470: my $string;
471: $string = '<select name="alt'.$sel.'"';
472: $string .= " onChange='selectchange($sel)'>";
473: $string .= "<option name='o0' value='0'>remove</option>";
474: for my $cur (1..$total) {
475: $string .= "<option name='o$cur' value='$cur'";
476: if ($cur == $sel) {
477: $string .= "selected";
478: }
479: $string .= ">$cur</option>";
480: }
481: $string .= "</select>\n";
482: return $string;
483: }
484:
485: 1;
486:
487: __END__
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>