5 /******************************************************************************************************/
6 /* setup JSAN and some initial libraries */
8 netscape.security.PrivilegeManager.enablePrivilege("UniversalXPConnect");
9 if (typeof JSAN == 'undefined') { throw( "The JSAN library object is missing."); }
10 JSAN.errorLevel = "die"; // none, warn, or die
11 JSAN.addRepository('/xul/server/');
12 JSAN.use('util.error'); g.error = new util.error();
13 g.error.sdump('D_TRACE','my_init() for cat/copy_editor.xul');
15 JSAN.use('util.functional');
16 JSAN.use('OpenILS.data'); g.data = new OpenILS.data(); g.data.init({'via':'stash'});
17 JSAN.use('util.network'); g.network = new util.network();
21 g.session = g.cgi.param('session') || g.cgi.param('ses');
22 g.docid = g.cgi.param('docid');
23 g.handle_update = g.cgi.param('handle_update');
25 /******************************************************************************************************/
26 /* Get the copy ids from various sources and flesh them */
29 if (g.cgi.param('copy_ids')) copy_ids = JSON2js( g.cgi.param('copy_ids') );
30 if (!copy_ids) copy_ids = [];
31 if (window.xulG && window.xulG.copy_ids) copy_ids = copy_ids.concat( window.xulG.copy_ids );
33 if (copy_ids.length > 0) g.copies = g.network.request(
34 api.FM_ACP_FLESHED_BATCH_RETRIEVE.app,
35 api.FM_ACP_FLESHED_BATCH_RETRIEVE.method,
39 /******************************************************************************************************/
40 /* And other fleshed copies if any */
42 if (!g.copies) g.copies = [];
43 if (window.xulG && window.xulG.copies) g.copies = g.copies.concat( window.xulG.copies );
44 if (g.cgi.param('copies')) g.copies = g.copies.concat( JSON2js( g.cgi.param('copies') ) );
46 /******************************************************************************************************/
47 /* We try to retrieve callnumbers for existing copies, but for new copies, we rely on this */
49 if (window.xulG && window.xulG.callnumbers) g.callnumbers = window.xulG.callnumbers;
50 if (g.cgi.param('callnumbers')) g.callnumbers = JSON2js( g.cgi.param('callnumbers') );
52 /******************************************************************************************************/
53 /* Is the interface an editor or a viewer? */
55 if (g.cgi.param('edit') == '1') {
57 document.getElementById('caption').setAttribute('label','Copy Editor');
58 document.getElementById('nav').setAttribute('hidden','false');
61 if (g.cgi.param('single_edit') == '1') {
63 document.getElementById('caption').setAttribute('label','Copy Editor');
64 document.getElementById('nav').setAttribute('hidden','false');
67 /******************************************************************************************************/
68 /* Show the Record Details? */
71 document.getElementById('brief_display').setAttribute(
73 urls.XUL_BIB_BRIEF + '?docid=' + g.docid
76 document.getElementById('brief_display').setAttribute('hidden','true');
79 /******************************************************************************************************/
80 /* Add stat cats to the right_pane_field_names */
82 var stat_cat_seen = {};
84 function add_stat_cat(sc) {
86 if (typeof g.data.hash.asc == 'undefined') { g.data.hash.asc = {}; g.data.stash('hash'); }
90 if (typeof sc == 'object') {
95 if (typeof stat_cat_seen[sc_id] != 'undefined') { return; }
97 stat_cat_seen[ sc_id ] = 1;
99 if (typeof sc != 'object') {
101 sc = g.network.simple_request(
102 'FM_ASC_BATCH_RETRIEVE',
103 [ g.session, [ sc_id ] ]
108 g.data.hash.asc[ sc.id() ] = sc; g.data.stash('hash');
110 var label_name = g.data.hash.aou[ sc.owner() ].shortname() + " : " + sc.name();
115 render: 'var l = util.functional.find_list( fm.stat_cat_entries(), function(e){ return e.stat_cat() == '
116 + sc.id() + '; } ); l ? l.value() : null;',
117 input: 'x = util.widgets.make_menulist( util.functional.map_list( g.data.hash.asc[' + sc.id()
118 + '].entries(), function(obj){ return [ obj.value(), obj.id() ]; } ).sort() ); '
119 + 'x.addEventListener("command",function(ev) { g.apply_stat_cat(' + sc.id()
120 + ', ev.target.value); } ,false);',
124 dump('temp_array = ' + js2JSON(temp_array) + '\n');
126 g.right_pane_field_names.push( temp_array );
129 /* The stat cats for the pertinent library */
130 for (var i = 0; i < g.data.list.my_asc.length; i++) {
131 add_stat_cat( g.data.list.my_asc[i] );
134 /* Other stat cats present on these copies */
135 for (var i = 0; i < g.copies.length; i++) {
136 var entries = g.copies[i].stat_cat_entries();
137 if (!entries) entries = [];
138 for (var j = 0; j < entries.length; j++) {
139 var sc_id = entries[j].stat_cat();
140 add_stat_cat( sc_id );
144 /******************************************************************************************************/
147 g.summarize( g.copies );
151 var err_msg = "!! This software has encountered an error. Please tell your friendly " +
152 "system administrator or software developer the following:\ncat/copy_editor.xul\n" + E + '\n';
153 try { g.error.sdump('D_ERROR',err_msg); } catch(E) { dump(err_msg); dump(js2JSON(E)); }
158 /******************************************************************************************************/
159 /* Apply a value to a specific field on all the copies being edited */
161 g.apply = function(field,value) {
162 g.error.sdump('D_TRACE','field = ' + field + ' value = ' + value + '\n');
163 for (var i = 0; i < g.copies.length; i++) {
164 var copy = g.copies[i];
166 copy[field]( value ); copy.ischanged('1');
173 /******************************************************************************************************/
174 /* Apply a stat cat entry to all the copies being edited */
176 g.apply_stat_cat = function(sc_id,entry_id) {
177 g.error.sdump('D_TRACE','sc_id = ' + sc_id + ' entry_id = ' + entry_id + '\n');
178 for (var i = 0; i < g.copies.length; i++) {
179 var copy = g.copies[i];
182 var temp = copy.stat_cat_entries();
183 if (!temp) temp = [];
184 temp = util.functional.filter_list(
187 return (obj.stat_cat() != sc_id);
191 util.functional.find_id_object_in_list(
192 g.data.hash.asc[sc_id].entries(),
196 copy.stat_cat_entries( temp );
205 /******************************************************************************************************/
206 /* These need data from the middle layer to render */
208 g.special_exception = {
209 'Call Number' : function(label,value) {
210 if (value>0) { /* an existing call number */
212 api.FM_ACN_RETRIEVE.app,
213 api.FM_ACN_RETRIEVE.method,
216 var cn = '??? id = ' + value;
218 cn = req.getResultObject().label();
220 g.error.sdump('D_ERROR','callnumber retrieve: ' + E);
222 label.setAttribute('value',cn);
225 } else { /* a yet to be created call number */
227 label.setAttribute('value',g.callnumbers[value]);
231 'Creator' : function(label,value) {
232 if (value == null || value == '' || value == 'null') return;
233 g.network.simple_request(
234 'FM_AU_RETRIEVE_VIA_ID',
235 [ g.session, value ],
237 var p = '??? id = ' + value;
239 p = req.getResultObject();
240 p = p.usrname() + ' : ' + p.family_name() + ', ' + p.first_given_name();
243 g.error.sdump('D_ERROR','patron retrieve: ' + E);
245 label.setAttribute('value',p);
249 'Last Editor' : function(label,value) {
250 if (value == null || value == '' || value == 'null') return;
251 g.network.simple_request(
252 'FM_AU_RETRIEVE_VIA_ID',
253 [ g.session, value ],
255 var p = '??? id = ' + value;
257 p = req.getResultObject();
258 p = p.usrname() + ' : ' + p.family_name() + ', ' + p.first_given_name();
261 g.error.sdump('D_ERROR','patron retrieve: ' + E);
263 label.setAttribute('value',p);
270 /******************************************************************************************************/
271 g.readonly_stat_cat_names = [];
272 g.editable_stat_cat_names = [];
274 /******************************************************************************************************/
275 /* These get show in the left panel */
277 g.left_pane_field_names = [
281 render: 'fm.barcode();',
287 render: 'fm.call_number();',
293 render: 'util.date.formatted_date( fm.create_date(), "%F");',
299 render: 'util.date.formatted_date( fm.edit_date(), "%F");',
305 /******************************************************************************************************/
306 /* These get shown in the right panel */
308 g.right_pane_field_names = [
312 render: 'fm.creator();',
318 render: 'fm.editor();',
324 render: 'fm.alert_message();',
325 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("alert_message",ev.target.value); }, false);',
331 render: 'fm.circ_as_type();',
332 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("circ_as_type",ev.target.value); }, false);',
336 "Circulation Library",
338 render: 'fm.circ_lib().shortname();',
339 input: 'x = util.widgets.make_menulist( util.functional.map_list( util.functional.filter_list(g.data.list.my_aou, function(obj) { return g.data.hash.aout[ obj.ou_type() ].can_have_vols(); }), function(obj) { return [ obj.shortname(), obj.id() ]; }).sort() ); x.addEventListener("command",function(ev) { g.apply("circ_lib",ev.target.value); }, false);',
343 "Circulation Modifier",
345 render: 'fm.circ_modifier();',
346 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("circ_modifier",ev.target.value); }, false);',
352 render: 'fm.circulate() ? "Yes" : "No";',
353 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("circulate",ev.target.value); }, false);',
359 render: 'fm.copy_number();',
360 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("copy_number",ev.target.value); }, false);',
366 render: 'fm.deposit() ? "Yes" : "No";',
367 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("deposit",ev.target.value); }, false);',
373 render: 'util.money.sanitize( fm.deposit_amount() );',
374 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("deposit_amount",ev.target.value); }, false);',
380 render: 'switch(fm.fine_level()){ case 1: "Low"; break; case 2: "Normal"; break; case 3: "High"; break; }',
381 input: 'x = util.widgets.make_menulist( [ [ "Low", "1" ], [ "Normal", "2" ], [ "High", "3" ] ] ); x.addEventListener("command",function(ev) { g.apply("fine_level",ev.target.value); }, false);',
387 render: 'fm.holdable() ? "Yes" : "No";',
388 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("holdable",ev.target.value); }, false);',
394 render: 'switch(fm.loan_duration()){ case 1: "Short"; break; case 2: "Normal"; break; case 3: "Long"; break; }',
395 input: 'x = util.widgets.make_menulist( [ [ "Short", "1" ], [ "Normal", "2" ], [ "Long", "3" ] ] ); x.addEventListener("command",function(ev) { g.apply("loan_duration",ev.target.value); }, false);',
402 render: 'fm.location().name();',
403 input: 'x = util.widgets.make_menulist( util.functional.map_list( g.data.list.acpl, function(obj) { return [ obj.name(), obj.id() ]; }).sort()); x.addEventListener("command",function(ev) { g.apply("location",ev.target.value); }, false);',
410 render: 'fm.opac_visible() ? "Yes" : "No";',
411 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("opac_visible",ev.target.value); }, false);',
417 render: 'util.money.sanitize( fm.price() );',
418 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("deposit_amount",ev.target.value); }, false);',
424 render: 'fm.ref() ? "Yes" : "No";',
425 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("ref",ev.target.value); }, false);',
431 render: 'fm.status().name();',
432 input: 'x = util.widgets.make_menulist( util.functional.map_list( g.data.list.ccs, function(obj) { return [ obj.name(), obj.id() ]; } ).sort() ); x.addEventListener("command",function(ev) { g.apply("status",ev.target.value); }, false);',
439 /******************************************************************************************************/
440 /* This loops through all our fieldnames and all the copies, tallying up counts for the different values */
442 g.summarize = function( copies ) {
443 /******************************************************************************************************/
446 JSAN.use('util.date'); JSAN.use('util.money');
448 g.field_names = g.left_pane_field_names;
449 g.field_names = g.field_names.concat( g.right_pane_field_names );
450 g.field_names = g.field_names.concat( g.editable_stat_cat_names );
451 g.field_names = g.field_names.concat( g.readonly_stat_cat_names );
453 /******************************************************************************************************/
454 /* Loop through the field names */
456 for (var i = 0; i < g.field_names.length; i++) {
458 var field_name = g.field_names[i][0];
459 var render = g.field_names[i][1].render;
460 g.summary[ field_name ] = {};
462 /******************************************************************************************************/
463 /* Loop through the copies */
465 for (var j = 0; j < copies.length; j++) {
468 var cmd = render || ('fm.' + field_name + '();');
471 /**********************************************************************************************/
472 /* Try to retrieve the value for this field for this copy */
477 g.error.sdump('D_ERROR','Attempted ' + cmd + '\n' + E + '\n');
479 if (typeof value == 'object' && value != null) {
480 alert('FIXME: field_name = ' + field_name + ' value = ' + js2JSON(value) + '\n');
483 /**********************************************************************************************/
484 /* Tally the count */
486 if (g.summary[ field_name ][ value ]) {
487 g.summary[ field_name ][ value ]++;
489 g.summary[ field_name ][ value ] = 1;
493 g.error.sdump('D_TRACE','summary = ' + js2JSON(g.summary) + '\n');
496 /******************************************************************************************************/
497 /* Display the summarized data and inputs for editing */
499 g.render = function() {
501 /******************************************************************************************************/
502 /* Library setup and clear any existing interface */
504 JSAN.use('util.widgets'); JSAN.use('util.date'); JSAN.use('util.money'); JSAN.use('util.functional');
506 var cns = document.getElementById('call_number_summary');
507 util.widgets.remove_children( cns );
508 var bcs = document.getElementById('barcode_summary');
509 util.widgets.remove_children( bcs );
510 var rp = document.getElementById('right_pane');
511 util.widgets.remove_children( rp );
513 /******************************************************************************************************/
514 /* Make the call number summary */
516 var grid = util.widgets.make_grid( [ { 'flex' : '1' } ] );
517 cns.appendChild(grid);
518 for (var i in g.summary['Call Number']) {
519 var cn_id = i; var count = g.summary['Call Number'][i];
520 var row = document.createElement('row'); grid.lastChild.appendChild(row);
521 var cn_label = document.createElement('label'); row.appendChild(cn_label);
522 g.special_exception['Call Number']( cn_label, cn_id );
523 var count_label = document.createElement('label'); row.appendChild(count_label);
524 var unit = count == 1 ? 'copy' : 'copies';
525 count_label.setAttribute('value',count + ' ' + unit);
528 /******************************************************************************************************/
529 /* List the copy barcodes */
531 for (var i in g.summary['Barcode']) {
533 var hbox = document.createElement('hbox'); bcs.appendChild(hbox);
534 var bc_label = document.createElement('label'); hbox.appendChild(bc_label);
535 bc_label.setAttribute('value',bc);
538 /******************************************************************************************************/
539 /* List the other non-editable fields in this pane */
541 var groupbox; var caption; var vbox; var grid; var rows;
542 for (var i = 0; i < g.left_pane_field_names.length; i++) {
544 var f = g.left_pane_field_names[i]; var fn = f[0];
545 if (fn == 'Call Number' || fn == 'Barcode') continue;
546 groupbox = document.createElement('groupbox'); bcs.parentNode.parentNode.appendChild(groupbox);
547 caption = document.createElement('caption'); groupbox.appendChild(caption);
548 caption.setAttribute('label',fn);
549 vbox = document.createElement('vbox'); groupbox.appendChild(vbox);
550 grid = util.widgets.make_grid( [ { 'flex' : 1 }, {}, {} ] ); vbox.appendChild(grid);
551 grid.setAttribute('flex','1');
552 rows = grid.lastChild;
555 /**************************************************************************************/
556 /* Loop through each value for the field */
558 for (var j in g.summary[fn]) {
559 var value = j; var count = g.summary[fn][j];
560 row = document.createElement('row'); rows.appendChild(row);
561 var label1 = document.createElement('label'); row.appendChild(label1);
562 if (g.special_exception[ fn ]) {
563 g.special_exception[ fn ]( label1, value );
565 label1.setAttribute('value',value);
567 var label2 = document.createElement('label'); row.appendChild(label2);
568 var unit = count == 1 ? 'copy' : 'copies';
569 label2.setAttribute('value',count + ' ' + unit);
571 var hbox = document.createElement('hbox');
572 vbox.appendChild(hbox);
574 g.error.sdump('D_ERROR','copy editor: ' + E + '\n');
578 /******************************************************************************************************/
579 /* Prepare the right panel, which is different for 1-copy view and multi-copy view */
583 /******************************************************************************************************/
584 /* For a less dangerous batch edit, choose one field here */
586 var gb = document.createElement('groupbox'); rp.appendChild(gb);
587 var c = document.createElement('caption'); gb.appendChild(c);
588 c.setAttribute('label','Choose a field to edit');
589 JSAN.use('util.widgets'); JSAN.use('util.functional');
590 var ml = util.widgets.make_menulist(
591 util.functional.map_list(
592 g.right_pane_field_names,
593 function(o,i) { return [ o[0], i ]; }
600 g.render_input(gb, g.right_pane_field_names[ ev.target.value ][1].input);
608 if (g.copies.length == 1) {
610 /******************************************************************************************************/
611 /* 1-copy mode has a single groupbox and each field is a row on a grid */
613 var groupbox; var caption; var vbox; var grid; var rows;
614 groupbox = document.createElement('groupbox'); rp.appendChild(groupbox);
615 caption = document.createElement('caption'); groupbox.appendChild(caption);
616 caption.setAttribute('label','Fields');
617 vbox = document.createElement('vbox'); groupbox.appendChild(vbox);
618 grid = util.widgets.make_grid( [ {}, { 'flex' : 1 } ] ); vbox.appendChild(grid);
619 grid.setAttribute('flex','1');
620 rows = grid.lastChild;
622 /******************************************************************************************************/
623 /* Loop through the field names */
625 for (var i = 0; i < g.right_pane_field_names.length; i++) {
627 var f = g.right_pane_field_names[i]; var fn = f[0];
630 /**************************************************************************************/
631 /* Loop through each value for the field */
633 for (var j in g.summary[fn]) {
634 var value = j; var count = g.summary[fn][j];
635 row = document.createElement('row'); rows.appendChild(row);
636 var label0 = document.createElement('label'); row.appendChild(label0);
637 label0.setAttribute('value',fn);
638 label0.setAttribute('style','font-weight: bold');
639 var label1 = document.createElement('label'); row.appendChild(label1);
640 if (g.special_exception[ fn ]) {
641 g.special_exception[ fn ]( label1, value );
643 label1.setAttribute('value',value);
648 /**************************************************************************************/
649 /* Render the input widget */
651 var hbox = document.createElement('hbox');
652 hbox.setAttribute('id',fn);
653 row.setAttribute('style','border-bottom: dotted black thin');
654 row.appendChild(hbox);
655 if (f[1].input && g.edit) {
656 g.render_input(hbox,f[1].input);
660 g.error.sdump('D_ERROR','copy editor: ' + E + '\n');
666 /******************************************************************************************************/
667 /* multi-copy mode has a groupbox for each field */
669 var groupbox; var caption; var vbox; var grid; var rows;
671 /******************************************************************************************************/
672 /* Loop through the field names */
674 for (var i = 0; i < g.right_pane_field_names.length; i++) {
676 var f = g.right_pane_field_names[i]; var fn = f[0];
677 groupbox = document.createElement('groupbox'); rp.appendChild(groupbox);
678 caption = document.createElement('caption'); groupbox.appendChild(caption);
679 caption.setAttribute('label',fn);
680 vbox = document.createElement('vbox'); groupbox.appendChild(vbox);
681 grid = util.widgets.make_grid( [ { 'flex' : 1 }, {}, {} ] ); vbox.appendChild(grid);
682 grid.setAttribute('flex','1');
683 rows = grid.lastChild;
686 /**************************************************************************************/
687 /* Loop through each value for the field */
689 for (var j in g.summary[fn]) {
690 var value = j; var count = g.summary[fn][j];
691 row = document.createElement('row'); rows.appendChild(row);
692 var label1 = document.createElement('label'); row.appendChild(label1);
693 if (g.special_exception[ fn ]) {
694 g.special_exception[ fn ]( label1, value );
696 label1.setAttribute('value',value);
698 var label2 = document.createElement('label'); row.appendChild(label2);
699 var unit = count == 1 ? 'copy' : 'copies';
700 label2.setAttribute('value',count + ' ' + unit);
703 var hbox = document.createElement('hbox');
704 hbox.setAttribute('id',fn);
705 vbox.appendChild(hbox);
707 /**************************************************************************************/
708 /* Render the input widget */
710 if (f[1].input && g.edit) {
711 g.render_input(hbox,f[1].input);
714 g.error.sdump('D_ERROR','copy editor: ' + E + '\n');
720 /******************************************************************************************************/
721 /* This actually draws the change button and input widget for a given field */
722 g.render_input = function(node,input_cmd) {
724 var spacer = document.createElement('spacer'); node.appendChild(spacer);
725 spacer.setAttribute('flex','1');
726 var deck = document.createElement('deck'); node.appendChild(deck);
727 var btn = document.createElement('button'); deck.appendChild(btn);
728 deck.setAttribute('style','width: 200px; min-width: 200px;');
729 btn.setAttribute('label','Change');
730 btn.setAttribute('oncommand','this.parentNode.selectedIndex = 1;');
731 var x; eval( input_cmd );
732 if (x) deck.appendChild(x);
735 g.error.sdump('D_ERROR',E + '\n');
739 /******************************************************************************************************/
740 /* store the copies in the global xpcom stash */
742 g.stash_and_close = function() {
743 if (g.handle_update) {
745 var r = g.network.request(
746 api.FM_ACP_FLESHED_BATCH_UPDATE.app,
747 api.FM_ACP_FLESHED_BATCH_UPDATE.method,
748 [ g.session, g.copies ]
750 /* FIXME -- revisit the return value here */
752 alert('copy update error: ' + js2JSON(E));
755 g.data.temp = js2JSON( g.copies );
756 g.error.sdump('D_CAT','in modal window, g.data.temp = \n' + g.data.temp + '\n');
757 g.data.stash('temp');