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.docid = g.cgi.param('docid');
22 g.handle_update = g.cgi.param('handle_update');
24 /******************************************************************************************************/
25 /* Get the copy ids from various sources and flesh them */
28 if (g.cgi.param('copy_ids')) copy_ids = JSON2js( g.cgi.param('copy_ids') );
29 if (window.xulG && window.xulG.copy_ids) copy_ids = copy_ids.concat( window.xulG.copy_ids );
30 if (typeof g.data.temp_copy_ids != 'undefined' && g.data.temp_copy_ids != null) copy_ids = copy_ids.concat( JSON2js( g.data.temp_copy_ids ) );
31 if (!copy_ids) 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') ) );
45 if (g.data.temp_copies != 'undefined' && g.data.temp_copies) g.copies = g.copies.concat( JSON2js( g.data.temp_copies ) );
46 g.data.temp_copies = null; g.data.stash('temp_copies');
48 /******************************************************************************************************/
49 /* We try to retrieve callnumbers for existing copies, but for new copies, we rely on this */
51 if (window.xulG && window.xulG.callnumbers) g.callnumbers = window.xulG.callnumbers;
52 if (g.cgi.param('callnumbers')) g.callnumbers = JSON2js( g.cgi.param('callnumbers') );
53 if (g.data.temp_callnumbers != 'undefined') g.callnumbers = JSON2js( g.data.temp_callnumbers );
55 /******************************************************************************************************/
56 /* Is the interface an editor or a viewer, single or multi copy, existing copies or new copies? */
58 if (g.cgi.param('edit') == '1') {
60 document.getElementById('caption').setAttribute('label','Copy Editor');
61 document.getElementById('save').setAttribute('hidden','false');
62 g.retrieve_templates();
64 $('top_nav').setAttribute('hidden','true');
67 if (g.copies.length > 0 && g.copies[0].id() < 0) {
68 document.getElementById('copy_notes').setAttribute('hidden','true');
69 g.apply("status",5 /* In Process */);
70 $('save').setAttribute('label','Create Copies');
72 g.panes_and_field_names.left_pane =
77 render: 'typeof fm.status() == "object" ? fm.status().name() : g.data.hash.ccs[ fm.status() ].name()',
78 input: g.safe_to_edit_copy_status() ? 'c = function(v){ g.apply("status",v); if (typeof post_c == "function") post_c(v); }; x = util.widgets.make_menulist( util.functional.map_list( g.data.list.ccs, function(obj) { return [ obj.name(), obj.id(), typeof my_constants.magical_statuses[obj.id()] != "undefined" ? true : false ]; } ).sort() ); x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);' : undefined,
79 //input: 'c = function(v){ g.apply("status",v); if (typeof post_c == "function") post_c(v); }; x = util.widgets.make_menulist( util.functional.map_list( util.functional.filter_list( g.data.list.ccs, function(obj) { return typeof my_constants.magical_statuses[obj.id()] == "undefined"; } ), function(obj) { return [ obj.name(), obj.id() ]; } ).sort() ); x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);',
82 ].concat(g.panes_and_field_names.left_pane);
85 if (g.copies.length != 1) {
86 document.getElementById('copy_notes').setAttribute('hidden','true');
89 /******************************************************************************************************/
90 /* Show the Record Details? */
93 document.getElementById('brief_display').setAttribute(
95 urls.XUL_BIB_BRIEF + '?docid=' + g.docid
98 document.getElementById('brief_display').setAttribute('hidden','true');
101 /******************************************************************************************************/
102 /* Add stat cats to the panes_and_field_names.right_pane4 */
104 g.stat_cat_seen = {};
106 function add_stat_cat(sc) {
108 if (typeof g.data.hash.asc == 'undefined') { g.data.hash.asc = {}; g.data.stash('hash'); }
112 if (typeof sc == 'object') {
117 if (typeof g.stat_cat_seen[sc_id] != 'undefined') { return; }
119 g.stat_cat_seen[ sc_id ] = 1;
121 if (typeof sc != 'object') {
123 sc = g.network.simple_request(
124 'FM_ASC_BATCH_RETRIEVE',
130 g.data.hash.asc[ sc.id() ] = sc; g.data.stash('hash');
132 var label_name = g.data.hash.aou[ sc.owner() ].shortname() + " : " + sc.name();
137 render: 'var l = util.functional.find_list( fm.stat_cat_entries(), function(e){ return e.stat_cat() == '
138 + sc.id() + '; } ); l ? l.value() : "<Unset>";',
139 input: 'c = function(v){ g.apply_stat_cat(' + sc.id() + ',v); if (typeof post_c == "function") post_c(v); }; x = util.widgets.make_menulist( [ [ "<Remove Stat Cat>", -1 ] ].concat( util.functional.map_list( g.data.hash.asc[' + sc.id()
140 + '].entries(), function(obj){ return [ obj.value(), obj.id() ]; } ) ).sort() ); '
141 //input: 'c = function(v){ g.apply_stat_cat(' + sc.id() + ',v); if (typeof post_c == "function") post_c(v); }; x = util.widgets.make_menulist( [ [ "<Remove Stat Cat>", null ] ].concat( util.functional.map_list( g.data.hash.asc[' + sc.id()
142 // + '].entries(), function(obj){ return [ obj.value(), obj.id() ]; } ).sort() ) ); '
143 + 'x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c),false);',
147 dump('temp_array = ' + js2JSON(temp_array) + '\n');
149 g.panes_and_field_names.right_pane4.push( temp_array );
152 /* The stat cats for the pertinent library */
153 for (var i = 0; i < g.data.list.my_asc.length; i++) {
154 add_stat_cat( g.data.list.my_asc[i] );
157 /* Other stat cats present on these copies */
158 for (var i = 0; i < g.copies.length; i++) {
159 var entries = g.copies[i].stat_cat_entries();
160 if (!entries) entries = [];
161 for (var j = 0; j < entries.length; j++) {
162 var sc_id = entries[j].stat_cat();
163 add_stat_cat( sc_id );
167 /******************************************************************************************************/
168 /* Backup copies :) */
170 g.original_copies = js2JSON( g.copies );
172 /******************************************************************************************************/
175 g.summarize( g.copies );
179 var err_msg = "!! This software has encountered an error. Please tell your friendly " +
180 "system administrator or software developer the following:\ncat/copy_editor.xul\n" + E + '\n';
181 try { g.error.sdump('D_ERROR',err_msg); } catch(E) { dump(err_msg); dump(js2JSON(E)); }
186 /******************************************************************************************************/
187 /* Retrieve Templates */
189 g.retrieve_templates = function() {
191 JSAN.use('util.widgets'); JSAN.use('util.functional');
193 var robj = g.network.simple_request('FM_AUS_RETRIEVE',[ses(),g.data.list.au[0].id()]);
194 if (typeof robj['staff_client.copy_editor.templates'] != 'undefined') {
195 g.templates = robj['staff_client.copy_editor.templates'];
197 util.widgets.remove_children('template_placeholder');
198 var list = util.functional.map_object_to_list( g.templates, function(obj,i) { return [i, i]; } );
200 g.template_menu = util.widgets.make_menulist( list );
201 $('template_placeholder').appendChild(g.template_menu);
203 g.error.standard_unexpected_error_alert('Error retrieving templates',E);
207 /******************************************************************************************************/
210 g.apply_template = function() {
212 var name = g.template_menu.value;
213 if (g.templates[ name ] != 'undefined') {
214 var template = g.templates[ name ];
215 for (var i in template) {
216 g.changed[ i ] = template[ i ];
217 switch( template[i].type ) {
219 g.apply(template[i].field,template[i].value);
222 if (g.stat_cat_seen[ template[i].field ]) g.apply_stat_cat(template[i].field,template[i].value);
226 g.summarize( g.copies );
230 g.error.standard_unexpected_error_alert('Error applying template',E);
234 /******************************************************************************************************/
235 /* Save as Template */
237 g.save_template = function() {
239 var name = window.prompt('Enter template name:','','Save As Template');
241 g.templates[name] = g.changed;
242 var robj = g.network.simple_request(
243 'FM_AUS_UPDATE',[ses(),g.data.list.au[0].id(), { 'staff_client.copy_editor.templates' : g.templates }]
245 if (typeof robj.ilsevent != 'undefined') {
248 alert('Template "' + name + '" saved.');
252 g.retrieve_templates();
254 g.error.standard_unexpected_error_alert('Error saving template',E);
260 g.error.standard_unexpected_error_alert('Error saving template',E);
264 /******************************************************************************************************/
265 /* Delete Template */
267 g.delete_template = function() {
269 var name = g.template_menu.value;
271 if (! window.confirm('Delete template "' + name + '"?') ) return;
272 delete(g.templates[name]);
273 var robj = g.network.simple_request(
274 'FM_AUS_UPDATE',[ses(),g.data.list.au[0].id(), { 'staff_client.copy_editor.templates' : g.templates }]
276 if (typeof robj.ilsevent != 'undefined') {
279 alert('Template "' + name + '" deleted.');
283 g.retrieve_templates();
285 g.error.standard_unexpected_error_alert('Error deleting template',E);
291 g.error.standard_unexpected_error_alert('Error deleting template',E);
295 /******************************************************************************************************/
296 /* Restore backup copies */
298 g.reset = function() {
300 g.copies = JSON2js( g.original_copies );
301 g.summarize( g.copies );
305 /******************************************************************************************************/
306 /* Apply a value to a specific field on all the copies being edited */
308 g.apply = function(field,value) {
309 g.error.sdump('D_TRACE','applying field = <' + field + '> value = <' + value + '>\n');
310 if (value == '<HACK:KLUDGE:NULL>') value = null;
311 for (var i = 0; i < g.copies.length; i++) {
312 var copy = g.copies[i];
314 copy[field]( value ); copy.ischanged('1');
321 /******************************************************************************************************/
322 /* Apply a stat cat entry to all the copies being edited. An entry_id of < 0 signifies the stat cat is being removed. */
324 g.apply_stat_cat = function(sc_id,entry_id) {
325 g.error.sdump('D_TRACE','sc_id = ' + sc_id + ' entry_id = ' + entry_id + '\n');
326 for (var i = 0; i < g.copies.length; i++) {
327 var copy = g.copies[i];
330 var temp = copy.stat_cat_entries();
331 if (!temp) temp = [];
332 temp = util.functional.filter_list(
335 return (obj.stat_cat() != sc_id);
338 if (entry_id > -1) temp.push(
339 util.functional.find_id_object_in_list(
340 g.data.hash.asc[sc_id].entries(),
344 copy.stat_cat_entries( temp );
347 g.error.standard_unexpected_error_alert('apply_stat_cat',E);
352 /******************************************************************************************************/
353 /* This returns true if none of the copies being edited have a magical status found in my_constants.magical_statuses */
355 g.safe_to_edit_copy_status = function() {
358 for (var i = 0; i < g.copies.length; i++) {
359 var status = g.copies[i].status(); if (typeof status == 'object') status = status.id();
360 if (typeof my_constants.magical_statuses[ status ] != 'undefined') safe = false;
364 g.error.standard_unexpected_error_alert('safe_to_edit_copy_status?',E);
369 /******************************************************************************************************/
370 /* This concats and uniques all the alert messages for use as the default value for a new alert message */
372 g.populate_alert_message_input = function(tb) {
374 var seen = {}; var s = '';
375 for (var i = 0; i < g.copies.length; i++) {
376 var msg = g.copies[i].alert_message();
378 if (typeof seen[msg] == 'undefined') {
384 tb.setAttribute('value',s);
386 g.error.standard_unexpected_error_alert('populate_alert_message_input',E);
390 /******************************************************************************************************/
391 /* This returns a list of acpl's appropriate for the copies being edited */
393 g.get_acpl_list = function() {
396 function get(lib_id) {
397 g.data.stash_retrieve();
398 var label = 'acpl_list_for_lib_'+lib_id;
399 if (typeof g.data[label] == 'undefined') {
400 var robj = g.network.simple_request('FM_ACPL_RETRIEVE', [ lib_id ]);
401 if (typeof robj.ilsevent != 'undefined') throw(robj);
403 for (var j = 0; j < robj.length; j++) {
404 var my_acpl = robj[j];
405 if (typeof g.data.hash.acpl[ my_acpl.id() ] == 'undefined') {
406 g.data.hash.acpl[ my_acpl.id() ] = my_acpl;
407 g.data.list.acpl.push( my_acpl );
409 temp_list.push( my_acpl );
411 g.data[label] = temp_list; g.data.stash(label,'hash','list');
413 return g.data[label];
416 var seen = {}; seen[ g.data.list.au[0].ws_ou() ] = true;
417 var list = get( g.data.list.au[0].ws_ou() ); //g.data.list.acpl;
418 for (var i = 0; i < g.copies.length; i++) {
419 var cn_id = g.copies[i].call_number();
421 var my_acn = g.network.simple_request('FM_ACN_RETRIEVE',[ cn_id ]);
422 var lib = my_acn.owning_lib();
423 if ( typeof seen[lib] == 'undefined' ) {
425 var r = get(my_acn.owning_lib());
426 list = list.concat( r );
432 g.error.standard_unexpected_error_alert('get_acpl_list',E);
438 /******************************************************************************************************/
439 /* This keeps track of what fields have been edited for styling purposes */
443 /******************************************************************************************************/
444 /* These need data from the middle layer to render */
446 g.special_exception = {
447 'Owning Lib : Call Number' : function(label,value) {
448 JSAN.use('util.widgets');
449 if (value>0) { /* an existing call number */
451 api.FM_ACN_RETRIEVE.app,
452 api.FM_ACN_RETRIEVE.method,
455 var cn = '??? id = ' + value;
457 cn = req.getResultObject();
459 g.error.sdump('D_ERROR','callnumber retrieve: ' + E);
461 util.widgets.set_text(label,g.data.hash.aou[ cn.owning_lib() ].shortname() + ' : ' + cn.label());
464 } else { /* a yet to be created call number */
466 util.widgets.set_text(label,g.data.hash.aou[ g.callnumbers[value].owning_lib ].shortname() + ' : ' + g.callnumbers[value].label);
470 'Creator' : function(label,value) {
471 if (value == null || value == '' || value == 'null') return;
472 g.network.simple_request(
473 'FM_AU_RETRIEVE_VIA_ID',
476 var p = '??? id = ' + value;
478 p = req.getResultObject();
482 g.error.sdump('D_ERROR','patron retrieve: ' + E);
484 JSAN.use('util.widgets');
485 util.widgets.set_text(label,p);
489 'Last Editor' : function(label,value) {
490 if (value == null || value == '' || value == 'null') return;
491 g.network.simple_request(
492 'FM_AU_RETRIEVE_VIA_ID',
495 var p = '??? id = ' + value;
497 p = req.getResultObject();
501 g.error.sdump('D_ERROR','patron retrieve: ' + E);
503 util.widgets.set_text(label,p);
510 /******************************************************************************************************/
511 g.readonly_stat_cat_names = [];
512 g.editable_stat_cat_names = [];
514 /******************************************************************************************************/
515 /* These get show in the left panel */
517 g.panes_and_field_names = {
524 render: 'fm.barcode();',
530 render: 'util.date.formatted_date( fm.create_date(), "%F");',
536 render: 'fm.creator();',
542 render: 'util.date.formatted_date( fm.edit_date(), "%F");',
548 render: 'fm.editor();',
559 render: 'typeof fm.location() == "object" ? fm.location().name() : g.data.lookup("acpl",fm.location()).name()',
560 input: 'c = function(v){ g.apply("location",v); if (typeof post_c == "function") post_c(v); }; x = util.widgets.make_menulist( util.functional.map_list( g.get_acpl_list(), function(obj) { return [ g.data.hash.aou[ obj.owning_lib() ].shortname() + " : " + obj.name(), obj.id() ]; }).sort()); x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);',
565 "Circulation Library",
567 render: 'typeof fm.circ_lib() == "object" ? fm.circ_lib().shortname() : g.data.hash.aou[ fm.circ_lib() ].shortname()',
568 input: 'c = function(v){ g.apply("circ_lib",v); if (typeof post_c == "function") post_c(v); }; 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("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);',
572 "Owning Lib : Call Number",
574 render: 'fm.call_number();',
580 render: 'fm.copy_number() == null ? "<Unset>" : fm.copy_number()',
581 input: 'c = function(v){ g.apply("copy_number",v); if (typeof post_c == "function") post_c(v); }; x = document.createElement("textbox"); x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);',
593 render: 'fm.circulate() == null ? "<Unset>" : ( get_bool( fm.circulate() ) ? "Yes" : "No" )',
594 input: 'c = function(v){ g.apply("circulate",v); if (typeof post_c == "function") post_c(v); }; x = util.widgets.make_menulist( [ [ "Yes", get_db_true() ], [ "No", get_db_false() ] ] ); x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);',
600 render: 'fm.holdable() == null ? "<Unset>" : ( get_bool( fm.holdable() ) ? "Yes" : "No" )',
601 input: 'c = function(v){ g.apply("holdable",v); if (typeof post_c == "function") post_c(v); }; x = util.widgets.make_menulist( [ [ "Yes", get_db_true() ], [ "No", get_db_false() ] ] ); x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);',
607 render: 'fm.age_protect() == null ? "<Unset>" : ( typeof fm.age_protect() == "object" ? fm.age_protect().name() : g.data.hash.crahp[ fm.age_protect() ].name() )',
608 input: 'c = function(v){ g.apply("age_protect",v); if (typeof post_c == "function") post_c(v); }; x = util.widgets.make_menulist( [ [ "<Remove Protection>", "<HACK:KLUDGE:NULL>" ] ].concat( util.functional.map_list( g.data.list.crahp, function(obj) { return [ obj.name(), obj.id() ]; }).sort() ) ); x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);',
615 render: 'switch(fm.loan_duration()){ case 1: case "1": "Short"; break; case 2: case "2": "Normal"; break; case 3:case "3": "Long"; break; }',
616 input: 'c = function(v){ g.apply("loan_duration",v); if (typeof post_c == "function") post_c(v); }; x = util.widgets.make_menulist( [ [ "Short", "1" ], [ "Normal", "2" ], [ "Long", "3" ] ] ); x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);',
623 render: 'switch(fm.fine_level()){ case 1: case "1": "Low"; break; case 2: case "2": "Normal"; break; case 3: case "3": "High"; break; }',
624 input: 'c = function(v){ g.apply("fine_level",v); if (typeof post_c == "function") post_c(v); }; x = util.widgets.make_menulist( [ [ "Low", "1" ], [ "Normal", "2" ], [ "High", "3" ] ] ); x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);',
631 render: 'fm.circ_as_type() == null ? "<Unset>" : g.data.hash.citm[ fm.circ_as_type() ].value()',
632 input: 'c = function(v){ g.apply("circ_as_type",v); if (typeof post_c == "function") post_c(v); }; x = util.widgets.make_menulist( util.functional.map_list( g.data.list.citm, function(n){return [ n.code() + " - " + n.value(), n.code()];} ).sort() ); x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);',
636 "Circulation Modifier",
638 render: 'fm.circ_modifier() == null ? "<Unset>" : fm.circ_modifier()',
639 /*input: 'c = function(v){ g.apply("circ_modifier",v); if (typeof post_c == "function") post_c(v); }; x = document.createElement("textbox"); x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);',*/
640 input: 'c = function(v){ g.apply("circ_modifier",v); if (typeof post_c == "function") post_c(v); }; x = util.widgets.make_menulist( util.functional.map_list( g.data.list.circ_modifier, function(obj) { return [ obj, obj ]; } ).sort() ); x.setAttribute("editable","true"); x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);',
649 render: 'fm.alert_message() == null ? "<Unset>" : fm.alert_message()',
650 input: 'c = function(v){ g.apply("alert_message",v); if (typeof post_c == "function") post_c(v); }; x = document.createElement("textbox"); x.setAttribute("multiline",true); g.populate_alert_message_input(x); x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);',
657 render: 'fm.deposit() == null ? "<Unset>" : ( get_bool( fm.deposit() ) ? "Yes" : "No" )',
658 input: 'c = function(v){ g.apply("deposit",v); if (typeof post_c == "function") post_c(v); }; x = util.widgets.make_menulist( [ [ "Yes", get_db_true() ], [ "No", get_db_false() ] ] ); x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);',
664 render: 'util.money.sanitize( fm.deposit_amount() );',
665 input: 'c = function(v){ g.apply("deposit_amount",v); if (typeof post_c == "function") post_c(v); }; x = document.createElement("textbox"); x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);',
671 render: 'util.money.sanitize( fm.price() );',
672 input: 'c = function(v){ g.apply("price",v); if (typeof post_c == "function") post_c(v); }; x = document.createElement("textbox"); x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);',
679 render: 'fm.opac_visible() == null ? "<Unset>" : ( get_bool( fm.opac_visible() ) ? "Yes" : "No" )',
680 input: 'c = function(v){ g.apply("opac_visible",v); if (typeof post_c == "function") post_c(v); }; x = util.widgets.make_menulist( [ [ "Yes", get_db_true() ], [ "No", get_db_false() ] ] ); x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);',
686 render: 'fm.ref() == null ? "<Unset>" : ( get_bool( fm.ref() ) ? "Yes" : "No" )',
687 input: 'c = function(v){ g.apply("ref",v); if (typeof post_c == "function") post_c(v); }; x = util.widgets.make_menulist( [ [ "Yes", get_db_true() ], [ "No", get_db_false() ] ] ); x.addEventListener("apply",function(f){ return function(ev) { f(ev.target.value); } }(c), false);',
698 /******************************************************************************************************/
699 /* This loops through all our fieldnames and all the copies, tallying up counts for the different values */
701 g.summarize = function( copies ) {
702 /******************************************************************************************************/
705 JSAN.use('util.date'); JSAN.use('util.money');
708 for (var i in g.panes_and_field_names) {
709 g.field_names = g.field_names.concat( g.panes_and_field_names[i] );
711 g.field_names = g.field_names.concat( g.editable_stat_cat_names );
712 g.field_names = g.field_names.concat( g.readonly_stat_cat_names );
714 /******************************************************************************************************/
715 /* Loop through the field names */
717 for (var i = 0; i < g.field_names.length; i++) {
719 var field_name = g.field_names[i][0];
720 var render = g.field_names[i][1].render;
721 g.summary[ field_name ] = {};
723 /******************************************************************************************************/
724 /* Loop through the copies */
726 for (var j = 0; j < copies.length; j++) {
729 var cmd = render || ('fm.' + field_name + '();');
732 /**********************************************************************************************/
733 /* Try to retrieve the value for this field for this copy */
738 g.error.sdump('D_ERROR','Attempted ' + cmd + '\n' + E + '\n');
740 if (typeof value == 'object' && value != null) {
741 alert('FIXME: field_name = <' + field_name + '> value = <' + js2JSON(value) + '>\n');
744 /**********************************************************************************************/
745 /* Tally the count */
747 if (g.summary[ field_name ][ value ]) {
748 g.summary[ field_name ][ value ]++;
750 g.summary[ field_name ][ value ] = 1;
754 g.error.sdump('D_TRACE','summary = ' + js2JSON(g.summary) + '\n');
757 /******************************************************************************************************/
758 /* Display the summarized data and inputs for editing */
760 g.render = function() {
762 /******************************************************************************************************/
763 /* Library setup and clear any existing interface */
765 JSAN.use('util.widgets'); JSAN.use('util.date'); JSAN.use('util.money'); JSAN.use('util.functional');
767 for (var i in g.panes_and_field_names) {
768 var p = document.getElementById(i);
769 if (p) util.widgets.remove_children(p);
772 /******************************************************************************************************/
773 /* Prepare the panes */
775 var groupbox; var caption; var vbox; var grid; var rows;
777 /******************************************************************************************************/
778 /* Loop through the field names */
780 for (h in g.panes_and_field_names) {
781 if (!document.getElementById(h)) continue;
782 for (var i = 0; i < g.panes_and_field_names[h].length; i++) {
784 var f = g.panes_and_field_names[h][i]; var fn = f[0];
785 groupbox = document.createElement('groupbox'); document.getElementById(h).appendChild(groupbox);
786 if (typeof g.changed[fn] != 'undefined') groupbox.setAttribute('class','copy_editor_field_changed');
787 caption = document.createElement('caption'); groupbox.appendChild(caption);
788 caption.setAttribute('label',fn);
789 vbox = document.createElement('vbox'); groupbox.appendChild(vbox);
790 grid = util.widgets.make_grid( [ { 'flex' : 1 }, {}, {} ] ); vbox.appendChild(grid);
791 grid.setAttribute('flex','1');
792 rows = grid.lastChild;
795 /**************************************************************************************/
796 /* Loop through each value for the field */
798 for (var j in g.summary[fn]) {
799 var value = j; var count = g.summary[fn][j];
800 row = document.createElement('row'); rows.appendChild(row);
801 var label1 = document.createElement('description'); row.appendChild(label1);
802 if (g.special_exception[ fn ]) {
803 g.special_exception[ fn ]( label1, value );
805 label1.appendChild( document.createTextNode(value) );
807 var label2 = document.createElement('description'); row.appendChild(label2);
808 var unit = count == 1 ? 'copy' : 'copies';
809 label2.appendChild( document.createTextNode(count + ' ' + unit) );
811 var hbox = document.createElement('hbox');
812 hbox.setAttribute('id',fn);
813 groupbox.appendChild(hbox);
814 var hbox2 = document.createElement('hbox');
815 groupbox.appendChild(hbox2);
817 /**************************************************************************************/
818 /* Render the input widget */
820 if (f[1].input && g.edit) {
821 g.render_input(hbox,f[1]);
825 g.error.sdump('D_ERROR','copy editor: ' + E + '\n');
831 /******************************************************************************************************/
832 /* This actually draws the change button and input widget for a given field */
833 g.render_input = function(node,blob) {
835 // node = hbox ; groupbox -> hbox, hbox
837 var groupbox = node.parentNode;
838 var caption = groupbox.firstChild;
839 var vbox = node.previousSibling;
841 var hbox2 = node.nextSibling;
843 var input_cmd = blob.input;
844 var render_cmd = blob.render;
846 var block = false; var first = true;
848 function on_mouseover(ev) {
849 groupbox.setAttribute('style','background: white');
852 function on_mouseout(ev) {
853 groupbox.setAttribute('style','');
856 vbox.addEventListener('mouseover',on_mouseover,false);
857 vbox.addEventListener('mouseout',on_mouseout,false);
858 groupbox.addEventListener('mouseover',on_mouseover,false);
859 groupbox.addEventListener('mouseout',on_mouseout,false);
860 groupbox.firstChild.addEventListener('mouseover',on_mouseover,false);
861 groupbox.firstChild.addEventListener('mouseout',on_mouseout,false);
863 function on_click(ev){
865 if (block) return; block = true;
870 var t = input_cmd.match('apply_stat_cat') ? 'stat_cat' : 'attribute';
874 f = input_cmd.match(/apply\("(.+?)",/)[1];
877 f = input_cmd.match(/apply_stat_cat\((.+?),/)[1];
880 g.changed[ hbox.id ] = { 'type' : t, 'field' : f, 'value' : v };
884 g.summarize( g.copies );
889 g.error.standard_unexpected_error_alert('post_c',E);
892 var x; var c; eval( input_cmd );
894 util.widgets.remove_children(vbox);
895 util.widgets.remove_children(hbox);
896 util.widgets.remove_children(hbox2);
898 var apply = document.createElement('button');
899 apply.setAttribute('label','Apply');
900 apply.setAttribute('accesskey','A');
901 hbox2.appendChild(apply);
902 apply.addEventListener('command',function() { c(x.value); },false);
903 var cancel = document.createElement('button');
904 cancel.setAttribute('label','Cancel');
905 cancel.addEventListener('command',function() { setTimeout( function() { g.summarize( g.copies ); g.render(); }, 0); }, false);
906 hbox2.appendChild(cancel);
907 setTimeout( function() { x.focus(); }, 0 );
910 g.error.standard_unexpected_error_alert('render_input',E);
913 vbox.addEventListener('click',on_click, false);
914 hbox.addEventListener('click',on_click, false);
915 caption.addEventListener('click',on_click, false);
916 caption.addEventListener('keypress',function(ev) {
917 if (ev.keyCode == 13 /* enter */ || ev.keyCode == 77 /* mac enter */) on_click();
919 caption.setAttribute('style','-moz-user-focus: normal');
920 caption.setAttribute('onfocus','this.setAttribute("class","outline_me")');
921 caption.setAttribute('onblur','this.setAttribute("class","")');
924 g.error.sdump('D_ERROR',E + '\n');
928 /******************************************************************************************************/
929 /* store the copies in the global xpcom stash */
931 g.stash_and_close = function() {
932 if (g.handle_update) {
934 var r = g.network.request(
935 api.FM_ACP_FLESHED_BATCH_UPDATE.app,
936 api.FM_ACP_FLESHED_BATCH_UPDATE.method,
937 [ ses(), g.copies, true ]
940 if (typeof r.ilsevent != 'undefined') {
941 g.error.standard_unexpected_error_alert('copy update',r);
943 /* FIXME -- revisit the return value here */
945 alert('copy update error: ' + js2JSON(E));
948 g.data.temp_copies = js2JSON( g.copies );
949 g.data.stash('temp_copies');
950 g.error.sdump('D_CAT','in modal window, g.data.temp_copies = \n' + g.data.temp_copies + '\n');
954 /******************************************************************************************************/
955 /* spawn copy notes interface */
957 g.copy_notes = function() {
958 JSAN.use('util.window'); var win = new util.window();
959 win.open(urls.XUL_COPY_NOTES + '?copy_id=' + window.escape(g.copies[0].id()),'Copy Notes','chrome,resizable,modal');