]> git.evergreen-ils.org Git - working/Evergreen.git/blob - Open-ILS/web/js/ui/default/staff/cat/catalog/app.js
webstaff: use activateItem to open the vol/copy editor
[working/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'])
11
12 .config(function($routeProvider, $locationProvider, $compileProvider) {
13     $locationProvider.html5Mode(true);
14     $compileProvider.aHrefSanitizationWhitelist(/^\s*(https?|blob):/); // grid export
15
16     var resolver = {delay : ['egCore','egStartup','egUser', function(egCore, egStartup, egUser) {
17         egCore.env.classLoaders.aous = function() {
18             return egCore.org.settings([
19                 'cat.marc_control_number_identifier'
20             ]).then(function(settings) {
21                 // local settings are cached within egOrg.  Caching them
22                 // again in egEnv just simplifies the syntax for access.
23                 egCore.env.aous = settings;
24             });
25         }
26         egCore.env.loadClasses.push('aous');
27         return egStartup.go()
28     }]};
29
30     $routeProvider.when('/cat/catalog/index', {
31         templateUrl: './cat/catalog/t_catalog',
32         controller: 'CatalogCtrl',
33         resolve : resolver
34     });
35
36     $routeProvider.when('/cat/catalog/retrieve_by_id', {
37         templateUrl: './cat/catalog/t_retrieve_by_id',
38         controller: 'CatalogRecordRetrieve',
39         resolve : resolver
40     });
41
42     $routeProvider.when('/cat/catalog/retrieve_by_tcn', {
43         templateUrl: './cat/catalog/t_retrieve_by_tcn',
44         controller: 'CatalogRecordRetrieve',
45         resolve : resolver
46     });
47
48     $routeProvider.when('/cat/catalog/new_bib', {
49         templateUrl: './cat/catalog/t_new_bib',
50         controller: 'NewBibCtrl',
51         resolve : resolver
52     });
53
54     // create some catalog page-specific mappings
55     $routeProvider.when('/cat/catalog/record/:record_id', {
56         templateUrl: './cat/catalog/t_catalog',
57         controller: 'CatalogCtrl',
58         resolve : resolver
59     });
60
61     // create some catalog page-specific mappings
62     $routeProvider.when('/cat/catalog/record/:record_id/:record_tab', {
63         templateUrl: './cat/catalog/t_catalog',
64         controller: 'CatalogCtrl',
65         resolve : resolver
66     });
67
68     $routeProvider.when('/cat/catalog/batchEdit', {
69         templateUrl: './cat/catalog/t_batchedit',
70         controller: 'BatchEditCtrl',
71         resolve : resolver
72     });
73
74     $routeProvider.when('/cat/catalog/batchEdit/:container_type/:container_id', {
75         templateUrl: './cat/catalog/t_batchedit',
76         controller: 'BatchEditCtrl',
77         resolve : resolver
78     });
79
80     $routeProvider.when('/cat/catalog/vandelay', {
81         templateUrl: './cat/catalog/t_vandelay',
82         controller: 'VandelayCtrl',
83         resolve : resolver
84     });
85
86     $routeProvider.when('/cat/catalog/verifyURLs', {
87         templateUrl: './cat/catalog/t_verifyurls',
88         controller: 'URLVerifyCtrl',
89         resolve : resolver
90     });
91
92     $routeProvider.when('/cat/catalog/manageAuthorities', {
93         templateUrl: './cat/catalog/t_manageauthorities',
94         controller: 'ManageAuthoritiesCtrl',
95         resolve : resolver
96     });
97
98     $routeProvider.when('/cat/catalog/authority/:authority_id/marc_edit', {
99         templateUrl: './cat/catalog/t_authority',
100         controller: 'AuthorityCtrl',
101         resolve : resolver
102     });
103
104     $routeProvider.otherwise({redirectTo : '/cat/catalog/index'});
105 })
106
107
108 /**
109  * */
110 .controller('CatalogRecordRetrieve',
111        ['$scope','$routeParams','$location','$q','egCore',
112 function($scope , $routeParams , $location , $q , egCore ) {
113
114     $scope.focusMe = true;
115
116     // jump to the patron checkout UI
117     function loadRecord(record_id) {
118         $location
119         .path('/cat/catalog/record/' + record_id);
120     }
121
122     $scope.submitId = function(args) {
123         $scope.recordNotFound = null;
124         if (!args.record_id) return;
125
126         // blur so next time it's set to true it will re-apply select()
127         $scope.selectMe = false;
128
129         return loadRecord(args.record_id);
130     }
131
132     $scope.submitTCN = function(args) {
133         $scope.recordNotFound = null;
134         $scope.moreRecordsFound = null;
135         if (!args.record_tcn) return;
136
137         // blur so next time it's set to true it will re-apply select()
138         $scope.selectMe = false;
139
140         // lookup TCN
141         egCore.net.request(
142             'open-ils.search',
143             'open-ils.search.biblio.tcn',
144             args.record_tcn)
145
146         .then(function(resp) { // get_barcodes
147
148             if (evt = egCore.evt.parse(resp)) {
149                 alert(evt); // FIXME
150                 return;
151             }
152
153             if (!resp.count) {
154                 $scope.recordNotFound = args.record_tcn;
155                 $scope.selectMe = true;
156                 return;
157             }
158
159             if (resp.count > 1) {
160                 $scope.moreRecordsFound = args.record_tcn;
161                 $scope.selectMe = true;
162                 return;
163             }
164
165             var record_id = resp.ids[0];
166             return loadRecord(record_id);
167         });
168     }
169
170 }])
171
172 .controller('NewBibCtrl',
173        ['$scope','$routeParams','$location','$window','$q','egCore',
174         'egGridDataProvider','egHoldGridActions','$timeout','holdingsSvc',
175 function($scope , $routeParams , $location , $window , $q , egCore) {
176
177     $scope.have_template = false;
178     $scope.marc_template = '';
179     $scope.stop_unload = false;
180     $scope.template_list = [];
181     $scope.template_name = '';
182     $scope.new_bib_id = 0;
183
184     egCore.net.request(
185         'open-ils.cat',
186         'open-ils.cat.marc_template.types.retrieve'
187     ).then(function(resp) {
188         angular.forEach(resp, function(name) {
189             $scope.template_list.push(name);
190         });
191         $scope.template_list.sort();
192     });
193     $scope.template_name = egCore.hatch.getSessionItem('eg.cat.last_bib_marc_template');
194     if (!$scope.template_name) {
195         egCore.hatch.getItem('cat.default_bib_marc_template').then(function(template) {
196             $scope.template_name = template;
197         });
198     }
199
200     $scope.loadTemplate = function() {
201         if ($scope.template_name) {
202             egCore.net.request(
203                 'open-ils.cat',
204                 'open-ils.cat.biblio.marc_template.retrieve',
205                 $scope.template_name
206             ).then(function(template) {
207                 $scope.marc_template = template;
208                 $scope.have_template = true;
209                 egCore.hatch.setSessionItem('eg.cat.last_bib_marc_template', $scope.template_name);
210             });
211         }
212     }
213
214     $scope.setDefaultTemplate = function() {
215         var hatch_key = "cat.default_bib_marc_template";
216         if ($scope.template_name) {
217             egCore.hatch.setItem(hatch_key, $scope.template_name);
218         } else {
219             egCore.hatch.removeItem(hatch_key);
220         }
221     }
222
223     $scope.$watch('new_bib_id', function(newVal, oldVal) {
224         if (newVal) {
225             $location.path('/cat/catalog/record/' + $scope.new_bib_id);
226         }
227     });
228     
229
230 }])
231
232 .controller('CatalogCtrl',
233        ['$scope','$routeParams','$location','$window','$q','egCore','egHolds','egCirc','egConfirmDialog',
234         'egGridDataProvider','egHoldGridActions','$timeout','$modal','holdingsSvc','egUser','conjoinedSvc',
235 function($scope , $routeParams , $location , $window , $q , egCore , egHolds , egCirc,  egConfirmDialog,
236          egGridDataProvider , egHoldGridActions , $timeout , $modal , holdingsSvc , egUser , conjoinedSvc) {
237
238     var holdingsSvcInst = new holdingsSvc();
239
240     // set record ID on page load if available...
241     $scope.record_id = $routeParams.record_id;
242     $scope.summary_pane_record;
243
244     if ($routeParams.record_id) $scope.from_route = true;
245     else $scope.from_route = false;
246
247     // will hold a ref to the opac iframe
248     $scope.opac_iframe = null;
249     $scope.parts_iframe = null;
250
251     $scope.in_opac_call = false;
252     $scope.opac_call = function (opac_frame_function, force_opac_tab) {
253         if ($scope.opac_iframe) {
254             if (force_opac_tab) $scope.record_tab = 'catalog';
255             $scope.in_opac_call = true;
256             $scope.opac_iframe.dom.contentWindow[opac_frame_function]();
257             if (opac_frame_function == 'rdetailBackToResults') {
258                 $location.update_path('/cat/catalog/index');
259             }
260         }
261     }
262
263     $scope.add_to_record_bucket = function() {
264         var recId = $scope.record_id;
265         return $modal.open({
266             templateUrl: './cat/catalog/t_add_to_bucket',
267             animation: true,
268             size: 'md',
269             controller:
270                    ['$scope','$modalInstance',
271             function($scope , $modalInstance) {
272
273                 $scope.bucket_id = 0;
274                 $scope.newBucketName = '';
275                 $scope.allBuckets = [];
276                 egCore.net.request(
277                     'open-ils.actor',
278                     'open-ils.actor.container.retrieve_by_class.authoritative',
279                     egCore.auth.token(), egCore.auth.user().id(),
280                     'biblio', 'staff_client'
281                 ).then(function(buckets) { $scope.allBuckets = buckets; });
282
283                 $scope.add_to_bucket = function() {
284                     var item = new egCore.idl.cbrebi();
285                     item.bucket($scope.bucket_id);
286                     item.target_biblio_record_entry(recId);
287                     egCore.net.request(
288                         'open-ils.actor',
289                         'open-ils.actor.container.item.create',
290                         egCore.auth.token(), 'biblio', item
291                     ).then(function(resp) {
292                         $modalInstance.close();
293                     });
294                 }
295
296                 $scope.add_to_new_bucket = function() {
297                     var bucket = new egCore.idl.cbreb();
298                     bucket.owner(egCore.auth.user().id());
299                     bucket.name($scope.newBucketName);
300                     bucket.description('');
301                     bucket.btype('staff_client');
302
303                     egCore.net.request(
304                         'open-ils.actor',
305                         'open-ils.actor.container.create',
306                         egCore.auth.token(), 'biblio', bucket
307                     ).then(function(bucket) {
308                         $scope.bucket_id = bucket;
309                         $scope.add_to_bucket();
310                     });
311                 }
312
313                 $scope.cancel = function() {
314                     $modalInstance.dismiss();
315                 }
316             }]
317         });
318     }
319
320     $scope.current_overlay_target     = egCore.hatch.getLocalItem('eg.cat.marked_overlay_record');
321     $scope.current_voltransfer_target = egCore.hatch.getLocalItem('eg.cat.marked_volume_transfer_record');
322     $scope.current_conjoined_target   = egCore.hatch.getLocalItem('eg.cat.marked_conjoined_record');
323
324     $scope.markConjoined = function () {
325         $scope.current_conjoined_target = $scope.record_id;
326         egCore.hatch.setLocalItem('eg.cat.marked_conjoined_record',$scope.record_id);
327     };
328
329     $scope.markVolTransfer = function () {
330         $scope.current_voltransfer_target = $scope.record_id;
331         egCore.hatch.setLocalItem('eg.cat.marked_volume_transfer_record',$scope.record_id);
332     };
333
334     $scope.markOverlay = function () {
335         $scope.current_overlay_target = $scope.record_id;
336         egCore.hatch.setLocalItem('eg.cat.marked_overlay_record',$scope.record_id);
337     };
338
339     $scope.clearRecordMarks = function () {
340         $scope.current_overlay_target     = null;
341         $scope.current_voltransfer_target = null;
342         $scope.current_conjoined_target   = null;
343         egCore.hatch.removeLocalItem('eg.cat.marked_volume_transfer_record');
344         egCore.hatch.removeLocalItem('eg.cat.marked_conjoined_record');
345         egCore.hatch.removeLocalItem('eg.cat.marked_overlay_record');
346     }
347
348     $scope.stop_unload = false;
349     $scope.$watch('stop_unload',
350         function(newVal, oldVal) {
351             if (newVal && newVal != oldVal && $scope.opac_iframe) {
352                 $($scope.opac_iframe.dom.contentWindow).on('beforeunload', function(){
353                     return 'There is unsaved data in this record.'
354                 });
355             } else {
356                 if ($scope.opac_iframe)
357                     $($scope.opac_iframe.dom.contentWindow).off('beforeunload');
358             }
359         }
360     );
361
362     // Set the "last bib" cookie, if we have that
363     if ($scope.record_id)
364         egCore.hatch.setLocalItem("eg.cat.last_record_retrieved", $scope.record_id);
365
366     $scope.refresh_record_callback = function (record_id) {
367         egCore.pcrud.retrieve('bre', record_id, {
368             flesh : 1,
369             flesh_fields : {
370                 bre : ['simple_record','creator','editor']
371             }
372         }).then(function(rec) {
373             rec.owner(egCore.org.get(rec.owner()));
374             $scope.summary_pane_record = rec;
375         });
376
377         return record_id;
378     }
379
380     // also set it when the iframe changes to a new record
381     $scope.handle_page = function(url) {
382
383         if (!url || url == 'about:blank') {
384             // nothing loaded.  If we already have a record ID, leave it.
385             return;
386         }
387
388         var match = url.match(/\/+opac\/+record\/+(\d+)/);
389         if (match) {
390             $scope.record_id = match[1];
391             egCore.hatch.setLocalItem("eg.cat.last_record_retrieved", $scope.record_id);
392             $scope.holdings_record_id_changed($scope.record_id);
393             conjoinedSvc.fetch($scope.record_id).then(function(){
394                 $scope.conjoinedGridDataProvider.refresh();
395             });
396             init_parts_url();
397             $location.update_path('/cat/catalog/record/' + $scope.record_id);
398         } else {
399             delete $scope.record_id;
400             $scope.from_route = false;
401         }
402
403         // child scope is executing this function, so our digest doesn't fire ... thus,
404         $scope.$apply();
405
406         if (!$scope.in_opac_call) {
407             if ($scope.record_id) {
408                 $scope.default_tab = egCore.hatch.getLocalItem( 'eg.cat.default_record_tab' );
409                 tab = $routeParams.record_tab || $scope.default_tab || 'catalog';
410             } else {
411                 tab = $routeParams.record_tab || 'catalog';
412             }
413             $scope.set_record_tab(tab);
414         } else {
415             $scope.in_opac_call = false;
416         }
417     }
418
419     // xulG catalog handlers
420     $scope.handlers = { }
421
422     // ------------------------------------------------------------------
423     // Conjoined items
424
425     $scope.conjoinedGridControls = {};
426     $scope.conjoinedGridDataProvider = egGridDataProvider.instance({
427         get : function(offset, count) {
428             return this.arrayNotifier(conjoinedSvc.items, offset, count);
429         }
430     });
431
432     $scope.changeConjoinedType = function () {
433         var peers = egCore.idl.Clone($scope.conjoinedGridControls.selectedItems());
434         angular.forEach(peers, function (p) {
435             p.target_copy(p.target_copy().id());
436             p.peer_type(p.peer_type().id());
437         });
438
439         var conjoinedGridDataProviderRef = $scope.conjoinedGridDataProvider;
440
441         return $modal.open({
442             templateUrl: './cat/catalog/t_conjoined_selector',
443             animation: true,
444             controller:
445                    ['$scope','$modalInstance',
446             function($scope , $modalInstance) {
447                 $scope.update = true;
448
449                 $scope.peer_type = null;
450                 $scope.peer_type_list = [];
451                 conjoinedSvc.get_peer_types().then(function(list){
452                     $scope.peer_type_list = list;
453                 });
454     
455                 $scope.ok = function(type) {
456                     var promises = [];
457     
458                     angular.forEach(peers, function (p) {
459                         p.ischanged(1);
460                         p.peer_type(type);
461                         promises.push(egCore.pcrud.update(p));
462                     });
463     
464                     return $q.all(promises)
465                         .then(function(){$modalInstance.close()})
466                         .then(function(){return conjoinedSvc.fetch()})
467                         .then(function(){conjoinedGridDataProviderRef.refresh()});
468                 }
469     
470                 $scope.cancel = function($event) {
471                     $modalInstance.dismiss();
472                     $event.preventDefault();
473                 }
474             }]
475         });
476         
477     }
478
479     $scope.refreshConjoined = function () {
480         conjoinedSvc.fetch($scope.record_id)
481         .then(function(){$scope.conjoinedGridDataProvider.refresh();});
482     }
483
484     $scope.deleteSelectedConjoined = function () {
485         var peers = $scope.conjoinedGridControls.selectedItems();
486
487         if (peers.length > 0) {
488             egConfirmDialog.open(
489                 egCore.strings.CONFIRM_DELETE_PEERS,
490                 egCore.strings.CONFIRM_DELETE_PEERS_MESSAGE,
491                 {peers : peers.length}
492             ).result.then(function() {
493                 angular.forEach(peers, function (p) {
494                     p.isdeleted(1);
495                 });
496
497                 egCore.pcrud.remove(peers).then(function() {
498                     return conjoinedSvc.fetch();
499                 }).then(function() {
500                     $scope.conjoinedGridDataProvider.refresh();
501                 });
502             });
503         }
504     }
505     if ($scope.record_id)
506         conjoinedSvc.fetch($scope.record_id);
507
508     // ------------------------------------------------------------------
509     // Holdings
510
511     $scope.holdingsGridControls = {
512         activateItem : function (item) {
513             $scope.selectedHoldingsVolCopyEdit();
514         }
515     };
516     $scope.holdingsGridDataProvider = egGridDataProvider.instance({
517         get : function(offset, count) {
518             return this.arrayNotifier(holdingsSvcInst.copies, offset, count);
519         }
520     });
521
522     $scope.add_copies_to_bucket = function() {
523         var copy_list = gatherSelectedHoldingsIds();
524         if (copy_list.length == 0) return;
525
526         return $modal.open({
527             templateUrl: './cat/catalog/t_add_to_bucket',
528             animation: true,
529             size: 'md',
530             controller:
531                    ['$scope','$modalInstance',
532             function($scope , $modalInstance) {
533
534                 $scope.bucket_id = 0;
535                 $scope.newBucketName = '';
536                 $scope.allBuckets = [];
537
538                 egCore.net.request(
539                     'open-ils.actor',
540                     'open-ils.actor.container.retrieve_by_class.authoritative',
541                     egCore.auth.token(), egCore.auth.user().id(),
542                     'copy', 'staff_client'
543                 ).then(function(buckets) { $scope.allBuckets = buckets; });
544
545                 $scope.add_to_bucket = function() {
546                     var promises = [];
547                     angular.forEach(copy_list, function (cp) {
548                         var item = new egCore.idl.ccbi()
549                         item.bucket($scope.bucket_id);
550                         item.target_copy(cp);
551                         promises.push(
552                             egCore.net.request(
553                                 'open-ils.actor',
554                                 'open-ils.actor.container.item.create',
555                                 egCore.auth.token(), 'copy', item
556                             )
557                         );
558
559                         return $q.all(promises).then(function() {
560                             $modalInstance.close();
561                         });
562                     });
563                 }
564
565                 $scope.add_to_new_bucket = function() {
566                     var bucket = new egCore.idl.ccb();
567                     bucket.owner(egCore.auth.user().id());
568                     bucket.name($scope.newBucketName);
569                     bucket.description('');
570                     bucket.btype('staff_client');
571
572                     return egCore.net.request(
573                         'open-ils.actor',
574                         'open-ils.actor.container.create',
575                         egCore.auth.token(), 'copy', bucket
576                     ).then(function(bucket) {
577                         $scope.bucket_id = bucket;
578                         $scope.add_to_bucket();
579                     });
580                 }
581
582                 $scope.cancel = function() {
583                     $modalInstance.dismiss();
584                 }
585             }]
586         });
587     }
588
589     $scope.requestItems = function() {
590         var copy_list = gatherSelectedHoldingsIds();
591         if (copy_list.length == 0) return;
592
593         return $modal.open({
594             templateUrl: './cat/catalog/t_request_items',
595             animation: true,
596             controller:
597                    ['$scope','$modalInstance',
598             function($scope , $modalInstance) {
599                 $scope.user = null;
600                 $scope.first_user_fetch = true;
601
602                 $scope.hold_data = {
603                     hold_type : 'C',
604                     copy_list : copy_list,
605                     pickup_lib: egCore.org.get(egCore.auth.user().ws_ou()),
606                     user      : egCore.auth.user().id()
607                 };
608
609                 egUser.get( $scope.hold_data.user ).then(function(u) {
610                     $scope.user = u;
611                     $scope.barcode = u.card().barcode();
612                     $scope.user_name = egUser.format_name(u);
613                     $scope.hold_data.user = u.id();
614                 });
615
616                 $scope.user_name = '';
617                 $scope.barcode = '';
618                 $scope.$watch('barcode', function (n) {
619                     if (!$scope.first_user_fetch) {
620                         egUser.getByBarcode(n).then(function(u) {
621                             $scope.user = u;
622                             $scope.user_name = egUser.format_name(u);
623                             $scope.hold_data.user = u.id();
624                         }, function() {
625                             $scope.user = null;
626                             $scope.user_name = '';
627                             delete $scope.hold_data.user;
628                         });
629                     }
630                     $scope.first_user_fetch = false;
631                 });
632
633                 $scope.ok = function(h) {
634                     var args = {
635                         patronid  : h.user,
636                         hold_type : h.hold_type,
637                         pickup_lib: h.pickup_lib.id(),
638                         depth     : 0
639                     };
640
641                     egCore.net.request(
642                         'open-ils.circ',
643                         'open-ils.circ.holds.test_and_create.batch.override',
644                         egCore.auth.token(), args, h.copy_list
645                     );
646
647                     $modalInstance.close();
648                 }
649
650                 $scope.cancel = function($event) {
651                     $modalInstance.dismiss();
652                     $event.preventDefault();
653                 }
654             }]
655         });
656     }
657
658     $scope.replaceBarcodes = function() {
659         var copy_list = gatherSelectedRawCopies();
660         if (copy_list.length == 0) return;
661
662         var holdingsGridDataProviderRef = $scope.holdingsGridDataProvider;
663
664         angular.forEach(copy_list, function (cp) {
665             $modal.open({
666                 templateUrl: './cat/share/t_replace_barcode',
667                 animation: true,
668                 controller:
669                            ['$scope','$modalInstance',
670                     function($scope , $modalInstance) {
671                         $scope.isModal = true;
672                         $scope.focusBarcode = false;
673                         $scope.focusBarcode2 = true;
674                         $scope.barcode1 = cp.barcode();
675
676                         $scope.updateBarcode = function() {
677                             $scope.copyNotFound = false;
678                             $scope.updateOK = false;
679                 
680                             egCore.pcrud.search('acp',
681                                 {deleted : 'f', barcode : $scope.barcode1})
682                             .then(function(copy) {
683                 
684                                 if (!copy) {
685                                     $scope.focusBarcode = true;
686                                     $scope.copyNotFound = true;
687                                     return;
688                                 }
689                 
690                                 $scope.copyId = copy.id();
691                                 copy.barcode($scope.barcode2);
692                 
693                                 egCore.pcrud.update(copy).then(function(stat) {
694                                     $scope.updateOK = stat;
695                                     $scope.focusBarcode = true;
696                                     holdingsSvc.fetchAgain().then(function (){
697                                         holdingsGridDataProviderRef.refresh();
698                                     });
699                                 });
700
701                             });
702                             $modalInstance.close();
703                         }
704
705                         $scope.cancel = function($event) {
706                             $modalInstance.dismiss();
707                             $event.preventDefault();
708                         }
709                     }
710                 ]
711             });
712         });
713     }
714
715     // refresh the list of holdings when the record_id is changed.
716     $scope.holdings_record_id_changed = function(id) {
717         if ($scope.record_id != id) $scope.record_id = id;
718         console.log('record id changed to ' + id + ', loading new holdings');
719         holdingsSvcInst.fetch({
720             rid : $scope.record_id,
721             org : $scope.holdings_ou,
722             copy: $scope.holdings_show_copies,
723             vol : $scope.holdings_show_vols,
724             empty: $scope.holdings_show_empty
725         }).then(function() {
726             $scope.holdingsGridDataProvider.refresh();
727         });
728     }
729
730     // refresh the list of holdings when the filter lib is changed.
731     $scope.holdings_ou = egCore.org.get(egCore.auth.user().ws_ou());
732     $scope.holdings_ou_changed = function(org) {
733         $scope.holdings_ou = org;
734         holdingsSvcInst.fetch({
735             rid : $scope.record_id,
736             org : $scope.holdings_ou,
737             copy: $scope.holdings_show_copies,
738             vol : $scope.holdings_show_vols,
739             empty: $scope.holdings_show_empty
740         }).then(function() {
741             $scope.holdingsGridDataProvider.refresh();
742         });
743     }
744
745     $scope.holdings_cb_changed = function(cb,newVal,norefresh) {
746         $scope[cb] = newVal;
747         egCore.hatch.setItem('cat.' + cb, newVal);
748         if (!norefresh) holdingsSvcInst.fetch({
749             rid : $scope.record_id,
750             org : $scope.holdings_ou,
751             copy: $scope.holdings_show_copies,
752             vol : $scope.holdings_show_vols,
753             empty: $scope.holdings_show_empty
754         }).then(function() {
755             $scope.holdingsGridDataProvider.refresh();
756         });
757     }
758
759     egCore.hatch.getItem('cat.holdings_show_vols').then(function(x){
760         if (typeof x ==  'undefined') x = true;
761         $scope.holdings_cb_changed('holdings_show_vols',x,true);
762         $('#holdings_show_vols').prop('checked', x);
763     }).then(function(){
764         egCore.hatch.getItem('cat.holdings_show_copies').then(function(x){
765             if (typeof x ==  'undefined') x = true;
766             $scope.holdings_cb_changed('holdings_show_copies',x,true);
767             $('#holdings_show_copies').prop('checked', x);
768         }).then(function(){
769             egCore.hatch.getItem('cat.holdings_show_empty').then(function(x){
770                 if (typeof x ==  'undefined') x = true;
771                 $scope.holdings_cb_changed('holdings_show_empty',x);
772                 $('#holdings_show_empty').prop('checked', x);
773             })
774         })
775     });
776
777     $scope.vols_not_shown = function () {
778         return !$scope.holdings_show_vols;
779     }
780
781     $scope.copies_not_shown = function () {
782         return !$scope.holdings_show_copies;
783     }
784
785     $scope.holdings_checkbox_handler = function (item) {
786         $scope.holdings_cb_changed(item.checkbox,item.checked);
787     }
788
789     function gatherSelectedHoldingsIds () {
790         var cp_id_list = [];
791         angular.forEach(
792             $scope.holdingsGridControls.selectedItems(),
793             function (item) { cp_id_list = cp_id_list.concat(item.id_list) }
794         );
795         return cp_id_list;
796     }
797
798     function gatherSelectedRawCopies () {
799         var cp_list = [];
800         angular.forEach(
801             $scope.holdingsGridControls.selectedItems(),
802             function (item) { if (item.raw) cp_list = cp_list.concat(item.raw) }
803         );
804         return cp_list;
805     }
806
807     function gatherSelectedEmptyVolumeIds () {
808         var cn_id_list = [];
809         angular.forEach(
810             $scope.holdingsGridControls.selectedItems(),
811             function (item) {
812                 if (item.copy_count == 0)
813                     cn_id_list.push(item.call_number.id)
814             }
815         );
816         return cn_id_list;
817     }
818
819     function gatherSelectedVolumeIds () {
820         var cn_id_list = [];
821         angular.forEach(
822             $scope.holdingsGridControls.selectedItems(),
823             function (item) {
824                 if (cn_id_list.indexOf(item.call_number.id) == -1)
825                     cn_id_list.push(item.call_number.id)
826             }
827         );
828         return cn_id_list;
829     }
830
831     $scope.selectedHoldingsDelete = function (vols, copies) {
832
833         var cnHash = {};
834         var perCnCopies = {};
835
836         var cn_count = 0;
837         var cp_count = 0;
838
839         angular.forEach(
840             $scope.holdingsGridControls.selectedItems(),
841             function (item) {
842                 if (vols && item.raw_call_number) {
843                     cnHash[item.call_number.id] = egCore.idl.Clone(item.raw_call_number);
844                     cnHash[item.call_number.id].isdeleted(1);
845                     cn_count++;
846                 } else if (copies) {
847                     angular.forEach(egCore.idl.Clone(item.raw), function (cp) {
848                         cp.isdeleted(1);
849                         cp_count++;
850                         var cn_id = cp.call_number().id();
851                         if (!cnHash[cn_id]) {
852                             cnHash[cn_id] = cp.call_number();
853                             perCnCopies[cn_id] = [cp];
854                         } else {
855                             perCnCopies[cn_id].push(cp);
856                         }
857                         cp.call_number(cn_id); // prevent loops in JSON-ification
858                     });
859
860                 }
861             }
862         );
863
864         angular.forEach(perCnCopies, function (v, k) {
865             if (vols) {
866                 cnHash[k].isdeleted(1);
867                 cn_count++;
868             }
869             cnHash[k].copies(v);
870         });
871
872         cnList = [];
873         angular.forEach(cnHash, function (v, k) {
874             cnList.push(v);
875         });
876
877         if (cnList.length == 0) return;
878
879         var flags = {};
880         if (vols && copies) flags.force_delete_copies = 1;
881
882         egConfirmDialog.open(
883             egCore.strings.CONFIRM_DELETE_COPIES_VOLUMES,
884             egCore.strings.CONFIRM_DELETE_COPIES_VOLUMES_MESSAGE,
885             {copies : cp_count, volumes : cn_count}
886         ).result.then(function() {
887             egCore.net.request(
888                 'open-ils.cat',
889                 'open-ils.cat.asset.volume.fleshed.batch.update.override',
890                 egCore.auth.token(), cnList, 1, flags
891             ).then(function(update_count) {
892                 $scope.holdingsGridDataProvider.refresh();
893             });
894         });
895     }
896     $scope.selectedHoldingsCopyDelete = function () { $scope.selectedHoldingsDelete(false,true) }
897     $scope.selectedHoldingsVolCopyDelete = function () { $scope.selectedHoldingsDelete(true,true) }
898     $scope.selectedHoldingsEmptyVolCopyDelete = function () { $scope.selectedHoldingsDelete(true,false) }
899
900     spawnHoldingsAdd = function (vols,copies){
901         var raw = [];
902         if (copies) { // just a copy on existing volumes
903             angular.forEach(gatherSelectedVolumeIds(), function (v) {
904                 raw.push( {callnumber : v} );
905             });
906         } else if (vols) {
907             angular.forEach(
908                 $scope.holdingsGridControls.selectedItems(),
909                 function (item) {
910                     raw.push({owner : item.owner_id});
911                 }
912             );
913         }
914
915         if (raw.length == 0) raw.push({});
916
917         egCore.net.request(
918             'open-ils.actor',
919             'open-ils.actor.anon_cache.set_value',
920             null, 'edit-these-copies', {
921                 record_id: $scope.record_id,
922                 raw: raw,
923                 hide_vols : false,
924                 hide_copies : false
925             }
926         ).then(function(key) {
927             if (key) {
928                 var url = egCore.env.basePath + 'cat/volcopy/' + key;
929                 $timeout(function() { $window.open(url, '_blank') });
930             } else {
931                 alert('Could not create anonymous cache key!');
932             }
933         });
934     }
935     $scope.selectedHoldingsVolCopyAdd = function () { spawnHoldingsAdd(true,false) }
936     $scope.selectedHoldingsCopyAdd = function () { spawnHoldingsAdd(false,true) }
937
938     spawnHoldingsEdit = function (hide_vols,hide_copies){
939         egCore.net.request(
940             'open-ils.actor',
941             'open-ils.actor.anon_cache.set_value',
942             null, 'edit-these-copies', {
943                 record_id: $scope.record_id,
944                 copies: gatherSelectedHoldingsIds(),
945                 raw: gatherSelectedEmptyVolumeIds().map(
946                     function(v){ return { callnumber : v } }
947                 ),
948                 hide_vols : hide_vols,
949                 hide_copies : hide_copies
950             }
951         ).then(function(key) {
952             if (key) {
953                 var url = egCore.env.basePath + 'cat/volcopy/' + key;
954                 $timeout(function() { $window.open(url, '_blank') });
955             } else {
956                 alert('Could not create anonymous cache key!');
957             }
958         });
959     }
960     $scope.selectedHoldingsVolCopyEdit = function () { spawnHoldingsEdit(false,false) }
961     $scope.selectedHoldingsVolEdit = function () { spawnHoldingsEdit(false,true) }
962     $scope.selectedHoldingsCopyEdit = function () { spawnHoldingsEdit(true,false) }
963
964     $scope.selectedHoldingsItemStatus = function (){
965         var url = egCore.env.basePath + 'cat/item/search/' + gatherSelectedHoldingsIds().join(',')
966         $timeout(function() { $window.open(url, '_blank') });
967     }
968
969     $scope.markVolAsItemTarget = function() {
970         if ($scope.holdingsGridControls.selectedItems()[0].call_number.id) { // cn.id missing when vols are collapsed
971             egCore.hatch.setLocalItem(
972                 'eg.cat.item_transfer_target',
973                 $scope.holdingsGridControls.selectedItems()[0].call_number.id
974             );
975         }
976     }
977
978     $scope.markLibAsVolTarget = function() {
979         egCore.hatch.setLocalItem(
980             'eg.cat.volume_transfer_target',
981             $scope.holdingsGridControls.selectedItems()[0].owner_id
982         );
983     }
984
985     $scope.selectedHoldingsItemStatusDetail = function (){
986         angular.forEach(
987             gatherSelectedHoldingsIds(),
988             function (cid) {
989                 var url = egCore.env.basePath +
990                           'cat/item/' + cid;
991                 $timeout(function() { $window.open(url, '_blank') });
992             }
993         );
994     }
995
996     $scope.transferVolumes = function (){
997         var xfer_target = egCore.hatch.getLocalItem('eg.cat.volume_transfer_target');
998
999         if (xfer_target) {
1000             egCore.net.request(
1001                 'open-ils.cat',
1002                 'open-ils.open-ils.cat.asset.volume.batch.transfer.override',
1003                 egCore.auth.token(), {
1004                     docid   : $scope.record_id,
1005                     lib     : xfer_target,
1006                     volumes : gatherSelectedVolumeIds()
1007                 }
1008             ).then(function(success) {
1009                 if (success) {
1010                     holdingsSvcInst.fetchAgain().then(function() {
1011                         $scope.holdingsGridDataProvider.refresh();
1012                     });
1013                 } else {
1014                     alert('Could not transfer volumes!');
1015                 }
1016             });
1017         }
1018         
1019     }
1020
1021     $scope.transferItems = function (){
1022         var xfer_target = egCore.hatch.getLocalItem('eg.cat.item_transfer_target');
1023         var copy_ids = gatherSelectedHoldingsIds();
1024         if (xfer_target && copy_ids.length > 0) {
1025             egCore.net.request(
1026                 'open-ils.cat',
1027                 'open-ils.cat.transfer_copies_to_volume',
1028                 egCore.auth.token(),
1029                 xfer_target,
1030                 copy_ids
1031             ).then(
1032                 function(resp) { // oncomplete
1033                     var evt = egCore.evt.parse(resp);
1034                     if (evt) {
1035                         egConfirmDialog.open(
1036                             egCore.strings.OVERRIDE_TRANSFER_COPIES_TO_MARKED_VOLUME_TITLE,
1037                             egCore.strings.OVERRIDE_TRANSFER_COPIES_TO_MARKED_VOLUME_BODY,
1038                             {'evt_desc': evt.desc}
1039                         ).result.then(function() {
1040                             egCore.net.request(
1041                                 'open-ils.cat',
1042                                 'open-ils.cat.transfer_copies_to_volume.override',
1043                                 egCore.auth.token(),
1044                                 xfer_target,
1045                                 copy_ids,
1046                                 { events: ['TITLE_LAST_COPY', 'COPY_DELETE_WARNING'] }
1047                             ).then(function(resp) {
1048                                 holdingsSvcInst.fetchAgain().then(function() {
1049                                     $scope.holdingsGridDataProvider.refresh();
1050                                 });
1051                             });
1052                         });
1053                     } else {
1054                         holdingsSvcInst.fetchAgain().then(function() {
1055                             $scope.holdingsGridDataProvider.refresh();
1056                         });
1057                     }
1058                 },
1059                 null, // onerror
1060                 null // onprogress
1061             )
1062         }
1063     }
1064
1065     $scope.selectedHoldingsItemStatusTgrEvt = function (){
1066         angular.forEach(
1067             gatherSelectedHoldingsIds(),
1068             function (cid) {
1069                 var url = egCore.env.basePath +
1070                           'cat/item/' + cid + '/triggered_events';
1071                 $timeout(function() { $window.open(url, '_blank') });
1072             }
1073         );
1074     }
1075
1076     $scope.selectedHoldingsItemStatusHolds = function (){
1077         angular.forEach(
1078             gatherSelectedHoldingsIds(),
1079             function (cid) {
1080                 var url = egCore.env.basePath +
1081                           'cat/item/' + cid + '/holds';
1082                 $timeout(function() { $window.open(url, '_blank') });
1083             }
1084         );
1085     }
1086
1087     $scope.selectedHoldingsDamaged = function () {
1088         egCirc.mark_damaged(gatherSelectedHoldingsIds()).then(function() {
1089             holdingsSvcInst.fetchAgain().then(function() {
1090                 $scope.holdingsGridDataProvider.refresh();
1091             });
1092         });
1093     }
1094
1095     $scope.selectedHoldingsMissing = function () {
1096         egCirc.mark_missing(gatherSelectedHoldingsIds()).then(function() {
1097             holdingsSvcInst.fetchAgain().then(function() {
1098                 $scope.holdingsGridDataProvider.refresh();
1099             });
1100         });
1101     }
1102
1103     $scope.attach_to_peer_bib = function() {
1104         var copy_list = gatherSelectedHoldingsIds();
1105         if (copy_list.length == 0) return;
1106
1107         egCore.hatch.getItem('eg.cat.marked_conjoined_record').then(function(target_record) {
1108             if (!target_record) return;
1109
1110             return $modal.open({
1111                 templateUrl: './cat/catalog/t_conjoined_selector',
1112                 animation: true,
1113                 controller:
1114                        ['$scope','$modalInstance',
1115                 function($scope , $modalInstance) {
1116                     $scope.update = false;
1117
1118                     $scope.peer_type = null;
1119                     $scope.peer_type_list = [];
1120                     conjoinedSvc.get_peer_types().then(function(list){
1121                         $scope.peer_type_list = list;
1122                     });
1123     
1124                     $scope.ok = function(type) {
1125                         var promises = [];
1126     
1127                         angular.forEach(copy_list, function (cp) {
1128                             var n = new egCore.idl.bpbcm();
1129                             n.isnew(true);
1130                             n.peer_record(target_record);
1131                             n.target_copy(cp);
1132                             n.peer_type(type);
1133                             promises.push(egCore.pcrud.create(n));
1134                         });
1135     
1136                         return $q.all(promises).then(function(){$modalInstance.close()});
1137                     }
1138     
1139                     $scope.cancel = function($event) {
1140                         $modalInstance.dismiss();
1141                         $event.preventDefault();
1142                     }
1143                 }]
1144             });
1145         });
1146     }
1147
1148
1149     // ------------------------------------------------------------------
1150     // Holds 
1151     var provider = egGridDataProvider.instance({});
1152     $scope.hold_grid_data_provider = provider;
1153     $scope.grid_actions = egHoldGridActions;
1154     $scope.grid_actions.refresh = function () { provider.refresh() };
1155     $scope.hold_grid_controls = {};
1156
1157     var hold_ids = []; // current list of holds
1158     function fetchHolds(offset, count) {
1159         var ids = hold_ids.slice(offset, offset + count);
1160         return egHolds.fetch_holds(ids).then(null, null,
1161             function(hold_data) { 
1162                 return hold_data;
1163             }
1164         );
1165     }
1166
1167     provider.get = function(offset, count) {
1168         if ($scope.record_tab != 'holds') return $q.when();
1169         var deferred = $q.defer();
1170         hold_ids = []; // no caching ATM
1171
1172         // fetch the IDs
1173         egCore.net.request(
1174             'open-ils.circ',
1175             'open-ils.circ.holds.retrieve_all_from_title',
1176             egCore.auth.token(), $scope.record_id, 
1177             {pickup_lib : egCore.org.descendants($scope.pickup_ou.id(), true)}
1178         ).then(
1179             function(hold_data) {
1180                 angular.forEach(hold_data, function(list, type) {
1181                     hold_ids = hold_ids.concat(list);
1182                 });
1183                 fetchHolds(offset, count).then(
1184                     deferred.resolve, null, deferred.notify);
1185             }
1186         );
1187
1188         return deferred.promise;
1189     }
1190
1191     $scope.detail_view = function(action, user_data, items) {
1192         if (h = items[0]) {
1193             $scope.detail_hold_id = h.hold.id();
1194         }
1195     }
1196
1197     $scope.list_view = function(items) {
1198          $scope.detail_hold_id = null;
1199     }
1200
1201     // refresh the list of record holds when the pickup lib is changed.
1202     $scope.pickup_ou = egCore.org.get(egCore.auth.user().ws_ou());
1203     $scope.pickup_ou_changed = function(org) {
1204         $scope.pickup_ou = org;
1205         provider.refresh();
1206     }
1207
1208     $scope.print_holds = function() {
1209         var holds = [];
1210         angular.forEach($scope.hold_grid_controls.allItems(), function(item) {
1211             holds.push({
1212                 hold : egCore.idl.toHash(item.hold),
1213                 patron_last : item.patron_last,
1214                 patron_alias : item.patron_alias,
1215                 patron_barcode : item.patron_barcode,
1216                 copy : egCore.idl.toHash(item.copy),
1217                 volume : egCore.idl.toHash(item.volume),
1218                 title : item.mvr.title(),
1219                 author : item.mvr.author()
1220             });
1221         });
1222
1223         egCore.print.print({
1224             context : 'receipt', 
1225             template : 'holds_for_bib', 
1226             scope : {holds : holds}
1227         });
1228     }
1229
1230     $scope.mark_hold_transfer_dest = function() {
1231         egCore.hatch.setLocalItem(
1232             'eg.circ.hold.title_transfer_target', $scope.record_id);
1233     }
1234
1235     // UI presents this option as "all holds"
1236     $scope.transfer_holds_to_marked = function() {
1237         var hold_ids = $scope.hold_grid_controls.allItems().map(
1238             function(hold_data) {return hold_data.hold.id()});
1239         egHolds.transfer_to_marked_title(hold_ids);
1240     }
1241
1242     // ------------------------------------------------------------------
1243     // Initialize the selected tab
1244
1245     function init_cat_url() {
1246         // Set the initial catalog URL.  This only happens once.
1247         // The URL is otherwise generated through user navigation.
1248         if ($scope.catalog_url) return; 
1249
1250         var url = $location.absUrl().replace(/\/staff.*/, '/opac/advanced');
1251
1252         // A record ID in the path indicates a request for the record-
1253         // specific page.
1254         if ($routeParams.record_id) {
1255             url = url.replace(/advanced/, '/record/' + $scope.record_id);
1256         }
1257
1258         $scope.catalog_url = url;
1259     }
1260
1261     function init_parts_url() {
1262         $scope.parts_url = $location
1263             .absUrl()
1264             .replace(
1265                 /\/staff.*/,
1266                 '/conify/global/biblio/monograph_part?r='+$scope.record_id
1267             );
1268     }
1269
1270     $scope.set_record_tab = function(tab) {
1271         $scope.record_tab = tab;
1272
1273         switch(tab) {
1274
1275             case 'monoparts':
1276                 init_parts_url();
1277                 break;
1278
1279             case 'catalog':
1280                 init_cat_url();
1281                 break;
1282
1283             case 'holds':
1284                 $scope.detail_hold_record_id = $scope.record_id; 
1285                 // refresh the holds grid
1286                 provider.refresh();
1287                 break;
1288         }
1289     }
1290
1291     $scope.set_default_record_tab = function() {
1292         egCore.hatch.setLocalItem(
1293             'eg.cat.default_record_tab', $scope.record_tab);
1294         $timeout(function(){$scope.default_tab = $scope.record_tab});
1295     }
1296
1297     var tab;
1298     if ($scope.record_id) {
1299         $scope.default_tab = egCore.hatch.getLocalItem( 'eg.cat.default_record_tab' );
1300         tab = $routeParams.record_tab || $scope.default_tab || 'catalog';
1301
1302     } else {
1303         tab = $routeParams.record_tab || 'catalog';
1304     }
1305     $scope.set_record_tab(tab);
1306
1307 }])
1308
1309 .controller('AuthorityCtrl',
1310        ['$scope','$routeParams','$location','$window','$q','egCore',
1311 function($scope , $routeParams , $location , $window , $q , egCore) {
1312
1313     // set record ID on page load if available...
1314     $scope.authority_id = $routeParams.authority_id;
1315
1316     if ($routeParams.authority_id) $scope.from_route = true;
1317     else $scope.from_route = false;
1318
1319     $scope.stop_unload = false;
1320 }])
1321
1322 .controller('URLVerifyCtrl',
1323        ['$scope','$location',
1324 function($scope , $location) {
1325     $scope.verifyurls_url = $location.absUrl().replace(/\/staff.*/, '/url_verify/sessions');
1326 }])
1327
1328 .controller('VandelayCtrl',
1329        ['$scope','$location',
1330 function($scope , $location) {
1331     $scope.vandelay_url = $location.absUrl().replace(/\/staff.*/, '/vandelay/vandelay');
1332 }])
1333
1334 .controller('ManageAuthoritiesCtrl',
1335        ['$scope','$location',
1336 function($scope , $location) {
1337     $scope.manageauthorities_url = $location.absUrl().replace(/\/staff.*/, '/cat/authority/list');
1338 }])
1339
1340 .controller('BatchEditCtrl',
1341        ['$scope','$location','$routeParams',
1342 function($scope , $location , $routeParams) {
1343     $scope.batchedit_url = $location.absUrl().replace(/\/eg.*/, '/opac/extras/merge_template');
1344     if ($routeParams.container_type) {
1345         switch ($routeParams.container_type) {
1346             case 'bucket':
1347                 $scope.batchedit_url += '?recordSource=b&containerid=' + $routeParams.container_id;
1348                 break;
1349             case 'record':
1350                 $scope.batchedit_url += '?recordSource=r&recid=' + $routeParams.container_id;
1351                 break;
1352         };
1353     }
1354 }])
1355
1356  
1357 .filter('boolText', function(){
1358     return function (v) {
1359         return v == 't';
1360     }
1361 })
1362
1363 .factory('conjoinedSvc', 
1364        ['egCore','$q',
1365 function(egCore , $q) {
1366
1367     var service = {
1368         items : [], // record search results
1369         index : 0, // search grid index
1370         rid : null
1371     };
1372
1373     service.flesh = {   
1374         flesh : 4, 
1375         flesh_fields : {
1376             bpbcm : ['target_copy','peer_type'],
1377             acp : ['call_number'],
1378             acn : ['record'],
1379             bre : ['simple_record']
1380         },
1381         // avoid fetching the MARC blob by specifying which
1382         // fields on the bre to select.  More may be needed.
1383         // note that fleshed fields are explicitly selected.
1384         select : { bre : ['id'] },
1385         order_by : { bpbcm : ['id'] },
1386     }
1387
1388     // resolved with the last received copy
1389     service.fetch = function(rid) {
1390         if (!rid && !service.rid) return $q.when();
1391
1392         if (rid) service.rid = rid;
1393         service.items = [];
1394         service.index = 0;
1395
1396         return egCore.pcrud.search(
1397             'bpbcm',
1398             {peer_record : service.rid},
1399             service.flesh,
1400             {atomic : true}
1401         ).then( function(list) { // finished
1402             service.items = list;
1403             return service.items;
1404         });
1405     }
1406
1407     // returns a promise resolved with the list of peer bib types
1408     service.get_peer_types = function() {
1409         if (egCore.env.bpt)
1410             return $q.when(egCore.env.bpt.list);
1411
1412         return egCore.pcrud.retrieveAll('bpt', null, {atomic : true})
1413         .then(function(list) {
1414             egCore.env.absorbList(list, 'bpt');
1415             return list;
1416         });
1417     };
1418
1419     return service;
1420 }])
1421
1422