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, existing copies or new copies? */
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 g.right_pane_field_names.push(
74 render: 'fm.status().name();',
75 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);',
81 if (g.copies.length != 1) {
82 document.getElementById('copy_notes').setAttribute('hidden','true');
85 /******************************************************************************************************/
86 /* Show the Record Details? */
89 document.getElementById('brief_display').setAttribute(
91 urls.XUL_BIB_BRIEF + '?docid=' + g.docid
94 document.getElementById('brief_display').setAttribute('hidden','true');
97 /******************************************************************************************************/
98 /* Add stat cats to the right_pane_field_names */
100 var stat_cat_seen = {};
102 function add_stat_cat(sc) {
104 if (typeof g.data.hash.asc == 'undefined') { g.data.hash.asc = {}; g.data.stash('hash'); }
108 if (typeof sc == 'object') {
113 if (typeof stat_cat_seen[sc_id] != 'undefined') { return; }
115 stat_cat_seen[ sc_id ] = 1;
117 if (typeof sc != 'object') {
119 sc = g.network.simple_request(
120 'FM_ASC_BATCH_RETRIEVE',
121 [ g.session, [ sc_id ] ]
126 g.data.hash.asc[ sc.id() ] = sc; g.data.stash('hash');
128 var label_name = g.data.hash.aou[ sc.owner() ].shortname() + " : " + sc.name();
133 render: 'var l = util.functional.find_list( fm.stat_cat_entries(), function(e){ return e.stat_cat() == '
134 + sc.id() + '; } ); l ? l.value() : null;',
135 input: 'x = util.widgets.make_menulist( util.functional.map_list( g.data.hash.asc[' + sc.id()
136 + '].entries(), function(obj){ return [ obj.value(), obj.id() ]; } ).sort() ); '
137 + 'x.addEventListener("command",function(ev) { g.apply_stat_cat(' + sc.id()
138 + ', ev.target.value); } ,false);',
142 dump('temp_array = ' + js2JSON(temp_array) + '\n');
144 g.right_pane_field_names.push( temp_array );
147 /* The stat cats for the pertinent library */
148 for (var i = 0; i < g.data.list.my_asc.length; i++) {
149 add_stat_cat( g.data.list.my_asc[i] );
152 /* Other stat cats present on these copies */
153 for (var i = 0; i < g.copies.length; i++) {
154 var entries = g.copies[i].stat_cat_entries();
155 if (!entries) entries = [];
156 for (var j = 0; j < entries.length; j++) {
157 var sc_id = entries[j].stat_cat();
158 add_stat_cat( sc_id );
162 /******************************************************************************************************/
165 g.summarize( g.copies );
169 var err_msg = "!! This software has encountered an error. Please tell your friendly " +
170 "system administrator or software developer the following:\ncat/copy_editor.xul\n" + E + '\n';
171 try { g.error.sdump('D_ERROR',err_msg); } catch(E) { dump(err_msg); dump(js2JSON(E)); }
176 /******************************************************************************************************/
177 /* Apply a value to a specific field on all the copies being edited */
179 g.apply = function(field,value) {
180 g.error.sdump('D_TRACE','field = ' + field + ' value = ' + value + '\n');
181 for (var i = 0; i < g.copies.length; i++) {
182 var copy = g.copies[i];
184 copy[field]( value ); copy.ischanged('1');
191 /******************************************************************************************************/
192 /* Apply a stat cat entry to all the copies being edited */
194 g.apply_stat_cat = function(sc_id,entry_id) {
195 g.error.sdump('D_TRACE','sc_id = ' + sc_id + ' entry_id = ' + entry_id + '\n');
196 for (var i = 0; i < g.copies.length; i++) {
197 var copy = g.copies[i];
200 var temp = copy.stat_cat_entries();
201 if (!temp) temp = [];
202 temp = util.functional.filter_list(
205 return (obj.stat_cat() != sc_id);
209 util.functional.find_id_object_in_list(
210 g.data.hash.asc[sc_id].entries(),
214 copy.stat_cat_entries( temp );
223 /******************************************************************************************************/
224 /* These need data from the middle layer to render */
226 g.special_exception = {
227 'Call Number' : function(label,value) {
228 if (value>0) { /* an existing call number */
230 api.FM_ACN_RETRIEVE.app,
231 api.FM_ACN_RETRIEVE.method,
234 var cn = '??? id = ' + value;
236 cn = req.getResultObject().label();
238 g.error.sdump('D_ERROR','callnumber retrieve: ' + E);
240 label.setAttribute('value',cn);
243 } else { /* a yet to be created call number */
245 label.setAttribute('value',g.callnumbers[value]);
249 'Creator' : 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);
267 'Last Editor' : function(label,value) {
268 if (value == null || value == '' || value == 'null') return;
269 g.network.simple_request(
270 'FM_AU_RETRIEVE_VIA_ID',
271 [ g.session, value ],
273 var p = '??? id = ' + value;
275 p = req.getResultObject();
276 p = p.usrname() + ' : ' + p.family_name() + ', ' + p.first_given_name();
279 g.error.sdump('D_ERROR','patron retrieve: ' + E);
281 label.setAttribute('value',p);
288 /******************************************************************************************************/
289 g.readonly_stat_cat_names = [];
290 g.editable_stat_cat_names = [];
292 /******************************************************************************************************/
293 /* These get show in the left panel */
295 g.left_pane_field_names = [
299 render: 'fm.barcode();',
305 render: 'fm.call_number();',
311 render: 'util.date.formatted_date( fm.create_date(), "%F");',
317 render: 'util.date.formatted_date( fm.edit_date(), "%F");',
323 /******************************************************************************************************/
324 /* These get shown in the right panel */
326 g.right_pane_field_names = [
330 render: 'fm.creator();',
336 render: 'fm.editor();',
342 render: 'fm.alert_message();',
343 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("alert_message",ev.target.value); }, false);',
349 render: 'fm.circ_as_type();',
350 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("circ_as_type",ev.target.value); }, false);',
354 "Circulation Library",
356 render: 'fm.circ_lib().shortname();',
357 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);',
361 "Circulation Modifier",
363 render: 'fm.circ_modifier();',
364 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("circ_modifier",ev.target.value); }, false);',
370 render: 'fm.circulate() ? "Yes" : "No";',
371 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("circulate",ev.target.value); }, false);',
377 render: 'fm.copy_number();',
378 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("copy_number",ev.target.value); }, false);',
384 render: 'fm.deposit() ? "Yes" : "No";',
385 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("deposit",ev.target.value); }, false);',
391 render: 'util.money.sanitize( fm.deposit_amount() );',
392 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("deposit_amount",ev.target.value); }, false);',
398 render: 'switch(fm.fine_level()){ case 1: "Low"; break; case 2: "Normal"; break; case 3: "High"; break; }',
399 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);',
405 render: 'fm.holdable() ? "Yes" : "No";',
406 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("holdable",ev.target.value); }, false);',
412 render: 'switch(fm.loan_duration()){ case 1: "Short"; break; case 2: "Normal"; break; case 3: "Long"; break; }',
413 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);',
420 render: 'fm.location().name();',
421 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);',
428 render: 'fm.opac_visible() ? "Yes" : "No";',
429 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("opac_visible",ev.target.value); }, false);',
435 render: 'util.money.sanitize( fm.price() );',
436 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("deposit_amount",ev.target.value); }, false);',
442 render: 'fm.ref() ? "Yes" : "No";',
443 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("ref",ev.target.value); }, false);',
448 /******************************************************************************************************/
449 /* This loops through all our fieldnames and all the copies, tallying up counts for the different values */
451 g.summarize = function( copies ) {
452 /******************************************************************************************************/
455 JSAN.use('util.date'); JSAN.use('util.money');
457 g.field_names = g.left_pane_field_names;
458 g.field_names = g.field_names.concat( g.right_pane_field_names );
459 g.field_names = g.field_names.concat( g.editable_stat_cat_names );
460 g.field_names = g.field_names.concat( g.readonly_stat_cat_names );
462 /******************************************************************************************************/
463 /* Loop through the field names */
465 for (var i = 0; i < g.field_names.length; i++) {
467 var field_name = g.field_names[i][0];
468 var render = g.field_names[i][1].render;
469 g.summary[ field_name ] = {};
471 /******************************************************************************************************/
472 /* Loop through the copies */
474 for (var j = 0; j < copies.length; j++) {
477 var cmd = render || ('fm.' + field_name + '();');
480 /**********************************************************************************************/
481 /* Try to retrieve the value for this field for this copy */
486 g.error.sdump('D_ERROR','Attempted ' + cmd + '\n' + E + '\n');
488 if (typeof value == 'object' && value != null) {
489 alert('FIXME: field_name = ' + field_name + ' value = ' + js2JSON(value) + '\n');
492 /**********************************************************************************************/
493 /* Tally the count */
495 if (g.summary[ field_name ][ value ]) {
496 g.summary[ field_name ][ value ]++;
498 g.summary[ field_name ][ value ] = 1;
502 g.error.sdump('D_TRACE','summary = ' + js2JSON(g.summary) + '\n');
505 /******************************************************************************************************/
506 /* Display the summarized data and inputs for editing */
508 g.render = function() {
510 /******************************************************************************************************/
511 /* Library setup and clear any existing interface */
513 JSAN.use('util.widgets'); JSAN.use('util.date'); JSAN.use('util.money'); JSAN.use('util.functional');
515 var cns = document.getElementById('call_number_summary');
516 util.widgets.remove_children( cns );
517 var bcs = document.getElementById('barcode_summary');
518 util.widgets.remove_children( bcs );
519 var rp = document.getElementById('right_pane');
520 util.widgets.remove_children( rp );
522 /******************************************************************************************************/
523 /* Make the call number summary */
525 var grid = util.widgets.make_grid( [ { 'flex' : '1' } ] );
526 cns.appendChild(grid);
527 for (var i in g.summary['Call Number']) {
528 var cn_id = i; var count = g.summary['Call Number'][i];
529 var row = document.createElement('row'); grid.lastChild.appendChild(row);
530 var cn_label = document.createElement('description'); row.appendChild(cn_label);
531 g.special_exception['Call Number']( cn_label, cn_id );
532 var count_label = document.createElement('description'); row.appendChild(count_label);
533 var unit = count == 1 ? 'copy' : 'copies';
534 count_label.appendChild( document.createTextNode(count + ' ' + unit) );
537 /******************************************************************************************************/
538 /* List the copy barcodes */
540 for (var i in g.summary['Barcode']) {
542 var hbox = document.createElement('hbox'); bcs.appendChild(hbox);
543 var bc_label = document.createElement('description'); hbox.appendChild(bc_label);
544 bc_label.appendChild( document.createTextNode(bc) );
547 /******************************************************************************************************/
548 /* List the other non-editable fields in this pane */
550 var groupbox; var caption; var vbox; var grid; var rows;
551 for (var i = 0; i < g.left_pane_field_names.length; i++) {
553 var f = g.left_pane_field_names[i]; var fn = f[0];
554 if (fn == 'Call Number' || fn == 'Barcode') continue;
555 groupbox = document.createElement('groupbox'); bcs.parentNode.parentNode.appendChild(groupbox);
556 caption = document.createElement('caption'); groupbox.appendChild(caption);
557 caption.setAttribute('label',fn);
558 vbox = document.createElement('vbox'); groupbox.appendChild(vbox);
559 grid = util.widgets.make_grid( [ { 'flex' : 1 }, {}, {} ] ); vbox.appendChild(grid);
560 grid.setAttribute('flex','1');
561 rows = grid.lastChild;
564 /**************************************************************************************/
565 /* Loop through each value for the field */
567 for (var j in g.summary[fn]) {
568 var value = j; var count = g.summary[fn][j];
569 row = document.createElement('row'); rows.appendChild(row);
570 var label1 = document.createElement('description'); row.appendChild(label1);
571 if (g.special_exception[ fn ]) {
572 g.special_exception[ fn ]( label1, value );
574 label1.appendChild( document.createTextNode(value) );
576 var label2 = document.createElement('description'); row.appendChild(label2);
577 var unit = count == 1 ? 'copy' : 'copies';
578 label2.appendChild( document.createTextNode(count + ' ' + unit) );
580 var hbox = document.createElement('hbox');
581 vbox.appendChild(hbox);
583 g.error.sdump('D_ERROR','copy editor: ' + E + '\n');
587 /******************************************************************************************************/
588 /* Prepare the right panel, which is different for 1-copy view and multi-copy view */
592 /******************************************************************************************************/
593 /* For a less dangerous batch edit, choose one field here */
595 var gb = document.createElement('groupbox'); rp.appendChild(gb);
596 var c = document.createElement('caption'); gb.appendChild(c);
597 c.setAttribute('label','Choose a field to edit');
598 JSAN.use('util.widgets'); JSAN.use('util.functional');
599 var ml = util.widgets.make_menulist(
600 util.functional.map_list(
601 g.right_pane_field_names,
602 function(o,i) { return [ o[0], i ]; }
609 g.render_input(gb, g.right_pane_field_names[ ev.target.value ][1].input);
617 if (g.copies.length == 1) {
619 /******************************************************************************************************/
620 /* 1-copy mode has a single groupbox and each field is a row on a grid */
622 var groupbox; var caption; var vbox; var grid; var rows;
623 groupbox = document.createElement('groupbox'); rp.appendChild(groupbox);
624 caption = document.createElement('caption'); groupbox.appendChild(caption);
625 caption.setAttribute('label','Fields');
626 vbox = document.createElement('vbox'); groupbox.appendChild(vbox);
627 grid = util.widgets.make_grid( [ {}, { 'flex' : 1 } ] ); vbox.appendChild(grid);
628 grid.setAttribute('flex','1');
629 rows = grid.lastChild;
631 /******************************************************************************************************/
632 /* Loop through the field names */
634 for (var i = 0; i < g.right_pane_field_names.length; i++) {
636 var f = g.right_pane_field_names[i]; var fn = f[0];
639 /**************************************************************************************/
640 /* Loop through each value for the field */
642 for (var j in g.summary[fn]) {
643 var value = j; var count = g.summary[fn][j];
644 row = document.createElement('row'); rows.appendChild(row);
645 var label0 = document.createElement('description'); row.appendChild(label0);
646 label0.appendChild( document.createTextNode(fn) );
647 label0.setAttribute('style','font-weight: bold');
648 var label1 = document.createElement('description'); row.appendChild(label1);
649 if (g.special_exception[ fn ]) {
650 g.special_exception[ fn ]( label1, value );
652 label1.appendChild( document.createTextNode(value) );
657 /**************************************************************************************/
658 /* Render the input widget */
660 var hbox = document.createElement('hbox');
661 hbox.setAttribute('id',fn);
662 row.setAttribute('style','border-bottom: dotted black thin');
663 row.appendChild(hbox);
664 if (f[1].input && g.edit) {
665 g.render_input(hbox,f[1].input);
669 g.error.sdump('D_ERROR','copy editor: ' + E + '\n');
675 /******************************************************************************************************/
676 /* multi-copy mode has a groupbox for each field */
678 var groupbox; var caption; var vbox; var grid; var rows;
680 /******************************************************************************************************/
681 /* Loop through the field names */
683 for (var i = 0; i < g.right_pane_field_names.length; i++) {
685 var f = g.right_pane_field_names[i]; var fn = f[0];
686 groupbox = document.createElement('groupbox'); rp.appendChild(groupbox);
687 caption = document.createElement('caption'); groupbox.appendChild(caption);
688 caption.setAttribute('label',fn);
689 vbox = document.createElement('vbox'); groupbox.appendChild(vbox);
690 grid = util.widgets.make_grid( [ { 'flex' : 1 }, {}, {} ] ); vbox.appendChild(grid);
691 grid.setAttribute('flex','1');
692 rows = grid.lastChild;
695 /**************************************************************************************/
696 /* Loop through each value for the field */
698 for (var j in g.summary[fn]) {
699 var value = j; var count = g.summary[fn][j];
700 row = document.createElement('row'); rows.appendChild(row);
701 var label1 = document.createElement('description'); row.appendChild(label1);
702 if (g.special_exception[ fn ]) {
703 g.special_exception[ fn ]( label1, value );
705 label1.appendChild( document.createTextNode(value) );
707 var label2 = document.createElement('description'); row.appendChild(label2);
708 var unit = count == 1 ? 'copy' : 'copies';
709 label2.appendChild( document.createTextNode(count + ' ' + unit) );
711 var hbox = document.createElement('hbox');
712 hbox.setAttribute('id',fn);
713 vbox.appendChild(hbox);
715 /**************************************************************************************/
716 /* Render the input widget */
718 if (f[1].input && g.edit) {
719 g.render_input(hbox,f[1].input);
722 g.error.sdump('D_ERROR','copy editor: ' + E + '\n');
728 /******************************************************************************************************/
729 /* This actually draws the change button and input widget for a given field */
730 g.render_input = function(node,input_cmd) {
732 var spacer = document.createElement('spacer'); node.appendChild(spacer);
733 spacer.setAttribute('flex','1');
734 var deck = document.createElement('deck'); node.appendChild(deck);
735 var btn = document.createElement('button'); deck.appendChild(btn);
736 deck.setAttribute('style','width: 200px; min-width: 200px;');
737 btn.setAttribute('label','Change');
738 btn.setAttribute('oncommand','this.parentNode.selectedIndex = 1;');
739 var x; eval( input_cmd );
740 if (x) deck.appendChild(x);
743 g.error.sdump('D_ERROR',E + '\n');
747 /******************************************************************************************************/
748 /* store the copies in the global xpcom stash */
750 g.stash_and_close = function() {
751 if (g.handle_update) {
753 var r = g.network.request(
754 api.FM_ACP_FLESHED_BATCH_UPDATE.app,
755 api.FM_ACP_FLESHED_BATCH_UPDATE.method,
756 [ g.session, g.copies ]
758 /* FIXME -- revisit the return value here */
760 alert('copy update error: ' + js2JSON(E));
763 g.data.temp = js2JSON( g.copies );
764 g.error.sdump('D_CAT','in modal window, g.data.temp = \n' + g.data.temp + '\n');
765 g.data.stash('temp');
769 /******************************************************************************************************/
770 /* spawn copy notes interface */
772 g.copy_notes = function() {
773 JSAN.use('util.window'); var win = new util.window();
774 win.open(urls.XUL_COPY_NOTES + '?copy_id=' + window.escape(g.copies[0].id()),'Copy Notes','chrome,resizable,modal');