]> git.evergreen-ils.org Git - Evergreen.git/blob - Open-ILS/web/js/ui/default/staff/cat/catalog/app.js
LP#1737812: Simplify holdings tranfser options
[Evergreen.git] / Open-ILS / web / js / ui / default / staff / cat / catalog / app.js
1 /**
2  * TPAC Frame App
3  *
4  * currently, this app doesn't use routes for each sub-ui, because 
5  * reloading the catalog each time is sloooow.  better so far to 
6  * swap out divs w/ ng-if / ng-show / ng-hide as needed.
7  *
8  */
9
10 angular.module('egCatalogApp', ['ui.bootstrap','ngRoute','ngLocationUpdate','egCoreMod','egGridMod', 'egMarcMod', 'egUserMod', 'egHoldingsMod', 'ngToast','egPatronSearchMod',
11 'egSerialsMod','egSerialsAppDep'])
12
13 .config(['ngToastProvider', function(ngToastProvider) {
14   ngToastProvider.configure({
15     verticalPosition: 'bottom',
16     animation: 'fade'
17   });
18 }])
19
20 .config(function($routeProvider, $locationProvider, $compileProvider) {
21     $locationProvider.html5Mode(true);
22     $compileProvider.aHrefSanitizationWhitelist(/^\s*(https?|mailto|blob):/); // grid export
23         
24     var resolver = {delay : ['egCore','egStartup','egUser', function(egCore, egStartup, egUser) {
25         egCore.env.classLoaders.aous = function() {
26             return egCore.org.settings([
27                 'cat.marc_control_number_identifier'
28             ]).then(function(settings) {
29                 // local settings are cached within egOrg.  Caching them
30                 // again in egEnv just simplifies the syntax for access.
31                 egCore.env.aous = settings;
32             });
33         }
34         egCore.env.loadClasses.push('aous');
35         return egStartup.go()
36     }]};
37
38     $routeProvider.when('/cat/catalog/index', {
39         templateUrl: './cat/catalog/t_catalog',
40         controller: 'CatalogCtrl',
41         resolve : resolver
42     });
43
44     // Jump directly to the results page.  Any URL parameter 
45     // supported by the embedded catalog is supported here.
46     $routeProvider.when('/cat/catalog/results', {
47         templateUrl: './cat/catalog/t_catalog',
48         controller: 'CatalogCtrl',
49         resolve : resolver
50     });
51
52     $routeProvider.when('/cat/catalog/retrieve_by_id', {
53         templateUrl: './cat/catalog/t_retrieve_by_id',
54         controller: 'CatalogRecordRetrieve',
55         resolve : resolver
56     });
57
58     $routeProvider.when('/cat/catalog/retrieve_by_tcn', {
59         templateUrl: './cat/catalog/t_retrieve_by_tcn',
60         controller: 'CatalogRecordRetrieve',
61         resolve : resolver
62     });
63
64     $routeProvider.when('/cat/catalog/new_bib', {
65         templateUrl: './cat/catalog/t_new_bib',
66         controller: 'NewBibCtrl',
67         resolve : resolver
68     });
69
70     // create some catalog page-specific mappings
71     $routeProvider.when('/cat/catalog/record/:record_id', {
72         templateUrl: './cat/catalog/t_catalog',
73         controller: 'CatalogCtrl',
74         resolve : resolver
75     });
76
77     // create some catalog page-specific mappings
78     $routeProvider.when('/cat/catalog/record/:record_id/:record_tab', {
79         templateUrl: './cat/catalog/t_catalog',
80         controller: 'CatalogCtrl',
81         resolve : resolver
82     });
83
84     $routeProvider.when('/cat/catalog/batchEdit', {
85         templateUrl: './cat/catalog/t_batchedit',
86         controller: 'BatchEditCtrl',
87         resolve : resolver
88     });
89
90     $routeProvider.when('/cat/catalog/batchEdit/:container_type/:container_id', {
91         templateUrl: './cat/catalog/t_batchedit',
92         controller: 'BatchEditCtrl',
93         resolve : resolver
94     });
95
96     $routeProvider.when('/cat/catalog/vandelay', {
97         templateUrl: './cat/catalog/t_vandelay',
98         controller: 'VandelayCtrl',
99         resolve : resolver
100     });
101
102     $routeProvider.when('/cat/catalog/verifyURLs', {
103         templateUrl: './cat/catalog/t_verifyurls',
104         controller: 'URLVerifyCtrl',
105         resolve : resolver
106     });
107
108     $routeProvider.when('/cat/catalog/manageAuthorities', {
109         templateUrl: './cat/catalog/t_manageauthorities',
110         controller: 'ManageAuthoritiesCtrl',
111         resolve : resolver
112     });
113
114     $routeProvider.when('/cat/catalog/authority/:authority_id/marc_edit', {
115         templateUrl: './cat/catalog/t_authority',
116         controller: 'AuthorityCtrl',
117         resolve : resolver
118     });
119
120     $routeProvider.otherwise({redirectTo : '/cat/catalog/index'});
121 })
122
123
124 /**
125  * */
126 .controller('CatalogRecordRetrieve',
127        ['$scope','$routeParams','$location','$q','egCore',
128 function($scope , $routeParams , $location , $q , egCore ) {
129
130     $scope.focusMe = true;
131
132     // jump to the patron checkout UI
133     function loadRecord(record_id) {
134         $location
135         .path('/cat/catalog/record/' + record_id);
136     }
137
138     $scope.submitId = function(args) {
139         $scope.recordNotFound = null;
140         if (!args.record_id) return;
141
142         // blur so next time it's set to true it will re-apply select()
143         $scope.selectMe = false;
144
145         return loadRecord(args.record_id);
146     }
147
148     $scope.submitTCN = function(args) {
149         $scope.recordNotFound = null;
150         $scope.moreRecordsFound = null;
151         if (!args.record_tcn) return;
152
153         // blur so next time it's set to true it will re-apply select()
154         $scope.selectMe = false;
155
156         // lookup TCN
157         egCore.net.request(
158             'open-ils.search',
159             'open-ils.search.biblio.tcn',
160             args.record_tcn)
161
162         .then(function(resp) { // get_barcodes
163
164             if (evt = egCore.evt.parse(resp)) {
165                 alert(evt); // FIXME
166                 return;
167             }
168
169             if (!resp.count) {
170                 $scope.recordNotFound = args.record_tcn;
171                 $scope.selectMe = true;
172                 return;
173             }
174
175             if (resp.count > 1) {
176                 $scope.moreRecordsFound = args.record_tcn;
177                 $scope.selectMe = true;
178                 return;
179             }
180
181             var record_id = resp.ids[0];
182             return loadRecord(record_id);
183         });
184     }
185
186 }])
187
188 .controller('NewBibCtrl',
189        ['$scope','$routeParams','$location','$window','$q','egCore',
190         'egGridDataProvider','egHoldGridActions','$timeout','holdingsSvc',
191 function($scope , $routeParams , $location , $window , $q , egCore) {
192
193     $scope.have_template = false;
194     $scope.marc_template = '';
195     $scope.stop_unload = false;
196     $scope.template_list = [];
197     $scope.template_name = '';
198     $scope.new_bib_id = 0;
199
200     egCore.net.request(
201         'open-ils.cat',
202         'open-ils.cat.marc_template.types.retrieve'
203     ).then(function(resp) {
204         angular.forEach(resp, function(name) {
205             $scope.template_list.push(name);
206         });
207         $scope.template_list.sort();
208     });
209     $scope.template_name = egCore.hatch.getSessionItem('eg.cat.last_bib_marc_template');
210     if (!$scope.template_name) {
211         egCore.hatch.getItem('cat.default_bib_marc_template').then(function(template) {
212             $scope.template_name = template;
213         });
214     }
215
216     $scope.loadTemplate = function() {
217         if ($scope.template_name) {
218             egCore.net.request(
219                 'open-ils.cat',
220                 'open-ils.cat.biblio.marc_template.retrieve',
221                 $scope.template_name
222             ).then(function(template) {
223                 $scope.marc_template = template;
224                 $scope.have_template = true;
225                 egCore.hatch.setSessionItem('eg.cat.last_bib_marc_template', $scope.template_name);
226             });
227         }
228     }
229
230     $scope.setDefaultTemplate = function() {
231         var hatch_key = "cat.default_bib_marc_template";
232         if ($scope.template_name) {
233             egCore.hatch.setItem(hatch_key, $scope.template_name);
234         } else {
235             egCore.hatch.removeItem(hatch_key);
236         }
237     }
238
239     $scope.$watch('new_bib_id', function(newVal, oldVal) {
240         if (newVal) {
241             $location.path('/cat/catalog/record/' + $scope.new_bib_id);
242         }
243     });
244     
245
246 }])
247 .controller('CatalogCtrl',
248        ['$scope','$routeParams','$location','$window','$q','egCore','egHolds','egCirc','egConfirmDialog','ngToast',
249         'egGridDataProvider','egHoldGridActions','egProgressDialog','$timeout','$uibModal','holdingsSvc','egUser','conjoinedSvc',
250         '$cookies','egSerialsCoreSvc',
251 function($scope , $routeParams , $location , $window , $q , egCore , egHolds , egCirc , egConfirmDialog , ngToast ,
252          egGridDataProvider , egHoldGridActions , egProgressDialog , $timeout , $uibModal , holdingsSvc , egUser , conjoinedSvc,
253          $cookies , egSerialsCoreSvc
254 ) {
255
256     var holdingsSvcInst = new holdingsSvc();
257
258     // set record ID on page load if available...
259     $scope.record_id = $routeParams.record_id;
260     $scope.summary_pane_record;
261
262     if ($scope.record_id) {
263         // TODO: Apply tab-specific title contexts
264         egCore.strings.setPageTitle(
265             egCore.strings.PAGE_TITLE_BIB_DETAIL,
266             egCore.strings.PAGE_TITLE_CATALOG_CONTEXT,
267             {record_id : $scope.record_id}
268         );
269     } else {
270         // Default to title = Catalog
271         egCore.strings.setPageTitle(
272             egCore.strings.PAGE_TITLE_CATALOG_CONTEXT);
273     }
274
275     if ($routeParams.record_id) $scope.from_route = true;
276     else $scope.from_route = false;
277
278     // set search and preferred library cookies
279     egCore.hatch.getItem('eg.search.search_lib').then(function(val) {
280         $cookies.put('eg_search_lib', val, { path : '/' });
281     });
282     egCore.hatch.getItem('eg.search.pref_lib').then(function(val) {
283         $cookies.put('eg_pref_lib', val, { path : '/' });
284     });
285
286     // will hold a ref to the opac iframe
287     $scope.opac_iframe = null;
288     $scope.parts_iframe = null;
289
290     $scope.search_result_index = 1;
291     $scope.search_result_hit_count = 1;
292
293     $scope.$watch(
294         'opac_iframe.dom.contentWindow.search_result_index',
295         function (n,o) {
296             if (!isNaN(parseInt(n)))
297                 $scope.search_result_index = n + 1;
298         }
299     );
300
301     $scope.$watch(
302         'opac_iframe.dom.contentWindow.search_result_hit_count',
303         function (n,o) {
304             if (!isNaN(parseInt(n)))
305                 $scope.search_result_hit_count = n;
306         }
307     );
308
309     $scope.in_opac_call = false;
310     $scope.opac_call = function (opac_frame_function, force_opac_tab) {
311         if ($scope.opac_iframe) {
312             if (force_opac_tab) $scope.record_tab = 'catalog';
313             $scope.in_opac_call = true;
314             $scope.opac_iframe.dom.contentWindow[opac_frame_function]();
315             if (opac_frame_function == 'rdetailBackToResults') {
316                 $location.update_path('/cat/catalog/index');
317             }
318         }
319     }
320
321     $scope.add_to_record_bucket = function() {
322         var recId = $scope.record_id;
323         return $uibModal.open({
324             templateUrl: './cat/catalog/t_add_to_bucket',
325             backdrop: 'static',
326             animation: true,
327             size: 'md',
328             controller:
329                    ['$scope','$uibModalInstance',
330             function($scope , $uibModalInstance) {
331
332                 $scope.bucket_id = 0;
333                 $scope.newBucketName = '';
334                 $scope.allBuckets = [];
335                 egCore.net.request(
336                     'open-ils.actor',
337                     'open-ils.actor.container.retrieve_by_class.authoritative',
338                     egCore.auth.token(), egCore.auth.user().id(),
339                     'biblio', 'staff_client'
340                 ).then(function(buckets) { $scope.allBuckets = buckets; });
341
342                 $scope.add_to_bucket = function() {
343                     var item = new egCore.idl.cbrebi();
344                     item.bucket($scope.bucket_id);
345                     item.target_biblio_record_entry(recId);
346                     egCore.net.request(
347                         'open-ils.actor',
348                         'open-ils.actor.container.item.create',
349                         egCore.auth.token(), 'biblio', item
350                     ).then(function(resp) {
351                         $uibModalInstance.close();
352                     });
353                 }
354
355                 $scope.add_to_new_bucket = function() {
356                     var bucket = new egCore.idl.cbreb();
357                     bucket.owner(egCore.auth.user().id());
358                     bucket.name($scope.newBucketName);
359                     bucket.description('');
360                     bucket.btype('staff_client');
361
362                     egCore.net.request(
363                         'open-ils.actor',
364                         'open-ils.actor.container.create',
365                         egCore.auth.token(), 'biblio', bucket
366                     ).then(function(bucket) {
367                         $scope.bucket_id = bucket;
368                         $scope.add_to_bucket();
369                     });
370                 }
371
372                 $scope.cancel = function() {
373                     $uibModalInstance.dismiss();
374                 }
375             }]
376         });
377     }
378
379     $scope.current_overlay_target     = egCore.hatch.getLocalItem('eg.cat.marked_overlay_record');
380     $scope.current_voltransfer_target = egCore.hatch.getLocalItem('eg.cat.marked_volume_transfer_record');
381     $scope.current_conjoined_target   = egCore.hatch.getLocalItem('eg.cat.marked_conjoined_record');
382
383     $scope.quickReceive = function () {
384         var list = [];
385         var next_per_stream = {};
386
387         var recId = $scope.record_id;
388         return $uibModal.open({
389             templateUrl: './share/t_subscription_select_dialog',
390             backdrop: 'static',
391             controller: ['$scope', '$uibModalInstance',
392                 function($scope, $uibModalInstance) {
393
394                     $scope.focus = true;
395                     $scope.rememberMe = 'eg.serials.quickreceive.last_org';
396                     $scope.record_id = recId;
397                     $scope.ssubId = null;
398
399                     $scope.ok = function() { $uibModalInstance.close($scope.ssubId) }
400                     $scope.cancel = function() { $uibModalInstance.dismiss(); }
401                 }
402             ]
403         }).result.then(function(ssubId) {
404             if (ssubId) {
405                 var promises = [];
406                 promises.push(egSerialsCoreSvc.fetchItemsForSub(ssubId,{status:'Expected'}).then(function(){
407                     angular.forEach(egSerialsCoreSvc.itemTree, function (item) {
408                         if (next_per_stream[item.stream().id()]) return;
409                         if (item.status() == 'Expected') {
410                             next_per_stream[item.stream().id()] = item;
411                             list.push(egCore.idl.Clone(item));
412                         }
413                     });
414                 }));
415
416                 return $q.all(promises).then(function() {
417
418                     if (!list.length) {
419                         ngToast.warning(egCore.strings.SERIALS_NO_ITEMS);
420                         return $q.reject();
421                     }
422
423                     return egSerialsCoreSvc.process_items(
424                         'receive',
425                         $scope.record_id,
426                         list,
427                         true, // barcode
428                         false,// bind
429                         false, // print by default
430                         function() { $scope.holdings_record_id_changed($scope.record_id) }
431                     );
432                 });
433             } else {
434                 ngToast.warning(egCore.strings.SERIALS_NO_SUBS);
435                 return $q.reject();
436             }
437         });
438     }
439
440     $scope.markConjoined = function () {
441         $scope.current_conjoined_target = $scope.record_id;
442         egCore.hatch.setLocalItem('eg.cat.marked_conjoined_record',$scope.record_id);
443         ngToast.create(egCore.strings.MARK_CONJ_TARGET);
444     };
445
446     $scope.markVolTransfer = function () {
447         ngToast.create(egCore.strings.MARK_VOL_TARGET);
448         $scope.current_voltransfer_target = $scope.record_id;
449         egCore.hatch.setLocalItem('eg.cat.marked_volume_transfer_record',$scope.record_id);
450         egCore.hatch.removeLocalItem('eg.cat.volume_transfer_target');
451     };
452
453     $scope.markOverlay = function () {
454         $scope.current_overlay_target = $scope.record_id;
455         egCore.hatch.setLocalItem('eg.cat.marked_overlay_record',$scope.record_id);
456         ngToast.create(egCore.strings.MARK_OVERLAY_TARGET);
457     };
458
459     $scope.clearRecordMarks = function () {
460         $scope.current_overlay_target     = null;
461         $scope.current_voltransfer_target = null;
462         $scope.current_conjoined_target   = null;
463         $scope.current_hold_transfer_dest = null;
464         egCore.hatch.removeLocalItem('eg.cat.marked_volume_transfer_record');
465         egCore.hatch.removeLocalItem('eg.cat.marked_conjoined_record');
466         egCore.hatch.removeLocalItem('eg.cat.marked_overlay_record');
467         egCore.hatch.removeLocalItem('eg.circ.hold.title_transfer_target');
468     }
469
470     $scope.stop_unload = false;
471     $scope.$watch('stop_unload',
472         function(newVal, oldVal) {
473             if (newVal && newVal != oldVal && $scope.opac_iframe) {
474                 $($scope.opac_iframe.dom.contentWindow).on('beforeunload', function(){
475                     return 'There is unsaved data in this record.'
476                 });
477             } else {
478                 if ($scope.opac_iframe)
479                     $($scope.opac_iframe.dom.contentWindow).off('beforeunload');
480             }
481         }
482     );
483
484     // Set the "last bib" cookie, if we have that
485     if ($scope.record_id)
486         egCore.hatch.setLocalItem("eg.cat.last_record_retrieved", $scope.record_id);
487
488     $scope.refresh_record_callback = function (record_id) {
489         egCore.pcrud.retrieve('bre', record_id, {
490             flesh : 1,
491             flesh_fields : {
492                 bre : ['simple_record','creator','editor']
493             }
494         }).then(function(rec) {
495             rec.owner(egCore.org.get(rec.owner()));
496             $scope.summary_pane_record = rec;
497         });
498
499         return record_id;
500     }
501
502     patron_search_dialog = function() {
503         return $uibModal.open({
504             templateUrl: './share/t_patron_selector',
505             backdrop: 'static',
506             size: 'lg',
507             animation: true,
508             controller:
509                    ['$scope','$uibModalInstance','$controller',
510             function($scope , $uibModalInstance , $controller) {
511                 angular.extend(this, $controller('BasePatronSearchCtrl', {$scope : $scope}));
512                 $scope.clearForm();
513                 $scope.need_one_selected = function() {
514                     var items = $scope.gridControls.selectedItems();
515                     return (items.length == 1) ? false : true
516                 }
517                 $scope.ok = function() {
518                     var items = $scope.gridControls.selectedItems();
519                     if (items.length == 1) {
520                         $uibModalInstance.close(items[0].card().barcode());
521                     } else {
522                         $uibModalInstance.close()
523                     }
524                 }
525                 $scope.cancel = function($event) {
526                     $uibModalInstance.dismiss();
527                     $event.preventDefault();
528                 }
529             }]
530         });
531     }
532
533     // also set it when the iframe changes to a new record
534     $scope.handle_page = function(url) {
535
536         if (!url || url == 'about:blank') {
537             // nothing loaded.  If we already have a record ID, leave it.
538             return;
539         }
540
541         var match = url.match(/\/+opac\/+record\/+(\d+)/);
542         if (match) {
543             $scope.record_id = match[1];
544             egCore.hatch.setLocalItem("eg.cat.last_record_retrieved", $scope.record_id);
545             $scope.holdings_record_id_changed($scope.record_id);
546             conjoinedSvc.fetch($scope.record_id).then(function(){
547                 $scope.conjoinedGridDataProvider.refresh();
548             });
549             egHolds.fetch_holds(hold_ids).then($scope.hold_grid_data_provider.refresh);
550             init_parts_url();
551             $location.update_path('/cat/catalog/record/' + $scope.record_id);
552             // update_path() bypasses the controller for path 
553             // /cat/catalog/record/:record_id. Manually set title here too.
554             egCore.strings.setPageTitle(
555                 egCore.strings.PAGE_TITLE_BIB_DETAIL,
556                 egCore.strings.PAGE_TITLE_CATALOG_CONTEXT,
557                 {record_id : $scope.record_id}
558             );
559         } else {
560             delete $scope.record_id;
561             $scope.from_route = false;
562         }
563
564         // child scope is executing this function, so our digest doesn't fire ... thus,
565         $scope.$apply();
566
567         if (!$scope.in_opac_call) {
568             if ($scope.record_id && !$scope.record_tab) {
569                 $scope.default_tab = egCore.hatch.getLocalItem( 'eg.cat.default_record_tab' );
570                 tab = $routeParams.record_tab || $scope.default_tab || 'catalog';
571             } else {
572                 tab = $routeParams.record_tab || 'catalog';
573             }
574             $scope.set_record_tab(tab);
575         } else {
576             $scope.in_opac_call = false;
577         }
578
579         if ($scope.opac_iframe && $location.path().match(/cat\/catalog/)) {
580             var doc = $scope.opac_iframe.dom.contentWindow.document;
581             $(doc).find('#hold_usr_search').show();
582             $(doc).find('#hold_usr_search').on('click', function() {
583                 patron_search_dialog().result.then(function(barc) {
584                     $(doc).find('#hold_usr_input').val(barc);
585                     $(doc).find('#hold_usr_input').change();
586                 });
587             })
588         }
589
590     }
591
592     // xulG catalog handlers
593     $scope.handlers = { }
594
595     // ------------------------------------------------------------------
596     // Conjoined items
597
598     $scope.conjoinedGridControls = {};
599     $scope.conjoinedGridDataProvider = egGridDataProvider.instance({
600         get : function(offset, count) {
601             return this.arrayNotifier(conjoinedSvc.items, offset, count);
602         }
603     });
604
605     $scope.changeConjoinedType = function () {
606         var peers = egCore.idl.Clone($scope.conjoinedGridControls.selectedItems());
607         angular.forEach(peers, function (p) {
608             p.target_copy(p.target_copy().id());
609             p.peer_type(p.peer_type().id());
610         });
611
612         var conjoinedGridDataProviderRef = $scope.conjoinedGridDataProvider;
613
614         return $uibModal.open({
615             templateUrl: './cat/catalog/t_conjoined_selector',
616             backdrop: 'static',
617             animation: true,
618             controller:
619                    ['$scope','$uibModalInstance',
620             function($scope , $uibModalInstance) {
621                 $scope.update = true;
622
623                 $scope.peer_type = null;
624                 $scope.peer_type_list = [];
625                 conjoinedSvc.get_peer_types().then(function(list){
626                     $scope.peer_type_list = list;
627                 });
628     
629                 $scope.ok = function(type) {
630                     var promises = [];
631     
632                     angular.forEach(peers, function (p) {
633                         p.ischanged(1);
634                         p.peer_type(type);
635                         promises.push(egCore.pcrud.update(p));
636                     });
637     
638                     return $q.all(promises)
639                         .then(function(){$uibModalInstance.close()})
640                         .then(function(){return conjoinedSvc.fetch()})
641                         .then(function(){conjoinedGridDataProviderRef.refresh()});
642                 }
643     
644                 $scope.cancel = function($event) {
645                     $uibModalInstance.dismiss();
646                     $event.preventDefault();
647                 }
648             }]
649         });
650         
651     }
652
653     $scope.refreshConjoined = function () {
654         conjoinedSvc.fetch($scope.record_id)
655         .then(function(){$scope.conjoinedGridDataProvider.refresh();});
656     }
657
658     $scope.deleteSelectedConjoined = function () {
659         var peers = $scope.conjoinedGridControls.selectedItems();
660
661         if (peers.length > 0) {
662             egConfirmDialog.open(
663                 egCore.strings.CONFIRM_DELETE_PEERS,
664                 egCore.strings.CONFIRM_DELETE_PEERS_MESSAGE,
665                 {peers : peers.length}
666             ).result.then(function() {
667                 angular.forEach(peers, function (p) {
668                     p.isdeleted(1);
669                 });
670
671                 egCore.pcrud.remove(peers).then(function() {
672                     return conjoinedSvc.fetch();
673                 }).then(function() {
674                     $scope.conjoinedGridDataProvider.refresh();
675                 });
676             });
677         }
678     }
679     if ($scope.record_id)
680         conjoinedSvc.fetch($scope.record_id);
681
682     // ------------------------------------------------------------------
683     // Holdings
684
685     $scope.holdingsGridControls = {
686         activateItem : function (item) {
687             $scope.selectedHoldingsVolCopyEdit();
688         }
689     };
690     $scope.holdingsGridDataProvider = egGridDataProvider.instance({
691         get : function(offset, count) {
692             return this.arrayNotifier(holdingsSvcInst.copies, offset, count);
693         }
694     });
695
696     $scope.add_copies_to_bucket = function() {
697         var copy_list = gatherSelectedHoldingsIds();
698         if (copy_list.length == 0) return;
699
700         return $uibModal.open({
701             templateUrl: './cat/catalog/t_add_to_bucket',
702             backdrop: 'static',
703             animation: true,
704             size: 'md',
705             controller:
706                    ['$scope','$uibModalInstance',
707             function($scope , $uibModalInstance) {
708
709                 $scope.bucket_id = 0;
710                 $scope.newBucketName = '';
711                 $scope.allBuckets = [];
712
713                 egCore.net.request(
714                     'open-ils.actor',
715                     'open-ils.actor.container.retrieve_by_class.authoritative',
716                     egCore.auth.token(), egCore.auth.user().id(),
717                     'copy', 'staff_client'
718                 ).then(function(buckets) { $scope.allBuckets = buckets; });
719
720                 $scope.add_to_bucket = function() {
721                     var promises = [];
722                     angular.forEach(copy_list, function (cp) {
723                         var item = new egCore.idl.ccbi()
724                         item.bucket($scope.bucket_id);
725                         item.target_copy(cp);
726                         promises.push(
727                             egCore.net.request(
728                                 'open-ils.actor',
729                                 'open-ils.actor.container.item.create',
730                                 egCore.auth.token(), 'copy', item
731                             )
732                         );
733
734                         return $q.all(promises).then(function() {
735                             $uibModalInstance.close();
736                         });
737                     });
738                 }
739
740                 $scope.add_to_new_bucket = function() {
741                     var bucket = new egCore.idl.ccb();
742                     bucket.owner(egCore.auth.user().id());
743                     bucket.name($scope.newBucketName);
744                     bucket.description('');
745                     bucket.btype('staff_client');
746
747                     return egCore.net.request(
748                         'open-ils.actor',
749                         'open-ils.actor.container.create',
750                         egCore.auth.token(), 'copy', bucket
751                     ).then(function(bucket) {
752                         $scope.bucket_id = bucket;
753                         $scope.add_to_bucket();
754                     });
755                 }
756
757                 $scope.cancel = function() {
758                     $uibModalInstance.dismiss();
759                 }
760             }]
761         });
762     }
763
764     // TODO: refactor common code between cat/catalog/app.js and cat/item/app.js 
765
766     $scope.need_one_selected = function() {
767         var items = $scope.holdingsGridControls.selectedItems();
768         if (items.length == 1) return false;
769         return true;
770     };
771
772     $scope.make_copies_bookable = function() {
773
774         var copies_by_record = {};
775         var record_list = [];
776         angular.forEach(
777             $scope.holdingsGridControls.selectedItems(),
778             function (item) {
779                 var record_id = item['call_number.record.id'];
780                 if (typeof copies_by_record[ record_id ] == 'undefined') {
781                     copies_by_record[ record_id ] = [];
782                     record_list.push( record_id );
783                 }
784                 copies_by_record[ record_id ].push(item.id);
785             }
786         );
787
788         var promises = [];
789         var combined_results = [];
790         angular.forEach(record_list, function(record_id) {
791             promises.push(
792                 egCore.net.request(
793                     'open-ils.booking',
794                     'open-ils.booking.resources.create_from_copies',
795                     egCore.auth.token(),
796                     copies_by_record[record_id]
797                 ).then(function(results) {
798                     if (results && results['brsrc']) {
799                         combined_results = combined_results.concat(results['brsrc']);
800                     }
801                 })
802             );
803         });
804
805         $q.all(promises).then(function() {
806             if (combined_results.length > 0) {
807                 $uibModal.open({
808                     template: '<eg-embed-frame url="booking_admin_url" handlers="funcs"></eg-embed-frame>',
809                     backdrop: 'static',
810                     animation: true,
811                     size: 'md',
812                     controller:
813                            ['$scope','$location','egCore','$uibModalInstance',
814                     function($scope , $location , egCore , $uibModalInstance) {
815
816                         $scope.funcs = {
817                             ses : egCore.auth.token(),
818                             resultant_brsrc : combined_results.map(function(o) { return o[0]; })
819                         }
820
821                         var booking_path = '/eg/conify/global/booking/resource';
822
823                         $scope.booking_admin_url =
824                             $location.absUrl().replace(/\/eg\/staff.*/, booking_path);
825                     }]
826                 });
827             }
828         });
829     }
830
831     $scope.book_copies_now = function() {
832         var copies_by_record = {};
833         var record_list = [];
834         angular.forEach(
835             $scope.holdingsGridControls.selectedItems(),
836             function (item) {
837                 var record_id = item['call_number.record.id'];
838                 if (typeof copies_by_record[ record_id ] == 'undefined') {
839                     copies_by_record[ record_id ] = [];
840                     record_list.push( record_id );
841                 }
842                 copies_by_record[ record_id ].push(item.id);
843             }
844         );
845
846         var promises = [];
847         var combined_brt = [];
848         var combined_brsrc = [];
849         angular.forEach(record_list, function(record_id) {
850             promises.push(
851                 egCore.net.request(
852                     'open-ils.booking',
853                     'open-ils.booking.resources.create_from_copies',
854                     egCore.auth.token(),
855                     copies_by_record[record_id]
856                 ).then(function(results) {
857                     if (results && results['brt']) {
858                         combined_brt = combined_brt.concat(results['brt']);
859                     }
860                     if (results && results['brsrc']) {
861                         combined_brsrc = combined_brsrc.concat(results['brsrc']);
862                     }
863                 })
864             );
865         });
866
867         $q.all(promises).then(function() {
868             if (combined_brt.length > 0 || combined_brsrc.length > 0) {
869                 $uibModal.open({
870                     template: '<eg-embed-frame url="booking_admin_url" handlers="funcs"></eg-embed-frame>',
871                     backdrop: 'static',
872                     animation: true,
873                     size: 'md',
874                     controller:
875                            ['$scope','$location','egCore','$uibModalInstance',
876                     function($scope , $location , egCore , $uibModalInstance) {
877
878                         $scope.funcs = {
879                             ses : egCore.auth.token(),
880                             bresv_interface_opts : {
881                                 booking_results : {
882                                      brt : combined_brt
883                                     ,brsrc : combined_brsrc
884                                 }
885                             }
886                         }
887
888                         var booking_path = '/eg/booking/reservation';
889
890                         $scope.booking_admin_url =
891                             $location.absUrl().replace(/\/eg\/staff.*/, booking_path);
892
893                     }]
894                 });
895             }
896         });
897     }
898
899
900     $scope.requestItems = function() {
901         var copy_list = gatherSelectedHoldingsIds();
902         if (copy_list.length == 0) return;
903
904         return $uibModal.open({
905             templateUrl: './cat/catalog/t_request_items',
906             animation: true,
907             controller:
908                    ['$scope','$uibModalInstance',
909             function($scope , $uibModalInstance) {
910                 $scope.user = null;
911                 $scope.first_user_fetch = true;
912
913                 $scope.hold_data = {
914                     hold_type : 'C',
915                     copy_list : copy_list,
916                     pickup_lib: egCore.org.get(egCore.auth.user().ws_ou()),
917                     user      : egCore.auth.user().id()
918                 };
919
920                 egUser.get( $scope.hold_data.user ).then(function(u) {
921                     $scope.user = u;
922                     $scope.barcode = u.card().barcode();
923                     $scope.user_name = egUser.format_name(u);
924                     $scope.hold_data.user = u.id();
925                 });
926
927                 $scope.user_name = '';
928                 $scope.barcode = '';
929                 $scope.$watch('barcode', function (n) {
930                     if (!$scope.first_user_fetch) {
931                         egUser.getByBarcode(n).then(function(u) {
932                             $scope.user = u;
933                             $scope.user_name = egUser.format_name(u);
934                             $scope.hold_data.user = u.id();
935                         }, function() {
936                             $scope.user = null;
937                             $scope.user_name = '';
938                             delete $scope.hold_data.user;
939                         });
940                     }
941                     $scope.first_user_fetch = false;
942                 });
943
944                 $scope.ok = function(h) {
945                     var args = {
946                         patronid  : h.user,
947                         hold_type : h.hold_type,
948                         pickup_lib: h.pickup_lib.id(),
949                         depth     : 0
950                     };
951
952                     egCore.net.request(
953                         'open-ils.circ',
954                         'open-ils.circ.holds.test_and_create.batch.override',
955                         egCore.auth.token(), args, h.copy_list
956                     );
957
958                     $uibModalInstance.close();
959                 }
960
961                 $scope.cancel = function($event) {
962                     $uibModalInstance.dismiss();
963                     $event.preventDefault();
964                 }
965             }]
966         });
967     }
968
969     $scope.view_place_orders = function() {
970         if (!$scope.record_id) return;
971         var url = egCore.env.basePath + 'acq/legacy/lineitem/related/' + $scope.record_id + '?target=bib';
972         $timeout(function() { $window.open(url, '_blank') });
973     }
974
975     $scope.replaceBarcodes = function() {
976         var copy_list = gatherSelectedRawCopies();
977         if (copy_list.length == 0) return;
978
979         var holdingsGridDataProviderRef = $scope.holdingsGridDataProvider;
980
981         angular.forEach(copy_list, function (cp) {
982             $uibModal.open({
983                 templateUrl: './cat/share/t_replace_barcode',
984                 backdrop: 'static',
985                 animation: true,
986                 controller:
987                            ['$scope','$uibModalInstance',
988                     function($scope , $uibModalInstance) {
989                         $scope.isModal = true;
990                         $scope.focusBarcode = false;
991                         $scope.focusBarcode2 = true;
992                         $scope.barcode1 = cp.barcode();
993
994                         $scope.updateBarcode = function() {
995                             $scope.copyNotFound = false;
996                             $scope.updateOK = false;
997                 
998                             egCore.pcrud.search('acp',
999                                 {deleted : 'f', barcode : $scope.barcode1})
1000                             .then(function(copy) {
1001                 
1002                                 if (!copy) {
1003                                     $scope.focusBarcode = true;
1004                                     $scope.copyNotFound = true;
1005                                     return;
1006                                 }
1007                 
1008                                 $scope.copyId = copy.id();
1009                                 copy.barcode($scope.barcode2);
1010                 
1011                                 egCore.pcrud.update(copy).then(function(stat) {
1012                                     $scope.updateOK = stat;
1013                                     $scope.focusBarcode = true;
1014                                     holdingsSvc.fetchAgain().then(function (){
1015                                         holdingsGridDataProviderRef.refresh();
1016                                     });
1017                                 });
1018
1019                             });
1020                             $uibModalInstance.close();
1021                         }
1022
1023                         $scope.cancel = function($event) {
1024                             $uibModalInstance.dismiss();
1025                             $event.preventDefault();
1026                         }
1027                     }
1028                 ]
1029             });
1030         });
1031     }
1032
1033     // refresh the list of holdings when the record_id is changed.
1034     $scope.holdings_record_id_changed = function(id) {
1035         if ($scope.record_id != id) $scope.record_id = id;
1036         console.log('record id changed to ' + id + ', loading new holdings');
1037         holdingsSvcInst.fetch({
1038             rid : $scope.record_id,
1039             org : $scope.holdings_ou,
1040             copy: $scope.holdings_show_vols ? $scope.holdings_show_copies : false,
1041             vol : $scope.holdings_show_vols,
1042             empty: $scope.holdings_show_empty,
1043             empty_org: $scope.holdings_show_empty_org
1044         }).then(function() {
1045             $scope.holdingsGridDataProvider.refresh();
1046         });
1047     }
1048
1049     // refresh the list of holdings when the filter lib is changed.
1050     $scope.holdings_ou = egCore.org.get(egCore.auth.user().ws_ou());
1051     $scope.holdings_ou_changed = function(org) {
1052         $scope.holdings_ou = org;
1053         holdingsSvcInst.fetch({
1054             rid : $scope.record_id,
1055             org : $scope.holdings_ou,
1056             copy: $scope.holdings_show_vols ? $scope.holdings_show_copies : false,
1057             vol : $scope.holdings_show_vols,
1058             empty: $scope.holdings_show_empty,
1059             empty_org: $scope.holdings_show_empty_org
1060         }).then(function() {
1061             $scope.holdingsGridDataProvider.refresh();
1062         });
1063     }
1064
1065     $scope.holdings_cb_changed = function(cb,newVal,norefresh) {
1066         $scope[cb] = newVal;
1067         var x = $scope.holdings_show_vols ? $scope.holdings_show_copies : false;
1068         $('#holdings_show_copies').prop('checked', x);
1069         egCore.hatch.setItem('cat.' + cb, newVal);
1070         if (!norefresh) holdingsSvcInst.fetch({
1071             rid : $scope.record_id,
1072             org : $scope.holdings_ou,
1073             copy: $scope.holdings_show_vols ? $scope.holdings_show_copies : false,
1074             vol : $scope.holdings_show_vols,
1075             empty: $scope.holdings_show_empty,
1076             empty_org: $scope.holdings_show_empty_org
1077         }).then(function() {
1078             $scope.holdingsGridDataProvider.refresh();
1079         });
1080     }
1081
1082     egCore.hatch.getItem('cat.holdings_show_vols').then(function(x){
1083         if (typeof x ==  'undefined') x = true;
1084         $scope.holdings_cb_changed('holdings_show_vols',x,true);
1085         $('#holdings_show_vols').prop('checked', x);
1086     }).then(function(){
1087         egCore.hatch.getItem('cat.holdings_show_copies').then(function(x){
1088             if (typeof x ==  'undefined') x = true;
1089             $scope.holdings_cb_changed('holdings_show_copies',x,true);
1090             x = $scope.holdings_show_vols ? x : false;
1091             $('#holdings_show_copies').prop('checked', x);
1092         }).then(function(){
1093             egCore.hatch.getItem('cat.holdings_show_empty').then(function(x){
1094                 if (typeof x ==  'undefined') x = true;
1095                 $scope.holdings_cb_changed('holdings_show_empty',x);
1096                 $('#holdings_show_empty').prop('checked', x);
1097             }).then(function(){
1098                 egCore.hatch.getItem('cat.holdings_show_empty_org').then(function(x){
1099                     if (typeof x ==  'undefined') x = true;
1100                     $scope.holdings_cb_changed('holdings_show_empty_org',x);
1101                     $('#holdings_show_empty_org').prop('checked', x);
1102                 })
1103             })
1104         })
1105     });
1106
1107     $scope.vols_not_shown = function () {
1108         return !$scope.holdings_show_vols;
1109     }
1110
1111     $scope.copies_not_shown = function () {
1112         return !$scope.holdings_show_copies;
1113     }
1114
1115     $scope.empty_org_not_shown = function () {
1116         return !$scope.holdings_show_empty_org;
1117     }
1118
1119     $scope.holdings_checkbox_handler = function (item) {
1120         $scope.holdings_cb_changed(item.checkbox,item.checked);
1121     }
1122
1123     function gatherSelectedOwners () {
1124         var owner_list = [];
1125         angular.forEach(
1126             $scope.holdingsGridControls.selectedItems(),
1127             function (item) { owner_list.push(item.owner_id) }
1128         );
1129         return owner_list;
1130     }
1131
1132     function gatherSelectedHoldingsIds () {
1133         var cp_id_list = [];
1134         angular.forEach(
1135             $scope.holdingsGridControls.selectedItems(),
1136             function (item) { cp_id_list = cp_id_list.concat(item.id_list) }
1137         );
1138         return cp_id_list;
1139     }
1140
1141     function gatherSelectedRawCopies () {
1142         var cp_list = [];
1143         angular.forEach(
1144             $scope.holdingsGridControls.selectedItems(),
1145             function (item) { if (item.raw) cp_list = cp_list.concat(item.raw) }
1146         );
1147         return cp_list;
1148     }
1149
1150     function gatherSelectedEmptyVolumeIds () {
1151         var cn_id_list = [];
1152         angular.forEach(
1153             $scope.holdingsGridControls.selectedItems(),
1154             function (item) {
1155                 if (item.copy_count == 0)
1156                     cn_id_list.push(item.call_number.id)
1157             }
1158         );
1159         return cn_id_list;
1160     }
1161
1162     function gatherSelectedVolumeIds () {
1163         var cn_id_list = [];
1164         angular.forEach(
1165             $scope.holdingsGridControls.selectedItems(),
1166             function (item) {
1167                 if (cn_id_list.indexOf(item.call_number.id) == -1)
1168                     cn_id_list.push(item.call_number.id)
1169             }
1170         );
1171         return cn_id_list;
1172     }
1173
1174     $scope.selectedHoldingsDelete = function (vols, copies) {
1175
1176         var cnHash = {};
1177         var perCnCopies = {};
1178
1179         var cn_count = 0;
1180         var cp_count = 0;
1181
1182         angular.forEach(
1183             $scope.holdingsGridControls.selectedItems(),
1184             function (item) {
1185                 if (vols && item.raw_call_number) {
1186                     cnHash[item.call_number.id] = egCore.idl.Clone(item.raw_call_number);
1187                     cnHash[item.call_number.id].isdeleted(1);
1188                     cn_count++;
1189                 } else if (copies) {
1190                     angular.forEach(egCore.idl.Clone(item.raw), function (cp) {
1191                         cp.isdeleted(1);
1192                         cp_count++;
1193                         var cn_id = cp.call_number().id();
1194                         if (!cnHash[cn_id]) {
1195                             cnHash[cn_id] = cp.call_number();
1196                             perCnCopies[cn_id] = [cp];
1197                         } else {
1198                             perCnCopies[cn_id].push(cp);
1199                         }
1200                         cp.call_number(cn_id); // prevent loops in JSON-ification
1201                     });
1202
1203                 }
1204             }
1205         );
1206
1207         angular.forEach(perCnCopies, function (v, k) {
1208             if (vols) {
1209                 cnHash[k].isdeleted(1);
1210                 cn_count++;
1211             }
1212             cnHash[k].copies(v);
1213         });
1214
1215         cnList = [];
1216         angular.forEach(cnHash, function (v, k) {
1217             cnList.push(v);
1218         });
1219
1220         if (cnList.length == 0) return;
1221
1222         var flags = {};
1223         if (vols && copies) flags.force_delete_copies = 1;
1224
1225         egConfirmDialog.open(
1226             egCore.strings.CONFIRM_DELETE_COPIES_VOLUMES,
1227             egCore.strings.CONFIRM_DELETE_COPIES_VOLUMES_MESSAGE,
1228             {copies : cp_count, volumes : cn_count}
1229         ).result.then(function() {
1230             egCore.net.request(
1231                 'open-ils.cat',
1232                 'open-ils.cat.asset.volume.fleshed.batch.update.override',
1233                 egCore.auth.token(), cnList, 1, flags
1234             ).then(function(update_count) {
1235                 holdingsSvcInst.fetchAgain().then(function() {
1236                     $scope.holdingsGridDataProvider.refresh();
1237                 });
1238             });
1239         });
1240     }
1241     $scope.selectedHoldingsCopyDelete = function () { $scope.selectedHoldingsDelete(false,true) }
1242     $scope.selectedHoldingsVolCopyDelete = function () { $scope.selectedHoldingsDelete(true,true) }
1243     $scope.selectedHoldingsEmptyVolCopyDelete = function () { $scope.selectedHoldingsDelete(true,false) }
1244
1245     spawnHoldingsAdd = function (vols,copies){
1246         var raw = [];
1247         if (copies) { // just a copy on existing volumes
1248             angular.forEach(gatherSelectedVolumeIds(), function (v) {
1249                 raw.push( {callnumber : v} );
1250             });
1251         } else if (vols) {
1252             if (typeof $scope.holdingsGridControls.selectedItems == "function" &&
1253                 $scope.holdingsGridControls.selectedItems().length > 0) {
1254                 angular.forEach($scope.holdingsGridControls.selectedItems(),
1255                     function (item) {
1256                         raw.push({
1257                             owner : item.owner_id,
1258                             label : item.call_number.label
1259                         });
1260                     });
1261             } else {
1262                 raw.push({
1263                     owner : egCore.auth.user().ws_ou()
1264                 });
1265             }
1266         }
1267
1268         if (raw.length == 0) raw.push({});
1269
1270         egCore.net.request(
1271             'open-ils.actor',
1272             'open-ils.actor.anon_cache.set_value',
1273             null, 'edit-these-copies', {
1274                 record_id: $scope.record_id,
1275                 raw: raw,
1276                 hide_vols : false,
1277                 hide_copies : false
1278             }
1279         ).then(function(key) {
1280             if (key) {
1281                 var url = egCore.env.basePath + 'cat/volcopy/' + key;
1282                 $timeout(function() { $window.open(url, '_blank') });
1283             } else {
1284                 alert('Could not create anonymous cache key!');
1285             }
1286         });
1287     }
1288     $scope.selectedHoldingsVolCopyAdd = function () { spawnHoldingsAdd(true,false) }
1289     $scope.selectedHoldingsCopyAdd = function () { spawnHoldingsAdd(false,true) }
1290
1291     spawnHoldingsEdit = function (hide_vols,hide_copies,add_vol){
1292
1293         var raw;
1294         if (add_vol) {
1295             raw = [{callnumber:null}];
1296         } else {
1297             raw = gatherSelectedEmptyVolumeIds().map(
1298                 function(v){ return { callnumber : v } }
1299             );
1300         }
1301
1302         egCore.net.request(
1303             'open-ils.actor',
1304             'open-ils.actor.anon_cache.set_value',
1305             null, 'edit-these-copies', {
1306                 record_id: $scope.record_id,
1307                 copies: gatherSelectedHoldingsIds(),
1308                 raw: raw,
1309                 hide_vols : hide_vols,
1310                 hide_copies : hide_copies,
1311                 only_add_vol : ((add_vol) ? true : false),
1312                 owners : gatherSelectedOwners()
1313             }
1314         ).then(function(key) {
1315             if (key) {
1316                 var url = egCore.env.basePath + 'cat/volcopy/' + key;
1317                 $timeout(function() { $window.open(url, '_blank') });
1318             } else {
1319                 alert('Could not create anonymous cache key!');
1320             }
1321         });
1322     }
1323     $scope.selectedHoldingsVolCopyEdit = function () { spawnHoldingsEdit(false,false) }
1324     $scope.selectedHoldingsVolEdit = function () { spawnHoldingsEdit(false,true) }
1325     $scope.selectedHoldingsCopyEdit = function () { spawnHoldingsEdit(true,false) }
1326
1327     $scope.selectedHoldingsVolAdd = function () { spawnHoldingsEdit(false,true,true) }
1328
1329     $scope.selectedHoldingsItemStatus = function (){
1330         var url = egCore.env.basePath + 'cat/item/search/' + gatherSelectedHoldingsIds().join(',')
1331         $timeout(function() { $window.open(url, '_blank') });
1332     }
1333
1334     $scope.markVolAsItemTarget = function() {
1335         if ($scope.holdingsGridControls.selectedItems()[0].call_number.id) { // cn.id missing when vols are collapsed
1336             egCore.hatch.setLocalItem(
1337                 'eg.cat.item_transfer_target',
1338                 $scope.holdingsGridControls.selectedItems()[0].call_number.id
1339             );
1340             ngToast.create(egCore.strings.MARK_ITEM_TARGET);
1341         }
1342     }
1343
1344     $scope.markLibAsVolTarget = function() {
1345         var recId = $scope.record_id;
1346         return $uibModal.open({
1347             templateUrl: './cat/catalog/t_choose_vol_target_lib',
1348             backdrop: 'static',
1349             animation: true,
1350             controller:
1351                    ['$scope','$uibModalInstance',
1352             function($scope , $uibModalInstance) {
1353
1354                 var orgId = egCore.hatch.getLocalItem('eg.cat.volume_transfer_target') || 1;
1355                 $scope.org = egCore.org.get(orgId);
1356                 $scope.cant_have_vols = function (id) { return !egCore.org.CanHaveVolumes(id); };
1357                 $scope.ok = function(org) {
1358                     egCore.hatch.setLocalItem(
1359                         'eg.cat.volume_transfer_target',
1360                         org.id()
1361                     );
1362                     egCore.hatch.setLocalItem(
1363                         'eg.cat.marked_volume_transfer_record',
1364                         recId
1365                     );
1366                     $uibModalInstance.close();
1367                 }
1368                 $scope.cancel = function($event) {
1369                     $uibModalInstance.dismiss();
1370                     $event.preventDefault();
1371                 }
1372             }]
1373         });
1374     }
1375     $scope.markLibFromSelectedAsVolTarget = function() {
1376         egCore.hatch.setLocalItem(
1377             'eg.cat.volume_transfer_target',
1378             $scope.holdingsGridControls.selectedItems()[0].owner_id
1379         );
1380         egCore.hatch.setLocalItem(
1381             'eg.cat.marked_volume_transfer_record',
1382             $scope.record_id
1383         );
1384         ngToast.create(egCore.strings.MARK_VOL_TARGET);
1385     }
1386
1387     $scope.selectedHoldingsItemStatusDetail = function (){
1388         angular.forEach(
1389             gatherSelectedHoldingsIds(),
1390             function (cid) {
1391                 var url = egCore.env.basePath +
1392                           'cat/item/' + cid;
1393                 $timeout(function() { $window.open(url, '_blank') });
1394             }
1395         );
1396     }
1397
1398     $scope.transferVolumesToRecord = function (){
1399         var target_record = egCore.hatch.getLocalItem('eg.cat.marked_volume_transfer_record');
1400         if (!target_record) return;
1401         if ($scope.record_id == target_record) return;
1402         var items = $scope.holdingsGridControls.selectedItems();
1403         if (!items.length) return;
1404
1405         var vols_to_move   = {};
1406         angular.forEach(items, function(item) {
1407             if (!(item.call_number.owning_lib in vols_to_move)) {
1408                 vols_to_move[item.call_number.owning_lib] = new Array;
1409             }
1410             vols_to_move[item.call_number.owning_lib].push(item.call_number.id);
1411         });
1412
1413         var promises = [];        
1414         angular.forEach(vols_to_move, function(vols, owning_lib) {
1415             promises.push(egCore.net.request(
1416                 'open-ils.cat',
1417                 'open-ils.cat.asset.volume.batch.transfer.override',
1418                 egCore.auth.token(), {
1419                     docid   : target_record,
1420                     lib     : owning_lib,
1421                     volumes : vols
1422                 }
1423             ));
1424         });
1425         $q.all(promises).then(function(success) {
1426             if (success) {
1427                 ngToast.create(egCore.strings.VOLS_TRANSFERED);
1428                 holdingsSvcInst.fetchAgain().then(function() {
1429                     $scope.holdingsGridDataProvider.refresh();
1430                 });
1431             } else {
1432                 alert('Could not transfer volumes!');
1433             }
1434         });
1435     }
1436
1437     function transferVolumes(new_record){
1438         var xfer_target = egCore.hatch.getLocalItem('eg.cat.volume_transfer_target');
1439
1440         if (xfer_target) {
1441             egCore.net.request(
1442                 'open-ils.cat',
1443                 'open-ils.cat.asset.volume.batch.transfer.override',
1444                 egCore.auth.token(), {
1445                     docid   : (new_record ? new_record : $scope.record_id),
1446                     lib     : xfer_target,
1447                     volumes : gatherSelectedVolumeIds()
1448                 }
1449             ).then(function(success) {
1450                 if (success) {
1451                     ngToast.create(egCore.strings.VOLS_TRANSFERED);
1452                     holdingsSvcInst.fetchAgain().then(function() {
1453                         $scope.holdingsGridDataProvider.refresh();
1454                     });
1455                 } else {
1456                     alert('Could not transfer volumes!');
1457                 }
1458             });
1459         }
1460         
1461     }
1462
1463     $scope.transferVolumesToLibrary = function() {
1464         transferVolumes();
1465     }
1466
1467     $scope.transferVolumesToRecordAndLibrary = function() {
1468         var target_record = egCore.hatch.getLocalItem('eg.cat.marked_volume_transfer_record');
1469         if (!target_record) return;
1470         transferVolumes(target_record);
1471     }
1472
1473     // this "transfers" selected copies to a new owning library,
1474     // auto-creating volumes and deleting unused volumes as required.
1475     $scope.changeItemOwningLib = function() {
1476         var xfer_target = egCore.hatch.getLocalItem('eg.cat.volume_transfer_target');
1477         var items = $scope.holdingsGridControls.selectedItems();
1478         if (!xfer_target || !items.length) {
1479             return;
1480         }
1481         var vols_to_move   = {};
1482         var copies_to_move = {};
1483         angular.forEach(items, function(item) {
1484             if (item.call_number.owning_lib != xfer_target) {
1485                 if (item.call_number.id in vols_to_move) {
1486                     copies_to_move[item.call_number.id].push(item.id);
1487                 } else {
1488                     vols_to_move[item.call_number.id] = item.call_number;
1489                     copies_to_move[item.call_number.id] = new Array;
1490                     copies_to_move[item.call_number.id].push(item.id);
1491                 }
1492             }
1493         });
1494     
1495         var promises = [];
1496         angular.forEach(vols_to_move, function(vol) {
1497             promises.push(egCore.net.request(
1498                 'open-ils.cat',
1499                 'open-ils.cat.call_number.find_or_create',
1500                 egCore.auth.token(),
1501                 vol.label,
1502                 vol.record,
1503                 xfer_target,
1504                 vol.prefix.id,
1505                 vol.suffix.id,
1506                 vol.label_class
1507             ).then(function(resp) {
1508                 var evt = egCore.evt.parse(resp);
1509                 if (evt) return;
1510                 return egCore.net.request(
1511                     'open-ils.cat',
1512                     'open-ils.cat.transfer_copies_to_volume',
1513                     egCore.auth.token(),
1514                     resp.acn_id,
1515                     copies_to_move[vol.id]
1516                 );
1517             }));
1518         });
1519         $q.all(promises).then(function() {
1520             ngToast.create(egCore.strings.ITEMS_TRANSFERED);
1521             holdingsSvcInst.fetchAgain().then(function() {
1522                 $scope.holdingsGridDataProvider.refresh();
1523             });
1524         });
1525     }
1526
1527     $scope.gridCellHandlers = {};
1528     $scope.gridCellHandlers.copyAlertsEdit = function(id) {
1529         egCirc.manage_copy_alerts([id]).then(function() {
1530             // update grid items?
1531         });
1532     };
1533
1534     $scope.transferItems = function (){
1535         var xfer_target = egCore.hatch.getLocalItem('eg.cat.item_transfer_target');
1536         var copy_ids = gatherSelectedHoldingsIds();
1537         if (xfer_target && copy_ids.length > 0) {
1538             egCore.net.request(
1539                 'open-ils.cat',
1540                 'open-ils.cat.transfer_copies_to_volume',
1541                 egCore.auth.token(),
1542                 xfer_target,
1543                 copy_ids
1544             ).then(
1545                 function(resp) { // oncomplete
1546                     var evt = egCore.evt.parse(resp);
1547                     if (evt) {
1548                         egConfirmDialog.open(
1549                             egCore.strings.OVERRIDE_TRANSFER_COPIES_TO_MARKED_VOLUME_TITLE,
1550                             egCore.strings.OVERRIDE_TRANSFER_COPIES_TO_MARKED_VOLUME_BODY,
1551                             {'evt_desc': evt.desc}
1552                         ).result.then(function() {
1553                             egCore.net.request(
1554                                 'open-ils.cat',
1555                                 'open-ils.cat.transfer_copies_to_volume.override',
1556                                 egCore.auth.token(),
1557                                 xfer_target,
1558                                 copy_ids,
1559                                 { events: ['TITLE_LAST_COPY', 'COPY_DELETE_WARNING'] }
1560                             ).then(function(resp) {
1561                                 holdingsSvcInst.fetchAgain().then(function() {
1562                                     $scope.holdingsGridDataProvider.refresh();
1563                                 });
1564                             });
1565                         });
1566                     } else {
1567                         ngToast.create(egCore.strings.ITEMS_TRANSFERED);
1568                         holdingsSvcInst.fetchAgain().then(function() {
1569                             $scope.holdingsGridDataProvider.refresh();
1570                         });
1571                     }
1572                 },
1573                 null, // onerror
1574                 null // onprogress
1575             )
1576         }
1577     }
1578
1579     $scope.selectedHoldingsItemStatusTgrEvt = function (){
1580         angular.forEach(
1581             gatherSelectedHoldingsIds(),
1582             function (cid) {
1583                 var url = egCore.env.basePath +
1584                           'cat/item/' + cid + '/triggered_events';
1585                 $timeout(function() { $window.open(url, '_blank') });
1586             }
1587         );
1588     }
1589
1590     $scope.selectedHoldingsItemStatusHolds = function (){
1591         angular.forEach(
1592             gatherSelectedHoldingsIds(),
1593             function (cid) {
1594                 var url = egCore.env.basePath +
1595                           'cat/item/' + cid + '/holds';
1596                 $timeout(function() { $window.open(url, '_blank') });
1597             }
1598         );
1599     }
1600
1601     $scope.selectedHoldingsPrintLabels = function() {
1602         egCore.net.request(
1603             'open-ils.actor',
1604             'open-ils.actor.anon_cache.set_value',
1605             null, 'print-labels-these-copies', {
1606                 copies : gatherSelectedHoldingsIds()
1607             }
1608         ).then(function(key) {
1609             if (key) {
1610                 var url = egCore.env.basePath + 'cat/printlabels/' + key;
1611                 $timeout(function() { $window.open(url, '_blank') });
1612             } else {
1613                 alert('Could not create anonymous cache key!');
1614             }
1615         });
1616     }
1617
1618     $scope.selectedHoldingsDamaged = function () {
1619         var copy_list = gatherSelectedRawCopies();
1620         if (copy_list.length == 0) return;
1621
1622         angular.forEach(copy_list, function(cp) {
1623             egCirc.mark_damaged({
1624                 id: cp.id(),
1625                 barcode: cp.barcode(),
1626                 circ_lib: cp.circ_lib().id()
1627             }).then(function() {
1628                 holdingsSvcInst.fetchAgain().then(function() {
1629                     $scope.holdingsGridDataProvider.refresh();
1630                 });
1631             });
1632         });
1633     }
1634
1635     $scope.selectedHoldingsMissing = function () {
1636         egCirc.mark_missing(gatherSelectedHoldingsIds()).then(function() {
1637             holdingsSvcInst.fetchAgain().then(function() {
1638                 $scope.holdingsGridDataProvider.refresh();
1639             });
1640         });
1641     }
1642
1643     $scope.selectedHoldingsCopyAlertsAdd = function() {
1644         egCirc.add_copy_alerts(gatherSelectedHoldingsIds()).then(function() {
1645             // no need to refresh grid
1646         });
1647     }
1648     $scope.selectedHoldingsCopyAlertsManage = function() {
1649         egCirc.manage_copy_alerts(gatherSelectedHoldingsIds()).then(function() {
1650             // no need to refresh grid
1651         });
1652     }
1653
1654     $scope.attach_to_peer_bib = function() {
1655         var copy_list = gatherSelectedHoldingsIds();
1656         if (copy_list.length == 0) return;
1657
1658         egCore.hatch.getItem('eg.cat.marked_conjoined_record').then(function(target_record) {
1659             if (!target_record) return;
1660
1661             return $uibModal.open({
1662                 templateUrl: './cat/catalog/t_conjoined_selector',
1663                 backdrop: 'static',
1664                 animation: true,
1665                 controller:
1666                        ['$scope','$uibModalInstance',
1667                 function($scope , $uibModalInstance) {
1668                     $scope.update = false;
1669
1670                     $scope.peer_type = null;
1671                     $scope.peer_type_list = [];
1672                     conjoinedSvc.get_peer_types().then(function(list){
1673                         $scope.peer_type_list = list;
1674                     });
1675     
1676                     $scope.ok = function(type) {
1677                         var promises = [];
1678     
1679                         angular.forEach(copy_list, function (cp) {
1680                             var n = new egCore.idl.bpbcm();
1681                             n.isnew(true);
1682                             n.peer_record(target_record);
1683                             n.target_copy(cp);
1684                             n.peer_type(type);
1685                             promises.push(egCore.pcrud.create(n));
1686                         });
1687     
1688                         return $q.all(promises).then(function(){$uibModalInstance.close()});
1689                     }
1690     
1691                     $scope.cancel = function($event) {
1692                         $uibModalInstance.dismiss();
1693                         $event.preventDefault();
1694                     }
1695                 }]
1696             });
1697         });
1698     }
1699
1700
1701     // ------------------------------------------------------------------
1702     // Holds 
1703     var provider = egGridDataProvider.instance({});
1704     $scope.hold_grid_data_provider = provider;
1705     $scope.grid_actions = egHoldGridActions;
1706     $scope.grid_actions.refresh = function () { provider.refresh() };
1707     $scope.hold_grid_controls = {};
1708
1709     var hold_ids = []; // current list of holds
1710     function fetchHolds(offset, count) {
1711         var ids = hold_ids.slice(offset, offset + count);
1712
1713         return egHolds.fetch_holds(ids).then(null, null,
1714             function(hold_data) { 
1715                 return hold_data;
1716             }
1717         );
1718     }
1719
1720     provider.get = function(offset, count) {
1721         if ($scope.record_tab != 'holds') return $q.when();
1722         var deferred = $q.defer();
1723         hold_ids = []; // no caching ATM
1724
1725         // open a determinate progress dialog, max value set below.
1726         egProgressDialog.open({max : 1, value : 0});
1727
1728         // fetch the IDs
1729         egCore.net.request(
1730             'open-ils.circ',
1731             'open-ils.circ.holds.retrieve_all_from_title',
1732             egCore.auth.token(), $scope.record_id, 
1733             {pickup_lib : egCore.org.descendants($scope.pickup_ou.id(), true)}
1734         ).then(
1735             function(hold_data) {
1736                 hold_ids = []; // clear the list of ids, hack to avoid dups
1737                 // TODO: fix the underlying problem, which is that
1738                 // this gets called twice when switching to the holds
1739                 // tab; once explicitly, and once via the change handler
1740                 // on the OU selector
1741                 angular.forEach(hold_data, function(list, type) {
1742                     hold_ids = hold_ids.concat(list);
1743                 });
1744
1745                 // Set the max value of the progress bar to the lesser of
1746                 // the total number of holds to fetch or the page size
1747                 // of the grid.
1748                 egProgressDialog.update(
1749                     {max : Math.min(hold_ids.length, count)});
1750
1751                 var holds_fetched = 0;
1752                 fetchHolds(offset, count)
1753                 .then(deferred.resolve, null, 
1754                     function(hold_data) {
1755                         holds_fetched++;
1756                         deferred.notify(hold_data);
1757                         egProgressDialog.increment();
1758                     }
1759                 )['finally'](egProgressDialog.close);
1760             }
1761         );
1762
1763         return deferred.promise;
1764     }
1765
1766     $scope.detail_view = function(action, user_data, items) {
1767         if (h = items[0]) {
1768             $scope.detail_hold_id = h.hold.id();
1769         }
1770     }
1771
1772     $scope.list_view = function(items) {
1773          $scope.detail_hold_id = null;
1774     }
1775
1776     // refresh the list of record holds when the pickup lib is changed.
1777     $scope.pickup_ou = egCore.org.get(egCore.auth.user().ws_ou());
1778     $scope.pickup_ou_changed = function(org) {
1779         $scope.pickup_ou = org;
1780         provider.refresh();
1781     }
1782
1783     $scope.print_holds = function() {
1784         var holds = [];
1785         angular.forEach($scope.hold_grid_controls.allItems(), function(item) {
1786             holds.push({
1787                 hold : egCore.idl.toHash(item.hold),
1788                 patron_last : item.patron_last,
1789                 patron_alias : item.patron_alias,
1790                 patron_barcode : item.patron_barcode,
1791                 copy : egCore.idl.toHash(item.copy),
1792                 volume : egCore.idl.toHash(item.volume),
1793                 title : item.mvr.title(),
1794                 author : item.mvr.author()
1795             });
1796         });
1797
1798         egCore.print.print({
1799             context : 'receipt', 
1800             template : 'holds_for_bib', 
1801             scope : {holds : holds}
1802         });
1803     }
1804
1805     $scope.current_hold_transfer_dest = egCore.hatch.getLocalItem ('eg.circ.hold.title_transfer_target');
1806
1807     $scope.mark_hold_transfer_dest = function() {
1808         $scope.current_hold_transfer_dest = $scope.record_id;
1809         egCore.hatch.setLocalItem(
1810             'eg.circ.hold.title_transfer_target', $scope.record_id);
1811         ngToast.create(egCore.strings.HOLD_TRANSFER_DEST_MARKED);
1812     }
1813
1814     // UI presents this option as "all holds"
1815     $scope.transfer_holds_to_marked = function() {
1816         var hold_ids = $scope.hold_grid_controls.allItems().map(
1817             function(hold_data) {return hold_data.hold.id()});
1818         egHolds.transfer_to_marked_title(hold_ids);
1819     }
1820
1821     // ------------------------------------------------------------------
1822     // Initialize the selected tab
1823
1824     // we explicitly initialize catalog_url because otherwise Firefox
1825     // ends up setting it to $BASE_URL/{{url}}, which then messes
1826     // things up. See LP#1708951
1827     $scope.catalog_url = '';
1828
1829     function init_cat_url() {
1830         // Set the initial catalog URL.  This only happens once.
1831         // The URL is otherwise generated through user navigation.
1832         if ($scope.catalog_url) return;
1833
1834         var url = $location.absUrl().replace(/\/staff.*/, '/opac/advanced');
1835
1836         // A record ID in the path indicates a request for the record-
1837         // specific page.
1838         if ($routeParams.record_id) {
1839             url = url.replace(/advanced/, '/record/' + $scope.record_id);
1840         }
1841
1842         // Jumping directly to the results page by passing a search
1843         // query via the URL.  Copy all URL params to the iframe url.
1844         if ($location.path().match(/catalog\/results/)) {
1845             url = url.replace(/advanced/, '/results?');
1846             var first = true;
1847             angular.forEach($location.search(), function(val, key) {
1848                 if (!first) url += '&';
1849                 first = false;
1850                 url += encodeURIComponent(key) 
1851                     + '=' + encodeURIComponent(val);
1852             });
1853         }
1854
1855         // if we're displaying the advanced search form, select
1856         // whatever default pane the user has chosen via workstation
1857         // preference
1858         if (url.match(/\/opac\/advanced$/)) {
1859             var adv_pane = egCore.hatch.getLocalItem('eg.search.adv_pane');
1860             if (adv_pane) {
1861                 url += '?pane=' + encodeURIComponent(adv_pane);
1862             }
1863         }
1864
1865         $scope.catalog_url = url;
1866     }
1867
1868     function init_parts_url() {
1869         $scope.parts_url = $location
1870             .absUrl()
1871             .replace(
1872                 /\/staff.*/,
1873                 '/conify/global/biblio/monograph_part?r='+$scope.record_id
1874             );
1875     }
1876
1877     $scope.set_record_tab = function(tab) {
1878         $scope.record_tab = tab;
1879
1880         switch(tab) {
1881
1882             case 'monoparts':
1883                 init_parts_url();
1884                 break;
1885
1886             case 'catalog':
1887                 init_cat_url();
1888                 break;
1889
1890             case 'holds':
1891                 $scope.detail_hold_record_id = $scope.record_id; 
1892                 // refresh the holds grid
1893                 provider.refresh();
1894
1895                 break;
1896         }
1897     }
1898
1899     $scope.set_default_record_tab = function() {
1900         egCore.hatch.setLocalItem(
1901             'eg.cat.default_record_tab', $scope.record_tab);
1902         $timeout(function(){$scope.default_tab = $scope.record_tab});
1903     }
1904
1905     var tab;
1906     if ($scope.record_id) {
1907         $scope.default_tab = egCore.hatch.getLocalItem( 'eg.cat.default_record_tab' );
1908         tab = $routeParams.record_tab || $scope.default_tab || 'catalog';
1909
1910     } else {
1911         tab = $routeParams.record_tab || 'catalog';
1912     }
1913     $scope.set_record_tab(tab);
1914
1915 }])
1916
1917 .controller('AuthorityCtrl',
1918        ['$scope','$routeParams','$location','$window','$q','egCore',
1919 function($scope , $routeParams , $location , $window , $q , egCore) {
1920
1921     // set record ID on page load if available...
1922     $scope.authority_id = $routeParams.authority_id;
1923
1924     if ($routeParams.authority_id) $scope.from_route = true;
1925     else $scope.from_route = false;
1926
1927     $scope.stop_unload = false;
1928 }])
1929
1930 .controller('URLVerifyCtrl',
1931        ['$scope','$location',
1932 function($scope , $location) {
1933     $scope.verifyurls_url = $location.absUrl().replace(/\/staff.*/, '/url_verify/sessions');
1934 }])
1935
1936 .controller('VandelayCtrl',
1937        ['$scope','$location', 'egCore', '$uibModal',
1938 function($scope , $location, egCore, $uibModal) {
1939     $scope.vandelay_url = $location.absUrl().replace(/\/staff\/cat\/catalog\/vandelay/, '/vandelay/vandelay');
1940     $scope.funcs = {};
1941     $scope.funcs.edit_marc_modal = function(bre, callback){
1942         var marcArgs = { 'marc_xml': bre.marc() };
1943         var vqbibrecId = bre.id();
1944         $uibModal.open({
1945             templateUrl: './cat/catalog/t_edit_marc_modal',
1946             backdrop: 'static',
1947             size: 'lg',
1948             controller: ['$scope', '$uibModalInstance', function($scope, $uibModalInstance) {
1949                 $scope.focusMe = true;
1950                 $scope.recordId = vqbibrecId;
1951                 $scope.args = marcArgs;
1952                 $scope.dirty_flag = false;
1953                 $scope.ok = function(marg){
1954                     $uibModalInstance.close(marg);
1955                 };
1956                 $scope.cancel = function(){ $uibModalInstance.dismiss() }
1957             }]
1958         }).result.then(function(res){
1959             var new_xml = res.marc_xml;
1960             egCore.pcrud.retrieve('vqbr', vqbibrecId).then(function(vqbib){
1961                 vqbib.marc(new_xml);
1962                 egCore.pcrud.update(vqbib).then( function(){ callback(vqbibrecId); });
1963             });
1964         });
1965     };
1966 }])
1967
1968 .controller('ManageAuthoritiesCtrl',
1969        ['$scope','$location',
1970 function($scope , $location) {
1971     $scope.manageauthorities_url = $location.absUrl().replace(/\/staff.*/, '/cat/authority/list');
1972 }])
1973
1974 .controller('BatchEditCtrl',
1975        ['$scope','$location','$routeParams',
1976 function($scope , $location , $routeParams) {
1977     $scope.batchedit_url = $location.absUrl().replace(/\/eg.*/, '/opac/extras/merge_template');
1978     if ($routeParams.container_type) {
1979         switch ($routeParams.container_type) {
1980             case 'bucket':
1981                 $scope.batchedit_url += '?recordSource=b&containerid=' + $routeParams.container_id;
1982                 break;
1983             case 'record':
1984                 $scope.batchedit_url += '?recordSource=r&recid=' + $routeParams.container_id;
1985                 break;
1986         };
1987     }
1988 }])
1989
1990  
1991 .filter('boolText', function(){
1992     return function (v) {
1993         return v == 't';
1994     }
1995 })
1996
1997 .factory('conjoinedSvc', 
1998        ['egCore','$q',
1999 function(egCore , $q) {
2000
2001     var service = {
2002         items : [], // record search results
2003         index : 0, // search grid index
2004         rid : null
2005     };
2006
2007     service.flesh = {   
2008         flesh : 4, 
2009         flesh_fields : {
2010             bpbcm : ['target_copy','peer_type'],
2011             acp : ['call_number'],
2012             acn : ['record'],
2013             bre : ['simple_record']
2014         },
2015         // avoid fetching the MARC blob by specifying which
2016         // fields on the bre to select.  More may be needed.
2017         // note that fleshed fields are explicitly selected.
2018         select : { bre : ['id'] },
2019         order_by : { bpbcm : ['id'] },
2020     }
2021
2022     // resolved with the last received copy
2023     service.fetch = function(rid) {
2024         if (!rid && !service.rid) return $q.when();
2025
2026         if (rid) service.rid = rid;
2027         service.items = [];
2028         service.index = 0;
2029
2030         return egCore.pcrud.search(
2031             'bpbcm',
2032             {peer_record : service.rid},
2033             service.flesh,
2034             {atomic : true}
2035         ).then( function(list) { // finished
2036             service.items = list;
2037             return service.items;
2038         });
2039     }
2040
2041     // returns a promise resolved with the list of peer bib types
2042     service.get_peer_types = function() {
2043         if (egCore.env.bpt)
2044             return $q.when(egCore.env.bpt.list);
2045
2046         return egCore.pcrud.retrieveAll('bpt', null, {atomic : true})
2047         .then(function(list) {
2048             egCore.env.absorbList(list, 'bpt');
2049             return list;
2050         });
2051     };
2052
2053     return service;
2054 }])
2055
2056