]> git.evergreen-ils.org Git - working/Evergreen.git/blob - Open-ILS/web/js/ui/default/staff/cat/catalog/app.js
webstaff: Allow UIs wrapping the opac to call functions in the opac iframe
[working/Evergreen.git] / Open-ILS / web / js / ui / default / staff / cat / catalog / app.js
1 /**
2  * TPAC Frame App
3  *
4  * currently, this app doesn't use routes for each sub-ui, because 
5  * reloading the catalog each time is sloooow.  better so far to 
6  * swap out divs w/ ng-if / ng-show / ng-hide as needed.
7  *
8  */
9
10 angular.module('egCatalogApp', ['ui.bootstrap','ngRoute','egCoreMod','egGridMod', 'egMarcMod'])
11
12 .config(function($routeProvider, $locationProvider, $compileProvider) {
13     $locationProvider.html5Mode(true);
14     $compileProvider.aHrefSanitizationWhitelist(/^\s*(https?|blob):/); // grid export
15
16     var resolver = {delay : 
17         ['egStartup', function(egStartup) {return egStartup.go()}]}
18
19     $routeProvider.when('/cat/catalog/index', {
20         templateUrl: './cat/catalog/t_catalog',
21         controller: 'CatalogCtrl',
22         resolve : resolver
23     });
24
25     $routeProvider.when('/cat/catalog/retrieve_by_id', {
26         templateUrl: './cat/catalog/t_retrieve_by_id',
27         controller: 'CatalogRecordRetrieve',
28         resolve : resolver
29     });
30
31     $routeProvider.when('/cat/catalog/retrieve_by_tcn', {
32         templateUrl: './cat/catalog/t_retrieve_by_tcn',
33         controller: 'CatalogRecordRetrieve',
34         resolve : resolver
35     });
36
37     // create some catalog page-specific mappings
38     $routeProvider.when('/cat/catalog/record/:record_id', {
39         templateUrl: './cat/catalog/t_catalog',
40         controller: 'CatalogCtrl',
41         resolve : resolver
42     });
43
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',
48         resolve : resolver
49     });
50
51     $routeProvider.otherwise({redirectTo : '/cat/catalog/index'});
52 })
53
54
55 /**
56  * */
57 .controller('CatalogRecordRetrieve',
58        ['$scope','$routeParams','$location','$q','egCore',
59 function($scope , $routeParams , $location , $q , egCore ) {
60
61     $scope.focusMe = true;
62
63     // jump to the patron checkout UI
64     function loadRecord(record_id) {
65         $location
66         .path('/cat/catalog/record/' + record_id);
67     }
68
69     $scope.submitId = function(args) {
70         $scope.recordNotFound = null;
71         if (!args.record_id) return;
72
73         // blur so next time it's set to true it will re-apply select()
74         $scope.selectMe = false;
75
76         return loadRecord(args.record_id);
77     }
78
79     $scope.submitTCN = function(args) {
80         $scope.recordNotFound = null;
81         $scope.moreRecordsFound = null;
82         if (!args.record_tcn) return;
83
84         // blur so next time it's set to true it will re-apply select()
85         $scope.selectMe = false;
86
87         // lookup TCN
88         egCore.net.request(
89             'open-ils.search',
90             'open-ils.search.biblio.tcn',
91             args.record_tcn)
92
93         .then(function(resp) { // get_barcodes
94
95             if (evt = egCore.evt.parse(resp)) {
96                 alert(evt); // FIXME
97                 return;
98             }
99
100             if (!resp.count) {
101                 $scope.recordNotFound = args.record_tcn;
102                 $scope.selectMe = true;
103                 return;
104             }
105
106             if (resp.count > 1) {
107                 $scope.moreRecordsFound = args.record_tcn;
108                 $scope.selectMe = true;
109                 return;
110             }
111
112             var record_id = resp.ids[0];
113             return loadRecord(record_id);
114         });
115     }
116
117 }])
118
119 .controller('CatalogCtrl',
120        ['$scope','$routeParams','$location','$q','egCore','egHolds',
121         'egGridDataProvider','egHoldGridActions',
122 function($scope , $routeParams , $location , $q , egCore , egHolds, 
123          egGridDataProvider , egHoldGridActions) {
124
125     // set record ID on page load if available...
126     $scope.record_id = $routeParams.record_id;
127
128     if ($routeParams.record_id) $scope.from_route = true;
129     else $scope.from_route = false;
130
131     // will hold a ref to the opac iframe
132     $scope.opac_iframe = null;
133     $scope.opac_call = function (thing) {
134         if ($scope.opac_iframe) $scope.opac_iframe.contentWindow[thing]()
135     }
136
137     // Set the "last bib" cookie, if we have that
138     if ($scope.record_id)
139         egCore.hatch.setLocalItem("eg.cat.last_record_retrieved", $scope.record_id);
140
141     // also set it when the iframe changes to a new record
142     $scope.handle_page = function(url) {
143
144         if (!url || url == 'about:blank') {
145             // nothing loaded.  If we already have a record ID, leave it.
146             return;
147         }
148
149         var match = url.match(/\/+opac\/+record\/+(\d+)/);
150         if (match) {
151             $scope.record_id = match[1];
152             egCore.hatch.setLocalItem("eg.cat.last_record_retrieved", $scope.record_id);
153
154             // force the record_id to show up in the page.  
155             // not sure why a $digest isn't occuring here.
156             try { $scope.$apply() } catch(E) {}
157         } else {
158             delete $scope.record_id;
159         }
160
161         if ($scope.record_id) {
162             var default_tab = egCore.hatch.getLocalItem( 'eg.cat.default_record_tab' );
163             tab = $routeParams.record_tab || default_tab || 'catalog';
164         } else {
165             tab = $routeParams.record_tab || 'catalog';
166         }
167         $scope.set_record_tab(tab);
168     }
169
170     // xulG catalog handlers
171     $scope.handlers = { }
172
173     // ------------------------------------------------------------------
174     // Holds 
175     var provider = egGridDataProvider.instance({});
176     $scope.hold_grid_data_provider = provider;
177     $scope.grid_actions = egHoldGridActions;
178     $scope.grid_actions.refresh = function () { provider.refresh() };
179     $scope.hold_grid_controls = {};
180
181     var hold_ids = []; // current list of holds
182     function fetchHolds(offset, count) {
183         var ids = hold_ids.slice(offset, offset + count);
184         return egHolds.fetch_holds(ids).then(null, null,
185             function(hold_data) { 
186                 return hold_data;
187             }
188         );
189     }
190
191     provider.get = function(offset, count) {
192         if ($scope.record_tab != 'holds') return $q.when();
193         var deferred = $q.defer();
194         hold_ids = []; // no caching ATM
195
196         // fetch the IDs
197         egCore.net.request(
198             'open-ils.circ',
199             'open-ils.circ.holds.retrieve_all_from_title',
200             egCore.auth.token(), $scope.record_id, 
201             {pickup_lib : egCore.org.descendants($scope.pickup_ou.id(), true)}
202         ).then(
203             function(hold_data) {
204                 angular.forEach(hold_data, function(list, type) {
205                     hold_ids = hold_ids.concat(list);
206                 });
207                 fetchHolds(offset, count).then(
208                     deferred.resolve, null, deferred.notify);
209             }
210         );
211
212         return deferred.promise;
213     }
214
215     $scope.detail_view = function(action, user_data, items) {
216         if (h = items[0]) {
217             $scope.detail_hold_id = h.hold.id();
218         }
219     }
220
221     $scope.list_view = function(items) {
222          $scope.detail_hold_id = null;
223     }
224
225     // refresh the list of record holds when the pickup lib is changed.
226     $scope.pickup_ou = egCore.org.get(egCore.auth.user().ws_ou());
227     $scope.pickup_ou_changed = function(org) {
228         $scope.pickup_ou = org;
229         provider.refresh();
230     }
231
232     $scope.print_holds = function() {
233         var holds = [];
234         angular.forEach($scope.hold_grid_controls.allItems(), function(item) {
235             holds.push({
236                 hold : egCore.idl.toHash(item.hold),
237                 patron_last : item.patron_last,
238                 patron_alias : item.patron_alias,
239                 patron_barcode : item.patron_barcode,
240                 copy : egCore.idl.toHash(item.copy),
241                 volume : egCore.idl.toHash(item.volume),
242                 title : item.mvr.title(),
243                 author : item.mvr.author()
244             });
245         });
246
247         egCore.print.print({
248             context : 'receipt', 
249             template : 'holds_for_bib', 
250             scope : {holds : holds}
251         });
252     }
253
254     $scope.mark_hold_transfer_dest = function() {
255         egCore.hatch.setLocalItem(
256             'eg.circ.hold.title_transfer_target', $scope.record_id);
257     }
258
259     // UI presents this option as "all holds"
260     $scope.transfer_holds_to_marked = function() {
261         var hold_ids = $scope.hold_grid_controls.allItems().map(
262             function(hold_data) {return hold_data.hold.id()});
263         egHolds.transfer_to_marked_title(hold_ids);
264     }
265
266     // ------------------------------------------------------------------
267     // Initialize the selected tab
268
269     function init_cat_url() {
270         // Set the initial catalog URL.  This only happens once.
271         // The URL is otherwise generated through user navigation.
272         if ($scope.catalog_url) return; 
273
274         var url = $location.absUrl().replace(/\/staff.*/, '/opac/advanced');
275
276         // A record ID in the path indicates a request for the record-
277         // specific page.
278         if ($routeParams.record_id) {
279             url = url.replace(/advanced/, '/record/' + $scope.record_id);
280         }
281
282         $scope.catalog_url = url;
283     }
284
285     $scope.set_record_tab = function(tab) {
286         $scope.record_tab = tab;
287
288         switch(tab) {
289
290             case 'catalog':
291                 init_cat_url();
292                 break;
293
294             case 'holds':
295                 $scope.detail_hold_record_id = $scope.record_id; 
296                 // refresh the holds grid
297                 provider.refresh();
298                 break;
299         }
300     }
301
302     $scope.set_default_record_tab = function() {
303         egCore.hatch.setLocalItem(
304             'eg.cat.default_record_tab', $scope.record_tab);
305     }
306
307     var tab;
308     if ($scope.record_id) {
309         var default_tab = egCore.hatch.getLocalItem( 'eg.cat.default_record_tab' );
310         tab = $routeParams.record_tab || default_tab || 'catalog';
311     } else {
312         tab = $routeParams.record_tab || 'catalog';
313     }
314     $scope.set_record_tab(tab);
315
316 }])
317