5 angular.module('egItemStatus',
6 ['ngRoute', 'ui.bootstrap', 'egCoreMod', 'egUiMod', 'egGridMod', 'egUserMod'])
8 .filter('boolText', function(){
14 .config(function($routeProvider, $locationProvider, $compileProvider) {
15 $locationProvider.html5Mode(true);
16 $compileProvider.aHrefSanitizationWhitelist(/^\s*(https?|mailto|blob):/); // grid export
18 var resolver = {delay : function(egStartup) {return egStartup.go()}};
20 // search page shows the list view by default
21 $routeProvider.when('/cat/item/search', {
22 templateUrl: './cat/item/t_list',
23 controller: 'ListCtrl',
27 // search page shows the list view by default
28 $routeProvider.when('/cat/item/search/:idList', {
29 templateUrl: './cat/item/t_list',
30 controller: 'ListCtrl',
34 $routeProvider.when('/cat/item/:id', {
35 templateUrl: './cat/item/t_view',
36 controller: 'ViewCtrl',
40 $routeProvider.when('/cat/item/:id/:tab', {
41 templateUrl: './cat/item/t_view',
42 controller: 'ViewCtrl',
46 // default page / bucket view
47 $routeProvider.otherwise({redirectTo : '/cat/item/search'});
51 * Search bar along the top of the page.
52 * Parent scope for list and detail views
54 .controller('SearchCtrl',
55 ['$scope','$q','$window','$location','$timeout','egCore','egNet','egGridDataProvider','egItem', 'egCirc',
56 function($scope , $q , $window , $location , $timeout , egCore , egNet , egGridDataProvider , itemSvc , egCirc) {
57 $scope.args = {}; // search args
59 // sub-scopes (search / detail-view) apply their version
60 // of retrieval function to $scope.context.search
61 // and display toggling via $scope.context.toggleDisplay
66 $scope.toggleView = function($event) {
67 $scope.context.toggleDisplay();
68 $event.preventDefault(); // avoid form submission
71 // The functions that follow in this controller are never called
72 // when the List View is active, only the Detail View.
74 // In this context, we're only ever dealing with 1 item, so
75 // we can simply refresh the page. These various itemSvc
76 // functions used to live in the ListCtrl, but they're now
77 // shared between SearchCtrl (for Actions for the Detail View)
78 // and ListCtrl (Actions in the egGrid)
79 itemSvc.add_barcode_to_list = function(b) {
80 //console.log('SearchCtrl: add_barcode_to_list',b);
81 // timeout so audible can happen upon checkin
82 $timeout(function() { location.href = location.href; }, 1000);
85 $scope.add_copies_to_bucket = function() {
86 itemSvc.add_copies_to_bucket([$scope.args.copyId]);
89 $scope.add_records_to_bucket = function() {
90 itemSvc.add_records_to_bucket([$scope.args.recordId], 'biblio');
93 $scope.show_in_catalog = function() {
94 window.open('/eg2/staff/catalog/record/' + $scope.args.recordId, '_blank');
97 $scope.print_labels = function() {
98 itemSvc.print_spine_labels([$scope.args.copyId]);
102 $scope.make_copies_bookable = function() {
103 itemSvc.make_copies_bookable([{
104 id : $scope.args.copyId,
105 'call_number.record.id' : $scope.args.recordId
109 $scope.book_copies_now = function() {
110 itemSvc.book_copies_now([$scope.args.copyBarcode]);
113 $scope.findAcquisition = function() {
116 $scope.openAcquisitionLineItem([$scope.args.copyId]);
119 $scope.openAcquisitionLineItem = function (cp_list) {
120 var hasResults = false;
123 angular.forEach(cp_list, function (copyId) {
127 'open-ils.acq.lineitem.retrieve.by_copy_id',
130 ).then(function (acqData) {
133 acqData = egCore.idl.toHash(acqData);
134 var url = '/eg/acq/po/view/' + acqData.purchase_order + '/' + acqData.id;
135 $timeout(function () { $window.open(url, '_blank') });
143 $q.all(promises).then(function () {
144 !hasResults ? alert('There is no corresponding purchase order for this item.') : false;
148 $scope.manage_reservations = function() {
149 itemSvc.manage_reservations([$scope.args.copyBarcode]);
152 $scope.requestItems = function() {
153 itemSvc.requestItems([$scope.args.copyId],[$scope.args.recordId]);
156 $scope.update_inventory = function() {
157 itemSvc.updateInventory([$scope.args.copyId], null)
158 .then(function(res) {
159 $timeout(function() { location.href = location.href; }, 1000);
163 $scope.show_triggered_events = function() {
164 $location.path('/cat/item/' + $scope.args.copyId + '/triggered_events');
167 $scope.show_item_holds = function() {
168 $location.path('/cat/item/' + $scope.args.copyId + '/holds');
171 $scope.show_record_holds = function() {
172 window.open('/eg2/staff/catalog/record/' + $scope.args.recordId + '/holds', '_blank');
175 $scope.add_item_alerts = function() {
176 egCirc.add_copy_alerts([$scope.args.copyId]);
179 $scope.manage_item_alerts = function() {
180 egCirc.manage_copy_alerts([$scope.args.copyId]);
184 $scope.attach_to_peer_bib = function() {
185 itemSvc.attach_to_peer_bib([{
186 id : $scope.args.copyId,
187 barcode : $scope.args.copyBarcode
191 $scope.selectedHoldingsCopyDelete = function () {
192 itemSvc.selectedHoldingsCopyDelete([{
193 id : $scope.args.copyId,
194 barcode : $scope.args.copyBarcode
198 $scope.checkin = function () {
200 id : $scope.args.copyId,
201 barcode : $scope.args.copyBarcode
205 $scope.renew = function () {
207 id : $scope.args.copyId,
208 barcode : $scope.args.copyBarcode
212 $scope.cancel_transit = function () {
213 itemSvc.cancel_transit([{
214 id : $scope.args.copyId,
215 barcode : $scope.args.copyBarcode
219 $scope.selectedHoldingsDamaged = function () {
220 itemSvc.selectedHoldingsDamaged([{
221 id : $scope.args.copyId,
222 barcode : $scope.args.copyBarcode,
227 $scope.selectedHoldingsDiscard = function () {
228 itemSvc.selectedHoldingsDiscard([{
229 id : $scope.args.copyId,
230 barcode : $scope.args.barcode
234 $scope.selectedHoldingsMissing = function () {
235 itemSvc.selectedHoldingsMissing([{
236 id : $scope.args.copyId,
237 barcode : $scope.args.barcode
241 $scope.selectedHoldingsVolCopyAdd = function () {
242 itemSvc.spawnHoldingsAdd([{
243 id : $scope.args.copyId,
244 'call_number.owning_lib' : $scope.args.cnOwningLib,
245 'call_number.record.id' : $scope.args.recordId,
246 barcode : $scope.args.copyBarcode
249 $scope.selectedHoldingsCopyAdd = function () {
250 itemSvc.spawnHoldingsAdd([{
251 id : $scope.args.copyId,
252 'call_number.id' : $scope.args.cnId,
253 'call_number.owning_lib' : $scope.args.cnOwningLib,
254 'call_number.record.id' : $scope.args.recordId,
255 barcode : $scope.args.copyBarcode
259 $scope.selectedHoldingsVolCopyEdit = function () {
260 itemSvc.spawnHoldingsEdit([{
261 id : $scope.args.copyId,
262 'call_number.id' : $scope.args.cnId,
263 'call_number.owning_lib' : $scope.args.cnOwningLib,
264 'call_number.record.id' : $scope.args.recordId,
265 barcode : $scope.args.copyBarcode
268 $scope.selectedHoldingsVolEdit = function () {
269 itemSvc.spawnHoldingsEdit([{
270 id : $scope.args.copyId,
271 'call_number.id' : $scope.args.cnId,
272 'call_number.owning_lib' : $scope.args.cnOwningLib,
273 'call_number.record.id' : $scope.args.recordId,
274 barcode : $scope.args.copyBarcode
277 $scope.selectedHoldingsCopyEdit = function () {
278 itemSvc.spawnHoldingsEdit([{
279 id : $scope.args.copyId,
280 'call_number.id' : $scope.args.cnId,
281 'call_number.owning_lib' : $scope.args.cnOwningLib,
282 'call_number.record.id' : $scope.args.recordId,
283 barcode : $scope.args.copyBarcode
287 $scope.replaceBarcodes = function() {
288 itemSvc.replaceBarcodes([{
289 id : $scope.args.copyId,
290 barcode : $scope.args.copyBarcode
294 $scope.changeItemOwningLib = function() {
295 itemSvc.changeItemOwningLib([{
296 id : $scope.args.copyId,
297 'call_number.id' : $scope.args.cnId,
298 'call_number.owning_lib' : $scope.args.cnOwningLib,
299 'call_number.record.id' : $scope.args.recordId,
300 'call_number.label' : $scope.args.cnLabel,
301 'call_number.label_class' : $scope.args.cnLabelClass,
302 'call_number.prefix.id' : $scope.args.cnPrefixId,
303 'call_number.suffix.id' : $scope.args.cnSuffixId,
304 barcode : $scope.args.copyBarcode
308 $scope.transferItems = function (){
309 itemSvc.transferItems([{
310 id : $scope.args.copyId,
311 barcode : $scope.args.copyBarcode
318 * List view - grid stuff
320 .controller('ListCtrl',
321 ['$scope','$q','$routeParams','$location','$timeout','$window','egCore',
322 'egGridDataProvider','egItem','egUser','$uibModal','egCirc','egConfirmDialog',
323 'egProgressDialog', 'ngToast',
324 // function($scope , $q , $routeParams , $location , $timeout , $window , egCore ,
325 // egGridDataProvider , itemSvc , egUser , $uibModal , egCirc , egConfirmDialog,
326 // egProgressDialog, ngToast) {
327 function($scope , $q , $routeParams , $location , $timeout , $window , egCore , egGridDataProvider , itemSvc , egUser , $uibModal , egCirc , egConfirmDialog,
328 egProgressDialog, ngToast) {
330 var cp_list = $routeParams.idList;
332 copyId = cp_list.split(',');
335 var modified_items = new Set();
337 $scope.context.page = 'list';
340 var provider = egGridDataProvider.instance();
341 provider.get = function(offset, count) {
345 $scope.gridDataProvider = egGridDataProvider.instance({
346 get : function(offset, count) {
347 //return provider.arrayNotifier(itemSvc.copies, offset, count);
348 return this.arrayNotifier(itemSvc.copies, offset, count);
352 // If a copy was just displayed in the detail view, ensure it's
353 // focused in the list view.
354 var selected = false;
355 var copyGrid = $scope.gridControls = {
356 itemRetrieved : function(item) {
357 if (selected || !itemSvc.copy) return;
358 if (itemSvc.copy.id() == item.id) {
359 copyGrid.selectItems([item.index]);
365 $scope.$watch('barcodesFromFile', function(newVal, oldVal) {
366 $scope.context.itemsNotFound = [];
367 $scope.context.fileDoneLoading = false;
368 $scope.context.numBarcodesInFile = 0;
369 if (newVal && newVal != oldVal) {
370 $scope.args.barcode = '';
373 angular.forEach(newVal.split(/\r?\n/), function(line) {
374 //remove all whitespace and commas
375 line = line.replace(/[\s,]+/g,'');
377 //Or remove leading/trailing whitespace
378 //line = line.replace(/(^[\s,]+|[\s,]+$/g,'');
384 // Serialize copy retrieval since there may be many, many copies.
385 function fetch_next_copy() {
386 var barcode = barcodes.pop();
387 egProgressDialog.increment();
389 if (barcode == undefined) { // All done here.
390 egProgressDialog.close();
392 if(itemSvc.copies[0]){ // Were any copies actually retrieved
393 copyGrid.selectItems([itemSvc.copies[0].index]);
395 $scope.context.fileDoneLoading = true;
399 itemSvc.fetch(barcode).then(function(item) {
401 $scope.context.itemsNotFound.push(barcode);
407 if (barcodes.length) {
408 $scope.context.numBarcodesInFile = barcodes.length;
409 egProgressDialog.open({value: 0, max: barcodes.length});
415 $scope.context.search = function(args) {
416 if (!args.barcode) return;
417 $scope.context.itemNotFound = false;
419 //check to see if there are multiple barcodes in CSV format
421 //split on commas and clean up barcodes
422 angular.forEach(args.barcode.split(/,/), function(line) {
423 //remove all whitespace and commas
424 line = line.replace(/[\s,]+/g,'');
426 //Or remove leading/trailing whitespace
427 //line = line.replace(/(^[\s,]+|[\s,]+$/g,'');
433 if(barcodes.length > 1){
434 //convert to newline seperated list and send to barcodesFromFile processor
435 $scope.barcodesFromFile = barcodes.join('\n');
436 //console.log('Barcodes: ',barcodes);
440 itemSvc.fetch(args.barcode).then(function(res) {
443 copyGrid.selectItems([res.index]);
444 $scope.args.barcode = '';
446 $scope.context.itemNotFound = true;
447 egCore.audio.play('warning.item_status.itemNotFound');
449 $scope.context.selectBarcode = true;
454 var add_barcode_to_list = function (b) {
455 //console.log('listCtrl: add_barcode_to_list',b);
456 $scope.context.search({barcode:b});
458 itemSvc.add_barcode_to_list = add_barcode_to_list;
460 $scope.context.toggleDisplay = function() {
461 var item = copyGrid.selectedItems()[0];
463 $location.path('/cat/item/' + item.id);
466 $scope.context.show_triggered_events = function() {
467 var item = copyGrid.selectedItems()[0];
469 $location.path('/cat/item/' + item.id + '/triggered_events');
472 function gatherSelectedRecordIds () {
475 copyGrid.selectedItems(),
477 if (rid_list.indexOf(item['call_number.record.id']) == -1)
478 rid_list.push(item['call_number.record.id'])
484 function gatherSelectedVolumeIds (rid) {
487 copyGrid.selectedItems(),
489 if (rid && item['call_number.record.id'] != rid) return;
490 if (cn_id_list.indexOf(item['call_number.id']) == -1)
491 cn_id_list.push(item['call_number.id'])
497 function gatherSelectedHoldingsIds (rid) {
500 copyGrid.selectedItems(),
502 if (rid && item['call_number.record.id'] != rid) return;
503 cp_id_list.push(item.id)
509 function gatherSelectedHoldingsRecords() {
510 var record_id_list = [];
512 copyGrid.selectedItems(),
514 record_id_list.push(item['call_number.record.id']);
517 return record_id_list;
520 $scope.refreshGridData = function() {
521 var chain = $q.when();
522 var all_items = itemSvc.copies.map(function(item) {
525 angular.forEach(all_items.reverse(), function(i) {
526 itemSvc.copies.shift();
527 chain = chain.then(function() {
528 return itemSvc.fetch(null, i);
531 return chain.then(function() {
537 $scope.add_copies_to_bucket = function() {
538 var copy_list = gatherSelectedHoldingsIds();
539 itemSvc.add_copies_to_bucket(copy_list);
542 $scope.add_records_to_bucket = function() {
543 var record_list = gatherSelectedHoldingsRecords();
544 itemSvc.add_copies_to_bucket(record_list, 'biblio');
547 $scope.locateAcquisition = function() {
548 if (gatherSelectedHoldingsIds) {
549 var cp_list = gatherSelectedHoldingsIds();
551 if (cp_list.length > 0) {
552 $scope.openAcquisitionLineItem(cp_list);
558 $scope.update_inventory = function() {
559 var copy_list = gatherSelectedHoldingsIds();
560 itemSvc.updateInventory(copy_list, $scope.gridControls.allItems()).then(function(res) {
562 $scope.gridControls.allItems(res);
563 ngToast.create(egCore.strings.SUCCESS_UPDATE_INVENTORY);
565 ngToast.warning(egCore.strings.FAIL_UPDATE_INVENTORY);
570 $scope.need_one_selected = function() {
571 var items = $scope.gridControls.selectedItems();
572 if (items.length == 1) return false;
576 $scope.make_copies_bookable = function() {
577 itemSvc.make_copies_bookable(copyGrid.selectedItems());
580 $scope.book_copies_now = function() {
581 var item = copyGrid.selectedItems()[0];
583 itemSvc.book_copies_now(item.barcode);
586 $scope.manage_reservations = function() {
587 var item = copyGrid.selectedItems()[0];
589 itemSvc.manage_reservations(item.barcode);
592 $scope.requestItems = function() {
593 var copy_list = gatherSelectedHoldingsIds();
594 var record_list = gatherSelectedRecordIds();
595 itemSvc.requestItems(copy_list,record_list);
598 $scope.replaceBarcodes = function() {
599 itemSvc.replaceBarcodes(copyGrid.selectedItems());
602 $scope.attach_to_peer_bib = function() {
603 itemSvc.attach_to_peer_bib(copyGrid.selectedItems());
606 $scope.selectedHoldingsCopyDelete = function () {
607 itemSvc.selectedHoldingsCopyDelete(copyGrid.selectedItems());
610 $scope.selectedHoldingsItemStatusTgrEvt= function() {
611 var item = copyGrid.selectedItems()[0];
613 $location.path('/cat/item/' + item.id + '/triggered_events');
616 $scope.selectedHoldingsItemStatusHolds= function() {
617 var item = copyGrid.selectedItems()[0];
619 $location.path('/cat/item/' + item.id + '/holds');
622 $scope.cancel_transit = function () {
623 itemSvc.cancel_transit(copyGrid.selectedItems());
626 $scope.selectedHoldingsDamaged = function () {
627 itemSvc.selectedHoldingsDamaged(copyGrid.selectedItems());
630 $scope.selectedHoldingsDiscard = function () {
631 itemSvc.selectedHoldingsDiscard(copyGrid.selectedItems());
634 $scope.selectedHoldingsMissing = function () {
635 itemSvc.selectedHoldingsMissing(copyGrid.selectedItems());
638 $scope.checkin = function () {
639 itemSvc.checkin(copyGrid.selectedItems());
642 $scope.renew = function () {
643 itemSvc.renew(copyGrid.selectedItems());
646 $scope.selectedHoldingsVolCopyAdd = function () {
647 itemSvc.spawnHoldingsAdd(copyGrid.selectedItems(),true,false);
649 $scope.selectedHoldingsCopyAdd = function () {
650 itemSvc.spawnHoldingsAdd(copyGrid.selectedItems(),false,true);
653 $scope.selectedHoldingsCopyAlertsAdd = function(items) {
655 angular.forEach(items, function(item) {
656 if (item.id) copy_ids.push(item.id);
658 egCirc.add_copy_alerts(copy_ids).then(function() {
659 // update grid items?
663 $scope.selectedHoldingsCopyAlertsEdit = function(items) {
665 angular.forEach(items, function(item) {
666 if (item.id) copy_ids.push(item.id);
668 egCirc.manage_copy_alerts(copy_ids).then(function() {
669 // update grid items?
673 $scope.gridCellHandlers = {};
674 $scope.gridCellHandlers.copyAlertsEdit = function(id) {
675 egCirc.manage_copy_alerts([id]).then(function() {
676 // update grid items?
680 $scope.showBibHolds = function () {
681 angular.forEach(gatherSelectedRecordIds(), function (r) {
682 var url = '/eg2/staff/catalog/record/' + r + '/holds';
683 $timeout(function() { $window.open(url, '_blank') });
687 $scope.selectedHoldingsVolCopyEdit = function () {
688 itemSvc.spawnHoldingsEdit(copyGrid.selectedItems(),false,false);
690 $scope.selectedHoldingsVolEdit = function () {
691 itemSvc.spawnHoldingsEdit(copyGrid.selectedItems(),false,true);
693 $scope.selectedHoldingsCopyEdit = function () {
694 itemSvc.spawnHoldingsEdit(copyGrid.selectedItems(),true,false);
697 $scope.changeItemOwningLib = function() {
698 itemSvc.changeItemOwningLib(copyGrid.selectedItems());
701 $scope.transferItems = function (){
702 itemSvc.transferItems(copyGrid.selectedItems());
705 $scope.print_labels = function() {
708 'open-ils.actor.anon_cache.set_value',
709 null, 'print-labels-these-copies', {
710 copies : gatherSelectedHoldingsIds()
712 ).then(function(key) {
714 var url = egCore.env.basePath + 'cat/printlabels/' + key;
715 $timeout(function() { $window.open(url, '_blank') });
717 alert('Could not create anonymous cache key!');
722 $scope.print_list = function() {
723 var print_data = { copies : copyGrid.allItems() };
725 if (print_data.copies.length == 0) return $q.when();
727 return egCore.print.print({
728 template : 'item_status',
733 $scope.show_in_catalog = function(){
734 itemSvc.show_in_catalog(copyGrid.selectedItems());
737 if (copyId.length > 0) {
739 angular.forEach(copyId, function (c) {
740 fetch_list.push(itemSvc.fetch(null,c));
743 return $q.all(fetch_list).then(function (res) { copyGrid.refresh(); });
746 $scope.statusIconColumn = {
748 template: function(item) {
750 if (modified_items.has(item['id'])) {
751 icon = '<span class="glyphicon glyphicon-floppy-saved"' +
752 'title="' + egCore.strings.ITEM_SUCCESSFULLY_MODIFIED + '" ' +
753 'aria-label="' + egCore.strings.ITEM_SUCCESSFULLY_MODIFIED + '">' +
760 if (typeof BroadcastChannel != 'undefined') {
761 var holdings_bChannel = new BroadcastChannel("eg.holdings.update");
762 holdings_bChannel.onmessage = function(e) {
763 angular.forEach(e.data.copies, function(i) {
764 modified_items.add(i);
766 ngToast.create(egCore.strings.ITEMS_SUCCESSFULLY_MODIFIED);
767 $scope.refreshGridData();
769 $scope.$on('$destroy', function() {
770 holdings_bChannel.close();
777 * Detail view -- shows one copy
779 .controller('ViewCtrl',
780 ['$scope','$q','egGridDataProvider','$location','$routeParams','$timeout','$window','egCore','egItem','egBilling','egCirc',
781 function($scope , $q , egGridDataProvider , $location , $routeParams , $timeout , $window , egCore , itemSvc , egBilling , egCirc) {
782 var copyId = $routeParams.id;
783 $scope.args.copyId = copyId;
784 $scope.tab = $routeParams.tab || 'summary';
785 $scope.context.page = 'detail';
786 $scope.summaryRecord = null;
787 $scope.courseModulesOptIn = fetchCourseOptIn();
788 $scope.has_course_perms = fetchCoursePerms();
790 if ($scope.tab == 'edit') {
791 $scope.tab = 'summary';
795 // use the cached record info
797 $scope.copy_alert_count = itemSvc.copy.copy_alerts().filter(function(aca) {
798 return !aca.ack_time();
800 $scope.recordId = itemSvc.copy.call_number().record().id();
801 $scope.args.recordId = $scope.recordId;
802 $scope.args.cnId = itemSvc.copy.call_number().id();
803 $scope.args.cnOwningLib = itemSvc.copy.call_number().owning_lib();
804 $scope.args.cnLabel = itemSvc.copy.call_number().label();
805 $scope.args.cnLabelClass = itemSvc.copy.call_number().label_class();
806 $scope.args.cnPrefixId = itemSvc.copy.call_number().prefix().id();
807 $scope.args.cnSuffixId = itemSvc.copy.call_number().suffix().id();
808 $scope.args.copyBarcode = itemSvc.copy.barcode();
811 function loadCopy(barcode) {
812 $scope.context.itemNotFound = false;
814 // Avoid re-fetching the same copy while jumping tabs.
815 // In addition to being quicker, this helps to avoid flickering
816 // of the top panel which is always visible in the detail view.
818 // 'barcode' represents the loading of a new item - refetch it
819 // regardless of whether it matches the current item.
820 if (!barcode && itemSvc.copy && itemSvc.copy.id() == copyId) {
821 $scope.copy = itemSvc.copy;
822 if (itemSvc.latest_inventory && itemSvc.latest_inventory.copy() == copyId) {
823 $scope.latest_inventory = itemSvc.latest_inventory;
825 $scope.copy_alert_count = itemSvc.copy.copy_alerts().filter(function(aca) {
826 return !aca.ack_time();
828 $scope.recordId = itemSvc.copy.call_number().record().id();
829 $scope.args.recordId = $scope.recordId;
830 $scope.args.cnId = itemSvc.copy.call_number().id();
831 $scope.args.cnOwningLib = itemSvc.copy.call_number().owning_lib();
832 $scope.args.cnLabel = itemSvc.copy.call_number().label();
833 $scope.args.cnLabelClass = itemSvc.copy.call_number().label_class();
834 $scope.args.cnPrefixId = itemSvc.copy.call_number().prefix().id();
835 $scope.args.cnSuffixId = itemSvc.copy.call_number().suffix().id();
836 $scope.args.copyBarcode = itemSvc.copy.barcode();
843 var deferred = $q.defer();
844 itemSvc.fetch(barcode, copyId, true).then(function(res) {
845 $scope.context.selectBarcode = true;
849 $scope.context.itemNotFound = true;
850 egCore.audio.play('warning.item_status.itemNotFound');
851 deferred.reject(); // avoid propagation of data fetch calls
857 if (res.latest_inventory) itemSvc.latest_inventory = res.latest_inventory;
861 $scope.latest_inventory = res.latest_inventory;
862 $scope.copy_alert_count = copy.copy_alerts().filter(function(aca) {
863 return !aca.ack_time();
865 console.debug($scope.copy_alert_count);
866 $scope.recordId = copy.call_number().record().id();
867 $scope.args.recordId = $scope.recordId;
868 $scope.args.cnId = itemSvc.copy.call_number().id();
869 $scope.args.cnOwningLib = itemSvc.copy.call_number().owning_lib();
870 $scope.args.cnLabel = itemSvc.copy.call_number().label();
871 $scope.args.cnLabelClass = itemSvc.copy.call_number().label_class();
872 $scope.args.cnPrefixId = itemSvc.copy.call_number().prefix().id();
873 $scope.args.cnSuffixId = itemSvc.copy.call_number().suffix().id();
874 $scope.args.copyBarcode = copy.barcode();
875 $scope.args.barcode = '';
877 // locally flesh org units
878 copy.circ_lib(egCore.org.get(copy.circ_lib()));
879 copy.call_number().owning_lib(
880 egCore.org.get(copy.call_number().owning_lib()));
882 var r = copy.call_number().record();
883 if (r.owner()) r.owner(egCore.org.get(r.owner()));
885 // make boolean for auto-magic true/false display
887 ['ref','opac_visible','holdable','circulate'],
888 function(field) { copy[field](Boolean(copy[field]() == 't')) }
891 // finally, if this is a different copy, redirect.
892 // Note that we flesh first since the copy we just
893 // fetched will be used after the redirect.
894 if (copyId && copyId != copy.id()) {
895 // if a new barcode is scanned in the detail view,
896 // update the url to match the ID of the new copy
897 $location.path('/cat/item/' + copy.id() + '/' + $scope.tab);
898 deferred.reject(); // avoid propagation of data fetch calls
906 return deferred.promise;
909 // load the two most recent circulations in /circs tab
910 function loadCurrentCirc() {
912 delete $scope.circ_summary;
913 delete $scope.prev_circ_summary;
914 delete $scope.prev_circ_usr;
918 itemSvc.copy.call_number().id() == -1 ?
919 itemSvc.copy.circ_lib().id() :
920 itemSvc.copy.call_number().owning_lib().id();
922 // since a user can still view patron checkout history here, check perms
923 egCore.perm.hasPermAt('VIEW_COPY_CHECKOUT_HISTORY', true)
924 .then(function(orgIds){
925 if(orgIds.indexOf(copy_org) == -1){
926 console.warn('User is not allowed to view circ history!');
930 return fetchMaxCircHistory();
932 .then(function(maxHistCount){
934 if (!maxHistCount) $scope.isMaxCircHistoryZero = true;
936 egCore.pcrud.search('aacs',
937 {target_copy : copyId},
943 'checkin_workstation',
946 'recurring_fine_rule'
950 order_by : {aacs : 'xact_start desc'},
954 ).then(null, null, function(circ) {
957 if (!circ) return $q.when();
959 // load the chain for this circ
962 'open-ils.circ.renewal_chain.retrieve_by_circ.summary',
963 egCore.auth.token(), $scope.circ.id()
964 ).then(function(summary) {
965 $scope.circ_summary = summary;
968 if (maxHistCount <= 1) return;
970 // load the chain for the previous circ, plus the user
973 'open-ils.circ.prev_renewal_chain.retrieve_by_circ.summary',
974 egCore.auth.token(), $scope.circ.id()
976 ).then(null, null, function(summary) {
977 $scope.prev_circ_summary = summary.summary;
979 if (summary.usr) { // aged circs have no 'usr'.
980 egCore.pcrud.retrieve('au', summary.usr,
981 {flesh : 1, flesh_fields : {au : ['card']}})
983 .then(function(user) { $scope.prev_circ_usr = user });
991 function fetchMaxCircHistory() {
992 if (maxHistory) return $q.when(maxHistory);
993 return egCore.org.settings(
994 'circ.item_checkout_history.max')
995 .then(function(set) {
996 maxHistory = set['circ.item_checkout_history.max'] || 4;
997 return Number(maxHistory);
1001 // Check for Course Modules Opt-In to enable Course Info tab
1002 function fetchCourseOptIn() {
1003 return egCore.org.settings(
1004 'circ.course_materials_opt_in'
1005 ).then(function(set) {
1006 $scope.courseModulesOptIn = set['circ.course_materials_opt_in'];
1008 return $scope.courseModulesOptIn;
1012 function fetchCoursePerms() {
1013 return egCore.perm.hasPermAt('MANAGE RESERVES', true).then(function(orgIds) {
1014 if(orgIds.indexOf(egCore.auth.user().ws_ou()) != -1){
1015 $scope.has_course_perms = true;
1017 return $scope.has_course_perms;
1022 $scope.addBilling = function(circ) {
1023 egBilling.showBillDialog({
1024 xact_id : circ.id(),
1029 $scope.retrieveAllPatrons = function() {
1030 var users = new Set();
1031 angular.forEach($scope.circ_list.map(function(circ) { return circ.usr(); }),function(usr) {
1032 // aged circs have no 'usr'.
1033 if (usr) users.add(usr);
1035 users.forEach(function(usr) {
1036 $timeout(function() {
1037 var url = $location.absUrl().replace(
1039 '/circ/patron/' + usr.id() + '/checkout');
1040 $window.open(url, '_blank')
1045 // load data for /circ_list tab
1046 function loadCircHistory() {
1047 $scope.circ_list = [];
1050 itemSvc.copy.call_number().id() == -1 ?
1051 itemSvc.copy.circ_lib().id() :
1052 itemSvc.copy.call_number().owning_lib().id();
1054 // there is an extra layer of permissibility over circ
1056 egCore.perm.hasPermAt('VIEW_COPY_CHECKOUT_HISTORY', true)
1057 .then(function(orgIds) {
1059 if (orgIds.indexOf(copy_org) == -1) {
1060 console.log('User is not allowed to view circ history');
1064 return fetchMaxCircHistory();
1066 }).then(function(maxHistCount) {
1068 if(!maxHistCount) $scope.isMaxCircHistoryZero = true;
1070 egCore.pcrud.search('aacs',
1071 {target_copy : copyId},
1077 'checkin_workstation',
1078 'recurring_fine_rule'
1082 order_by : {aacs : 'xact_start desc'},
1083 // fetch at least one to see if copy ever circulated
1084 limit : $scope.isMaxCircHistoryZero ? 1 : maxHistCount
1087 ).then(null, null, function(circ) {
1091 // flesh circ_lib locally
1092 circ.circ_lib(egCore.org.get(circ.circ_lib()));
1093 circ.checkin_lib(egCore.org.get(circ.checkin_lib()));
1094 $scope.circ_list.push(circ);
1100 function loadCircCounts() {
1102 delete $scope.circ_counts;
1103 $scope.total_circs = 0;
1104 $scope.total_circs_this_year = 0;
1105 $scope.total_circs_prev_year = 0;
1106 if (!copyId) return;
1108 egCore.pcrud.search('circbyyr',
1109 {copy : copyId}, null, {atomic : true})
1111 .then(function(counts) {
1112 $scope.circ_counts = counts;
1114 angular.forEach(counts, function(count) {
1115 $scope.total_circs += Number(count.count());
1118 var this_year = counts.filter(function(c) {
1119 return c.year() == new Date().getFullYear();
1122 $scope.total_circs_this_year = (function() {
1124 if (this_year.length == 2) {
1125 total = (Number(this_year[0].count()) + Number(this_year[1].count()));
1126 } else if (this_year.length == 1) {
1127 total = Number(this_year[0].count());
1132 var prev_year = counts.filter(function(c) {
1133 return c.year() == new Date().getFullYear() - 1;
1136 $scope.total_circs_prev_year = (function() {
1138 if (prev_year.length == 2) {
1139 total = (Number(prev_year[0].count()) + Number(prev_year[1].count()));
1140 } else if (prev_year.length == 1) {
1141 total = Number(prev_year[0].count());
1149 function loadHolds() {
1151 if (!copyId) return;
1153 egCore.pcrud.search('ahr',
1154 { current_copy : copyId,
1156 fulfillment_time : null,
1157 capture_time : {'<>' : null}
1161 ahr : ['requestor', 'usr'],
1165 ).then(null, null, function(hold) {
1167 hold.pickup_lib(egCore.org.get(hold.pickup_lib()));
1168 if (hold.current_shelf_lib()) {
1169 hold.current_shelf_lib(
1170 egCore.org.get(hold.current_shelf_lib()));
1172 hold.behind_desk(Boolean(hold.behind_desk() == 't'));
1176 function loadMostRecentTransit() {
1177 delete $scope.transit;
1178 delete $scope.hold_transit;
1179 if (!copyId) return;
1181 egCore.pcrud.search('atc',
1182 {target_copy : copyId},
1184 order_by : {atc : 'source_send_time DESC'},
1188 ).then(null, null, function(transit) {
1189 // use progress callback since we'll get up to one result
1190 $scope.transit = transit;
1191 transit.source(egCore.org.get(transit.source()));
1192 transit.dest(egCore.org.get(transit.dest()));
1196 function loadCourseInfo() {
1197 delete $scope.courses;
1198 delete $scope.instructors;
1199 delete $scope.course_ids;
1200 delete $scope.instructors_exist;
1201 if (!copyId) return;
1202 $scope.course_ids = [];
1203 $scope.courses = [];
1204 $scope.instructors = {};
1206 egCore.pcrud.search('acmcm', {
1212 }, order_by: {acmc : 'id desc'}
1213 }).then(null, null, function(material) {
1215 $scope.courses.push(material.course());
1218 'open-ils.circ.course_users.retrieve',
1219 material.course().id()
1220 ).then(null, null, function(instructors) {
1221 angular.forEach(instructors, function(instructor) {
1222 var patron_id = instructor.patron_id.toString();
1223 if (!$scope.instructors[patron_id]) {
1224 $scope.instructors[patron_id] = instructor;
1225 $scope.instructors_exist = true;
1226 $scope.instructors[patron_id]._linked_course = [];
1228 $scope.instructors[patron_id]._linked_course.push({
1229 role: instructor.usr_role,
1230 course: material.course().name()
1238 // we don't need all data on all tabs, so fetch what's needed when needed.
1239 function loadTabData() {
1240 switch($scope.tab) {
1256 loadMostRecentTransit();
1263 case 'triggered_events':
1264 var url = $location.absUrl().replace(/\/staff.*/, '/actor/user/event_log');
1265 url += '?copy_id=' + encodeURIComponent(copyId);
1266 $scope.triggered_events_url = url;
1273 'open-ils.actor.anon_cache.set_value',
1274 null, 'edit-these-copies', {
1275 record_id: $scope.recordId,
1280 ).then(function(key) {
1282 var url = egCore.env.basePath + 'cat/volcopy/' + key;
1283 $window.location.href = url;
1285 alert('Could not create anonymous cache key!');
1293 $scope.addCopyAlerts = function(copy_id) {
1294 egCirc.add_copy_alerts([copy_id]).then(function() {
1296 loadCopy($scope.copy.barcode()).then(loadTabData);
1299 $scope.manageCopyAlerts = function(copy_id) {
1300 egCirc.manage_copy_alerts([copy_id]).then(function() {
1302 loadCopy($scope.copy.barcode()).then(loadTabData);
1306 $scope.context.toggleDisplay = function() {
1307 $location.path('/cat/item/search');
1310 // handle the barcode scan box, which will replace our current copy
1311 $scope.context.search = function(args) {
1312 loadCopy(args.barcode).then(loadTabData);
1315 $scope.context.show_triggered_events = function() {
1316 $location.path('/cat/item/' + copyId + '/triggered_events');
1319 loadCopy().then(loadTabData);