]> git.evergreen-ils.org Git - working/Evergreen.git/blob - Open-ILS/xul/staff_client/server/cat/marcedit.js
4292e78ac7632f0e6937029f65f5ec8de16a7c67
[working/Evergreen.git] / Open-ILS / xul / staff_client / server / cat / marcedit.js
1 /* vim: et:sw=4:ts=4:
2  *
3  * Copyright (C) 2004-2008  Georgia Public Library Service
4  * Copyright (C) 2008-2010  Equinox Software, Inc.
5  * Mike Rylander <miker@esilibrary.com> 
6  *
7  * Copyright (C) 2010 Dan Scott <dan@coffeecode.net>
8  * Copyright (C) 2010 Internationaal Instituut voor Sociale Geschiedenis <info@iisg.nl>
9  *
10  * This program is free software; you can redistribute it and/or
11  * modify it under the terms of the GNU General Public License
12  * as published by the Free Software Foundation; either version 2
13  * of the License, or (at your option) any later version.
14  *
15  * This program is distributed in the hope that it will be useful,
16  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18  * GNU General Public License for more details.  
19  *
20  */
21 // Pretty printing kills whitespace too, so disable it.
22 XML.prettyPrinting = false;
23 var xmlDeclaration = /^<\?xml version[^>]+?>/;
24
25 var serializer = new XMLSerializer();
26 var marcns = new Namespace("http://www.loc.gov/MARC21/slim");
27 var gw = new Namespace("http://opensrf.org/-/namespaces/gateway/v1");
28 var xulns = new Namespace("http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul");
29 default xml namespace = marcns;
30
31 var tooltip_hash = {};
32 var current_focus;
33 var _record;
34 var _record_type;
35 var bib_data;
36
37 var xml_record;
38
39 var context_menus;
40 var tag_menu;
41 var p;
42 var auth_pages = {};
43 var show_auth_menu = false;
44
45 function $(id) { return document.getElementById(id); }
46
47 var acs; // AuthorityControlSet
48
49 function mangle_005() {
50     var now = new Date();
51     var y = now.getUTCFullYear();
52
53     var m = now.getUTCMonth() + 1;
54     if (m < 10) m = '0' + m;
55     
56     var d = now.getUTCDate();
57     if (d < 10) d = '0' + d;
58     
59     var H = now.getUTCHours();
60     if (H < 10) H = '0' + H;
61     
62     var M = now.getUTCMinutes();
63     if (M < 10) M = '0' + M;
64     
65     var S = now.getUTCSeconds();
66     if (S < 10) S = '0' + S;
67     
68
69     var stamp = '' + y + m + d + H + M + S + '.0';
70     createControlField('005',stamp);
71
72 }
73
74 function createControlField (tag,data) {
75     // first, remove the old field, if any;
76     for (var i in xml_record.controlfield.(@tag == tag)) delete xml_record.controlfield.(@tag == tag)[i];
77
78     var cf = <controlfield tag="" xmlns="http://www.loc.gov/MARC21/slim">{ data }</controlfield>;
79     cf.@tag = tag;
80
81     // then, find the right position and insert it
82     var done = 0;
83     var cfields = xml_record.controlfield;
84     var base = Number(tag.substring(2));
85     for (var i in cfields) {
86         var t = Number(cfields[i].@tag.toString().substring(2));
87         if (t > base) {
88             xml_record.insertChildBefore( cfields[i], cf );
89             done = 1
90             break;
91         }
92     }
93
94     if (!done) xml_record.insertChildBefore( xml_record.datafield[0], cf );
95
96     return cf;
97 }
98
99 function xml_escape_unicode ( str ) {
100     return str.replace(
101         /([\u0080-\ufffe])/g,
102         function (r,s) { return "&#x" + s.charCodeAt(0).toString(16) + ";"; }
103     );
104 }
105
106 function wrap_long_fields (node) {
107     var text_size = dojo.attr(node, 'size');
108     var hard_width = 100; 
109     if (text_size > hard_width) {
110         dojo.attr(node, 'multiline', 'true');
111         dojo.attr(node, 'cols', hard_width);
112         var text_rows = (text_size / hard_width) + 1;
113         dojo.attr(node, 'rows', text_rows);
114     }
115 }
116
117 function set_flat_editor (useFlatText) {
118
119     var xe = $('xul-editor');
120     var te = $('text-editor');
121
122     if (useFlatText) {
123         if (xe.hidden) { return; }
124         te.hidden = false;
125         xe.hidden = true;
126     } else {
127         if (te.hidden) { return; }
128         te.hidden = true;
129         xe.hidden = false;
130     }
131
132     if (te.hidden) {
133         // get the marcxml from the text box
134         var xml_string = new MARC.Record({
135             marcbreaker : $('text-editor-box').value,
136             delimiter : '$'
137         }).toXmlString();
138
139         // reset the xml record and rerender it
140         xml_record = new XML( xml_string );
141         if (xml_record..record[0]) xml_record = xml_record..record[0];
142         loadRecord();
143     } else {
144         var xml_string = xml_record.toXMLString();
145
146         // push the xml record into the textbox
147         var rec = new MARC.Record ({ delimiter : '$', marcxml : xml_string });
148         $('text-editor-box').value = rec.toBreaker();
149     }
150 }
151
152 function my_init() {
153     try {
154
155         if (typeof JSAN == 'undefined') { throw( $("commonStrings").getString('common.jsan.missing') ); }
156         JSAN.errorLevel = "die"; // none, warn, or die
157         JSAN.addRepository('/xul/server/');
158
159         dojo.require('openils.AuthorityControlSet');
160         acs = new openils.AuthorityControlSet ();
161
162         // Fake xulG for standalone...
163         try {
164             window.xulG.record;
165         } catch (e) {
166             window.xulG = {};
167             window.xulG.record = {};
168             window.xulG.save = {};
169             window.xulG.marc_control_number_identifier = 'CONS';
170
171             window.xulG.save.label = $('catStrings').getString('staff.cat.marcedit.save.label');
172             window.xulG.save.func = function (r) { alert(r); }
173
174             var cgi = new CGI();
175             var _rid = cgi.param('record');
176             if (_rid) {
177                 window.xulG.record.id = _rid;
178                 window.xulG.record.url = '/opac/extras/supercat/retrieve/marcxml/record/' + _rid;
179             }
180         }
181
182         // End faking part...
183
184         /* Check for an explicitly passed record type
185          * This is not the same as the fixed-field record type; we can't trust
186          * the fixed fields when making modifications to the attributes for a
187          * given record (in particular, config.bib_source only applies for bib
188          * records, but an auth or MFHD record with the same ID and bad fixed
189          * fields could trample the config.bib_source value for the
190          * corresponding bib record if we're not careful.
191          *
192          * are = authority record
193          * sre = serial record (MFHD)
194          * bre = bibliographic record
195          */
196         if (!window.xulG.record.rtype) {
197             var cgi = new CGI();
198             window.xulG.record.rtype = cgi.param('rtype') || false;
199         }
200
201         document.getElementById('save-button').setAttribute('label', window.xulG.save.label);
202         document.getElementById('save-button').setAttribute('oncommand',
203             'if ($("xul-editor").hidden) set_flat_editor(false); ' +
204             'mangle_005(); ' + 
205             'var xml_string = xml_escape_unicode( xml_record.toXMLString() ); ' + 
206             'save_attempt( xml_string ); ' +
207             'loadRecord();'
208         );
209
210         if (window.xulG.record.url) {
211             var req =  new XMLHttpRequest();
212             req.open('POST',window.xulG.record.url,false);
213             req.send(null);
214             window.xulG.record.marc = req.responseText.replace(xmlDeclaration, '');
215         }
216
217         xml_record = new XML( window.xulG.record.marc );
218         if (xml_record..record[0]) xml_record = xml_record..record[0];
219
220         // Get the tooltip xml all async like
221         req =  new XMLHttpRequest();
222
223         // Set a default locale in case preferences fail us
224         var locale = "en-US";
225
226         // Try to get the locale from our preferences
227         try {
228             const Cc = Components.classes;
229             const Ci = Components.interfaces;
230             locale = Cc["@mozilla.org/preferences-service;1"].
231                 getService(Ci.nsIPrefBranch).
232                 getCharPref("general.useragent.locale");
233         }
234         catch (e) { }
235
236         // TODO: We should send a HEAD request to check for the existence of the desired file
237         // then fall back to the default locale if preferred locale is not necessary;
238         // however, for now we have a simplistic check:
239         //
240         // we currently have translations for only three locales; in the absence of a
241         // valid locale, default to the almighty en-US
242         if (locale != 'en-US' && locale != 'fr-CA' && locale != 'fi-FI') {
243             locale = 'en-US';
244         }
245
246         // grab the right tooltip based on MARC type
247         var tooltip_doc = 'marcedit-tooltips.xml';
248         switch (window.xulG.record.rtype) {
249             case 'bre':
250                 tooltip_doc = 'marcedit-tooltips.xml';
251                 break; 
252             case 'are':
253                 tooltip_doc = 'marcedit-tooltips-authority.xml';
254                 locale = 'en-US'; // FIXME - note TODO above; at moment only en-US has this
255                 break; 
256             case 'sre':
257                 tooltip_doc = 'marcedit-tooltips-mfhd.xml';
258                 locale = 'en-US'; // FIXME - note TODO above; at moment only en-US has this
259                 break; 
260             default: 
261                 tooltip_doc = 'marcedit-tooltips.xml';
262         }
263
264         // Get the locale-specific tooltips
265         req.open('GET','/xul/server/locale/' + locale + '/' + tooltip_doc,true);
266
267         context_menus = createComplexXULElement('popupset');
268         document.documentElement.appendChild( context_menus );
269
270         tag_menu = createMenuPopup({position : 'after_start', id : 'tags_popup'});
271         context_menus.appendChild( tag_menu );
272
273         tag_menu.appendChild(
274             createMenuitem(
275                 { label : $('catStrings').getString('staff.cat.marcedit.add_row.label'),
276                   oncommand : 
277                     'var e = document.createEvent("KeyEvents");' +
278                     'e.initKeyEvent("keypress",1,1,null,1,0,0,0,13,0);' +
279                     'current_focus.inputField.dispatchEvent(e);'
280                  }
281             )
282         );
283
284         tag_menu.appendChild(
285             createMenuitem(
286                 { label : $('catStrings').getString('staff.cat.marcedit.insert_row.label'),
287                   oncommand : 
288                     'var e = document.createEvent("KeyEvents");' +
289                     'e.initKeyEvent("keypress",1,1,null,1,0,1,0,13,0);' +
290                     'current_focus.inputField.dispatchEvent(e);'
291                  }
292             )
293         );
294
295         tag_menu.appendChild(
296             createMenuitem(
297                 { label : $('catStrings').getString('staff.cat.marcedit.remove_row.label'),
298                   oncommand : 
299                     'var e = document.createEvent("KeyEvents");' +
300                     'e.initKeyEvent("keypress",1,1,null,1,0,0,0,46,0);' +
301                     'current_focus.inputField.dispatchEvent(e);'
302                 }
303             )
304         );
305
306         tag_menu.appendChild( createComplexXULElement( 'separator' ) );
307
308         tag_menu.appendChild(
309             createMenuitem(
310                 { label : $('catStrings').getString('staff.cat.marcedit.replace_006.label'),
311                   oncommand : 
312                     'var e = document.createEvent("KeyEvents");' +
313                     'e.initKeyEvent("keypress",1,1,null,1,0,0,0,117,0);' +
314                     'current_focus.inputField.dispatchEvent(e);'
315                  }
316             )
317         );
318
319         tag_menu.appendChild(
320             createMenuitem(
321                 { label : $('catStrings').getString('staff.cat.marcedit.replace_007.label'),
322                   oncommand : 
323                     'var e = document.createEvent("KeyEvents");' +
324                     'e.initKeyEvent("keypress",1,1,null,1,0,0,0,118,0);' +
325                     'current_focus.inputField.dispatchEvent(e);'
326                 }
327             )
328         );
329
330         tag_menu.appendChild(
331             createMenuitem(
332                 { label : $('catStrings').getString('staff.cat.marcedit.replace_008.label'),
333                   oncommand : 
334                     'var e = document.createEvent("KeyEvents");' +
335                     'e.initKeyEvent("keypress",1,1,null,1,0,0,0,119,0);' +
336                     'current_focus.inputField.dispatchEvent(e);'
337                 }
338             )
339         );
340
341         tag_menu.appendChild( createComplexXULElement( 'separator' ) );
342
343         p = createComplexXULElement('popupset');
344         document.documentElement.appendChild( p );
345
346         req.onreadystatechange = function () {
347             if (req.readyState == 4) {
348                 bib_data = new XML( req.responseText.replace(xmlDeclaration, '') );
349                 genToolTips();
350             }
351         }
352         req.send(null);
353
354         loadRecord();
355
356         if (! xulG.fast_add_item) {
357             document.getElementById('fastItemAdd_checkbox').hidden = true;
358         }
359         document.getElementById('fastItemAdd_textboxes').hidden = document.getElementById('fastItemAdd_checkbox').hidden || !document.getElementById('fastItemAdd_checkbox').checked;
360
361         // Only show bib sources for bib records that already exist in the database
362         if (xulG.record.rtype == 'bre' && xulG.record.id) {
363             dojo.require('openils.PermaCrud');
364             var authtoken = ses();
365             // Retrieve the current record attributes
366             var bib = new openils.PermaCrud({"authtoken": authtoken}).retrieve('bre', xulG.record.id);
367
368             // Remember the current bib source of the record
369             xulG.record.bre = bib;
370
371             buildBibSourceList(authtoken, xulG.record.id);
372         }
373
374         dojo.require('MARC.FixedFields');
375
376     } catch(E) {
377         alert('FIXME, MARC Editor, my_init: ' + E);
378     }
379 }
380
381
382 function createComplexHTMLElement (e, attrs, objects, text) {
383     var l = document.createElementNS('http://www.w3.org/1999/xhtml',e);
384
385     if (attrs) {
386         for (var i in attrs) l.setAttribute(i,attrs[i]);
387     }
388
389     if (objects) {
390         for ( var i in objects ) l.appendChild( objects[i] );
391     }
392
393     if (text) {
394         l.appendChild( document.createTextNode(text) )
395     }
396
397     return l;
398 }
399
400 function createComplexXULElement (e, attrs, objects) {
401     var l = document.createElementNS('http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul',e);
402
403     if (attrs) {
404         for (var i in attrs) {
405             if (typeof attrs[i] == 'function') {
406                 l.addEventListener( i, attrs[i], true );
407             } else {
408                 l.setAttribute(i,attrs[i]);
409             }
410         }
411     } 
412
413     if (objects) {
414         for ( var i in objects ) l.appendChild( objects[i] );
415     }
416
417     return l;
418 }
419
420 function createDescription (attrs) {
421     return createComplexXULElement('description', attrs, Array.prototype.slice.apply(arguments, [1]) );
422 }
423
424 function createTooltip (attrs) {
425     return createComplexXULElement('tooltip', attrs, Array.prototype.slice.apply(arguments, [1]) );
426 }
427
428 function createLabel (attrs) {
429     return createComplexXULElement('label', attrs, Array.prototype.slice.apply(arguments, [1]) );
430 }
431
432 function createVbox (attrs) {
433     return createComplexXULElement('vbox', attrs, Array.prototype.slice.apply(arguments, [1]) );
434 }
435
436 function createHbox (attrs) {
437     return createComplexXULElement('hbox', attrs, Array.prototype.slice.apply(arguments, [1]) );
438 }
439
440 function createRow (attrs) {
441     return createComplexXULElement('row', attrs, Array.prototype.slice.apply(arguments, [1]) );
442 }
443
444 function createTextbox (attrs) {
445     return createComplexXULElement('textbox', attrs, Array.prototype.slice.apply(arguments, [1]) );
446 }
447
448 function createMenu (attrs) {
449     return createComplexXULElement('menu', attrs, Array.prototype.slice.apply(arguments, [1]) );
450 }
451
452 function createMenuPopup (attrs) {
453     return createComplexXULElement('menupopup', attrs, Array.prototype.slice.apply(arguments, [1]) );
454 }
455
456 function createPopup (attrs) {
457     return createComplexXULElement('popup', attrs, Array.prototype.slice.apply(arguments, [1]) );
458 }
459
460 function createMenuitem (attrs) {
461     return createComplexXULElement('menuitem', attrs, Array.prototype.slice.apply(arguments, [1]) );
462 }
463
464 function createCheckbox (attrs) {
465     return createComplexXULElement('checkbox', attrs, Array.prototype.slice.apply(arguments, [1]) );
466 }
467
468 // Find the next textbox that we can use for a focus point
469 // For control fields, use the first editable text box
470 // For data fields, focus on the first subfield text box
471 function setFocusToNextTag (row, direction) {
472     var keep_looking = true;
473     while (keep_looking && (direction == 'up' ? row = row.previousSibling : row = row.nextSibling)) {
474         // Is it a datafield?
475         dojo.query('hbox', row).query('hbox').query('textbox').forEach(function(node, index, arr) {
476             node.focus();
477             keep_looking = false;
478         });
479
480         // No, it's a control field; use the first textbox
481         if (keep_looking) {
482             dojo.query('textbox', row).forEach(function(node, index, arr) {
483                 node.focus();
484                 keep_looking = false;
485             });
486         }
487     }
488
489     return true;
490 }
491
492 function set_lock_on_keypress(ev) {
493     try {
494         //dump('keypress: isChar = ' + ev.isChar + ' char = ' + ev.char + ' charCode = ' + ev.charCode + ' key = ' + ev.key + ' keyCode = ' + ev.keyCode + '\n');
495         if (! /* NOT */(
496                 ev.altKey
497                 || ev.ctrlKey
498                 || ev.metaKey
499                 || ev.keyCode == ev.DOM_VK_F1
500                 || ev.keyCode == ev.DOM_VK_F2
501                 || ev.keyCode == ev.DOM_VK_F3
502                 || ev.keyCode == ev.DOM_VK_F4
503                 || ev.keyCode == ev.DOM_VK_F5
504                 || ev.keyCode == ev.DOM_VK_F6
505                 || ev.keyCode == ev.DOM_VK_F7
506                 || ev.keyCode == ev.DOM_VK_F8
507                 || ev.keyCode == ev.DOM_VK_F9
508                 || ev.keyCode == ev.DOM_VK_F10
509                 || ev.keyCode == ev.DOM_VK_F11
510                 || ev.keyCode == ev.DOM_VK_F12
511                 || ev.keyCode == ev.DOM_VK_F13
512                 || ev.keyCode == ev.DOM_VK_F14
513                 || ev.keyCode == ev.DOM_VK_F15
514                 || ev.keyCode == ev.DOM_VK_F16
515                 || ev.keyCode == ev.DOM_VK_F17
516                 || ev.keyCode == ev.DOM_VK_F18
517                 || ev.keyCode == ev.DOM_VK_F19
518                 || ev.keyCode == ev.DOM_VK_F20
519                 || ev.keyCode == ev.DOM_VK_F21
520                 || ev.keyCode == ev.DOM_VK_F22
521                 || ev.keyCode == ev.DOM_VK_F23
522                 || ev.keyCode == ev.DOM_VK_F24
523         )) {
524             oils_lock_page();
525         }
526     } catch(E) {
527         alert(E);
528     }
529 }
530
531 function createMARCTextbox (element,attrs) {
532
533     var box = createComplexXULElement('textbox', attrs, Array.prototype.slice.apply(arguments, [2]) );
534     box.addEventListener(
535         'keypress',
536         set_lock_on_keypress,
537         false
538     );
539     box.onkeypress = function (event) {
540         var root_node;
541         var node = element;
542         while(node = node.parent()) {
543             root_node = node;
544         }
545
546         var row = event.target;
547         while (row.tagName != 'row') row = row.parentNode;
548
549         if (element.nodeKind() == 'attribute') element[0]=box.value;
550         else element.setChildren( box.value );
551
552         if (element.localName() != 'controlfield') {
553             if ((event.charCode == 100 || event.charCode == 105) && event.ctrlKey) { // ctrl+d or ctrl+i
554
555                 var index_sf, target, move_data;
556                 if (element.localName() == 'subfield') {
557                     index_sf = element;
558                     target = event.target.parentNode;
559
560                     var start = event.target.selectionStart;
561                     var end = event.target.selectionEnd - event.target.selectionStart ?
562                             event.target.selectionEnd :
563                             event.target.value.length;
564
565                     move_data = event.target.value.substring(start,end);
566                     event.target.value = event.target.value.substring(0,start) + event.target.value.substring(end);
567                     event.target.setAttribute('size', event.target.value.length + 2);
568     
569                     element.setChildren( event.target.value );
570
571                 } else if (element.localName() == 'code') {
572                     index_sf = element.parent();
573                     target = event.target.parentNode;
574                 } else if (element.localName() == 'tag' || element.localName() == 'ind1' || element.localName() == 'ind2') {
575                     index_sf = element.parent().children()[element.parent().children().length() - 1];
576                     target = event.target.parentNode.lastChild.lastChild;
577                 }
578
579                 var sf = <subfield code="" xmlns="http://www.loc.gov/MARC21/slim">{ move_data }</subfield>;
580
581                 index_sf.parent().insertChildAfter( index_sf, sf );
582
583                 var new_sf = marcSubfield(sf);
584
585                 if (target === target.parentNode.lastChild) {
586                     target.parentNode.appendChild( new_sf );
587                 } else {
588                     target.parentNode.insertBefore( new_sf, target.nextSibling );
589                 }
590
591                 new_sf.firstChild.nextSibling.focus();
592
593                 event.preventDefault();
594                 return false;
595
596             } else if (event.keyCode == 13 || event.keyCode == 77) {
597                 if (event.ctrlKey) { // ctrl+enter
598
599                     var index;
600                     if (element.localName() == 'subfield') index = element.parent();
601                     if (element.localName() == 'code') index = element.parent().parent();
602                     if (element.localName() == 'tag') index = element.parent();
603                     if (element.localName() == 'ind1') index = element.parent();
604                     if (element.localName() == 'ind2') index = element.parent();
605
606                     var df = <datafield tag="" ind1="" ind2="" xmlns="http://www.loc.gov/MARC21/slim"><subfield code="" /></datafield>;
607
608                     if (event.shiftKey) { // ctrl+shift+enter
609                         index.parent().insertChildBefore( index, df );
610                     } else {
611                         index.parent().insertChildAfter( index, df );
612                     }
613
614                     var new_df = marcDatafield(df);
615
616                     if (row.parentNode.lastChild === row) {
617                         row.parentNode.appendChild( new_df );
618                     } else {
619                         if (event.shiftKey) { // ctrl+shift+enter
620                             row.parentNode.insertBefore( new_df, row );
621                         } else {
622                             row.parentNode.insertBefore( new_df, row.nextSibling );
623                         }
624                     }
625
626                     new_df.firstChild.focus();
627
628                     event.preventDefault();
629                     return false;
630
631                 } else if (event.shiftKey) {
632                     if (row.previousSibling.className.match('marcDatafieldRow'))
633                         row.previousSibling.firstChild.focus();
634                 } else {
635                     row.nextSibling.firstChild.focus();
636                 }
637
638             } else if (event.keyCode == 38 || event.keyCode == 40) { // up-arrow or down-arrow
639                 if (event.ctrlKey) { // CTRL key: copy the field
640                     var index;
641                     if (element.localName() == 'subfield') index = element.parent();
642                     if (element.localName() == 'code') index = element.parent().parent();
643                     if (element.localName() == 'tag') index = element.parent();
644                     if (element.localName() == 'ind1') index = element.parent();
645                     if (element.localName() == 'ind2') index = element.parent();
646
647                     var copyField = index.copy();
648
649                     if (event.keyCode == 38) { // ctrl+up-arrow
650                         index.parent().insertChildBefore( index, copyField );
651                     } else {
652                         index.parent().insertChildAfter( index, copyField );
653                     }
654
655                     var new_df = marcDatafield(copyField);
656
657                     if (row.parentNode.lastChild === row) {
658                         row.parentNode.appendChild( new_df );
659                     } else {
660                         if (event.keyCode == 38) { // ctrl+up-arrow
661                             row.parentNode.insertBefore( new_df, row );
662                         } else { // ctrl+down-arrow
663                             row.parentNode.insertBefore( new_df, row.nextSibling );
664                         }
665                     }
666
667                     new_df.firstChild.focus();
668
669                     event.preventDefault();
670
671                     return false;
672                 } else {
673                     if (event.keyCode == 38) {
674                         return setFocusToNextTag(row, 'up');
675                     }
676                     if (event.keyCode == 40) {
677                         return setFocusToNextTag(row, 'down');
678                     }
679                     return false;
680                 }
681
682             } else if (event.keyCode == 46 && event.ctrlKey) { // ctrl+del
683
684                 var index;
685                 if (element.localName() == 'subfield') index = element.parent();
686                 if (element.localName() == 'code') index = element.parent().parent();
687                 if (element.localName() == 'tag') index = element.parent();
688                 if (element.localName() == 'ind1') index = element.parent();
689                 if (element.localName() == 'ind2') index = element.parent();
690
691                 for (var i in index.parent().children()) {
692                     if (index === index.parent().children()[i]) {
693                         delete index.parent().children()[i];
694                         break;
695                     }
696                 }
697
698                 row.previousSibling.firstChild.focus();
699                 row.parentNode.removeChild(row);
700
701                 event.preventDefault();
702                 return false;
703
704             } else if (event.keyCode == 46 && event.shiftKey) { // shift+del
705
706                 var index;
707                 if (element.localName() == 'subfield') index = element;
708                 if (element.localName() == 'code') index = element.parent();
709
710                 if (index) {
711                     for (var i in index.parent().children()) {
712                         if (index === index.parent().children()[i]) {
713                             delete index.parent().children()[i];
714                             break;
715                         }
716                     }
717
718                     if (event.target.parentNode === event.target.parentNode.parentNode.lastChild) {
719                         event.target.parentNode.previousSibling.lastChild.focus();
720                     } else {
721                         event.target.parentNode.nextSibling.firstChild.nextSibling.focus();
722                     }
723
724                     event.target.parentNode.parentNode.removeChild(event.target.parentNode);
725
726                     event.preventDefault();
727                     return false;
728                 }
729             } else if (event.keyCode == 117 && event.ctrlKey) { // ctrl + F6
730                 box = null;
731                 createControlField('006','                                        ');
732                 loadRecord();
733             } else if (event.keyCode == 118 && event.ctrlKey) { // ctrl + F7
734                 box = null;
735                 createControlField('007','                                        ');
736                 loadRecord();
737             } else if (event.keyCode == 119 && event.ctrlKey) { // ctrl + F8
738                 box = null;
739                 createControlField('008','                                        ');
740                 loadRecord();
741             }
742
743             return true;
744
745         } else { // event on a control field
746             if (event.keyCode == 38) { 
747                 return setFocusToNextTag(row, 'up'); 
748             } else if (event.keyCode == 40) { 
749                 return setFocusToNextTag(row, 'down');
750             }
751         }
752     };
753
754     box.addEventListener(
755         'keypress', 
756         function () {
757             if (element.nodeKind() == 'attribute') element[0]=box.value;
758             else element.setChildren( box.value );
759             return true;
760         },
761         false
762     );
763
764     box.addEventListener(
765         'change', 
766         function () {
767             if (element.nodeKind() == 'attribute') element[0]=box.value;
768             else element.setChildren( box.value );
769             return true;
770         },
771         false
772     );
773
774     box.addEventListener(
775         'keypress', 
776         function () {
777             if (element.nodeKind() == 'attribute') element[0]=box.value;
778             else element.setChildren( box.value );
779             return true;
780         },
781         true
782     );
783
784     // 'input' event catches the box value after the keypress
785     box.addEventListener(
786         'input', 
787         function () {
788             if (element.nodeKind() == 'attribute') element[0]=box.value;
789             else element.setChildren( box.value );
790             return true;
791         },
792         true
793     );
794
795     box.addEventListener(
796         'keyup', 
797         function () {
798             if (element.localName() == 'controlfield')
799                 eval('fillFixedFields();');
800         },
801         true
802     );
803
804     return box;
805 }
806
807 function toggleFFE () {
808     var grid = document.getElementById('leaderGrid');
809     if (grid.hidden) {
810         grid.hidden = false;
811     } else {
812         grid.hidden = true;
813     }
814     return true;
815 }
816
817 function changeFFEditor (type) {
818     var grid = document.getElementById('leaderGrid');
819     grid.setAttribute('type',type);
820     document.getElementById('recordTypeLabel').setAttribute('value',type);
821
822     // Hide FFEditor rows that we don't need for our current type
823     // If all of the labels for a given row do not include our
824     // desired type in their set attribute, we can hide that row
825     dojo.query('rows', grid).query('row').forEach(function(node, index, arr) {
826         if (dojo.query('label[set~=' + type + ']', node).length == 0) {
827             node.hidden = true;
828         }
829     });
830
831 }
832
833 function fillFixedFields () {
834     try {
835             var grid = document.getElementById('leaderGrid');
836             var marc_rec = new MARC.Record ({ delimiter : '$', marcxml : xml_record.toXMLString() });
837
838             var list = [];
839             var pre_list = grid.getElementsByTagName('label');
840             for (var i in pre_list) {
841                 if ( pre_list[i].getAttribute && pre_list[i].getAttribute('set').indexOf(grid.getAttribute('type')) > -1 ) {
842                     list.push( pre_list[i] );
843                 }
844             }
845
846             for (var i in list) {
847                 var name = list[i].getAttribute('name');
848                 var value = marc_rec.extractFixedField(name, true);
849
850                 if (value === null) continue;
851
852                 list[i].nextSibling.value = value;
853             }
854
855             return true;
856     } catch(E) {
857         alert('FIXME, MARC Editor, fillFixedFields: ' + E);
858     }
859 }
860
861 function updateFixedFields (element) {
862     var grid = document.getElementById('leaderGrid');
863     var recGrid = document.getElementById('recGrid');
864     var new_value = element.value;
865     // Don't take focus away/adjust the record on partial changes
866     var length = element.getAttribute('maxlength');
867     if(new_value.length < length) return true;
868
869     var marc_rec = new MARC.Record ({ delimiter : '$', marcxml : xml_record.toXMLString() });
870     marc_rec.setFixedField(element.getAttribute('name'), new_value);
871
872     var xml_string = marc_rec.toXmlString();
873     xml_record = new XML( xml_string );
874     if (xml_record..record[0]) xml_record = xml_record..record[0];
875     loadRecord();
876     // Put the cursor back to the current fixed field
877     element.select();
878
879     return true;
880 }
881
882 function marcLeader (leader) {
883     var row = createRow(
884         { class : 'marcLeaderRow',
885           tag : 'ldr' },
886         createLabel(
887             { value : 'LDR',
888               class : 'marcTag',
889               tooltiptext : $('catStrings').getString('staff.cat.marcedit.marcTag.LDR.label') } ),
890         createLabel(
891             { value : '',
892               class : 'marcInd1' } ),
893         createLabel(
894             { value : '',
895               class : 'marcInd2' } ),
896         createLabel(
897             { value : leader.text(),
898               class : 'marcLeader' } )
899     );
900
901     return row;
902 }
903
904 function marcControlfield (field) {
905     tagname = field.@tag.toString().substr(2);
906     var row;
907     if (tagname == '1' || tagname == '3' || tagname == '6' || tagname == '7' || tagname == '8') {
908         row = createRow(
909             { class : 'marcControlfieldRow',
910               tag : '_' + tagname },
911             createLabel(
912                 { value : field.@tag,
913                   class : 'marcTag',
914                   context : 'tags_popup',
915                   onmouseover : 'getTooltip(this, "tag");',
916                   tooltipid : 'tag' + field.@tag } ),
917             createLabel(
918                 { value : field.@ind1,
919                   class : 'marcInd1',
920                   onmouseover : 'getTooltip(this, "ind1");',
921                   tooltipid : 'tag' + field.@tag + 'ind1val' + field.@ind1 } ),
922             createLabel(
923                 { value : field.@ind2,
924                   class : 'marcInd2',
925                   onmouseover : 'getTooltip(this, "ind2");',
926                   tooltipid : 'tag' + field.@tag + 'ind2val' + field.@ind2 } ),
927             createMARCTextbox(
928                 field,
929                 { value : field.text(),
930                   class : 'plain marcEditableControlfield',
931                   name : 'CONTROL' + tagname,
932                   context : 'clipboard',
933                   size : 50,
934                   maxlength : 50 } )
935             );
936     } else {
937         row = createRow(
938             { class : 'marcControlfieldRow',
939               tag : '_' + tagname },
940             createLabel(
941                 { value : field.@tag,
942                   class : 'marcTag',
943                   onmouseover : 'getTooltip(this, "tag");',
944                   tooltipid : 'tag' + field.@tag } ),
945             createLabel(
946                 { value : field.@ind1,
947                   class : 'marcInd1',
948                   onmouseover : 'getTooltip(this, "ind1");',
949                   tooltipid : 'tag' + field.@tag + 'ind1val' + field.@ind1 } ),
950             createLabel(
951                 { value : field.@ind2,
952                   class : 'marcInd2',
953                   onmouseover : 'getTooltip(this, "ind2");',
954                   tooltipid : 'tag' + field.@tag + 'ind2val' + field.@ind2 } ),
955             createLabel(
956                 { value : field.text(),
957                   class : 'marcControlfield' } )
958         );
959     }
960
961     return row;
962 }
963
964 function stackSubfields(checkbox) {
965     var list = document.getElementsByAttribute('name','sf_box');
966
967     var o = 'vertical';
968     if (!checkbox.checked) o = 'horizontal';
969     
970     for (var i = 0; i < list.length; i++) {
971         if (list[i]) list[i].setAttribute('orient',o);
972     }
973 }
974
975 function fastItemAdd_toggle(checkbox) {
976     var x = document.getElementById('fastItemAdd_textboxes');
977     if (checkbox.checked) {
978         x.hidden = false;
979         document.getElementById('fastItemAdd_callnumber').focus();
980         document.getElementById('fastItemAdd_callnumber').select();
981     } else {
982         x.hidden = true;
983     }
984 }
985
986 function fastItemAdd_attempt(doc_id) {
987     try {
988         if (typeof window.xulG.fast_add_item != 'function') { return; }
989         if (!document.getElementById('fastItemAdd_checkbox').checked) { return; }
990         if (!document.getElementById('fastItemAdd_callnumber').value) { return; }
991         if (!document.getElementById('fastItemAdd_barcode').value) { return; }
992         window.xulG.fast_add_item( doc_id, document.getElementById('fastItemAdd_callnumber').value, document.getElementById('fastItemAdd_barcode').value );
993         document.getElementById('fastItemAdd_barcode').value = '';
994         return true;
995     } catch(E) {
996         alert('fastItemAdd_attempt: ' + E);
997     }
998 }
999
1000 function save_attempt(xml_string) {
1001     try {
1002         var result = window.xulG.save.func( xml_string );
1003         // I'd prefer to pass on_complete on through to fast_item_add,
1004         // but with the way these window scopes get destroyed with
1005         // tab replacement, maybe not a good idea
1006         var replace_on_complete = false;
1007         if (result) {
1008             oils_unlock_page();
1009             if (result.id) {
1010                 replace_on_complete = fastItemAdd_attempt(result.id);
1011             }
1012             if (!replace_on_complete && typeof result.on_complete == 'function') {
1013                 result.on_complete();
1014             }
1015         }
1016     } catch(E) {
1017         alert('save_attempt: ' + E);
1018     }
1019 }
1020
1021 function marcDatafield (field) {
1022     var row = createRow(
1023         { class : 'marcDatafieldRow' },
1024         createMARCTextbox(
1025             field.@tag,
1026             { value : field.@tag,
1027               class : 'plain marcTag',
1028               name : 'marcTag',
1029               context : 'tags_popup',
1030               oninput : 'if (this.value.length == 3) { this.nextSibling.focus(); }',
1031               size : 3,
1032               maxlength : 3,
1033               onmouseover : 'current_focus = this; getTooltip(this, "tag");' } ),
1034         createMARCTextbox(
1035             field.@ind1,
1036             { value : field.@ind1,
1037               class : 'plain marcInd1',
1038               name : 'marcInd1',
1039               oninput : 'if (this.value.length == 1) { this.nextSibling.focus(); }',
1040               size : 1,
1041               maxlength : 1,
1042               onmouseover : 'current_focus = this; getContextMenu(this, "ind1"); getTooltip(this, "ind1");',
1043               oncontextmenu : 'getContextMenu(this, "ind1");' } ),
1044         createMARCTextbox(
1045             field.@ind2,
1046             { value : field.@ind2,
1047               class : 'plain marcInd2',
1048               name : 'marcInd2',
1049               oninput : 'if (this.value.length == 1) { this.nextSibling.firstChild.firstChild.focus(); }',
1050               size : 1,
1051               maxlength : 1,
1052               onmouseover : 'current_focus = this; getContextMenu(this, "ind2"); getTooltip(this, "ind2");',
1053               oncontextmenu : 'getContextMenu(this, "ind2");' } ),
1054         createHbox({ name : 'sf_box' })
1055     );
1056
1057     if (!current_focus && field.@tag == '') current_focus = row.childNodes[0];
1058     if (!current_focus && field.@ind1 == '') current_focus = row.childNodes[1];
1059     if (!current_focus && field.@ind2 == '') current_focus = row.childNodes[2];
1060
1061     var sf_box = row.lastChild;
1062     if (document.getElementById('stackSubfields').checked)
1063         sf_box.setAttribute('orient','vertical');
1064
1065     sf_box.addEventListener(
1066         'click',
1067         function (e) {
1068             if (sf_box === e.target) {
1069                 sf_box.lastChild.lastChild.focus();
1070             } else if (e.target.parentNode === sf_box) {
1071                 e.target.lastChild.focus();
1072             }
1073         },
1074         false
1075     );
1076
1077
1078     for (var i in field.subfield) {
1079         var sf = field.subfield[i];
1080         sf_box.appendChild(
1081             marcSubfield(sf)
1082         );
1083
1084         dojo.query('.marcSubfield', sf_box).forEach(wrap_long_fields);
1085
1086         if (sf.@code == '' && (!current_focus || current_focus.className.match(/Ind/)))
1087             current_focus = sf_box.lastChild.childNodes[1];
1088     }
1089
1090     return row;
1091 }
1092
1093 function marcSubfield (sf) {            
1094     return createHbox(
1095         { class : 'marcSubfieldBox' },
1096         createLabel(
1097             { value : "\u2021",
1098               class : 'plain marcSubfieldDelimiter',
1099               onmouseover : 'getTooltip(this.nextSibling, "subfield");',
1100               oncontextmenu : 'getContextMenu(this.nextSibling, "subfield");',
1101                 //onclick : 'this.nextSibling.focus();',
1102                 onfocus : 'this.nextSibling.focus();',
1103               size : 2 } ),
1104         createMARCTextbox(
1105             sf.@code,
1106             { value : sf.@code,
1107               class : 'plain marcSubfieldCode',
1108               align: 'start',
1109               name : 'marcSubfieldCode',
1110               onmouseover : 'current_focus = this; getContextMenu(this, "subfield"); getTooltip(this, "subfield");',
1111               oncontextmenu : 'getContextMenu(this, "subfield");',
1112               oninput : 'if (this.value.length == 1) { this.nextSibling.focus(); }',
1113               size : 2,
1114               maxlength : 1 } ),
1115         createMARCTextbox(
1116             sf,
1117             { value : sf.text(),
1118               name : sf.parent().@tag + ':' + sf.@code,
1119               class : 'plain marcSubfield', 
1120               align: 'start',
1121               onmouseover : 'getTooltip(this, "subfield");',
1122               contextmenu : function (event) { getAuthorityContextMenu(event.target, sf) },
1123               size : new String(sf.text()).length + 2,
1124               oninput : "this.setAttribute('size', this.value.length + 2);"
1125             } )
1126     );
1127 }
1128
1129 function loadRecord() {
1130     try {
1131             var grid_rows = document.getElementById('recGrid').lastChild;
1132
1133             while (grid_rows.firstChild) grid_rows.removeChild(grid_rows.firstChild);
1134
1135             grid_rows.appendChild( marcLeader( xml_record.leader ) );
1136
1137             for (var i in xml_record.controlfield) {
1138                 grid_rows.appendChild( marcControlfield( xml_record.controlfield[i] ) );
1139             }
1140
1141             for (var i in xml_record.datafield) {
1142                 grid_rows.appendChild( marcDatafield( xml_record.datafield[i] ) );
1143             }
1144
1145             grid_rows.getElementsByAttribute('class','marcDatafieldRow')[0].firstChild.focus();
1146
1147             var marc_rec = new MARC.Record ({ delimiter : '$', marcxml : xml_record.toXMLString() });
1148             changeFFEditor(marc_rec.recordType());
1149             fillFixedFields();
1150     } catch(E) {
1151         alert('FIXME, MARC Editor, loadRecord: ' + E);
1152     }
1153 }
1154
1155
1156 function genToolTips () {
1157     for (var i in bib_data.field) {
1158         var f = bib_data.field[i];
1159     
1160         tag_menu.appendChild(
1161             createMenuitem(
1162                 { label : f.@tag,
1163                   oncommand : 
1164                       'current_focus.value = "' + f.@tag + '";' +
1165                     'var e = document.createEvent("MutationEvents");' +
1166                     'e.initMutationEvent("change",1,1,null,0,0,0,0);' +
1167                     'current_focus.inputField.dispatchEvent(e);',
1168                   disabled : f.@tag < '010' ? "true" : "false",
1169                   tooltiptext : f.description }
1170             )
1171         );
1172     
1173         var i1_popup = createMenuPopup({position : 'after_start', id : 't' + f.@tag + 'i1' });
1174         context_menus.appendChild( i1_popup );
1175     
1176         var i2_popup = createMenuPopup({position : 'after_start', id : 't' + f.@tag + 'i2' });
1177         context_menus.appendChild( i2_popup );
1178     
1179         var sf_popup = createMenuPopup({position : 'after_start', id : 't' + f.@tag + 'sf' });
1180         context_menus.appendChild( sf_popup );
1181     
1182         tooltip_hash['tag' + f.@tag] = f.description;
1183         for (var j in f.indicator) {
1184             var ind = f.indicator[j];
1185             tooltip_hash['tag' + f.@tag + 'ind' + ind.@position + 'val' + ind.@value] = ind.description;
1186     
1187             if (ind.@position == 1) {
1188                 i1_popup.appendChild(
1189                     createMenuitem(
1190                         { label : ind.@value,
1191                           oncommand : 
1192                               'current_focus.value = "' + ind.@value + '";' +
1193                             'var e = document.createEvent("MutationEvents");' +
1194                             'e.initMutationEvent("change",1,1,null,0,0,0,0);' +
1195                             'current_focus.inputField.dispatchEvent(e);',
1196                           tooltiptext : ind.description }
1197                     )
1198                 );
1199             }
1200     
1201             if (ind.@position == 2) {
1202                 i2_popup.appendChild(
1203                     createMenuitem(
1204                         { label : ind.@value,
1205                           oncommand : 
1206                               'current_focus.value = "' + ind.@value + '";' +
1207                             'var e = document.createEvent("MutationEvents");' +
1208                             'e.initMutationEvent("change",1,1,null,0,0,0,0);' +
1209                             'current_focus.inputField.dispatchEvent(e);',
1210                           tooltiptext : ind.description }
1211                     )
1212                 );
1213             }
1214         }
1215     
1216         for (var j in f.subfield) {
1217             var sf = f.subfield[j];
1218             tooltip_hash['tag' + f.@tag + 'sf' + sf.@code] = sf.description;
1219     
1220             sf_popup.appendChild(
1221                 createMenuitem(
1222                     { label : sf.@code,
1223                       oncommand : 
1224                           'current_focus.value = "' + sf.@code + '";' +
1225                         'var e = document.createEvent("MutationEvents");' +
1226                         'e.initMutationEvent("change",1,1,null,0,0,0,0);' +
1227                         'current_focus.inputField.dispatchEvent(e);',
1228                       tooltiptext : sf.description
1229                     }
1230                 )
1231             );
1232         }
1233     }
1234 }
1235
1236 function getTooltip (target, type) {
1237
1238     var tt = '';
1239     if (type == 'subfield')
1240         tt = 'tag' + target.parentNode.parentNode.parentNode.firstChild.value + 'sf' + target.parentNode.childNodes[1].value;
1241
1242     if (type == 'ind1')
1243         tt = 'tag' + target.parentNode.firstChild.value + 'ind1val' + target.value;
1244
1245     if (type == 'ind2')
1246         tt = 'tag' + target.parentNode.firstChild.value + 'ind2val' + target.value;
1247
1248     if (type == 'tag')
1249         tt = 'tag' + target.parentNode.firstChild.value;
1250
1251     if (!document.getElementById( tt )) {
1252         p.appendChild(
1253             createTooltip(
1254                 { id : tt,
1255                   flex : "1",
1256                   orient : 'vertical',
1257                   onpopupshown : 'this.width = this.firstChild.boxObject.width + 10; this.height = this.firstChild.boxObject.height + 10;',
1258                   class : 'tooltip' },
1259                 createDescription({}, document.createTextNode( tooltip_hash[tt] ) )
1260             )
1261         );
1262     }
1263
1264     target.tooltip = tt;
1265     return true;
1266 }
1267
1268 function getContextMenu (target, type) {
1269
1270     var tt = '';
1271     if (type == 'subfield')
1272         tt = 't' + target.parentNode.parentNode.parentNode.firstChild.value + 'sf';
1273
1274     if (type == 'ind1')
1275         tt = 't' + target.parentNode.firstChild.value + 'i1';
1276
1277     if (type == 'ind2')
1278         tt = 't' + target.parentNode.firstChild.value + 'i2';
1279
1280     target.setAttribute('context', tt);
1281     return true;
1282 }
1283
1284 var control_map = {
1285     100 : {
1286         'a' : { 100 : 'a' },
1287         'd' : { 100 : 'd' },
1288         'e' : { 100 : 'e' },
1289         'q' : { 100 : 'q' }
1290     },
1291     110 : {
1292         'a' : { 110 : 'a' },
1293         'd' : { 110 : 'd' }
1294     },
1295     111 : {
1296         'a' : { 111 : 'a' },
1297         'd' : { 111 : 'd' }
1298     },
1299     130 : {
1300         'a' : { 130 : 'a' },
1301         'd' : { 130 : 'd' }
1302     },
1303     240 : {
1304         'a' : { 130 : 'a' },
1305         'd' : { 130 : 'd' }
1306     },
1307     400 : {
1308         'a' : { 100 : 'a' },
1309         'd' : { 100 : 'd' }
1310     },
1311     410 : {
1312         'a' : { 110 : 'a' },
1313         'd' : { 110 : 'd' }
1314     },
1315     411 : {
1316         'a' : { 111 : 'a' },
1317         'd' : { 111 : 'd' }
1318     },
1319     440 : {
1320         'a' : { 130 : 'a' },
1321         'n' : { 130 : 'n' },
1322         'p' : { 130 : 'p' }
1323     },
1324     700 : {
1325         'a' : { 100 : 'a' },
1326         'd' : { 100 : 'd' },
1327         'q' : { 100 : 'q' },
1328         't' : { 100 : 't' }
1329     },
1330     710 : {
1331         'a' : { 110 : 'a' },
1332         'd' : { 110 : 'd' }
1333     },
1334     711 : {
1335         'a' : { 111 : 'a' },
1336         'c' : { 111 : 'c' },
1337         'd' : { 111 : 'd' }
1338     },
1339     730 : {
1340         'a' : { 130 : 'a' },
1341         'd' : { 130 : 'd' }
1342     },
1343     800 : {
1344         'a' : { 100 : 'a' },
1345         'd' : { 100 : 'd' }
1346     },
1347     810 : {
1348         'a' : { 110 : 'a' },
1349         'd' : { 110 : 'd' }
1350     },
1351     811 : {
1352         'a' : { 111 : 'a' },
1353         'd' : { 111 : 'd' }
1354     },
1355     830 : {
1356         'a' : { 130 : 'a' },
1357         'd' : { 130 : 'd' }
1358     },
1359     600 : {
1360         'a' : { 100 : 'a' },
1361         'd' : { 100 : 'd' },
1362         'q' : { 100 : 'q' },
1363         't' : { 100 : 't' },
1364         'v' : { 180 : 'v',
1365             100 : 'v',
1366             181 : 'v',
1367             182 : 'v',
1368             185 : 'v'
1369         },
1370         'x' : { 180 : 'x',
1371             100 : 'x',
1372             181 : 'x',
1373             182 : 'x',
1374             185 : 'x'
1375         },
1376         'y' : { 180 : 'y',
1377             100 : 'y',
1378             181 : 'y',
1379             182 : 'y',
1380             185 : 'y'
1381         },
1382         'z' : { 180 : 'z',
1383             100 : 'z',
1384             181 : 'z',
1385             182 : 'z',
1386             185 : 'z'
1387         }
1388     },
1389     610 : {
1390         'a' : { 110 : 'a' },
1391         'd' : { 110 : 'd' },
1392         't' : { 110 : 't' },
1393         'v' : { 180 : 'v',
1394             110 : 'v',
1395             181 : 'v',
1396             182 : 'v',
1397             185 : 'v'
1398         },
1399         'x' : { 180 : 'x',
1400             110 : 'x',
1401             181 : 'x',
1402             182 : 'x',
1403             185 : 'x'
1404         },
1405         'y' : { 180 : 'y',
1406             110 : 'y',
1407             181 : 'y',
1408             182 : 'y',
1409             185 : 'y'
1410         },
1411         'z' : { 180 : 'z',
1412             110 : 'z',
1413             181 : 'z',
1414             182 : 'z',
1415             185 : 'z'
1416         }
1417     },
1418     611 : {
1419         'a' : { 111 : 'a' },
1420         'd' : { 111 : 'd' },
1421         't' : { 111 : 't' },
1422         'v' : { 180 : 'v',
1423             111 : 'v',
1424             181 : 'v',
1425             182 : 'v',
1426             185 : 'v'
1427         },
1428         'x' : { 180 : 'x',
1429             111 : 'x',
1430             181 : 'x',
1431             182 : 'x',
1432             185 : 'x'
1433         },
1434         'y' : { 180 : 'y',
1435             111 : 'y',
1436             181 : 'y',
1437             182 : 'y',
1438             185 : 'y'
1439         },
1440         'z' : { 180 : 'z',
1441             111 : 'z',
1442             181 : 'z',
1443             182 : 'z',
1444             185 : 'z'
1445         }
1446     },
1447     630 : {
1448         'a' : { 130 : 'a' },
1449         'd' : { 130 : 'd' }
1450     },
1451     648 : {
1452         'a' : { 148 : 'a' },
1453         'v' : { 148 : 'v' },
1454         'x' : { 148 : 'x' },
1455         'y' : { 148 : 'y' },
1456         'z' : { 148 : 'z' }
1457     },
1458     650 : {
1459         'a' : { 150 : 'a' },
1460         'b' : { 150 : 'b' },
1461         'v' : { 180 : 'v',
1462             150 : 'v',
1463             181 : 'v',
1464             182 : 'v',
1465             185 : 'v'
1466         },
1467         'x' : { 180 : 'x',
1468             150 : 'x',
1469             181 : 'x',
1470             182 : 'x',
1471             185 : 'x'
1472         },
1473         'y' : { 180 : 'y',
1474             150 : 'y',
1475             181 : 'y',
1476             182 : 'y',
1477             185 : 'y'
1478         },
1479         'z' : { 180 : 'z',
1480             150 : 'z',
1481             181 : 'z',
1482             182 : 'z',
1483             185 : 'z'
1484         }
1485     },
1486     651 : {
1487         'a' : { 151 : 'a' },
1488         'v' : { 180 : 'v',
1489             151 : 'v',
1490             181 : 'v',
1491             182 : 'v',
1492             185 : 'v'
1493         },
1494         'x' : { 180 : 'x',
1495             151 : 'x',
1496             181 : 'x',
1497             182 : 'x',
1498             185 : 'x'
1499         },
1500         'y' : { 180 : 'y',
1501             151 : 'y',
1502             181 : 'y',
1503             182 : 'y',
1504             185 : 'y'
1505         },
1506         'z' : { 180 : 'z',
1507             151 : 'z',
1508             181 : 'z',
1509             182 : 'z',
1510             185 : 'z'
1511         }
1512     },
1513     655 : {
1514         'a' : { 155 : 'a' },
1515         'v' : { 180 : 'v',
1516             155 : 'v',
1517             181 : 'v',
1518             182 : 'v',
1519             185 : 'v'
1520         },
1521         'x' : { 180 : 'x',
1522             155 : 'x',
1523             181 : 'x',
1524             182 : 'x',
1525             185 : 'x'
1526         },
1527         'y' : { 180 : 'y',
1528             155 : 'y',
1529             181 : 'y',
1530             182 : 'y',
1531             185 : 'y'
1532         },
1533         'z' : { 180 : 'z',
1534             155 : 'z',
1535             181 : 'z',
1536             182 : 'z',
1537             185 : 'z'
1538         }
1539     }
1540 };
1541
1542 function getAuthorityContextMenu (target, sf) {
1543     var menu_id = sf.parent().@tag + ':' + sf.@code + '-authority-context-' + sf;
1544
1545     var page = 0;
1546     var old = dojo.byId( menu_id );
1547     if (old) {
1548         page = auth_pages[menu_id];
1549         old.parentNode.removeChild(old);
1550     } else {
1551         auth_pages[menu_id] = 0;
1552     }
1553
1554     var sf_popup = createMenuPopup({ id : menu_id, flex : 1 });
1555
1556     sf_popup.addEventListener("popuphiding", function(event) {
1557         if (show_auth_menu) {
1558             show_auth_menu = false;
1559             getAuthorityContextMenu(target, sf);
1560             dojo.byId(menu_id).openPopup();
1561         }  
1562     }, false);
1563
1564     context_menus.appendChild( sf_popup );
1565
1566     var found_acs = [];
1567     dojo.forEach( acs.controlSetList(), function (acs_id) {
1568         if (acs.controlSet(acs_id).control_map[sf.parent().@tag]) found_acs.push(acs_id);
1569     });
1570
1571     if (!found_acs.length) {
1572         sf_popup.appendChild(createLabel( { value : $('catStrings').getString('staff.cat.marcedit.not_authority_field.label') } ) );
1573         target.setAttribute('context', 'clipboard');
1574         return false;
1575     }
1576
1577     if (sf.toString().replace(/\s*/, '')) {
1578         return browseAuthority(sf_popup, menu_id, target, sf, 20, page);
1579     }
1580
1581     return true;
1582 }
1583
1584 /* Apply the complete 1xx */
1585 function applyFullAuthority ( target, ui_sf, e4x_sf ) {
1586     var new_vals = dojo.query('*[tag^="1"]', target);
1587     return applyAuthority( target, ui_sf, e4x_sf, new_vals );
1588 }
1589
1590 function applySelectedAuthority ( target, ui_sf, e4x_sf ) {
1591     var new_vals = target.getElementsByAttribute('checked','true');
1592     return applyAuthority( target, ui_sf, e4x_sf, new_vals );
1593 }
1594
1595 function applyAuthority ( target, ui_sf, e4x_sf, new_vals ) {
1596     var field = e4x_sf.parent();
1597
1598     for (var i = 0; i < new_vals.length; i++) {
1599
1600         var sf_list = field.subfield;
1601         for (var j in sf_list) {
1602
1603             if (sf_list[j].@code == new_vals[i].getAttribute('subfield')) {
1604                 sf_list[j] = new_vals[i].getAttribute('value');
1605                 new_vals[i].setAttribute('subfield','');
1606                 break;
1607             }
1608         }
1609     }
1610
1611     for (var i = 0; i < new_vals.length; i++) {
1612
1613         /* indicators for the authority datafield are carried over in the main entry linking subfield */
1614         if (new_vals[i].getAttribute('subfield') == '0') {
1615             field.@ind1 = new_vals[i].getAttribute('ind1');
1616             field.@ind2 = new_vals[i].getAttribute('ind2');
1617         }
1618
1619         if (!new_vals[i].getAttribute('subfield')) continue;
1620
1621         var val = new_vals[i].getAttribute('value');
1622
1623         var sf = <subfield code="" xmlns="http://www.loc.gov/MARC21/slim">{val}</subfield>;
1624         sf.@code = new_vals[i].getAttribute('subfield');
1625
1626         field.insertChildAfter(field.subfield[field.subfield.length() - 1], sf);
1627     }
1628
1629     var row = marcDatafield( field );
1630
1631     var node = ui_sf;
1632     while (node.nodeName != 'row') {
1633         node = node.parentNode;
1634     }
1635
1636     node.parentNode.replaceChild( row, node );
1637     return true;
1638 }
1639
1640 function validateAuthority (button) {
1641     var grid = document.getElementById('recGrid');
1642     var label = button.getAttribute('label');
1643
1644     //loop over rows
1645     var rows = grid.lastChild.childNodes;
1646     for (var i = 0; i < rows.length; i++) {
1647         var row = rows[i];
1648         var tag = row.firstChild;
1649
1650         var done = false;
1651         dojo.forEach(acs.controlSetList(), function (acs_id) {
1652             if (done) return;
1653             var control_map = acs.controlSet(acs_id).control_map;
1654     
1655             if (!control_map[tag.value]) return;
1656             button.setAttribute('label', label + ' - ' + tag.value);
1657     
1658             var ind1 = tag.nextSibling;
1659             var ind2 = ind1.nextSibling;
1660             var subfields = ind2.nextSibling.childNodes;
1661     
1662             var sf_list = [];
1663             for (var j = 0; j < subfields.length; j++) {
1664                 var sf = subfields[j];
1665                 sf_list.push( [ sf.childNodes[1].value, sf.childNodes[2].value ] );
1666             }
1667
1668             var matches = acs.findMatchingAuthorities(
1669                 new MARC.Field({
1670                     'tag'       : tag.value,
1671                     'subfields' : sf_list
1672                 })
1673             );
1674
1675             // matches = [ { "$csetId" : [ ... ] } ]
1676
1677             var found = false;
1678             if (matches[0]) { // probably set
1679                 for (var cset in matches[0]) {
1680                     var arr = matches[0][cset];
1681                     if (arr.length) {
1682                         // protect against errant empty string values
1683                         if (arr.length == 1 && arr[0] == '')
1684                             continue;
1685                         found = true;
1686                         break;
1687                     }
1688                 }
1689             }
1690
1691     
1692             // XXX If adt, etc should be validated separately from vxz, etc then move this up into the above for loop
1693             for (var j = 0; j < subfields.length; j++) {
1694                 var sf = subfields[j];
1695                 if (!found) {
1696                     dojo.removeClass(sf.childNodes[2], 'marcValidated');
1697                     dojo.addClass(sf.childNodes[2], 'marcUnvalidated');
1698                 } else {
1699                     dojo.removeClass(sf.childNodes[2], 'marcUnvalidated');
1700                     dojo.addClass(sf.childNodes[2], 'marcValidated');
1701                 }
1702             }
1703
1704             if (found) done = true;
1705         });
1706     }
1707
1708     button.setAttribute('label', label);
1709
1710     return true;
1711 }
1712
1713
1714 /*
1715 function validateBibField (tags, searches) {
1716     var url = "/gateway?input_format=json&format=xml&service=open-ils.search&method=open-ils.search.authority.validate.tag";
1717     url += '&param="tags"&param=' + js2JSON(tags);
1718     url += '&param="searches"&param=' + js2JSON(searches);
1719
1720
1721     var req = new XMLHttpRequest();
1722     req.open('GET',url,false);
1723     req.send(null);
1724
1725     return req;
1726
1727 }
1728 */
1729
1730 function searchAuthority (term, tag, sf, limit) {
1731     var url = "/gateway?input_format=json&format=xml&service=open-ils.search&method=open-ils.search.authority.fts";
1732     url += '&param="term"&param="' + term + '"';
1733     url += '&param="limit"&param=' + limit;
1734     url += '&param="tag"&param=' + tag;
1735     url += '&param="subfield"&param="' + sf + '"';
1736
1737
1738     var req = new XMLHttpRequest();
1739     req.open('GET',url,false);
1740     req.send(null);
1741
1742     return req;
1743
1744 }
1745
1746 /* TODO new authority browse support for context sets, and use that here */
1747 function browseAuthority (sf_popup, menu_id, target, sf, limit, page) {
1748     dojo.require('dojox.xml.parser');
1749
1750     // map tag + subfield to the appropriate authority browse axis:
1751     // currently authority.author, authority.subject, authority.title, authority.topic
1752     // based on mappings in OpenILS::Application::SuperCat, though Authority Control
1753     // Sets will change that
1754
1755     var axis_list = acs.bibFieldBrowseAxes( sf.parent().@tag.toString() );
1756
1757     // No matching tag means no authorities to search - shortcut
1758     if (axis_list.length == 0) {
1759         target.setAttribute('context', 'clipboard');
1760         return false;
1761     }
1762
1763     var type = 'authority.' + axis_list[0]; // Just take the first for now
1764                                             // TODO support multiple axes ... loop?
1765     if (!limit) {
1766         limit = 10;
1767     }
1768
1769     if (!page) {
1770         page = 0;
1771     }
1772
1773     var sf_string = '';
1774     var sf_list = sf.parent().subfield;
1775     for ( var i in sf_list) {
1776         sf_string += sf_list[i].toString() + ' ';
1777         if (sf_list[i] === sf) break;
1778     }
1779
1780     var url = '/opac/extras/browse/marcxml/'
1781         + type + '.refs'
1782         + '/1' // OU - currently unscoped
1783         + '/' + sf_string
1784         + '/' + page
1785         + '/' + limit
1786     ;
1787
1788     // would be good to carve this out into a separate function
1789     dojo.xhrGet({"url":url, "sync": true, "preventCache": true, "handleAs":"xml", "load": function(records) {
1790         var create_menu = createMenu({ label: $('catStrings').getString('staff.cat.marcedit.create_authority.label')});
1791
1792         var cm_popup = create_menu.appendChild(
1793             createMenuPopup()
1794         );
1795
1796         cm_popup.appendChild(
1797             createMenuitem({ label : $('catStrings').getString('staff.cat.marcedit.create_authority_now.label'),
1798                 command : function() { 
1799                     // Call middle-layer function to create and save the new authority
1800                     var source_f = summarizeField(sf);
1801                     var new_auth = fieldmapper.standardRequest(
1802                         ["open-ils.cat", "open-ils.cat.authority.record.create_from_bib"],
1803                         [source_f, xulG.marc_control_number_identifier, ses()]
1804                     );
1805                     if (new_auth && new_auth.id()) {
1806                         addNewAuthorityID(new_auth, sf, target);
1807                     }
1808                 }
1809             })
1810         );
1811
1812         cm_popup.appendChild(
1813             createMenuitem({ label : $('catStrings').getString('staff.cat.marcedit.create_authority_edit.label'),
1814                 command : function() { 
1815                     // Generate the new authority by calling the new middle-layer
1816                     // function (a non-saving variant), then display in another
1817                     // MARC editor
1818                     var source_f = summarizeField(sf);
1819                     var authtoken = ses();
1820                     dojo.require('openils.PermaCrud');
1821                     var pcrud = new openils.PermaCrud({"authtoken": authtoken});
1822                     var rec = fieldmapper.standardRequest(
1823                         ["open-ils.cat", "open-ils.cat.authority.record.create_from_bib.readonly"],
1824                         { "params": [source_f, xulG.marc_control_number_identifier] }
1825                     );
1826                     loadMarcEditor(pcrud, rec, target, sf);
1827                 }
1828             })
1829         );
1830
1831         sf_popup.appendChild(create_menu);
1832         sf_popup.appendChild( createComplexXULElement( 'menuseparator' ) );
1833
1834         // append "Previous page" results browser
1835         sf_popup.appendChild(
1836             createMenuitem({ label : $('catStrings').getString('staff.cat.marcedit.previous_page.label'),
1837                 command : function(event) { 
1838                     auth_pages[menu_id] -= 1;
1839                     show_auth_menu = true;
1840                 }
1841             })
1842         );
1843         sf_popup.appendChild( createComplexXULElement( 'menuseparator' ) );
1844
1845         dojo.query('record', records).forEach(function(record) {
1846             var main_text = '';
1847             var see_from = [];
1848             var see_also = [];
1849             var auth_id = dojox.xml.parser.textContent(dojo.query('datafield[tag="901"]', record).query('subfield[code="c"]')[0]);
1850             var auth_org = '';
1851             if (dojo.query('controlfield[tag="003"]', record).length > 0) {
1852                 auth_org = dojox.xml.parser.textContent(dojo.query('controlfield[tag="003"]', record)[0]);
1853             }
1854
1855             // Grab the fields with tags beginning with 1 (main entries) and iterate through the subfields
1856             dojo.query('datafield[tag^="1"]', record).forEach(function(field) {
1857                 dojo.query('subfield', field).forEach(function(subfield) {
1858                     if (main_text) {
1859                         main_text += ' / ';
1860                     }
1861                     main_text += dojox.xml.parser.textContent(subfield);
1862                 });
1863             });
1864
1865             // Grab the fields with tags beginning with 4 (see from entries) and iterate through the subfields
1866             dojo.query('datafield[tag^="4"]', record).forEach(function(field) {
1867                 var see_text = '';
1868                 dojo.query('subfield', field).forEach(function(subfield) {
1869                     if (see_text) {
1870                         see_text += ' / ';
1871                     }
1872                     see_text += dojox.xml.parser.textContent(subfield);
1873                 });
1874                 see_from.push($('catStrings').getFormattedString('staff.cat.marcedit.authority_see_from', [see_text]));
1875             });
1876
1877             // Grab the fields with tags beginning with 5 (see also entries) and iterate through the subfields
1878             dojo.query('datafield[tag^="5"]', record).forEach(function(field) {
1879                 var see_text = '';
1880                 dojo.query('subfield', field).forEach(function(subfield) {
1881                     if (see_text) {
1882                         see_text += ' / ';
1883                     }
1884                     see_text += dojox.xml.parser.textContent(subfield);
1885                 });
1886                 see_also.push($('catStrings').getFormattedString('staff.cat.marcedit.authority_see_also', [see_text]));
1887             });
1888
1889             buildAuthorityPopup(main_text, record, auth_org, auth_id, sf_popup, target, sf);
1890
1891             dojo.forEach(see_from, function(entry_text) {
1892                 buildAuthorityPopup(entry_text, record, auth_org, auth_id, sf_popup, target, sf, "font-style: italic; margin-left: 2em;");
1893             });
1894
1895             // To-do: instead of launching the standard selector menu, invoke
1896             // a new authority search using the 5XX entry text
1897             dojo.forEach(see_also, function(entry_text) {
1898                 buildAuthorityPopup(entry_text, record, auth_org, auth_id, sf_popup, target, sf, "font-style: italic; margin-left: 2em;");
1899             });
1900
1901         });
1902
1903         if (sf_popup.childNodes.length == 0) {
1904             sf_popup.appendChild(createLabel( { value : $('catStrings').getString('staff.cat.marcedit.no_authority_match.label') } ) );
1905         } else {
1906             // append "Next page" results browser
1907             sf_popup.appendChild( createComplexXULElement( 'menuseparator' ) );
1908             sf_popup.appendChild(
1909                 createMenuitem({ label : $('catStrings').getString('staff.cat.marcedit.next_page.label'),
1910                     command : function(event) { 
1911                         auth_pages[menu_id] += 1;
1912                         show_auth_menu = true;
1913                     }
1914                 })
1915             );
1916         }
1917
1918         target.setAttribute('context', menu_id);
1919         return true;
1920     }});
1921
1922 }
1923
1924 function buildAuthorityPopup (entry_text, record, auth_org, auth_id, sf_popup, target, sf, style) {
1925     var grid = dojo.query('[name="authority-marc-template"]')[0].cloneNode(true);
1926     grid.setAttribute('name','-none-');
1927     grid.setAttribute('style','overflow:scroll');
1928
1929     var submenu = createMenu( { "label": entry_text } );
1930
1931     var popup = createMenuPopup({ "flex": "1" });
1932     if (style) {
1933         submenu.setAttribute('style', style);
1934         popup.setAttribute('style', 'font-style: normal; margin-left: 0em;');
1935     }
1936     submenu.appendChild(popup);
1937
1938     dojo.query('datafield[tag^="1"]', record).forEach(function(field) {
1939         buildAuthorityPopupSelector(field, grid, auth_org, auth_id);
1940     });
1941     dojo.query('datafield[tag^="4"]', record).forEach(function(field) {
1942         buildAuthorityPopupSelector(field, grid, auth_org, auth_id);
1943     });
1944     dojo.query('datafield[tag^="5"]', record).forEach(function(field) {
1945         buildAuthorityPopupSelector(field, grid, auth_org, auth_id);
1946     });
1947
1948     grid.hidden = false;
1949     popup.appendChild( grid );
1950
1951     popup.appendChild(
1952         createMenuitem(
1953             { label : $('catStrings').getString('staff.cat.marcedit.apply_selected.label'),
1954               command : function (event) {
1955                     applySelectedAuthority(event.target.previousSibling, target, sf);
1956                     return true;
1957               }
1958             }
1959         )
1960     );
1961
1962     popup.appendChild( createComplexXULElement( 'menuseparator' ) );
1963
1964     popup.appendChild(
1965         createMenuitem(
1966             { label : $('catStrings').getString('staff.cat.marcedit.apply_full.label'),
1967               command : function (event) {
1968                     applyFullAuthority(event.target.previousSibling.previousSibling.previousSibling, target, sf);
1969                     return true;
1970               }
1971             }
1972         )
1973     );
1974
1975     sf_popup.appendChild( submenu );
1976 }
1977
1978 function buildAuthorityPopupSelector (field, grid, auth_org, auth_id) {
1979     var row = createRow(
1980         { },
1981         createLabel( { "value" : dojo.attr(field, 'tag') } ),
1982         createLabel( { "value" : dojo.attr(field, 'ind1') } ),
1983         createLabel( { "value" : dojo.attr(field, 'ind2') } )
1984     );
1985
1986     var sf_box = createHbox();
1987     dojo.query('subfield', field).forEach(function(subfield) {
1988         sf_box.appendChild(
1989             createCheckbox(
1990                 { "label"    : '\u2021' + dojo.attr(subfield, 'code') + ' ' + dojox.xml.parser.textContent(subfield),
1991                   "subfield" : dojo.attr(subfield, 'code'),
1992                   "tag"      : dojo.attr(field, 'tag'),
1993                   "value"    : dojox.xml.parser.textContent(subfield)
1994                 }
1995             )
1996         );
1997         row.appendChild(sf_box);
1998     });
1999
2000     // Append the authority linking subfield only for main entries
2001     if (dojo.attr(field, 'tag').charAt(0) == '1') {
2002         sf_box.appendChild(
2003             createCheckbox(
2004                 { "label"    : '\u2021' + '0' + ' (' + auth_org + ')' + auth_id,
2005                   "subfield" : '0',
2006                   "tag"      : dojo.attr(field, 'tag'),
2007                   "ind1"     : dojo.attr(field, 'ind1'),
2008                   "ind2"     : dojo.attr(field, 'ind2'),
2009                   "value"    : '(' + auth_org + ')' + auth_id
2010                 }
2011             )
2012         );
2013     }
2014     row.appendChild(sf_box);
2015
2016     grid.lastChild.appendChild(row);
2017 }
2018
2019 function summarizeField(sf) {
2020     var source_f= {
2021         "tag": '',
2022         "ind1": '',
2023         "ind2": '',
2024         "subfields": []
2025     };
2026
2027     source_f.tag = sf.parent().@tag.toString();
2028     source_f.ind1 = sf.parent().@ind1.toString();
2029     source_f.ind2 = sf.parent().@ind2.toString();
2030
2031     var found_acs = [];
2032     dojo.forEach( acs.controlSetList(), function (acs_id) {
2033         if (acs.controlSet(acs_id).control_map[sf.parent().@tag]) found_acs.push(acs_id);
2034     });
2035
2036     var cmap;
2037     if (!found_acs.length) {
2038         return false;
2039     } else {
2040         cmap = acs.controlSet(found_acs[0]).control_map;
2041     }
2042
2043     for (var i = 0; i < sf.parent().subfield.length(); i++) {
2044         var sf_iter = sf.parent().subfield[i];
2045
2046         /* Filter out subfields that are not controlled for this tag */
2047         if (!cmap[source_f.tag][sf_iter.@code.toString()]) {
2048             continue;
2049         }
2050
2051         source_f.subfields.push([sf_iter.@code.toString(), sf_iter.toString()]);
2052     }
2053
2054     return source_f;
2055 }
2056
2057 function buildBibSourceList (authtoken, recId) {
2058     /* TODO: Work out how to set the bib source of the bre that does not yet
2059      * exist - this is specifically in the case of Z39.50 imports. Right now
2060      * we just avoid populating and showing the config.bib_source list
2061      */
2062     if (!recId) {
2063         return false;
2064     }
2065
2066     var bib = xulG.record.bre;
2067
2068     dojo.require('openils.PermaCrud');
2069
2070     // cbsList = the XUL menulist that contains the available bib sources 
2071     var cbsList = dojo.byId('bib-source-list');
2072
2073     // bibSources = an array containing all of the bib source objects
2074     var bibSources = new openils.PermaCrud({"authtoken": authtoken}).retrieveAll('cbs');
2075
2076     // A tad ugly, but gives us the index of the bib source ID in cbsList
2077     var x = 0;
2078     var cbsListArr = [];
2079     dojo.forEach(bibSources, function (item) {
2080         cbsList.appendItem(item.source(), item.id());
2081         cbsListArr[item.id()] = x;
2082         x++;
2083     });
2084
2085     // Show the current value of the bib source for this record
2086     cbsList.selectedIndex = cbsListArr[bib.source()];
2087
2088     // Display the bib source selection widget
2089     dojo.byId('bib-source-list-caption').hidden = false;
2090     dojo.byId('bib-source-list').hidden = false;
2091     dojo.byId('bib-source-list-button').disabled = true;
2092     dojo.byId('bib-source-list-button').hidden = false;
2093 }
2094
2095 // Fired when the "Update Source" button is clicked
2096 // Updates the value of the bib source for the current record
2097 function updateBibSource() {
2098     var authtoken = ses();
2099     var cbs = dojo.byId('bib-source-list').selectedItem.value;
2100     var recId = xulG.record.id;
2101     var pcrud = new openils.PermaCrud({"authtoken": authtoken});
2102     var bib = pcrud.retrieve('bre', recId);
2103     if (bib.source() != cbs) {
2104         bib.source(cbs);
2105         bib.ischanged = true;
2106         pcrud.update(bib);
2107     }
2108 }
2109
2110 function onBibSourceSelect() {
2111     var cbs = dojo.byId('bib-source-list').selectedItem.value;
2112     var bib = xulG.record.bre;
2113     if (bib.source() != cbs) {
2114         dojo.byId('bib-source-list-button').disabled = false;   
2115     } else {
2116         dojo.byId('bib-source-list-button').disabled = true;   
2117     }
2118 }
2119
2120 function addNewAuthorityID(authority, sf, target) {
2121     var id_sf = <subfield code="0" xmlns="http://www.loc.gov/MARC21/slim">({xulG.marc_control_number_identifier}){authority.id()}</subfield>;
2122     sf.parent().appendChild(id_sf);
2123     var new_sf = marcSubfield(id_sf);
2124
2125     var node = target;
2126     while (dojo.attr(node, 'name') != 'sf_box') {
2127         node = node.parentNode;
2128     }
2129     node.appendChild( new_sf );
2130
2131     alert($('catStrings').getString('staff.cat.marcedit.create_authority_success.label'));
2132 }
2133
2134 function loadMarcEditor(pcrud, marcxml, target, sf) {
2135     /*
2136        To run in Firefox directly, must set signed.applets.codebase_principal_support
2137        to true in about:config
2138      */
2139     win = window.open('/xul/server/cat/marcedit.xul', '_blank', 'chrome'); // XXX version?
2140
2141     // Match marc2are.pl last_xact_id format, roughly
2142     var now = new Date;
2143     var xact_id = 'IMPORT-' + Date.parse(now);
2144     
2145     win.xulG = {
2146         "record": {"marc": marcxml, "rtype": "are"},
2147         "save": {
2148             "label": $('catStrings').getString('staff.cat.marcedit.save.label'),
2149             "func": function(xmlString) {
2150                 var rec = new are();
2151                 rec.marc(xmlString);
2152                 rec.last_xact_id(xact_id);
2153                 rec.isnew(true);
2154                 pcrud.create(rec, {
2155                     "oncomplete": function (r, objs) {
2156                         var new_rec = objs[0];
2157                         if (!new_rec) {
2158                             return '';
2159                         }
2160
2161                         addNewAuthorityID(new_rec, sf, target);
2162
2163                         win.close();
2164                     }
2165                 });
2166             }
2167         }
2168     };
2169 }
2170
2171