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'])
12 .config(function($routeProvider, $locationProvider, $compileProvider) {
13 $locationProvider.html5Mode(true);
14 $compileProvider.aHrefSanitizationWhitelist(/^\s*(https?|blob):/); // grid export
16 var resolver = {delay :
17 ['egStartup', function(egStartup) {return egStartup.go()}]}
19 $routeProvider.when('/cat/catalog/index', {
20 templateUrl: './cat/catalog/t_catalog',
21 controller: 'CatalogCtrl',
25 $routeProvider.when('/cat/catalog/retrieve_by_id', {
26 templateUrl: './cat/catalog/t_retrieve_by_id',
27 controller: 'CatalogRecordRetrieve',
31 $routeProvider.when('/cat/catalog/retrieve_by_tcn', {
32 templateUrl: './cat/catalog/t_retrieve_by_tcn',
33 controller: 'CatalogRecordRetrieve',
37 // create some catalog page-specific mappings
38 $routeProvider.when('/cat/catalog/record/:record_id', {
39 templateUrl: './cat/catalog/t_catalog',
40 controller: 'CatalogCtrl',
44 // create some catalog page-specific mappings
45 $routeProvider.when('/cat/catalog/record/:record_id/:record_tab', {
46 templateUrl: './cat/catalog/t_catalog',
47 controller: 'CatalogCtrl',
51 $routeProvider.otherwise({redirectTo : '/cat/catalog/index'});
57 .controller('CatalogRecordRetrieve',
58 ['$scope','$routeParams','$location','$q','egCore',
59 function($scope , $routeParams , $location , $q , egCore ) {
61 $scope.focusMe = true;
63 // jump to the patron checkout UI
64 function loadRecord(record_id) {
66 .path('/cat/catalog/record/' + record_id);
69 $scope.submitId = function(args) {
70 $scope.recordNotFound = null;
71 if (!args.record_id) return;
73 // blur so next time it's set to true it will re-apply select()
74 $scope.selectMe = false;
76 return loadRecord(args.record_id);
79 $scope.submitTCN = function(args) {
80 $scope.recordNotFound = null;
81 $scope.moreRecordsFound = null;
82 if (!args.record_tcn) return;
84 // blur so next time it's set to true it will re-apply select()
85 $scope.selectMe = false;
90 'open-ils.search.biblio.tcn',
93 .then(function(resp) { // get_barcodes
95 if (evt = egCore.evt.parse(resp)) {
101 $scope.recordNotFound = args.record_tcn;
102 $scope.selectMe = true;
106 if (resp.count > 1) {
107 $scope.moreRecordsFound = args.record_tcn;
108 $scope.selectMe = true;
112 var record_id = resp.ids[0];
113 return loadRecord(record_id);
119 .controller('CatalogCtrl',
120 ['$scope','$routeParams','$location','$q','egCore','egHolds',
121 'egGridDataProvider','egHoldGridActions',
122 function($scope , $routeParams , $location , $q , egCore , egHolds,
123 egGridDataProvider , egHoldGridActions) {
125 // set record ID on page load if available...
126 $scope.record_id = $routeParams.record_id;
128 if ($routeParams.record_id) $scope.from_route = true;
129 else $scope.from_route = false;
131 // will hold a ref to the opac iframe
132 $scope.opac_iframe = null;
133 $scope.opac_call = function (opac_frame_function, force_opac_tab) {
134 if ($scope.opac_iframe) {
135 if (force_opac_tab) $scope.record_tab = 'catalog';
136 $scope.opac_iframe.contentWindow[opac_frame_function]();
140 $scope.stop_unload = false;
141 $scope.$watch('stop_unload',
142 function(newVal, oldVal) {
143 if (newVal && newVal != oldVal && $scope.opac_iframe) {
144 $($scope.opac_iframe.contentWindow).on('beforeunload', function(){
145 return 'There is unsaved data in this record.'
148 $($scope.opac_iframe.contentWindow).off('beforeunload');
153 // Set the "last bib" cookie, if we have that
154 if ($scope.record_id)
155 egCore.hatch.setLocalItem("eg.cat.last_record_retrieved", $scope.record_id);
157 // also set it when the iframe changes to a new record
158 $scope.handle_page = function(url) {
160 if (!url || url == 'about:blank') {
161 // nothing loaded. If we already have a record ID, leave it.
165 var match = url.match(/\/+opac\/+record\/+(\d+)/);
167 $scope.record_id = match[1];
168 egCore.hatch.setLocalItem("eg.cat.last_record_retrieved", $scope.record_id);
170 // force the record_id to show up in the page.
171 // not sure why a $digest isn't occuring here.
172 try { $scope.$apply() } catch(E) {}
174 delete $scope.record_id;
177 if ($scope.record_id) {
178 var default_tab = egCore.hatch.getLocalItem( 'eg.cat.default_record_tab' );
179 tab = $routeParams.record_tab || default_tab || 'catalog';
181 tab = $routeParams.record_tab || 'catalog';
183 $scope.set_record_tab(tab);
186 // xulG catalog handlers
187 $scope.handlers = { }
189 // ------------------------------------------------------------------
191 var provider = egGridDataProvider.instance({});
192 $scope.hold_grid_data_provider = provider;
193 $scope.grid_actions = egHoldGridActions;
194 $scope.grid_actions.refresh = function () { provider.refresh() };
195 $scope.hold_grid_controls = {};
197 var hold_ids = []; // current list of holds
198 function fetchHolds(offset, count) {
199 var ids = hold_ids.slice(offset, offset + count);
200 return egHolds.fetch_holds(ids).then(null, null,
201 function(hold_data) {
207 provider.get = function(offset, count) {
208 if ($scope.record_tab != 'holds') return $q.when();
209 var deferred = $q.defer();
210 hold_ids = []; // no caching ATM
215 'open-ils.circ.holds.retrieve_all_from_title',
216 egCore.auth.token(), $scope.record_id,
217 {pickup_lib : egCore.org.descendants($scope.pickup_ou.id(), true)}
219 function(hold_data) {
220 angular.forEach(hold_data, function(list, type) {
221 hold_ids = hold_ids.concat(list);
223 fetchHolds(offset, count).then(
224 deferred.resolve, null, deferred.notify);
228 return deferred.promise;
231 $scope.detail_view = function(action, user_data, items) {
233 $scope.detail_hold_id = h.hold.id();
237 $scope.list_view = function(items) {
238 $scope.detail_hold_id = null;
241 // refresh the list of record holds when the pickup lib is changed.
242 $scope.pickup_ou = egCore.org.get(egCore.auth.user().ws_ou());
243 $scope.pickup_ou_changed = function(org) {
244 $scope.pickup_ou = org;
248 $scope.print_holds = function() {
250 angular.forEach($scope.hold_grid_controls.allItems(), function(item) {
252 hold : egCore.idl.toHash(item.hold),
253 patron_last : item.patron_last,
254 patron_alias : item.patron_alias,
255 patron_barcode : item.patron_barcode,
256 copy : egCore.idl.toHash(item.copy),
257 volume : egCore.idl.toHash(item.volume),
258 title : item.mvr.title(),
259 author : item.mvr.author()
265 template : 'holds_for_bib',
266 scope : {holds : holds}
270 $scope.mark_hold_transfer_dest = function() {
271 egCore.hatch.setLocalItem(
272 'eg.circ.hold.title_transfer_target', $scope.record_id);
275 // UI presents this option as "all holds"
276 $scope.transfer_holds_to_marked = function() {
277 var hold_ids = $scope.hold_grid_controls.allItems().map(
278 function(hold_data) {return hold_data.hold.id()});
279 egHolds.transfer_to_marked_title(hold_ids);
282 // ------------------------------------------------------------------
283 // Initialize the selected tab
285 function init_cat_url() {
286 // Set the initial catalog URL. This only happens once.
287 // The URL is otherwise generated through user navigation.
288 if ($scope.catalog_url) return;
290 var url = $location.absUrl().replace(/\/staff.*/, '/opac/advanced');
292 // A record ID in the path indicates a request for the record-
294 if ($routeParams.record_id) {
295 url = url.replace(/advanced/, '/record/' + $scope.record_id);
298 $scope.catalog_url = url;
301 $scope.set_record_tab = function(tab) {
302 $scope.record_tab = tab;
311 $scope.detail_hold_record_id = $scope.record_id;
312 // refresh the holds grid
318 $scope.set_default_record_tab = function() {
319 egCore.hatch.setLocalItem(
320 'eg.cat.default_record_tab', $scope.record_tab);
324 if ($scope.record_id) {
325 var default_tab = egCore.hatch.getLocalItem( 'eg.cat.default_record_tab' );
326 tab = $routeParams.record_tab || default_tab || 'catalog';
328 tab = $routeParams.record_tab || 'catalog';
330 $scope.set_record_tab(tab);