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, single or multi copy? */
55 if (g.cgi.param('edit') == '1') {
57 document.getElementById('caption').setAttribute('label','Copy Editor');
58 document.getElementById('save').setAttribute('hidden','false');
61 if (g.cgi.param('single_edit') == '1') {
63 document.getElementById('caption').setAttribute('label','Copy Editor');
64 document.getElementById('save').setAttribute('hidden','false');
67 if (g.copies[0].id() < 0) {
68 document.getElementById('copy_notes').setAttribute('hidden','true');
70 if (g.copies.length != 1) {
71 document.getElementById('copy_notes').setAttribute('hidden','true');
74 /******************************************************************************************************/
75 /* Show the Record Details? */
78 document.getElementById('brief_display').setAttribute(
80 urls.XUL_BIB_BRIEF + '?docid=' + g.docid
83 document.getElementById('brief_display').setAttribute('hidden','true');
86 /******************************************************************************************************/
87 /* Add stat cats to the right_pane_field_names */
89 var stat_cat_seen = {};
91 function add_stat_cat(sc) {
93 if (typeof g.data.hash.asc == 'undefined') { g.data.hash.asc = {}; g.data.stash('hash'); }
97 if (typeof sc == 'object') {
102 if (typeof stat_cat_seen[sc_id] != 'undefined') { return; }
104 stat_cat_seen[ sc_id ] = 1;
106 if (typeof sc != 'object') {
108 sc = g.network.simple_request(
109 'FM_ASC_BATCH_RETRIEVE',
110 [ g.session, [ sc_id ] ]
115 g.data.hash.asc[ sc.id() ] = sc; g.data.stash('hash');
117 var label_name = g.data.hash.aou[ sc.owner() ].shortname() + " : " + sc.name();
122 render: 'var l = util.functional.find_list( fm.stat_cat_entries(), function(e){ return e.stat_cat() == '
123 + sc.id() + '; } ); l ? l.value() : null;',
124 input: 'x = util.widgets.make_menulist( util.functional.map_list( g.data.hash.asc[' + sc.id()
125 + '].entries(), function(obj){ return [ obj.value(), obj.id() ]; } ).sort() ); '
126 + 'x.addEventListener("command",function(ev) { g.apply_stat_cat(' + sc.id()
127 + ', ev.target.value); } ,false);',
131 dump('temp_array = ' + js2JSON(temp_array) + '\n');
133 g.right_pane_field_names.push( temp_array );
136 /* The stat cats for the pertinent library */
137 for (var i = 0; i < g.data.list.my_asc.length; i++) {
138 add_stat_cat( g.data.list.my_asc[i] );
141 /* Other stat cats present on these copies */
142 for (var i = 0; i < g.copies.length; i++) {
143 var entries = g.copies[i].stat_cat_entries();
144 if (!entries) entries = [];
145 for (var j = 0; j < entries.length; j++) {
146 var sc_id = entries[j].stat_cat();
147 add_stat_cat( sc_id );
151 /******************************************************************************************************/
154 g.summarize( g.copies );
158 var err_msg = "!! This software has encountered an error. Please tell your friendly " +
159 "system administrator or software developer the following:\ncat/copy_editor.xul\n" + E + '\n';
160 try { g.error.sdump('D_ERROR',err_msg); } catch(E) { dump(err_msg); dump(js2JSON(E)); }
165 /******************************************************************************************************/
166 /* Apply a value to a specific field on all the copies being edited */
168 g.apply = function(field,value) {
169 g.error.sdump('D_TRACE','field = ' + field + ' value = ' + value + '\n');
170 for (var i = 0; i < g.copies.length; i++) {
171 var copy = g.copies[i];
173 copy[field]( value ); copy.ischanged('1');
180 /******************************************************************************************************/
181 /* Apply a stat cat entry to all the copies being edited */
183 g.apply_stat_cat = function(sc_id,entry_id) {
184 g.error.sdump('D_TRACE','sc_id = ' + sc_id + ' entry_id = ' + entry_id + '\n');
185 for (var i = 0; i < g.copies.length; i++) {
186 var copy = g.copies[i];
189 var temp = copy.stat_cat_entries();
190 if (!temp) temp = [];
191 temp = util.functional.filter_list(
194 return (obj.stat_cat() != sc_id);
198 util.functional.find_id_object_in_list(
199 g.data.hash.asc[sc_id].entries(),
203 copy.stat_cat_entries( temp );
212 /******************************************************************************************************/
213 /* These need data from the middle layer to render */
215 g.special_exception = {
216 'Call Number' : function(label,value) {
217 if (value>0) { /* an existing call number */
219 api.FM_ACN_RETRIEVE.app,
220 api.FM_ACN_RETRIEVE.method,
223 var cn = '??? id = ' + value;
225 cn = req.getResultObject().label();
227 g.error.sdump('D_ERROR','callnumber retrieve: ' + E);
229 label.setAttribute('value',cn);
232 } else { /* a yet to be created call number */
234 label.setAttribute('value',g.callnumbers[value]);
238 'Creator' : function(label,value) {
239 if (value == null || value == '' || value == 'null') return;
240 g.network.simple_request(
241 'FM_AU_RETRIEVE_VIA_ID',
242 [ g.session, value ],
244 var p = '??? id = ' + value;
246 p = req.getResultObject();
247 p = p.usrname() + ' : ' + p.family_name() + ', ' + p.first_given_name();
250 g.error.sdump('D_ERROR','patron retrieve: ' + E);
252 label.setAttribute('value',p);
256 'Last Editor' : function(label,value) {
257 if (value == null || value == '' || value == 'null') return;
258 g.network.simple_request(
259 'FM_AU_RETRIEVE_VIA_ID',
260 [ g.session, value ],
262 var p = '??? id = ' + value;
264 p = req.getResultObject();
265 p = p.usrname() + ' : ' + p.family_name() + ', ' + p.first_given_name();
268 g.error.sdump('D_ERROR','patron retrieve: ' + E);
270 label.setAttribute('value',p);
277 /******************************************************************************************************/
278 g.readonly_stat_cat_names = [];
279 g.editable_stat_cat_names = [];
281 /******************************************************************************************************/
282 /* These get show in the left panel */
284 g.left_pane_field_names = [
288 render: 'fm.barcode();',
294 render: 'fm.call_number();',
300 render: 'util.date.formatted_date( fm.create_date(), "%F");',
306 render: 'util.date.formatted_date( fm.edit_date(), "%F");',
312 /******************************************************************************************************/
313 /* These get shown in the right panel */
315 g.right_pane_field_names = [
319 render: 'fm.creator();',
325 render: 'fm.editor();',
331 render: 'fm.alert_message();',
332 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("alert_message",ev.target.value); }, false);',
338 render: 'fm.circ_as_type();',
339 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("circ_as_type",ev.target.value); }, false);',
343 "Circulation Library",
345 render: 'fm.circ_lib().shortname();',
346 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);',
350 "Circulation Modifier",
352 render: 'fm.circ_modifier();',
353 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("circ_modifier",ev.target.value); }, false);',
359 render: 'fm.circulate() ? "Yes" : "No";',
360 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("circulate",ev.target.value); }, false);',
366 render: 'fm.copy_number();',
367 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("copy_number",ev.target.value); }, false);',
373 render: 'fm.deposit() ? "Yes" : "No";',
374 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("deposit",ev.target.value); }, false);',
380 render: 'util.money.sanitize( fm.deposit_amount() );',
381 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("deposit_amount",ev.target.value); }, false);',
387 render: 'switch(fm.fine_level()){ case 1: "Low"; break; case 2: "Normal"; break; case 3: "High"; break; }',
388 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);',
394 render: 'fm.holdable() ? "Yes" : "No";',
395 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("holdable",ev.target.value); }, false);',
401 render: 'switch(fm.loan_duration()){ case 1: "Short"; break; case 2: "Normal"; break; case 3: "Long"; break; }',
402 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);',
409 render: 'fm.location().name();',
410 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);',
417 render: 'fm.opac_visible() ? "Yes" : "No";',
418 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("opac_visible",ev.target.value); }, false);',
424 render: 'util.money.sanitize( fm.price() );',
425 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("deposit_amount",ev.target.value); }, false);',
431 render: 'fm.ref() ? "Yes" : "No";',
432 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("ref",ev.target.value); }, false);',
438 render: 'fm.status().name();',
439 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);',
446 /******************************************************************************************************/
447 /* This loops through all our fieldnames and all the copies, tallying up counts for the different values */
449 g.summarize = function( copies ) {
450 /******************************************************************************************************/
453 JSAN.use('util.date'); JSAN.use('util.money');
455 g.field_names = g.left_pane_field_names;
456 g.field_names = g.field_names.concat( g.right_pane_field_names );
457 g.field_names = g.field_names.concat( g.editable_stat_cat_names );
458 g.field_names = g.field_names.concat( g.readonly_stat_cat_names );
460 /******************************************************************************************************/
461 /* Loop through the field names */
463 for (var i = 0; i < g.field_names.length; i++) {
465 var field_name = g.field_names[i][0];
466 var render = g.field_names[i][1].render;
467 g.summary[ field_name ] = {};
469 /******************************************************************************************************/
470 /* Loop through the copies */
472 for (var j = 0; j < copies.length; j++) {
475 var cmd = render || ('fm.' + field_name + '();');
478 /**********************************************************************************************/
479 /* Try to retrieve the value for this field for this copy */
484 g.error.sdump('D_ERROR','Attempted ' + cmd + '\n' + E + '\n');
486 if (typeof value == 'object' && value != null) {
487 alert('FIXME: field_name = ' + field_name + ' value = ' + js2JSON(value) + '\n');
490 /**********************************************************************************************/
491 /* Tally the count */
493 if (g.summary[ field_name ][ value ]) {
494 g.summary[ field_name ][ value ]++;
496 g.summary[ field_name ][ value ] = 1;
500 g.error.sdump('D_TRACE','summary = ' + js2JSON(g.summary) + '\n');
503 /******************************************************************************************************/
504 /* Display the summarized data and inputs for editing */
506 g.render = function() {
508 /******************************************************************************************************/
509 /* Library setup and clear any existing interface */
511 JSAN.use('util.widgets'); JSAN.use('util.date'); JSAN.use('util.money'); JSAN.use('util.functional');
513 var cns = document.getElementById('call_number_summary');
514 util.widgets.remove_children( cns );
515 var bcs = document.getElementById('barcode_summary');
516 util.widgets.remove_children( bcs );
517 var rp = document.getElementById('right_pane');
518 util.widgets.remove_children( rp );
520 /******************************************************************************************************/
521 /* Make the call number summary */
523 var grid = util.widgets.make_grid( [ { 'flex' : '1' } ] );
524 cns.appendChild(grid);
525 for (var i in g.summary['Call Number']) {
526 var cn_id = i; var count = g.summary['Call Number'][i];
527 var row = document.createElement('row'); grid.lastChild.appendChild(row);
528 var cn_label = document.createElement('description'); row.appendChild(cn_label);
529 g.special_exception['Call Number']( cn_label, cn_id );
530 var count_label = document.createElement('description'); row.appendChild(count_label);
531 var unit = count == 1 ? 'copy' : 'copies';
532 count_label.appendChild( document.createTextNode(count + ' ' + unit) );
535 /******************************************************************************************************/
536 /* List the copy barcodes */
538 for (var i in g.summary['Barcode']) {
540 var hbox = document.createElement('hbox'); bcs.appendChild(hbox);
541 var bc_label = document.createElement('description'); hbox.appendChild(bc_label);
542 bc_label.appendChild( document.createTextNode(bc) );
545 /******************************************************************************************************/
546 /* List the other non-editable fields in this pane */
548 var groupbox; var caption; var vbox; var grid; var rows;
549 for (var i = 0; i < g.left_pane_field_names.length; i++) {
551 var f = g.left_pane_field_names[i]; var fn = f[0];
552 if (fn == 'Call Number' || fn == 'Barcode') continue;
553 groupbox = document.createElement('groupbox'); bcs.parentNode.parentNode.appendChild(groupbox);
554 caption = document.createElement('caption'); groupbox.appendChild(caption);
555 caption.setAttribute('label',fn);
556 vbox = document.createElement('vbox'); groupbox.appendChild(vbox);
557 grid = util.widgets.make_grid( [ { 'flex' : 1 }, {}, {} ] ); vbox.appendChild(grid);
558 grid.setAttribute('flex','1');
559 rows = grid.lastChild;
562 /**************************************************************************************/
563 /* Loop through each value for the field */
565 for (var j in g.summary[fn]) {
566 var value = j; var count = g.summary[fn][j];
567 row = document.createElement('row'); rows.appendChild(row);
568 var label1 = document.createElement('description'); row.appendChild(label1);
569 if (g.special_exception[ fn ]) {
570 g.special_exception[ fn ]( label1, value );
572 label1.appendChild( document.createTextNode(value) );
574 var label2 = document.createElement('description'); row.appendChild(label2);
575 var unit = count == 1 ? 'copy' : 'copies';
576 label2.appendChild( document.createTextNode(count + ' ' + unit) );
578 var hbox = document.createElement('hbox');
579 vbox.appendChild(hbox);
581 g.error.sdump('D_ERROR','copy editor: ' + E + '\n');
585 /******************************************************************************************************/
586 /* Prepare the right panel, which is different for 1-copy view and multi-copy view */
590 /******************************************************************************************************/
591 /* For a less dangerous batch edit, choose one field here */
593 var gb = document.createElement('groupbox'); rp.appendChild(gb);
594 var c = document.createElement('caption'); gb.appendChild(c);
595 c.setAttribute('label','Choose a field to edit');
596 JSAN.use('util.widgets'); JSAN.use('util.functional');
597 var ml = util.widgets.make_menulist(
598 util.functional.map_list(
599 g.right_pane_field_names,
600 function(o,i) { return [ o[0], i ]; }
607 g.render_input(gb, g.right_pane_field_names[ ev.target.value ][1].input);
615 if (g.copies.length == 1) {
617 /******************************************************************************************************/
618 /* 1-copy mode has a single groupbox and each field is a row on a grid */
620 var groupbox; var caption; var vbox; var grid; var rows;
621 groupbox = document.createElement('groupbox'); rp.appendChild(groupbox);
622 caption = document.createElement('caption'); groupbox.appendChild(caption);
623 caption.setAttribute('label','Fields');
624 vbox = document.createElement('vbox'); groupbox.appendChild(vbox);
625 grid = util.widgets.make_grid( [ {}, { 'flex' : 1 } ] ); vbox.appendChild(grid);
626 grid.setAttribute('flex','1');
627 rows = grid.lastChild;
629 /******************************************************************************************************/
630 /* Loop through the field names */
632 for (var i = 0; i < g.right_pane_field_names.length; i++) {
634 var f = g.right_pane_field_names[i]; var fn = f[0];
637 /**************************************************************************************/
638 /* Loop through each value for the field */
640 for (var j in g.summary[fn]) {
641 var value = j; var count = g.summary[fn][j];
642 row = document.createElement('row'); rows.appendChild(row);
643 var label0 = document.createElement('description'); row.appendChild(label0);
644 label0.appendChild( document.createTextNode(fn) );
645 label0.setAttribute('style','font-weight: bold');
646 var label1 = document.createElement('description'); row.appendChild(label1);
647 if (g.special_exception[ fn ]) {
648 g.special_exception[ fn ]( label1, value );
650 label1.appendChild( document.createTextNode(value) );
655 /**************************************************************************************/
656 /* Render the input widget */
658 var hbox = document.createElement('hbox');
659 hbox.setAttribute('id',fn);
660 row.setAttribute('style','border-bottom: dotted black thin');
661 row.appendChild(hbox);
662 if (f[1].input && g.edit) {
663 g.render_input(hbox,f[1].input);
667 g.error.sdump('D_ERROR','copy editor: ' + E + '\n');
673 /******************************************************************************************************/
674 /* multi-copy mode has a groupbox for each field */
676 var groupbox; var caption; var vbox; var grid; var rows;
678 /******************************************************************************************************/
679 /* Loop through the field names */
681 for (var i = 0; i < g.right_pane_field_names.length; i++) {
683 var f = g.right_pane_field_names[i]; var fn = f[0];
684 groupbox = document.createElement('groupbox'); rp.appendChild(groupbox);
685 caption = document.createElement('caption'); groupbox.appendChild(caption);
686 caption.setAttribute('label',fn);
687 vbox = document.createElement('vbox'); groupbox.appendChild(vbox);
688 grid = util.widgets.make_grid( [ { 'flex' : 1 }, {}, {} ] ); vbox.appendChild(grid);
689 grid.setAttribute('flex','1');
690 rows = grid.lastChild;
693 /**************************************************************************************/
694 /* Loop through each value for the field */
696 for (var j in g.summary[fn]) {
697 var value = j; var count = g.summary[fn][j];
698 row = document.createElement('row'); rows.appendChild(row);
699 var label1 = document.createElement('description'); row.appendChild(label1);
700 if (g.special_exception[ fn ]) {
701 g.special_exception[ fn ]( label1, value );
703 label1.appendChild( document.createTextNode(value) );
705 var label2 = document.createElement('description'); row.appendChild(label2);
706 var unit = count == 1 ? 'copy' : 'copies';
707 label2.appendChild( document.createTextNode(count + ' ' + unit) );
709 var hbox = document.createElement('hbox');
710 hbox.setAttribute('id',fn);
711 vbox.appendChild(hbox);
713 /**************************************************************************************/
714 /* Render the input widget */
716 if (f[1].input && g.edit) {
717 g.render_input(hbox,f[1].input);
720 g.error.sdump('D_ERROR','copy editor: ' + E + '\n');
726 /******************************************************************************************************/
727 /* This actually draws the change button and input widget for a given field */
728 g.render_input = function(node,input_cmd) {
730 var spacer = document.createElement('spacer'); node.appendChild(spacer);
731 spacer.setAttribute('flex','1');
732 var deck = document.createElement('deck'); node.appendChild(deck);
733 var btn = document.createElement('button'); deck.appendChild(btn);
734 deck.setAttribute('style','width: 200px; min-width: 200px;');
735 btn.setAttribute('label','Change');
736 btn.setAttribute('oncommand','this.parentNode.selectedIndex = 1;');
737 var x; eval( input_cmd );
738 if (x) deck.appendChild(x);
741 g.error.sdump('D_ERROR',E + '\n');
745 /******************************************************************************************************/
746 /* store the copies in the global xpcom stash */
748 g.stash_and_close = function() {
749 if (g.handle_update) {
751 var r = g.network.request(
752 api.FM_ACP_FLESHED_BATCH_UPDATE.app,
753 api.FM_ACP_FLESHED_BATCH_UPDATE.method,
754 [ g.session, g.copies ]
756 /* FIXME -- revisit the return value here */
758 alert('copy update error: ' + js2JSON(E));
761 g.data.temp = js2JSON( g.copies );
762 g.error.sdump('D_CAT','in modal window, g.data.temp = \n' + g.data.temp + '\n');
763 g.data.stash('temp');
767 /******************************************************************************************************/
768 /* spawn copy notes interface */
770 g.copy_notes = function() {
771 JSAN.use('util.window'); var win = new util.window();
772 win.open(urls.XUL_COPY_NOTES + '?copy_id=' + window.escape(g.copies[0].id()),'Copy Notes','chrome,resizable,modal');