2 * App to drive the base page.
7 angular.module('egWorkstationAdmin',
8 ['ngRoute', 'ui.bootstrap', 'egCoreMod', 'egUiMod'])
10 .config(['$routeProvider','$locationProvider','$compileProvider',
11 function($routeProvider , $locationProvider , $compileProvider) {
13 $locationProvider.html5Mode(true);
14 $compileProvider.aHrefSanitizationWhitelist(/^\s*(https?|mailto|blob):/);
15 var resolver = {delay : function(egStartup) {return egStartup.go()}};
17 $routeProvider.when('/admin/workstation/workstations', {
18 templateUrl: './admin/workstation/t_workstations',
19 controller: 'WSRegCtrl',
23 $routeProvider.when('/admin/workstation/print/config', {
24 templateUrl: './admin/workstation/t_print_config',
25 controller: 'PrintConfigCtrl',
29 $routeProvider.when('/admin/workstation/print/templates', {
30 templateUrl: './admin/workstation/t_print_templates',
31 controller: 'PrintTemplatesCtrl',
35 $routeProvider.when('/admin/workstation/stored_prefs', {
36 templateUrl: './admin/workstation/t_stored_prefs',
37 controller: 'StoredPrefsCtrl',
41 $routeProvider.when('/admin/workstation/hatch', {
42 templateUrl: './admin/workstation/t_hatch',
43 controller: 'HatchCtrl',
48 $routeProvider.otherwise({
49 templateUrl : './admin/workstation/t_splash',
50 controller : 'SplashCtrl',
55 .config(['ngToastProvider', function(ngToastProvider) {
56 ngToastProvider.configure({
57 verticalPosition: 'bottom',
62 .factory('workstationSvc',
63 ['$q','$timeout','$location','egCore','egConfirmDialog',
64 function($q , $timeout , $location , egCore , egConfirmDialog) {
68 service.get_all = function() {
69 return egCore.hatch.getItem('eg.workstation.all')
70 .then(function(all) { return all || [] });
73 service.get_default = function() {
74 return egCore.hatch.getItem('eg.workstation.default');
77 service.set_default = function(name) {
78 return egCore.hatch.setItem('eg.workstation.default', name);
81 service.register_workstation = function(base_name, name, org_id) {
82 return service.register_ws_api(base_name, name, org_id)
83 .then(function(ws_id) {
84 return service.track_new_ws(ws_id, name, org_id);
88 service.register_ws_api =
89 function(base_name, name, org_id, override, deferred) {
90 if (!deferred) deferred = $q.defer();
92 var method = 'open-ils.actor.workstation.register';
93 if (override) method += '.override';
96 'open-ils.actor', method, egCore.auth.token(), name, org_id)
98 .then(function(resp) {
100 if (evt = egCore.evt.parse(resp)) {
101 console.log('register returned ' + evt.toString());
103 if (evt.textcode == 'WORKSTATION_NAME_EXISTS' && !override) {
105 egConfirmDialog.open(
106 egCore.strings.WS_EXISTS, base_name, {
108 service.register_ws_api(
109 base_name, name, org_id, true, deferred)
111 cancel : function() {
118 alert(evt.toString());
122 console.log('Resolving register promise with: ' + resp);
123 deferred.resolve(resp);
127 return deferred.promise;
130 service.track_new_ws = function(ws_id, ws_name, owning_lib) {
131 console.log('Tracking newly created WS with ID ' + ws_id);
132 var new_ws = {id : ws_id, name : ws_name, owning_lib : owning_lib};
134 return service.get_all()
135 .then(function(all) {
137 return egCore.hatch.setItem('eg.workstation.all', all)
138 .then(function() { return new_ws });
142 // Remove all traces of the workstation locally.
143 // This does not remove the WS from the server.
144 service.remove_workstation = function(name) {
145 console.debug('Removing workstation: ' + name);
147 return egCore.hatch.getItem('eg.workstation.all')
149 // remove from list of all workstations
150 .then(function(all) {
152 var keep = all.filter(function(ws) {return ws.name != name});
153 return egCore.hatch.setItem('eg.workstation.all', keep)
157 return service.get_default()
159 }).then(function(def) {
161 console.debug('Removing default workstation: ' + name);
162 return egCore.hatch.removeItem('eg.workstation.default');
171 .controller('SplashCtrl',
172 ['$scope','$window','$location','egCore','egConfirmDialog',
173 function($scope , $window , $location , egCore , egConfirmDialog) {
175 egCore.hatch.getItem('eg.audio.disable').then(function(val) {
176 $scope.disable_sound = val;
179 egCore.hatch.getItem('eg.search.search_lib').then(function(val) {
180 $scope.search_lib = egCore.org.get(val);
182 $scope.handle_search_lib_changed = function(org) {
183 egCore.hatch.setItem('eg.search.search_lib', org.id());
186 egCore.hatch.getItem('eg.search.pref_lib').then(function(val) {
187 $scope.pref_lib = egCore.org.get(val);
189 $scope.handle_pref_lib_changed = function(org) {
190 egCore.hatch.setItem('eg.search.pref_lib', org.id());
193 $scope.adv_pane = 'advanced'; // default value if not explicitly set
194 egCore.hatch.getItem('eg.search.adv_pane').then(function(val) {
195 $scope.adv_pane = val;
197 $scope.$watch('adv_pane', function(newVal, oldVal) {
198 if (typeof newVal != 'undefined' && newVal != oldVal) {
199 egCore.hatch.setItem('eg.search.adv_pane', newVal);
203 $scope.apply_sound = function() {
204 if ($scope.disable_sound) {
205 egCore.hatch.setItem('eg.audio.disable', true);
207 egCore.hatch.removeItem('eg.audio.disable');
211 $scope.test_audio = function(sound) {
212 egCore.audio.play(sound);
217 .controller('PrintConfigCtrl',
219 function($scope , egCore) {
221 $scope.printConfig = {};
222 $scope.setContext = function(ctx) {
223 $scope.context = ctx;
224 $scope.isTestView = false;
226 $scope.setContext('default');
228 $scope.setContentType = function(type) { $scope.contentType = type }
229 $scope.setContentType('text/plain');
231 $scope.useHatchPrinting = function() {
232 return egCore.hatch.usePrinting();
235 $scope.hatchIsOpen = function() {
236 return egCore.hatch.hatchAvailable;
239 $scope.getPrinterByAttr = function(attr, value) {
241 angular.forEach($scope.printers, function(p) {
242 if (p[attr] == value) printer = p;
247 $scope.resetPrinterSettings = function(context) {
248 $scope.printConfig[context] = {
250 printer : $scope.defaultPrinter ? $scope.defaultPrinter.name : null,
257 $scope.savePrinterSettings = function(context) {
258 return egCore.hatch.setPrintConfig(
259 context, $scope.printConfig[context]);
262 $scope.printerConfString = function() {
263 if ($scope.printConfigError) return $scope.printConfigError;
264 if (!$scope.printConfig) return;
265 if (!$scope.printConfig[$scope.context]) return;
266 return JSON.stringify(
267 $scope.printConfig[$scope.context], undefined, 2);
270 function loadPrinterOptions(name) {
271 egCore.hatch.getPrinterOptions(name).then(
272 function(options) {$scope.printerOptions = options});
275 $scope.setPrinter = function(name) {
276 $scope.printConfig[$scope.context].printer = name;
277 loadPrinterOptions(name);
280 $scope.testPrint = function(withDialog) {
281 if ($scope.contentType == 'text/plain') {
283 context : $scope.context,
284 content_type : $scope.contentType,
285 content : $scope.textPrintContent,
286 show_dialog : withDialog
290 context : $scope.context,
291 content_type : $scope.contentType,
292 content : $scope.htmlPrintContent,
294 value1 : 'Value One',
295 value2 : 'Value Two',
296 date_value : '2015-02-04T14:04:34-0400'
298 show_dialog : withDialog
303 // Load startup data....
304 // Don't bother talking to Hatch if it's not there.
305 if (!egCore.hatch.hatchAvailable) return;
307 // fetch info on all remote printers
308 egCore.hatch.getPrinters()
309 .then(function(printers) {
310 $scope.printers = printers;
312 var def = $scope.getPrinterByAttr('is-default', true);
313 if (!def && printers.length) def = printers[0];
316 $scope.defaultPrinter = def;
317 loadPrinterOptions(def.name);
321 ['default','receipt','label','mail','offline'],
323 egCore.hatch.getPrintConfig(ctx).then(function(conf) {
325 $scope.printConfig[ctx] = conf;
327 $scope.resetPrinterSettings(ctx);
336 .controller('PrintTemplatesCtrl',
337 ['$scope','$q','egCore','ngToast',
338 function($scope , $q , egCore , ngToast) {
341 template_name : 'bills_current',
342 template_output : '',
343 template_context : 'default'
346 // print preview scope data
347 // TODO: consider moving the template-specific bits directly
348 // into the templates or storing template- specific script files
349 // alongside the templates.
350 // NOTE: A lot of this data can be shared across templates.
352 first_given_name : 'Slow',
353 second_given_name : 'Joe',
354 family_name : 'Jones',
356 barcode : '30393830393'
360 street1 : '123 Apple Rd',
369 title : 'Traveling Pants!!',
370 author : 'Jane Jones',
375 barcode : '33434322323',
380 'title' : 'Test Title'
385 name : 'General Collection'
387 // flattened versions for item status template
388 // TODO - make this go away
389 'call_number.label' : '636.8 JON',
390 'call_number.record.simple_record.title' : 'Test Title',
391 'location.name' : 'General Colleciton'
396 phone_notify : '111-222-3333',
397 sms_notify : '111-222-3333',
398 email_notify : 'user@example.org',
399 request_time : new Date().toISOString(),
401 shelf_expire_time : new Date().toISOString()
408 holds_address : seed_addr
413 holds_address : seed_addr
415 source_send_time : new Date().toISOString(),
416 target_copy : seed_copy
419 $scope.preview_scope = {
424 xact_start : new Date().toISOString(),
426 xact_type : 'circulation',
427 last_billing_type : 'Overdue materials',
429 last_payment_note : 'Test Note 1',
430 last_payment_type : 'cash_payment',
436 xact_start : new Date().toISOString(),
438 xact_type : 'circulation',
439 last_billing_type : 'Overdue materials',
441 last_payment_note : 'Test Note 2',
442 last_payment_type : 'credit_payment',
450 copies : [ seed_copy ],
454 due_date : new Date().toISOString(),
457 target_copy : seed_copy,
458 copy_barcode : seed_copy.barcode,
459 call_number : seed_copy.call_number,
460 title : seed_record.title
467 due_date : new Date().toISOString(),
472 title : seed_record.title,
473 author : seed_record.author
487 title : seed_record.title
491 previous_balance : 8.45,
492 payment_total : 2.00,
493 payment_applied : 2.00,
497 payment_type : 'cash_payment',
498 payment_note : 'Here is a payment note',
500 create_date : new Date().toISOString(),
501 title : 'Test Note Title',
503 value : 'This patron is super nice!'
506 transit : seed_transit,
507 transits : [ seed_transit ],
508 title : seed_record.title,
509 author : seed_record.author,
512 dest_location : egCore.idl.toHash(egCore.org.get(egCore.auth.user().ws_ou())),
513 dest_address : seed_addr,
517 hold : one_hold, title : 'Some Title 1', author : 'Some Author 1',
518 volume : { label : '646.4 SOM' }, copy : seed_copy,
519 part : { label : 'v. 1' },
520 patron_barcode : 'S52802662',
521 patron_alias : 'XYZ', patron_last : 'Smith', patron_first : 'Jane',
522 status_string : 'Ready for Pickup'
525 hold : one_hold, title : 'Some Title 2', author : 'Some Author 2',
526 volume : { label : '646.4 SOM' }, copy : seed_copy,
527 part : { label : 'v. 1' },
528 patron_barcode : 'S52802662',
529 patron_alias : 'XYZ', patron_last : 'Smith', patron_first : 'Jane',
530 status_string : 'Ready for Pickup'
533 hold : one_hold, title : 'Some Title 3', author : 'Some Author 3',
534 volume : { label : '646.4 SOM' }, copy : seed_copy,
535 part : { label : 'v. 1' },
536 patron_barcode : 'S52802662',
537 patron_alias : 'XYZ', patron_last : 'Smith', patron_first : 'Jane',
538 status_string : 'In Transit'
543 $scope.preview_scope.payments = [
544 {amount : 1.00, xact : $scope.preview_scope.transactions[0]},
545 {amount : 1.00, xact : $scope.preview_scope.transactions[1]}
547 $scope.preview_scope.payments[0].xact.title = 'Hali Bote Azikaban de tao fan';
548 $scope.preview_scope.payments[0].xact.copy_barcode = '334343434';
549 $scope.preview_scope.payments[1].xact.title = seed_record.title;
550 $scope.preview_scope.payments[1].xact.copy_barcode = seed_copy.barcode;
552 // today, staff, current_location, etc.
553 egCore.print.fleshPrintScope($scope.preview_scope);
555 $scope.template_changed = function() {
556 $scope.print.load_failed = false;
557 egCore.print.getPrintTemplate($scope.print.template_name)
560 $scope.print.template_content = html;
561 console.log('set template content');
564 $scope.print.template_content = '';
565 $scope.print.load_failed = true;
568 egCore.print.getPrintTemplateContext($scope.print.template_name)
569 .then(function(template_context) {
570 $scope.print.template_context = template_context;
574 $scope.reset_to_default = function() {
575 egCore.print.removePrintTemplate(
576 $scope.print.template_name
578 egCore.print.removePrintTemplateContext(
579 $scope.print.template_name
581 $scope.template_changed();
584 $scope.save_locally = function() {
585 egCore.print.storePrintTemplate(
586 $scope.print.template_name,
587 $scope.print.template_content
589 egCore.print.storePrintTemplateContext(
590 $scope.print.template_name,
591 $scope.print.template_context
595 $scope.exportable_templates = function() {
598 var deferred = $q.defer();
600 egCore.hatch.getKeys('eg.print.template').then(function(keys) {
601 angular.forEach(keys, function(key) {
602 if (key.match(/^eg\.print\.template\./)) {
603 promises.push(egCore.hatch.getItem(key).then(function(value) {
604 templates[key.replace('eg.print.template.', '')] = value;
607 promises.push(egCore.hatch.getItem(key).then(function(value) {
608 contexts[key.replace('eg.print.template_context.', '')] = value;
612 $q.all(promises).then(function() {
613 if (Object.keys(templates).length) {
615 templates: templates,
619 ngToast.warning(egCore.strings.PRINT_TEMPLATES_FAIL_EXPORT);
624 return deferred.promise;
627 $scope.imported_print_templates = { data : '' };
628 $scope.$watch('imported_print_templates.data', function(newVal, oldVal) {
629 if (newVal && newVal != oldVal) {
631 var data = JSON.parse(newVal);
632 angular.forEach(data.templates, function(template_content, template_name) {
633 egCore.print.storePrintTemplate(template_name, template_content);
635 angular.forEach(data.contexts, function(template_context, template_name) {
636 egCore.print.storePrintTemplateContext(template_name, template_context);
638 $scope.template_changed(); // refresh
639 ngToast.create(egCore.strings.PRINT_TEMPLATES_SUCCESS_IMPORT);
641 ngToast.warning(egCore.strings.PRINT_TEMPLATES_FAIL_IMPORT);
646 $scope.template_changed(); // load the default
650 .directive('egPrintTemplateOutput', ['$compile',function($compile) {
651 return function(scope, element, attrs) {
654 return scope.$eval(attrs.content);
657 // create an isolate scope and copy the print context
658 // data into the new scope.
659 // TODO: see also print security concerns in egHatch
660 var result = element.html(value);
661 var context = scope.$eval(attrs.context);
662 var print_scope = scope.$new(true);
663 angular.forEach(context, function(val, key) {
664 print_scope[key] = val;
666 $compile(element.contents())(print_scope);
672 .controller('StoredPrefsCtrl',
673 ['$scope','$q','egCore','egConfirmDialog',
674 function($scope , $q , egCore , egConfirmDialog) {
675 console.log('StoredPrefsCtrl');
677 $scope.setContext = function(ctx) {
678 $scope.context = ctx;
680 $scope.setContext('local');
682 // grab the edit perm
683 $scope.userHasDeletePerm = false;
684 egCore.perm.hasPermHere('ADMIN_WORKSTATION')
685 .then(function(bool) { $scope.userHasDeletePerm = bool });
689 function refreshKeys() {
690 $scope.keys = {local : [], remote : []};
692 if (egCore.hatch.hatchAvailable) {
693 egCore.hatch.getRemoteKeys().then(
694 function(keys) { $scope.keys.remote = keys.sort() })
697 // local calls are non-async
698 $scope.keys.local = egCore.hatch.getLocalKeys();
702 $scope.selectKey = function(key) {
703 $scope.currentKey = key;
704 $scope.currentKeyContent = null;
706 if ($scope.context == 'local') {
707 $scope.currentKeyContent = egCore.hatch.getLocalItem(key);
709 egCore.hatch.getRemoteItem(key)
710 .then(function(content) {
711 $scope.currentKeyContent = content
716 $scope.getCurrentKeyContent = function() {
717 return JSON.stringify($scope.currentKeyContent, null, 2);
720 $scope.removeKey = function(key) {
721 egConfirmDialog.open(
722 egCore.strings.PREFS_REMOVE_KEY_CONFIRM, '',
725 if ($scope.context == 'local') {
726 egCore.hatch.removeLocalItem(key);
729 egCore.hatch.removeItem(key)
730 .then(function() { refreshKeys() });
733 cancel : function() {} // user canceled, nothing to do
739 .controller('WSRegCtrl',
740 ['$scope','$q','$window','$location','egCore','egAlertDialog','workstationSvc',
741 function($scope , $q , $window , $location , egCore , egAlertDialog , workstationSvc) {
743 var all_workstations = [];
744 var reg_perm_orgs = [];
746 $scope.page_loaded = false;
747 $scope.contextOrg = egCore.org.get(egCore.auth.user().ws_ou());
748 $scope.wsOrgChanged = function(org) { $scope.contextOrg = org; }
750 console.log('set context org to ' + $scope.contextOrg);
752 // fetch workstation reg perms
753 egCore.perm.hasPermAt('REGISTER_WORKSTATION', true)
754 .then(function(orgList) {
755 reg_perm_orgs = orgList;
757 // hide orgs in the context org selector where this login
758 // does not have the reg_ws perm or the org can't have users
759 $scope.wsOrgHidden = function(id) {
760 return reg_perm_orgs.indexOf(id) == -1
761 || $scope.cant_have_users(id);
764 // fetch the locally stored workstation data
766 return workstationSvc.get_all()
768 }).then(function(all) {
769 all_workstations = all || [];
770 $scope.workstations =
771 all_workstations.map(function(w) { return w.name });
772 return workstationSvc.get_default()
774 // fetch the default workstation
775 }).then(function(def) {
776 $scope.defaultWS = def;
777 $scope.activeWS = $scope.selectedWS = egCore.auth.workstation() || def;
779 // Handle any URL commands.
781 var remove = $location.search().remove;
783 console.log('Removing WS via URL request: ' + remove);
784 return $scope.remove_ws(remove).then(
785 function() { $scope.page_loaded = true; });
787 $scope.page_loaded = true;
790 $scope.get_ws_label = function(ws) {
791 return ws == $scope.defaultWS ?
792 egCore.strings.$replace(egCore.strings.DEFAULT_WS_LABEL, {ws:ws}) : ws;
795 $scope.set_default_ws = function(name) {
796 delete $scope.removing_ws;
797 $scope.defaultWS = name;
798 workstationSvc.set_default(name);
801 $scope.cant_have_users =
802 function (id) { return !egCore.org.CanHaveUsers(id); };
803 $scope.cant_have_volumes =
804 function (id) { return !egCore.org.CanHaveVolumes(id); };
806 // Log out and return to login page with selected WS
807 $scope.use_now = function() {
808 egCore.auth.logout();
809 $window.location.href = $location
811 .search({ws : $scope.selectedWS})
815 $scope.can_delete_ws = function(name) {
816 var ws = all_workstations.filter(
817 function(ws) { return ws.name == name })[0];
818 return ws && reg_perm_orgs.indexOf(ws.owning_lib) != -1;
821 $scope.remove_ws = function(remove_me) {
822 $scope.removing_ws = remove_me;
824 // Perm is used to disable Remove button in UI, but have to check
825 // again here in case we're removing a WS based on URL params.
826 if (!$scope.can_delete_ws(remove_me)) return $q.when();
828 $scope.is_removing = true;
829 return workstationSvc.remove_workstation(remove_me)
832 all_workstations = all_workstations.filter(
833 function(ws) { return ws.name != remove_me });
835 $scope.workstations = $scope.workstations.filter(
836 function(ws) { return ws != remove_me });
838 if ($scope.selectedWS == remove_me)
839 $scope.selectedWS = $scope.workstations[0];
841 if ($scope.defaultWS == remove_me)
842 $scope.defaultWS = '';
844 $scope.is_removing = false;
848 $scope.register_ws = function() {
849 delete $scope.removing_ws;
852 $scope.contextOrg.shortname() + '-' + $scope.newWSName;
854 if ($scope.workstations.indexOf(full_name) > -1) {
855 // avoid duplicate local registrations
856 return egAlertDialog.open(egCore.strings.WS_USED);
859 $scope.is_registering = true;
860 workstationSvc.register_workstation(
861 $scope.newWSName, full_name,
862 $scope.contextOrg.id()
864 ).then(function(new_ws) {
865 $scope.workstations.push(new_ws.name);
866 all_workstations.push(new_ws);
867 $scope.is_registering = false;
869 if (!$scope.selectedWS) {
870 $scope.selectedWS = new_ws.name;
872 if (!$scope.defaultWS) {
873 return $scope.set_default_ws(new_ws.name);
875 $scope.newWSName = '';
877 $scope.is_registering = false;
882 .controller('HatchCtrl',
883 ['$scope','egCore','ngToast',
884 function($scope , egCore , ngToast) {
885 var hatch = egCore.hatch; // convenience
887 $scope.hatch_available = hatch.hatchAvailable;
888 $scope.hatch_printing = hatch.usePrinting();
889 $scope.hatch_settings = hatch.useSettings();
890 $scope.hatch_offline = hatch.useOffline();
892 // Apply Hatch settings as changes occur in the UI.
894 $scope.$watch('hatch_printing', function(newval) {
895 if (typeof newval != 'boolean') return;
896 hatch.setLocalItem('eg.hatch.enable.printing', newval);
899 $scope.$watch('hatch_settings', function(newval) {
900 if (typeof newval != 'boolean') return;
901 hatch.setLocalItem('eg.hatch.enable.settings', newval);
904 $scope.$watch('hatch_offline', function(newval) {
905 if (typeof newval != 'boolean') return;
906 hatch.setLocalItem('eg.hatch.enable.offline', newval);
909 $scope.copy_to_hatch = function() {
910 hatch.copySettingsToHatch().then(
912 ngToast.create(egCore.strings.HATCH_SETTINGS_MIGRATION_SUCCESS)},
914 ngToast.warning(egCore.strings.HATCH_SETTINGS_MIGRATION_FAILURE)}
918 $scope.copy_to_local = function() {
919 hatch.copySettingsToLocal().then(
921 ngToast.create(egCore.strings.HATCH_SETTINGS_MIGRATION_SUCCESS)},
923 ngToast.warning(egCore.strings.HATCH_SETTINGS_MIGRATION_FAILURE)}