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;
234 api.FM_AU_RETRIEVE_VIA_ID.app,
235 api.FM_AU_RETRIEVE_VIA_ID.method,
236 [ g.session, value ],
238 var p = '??? id = ' + value;
240 p = req.getResultObject();
241 p = p.card().barcode() + ' : ' + p.family_name();
244 g.error.sdump('D_ERROR','patron retrieve: ' + E);
246 label.setAttribute('value',p);
250 'Last Editor' : function(label,value) {
251 if (value == null || value == '' || value == 'null') return;
253 api.FM_AU_RETRIEVE_VIA_ID.app,
254 api.FM_AU_RETRIEVE_VIA_ID.method,
255 [ g.session, value ],
257 var p = '??? id = ' + value;
259 p = req.getResultObject();
260 p = p.card().barcode() + ' : ' + p.family_name();
263 g.error.sdump('D_ERROR','patron retrieve: ' + E);
265 label.setAttribute('value',p);
272 /******************************************************************************************************/
273 g.readonly_stat_cat_names = [];
274 g.editable_stat_cat_names = [];
276 /******************************************************************************************************/
277 /* These get show in the left panel */
279 g.left_pane_field_names = [
283 render: 'fm.barcode();',
289 render: 'fm.call_number();',
295 render: 'util.date.formatted_date( fm.create_date(), "%F");',
301 render: 'util.date.formatted_date( fm.edit_date(), "%F");',
307 /******************************************************************************************************/
308 /* These get shown in the right panel */
310 g.right_pane_field_names = [
314 render: 'fm.creator();',
320 render: 'fm.editor();',
326 render: 'fm.alert_message();',
327 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("alert_message",ev.target.value); }, false);',
333 render: 'fm.circ_as_type();',
334 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("circ_as_type",ev.target.value); }, false);',
338 "Circulation Library",
340 render: 'fm.circ_lib().shortname();',
341 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);',
345 "Circulation Modifier",
347 render: 'fm.circ_modifier();',
348 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("circ_modifier",ev.target.value); }, false);',
354 render: 'fm.circulate() ? "Yes" : "No";',
355 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("circulate",ev.target.value); }, false);',
361 render: 'fm.copy_number();',
362 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("copy_number",ev.target.value); }, false);',
368 render: 'fm.deposit() ? "Yes" : "No";',
369 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("deposit",ev.target.value); }, false);',
375 render: 'util.money.sanitize( fm.deposit_amount() );',
376 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("deposit_amount",ev.target.value); }, false);',
382 render: 'switch(fm.fine_level()){ case 1: "Low"; break; case 2: "Normal"; break; case 3: "High"; break; }',
383 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);',
389 render: 'fm.holdable() ? "Yes" : "No";',
390 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("holdable",ev.target.value); }, false);',
396 render: 'switch(fm.loan_duration()){ case 1: "Short"; break; case 2: "Normal"; break; case 3: "Long"; break; }',
397 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);',
404 render: 'fm.location().name();',
405 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);',
412 render: 'fm.opac_visible() ? "Yes" : "No";',
413 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("opac_visible",ev.target.value); }, false);',
419 render: 'util.money.sanitize( fm.price() );',
420 input: 'x = document.createElement("textbox"); x.addEventListener("change",function(ev) { g.apply("deposit_amount",ev.target.value); }, false);',
426 render: 'fm.ref() ? "Yes" : "No";',
427 input: 'x = util.widgets.make_menulist( [ [ "Yes", "1" ], [ "No", "0" ] ] ); x.addEventListener("command",function(ev) { g.apply("ref",ev.target.value); }, false);',
433 render: 'fm.status().name();',
434 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);',
441 /******************************************************************************************************/
442 /* This loops through all our fieldnames and all the copies, tallying up counts for the different values */
444 g.summarize = function( copies ) {
445 /******************************************************************************************************/
448 JSAN.use('util.date'); JSAN.use('util.money');
450 g.field_names = g.left_pane_field_names;
451 g.field_names = g.field_names.concat( g.right_pane_field_names );
452 g.field_names = g.field_names.concat( g.editable_stat_cat_names );
453 g.field_names = g.field_names.concat( g.readonly_stat_cat_names );
455 /******************************************************************************************************/
456 /* Loop through the field names */
458 for (var i = 0; i < g.field_names.length; i++) {
460 var field_name = g.field_names[i][0];
461 var render = g.field_names[i][1].render;
462 g.summary[ field_name ] = {};
464 /******************************************************************************************************/
465 /* Loop through the copies */
467 for (var j = 0; j < copies.length; j++) {
470 var cmd = render || ('fm.' + field_name + '();');
473 /**********************************************************************************************/
474 /* Try to retrieve the value for this field for this copy */
479 g.error.sdump('D_ERROR','Attempted ' + cmd + '\n' + E + '\n');
481 if (typeof value == 'object' && value != null) {
482 alert('FIXME: field_name = ' + field_name + ' value = ' + js2JSON(value) + '\n');
485 /**********************************************************************************************/
486 /* Tally the count */
488 if (g.summary[ field_name ][ value ]) {
489 g.summary[ field_name ][ value ]++;
491 g.summary[ field_name ][ value ] = 1;
495 g.error.sdump('D_TRACE','summary = ' + js2JSON(g.summary) + '\n');
498 /******************************************************************************************************/
499 /* Display the summarized data and inputs for editing */
501 g.render = function() {
503 /******************************************************************************************************/
504 /* Library setup and clear any existing interface */
506 JSAN.use('util.widgets'); JSAN.use('util.date'); JSAN.use('util.money'); JSAN.use('util.functional');
508 var cns = document.getElementById('call_number_summary');
509 util.widgets.remove_children( cns );
510 var bcs = document.getElementById('barcode_summary');
511 util.widgets.remove_children( bcs );
512 var rp = document.getElementById('right_pane');
513 util.widgets.remove_children( rp );
515 /******************************************************************************************************/
516 /* Make the call number summary */
518 var grid = util.widgets.make_grid( [ { 'flex' : '1' } ] );
519 cns.appendChild(grid);
520 for (var i in g.summary['Call Number']) {
521 var cn_id = i; var count = g.summary['Call Number'][i];
522 var row = document.createElement('row'); grid.lastChild.appendChild(row);
523 var cn_label = document.createElement('label'); row.appendChild(cn_label);
524 g.special_exception['Call Number']( cn_label, cn_id );
525 var count_label = document.createElement('label'); row.appendChild(count_label);
526 var unit = count == 1 ? 'copy' : 'copies';
527 count_label.setAttribute('value',count + ' ' + unit);
530 /******************************************************************************************************/
531 /* List the copy barcodes */
533 for (var i in g.summary['Barcode']) {
535 var hbox = document.createElement('hbox'); bcs.appendChild(hbox);
536 var bc_label = document.createElement('label'); hbox.appendChild(bc_label);
537 bc_label.setAttribute('value',bc);
540 /******************************************************************************************************/
541 /* List the other non-editable fields in this pane */
543 var groupbox; var caption; var vbox; var grid; var rows;
544 for (var i = 0; i < g.left_pane_field_names.length; i++) {
546 var f = g.left_pane_field_names[i]; var fn = f[0];
547 if (fn == 'Call Number' || fn == 'Barcode') continue;
548 groupbox = document.createElement('groupbox'); bcs.parentNode.parentNode.appendChild(groupbox);
549 caption = document.createElement('caption'); groupbox.appendChild(caption);
550 caption.setAttribute('label',fn);
551 vbox = document.createElement('vbox'); groupbox.appendChild(vbox);
552 grid = util.widgets.make_grid( [ { 'flex' : 1 }, {}, {} ] ); vbox.appendChild(grid);
553 grid.setAttribute('flex','1');
554 rows = grid.lastChild;
557 /**************************************************************************************/
558 /* Loop through each value for the field */
560 for (var j in g.summary[fn]) {
561 var value = j; var count = g.summary[fn][j];
562 row = document.createElement('row'); rows.appendChild(row);
563 var label1 = document.createElement('label'); row.appendChild(label1);
564 if (g.special_exception[ fn ]) {
565 g.special_exception[ fn ]( label1, value );
567 label1.setAttribute('value',value);
569 var label2 = document.createElement('label'); row.appendChild(label2);
570 var unit = count == 1 ? 'copy' : 'copies';
571 label2.setAttribute('value',count + ' ' + unit);
573 var hbox = document.createElement('hbox');
574 vbox.appendChild(hbox);
576 g.error.sdump('D_ERROR','copy editor: ' + E + '\n');
580 /******************************************************************************************************/
581 /* Prepare the right panel, which is different for 1-copy view and multi-copy view */
585 /******************************************************************************************************/
586 /* For a less dangerous batch edit, choose one field here */
588 var gb = document.createElement('groupbox'); rp.appendChild(gb);
589 var c = document.createElement('caption'); gb.appendChild(c);
590 c.setAttribute('label','Choose a field to edit');
591 JSAN.use('util.widgets'); JSAN.use('util.functional');
592 var ml = util.widgets.make_menulist(
593 util.functional.map_list(
594 g.right_pane_field_names,
595 function(o,i) { return [ o[0], i ]; }
602 g.render_input(gb, g.right_pane_field_names[ ev.target.value ][1].input);
610 if (g.copies.length == 1) {
612 /******************************************************************************************************/
613 /* 1-copy mode has a single groupbox and each field is a row on a grid */
615 var groupbox; var caption; var vbox; var grid; var rows;
616 groupbox = document.createElement('groupbox'); rp.appendChild(groupbox);
617 caption = document.createElement('caption'); groupbox.appendChild(caption);
618 caption.setAttribute('label','Fields');
619 vbox = document.createElement('vbox'); groupbox.appendChild(vbox);
620 grid = util.widgets.make_grid( [ {}, { 'flex' : 1 } ] ); vbox.appendChild(grid);
621 grid.setAttribute('flex','1');
622 rows = grid.lastChild;
624 /******************************************************************************************************/
625 /* Loop through the field names */
627 for (var i = 0; i < g.right_pane_field_names.length; i++) {
629 var f = g.right_pane_field_names[i]; var fn = f[0];
632 /**************************************************************************************/
633 /* Loop through each value for the field */
635 for (var j in g.summary[fn]) {
636 var value = j; var count = g.summary[fn][j];
637 row = document.createElement('row'); rows.appendChild(row);
638 var label0 = document.createElement('label'); row.appendChild(label0);
639 label0.setAttribute('value',fn);
640 label0.setAttribute('style','font-weight: bold');
641 var label1 = document.createElement('label'); row.appendChild(label1);
642 if (g.special_exception[ fn ]) {
643 g.special_exception[ fn ]( label1, value );
645 label1.setAttribute('value',value);
650 /**************************************************************************************/
651 /* Render the input widget */
653 var hbox = document.createElement('hbox');
654 hbox.setAttribute('id',fn);
655 row.setAttribute('style','border-bottom: dotted black thin');
656 row.appendChild(hbox);
657 if (f[1].input && g.edit) {
658 g.render_input(hbox,f[1].input);
662 g.error.sdump('D_ERROR','copy editor: ' + E + '\n');
668 /******************************************************************************************************/
669 /* multi-copy mode has a groupbox for each field */
671 var groupbox; var caption; var vbox; var grid; var rows;
673 /******************************************************************************************************/
674 /* Loop through the field names */
676 for (var i = 0; i < g.right_pane_field_names.length; i++) {
678 var f = g.right_pane_field_names[i]; var fn = f[0];
679 groupbox = document.createElement('groupbox'); rp.appendChild(groupbox);
680 caption = document.createElement('caption'); groupbox.appendChild(caption);
681 caption.setAttribute('label',fn);
682 vbox = document.createElement('vbox'); groupbox.appendChild(vbox);
683 grid = util.widgets.make_grid( [ { 'flex' : 1 }, {}, {} ] ); vbox.appendChild(grid);
684 grid.setAttribute('flex','1');
685 rows = grid.lastChild;
688 /**************************************************************************************/
689 /* Loop through each value for the field */
691 for (var j in g.summary[fn]) {
692 var value = j; var count = g.summary[fn][j];
693 row = document.createElement('row'); rows.appendChild(row);
694 var label1 = document.createElement('label'); row.appendChild(label1);
695 if (g.special_exception[ fn ]) {
696 g.special_exception[ fn ]( label1, value );
698 label1.setAttribute('value',value);
700 var label2 = document.createElement('label'); row.appendChild(label2);
701 var unit = count == 1 ? 'copy' : 'copies';
702 label2.setAttribute('value',count + ' ' + unit);
705 var hbox = document.createElement('hbox');
706 hbox.setAttribute('id',fn);
707 vbox.appendChild(hbox);
709 /**************************************************************************************/
710 /* Render the input widget */
712 if (f[1].input && g.edit) {
713 g.render_input(hbox,f[1].input);
716 g.error.sdump('D_ERROR','copy editor: ' + E + '\n');
722 /******************************************************************************************************/
723 /* This actually draws the change button and input widget for a given field */
724 g.render_input = function(node,input_cmd) {
726 var spacer = document.createElement('spacer'); node.appendChild(spacer);
727 spacer.setAttribute('flex','1');
728 var deck = document.createElement('deck'); node.appendChild(deck);
729 var btn = document.createElement('button'); deck.appendChild(btn);
730 deck.setAttribute('style','width: 200px; min-width: 200px;');
731 btn.setAttribute('label','Change');
732 btn.setAttribute('oncommand','this.parentNode.selectedIndex = 1;');
733 var x; eval( input_cmd );
734 if (x) deck.appendChild(x);
737 g.error.sdump('D_ERROR',E + '\n');
741 /******************************************************************************************************/
742 /* store the copies in the global xpcom stash */
744 g.stash_and_close = function() {
745 if (g.handle_update) {
747 var r = g.network.request(
748 api.FM_ACP_FLESHED_BATCH_UPDATE.app,
749 api.FM_ACP_FLESHED_BATCH_UPDATE.method,
750 [ g.session, g.copies ]
752 /* FIXME -- revisit the return value here */
754 alert('copy update error: ' + js2JSON(E));
757 g.data.temp = js2JSON( g.copies );
758 g.error.sdump('D_CAT','in modal window, g.data.temp = \n' + g.data.temp + '\n');
759 g.data.stash('temp');