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','egCoreMod','egGridMod', 'egMarcMod', 'egUserMod'])
12 .config(function($routeProvider, $locationProvider, $compileProvider) {
13 $locationProvider.html5Mode(true);
14 $compileProvider.aHrefSanitizationWhitelist(/^\s*(https?|blob):/); // grid export
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;
26 egCore.env.loadClasses.push('aous');
30 $routeProvider.when('/cat/catalog/index', {
31 templateUrl: './cat/catalog/t_catalog',
32 controller: 'CatalogCtrl',
36 $routeProvider.when('/cat/catalog/retrieve_by_id', {
37 templateUrl: './cat/catalog/t_retrieve_by_id',
38 controller: 'CatalogRecordRetrieve',
42 $routeProvider.when('/cat/catalog/retrieve_by_tcn', {
43 templateUrl: './cat/catalog/t_retrieve_by_tcn',
44 controller: 'CatalogRecordRetrieve',
48 $routeProvider.when('/cat/catalog/new_bib', {
49 templateUrl: './cat/catalog/t_new_bib',
50 controller: 'NewBibCtrl',
54 // create some catalog page-specific mappings
55 $routeProvider.when('/cat/catalog/record/:record_id', {
56 templateUrl: './cat/catalog/t_catalog',
57 controller: 'CatalogCtrl',
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',
68 $routeProvider.when('/cat/catalog/batchEdit', {
69 templateUrl: './cat/catalog/t_batchedit',
70 controller: 'BatchEditCtrl',
74 $routeProvider.when('/cat/catalog/batchEdit/:container_type/:container_id', {
75 templateUrl: './cat/catalog/t_batchedit',
76 controller: 'BatchEditCtrl',
80 $routeProvider.when('/cat/catalog/vandelay', {
81 templateUrl: './cat/catalog/t_vandelay',
82 controller: 'VandelayCtrl',
86 $routeProvider.when('/cat/catalog/verifyURLs', {
87 templateUrl: './cat/catalog/t_verifyurls',
88 controller: 'URLVerifyCtrl',
92 $routeProvider.when('/cat/catalog/manageAuthorities', {
93 templateUrl: './cat/catalog/t_manageauthorities',
94 controller: 'ManageAuthoritiesCtrl',
98 $routeProvider.when('/cat/catalog/authority/:authority_id/marc_edit', {
99 templateUrl: './cat/catalog/t_authority',
100 controller: 'AuthorityCtrl',
104 $routeProvider.otherwise({redirectTo : '/cat/catalog/index'});
110 .controller('CatalogRecordRetrieve',
111 ['$scope','$routeParams','$location','$q','egCore',
112 function($scope , $routeParams , $location , $q , egCore ) {
114 $scope.focusMe = true;
116 // jump to the patron checkout UI
117 function loadRecord(record_id) {
119 .path('/cat/catalog/record/' + record_id);
122 $scope.submitId = function(args) {
123 $scope.recordNotFound = null;
124 if (!args.record_id) return;
126 // blur so next time it's set to true it will re-apply select()
127 $scope.selectMe = false;
129 return loadRecord(args.record_id);
132 $scope.submitTCN = function(args) {
133 $scope.recordNotFound = null;
134 $scope.moreRecordsFound = null;
135 if (!args.record_tcn) return;
137 // blur so next time it's set to true it will re-apply select()
138 $scope.selectMe = false;
143 'open-ils.search.biblio.tcn',
146 .then(function(resp) { // get_barcodes
148 if (evt = egCore.evt.parse(resp)) {
154 $scope.recordNotFound = args.record_tcn;
155 $scope.selectMe = true;
159 if (resp.count > 1) {
160 $scope.moreRecordsFound = args.record_tcn;
161 $scope.selectMe = true;
165 var record_id = resp.ids[0];
166 return loadRecord(record_id);
172 .controller('NewBibCtrl',
173 ['$scope','$routeParams','$location','$window','$q','egCore',
174 'egGridDataProvider','egHoldGridActions','$timeout','holdingsSvc',
175 function($scope , $routeParams , $location , $window , $q , egCore) {
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;
186 'open-ils.cat.marc_template.types.retrieve'
187 ).then(function(resp) {
188 angular.forEach(resp, function(name) {
189 $scope.template_list.push(name);
191 $scope.template_list.sort();
193 egCore.hatch.getItem('cat.default_bib_marc_template').then(function(template) {
194 $scope.template_name = template;
197 $scope.loadTemplate = function() {
198 if ($scope.template_name) {
201 'open-ils.cat.biblio.marc_template.retrieve',
203 ).then(function(template) {
204 $scope.marc_template = template;
205 $scope.have_template = true;
210 $scope.setDefaultTemplate = function() {
211 var hatch_key = "cat.default_bib_marc_template";
212 if ($scope.template_name) {
213 egCore.hatch.setItem(hatch_key, $scope.template_name);
215 egCore.hatch.removeItem(hatch_key);
219 $scope.$watch('new_bib_id', function(newVal, oldVal) {
221 $location.path('/cat/catalog/record/' + $scope.new_bib_id);
228 .controller('CatalogCtrl',
229 ['$scope','$routeParams','$location','$window','$q','egCore','egHolds','egCirc','egConfirmDialog',
230 'egGridDataProvider','egHoldGridActions','$timeout','$modal','holdingsSvc','egUser','conjoinedSvc',
231 function($scope , $routeParams , $location , $window , $q , egCore , egHolds , egCirc, egConfirmDialog,
232 egGridDataProvider , egHoldGridActions , $timeout , $modal , holdingsSvc , egUser , conjoinedSvc) {
234 // set record ID on page load if available...
235 $scope.record_id = $routeParams.record_id;
237 if ($routeParams.record_id) $scope.from_route = true;
238 else $scope.from_route = false;
240 // will hold a ref to the opac iframe
241 $scope.opac_iframe = null;
242 $scope.parts_iframe = null;
244 $scope.in_opac_call = false;
245 $scope.opac_call = function (opac_frame_function, force_opac_tab) {
246 if ($scope.opac_iframe) {
247 if (force_opac_tab) $scope.record_tab = 'catalog';
248 $scope.in_opac_call = true;
249 $scope.opac_iframe.dom.contentWindow[opac_frame_function]();
253 $scope.add_to_record_bucket = function() {
254 var recId = $scope.record_id;
256 templateUrl: './cat/catalog/t_add_to_bucket',
260 ['$scope','$modalInstance',
261 function($scope , $modalInstance) {
263 $scope.bucket_id = 0;
264 $scope.newBucketName = '';
265 $scope.allBuckets = [];
268 'open-ils.actor.container.retrieve_by_class.authoritative',
269 egCore.auth.token(), egCore.auth.user().id(),
270 'biblio', 'staff_client'
271 ).then(function(buckets) { $scope.allBuckets = buckets; });
273 $scope.add_to_bucket = function() {
274 var item = new egCore.idl.cbrebi();
275 item.bucket($scope.bucket_id);
276 item.target_biblio_record_entry(recId);
279 'open-ils.actor.container.item.create',
280 egCore.auth.token(), 'biblio', item
281 ).then(function(resp) {
282 $modalInstance.close();
286 $scope.add_to_new_bucket = function() {
287 var bucket = new egCore.idl.cbreb();
288 bucket.owner(egCore.auth.user().id());
289 bucket.name($scope.newBucketName);
290 bucket.description('');
291 bucket.btype('staff_client');
295 'open-ils.actor.container.create',
296 egCore.auth.token(), 'biblio', bucket
297 ).then(function(bucket) {
298 $scope.bucket_id = bucket;
299 $scope.add_to_bucket();
303 $scope.cancel = function() {
304 $modalInstance.dismiss();
310 $scope.stop_unload = false;
311 $scope.$watch('stop_unload',
312 function(newVal, oldVal) {
313 if (newVal && newVal != oldVal && $scope.opac_iframe) {
314 $($scope.opac_iframe.dom.contentWindow).on('beforeunload', function(){
315 return 'There is unsaved data in this record.'
318 if ($scope.opac_iframe)
319 $($scope.opac_iframe.dom.contentWindow).off('beforeunload');
324 // Set the "last bib" cookie, if we have that
325 if ($scope.record_id)
326 egCore.hatch.setLocalItem("eg.cat.last_record_retrieved", $scope.record_id);
328 // also set it when the iframe changes to a new record
329 $scope.handle_page = function(url) {
331 if (!url || url == 'about:blank') {
332 // nothing loaded. If we already have a record ID, leave it.
336 var match = url.match(/\/+opac\/+record\/+(\d+)/);
338 $scope.record_id = match[1];
339 egCore.hatch.setLocalItem("eg.cat.last_record_retrieved", $scope.record_id);
340 $scope.holdings_record_id_changed($scope.record_id);
341 conjoinedSvc.fetch($scope.record_id).then(function(){
342 $scope.conjoinedGridDataProvider.refresh();
346 delete $scope.record_id;
347 $scope.from_route = false;
350 // child scope is executing this function, so our digest doesn't fire ... thus,
353 if (!$scope.in_opac_call) {
354 if ($scope.record_id) {
355 $scope.default_tab = egCore.hatch.getLocalItem( 'eg.cat.default_record_tab' );
356 tab = $routeParams.record_tab || $scope.default_tab || 'catalog';
358 tab = $routeParams.record_tab || 'catalog';
360 $scope.set_record_tab(tab);
362 $scope.in_opac_call = false;
366 // xulG catalog handlers
367 $scope.handlers = { }
369 // ------------------------------------------------------------------
372 $scope.conjoinedGridControls = {};
373 $scope.conjoinedGridDataProvider = egGridDataProvider.instance({
374 get : function(offset, count) {
375 return this.arrayNotifier(conjoinedSvc.items, offset, count);
379 $scope.changeConjoinedType = function () {
380 var peers = egCore.idl.Clone($scope.conjoinedGridControls.selectedItems());
381 angular.forEach(peers, function (p) {
382 p.target_copy(p.target_copy().id());
383 p.peer_type(p.peer_type().id());
386 var conjoinedGridDataProviderRef = $scope.conjoinedGridDataProvider;
389 templateUrl: './cat/catalog/t_conjoined_selector',
392 ['$scope','$modalInstance',
393 function($scope , $modalInstance) {
394 $scope.update = true;
396 $scope.peer_type = null;
397 $scope.peer_type_list = [];
398 conjoinedSvc.get_peer_types().then(function(list){
399 $scope.peer_type_list = list;
402 $scope.ok = function(type) {
405 angular.forEach(peers, function (p) {
408 promises.push(egCore.pcrud.update(p));
411 return $q.all(promises)
412 .then(function(){$modalInstance.close()})
413 .then(function(){return conjoinedSvc.fetch()})
414 .then(function(){conjoinedGridDataProviderRef.refresh()});
417 $scope.cancel = function($event) {
418 $modalInstance.dismiss();
419 $event.preventDefault();
426 $scope.refreshConjoined = function () {
427 conjoinedSvc.fetch($scope.record_id)
428 .then(function(){$scope.conjoinedGridDataProvider.refresh();});
431 $scope.deleteSelectedConjoined = function () {
432 var peers = $scope.conjoinedGridControls.selectedItems();
434 if (peers.length > 0) {
435 egConfirmDialog.open(
436 egCore.strings.CONFIRM_DELETE_PEERS,
437 egCore.strings.CONFIRM_DELETE_PEERS_MESSAGE,
438 {peers : peers.length}
439 ).result.then(function() {
440 angular.forEach(peers, function (p) {
444 egCore.pcrud.remove(peers).then(function() {
445 return conjoinedSvc.fetch();
447 $scope.conjoinedGridDataProvider.refresh();
452 if ($scope.record_id)
453 conjoinedSvc.fetch($scope.record_id);
455 // ------------------------------------------------------------------
458 $scope.holdingsGridControls = {};
459 $scope.holdingsGridDataProvider = egGridDataProvider.instance({
460 get : function(offset, count) {
461 return this.arrayNotifier(holdingsSvc.copies, offset, count);
465 $scope.add_copies_to_bucket = function() {
466 var copy_list = gatherSelectedHoldingsIds();
467 if (copy_list.length == 0) return;
470 templateUrl: './cat/catalog/t_add_to_bucket',
474 ['$scope','$modalInstance',
475 function($scope , $modalInstance) {
477 $scope.bucket_id = 0;
478 $scope.newBucketName = '';
479 $scope.allBuckets = [];
483 'open-ils.actor.container.retrieve_by_class.authoritative',
484 egCore.auth.token(), egCore.auth.user().id(),
485 'copy', 'staff_client'
486 ).then(function(buckets) { $scope.allBuckets = buckets; });
488 $scope.add_to_bucket = function() {
490 angular.forEach(copy_list, function (cp) {
491 var item = new egCore.idl.ccbi()
492 item.bucket($scope.bucket_id);
493 item.target_copy(cp);
497 'open-ils.actor.container.item.create',
498 egCore.auth.token(), 'copy', item
502 return $q.all(promises).then(function() {
503 $modalInstance.close();
508 $scope.add_to_new_bucket = function() {
509 var bucket = new egCore.idl.ccb();
510 bucket.owner(egCore.auth.user().id());
511 bucket.name($scope.newBucketName);
512 bucket.description('');
513 bucket.btype('staff_client');
515 return egCore.net.request(
517 'open-ils.actor.container.create',
518 egCore.auth.token(), 'copy', bucket
519 ).then(function(bucket) {
520 $scope.bucket_id = bucket;
521 $scope.add_to_bucket();
525 $scope.cancel = function() {
526 $modalInstance.dismiss();
532 $scope.requestItems = function() {
533 var copy_list = gatherSelectedHoldingsIds();
534 if (copy_list.length == 0) return;
537 templateUrl: './cat/catalog/t_request_items',
540 ['$scope','$modalInstance',
541 function($scope , $modalInstance) {
543 $scope.first_user_fetch = true;
547 copy_list : copy_list,
548 pickup_lib: egCore.org.get(egCore.auth.user().ws_ou()),
549 user : egCore.auth.user().id()
552 egUser.get( $scope.hold_data.user ).then(function(u) {
554 $scope.barcode = u.card().barcode();
555 $scope.user_name = egUser.format_name(u);
556 $scope.hold_data.user = u.id();
559 $scope.user_name = '';
561 $scope.$watch('barcode', function (n) {
562 if (!$scope.first_user_fetch) {
563 egUser.getByBarcode(n).then(function(u) {
565 $scope.user_name = egUser.format_name(u);
566 $scope.hold_data.user = u.id();
569 $scope.user_name = '';
570 delete $scope.hold_data.user;
573 $scope.first_user_fetch = false;
576 $scope.ok = function(h) {
579 hold_type : h.hold_type,
580 pickup_lib: h.pickup_lib.id(),
586 'open-ils.circ.holds.test_and_create.batch.override',
587 egCore.auth.token(), args, h.copy_list
590 $modalInstance.close();
593 $scope.cancel = function($event) {
594 $modalInstance.dismiss();
595 $event.preventDefault();
601 $scope.replaceBarcodes = function() {
602 var copy_list = gatherSelectedRawCopies();
603 if (copy_list.length == 0) return;
605 var holdingsGridDataProviderRef = $scope.holdingsGridDataProvider;
607 angular.forEach(copy_list, function (cp) {
609 templateUrl: './cat/share/t_replace_barcode',
612 ['$scope','$modalInstance',
613 function($scope , $modalInstance) {
614 $scope.isModal = true;
615 $scope.focusBarcode = false;
616 $scope.focusBarcode2 = true;
617 $scope.barcode1 = cp.barcode();
619 $scope.updateBarcode = function() {
620 $scope.copyNotFound = false;
621 $scope.updateOK = false;
623 egCore.pcrud.search('acp',
624 {deleted : 'f', barcode : $scope.barcode1})
625 .then(function(copy) {
628 $scope.focusBarcode = true;
629 $scope.copyNotFound = true;
633 $scope.copyId = copy.id();
634 copy.barcode($scope.barcode2);
636 egCore.pcrud.update(copy).then(function(stat) {
637 $scope.updateOK = stat;
638 $scope.focusBarcode = true;
639 holdingsSvc.fetchAgain().then(function (){
640 holdingsGridDataProviderRef.refresh();
645 $modalInstance.close();
648 $scope.cancel = function($event) {
649 $modalInstance.dismiss();
650 $event.preventDefault();
658 // refresh the list of holdings when the record_id is changed.
659 $scope.holdings_record_id_changed = function(id) {
660 if ($scope.record_id != id) $scope.record_id = id;
661 console.log('record id changed to ' + id + ', loading new holdings');
663 rid : $scope.record_id,
664 org : $scope.holdings_ou,
665 copy: $scope.holdings_show_copies,
666 vol : $scope.holdings_show_vols,
667 empty: $scope.holdings_show_empty
669 $scope.holdingsGridDataProvider.refresh();
673 // refresh the list of holdings when the filter lib is changed.
674 $scope.holdings_ou = egCore.org.get(egCore.auth.user().ws_ou());
675 $scope.holdings_ou_changed = function(org) {
676 $scope.holdings_ou = org;
678 rid : $scope.record_id,
679 org : $scope.holdings_ou,
680 copy: $scope.holdings_show_copies,
681 vol : $scope.holdings_show_vols,
682 empty: $scope.holdings_show_empty
684 $scope.holdingsGridDataProvider.refresh();
688 $scope.holdings_cb_changed = function(cb,newVal,norefresh) {
690 egCore.hatch.setItem('cat.' + cb, newVal);
691 if (!norefresh) holdingsSvc.fetch({
692 rid : $scope.record_id,
693 org : $scope.holdings_ou,
694 copy: $scope.holdings_show_copies,
695 vol : $scope.holdings_show_vols,
696 empty: $scope.holdings_show_empty
698 $scope.holdingsGridDataProvider.refresh();
702 egCore.hatch.getItem('cat.holdings_show_vols').then(function(x){
703 if (typeof x == 'undefined') x = true;
704 $scope.holdings_cb_changed('holdings_show_vols',x,true);
705 $('#holdings_show_vols').prop('checked', x);
707 egCore.hatch.getItem('cat.holdings_show_copies').then(function(x){
708 if (typeof x == 'undefined') x = true;
709 $scope.holdings_cb_changed('holdings_show_copies',x,true);
710 $('#holdings_show_copies').prop('checked', x);
712 egCore.hatch.getItem('cat.holdings_show_empty').then(function(x){
713 if (typeof x == 'undefined') x = true;
714 $scope.holdings_cb_changed('holdings_show_empty',x);
715 $('#holdings_show_empty').prop('checked', x);
720 $scope.vols_not_shown = function () {
721 return !$scope.holdings_show_vols;
724 $scope.copies_not_shown = function () {
725 return !$scope.holdings_show_copies;
728 $scope.holdings_checkbox_handler = function (item) {
729 $scope.holdings_cb_changed(item.checkbox,item.checked);
732 function gatherSelectedHoldingsIds () {
735 $scope.holdingsGridControls.selectedItems(),
736 function (item) { cp_id_list = cp_id_list.concat(item.id_list) }
741 function gatherSelectedRawCopies () {
744 $scope.holdingsGridControls.selectedItems(),
745 function (item) { if (item.raw) cp_list = cp_list.concat(item.raw) }
750 function gatherSelectedVolumeIds () {
753 $scope.holdingsGridControls.selectedItems(),
755 if (cn_id_list.indexOf(item.call_number.id) == -1)
756 cn_id_list.push(item.call_number.id)
762 $scope.selectedHoldingsDelete = function (vols, copies) {
765 var perCnCopies = {};
771 $scope.holdingsGridControls.selectedItems(),
773 if (vols && item.raw_call_number) {
774 cnHash[item.call_number.id] = egCore.idl.Clone(item.raw_call_number);
775 cnHash[item.call_number.id].isdeleted(1);
778 angular.forEach(egCore.idl.Clone(item.raw), function (cp) {
781 var cn_id = cp.call_number().id();
782 if (!cnHash[cn_id]) {
783 cnHash[cn_id] = cp.call_number();
784 perCnCopies[cn_id] = [cp];
786 perCnCopies[cn_id].push(cp);
788 cp.call_number(cn_id); // prevent loops in JSON-ification
795 angular.forEach(perCnCopies, function (v, k) {
797 cnHash[k].isdeleted(1);
804 angular.forEach(cnHash, function (v, k) {
808 if (cnList.length == 0) return;
810 egConfirmDialog.open(
811 egCore.strings.CONFIRM_DELETE_COPIES_VOLUMES,
812 egCore.strings.CONFIRM_DELETE_COPIES_VOLUMES_MESSAGE,
813 {copies : cp_count, volumes : cn_count}
814 ).result.then(function() {
817 'open-ils.cat.asset.volume.fleshed.batch.update.override',
818 egCore.auth.token(), cnList, 1, {}
819 ).then(function(update_count) {
820 $scope.holdingsGridDataProvider.refresh();
824 $scope.selectedHoldingsCopyDelete = function () { $scope.selectedHoldingsDelete(false,true) }
825 $scope.selectedHoldingsVolCopyDelete = function () { $scope.selectedHoldingsDelete(true,true) }
826 $scope.selectedHoldingsEmptyVolCopyDelete = function () { $scope.selectedHoldingsDelete(true,false) }
828 spawnHoldingsAdd = function (vols,copies){
830 if (copies) { // just a copy on existing volumes
831 angular.forEach(gatherSelectedVolumeIds(), function (v) {
832 raw.push( {callnumber : v} );
836 $scope.holdingsGridControls.selectedItems(),
838 raw.push({owner : item.owner_id});
843 if (raw.length == 0) raw.push({});
847 'open-ils.actor.anon_cache.set_value',
848 null, 'edit-these-copies', {
849 record_id: $scope.record_id,
854 ).then(function(key) {
856 var url = egCore.env.basePath + 'cat/volcopy/' + key;
857 $timeout(function() { $window.open(url, '_blank') });
859 alert('Could not create anonymous cache key!');
863 $scope.selectedHoldingsVolCopyAdd = function () { spawnHoldingsAdd(true,false) }
864 $scope.selectedHoldingsCopyAdd = function () { spawnHoldingsAdd(false,true) }
866 spawnHoldingsEdit = function (hide_vols,hide_copies){
869 'open-ils.actor.anon_cache.set_value',
870 null, 'edit-these-copies', {
871 record_id: $scope.record_id,
872 copies: gatherSelectedHoldingsIds(),
873 hide_vols : hide_vols,
874 hide_copies : hide_copies
876 ).then(function(key) {
878 var url = egCore.env.basePath + 'cat/volcopy/' + key;
879 $timeout(function() { $window.open(url, '_blank') });
881 alert('Could not create anonymous cache key!');
885 $scope.selectedHoldingsVolCopyEdit = function () { spawnHoldingsEdit(false,false) }
886 $scope.selectedHoldingsVolEdit = function () { spawnHoldingsEdit(false,true) }
887 $scope.selectedHoldingsCopyEdit = function () { spawnHoldingsEdit(true,false) }
889 $scope.selectedHoldingsItemStatus = function (){
890 var url = egCore.env.basePath + 'cat/item/search/' + gatherSelectedHoldingsIds().join(',')
891 $timeout(function() { $window.open(url, '_blank') });
894 $scope.markVolAsItemTarget = function() {
895 if ($scope.holdingsGridControls.selectedItems()[0].call_number.id) { // cn.id missing when vols are collapsed
896 egCore.hatch.setLocalItem(
897 'eg.cat.item_transfer_target',
898 $scope.holdingsGridControls.selectedItems()[0].call_number.id
903 $scope.markLibAsVolTarget = function() {
904 egCore.hatch.setLocalItem(
905 'eg.cat.volume_transfer_target',
906 $scope.holdingsGridControls.selectedItems()[0].owner_id
910 $scope.selectedHoldingsItemStatusDetail = function (){
912 gatherSelectedHoldingsIds(),
914 var url = egCore.env.basePath +
916 $timeout(function() { $window.open(url, '_blank') });
921 $scope.transferVolumes = function (){
922 var xfer_target = egCore.hatch.getLocalItem('eg.cat.volume_transfer_target');
927 'open-ils.open-ils.cat.asset.volume.batch.transfer.override',
928 egCore.auth.token(), {
929 docid : $scope.record_id,
931 volumes : gatherSelectedVolumeIds()
933 ).then(function(success) {
935 holdingsSvc.fetchAgain().then(function() {
936 $scope.holdingsGridDataProvider.refresh();
939 alert('Could not transfer volumes!');
946 $scope.transferItems = function (){
947 var xfer_target = egCore.hatch.getLocalItem('eg.cat.item_transfer_target');
948 var copy_ids = gatherSelectedHoldingsIds();
949 if (xfer_target && copy_ids.length > 0) {
952 'open-ils.cat.transfer_copies_to_volume',
957 function(resp) { // oncomplete
958 var evt = egCore.evt.parse(resp);
960 egConfirmDialog.open(
961 egCore.strings.OVERRIDE_TRANSFER_COPIES_TO_MARKED_VOLUME_TITLE,
962 egCore.strings.OVERRIDE_TRANSFER_COPIES_TO_MARKED_VOLUME_BODY,
963 {'evt_desc': evt.desc}
964 ).result.then(function() {
967 'open-ils.cat.transfer_copies_to_volume.override',
971 { events: ['TITLE_LAST_COPY', 'COPY_DELETE_WARNING'] }
972 ).then(function(resp) {
973 holdingsSvc.fetchAgain().then(function() {
974 $scope.holdingsGridDataProvider.refresh();
979 holdingsSvc.fetchAgain().then(function() {
980 $scope.holdingsGridDataProvider.refresh();
990 $scope.selectedHoldingsItemStatusTgrEvt = function (){
992 gatherSelectedHoldingsIds(),
994 var url = egCore.env.basePath +
995 'cat/item/' + cid + '/triggered_events';
996 $timeout(function() { $window.open(url, '_blank') });
1001 $scope.selectedHoldingsItemStatusHolds = function (){
1003 gatherSelectedHoldingsIds(),
1005 var url = egCore.env.basePath +
1006 'cat/item/' + cid + '/holds';
1007 $timeout(function() { $window.open(url, '_blank') });
1012 $scope.selectedHoldingsDamaged = function () {
1013 egCirc.mark_damaged(gatherSelectedHoldingsIds()).then(function() {
1014 holdingsSvc.fetchAgain().then(function() {
1015 $scope.holdingsGridDataProvider.refresh();
1020 $scope.selectedHoldingsMissing = function () {
1021 egCirc.mark_missing(gatherSelectedHoldingsIds()).then(function() {
1022 holdingsSvc.fetchAgain().then(function() {
1023 $scope.holdingsGridDataProvider.refresh();
1028 $scope.attach_to_peer_bib = function() {
1029 var copy_list = gatherSelectedHoldingsIds();
1030 if (copy_list.length == 0) return;
1032 egCore.hatch.getItem('eg.cat.marked_conjoined_record').then(function(target_record) {
1033 if (!target_record) return;
1035 return $modal.open({
1036 templateUrl: './cat/catalog/t_conjoined_selector',
1039 ['$scope','$modalInstance',
1040 function($scope , $modalInstance) {
1041 $scope.update = false;
1043 $scope.peer_type = null;
1044 $scope.peer_type_list = [];
1045 conjoinedSvc.get_peer_types().then(function(list){
1046 $scope.peer_type_list = list;
1049 $scope.ok = function(type) {
1052 angular.forEach(copy_list, function (cp) {
1053 var n = new egCore.idl.bpbcm();
1055 n.peer_record(target_record);
1058 promises.push(egCore.pcrud.create(n));
1061 return $q.all(promises).then(function(){$modalInstance.close()});
1064 $scope.cancel = function($event) {
1065 $modalInstance.dismiss();
1066 $event.preventDefault();
1074 // ------------------------------------------------------------------
1076 var provider = egGridDataProvider.instance({});
1077 $scope.hold_grid_data_provider = provider;
1078 $scope.grid_actions = egHoldGridActions;
1079 $scope.grid_actions.refresh = function () { provider.refresh() };
1080 $scope.hold_grid_controls = {};
1082 var hold_ids = []; // current list of holds
1083 function fetchHolds(offset, count) {
1084 var ids = hold_ids.slice(offset, offset + count);
1085 return egHolds.fetch_holds(ids).then(null, null,
1086 function(hold_data) {
1092 provider.get = function(offset, count) {
1093 if ($scope.record_tab != 'holds') return $q.when();
1094 var deferred = $q.defer();
1095 hold_ids = []; // no caching ATM
1100 'open-ils.circ.holds.retrieve_all_from_title',
1101 egCore.auth.token(), $scope.record_id,
1102 {pickup_lib : egCore.org.descendants($scope.pickup_ou.id(), true)}
1104 function(hold_data) {
1105 angular.forEach(hold_data, function(list, type) {
1106 hold_ids = hold_ids.concat(list);
1108 fetchHolds(offset, count).then(
1109 deferred.resolve, null, deferred.notify);
1113 return deferred.promise;
1116 $scope.detail_view = function(action, user_data, items) {
1118 $scope.detail_hold_id = h.hold.id();
1122 $scope.list_view = function(items) {
1123 $scope.detail_hold_id = null;
1126 // refresh the list of record holds when the pickup lib is changed.
1127 $scope.pickup_ou = egCore.org.get(egCore.auth.user().ws_ou());
1128 $scope.pickup_ou_changed = function(org) {
1129 $scope.pickup_ou = org;
1133 $scope.print_holds = function() {
1135 angular.forEach($scope.hold_grid_controls.allItems(), function(item) {
1137 hold : egCore.idl.toHash(item.hold),
1138 patron_last : item.patron_last,
1139 patron_alias : item.patron_alias,
1140 patron_barcode : item.patron_barcode,
1141 copy : egCore.idl.toHash(item.copy),
1142 volume : egCore.idl.toHash(item.volume),
1143 title : item.mvr.title(),
1144 author : item.mvr.author()
1148 egCore.print.print({
1149 context : 'receipt',
1150 template : 'holds_for_bib',
1151 scope : {holds : holds}
1155 $scope.mark_hold_transfer_dest = function() {
1156 egCore.hatch.setLocalItem(
1157 'eg.circ.hold.title_transfer_target', $scope.record_id);
1160 // UI presents this option as "all holds"
1161 $scope.transfer_holds_to_marked = function() {
1162 var hold_ids = $scope.hold_grid_controls.allItems().map(
1163 function(hold_data) {return hold_data.hold.id()});
1164 egHolds.transfer_to_marked_title(hold_ids);
1167 // ------------------------------------------------------------------
1168 // Initialize the selected tab
1170 function init_cat_url() {
1171 // Set the initial catalog URL. This only happens once.
1172 // The URL is otherwise generated through user navigation.
1173 if ($scope.catalog_url) return;
1175 var url = $location.absUrl().replace(/\/staff.*/, '/opac/advanced');
1177 // A record ID in the path indicates a request for the record-
1179 if ($routeParams.record_id) {
1180 url = url.replace(/advanced/, '/record/' + $scope.record_id);
1183 $scope.catalog_url = url;
1186 function init_parts_url() {
1187 $scope.parts_url = $location
1191 '/conify/global/biblio/monograph_part?r='+$scope.record_id
1195 $scope.set_record_tab = function(tab) {
1196 $scope.record_tab = tab;
1209 $scope.detail_hold_record_id = $scope.record_id;
1210 // refresh the holds grid
1216 $scope.set_default_record_tab = function() {
1217 egCore.hatch.setLocalItem(
1218 'eg.cat.default_record_tab', $scope.record_tab);
1219 $timeout(function(){$scope.default_tab = $scope.record_tab});
1223 if ($scope.record_id) {
1224 $scope.default_tab = egCore.hatch.getLocalItem( 'eg.cat.default_record_tab' );
1225 tab = $routeParams.record_tab || $scope.default_tab || 'catalog';
1228 tab = $routeParams.record_tab || 'catalog';
1230 $scope.set_record_tab(tab);
1234 .controller('AuthorityCtrl',
1235 ['$scope','$routeParams','$location','$window','$q','egCore',
1236 function($scope , $routeParams , $location , $window , $q , egCore) {
1238 // set record ID on page load if available...
1239 $scope.authority_id = $routeParams.authority_id;
1241 if ($routeParams.authority_id) $scope.from_route = true;
1242 else $scope.from_route = false;
1244 $scope.stop_unload = false;
1247 .controller('URLVerifyCtrl',
1248 ['$scope','$location',
1249 function($scope , $location) {
1250 $scope.verifyurls_url = $location.absUrl().replace(/\/staff.*/, '/url_verify/sessions');
1253 .controller('VandelayCtrl',
1254 ['$scope','$location',
1255 function($scope , $location) {
1256 $scope.vandelay_url = $location.absUrl().replace(/\/staff.*/, '/vandelay/vandelay');
1259 .controller('ManageAuthoritiesCtrl',
1260 ['$scope','$location',
1261 function($scope , $location) {
1262 $scope.manageauthorities_url = $location.absUrl().replace(/\/staff.*/, '/cat/authority/list');
1265 .controller('BatchEditCtrl',
1266 ['$scope','$location','$routeParams',
1267 function($scope , $location , $routeParams) {
1268 $scope.batchedit_url = $location.absUrl().replace(/\/eg.*/, '/opac/extras/merge_template');
1269 if ($routeParams.container_type) {
1270 switch ($routeParams.container_type) {
1272 $scope.batchedit_url += '?recordSource=b&containerid=' + $routeParams.container_id;
1275 $scope.batchedit_url += '?recordSource=r&recid=' + $routeParams.container_id;
1282 .filter('boolText', function(){
1283 return function (v) {
1288 .factory('holdingsSvc',
1290 function(egCore , $q) {
1294 copies : [], // record search results
1295 index : 0, // search grid index
1303 acp : ['status','location'],
1304 acn : ['prefix','suffix','copies']
1308 service.fetchAgain = function() {
1309 return service.fetch({
1314 empty: service.empty
1318 // resolved with the last received copy
1319 service.fetch = function(opts) {
1320 if (service.ongoing) {
1321 console.log('Skipping fetch, ongoing = true');
1327 var copy = opts.copy;
1329 var empty = opts.empty;
1331 if (!rid) return $q.when();
1332 if (!org) return $q.when();
1334 service.ongoing = true;
1338 service.copy = opts.copy;
1339 service.vol = opts.vol;
1340 service.empty = opts.empty;
1342 service.copies = [];
1345 var org_list = egCore.org.descendants(org.id(), true);
1346 console.log('Holdings fetch with: rid='+rid+' org='+org_list+' copy='+copy+' vol='+vol+' empty='+empty);
1348 return egCore.pcrud.search(
1350 {record : rid, owning_lib : org_list, deleted : 'f'},
1353 function() { // finished
1354 service.copies = service.copies.sort(
1356 function compare_array (x, y, i) {
1357 if (x[i] && y[i]) { // both have values
1358 if (x[i] == y[i]) { // need to look deeper
1359 return compare_array(x, y, ++i);
1362 if (x[i] < y[i]) { // x is first
1364 } else if (x[i] > y[i]) { // y is first
1368 } else { // no orgs to compare ...
1369 if (x[i]) return -1;
1375 var owner_order = compare_array(a.owner_list, b.owner_list, 0);
1377 // now compare on CN label
1378 if (a.call_number.label < b.call_number.label) return -1;
1379 if (a.call_number.label > b.call_number.label) return 1;
1382 if (a.copy_number < b.copy_number) return -1;
1383 if (a.copy_number > b.copy_number) return 1;
1386 if (a.barcode < b.barcode) return -1;
1387 if (a.barcode > b.barcode) return 1;
1393 // create a label using just the unique part of the owner list
1395 var prev_owner_list;
1396 angular.forEach(service.copies, function (cp) {
1397 if (!prev_owner_list) {
1398 cp.owner_label = cp.owner_list.join(' ... ');
1400 var current_owner_list = cp.owner_list.slice();
1401 while (current_owner_list[1] && prev_owner_list[1] && current_owner_list[0] == prev_owner_list[0]) {
1402 current_owner_list.shift();
1403 prev_owner_list.shift();
1405 cp.owner_label = current_owner_list.join(' ... ');
1409 prev_owner_list = cp.owner_list.slice();
1412 var new_list = service.copies;
1413 if (!copy || !vol) { // collapse copy rows, supply a count instead
1418 var current_blob = { copy_count : 0 };
1419 angular.forEach(new_list, function (cp) {
1421 prev_key = cp.owner_list.join('') + cp.call_number.label;
1422 if (cp.barcode) current_blob.copy_count = 1;
1423 current_blob.index = index++;
1424 current_blob.id_list = cp.id_list;
1425 if (cp.raw) current_blob.raw = cp.raw;
1426 current_blob.call_number = cp.call_number;
1427 current_blob.owner_list = cp.owner_list;
1428 current_blob.owner_label = cp.owner_label;
1429 current_blob.owner_id = cp.owner_id;
1431 var current_key = cp.owner_list.join('') + cp.call_number.label;
1432 if (prev_key == current_key) { // collapse into current_blob
1433 current_blob.copy_count++;
1434 current_blob.id_list = current_blob.id_list.concat(cp.id_list);
1435 current_blob.raw = current_blob.raw.concat(cp.raw);
1437 current_blob.barcode = current_blob.copy_count;
1438 cp_list.push(current_blob);
1439 prev_key = current_key;
1440 current_blob = { copy_count : 0 };
1441 if (cp.barcode) current_blob.copy_count = 1;
1442 current_blob.index = index++;
1443 current_blob.id_list = cp.id_list;
1444 if (cp.raw) current_blob.raw = cp.raw;
1445 current_blob.owner_label = cp.owner_label;
1446 current_blob.owner_id = cp.owner_id;
1447 current_blob.call_number = cp.call_number;
1448 current_blob.owner_list = cp.owner_list;
1453 current_blob.barcode = current_blob.copy_count;
1454 cp_list.push(current_blob);
1457 if (!vol) { // do the same for vol rows
1462 current_blob = { copy_count : 0 };
1463 angular.forEach(cp_list, function (cp) {
1465 prev_key = cp.owner_list.join('');
1466 current_blob.index = index++;
1467 current_blob.id_list = cp.id_list;
1468 if (cp.raw) current_blob.raw = cp.raw;
1469 current_blob.cn_count = 1;
1470 current_blob.copy_count = cp.copy_count;
1471 current_blob.owner_list = cp.owner_list;
1472 current_blob.owner_label = cp.owner_label;
1473 current_blob.owner_id = cp.owner_id;
1475 var current_key = cp.owner_list.join('');
1476 if (prev_key == current_key) { // collapse into current_blob
1477 current_blob.cn_count++;
1478 current_blob.copy_count += cp.copy_count;
1479 current_blob.id_list = current_blob.id_list.concat(cp.id_list);
1480 if (cp.raw) current_blob.raw = current_blob.raw.concat(cp.raw);
1482 current_blob.barcode = current_blob.copy_count;
1483 current_blob.call_number = { label : current_blob.cn_count };
1484 cn_list.push(current_blob);
1485 prev_key = current_key;
1486 current_blob = { copy_count : 0 };
1487 current_blob.index = index++;
1488 current_blob.id_list = cp.id_list;
1489 if (cp.raw) current_blob.raw = cp.raw;
1490 current_blob.owner_label = cp.owner_label;
1491 current_blob.owner_id = cp.owner_id;
1492 current_blob.cn_count = 1;
1493 current_blob.copy_count = cp.copy_count;
1494 current_blob.owner_list = cp.owner_list;
1499 current_blob.barcode = current_blob.copy_count;
1500 current_blob.call_number = { label : current_blob.cn_count };
1501 cn_list.push(current_blob);
1507 service.copies = new_list;
1508 service.ongoing = false;
1513 // notify reads the stream of copies, one at a time.
1516 var copies = cn.copies().filter(function(cp){ return cp.deleted() == 'f' });
1519 angular.forEach(copies, function (cp) {
1523 var owner_id = cn.owning_lib();
1524 var owner = egCore.org.get(owner_id);
1526 var owner_name_list = [];
1527 while (owner.parent_ou()) { // we're going to skip the top of the tree...
1528 owner_name_list.unshift(owner.name());
1529 owner = egCore.org.get(owner.parent_ou());
1534 angular.forEach(copies, function (cp) {
1535 var flat_cp = egCore.idl.toHash(cp);
1536 flat_cp.owner_id = owner_id;
1537 flat_cp.owner_list = owner_name_list;
1538 flat_cp.id_list = [flat_cp.id];
1543 service.copies = service.copies.concat(flat);
1545 service.copies.push({
1546 owner_id : owner_id,
1547 owner_list : owner_name_list,
1548 call_number: egCore.idl.toHash(cn),
1561 .factory('conjoinedSvc',
1563 function(egCore , $q) {
1566 items : [], // record search results
1567 index : 0, // search grid index
1574 bpbcm : ['target_copy','peer_type'],
1575 acp : ['call_number'],
1577 bre : ['simple_record']
1579 // avoid fetching the MARC blob by specifying which
1580 // fields on the bre to select. More may be needed.
1581 // note that fleshed fields are explicitly selected.
1582 select : { bre : ['id'] },
1583 order_by : { bpbcm : ['id'] },
1586 // resolved with the last received copy
1587 service.fetch = function(rid) {
1588 if (!rid && !service.rid) return $q.when();
1590 if (rid) service.rid = rid;
1594 return egCore.pcrud.search(
1596 {peer_record : service.rid},
1599 ).then( function(list) { // finished
1600 service.items = list;
1601 return service.items;
1605 // returns a promise resolved with the list of peer bib types
1606 service.get_peer_types = function() {
1608 return $q.when(egCore.env.bpt.list);
1610 return egCore.pcrud.retrieveAll('bpt', null, {atomic : true})
1611 .then(function(list) {
1612 egCore.env.absorbList(list, 'bpt');