3 * Copyright (C) 2004-2008 Georgia Public Library Service
4 * Copyright (C) 2008-2010 Equinox Software, Inc.
5 * Mike Rylander <miker@esilibrary.com>
7 * Copyright (C) 2010 Dan Scott <dan@coffeecode.net>
8 * Copyright (C) 2010 Internationaal Instituut voor Sociale Geschiedenis <info@iisg.nl>
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.
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.
21 // Pretty printing kills whitespace too, so disable it.
22 XML.prettyPrinting = false;
23 var xmlDeclaration = /^<\?xml version[^>]+?>/;
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;
31 var tooltip_hash = {};
43 var show_auth_menu = false;
45 var _fixed_field_values = {}, _fixed_field_context_menus = {};
46 var _fixed_field_anonymous_func_counter = 0;
48 function $(id) { return document.getElementById(id); }
50 var acs; // AuthorityControlSet
52 function get_new_008() {
55 var y = now.getUTCFullYear().toString().substr(2,2);
56 var m = now.getUTCMonth() + 1;
57 if (m < 10) m = '0' + m;
58 var d = now.getUTCDate();
59 if (d < 10) d = '0' + d;
61 if (xml_record.controlfield.(@tag == '008')) {
62 var field = xml_record.controlfield.(@tag == '008')[0];
63 orig008 = field.text();
66 /* lang code from 041a */
67 var lang = orig008.substr(35, 3);
68 if (xml_record.datafield.(@tag == '041')) {
69 var field = xml_record.datafield.(@tag == '041')[0];
70 if (field && field.subfield.(@code == 'a')) {
71 lang = field.subfield.(@code == 'a')[0];
75 /* country code from 044a */
76 var country = orig008.substr(15, 3);
77 if (xml_record.datafield.(@tag == '044')) {
78 var field = xml_record.datafield.(@tag == '044')[0];
79 if (field && field.subfield.(@code == 'a')) {
80 country = field.subfield.(@code == 'a')[0];
83 while (country.length < 3) country = country + ' ';
84 if (country.length > 3) country = country.substr(0,3);
87 var date1 = now.getUTCFullYear().toString();
88 if (xml_record.datafield.(@tag == '260')) {
89 var field = xml_record.datafield.(@tag == '260')[0];
90 if (field && field.subfield.(@code == 'c')) {
91 var tmpd = field.subfield.(@code == 'c')[0].replace(/[^0-9]/g, '');
92 if (tmpd.match(/^\d\d\d\d/)) {
93 date1 = tmpd.substr(0, 4);
98 var date2 = orig008.substr(11, 4);
99 var datetype = orig008.substr(6, 1);
100 var modded = orig008.substr(38, 1);
101 var catsrc = orig008.substr(39, 1);
103 return '' + y + m + d + datetype + date1 + date2 + country + ' ' + lang + modded + catsrc;
107 function mangle_005() {
108 var now = new Date();
109 var y = now.getUTCFullYear();
111 var m = now.getUTCMonth() + 1;
112 if (m < 10) m = '0' + m;
114 var d = now.getUTCDate();
115 if (d < 10) d = '0' + d;
117 var H = now.getUTCHours();
118 if (H < 10) H = '0' + H;
120 var M = now.getUTCMinutes();
121 if (M < 10) M = '0' + M;
123 var S = now.getUTCSeconds();
124 if (S < 10) S = '0' + S;
127 var stamp = '' + y + m + d + H + M + S + '.0';
128 createControlField('005',stamp);
132 function createControlField (tag,data) {
133 // first, remove the old field, if any;
134 for (var i in xml_record.controlfield.(@tag == tag)) delete xml_record.controlfield.(@tag == tag)[i];
136 var cf = <controlfield tag="" xmlns="http://www.loc.gov/MARC21/slim">{ data }</controlfield>;
139 // then, find the right position and insert it
141 var cfields = xml_record.controlfield;
142 var base = Number(tag.substring(2));
143 for (var i in cfields) {
144 var t = Number(cfields[i].@tag.toString().substring(2));
146 xml_record.insertChildBefore( cfields[i], cf );
152 if (!done) xml_record.insertChildBefore( xml_record.datafield[0], cf );
157 function xml_escape_unicode ( str ) {
159 /([\u0080-\ufffe])/g,
160 function (r,s) { return "&#x" + s.charCodeAt(0).toString(16) + ";"; }
164 function wrap_long_fields (node) {
165 var text_size = dojo.attr(node, 'size');
166 var hard_width = 100;
167 if (text_size > hard_width) {
168 dojo.attr(node, 'multiline', 'true');
169 dojo.attr(node, 'cols', hard_width);
170 var text_rows = (text_size / hard_width) + 1;
171 dojo.attr(node, 'rows', text_rows);
175 function set_flat_editor (useFlatText) {
177 var xe = $('xul-editor');
178 var te = $('text-editor');
181 if (xe.hidden) { return; }
185 if (te.hidden) { return; }
191 // get the marcxml from the text box
192 var xml_string = new MARC.Record({
193 marcbreaker : $('text-editor-box').value,
197 // reset the xml record and rerender it
198 xml_record = new XML( xml_string );
199 if (xml_record..record[0]) xml_record = xml_record..record[0];
202 var xml_string = xml_record.toXMLString();
204 // push the xml record into the textbox
205 var rec = new MARC.Record ({ delimiter : '$', marcxml : xml_string });
206 $('text-editor-box').value = rec.toBreaker();
213 if (typeof JSAN == 'undefined') { throw( $("commonStrings").getString('common.jsan.missing') ); }
214 JSAN.errorLevel = "die"; // none, warn, or die
215 JSAN.addRepository('/xul/server/');
217 dojo.require('openils.AuthorityControlSet');
218 acs = new openils.AuthorityControlSet ();
220 // Fake xulG for standalone...
225 window.xulG.record = {};
226 window.xulG.save = {};
227 window.xulG.marc_control_number_identifier = 'CONS';
229 window.xulG.save.label = $('catStrings').getString('staff.cat.marcedit.save.label');
230 window.xulG.save.func = function (r) { alert(r); }
233 var _rid = cgi.param('record');
235 window.xulG.record.id = _rid;
236 window.xulG.record.url = '/opac/extras/supercat/retrieve/marcxml/record/' + _rid;
240 // End faking part...
242 /* Check for an explicitly passed record type
243 * This is not the same as the fixed-field record type; we can't trust
244 * the fixed fields when making modifications to the attributes for a
245 * given record (in particular, config.bib_source only applies for bib
246 * records, but an auth or MFHD record with the same ID and bad fixed
247 * fields could trample the config.bib_source value for the
248 * corresponding bib record if we're not careful.
250 * are = authority record
251 * sre = serial record (MFHD)
252 * bre = bibliographic record
254 if (!window.xulG.record.rtype) {
256 window.xulG.record.rtype = cgi.param('rtype') || false;
259 document.getElementById('save-button').setAttribute('label', window.xulG.save.label);
260 /* Ugh. Sorry about the spaghetti. */
261 document.getElementById('save-button').setAttribute('oncommand',
262 'var to_save = function() { ' + /* begin to_save() */
263 'if ($("xul-editor").hidden) set_flat_editor(false); ' +
265 'var xml_string = xml_escape_unicode( xml_record.toXMLString() ); ' +
266 'save_attempt( xml_string ); ' +
268 '}; ' + /* end to_save() */
270 'if (typeof _owPCW == "object") { ' +
271 ' for (var k in _owPCW) { ' +
272 ' if (_owPCW[k].active) { ' +
273 ' try { _owPCW[k].apply(to_save); to_save.ran = true; } ' +
274 ' catch (E) { alert("_ow_PCW[" + k + "]: " + E); } ' +
279 'if (!to_save.ran) to_save();'
282 if (window.xulG.record.url) {
283 var req = new XMLHttpRequest();
284 req.open('POST',window.xulG.record.url,false);
286 window.xulG.record.marc = req.responseText.replace(xmlDeclaration, '');
289 xml_record = new XML( window.xulG.record.marc );
290 if (xml_record..record[0]) xml_record = xml_record..record[0];
292 // Get the tooltip xml all async like
293 req = new XMLHttpRequest();
295 // Set a default locale in case preferences fail us
296 var locale = "en-US";
298 // Try to get the locale from our preferences
300 const Cc = Components.classes;
301 const Ci = Components.interfaces;
302 locale = Cc["@mozilla.org/preferences-service;1"].
303 getService(Ci.nsIPrefBranch).
304 getCharPref("general.useragent.locale");
308 // TODO: We should send a HEAD request to check for the existence of the desired file
309 // then fall back to the default locale if preferred locale is not necessary;
310 // however, for now we have a simplistic check:
312 // we currently have translations for only three locales; in the absence of a
313 // valid locale, default to the almighty en-US
314 if (locale != 'en-US' && locale != 'fr-CA' && locale != 'fi-FI') {
318 // grab the right tooltip based on MARC type
319 var tooltip_doc = 'marcedit-tooltips.xml';
320 switch (window.xulG.record.rtype) {
322 tooltip_doc = 'marcedit-tooltips.xml';
325 tooltip_doc = 'marcedit-tooltips-authority.xml';
326 locale = 'en-US'; // FIXME - note TODO above; at moment only en-US has this
329 tooltip_doc = 'marcedit-tooltips-mfhd.xml';
330 locale = 'en-US'; // FIXME - note TODO above; at moment only en-US has this
333 tooltip_doc = 'marcedit-tooltips.xml';
336 // Get the locale-specific tooltips
337 req.open('GET','/xul/server/locale/' + locale + '/' + tooltip_doc,true);
339 context_menus = createComplexXULElement('popupset');
340 document.documentElement.appendChild( context_menus );
342 tag_menu = createMenuPopup({position : 'after_start', id : 'tags_popup'});
343 context_menus.appendChild( tag_menu );
345 tag_menu.appendChild(
347 { label : $('catStrings').getString('staff.cat.marcedit.add_row.label'),
349 'var e = document.createEvent("KeyEvents");' +
350 'e.initKeyEvent("keypress",1,1,null,1,0,0,0,13,0);' +
351 'current_focus.inputField.dispatchEvent(e);'
356 tag_menu.appendChild(
358 { label : $('catStrings').getString('staff.cat.marcedit.insert_row.label'),
360 'var e = document.createEvent("KeyEvents");' +
361 'e.initKeyEvent("keypress",1,1,null,1,0,1,0,13,0);' +
362 'current_focus.inputField.dispatchEvent(e);'
367 tag_menu.appendChild(
369 { label : $('catStrings').getString('staff.cat.marcedit.remove_row.label'),
371 'var e = document.createEvent("KeyEvents");' +
372 'e.initKeyEvent("keypress",1,1,null,1,0,0,0,46,0);' +
373 'current_focus.inputField.dispatchEvent(e);'
378 tag_menu.appendChild( createComplexXULElement( 'separator' ) );
380 tag_menu.appendChild(
382 { label : $('catStrings').getString('staff.cat.marcedit.replace_006.label'),
384 'var e = document.createEvent("KeyEvents");' +
385 'e.initKeyEvent("keypress",1,1,null,1,0,0,0,117,0);' +
386 'current_focus.inputField.dispatchEvent(e);'
391 tag_menu.appendChild(
393 { label : $('catStrings').getString('staff.cat.marcedit.replace_007.label'),
395 'var e = document.createEvent("KeyEvents");' +
396 'e.initKeyEvent("keypress",1,1,null,1,0,0,0,118,0);' +
397 'current_focus.inputField.dispatchEvent(e);'
402 tag_menu.appendChild(
404 { label : $('catStrings').getString('staff.cat.marcedit.replace_008.label'),
406 'var e = document.createEvent("KeyEvents");' +
407 'e.initKeyEvent("keypress",1,1,null,1,0,0,0,119,0);' +
408 'current_focus.inputField.dispatchEvent(e);'
413 tag_menu.appendChild( createComplexXULElement( 'separator' ) );
415 p = createComplexXULElement('popupset');
416 document.documentElement.appendChild( p );
418 req.onreadystatechange = function () {
419 if (req.readyState == 4) {
420 bib_data = new XML( req.responseText.replace(xmlDeclaration, '') );
429 if (! xulG.fast_add_item) {
430 document.getElementById('fastItemAdd_checkbox').hidden = true;
432 document.getElementById('fastItemAdd_textboxes').hidden = document.getElementById('fastItemAdd_checkbox').hidden || !document.getElementById('fastItemAdd_checkbox').checked;
434 // Only show bib sources for bib records that already exist in the database
435 if (xulG.record.rtype == 'bre' && xulG.record.id) {
436 dojo.require('openils.PermaCrud');
437 var authtoken = ses();
438 // Retrieve the current record attributes
439 var bib = new openils.PermaCrud({"authtoken": authtoken}).retrieve('bre', xulG.record.id);
441 // Remember the current bib source of the record
442 xulG.record.bre = bib;
444 buildBibSourceList(authtoken, xulG.record.id);
447 preparePhysCharWizardContext();
448 dojo.require('MARC.FixedFields');
449 dojo.require("openils.widget.PhysCharWizard");
452 alert('FIXME, MARC Editor, my_init: ' + E);
457 function createComplexHTMLElement (e, attrs, objects, text) {
458 var l = document.createElementNS('http://www.w3.org/1999/xhtml',e);
461 for (var i in attrs) l.setAttribute(i,attrs[i]);
465 for ( var i in objects ) l.appendChild( objects[i] );
469 l.appendChild( document.createTextNode(text) )
475 function createComplexXULElement (e, attrs, objects) {
476 var l = document.createElementNS('http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul',e);
479 for (var i in attrs) {
480 if (typeof attrs[i] == 'function') {
481 l.addEventListener( i, attrs[i], true );
483 l.setAttribute(i,attrs[i]);
489 for ( var i in objects ) l.appendChild( objects[i] );
495 function createDescription (attrs) {
496 return createComplexXULElement('description', attrs, Array.prototype.slice.apply(arguments, [1]) );
499 function createTooltip (attrs) {
500 return createComplexXULElement('tooltip', attrs, Array.prototype.slice.apply(arguments, [1]) );
503 function createLabel (attrs) {
504 return createComplexXULElement('label', attrs, Array.prototype.slice.apply(arguments, [1]) );
507 function createVbox (attrs) {
508 return createComplexXULElement('vbox', attrs, Array.prototype.slice.apply(arguments, [1]) );
511 function createHbox (attrs) {
512 return createComplexXULElement('hbox', attrs, Array.prototype.slice.apply(arguments, [1]) );
515 function createRow (attrs) {
516 return createComplexXULElement('row', attrs, Array.prototype.slice.apply(arguments, [1]) );
519 function createTextbox (attrs) {
520 return createComplexXULElement('textbox', attrs, Array.prototype.slice.apply(arguments, [1]) );
523 function createMenu (attrs) {
524 return createComplexXULElement('menu', attrs, Array.prototype.slice.apply(arguments, [1]) );
527 function createMenuPopup (attrs) {
528 return createComplexXULElement('menupopup', attrs, Array.prototype.slice.apply(arguments, [1]) );
531 function createPopup (attrs) {
532 return createComplexXULElement('popup', attrs, Array.prototype.slice.apply(arguments, [1]) );
535 function createMenuitem (attrs) {
536 return createComplexXULElement('menuitem', attrs, Array.prototype.slice.apply(arguments, [1]) );
539 function createCheckbox (attrs) {
540 return createComplexXULElement('checkbox', attrs, Array.prototype.slice.apply(arguments, [1]) );
543 // Find the next textbox that we can use for a focus point
544 // For control fields, use the first editable text box
545 // For data fields, focus on the first subfield text box
546 function setFocusToNextTag (row, direction) {
547 var keep_looking = true;
548 while (keep_looking && (direction == 'up' ? row = row.previousSibling : row = row.nextSibling)) {
549 // Is it a datafield?
550 dojo.query('hbox', row).query('hbox').query('textbox').forEach(function(node, index, arr) {
552 keep_looking = false;
555 // No, it's a control field; use the first textbox
557 dojo.query('textbox', row).forEach(function(node, index, arr) {
559 keep_looking = false;
567 function set_lock_on_keypress(ev) {
569 var tabs = window.parent.parent.document.getElementById('main_tabs');
572 var idx = tabs.selectedIndex;
573 var tab = tabs.childNodes[idx];
575 //dump('keypress: isChar = ' + ev.isChar + ' char = ' + ev.char + ' charCode = ' + ev.charCode + ' key = ' + ev.key + ' keyCode = ' + ev.keyCode + '\n');
581 || ev.keyCode == ev.DOM_VK_F1
582 || ev.keyCode == ev.DOM_VK_F2
583 || ev.keyCode == ev.DOM_VK_F3
584 || ev.keyCode == ev.DOM_VK_F4
585 || ev.keyCode == ev.DOM_VK_F5
586 || ev.keyCode == ev.DOM_VK_F6
587 || ev.keyCode == ev.DOM_VK_F7
588 || ev.keyCode == ev.DOM_VK_F8
589 || ev.keyCode == ev.DOM_VK_F9
590 || ev.keyCode == ev.DOM_VK_F10
591 || ev.keyCode == ev.DOM_VK_F11
592 || ev.keyCode == ev.DOM_VK_F12
593 || ev.keyCode == ev.DOM_VK_F13
594 || ev.keyCode == ev.DOM_VK_F14
595 || ev.keyCode == ev.DOM_VK_F15
596 || ev.keyCode == ev.DOM_VK_F16
597 || ev.keyCode == ev.DOM_VK_F17
598 || ev.keyCode == ev.DOM_VK_F18
599 || ev.keyCode == ev.DOM_VK_F19
600 || ev.keyCode == ev.DOM_VK_F20
601 || ev.keyCode == ev.DOM_VK_F21
602 || ev.keyCode == ev.DOM_VK_F22
603 || ev.keyCode == ev.DOM_VK_F23
604 || ev.keyCode == ev.DOM_VK_F24
608 params.allow_multiple_locks = tab.marc_edit_allow_multiple_locks;
610 oils_lock_page(params);
617 function createMARCTextbox (element,attrs) {
618 var box = createComplexXULElement('textbox', attrs, Array.prototype.slice.apply(arguments, [2]) );
619 box.addEventListener(
621 set_lock_on_keypress,
624 if ($('symbol-panel')) {
625 addSymbolTrigger(box);
627 box.onkeypress = function (event) {
630 while(node = node.parent()) {
634 var row = event.target;
635 while (row.tagName != 'row') row = row.parentNode;
637 if (element.nodeKind() == 'attribute') element[0]=box.value;
638 else element.setChildren( box.value );
640 if (element.localName() != 'controlfield') {
641 if ((event.charCode == 100 || event.charCode == 105) && event.ctrlKey) { // ctrl+d or ctrl+i
643 var index_sf, target, move_data;
644 if (element.localName() == 'subfield') {
646 target = event.target.parentNode;
648 var start = event.target.selectionStart;
649 var end = event.target.selectionEnd - event.target.selectionStart ?
650 event.target.selectionEnd :
651 event.target.value.length;
653 move_data = event.target.value.substring(start,end);
654 event.target.value = event.target.value.substring(0,start) + event.target.value.substring(end);
655 event.target.setAttribute('size', event.target.value.length + 2);
657 element.setChildren( event.target.value );
659 } else if (element.localName() == 'code') {
660 index_sf = element.parent();
661 target = event.target.parentNode;
662 } else if (element.localName() == 'tag' || element.localName() == 'ind1' || element.localName() == 'ind2') {
663 index_sf = element.parent().children()[element.parent().children().length() - 1];
664 target = event.target.parentNode.lastChild.lastChild;
667 var sf = <subfield code="" xmlns="http://www.loc.gov/MARC21/slim">{ move_data }</subfield>;
669 index_sf.parent().insertChildAfter( index_sf, sf );
671 var new_sf = marcSubfield(sf);
673 if (target === target.parentNode.lastChild) {
674 target.parentNode.appendChild( new_sf );
676 target.parentNode.insertBefore( new_sf, target.nextSibling );
679 new_sf.firstChild.nextSibling.focus();
681 event.preventDefault();
684 } else if (event.keyCode == 13 || event.keyCode == 77) {
685 if (event.ctrlKey) { // ctrl+enter
688 if (element.localName() == 'subfield') index = element.parent();
689 if (element.localName() == 'code') index = element.parent().parent();
690 if (element.localName() == 'tag') index = element.parent();
691 if (element.localName() == 'ind1') index = element.parent();
692 if (element.localName() == 'ind2') index = element.parent();
694 var df = <datafield tag="" ind1="" ind2="" xmlns="http://www.loc.gov/MARC21/slim"><subfield code="" /></datafield>;
696 if (event.shiftKey) { // ctrl+shift+enter
697 index.parent().insertChildBefore( index, df );
699 index.parent().insertChildAfter( index, df );
702 var new_df = marcDatafield(df);
704 if (row.parentNode.lastChild === row) {
705 row.parentNode.appendChild( new_df );
707 if (event.shiftKey) { // ctrl+shift+enter
708 row.parentNode.insertBefore( new_df, row );
710 row.parentNode.insertBefore( new_df, row.nextSibling );
714 new_df.firstChild.focus();
716 event.preventDefault();
719 } else if (event.shiftKey) {
720 if (row.previousSibling.className.match('marcDatafieldRow'))
721 row.previousSibling.firstChild.focus();
723 row.nextSibling.firstChild.focus();
726 } else if (event.keyCode == 38 || event.keyCode == 40) { // up-arrow or down-arrow
727 if (event.ctrlKey) { // CTRL key: copy the field
729 if (element.localName() == 'subfield') index = element.parent();
730 if (element.localName() == 'code') index = element.parent().parent();
731 if (element.localName() == 'tag') index = element.parent();
732 if (element.localName() == 'ind1') index = element.parent();
733 if (element.localName() == 'ind2') index = element.parent();
735 var copyField = index.copy();
737 if (event.keyCode == 38) { // ctrl+up-arrow
738 index.parent().insertChildBefore( index, copyField );
740 index.parent().insertChildAfter( index, copyField );
743 var new_df = marcDatafield(copyField);
745 if (row.parentNode.lastChild === row) {
746 row.parentNode.appendChild( new_df );
748 if (event.keyCode == 38) { // ctrl+up-arrow
749 row.parentNode.insertBefore( new_df, row );
750 } else { // ctrl+down-arrow
751 row.parentNode.insertBefore( new_df, row.nextSibling );
755 new_df.firstChild.focus();
757 event.preventDefault();
761 if (event.keyCode == 38) {
762 return setFocusToNextTag(row, 'up');
764 if (event.keyCode == 40) {
765 return setFocusToNextTag(row, 'down');
770 } else if (event.keyCode == 46 && event.ctrlKey) { // ctrl+del
773 if (element.localName() == 'subfield') index = element.parent();
774 if (element.localName() == 'code') index = element.parent().parent();
775 if (element.localName() == 'tag') index = element.parent();
776 if (element.localName() == 'ind1') index = element.parent();
777 if (element.localName() == 'ind2') index = element.parent();
779 for (var i in index.parent().children()) {
780 if (index === index.parent().children()[i]) {
781 delete index.parent().children()[i];
786 row.previousSibling.firstChild.focus();
787 row.parentNode.removeChild(row);
789 event.preventDefault();
792 } else if (event.keyCode == 46 && event.shiftKey) { // shift+del
795 if (element.localName() == 'subfield') index = element;
796 if (element.localName() == 'code') index = element.parent();
799 for (var i in index.parent().children()) {
800 if (index === index.parent().children()[i]) {
801 delete index.parent().children()[i];
806 if (event.target.parentNode === event.target.parentNode.parentNode.lastChild) {
807 event.target.parentNode.previousSibling.lastChild.focus();
809 event.target.parentNode.nextSibling.firstChild.nextSibling.focus();
812 event.target.parentNode.parentNode.removeChild(event.target.parentNode);
814 event.preventDefault();
817 } else if (event.keyCode == 117 && event.ctrlKey) { // ctrl + F6
819 createControlField('006',' ');
821 } else if (event.keyCode == 118 && event.ctrlKey) { // ctrl + F7
823 createControlField('007',' ');
825 } else if (event.keyCode == 119 && event.ctrlKey) { // ctrl + F8
827 createControlField('008', get_new_008());
833 } else { // event on a control field
834 if (event.keyCode == 38) {
835 return setFocusToNextTag(row, 'up');
836 } else if (event.keyCode == 40) {
837 return setFocusToNextTag(row, 'down');
842 box.addEventListener(
845 if (element.nodeKind() == 'attribute') element[0]=box.value;
846 else element.setChildren( box.value );
852 box.addEventListener(
855 if (element.nodeKind() == 'attribute') element[0]=box.value;
856 else element.setChildren( box.value );
862 box.addEventListener(
865 if (element.nodeKind() == 'attribute') element[0]=box.value;
866 else element.setChildren( box.value );
872 // 'input' event catches the box value after the keypress
873 box.addEventListener(
876 if (element.nodeKind() == 'attribute') element[0]=box.value;
877 else element.setChildren( box.value );
883 box.addEventListener(
886 if (element.localName() == 'controlfield')
887 eval('fillFixedFields();');
895 function toggleFFE () {
896 var grid = document.getElementById('leaderGrid');
905 function changeFFEditor (type) {
906 var grid = document.getElementById('leaderGrid');
907 grid.setAttribute('type',type);
908 document.getElementById('recordTypeLabel').setAttribute('value',type);
910 // Hide FFEditor rows that we don't need for our current type
911 // If all of the labels for a given row do not include our
912 // desired type in their set attribute, we can hide that row
913 dojo.query('rows', grid).query('row').forEach(function(node, index, arr) {
914 if (dojo.query('label[set~=' + type + ']', node).length == 0) {
922 updateFFEditorContexts(grid, type);
925 alert( /* XXX i18n - the marc editor either isn't tied in to
926 an overall i18n infrastructure, or it's different
927 enough from other Evergreen parts that I don't
928 understand the right way to do i18n here */
929 "failed to load fixed field values for rec_type '" +
936 function getFFValuesForType(type, callback, errback) {
937 if (_fixed_field_values[type]) {
940 dojo.require("openils.Util");
942 fieldmapper.standardRequest(
944 "open-ils.cat.biblio.fixed_field_values.by_rec_type"], {
947 "oncomplete": function(r) {
948 if (r = openils.Util.readResponse(r)) {
949 _fixed_field_values[type] = r;
961 function updateFFEditorContexts(grid, type) {
963 /* XXX Dojo to navigate a XUL DOM. Bad form? We do it elsewhere in this
964 * file, but I think I've heard it scoffed at. */
966 var rows_node = dojo.query("rows", grid)[0];
967 dojo.query("row", rows_node).forEach(
969 dojo.query("textbox", row).forEach(
971 var name = tb.getAttribute("name");
972 if (_fixed_field_values[type][name]) {
974 "context", getFFContextMenu(type, name)
977 tb.setAttribute("context", "clipboard");
985 function getFFContextMenu(type, name) {
986 if (!_fixed_field_context_menus[type]) {
987 _fixed_field_context_menus[type] = {};
990 var context_menu_id = "_fixed_field_context_menus_" + type + "_" + name;
992 if (!_fixed_field_context_menus[type][name]) {
993 var p = document.getElementsByTagName("popupset")[0];
994 var m = document.createElement("menupopup");
995 m.setAttribute("id", context_menu_id);
996 _fixed_field_values[type][name].forEach(
998 var mi = document.createElement("menuitem");
999 var funcname = "_ff_anon_" +
1000 String(_fixed_field_anonymous_func_counter++);
1002 /* These anon functions linger and take up memory, but due
1003 * to caching there's a limit on how much, and that limit is
1004 * a function of the size of the dataset defined in the
1005 * ccvm/cmfpm tables. */
1009 var el = document.getElementById(name + "_tb");
1010 window[funcname] = function() {
1012 while (el.value.length < len) {
1015 updateFixedFields(el);
1019 /* In XUL land we can't set an element's
1020 * attribute like oncommand to a code reference. It has to be
1021 * an actual string to be eval'd. */
1022 mi.setAttribute("oncommand", funcname + "()");
1023 mi.setAttribute("label", v[0] + ": " + v[1]); /* XXX i18n ? */
1027 _fixed_field_context_menus[type][name] = m;
1031 return context_menu_id;
1034 /* This just sets up a special context menu for a 007 data field to use, so
1035 * that users can right-click for a menu and get a choice to launch the
1036 * Physical Characteristics Wizard.
1038 function preparePhysCharWizardContext() {
1039 var menu = document.getElementById("physCharWizardContext");
1040 menu.appendChild(document.createElement("menuseparator"));
1042 var clipb_children = document.getElementById("clipboard").childNodes;
1043 for (var i = 0; i < clipb_children.length; i++) /* collection not array */ {
1044 var child = clipb_children[i];
1045 if (child.nodeName == 'menuitem')
1046 menu.appendChild(child.cloneNode(true));
1050 function launchPhysCharWizard(popup_node) {
1052 new openils.widget.PhysCharWizard({
1054 "onapply": function(v) {
1055 createControlField("007", v);
1061 alert("Exception raised by openils.widget.PhysCharWizard:\n" + E);
1065 function fillFixedFields () {
1067 var grid = document.getElementById('leaderGrid');
1068 var marc_rec = new MARC.Record ({ delimiter : '$', marcxml : xml_record.toXMLString() });
1071 var pre_list = grid.getElementsByTagName('label');
1072 for (var i in pre_list) {
1073 if ( pre_list[i].getAttribute && pre_list[i].getAttribute('set').indexOf(grid.getAttribute('type')) > -1 ) {
1074 list.push( pre_list[i] );
1078 for (var i in list) {
1079 var name = list[i].getAttribute('name');
1080 var value = marc_rec.extractFixedField(name, true);
1082 if (value === null) continue;
1084 list[i].nextSibling.value = value;
1089 alert('FIXME, MARC Editor, fillFixedFields: ' + E);
1093 function updateFixedFields (element) {
1094 var new_value = element.value;
1096 var marc_rec = new MARC.Record ({ delimiter : '$', marcxml : xml_record.toXMLString() });
1097 marc_rec.setFixedField(element.getAttribute('name'), new_value);
1099 var xml_string = marc_rec.toXmlString();
1100 xml_record = new XML( xml_string );
1101 if (xml_record..record[0]) xml_record = xml_record..record[0];
1103 // Put the cursor back to the current fixed field
1109 function marcLeader (leader) {
1110 var row = createRow(
1111 { class : 'marcLeaderRow',
1116 tooltiptext : $('catStrings').getString('staff.cat.marcedit.marcTag.LDR.label') } ),
1119 class : 'marcInd1' } ),
1122 class : 'marcInd2' } ),
1124 { value : leader.text(),
1125 class : 'marcLeader' } )
1131 function marcControlfield (field) {
1132 tagname = field.@tag.toString().substr(2);
1134 if (tagname == '1' || tagname == '3' || tagname == '6' || tagname == '7' || tagname == '8') {
1136 { class : 'marcControlfieldRow',
1137 tag : '_' + tagname },
1139 { value : field.@tag,
1141 context : 'tags_popup',
1142 onmouseover : 'getTooltip(this, "tag");',
1143 tooltipid : 'tag' + field.@tag } ),
1145 { value : field.@ind1,
1147 onmouseover : 'getTooltip(this, "ind1");',
1148 tooltipid : 'tag' + field.@tag + 'ind1val' + field.@ind1 } ),
1150 { value : field.@ind2,
1152 onmouseover : 'getTooltip(this, "ind2");',
1153 tooltipid : 'tag' + field.@tag + 'ind2val' + field.@ind2 } ),
1156 { value : field.text(),
1157 class : 'plain marcEditableControlfield',
1158 name : 'CONTROL' + tagname,
1159 context : tagname == 7 ? 'physCharWizardContext': 'clipboard',
1165 { class : 'marcControlfieldRow',
1166 tag : '_' + tagname },
1168 { value : field.@tag,
1170 onmouseover : 'getTooltip(this, "tag");',
1171 tooltipid : 'tag' + field.@tag } ),
1173 { value : field.@ind1,
1175 onmouseover : 'getTooltip(this, "ind1");',
1176 tooltipid : 'tag' + field.@tag + 'ind1val' + field.@ind1 } ),
1178 { value : field.@ind2,
1180 onmouseover : 'getTooltip(this, "ind2");',
1181 tooltipid : 'tag' + field.@tag + 'ind2val' + field.@ind2 } ),
1183 { value : field.text(),
1184 class : 'marcControlfield' } )
1191 function stackSubfields(checkbox) {
1192 var list = document.getElementsByAttribute('name','sf_box');
1195 if (!checkbox.checked) o = 'horizontal';
1197 for (var i = 0; i < list.length; i++) {
1198 if (list[i]) list[i].setAttribute('orient',o);
1202 function fastItemAdd_toggle(checkbox) {
1203 var x = document.getElementById('fastItemAdd_textboxes');
1204 if (checkbox.checked) {
1206 document.getElementById('fastItemAdd_callnumber').focus();
1207 document.getElementById('fastItemAdd_callnumber').select();
1213 function fastItemAdd_attempt(doc_id) {
1215 if (typeof window.xulG.fast_add_item != 'function') { return; }
1216 if (!document.getElementById('fastItemAdd_checkbox').checked) { return; }
1217 if (!document.getElementById('fastItemAdd_callnumber').value) { return; }
1218 if (!document.getElementById('fastItemAdd_barcode').value) { return; }
1219 window.xulG.fast_add_item( doc_id, document.getElementById('fastItemAdd_callnumber').value, document.getElementById('fastItemAdd_barcode').value );
1220 document.getElementById('fastItemAdd_barcode').value = '';
1223 alert('fastItemAdd_attempt: ' + E);
1227 function save_attempt(xml_string) {
1229 var tabs = window.parent.parent.document.getElementById('main_tabs');
1232 var idx = tabs.selectedIndex;
1233 var tab = tabs.childNodes[idx];
1235 var result = window.xulG.save.func( xml_string );
1236 // I'd prefer to pass on_complete on through to fast_item_add,
1237 // but with the way these window scopes get destroyed with
1238 // tab replacement, maybe not a good idea
1239 var replace_on_complete = false;
1242 tab.marc_edit_changed = false;
1245 oils_unlock_page(); /* kludge for LP1491875 */
1247 replace_on_complete = fastItemAdd_attempt(result.id);
1249 // When fastItemAdd is not used, replace_on_complete is undefined
1250 // When it is used, and successful, replace_on_complete will be set to true
1251 // then we need it to execute on_complete()
1252 if ((replace_on_complete==undefined || replace_on_complete ) && typeof result.on_complete == 'function') {
1253 result.on_complete();
1257 alert('save_attempt: ' + E);
1261 function marcDatafield (field) {
1262 var row = createRow(
1263 { class : 'marcDatafieldRow' },
1266 { value : field.@tag,
1267 class : 'plain marcTag',
1269 context : 'tags_popup',
1270 oninput : 'if (this.value.length == 3) { this.nextSibling.focus(); }',
1273 onmouseover : 'current_focus = this; getTooltip(this, "tag");' } ),
1276 { value : field.@ind1,
1277 class : 'plain marcInd1',
1279 oninput : 'if (this.value.length == 1) { this.nextSibling.focus(); }',
1282 onmouseover : 'current_focus = this; getContextMenu(this, "ind1"); getTooltip(this, "ind1");',
1283 oncontextmenu : 'getContextMenu(this, "ind1");' } ),
1286 { value : field.@ind2,
1287 class : 'plain marcInd2',
1289 oninput : 'if (this.value.length == 1) { this.nextSibling.firstChild.firstChild.focus(); }',
1292 onmouseover : 'current_focus = this; getContextMenu(this, "ind2"); getTooltip(this, "ind2");',
1293 oncontextmenu : 'getContextMenu(this, "ind2");' } ),
1294 createHbox({ name : 'sf_box' })
1297 if (!current_focus && field.@tag == '') current_focus = row.childNodes[0];
1298 if (!current_focus && field.@ind1 == '') current_focus = row.childNodes[1];
1299 if (!current_focus && field.@ind2 == '') current_focus = row.childNodes[2];
1301 var sf_box = row.lastChild;
1302 if (document.getElementById('stackSubfields').checked)
1303 sf_box.setAttribute('orient','vertical');
1305 sf_box.addEventListener(
1308 if (sf_box === e.target) {
1309 sf_box.lastChild.lastChild.focus();
1310 } else if (e.target.parentNode === sf_box) {
1311 e.target.lastChild.focus();
1318 for (var i in field.subfield) {
1319 var sf = field.subfield[i];
1324 dojo.query('.marcSubfield', sf_box).forEach(wrap_long_fields);
1326 if (sf.@code == '' && (!current_focus || current_focus.className.match(/Ind/)))
1327 current_focus = sf_box.lastChild.childNodes[1];
1333 function marcSubfield (sf) {
1335 { class : 'marcSubfieldBox' },
1338 class : 'plain marcSubfieldDelimiter',
1339 onmouseover : 'getTooltip(this.nextSibling, "subfield");',
1340 oncontextmenu : 'getContextMenu(this.nextSibling, "subfield");',
1341 //onclick : 'this.nextSibling.focus();',
1342 onfocus : 'this.nextSibling.focus();',
1347 class : 'plain marcSubfieldCode',
1349 name : 'marcSubfieldCode',
1350 onmouseover : 'current_focus = this; getContextMenu(this, "subfield"); getTooltip(this, "subfield");',
1351 oncontextmenu : 'getContextMenu(this, "subfield");',
1352 oninput : 'if (this.value.length == 1) { this.nextSibling.focus(); }',
1357 { value : sf.text(),
1358 name : sf.parent().@tag + ':' + sf.@code,
1359 class : 'plain marcSubfield',
1361 onmouseover : 'getTooltip(this, "subfield");',
1362 contextmenu : function (event) { getAuthorityContextMenu(event.target, sf) },
1363 size : new String(sf.text()).length + 2,
1364 oninput : "this.setAttribute('size', this.value.length + 2);"
1369 function loadRecord() {
1371 var tabs = window.parent.parent.document.getElementById('main_tabs');
1374 var idx = tabs.selectedIndex;
1375 var tab = tabs.childNodes[idx];
1377 tab.marc_edit_changed = false;
1378 tab.marc_edit_allow_multiple_locks = true;
1382 var grid_rows = document.getElementById('recGrid').lastChild;
1384 while (grid_rows.firstChild) grid_rows.removeChild(grid_rows.firstChild);
1386 grid_rows.appendChild( marcLeader( xml_record.leader ) );
1388 for (var i in xml_record.controlfield) {
1389 grid_rows.appendChild( marcControlfield( xml_record.controlfield[i] ) );
1392 for (var i in xml_record.datafield) {
1393 grid_rows.appendChild( marcDatafield( xml_record.datafield[i] ) );
1396 grid_rows.getElementsByAttribute('class','marcDatafieldRow')[0].firstChild.focus();
1398 var marc_rec = new MARC.Record ({ delimiter : '$', marcxml : xml_record.toXMLString() });
1399 changeFFEditor(marc_rec.recordType());
1402 alert('FIXME, MARC Editor, loadRecord: ' + E);
1407 function genToolTips () {
1408 for (var i in bib_data.field) {
1409 var f = bib_data.field[i];
1411 tag_menu.appendChild(
1415 'current_focus.value = "' + f.@tag + '";' +
1416 'var e = document.createEvent("MutationEvents");' +
1417 'e.initMutationEvent("change",1,1,null,0,0,0,0);' +
1418 'current_focus.inputField.dispatchEvent(e);',
1419 disabled : f.@tag < '010' ? "true" : "false",
1420 tooltiptext : f.description }
1424 var i1_popup = createMenuPopup({position : 'after_start', id : 't' + f.@tag + 'i1' });
1425 context_menus.appendChild( i1_popup );
1427 var i2_popup = createMenuPopup({position : 'after_start', id : 't' + f.@tag + 'i2' });
1428 context_menus.appendChild( i2_popup );
1430 var sf_popup = createMenuPopup({position : 'after_start', id : 't' + f.@tag + 'sf' });
1431 context_menus.appendChild( sf_popup );
1433 tooltip_hash['tag' + f.@tag] = f.description;
1434 for (var j in f.indicator) {
1435 var ind = f.indicator[j];
1436 tooltip_hash['tag' + f.@tag + 'ind' + ind.@position + 'val' + ind.@value] = ind.description;
1438 if (ind.@position == 1) {
1439 i1_popup.appendChild(
1441 { label : ind.@value,
1443 'current_focus.value = "' + ind.@value + '";' +
1444 'var e = document.createEvent("MutationEvents");' +
1445 'e.initMutationEvent("change",1,1,null,0,0,0,0);' +
1446 'current_focus.inputField.dispatchEvent(e);',
1447 tooltiptext : ind.description }
1452 if (ind.@position == 2) {
1453 i2_popup.appendChild(
1455 { label : ind.@value,
1457 'current_focus.value = "' + ind.@value + '";' +
1458 'var e = document.createEvent("MutationEvents");' +
1459 'e.initMutationEvent("change",1,1,null,0,0,0,0);' +
1460 'current_focus.inputField.dispatchEvent(e);',
1461 tooltiptext : ind.description }
1467 for (var j in f.subfield) {
1468 var sf = f.subfield[j];
1469 tooltip_hash['tag' + f.@tag + 'sf' + sf.@code] = sf.description;
1471 sf_popup.appendChild(
1475 'current_focus.value = "' + sf.@code + '";' +
1476 'var e = document.createEvent("MutationEvents");' +
1477 'e.initMutationEvent("change",1,1,null,0,0,0,0);' +
1478 'current_focus.inputField.dispatchEvent(e);',
1479 tooltiptext : sf.description
1487 function getTooltip (target, type) {
1490 if (type == 'subfield')
1491 tt = 'tag' + target.parentNode.parentNode.parentNode.firstChild.value + 'sf' + target.parentNode.childNodes[1].value;
1494 tt = 'tag' + target.parentNode.firstChild.value + 'ind1val' + target.value;
1497 tt = 'tag' + target.parentNode.firstChild.value + 'ind2val' + target.value;
1500 tt = 'tag' + target.parentNode.firstChild.value;
1502 if (!document.getElementById( tt )) {
1507 orient : 'vertical',
1508 onpopupshown : 'this.width = this.firstChild.boxObject.width + 10; this.height = this.firstChild.boxObject.height + 10;',
1509 class : 'tooltip' },
1510 createDescription({}, document.createTextNode( tooltip_hash[tt] ) )
1515 target.tooltip = tt;
1519 function getContextMenu (target, type) {
1522 if (type == 'subfield')
1523 tt = 't' + target.parentNode.parentNode.parentNode.firstChild.value + 'sf';
1526 tt = 't' + target.parentNode.firstChild.value + 'i1';
1529 tt = 't' + target.parentNode.firstChild.value + 'i2';
1531 target.setAttribute('context', tt);
1537 'a' : { 100 : 'a' },
1538 'd' : { 100 : 'd' },
1539 'e' : { 100 : 'e' },
1543 'a' : { 110 : 'a' },
1547 'a' : { 111 : 'a' },
1551 'a' : { 130 : 'a' },
1555 'a' : { 130 : 'a' },
1559 'a' : { 100 : 'a' },
1563 'a' : { 110 : 'a' },
1567 'a' : { 111 : 'a' },
1571 'a' : { 130 : 'a' },
1572 'n' : { 130 : 'n' },
1576 'a' : { 100 : 'a' },
1577 'd' : { 100 : 'd' },
1578 'q' : { 100 : 'q' },
1582 'a' : { 110 : 'a' },
1586 'a' : { 111 : 'a' },
1587 'c' : { 111 : 'c' },
1591 'a' : { 130 : 'a' },
1595 'a' : { 100 : 'a' },
1599 'a' : { 110 : 'a' },
1603 'a' : { 111 : 'a' },
1607 'a' : { 130 : 'a' },
1611 'a' : { 100 : 'a' },
1612 'd' : { 100 : 'd' },
1613 'q' : { 100 : 'q' },
1614 't' : { 100 : 't' },
1641 'a' : { 110 : 'a' },
1642 'd' : { 110 : 'd' },
1643 't' : { 110 : 't' },
1670 'a' : { 111 : 'a' },
1671 'd' : { 111 : 'd' },
1672 't' : { 111 : 't' },
1699 'a' : { 130 : 'a' },
1703 'a' : { 148 : 'a' },
1704 'v' : { 148 : 'v' },
1705 'x' : { 148 : 'x' },
1706 'y' : { 148 : 'y' },
1710 'a' : { 150 : 'a' },
1711 'b' : { 150 : 'b' },
1738 'a' : { 151 : 'a' },
1765 'a' : { 155 : 'a' },
1793 function getAuthorityContextMenu (target, sf) {
1794 var menu_id = sf.parent().@tag + ':' + sf.@code + '-authority-context-' + sf;
1797 var old = dojo.byId( menu_id );
1799 page = auth_pages[menu_id];
1800 old.parentNode.removeChild(old);
1802 auth_pages[menu_id] = 0;
1805 var sf_popup = createMenuPopup({ id : menu_id, flex : 1 });
1807 sf_popup.addEventListener("popuphiding", function(event) {
1808 if (show_auth_menu) {
1809 show_auth_menu = false;
1810 getAuthorityContextMenu(target, sf);
1811 dojo.byId(menu_id).openPopup();
1815 context_menus.appendChild( sf_popup );
1818 dojo.forEach( acs.controlSetList(), function (acs_id) {
1819 if (acs.controlSet(acs_id).control_map[sf.parent().@tag]) found_acs.push(acs_id);
1822 if (!found_acs.length) {
1823 sf_popup.appendChild(createLabel( { value : $('catStrings').getString('staff.cat.marcedit.not_authority_field.label') } ) );
1824 target.setAttribute('context', 'clipboard');
1828 if (sf.toString().replace(/\s*/, '')) {
1829 return browseAuthority(sf_popup, menu_id, target, sf, 20, page);
1835 /* Apply the complete 1xx */
1836 function applyFullAuthority ( target, ui_sf, e4x_sf ) {
1837 var new_vals = dojo.query('*[tag^="1"]', target);
1838 return applyAuthority( target, ui_sf, e4x_sf, new_vals );
1841 function applySelectedAuthority ( target, ui_sf, e4x_sf ) {
1842 var new_vals = target.getElementsByAttribute('checked','true');
1843 return applyAuthority( target, ui_sf, e4x_sf, new_vals );
1846 function applyAuthority ( target, ui_sf, e4x_sf, new_vals ) {
1847 var field = e4x_sf.parent();
1849 for (var i = 0; i < new_vals.length; i++) {
1851 var sf_list = field.subfield;
1852 for (var j in sf_list) {
1854 if (sf_list[j].@code == new_vals[i].getAttribute('subfield')) {
1855 sf_list[j] = new_vals[i].getAttribute('value');
1856 new_vals[i].setAttribute('subfield','');
1862 for (var i = 0; i < new_vals.length; i++) {
1864 /* indicators for the authority datafield are carried over in the main entry linking subfield */
1865 if (new_vals[i].getAttribute('subfield') == '0') {
1866 field.@ind1 = new_vals[i].getAttribute('ind1');
1867 field.@ind2 = new_vals[i].getAttribute('ind2');
1870 if (!new_vals[i].getAttribute('subfield')) continue;
1872 var val = new_vals[i].getAttribute('value');
1874 var sf = <subfield code="" xmlns="http://www.loc.gov/MARC21/slim">{val}</subfield>;
1875 sf.@code = new_vals[i].getAttribute('subfield');
1877 field.insertChildAfter(field.subfield[field.subfield.length() - 1], sf);
1880 var row = marcDatafield( field );
1883 while (node.nodeName != 'row') {
1884 node = node.parentNode;
1887 node.parentNode.replaceChild( row, node );
1891 function validateAuthority (button) {
1892 var grid = document.getElementById('recGrid');
1893 var label = button.getAttribute('label');
1896 var rows = grid.lastChild.childNodes;
1897 for (var i = 0; i < rows.length; i++) {
1899 var tag = row.firstChild;
1902 dojo.forEach(acs.controlSetList(), function (acs_id) {
1904 var control_map = acs.controlSet(acs_id).control_map;
1906 if (!control_map[tag.value]) return;
1907 button.setAttribute('label', label + ' - ' + tag.value);
1909 var ind1 = tag.nextSibling;
1910 var ind2 = ind1.nextSibling;
1911 var subfields = ind2.nextSibling.childNodes;
1914 for (var j = 0; j < subfields.length; j++) {
1915 var sf = subfields[j];
1916 sf_list.push( [ sf.childNodes[1].value, sf.childNodes[2].value ] );
1919 var matches = acs.findMatchingAuthorities(
1922 'subfields' : sf_list
1926 // matches = [ { "$csetId" : [ ... ] } ]
1929 if (matches[0]) { // probably set
1930 for (var cset in matches[0]) {
1931 var arr = matches[0][cset];
1933 // protect against errant empty string values
1934 if (arr.length == 1 && arr[0] == '')
1943 // XXX If adt, etc should be validated separately from vxz, etc then move this up into the above for loop
1944 for (var j = 0; j < subfields.length; j++) {
1945 var sf = subfields[j];
1947 dojo.removeClass(sf.childNodes[2], 'marcValidated');
1948 dojo.addClass(sf.childNodes[2], 'marcUnvalidated');
1950 dojo.removeClass(sf.childNodes[2], 'marcUnvalidated');
1951 dojo.addClass(sf.childNodes[2], 'marcValidated');
1955 if (found) done = true;
1959 button.setAttribute('label', label);
1966 function validateBibField (tags, searches) {
1967 var url = "/gateway?input_format=json&format=xml&service=open-ils.search&method=open-ils.search.authority.validate.tag";
1968 url += '¶m="tags"¶m=' + js2JSON(tags);
1969 url += '¶m="searches"¶m=' + js2JSON(searches);
1972 var req = new XMLHttpRequest();
1973 req.open('GET',url,false);
1981 function searchAuthority (term, tag, sf, limit) {
1982 var url = "/gateway?input_format=json&format=xml&service=open-ils.search&method=open-ils.search.authority.fts";
1983 url += '¶m="term"¶m="' + term + '"';
1984 url += '¶m="limit"¶m=' + limit;
1985 url += '¶m="tag"¶m=' + tag;
1986 url += '¶m="subfield"¶m="' + sf + '"';
1989 var req = new XMLHttpRequest();
1990 req.open('GET',url,false);
1997 /* TODO new authority browse support for context sets, and use that here */
1998 function browseAuthority (sf_popup, menu_id, target, sf, limit, page) {
1999 dojo.require('dojox.xml.parser');
2001 var target_tag = sf.parent().@tag.toString();
2004 dojo.forEach( acs.controlSetList(), function (acs_id) {
2005 if (acs.controlSet(acs_id).control_map[target_tag]) found_acs.push(acs_id);
2009 if (!found_acs.length) {
2010 target.setAttribute('context', 'clipboard');
2013 cmap = acs.controlSet(found_acs[0]).control_map;
2016 // map tag + subfield to the appropriate authority browse axis:
2017 // currently authority.author, authority.subject, authority.title, authority.topic
2018 // based on mappings in OpenILS::Application::SuperCat, though Authority Control
2019 // Sets will change that
2021 var axis_list = acs.bibFieldBrowseAxes( target_tag );
2023 // No matching tag means no authorities to search - shortcut
2024 if (axis_list.length == 0) {
2025 target.setAttribute('context', 'clipboard');
2029 var type = 'authority.' + axis_list[0]; // Just take the first for now
2030 // TODO support multiple axes ... loop?
2040 var sf_list = sf.parent().subfield;
2041 for ( var i in sf_list) {
2042 if (!cmap[target_tag][sf_list[i].@code.toString()]) {
2043 continue; // This is not a controlled subfield, such as $0
2045 sf_string += sf_list[i].toString() + ' ';
2046 if (sf_list[i] === sf) break;
2049 var url = '/opac/extras/browse/marcxml/'
2051 + '/1' // OU - currently unscoped
2057 // would be good to carve this out into a separate function
2058 dojo.xhrGet({"url":url, "sync": true, "preventCache": true, "handleAs":"xml", "load": function(records) {
2059 var create_menu = createMenu({ label: $('catStrings').getString('staff.cat.marcedit.create_authority.label')});
2061 var cm_popup = create_menu.appendChild(
2065 cm_popup.appendChild(
2066 createMenuitem({ label : $('catStrings').getString('staff.cat.marcedit.create_authority_now.label'),
2067 command : function() {
2068 // Call middle-layer function to create and save the new authority
2069 var source_f = summarizeField(sf);
2070 var new_auth = fieldmapper.standardRequest(
2071 ["open-ils.cat", "open-ils.cat.authority.record.create_from_bib"],
2072 [source_f, xulG.marc_control_number_identifier, ses()]
2074 if (new_auth && new_auth.id()) {
2075 addNewAuthorityID(new_auth, sf, target);
2081 cm_popup.appendChild(
2082 createMenuitem({ label : $('catStrings').getString('staff.cat.marcedit.create_authority_edit.label'),
2083 command : function() {
2084 // Generate the new authority by calling the new middle-layer
2085 // function (a non-saving variant), then display in another
2087 var source_f = summarizeField(sf);
2088 var authtoken = ses();
2089 dojo.require('openils.PermaCrud');
2090 var pcrud = new openils.PermaCrud({"authtoken": authtoken});
2091 var rec = fieldmapper.standardRequest(
2092 ["open-ils.cat", "open-ils.cat.authority.record.create_from_bib.readonly"],
2093 { "params": [source_f, xulG.marc_control_number_identifier] }
2095 loadMarcEditor(pcrud, rec, target, sf);
2100 sf_popup.appendChild(create_menu);
2101 sf_popup.appendChild( createComplexXULElement( 'menuseparator' ) );
2103 // append "Previous page" results browser
2104 sf_popup.appendChild(
2105 createMenuitem({ label : $('catStrings').getString('staff.cat.marcedit.previous_page.label'),
2106 command : function(event) {
2107 auth_pages[menu_id] -= 1;
2108 show_auth_menu = true;
2112 sf_popup.appendChild( createComplexXULElement( 'menuseparator' ) );
2114 dojo.query('record', records).forEach(function(record) {
2118 var auth_id = dojox.xml.parser.textContent(dojo.query('datafield[tag="901"]', record).query('subfield[code="c"]')[0]);
2120 if (dojo.query('controlfield[tag="003"]', record).length > 0) {
2121 auth_org = dojox.xml.parser.textContent(dojo.query('controlfield[tag="003"]', record)[0]);
2124 // Grab the fields with tags beginning with 1 (main entries) and iterate through the subfields
2125 dojo.query('datafield[tag^="1"]', record).forEach(function(field) {
2126 dojo.query('subfield', field).forEach(function(subfield) {
2130 main_text += dojox.xml.parser.textContent(subfield);
2134 // Grab the fields with tags beginning with 4 (see from entries) and iterate through the subfields
2135 dojo.query('datafield[tag^="4"]', record).forEach(function(field) {
2137 dojo.query('subfield', field).forEach(function(subfield) {
2141 see_text += dojox.xml.parser.textContent(subfield);
2143 see_from.push($('catStrings').getFormattedString('staff.cat.marcedit.authority_see_from', [see_text]));
2146 // Grab the fields with tags beginning with 5 (see also entries) and iterate through the subfields
2147 dojo.query('datafield[tag^="5"]', record).forEach(function(field) {
2149 dojo.query('subfield', field).forEach(function(subfield) {
2153 see_text += dojox.xml.parser.textContent(subfield);
2155 see_also.push($('catStrings').getFormattedString('staff.cat.marcedit.authority_see_also', [see_text]));
2158 buildAuthorityPopup(main_text, record, auth_org, auth_id, sf_popup, target, sf);
2160 dojo.forEach(see_from, function(entry_text) {
2161 buildAuthorityPopup(entry_text, record, auth_org, auth_id, sf_popup, target, sf, "font-style: italic; margin-left: 2em;");
2164 // To-do: instead of launching the standard selector menu, invoke
2165 // a new authority search using the 5XX entry text
2166 dojo.forEach(see_also, function(entry_text) {
2167 buildAuthorityPopup(entry_text, record, auth_org, auth_id, sf_popup, target, sf, "font-style: italic; margin-left: 2em;");
2172 if (sf_popup.childNodes.length == 0) {
2173 sf_popup.appendChild(createLabel( { value : $('catStrings').getString('staff.cat.marcedit.no_authority_match.label') } ) );
2175 // append "Next page" results browser
2176 sf_popup.appendChild( createComplexXULElement( 'menuseparator' ) );
2177 sf_popup.appendChild(
2178 createMenuitem({ label : $('catStrings').getString('staff.cat.marcedit.next_page.label'),
2179 command : function(event) {
2180 auth_pages[menu_id] += 1;
2181 show_auth_menu = true;
2187 target.setAttribute('context', menu_id);
2193 function buildAuthorityPopup (entry_text, record, auth_org, auth_id, sf_popup, target, sf, style) {
2194 var grid = dojo.query('[name="authority-marc-template"]')[0].cloneNode(true);
2195 grid.setAttribute('name','-none-');
2196 grid.setAttribute('style','overflow:scroll');
2198 var submenu = createMenu( { "label": entry_text } );
2200 var popup = createMenuPopup({ "flex": "1" });
2202 submenu.setAttribute('style', style);
2203 popup.setAttribute('style', 'font-style: normal; margin-left: 0em;');
2205 submenu.appendChild(popup);
2207 dojo.query('datafield[tag^="1"]', record).forEach(function(field) {
2208 buildAuthorityPopupSelector(field, grid, auth_org, auth_id);
2210 dojo.query('datafield[tag^="4"]', record).forEach(function(field) {
2211 buildAuthorityPopupSelector(field, grid, auth_org, auth_id);
2213 dojo.query('datafield[tag^="5"]', record).forEach(function(field) {
2214 buildAuthorityPopupSelector(field, grid, auth_org, auth_id);
2217 grid.hidden = false;
2218 popup.appendChild( grid );
2222 { label : $('catStrings').getString('staff.cat.marcedit.apply_selected.label'),
2223 command : function (event) {
2224 applySelectedAuthority(event.target.previousSibling, target, sf);
2231 popup.appendChild( createComplexXULElement( 'menuseparator' ) );
2235 { label : $('catStrings').getString('staff.cat.marcedit.apply_full.label'),
2236 command : function (event) {
2237 applyFullAuthority(event.target.previousSibling.previousSibling.previousSibling, target, sf);
2244 sf_popup.appendChild( submenu );
2247 function buildAuthorityPopupSelector (field, grid, auth_org, auth_id) {
2248 var row = createRow(
2250 createLabel( { "value" : dojo.attr(field, 'tag') } ),
2251 createLabel( { "value" : dojo.attr(field, 'ind1') } ),
2252 createLabel( { "value" : dojo.attr(field, 'ind2') } )
2255 var sf_box = createHbox();
2256 dojo.query('subfield', field).forEach(function(subfield) {
2259 { "label" : '\u2021' + dojo.attr(subfield, 'code') + ' ' + dojox.xml.parser.textContent(subfield),
2260 "subfield" : dojo.attr(subfield, 'code'),
2261 "tag" : dojo.attr(field, 'tag'),
2262 "value" : dojox.xml.parser.textContent(subfield)
2266 row.appendChild(sf_box);
2269 // Append the authority linking subfield only for main entries
2270 if (dojo.attr(field, 'tag').charAt(0) == '1') {
2273 { "label" : '\u2021' + '0' + ' (' + auth_org + ')' + auth_id,
2275 "tag" : dojo.attr(field, 'tag'),
2276 "ind1" : dojo.attr(field, 'ind1'),
2277 "ind2" : dojo.attr(field, 'ind2'),
2278 "value" : '(' + auth_org + ')' + auth_id
2283 row.appendChild(sf_box);
2285 grid.lastChild.appendChild(row);
2288 function summarizeField(sf) {
2296 source_f.tag = sf.parent().@tag.toString();
2297 source_f.ind1 = sf.parent().@ind1.toString();
2298 source_f.ind2 = sf.parent().@ind2.toString();
2301 dojo.forEach( acs.controlSetList(), function (acs_id) {
2302 if (acs.controlSet(acs_id).control_map[sf.parent().@tag]) found_acs.push(acs_id);
2306 if (!found_acs.length) {
2309 cmap = acs.controlSet(found_acs[0]).control_map;
2312 for (var i = 0; i < sf.parent().subfield.length(); i++) {
2313 var sf_iter = sf.parent().subfield[i];
2315 /* Filter out subfields that are not controlled for this tag */
2316 if (!cmap[source_f.tag][sf_iter.@code.toString()]) {
2320 source_f.subfields.push([sf_iter.@code.toString(), sf_iter.toString()]);
2326 function buildBibSourceList (authtoken, recId) {
2327 /* TODO: Work out how to set the bib source of the bre that does not yet
2328 * exist - this is specifically in the case of Z39.50 imports. Right now
2329 * we just avoid populating and showing the config.bib_source list
2335 var bib = xulG.record.bre;
2337 dojo.require('openils.PermaCrud');
2339 // cbsList = the XUL menulist that contains the available bib sources
2340 var cbsList = dojo.byId('bib-source-list');
2342 // bibSources = an array containing all of the bib source objects
2343 var bibSources = new openils.PermaCrud({"authtoken": authtoken}).retrieveAll('cbs');
2345 // A tad ugly, but gives us the index of the bib source ID in cbsList
2347 var cbsListArr = [];
2348 dojo.forEach(bibSources, function (item) {
2349 cbsList.appendItem(item.source(), item.id());
2350 cbsListArr[item.id()] = x;
2354 // Show the current value of the bib source for this record
2355 cbsList.selectedIndex = cbsListArr[bib.source()];
2357 // Display the bib source selection widget
2358 dojo.byId('bib-source-list-caption').hidden = false;
2359 dojo.byId('bib-source-list').hidden = false;
2360 dojo.byId('bib-source-list-button').disabled = true;
2361 dojo.byId('bib-source-list-button').hidden = false;
2364 // Fired when the "Update Source" button is clicked
2365 // Updates the value of the bib source for the current record
2366 function updateBibSource() {
2367 var authtoken = ses();
2368 var cbs = dojo.byId('bib-source-list').selectedItem.value;
2369 var recId = xulG.record.id;
2370 var pcrud = new openils.PermaCrud({"authtoken": authtoken});
2371 var bib = pcrud.retrieve('bre', recId);
2372 if (bib.source() != cbs) {
2374 bib.ischanged = true;
2379 function onBibSourceSelect() {
2380 var cbs = dojo.byId('bib-source-list').selectedItem.value;
2381 var bib = xulG.record.bre;
2382 if (bib.source() != cbs) {
2383 dojo.byId('bib-source-list-button').disabled = false;
2385 dojo.byId('bib-source-list-button').disabled = true;
2389 function addNewAuthorityID(authority, sf, target) {
2390 var id_sf = <subfield code="0" xmlns="http://www.loc.gov/MARC21/slim">({xulG.marc_control_number_identifier}){authority.id()}</subfield>;
2391 sf.parent().appendChild(id_sf);
2392 var new_sf = marcSubfield(id_sf);
2395 while (dojo.attr(node, 'name') != 'sf_box') {
2396 node = node.parentNode;
2398 node.appendChild( new_sf );
2400 alert($('catStrings').getString('staff.cat.marcedit.create_authority_success.label'));
2403 function loadMarcEditor(pcrud, marcxml, target, sf) {
2405 To run in Firefox directly, must set signed.applets.codebase_principal_support
2406 to true in about:config
2408 win = window.open('/xul/server/cat/marcedit.xul', '_blank', 'chrome'); // XXX version?
2410 // Match marc2are.pl last_xact_id format, roughly
2412 var xact_id = 'IMPORT-' + Date.parse(now);
2415 "record": {"marc": marcxml, "rtype": "are"},
2417 "label": $('catStrings').getString('staff.cat.marcedit.save.label'),
2418 "func": function(xmlString) {
2419 var rec = new are();
2420 rec.marc(xmlString);
2421 rec.last_xact_id(xact_id);
2424 "oncomplete": function (r, objs) {
2425 var new_rec = objs[0];
2430 addNewAuthorityID(new_rec, sf, target);