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.
10 angular.module('egCatalogApp', ['ui.bootstrap','ngRoute','ngLocationUpdate','egCoreMod','egGridMod', 'egMarcMod', 'egUserMod', 'egHoldingsMod', 'ngToast','egPatronSearchMod',
11 'egSerialsMod','egSerialsAppDep'])
13 .config(['ngToastProvider', function(ngToastProvider) {
14 ngToastProvider.configure({
15 verticalPosition: 'bottom',
20 .config(function($routeProvider, $locationProvider, $compileProvider) {
21 $locationProvider.html5Mode(true);
22 $compileProvider.aHrefSanitizationWhitelist(/^\s*(https?|mailto|blob):/); // grid export
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;
34 egCore.env.loadClasses.push('aous');
38 $routeProvider.when('/cat/catalog/index', {
39 templateUrl: './cat/catalog/t_catalog',
40 controller: 'CatalogCtrl',
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',
52 $routeProvider.when('/cat/catalog/retrieve_by_id', {
53 templateUrl: './cat/catalog/t_retrieve_by_id',
54 controller: 'CatalogRecordRetrieve',
58 $routeProvider.when('/cat/catalog/retrieve_by_tcn', {
59 templateUrl: './cat/catalog/t_retrieve_by_tcn',
60 controller: 'CatalogRecordRetrieve',
64 $routeProvider.when('/cat/catalog/retrieve_by_authority_id', {
65 templateUrl: './cat/catalog/t_retrieve_by_authority_id',
66 controller: 'CatalogRecordRetrieve',
70 $routeProvider.when('/cat/catalog/new_bib', {
71 templateUrl: './cat/catalog/t_new_bib',
72 controller: 'NewBibCtrl',
76 // create some catalog page-specific mappings
77 $routeProvider.when('/cat/catalog/record/:record_id', {
78 templateUrl: './cat/catalog/t_catalog',
79 controller: 'CatalogCtrl',
83 // create some catalog page-specific mappings
84 $routeProvider.when('/cat/catalog/record/:record_id/:record_tab', {
85 templateUrl: './cat/catalog/t_catalog',
86 controller: 'CatalogCtrl',
90 $routeProvider.when('/cat/catalog/batchEdit', {
91 templateUrl: './cat/catalog/t_batchedit',
92 controller: 'BatchEditCtrl',
96 $routeProvider.when('/cat/catalog/batchEdit/:container_type/:container_id', {
97 templateUrl: './cat/catalog/t_batchedit',
98 controller: 'BatchEditCtrl',
102 $routeProvider.when('/cat/catalog/vandelay', {
103 templateUrl: './cat/catalog/t_vandelay',
104 controller: 'VandelayCtrl',
108 $routeProvider.when('/cat/catalog/verifyURLs', {
109 templateUrl: './cat/catalog/t_verifyurls',
110 controller: 'URLVerifyCtrl',
114 $routeProvider.when('/cat/catalog/manageAuthorities', {
115 templateUrl: './cat/catalog/t_manageauthorities',
116 controller: 'ManageAuthoritiesCtrl',
120 $routeProvider.when('/cat/catalog/authority/:authority_id/marc_edit', {
121 templateUrl: './cat/catalog/t_authority',
122 controller: 'AuthorityCtrl',
126 $routeProvider.otherwise({redirectTo : '/cat/catalog/index'});
132 .controller('CatalogRecordRetrieve',
133 ['$scope','$routeParams','$location','$q','egCore',
134 function($scope , $routeParams , $location , $q , egCore ) {
136 $scope.focusMe = true;
138 // jump to the patron checkout UI
139 function loadRecord(record_id) {
141 .path('/cat/catalog/record/' + record_id);
144 function loadAuthorityRecord(record_id) {
146 .path('/cat/catalog/authority/' + record_id + '/marc_edit');
149 $scope.submitId = function(args) {
150 $scope.recordNotFound = null;
151 if (!args.record_id) return;
153 // blur so next time it's set to true it will re-apply select()
154 $scope.selectMe = false;
156 return loadRecord(args.record_id);
159 $scope.submitAuthorityId = function(args) {
160 if (!args.record_id) return;
162 // blur so next time it's set to true it will re-apply select()
163 $scope.selectMe = false;
165 return loadAuthorityRecord(args.record_id);
168 $scope.submitTCN = function(args) {
169 $scope.recordNotFound = null;
170 $scope.moreRecordsFound = null;
171 if (!args.record_tcn) return;
173 // blur so next time it's set to true it will re-apply select()
174 $scope.selectMe = false;
179 'open-ils.search.biblio.tcn',
182 .then(function(resp) { // get_barcodes
185 return $q.when(resp);
187 // Search again including deleted records
188 return egCore.net.request('open-ils.search',
189 'open-ils.search.biblio.tcn', args.record_tcn, true);
192 }).then(function(resp2) {
195 $scope.recordNotFound = args.record_tcn;
196 $scope.selectMe = true;
200 if (resp2.count > 1) {
201 $scope.moreRecordsFound = args.record_tcn;
202 $scope.selectMe = true;
206 var record_id = resp2.ids[0];
207 return loadRecord(record_id);
213 .controller('NewBibCtrl',
214 ['$scope','$routeParams','$location','$window','$q','egCore',
215 'egGridDataProvider','egHoldGridActions','$timeout','holdingsSvc',
216 function($scope , $routeParams , $location , $window , $q , egCore) {
218 $scope.have_template = false;
219 $scope.marc_template = '';
220 $scope.stop_unload = false;
221 $scope.template_list = [];
222 $scope.template_name = '';
223 $scope.new_bib_id = 0;
227 'open-ils.cat.marc_template.types.retrieve'
228 ).then(function(resp) {
229 angular.forEach(resp, function(name) {
230 $scope.template_list.push(name);
232 $scope.template_list.sort();
234 $scope.template_name = egCore.hatch.getSessionItem('eg.cat.last_bib_marc_template');
235 if (!$scope.template_name) {
236 egCore.hatch.getItem('cat.default_bib_marc_template').then(function(template) {
237 $scope.template_name = template;
241 $scope.loadTemplate = function() {
242 if ($scope.template_name) {
245 'open-ils.cat.biblio.marc_template.retrieve',
247 ).then(function(template) {
248 $scope.marc_template = template;
249 $scope.have_template = true;
250 egCore.hatch.setSessionItem('eg.cat.last_bib_marc_template', $scope.template_name);
255 $scope.setDefaultTemplate = function() {
256 var hatch_key = "cat.default_bib_marc_template";
257 if ($scope.template_name) {
258 egCore.hatch.setItem(hatch_key, $scope.template_name);
260 egCore.hatch.removeItem(hatch_key);
264 $scope.$watch('new_bib_id', function(newVal, oldVal) {
266 location.href = '/eg2/staff/catalog/record/' + $scope.new_bib_id;
272 .controller('CatalogCtrl',
273 ['$scope','$routeParams','$location','$window','$q','egCore','egHolds','egCirc','egConfirmDialog','ngToast',
274 'egGridDataProvider','egHoldGridActions','egProgressDialog','$timeout','$uibModal','holdingsSvc','egUser','conjoinedSvc',
275 '$cookies','egSerialsCoreSvc',
276 function($scope , $routeParams , $location , $window , $q , egCore , egHolds , egCirc , egConfirmDialog , ngToast ,
277 egGridDataProvider , egHoldGridActions , egProgressDialog , $timeout , $uibModal , holdingsSvc , egUser , conjoinedSvc,
278 $cookies , egSerialsCoreSvc
281 var holdingsSvcInst = new holdingsSvc();
283 // set record ID on page load if available...
284 $scope.record_id = $routeParams.record_id;
285 $scope.summary_pane_record;
287 if ($scope.record_id) {
288 // TODO: Apply tab-specific title contexts
289 egCore.strings.setPageTitle(
290 egCore.strings.PAGE_TITLE_BIB_DETAIL,
291 egCore.strings.PAGE_TITLE_CATALOG_CONTEXT,
292 {record_id : $scope.record_id}
295 // Default to title = Catalog
296 egCore.strings.setPageTitle(
297 egCore.strings.PAGE_TITLE_CATALOG_CONTEXT);
300 if ($routeParams.record_id) $scope.from_route = true;
301 else $scope.from_route = false;
303 // set search and preferred library cookies
304 egCore.hatch.getItem('eg.search.search_lib').then(function(val) {
305 $cookies.put('eg_search_lib', val, { path : '/' });
307 egCore.hatch.getItem('eg.search.pref_lib').then(function(val) {
308 $cookies.put('eg_pref_lib', val, { path : '/' });
311 // will hold a ref to the opac iframe
312 $scope.opac_iframe = null;
313 $scope.parts_iframe = null;
315 $scope.search_result_index = 1;
316 $scope.search_result_hit_count = 1;
319 'opac_iframe.dom.contentWindow.search_result_index',
321 if (!isNaN(parseInt(n)))
322 $scope.search_result_index = n + 1;
327 'opac_iframe.dom.contentWindow.search_result_hit_count',
329 if (!isNaN(parseInt(n)))
330 $scope.search_result_hit_count = n;
334 $scope.in_opac_call = false;
335 $scope.opac_call = function (opac_frame_function, force_opac_tab) {
336 if ($scope.opac_iframe) {
337 if (force_opac_tab) $scope.record_tab = 'catalog';
338 $scope.in_opac_call = true;
339 $scope.opac_iframe.dom.contentWindow[opac_frame_function]();
340 if (opac_frame_function == 'rdetailBackToResults') {
341 $location.update_path('/cat/catalog/index');
346 $scope.add_cart_to_record_bucket = function() {
347 var cartkey = $cookies.get('cartcache');
348 if (!cartkey) return;
351 'open-ils.actor.anon_cache.get_value',
354 ).then(function(list) {
355 list = list.map(function(x) {
358 $scope.add_to_record_bucket(list);
362 $scope.add_to_record_bucket = function(recs) {
363 if (!angular.isArray(recs)) {
364 recs = [ $scope.record_id ];
366 return $uibModal.open({
367 templateUrl: './cat/catalog/t_add_to_bucket',
372 ['$scope','$uibModalInstance',
373 function($scope , $uibModalInstance) {
375 $scope.bucket_id = 0;
376 $scope.newBucketName = '';
377 $scope.allBuckets = [];
380 'open-ils.actor.container.retrieve_by_class.authoritative',
381 egCore.auth.token(), egCore.auth.user().id(),
382 'biblio', 'staff_client'
383 ).then(function(buckets) { $scope.allBuckets = buckets; });
385 $scope.add_to_bucket = function() {
387 angular.forEach(recs, function(recId) {
388 var item = new egCore.idl.cbrebi();
389 item.bucket($scope.bucket_id);
390 item.target_biblio_record_entry(recId);
391 promises.push(egCore.net.request(
393 'open-ils.actor.container.item.create',
394 egCore.auth.token(), 'biblio', item
397 $q.all(promises).then(function(resp) {
398 $uibModalInstance.close();
402 $scope.add_to_new_bucket = function() {
403 var bucket = new egCore.idl.cbreb();
404 bucket.owner(egCore.auth.user().id());
405 bucket.name($scope.newBucketName);
406 bucket.description('');
407 bucket.btype('staff_client');
411 'open-ils.actor.container.create',
412 egCore.auth.token(), 'biblio', bucket
413 ).then(function(bucket) {
414 $scope.bucket_id = bucket;
415 $scope.add_to_bucket();
419 $scope.cancel = function() {
420 $uibModalInstance.dismiss();
426 $scope.carousels_available = false;
429 'open-ils.actor.carousel.retrieve_manual_by_staff',
431 ).then(function(carousels) { $scope.carousels_available = true; });
433 $scope.add_to_carousel = function(recs) {
434 if (!angular.isArray(recs)) {
435 recs = [ $scope.record_id ];
437 return $uibModal.open({
438 templateUrl: './cat/catalog/t_add_to_carousel',
443 ['$scope','$uibModalInstance',
444 function($scope , $uibModalInstance) {
445 $scope.bucket_id = 0;
446 $scope.allCarousels = [];
449 'open-ils.actor.carousel.retrieve_manual_by_staff',
451 ).then(function(carousels) { $scope.allCarousels = carousels; });
453 $scope.add_to_carousel = function() {
454 // or more precisely, the carousel's bucket
456 angular.forEach(recs, function(recId) {
457 var item = new egCore.idl.cbrebi();
458 item.bucket($scope.bucket_id);
459 item.target_biblio_record_entry(recId);
460 promises.push(egCore.net.request(
462 'open-ils.actor.container.item.create',
463 egCore.auth.token(), 'biblio', item
466 $q.all(promises).then(function(resp) {
467 $uibModalInstance.close();
471 $scope.cancel = function() {
472 $uibModalInstance.dismiss();
478 $scope.current_overlay_target = egCore.hatch.getLocalItem('eg.cat.marked_overlay_record');
479 $scope.current_transfer_target = egCore.hatch.getLocalItem('eg.cat.transfer_target_record');
480 $scope.current_conjoined_target = egCore.hatch.getLocalItem('eg.cat.marked_conjoined_record');
482 $scope.quickReceive = function () {
484 var next_per_stream = {};
486 var recId = $scope.record_id;
487 return $uibModal.open({
488 templateUrl: './share/t_subscription_select_dialog',
490 controller: ['$scope', '$uibModalInstance',
491 function($scope, $uibModalInstance) {
494 $scope.rememberMe = 'eg.serials.quickreceive.last_org';
495 $scope.record_id = recId;
496 $scope.ssubId = null;
498 $scope.ok = function() { $uibModalInstance.close($scope.ssubId) }
499 $scope.cancel = function() { $uibModalInstance.dismiss(); }
502 }).result.then(function(ssubId) {
505 promises.push(egSerialsCoreSvc.fetchItemsForSub(ssubId,{status:'Expected'}).then(function(){
506 angular.forEach(egSerialsCoreSvc.itemTree, function (item) {
507 if (next_per_stream[item.stream().id()]) return;
508 if (item.status() == 'Expected') {
509 next_per_stream[item.stream().id()] = item;
510 list.push(egCore.idl.Clone(item));
515 return $q.all(promises).then(function() {
518 ngToast.warning(egCore.strings.SERIALS_NO_ITEMS);
522 return egSerialsCoreSvc.process_items(
528 false, // print by default
529 function() { $scope.holdings_record_id_changed($scope.record_id) }
533 ngToast.warning(egCore.strings.SERIALS_NO_SUBS);
539 $scope.markConjoined = function () {
540 $scope.current_conjoined_target = $scope.record_id;
541 egCore.hatch.setLocalItem('eg.cat.marked_conjoined_record',$scope.record_id);
542 ngToast.create(egCore.strings.MARK_CONJ_TARGET);
545 $scope.markHoldingsTransfer = function () {
546 $scope.current_transfer_target = $scope.record_id;
547 egCore.hatch.setLocalItem('eg.cat.transfer_target_record',$scope.record_id);
548 egCore.hatch.removeLocalItem('eg.cat.transfer_target_lib');
549 egCore.hatch.removeLocalItem('eg.cat.transfer_target_vol');
550 ngToast.create(egCore.strings.MARK_HOLDINGS_TARGET);
553 $scope.markOverlay = function () {
554 $scope.current_overlay_target = $scope.record_id;
555 egCore.hatch.setLocalItem('eg.cat.marked_overlay_record',$scope.record_id);
556 ngToast.create(egCore.strings.MARK_OVERLAY_TARGET);
559 $scope.clearRecordMarks = function () {
560 $scope.current_overlay_target = null;
561 $scope.current_transfer_target = null;
562 $scope.current_conjoined_target = null;
563 $scope.current_hold_transfer_dest = null;
564 egCore.hatch.removeLocalItem('eg.cat.transfer_target_record');
565 egCore.hatch.removeLocalItem('eg.cat.marked_conjoined_record');
566 egCore.hatch.removeLocalItem('eg.cat.marked_overlay_record');
567 egCore.hatch.removeLocalItem('eg.circ.hold.title_transfer_target');
570 $scope.stop_unload = false;
571 $scope.$watch('stop_unload',
572 function(newVal, oldVal) {
573 if (newVal && newVal != oldVal && $scope.opac_iframe) {
574 $($scope.opac_iframe.dom.contentWindow).on('beforeunload', function(){
575 return 'There is unsaved data in this record.'
578 if ($scope.opac_iframe)
579 $($scope.opac_iframe.dom.contentWindow).off('beforeunload');
584 // Set the "last bib" cookie, if we have that
585 if ($scope.record_id)
586 egCore.hatch.setLocalItem("eg.cat.last_record_retrieved", $scope.record_id);
588 $scope.refresh_record_callback = function (record_id) {
589 egCore.pcrud.retrieve('bre', record_id, {
592 bre : ['simple_record','creator','editor']
594 }).then(function(rec) {
595 rec.owner(egCore.org.get(rec.owner()));
596 $scope.summary_pane_record = rec;
602 patron_search_dialog = function() {
603 return $uibModal.open({
604 templateUrl: './share/t_patron_selector',
609 ['$scope','$uibModalInstance','$controller',
610 function($scope , $uibModalInstance , $controller) {
611 angular.extend(this, $controller('BasePatronSearchCtrl', {$scope : $scope}));
613 $scope.need_one_selected = function() {
614 var items = $scope.gridControls.selectedItems();
615 return (items.length == 1) ? false : true
617 $scope.ok = function() {
618 var items = $scope.gridControls.selectedItems();
619 if (items.length == 1) {
620 $uibModalInstance.close(items[0].card().barcode());
622 $uibModalInstance.close()
625 $scope.cancel = function($event) {
626 $uibModalInstance.dismiss();
627 $event.preventDefault();
633 // Map the Angular catalog-only 'item_table' tab to the AngJS
635 function get_default_record_tab() {
636 var tab = egCore.hatch.getLocalItem('eg.cat.default_record_tab');
637 if (!tab || tab === 'item_table') { return 'catalog'; }
641 // also set it when the iframe changes to a new record
642 $scope.handle_page = function(url) {
644 if (!url || url == 'about:blank') {
645 // nothing loaded. If we already have a record ID, leave it.
649 var prev_record_id = $scope.record_id;
650 var match = url.match(/\/+opac\/+record\/+(\d+)/);
652 $scope.record_id = match[1];
653 egCore.hatch.setLocalItem("eg.cat.last_record_retrieved", $scope.record_id);
654 $scope.holdings_record_id_changed($scope.record_id);
655 conjoinedSvc.fetch($scope.record_id).then(function(){
656 $scope.conjoinedGridDataProvider.refresh();
659 $scope.grid_actions.refresh();
660 $location.update_path('/cat/catalog/record/' + $scope.record_id);
661 // update_path() bypasses the controller for path
662 // /cat/catalog/record/:record_id. Manually set title here too.
663 egCore.strings.setPageTitle(
664 egCore.strings.PAGE_TITLE_BIB_DETAIL,
665 egCore.strings.PAGE_TITLE_CATALOG_CONTEXT,
666 {record_id : $scope.record_id}
669 delete $scope.record_id;
670 $scope.from_route = false;
673 // child scope is executing this function, so our digest doesn't fire ... thus,
676 // don't change tabs if we are using the OPAC nav buttons,
677 // or we didn't change records on the OPAC load
678 if (!$scope.in_opac_call && ($scope.record_id != prev_record_id)) {
679 if ($scope.record_id) {
680 $scope.default_tab = get_default_record_tab();
681 tab = $routeParams.record_tab || $scope.default_tab;
683 tab = $routeParams.record_tab || 'catalog';
685 $scope.set_record_tab(tab);
687 $scope.in_opac_call = false;
690 if ($scope.opac_iframe && $location.path().match(/cat\/catalog/)) {
691 var doc = $scope.opac_iframe.dom.contentWindow.document;
692 $(doc).find('#hold_usr_search').show();
693 $(doc).find('#hold_usr_search').on('click', function() {
694 patron_search_dialog().result.then(function(barc) {
695 $(doc).find('#hold_usr_input').val(barc);
696 $(doc).find('#hold_usr_input').trigger($.Event('keydown', {which: 13}));
699 // Add Cart to Record Bucket, in two flavors:
700 // First, the traditional TPAC, which uses a <select> menu
701 $(doc).find('#select_basket_action').on('change', function() {
702 if (this.options[this.selectedIndex].value && this.options[this.selectedIndex].value == "add_cart_to_bucket") {
703 $scope.add_cart_to_record_bucket();
706 // Second, the bootstrap OPAC, which uses a bunch of <a>s styled as a dropdown
707 $(doc).find('a[href="add_cart_to_bucket"]').on('click', function (event) {
708 event.preventDefault();
709 $scope.add_cart_to_record_bucket();
715 // xulG catalog handlers
716 $scope.handlers = { }
718 // ------------------------------------------------------------------
721 $scope.conjoinedGridControls = {};
722 $scope.conjoinedGridDataProvider = egGridDataProvider.instance({
723 get : function(offset, count) {
724 return this.arrayNotifier(conjoinedSvc.items, offset, count);
728 $scope.changeConjoinedType = function () {
729 var peers = egCore.idl.Clone($scope.conjoinedGridControls.selectedItems());
730 angular.forEach(peers, function (p) {
731 p.target_copy(p.target_copy().id());
732 p.peer_type(p.peer_type().id());
735 var conjoinedGridDataProviderRef = $scope.conjoinedGridDataProvider;
737 return $uibModal.open({
738 templateUrl: './cat/catalog/t_conjoined_selector',
742 ['$scope','$uibModalInstance',
743 function($scope , $uibModalInstance) {
744 $scope.update = true;
746 $scope.peer_type = null;
747 $scope.peer_type_list = [];
748 conjoinedSvc.get_peer_types().then(function(list){
749 $scope.peer_type_list = list;
752 $scope.ok = function(type) {
755 angular.forEach(peers, function (p) {
758 promises.push(egCore.pcrud.update(p));
761 return $q.all(promises)
762 .then(function(){$uibModalInstance.close()})
763 .then(function(){return conjoinedSvc.fetch()})
764 .then(function(){conjoinedGridDataProviderRef.refresh()});
767 $scope.cancel = function($event) {
768 $uibModalInstance.dismiss();
769 $event.preventDefault();
776 $scope.refreshConjoined = function () {
777 conjoinedSvc.fetch($scope.record_id)
778 .then(function(){$scope.conjoinedGridDataProvider.refresh();});
781 $scope.deleteSelectedConjoined = function () {
782 var peers = $scope.conjoinedGridControls.selectedItems();
784 if (peers.length > 0) {
785 egConfirmDialog.open(
786 egCore.strings.CONFIRM_DELETE_PEERS,
787 egCore.strings.CONFIRM_DELETE_PEERS_MESSAGE,
788 {peers : peers.length}
789 ).result.then(function() {
790 angular.forEach(peers, function (p) {
794 egCore.pcrud.remove(peers).then(function() {
795 return conjoinedSvc.fetch();
797 $scope.conjoinedGridDataProvider.refresh();
802 if ($scope.record_id)
803 conjoinedSvc.fetch($scope.record_id);
805 // ------------------------------------------------------------------
808 $scope.holdingsGridControls = {
809 activateItem : function (item) {
810 $scope.selectedHoldingsVolCopyEdit();
813 $scope.holdingsGridDataProvider = egGridDataProvider.instance({
814 get : function(offset, count) {
815 return this.arrayNotifier(holdingsSvcInst.copies, offset, count);
819 $scope.add_copies_to_bucket = function() {
820 var copy_list = gatherSelectedHoldingsIds();
821 if (copy_list.length == 0) return;
823 return $uibModal.open({
824 templateUrl: './cat/catalog/t_add_to_bucket',
829 ['$scope','$uibModalInstance',
830 function($scope , $uibModalInstance) {
832 $scope.bucket_id = 0;
833 $scope.newBucketName = '';
834 $scope.allBuckets = [];
838 'open-ils.actor.container.retrieve_by_class.authoritative',
839 egCore.auth.token(), egCore.auth.user().id(),
840 'copy', 'staff_client'
841 ).then(function(buckets) { $scope.allBuckets = buckets; });
843 $scope.add_to_bucket = function() {
845 angular.forEach(copy_list, function (cp) {
846 var item = new egCore.idl.ccbi()
847 item.bucket($scope.bucket_id);
848 item.target_copy(cp);
852 'open-ils.actor.container.item.create',
853 egCore.auth.token(), 'copy', item
857 return $q.all(promises).then(function() {
858 $uibModalInstance.close();
863 $scope.add_to_new_bucket = function() {
864 var bucket = new egCore.idl.ccb();
865 bucket.owner(egCore.auth.user().id());
866 bucket.name($scope.newBucketName);
867 bucket.description('');
868 bucket.btype('staff_client');
870 return egCore.net.request(
872 'open-ils.actor.container.create',
873 egCore.auth.token(), 'copy', bucket
874 ).then(function(bucket) {
875 $scope.bucket_id = bucket;
876 $scope.add_to_bucket();
880 $scope.cancel = function() {
881 $uibModalInstance.dismiss();
887 // TODO: refactor common code between cat/catalog/app.js and cat/item/app.js
889 $scope.need_one_selected = function() {
890 var items = $scope.holdingsGridControls.selectedItems();
891 if (items.length == 1) return false;
895 $scope.make_copies_bookable = function() {
897 var copies_by_record = {};
898 var record_list = [];
900 $scope.holdingsGridControls.selectedItems(),
902 var record_id = item['call_number.record.id'];
903 if (typeof copies_by_record[ record_id ] == 'undefined') {
904 copies_by_record[ record_id ] = [];
905 record_list.push( record_id );
907 copies_by_record[ record_id ].push(item.id);
912 var combined_results = [];
913 angular.forEach(record_list, function(record_id) {
917 'open-ils.booking.resources.create_from_copies',
919 copies_by_record[record_id]
920 ).then(function(results) {
921 if (results && results['brsrc']) {
922 combined_results = combined_results.concat(results['brsrc']);
928 $q.all(promises).then(function() {
929 if (combined_results.length > 0) {
931 template: '<eg-embed-frame url="booking_admin_url" handlers="funcs"></eg-embed-frame>',
936 ['$scope','$location','egCore','$uibModalInstance',
937 function($scope , $location , egCore , $uibModalInstance) {
940 ses : egCore.auth.token(),
941 resultant_brsrc : combined_results.map(function(o) { return o[0]; })
944 var booking_path = '/eg/conify/global/booking/resource';
946 $scope.booking_admin_url =
947 $location.absUrl().replace(/\/eg\/staff.*/, booking_path);
954 $scope.book_copies_now = function(items) {
955 location.href = "/eg2/staff/booking/create_reservation/for_resource/" + items[0]['barcode'];
958 $scope.requestItems = function() {
959 var copy_list = gatherSelectedHoldingsIds();
960 if (copy_list.length == 0) return;
962 return $uibModal.open({
963 templateUrl: './cat/catalog/t_request_items',
966 ['$scope','$uibModalInstance',
967 function($scope , $uibModalInstance) {
969 $scope.first_user_fetch = true;
973 copy_list : copy_list,
974 pickup_lib: egCore.org.get(egCore.auth.user().ws_ou()),
975 user : egCore.auth.user().id()
978 egUser.get( $scope.hold_data.user ).then(function(u) {
980 $scope.barcode = u.card().barcode();
981 $scope.user_name = egUser.format_name(u);
982 $scope.hold_data.user = u.id();
985 $scope.user_name = '';
987 $scope.$watch('barcode', function (n) {
988 if (!$scope.first_user_fetch) {
989 egUser.getByBarcode(n).then(function(u) {
991 $scope.user_name = egUser.format_name(u);
992 $scope.hold_data.user = u.id();
995 $scope.user_name = '';
996 delete $scope.hold_data.user;
999 $scope.first_user_fetch = false;
1002 $scope.ok = function(h) {
1005 hold_type : h.hold_type,
1006 pickup_lib: h.pickup_lib.id(),
1012 'open-ils.circ.holds.test_and_create.batch.override',
1013 egCore.auth.token(), args, h.copy_list
1015 holds = []; // force the holds grid to refetch data.
1016 $uibModalInstance.close();
1020 $scope.cancel = function($event) {
1021 $uibModalInstance.dismiss();
1022 $event.preventDefault();
1028 $scope.manage_reservations = function() {
1029 var item = $scope.holdingsGridControls.selectedItems()[0];
1031 location.href = "/eg2/staff/booking/manage_reservations/by_resource/" + item.barcode;
1035 $scope.view_place_orders = function() {
1036 if (!$scope.record_id) return;
1037 var url = egCore.env.basePath + 'acq/legacy/lineitem/related/' + $scope.record_id + '?target=bib';
1038 $timeout(function() { $window.open(url, '_blank') });
1041 $scope.replaceBarcodes = function() {
1042 var copy_list = gatherSelectedRawCopies();
1043 if (copy_list.length == 0) return;
1045 var holdingsGridDataProviderRef = $scope.holdingsGridDataProvider;
1047 angular.forEach(copy_list, function (cp) {
1049 templateUrl: './cat/share/t_replace_barcode',
1053 ['$scope','$uibModalInstance',
1054 function($scope , $uibModalInstance) {
1055 $scope.duplicate_barcode = false;
1056 $scope.isModal = true;
1057 $scope.focusBarcode = false;
1058 $scope.focusBarcode2 = true;
1059 $scope.barcode1 = cp.barcode();
1061 // check input to see if it's a duplicate barcode
1062 $scope.checkCurrentBarcode = function() {
1063 if (!$scope.duplicate_barcode_string) {
1064 $scope.duplicate_barcode_string = window.duplicate_barcode_string;
1066 var searchParams = {
1068 'barcode' : $scope.barcode2,
1069 id : { '!=' : $scope.copyId }
1071 egCore.pcrud.search('acp', searchParams).then(function (res) {
1072 $scope.duplicate_barcode = res;
1076 $scope.updateBarcode = function() {
1077 $scope.copyNotFound = false;
1078 $scope.updateOK = false;
1080 egCore.pcrud.search('acp',
1081 {deleted : 'f', barcode : $scope.barcode1})
1082 .then(function(copy) {
1085 $scope.focusBarcode = true;
1086 $scope.copyNotFound = true;
1090 $scope.copyId = copy.id();
1091 copy.barcode($scope.barcode2);
1093 egCore.pcrud.update(copy).then(function(stat) {
1094 $scope.updateOK = stat;
1095 $scope.focusBarcode = true;
1096 holdingsSvc.fetchAgain().then(function (){
1097 holdingsGridDataProviderRef.refresh();
1102 $uibModalInstance.close();
1105 $scope.cancel = function($event) {
1106 $uibModalInstance.dismiss();
1107 $event.preventDefault();
1115 var holdings_bChannel = null;
1116 // subscribe to BroadcastChannel for any child VolCopy tabs
1117 // refresh grid if needed to show new updates
1118 // if ($scope.record_tab === 'holdings'){
1119 $scope.$watch('record_tab', function(n){
1121 if (n === 'holdings'){
1122 if (typeof BroadcastChannel != 'undefined') {
1123 // we're in holdings tab, connect 2 bChannel
1124 holdings_bChannel = new BroadcastChannel('eg.holdings.update');
1125 holdings_bChannel.onmessage = function(e){
1128 && e.data.records.length
1129 && e.data.records.includes(Number($scope.record_id))
1130 ){ // it's for us, refresh grid!
1131 console.log("Got broadcast from channel eg.holdings.update for records " + e.data.records);
1132 $scope.holdings_record_id_changed($scope.record_id);
1137 } else if (holdings_bChannel){ // we're leaving holding tab, close bChannel
1138 holdings_bChannel.close();
1143 // refresh the list of holdings when the record_id is changed.
1144 $scope.holdings_record_id_changed = function(id) {
1145 if ($scope.record_id != id) $scope.record_id = id;
1146 console.log('record id changed to ' + id + ', loading new holdings');
1147 holdingsSvcInst.fetch({
1148 rid : $scope.record_id,
1149 org : $scope.holdings_ou,
1150 copy: $scope.holdings_show_vols ? $scope.holdings_show_copies : false,
1151 vol : $scope.holdings_show_vols,
1152 empty: $scope.holdings_show_empty,
1153 empty_org: $scope.holdings_show_empty_org
1154 }).then(function() {
1155 $scope.holdingsGridDataProvider.refresh();
1159 // refresh the list of holdings when the filter lib is changed.
1160 $scope.holdings_ou = egCore.org.get(egCore.auth.user().ws_ou());
1161 $scope.holdings_ou_changed = function(org) {
1162 $scope.holdings_ou = org;
1163 holdingsSvcInst.fetch({
1164 rid : $scope.record_id,
1165 org : $scope.holdings_ou,
1166 copy: $scope.holdings_show_vols ? $scope.holdings_show_copies : false,
1167 vol : $scope.holdings_show_vols,
1168 empty: $scope.holdings_show_empty,
1169 empty_org: $scope.holdings_show_empty_org
1170 }).then(function() {
1171 $scope.holdingsGridDataProvider.refresh();
1175 $scope.holdings_cb_changed = function(cb,newVal,norefresh) {
1176 $scope[cb] = newVal;
1177 var x = $scope.holdings_show_vols ? $scope.holdings_show_copies : false;
1178 $('#holdings_show_copies').prop('checked', x);
1179 egCore.hatch.setItem('cat.' + cb, newVal);
1180 if (!norefresh) holdingsSvcInst.fetch({
1181 rid : $scope.record_id,
1182 org : $scope.holdings_ou,
1183 copy: $scope.holdings_show_vols ? $scope.holdings_show_copies : false,
1184 vol : $scope.holdings_show_vols,
1185 empty: $scope.holdings_show_empty,
1186 empty_org: $scope.holdings_show_empty_org
1187 }).then(function() {
1188 $scope.holdingsGridDataProvider.refresh();
1192 egCore.hatch.getItem('cat.holdings_show_vols').then(function(x){
1193 if (typeof x == 'undefined') x = true;
1194 $scope.holdings_cb_changed('holdings_show_vols',x,true);
1195 $('#holdings_show_vols').prop('checked', x);
1197 egCore.hatch.getItem('cat.holdings_show_copies').then(function(x){
1198 if (typeof x == 'undefined') x = true;
1199 $scope.holdings_cb_changed('holdings_show_copies',x,true);
1200 x = $scope.holdings_show_vols ? x : false;
1201 $('#holdings_show_copies').prop('checked', x);
1203 egCore.hatch.getItem('cat.holdings_show_empty').then(function(x){
1204 if (typeof x == 'undefined') x = true;
1205 $scope.holdings_cb_changed('holdings_show_empty',x);
1206 $('#holdings_show_empty').prop('checked', x);
1208 egCore.hatch.getItem('cat.holdings_show_empty_org').then(function(x){
1209 if (typeof x == 'undefined') x = true;
1210 $scope.holdings_cb_changed('holdings_show_empty_org',x);
1211 $('#holdings_show_empty_org').prop('checked', x);
1217 $scope.vols_not_shown = function () {
1218 return !$scope.holdings_show_vols;
1221 $scope.copies_not_shown = function () {
1222 return !$scope.holdings_show_copies;
1225 $scope.empty_org_not_shown = function () {
1226 return !$scope.holdings_show_empty_org;
1229 $scope.holdings_checkbox_handler = function (item) {
1230 $scope.holdings_cb_changed(item.checkbox,item.checked);
1233 function gatherSelectedHoldingsIds () {
1234 var cp_id_list = [];
1236 $scope.holdingsGridControls.selectedItems(),
1237 function (item) { cp_id_list = cp_id_list.concat(item.id_list) }
1242 function gatherSelectedRawCopies () {
1245 $scope.holdingsGridControls.selectedItems(),
1246 function (item) { if (item.raw) cp_list = cp_list.concat(item.raw) }
1251 function gatherSelectedEmptyVolumeIds () {
1252 var cn_id_list = [];
1254 $scope.holdingsGridControls.selectedItems(),
1256 if (item.copy_count == 0 || (!item.id && item.call_number))
1257 // we are in a compressed row with no copies, or we are in a single
1258 // call number row with no copy (testing for presence of 'id')
1259 // In either case, the call number is 'empty'
1260 cn_id_list.push(item.call_number.id)
1266 function gatherSelectedVolumeIds () {
1267 var cn_id_list = [];
1269 $scope.holdingsGridControls.selectedItems(),
1271 if (cn_id_list.indexOf(item.call_number.id) == -1)
1272 cn_id_list.push(item.call_number.id)
1278 $scope.selectedHoldingsDelete = function (vols, copies) {
1281 var perCnCopies = {};
1287 $scope.holdingsGridControls.selectedItems(),
1289 if (vols && item.raw_call_number) {
1290 cnHash[item.call_number.id] = egCore.idl.Clone(item.raw_call_number);
1291 cnHash[item.call_number.id].isdeleted(1);
1293 } else if (copies) {
1294 angular.forEach(egCore.idl.Clone(item.raw), function (cp) {
1297 var cn_id = cp.call_number().id();
1298 if (!cnHash[cn_id]) {
1299 cnHash[cn_id] = cp.call_number();
1300 perCnCopies[cn_id] = [cp];
1302 perCnCopies[cn_id].push(cp);
1304 cp.call_number(cn_id); // prevent loops in JSON-ification
1311 angular.forEach(perCnCopies, function (v, k) {
1313 cnHash[k].isdeleted(1);
1316 cnHash[k].copies(v);
1320 angular.forEach(cnHash, function (v, k) {
1324 if (cnList.length == 0) return;
1327 if (vols && copies) flags.force_delete_copies = 1;
1329 egConfirmDialog.open(
1330 egCore.strings.CONFIRM_DELETE_COPIES_VOLUMES,
1331 egCore.strings.CONFIRM_DELETE_COPIES_VOLUMES_MESSAGE,
1332 {copies : cp_count, volumes : cn_count}
1333 ).result.then(function() {
1336 'open-ils.cat.asset.volume.fleshed.batch.update',
1337 egCore.auth.token(), cnList, 1, flags
1338 ).then(function(resp) {
1339 var evt = egCore.evt.parse(resp);
1341 egConfirmDialog.open(
1342 egCore.strings.OVERRIDE_DELETE_ITEMS_FROM_CATALOG_TITLE,
1343 egCore.strings.OVERRIDE_DELETE_ITEMS_FROM_CATALOG_BODY,
1344 {'evt_desc': evt.desc}
1345 ).result.then(function() {
1348 'open-ils.cat.asset.volume.fleshed.batch.update.override',
1349 egCore.auth.token(), cnList, 1,
1350 { events: ['TITLE_LAST_COPY', 'COPY_DELETE_WARNING'] }
1352 holdingsSvcInst.fetchAgain().then(function() {
1353 $scope.holdingsGridDataProvider.refresh();
1358 holdingsSvcInst.fetchAgain().then(function() {
1359 $scope.holdingsGridDataProvider.refresh();
1365 $scope.selectedHoldingsCopyDelete = function () { $scope.selectedHoldingsDelete(false,true) }
1366 $scope.selectedHoldingsVolCopyDelete = function () { $scope.selectedHoldingsDelete(true,true) }
1367 $scope.selectedHoldingsEmptyVolCopyDelete = function () { $scope.selectedHoldingsDelete(true,false) }
1369 spawnHoldingsAdd = function (add_vols,add_copies){
1371 if (!add_vols && add_copies) { // just a copy on existing volumes
1372 angular.forEach(gatherSelectedVolumeIds(), function (v) {
1373 raw.push( {callnumber : v} );
1375 } else if (add_vols) {
1376 if (typeof $scope.holdingsGridControls.selectedItems == "function" &&
1377 $scope.holdingsGridControls.selectedItems().length > 0) {
1378 angular.forEach($scope.holdingsGridControls.selectedItems(),
1381 owner : item.owner_id,
1382 label : ((item.call_number) ? item.call_number.label : null)
1387 owner : egCore.auth.user().ws_ou()
1392 if (raw.length == 0) raw.push({});
1396 'open-ils.actor.anon_cache.set_value',
1397 null, 'edit-these-copies', {
1398 record_id: $scope.record_id,
1401 hide_copies : !add_copies
1403 ).then(function(key) {
1405 var url = egCore.env.basePath + 'cat/volcopy/' + key;
1406 $timeout(function() { $window.open(url, '_blank') });
1408 alert('Could not create anonymous cache key!');
1412 $scope.selectedHoldingsVolCopyAdd = function () { spawnHoldingsAdd(true,true) }
1413 $scope.selectedHoldingsCopyAdd = function () { spawnHoldingsAdd(false,true) }
1414 $scope.selectedHoldingsVolAdd = function () { spawnHoldingsAdd(true,false) }
1416 spawnHoldingsEdit = function (hide_vols,hide_copies){
1419 'open-ils.actor.anon_cache.set_value',
1420 null, 'edit-these-copies', {
1421 record_id: $scope.record_id,
1422 copies: gatherSelectedHoldingsIds(),
1423 raw: gatherSelectedEmptyVolumeIds().map(
1424 function(v){ return { callnumber : v } }
1426 hide_vols : hide_vols,
1427 hide_copies : hide_copies
1429 ).then(function(key) {
1431 var url = egCore.env.basePath + 'cat/volcopy/' + key;
1432 $timeout(function() { $window.open(url, '_blank') });
1434 alert('Could not create anonymous cache key!');
1438 $scope.selectedHoldingsVolCopyEdit = function () { spawnHoldingsEdit(false,false) }
1439 $scope.selectedHoldingsVolEdit = function () { spawnHoldingsEdit(false,true) }
1440 $scope.selectedHoldingsCopyEdit = function () { spawnHoldingsEdit(true,false) }
1442 $scope.selectedHoldingsItemStatus = function (){
1443 var url = egCore.env.basePath + 'cat/item/search/' + gatherSelectedHoldingsIds().join(',')
1444 $timeout(function() { $window.open(url, '_blank') });
1447 $scope.markFromSelectedAsHoldingsTarget = function() {
1448 egCore.hatch.setLocalItem(
1449 'eg.cat.transfer_target_lib',
1450 $scope.holdingsGridControls.selectedItems()[0].owner_id
1452 egCore.hatch.setLocalItem(
1453 'eg.cat.transfer_target_record',
1456 if ($scope.holdingsGridControls.selectedItems()[0].call_number.id) { // cn.id missing when vols are collapsed, or we are on an empty lib
1457 egCore.hatch.setLocalItem(
1458 'eg.cat.transfer_target_vol',
1459 $scope.holdingsGridControls.selectedItems()[0].call_number.id
1462 // clear out the stale value if we're on a lib-only
1463 // or vol-collapsed row
1464 egCore.hatch.removeLocalItem('eg.cat.transfer_target_vol');
1466 ngToast.create(egCore.strings.MARK_HOLDINGS_TARGET);
1469 $scope.selectedHoldingsItemStatusDetail = function (){
1471 gatherSelectedHoldingsIds(),
1473 var url = egCore.env.basePath +
1475 $timeout(function() { $window.open(url, '_blank') });
1480 $scope.transferVolumes = function (){
1481 var target_record = egCore.hatch.getLocalItem('eg.cat.transfer_target_record');
1482 var target_lib = egCore.hatch.getLocalItem('eg.cat.transfer_target_lib');
1484 && (!target_record || ($scope.record_id == target_record) )
1487 var vols_to_move = {};
1489 // we're moving volumes to a different library
1490 var vol_ids = gatherSelectedVolumeIds();
1491 if (vol_ids.length) {
1492 vols_to_move[target_lib] = vol_ids;
1494 // if we're *only* switching libs,
1495 // grab the current record as the target
1496 target_record = target_record || $scope.record_id;
1499 // we're moving volumes to the same library they exist in
1500 // currently, but on a different record
1501 var items = $scope.holdingsGridControls.selectedItems();
1502 angular.forEach(items, function(item) {
1503 if (!(item.call_number.owning_lib in vols_to_move)) {
1504 vols_to_move[item.call_number.owning_lib] = new Array;
1506 vols_to_move[item.call_number.owning_lib].push(item.call_number.id);
1511 angular.forEach(vols_to_move, function(vols, owning_lib) {
1512 promises.push(egCore.net.request(
1514 'open-ils.cat.asset.volume.batch.transfer.override',
1515 egCore.auth.token(), {
1516 docid : target_record,
1522 $q.all(promises).then(function(success) {
1524 ngToast.create(egCore.strings.VOLS_TRANSFERED);
1525 holdingsSvcInst.fetchAgain().then(function() {
1526 $scope.holdingsGridDataProvider.refresh();
1529 alert('Could not transfer volumes!');
1534 // this "transfers" selected copies to a new owning library,
1535 // auto-creating volumes as required
1536 $scope.transferItemsAutoFill = function() {
1537 var target_record = egCore.hatch.getLocalItem('eg.cat.transfer_target_record');
1538 var target_lib = egCore.hatch.getLocalItem('eg.cat.transfer_target_lib');
1540 && (!target_record || ($scope.record_id == target_record) )
1543 var items = $scope.holdingsGridControls.selectedItems();
1544 if (!items.length) {
1548 var vols_to_move = {};
1549 var copies_to_move = {};
1550 angular.forEach(items, function(item) {
1551 var needs_move = false;
1553 && (item.call_number.owning_lib != target_lib)) {
1554 item.call_number.owning_lib = target_lib;
1558 && (item.call_number.record != target_record)) {
1559 item.call_number.record = target_record;
1563 if (item.call_number.id in vols_to_move) {
1564 copies_to_move[item.call_number.id].push(item.id);
1566 vols_to_move[item.call_number.id] = item.call_number;
1567 copies_to_move[item.call_number.id] = new Array;
1568 copies_to_move[item.call_number.id].push(item.id);
1574 angular.forEach(vols_to_move, function(vol) {
1575 promises.push(egCore.net.request(
1577 'open-ils.cat.call_number.find_or_create',
1578 egCore.auth.token(),
1580 vol.record, // may be new
1581 vol.owning_lib, // may be new
1585 ).then(function(resp) {
1586 var evt = egCore.evt.parse(resp);
1588 return egCore.net.request(
1590 'open-ils.cat.transfer_copies_to_volume',
1591 egCore.auth.token(),
1593 copies_to_move[vol.id]
1597 $q.all(promises).then(function() {
1598 ngToast.create(egCore.strings.ITEMS_TRANSFERED);
1599 holdingsSvcInst.fetchAgain().then(function() {
1600 $scope.holdingsGridDataProvider.refresh();
1605 $scope.gridCellHandlers = {};
1606 $scope.gridCellHandlers.copyAlertsEdit = function(id) {
1607 egCirc.manage_copy_alerts([id]).then(function() {
1608 // update grid items?
1612 $scope.transferItems = function (){
1613 var xfer_target = egCore.hatch.getLocalItem('eg.cat.transfer_target_vol');
1616 // we have no specific volume, let's try to fill in the
1618 return $scope.transferItemsAutoFill();
1621 var copy_ids = gatherSelectedHoldingsIds();
1622 if (copy_ids.length > 0) {
1625 'open-ils.cat.transfer_copies_to_volume',
1626 egCore.auth.token(),
1630 function(resp) { // oncomplete
1631 var evt = egCore.evt.parse(resp);
1633 egConfirmDialog.open(
1634 egCore.strings.OVERRIDE_TRANSFER_COPIES_TO_MARKED_VOLUME_TITLE,
1635 egCore.strings.OVERRIDE_TRANSFER_COPIES_TO_MARKED_VOLUME_BODY,
1636 {'evt_desc': evt.desc}
1637 ).result.then(function() {
1640 'open-ils.cat.transfer_copies_to_volume.override',
1641 egCore.auth.token(),
1644 { events: ['TITLE_LAST_COPY', 'COPY_DELETE_WARNING'] }
1645 ).then(function(resp) {
1646 holdingsSvcInst.fetchAgain().then(function() {
1647 $scope.holdingsGridDataProvider.refresh();
1652 ngToast.create(egCore.strings.ITEMS_TRANSFERED);
1653 holdingsSvcInst.fetchAgain().then(function() {
1654 $scope.holdingsGridDataProvider.refresh();
1664 $scope.selectedHoldingsItemStatusTgrEvt = function (){
1666 gatherSelectedHoldingsIds(),
1668 var url = '/eg2/staff/circ/item/event-log/' + cid;
1669 $timeout(function() { $window.open(url, '_blank') });
1674 $scope.selectedHoldingsItemStatusHolds = function (){
1676 gatherSelectedHoldingsIds(),
1678 var url = egCore.env.basePath +
1679 'cat/item/' + cid + '/holds';
1680 $timeout(function() { $window.open(url, '_blank') });
1685 $scope.selectedHoldingsPrintLabels = function() {
1688 'open-ils.actor.anon_cache.set_value',
1689 null, 'print-labels-these-copies', {
1690 copies : gatherSelectedHoldingsIds()
1692 ).then(function(key) {
1694 var url = egCore.env.basePath + 'cat/printlabels/' + key;
1695 $timeout(function() { $window.open(url, '_blank') });
1697 alert('Could not create anonymous cache key!');
1702 $scope.selectedHoldingsDamaged = function () {
1703 var copy_list = gatherSelectedRawCopies();
1704 if (copy_list.length == 0) return;
1706 angular.forEach(copy_list, function(cp) {
1707 egCirc.mark_damaged({
1709 barcode: cp.barcode(),
1710 circ_lib: cp.circ_lib().id()
1711 }).then(function() {
1712 holdingsSvcInst.fetchAgain().then(function() {
1713 $scope.holdingsGridDataProvider.refresh();
1719 $scope.selectedHoldingsDiscard = function () {
1720 var copy_list = gatherSelectedRawCopies();
1721 if (copy_list.length == 0) return;
1722 egCirc.mark_discard(copy_list.map(function(cp) {
1723 return {id: cp.id(), barcode: cp.barcode()};})).then(function() {
1724 holdingsSvcInst.fetchAgain().then(function() {
1725 $scope.holdingsGridDataProvider.refresh();
1730 $scope.selectedHoldingsMissing = function () {
1731 var copy_list = gatherSelectedRawCopies();
1732 if (copy_list.length == 0) return;
1733 egCirc.mark_missing(copy_list.map(function(cp) {
1734 return {id: cp.id(), barcode: cp.barcode()};})).then(function() {
1735 holdingsSvcInst.fetchAgain().then(function() {
1736 $scope.holdingsGridDataProvider.refresh();
1741 $scope.selectedHoldingsCopyAlertsAdd = function() {
1742 egCirc.add_copy_alerts(gatherSelectedHoldingsIds()).then(function() {
1743 // no need to refresh grid
1746 $scope.selectedHoldingsCopyAlertsManage = function() {
1747 egCirc.manage_copy_alerts(gatherSelectedHoldingsIds()).then(function() {
1748 // no need to refresh grid
1752 $scope.attach_to_peer_bib = function() {
1753 var copy_list = gatherSelectedHoldingsIds();
1754 if (copy_list.length == 0) return;
1756 egCore.hatch.getItem('eg.cat.marked_conjoined_record').then(function(target_record) {
1757 if (!target_record) return;
1759 return $uibModal.open({
1760 templateUrl: './cat/catalog/t_conjoined_selector',
1764 ['$scope','$uibModalInstance',
1765 function($scope , $uibModalInstance) {
1766 $scope.update = false;
1768 $scope.peer_type = null;
1769 $scope.peer_type_list = [];
1770 conjoinedSvc.get_peer_types().then(function(list){
1771 $scope.peer_type_list = list;
1774 $scope.ok = function(type) {
1777 angular.forEach(copy_list, function (cp) {
1778 var n = new egCore.idl.bpbcm();
1780 n.peer_record(target_record);
1783 promises.push(egCore.pcrud.create(n));
1786 return $q.all(promises).then(function(){$uibModalInstance.close()});
1789 $scope.cancel = function($event) {
1790 $uibModalInstance.dismiss();
1791 $event.preventDefault();
1799 // ------------------------------------------------------------------
1801 var provider = egGridDataProvider.instance({});
1802 var holds = []; // current list of holds
1804 var hold_grid_load_promise;
1806 $scope.hold_grid_data_provider = provider;
1807 $scope.grid_actions = egHoldGridActions;
1808 $scope.grid_actions.refresh = function () { holds = []; hold_count = 0; provider.refresh() };
1809 $scope.hold_grid_controls = {};
1811 provider.get = function(offset, count) {
1812 if ($scope.record_tab != 'holds') return $q.when();
1814 if (hold_grid_load_promise) {
1815 // Active load in progress.
1816 console.debug('Exiting concurrent hold fetch');
1817 return hold_grid_load_promise;
1820 // see if we have the requested range cached
1821 if (holds[offset]) {
1823 'Serving holds from cache with pickup lib', $scope.pickup_ou.id());
1824 return provider.arrayNotifier(holds, offset, count);
1829 var restrictions = {
1830 is_staff_request : 'true',
1831 fulfillment_time : null,
1833 record_id : $scope.record_id,
1834 pickup_lib : egCore.org.descendants($scope.pickup_ou.id(), true)
1837 var order_by = [{ request_time : null }];
1838 // NOTE: Server sort is disabled for now. See the comment on
1839 // similar code in circ/holds/app.js for details.
1840 if (false && provider.sort && provider.sort.length) {
1842 angular.forEach(provider.sort, function (c) {
1843 if (!angular.isObject(c)) {
1844 if (c.match(/^hold\./)) {
1845 var i = c.replace('hold.','');
1851 var i = Object.keys(c)[0];
1852 var direction = c[i];
1853 if (i.match(/^hold\./)) {
1854 i = i.replace('hold.','');
1856 ob[i] = {dir:direction};
1864 'Fetching holds from network with PU lib', $scope.pickup_ou.id());
1866 egProgressDialog.open({max : 1, value : 0});
1868 hold_grid_load_promise = egHolds.fetch_wide_holds(
1871 ).then(function () {
1872 hold_grid_load_promise = null;
1873 return provider.arrayNotifier(holds, offset, count);
1876 function(hold_data) {
1878 hold_count = hold_data;
1880 egProgressDialog.update({max:hold_count});
1882 egProgressDialog.increment();
1883 var new_item = { id : hold_data.id, hold : hold_data };
1884 new_item.status_string =
1885 egCore.strings['HOLD_STATUS_' + hold_data.hold_status]
1886 || hold_data.hold_status;
1888 holds.push(new_item);
1891 ).finally(function() {
1892 hold_grid_load_promise = null;
1893 egProgressDialog.close();
1896 return hold_grid_load_promise;
1899 $scope.detail_view = function(action, user_data, items) {
1901 $scope.detail_hold_id = h.hold.id;
1905 $scope.list_view = function(items) {
1906 $scope.detail_hold_id = null;
1909 // refresh the list of record holds when the pickup lib is changed.
1910 $scope.pickup_ou = egCore.org.get(egCore.auth.user().ws_ou());
1911 $scope.pickup_ou_changed = function(org) {
1912 if ($scope.pickup_ou && $scope.pickup_ou.id() == org.id()) {
1913 // This fires on every component render, even though the
1914 // value we already have may match. Avoid duplicate lookups.
1918 var promise = hold_grid_load_promise || $q.when();
1920 // Avoid refreshing the grid if it's currently loading data.
1921 promise.finally(function() {
1923 // Previous grid data load complete. Timeout gives the
1924 // grid a chance to mark itself as load-completed, which
1925 // happens after the data load promise is done.
1926 setTimeout(function() {
1927 console.debug('Refreshing holds after PU lib change to ', org.id());
1928 $scope.pickup_ou = org;
1936 function map_prefix_to_subhash (h,pf) {
1938 angular.forEach(Object.keys(h), function (k) {
1939 if (k.startsWith(pf)) {
1940 var nk = k.substr(pf.length);
1947 $scope.print_holds = function() {
1949 angular.forEach(holds, function(item) {
1952 status_string : item.status_string,
1953 patron_first : item.hold.usr_first_given_name,
1954 patron_last : item.hold.usr_family_name,
1955 patron_alias : item.hold.usr_alias,
1956 patron_barcode : item.hold.ucard_barcode,
1957 copy : map_prefix_to_subhash(item.hold,'cp_'),
1958 volume : map_prefix_to_subhash(item.hold,'cn_'),
1959 title : item.hold.title,
1960 author : item.hold.author
1964 egCore.print.print({
1965 context : 'receipt',
1966 template : 'holds_for_bib',
1967 scope : {holds : pholds}
1971 $scope.current_hold_transfer_dest = egCore.hatch.getLocalItem ('eg.circ.hold.title_transfer_target');
1973 $scope.mark_hold_transfer_dest = function() {
1974 $scope.current_hold_transfer_dest = $scope.record_id;
1975 egCore.hatch.setLocalItem(
1976 'eg.circ.hold.title_transfer_target', $scope.record_id);
1977 ngToast.create(egCore.strings.HOLD_TRANSFER_DEST_MARKED);
1980 // UI presents this option as "all holds"
1981 $scope.transfer_holds_to_marked = function() {
1982 var hold_ids = $scope.hold_grid_controls.allItems().map(
1983 function(hold_data) {return hold_data.hold.id});
1984 egHolds.transfer_to_marked_title(hold_ids);
1987 // ------------------------------------------------------------------
1988 // Initialize the selected tab
1990 // we explicitly initialize catalog_url because otherwise Firefox
1991 // ends up setting it to $BASE_URL/{{url}}, which then messes
1992 // things up. See LP#1708951
1993 $scope.catalog_url = '';
1995 function init_cat_url() {
1996 // Set the initial catalog URL. This only happens once.
1997 // The URL is otherwise generated through user navigation.
1998 if ($scope.catalog_url) return;
2000 var url = $location.absUrl().replace(/\/staff.*/, '/opac/advanced');
2002 // A record ID in the path indicates a request for the record-
2004 if ($routeParams.record_id) {
2005 url = url.replace(/\/advanced/, '/record/' + $scope.record_id);
2008 // Jumping directly to the results page by passing a search
2009 // query via the URL. Copy all URL params to the iframe url.
2010 if ($location.path().match(/catalog\/results/)) {
2011 url = url.replace(/\/advanced/, '/results?');
2013 angular.forEach($location.search(), function(val, key) {
2014 if (!first) url += '&';
2016 url += encodeURIComponent(key)
2017 + '=' + encodeURIComponent(val);
2021 // if we're displaying the advanced search form, select
2022 // whatever default pane the user has chosen via workstation
2024 if (url.match(/\/opac\/advanced$/)) {
2025 egCore.hatch.getItem('eg.search.adv_pane').then(function(adv_pane_val){
2027 url += '?pane=' + encodeURIComponent(adv_pane_val);
2030 $scope.catalog_url = url;
2033 $scope.catalog_url = url;
2038 function init_parts_url() {
2039 $scope.parts_url = $location
2043 '/conify/global/biblio/monograph_part?r='+$scope.record_id
2047 $scope.set_record_tab = function(tab) {
2048 $scope.record_tab = tab;
2061 $scope.detail_hold_record_id = $scope.record_id;
2062 // refresh the holds grid
2069 $scope.set_default_record_tab = function() {
2070 egCore.hatch.setLocalItem(
2071 'eg.cat.default_record_tab', $scope.record_tab);
2072 $timeout(function(){$scope.default_tab = $scope.record_tab});
2076 if ($scope.record_id) {
2077 $scope.default_tab = get_default_record_tab();
2078 tab = $routeParams.record_tab || $scope.default_tab;
2081 tab = $routeParams.record_tab || 'catalog';
2083 $scope.set_record_tab(tab);
2087 .controller('AuthorityCtrl',
2088 ['$scope','$routeParams','$location','$window','$q','egCore',
2089 function($scope , $routeParams , $location , $window , $q , egCore) {
2091 // set record ID on page load if available...
2092 $scope.authority_id = $routeParams.authority_id;
2094 if ($routeParams.authority_id) $scope.from_route = true;
2095 else $scope.from_route = false;
2097 $scope.stop_unload = false;
2100 .controller('URLVerifyCtrl',
2101 ['$scope','$location',
2102 function($scope , $location) {
2103 $scope.verifyurls_url = $location.absUrl().replace(/\/staff.*/, '/url_verify/sessions');
2106 .controller('VandelayCtrl',
2107 ['$scope','$location', 'egCore', '$uibModal',
2108 function($scope , $location, egCore, $uibModal) {
2109 $scope.vandelay_url = $location.absUrl().replace(/\/staff\/cat\/catalog\/vandelay/, '/vandelay/vandelay');
2111 $scope.funcs.edit_marc_modal = function(bre, callback){
2112 var marcArgs = { 'marc_xml': bre.marc() };
2113 var vqbibrecId = bre.id();
2115 templateUrl: './cat/catalog/t_edit_marc_modal',
2118 controller: ['$scope', '$uibModalInstance', function($scope, $uibModalInstance) {
2119 $scope.focusMe = true;
2120 $scope.recordId = vqbibrecId;
2121 $scope.args = marcArgs;
2122 $scope.dirty_flag = false;
2123 $scope.ok = function(marg){
2124 $uibModalInstance.close(marg);
2126 $scope.cancel = function(){ $uibModalInstance.dismiss() }
2128 }).result.then(function(res){
2129 var new_xml = res.marc_xml;
2130 egCore.pcrud.retrieve('vqbr', vqbibrecId).then(function(vqbib){
2131 vqbib.marc(new_xml);
2132 egCore.pcrud.update(vqbib).then( function(){ callback(vqbibrecId); });
2138 .controller('ManageAuthoritiesCtrl',
2139 ['$scope','$location',
2140 function($scope , $location) {
2141 $scope.manageauthorities_url = $location.absUrl().replace(/\/staff.*/, '/cat/authority/list');
2144 .controller('BatchEditCtrl',
2145 ['$scope','$location','$routeParams',
2146 function($scope , $location , $routeParams) {
2147 $scope.batchedit_url = $location.absUrl().replace(/\/eg.*/, '/opac/extras/merge_template');
2148 if ($routeParams.container_type) {
2149 switch ($routeParams.container_type) {
2151 $scope.batchedit_url += '?recordSource=b&containerid=' + $routeParams.container_id;
2154 $scope.batchedit_url += '?recordSource=r&recid=' + $routeParams.container_id;
2161 .filter('boolText', function(){
2162 return function (v) {
2167 .factory('conjoinedSvc',
2169 function(egCore , $q) {
2172 items : [], // record search results
2173 index : 0, // search grid index
2180 bpbcm : ['target_copy','peer_type'],
2181 acp : ['call_number'],
2183 bre : ['simple_record']
2185 // avoid fetching the MARC blob by specifying which
2186 // fields on the bre to select. More may be needed.
2187 // note that fleshed fields are explicitly selected.
2188 select : { bre : ['id'] },
2189 order_by : { bpbcm : ['id'] },
2192 // resolved with the last received copy
2193 service.fetch = function(rid) {
2194 if (!rid && !service.rid) return $q.when();
2196 if (rid) service.rid = rid;
2200 return egCore.pcrud.search(
2202 {peer_record : service.rid},
2205 ).then( function(list) { // finished
2206 service.items = list;
2207 return service.items;
2211 // returns a promise resolved with the list of peer bib types
2212 service.get_peer_types = function() {
2214 return $q.when(egCore.env.bpt.list);
2216 return egCore.pcrud.retrieveAll('bpt', null, {atomic : true})
2217 .then(function(list) {
2218 egCore.env.absorbList(list, 'bpt');