]> git.evergreen-ils.org Git - working/Evergreen.git/blob - Open-ILS/web/opac/skin/default/js/holds.js
Merge branch 'hold_queue_position_sorting'
[working/Evergreen.git] / Open-ILS / web / opac / skin / default / js / holds.js
1 var holdsOrgSelectorBuilt = false;
2 var holdArgs;
3
4 /* 
5 note: metarecord holds have a holdable_formats field that contains
6 item_type(s)-item_forms(s)-language
7 item_form and language are optional - if language exist and no 
8 item_form is specified, use item_type(s)--language
9 */
10
11 var noEmailMessage;
12 var noEmailMessageXUL;
13
14 var holdTargetTypeMap = {
15     M : 'metarecord',
16     T : 'record',
17     V : 'volume',
18     I : 'issuance',
19     C : 'copy',
20     P : 'part'
21 };
22
23
24
25 function holdsHandleStaff() {
26
27     // if we know the recipient's barcode, use it
28     if(xulG.patron_barcode) return _holdsHandleStaff();
29
30         swapCanvas($('xulholds_box'));
31         $('xul_recipient_barcode').focus();
32         $('xul_recipient_barcode').onkeypress = function(evt) 
33                 {if(userPressedEnter(evt)) { _holdsHandleStaff(); } };
34         $('xul_recipient_barcode_submit').onclick = _holdsHandleStaff;
35         $('xul_recipient_me').onclick = _holdsHandleStaffMe;
36
37         $('xul_recipient_barcode').onkeyup = function(evt) {
38         if($('xul_recipient_barcode').value == '') 
39             $('xul_recipient_me').disabled = false;
40         else
41             $('xul_recipient_me').disabled = true;
42     };
43 }
44
45 $('holds_frozen_thaw_input').onchange = 
46         function(){holdsVerifyThawDateUI('holds_frozen_thaw_input');}
47 $('holds_frozen_thaw_input').onkeyup = 
48         function(){holdsVerifyThawDateUI('holds_frozen_thaw_input');}
49
50 function _holdsHandleStaffMe() {
51         holdArgs.recipient = G.user;
52         holdsDrawEditor();
53 }
54
55 function _holdsHandleStaff() {
56         var barcode = xulG.patron_barcode;
57     if(!barcode) {
58         barcode = $('xul_recipient_barcode').value;
59         if(xulG.get_barcode) {
60             // We have a "complete the barcode" function, call it (actor = users only)
61             var new_barcode = xulG.get_barcode(window, 'actor', barcode);
62             // If we got a result (boolean false is "no result") check it
63             if(new_barcode) {
64                 // user_false string means they picked "None of the above"
65                 // Abort before any other events can fire
66                 if(new_barcode == "user_false") return;
67                 // No error means we have a (hopefully valid) completed barcode to use.
68                 // Otherwise, fall through to other methods of checking
69                 if(typeof new_barcode.ilsevent == 'undefined')
70                     barcode = new_barcode.barcode;
71             }
72         }
73     }
74         var user = grabUserByBarcode( G.user.session, barcode );
75
76         var evt;
77         if(evt = checkILSEvent(user)) {
78                 alertILSEvent(user);
79                 return;
80         }
81
82         if(!barcode || !user) {
83                 alertId('holds_invalid_recipient', barcode);
84                 return
85         }
86
87         grabUserPrefs(user);
88
89         holdArgs.recipient = user;
90         holdsDrawEditor();
91 }
92
93
94
95 /** args:
96   * record, volume, copy (ids)
97   * request, recipient, editHold (objects)
98   */
99
100 function holdsDrawEditor(args) {
101
102         holdArgs = (args) ? args : holdArgs;
103
104     if(!noEmailMessage)
105         noEmailMessage = $('holds_email').removeChild($('holds.no_email'));
106
107     if(!noEmailMessageXUL)
108         noEmailMessageXUL = $('holds_email').removeChild($('holds.no_email.xul'));
109
110         if(isXUL() && holdArgs.recipient == null 
111                         && holdArgs.editHold == null) {
112                 holdsHandleStaff();
113                 return;
114         }
115
116         if(!holdArgs.recipient) holdArgs.recipient = G.user;
117         if(!holdArgs.requestor) holdArgs.requestor = G.user;
118
119         if(!(holdArgs.requestor && holdArgs.requestor.session)) {
120                 detachAllEvt('common','locationChanged');
121                 attachEvt('common','loggedIn', holdsDrawEditor)
122                 initLogin();
123                 return;
124         }
125
126         if(holdArgs.editHold) // flesh the args with the existing hold 
127                 holdArgsFromHold(holdArgs.editHold, holdArgs);
128
129 //    $('holds_parts_selector').style.border = 'auto';
130     holdArgs.partsSuggestionMade = false;
131
132         holdsDrawWindow();
133 }
134
135
136 // updates the edit window with the existing hold's data 
137 function _holdsUpdateEditHold() {
138
139         var hold = holdArgs.editHold;
140         var qstats = holdArgs.status;
141
142         var orgsel = $('holds_org_selector');
143     var frozenbox = $('holds_frozen_chkbox');
144
145         setSelector(orgsel, hold.pickup_lib());
146
147         if( hold.capture_time() || qstats.status > 2 ) {
148         frozenbox.disabled = true;
149         $('holds_frozen_thaw_input').disabled = true;
150         if(qstats.status == 3) {
151             // no pickup lib changes while in-transit
152                     orgsel.disabled = true;
153         } else {
154             var orgs = fetchPermOrgs('UPDATE_PICKUP_LIB_FROM_HOLDS_SHELF');
155             if(orgs[0] == -1)
156                         orgsel.disabled = true;
157         }
158     } else {
159                 orgsel.disabled = false;
160         frozenbox.disabled = false;
161     }
162
163
164         $('holds_submit').onclick = holdsEditHold;
165         $('holds_update').onclick = holdsEditHold;
166
167         if(hold.phone_notify()) {
168                 $('holds_enable_phone').checked = true;
169                 $('holds_phone').value = hold.phone_notify();
170
171         } else {
172                 $('holds_phone').disabled = true;
173                 $('holds_enable_phone').checked = false;
174         }
175
176         if(isTrue(hold.email_notify())) {
177                 $('holds_enable_email').checked = true;
178
179         } else {
180                 $('holds_enable_email').checked = false;
181         }
182
183     dijit.byId('holds_expire_time').setValue(dojo.date.stamp.fromISOString(hold.expire_time()));
184
185     /* populate the hold freezing info */
186     if(!frozenbox.disabled && isTrue(hold.frozen())) {
187         frozenbox.checked = true;
188         unHideMe($('hold_frozen_thaw_row'));
189         if(hold.thaw_date()) {
190             dijit.byId('holds_frozen_thaw_input').setValue(dojo.date.stamp.fromISOString(hold.thaw_date()));
191         } else {
192             dijit.byId('holds_frozen_thaw_input').setValue('');
193         }
194     } else {
195         frozenbox.checked = false;
196         dijit.byId('holds_frozen_thaw_input').setValue('');
197         hideMe($('hold_frozen_thaw_row'));
198     }
199 }
200
201 function holdsEditHold() {
202         var hold = holdsBuildHoldFromWindow();
203         if(!hold) return;
204         holdsUpdate(hold);
205         showCanvas();
206         if(holdArgs.onComplete)
207                 holdArgs.onComplete(hold);
208 }
209
210 function holdArgsFromHold(hold, oargs) {
211         var args = (oargs) ? oargs : {};
212         args.type = hold.hold_type();
213         var target = hold.target();
214     args[holdTargetTypeMap[args.type]] = target;
215         return args;
216 }
217
218 function holdFetchObjects(hold, doneCallback) {
219
220         var args = (hold) ? holdArgsFromHold(hold) : holdArgs;
221
222         var type = args.type;
223
224         if( type == 'C' ) {
225
226                 if( args.copyObject ) {
227
228                         args.copy = args.copyObject.id();
229                         args.volume = args.copyObject.call_number();
230                         _h_set_vol(args, doneCallback);
231
232                 } else {
233                         var creq = new Request(FETCH_COPY, args.copy);
234
235                         creq.callback(
236                                 function(r) {
237                                         var cp = r.getResultObject();
238                                         args.copyObject = cp;
239                                         args.volume = args.copyObject.call_number();
240                                         _h_set_vol(args, doneCallback);
241                                 }
242                         );
243                         creq.send();
244                 }
245         } else {
246                 if( type == 'V' ) {
247                         _h_set_vol(args, doneCallback);
248
249         } else if( type == 'I' ) {
250             _h_set_issuance(args, doneCallback);
251
252         } else if( type == 'P' ) {
253             _h_set_parts(args, doneCallback);
254
255                 } else {
256                         if( type == 'T') {
257                                 _h_set_rec(args, doneCallback);
258                         } else {
259                                 _h_set_rec_descriptors(args, doneCallback);
260                         }
261                 }
262         }
263
264         return args;
265 }
266
267 function _h_set_parts(args, doneCallback) {
268
269     var preq = new Request(
270         'open-ils.fielder:open-ils.fielder.bmp.atomic',
271         {"cache":1, "fields":["label", "record"],"query": {"id":args.part}}
272     );
273
274     preq.callback(
275         function(r) {
276             var part = r.getResultObject()[0];
277             args.record = part.record;
278             args.partObject = part;
279             _h_set_rec(args, doneCallback);
280         }
281     );
282
283     preq.send();
284 }
285
286 function _h_set_vol(args, doneCallback) {
287
288         if( args.volumeObject ) {
289                 args.volume = args.volumeObject.id();
290                 args.record = args.volumeObject.record();
291                 _h_set_rec(args, doneCallback);
292
293         } else {
294
295                 var vreq = new Request(FETCH_VOLUME, args.volume);
296                 vreq.callback(
297                         function(r) {
298                                 var vol = r.getResultObject();
299                                 args.volumeObject = vol;
300                                 args.record = vol.record();
301                                 _h_set_rec(args, doneCallback);
302                         }
303                 );
304                 vreq.send();
305         }
306 }
307
308 function _h_set_issuance(args, doneCallback) {
309
310         if( args.issuanceObject ) {
311                 args.issuance = args.issuanceObject.id();
312                 args.record = args.issuanceObject.subscription().record_entry();
313                 _h_set_rec(args, doneCallback);
314
315         } else {
316
317                 var vreq = new Request(FETCH_ISSUANCE, [args.issuance]);
318                 vreq.callback(
319                         function(r) {
320                                 var issuance = r.getResultObject()[0];
321                                 args.issuanceObject = issuance;
322                                 args.record = issuance.subscription().record_entry();
323                                 _h_set_rec(args, doneCallback);
324                         }
325                 );
326                 vreq.send();
327         }
328 }
329
330 function _h_set_rec(args, doneCallback) {
331
332         if(args.recordObject) 
333                 args.record = args.recordObject.doc_id();
334         else 
335                 args.recordObject = findRecord( args.record, 'T' );
336         
337         if( args.type == 'T' || args.type == 'M' )  {
338                 _h_set_rec_descriptors(args, doneCallback);
339         //} else if(args.type == 'P') {
340         //_h_get_parts(args, doneCallback);
341     } else {
342                 if(doneCallback) doneCallback(args);
343     }
344 }
345
346
347 function _h_set_rec_descriptors(args, doneCallback) {
348
349     if( ! args.pickup_lib )
350         args.pickup_lib = getSelectorVal($('holds_org_selector'));
351
352     if(args.pickup_lib === null)
353         args.pickup_lib = args.recipient.home_ou();
354
355         // grab the list of record desciptors attached to this records metarecord 
356         if( ! args.recordDescriptors )  {
357                 var params = { pickup_lib: args.pickup_lib };
358
359         if (args.type == 'M') {
360                 if( !args.metarecord && args.record) {
361                 params.metarecord = args.metarecord = args.record;
362                 delete(args.record);
363                 } else {
364                                 params.metarecord = args.metarecordObject.doc_id();
365                 }
366         } else {
367                 params.record = args.record;
368         }
369
370                 if( ! args.record ) {
371                         if( args.metarecord )
372                                 params.metarecord = args.metarecord;
373                         else 
374                                 params.metarecord = args.metarecordObject.doc_id();
375                 }
376
377                 var req = new Request(FETCH_MR_DESCRIPTORS, params );
378                 req.callback(
379                         function(r) {
380                                 var data = r.getResultObject();
381                                 args.recordDescriptors = args.recordDescriptors = data.descriptors;
382                                 args.metarecord = args.metarecord = data.metarecord;
383                                 if( args.type == 'M' && ! args.metarecordObject) 
384                                         args.metarecordObject = args.metarecordObject = findRecord(args.metarecord, 'M');       
385
386                 _h_get_parts(args, doneCallback);
387                         }
388                 );
389                 req.send();
390
391         } else {
392         _h_get_parts(args, doneCallback);
393         }
394
395         return args;
396 }
397
398 function _h_get_parts(args, doneCallback) {
399
400     if(args.type == 'M' || args.editHold || args.holdParts) {
401         if(doneCallback) 
402             doneCallback(args);
403
404     } else {
405
406                 var req = new Request(
407             'open-ils.search:open-ils.search.biblio.record_hold_parts', 
408                     {pickup_lib: args.pickup_lib, record: args.record}
409         );
410
411                 req.callback(
412                         function(r) {
413                                 args.recordParts = r.getResultObject();
414                 if(doneCallback)
415                     doneCallback(args);
416                         }
417                 );
418                 req.send();
419     }
420 }
421
422
423
424 function holdsDrawWindow() {
425         swapCanvas($('holds_box'));
426         $('holds_cancel').onclick = function(){ runEvt('common', 'holdUpdateCanceled'), showCanvas() };
427         $('holds_submit').onclick = function(){holdsPlaceHold(holdsBuildHoldFromWindow())};
428         $('holds_update').onclick = function(){holdsPlaceHold(holdsBuildHoldFromWindow())};
429     $('holds_org_selector').onchange = function(){holdsDrawWindow()};
430         holdFetchObjects(null, 
431                 function(){
432                         __holdsDrawWindow();
433
434                         if(holdArgs.editHold) {
435                                 hideMe($('holds_submit'));
436                                 unHideMe($('holds_update'));
437                                 var req = new Request(FETCH_HOLD_STATUS, 
438                                         G.user.session, holdArgs.editHold.id());
439                                 req.send(true);
440                                 holdArgs.status = req.result();
441                                 _holdsUpdateEditHold();
442                         }  
443                 }
444         );
445 }
446
447 function __holdsDrawWindow() {
448
449         var rec = holdArgs.recordObject;
450         var vol = holdArgs.volumeObject;
451         var copy = holdArgs.copyObject;
452         var mr = holdArgs.metarecordObject;
453
454         rec = (rec) ? rec : mr;
455
456         if(!holdsOrgSelectorBuilt) {
457                 holdsBuildOrgSelector(null,0);
458                 holdsOrgSelectorBuilt = true;
459                 var selector = $('holds_org_selector');
460
461                 /*
462                 var o_loc = findOrgUnit(getOrigLocation());
463                 var t = findOrgType(o_loc.ou_type());
464                 if( t.can_have_users() ) 
465                         setSelector(selector, o_loc.id());
466                 else 
467                 */
468
469                 setSelector(selector, holdArgs.recipient.home_ou());
470         
471         }
472
473         /*
474         if(isXUL()) {
475                 var dsel = $('holds_depth_selector');
476                 unHideMe($('holds_depth_selector_row'));
477                 if(dsel.getElementsByTagName('option').length == 0) {
478                         var types = globalOrgTypes;
479                         var depth = findOrgDepth(G.user.ws_ou());
480                         iterate(types, 
481                                 function(t) {
482                                         if(t.depth() > depth) return;
483                                         insertSelectorVal(dsel, -1, t.opac_label(), t.depth());
484                                 }
485                         );
486                 }
487         }
488         */
489
490         appendClear($('holds_recipient'), text(
491                 holdArgs.recipient.family_name() + ', ' +  
492                         holdArgs.recipient.first_given_name()));
493         appendClear($('holds_title'), text(rec.title()));
494         appendClear($('holds_author'), text(rec.author()));
495
496     if( holdArgs.type == 'I' ) {
497                 unHideMe($('holds_type_row'));
498         unHideMe($('holds_is_issuance'));
499         unHideMe($('holds_issuance_row'));
500         appendClear($('holds_issuance_label'), text(holdArgs.issuanceObject.label()));
501
502     } else if( holdArgs.type == 'V' || holdArgs.type == 'C' ) {
503
504                 unHideMe($('holds_type_row'));
505                 unHideMe($('holds_cn_row'));
506                 appendClear($('holds_cn'), text(holdArgs.volumeObject.label()));
507
508                 if( holdArgs.type == 'V'  ) {
509                         unHideMe($('holds_is_cn'));
510                         hideMe($('holds_is_copy'));
511
512                 } else {
513                         hideMe($('holds_is_cn'));
514                         unHideMe($('holds_is_copy'));
515                         unHideMe($('holds_copy_row'));
516                         appendClear($('holds_copy'), text(holdArgs.copyObject.barcode()));
517                 }
518
519         } else {
520                 hideMe($('holds_type_row'));
521                 hideMe($('holds_copy_row'));
522                 hideMe($('holds_cn_row'));
523                 hideMe($('holds_issuance_row'));
524         }
525
526     if(holdArgs.recordParts && holdArgs.recordParts.length) {
527         var selector = $('holds_parts_selector');
528         unHideMe($('holds_parts_row'));
529         unHideMe(selector);
530
531         var nodeList = [];
532         dojo.forEach(selector.options, 
533             function(node) { if(node.value != '') nodeList.push(node) } );
534
535         dojo.forEach(nodeList, function(node) { selector.removeChild(node); });
536
537         dojo.forEach(
538             holdArgs.recordParts, 
539             function(part) {
540                 insertSelectorVal(selector, -1, part.label, part.id);
541             }
542         );
543
544     } else if(holdArgs.type == 'P') {
545         unHideMe($('holds_parts_row'));
546         unHideMe($('holds_parts_label'));
547             appendClear( $('holds_parts_label'), text(holdArgs.partObject.label));
548     }
549
550         removeChildren($('holds_format'));
551
552         var mods_formats = rec.types_of_resource();
553         var formats;
554
555         if(holdArgs.recordDescriptors)
556                 formats = holdArgs.recordDescriptors[0].item_type();
557
558         if( holdArgs.type == 'T' ) {
559                 var desc = grep( holdArgs.recordDescriptors,
560                         function(i) {
561                                 return (i.record() == holdArgs.record); 
562                         }
563                 );
564                 formats = desc[0].item_type();
565         }
566
567         if( holdArgs.type == 'M' ) {
568                 var mr_formats;
569                 if(holdArgs.editHold){
570                         mr_formats = holdArgs.editHold.holdable_formats();
571                 }else{
572                         mr_formats = ''; // collect the item_type()s from all holdArgs.recordDescriptors
573                         for(var desc in holdArgs.recordDescriptors){
574                 if (!holdArgs.recordDescriptors[desc].item_type()) continue;
575                                 mr_formats += holdArgs.recordDescriptors[desc].item_type();
576                         }
577
578             var first_form = 1;
579                         for(var desc in holdArgs.recordDescriptors){
580                 if (!holdArgs.recordDescriptors[desc].item_form()) continue;
581                 if (first_form) {
582                     mr_formats += '-';
583                     first_form = 0;
584                 }
585                                 mr_formats += holdArgs.recordDescriptors[desc].item_form();
586                         }
587
588
589                 }
590                 
591                 var data = holdsParseMRFormats(mr_formats);
592                 mods_formats = data.mods_formats;
593                 formats = data.formats;
594         }
595
596
597         for( var i in mods_formats ) {
598                 var res = mods_formats[i];
599                 var img = elem("img");
600                 setResourcePic(img, res);
601                 $('holds_format').appendChild(img);
602                 if(formats)
603                         $('holds_format').appendChild(text(' '+ MARCTypeToFriendly(formats[i]) +' '));
604                 else
605                         $('holds_format').appendChild(text(' '+ mods_formats[i] +' '));
606                 $('holds_format').appendChild(elem('br'));
607         }
608
609
610         $('holds_phone').value = holdArgs.recipient.day_phone();
611         appendClear( $('holds_email'), text(holdArgs.recipient.email()));
612
613         var pref = holdArgs.recipient.prefs[PREF_HOLD_NOTIFY];
614
615         if(pref) {
616                 if( ! pref.match(/email/i) ) {
617                         $('holds_enable_email').checked = false;
618                 } else {
619                         $('holds_enable_email').checked = true;
620                 }
621
622                 if( ! pref.match(/phone/i) ) {
623                         $('holds_phone').disabled = true;
624                         $('holds_enable_phone').checked = false;
625                 } else {
626                         $('holds_phone').disabled = false;
627                         $('holds_enable_phone').checked = true;
628                 }
629         }
630
631     if(!holdArgs.recipient.email()) {
632                 $('holds_enable_email').checked = false;        
633                 $('holds_enable_email').disabled = true;
634         var message;
635         if(isXUL()) {
636             message = noEmailMessageXUL.cloneNode(true);
637                 appendClear($('holds_email'), message);
638         } else {
639             message = noEmailMessage.cloneNode(true);
640                 appendClear($('holds_email'), message);
641             $('holds.no_email.my_account').setAttribute('href', buildOPACLink({page:MYOPAC},null,true));
642         }
643         unHideMe(message);
644     }
645
646         if(!$('holds_phone').value) 
647                 $('holds_enable_phone').checked = false;        
648
649         appendClear($('holds_physical_desc'), text(rec.physical_description()));
650
651         if(holdArgs.type == 'M') hideMe($('hold_physical_desc_row'));
652
653         holdsSetFormatSelector();
654
655     $('holds_frozen_chkbox').checked = false;
656     hideMe($('hold_frozen_thaw_row'));
657
658     var interval = fetchOrgSettingDefault(holdArgs.recipient.home_ou(), 'circ.hold_expire_interval');
659     var secs = 0;
660     if(interval) {
661         secs = interval_to_seconds(interval);
662         var expire = new Date();
663         expire.setTime(expire.getTime() + Number(secs + '000'));
664         dijit.byId('holds_expire_time').setValue(expire);
665     }
666 }
667
668 function holdsParseMRFormats(str) {
669         var data = str.split(/-/);      
670
671         var formats = [];
672         var mods_formats = [];
673
674         for( var i = 0; i < data[0].length; i++ ) {
675                 formats.push( data[0].charAt(i) );
676                 mods_formats.push( MARCFormatToMods( formats[i] ) );
677         }
678         
679         formats = uniquify(formats);
680         mods_formats = uniquify(mods_formats);
681
682         return {
683                 formats                 : formats,
684                 mods_formats    : mods_formats,
685                 lang                            : data[2],
686                 largeprint              : data[1]
687         };
688 }
689
690
691 function holdsSetFormatSelector() {
692         var type = holdArgs.type;
693         if( type == 'C' || type == 'V' || type == "I" || holdArgs.editHold ) return;
694
695         var data                                = holdsGetFormats();
696         var avail_formats       = data.avail_formats;
697         var sel_formats = data.sel_formats;
698         holdArgs.language = data.lang;
699         if( type=='M'){         
700                 hideMe($('holds_alt_formats_row_extras'));
701                 unHideMe($('holds_alt_formats_row'));   
702         }else{
703                 unHideMe($('holds_alt_formats_row_extras'));
704         }
705
706         var selector = $('hold_alt_form_selector');
707
708     for( var i = 0; i < selector.options.length; i++ ) {
709         if (selector.options[i].className.indexOf('hide_me') == -1)
710             hideMe(selector.options[i]);
711     }
712
713         for( var i = 0; i < avail_formats.length; i++ ) {
714                 var form = avail_formats[i];
715                 var opt = findFormatSelectorOptByParts(selector,form);
716         if (!opt) continue;
717                 if(type=='M') opt.selected=true;
718                 unHideMe(opt);
719         }
720
721     // If the user selects a format, P-type holds are no longer an option
722     // disable and reset the P-type form control
723     selector.onchange = function() {
724         var partsSel = $('holds_parts_selector');
725         for(var i = 0; i < selector.options.length; i++) {
726             if(selector.options[i].selected) {
727                 partsSel.selectedIndex = 0; // none selected
728                 partsSel.disabled = true;
729                 return;
730             }
731         }
732         partsSel.disabled = false;
733     }
734 }
735
736 function findFormatSelectorOptByParts( sel, val ) {
737     var parts = val.split('-');
738     for( var i = 0; i < sel.options.length; i++ ) {
739         var opt = sel.options[i];
740         var oval = opt.value;
741         var oparts = oval.split('-');
742         if( oparts[0].indexOf(parts[0]) > -1 && (!parts[1] || oparts[1].indexOf(parts[1]) > -1) ) return opt;
743     }
744     return null;
745 }
746
747 function holdsGetFormats() {
748
749         var lang;
750         var formats = [];
751         var sformats = []; // selected formats 
752
753         var type = holdArgs.type;
754         var desc = holdArgs.recordDescriptors;
755         var rec = holdArgs.record;
756         var mrec = holdArgs.metarecord;
757
758
759         if( type == 'T') {
760
761                 for( var i = 0; i < desc.length; i++ ) {
762                         var d = desc[i];
763                         if( d.record() == holdArgs.record ) {
764                                 lang = d.item_lang();
765                                 holdArgs.myFormat =  _t_f_2_format(d.item_type(), d.item_form());
766                                 sformats.push(holdArgs.myFormat);
767                                 break;
768                         }
769                 }
770
771         for( var i = 0; i < desc.length; i++ ) {
772                 var d = desc[i];
773                     if( type == 'T' && d.item_lang() != lang ) continue;
774                 formats.push( _t_f_2_format(d.item_type(), d.item_form()));
775             }
776
777         } else if( type =='M') {
778
779         // All available formats are selected by default in MR holds
780         for( var i = 0; i < desc.length; i++ ) {
781                 var d = desc[i];
782                     var _tmp_f = _t_f_2_format(d.item_type(), d.item_form());
783                 formats.push( _tmp_f );
784                 sformats.push( _tmp_f );
785         }
786         }
787
788         formats = uniquify(formats);
789         sformats = uniquify(sformats);
790
791         return {
792                 lang : lang,
793                 avail_formats : formats, 
794                 sel_formats : sformats
795         };
796 }
797
798
799
800 function _t_f_2_format(type, form) {
801         if( (type == 'a' || type == 't') && form == 's' ) return 'at-s';
802         if( form == 'd' ) return 'at-d';
803         return (type == 'a' || type == 't') ? 'at' : type;
804 }
805
806 function holdsSetSelectedFormats() {
807
808         var cn = $('holds_alt_formats_row').className;
809         if( cn && cn.match(/hide_me/) ) return;
810
811         var selector = $('hold_alt_form_selector');
812         var vals = getSelectedList(selector);
813
814         if(vals.length == 0) return;
815
816         var fstring = "";
817
818         if( contains(vals, 'at-d') || contains(vals, 'at-s') || contains(vals, 'at')) {
819                 if( contains(vals, 'at') ) {
820                         fstring = 'at';
821                 } else if (contains(vals, 'at-s') && contains(vals, 'at-d')) {
822                         fstring = 'at-sd';
823                 } else if (!contains(vals, 'at-s')) {
824                         fstring = 'at-d';
825                 } else {
826                         fstring = 'at-s';
827                 }
828         }
829
830         for( var i = 0; i < vals.length; i++ ) {
831                 var val = vals[i];
832                 if( !val.match(/at/) ) fstring = val + fstring;
833         }
834
835         if( holdArgs.language ) {
836                 if( fstring.match(/-/) )
837                         fstring = fstring + '-' + holdArgs.language;
838                 else
839                         fstring = fstring + '--' + holdArgs.language;
840         }
841
842
843         return fstring;
844 }
845
846
847 function holdsCheckPossibility(pickuplib, hold, recurse) {
848
849         var args = { 
850                 titleid : holdArgs.record,
851                 mrid : holdArgs.metarecord,
852                 volume_id : holdArgs.volume,
853                 issuanceid : holdArgs.issuance,
854                 copy_id : holdArgs.copy,
855                 hold_type : holdArgs.type,
856                 holdable_formats : holdArgs.holdable_formats,
857                 patronid : holdArgs.recipient.id(),
858                 depth : 0, 
859                 pickup_lib : pickuplib,
860         partid : holdArgs.part
861         };
862
863         if(recurse) {
864                 /* if we're calling create again (recursing), 
865                         we know that the hold possibility check already succeeded */
866                 holdHandleCreateResponse({_recurse:true, _hold:hold}, true );
867
868         } else {
869                 _debug("hold possible args = "+js2JSON(args));
870         
871                 var req = new Request(CHECK_HOLD_POSSIBLE, G.user.session, args );
872         
873                 req.request.alertEvent = false;
874                 req.request._hold = hold;
875                 req.request._recurse = recurse;
876                 req.callback(holdHandleCreateResponse);
877                 req.send();
878         }
879 }
880
881
882 function holdsBuildOrgSelector(node) {
883
884         if(!node) node = globalOrgTree;
885         if(!isTrue(node.opac_visible()) && !isXUL()) return;
886
887         var render_this_org = true;
888         var orgHiding = checkOrgHiding(); // value here is cached so not too painful with the recursion
889         if (orgHiding) {
890                 if (node.id() == globalOrgTree.id()) {
891                         node = orgHiding.org; // top of tree = org hiding context org
892                 }
893                 if ( ! orgIsMine( orgHiding.org, node, orgHiding.depth ) ) {
894                         render_this_org = false;
895                 }
896         }
897
898         if (render_this_org) {
899                 var selector = $('holds_org_selector');
900                 var index = selector.options.length;
901
902                 var type = findOrgType(node.ou_type());
903                 var indent = type.depth() - 1;
904
905                 var opt = setSelectorVal( selector, index, node.name(), node.id(), null, indent );
906                 if(!type.can_have_users()) {
907                         opt.disabled = true;
908                         addCSSClass(opt, 'disabled_option');
909                 }
910         }
911         
912         for( var i in node.children() ) {
913                 var child = node.children()[i];
914                 if(child) holdsBuildOrgSelector(child);
915         }
916 }
917
918 function holdsBuildHoldFromWindow() {
919
920         var org = getSelectorVal($('holds_org_selector'));
921         var node = findOrgUnit(org);
922         var ntype = findOrgType(node.ou_type());
923         if(!ntype.can_have_users()) {
924                 alertId('holds_pick_good_org');
925                 return;
926         }
927
928     fieldmapper.IDL.load(['ahr']);
929         var hold = new ahr();
930         if(holdArgs.editHold) {
931                 hold = holdArgs.editHold;
932                 holdArgs.editHold = null;
933         }
934
935         if( $('holds_enable_phone').checked ) {
936                 var phone = $('holds_phone').value;
937                 if( !phone || !phone.match(REGEX_PHONE) ) {
938                         alert($('holds_bad_phone').innerHTML);
939                         return null;
940                 }
941                 hold.phone_notify(phone);
942
943         } else {
944                 hold.phone_notify("");
945         }
946
947         if( $('holds_enable_email').checked ) 
948                 hold.email_notify(1);
949         else
950                 hold.email_notify(0);
951
952     var part = getSelectorVal($('holds_parts_selector'));
953     if(part) {
954         holdArgs.type = 'P';
955         holdArgs.part = part;
956     }
957
958         var target = holdArgs[holdTargetTypeMap[holdArgs.type]];
959
960     // a mono part is selected
961
962         hold.pickup_lib(org); 
963         //hold.request_lib(org); 
964         hold.requestor(holdArgs.requestor.id());
965         hold.usr(holdArgs.recipient.id());
966         hold.target(target);
967         hold.hold_type(holdArgs.type);
968
969     var expDate = dijit.byId('holds_expire_time').getValue();
970     if(expDate) {
971         var expireDate = dojo.date.stamp.toISOString(expDate);
972         expireDate = holdsVerifyThawDate(expireDate); 
973         if(expireDate)
974             hold.expire_time(expireDate);
975         else 
976             return;
977     }
978
979     // see if this hold should be frozen and for how long
980     if($('holds_frozen_chkbox').checked) {
981         hold.frozen('t');
982         unHideMe($('hold_frozen_thaw_row'));
983         var thawDate = dijit.byId('holds_frozen_thaw_input').attr('value');
984         if(thawDate) {
985             thawDate = dojo.date.stamp.toISOString(thawDate);
986             thawDate = holdsVerifyThawDate(thawDate); 
987             if(thawDate) 
988                 hold.thaw_date(thawDate);
989             else
990                 return;
991         } else {
992             hold.thaw_date(null);
993         }
994     } else {
995         hold.frozen('f');
996         hold.thaw_date(null);
997     }
998
999         //check for alternate hold formats 
1000         var fstring = holdsSetSelectedFormats();
1001         if(fstring) { 
1002                 hold.hold_type('M'); 
1003                 hold.holdable_formats(fstring);
1004                 if (fstring)
1005                         holdArgs.holdable_formats = fstring;
1006                 hold.target(holdArgs.metarecord);
1007         }
1008         return hold;
1009 }
1010         
1011 function holdsPlaceHold(hold, recurse) {
1012         if(!hold) return;
1013         swapCanvas($('check_holds_box'));
1014         holdsCheckPossibility(hold.pickup_lib(), hold, recurse);
1015 }
1016
1017
1018 function holdHandleCreateResponse(r, recurse) {
1019
1020         if(!recurse) {
1021                 var res = r.getResultObject();
1022                 if(checkILSEvent(res) || res.success != 1) {
1023                         if(res.success != 1) {
1024
1025                 if(!holdArgs.partsSuggestionMade && holdArgs.recordParts && 
1026                         holdArgs.recordParts.length && holdArgs.type == 'T') {
1027                     // T holds on records that have parts are OK, but if the record has no non-part
1028                     // copies, the hold will ultimately fail.  Suggest selecting a part to the user.
1029                     $('holds_parts_selector').style.border = '2px solid red';
1030                     holdArgs.partsSuggestionMade = true;
1031                     alert($('hold_has_parts').innerHTML);
1032                 } else {
1033                                     alert($('hold_not_allowed').innerHTML);
1034                 }
1035                         } else {
1036                                 if( res.textcode == 'PATRON_BARRED' ) {
1037                                         alertId('hold_failed_patron_barred');
1038                         } else {
1039                                         alert($('hold_not_allowed').innerHTML);
1040                                 }
1041                         }
1042                         swapCanvas($('holds_box'));
1043                         return;
1044                 }
1045         r._hold.selection_depth(res.depth);
1046         }       
1047
1048         holdCreateHold(r._recurse, r._hold);
1049 }
1050
1051
1052 function holdCreateHold( recurse, hold ) {
1053         var method = CREATE_HOLD;
1054         if(recurse) method = CREATE_HOLD_OVERRIDE;
1055         var req = new Request( method, holdArgs.requestor.session, hold );
1056         req.request.alertEvent = false;
1057         req.send(true);
1058         var res = req.result();
1059         holdProcessResult(hold, res, recurse);
1060         
1061         showCanvas();
1062
1063         runEvt('common', 'holdUpdated');
1064 }
1065
1066
1067 function holdProcessResult( hold, res, recurse ) {
1068
1069         if( res && res > -1 ) {
1070                 alert($('holds_success').innerHTML);
1071                 holdArgs = null;
1072         if(isXUL() && typeof xulG.opac_hold_placed == 'function')
1073             xulG.opac_hold_placed(res);
1074
1075         } else {
1076
1077                 if( recurse ) {
1078                         alert($('holds_failure').innerHTML);
1079                         return;
1080                 }
1081
1082                 if( grep(res, function(e) { return (e.textcode == 'HOLD_EXISTS'); }) ) {
1083                         if( fetchPermOrgs('HOLD_EXISTS.override')[0] != -1 ) {
1084                                 if( confirm($('hold_dup_exists_override').innerHTML) ) {
1085                                         return holdsPlaceHold(hold, true);
1086                                 }
1087                 return;
1088
1089                         } else {
1090                                 return alert($('hold_dup_exists').innerHTML);
1091                         }
1092                 }
1093
1094                 if( grep(res, function(e) { return (e.textcode == 'HOLD_ITEM_CHECKED_OUT'); }) ) {
1095                         if( fetchPermOrgs('HOLD_ITEM_CHECKED_OUT.override')[0] != -1 ) {
1096                                 if( confirm($('hold_checked_out_override').innerHTML) ) {
1097                                         return holdsPlaceHold(hold, true);
1098                                 }
1099                 return;
1100
1101                         } else {
1102                                 return alert($('hold_checked_out').innerHTML);
1103                         }
1104                 }
1105
1106
1107                 alert($('holds_failure').innerHTML);
1108         }
1109 }
1110
1111
1112 function holdsCancel(holdid, user) {
1113         if(!user) user = G.user;
1114         var req = new Request(CANCEL_HOLD, user.session, holdid, /* Patron via OPAC */ 6);
1115         req.send(true);
1116         return req.result();
1117         runEvt('common', 'holdUpdated');
1118 }
1119
1120 function holdsUpdate(hold, user) {
1121         if(!user) user = G.user;
1122         var req = new Request(UPDATE_HOLD, user.session, hold);
1123         req.send(true);
1124         var x = req.result(); // cause an exception if there is one 
1125         runEvt('common', 'holdUpdated');
1126 }
1127
1128 /* verify that the thaw date is valid and after today */
1129 function holdsVerifyThawDate(dateString, isGreater) {
1130     thawDate = dojo.date.stamp.fromISOString(dateString);
1131     if(thawDate) {
1132         if(isGreater) {
1133             if(dojo.date.compare(thawDate) > 0) {
1134                 return dojo.date.stamp.toISOString(thawDate);
1135             }
1136         } else {
1137             return dojo.date.stamp.toISOString(thawDate);
1138         }
1139     }
1140     return null;
1141 }
1142
1143 function holdsVerifyThawDateUI(element) {
1144     value = dojo.date.stamp.toISOString(dijit.byId(element).getValue());
1145
1146     if(!value) {
1147         removeCSSClass($(element), 'invalid_field');
1148         return;
1149     }
1150
1151     if(!holdsVerifyThawDate(value, true)) {
1152         addCSSClass($(element), 'invalid_field');
1153     } else {
1154         removeCSSClass($(element), 'invalid_field');
1155     }
1156 }
1157