de6e94f5fbfa245b50ca17670093d229bd8f0b37
[working/Evergreen.git] / Open-ILS / web / js / ui / default / staff / admin / workstation / app.js
1 /**
2  * App to drive the base page. 
3  * Login Form
4  * Splash Page
5  */
6
7 angular.module('egWorkstationAdmin', 
8     ['ngRoute', 'ui.bootstrap', 'egCoreMod', 'egUiMod'])
9
10 .config(['$routeProvider','$locationProvider','$compileProvider', 
11  function($routeProvider , $locationProvider , $compileProvider) {
12
13     $locationProvider.html5Mode(true);
14     $compileProvider.aHrefSanitizationWhitelist(/^\s*(https?|mailto|blob):/); 
15     var resolver = {delay : function(egStartup) {return egStartup.go()}};
16
17     $routeProvider.when('/admin/workstation/workstations', {
18         templateUrl: './admin/workstation/t_workstations',
19         controller: 'WSRegCtrl',
20         resolve : resolver
21     });
22
23     $routeProvider.when('/admin/workstation/print/config', {
24         templateUrl: './admin/workstation/t_print_config',
25         controller: 'PrintConfigCtrl',
26         resolve : resolver
27     });
28
29     $routeProvider.when('/admin/workstation/print/templates', {
30         templateUrl: './admin/workstation/t_print_templates',
31         controller: 'PrintTemplatesCtrl',
32         resolve : resolver
33     });
34
35     $routeProvider.when('/admin/workstation/stored_prefs', {
36         templateUrl: './admin/workstation/t_stored_prefs',
37         controller: 'StoredPrefsCtrl',
38         resolve : resolver
39     });
40
41     $routeProvider.when('/admin/workstation/hatch', {
42         templateUrl: './admin/workstation/t_hatch',
43         controller: 'HatchCtrl',
44         resolve : resolver
45     });
46
47     $routeProvider.when('/admin/workstation/tests', {
48         templateUrl: './admin/workstation/t_tests',
49         controller: 'testsCtrl',
50         resolve : resolver
51     });
52     
53     // default page 
54     $routeProvider.otherwise({
55         templateUrl : './admin/workstation/t_splash',
56         controller : 'SplashCtrl',
57         resolve : resolver
58     });
59 }])
60
61 .config(['ngToastProvider', function(ngToastProvider) {
62   ngToastProvider.configure({
63     verticalPosition: 'bottom',
64     animation: 'fade'
65   });
66 }])
67
68 .factory('workstationSvc',
69        ['$q','$timeout','$location','egCore','egConfirmDialog',
70 function($q , $timeout , $location , egCore , egConfirmDialog) {
71     
72     var service = {};
73
74     service.get_all = function() {
75         return egCore.hatch.getItem('eg.workstation.all')
76         .then(function(all) { return all || [] });
77     }
78
79     service.get_default = function() {
80         return egCore.hatch.getItem('eg.workstation.default');
81     }
82
83     service.set_default = function(name) {
84         return egCore.hatch.setItem('eg.workstation.default', name);
85     }
86
87     service.register_workstation = function(base_name, name, org_id) {
88         return service.register_ws_api(base_name, name, org_id)
89         .then(function(ws_id) {
90             return service.track_new_ws(ws_id, name, org_id);
91         });
92     };
93
94     service.register_ws_api = 
95         function(base_name, name, org_id, override, deferred) {
96         if (!deferred) deferred = $q.defer();
97
98         var method = 'open-ils.actor.workstation.register';
99         if (override) method += '.override';
100
101         egCore.net.request(
102             'open-ils.actor', method, egCore.auth.token(), name, org_id)
103
104         .then(function(resp) {
105
106             if (evt = egCore.evt.parse(resp)) {
107                 console.log('register returned ' + evt.toString());
108
109                 if (evt.textcode == 'WORKSTATION_NAME_EXISTS' && !override) {
110
111                     egConfirmDialog.open(
112                         egCore.strings.WS_EXISTS, base_name, {  
113                             ok : function() {
114                                 service.register_ws_api(
115                                     base_name, name, org_id, true, deferred)
116                             },
117                             cancel : function() {
118                                 deferred.reject();
119                             }
120                         }
121                     );
122
123                 } else {
124                     alert(evt.toString());
125                     deferred.reject();
126                 }
127             } else if (resp) {
128                 console.log('Resolving register promise with: ' + resp);
129                 deferred.resolve(resp);
130             }
131         });
132
133         return deferred.promise;
134     }
135
136     service.track_new_ws = function(ws_id, ws_name, owning_lib) {
137         console.log('Tracking newly created WS with ID ' + ws_id);
138         var new_ws = {id : ws_id, name : ws_name, owning_lib : owning_lib};
139
140         return service.get_all()
141         .then(function(all) {
142             all.push(new_ws);
143             return egCore.hatch.setItem('eg.workstation.all', all)
144             .then(function() { return new_ws });
145         });
146     }
147
148     // Remove all traces of the workstation locally.
149     // This does not remove the WS from the server.
150     service.remove_workstation = function(name) {
151         console.debug('Removing workstation: ' + name);
152
153         return egCore.hatch.getItem('eg.workstation.all')
154
155         // remove from list of all workstations
156         .then(function(all) {
157             if (!all) all = [];
158             var keep = all.filter(function(ws) {return ws.name != name});
159             return egCore.hatch.setItem('eg.workstation.all', keep)
160
161         }).then(function() { 
162
163             return service.get_default()
164
165         }).then(function(def) {
166             if (def == name) {
167                 console.debug('Removing default workstation: ' + name);
168                 return egCore.hatch.removeItem('eg.workstation.default');
169             }
170         });
171     }
172
173     return service;
174 }])
175
176
177 .controller('SplashCtrl',
178        ['$scope','$window','$location','egCore','egConfirmDialog',
179 function($scope , $window , $location , egCore , egConfirmDialog) {
180
181     egCore.hatch.getItem('eg.audio.disable').then(function(val) {
182         $scope.disable_sound = val;
183     });
184
185     egCore.hatch.getItem('eg.search.search_lib').then(function(val) {
186         $scope.search_lib = egCore.org.get(val);
187     });
188     $scope.handle_search_lib_changed = function(org) {
189         egCore.hatch.setItem('eg.search.search_lib', org.id());
190     };
191
192     egCore.hatch.getItem('eg.search.pref_lib').then(function(val) {
193         $scope.pref_lib = egCore.org.get(val);
194     });
195     $scope.handle_pref_lib_changed = function(org) {
196         egCore.hatch.setItem('eg.search.pref_lib', org.id());
197     };
198
199     $scope.adv_pane = 'advanced'; // default value if not explicitly set
200     egCore.hatch.getItem('eg.search.adv_pane').then(function(val) {
201         $scope.adv_pane = val;
202     });
203     $scope.$watch('adv_pane', function(newVal, oldVal) {
204         if (typeof newVal != 'undefined' && newVal != oldVal) {
205             egCore.hatch.setItem('eg.search.adv_pane', newVal);
206         }
207     });
208
209     $scope.apply_sound = function() {
210         if ($scope.disable_sound) {
211             egCore.hatch.setItem('eg.audio.disable', true);
212         } else {
213             egCore.hatch.removeItem('eg.audio.disable');
214         }
215     }
216
217     $scope.test_audio = function(sound) {
218         egCore.audio.play(sound);
219     }
220
221 }])
222
223 .controller('PrintConfigCtrl',
224        ['$scope','egCore',
225 function($scope , egCore) {
226
227     $scope.printConfig = {};
228     $scope.setContext = function(ctx) { 
229         $scope.context = ctx; 
230         $scope.isTestView = false;
231     }
232     $scope.setContext('default');
233
234     $scope.setContentType = function(type) { $scope.contentType = type }
235     $scope.setContentType('text/plain');
236
237     $scope.useHatchPrinting = function() {
238         return egCore.hatch.usePrinting();
239     }
240
241     $scope.hatchIsOpen = function() {
242         return egCore.hatch.hatchAvailable;
243     }
244
245     $scope.getPrinterByAttr = function(attr, value) {
246         var printer;
247         angular.forEach($scope.printers, function(p) {
248             if (p[attr] == value) printer = p;
249         });
250         return printer;
251     }
252
253     $scope.resetPrinterSettings = function(context) {
254         $scope.printConfig[context] = {
255             context : context,
256             printer : $scope.defaultPrinter ? $scope.defaultPrinter.name : null,
257             autoMargins : true, 
258             allPages : true,
259             pageRanges : []
260         };
261     }
262
263     $scope.savePrinterSettings = function(context) {
264         return egCore.hatch.setPrintConfig(
265             context, $scope.printConfig[context]);
266     }
267
268     $scope.printerConfString = function() {
269         if ($scope.printConfigError) return $scope.printConfigError;
270         if (!$scope.printConfig) return;
271         if (!$scope.printConfig[$scope.context]) return;
272         return JSON.stringify(
273             $scope.printConfig[$scope.context], undefined, 2);
274     }
275
276     function loadPrinterOptions(name) {
277         egCore.hatch.getPrinterOptions(name).then(
278             function(options) {$scope.printerOptions = options});
279     }
280
281     $scope.setPrinter = function(name) {
282         $scope.printConfig[$scope.context].printer = name;
283         loadPrinterOptions(name);
284     }
285
286     $scope.testPrint = function(withDialog) {
287         if ($scope.contentType == 'text/plain') {
288             egCore.print.print({
289                 context : $scope.context, 
290                 content_type : $scope.contentType, 
291                 content : $scope.textPrintContent,
292                 show_dialog : withDialog
293             });
294         } else {
295             egCore.print.print({
296                 context : $scope.context,
297                 content_type : $scope.contentType, 
298                 content : $scope.htmlPrintContent, 
299                 scope : {
300                     value1 : 'Value One', 
301                     value2 : 'Value Two',
302                     date_value : '2015-02-04T14:04:34-0400'
303                 },
304                 show_dialog : withDialog
305             });
306         }
307     }
308
309     // Load startup data....
310     // Don't bother talking to Hatch if it's not there.
311     if (!egCore.hatch.hatchAvailable) return;
312
313     // fetch info on all remote printers
314     egCore.hatch.getPrinters()
315     .then(function(printers) { 
316         $scope.printers = printers;
317
318         var def = $scope.getPrinterByAttr('is-default', true);
319         if (!def && printers.length) def = printers[0];
320
321         if (def) {
322             $scope.defaultPrinter = def;
323             loadPrinterOptions(def.name);
324         }
325     }).then(function() {
326         angular.forEach(
327             ['default','receipt','label','mail','offline'],
328             function(ctx) {
329                 egCore.hatch.getPrintConfig(ctx).then(function(conf) {
330                     if (conf) {
331                         $scope.printConfig[ctx] = conf;
332                     } else {
333                         $scope.resetPrinterSettings(ctx);
334                     }
335                 });
336             }
337         );
338     });
339
340 }])
341
342 .controller('PrintTemplatesCtrl',
343        ['$scope','$q','egCore','ngToast',
344 function($scope , $q , egCore , ngToast) {
345
346     $scope.print = {
347         template_name : 'bills_current',
348         template_output : '',
349         template_context : 'default'
350     };
351
352     // print preview scope data
353     // TODO: consider moving the template-specific bits directly
354     // into the templates or storing template- specific script files
355     // alongside the templates.
356     // NOTE: A lot of this data can be shared across templates.
357     var seed_user = {
358         prefix : 'Mr',
359         first_given_name : 'Slow',
360         second_given_name : 'Joe',
361         family_name : 'Jones',
362         suffix : 'III',
363         card : {
364             barcode : '30393830393'
365         },
366         money_summary : {
367             balance_owed : 4, // This is currently how these values are returned to the client
368             total_billed : '5.00',
369             total_paid : '1.00'
370         },
371         expire_date : '2020-12-31',
372         alias : 'the dude',
373         has_email : true,
374         has_phone : false
375     }
376     var seed_addr = {
377         street1 : '123 Apple Rd',
378         street2 : 'Suite B',
379         city : 'Anywhere',
380         state : 'XX',
381         country : 'US',
382         post_code : '12345'
383     }
384
385     var seed_record = {
386         title : 'Traveling Pants!!',
387         author : 'Jane Jones',
388         isbn : '1231312123'
389     };
390
391     var seed_copy = {
392         barcode : '33434322323',
393         call_number : {
394             label : '636.8 JON',
395             record : {
396                 simple_record : {
397                     'title' : 'Test Title'
398                 }
399             }
400         },
401         location : {
402             name : 'General Collection'
403         },
404         // flattened versions for item status template
405         // TODO - make this go away
406         'call_number.label' : '636.8 JON',
407         'call_number.record.simple_record.title' : 'Test Title',
408         'location.name' : 'General Colleciton'
409     }
410
411     var one_hold = {
412         behind_desk : 'f',
413         phone_notify : '111-222-3333',
414         sms_notify : '111-222-3333',
415         email_notify : 'user@example.org',
416         request_time : new Date().toISOString(),
417         hold_type : 'T',
418         shelf_expire_time : new Date().toISOString()
419     }
420
421     var seed_transit = {
422         source : {
423             name : 'Library Y',
424             shortname : 'LY',
425             holds_address : seed_addr
426         },
427         dest : {
428             name : 'Library X',
429             shortname : 'LX',
430             holds_address : seed_addr
431         },
432         source_send_time : new Date().toISOString(),
433         target_copy : seed_copy
434     }
435
436     $scope.preview_scope = {
437         //bills
438         transactions : [
439             {
440                 id : 1,
441                 xact_start : new Date().toISOString(),
442                 summary : {
443                     xact_type : 'circulation',
444                     last_billing_type : 'Overdue materials',
445                     total_owed : 1.50,
446                     last_payment_note : 'Test Note 1',
447                     last_payment_type : 'cash_payment',
448                     total_paid : 0.50,
449                     balance_owed : 1.00
450                 }
451             }, {
452                 id : 2,
453                 xact_start : new Date().toISOString(),
454                 summary : {
455                     xact_type : 'circulation',
456                     last_billing_type : 'Overdue materials',
457                     total_owed : 2.50,
458                     last_payment_note : 'Test Note 2',
459                     last_payment_type : 'credit_payment',
460                     total_paid : 0.50,
461                     balance_owed : 2.00
462                 }
463             }
464         ],
465
466         copy : seed_copy,
467         copies : [ seed_copy ],
468
469         checkins : [
470             {
471                 due_date : new Date().toISOString(),
472                 circ_lib : 1,
473                 duration : '7 days',
474                 target_copy : seed_copy,
475                 copy_barcode : seed_copy.barcode,
476                 call_number : seed_copy.call_number,
477                 title : seed_record.title
478             },
479         ],
480
481         circulations : [
482             {
483                 circ : {
484                     due_date : new Date().toISOString(),
485                     circ_lib : 1,
486                     duration : '7 days'
487                 },
488                 copy : seed_copy,
489                 title : seed_record.title,
490                 author : seed_record.author
491             },
492         ],
493
494         patron_money : {
495             balance_owed : 5.01,
496             total_owed : 10.12,
497             total_paid : 5.11
498         },
499
500         in_house_uses : [
501             {
502                 num_uses : 3,
503                 copy : seed_copy,
504                 title : seed_record.title
505             }
506         ],
507
508         previous_balance : 8.45,
509         payment_total : 2.00,
510         payment_applied : 2.00,
511         new_balance : 6.45,
512         amount_voided : 0,
513         change_given : 0,
514         payment_type : 'cash_payment',
515         payment_note : 'Here is a payment note',
516         note : {
517             create_date : new Date().toISOString(), 
518             title : 'Test Note Title',
519             usr : seed_user,
520             value : 'This patron is super nice!'
521         },
522
523         transit : seed_transit,
524         transits : [ seed_transit ],
525         title : seed_record.title,
526         author : seed_record.author,
527         patron : seed_user,
528         address : seed_addr,
529         dest_location : egCore.idl.toHash(egCore.org.get(egCore.auth.user().ws_ou())),
530         dest_address : seed_addr,
531         hold : one_hold,
532         holds : [
533             {
534                 hold : one_hold, title : 'Some Title 1', author : 'Some Author 1',
535                 volume : { label : '646.4 SOM' }, copy : seed_copy,
536                 part : { label : 'v. 1' },
537                 patron_barcode : 'S52802662',
538                 patron_alias : 'XYZ', patron_last : 'Smith', patron_first : 'Jane',
539                 status_string : 'Ready for Pickup'
540             },
541             {
542                 hold : one_hold, title : 'Some Title 2', author : 'Some Author 2',
543                 volume : { label : '646.4 SOM' }, copy : seed_copy,
544                 part : { label : 'v. 1' },
545                 patron_barcode : 'S52802662',
546                 patron_alias : 'XYZ', patron_last : 'Smith', patron_first : 'Jane',
547                 status_string : 'Ready for Pickup'
548             },
549             {
550                 hold : one_hold, title : 'Some Title 3', author : 'Some Author 3',
551                 volume : { label : '646.4 SOM' }, copy : seed_copy,
552                 part : { label : 'v. 1' },
553                 patron_barcode : 'S52802662',
554                 patron_alias : 'XYZ', patron_last : 'Smith', patron_first : 'Jane',
555                 status_string : 'Canceled'
556             }
557         ]
558     }
559
560     $scope.preview_scope.payments = [
561         {amount : 1.00, xact : $scope.preview_scope.transactions[0]}, 
562         {amount : 1.00, xact : $scope.preview_scope.transactions[1]}
563     ]
564     $scope.preview_scope.payments[0].xact.title = 'Hali Bote Azikaban de tao fan';
565     $scope.preview_scope.payments[0].xact.copy_barcode = '334343434';
566     $scope.preview_scope.payments[1].xact.title = seed_record.title;
567     $scope.preview_scope.payments[1].xact.copy_barcode = seed_copy.barcode;
568
569     // today, staff, current_location, etc.
570     egCore.print.fleshPrintScope($scope.preview_scope);
571
572     $scope.template_changed = function() {
573         $scope.print.load_failed = false;
574         egCore.print.getPrintTemplate($scope.print.template_name)
575         .then(
576             function(html) { 
577                 $scope.print.template_content = html;
578                 console.log('set template content');
579             },
580             function() {
581                 $scope.print.template_content = '';
582                 $scope.print.load_failed = true;
583             }
584         );
585         egCore.print.getPrintTemplateContext($scope.print.template_name)
586         .then(function(template_context) {
587             $scope.print.template_context = template_context;
588         });
589     }
590
591     $scope.reset_to_default = function() {
592         egCore.print.removePrintTemplate(
593             $scope.print.template_name
594         );
595         egCore.print.removePrintTemplateContext(
596             $scope.print.template_name
597         );
598         $scope.template_changed();
599     }
600
601     $scope.save_locally = function() {
602         egCore.print.storePrintTemplate(
603             $scope.print.template_name,
604             $scope.print.template_content
605         );
606         egCore.print.storePrintTemplateContext(
607             $scope.print.template_name,
608             $scope.print.template_context
609         );
610     }
611
612     $scope.exportable_templates = function() {
613         var templates = {};
614         var contexts = {};
615         var deferred = $q.defer();
616         var promises = [];
617         egCore.hatch.getKeys('eg.print.template').then(function(keys) {
618             angular.forEach(keys, function(key) {
619                 if (key.match(/^eg\.print\.template\./)) {
620                     promises.push(egCore.hatch.getItem(key).then(function(value) {
621                         templates[key.replace('eg.print.template.', '')] = value;
622                     }));
623                 } else {
624                     promises.push(egCore.hatch.getItem(key).then(function(value) {
625                         contexts[key.replace('eg.print.template_context.', '')] = value;
626                     }));
627                 }
628             });
629             $q.all(promises).then(function() {
630                 if (Object.keys(templates).length) {
631                     deferred.resolve({
632                         templates: templates,
633                         contexts: contexts
634                     });
635                 } else {
636                     ngToast.warning(egCore.strings.PRINT_TEMPLATES_FAIL_EXPORT);
637                     deferred.reject();
638                 }
639             });
640         });
641         return deferred.promise;
642     }
643
644     $scope.imported_print_templates = { data : '' };
645     $scope.$watch('imported_print_templates.data', function(newVal, oldVal) {
646         if (newVal && newVal != oldVal) {
647             try {
648                 var data = JSON.parse(newVal);
649                 angular.forEach(data.templates, function(template_content, template_name) {
650                     egCore.print.storePrintTemplate(template_name, template_content);
651                 });
652                 angular.forEach(data.contexts, function(template_context, template_name) {
653                     egCore.print.storePrintTemplateContext(template_name, template_context);
654                 });
655                 $scope.template_changed(); // refresh
656                 ngToast.create(egCore.strings.PRINT_TEMPLATES_SUCCESS_IMPORT);
657             } catch (E) {
658                 ngToast.warning(egCore.strings.PRINT_TEMPLATES_FAIL_IMPORT);
659             }
660         }
661     });
662
663     $scope.template_changed(); // load the default
664 }])
665
666 // 
667 .directive('egPrintTemplateOutput', ['$compile',function($compile) {
668     return function(scope, element, attrs) {
669         scope.$watch(
670             function(scope) {
671                 return scope.$eval(attrs.content);
672             },
673             function(value) {
674                 // create an isolate scope and copy the print context
675                 // data into the new scope.
676                 // TODO: see also print security concerns in egHatch
677                 var result = element.html(value);
678                 var context = scope.$eval(attrs.context);
679                 var print_scope = scope.$new(true);
680                 angular.forEach(context, function(val, key) {
681                     print_scope[key] = val;
682                 })
683                 $compile(element.contents())(print_scope);
684             }
685         );
686     };
687 }])
688
689 .controller('StoredPrefsCtrl',
690        ['$scope','$q','egCore','egConfirmDialog',
691 function($scope , $q , egCore , egConfirmDialog) {
692     console.log('StoredPrefsCtrl');
693
694     $scope.setContext = function(ctx) {
695         $scope.context = ctx;
696     }
697     $scope.setContext('local');
698
699     // grab the edit perm
700     $scope.userHasDeletePerm = false;
701     egCore.perm.hasPermHere('ADMIN_WORKSTATION')
702     .then(function(bool) { $scope.userHasDeletePerm = bool });
703
704     // fetch the keys
705
706     function refreshKeys() {
707         $scope.keys = {local : [], remote : []};
708
709         if (egCore.hatch.hatchAvailable) {
710             egCore.hatch.getRemoteKeys().then(
711                 function(keys) { $scope.keys.remote = keys.sort() })
712         }
713     
714         // local calls are non-async
715         $scope.keys.local = egCore.hatch.getLocalKeys();
716     }
717     refreshKeys();
718
719     $scope.selectKey = function(key) {
720         $scope.currentKey = key;
721         $scope.currentKeyContent = null;
722
723         if ($scope.context == 'local') {
724             $scope.currentKeyContent = egCore.hatch.getLocalItem(key);
725         } else {
726             egCore.hatch.getRemoteItem(key)
727             .then(function(content) {
728                 $scope.currentKeyContent = content
729             });
730         }
731     }
732
733     $scope.getCurrentKeyContent = function() {
734         return JSON.stringify($scope.currentKeyContent, null, 2);
735     }
736
737     $scope.removeKey = function(key) {
738         egConfirmDialog.open(
739             egCore.strings.PREFS_REMOVE_KEY_CONFIRM, '',
740             {   deleteKey : key,
741                 ok : function() {
742                     if ($scope.context == 'local') {
743                         egCore.hatch.removeLocalItem(key);
744                         refreshKeys();
745                     } else {
746                         egCore.hatch.removeItem(key)
747                         .then(function() { refreshKeys() });
748                     }
749                 },
750                 cancel : function() {} // user canceled, nothing to do
751             }
752         );
753     }
754 }])
755
756 .controller('WSRegCtrl',
757        ['$scope','$q','$window','$location','egCore','egAlertDialog','workstationSvc',
758 function($scope , $q , $window , $location , egCore , egAlertDialog , workstationSvc) {
759
760     var all_workstations = [];
761     var reg_perm_orgs = [];
762
763     $scope.page_loaded = false;
764     $scope.contextOrg = egCore.org.get(egCore.auth.user().ws_ou());
765     $scope.wsOrgChanged = function(org) { $scope.contextOrg = org; }
766
767     console.log('set context org to ' + $scope.contextOrg);
768
769     // fetch workstation reg perms
770     egCore.perm.hasPermAt('REGISTER_WORKSTATION', true)
771     .then(function(orgList) { 
772         reg_perm_orgs = orgList;
773
774         // hide orgs in the context org selector where this login
775         // does not have the reg_ws perm or the org can't have users
776         $scope.wsOrgHidden = function(id) {
777             return reg_perm_orgs.indexOf(id) == -1
778                 || $scope.cant_have_users(id);
779         }
780
781     // fetch the locally stored workstation data
782     }).then(function() {
783         return workstationSvc.get_all()
784         
785     }).then(function(all) {
786         all_workstations = all || [];
787         $scope.workstations = 
788             all_workstations.map(function(w) { return w.name });
789         return workstationSvc.get_default()
790
791     // fetch the default workstation
792     }).then(function(def) { 
793         $scope.defaultWS = def;
794         $scope.activeWS = $scope.selectedWS = egCore.auth.workstation() || def;
795
796     // Handle any URL commands.
797     }).then(function() {
798         var remove = $location.search().remove;
799          if (remove) {
800             console.log('Removing WS via URL request: ' + remove);
801             return $scope.remove_ws(remove).then(
802                 function() { $scope.page_loaded = true; });
803         }
804         $scope.page_loaded = true;
805     });
806
807     $scope.get_ws_label = function(ws) {
808         return ws == $scope.defaultWS ? 
809             egCore.strings.$replace(egCore.strings.DEFAULT_WS_LABEL, {ws:ws}) : ws;
810     }
811
812     $scope.set_default_ws = function(name) {
813         delete $scope.removing_ws;
814         $scope.defaultWS = name;
815         workstationSvc.set_default(name);
816     }
817
818     $scope.cant_have_users = 
819         function (id) { return !egCore.org.CanHaveUsers(id); };
820     $scope.cant_have_volumes = 
821         function (id) { return !egCore.org.CanHaveVolumes(id); };
822
823     // Log out and return to login page with selected WS 
824     $scope.use_now = function() {
825         egCore.auth.logout();
826         $window.location.href = $location
827             .path('/login')
828             .search({ws : $scope.selectedWS})
829             .absUrl();
830     }
831
832     $scope.can_delete_ws = function(name) {
833         var ws = all_workstations.filter(
834             function(ws) { return ws.name == name })[0];
835         return ws && reg_perm_orgs.indexOf(ws.owning_lib) != -1;
836     }
837
838     $scope.remove_ws = function(remove_me) {
839         $scope.removing_ws = remove_me;
840
841         // Perm is used to disable Remove button in UI, but have to check
842         // again here in case we're removing a WS based on URL params.
843         if (!$scope.can_delete_ws(remove_me)) return $q.when();
844
845         $scope.is_removing = true;
846         return workstationSvc.remove_workstation(remove_me)
847         .then(function() {
848
849             all_workstations = all_workstations.filter(
850                 function(ws) { return ws.name != remove_me });
851
852             $scope.workstations = $scope.workstations.filter(
853                 function(ws) { return ws != remove_me });
854
855             if ($scope.selectedWS == remove_me) 
856                 $scope.selectedWS = $scope.workstations[0];
857
858             if ($scope.defaultWS == remove_me) 
859                 $scope.defaultWS = '';
860
861             $scope.is_removing = false;
862         });
863     }
864
865     $scope.register_ws = function() {
866         delete $scope.removing_ws;
867
868         var full_name = 
869             $scope.contextOrg.shortname() + '-' + $scope.newWSName;
870
871         if ($scope.workstations.indexOf(full_name) > -1) {
872             // avoid duplicate local registrations
873             return egAlertDialog.open(egCore.strings.WS_USED);
874         }
875
876         $scope.is_registering = true;
877         workstationSvc.register_workstation(
878             $scope.newWSName, full_name,
879             $scope.contextOrg.id()
880
881         ).then(function(new_ws) {
882             $scope.workstations.push(new_ws.name);
883             all_workstations.push(new_ws);  
884             $scope.is_registering = false;
885
886             if (!$scope.selectedWS) {
887                 $scope.selectedWS = new_ws.name;
888             }
889             if (!$scope.defaultWS) {
890                 return $scope.set_default_ws(new_ws.name);
891             }
892             $scope.newWSName = '';
893         }, function(err) {
894             $scope.is_registering = false;
895         });
896     }
897 }])
898
899 .controller('HatchCtrl',
900        ['$scope','egCore','ngToast',
901 function($scope , egCore , ngToast) {
902     var hatch = egCore.hatch;  // convenience
903
904     $scope.hatch_available = hatch.hatchAvailable;
905     $scope.hatch_printing = hatch.usePrinting();
906     $scope.hatch_settings = hatch.useSettings();
907     $scope.hatch_offline  = hatch.useOffline();
908
909     // Apply Hatch settings as changes occur in the UI.
910     
911     $scope.$watch('hatch_printing', function(newval) {
912         if (typeof newval != 'boolean') return;
913         hatch.setLocalItem('eg.hatch.enable.printing', newval);
914     });
915
916     $scope.$watch('hatch_settings', function(newval) {
917         if (typeof newval != 'boolean') return;
918         hatch.setLocalItem('eg.hatch.enable.settings', newval);
919     });
920
921     $scope.$watch('hatch_offline', function(newval) {
922         if (typeof newval != 'boolean') return;
923         hatch.setLocalItem('eg.hatch.enable.offline', newval);
924     });
925
926     $scope.copy_to_hatch = function() {
927         hatch.copySettingsToHatch().then(
928             function() {
929                 ngToast.create(egCore.strings.HATCH_SETTINGS_MIGRATION_SUCCESS)},
930             function() {
931                 ngToast.warning(egCore.strings.HATCH_SETTINGS_MIGRATION_FAILURE)}
932         );
933     }
934
935     $scope.copy_to_local = function() {
936         hatch.copySettingsToLocal().then(
937             function() {
938                 ngToast.create(egCore.strings.HATCH_SETTINGS_MIGRATION_SUCCESS)},
939             function() {
940                 ngToast.warning(egCore.strings.HATCH_SETTINGS_MIGRATION_FAILURE)}
941         );
942     }
943
944 }])
945
946 /*
947  * Home of the Latency tester
948  * */
949 .controller('testsCtrl', ['$scope', '$location', 'egCore', function($scope, $location, egCore) {
950     $scope.hostname = $location.host();
951
952     $scope.tests = [];
953     $scope.clearTestData = function(){
954         $scope.tests = [];
955         numPings = 0;
956     }
957
958     $scope.isTesting = false;
959     $scope.avrg = 0; // avrg latency
960     $scope.canCopyCommand = document.queryCommandSupported('copy');
961     var numPings = 0;
962     // initially fetch first 10 (gets a decent average)
963
964     function calcAverage(){
965
966         if ($scope.tests.length == 0) return 0;
967
968         if ($scope.tests.length == 1) return $scope.tests[0].l;
969
970         var sum = 0;
971         angular.forEach($scope.tests, function(t){
972             sum += t.l;
973         });
974
975         return sum / $scope.tests.length;
976     }
977
978     function ping(){
979         $scope.isTesting = true;
980         var t = Date.now();
981         return egCore.net.request(
982             "open-ils.pcrud", "opensrf.system.echo", "ping"
983         ).then(function(resp){
984             var t2 = Date.now();
985             var latency = t2 - t;
986             $scope.tests.push({t: new Date(t), l: latency});
987             console.log("Start: " + t + " and end: " + t2);
988             console.log("Latency: " + latency);
989             console.log(resp);
990         }).then(function(){
991             $scope.avrg = calcAverage();
992             numPings++;
993             $scope.isTesting = false;
994         });
995     }
996
997     $scope.testLatency = function(){
998
999         if (numPings >= 10){
1000             ping(); // just ping once after the initial ten
1001         } else {
1002             ping()
1003                 .then($scope.testLatency)
1004                 .then(function(){
1005                     if (numPings == 9){
1006                         $scope.tests.shift(); // toss first result
1007                         $scope.avrg = calcAverage();
1008                     }
1009                 });
1010         }
1011     }
1012
1013     $scope.copyTests = function(){
1014
1015         var lNode = document.querySelector('#pingData');
1016         var r = document.createRange();
1017         r.selectNode(lNode);
1018         var sel = window.getSelection();
1019         sel.removeAllRanges();
1020         sel.addRange(r);
1021         document.execCommand('copy');
1022     }
1023
1024 }])
1025
1026