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',
47 $routeProvider.when('/admin/workstation/tests', {
48 templateUrl: './admin/workstation/t_tests',
49 controller: 'testsCtrl',
54 $routeProvider.otherwise({
55 templateUrl : './admin/workstation/t_splash',
56 controller : 'SplashCtrl',
61 .config(['ngToastProvider', function(ngToastProvider) {
62 ngToastProvider.configure({
63 verticalPosition: 'bottom',
68 .factory('workstationSvc',
69 ['$q','$timeout','$location','egCore','egConfirmDialog',
70 function($q , $timeout , $location , egCore , egConfirmDialog) {
74 service.get_all = function() {
75 return egCore.hatch.getWorkstations()
76 .then(function(all) { return all || [] });
79 service.get_default = function() {
80 return egCore.hatch.getDefaultWorkstation();
83 service.set_default = function(name) {
84 return egCore.hatch.setDefaultWorkstation(name);
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);
94 service.register_ws_api =
95 function(base_name, name, org_id, override, deferred) {
96 if (!deferred) deferred = $q.defer();
98 var method = 'open-ils.actor.workstation.register';
99 if (override) method += '.override';
102 'open-ils.actor', method, egCore.auth.token(), name, org_id)
104 .then(function(resp) {
106 if (evt = egCore.evt.parse(resp)) {
107 console.log('register returned ' + evt.toString());
109 if (evt.textcode == 'WORKSTATION_NAME_EXISTS' && !override) {
111 egConfirmDialog.open(
112 egCore.strings.WS_EXISTS, base_name, {
114 service.register_ws_api(
115 base_name, name, org_id, true, deferred)
117 cancel : function() {
124 alert(evt.toString());
128 console.log('Resolving register promise with: ' + resp);
129 deferred.resolve(resp);
133 return deferred.promise;
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};
140 return service.get_all()
141 .then(function(all) {
143 return egCore.hatch.setWorkstations(all)
144 .then(function() { return new_ws });
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);
153 return egCore.hatch.getWorkstations()
155 // remove from list of all workstations
156 .then(function(all) {
158 var keep = all.filter(function(ws) {return ws.name != name});
159 return egCore.hatch.setWorkstations(keep);
163 return service.get_default()
165 }).then(function(def) {
167 console.debug('Removing default workstation: ' + name);
168 return egCore.hatch.removeDefaultWorkstation();
177 .controller('SplashCtrl',
178 ['$scope','$window','$location','egCore','egConfirmDialog',
179 function($scope , $window , $location , egCore , egConfirmDialog) {
181 egCore.hatch.getItem('eg.audio.disable').then(function(val) {
182 $scope.disable_sound = val;
185 egCore.hatch.getItem('eg.search.search_lib').then(function(val) {
186 $scope.search_lib = egCore.org.get(val);
188 $scope.handle_search_lib_changed = function(org) {
189 egCore.hatch.setItem('eg.search.search_lib', org.id());
192 egCore.hatch.getItem('eg.search.pref_lib').then(function(val) {
193 $scope.pref_lib = egCore.org.get(val);
195 $scope.handle_pref_lib_changed = function(org) {
196 egCore.hatch.setItem('eg.search.pref_lib', org.id());
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;
203 $scope.$watch('adv_pane', function(newVal, oldVal) {
204 if (typeof newVal != 'undefined' && newVal != oldVal) {
205 egCore.hatch.setItem('eg.search.adv_pane', newVal);
209 $scope.apply_sound = function() {
210 if ($scope.disable_sound) {
211 egCore.hatch.setItem('eg.audio.disable', true);
213 egCore.hatch.removeItem('eg.audio.disable');
217 $scope.test_audio = function(sound) {
218 egCore.audio.play(sound);
223 .controller('PrintConfigCtrl',
225 function($scope , egCore) {
227 $scope.printConfig = {};
228 $scope.setContext = function(ctx) {
229 $scope.context = ctx;
230 $scope.isTestView = false;
232 $scope.setContext('default');
234 $scope.setContentType = function(type) { $scope.contentType = type }
235 $scope.setContentType('text/plain');
237 var hatchPrinting = false;
238 egCore.hatch.usePrinting().then(function(answer) {
239 hatchPrinting = answer;
242 $scope.useHatchPrinting = function() {
243 return hatchPrinting;
246 $scope.hatchIsOpen = function() {
247 return egCore.hatch.hatchAvailable;
250 $scope.getPrinterByAttr = function(attr, value) {
252 angular.forEach($scope.printers, function(p) {
253 if (p[attr] == value) printer = p;
258 $scope.resetPrinterSettings = function(context) {
259 $scope.printConfig[context] = {
261 printer : $scope.defaultPrinter ? $scope.defaultPrinter.name : null,
268 $scope.savePrinterSettings = function(context) {
269 return egCore.hatch.setPrintConfig(
270 context, $scope.printConfig[context]);
273 $scope.printerConfString = function() {
274 if ($scope.printConfigError) return $scope.printConfigError;
275 if (!$scope.printConfig) return;
276 if (!$scope.printConfig[$scope.context]) return;
277 return JSON.stringify(
278 $scope.printConfig[$scope.context], undefined, 2);
281 function loadPrinterOptions(name) {
282 if (name == 'hatch_file_writer') {
283 $scope.printerOptions = {};
285 egCore.hatch.getPrinterOptions(name).then(
286 function(options) {$scope.printerOptions = options});
290 $scope.setPrinter = function(name) {
291 $scope.printConfig[$scope.context].printer = name;
292 loadPrinterOptions(name);
295 $scope.testPrint = function(withDialog) {
296 if ($scope.contentType == 'text/plain') {
298 context : $scope.context,
299 content_type : $scope.contentType,
300 content : $scope.textPrintContent,
301 show_dialog : withDialog
305 context : $scope.context,
306 content_type : $scope.contentType,
307 content : $scope.htmlPrintContent,
309 value1 : 'Value One',
310 value2 : 'Value Two',
311 date_value : '2015-02-04T14:04:34-0400'
313 show_dialog : withDialog
318 $scope.useFileWriter = function() {
320 $scope.printConfig[$scope.context] &&
321 $scope.printConfig[$scope.context].printer == 'hatch_file_writer'
325 // Load startup data....
326 // Don't bother talking to Hatch if it's not there.
327 if (!egCore.hatch.hatchAvailable) return;
329 // fetch info on all remote printers
330 egCore.hatch.getPrinters()
331 .then(function(printers) {
332 $scope.printers = printers;
335 // We need a static name for saving configs.
336 // Human-friendly label is set in the template.
337 name: 'hatch_file_writer'
340 var def = $scope.getPrinterByAttr('is-default', true);
341 if (!def && printers.length) def = printers[0];
344 $scope.defaultPrinter = def;
345 loadPrinterOptions(def.name);
349 ['default','receipt','label','mail','offline'],
351 egCore.hatch.getPrintConfig(ctx).then(function(conf) {
353 $scope.printConfig[ctx] = conf;
355 $scope.resetPrinterSettings(ctx);
364 .controller('PrintTemplatesCtrl',
365 ['$scope','$q','egCore','ngToast',
366 function($scope , $q , egCore , ngToast) {
369 template_name : 'bills_current',
370 template_output : '',
371 template_context : 'default'
374 // print preview scope data
375 // TODO: consider moving the template-specific bits directly
376 // into the templates or storing template- specific script files
377 // alongside the templates.
378 // NOTE: A lot of this data can be shared across templates.
381 first_given_name : 'Joseph',
382 second_given_name : 'Martin',
383 family_name : 'Jones',
385 pref_first_given_name : 'Martin',
386 pref_second_given_name : 'Joe',
387 pref_family_name : 'Smith',
389 barcode : '30393830393'
392 balance_owed : 4, // This is currently how these values are returned to the client
393 total_billed : '5.00',
396 expire_date : '2020-12-31',
400 dob : '1980-01-01T00:00:00-8:00',
402 usrname : '30393830393',
403 day_phone : '111-222-3333',
404 evening_phone : '222-333-1111',
405 other_phone : '333-111-2222',
406 email : 'user@example.com',
407 home_ou : {name: function() {return 'BR1'}},
408 profile : {name: function() {return 'Patrons'}},
409 net_access_level : {name: function() {return 'Filtered'}},
412 master_account : 'f',
413 claims_returned_count : '0',
414 claims_never_checked_out_count : '0',
415 alert_message : 'Coat is in the lost-and-found behind the circ desk',
416 ident_type: {name: function() {return 'Drivers License'}},
417 ident_value: '11332445',
418 ident_type2: {name: function() {return 'Other'}},
419 ident_value2 : '55442211',
423 stat_cat : {'name' : 'Favorite Donut'},
424 'stat_cat_entry' : 'Maple'
426 stat_cat : {'name' : 'Favorite Book'},
427 'stat_cat_entry' : 'Beasts Made of Night'
433 address_type : 'MAILING',
434 street1 : '123 Apple Rd',
437 county : 'Great County',
442 within_city_limits: 't'
445 seed_user.addresses.push(seed_addr);
448 title : 'Traveling Pants!!',
449 author : 'Jane Jones',
454 barcode : '33434322323',
459 'title' : 'Test Title'
463 name : 'Ankers Memorial Library',
471 name : 'General Collection'
476 // flattened versions for item status template
477 // TODO - make this go away
478 'call_number.label' : '636.8 JON',
479 'call_number.record.simple_record.title' : 'Test Title',
480 'location.name' : 'General Collection',
481 'call_number.owning_lib.name' : 'Ankers Memorial Library',
482 'call_number.owning_lib.shortname' : 'Ankers'
487 phone_notify : '111-222-3333',
488 sms_notify : '111-222-3333',
489 email_notify : 'user@example.org',
490 request_time : new Date().toISOString(),
492 shelf_expire_time : new Date().toISOString()
499 holds_address : seed_addr
504 holds_address : seed_addr
506 source_send_time : new Date().toISOString(),
507 target_copy : seed_copy
510 $scope.preview_scope = {
515 xact_start : new Date().toISOString(),
516 xact_finish : new Date().toISOString(),
519 prefix : "biography",
520 suffix : "Closed Stacks",
522 name : "Mineola Public Library",
523 shortname : "Mineola"
527 xact_type : 'circulation',
528 last_billing_type : 'Overdue materials',
530 last_payment_note : 'Test Note 1',
531 last_payment_type : 'cash_payment',
532 last_payment_ts : new Date().toISOString(),
538 xact_start : new Date().toISOString(),
539 xact_finish : new Date().toISOString(),
545 name : "Rogers Reading Room",
550 xact_type : 'circulation',
551 last_billing_type : 'Overdue materials',
553 last_payment_note : 'Test Note 2',
554 last_payment_type : 'credit_payment',
555 last_payment_ts : new Date().toISOString(),
563 copies : [ seed_copy ],
567 due_date : new Date().toISOString(),
570 target_copy : seed_copy,
571 copy_barcode : seed_copy.barcode,
572 call_number : seed_copy.call_number,
573 title : seed_record.title
580 due_date : new Date().toISOString(),
583 renewal_remaining : 2
586 title : seed_record.title,
587 author : seed_record.author
601 title : seed_record.title
605 previous_balance : 8.45,
606 payment_total : 2.00,
607 payment_applied : 2.00,
611 payment_type : 'cash_payment',
612 payment_note : 'Here is a payment note',
614 create_date : new Date().toISOString(),
615 title : 'Test Note Title',
617 value : 'This patron is super nice!'
620 transit : seed_transit,
621 transits : [ seed_transit ],
622 title : seed_record.title,
623 author : seed_record.author,
626 dest_location : egCore.idl.toHash(egCore.org.get(egCore.auth.user().ws_ou())),
627 dest_courier_code : 'ABC 123',
628 dest_address : seed_addr,
632 hold : one_hold, title : 'Some Title 1', author : 'Some Author 1',
633 volume : { label : '646.4 SOM' }, copy : seed_copy,
634 part : { label : 'v. 1' },
635 patron_barcode : 'S52802662',
636 patron_alias : 'XYZ', patron_last : 'Smith', patron_first : 'Jane',
637 status_string : 'Ready for Pickup'
640 hold : one_hold, title : 'Some Title 2', author : 'Some Author 2',
641 volume : { label : '646.4 SOM' }, copy : seed_copy,
642 part : { label : 'v. 1' },
643 patron_barcode : 'S52802662',
644 patron_alias : 'XYZ', patron_last : 'Smith', patron_first : 'Jane',
645 status_string : 'Ready for Pickup'
648 hold : one_hold, title : 'Some Title 3', author : 'Some Author 3',
649 volume : { label : '646.4 SOM' }, copy : seed_copy,
650 part : { label : 'v. 1' },
651 patron_barcode : 'S52802662',
652 patron_alias : 'XYZ', patron_last : 'Smith', patron_first : 'Jane',
653 status_string : 'Canceled'
658 $scope.preview_scope.payments = [
659 {amount : 1.00, xact : $scope.preview_scope.transactions[0]},
660 {amount : 1.00, xact : $scope.preview_scope.transactions[1]}
662 $scope.preview_scope.payments[0].xact.title = 'Hali Bote Azikaban de tao fan';
663 $scope.preview_scope.payments[0].xact.copy_barcode = '334343434';
664 $scope.preview_scope.payments[1].xact.title = seed_record.title;
665 $scope.preview_scope.payments[1].xact.copy_barcode = seed_copy.barcode;
667 // today, staff, current_location, etc.
668 egCore.print.fleshPrintScope($scope.preview_scope);
670 $scope.template_changed = function() {
671 $scope.print.load_failed = false;
672 egCore.print.getPrintTemplate($scope.print.template_name)
675 $scope.print.template_content = html;
676 console.log('set template content');
679 $scope.print.template_content = '';
680 $scope.print.load_failed = true;
683 egCore.print.getPrintTemplateContext($scope.print.template_name)
684 .then(function(template_context) {
685 $scope.print.template_context = template_context;
689 $scope.reset_to_default = function() {
690 egCore.print.removePrintTemplate(
691 $scope.print.template_name
693 egCore.print.removePrintTemplateContext(
694 $scope.print.template_name
696 $scope.template_changed();
699 $scope.save_locally = function() {
700 egCore.print.storePrintTemplate(
701 $scope.print.template_name,
702 $scope.print.template_content
704 egCore.print.storePrintTemplateContext(
705 $scope.print.template_name,
706 $scope.print.template_context
710 $scope.exportable_templates = function() {
713 var deferred = $q.defer();
715 egCore.hatch.getKeys('eg.print.template').then(function(keys) {
716 angular.forEach(keys, function(key) {
717 if (key.match(/^eg\.print\.template\./)) {
718 promises.push(egCore.hatch.getItem(key).then(function(value) {
719 templates[key.replace('eg.print.template.', '')] = value;
722 promises.push(egCore.hatch.getItem(key).then(function(value) {
723 contexts[key.replace('eg.print.template_context.', '')] = value;
727 $q.all(promises).then(function() {
728 if (Object.keys(templates).length) {
730 templates: templates,
734 ngToast.warning(egCore.strings.PRINT_TEMPLATES_FAIL_EXPORT);
739 return deferred.promise;
742 $scope.imported_print_templates = { data : '' };
743 $scope.$watch('imported_print_templates.data', function(newVal, oldVal) {
744 if (newVal && newVal != oldVal) {
746 var data = JSON.parse(newVal);
747 angular.forEach(data.templates, function(template_content, template_name) {
748 egCore.print.storePrintTemplate(template_name, template_content);
750 angular.forEach(data.contexts, function(template_context, template_name) {
751 egCore.print.storePrintTemplateContext(template_name, template_context);
753 $scope.template_changed(); // refresh
754 ngToast.create(egCore.strings.PRINT_TEMPLATES_SUCCESS_IMPORT);
756 ngToast.warning(egCore.strings.PRINT_TEMPLATES_FAIL_IMPORT);
761 $scope.template_changed(); // load the default
765 .directive('egPrintTemplateOutput', ['$compile',function($compile) {
766 return function(scope, element, attrs) {
769 return scope.$eval(attrs.content);
772 // create an isolate scope and copy the print context
773 // data into the new scope.
774 // TODO: see also print security concerns in egHatch
775 var result = element.html(value);
776 var context = scope.$eval(attrs.context);
777 var print_scope = scope.$new(true);
778 angular.forEach(context, function(val, key) {
779 print_scope[key] = val;
781 $compile(element.contents())(print_scope);
787 .controller('StoredPrefsCtrl',
788 ['$scope','$q','egCore','egConfirmDialog',
789 function($scope , $q , egCore , egConfirmDialog) {
790 console.log('StoredPrefsCtrl');
792 $scope.setContext = function(ctx) {
793 $scope.context = ctx;
795 $scope.setContext('local');
797 // grab the edit perm
798 $scope.userHasDeletePerm = false;
799 egCore.perm.hasPermHere('ADMIN_WORKSTATION')
800 .then(function(bool) { $scope.userHasDeletePerm = bool });
804 function refreshKeys() {
805 $scope.keys = {local : [], remote : [], server_workstation: []};
807 if (egCore.hatch.hatchAvailable) {
808 egCore.hatch.getRemoteKeys().then(
809 function(keys) { $scope.keys.remote = keys.sort() })
812 // local calls are non-async
813 $scope.keys.local = egCore.hatch.getLocalKeys();
815 egCore.hatch.getServerKeys(null, {workstation_only: true}).then(
816 function(keys) {$scope.keys.server_workstation = keys});
820 $scope.selectKey = function(key) {
821 $scope.currentKey = key;
822 $scope.currentKeyContent = null;
824 if ($scope.context == 'local') {
825 $scope.currentKeyContent = egCore.hatch.getLocalItem(key);
826 } else if ($scope.context == 'remote') {
827 egCore.hatch.getRemoteItem(key)
828 .then(function(content) {
829 $scope.currentKeyContent = content
831 } else if ($scope.context == 'server_workstation') {
832 egCore.hatch.getServerItem(key).then(function(content) {
833 $scope.currentKeyContent = content;
838 $scope.getCurrentKeyContent = function() {
839 return JSON.stringify($scope.currentKeyContent, null, 2);
842 $scope.removeKey = function(key) {
843 egConfirmDialog.open(
844 egCore.strings.PREFS_REMOVE_KEY_CONFIRM, '',
847 if ($scope.context == 'local') {
848 egCore.hatch.removeLocalItem(key);
850 } else if ($scope.context == 'remote') {
851 // Honor requests to remove items from Hatch even
852 // when Hatch is configured for data storage.
853 egCore.hatch.removeRemoteItem(key)
854 .then(function() { refreshKeys() });
855 } else if ($scope.context == 'server_workstation') {
856 egCore.hatch.removeServerItem(key)
857 .then(function() { refreshKeys() });
860 cancel : function() {} // user canceled, nothing to do
866 .controller('WSRegCtrl',
867 ['$scope','$q','$window','$location','egCore','egAlertDialog','workstationSvc',
868 function($scope , $q , $window , $location , egCore , egAlertDialog , workstationSvc) {
870 var all_workstations = [];
871 var reg_perm_orgs = [];
873 $scope.page_loaded = false;
874 $scope.contextOrg = egCore.org.get(egCore.auth.user().ws_ou());
875 $scope.wsOrgChanged = function(org) { $scope.contextOrg = org; }
877 console.log('set context org to ' + $scope.contextOrg);
879 egCore.hatch.hostname().then(function(name) {
880 $scope.newWSName = name || '';
883 // fetch workstation reg perms
884 egCore.perm.hasPermAt('REGISTER_WORKSTATION', true)
885 .then(function(orgList) {
886 reg_perm_orgs = orgList;
888 // hide orgs in the context org selector where this login
889 // does not have the reg_ws perm or the org can't have users
890 $scope.wsOrgHidden = function(id) {
891 return reg_perm_orgs.indexOf(id) == -1
892 || $scope.cant_have_users(id);
895 // fetch the locally stored workstation data
897 return workstationSvc.get_all()
899 }).then(function(all) {
900 all_workstations = all || [];
901 $scope.workstations =
902 all_workstations.map(function(w) { return w.name });
903 return workstationSvc.get_default()
905 // fetch the default workstation
906 }).then(function(def) {
907 $scope.defaultWS = def;
908 $scope.activeWS = $scope.selectedWS = egCore.auth.workstation() || def;
910 // Handle any URL commands.
912 var remove = $location.search().remove;
914 console.log('Removing WS via URL request: ' + remove);
915 return $scope.remove_ws(remove).then(
916 function() { $scope.page_loaded = true; });
918 $scope.page_loaded = true;
921 $scope.get_ws_label = function(ws) {
922 return ws == $scope.defaultWS ?
923 egCore.strings.$replace(egCore.strings.DEFAULT_WS_LABEL, {ws:ws}) : ws;
926 $scope.set_default_ws = function(name) {
927 delete $scope.removing_ws;
928 $scope.defaultWS = name;
929 workstationSvc.set_default(name);
932 $scope.cant_have_users =
933 function (id) { return !egCore.org.CanHaveUsers(id); };
934 $scope.cant_have_volumes =
935 function (id) { return !egCore.org.CanHaveVolumes(id); };
937 // Log out and return to login page with selected WS
938 $scope.use_now = function() {
939 egCore.auth.logout();
940 $window.location.href = $location
942 .search({ws : $scope.selectedWS})
946 $scope.can_delete_ws = function(name) {
947 var ws = all_workstations.filter(
948 function(ws) { return ws.name == name })[0];
949 return ws && reg_perm_orgs.indexOf(ws.owning_lib) != -1;
952 $scope.remove_ws = function(remove_me) {
953 $scope.removing_ws = remove_me;
955 // Perm is used to disable Remove button in UI, but have to check
956 // again here in case we're removing a WS based on URL params.
957 if (!$scope.can_delete_ws(remove_me)) return $q.when();
959 $scope.is_removing = true;
960 return workstationSvc.remove_workstation(remove_me)
963 all_workstations = all_workstations.filter(
964 function(ws) { return ws.name != remove_me });
966 $scope.workstations = $scope.workstations.filter(
967 function(ws) { return ws != remove_me });
969 if ($scope.selectedWS == remove_me)
970 $scope.selectedWS = $scope.workstations[0];
972 if ($scope.defaultWS == remove_me)
973 $scope.defaultWS = '';
975 $scope.is_removing = false;
979 $scope.register_ws = function() {
980 delete $scope.removing_ws;
983 $scope.contextOrg.shortname() + '-' + $scope.newWSName;
985 if ($scope.workstations.indexOf(full_name) > -1) {
986 // avoid duplicate local registrations
987 return egAlertDialog.open(egCore.strings.WS_USED);
990 $scope.is_registering = true;
991 workstationSvc.register_workstation(
992 $scope.newWSName, full_name,
993 $scope.contextOrg.id()
995 ).then(function(new_ws) {
996 $scope.workstations.push(new_ws.name);
997 all_workstations.push(new_ws);
998 $scope.is_registering = false;
1000 if (!$scope.selectedWS) {
1001 $scope.selectedWS = new_ws.name;
1003 if (!$scope.defaultWS) {
1004 return $scope.set_default_ws(new_ws.name);
1006 $scope.newWSName = '';
1008 $scope.is_registering = false;
1013 .controller('HatchCtrl',
1014 ['$scope','egCore','ngToast',
1015 function($scope , egCore , ngToast) {
1016 var hatch = egCore.hatch; // convenience
1018 $scope.hatch_available = hatch.hatchAvailable;
1019 $scope.hatch_settings = hatch.useSettings();
1020 $scope.hatch_offline = hatch.useOffline();
1022 hatch.usePrinting().then(function(answer) {
1023 $scope.hatch_printing = answer;
1026 // Apply Hatch settings as changes occur in the UI.
1028 $scope.$watch('hatch_printing', function(newval) {
1029 if (typeof newval != 'boolean') return;
1030 hatch.setItem('eg.hatch.enable.printing', newval);
1033 $scope.$watch('hatch_settings', function(newval) {
1034 if (typeof newval != 'boolean') return;
1035 hatch.setLocalItem('eg.hatch.enable.settings', newval);
1038 $scope.$watch('hatch_offline', function(newval) {
1039 if (typeof newval != 'boolean') return;
1040 hatch.setLocalItem('eg.hatch.enable.offline', newval);
1043 $scope.copy_to_hatch = function() {
1044 hatch.copySettingsToHatch().then(
1046 ngToast.create(egCore.strings.HATCH_SETTINGS_MIGRATION_SUCCESS)},
1048 ngToast.warning(egCore.strings.HATCH_SETTINGS_MIGRATION_FAILURE)}
1052 $scope.copy_to_local = function() {
1053 hatch.copySettingsToLocal().then(
1055 ngToast.create(egCore.strings.HATCH_SETTINGS_MIGRATION_SUCCESS)},
1057 ngToast.warning(egCore.strings.HATCH_SETTINGS_MIGRATION_FAILURE)}
1064 * Home of the Latency tester
1066 .controller('testsCtrl', ['$scope', '$location', 'egCore', function($scope, $location, egCore) {
1067 $scope.hostname = $location.host();
1070 $scope.clearTestData = function(){
1075 $scope.isTesting = false;
1076 $scope.avrg = 0; // avrg latency
1077 $scope.canCopyCommand = document.queryCommandSupported('copy');
1079 // initially fetch first 10 (gets a decent average)
1081 function calcAverage(){
1083 if ($scope.tests.length == 0) return 0;
1085 if ($scope.tests.length == 1) return $scope.tests[0].l;
1088 angular.forEach($scope.tests, function(t){
1092 return sum / $scope.tests.length;
1096 $scope.isTesting = true;
1098 return egCore.net.request(
1099 "open-ils.pcrud", "opensrf.system.echo", "ping"
1100 ).then(function(resp){
1101 var t2 = Date.now();
1102 var latency = t2 - t;
1103 $scope.tests.push({t: new Date(t), l: latency});
1104 console.log("Start: " + t + " and end: " + t2);
1105 console.log("Latency: " + latency);
1108 $scope.avrg = calcAverage();
1110 $scope.isTesting = false;
1114 $scope.testLatency = function(){
1116 if (numPings >= 10){
1117 ping(); // just ping once after the initial ten
1120 .then($scope.testLatency)
1123 $scope.tests.shift(); // toss first result
1124 $scope.avrg = calcAverage();
1130 $scope.copyTests = function(){
1132 var lNode = document.querySelector('#pingData');
1133 var r = document.createRange();
1134 r.selectNode(lNode);
1135 var sel = window.getSelection();
1136 sel.removeAllRanges();
1138 document.execCommand('copy');