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.34 2005/06/10 02:19:51 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)=@_;
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: while ($token = $parser->get_token) {
213: if ($token->[0] eq 'S') {
214: if ($token->[1] eq 'resource') {
215: if ($env{'form.recover'}) {
216: if ($token->[2]->{'type'} ne 'zombie') { next; }
217: } else {
218: if ($token->[2]->{'type'} eq 'zombie') { next; }
219: }
220: my $name=$token->[2]->{'title'};
221: my $url=$token->[2]->{'src'};
222: $name=~s/ \[\((\d+)\,(\w+)\,(\w+)\)\]$//;
223: if ($1) {
224: $name.='<br />'.&mt('Removed by ').
225: &Apache::loncommon::plainname($2,$3).', '.
226: &Apache::lonlocal::locallocaltime($1);
227: }
228: $r->print('<br />'.$name);
229: }
230: }
231: }
232: }
233: return ($shash,$thash);
234: }
235:
236: # ---------------------------------------------------------------- Main Handler
237: sub handler {
238: my $r = shift;
239:
240: &Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'},
241: ['acts','catalogmode','mode','readfile','recover']);
242: # color scheme
243: my $fileclr = '#ffffe6';
244: my $titleclr = '#ddffff';
245:
246: &Apache::loncommon::content_type($r,'text/html');
247: $r->send_http_header;
248: return OK if $r->header_only;
249:
250: # finish_import looks different for graphical or "simple" RAT
251: my $finishimport='';
252: if ($env{'form.mode'} eq 'simple' || $env{'form.mode'} eq '') {
253: $finishimport=(<<ENDSMP);
254: function finish_import() {
255: opener.document.forms.simpleedit.importdetail.value='';
256: for (var num=0; num<document.forms.groupsort.fnum.value; num++) {
257: opener.document.forms.simpleedit.importdetail.value+='&'+
258: escape(eval("document.forms.groupsort.title"+num+".value"))+'='+
259: escape(eval("document.forms.groupsort.filelink"+num+".value"));
260: }
261: opener.document.forms.simpleedit.submit();
262: self.close();
263: }
264: ENDSMP
265: } else {
266: $finishimport=(<<ENDADV);
267: function finish_import() {
268: var linkflag=false;
269: for (var num=0; num<document.forms.groupsort.fnum.value; num++) {
270: insertRowInLastRow();
271: placeResourceInLastRow(
272: eval("document.forms.groupsort.title"+num+".value"),
273: eval("document.forms.groupsort.filelink"+num+".value"),
274: linkflag
275: );
276: linkflag=true;
277: }
278: opener.editmode=0;
279: opener.notclear=0;
280: opener.linkmode=0;
281: opener.draw();
282: self.close();
283: }
284: ENDADV
285: }
286:
287: # output start of web page
288: my $html=&Apache::lonxml::xmlbegin();
289: $r->print(<<END);
290: $html
291: <head>
292: <title>The LearningOnline Network With CAPA Group Sorter</title>
293: <script language='javascript'>
294: function insertRowInLastRow() {
295: opener.insertrow(opener.maxrow);
296: opener.addobj(opener.maxrow,'e&2');
297: }
298: function placeResourceInLastRow (title,url,linkflag) {
299: opener.mostrecent=opener.newresource(opener.maxrow,2,opener.escape(title),
300: opener.escape(url),'false','normal');
301: opener.save();
302: if (linkflag) {
303: opener.joinres(opener.linkmode,opener.mostrecent,0);
304: }
305: opener.linkmode=opener.mostrecent;
306: }
307: $finishimport
308: function selectchange(val) {
309: var newval=0+eval("document.forms.groupsort.alt"+val+".selectedIndex");
310: orderchange(val,newval);
311: }
312: function move(val,newval) {
313: orderchange(val,newval);
314: }
315: function orderchange(val,newval) {
316: document.forms.groupsort.oldval.value=val;
317: document.forms.groupsort.newval.value=newval;
318: document.forms.groupsort.submit();
319: }
320: </script>
321: </head>
322: END
323: # read pertinent machine configuration
324: my $domain = $r->dir_config('lonDefDomain');
325: $iconpath = $r->dir_config('lonIconsURL') . "/";
326:
327: my %shash; # sort order (key is resource location, value is sort order)
328: my %thash; # title (key is resource location, value is title)
329:
330: if ($env{'form.readfile'}) {
331: &readfromfile($r,\%shash,\%thash);
332: } else {
333: &readfromdb($r,\%shash,\%thash);
334: }
335:
336: my $ctr = 0;
337: my $clen = scalar(keys %shash);
338: if ($clen > 1) {
339: my %lt=&Apache::lonlocal::texthash(
340: 'fin'=> 'Finalize order of resources',
341: 'gb' => 'Go Back',
342: 'ns' => 'New Search',
343: 'fi' => 'Finish Import',
344: 'ca' => 'Cancel',
345: 'co' => 'Change Order',
346: 'ti' => 'Title',
347: 'pa' => 'Path'
348: );
349: $r->print(&Apache::loncommon::bodytag('Sort Imported Resources'));
350: $r->print(<<END);
351: <b><font color="#888888">$lt{'fin'}</font></b>
352: <form method='post' action='/adm/groupsort' name='groupsort'
353: enctype='application/x-www-form-urlencoded'>
354: <input type="hidden" name="fnum" value="$clen" />
355: <input type="hidden" name="oldval" value="" />
356: <input type="hidden" name="newval" value="" />
357: <input type="hidden" name="mode" value="$env{'form.mode'}" />
358: END
359:
360: # --- Expand here if "GO BACK" button desired
361: if ($env{'form.catalogmode'} eq 'groupimport') {
362: my $resurl = &Apache::loncommon::lastresurl();
363: $r->print(<<END);
364: <input type="button" name="alter" value="$lt{'gb'}"
365: onClick="window.location='$resurl?catalogmode=groupimport'" />
366: END
367: }
368: if ($env{'form.catalogmode'} eq 'groupsearch') {
369: $r->print(<<END);
370: <input type="button" name="alter" value="$lt{'ns'}"
371: onClick="window.location='/adm/searchcat?catalogmode=groupsearch&cleargroupsort=1'" />
372: END
373: }
374: # ---
375:
376: $r->print(<<END);
377: <input type="button" name="alter" value="$lt{'fi'}"
378: onClick="finish_import()" />
379: <input type="button" name="alter" value="$lt{'ca'}" onClick="self.close()" />
380: END
381: $r->print("<table border='0'><tr><td bgcolor='#eeeeee'>");
382: $r->print("<table border=0><tr>\n");
383: $r->print("<td colspan='2' bgcolor='$titleclr'><b>$lt{'co'}</b></td>\n");
384: $r->print("<td colspan='2' bgcolor='$titleclr'><b>$lt{'ti'}</b></td>\n");
385: $r->print("<td bgcolor='$titleclr'><b>$lt{'pa'}</b></td></tr>\n");
386: } else {
387: $r->print(<<END);
388: <body>
389: <form method='post' action='/adm/groupsort' name='groupsort'
390: enctype='application/x-www-form-urlencoded'>
391: <input type="hidden" name="fnum" value="$clen" />
392: <input type="hidden" name="oldval" value="" />
393: <input type="hidden" name="newval" value="" />
394: <input type="hidden" name="mode" value="$env{'form.mode'}" />
395: END
396: }
397: foreach (sort {$shash{$a}<=>$shash{$b}} (keys %shash)) {
398: my $key=$_;
399: $ctr++;
400: my $iconname=&Apache::loncommon::icon($key);
401: if ($clen > 1) {
402: $r->print("<tr><td bgcolor='$fileclr'>");
403: $r->print(&movers($clen,$ctr));
404: }
405: $r->print(&hidden($ctr-1,$thash{$key},$key));
406: if ($clen > 1) {
407: $r->print("</td><td bgcolor='$fileclr'>");
408: $r->print(&select_box($clen,$ctr));
409: $r->print("</td><td bgcolor='$fileclr'>");
410: $r->print("<img src='$iconname' />");
411: $r->print("</td><td bgcolor='$fileclr'>");
412: $r->print("$thash{$key}</td><td bgcolor='$fileclr'>\n");
413: $r->print("$key</td></tr>\n");
414: }
415: }
416: if ($clen > 1) {
417: $r->print("</table></td></tr></table></form>");
418: } else {
419: $r->print(<<END);
420: <script type="text/javascript">
421: finish_import();
422: </script>
423: END
424: }
425: $r->print(<<END);
426: </body>
427: </html>
428: END
429:
430: return OK;
431: }
432:
433: # --------------------------------------- Hidden values (returns scalar string)
434: sub hidden {
435: my ($sel,$title,$filelink) = @_;
436: my $string = '<input type="hidden" name="title'.$sel.'" value="'.$title.
437: '" />';
438: $string .= '<input type="hidden" name="filelink'.$sel.'" value="'.
439: $filelink.'" />';
440: return $string;
441: }
442:
443: # --------------------------------------- Moving arrows (returns scalar string)
444: sub movers {
445: my ($total,$sel) = @_;
446: my $dsel = $sel-1;
447: my $usel = $sel+1;
448: $usel = 1 if $usel > $total;
449: $dsel = $total if $dsel < 1;
450: my $string;
451: $string = (<<END);
452: <table border='0' cellspacing='0' cellpadding='0'>
453: <tr><td><a href='javascript:move($sel,$dsel)'>
454: <img src="${iconpath}move_up.gif" alt='UP' border='0' /></a></td></tr>
455: <tr><td><a href='javascript:move($sel,$usel)'>
456: <img src="${iconpath}move_down.gif" alt='DOWN' border='0' /></a></td></tr>
457: </table>
458: END
459: return $string;
460: }
461:
462: # ------------------------------------------ Select box (returns scalar string)
463: sub select_box {
464: my ($total,$sel) = @_;
465: my $string;
466: $string = '<select name="alt'.$sel.'"';
467: $string .= " onChange='selectchange($sel)'>";
468: $string .= "<option name='o0' value='0'>remove</option>";
469: for my $cur (1..$total) {
470: $string .= "<option name='o$cur' value='$cur'";
471: if ($cur == $sel) {
472: $string .= "selected";
473: }
474: $string .= ">$cur</option>";
475: }
476: $string .= "</select>\n";
477: return $string;
478: }
479:
480: 1;
481:
482: __END__
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>