From 9056153e76ca0c8e9a7a21771dc600c6232fe69f Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Thu, 6 Dec 2018 17:33:04 -0300 Subject: [PATCH 001/606] Create feature model --- backend/models/__init__.py | 3 ++- backend/models/feature.py | 34 ++++++++++++++++++++++++++++++++++ frontend/sw.js | 2 +- 3 files changed, 37 insertions(+), 2 deletions(-) create mode 100644 backend/models/feature.py diff --git a/backend/models/__init__.py b/backend/models/__init__.py index aed79fcab..7cbe1f8b0 100644 --- a/backend/models/__init__.py +++ b/backend/models/__init__.py @@ -18,6 +18,7 @@ from .post import * from .survey_post import * from .factory_post import * +from .feature import * models = [ @@ -25,7 +26,7 @@ invite_institution_children, invite_institution_parent, invite_user, request, request_user, request_institution_parent, request_institution_children, invite_user_adm, request_institution, factory_invites, post, - survey_post, factory_post + survey_post, factory_post, feature ] __all__ = [prop for model in models for prop in model.__all__] diff --git a/backend/models/feature.py b/backend/models/feature.py new file mode 100644 index 000000000..153427a69 --- /dev/null +++ b/backend/models/feature.py @@ -0,0 +1,34 @@ +"""Feature model.""" +from google.appengine.ext import ndb + +__all__ = ['Feature'] + +class Feature(ndb.Model): + name = ndb.StringProperty() + enabled = ndb.BooleanProperty() + + @staticmethod + def create(name, enable): + feature = Feature(id=name, name=name, enabled=enable) + feature.put() + return feature + + @staticmethod + def enable(feature_name, enabled): + feature = Feature.get_by_id(feature_name) + + if feature: + feature.enabled = enabled + feature.put() + return feature + else: + raise Exception("Feature not found!") + + @staticmethod + def isEnabled(feature_name): + feature = Feature.get_by_id(feature_name) + + if feature: + return feature.enabled + else: + raise Exception("Feature not found!") \ No newline at end of file diff --git a/frontend/sw.js b/frontend/sw.js index 4bca0fe8a..3d30b829c 100644 --- a/frontend/sw.js +++ b/frontend/sw.js @@ -7,7 +7,7 @@ importScripts('app/firebase-config.js'); importScripts('app/config.js'); // if the line number of the code below changes, modify the /ecis script. - const CACHE_SUFIX = 'master'; + const CACHE_SUFIX = 'create-feature-toggles'; let messaging; From 1ea3a86fbb38a0cff9fa8d60201151a2e7f34a9c Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Fri, 7 Dec 2018 16:43:04 -0300 Subject: [PATCH 002/606] Create feature toggle handler --- backend/handlers/__init__.py | 3 +- backend/handlers/feature_toggle_handler.py | 35 ++++++++++++++++++++++ backend/main.py | 2 ++ backend/models/feature.py | 22 +++++++------- 4 files changed, 51 insertions(+), 11 deletions(-) create mode 100644 backend/handlers/feature_toggle_handler.py diff --git a/backend/handlers/__init__.py b/backend/handlers/__init__.py index 184916a15..236a06b38 100644 --- a/backend/handlers/__init__.py +++ b/backend/handlers/__init__.py @@ -42,6 +42,7 @@ from .invite_user_handler import * from .institution_parent_handler import * from .institution_children_handler import * +from .feature_toggle_handler import * handlers = [ base_handler, erro_handler, event_collection_handler, event_handler, @@ -61,7 +62,7 @@ user_request_collection_handler, user_timeline_handler, vote_handler, invite_hierarchy_collection_handler, invite_user_collection_handler, invite_institution_handler, invite_user_handler, institution_parent_handler, - institution_children_handler + institution_children_handler, feature_toggle_handler ] __all__ = [prop for handler in handlers for prop in handler.__all__] diff --git a/backend/handlers/feature_toggle_handler.py b/backend/handlers/feature_toggle_handler.py new file mode 100644 index 000000000..9130b58cb --- /dev/null +++ b/backend/handlers/feature_toggle_handler.py @@ -0,0 +1,35 @@ +"""Feature Toogle handler.""" + +import json +from . import BaseHandler +from utils import json_response +from util import login_required +from models import Feature + + +__all__ = ['FeatureToggleHander'] + +def to_json(feature_list): + features = { + feature.name: { + "enabled": feature.enabled, + "group": feature.group + } for feature in feature_list + } + + return features + +class FeatureToggleHander(BaseHandler): + + @login_required + @json_response + def get(self, user): + features = Feature.query().fetch() + self.response.write(json.dumps(to_json(features))) + + @login_required + @json_response + def put(self, user): + features_body = json.loads(self.request.body) + features = Feature.enable_all(features_body) + self.response.write(json.dumps(to_json(features))) \ No newline at end of file diff --git a/backend/main.py b/backend/main.py index 4bbdb7f86..3d6afff84 100644 --- a/backend/main.py +++ b/backend/main.py @@ -45,6 +45,7 @@ from handlers import InviteHandler from handlers import InstitutionParentHandler from handlers import InstitutionChildrenHandler +from handlers import FeatureToggleHander methods = set(webapp2.WSGIApplication.allowed_methods) methods.add('PATCH') @@ -96,6 +97,7 @@ ("/api/user/institutions/(.*)", UserHandler), ("/api/user/timeline.*", UserTimelineHandler), ("/api/search/institution", SearchHandler), + ("/api/feature-toggle", FeatureToggleHander), ("/login", LoginHandler), ("/logout", LogoutHandler), ("/api/.*", ErroHandler) diff --git a/backend/models/feature.py b/backend/models/feature.py index 153427a69..b3b21107c 100644 --- a/backend/models/feature.py +++ b/backend/models/feature.py @@ -6,23 +6,25 @@ class Feature(ndb.Model): name = ndb.StringProperty() enabled = ndb.BooleanProperty() + group = ndb.StringProperty( + choices=set(["ADMIN", "COMMOM", "ALL"])) @staticmethod - def create(name, enable): - feature = Feature(id=name, name=name, enabled=enable) + def create(name, enable, group="ALL"): + feature = Feature(id=name, name=name, enabled=enable, group=group) feature.put() return feature @staticmethod - def enable(feature_name, enabled): - feature = Feature.get_by_id(feature_name) + def enable_all(features_dict): + features = Feature.query(Feature.name.IN(features_dict.keys())).fetch() - if feature: - feature.enabled = enabled - feature.put() - return feature - else: - raise Exception("Feature not found!") + for feature in features: + feature.enabled = features_dict[feature.name]['enabled'] + feature.group = features_dict[feature.name]['group'] + + ndb.put_multi(features) + return features @staticmethod def isEnabled(feature_name): From c04ee9d6bebee76ce2bc2dac95e1ab62aa5311ca Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Fri, 7 Dec 2018 17:30:52 -0300 Subject: [PATCH 003/606] Create feature toggle service --- backend/handlers/feature_toggle_handler.py | 2 +- backend/models/feature.py | 2 +- frontend/utils/featureToggleService.js | 13 +++++++++++++ 3 files changed, 15 insertions(+), 2 deletions(-) create mode 100644 frontend/utils/featureToggleService.js diff --git a/backend/handlers/feature_toggle_handler.py b/backend/handlers/feature_toggle_handler.py index 9130b58cb..310e1bec9 100644 --- a/backend/handlers/feature_toggle_handler.py +++ b/backend/handlers/feature_toggle_handler.py @@ -32,4 +32,4 @@ def get(self, user): def put(self, user): features_body = json.loads(self.request.body) features = Feature.enable_all(features_body) - self.response.write(json.dumps(to_json(features))) \ No newline at end of file + self.response.write(json.dumps(to_json(features))) diff --git a/backend/models/feature.py b/backend/models/feature.py index b3b21107c..78969351e 100644 --- a/backend/models/feature.py +++ b/backend/models/feature.py @@ -33,4 +33,4 @@ def isEnabled(feature_name): if feature: return feature.enabled else: - raise Exception("Feature not found!") \ No newline at end of file + raise Exception("Feature not found!") diff --git a/frontend/utils/featureToggleService.js b/frontend/utils/featureToggleService.js new file mode 100644 index 000000000..c66bf1894 --- /dev/null +++ b/frontend/utils/featureToggleService.js @@ -0,0 +1,13 @@ +(function() { + 'use strict'; + + const app = angular.module('app'); + + app.service('FeatureToggleSercvice', function(HttpService) { + const service = this; + + service.getFeatures = function getFeatures() { + + }; + }); +})(); \ No newline at end of file From 4eb6a7cc2b28cb76dcb22809e74abbe868f51dcd Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Mon, 10 Dec 2018 11:07:30 -0300 Subject: [PATCH 004/606] Create method to get feature toggles --- frontend/index.html | 1 + frontend/user/userService.js | 2 +- frontend/utils/featureToggleService.js | 9 +++++++-- 3 files changed, 9 insertions(+), 3 deletions(-) diff --git a/frontend/index.html b/frontend/index.html index 8fafa97a3..6f361b737 100644 --- a/frontend/index.html +++ b/frontend/index.html @@ -138,6 +138,7 @@ + diff --git a/frontend/user/userService.js b/frontend/user/userService.js index 46a251547..5b87e9f91 100644 --- a/frontend/user/userService.js +++ b/frontend/user/userService.js @@ -3,7 +3,7 @@ (function() { var app = angular.module("app"); - app.service("UserService", function UserService(HttpService, $q) { + app.service("UserService", function UserService(HttpService, $q, FeatureToggleService) { var service = this; var USER_URI = "/api/user"; diff --git a/frontend/utils/featureToggleService.js b/frontend/utils/featureToggleService.js index c66bf1894..2787fb0d1 100644 --- a/frontend/utils/featureToggleService.js +++ b/frontend/utils/featureToggleService.js @@ -3,11 +3,16 @@ const app = angular.module('app'); - app.service('FeatureToggleSercvice', function(HttpService) { + app.service('FeatureToggleService', function FeatureToggleService(HttpService) { const service = this; + const uri = '/api/feature-toggle'; service.getFeatures = function getFeatures() { - + HttpService.get(uri).then(function(response) { + console.log(response); + }); }; + + service.getFeatures(); }); })(); \ No newline at end of file From 81a88d676f3347132f7ab5b6dde1095e625e63f3 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Tue, 11 Dec 2018 08:51:50 -0300 Subject: [PATCH 005/606] Change feature model on backend --- backend/handlers/feature_toggle_handler.py | 21 ++++++++++---------- backend/main.py | 2 +- backend/models/feature.py | 23 ++++++++++++++++++++-- frontend/utils/featureToggleService.js | 16 +++++++++++---- 4 files changed, 44 insertions(+), 18 deletions(-) diff --git a/backend/handlers/feature_toggle_handler.py b/backend/handlers/feature_toggle_handler.py index 310e1bec9..47335b2e4 100644 --- a/backend/handlers/feature_toggle_handler.py +++ b/backend/handlers/feature_toggle_handler.py @@ -10,22 +10,21 @@ __all__ = ['FeatureToggleHander'] def to_json(feature_list): - features = { - feature.name: { - "enabled": feature.enabled, - "group": feature.group - } for feature in feature_list - } - + features = [feature.make() for feature in feature_list] return features class FeatureToggleHander(BaseHandler): - @login_required @json_response - def get(self, user): - features = Feature.query().fetch() - self.response.write(json.dumps(to_json(features))) + def get(self): + feature_name = self.request.get('name') + + if feature_name: + features = Feature.get_feature(feature_name).make() + else: + features = to_json(Feature.get_all_features()) + + self.response.write(json.dumps(features)) @login_required @json_response diff --git a/backend/main.py b/backend/main.py index 3d6afff84..d1b26ce51 100644 --- a/backend/main.py +++ b/backend/main.py @@ -97,7 +97,7 @@ ("/api/user/institutions/(.*)", UserHandler), ("/api/user/timeline.*", UserTimelineHandler), ("/api/search/institution", SearchHandler), - ("/api/feature-toggle", FeatureToggleHander), + ("/api/feature-toggle.*", FeatureToggleHander), ("/login", LoginHandler), ("/logout", LogoutHandler), ("/api/.*", ErroHandler) diff --git a/backend/models/feature.py b/backend/models/feature.py index 78969351e..56b8abe3a 100644 --- a/backend/models/feature.py +++ b/backend/models/feature.py @@ -27,10 +27,29 @@ def enable_all(features_dict): return features @staticmethod - def isEnabled(feature_name): + def get_all_features(): + features = Feature.query().fetch() + return features + + @staticmethod + def get_feature(feature_name): feature = Feature.get_by_id(feature_name) if feature: - return feature.enabled + return feature else: raise Exception("Feature not found!") + + @staticmethod + def isEnabled(feature_name): + feature = Feature.get_feature(feature_name) + return feature.enabled + + def make(self): + make_obj = { + 'name': self.name, + 'enabled': self.enabled, + 'group': self.group + } + + return make_obj \ No newline at end of file diff --git a/frontend/utils/featureToggleService.js b/frontend/utils/featureToggleService.js index 2787fb0d1..1a7808881 100644 --- a/frontend/utils/featureToggleService.js +++ b/frontend/utils/featureToggleService.js @@ -7,12 +7,20 @@ const service = this; const uri = '/api/feature-toggle'; - service.getFeatures = function getFeatures() { - HttpService.get(uri).then(function(response) { - console.log(response); + service.getFeatures = function getFeatures(feature_name) { + const query = (feature_name) ? `?name=${feature_name}` : ''; + return HttpService.get(`${uri}${query}`); + }; + + + service.isEnabled = function isEnabled(feature_name) { + return service.getFeatures(feature_name).then(function(response) { + return response[feature_name].enabled; }); }; - service.getFeatures(); + service.isEnabled('other').then(function(response) { + console.log(response); + }); }); })(); \ No newline at end of file From 1208eb56a8af8c22e55896f96d349dec05180987 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Tue, 11 Dec 2018 14:57:47 -0300 Subject: [PATCH 006/606] Fix get feature by name --- backend/handlers/feature_toggle_handler.py | 2 +- backend/models/feature.py | 11 +++++++++-- frontend/user/userService.js | 2 +- frontend/utils/featureToggleService.js | 6 +----- 4 files changed, 12 insertions(+), 9 deletions(-) diff --git a/backend/handlers/feature_toggle_handler.py b/backend/handlers/feature_toggle_handler.py index 47335b2e4..52d5b130d 100644 --- a/backend/handlers/feature_toggle_handler.py +++ b/backend/handlers/feature_toggle_handler.py @@ -20,7 +20,7 @@ def get(self): feature_name = self.request.get('name') if feature_name: - features = Feature.get_feature(feature_name).make() + features = [Feature.get_feature(feature_name).make()] else: features = to_json(Feature.get_all_features()) diff --git a/backend/models/feature.py b/backend/models/feature.py index 56b8abe3a..3786a2c16 100644 --- a/backend/models/feature.py +++ b/backend/models/feature.py @@ -16,7 +16,14 @@ def create(name, enable, group="ALL"): return feature @staticmethod - def enable_all(features_dict): + def enable_all(features_list): + features_dict = { + feature['name']: { + 'enabled': feature['enabled'], + 'group': feature['group'] + } for feature in features_list + } + features = Feature.query(Feature.name.IN(features_dict.keys())).fetch() for feature in features: @@ -52,4 +59,4 @@ def make(self): 'group': self.group } - return make_obj \ No newline at end of file + return make_obj diff --git a/frontend/user/userService.js b/frontend/user/userService.js index 5b87e9f91..46a251547 100644 --- a/frontend/user/userService.js +++ b/frontend/user/userService.js @@ -3,7 +3,7 @@ (function() { var app = angular.module("app"); - app.service("UserService", function UserService(HttpService, $q, FeatureToggleService) { + app.service("UserService", function UserService(HttpService, $q) { var service = this; var USER_URI = "/api/user"; diff --git a/frontend/utils/featureToggleService.js b/frontend/utils/featureToggleService.js index 1a7808881..ddf7640d9 100644 --- a/frontend/utils/featureToggleService.js +++ b/frontend/utils/featureToggleService.js @@ -15,12 +15,8 @@ service.isEnabled = function isEnabled(feature_name) { return service.getFeatures(feature_name).then(function(response) { - return response[feature_name].enabled; + return response[0].enabled; }); }; - - service.isEnabled('other').then(function(response) { - console.log(response); - }); }); })(); \ No newline at end of file From 4ce85aa3964c9e39f442001864dbcb27c6515bf4 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Tue, 11 Dec 2018 17:44:38 -0300 Subject: [PATCH 007/606] Create state to feature constant --- frontend/app.js | 31 ++++++++++++++++++++++++++ frontend/index.html | 1 + frontend/utils/featureByState.js | 9 ++++++++ frontend/utils/featureToggleService.js | 1 - 4 files changed, 41 insertions(+), 1 deletion(-) create mode 100644 frontend/utils/featureByState.js diff --git a/frontend/app.js b/frontend/app.js index 9a375894d..55646c238 100644 --- a/frontend/app.js +++ b/frontend/app.js @@ -505,6 +505,37 @@ }); }); + app.run(function featureToggleInterceptor(FeatureToggleService, STATE_TO_FEATURE, $transitions, $state, STATES) { + let verified = false; + + $transitions.onStart({ + to: function(state) { + const stateName = state.name; + if (!verified && (stateName in STATE_TO_FEATURE)) { + return true; + } else { + verified = false; + return false; + } + } + }, function (transition) { + transition.abort(); + const targetState = transition._targetState; + + FeatureToggleService.isEnabled(STATE_TO_FEATURE[targetState._identifier]).then(function(enabled) { + if (enabled) { + verified = true; + $state.go(targetState._identifier, targetState._params); + } else { + $state.go(STATES.ERROR, { + "msg": "Desculpa! Este recurso ainda não está disponível.", + "status": "401" + }); + } + }); + }); + }); + function initServiceWorker() { if ('serviceWorker' in navigator) { navigator.serviceWorker.register('/sw.js').then(function (registration) { diff --git a/frontend/index.html b/frontend/index.html index d5d066ccd..0f543df44 100644 --- a/frontend/index.html +++ b/frontend/index.html @@ -140,6 +140,7 @@ + diff --git a/frontend/utils/featureByState.js b/frontend/utils/featureByState.js new file mode 100644 index 000000000..edf779609 --- /dev/null +++ b/frontend/utils/featureByState.js @@ -0,0 +1,9 @@ +(function() { + 'use strict'; + + const app = angular.module('app'); + + app.constant("STATE_TO_FEATURE", { + "app.manage_institution.edit_info": "manage-institution" + }); +})(); \ No newline at end of file diff --git a/frontend/utils/featureToggleService.js b/frontend/utils/featureToggleService.js index ddf7640d9..e0d9415d7 100644 --- a/frontend/utils/featureToggleService.js +++ b/frontend/utils/featureToggleService.js @@ -12,7 +12,6 @@ return HttpService.get(`${uri}${query}`); }; - service.isEnabled = function isEnabled(feature_name) { return service.getFeatures(feature_name).then(function(response) { return response[0].enabled; From cdad9c268589a4dabac1f4ee6bb099e3c514d76e Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Wed, 12 Dec 2018 16:31:23 -0300 Subject: [PATCH 008/606] Create device check on feature toggle] --- backend/models/feature.py | 14 +++++++++----- frontend/utils/featureToggleService.js | 11 ++++++++++- 2 files changed, 19 insertions(+), 6 deletions(-) diff --git a/backend/models/feature.py b/backend/models/feature.py index 3786a2c16..2f11c77c8 100644 --- a/backend/models/feature.py +++ b/backend/models/feature.py @@ -7,11 +7,12 @@ class Feature(ndb.Model): name = ndb.StringProperty() enabled = ndb.BooleanProperty() group = ndb.StringProperty( - choices=set(["ADMIN", "COMMOM", "ALL"])) + choices=set(["SUPER_USER", "ADMIN", "ALL"])) + device = ndb.StringProperty(choices=set(["MOBILE", "DESKTOP", "ALL"])) @staticmethod - def create(name, enable, group="ALL"): - feature = Feature(id=name, name=name, enabled=enable, group=group) + def create(name, enable, group="ALL", device="ALL"): + feature = Feature(id=name, name=name, enabled=enable, group=group, device=device) feature.put() return feature @@ -20,7 +21,8 @@ def enable_all(features_list): features_dict = { feature['name']: { 'enabled': feature['enabled'], - 'group': feature['group'] + 'group': feature['group'], + 'device': feature['device'] } for feature in features_list } @@ -29,6 +31,7 @@ def enable_all(features_list): for feature in features: feature.enabled = features_dict[feature.name]['enabled'] feature.group = features_dict[feature.name]['group'] + feature.device = features_dict[feature.name]['device'] ndb.put_multi(features) return features @@ -56,7 +59,8 @@ def make(self): make_obj = { 'name': self.name, 'enabled': self.enabled, - 'group': self.group + 'group': self.group, + 'device': self.device } return make_obj diff --git a/frontend/utils/featureToggleService.js b/frontend/utils/featureToggleService.js index e0d9415d7..0247dcef9 100644 --- a/frontend/utils/featureToggleService.js +++ b/frontend/utils/featureToggleService.js @@ -14,7 +14,16 @@ service.isEnabled = function isEnabled(feature_name) { return service.getFeatures(feature_name).then(function(response) { - return response[0].enabled; + const feature = response[0]; + + if (!feature.enabled) + return false; + + const enableMobile = feature.device === 'MOBILE' && Utils.isMobileScreen(); + const enableDesktop = feature.device === 'DESKTOP' && !Utils.isMobileScreen(); + const enableALL = feature.device === 'ALL'; + + return enableALL || enableMobile || enableDesktop; }); }; }); From a7c11e69fe763115bf1e51329e67fcf3000636a4 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Wed, 12 Dec 2018 17:18:57 -0300 Subject: [PATCH 009/606] Change properties of feature models --- backend/models/feature.py | 27 ++++++++++++-------------- frontend/utils/featureToggleService.js | 10 ++++------ 2 files changed, 16 insertions(+), 21 deletions(-) diff --git a/backend/models/feature.py b/backend/models/feature.py index 2f11c77c8..05aba93ce 100644 --- a/backend/models/feature.py +++ b/backend/models/feature.py @@ -5,14 +5,14 @@ class Feature(ndb.Model): name = ndb.StringProperty() - enabled = ndb.BooleanProperty() - group = ndb.StringProperty( - choices=set(["SUPER_USER", "ADMIN", "ALL"])) - device = ndb.StringProperty(choices=set(["MOBILE", "DESKTOP", "ALL"])) + enable_mobile = ndb.StringProperty( + choices=set(["SUPER_USER", "ADMIN", "ALL", "DISABLED"])) + enable_desktop = ndb.StringProperty( + choices=set(["SUPER_USER", "ADMIN", "ALL", "DISABLED"])) @staticmethod - def create(name, enable, group="ALL", device="ALL"): - feature = Feature(id=name, name=name, enabled=enable, group=group, device=device) + def create(name, enable_mobile="ALL", enable_desktop="ALL"): + feature = Feature(id=name, name=name, enable_desktop=enable_desktop, enable_mobile=enable_mobile) feature.put() return feature @@ -20,18 +20,16 @@ def create(name, enable, group="ALL", device="ALL"): def enable_all(features_list): features_dict = { feature['name']: { - 'enabled': feature['enabled'], - 'group': feature['group'], - 'device': feature['device'] + 'enable_mobile': feature['enable_mobile'], + 'enable_desktop': feature['enable_desktop'] } for feature in features_list } features = Feature.query(Feature.name.IN(features_dict.keys())).fetch() for feature in features: - feature.enabled = features_dict[feature.name]['enabled'] - feature.group = features_dict[feature.name]['group'] - feature.device = features_dict[feature.name]['device'] + feature.enable_desktop = features_dict[feature.name]['enable_desktop'] + feature.enable_mobile = features_dict[feature.name]['enable_mobile'] ndb.put_multi(features) return features @@ -58,9 +56,8 @@ def isEnabled(feature_name): def make(self): make_obj = { 'name': self.name, - 'enabled': self.enabled, - 'group': self.group, - 'device': self.device + 'enable_mobile': self.enable_mobile, + 'enable_desktop': self.enable_desktop } return make_obj diff --git a/frontend/utils/featureToggleService.js b/frontend/utils/featureToggleService.js index 0247dcef9..d2427ffa7 100644 --- a/frontend/utils/featureToggleService.js +++ b/frontend/utils/featureToggleService.js @@ -15,15 +15,13 @@ service.isEnabled = function isEnabled(feature_name) { return service.getFeatures(feature_name).then(function(response) { const feature = response[0]; + const disabledMobile = feature.enable_mobile === 'DISABLED'; + const disabeldDesktop = feature.enable_desktop === 'DISABLED'; - if (!feature.enabled) + if (disabledMobile && disabeldDesktop) return false; - const enableMobile = feature.device === 'MOBILE' && Utils.isMobileScreen(); - const enableDesktop = feature.device === 'DESKTOP' && !Utils.isMobileScreen(); - const enableALL = feature.device === 'ALL'; - - return enableALL || enableMobile || enableDesktop; + return !disabeldDesktop && !Utils.isMobileScreen() || !disabledMobile && Utils.isMobileScreen(); }); }; }); From 1a923d072a9bf4f5e674e9bce37d3d54af4c7b36 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Thu, 13 Dec 2018 14:05:17 -0300 Subject: [PATCH 010/606] Create mapStateToFeatureService --- frontend/app.js | 6 +++--- frontend/index.html | 2 +- frontend/utils/featureByState.js | 9 --------- frontend/utils/mapStateToFeatureService.js | 22 ++++++++++++++++++++++ 4 files changed, 26 insertions(+), 13 deletions(-) delete mode 100644 frontend/utils/featureByState.js create mode 100644 frontend/utils/mapStateToFeatureService.js diff --git a/frontend/app.js b/frontend/app.js index 55646c238..5ef32eaa6 100644 --- a/frontend/app.js +++ b/frontend/app.js @@ -505,13 +505,13 @@ }); }); - app.run(function featureToggleInterceptor(FeatureToggleService, STATE_TO_FEATURE, $transitions, $state, STATES) { + app.run(function featureToggleInterceptor(FeatureToggleService, MapStateToFeatureService, $transitions, $state, STATES) { let verified = false; $transitions.onStart({ to: function(state) { const stateName = state.name; - if (!verified && (stateName in STATE_TO_FEATURE)) { + if (!verified && MapStateToFeatureService.containsFeature(stateName)) { return true; } else { verified = false; @@ -522,7 +522,7 @@ transition.abort(); const targetState = transition._targetState; - FeatureToggleService.isEnabled(STATE_TO_FEATURE[targetState._identifier]).then(function(enabled) { + FeatureToggleService.isEnabled(MapStateToFeatureService.getFeatureByState(targetState._identifier)).then(function(enabled) { if (enabled) { verified = true; $state.go(targetState._identifier, targetState._params); diff --git a/frontend/index.html b/frontend/index.html index 0f543df44..53399ee35 100644 --- a/frontend/index.html +++ b/frontend/index.html @@ -140,7 +140,7 @@ - + diff --git a/frontend/utils/featureByState.js b/frontend/utils/featureByState.js deleted file mode 100644 index edf779609..000000000 --- a/frontend/utils/featureByState.js +++ /dev/null @@ -1,9 +0,0 @@ -(function() { - 'use strict'; - - const app = angular.module('app'); - - app.constant("STATE_TO_FEATURE", { - "app.manage_institution.edit_info": "manage-institution" - }); -})(); \ No newline at end of file diff --git a/frontend/utils/mapStateToFeatureService.js b/frontend/utils/mapStateToFeatureService.js new file mode 100644 index 000000000..096a0d4cb --- /dev/null +++ b/frontend/utils/mapStateToFeatureService.js @@ -0,0 +1,22 @@ +(function() { + 'use strict'; + + const app = angular.module('app'); + + + app.service('MapStateToFeatureService', function(STATES) { + const service = this; + + service._statesToFeature = { + [STATES.MANAGE_INST_EDIT]: 'manage-inst-edit' + }; + + service.getFeatureByState = function getFeatureByState(stateName) { + return _.get(service._statesToFeature, stateName, null); + }; + + service.containsFeature = function containsFeature(stateName) { + return stateName in service._statesToFeature; + }; + }); +})(); \ No newline at end of file From a03e561a815d868775b5c2d5cebda1bc2aa98848 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Thu, 13 Dec 2018 15:16:29 -0300 Subject: [PATCH 011/606] Add docstring of backend methods --- backend/handlers/feature_toggle_handler.py | 16 +++++++ backend/models/feature.py | 50 ++++++++++++++++++++-- frontend/sw.js | 2 +- frontend/utils/mapStateToFeatureService.js | 1 - 4 files changed, 63 insertions(+), 6 deletions(-) diff --git a/backend/handlers/feature_toggle_handler.py b/backend/handlers/feature_toggle_handler.py index 52d5b130d..d67661e80 100644 --- a/backend/handlers/feature_toggle_handler.py +++ b/backend/handlers/feature_toggle_handler.py @@ -10,13 +10,25 @@ __all__ = ['FeatureToggleHander'] def to_json(feature_list): + """ + Method to generate list of feature models in json format object. + + Params: + feature_list -- List of features objects + """ + features = [feature.make() for feature in feature_list] return features class FeatureToggleHander(BaseHandler): + """Feature toggle hanler.""" @json_response def get(self): + """ + Method to get all features or filter by name using query parameter. + """ + feature_name = self.request.get('name') if feature_name: @@ -29,6 +41,10 @@ def get(self): @login_required @json_response def put(self, user): + """ + Method for modifying the properties of one or more features. + """ + features_body = json.loads(self.request.body) features = Feature.enable_all(features_body) self.response.write(json.dumps(to_json(features))) diff --git a/backend/models/feature.py b/backend/models/feature.py index 05aba93ce..9836b4690 100644 --- a/backend/models/feature.py +++ b/backend/models/feature.py @@ -4,6 +4,10 @@ __all__ = ['Feature'] class Feature(ndb.Model): + """ + Model of Feature + """ + name = ndb.StringProperty() enable_mobile = ndb.StringProperty( choices=set(["SUPER_USER", "ADMIN", "ALL", "DISABLED"])) @@ -11,13 +15,31 @@ class Feature(ndb.Model): choices=set(["SUPER_USER", "ADMIN", "ALL", "DISABLED"])) @staticmethod + @ndb.transactional(xg=True) def create(name, enable_mobile="ALL", enable_desktop="ALL"): + """ + Method to create new feature. + + Params: + name -- Name of the new feature + enable_mobile -- (Optional) User group to which the feature is enabled in the mobile version. If not received will be enabled for everyone. + enable_desktop -- (Optional) User group to which the feature is enabled in the desktop version. If not received will be enabled for everyone. + """ + feature = Feature(id=name, name=name, enable_desktop=enable_desktop, enable_mobile=enable_mobile) feature.put() return feature @staticmethod + @ndb.transactional(xg=True) def enable_all(features_list): + """ + Method to enable or disable multiple features. + + Params: + features_list -- list of dictionaries containing the properties of the features model to be modified. + """ + features_dict = { feature['name']: { 'enable_mobile': feature['enable_mobile'], @@ -36,11 +58,22 @@ def enable_all(features_list): @staticmethod def get_all_features(): + """ + Method to get all stored features. + """ + features = Feature.query().fetch() return features @staticmethod def get_feature(feature_name): + """ + Method to get feature by name. + + Params: + feature_name -- name of the requested feature + """ + feature = Feature.get_by_id(feature_name) if feature: @@ -48,12 +81,21 @@ def get_feature(feature_name): else: raise Exception("Feature not found!") - @staticmethod - def isEnabled(feature_name): - feature = Feature.get_feature(feature_name) - return feature.enabled + def is_enabled(self): + """ + Method to verify if the feature is enabled. + """ + + disabled_mobile = self.enable_mobile == 'DISABLED' + disabled_desktop = self.enable_desktop == 'DISABLED' + + return not (disabled_desktop and disabled_mobile) def make(self): + """ + Method to make feature. + """ + make_obj = { 'name': self.name, 'enable_mobile': self.enable_mobile, diff --git a/frontend/sw.js b/frontend/sw.js index 3d30b829c..4bca0fe8a 100644 --- a/frontend/sw.js +++ b/frontend/sw.js @@ -7,7 +7,7 @@ importScripts('app/firebase-config.js'); importScripts('app/config.js'); // if the line number of the code below changes, modify the /ecis script. - const CACHE_SUFIX = 'create-feature-toggles'; + const CACHE_SUFIX = 'master'; let messaging; diff --git a/frontend/utils/mapStateToFeatureService.js b/frontend/utils/mapStateToFeatureService.js index 096a0d4cb..1894488a6 100644 --- a/frontend/utils/mapStateToFeatureService.js +++ b/frontend/utils/mapStateToFeatureService.js @@ -3,7 +3,6 @@ const app = angular.module('app'); - app.service('MapStateToFeatureService', function(STATES) { const service = this; From 2b88b61260922871b0788665c277d89a965c5e47 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Thu, 13 Dec 2018 16:21:48 -0300 Subject: [PATCH 012/606] Add documentation of frontend functions --- backend/models/feature.py | 5 ++--- frontend/app.js | 2 +- frontend/utils/featureToggleService.js | 23 ++++++++++++++++++---- frontend/utils/mapStateToFeatureService.js | 12 +++++++++++ 4 files changed, 34 insertions(+), 8 deletions(-) diff --git a/backend/models/feature.py b/backend/models/feature.py index 9836b4690..cf8968819 100644 --- a/backend/models/feature.py +++ b/backend/models/feature.py @@ -5,9 +5,9 @@ class Feature(ndb.Model): """ - Model of Feature + Model of Feature. """ - + name = ndb.StringProperty() enable_mobile = ndb.StringProperty( choices=set(["SUPER_USER", "ADMIN", "ALL", "DISABLED"])) @@ -31,7 +31,6 @@ def create(name, enable_mobile="ALL", enable_desktop="ALL"): return feature @staticmethod - @ndb.transactional(xg=True) def enable_all(features_list): """ Method to enable or disable multiple features. diff --git a/frontend/app.js b/frontend/app.js index 5ef32eaa6..2cfc80af0 100644 --- a/frontend/app.js +++ b/frontend/app.js @@ -529,7 +529,7 @@ } else { $state.go(STATES.ERROR, { "msg": "Desculpa! Este recurso ainda não está disponível.", - "status": "401" + "status": "500" }); } }); diff --git a/frontend/utils/featureToggleService.js b/frontend/utils/featureToggleService.js index d2427ffa7..25302e853 100644 --- a/frontend/utils/featureToggleService.js +++ b/frontend/utils/featureToggleService.js @@ -7,13 +7,28 @@ const service = this; const uri = '/api/feature-toggle'; - service.getFeatures = function getFeatures(feature_name) { - const query = (feature_name) ? `?name=${feature_name}` : ''; + + /** + * Function to get all features or filter by name + * + * @param {String} featureName - (Optional) Feature name to use in query parameter + * @return {Promise} Promise that when it is resolved it returns the searched feature. + */ + service.getFeatures = function getFeatures(featureName) { + const query = (featureName) ? `?name=${featureName}` : ''; return HttpService.get(`${uri}${query}`); }; - service.isEnabled = function isEnabled(feature_name) { - return service.getFeatures(feature_name).then(function(response) { + + /** + * This function checks whether the feature passed by parameter + * is enabled for the logged in user and the device that he uses. + * + * @param {String} featureName - Feature name to check if is enabled + * @return {Promise} Promise that when it is resolved it returns a boolean indicating whether or not it is enabled. + */ + service.isEnabled = function isEnabled(featureName) { + return service.getFeatures(featureName).then(function(response) { const feature = response[0]; const disabledMobile = feature.enable_mobile === 'DISABLED'; const disabeldDesktop = feature.enable_desktop === 'DISABLED'; diff --git a/frontend/utils/mapStateToFeatureService.js b/frontend/utils/mapStateToFeatureService.js index 1894488a6..09ddff7bb 100644 --- a/frontend/utils/mapStateToFeatureService.js +++ b/frontend/utils/mapStateToFeatureService.js @@ -10,10 +10,22 @@ [STATES.MANAGE_INST_EDIT]: 'manage-inst-edit' }; + /** + * Function to get feature by state name + * + * @param {String} stateName - name of state to get related feature + * @return {String} Feature name if the state is registered, otherwise returns null + */ service.getFeatureByState = function getFeatureByState(stateName) { return _.get(service._statesToFeature, stateName, null); }; + /** + * Function to check if exists feature for state passed by parameter + * + * @param {String} stateName - Name of state to be check + * @return {Boolean} True if exists, otherwise, false + */ service.containsFeature = function containsFeature(stateName) { return stateName in service._statesToFeature; }; From 16782f1366c985b1a28816995eaf3f2578bad550 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Thu, 13 Dec 2018 17:41:03 -0300 Subject: [PATCH 013/606] Refact state transition interceptor --- frontend/app.js | 22 +++++++--------------- 1 file changed, 7 insertions(+), 15 deletions(-) diff --git a/frontend/app.js b/frontend/app.js index 2cfc80af0..9f090eeb1 100644 --- a/frontend/app.js +++ b/frontend/app.js @@ -505,29 +505,21 @@ }); }); - app.run(function featureToggleInterceptor(FeatureToggleService, MapStateToFeatureService, $transitions, $state, STATES) { - let verified = false; + app.run(function featureToggleInterceptor(FeatureToggleService, MapStateToFeatureService, $transitions, STATES) { - $transitions.onStart({ + $transitions.onBefore({ to: function(state) { const stateName = state.name; - if (!verified && MapStateToFeatureService.containsFeature(stateName)) { - return true; - } else { - verified = false; - return false; - } + return MapStateToFeatureService.containsFeature(stateName); } }, function (transition) { - transition.abort(); - const targetState = transition._targetState; + const targetStateName = transition.to().name; - FeatureToggleService.isEnabled(MapStateToFeatureService.getFeatureByState(targetState._identifier)).then(function(enabled) { + return FeatureToggleService.isEnabled(MapStateToFeatureService.getFeatureByState(targetStateName)).then(function(enabled) { if (enabled) { - verified = true; - $state.go(targetState._identifier, targetState._params); + return transition; } else { - $state.go(STATES.ERROR, { + return transition.router.stateService.target(STATES.ERROR, { "msg": "Desculpa! Este recurso ainda não está disponível.", "status": "500" }); From 0b8aa755e5082a9ec0c571cd1331a00ac144f8b2 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Fri, 14 Dec 2018 15:12:54 -0300 Subject: [PATCH 014/606] Fix issues --- backend/handlers/feature_toggle_handler.py | 19 ++++++++++--------- backend/main.py | 4 ++-- frontend/utils/featureToggleService.js | 10 +++++----- frontend/utils/mapStateToFeatureService.js | 4 ++-- 4 files changed, 19 insertions(+), 18 deletions(-) diff --git a/backend/handlers/feature_toggle_handler.py b/backend/handlers/feature_toggle_handler.py index d67661e80..b0a59f476 100644 --- a/backend/handlers/feature_toggle_handler.py +++ b/backend/handlers/feature_toggle_handler.py @@ -1,4 +1,4 @@ -"""Feature Toogle handler.""" +"""Feature Toggle handler.""" import json from . import BaseHandler @@ -7,7 +7,7 @@ from models import Feature -__all__ = ['FeatureToggleHander'] +__all__ = ['FeatureToggleHandler'] def to_json(feature_list): """ @@ -18,13 +18,14 @@ def to_json(feature_list): """ features = [feature.make() for feature in feature_list] - return features + return json.dumps(features) -class FeatureToggleHander(BaseHandler): +class FeatureToggleHandler(BaseHandler): """Feature toggle hanler.""" @json_response - def get(self): + @login_required + def get(self, user): """ Method to get all features or filter by name using query parameter. """ @@ -32,11 +33,11 @@ def get(self): feature_name = self.request.get('name') if feature_name: - features = [Feature.get_feature(feature_name).make()] + features = [Feature.get_feature(feature_name)] else: - features = to_json(Feature.get_all_features()) + features = Feature.get_all_features() - self.response.write(json.dumps(features)) + self.response.write(to_json(features)) @login_required @json_response @@ -47,4 +48,4 @@ def put(self, user): features_body = json.loads(self.request.body) features = Feature.enable_all(features_body) - self.response.write(json.dumps(to_json(features))) + self.response.write(to_json(features)) diff --git a/backend/main.py b/backend/main.py index d1b26ce51..ea8577b1e 100644 --- a/backend/main.py +++ b/backend/main.py @@ -45,7 +45,7 @@ from handlers import InviteHandler from handlers import InstitutionParentHandler from handlers import InstitutionChildrenHandler -from handlers import FeatureToggleHander +from handlers import FeatureToggleHandler methods = set(webapp2.WSGIApplication.allowed_methods) methods.add('PATCH') @@ -97,7 +97,7 @@ ("/api/user/institutions/(.*)", UserHandler), ("/api/user/timeline.*", UserTimelineHandler), ("/api/search/institution", SearchHandler), - ("/api/feature-toggle.*", FeatureToggleHander), + ("/api/feature-toggle.*", FeatureToggleHandler), ("/login", LoginHandler), ("/logout", LogoutHandler), ("/api/.*", ErroHandler) diff --git a/frontend/utils/featureToggleService.js b/frontend/utils/featureToggleService.js index 25302e853..ebfb3fa5c 100644 --- a/frontend/utils/featureToggleService.js +++ b/frontend/utils/featureToggleService.js @@ -29,14 +29,14 @@ */ service.isEnabled = function isEnabled(featureName) { return service.getFeatures(featureName).then(function(response) { - const feature = response[0]; - const disabledMobile = feature.enable_mobile === 'DISABLED'; - const disabeldDesktop = feature.enable_desktop === 'DISABLED'; + const feature = _.first(response); + const disableMobile = _.get(feature, 'enable_mobile') === 'DISABLED'; + const disableDesktop = _.get(feature, 'enable_desktop') === 'DISABLED'; - if (disabledMobile && disabeldDesktop) + if (disableMobile && disableDesktop) return false; - return !disabeldDesktop && !Utils.isMobileScreen() || !disabledMobile && Utils.isMobileScreen(); + return !disableDesktop && !Utils.isMobileScreen() || !disableMobile && Utils.isMobileScreen(); }); }; }); diff --git a/frontend/utils/mapStateToFeatureService.js b/frontend/utils/mapStateToFeatureService.js index 09ddff7bb..9b8d6658f 100644 --- a/frontend/utils/mapStateToFeatureService.js +++ b/frontend/utils/mapStateToFeatureService.js @@ -14,10 +14,10 @@ * Function to get feature by state name * * @param {String} stateName - name of state to get related feature - * @return {String} Feature name if the state is registered, otherwise returns null + * @return {String} Feature name if the state is registered, otherwise returns undefined */ service.getFeatureByState = function getFeatureByState(stateName) { - return _.get(service._statesToFeature, stateName, null); + return _.get(service._statesToFeature, stateName); }; /** From 046bc0827ccc2b5f974b258b7cd7b25a8c7ff295 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Fri, 14 Dec 2018 17:07:13 -0300 Subject: [PATCH 015/606] Create Feature model test --- backend/test/model_test/feature_test.py | 34 +++++++++++++++++++++++++ frontend/app.js | 2 +- frontend/utils/featureToggleService.js | 12 +++++++-- 3 files changed, 45 insertions(+), 3 deletions(-) create mode 100644 backend/test/model_test/feature_test.py diff --git a/backend/test/model_test/feature_test.py b/backend/test/model_test/feature_test.py new file mode 100644 index 000000000..259b7bd92 --- /dev/null +++ b/backend/test/model_test/feature_test.py @@ -0,0 +1,34 @@ +"""Feature Test.""" + +from .. import mocks +from ..test_base import TestBase +from models import Feature + + +class FeatureTest(TestBase): + + @classmethod + def setUp(cls): + """Provide the base for the tests.""" + cls.test = cls.testbed.Testbed() + cls.test.activate() + cls.policy = cls.datastore.PseudoRandomHRConsistencyPolicy( + probability=1) + cls.test.init_datastore_v3_stub(consistency_policy=cls.policy) + cls.test.init_memcache_stub() + + def test_create(self): + feature = Feature.create('feature-test') + self.assertEqual(feature.name, 'feature-test') + self.assertEqual(feature.enable_desktop, 'ALL') + self.assertEqual(feature.enable_mobile, 'ALL') + + feature = Feature.create('feature-test2', 'DISABLED') + self.assertEqual(feature.name, 'feature-test2') + self.assertEqual(feature.enable_desktop, 'ALL') + self.assertEqual(feature.enable_mobile, 'DISABLED') + + feature = Feature.create('feature-test3', 'DISABLED', 'DISABLED') + self.assertEqual(feature.name, 'feature-test3') + self.assertEqual(feature.enable_desktop, 'DISABLED') + self.assertEqual(feature.enable_mobile, 'DISABLED') diff --git a/frontend/app.js b/frontend/app.js index 9f090eeb1..46912c00a 100644 --- a/frontend/app.js +++ b/frontend/app.js @@ -521,7 +521,7 @@ } else { return transition.router.stateService.target(STATES.ERROR, { "msg": "Desculpa! Este recurso ainda não está disponível.", - "status": "500" + "status": "403" }); } }); diff --git a/frontend/utils/featureToggleService.js b/frontend/utils/featureToggleService.js index ebfb3fa5c..e8e976320 100644 --- a/frontend/utils/featureToggleService.js +++ b/frontend/utils/featureToggleService.js @@ -13,12 +13,20 @@ * * @param {String} featureName - (Optional) Feature name to use in query parameter * @return {Promise} Promise that when it is resolved it returns the searched feature. + * @private */ - service.getFeatures = function getFeatures(featureName) { + service._getFeatures = function getFeatures(featureName) { const query = (featureName) ? `?name=${featureName}` : ''; return HttpService.get(`${uri}${query}`); }; + service.getFeature = function getFeature(featureName) { + return service._getFeatures(featureName); + }; + + service.getAllFeatures = function getAllFeatures() { + return service._getFeatures(); + }; /** * This function checks whether the feature passed by parameter @@ -28,7 +36,7 @@ * @return {Promise} Promise that when it is resolved it returns a boolean indicating whether or not it is enabled. */ service.isEnabled = function isEnabled(featureName) { - return service.getFeatures(featureName).then(function(response) { + return service.getFeature(featureName).then(function(response) { const feature = _.first(response); const disableMobile = _.get(feature, 'enable_mobile') === 'DISABLED'; const disableDesktop = _.get(feature, 'enable_desktop') === 'DISABLED'; From b0c30afc1bb25a62d1231ca3f2ffe03079f2b41f Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Mon, 17 Dec 2018 14:02:55 -0300 Subject: [PATCH 016/606] Add more tests --- backend/test/model_test/feature_test.py | 3 +++ 1 file changed, 3 insertions(+) diff --git a/backend/test/model_test/feature_test.py b/backend/test/model_test/feature_test.py index 259b7bd92..d3c794e6f 100644 --- a/backend/test/model_test/feature_test.py +++ b/backend/test/model_test/feature_test.py @@ -32,3 +32,6 @@ def test_create(self): self.assertEqual(feature.name, 'feature-test3') self.assertEqual(feature.enable_desktop, 'DISABLED') self.assertEqual(feature.enable_mobile, 'DISABLED') + + with self.assertRaises(Exception) as raises_context: + Feature.create('feature-test', 'asjdkhd') From 0d0a9588e7a71794ecac6dc05c5670b95accdf63 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Mon, 17 Dec 2018 15:14:47 -0300 Subject: [PATCH 017/606] Add more tests --- backend/test/model_test/feature_test.py | 47 ++++++++++++++++++++++++- frontend/sw.js | 2 +- 2 files changed, 47 insertions(+), 2 deletions(-) diff --git a/backend/test/model_test/feature_test.py b/backend/test/model_test/feature_test.py index d3c794e6f..60fe50141 100644 --- a/backend/test/model_test/feature_test.py +++ b/backend/test/model_test/feature_test.py @@ -1,6 +1,5 @@ """Feature Test.""" -from .. import mocks from ..test_base import TestBase from models import Feature @@ -35,3 +34,49 @@ def test_create(self): with self.assertRaises(Exception) as raises_context: Feature.create('feature-test', 'asjdkhd') + + exception_message = raises_context.exception + self.assertEquals( + str(exception_message), + "Value 'asjdkhd' for property enable_mobile is not an allowed choice" + ) + + def test_get_all_features(self): + feature = Feature.create('feature-test') + feature2 = Feature.create('feature-test2') + + features = Feature.get_all_features() + self.assertIn(feature, features) + self.assertIn(feature2, features) + + feature3 = Feature.create('feature-test3') + self.assertNotIn(feature3, features) + + features = Feature.get_all_features() + self.assertIn(feature3, features) + + def test_get_feature(self): + feature = Feature.create('feature-test') + feature2 = Feature.create('feature-test2') + + self.assertEqual(feature, Feature.get_feature('feature-test')) + self.assertEqual(feature2, Feature.get_feature('feature-test2')) + + with self.assertRaises(Exception) as raises_context: + Feature.get_feature('djsasadj') + + exception_message = raises_context.exception + self.assertEquals( + str(exception_message), + "Feature not found!" + ) + + def test_make(self): + feature = Feature.create('feature-test') + make = { + 'name': 'feature-test', + 'enable_mobile': 'ALL', + 'enable_desktop': 'ALL' + } + + self.assertEqual(make, feature.make()) diff --git a/frontend/sw.js b/frontend/sw.js index 4bca0fe8a..3d30b829c 100644 --- a/frontend/sw.js +++ b/frontend/sw.js @@ -7,7 +7,7 @@ importScripts('app/firebase-config.js'); importScripts('app/config.js'); // if the line number of the code below changes, modify the /ecis script. - const CACHE_SUFIX = 'master'; + const CACHE_SUFIX = 'create-feature-toggles'; let messaging; From 3e540d9fecc44f061d51d0b107617917aac9366c Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Mon, 17 Dec 2018 17:05:13 -0300 Subject: [PATCH 018/606] Change name of method from enable_all to set_visibility --- backend/handlers/feature_toggle_handler.py | 2 +- backend/models/feature.py | 2 +- backend/test/model_test/feature_test.py | 22 ++++++++++++++++++++++ 3 files changed, 24 insertions(+), 2 deletions(-) diff --git a/backend/handlers/feature_toggle_handler.py b/backend/handlers/feature_toggle_handler.py index b0a59f476..1ee5a5c9a 100644 --- a/backend/handlers/feature_toggle_handler.py +++ b/backend/handlers/feature_toggle_handler.py @@ -47,5 +47,5 @@ def put(self, user): """ features_body = json.loads(self.request.body) - features = Feature.enable_all(features_body) + features = Feature.set_visibility(features_body) self.response.write(to_json(features)) diff --git a/backend/models/feature.py b/backend/models/feature.py index cf8968819..1cb30a3d0 100644 --- a/backend/models/feature.py +++ b/backend/models/feature.py @@ -31,7 +31,7 @@ def create(name, enable_mobile="ALL", enable_desktop="ALL"): return feature @staticmethod - def enable_all(features_list): + def set_visibility(features_list): """ Method to enable or disable multiple features. diff --git a/backend/test/model_test/feature_test.py b/backend/test/model_test/feature_test.py index 60fe50141..984696066 100644 --- a/backend/test/model_test/feature_test.py +++ b/backend/test/model_test/feature_test.py @@ -5,6 +5,7 @@ class FeatureTest(TestBase): + """Feature model test.""" @classmethod def setUp(cls): @@ -17,6 +18,7 @@ def setUp(cls): cls.test.init_memcache_stub() def test_create(self): + """Teste create a new feature.""" feature = Feature.create('feature-test') self.assertEqual(feature.name, 'feature-test') self.assertEqual(feature.enable_desktop, 'ALL') @@ -41,7 +43,25 @@ def test_create(self): "Value 'asjdkhd' for property enable_mobile is not an allowed choice" ) + def test_set_visibility(self): + """Test set visibility.""" + Feature.create('feature-test') + + feature_dict = { + 'name': 'feature-test', + 'enable_mobile': 'DISABLED', + 'enable_desktop': 'ADMIN' + } + + Feature.set_visibility([feature_dict]) + + feature = Feature.get_feature('feature-test') + + self.assertEqual(feature.enable_mobile, 'DISABLED') + self.assertEqual(feature.enable_desktop, 'ADMIN') + def test_get_all_features(self): + """Test get all features.""" feature = Feature.create('feature-test') feature2 = Feature.create('feature-test2') @@ -56,6 +76,7 @@ def test_get_all_features(self): self.assertIn(feature3, features) def test_get_feature(self): + """Test get feature.""" feature = Feature.create('feature-test') feature2 = Feature.create('feature-test2') @@ -72,6 +93,7 @@ def test_get_feature(self): ) def test_make(self): + """Test make feature.""" feature = Feature.create('feature-test') make = { 'name': 'feature-test', From a24378d9ed00e26de02b5acf7cb59270773fad60 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Tue, 18 Dec 2018 10:35:07 -0300 Subject: [PATCH 019/606] Add feature toggle handler test --- .../feature_toggle_handler_tests/__init__.py | 0 .../feature_toggle_handler_test.py | 53 +++++++++++++++++++ 2 files changed, 53 insertions(+) create mode 100644 backend/test/feature_toggle_handler_tests/__init__.py create mode 100644 backend/test/feature_toggle_handler_tests/feature_toggle_handler_test.py diff --git a/backend/test/feature_toggle_handler_tests/__init__.py b/backend/test/feature_toggle_handler_tests/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/backend/test/feature_toggle_handler_tests/feature_toggle_handler_test.py b/backend/test/feature_toggle_handler_tests/feature_toggle_handler_test.py new file mode 100644 index 000000000..39fa89819 --- /dev/null +++ b/backend/test/feature_toggle_handler_tests/feature_toggle_handler_test.py @@ -0,0 +1,53 @@ +"""Feature toggle handler test""" + +from ..test_base_handler import TestBaseHandler +from handlers import FeatureToggleHandler +from models import Feature +from mock import patch + + +USER = {'email': 'user@email.com'} + +class FeatureToggleHandlerTest(TestBaseHandler): + + @classmethod + def setUp(cls): + """Provide the base for the tests.""" + super(FeatureToggleHandlerTest, cls).setUp() + app = cls.webapp2.WSGIApplication( + [("/api/feature-toggles.*", + FeatureToggleHandler) + ], debug=True) + cls.testapp = cls.webtest.TestApp(app) + + cls.feature = Feature.create('feature-test') + cls.other_feature = Feature.create('feature-test-other') + + + @patch('util.login_service.verify_token', return_value=USER) + def test_get_all(self, verify_token): + features = self.testapp.get('/api/feature-toggles').json + features_make = [self.feature.make(), self.other_feature.make()] + + self.assertEquals(len(features), 2) + self.assertItemsEqual(features, features_make) + + @patch('util.login_service.verify_token', return_value=USER) + def test_get_by_query(self, verify_token): + feature = self.testapp.get('/api/feature-toggles?name=feature-test').json + feature_make = [self.feature.make()] + + self.assertListEqual(feature, feature_make) + + + feature = self.testapp.get('/api/feature-toggles?name=feature-test-other').json + feature_make = [self.other_feature.make()] + + self.assertListEqual(feature, feature_make) + + with self.assertRaises(Exception) as raises_context: + self.testapp.get('/api/feature-toggles?name=sfjkldh') + + exception_message = self.get_message_exception(str(raises_context.exception)) + + self.assertEquals(exception_message, "Error! Feature not found!") \ No newline at end of file From 81369f79c8f9bb57d986829a6154dae847432d61 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Wed, 19 Dec 2018 13:05:08 -0300 Subject: [PATCH 020/606] Add tests off put method --- backend/handlers/feature_toggle_handler.py | 14 ++++++++++-- .../feature_toggle_handler_test.py | 22 ++++++++++++++++++- 2 files changed, 33 insertions(+), 3 deletions(-) diff --git a/backend/handlers/feature_toggle_handler.py b/backend/handlers/feature_toggle_handler.py index 1ee5a5c9a..36aa9cdd8 100644 --- a/backend/handlers/feature_toggle_handler.py +++ b/backend/handlers/feature_toggle_handler.py @@ -1,10 +1,12 @@ +# -*- coding: utf-8 -*- """Feature Toggle handler.""" import json from . import BaseHandler -from utils import json_response +from utils import json_response, Utils from util import login_required -from models import Feature +from models import Feature, get_deciis +from custom_exceptions import NotAuthorizedException __all__ = ['FeatureToggleHandler'] @@ -46,6 +48,14 @@ def put(self, user): Method for modifying the properties of one or more features. """ + """ + The super user is the admin of + 'Departamento do Complexo Industrial e Inovação em Saúde". + """ + super_user = get_deciis().admin + + Utils._assert(not (super_user == user.key), "User not allowed to modify features!", NotAuthorizedException) + features_body = json.loads(self.request.body) features = Feature.set_visibility(features_body) self.response.write(to_json(features)) diff --git a/backend/test/feature_toggle_handler_tests/feature_toggle_handler_test.py b/backend/test/feature_toggle_handler_tests/feature_toggle_handler_test.py index 39fa89819..c0e726b76 100644 --- a/backend/test/feature_toggle_handler_tests/feature_toggle_handler_test.py +++ b/backend/test/feature_toggle_handler_tests/feature_toggle_handler_test.py @@ -1,9 +1,11 @@ """Feature toggle handler test""" +import json from ..test_base_handler import TestBaseHandler from handlers import FeatureToggleHandler from models import Feature from mock import patch +from .. import mocks USER = {'email': 'user@email.com'} @@ -50,4 +52,22 @@ def test_get_by_query(self, verify_token): exception_message = self.get_message_exception(str(raises_context.exception)) - self.assertEquals(exception_message, "Error! Feature not found!") \ No newline at end of file + self.assertEquals(exception_message, "Error! Feature not found!") + + @patch('util.login_service.verify_token', return_value=USER) + def test_put(self, verify_token): + user_admin = mocks.create_user('user@email.com') + deciis = mocks.create_institution('DECIIS') + deciis.trusted = True + deciis.add_member(user_admin) + deciis.set_admin(user_admin.key) + user_admin.add_institution(deciis.key) + user_admin.add_institution_admin(deciis.key) + + feature = self.feature.make() + other_feature = self.other_feature.make() + + feature['enable_mobile'] = 'DISABLED' + other_feature['enable_desktop'] = 'DISABLED' + + #self.testapp.put_json('/api/feature-toggle',) From 0a71acb214752d403074fcae619496faa0077433 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Wed, 19 Dec 2018 15:59:19 -0300 Subject: [PATCH 021/606] Finish feature toggle heandler test --- backend/models/feature.py | 10 ------- .../feature_toggle_handler_test.py | 29 ++++++++++++++++++- 2 files changed, 28 insertions(+), 11 deletions(-) diff --git a/backend/models/feature.py b/backend/models/feature.py index 1cb30a3d0..127c396a8 100644 --- a/backend/models/feature.py +++ b/backend/models/feature.py @@ -80,16 +80,6 @@ def get_feature(feature_name): else: raise Exception("Feature not found!") - def is_enabled(self): - """ - Method to verify if the feature is enabled. - """ - - disabled_mobile = self.enable_mobile == 'DISABLED' - disabled_desktop = self.enable_desktop == 'DISABLED' - - return not (disabled_desktop and disabled_mobile) - def make(self): """ Method to make feature. diff --git a/backend/test/feature_toggle_handler_tests/feature_toggle_handler_test.py b/backend/test/feature_toggle_handler_tests/feature_toggle_handler_test.py index c0e726b76..0ead984fb 100644 --- a/backend/test/feature_toggle_handler_tests/feature_toggle_handler_test.py +++ b/backend/test/feature_toggle_handler_tests/feature_toggle_handler_test.py @@ -11,6 +11,9 @@ USER = {'email': 'user@email.com'} class FeatureToggleHandlerTest(TestBaseHandler): + """ + Feature Toggle Handler Test. + """ @classmethod def setUp(cls): @@ -28,6 +31,8 @@ def setUp(cls): @patch('util.login_service.verify_token', return_value=USER) def test_get_all(self, verify_token): + """Test get all features.""" + features = self.testapp.get('/api/feature-toggles').json features_make = [self.feature.make(), self.other_feature.make()] @@ -36,6 +41,8 @@ def test_get_all(self, verify_token): @patch('util.login_service.verify_token', return_value=USER) def test_get_by_query(self, verify_token): + """Test get feature with query parameter.""" + feature = self.testapp.get('/api/feature-toggles?name=feature-test').json feature_make = [self.feature.make()] @@ -56,7 +63,10 @@ def test_get_by_query(self, verify_token): @patch('util.login_service.verify_token', return_value=USER) def test_put(self, verify_token): + """Test put features.""" + user_admin = mocks.create_user('user@email.com') + user = mocks.create_user() deciis = mocks.create_institution('DECIIS') deciis.trusted = True deciis.add_member(user_admin) @@ -70,4 +80,21 @@ def test_put(self, verify_token): feature['enable_mobile'] = 'DISABLED' other_feature['enable_desktop'] = 'DISABLED' - #self.testapp.put_json('/api/feature-toggle',) + self.testapp.put_json('/api/feature-toggles', [feature, other_feature]).json + + self.feature = self.feature.key.get() + self.other_feature = self.other_feature.key.get() + + self.assertEquals(self.feature.enable_desktop, 'ALL') + self.assertEquals(self.feature.enable_mobile, 'DISABLED') + self.assertEquals(self.other_feature.enable_desktop, 'DISABLED') + self.assertEquals(self.other_feature.enable_mobile, 'ALL') + + verify_token._mock_return_value = {'email': user.email[0]} + + with self.assertRaises(Exception) as raises_context: + self.testapp.put_json('/api/feature-toggles', [feature, other_feature]).json + + exception_message = self.get_message_exception(str(raises_context.exception)) + + self.assertEquals(exception_message, 'Error! User not allowed to modify features!') From 06a287b3bfa4a98a738a5ba0e78108e451af1eb3 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Thu, 20 Dec 2018 16:29:43 -0300 Subject: [PATCH 022/606] Create tests for featureToggleService --- frontend/sw.js | 2 +- .../specs/utils/featureToggleServiceSpec.js | 129 ++++++++++++++++++ 2 files changed, 130 insertions(+), 1 deletion(-) create mode 100644 frontend/test/specs/utils/featureToggleServiceSpec.js diff --git a/frontend/sw.js b/frontend/sw.js index 3d30b829c..4bca0fe8a 100644 --- a/frontend/sw.js +++ b/frontend/sw.js @@ -7,7 +7,7 @@ importScripts('app/firebase-config.js'); importScripts('app/config.js'); // if the line number of the code below changes, modify the /ecis script. - const CACHE_SUFIX = 'create-feature-toggles'; + const CACHE_SUFIX = 'master'; let messaging; diff --git a/frontend/test/specs/utils/featureToggleServiceSpec.js b/frontend/test/specs/utils/featureToggleServiceSpec.js new file mode 100644 index 000000000..41534ad54 --- /dev/null +++ b/frontend/test/specs/utils/featureToggleServiceSpec.js @@ -0,0 +1,129 @@ +'use strict'; + +describe('Test FeatureToggleService', function() { + beforeEach(module('app')); + + let featureToggleService, httpBackend; + + const feature = { + name: 'test-feature', + enable_mobile: 'ALL', + enable_desktop: 'DISABLED' + }; + + const otherFeature = { + name: 'test-other-feature', + enable_mobile: 'DISABLED', + enable_desktop: 'ALL' + }; + + beforeEach(inject(function($httpBackend, FeatureToggleService) { + featureToggleService = FeatureToggleService; + httpBackend = $httpBackend; + + $httpBackend.whenGET(`/api/feature-toggle?name=${feature.name}`).respond([feature]); + $httpBackend.whenGET(`/api/feature-toggle?name=${otherFeature.name}`).respond([otherFeature]); + $httpBackend.whenGET('/api/feature-toggle').respond([feature, otherFeature]); + })); + + afterEach(function() { + httpBackend.verifyNoOutstandingExpectation(); + httpBackend.verifyNoOutstandingRequest(); + }); + + describe('Test _getFeature', function() { + + it('Should be return all features', function(done) { + featureToggleService._getFeatures().then(function(response) { + expect(response).toEqual([feature, otherFeature]); + done(); + }); + httpBackend.flush(); + }); + + it('Should be return feature test-feature', function(done) { + featureToggleService._getFeatures(feature.name).then(function(response) { + expect(response).toEqual([feature]); + done(); + }); + httpBackend.flush(); + }); + + it('Should be return feature test-other-feature', function(done) { + featureToggleService._getFeatures(otherFeature.name).then(function(response) { + expect(response).toEqual([otherFeature]); + done(); + }); + httpBackend.flush(); + }); + }); + + describe('Test getAllFeatures', function() { + + it('Should be return all features', function(done) { + featureToggleService.getAllFeatures().then(function(response) { + expect(response).toEqual([feature, otherFeature]); + done(); + }); + httpBackend.flush(); + }); + }); + + describe('Test getFeature', function() { + it('Should be return feature test-feature', function(done) { + featureToggleService.getFeature(feature.name).then(function(response) { + expect(response).toEqual([feature]); + done(); + }); + httpBackend.flush(); + }); + + it('Should be return feature test-other-feature', function(done) { + featureToggleService.getFeature(otherFeature.name).then(function(response) { + expect(response).toEqual([otherFeature]); + done(); + }); + httpBackend.flush(); + }); + }); + + + describe('Test isEnabled', function() { + it('Should be return true with test-feature', function(done) { + window.screen = {width: 100}; + featureToggleService.isEnabled(feature.name).then(function(response) { + expect(response).toBeTruthy(); + done(); + }); + httpBackend.flush(); + }); + + it('Should be return true with test-other-feature', function(done) { + window.screen = {width: 1000}; + featureToggleService.isEnabled(otherFeature.name).then(function(response) { + expect(response).toBeTruthy(); + done(); + }); + httpBackend.flush(); + }); + + + it('Should be return false with test-feature', function(done) { + window.screen = {width: 1000}; + featureToggleService.isEnabled(feature.name).then(function(response) { + expect(response).toBeFalsy(); + done(); + }); + httpBackend.flush(); + }); + + it('Should be return false with test-other-feature', function(done) { + window.screen = {width: 100}; + featureToggleService.isEnabled(otherFeature.name).then(function(response) { + expect(response).toBeFalsy(); + done(); + }); + httpBackend.flush(); + }); + }); +}); \ No newline at end of file From 551f99142135ddfa44f9f04e809a040919cbf86b Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Fri, 21 Dec 2018 15:50:14 -0300 Subject: [PATCH 023/606] Create tests of mapStateToFeatureService --- .../utils/mapStateToFeatureServiceSpec.js | 38 +++++++++++++++++++ 1 file changed, 38 insertions(+) create mode 100644 frontend/test/specs/utils/mapStateToFeatureServiceSpec.js diff --git a/frontend/test/specs/utils/mapStateToFeatureServiceSpec.js b/frontend/test/specs/utils/mapStateToFeatureServiceSpec.js new file mode 100644 index 000000000..79046fff4 --- /dev/null +++ b/frontend/test/specs/utils/mapStateToFeatureServiceSpec.js @@ -0,0 +1,38 @@ +'use strict'; + + +describe('MapStateToFeatureService Test', function() { + beforeEach(module('app')); + + let mapStateToFeatureService; + + beforeEach(inject(function(MapStateToFeatureService) { + mapStateToFeatureService = MapStateToFeatureService; + + mapStateToFeatureService._statesToFeature = { + 'app.manage-user': 'manage-user', + 'app.edit-inst': 'edit-inst' + }; + })); + + describe('Test getFeatureByState', function() { + + it('Should be return feature', function() { + let feature = mapStateToFeatureService.getFeatureByState('app.manage-user'); + expect(feature).toEqual('manage-user'); + + feature = mapStateToFeatureService.getFeatureByState('app.edit-inst'); + expect(feature).toEqual('edit-inst'); + }); + + it('Should be return undefined with unregistred state', function() { + let feature = mapStateToFeatureService.getFeatureByState('app.manage-inst'); + expect(feature).toBeUndefined(); + }); + }); + + describe('Test containsFeature', function() { + + + }); +}); \ No newline at end of file From 7174cc8d4584df93b990432d066fc803750bb729 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Fri, 21 Dec 2018 16:49:15 -0300 Subject: [PATCH 024/606] Finalize mapStateToFeature test --- .../test/specs/utils/featureToggleServiceSpec.js | 4 ++-- .../specs/utils/mapStateToFeatureServiceSpec.js | 14 +++++++++++++- 2 files changed, 15 insertions(+), 3 deletions(-) diff --git a/frontend/test/specs/utils/featureToggleServiceSpec.js b/frontend/test/specs/utils/featureToggleServiceSpec.js index 41534ad54..97b4cf97c 100644 --- a/frontend/test/specs/utils/featureToggleServiceSpec.js +++ b/frontend/test/specs/utils/featureToggleServiceSpec.js @@ -111,7 +111,7 @@ describe('Test FeatureToggleService', function() { it('Should be return false with test-feature', function(done) { window.screen = {width: 1000}; featureToggleService.isEnabled(feature.name).then(function(response) { - expect(response).toBeFalsy(); + expect(response).toEqual(false); done(); }); httpBackend.flush(); @@ -120,7 +120,7 @@ describe('Test FeatureToggleService', function() { it('Should be return false with test-other-feature', function(done) { window.screen = {width: 100}; featureToggleService.isEnabled(otherFeature.name).then(function(response) { - expect(response).toBeFalsy(); + expect(response).toEqual(false); done(); }); httpBackend.flush(); diff --git a/frontend/test/specs/utils/mapStateToFeatureServiceSpec.js b/frontend/test/specs/utils/mapStateToFeatureServiceSpec.js index 79046fff4..39a4c711e 100644 --- a/frontend/test/specs/utils/mapStateToFeatureServiceSpec.js +++ b/frontend/test/specs/utils/mapStateToFeatureServiceSpec.js @@ -33,6 +33,18 @@ describe('MapStateToFeatureService Test', function() { describe('Test containsFeature', function() { - + it('Should be return true', function() { + let contain = mapStateToFeatureService.containsFeature('app.edit-inst'); + expect(contain).toBeTruthy(); + + contain = mapStateToFeatureService.containsFeature('app.manage-user'); + expect(contain).toBeTruthy(); + }); + + + it('Should be return false', function() { + let contain = mapStateToFeatureService.containsFeature('app.manage-inst'); + expect(contain).toEqual(false); + }); }); }); \ No newline at end of file From 937b786f442542ec1a179fe9be039dd495d4324e Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Mon, 7 Jan 2019 15:02:23 -0300 Subject: [PATCH 025/606] send exception if param featureName is undefined --- frontend/test/specs/utils/featureToggleServiceSpec.js | 4 ++++ frontend/utils/featureToggleService.js | 3 +++ 2 files changed, 7 insertions(+) diff --git a/frontend/test/specs/utils/featureToggleServiceSpec.js b/frontend/test/specs/utils/featureToggleServiceSpec.js index 97b4cf97c..368bb38aa 100644 --- a/frontend/test/specs/utils/featureToggleServiceSpec.js +++ b/frontend/test/specs/utils/featureToggleServiceSpec.js @@ -125,5 +125,9 @@ describe('Test FeatureToggleService', function() { }); httpBackend.flush(); }); + + it('Should be generated exception', function() { + expect(() => featureToggleService.isEnabled()).toThrow(new Error("Required param featureName")); + }); }); }); \ No newline at end of file diff --git a/frontend/utils/featureToggleService.js b/frontend/utils/featureToggleService.js index e8e976320..ed7eafcbb 100644 --- a/frontend/utils/featureToggleService.js +++ b/frontend/utils/featureToggleService.js @@ -36,6 +36,9 @@ * @return {Promise} Promise that when it is resolved it returns a boolean indicating whether or not it is enabled. */ service.isEnabled = function isEnabled(featureName) { + if (!featureName) + throw new Error("Required param featureName"); + return service.getFeature(featureName).then(function(response) { const feature = _.first(response); const disableMobile = _.get(feature, 'enable_mobile') === 'DISABLED'; From 2adea08f67fe776bcdfc083465945722714a2104 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Mon, 7 Jan 2019 15:18:38 -0300 Subject: [PATCH 026/606] return rejected promise if param featureName is undefined --- frontend/app.js | 8 +++++++- frontend/test/specs/utils/featureToggleServiceSpec.js | 7 +++++-- frontend/utils/featureToggleService.js | 4 +++- 3 files changed, 15 insertions(+), 4 deletions(-) diff --git a/frontend/app.js b/frontend/app.js index 46912c00a..d44cef127 100644 --- a/frontend/app.js +++ b/frontend/app.js @@ -505,7 +505,7 @@ }); }); - app.run(function featureToggleInterceptor(FeatureToggleService, MapStateToFeatureService, $transitions, STATES) { + app.run(function featureToggleInterceptor(FeatureToggleService, MapStateToFeatureService, $transitions, STATES, MessageService) { $transitions.onBefore({ to: function(state) { @@ -524,6 +524,12 @@ "status": "403" }); } + }).catch(function(message) { + MessageService.showToast(message); + return transition.router.stateService.target(STATES.ERROR, { + "msg": "Desculpa! Este recurso ainda não está disponível.", + "status": "403" + }); }); }); }); diff --git a/frontend/test/specs/utils/featureToggleServiceSpec.js b/frontend/test/specs/utils/featureToggleServiceSpec.js index 368bb38aa..3f88638ab 100644 --- a/frontend/test/specs/utils/featureToggleServiceSpec.js +++ b/frontend/test/specs/utils/featureToggleServiceSpec.js @@ -126,8 +126,11 @@ describe('Test FeatureToggleService', function() { httpBackend.flush(); }); - it('Should be generated exception', function() { - expect(() => featureToggleService.isEnabled()).toThrow(new Error("Required param featureName")); + it('Should be generated exception', function(done) { + featureToggleService.isEnabled().catch(function(message) { + expect(message).toEqual("Required param featureName"); + done(); + }); }); }); }); \ No newline at end of file diff --git a/frontend/utils/featureToggleService.js b/frontend/utils/featureToggleService.js index ed7eafcbb..18f89d39f 100644 --- a/frontend/utils/featureToggleService.js +++ b/frontend/utils/featureToggleService.js @@ -37,7 +37,9 @@ */ service.isEnabled = function isEnabled(featureName) { if (!featureName) - throw new Error("Required param featureName"); + return new Promise((resolve, reject) => { + reject("Required param featureName"); + }); return service.getFeature(featureName).then(function(response) { const feature = _.first(response); From 922aad98c8e4adf6fac34e9fa479c0792a8e2ec2 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Mon, 7 Jan 2019 17:08:51 -0300 Subject: [PATCH 027/606] Create ResetFeatureHandler in admin.py --- backend/admin.py | 25 +++++++++++++++++++++++++ 1 file changed, 25 insertions(+) diff --git a/backend/admin.py b/backend/admin.py index d96e5a8d1..ca32d57ee 100644 --- a/backend/admin.py +++ b/backend/admin.py @@ -17,6 +17,7 @@ from models import Comment from models import Invite from models import Event +from models import Feature from utils import NotAuthorizedException from google.appengine.ext import ndb from google.appengine.api import search @@ -37,6 +38,21 @@ delectus, ut aut reiciendis voluptatibus maiores alias consequatur \ aut perferendis doloribus asperiores repellat.' +features = [ + { + "name": 'manage-inst-edit', + "enable_mobile": "DISABLED", + "enable_desktop": "ALL" + } +] + +def reset_features(): + features_query = Feature.query().fetch(keys_only=True) + ndb.delete_multi(features_query) + + for feature in features: + Feature.create(**feature) + def add_comments_to_post(user, user_reply, post, institution, comments_qnt=3): """Add comments to post.""" @@ -520,11 +536,20 @@ def get(self): splab.posts = [] splab.put() + reset_features() + jsonList.append({"msg": "database initialized with a few posts"}) self.response.write(json.dumps(jsonList)) + +class ResetFeaturesHandler(BaseHandler): + def get(self): + reset_features() + self.response.write({"msg": "database initialized with a few features"}) + app = webapp2.WSGIApplication([ ('/admin/reset', ResetHandler), + ('/admin/reset-features', ResetFeaturesHandler) ], debug=True) From aeaca9415c8c97b10eabe8756b4ded4e9a80278d Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Tue, 8 Jan 2019 12:10:40 -0300 Subject: [PATCH 028/606] create initial layout of feature toggles service --- ecis | 36 +- feature-toggles/app.js | 44 + feature-toggles/config.js | 10 + feature-toggles/favicon.ico | Bin 0 -> 32038 bytes feature-toggles/feature.yaml | 17 + feature-toggles/index.html | 46 + feature-toggles/libs/angular-ui-router.js | 10075 +++++++++++++++++++ feature-toggles/main/main.html | 3 + feature-toggles/main/mainControler.js | 10 + feature-toggles/manage/manage-toggles.html | 1 + feature-toggles/manage/manageToggles.js | 9 + 11 files changed, 10244 insertions(+), 7 deletions(-) create mode 100644 feature-toggles/app.js create mode 100644 feature-toggles/config.js create mode 100644 feature-toggles/favicon.ico create mode 100644 feature-toggles/feature.yaml create mode 100644 feature-toggles/index.html create mode 100644 feature-toggles/libs/angular-ui-router.js create mode 100644 feature-toggles/main/main.html create mode 100644 feature-toggles/main/mainControler.js create mode 100644 feature-toggles/manage/manage-toggles.html create mode 100644 feature-toggles/manage/manageToggles.js diff --git a/ecis b/ecis index 093ef874e..fe869d50f 100755 --- a/ecis +++ b/ecis @@ -14,18 +14,22 @@ FRONTEND_YAML=frontend/frontend.yaml BACKEND_YAML=backend/backend.yaml WORKER_YAML=backend/worker.yaml SUPPORT_YAML=support/support.yaml +FEATURE_YAML=feature-toggles/feature.yaml LANDINGPAGE_LOCAL="localhost:8080" FRONTEND_LOCAL="localhost:8081" BACKEND_LOCAL="localhost:8082" SUPPORT_LOCAL="localhost:8083" +FEATURE_LOCAL="localhost:8084" + FRONTEND_CONFIG_FILE="frontend/config.js" SUPPORT_CONFIG_FILE="support/config.js" LANDINGPAGE_CONFIG_FILE="landing/config.js" +FEATURE_CONFIG_FILE="feature-toggles/config.js" SW_FILE="frontend/sw.js" -git update-index --skip-worktree $FRONTEND_CONFIG_FILE $SUPPORT_CONFIG_FILE $LANDINGPAGE_CONFIG_FILE +git update-index --skip-worktree $FRONTEND_CONFIG_FILE $SUPPORT_CONFIG_FILE $LANDINGPAGE_CONFIG_FILE $FEATURE_CONFIG_FILE PY_ENV=backend/py_env @@ -77,6 +81,7 @@ function set_config_file { sed -i "$line s|.*| $service: '$url',|" $FRONTEND_CONFIG_FILE sed -i "$line s|.*| $service: '$url',|" $SUPPORT_CONFIG_FILE sed -i "$line s|.*| $service: '$url',|" $LANDINGPAGE_CONFIG_FILE + sed -i "$line s|.*| $service: '$url',|" $FEATURE_CONFIG_FILE catch_error $? "Frontend will use ${bold}$url${_bold} as $service_name." catch_error $? "Support will use ${bold}$url${_bold} as $service_name." @@ -114,13 +119,23 @@ function set_frontend_url { set_config_file $line $service $url $service_name; } +function set_feature_url { + url=$(set_service_url $1 $2); + service="FEATURE_URL"; + line=7; + service_name="feature"; + set_config_file $line $service $url $service_name; +} + function set_app_version_config { - sed -i "8s|.*| APP_VERSION: '$1'|" $FRONTEND_CONFIG_FILE + sed -i "9s|.*| APP_VERSION: '$1'|" $FRONTEND_CONFIG_FILE catch_error $? "APP VERSION on Frontend $1" - sed -i "8s|.*| APP_VERSION: '$1'|" $SUPPORT_CONFIG_FILE + sed -i "9s|.*| APP_VERSION: '$1'|" $SUPPORT_CONFIG_FILE catch_error $? "APP VERSION on Support $1" - sed -i "8s|.*| APP_VERSION: '$1'|" $LANDINGPAGE_CONFIG_FILE + sed -i "9s|.*| APP_VERSION: '$1'|" $LANDINGPAGE_CONFIG_FILE catch_error $? "APP VERSION on LandingPage $1" + sed -i "9s|.*| APP_VERSION: '$1'|" $FEATURE_CONFIG_FILE + catch_error $? "APP VERSION on Feature $1" } function set_cache_suffix_sw { @@ -212,6 +227,7 @@ case "$1" in set_landingpage_url local $LANDINGPAGE_LOCAL set_support_url local $SUPPORT_LOCAL set_frontend_url local $FRONTEND_LOCAL + set_feature_url local $FRONTEND_LOCAL if [[ -n $2 ]] && [ $2 = "--enable_datastore_emulator" ] ; then gcloud beta emulators datastore start --host-port=0.0.0.0:8586 & @@ -223,7 +239,7 @@ case "$1" in fi $(gcloud beta emulators datastore env-init) - dev_appserver.py $APP_YAML $FRONTEND_YAML $BACKEND_YAML $SUPPORT_YAML $WORKER_YAML -A development-cis --support_datastore_emulator=$emulator $other_parameter + dev_appserver.py $APP_YAML $FRONTEND_YAML $BACKEND_YAML $SUPPORT_YAML $FEATURE_YAML $WORKER_YAML -A development-cis --support_datastore_emulator=$emulator $other_parameter $(gcloud beta emulators datastore env-unset) ;; @@ -295,12 +311,14 @@ case "$1" in APP_NAME="development-cis" SUPPORT_DOMAIN="support-dot-$APP_NAME.appspot.com" FRONTEND_DOMAIN="frontend-dot-$APP_NAME.appspot.com" + FEATURE_DOMAIN="frontend-dot-$APP_NAME.appspot.com" break ;; "eciis-splab") APP_NAME="eciis-splab" SUPPORT_DOMAIN="support.plataformacis.org" FRONTEND_DOMAIN="frontend.plataformacis.org" + FEATURE_DOMAIN="frontend.plataformacis.org" break ;; "Other") @@ -354,17 +372,21 @@ case "$1" in url=$version"."$FRONTEND_DOMAIN set_frontend_url deploy $url + url=$version"."$FEATURE_DOMAIN + set_frontend_url deploy $url + if [ ! -z $3 ]; then # Especified one or more yaml configuration files gcloud app deploy --version $version --no-promote $3 else - gcloud app deploy --version $version --no-promote $APP_YAML $FRONTEND_YAML $BACKEND_YAML $SUPPORT_YAML $WORKER_YAML queue.yaml + gcloud app deploy --version $version --no-promote $APP_YAML $FRONTEND_YAML $BACKEND_YAML $SUPPORT_YAML $FEATURE_YAML $WORKER_YAML queue.yaml fi else set_backend_url deploy $BACKEND_DOMAIN set_landingpage_url deploy $DOMAIN set_support_url deploy $SUPPORT_DOMAIN set_frontend_url deploy $FRONTEND_DOMAIN - gcloud app deploy $APP_YAML $FRONTEND_YAML $BACKEND_YAML $SUPPORT_YAML $WORKER_YAML + set_frontend_url deploy $FEATURE_DOMAIN + gcloud app deploy $APP_YAML $FRONTEND_YAML $BACKEND_YAML $SUPPORT_YAML $FEATURE_YAML $WORKER_YAML fi exit 0 diff --git a/feature-toggles/app.js b/feature-toggles/app.js new file mode 100644 index 000000000..4d6b3020b --- /dev/null +++ b/feature-toggles/app.js @@ -0,0 +1,44 @@ +(function() { + 'use strict'; + + const app = angular.module('app', [ + 'ngMaterial', + 'ui.router', + 'ngAnimate', + 'ngMessages', + ]); + + + app.config(function($mdIconProvider, $mdThemingProvider, $urlMatcherFactoryProvider, $urlRouterProvider, $locationProvider, $stateProvider) { + $mdIconProvider.fontSet('md', 'material-icons'); + $mdThemingProvider.theme('docs-dark'); + $mdThemingProvider.theme('input') + .primaryPalette('green'); + $mdThemingProvider.theme('dialogTheme') + .primaryPalette('teal'); + + $urlMatcherFactoryProvider.caseInsensitive(true); + + $stateProvider + .state("main", { + abstract: true, + views: { + main: { + templateUrl: "app/main/main.html", + controller: "MainController as mainCtrl" + } + } + }).state("manage-features", { + url: "/", + views: { + content: { + templateUrl: "app/manage/manage-toggles.html" + } + } + }); + + $urlRouterProvider.otherwise("/"); + + $locationProvider.html5Mode(true); + }); +})(); \ No newline at end of file diff --git a/feature-toggles/config.js b/feature-toggles/config.js new file mode 100644 index 000000000..336ccee67 --- /dev/null +++ b/feature-toggles/config.js @@ -0,0 +1,10 @@ +"use strict"; + +var Config = { + BACKEND_URL: 'http://localhost:8082', + LANDINGPAGE_URL: 'http://localhost:8080', + SUPPORT_URL: 'http://localhost:8083', + FEATURE_URL: 'http://localhost:8081', + FEATURE_URL: 'http://localhost:8083', + APP_VERSION: 'create-feature-toggles-service' +}; diff --git a/feature-toggles/favicon.ico b/feature-toggles/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..1d395073d639a42d5ed78f8ce87e6ca95d6386fe GIT binary patch literal 32038 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 M1Ni3te-{V-ABs94^8f$< literal 0 HcmV?d00001 diff --git a/feature-toggles/feature.yaml b/feature-toggles/feature.yaml new file mode 100644 index 000000000..813fcd904 --- /dev/null +++ b/feature-toggles/feature.yaml @@ -0,0 +1,17 @@ +runtime: python27 +api_version: 1 +threadsafe: yes +service: feature + +default_expiration: "1s" + +handlers: +- url: /app/(.*) + secure: always + static_files: \1 + upload: (.*) + +- url: /(.*) + secure: always + static_files: index.html + upload: index.html \ No newline at end of file diff --git a/feature-toggles/index.html b/feature-toggles/index.html new file mode 100644 index 000000000..bebb9401d --- /dev/null +++ b/feature-toggles/index.html @@ -0,0 +1,46 @@ + + + + + + + + Plataforma CIS + + + + + + + + + + + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/feature-toggles/libs/angular-ui-router.js b/feature-toggles/libs/angular-ui-router.js new file mode 100644 index 000000000..4e69b5e20 --- /dev/null +++ b/feature-toggles/libs/angular-ui-router.js @@ -0,0 +1,10075 @@ +/** + * State-based routing for AngularJS 1.x + * NOTICE: This monolithic bundle also bundles the @uirouter/core code. + * This causes it to be incompatible with plugins that depend on @uirouter/core. + * We recommend switching to the ui-router-core.js and ui-router-angularjs.js bundles instead. + * For more information, see https://ui-router.github.io/blog/uirouter-for-angularjs-umd-bundles + * @version v1.0.10 + * @link https://ui-router.github.io + * @license MIT License, http://www.opensource.org/licenses/MIT + */ +(function (global, factory) { + typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports, require('angular')) : + typeof define === 'function' && define.amd ? define(['exports', 'angular'], factory) : + (factory((global['@uirouter/angularjs'] = {}),global.angular)); +}(this, (function (exports,ng_from_import) { 'use strict'; + +var ng_from_global = angular; +var ng = (ng_from_import && ng_from_import.module) ? ng_from_import : ng_from_global; + +/** + * Higher order functions + * + * These utility functions are exported, but are subject to change without notice. + * + * @module common_hof + */ /** */ +/** + * Returns a new function for [Partial Application](https://en.wikipedia.org/wiki/Partial_application) of the original function. + * + * Given a function with N parameters, returns a new function that supports partial application. + * The new function accepts anywhere from 1 to N parameters. When that function is called with M parameters, + * where M is less than N, it returns a new function that accepts the remaining parameters. It continues to + * accept more parameters until all N parameters have been supplied. + * + * + * This contrived example uses a partially applied function as an predicate, which returns true + * if an object is found in both arrays. + * @example + * ``` + * // returns true if an object is in both of the two arrays + * function inBoth(array1, array2, object) { + * return array1.indexOf(object) !== -1 && + * array2.indexOf(object) !== 1; + * } + * let obj1, obj2, obj3, obj4, obj5, obj6, obj7 + * let foos = [obj1, obj3] + * let bars = [obj3, obj4, obj5] + * + * // A curried "copy" of inBoth + * let curriedInBoth = curry(inBoth); + * // Partially apply both the array1 and array2 + * let inFoosAndBars = curriedInBoth(foos, bars); + * + * // Supply the final argument; since all arguments are + * // supplied, the original inBoth function is then called. + * let obj1InBoth = inFoosAndBars(obj1); // false + * + * // Use the inFoosAndBars as a predicate. + * // Filter, on each iteration, supplies the final argument + * let allObjs = [ obj1, obj2, obj3, obj4, obj5, obj6, obj7 ]; + * let foundInBoth = allObjs.filter(inFoosAndBars); // [ obj3 ] + * + * ``` + * + * Stolen from: http://stackoverflow.com/questions/4394747/javascript-curry-function + * + * @param fn + * @returns {*|function(): (*|any)} + */ +function curry(fn) { + var initial_args = [].slice.apply(arguments, [1]); + var func_args_length = fn.length; + function curried(args) { + if (args.length >= func_args_length) + return fn.apply(null, args); + return function () { + return curried(args.concat([].slice.apply(arguments))); + }; + } + return curried(initial_args); +} +/** + * Given a varargs list of functions, returns a function that composes the argument functions, right-to-left + * given: f(x), g(x), h(x) + * let composed = compose(f,g,h) + * then, composed is: f(g(h(x))) + */ +function compose() { + var args = arguments; + var start = args.length - 1; + return function () { + var i = start, result = args[start].apply(this, arguments); + while (i--) + result = args[i].call(this, result); + return result; + }; +} +/** + * Given a varargs list of functions, returns a function that is composes the argument functions, left-to-right + * given: f(x), g(x), h(x) + * let piped = pipe(f,g,h); + * then, piped is: h(g(f(x))) + */ +function pipe() { + var funcs = []; + for (var _i = 0; _i < arguments.length; _i++) { + funcs[_i] = arguments[_i]; + } + return compose.apply(null, [].slice.call(arguments).reverse()); +} +/** + * Given a property name, returns a function that returns that property from an object + * let obj = { foo: 1, name: "blarg" }; + * let getName = prop("name"); + * getName(obj) === "blarg" + */ +var prop = function (name) { + return function (obj) { return obj && obj[name]; }; +}; +/** + * Given a property name and a value, returns a function that returns a boolean based on whether + * the passed object has a property that matches the value + * let obj = { foo: 1, name: "blarg" }; + * let getName = propEq("name", "blarg"); + * getName(obj) === true + */ +var propEq = curry(function (name, val, obj) { return obj && obj[name] === val; }); +/** + * Given a dotted property name, returns a function that returns a nested property from an object, or undefined + * let obj = { id: 1, nestedObj: { foo: 1, name: "blarg" }, }; + * let getName = prop("nestedObj.name"); + * getName(obj) === "blarg" + * let propNotFound = prop("this.property.doesnt.exist"); + * propNotFound(obj) === undefined + */ +var parse = function (name) { + return pipe.apply(null, name.split(".").map(prop)); +}; +/** + * Given a function that returns a truthy or falsey value, returns a + * function that returns the opposite (falsey or truthy) value given the same inputs + */ +var not = function (fn) { + return function () { + var args = []; + for (var _i = 0; _i < arguments.length; _i++) { + args[_i] = arguments[_i]; + } + return !fn.apply(null, args); + }; +}; +/** + * Given two functions that return truthy or falsey values, returns a function that returns truthy + * if both functions return truthy for the given arguments + */ +function and(fn1, fn2) { + return function () { + var args = []; + for (var _i = 0; _i < arguments.length; _i++) { + args[_i] = arguments[_i]; + } + return fn1.apply(null, args) && fn2.apply(null, args); + }; +} +/** + * Given two functions that return truthy or falsey values, returns a function that returns truthy + * if at least one of the functions returns truthy for the given arguments + */ +function or(fn1, fn2) { + return function () { + var args = []; + for (var _i = 0; _i < arguments.length; _i++) { + args[_i] = arguments[_i]; + } + return fn1.apply(null, args) || fn2.apply(null, args); + }; +} +/** + * Check if all the elements of an array match a predicate function + * + * @param fn1 a predicate function `fn1` + * @returns a function which takes an array and returns true if `fn1` is true for all elements of the array + */ +var all = function (fn1) { + return function (arr) { return arr.reduce(function (b, x) { return b && !!fn1(x); }, true); }; +}; +var any = function (fn1) { + return function (arr) { return arr.reduce(function (b, x) { return b || !!fn1(x); }, false); }; +}; +/** Given a class, returns a Predicate function that returns true if the object is of that class */ +var is = function (ctor) { + return function (obj) { + return (obj != null && obj.constructor === ctor || obj instanceof ctor); + }; +}; +/** Given a value, returns a Predicate function that returns true if another value is === equal to the original value */ +var eq = function (val) { return function (other) { + return val === other; +}; }; +/** Given a value, returns a function which returns the value */ +var val = function (v) { return function () { return v; }; }; +function invoke(fnName, args) { + return function (obj) { + return obj[fnName].apply(obj, args); + }; +} +/** + * Sorta like Pattern Matching (a functional programming conditional construct) + * + * See http://c2.com/cgi/wiki?PatternMatching + * + * This is a conditional construct which allows a series of predicates and output functions + * to be checked and then applied. Each predicate receives the input. If the predicate + * returns truthy, then its matching output function (mapping function) is provided with + * the input and, then the result is returned. + * + * Each combination (2-tuple) of predicate + output function should be placed in an array + * of size 2: [ predicate, mapFn ] + * + * These 2-tuples should be put in an outer array. + * + * @example + * ``` + * + * // Here's a 2-tuple where the first element is the isString predicate + * // and the second element is a function that returns a description of the input + * let firstTuple = [ angular.isString, (input) => `Heres your string ${input}` ]; + * + * // Second tuple: predicate "isNumber", mapfn returns a description + * let secondTuple = [ angular.isNumber, (input) => `(${input}) That's a number!` ]; + * + * let third = [ (input) => input === null, (input) => `Oh, null...` ]; + * + * let fourth = [ (input) => input === undefined, (input) => `notdefined` ]; + * + * let descriptionOf = pattern([ firstTuple, secondTuple, third, fourth ]); + * + * console.log(descriptionOf(undefined)); // 'notdefined' + * console.log(descriptionOf(55)); // '(55) That's a number!' + * console.log(descriptionOf("foo")); // 'Here's your string foo' + * ``` + * + * @param struct A 2D array. Each element of the array should be an array, a 2-tuple, + * with a Predicate and a mapping/output function + * @returns {function(any): *} + */ +function pattern(struct) { + return function (x) { + for (var i = 0; i < struct.length; i++) { + if (struct[i][0](x)) + return struct[i][1](x); + } + }; +} + +/** + * @coreapi + * @module core + */ +/** + * Matches state names using glob-like pattern strings. + * + * Globs can be used in specific APIs including: + * + * - [[StateService.is]] + * - [[StateService.includes]] + * - The first argument to Hook Registration functions like [[TransitionService.onStart]] + * - [[HookMatchCriteria]] and [[HookMatchCriterion]] + * + * A `Glob` string is a pattern which matches state names. + * Nested state names are split into segments (separated by a dot) when processing. + * The state named `foo.bar.baz` is split into three segments ['foo', 'bar', 'baz'] + * + * Globs work according to the following rules: + * + * ### Exact match: + * + * The glob `'A.B'` matches the state named exactly `'A.B'`. + * + * | Glob |Matches states named|Does not match state named| + * |:------------|:--------------------|:---------------------| + * | `'A'` | `'A'` | `'B'` , `'A.C'` | + * | `'A.B'` | `'A.B'` | `'A'` , `'A.B.C'` | + * | `'foo'` | `'foo'` | `'FOO'` , `'foo.bar'`| + * + * ### Single star (`*`) + * + * A single star (`*`) is a wildcard that matches exactly one segment. + * + * | Glob |Matches states named |Does not match state named | + * |:------------|:---------------------|:--------------------------| + * | `'*'` | `'A'` , `'Z'` | `'A.B'` , `'Z.Y.X'` | + * | `'A.*'` | `'A.B'` , `'A.C'` | `'A'` , `'A.B.C'` | + * | `'A.*.*'` | `'A.B.C'` , `'A.X.Y'`| `'A'`, `'A.B'` , `'Z.Y.X'`| + * + * ### Double star (`**`) + * + * A double star (`'**'`) is a wildcard that matches *zero or more segments* + * + * | Glob |Matches states named |Does not match state named | + * |:------------|:----------------------------------------------|:----------------------------------| + * | `'**'` | `'A'` , `'A.B'`, `'Z.Y.X'` | (matches all states) | + * | `'A.**'` | `'A'` , `'A.B'` , `'A.C.X'` | `'Z.Y.X'` | + * | `'**.X'` | `'X'` , `'A.X'` , `'Z.Y.X'` | `'A'` , `'A.login.Z'` | + * | `'A.**.X'` | `'A.X'` , `'A.B.X'` , `'A.B.C.X'` | `'A'` , `'A.B.C'` | + * + */ +var Glob = /** @class */ (function () { + function Glob(text) { + this.text = text; + this.glob = text.split('.'); + var regexpString = this.text.split('.') + .map(function (seg) { + if (seg === '**') + return '(?:|(?:\\.[^.]*)*)'; + if (seg === '*') + return '\\.[^.]*'; + return '\\.' + seg; + }).join(''); + this.regexp = new RegExp("^" + regexpString + "$"); + } + Glob.prototype.matches = function (name) { + return this.regexp.test('.' + name); + }; + /** Returns true if the string has glob-like characters in it */ + Glob.is = function (text) { + return !!/[!,*]+/.exec(text); + }; + /** Returns a glob from the string, or null if the string isn't Glob-like */ + Glob.fromString = function (text) { + return Glob.is(text) ? new Glob(text) : null; + }; + return Glob; +}()); + +/** + * Internal representation of a UI-Router state. + * + * Instances of this class are created when a [[StateDeclaration]] is registered with the [[StateRegistry]]. + * + * A registered [[StateDeclaration]] is augmented with a getter ([[StateDeclaration.$$state]]) which returns the corresponding [[StateObject]] object. + * + * This class prototypally inherits from the corresponding [[StateDeclaration]]. + * Each of its own properties (i.e., `hasOwnProperty`) are built using builders from the [[StateBuilder]]. + */ +var StateObject = /** @class */ (function () { + /** @deprecated use State.create() */ + function StateObject(config) { + return StateObject.create(config || {}); + } + /** + * Create a state object to put the private/internal implementation details onto. + * The object's prototype chain looks like: + * (Internal State Object) -> (Copy of State.prototype) -> (State Declaration object) -> (State Declaration's prototype...) + * + * @param stateDecl the user-supplied State Declaration + * @returns {StateObject} an internal State object + */ + StateObject.create = function (stateDecl) { + stateDecl = StateObject.isStateClass(stateDecl) ? new stateDecl() : stateDecl; + var state = inherit(inherit(stateDecl, StateObject.prototype)); + stateDecl.$$state = function () { return state; }; + state.self = stateDecl; + state.__stateObjectCache = { + nameGlob: Glob.fromString(state.name) // might return null + }; + return state; + }; + /** + * Returns true if the provided parameter is the same state. + * + * Compares the identity of the state against the passed value, which is either an object + * reference to the actual `State` instance, the original definition object passed to + * `$stateProvider.state()`, or the fully-qualified name. + * + * @param ref Can be one of (a) a `State` instance, (b) an object that was passed + * into `$stateProvider.state()`, (c) the fully-qualified name of a state as a string. + * @returns Returns `true` if `ref` matches the current `State` instance. + */ + StateObject.prototype.is = function (ref) { + return this === ref || this.self === ref || this.fqn() === ref; + }; + /** + * @deprecated this does not properly handle dot notation + * @returns Returns a dot-separated name of the state. + */ + StateObject.prototype.fqn = function () { + if (!this.parent || !(this.parent instanceof this.constructor)) + return this.name; + var name = this.parent.fqn(); + return name ? name + "." + this.name : this.name; + }; + /** + * Returns the root node of this state's tree. + * + * @returns The root of this state's tree. + */ + StateObject.prototype.root = function () { + return this.parent && this.parent.root() || this; + }; + /** + * Gets the state's `Param` objects + * + * Gets the list of [[Param]] objects owned by the state. + * If `opts.inherit` is true, it also includes the ancestor states' [[Param]] objects. + * If `opts.matchingKeys` exists, returns only `Param`s whose `id` is a key on the `matchingKeys` object + * + * @param opts options + */ + StateObject.prototype.parameters = function (opts) { + opts = defaults(opts, { inherit: true, matchingKeys: null }); + var inherited = opts.inherit && this.parent && this.parent.parameters() || []; + return inherited.concat(values(this.params)) + .filter(function (param) { return !opts.matchingKeys || opts.matchingKeys.hasOwnProperty(param.id); }); + }; + /** + * Returns a single [[Param]] that is owned by the state + * + * If `opts.inherit` is true, it also searches the ancestor states` [[Param]]s. + * @param id the name of the [[Param]] to return + * @param opts options + */ + StateObject.prototype.parameter = function (id, opts) { + if (opts === void 0) { opts = {}; } + return (this.url && this.url.parameter(id, opts) || + find(values(this.params), propEq('id', id)) || + opts.inherit && this.parent && this.parent.parameter(id)); + }; + StateObject.prototype.toString = function () { + return this.fqn(); + }; + /** Predicate which returns true if the object is an class with @State() decorator */ + StateObject.isStateClass = function (stateDecl) { + return isFunction(stateDecl) && stateDecl['__uiRouterState'] === true; + }; + /** Predicate which returns true if the object is an internal [[StateObject]] object */ + StateObject.isState = function (obj) { + return isObject(obj['__stateObjectCache']); + }; + return StateObject; +}()); + +/** Predicates + * + * These predicates return true/false based on the input. + * Although these functions are exported, they are subject to change without notice. + * + * @module common_predicates + */ +/** */ +var toStr = Object.prototype.toString; +var tis = function (t) { return function (x) { return typeof (x) === t; }; }; +var isUndefined = tis('undefined'); +var isDefined = not(isUndefined); +var isNull = function (o) { return o === null; }; +var isNullOrUndefined = or(isNull, isUndefined); +var isFunction = tis('function'); +var isNumber = tis('number'); +var isString = tis('string'); +var isObject = function (x) { return x !== null && typeof x === 'object'; }; +var isArray = Array.isArray; +var isDate = (function (x) { return toStr.call(x) === '[object Date]'; }); +var isRegExp = (function (x) { return toStr.call(x) === '[object RegExp]'; }); +var isState = StateObject.isState; +/** + * Predicate which checks if a value is injectable + * + * A value is "injectable" if it is a function, or if it is an ng1 array-notation-style array + * where all the elements in the array are Strings, except the last one, which is a Function + */ +function isInjectable(val$$1) { + if (isArray(val$$1) && val$$1.length) { + var head = val$$1.slice(0, -1), tail = val$$1.slice(-1); + return !(head.filter(not(isString)).length || tail.filter(not(isFunction)).length); + } + return isFunction(val$$1); +} +/** + * Predicate which checks if a value looks like a Promise + * + * It is probably a Promise if it's an object, and it has a `then` property which is a Function + */ +var isPromise = and(isObject, pipe(prop('then'), isFunction)); + +var notImplemented = function (fnname) { return function () { + throw new Error(fnname + "(): No coreservices implementation for UI-Router is loaded."); +}; }; +var services = { + $q: undefined, + $injector: undefined, +}; + +/** + * Random utility functions used in the UI-Router code + * + * These functions are exported, but are subject to change without notice. + * + * @preferred + * @module common + */ +/** for typedoc */ +var root = (typeof self === 'object' && self.self === self && self) || + (typeof global === 'object' && global.global === global && global) || undefined; +var angular$1 = root.angular || {}; +var fromJson = angular$1.fromJson || JSON.parse.bind(JSON); +var toJson = angular$1.toJson || JSON.stringify.bind(JSON); +var forEach = angular$1.forEach || _forEach; +var extend = Object.assign || _extend; +var equals = angular$1.equals || _equals; +function identity(x) { return x; } +function noop$1() { } +/** + * Builds proxy functions on the `to` object which pass through to the `from` object. + * + * For each key in `fnNames`, creates a proxy function on the `to` object. + * The proxy function calls the real function on the `from` object. + * + * + * #### Example: + * This example creates an new class instance whose functions are prebound to the new'd object. + * ```js + * class Foo { + * constructor(data) { + * // Binds all functions from Foo.prototype to 'this', + * // then copies them to 'this' + * bindFunctions(Foo.prototype, this, this); + * this.data = data; + * } + * + * log() { + * console.log(this.data); + * } + * } + * + * let myFoo = new Foo([1,2,3]); + * var logit = myFoo.log; + * logit(); // logs [1, 2, 3] from the myFoo 'this' instance + * ``` + * + * #### Example: + * This example creates a bound version of a service function, and copies it to another object + * ``` + * + * var SomeService = { + * this.data = [3, 4, 5]; + * this.log = function() { + * console.log(this.data); + * } + * } + * + * // Constructor fn + * function OtherThing() { + * // Binds all functions from SomeService to SomeService, + * // then copies them to 'this' + * bindFunctions(SomeService, this, SomeService); + * } + * + * let myOtherThing = new OtherThing(); + * myOtherThing.log(); // logs [3, 4, 5] from SomeService's 'this' + * ``` + * + * @param source A function that returns the source object which contains the original functions to be bound + * @param target A function that returns the target object which will receive the bound functions + * @param bind A function that returns the object which the functions will be bound to + * @param fnNames The function names which will be bound (Defaults to all the functions found on the 'from' object) + * @param latebind If true, the binding of the function is delayed until the first time it's invoked + */ +function createProxyFunctions(source, target, bind, fnNames, latebind) { + if (latebind === void 0) { latebind = false; } + var bindFunction = function (fnName) { + return source()[fnName].bind(bind()); + }; + var makeLateRebindFn = function (fnName) { return function lateRebindFunction() { + target[fnName] = bindFunction(fnName); + return target[fnName].apply(null, arguments); + }; }; + fnNames = fnNames || Object.keys(source()); + return fnNames.reduce(function (acc, name) { + acc[name] = latebind ? makeLateRebindFn(name) : bindFunction(name); + return acc; + }, target); +} +/** + * prototypal inheritance helper. + * Creates a new object which has `parent` object as its prototype, and then copies the properties from `extra` onto it + */ +var inherit = function (parent, extra) { + return extend(Object.create(parent), extra); +}; +/** Given an array, returns true if the object is found in the array, (using indexOf) */ +var inArray = curry(_inArray); +function _inArray(array, obj) { + return array.indexOf(obj) !== -1; +} +/** + * Given an array, and an item, if the item is found in the array, it removes it (in-place). + * The same array is returned + */ +var removeFrom = curry(_removeFrom); +function _removeFrom(array, obj) { + var idx = array.indexOf(obj); + if (idx >= 0) + array.splice(idx, 1); + return array; +} +/** pushes a values to an array and returns the value */ +var pushTo = curry(_pushTo); +function _pushTo(arr, val$$1) { + return (arr.push(val$$1), val$$1); +} +/** Given an array of (deregistration) functions, calls all functions and removes each one from the source array */ +var deregAll = function (functions) { + return functions.slice().forEach(function (fn) { + typeof fn === 'function' && fn(); + removeFrom(functions, fn); + }); +}; +/** + * Applies a set of defaults to an options object. The options object is filtered + * to only those properties of the objects in the defaultsList. + * Earlier objects in the defaultsList take precedence when applying defaults. + */ +function defaults(opts) { + var defaultsList = []; + for (var _i = 1; _i < arguments.length; _i++) { + defaultsList[_i - 1] = arguments[_i]; + } + var _defaultsList = defaultsList.concat({}).reverse(); + var defaultVals = extend.apply(null, _defaultsList); + return extend({}, defaultVals, pick(opts || {}, Object.keys(defaultVals))); +} +/** Reduce function that merges each element of the list into a single object, using extend */ +var mergeR = function (memo, item) { return extend(memo, item); }; +/** + * Finds the common ancestor path between two states. + * + * @param {Object} first The first state. + * @param {Object} second The second state. + * @return {Array} Returns an array of state names in descending order, not including the root. + */ +function ancestors(first, second) { + var path = []; + for (var n in first.path) { + if (first.path[n] !== second.path[n]) + break; + path.push(first.path[n]); + } + return path; +} +/** + * Return a copy of the object only containing the whitelisted properties. + * + * #### Example: + * ``` + * var foo = { a: 1, b: 2, c: 3 }; + * var ab = pick(foo, ['a', 'b']); // { a: 1, b: 2 } + * ``` + * @param obj the source object + * @param propNames an Array of strings, which are the whitelisted property names + */ +function pick(obj, propNames) { + var objCopy = {}; + for (var prop_1 in obj) { + if (propNames.indexOf(prop_1) !== -1) { + objCopy[prop_1] = obj[prop_1]; + } + } + return objCopy; +} +/** + * Return a copy of the object omitting the blacklisted properties. + * + * @example + * ``` + * + * var foo = { a: 1, b: 2, c: 3 }; + * var ab = omit(foo, ['a', 'b']); // { c: 3 } + * ``` + * @param obj the source object + * @param propNames an Array of strings, which are the blacklisted property names + */ +function omit(obj, propNames) { + return Object.keys(obj) + .filter(not(inArray(propNames))) + .reduce(function (acc, key) { return (acc[key] = obj[key], acc); }, {}); +} +/** + * Maps an array, or object to a property (by name) + */ +function pluck(collection, propName) { + return map(collection, prop(propName)); +} +/** Filters an Array or an Object's properties based on a predicate */ +function filter(collection, callback) { + var arr = isArray(collection), result = arr ? [] : {}; + var accept = arr ? function (x) { return result.push(x); } : function (x, key) { return result[key] = x; }; + forEach(collection, function (item, i) { + if (callback(item, i)) + accept(item, i); + }); + return result; +} +/** Finds an object from an array, or a property of an object, that matches a predicate */ +function find(collection, callback) { + var result; + forEach(collection, function (item, i) { + if (result) + return; + if (callback(item, i)) + result = item; + }); + return result; +} +/** Given an object, returns a new object, where each property is transformed by the callback function */ +var mapObj = map; +/** Maps an array or object properties using a callback function */ +function map(collection, callback) { + var result = isArray(collection) ? [] : {}; + forEach(collection, function (item, i) { return result[i] = callback(item, i); }); + return result; +} +/** + * Given an object, return its enumerable property values + * + * @example + * ``` + * + * let foo = { a: 1, b: 2, c: 3 } + * let vals = values(foo); // [ 1, 2, 3 ] + * ``` + */ +var values = function (obj) { + return Object.keys(obj).map(function (key) { return obj[key]; }); +}; +/** + * Reduce function that returns true if all of the values are truthy. + * + * @example + * ``` + * + * let vals = [ 1, true, {}, "hello world"]; + * vals.reduce(allTrueR, true); // true + * + * vals.push(0); + * vals.reduce(allTrueR, true); // false + * ``` + */ +var allTrueR = function (memo, elem) { return memo && elem; }; +/** + * Reduce function that returns true if any of the values are truthy. + * + * * @example + * ``` + * + * let vals = [ 0, null, undefined ]; + * vals.reduce(anyTrueR, true); // false + * + * vals.push("hello world"); + * vals.reduce(anyTrueR, true); // true + * ``` + */ +var anyTrueR = function (memo, elem) { return memo || elem; }; +/** + * Reduce function which un-nests a single level of arrays + * @example + * ``` + * + * let input = [ [ "a", "b" ], [ "c", "d" ], [ [ "double", "nested" ] ] ]; + * input.reduce(unnestR, []) // [ "a", "b", "c", "d", [ "double, "nested" ] ] + * ``` + */ +var unnestR = function (memo, elem) { return memo.concat(elem); }; +/** + * Reduce function which recursively un-nests all arrays + * + * @example + * ``` + * + * let input = [ [ "a", "b" ], [ "c", "d" ], [ [ "double", "nested" ] ] ]; + * input.reduce(unnestR, []) // [ "a", "b", "c", "d", "double, "nested" ] + * ``` + */ +var flattenR = function (memo, elem) { + return isArray(elem) ? memo.concat(elem.reduce(flattenR, [])) : pushR(memo, elem); +}; +/** + * Reduce function that pushes an object to an array, then returns the array. + * Mostly just for [[flattenR]] and [[uniqR]] + */ +function pushR(arr, obj) { + arr.push(obj); + return arr; +} +/** Reduce function that filters out duplicates */ +var uniqR = function (acc, token) { + return inArray(acc, token) ? acc : pushR(acc, token); +}; +/** + * Return a new array with a single level of arrays unnested. + * + * @example + * ``` + * + * let input = [ [ "a", "b" ], [ "c", "d" ], [ [ "double", "nested" ] ] ]; + * unnest(input) // [ "a", "b", "c", "d", [ "double, "nested" ] ] + * ``` + */ +var unnest = function (arr) { return arr.reduce(unnestR, []); }; +/** + * Return a completely flattened version of an array. + * + * @example + * ``` + * + * let input = [ [ "a", "b" ], [ "c", "d" ], [ [ "double", "nested" ] ] ]; + * flatten(input) // [ "a", "b", "c", "d", "double, "nested" ] + * ``` + */ +var flatten = function (arr) { return arr.reduce(flattenR, []); }; +/** + * Given a .filter Predicate, builds a .filter Predicate which throws an error if any elements do not pass. + * @example + * ``` + * + * let isNumber = (obj) => typeof(obj) === 'number'; + * let allNumbers = [ 1, 2, 3, 4, 5 ]; + * allNumbers.filter(assertPredicate(isNumber)); //OK + * + * let oneString = [ 1, 2, 3, 4, "5" ]; + * oneString.filter(assertPredicate(isNumber, "Not all numbers")); // throws Error(""Not all numbers""); + * ``` + */ +var assertPredicate = assertFn; +/** + * Given a .map function, builds a .map function which throws an error if any mapped elements do not pass a truthyness test. + * @example + * ``` + * + * var data = { foo: 1, bar: 2 }; + * + * let keys = [ 'foo', 'bar' ] + * let values = keys.map(assertMap(key => data[key], "Key not found")); + * // values is [1, 2] + * + * let keys = [ 'foo', 'bar', 'baz' ] + * let values = keys.map(assertMap(key => data[key], "Key not found")); + * // throws Error("Key not found") + * ``` + */ +var assertMap = assertFn; +function assertFn(predicateOrMap, errMsg) { + if (errMsg === void 0) { errMsg = "assert failure"; } + return function (obj) { + var result = predicateOrMap(obj); + if (!result) { + throw new Error(isFunction(errMsg) ? errMsg(obj) : errMsg); + } + return result; + }; +} +/** + * Like _.pairs: Given an object, returns an array of key/value pairs + * + * @example + * ``` + * + * pairs({ foo: "FOO", bar: "BAR }) // [ [ "foo", "FOO" ], [ "bar": "BAR" ] ] + * ``` + */ +var pairs = function (obj) { + return Object.keys(obj).map(function (key) { return [key, obj[key]]; }); +}; +/** + * Given two or more parallel arrays, returns an array of tuples where + * each tuple is composed of [ a[i], b[i], ... z[i] ] + * + * @example + * ``` + * + * let foo = [ 0, 2, 4, 6 ]; + * let bar = [ 1, 3, 5, 7 ]; + * let baz = [ 10, 30, 50, 70 ]; + * arrayTuples(foo, bar); // [ [0, 1], [2, 3], [4, 5], [6, 7] ] + * arrayTuples(foo, bar, baz); // [ [0, 1, 10], [2, 3, 30], [4, 5, 50], [6, 7, 70] ] + * ``` + */ +function arrayTuples() { + var args = []; + for (var _i = 0; _i < arguments.length; _i++) { + args[_i] = arguments[_i]; + } + if (args.length === 0) + return []; + var maxArrayLen = args.reduce(function (min, arr) { return Math.min(arr.length, min); }, 9007199254740991); // aka 2^53 − 1 aka Number.MAX_SAFE_INTEGER + var i, result = []; + for (i = 0; i < maxArrayLen; i++) { + // This is a hot function + // Unroll when there are 1-4 arguments + switch (args.length) { + case 1: + result.push([args[0][i]]); + break; + case 2: + result.push([args[0][i], args[1][i]]); + break; + case 3: + result.push([args[0][i], args[1][i], args[2][i]]); + break; + case 4: + result.push([args[0][i], args[1][i], args[2][i], args[3][i]]); + break; + default: + result.push(args.map(function (array) { return array[i]; })); + break; + } + } + return result; +} +/** + * Reduce function which builds an object from an array of [key, value] pairs. + * + * Each iteration sets the key/val pair on the memo object, then returns the memo for the next iteration. + * + * Each keyValueTuple should be an array with values [ key: string, value: any ] + * + * @example + * ``` + * + * var pairs = [ ["fookey", "fooval"], ["barkey", "barval"] ] + * + * var pairsToObj = pairs.reduce((memo, pair) => applyPairs(memo, pair), {}) + * // pairsToObj == { fookey: "fooval", barkey: "barval" } + * + * // Or, more simply: + * var pairsToObj = pairs.reduce(applyPairs, {}) + * // pairsToObj == { fookey: "fooval", barkey: "barval" } + * ``` + */ +function applyPairs(memo, keyValTuple) { + var key, value; + if (isArray(keyValTuple)) + key = keyValTuple[0], value = keyValTuple[1]; + if (!isString(key)) + throw new Error("invalid parameters to applyPairs"); + memo[key] = value; + return memo; +} +/** Get the last element of an array */ +function tail(arr) { + return arr.length && arr[arr.length - 1] || undefined; +} +/** + * shallow copy from src to dest + */ +function copy(src, dest) { + if (dest) + Object.keys(dest).forEach(function (key) { return delete dest[key]; }); + if (!dest) + dest = {}; + return extend(dest, src); +} +/** Naive forEach implementation works with Objects or Arrays */ +function _forEach(obj, cb, _this) { + if (isArray(obj)) + return obj.forEach(cb, _this); + Object.keys(obj).forEach(function (key) { return cb(obj[key], key); }); +} +function _extend(toObj) { + for (var i = 1; i < arguments.length; i++) { + var obj = arguments[i]; + if (!obj) + continue; + var keys = Object.keys(obj); + for (var j = 0; j < keys.length; j++) { + toObj[keys[j]] = obj[keys[j]]; + } + } + return toObj; +} +function _equals(o1, o2) { + if (o1 === o2) + return true; + if (o1 === null || o2 === null) + return false; + if (o1 !== o1 && o2 !== o2) + return true; // NaN === NaN + var t1 = typeof o1, t2 = typeof o2; + if (t1 !== t2 || t1 !== 'object') + return false; + var tup = [o1, o2]; + if (all(isArray)(tup)) + return _arraysEq(o1, o2); + if (all(isDate)(tup)) + return o1.getTime() === o2.getTime(); + if (all(isRegExp)(tup)) + return o1.toString() === o2.toString(); + if (all(isFunction)(tup)) + return true; // meh + var predicates = [isFunction, isArray, isDate, isRegExp]; + if (predicates.map(any).reduce(function (b, fn) { return b || !!fn(tup); }, false)) + return false; + var key, keys = {}; + for (key in o1) { + if (!_equals(o1[key], o2[key])) + return false; + keys[key] = true; + } + for (key in o2) { + if (!keys[key]) + return false; + } + return true; +} +function _arraysEq(a1, a2) { + if (a1.length !== a2.length) + return false; + return arrayTuples(a1, a2).reduce(function (b, t) { return b && _equals(t[0], t[1]); }, true); +} +// issue #2676 +var silenceUncaughtInPromise = function (promise) { + return promise.catch(function (e) { return 0; }) && promise; +}; +var silentRejection = function (error) { + return silenceUncaughtInPromise(services.$q.reject(error)); +}; + +/** + * @module common + */ /** for typedoc */ +var Queue = /** @class */ (function () { + function Queue(_items, _limit) { + if (_items === void 0) { _items = []; } + if (_limit === void 0) { _limit = null; } + this._items = _items; + this._limit = _limit; + } + Queue.prototype.enqueue = function (item) { + var items = this._items; + items.push(item); + if (this._limit && items.length > this._limit) + items.shift(); + return item; + }; + Queue.prototype.dequeue = function () { + if (this.size()) + return this._items.splice(0, 1)[0]; + }; + Queue.prototype.clear = function () { + var current = this._items; + this._items = []; + return current; + }; + Queue.prototype.size = function () { + return this._items.length; + }; + Queue.prototype.remove = function (item) { + var idx = this._items.indexOf(item); + return idx > -1 && this._items.splice(idx, 1)[0]; + }; + Queue.prototype.peekTail = function () { + return this._items[this._items.length - 1]; + }; + Queue.prototype.peekHead = function () { + if (this.size()) + return this._items[0]; + }; + return Queue; +}()); + +/** + * @coreapi + * @module transition + */ /** for typedoc */ +"use strict"; + +(function (RejectType) { + RejectType[RejectType["SUPERSEDED"] = 2] = "SUPERSEDED"; + RejectType[RejectType["ABORTED"] = 3] = "ABORTED"; + RejectType[RejectType["INVALID"] = 4] = "INVALID"; + RejectType[RejectType["IGNORED"] = 5] = "IGNORED"; + RejectType[RejectType["ERROR"] = 6] = "ERROR"; +})(exports.RejectType || (exports.RejectType = {})); +/** @hidden */ var id = 0; +var Rejection = /** @class */ (function () { + function Rejection(type, message, detail) { + this.$id = id++; + this.type = type; + this.message = message; + this.detail = detail; + } + Rejection.prototype.toString = function () { + var detailString = function (d) { + return d && d.toString !== Object.prototype.toString ? d.toString() : stringify(d); + }; + var detail = detailString(this.detail); + var _a = this, $id = _a.$id, type = _a.type, message = _a.message; + return "Transition Rejection($id: " + $id + " type: " + type + ", message: " + message + ", detail: " + detail + ")"; + }; + Rejection.prototype.toPromise = function () { + return extend(silentRejection(this), { _transitionRejection: this }); + }; + /** Returns true if the obj is a rejected promise created from the `asPromise` factory */ + Rejection.isRejectionPromise = function (obj) { + return obj && (typeof obj.then === 'function') && is(Rejection)(obj._transitionRejection); + }; + /** Returns a Rejection due to transition superseded */ + Rejection.superseded = function (detail, options) { + var message = "The transition has been superseded by a different transition"; + var rejection = new Rejection(exports.RejectType.SUPERSEDED, message, detail); + if (options && options.redirected) { + rejection.redirected = true; + } + return rejection; + }; + /** Returns a Rejection due to redirected transition */ + Rejection.redirected = function (detail) { + return Rejection.superseded(detail, { redirected: true }); + }; + /** Returns a Rejection due to invalid transition */ + Rejection.invalid = function (detail) { + var message = "This transition is invalid"; + return new Rejection(exports.RejectType.INVALID, message, detail); + }; + /** Returns a Rejection due to ignored transition */ + Rejection.ignored = function (detail) { + var message = "The transition was ignored"; + return new Rejection(exports.RejectType.IGNORED, message, detail); + }; + /** Returns a Rejection due to aborted transition */ + Rejection.aborted = function (detail) { + var message = "The transition has been aborted"; + return new Rejection(exports.RejectType.ABORTED, message, detail); + }; + /** Returns a Rejection due to aborted transition */ + Rejection.errored = function (detail) { + var message = "The transition errored"; + return new Rejection(exports.RejectType.ERROR, message, detail); + }; + /** + * Returns a Rejection + * + * Normalizes a value as a Rejection. + * If the value is already a Rejection, returns it. + * Otherwise, wraps and returns the value as a Rejection (Rejection type: ERROR). + * + * @returns `detail` if it is already a `Rejection`, else returns an ERROR Rejection. + */ + Rejection.normalize = function (detail) { + return is(Rejection)(detail) ? detail : Rejection.errored(detail); + }; + return Rejection; +}()); + +/** + * # Transition tracing (debug) + * + * Enable transition tracing to print transition information to the console, + * in order to help debug your application. + * Tracing logs detailed information about each Transition to your console. + * + * To enable tracing, import the [[Trace]] singleton and enable one or more categories. + * + * ### ES6 + * ```js + * import {trace} from "ui-router-ng2"; // or "angular-ui-router" + * trace.enable(1, 5); // TRANSITION and VIEWCONFIG + * ``` + * + * ### CJS + * ```js + * let trace = require("angular-ui-router").trace; // or "ui-router-ng2" + * trace.enable("TRANSITION", "VIEWCONFIG"); + * ``` + * + * ### Globals + * ```js + * let trace = window["angular-ui-router"].trace; // or "ui-router-ng2" + * trace.enable(); // Trace everything (very verbose) + * ``` + * + * ### Angular 1: + * ```js + * app.run($trace => $trace.enable()); + * ``` + * + * @coreapi + * @module trace + */ /** for typedoc */ +/** @hidden */ +function uiViewString(uiview) { + if (!uiview) + return 'ui-view (defunct)'; + var state = uiview.creationContext ? uiview.creationContext.name || '(root)' : '(none)'; + return "[ui-view#" + uiview.id + " " + uiview.$type + ":" + uiview.fqn + " (" + uiview.name + "@" + state + ")]"; +} +/** @hidden */ +var viewConfigString = function (viewConfig) { + var view = viewConfig.viewDecl; + var state = view.$context.name || '(root)'; + return "[View#" + viewConfig.$id + " from '" + state + "' state]: target ui-view: '" + view.$uiViewName + "@" + view.$uiViewContextAnchor + "'"; +}; +/** @hidden */ +function normalizedCat(input) { + return isNumber(input) ? exports.Category[input] : exports.Category[exports.Category[input]]; +} +/** @hidden */ +var consoleLog = Function.prototype.bind.call(console.log, console); +/** @hidden */ +var consoletable = isFunction(console.table) ? console.table.bind(console) : consoleLog.bind(console); +/** + * Trace categories Enum + * + * Enable or disable a category using [[Trace.enable]] or [[Trace.disable]] + * + * `trace.enable(Category.TRANSITION)` + * + * These can also be provided using a matching string, or position ordinal + * + * `trace.enable("TRANSITION")` + * + * `trace.enable(1)` + */ + +(function (Category) { + Category[Category["RESOLVE"] = 0] = "RESOLVE"; + Category[Category["TRANSITION"] = 1] = "TRANSITION"; + Category[Category["HOOK"] = 2] = "HOOK"; + Category[Category["UIVIEW"] = 3] = "UIVIEW"; + Category[Category["VIEWCONFIG"] = 4] = "VIEWCONFIG"; +})(exports.Category || (exports.Category = {})); +/** @hidden */ var _tid = parse("$id"); +/** @hidden */ var _rid = parse("router.$id"); +/** @hidden */ var transLbl = function (trans) { return "Transition #" + _tid(trans) + "-" + _rid(trans); }; +/** + * Prints UI-Router Transition trace information to the console. + */ +var Trace = /** @class */ (function () { + /** @hidden */ + function Trace() { + /** @hidden */ + this._enabled = {}; + this.approximateDigests = 0; + } + /** @hidden */ + Trace.prototype._set = function (enabled, categories) { + var _this = this; + if (!categories.length) { + categories = Object.keys(exports.Category) + .map(function (k) { return parseInt(k, 10); }) + .filter(function (k) { return !isNaN(k); }) + .map(function (key) { return exports.Category[key]; }); + } + categories.map(normalizedCat).forEach(function (category) { return _this._enabled[category] = enabled; }); + }; + Trace.prototype.enable = function () { + var categories = []; + for (var _i = 0; _i < arguments.length; _i++) { + categories[_i] = arguments[_i]; + } + this._set(true, categories); + }; + Trace.prototype.disable = function () { + var categories = []; + for (var _i = 0; _i < arguments.length; _i++) { + categories[_i] = arguments[_i]; + } + this._set(false, categories); + }; + /** + * Retrieves the enabled stateus of a [[Category]] + * + * ```js + * trace.enabled("VIEWCONFIG"); // true or false + * ``` + * + * @returns boolean true if the category is enabled + */ + Trace.prototype.enabled = function (category) { + return !!this._enabled[normalizedCat(category)]; + }; + /** @internalapi called by ui-router code */ + Trace.prototype.traceTransitionStart = function (trans) { + if (!this.enabled(exports.Category.TRANSITION)) + return; + console.log(transLbl(trans) + ": Started -> " + stringify(trans)); + }; + /** @internalapi called by ui-router code */ + Trace.prototype.traceTransitionIgnored = function (trans) { + if (!this.enabled(exports.Category.TRANSITION)) + return; + console.log(transLbl(trans) + ": Ignored <> " + stringify(trans)); + }; + /** @internalapi called by ui-router code */ + Trace.prototype.traceHookInvocation = function (step, trans, options) { + if (!this.enabled(exports.Category.HOOK)) + return; + var event = parse("traceData.hookType")(options) || "internal", context = parse("traceData.context.state.name")(options) || parse("traceData.context")(options) || "unknown", name = functionToString(step.registeredHook.callback); + console.log(transLbl(trans) + ": Hook -> " + event + " context: " + context + ", " + maxLength(200, name)); + }; + /** @internalapi called by ui-router code */ + Trace.prototype.traceHookResult = function (hookResult, trans, transitionOptions) { + if (!this.enabled(exports.Category.HOOK)) + return; + console.log(transLbl(trans) + ": <- Hook returned: " + maxLength(200, stringify(hookResult))); + }; + /** @internalapi called by ui-router code */ + Trace.prototype.traceResolvePath = function (path, when, trans) { + if (!this.enabled(exports.Category.RESOLVE)) + return; + console.log(transLbl(trans) + ": Resolving " + path + " (" + when + ")"); + }; + /** @internalapi called by ui-router code */ + Trace.prototype.traceResolvableResolved = function (resolvable, trans) { + if (!this.enabled(exports.Category.RESOLVE)) + return; + console.log(transLbl(trans) + ": <- Resolved " + resolvable + " to: " + maxLength(200, stringify(resolvable.data))); + }; + /** @internalapi called by ui-router code */ + Trace.prototype.traceError = function (reason, trans) { + if (!this.enabled(exports.Category.TRANSITION)) + return; + console.log(transLbl(trans) + ": <- Rejected " + stringify(trans) + ", reason: " + reason); + }; + /** @internalapi called by ui-router code */ + Trace.prototype.traceSuccess = function (finalState, trans) { + if (!this.enabled(exports.Category.TRANSITION)) + return; + console.log(transLbl(trans) + ": <- Success " + stringify(trans) + ", final state: " + finalState.name); + }; + /** @internalapi called by ui-router code */ + Trace.prototype.traceUIViewEvent = function (event, viewData, extra) { + if (extra === void 0) { extra = ""; } + if (!this.enabled(exports.Category.UIVIEW)) + return; + console.log("ui-view: " + padString(30, event) + " " + uiViewString(viewData) + extra); + }; + /** @internalapi called by ui-router code */ + Trace.prototype.traceUIViewConfigUpdated = function (viewData, context) { + if (!this.enabled(exports.Category.UIVIEW)) + return; + this.traceUIViewEvent("Updating", viewData, " with ViewConfig from context='" + context + "'"); + }; + /** @internalapi called by ui-router code */ + Trace.prototype.traceUIViewFill = function (viewData, html) { + if (!this.enabled(exports.Category.UIVIEW)) + return; + this.traceUIViewEvent("Fill", viewData, " with: " + maxLength(200, html)); + }; + /** @internalapi called by ui-router code */ + Trace.prototype.traceViewSync = function (pairs) { + if (!this.enabled(exports.Category.VIEWCONFIG)) + return; + var mapping = pairs.map(function (_a) { + var uiViewData = _a[0], config = _a[1]; + var uiView = uiViewData.$type + ":" + uiViewData.fqn; + var view = config && config.viewDecl.$context.name + ": " + config.viewDecl.$name + " (" + config.viewDecl.$type + ")"; + return { 'ui-view fqn': uiView, 'state: view name': view }; + }).sort(function (a, b) { return a['ui-view fqn'].localeCompare(b['ui-view fqn']); }); + consoletable(mapping); + }; + /** @internalapi called by ui-router code */ + Trace.prototype.traceViewServiceEvent = function (event, viewConfig) { + if (!this.enabled(exports.Category.VIEWCONFIG)) + return; + console.log("VIEWCONFIG: " + event + " " + viewConfigString(viewConfig)); + }; + /** @internalapi called by ui-router code */ + Trace.prototype.traceViewServiceUIViewEvent = function (event, viewData) { + if (!this.enabled(exports.Category.VIEWCONFIG)) + return; + console.log("VIEWCONFIG: " + event + " " + uiViewString(viewData)); + }; + return Trace; +}()); +/** + * The [[Trace]] singleton + * + * #### Example: + * ```js + * import {trace} from "angular-ui-router"; + * trace.enable(1, 5); + * ``` + */ +var trace = new Trace(); + +(function (TransitionHookPhase) { + TransitionHookPhase[TransitionHookPhase["CREATE"] = 0] = "CREATE"; + TransitionHookPhase[TransitionHookPhase["BEFORE"] = 1] = "BEFORE"; + TransitionHookPhase[TransitionHookPhase["RUN"] = 2] = "RUN"; + TransitionHookPhase[TransitionHookPhase["SUCCESS"] = 3] = "SUCCESS"; + TransitionHookPhase[TransitionHookPhase["ERROR"] = 4] = "ERROR"; +})(exports.TransitionHookPhase || (exports.TransitionHookPhase = {})); + +(function (TransitionHookScope) { + TransitionHookScope[TransitionHookScope["TRANSITION"] = 0] = "TRANSITION"; + TransitionHookScope[TransitionHookScope["STATE"] = 1] = "STATE"; +})(exports.TransitionHookScope || (exports.TransitionHookScope = {})); + +/** + * @coreapi + * @module state + */ /** for typedoc */ +/** + * Encapsulate the target (destination) state/params/options of a [[Transition]]. + * + * This class is frequently used to redirect a transition to a new destination. + * + * See: + * + * - [[HookResult]] + * - [[TransitionHookFn]] + * - [[TransitionService.onStart]] + * + * To create a `TargetState`, use [[StateService.target]]. + * + * --- + * + * This class wraps: + * + * 1) an identifier for a state + * 2) a set of parameters + * 3) and transition options + * 4) the registered state object (the [[StateDeclaration]]) + * + * Many UI-Router APIs such as [[StateService.go]] take a [[StateOrName]] argument which can + * either be a *state object* (a [[StateDeclaration]] or [[StateObject]]) or a *state name* (a string). + * The `TargetState` class normalizes those options. + * + * A `TargetState` may be valid (the state being targeted exists in the registry) + * or invalid (the state being targeted is not registered). + */ +var TargetState = /** @class */ (function () { + /** + * The TargetState constructor + * + * Note: Do not construct a `TargetState` manually. + * To create a `TargetState`, use the [[StateService.target]] factory method. + * + * @param _stateRegistry The StateRegistry to use to look up the _definition + * @param _identifier An identifier for a state. + * Either a fully-qualified state name, or the object used to define the state. + * @param _params Parameters for the target state + * @param _options Transition options. + * + * @internalapi + */ + function TargetState(_stateRegistry, _identifier, _params, _options) { + this._stateRegistry = _stateRegistry; + this._identifier = _identifier; + this._identifier = _identifier; + this._params = extend({}, _params || {}); + this._options = extend({}, _options || {}); + this._definition = _stateRegistry.matcher.find(_identifier, this._options.relative); + } + /** The name of the state this object targets */ + TargetState.prototype.name = function () { + return this._definition && this._definition.name || this._identifier; + }; + /** The identifier used when creating this TargetState */ + TargetState.prototype.identifier = function () { + return this._identifier; + }; + /** The target parameter values */ + TargetState.prototype.params = function () { + return this._params; + }; + /** The internal state object (if it was found) */ + TargetState.prototype.$state = function () { + return this._definition; + }; + /** The internal state declaration (if it was found) */ + TargetState.prototype.state = function () { + return this._definition && this._definition.self; + }; + /** The target options */ + TargetState.prototype.options = function () { + return this._options; + }; + /** True if the target state was found */ + TargetState.prototype.exists = function () { + return !!(this._definition && this._definition.self); + }; + /** True if the object is valid */ + TargetState.prototype.valid = function () { + return !this.error(); + }; + /** If the object is invalid, returns the reason why */ + TargetState.prototype.error = function () { + var base = this.options().relative; + if (!this._definition && !!base) { + var stateName = base.name ? base.name : base; + return "Could not resolve '" + this.name() + "' from state '" + stateName + "'"; + } + if (!this._definition) + return "No such state '" + this.name() + "'"; + if (!this._definition.self) + return "State '" + this.name() + "' has an invalid definition"; + }; + TargetState.prototype.toString = function () { + return "'" + this.name() + "'" + stringify(this.params()); + }; + /** + * Returns a copy of this TargetState which targets a different state. + * The new TargetState has the same parameter values and transition options. + * + * @param state The new state that should be targeted + */ + TargetState.prototype.withState = function (state) { + return new TargetState(this._stateRegistry, state, this._params, this._options); + }; + /** + * Returns a copy of this TargetState, using the specified parameter values. + * + * @param params the new parameter values to use + * @param replace When false (default) the new parameter values will be merged with the current values. + * When true the parameter values will be used instead of the current values. + */ + TargetState.prototype.withParams = function (params, replace) { + if (replace === void 0) { replace = false; } + var newParams = replace ? params : extend({}, this._params, params); + return new TargetState(this._stateRegistry, this._identifier, newParams, this._options); + }; + /** + * Returns a copy of this TargetState, using the specified Transition Options. + * + * @param options the new options to use + * @param replace When false (default) the new options will be merged with the current options. + * When true the options will be used instead of the current options. + */ + TargetState.prototype.withOptions = function (options, replace) { + if (replace === void 0) { replace = false; } + var newOpts = replace ? options : extend({}, this._options, options); + return new TargetState(this._stateRegistry, this._identifier, this._params, newOpts); + }; + /** Returns true if the object has a state property that might be a state or state name */ + TargetState.isDef = function (obj) { + return obj && obj.state && (isString(obj.state) || isString(obj.state.name)); + }; + return TargetState; +}()); + +/** + * @coreapi + * @module transition + */ +/** for typedoc */ +var defaultOptions = { + current: noop$1, + transition: null, + traceData: {}, + bind: null, +}; +/** @hidden */ +var TransitionHook = /** @class */ (function () { + function TransitionHook(transition, stateContext, registeredHook, options) { + var _this = this; + this.transition = transition; + this.stateContext = stateContext; + this.registeredHook = registeredHook; + this.options = options; + this.isSuperseded = function () { + return _this.type.hookPhase === exports.TransitionHookPhase.RUN && !_this.options.transition.isActive(); + }; + this.options = defaults(options, defaultOptions); + this.type = registeredHook.eventType; + } + TransitionHook.prototype.logError = function (err) { + this.transition.router.stateService.defaultErrorHandler()(err); + }; + TransitionHook.prototype.invokeHook = function () { + var _this = this; + var hook = this.registeredHook; + if (hook._deregistered) + return; + var notCurrent = this.getNotCurrentRejection(); + if (notCurrent) + return notCurrent; + var options = this.options; + trace.traceHookInvocation(this, this.transition, options); + var invokeCallback = function () { + return hook.callback.call(options.bind, _this.transition, _this.stateContext); + }; + var normalizeErr = function (err) { + return Rejection.normalize(err).toPromise(); + }; + var handleError = function (err) { + return hook.eventType.getErrorHandler(_this)(err); + }; + var handleResult = function (result) { + return hook.eventType.getResultHandler(_this)(result); + }; + try { + var result = invokeCallback(); + if (!this.type.synchronous && isPromise(result)) { + return result.catch(normalizeErr) + .then(handleResult, handleError); + } + else { + return handleResult(result); + } + } + catch (err) { + // If callback throws (synchronously) + return handleError(Rejection.normalize(err)); + } + finally { + if (hook.invokeLimit && ++hook.invokeCount >= hook.invokeLimit) { + hook.deregister(); + } + } + }; + /** + * This method handles the return value of a Transition Hook. + * + * A hook can return false (cancel), a TargetState (redirect), + * or a promise (which may later resolve to false or a redirect) + * + * This also handles "transition superseded" -- when a new transition + * was started while the hook was still running + */ + TransitionHook.prototype.handleHookResult = function (result) { + var _this = this; + var notCurrent = this.getNotCurrentRejection(); + if (notCurrent) + return notCurrent; + // Hook returned a promise + if (isPromise(result)) { + // Wait for the promise, then reprocess with the resulting value + return result.then(function (val$$1) { return _this.handleHookResult(val$$1); }); + } + trace.traceHookResult(result, this.transition, this.options); + // Hook returned false + if (result === false) { + // Abort this Transition + return Rejection.aborted("Hook aborted transition").toPromise(); + } + var isTargetState = is(TargetState); + // hook returned a TargetState + if (isTargetState(result)) { + // Halt the current Transition and redirect (a new Transition) to the TargetState. + return Rejection.redirected(result).toPromise(); + } + }; + /** + * Return a Rejection promise if the transition is no longer current due + * to a stopped router (disposed), or a new transition has started and superseded this one. + */ + TransitionHook.prototype.getNotCurrentRejection = function () { + var router = this.transition.router; + // The router is stopped + if (router._disposed) { + return Rejection.aborted("UIRouter instance #" + router.$id + " has been stopped (disposed)").toPromise(); + } + if (this.transition._aborted) { + return Rejection.aborted().toPromise(); + } + // This transition is no longer current. + // Another transition started while this hook was still running. + if (this.isSuperseded()) { + // Abort this transition + return Rejection.superseded(this.options.current()).toPromise(); + } + }; + TransitionHook.prototype.toString = function () { + var _a = this, options = _a.options, registeredHook = _a.registeredHook; + var event = parse("traceData.hookType")(options) || "internal", context = parse("traceData.context.state.name")(options) || parse("traceData.context")(options) || "unknown", name = fnToString(registeredHook.callback); + return event + " context: " + context + ", " + maxLength(200, name); + }; + /** + * Chains together an array of TransitionHooks. + * + * Given a list of [[TransitionHook]] objects, chains them together. + * Each hook is invoked after the previous one completes. + * + * #### Example: + * ```js + * var hooks: TransitionHook[] = getHooks(); + * let promise: Promise = TransitionHook.chain(hooks); + * + * promise.then(handleSuccess, handleError); + * ``` + * + * @param hooks the list of hooks to chain together + * @param waitFor if provided, the chain is `.then()`'ed off this promise + * @returns a `Promise` for sequentially invoking the hooks (in order) + */ + TransitionHook.chain = function (hooks, waitFor) { + // Chain the next hook off the previous + var createHookChainR = function (prev, nextHook) { + return prev.then(function () { return nextHook.invokeHook(); }); + }; + return hooks.reduce(createHookChainR, waitFor || services.$q.when()); + }; + /** + * Invokes all the provided TransitionHooks, in order. + * Each hook's return value is checked. + * If any hook returns a promise, then the rest of the hooks are chained off that promise, and the promise is returned. + * If no hook returns a promise, then all hooks are processed synchronously. + * + * @param hooks the list of TransitionHooks to invoke + * @param doneCallback a callback that is invoked after all the hooks have successfully completed + * + * @returns a promise for the async result, or the result of the callback + */ + TransitionHook.invokeHooks = function (hooks, doneCallback) { + for (var idx = 0; idx < hooks.length; idx++) { + var hookResult = hooks[idx].invokeHook(); + if (isPromise(hookResult)) { + var remainingHooks = hooks.slice(idx + 1); + return TransitionHook.chain(remainingHooks, hookResult) + .then(doneCallback); + } + } + return doneCallback(); + }; + /** + * Run all TransitionHooks, ignoring their return value. + */ + TransitionHook.runAllHooks = function (hooks) { + hooks.forEach(function (hook) { return hook.invokeHook(); }); + }; + /** + * These GetResultHandler(s) are used by [[invokeHook]] below + * Each HookType chooses a GetResultHandler (See: [[TransitionService._defineCoreEvents]]) + */ + TransitionHook.HANDLE_RESULT = function (hook) { return function (result) { + return hook.handleHookResult(result); + }; }; + /** + * If the result is a promise rejection, log it. + * Otherwise, ignore the result. + */ + TransitionHook.LOG_REJECTED_RESULT = function (hook) { return function (result) { + isPromise(result) && result.catch(function (err) { + return hook.logError(Rejection.normalize(err)); + }); + return undefined; + }; }; + /** + * These GetErrorHandler(s) are used by [[invokeHook]] below + * Each HookType chooses a GetErrorHandler (See: [[TransitionService._defineCoreEvents]]) + */ + TransitionHook.LOG_ERROR = function (hook) { return function (error) { + return hook.logError(error); + }; }; + TransitionHook.REJECT_ERROR = function (hook) { return function (error) { + return silentRejection(error); + }; }; + TransitionHook.THROW_ERROR = function (hook) { return function (error) { + throw error; + }; }; + return TransitionHook; +}()); + +/** + * @coreapi + * @module transition + */ /** for typedoc */ +/** + * Determines if the given state matches the matchCriteria + * + * @hidden + * + * @param state a State Object to test against + * @param criterion + * - If a string, matchState uses the string as a glob-matcher against the state name + * - If an array (of strings), matchState uses each string in the array as a glob-matchers against the state name + * and returns a positive match if any of the globs match. + * - If a function, matchState calls the function with the state and returns true if the function's result is truthy. + * @returns {boolean} + */ +function matchState(state, criterion) { + var toMatch = isString(criterion) ? [criterion] : criterion; + function matchGlobs(_state) { + var globStrings = toMatch; + for (var i = 0; i < globStrings.length; i++) { + var glob = new Glob(globStrings[i]); + if ((glob && glob.matches(_state.name)) || (!glob && globStrings[i] === _state.name)) { + return true; + } + } + return false; + } + var matchFn = (isFunction(toMatch) ? toMatch : matchGlobs); + return !!matchFn(state); +} +/** + * @internalapi + * The registration data for a registered transition hook + */ +var RegisteredHook = /** @class */ (function () { + function RegisteredHook(tranSvc, eventType, callback, matchCriteria, removeHookFromRegistry, options) { + if (options === void 0) { options = {}; } + this.tranSvc = tranSvc; + this.eventType = eventType; + this.callback = callback; + this.matchCriteria = matchCriteria; + this.removeHookFromRegistry = removeHookFromRegistry; + this.invokeCount = 0; + this._deregistered = false; + this.priority = options.priority || 0; + this.bind = options.bind || null; + this.invokeLimit = options.invokeLimit; + } + /** + * Gets the matching [[PathNode]]s + * + * Given an array of [[PathNode]]s, and a [[HookMatchCriterion]], returns an array containing + * the [[PathNode]]s that the criteria matches, or `null` if there were no matching nodes. + * + * Returning `null` is significant to distinguish between the default + * "match-all criterion value" of `true` compared to a `() => true` function, + * when the nodes is an empty array. + * + * This is useful to allow a transition match criteria of `entering: true` + * to still match a transition, even when `entering === []`. Contrast that + * with `entering: (state) => true` which only matches when a state is actually + * being entered. + */ + RegisteredHook.prototype._matchingNodes = function (nodes, criterion) { + if (criterion === true) + return nodes; + var matching = nodes.filter(function (node) { return matchState(node.state, criterion); }); + return matching.length ? matching : null; + }; + /** + * Gets the default match criteria (all `true`) + * + * Returns an object which has all the criteria match paths as keys and `true` as values, i.e.: + * + * ```js + * { + * to: true, + * from: true, + * entering: true, + * exiting: true, + * retained: true, + * } + */ + RegisteredHook.prototype._getDefaultMatchCriteria = function () { + return map(this.tranSvc._pluginapi._getPathTypes(), function () { return true; }); + }; + /** + * Gets matching nodes as [[IMatchingNodes]] + * + * Create a IMatchingNodes object from the TransitionHookTypes that is roughly equivalent to: + * + * ```js + * let matches: IMatchingNodes = { + * to: _matchingNodes([tail(treeChanges.to)], mc.to), + * from: _matchingNodes([tail(treeChanges.from)], mc.from), + * exiting: _matchingNodes(treeChanges.exiting, mc.exiting), + * retained: _matchingNodes(treeChanges.retained, mc.retained), + * entering: _matchingNodes(treeChanges.entering, mc.entering), + * }; + * ``` + */ + RegisteredHook.prototype._getMatchingNodes = function (treeChanges) { + var _this = this; + var criteria = extend(this._getDefaultMatchCriteria(), this.matchCriteria); + var paths = values(this.tranSvc._pluginapi._getPathTypes()); + return paths.reduce(function (mn, pathtype) { + // STATE scope criteria matches against every node in the path. + // TRANSITION scope criteria matches against only the last node in the path + var isStateHook = pathtype.scope === exports.TransitionHookScope.STATE; + var path = treeChanges[pathtype.name] || []; + var nodes = isStateHook ? path : [tail(path)]; + mn[pathtype.name] = _this._matchingNodes(nodes, criteria[pathtype.name]); + return mn; + }, {}); + }; + /** + * Determines if this hook's [[matchCriteria]] match the given [[TreeChanges]] + * + * @returns an IMatchingNodes object, or null. If an IMatchingNodes object is returned, its values + * are the matching [[PathNode]]s for each [[HookMatchCriterion]] (to, from, exiting, retained, entering) + */ + RegisteredHook.prototype.matches = function (treeChanges) { + var matches = this._getMatchingNodes(treeChanges); + // Check if all the criteria matched the TreeChanges object + var allMatched = values(matches).every(identity); + return allMatched ? matches : null; + }; + RegisteredHook.prototype.deregister = function () { + this.removeHookFromRegistry(this); + this._deregistered = true; + }; + return RegisteredHook; +}()); +/** @hidden Return a registration function of the requested type. */ +function makeEvent(registry, transitionService, eventType) { + // Create the object which holds the registered transition hooks. + var _registeredHooks = registry._registeredHooks = (registry._registeredHooks || {}); + var hooks = _registeredHooks[eventType.name] = []; + var removeHookFn = removeFrom(hooks); + // Create hook registration function on the IHookRegistry for the event + registry[eventType.name] = hookRegistrationFn; + function hookRegistrationFn(matchObject, callback, options) { + if (options === void 0) { options = {}; } + var registeredHook = new RegisteredHook(transitionService, eventType, callback, matchObject, removeHookFn, options); + hooks.push(registeredHook); + return registeredHook.deregister.bind(registeredHook); + } + return hookRegistrationFn; +} + +/** + * @coreapi + * @module transition + */ /** for typedoc */ +/** + * This class returns applicable TransitionHooks for a specific Transition instance. + * + * Hooks ([[RegisteredHook]]) may be registered globally, e.g., $transitions.onEnter(...), or locally, e.g. + * myTransition.onEnter(...). The HookBuilder finds matching RegisteredHooks (where the match criteria is + * determined by the type of hook) + * + * The HookBuilder also converts RegisteredHooks objects to TransitionHook objects, which are used to run a Transition. + * + * The HookBuilder constructor is given the $transitions service and a Transition instance. Thus, a HookBuilder + * instance may only be used for one specific Transition object. (side note: the _treeChanges accessor is private + * in the Transition class, so we must also provide the Transition's _treeChanges) + * + */ +var HookBuilder = /** @class */ (function () { + function HookBuilder(transition) { + this.transition = transition; + } + HookBuilder.prototype.buildHooksForPhase = function (phase) { + var _this = this; + var $transitions = this.transition.router.transitionService; + return $transitions._pluginapi._getEvents(phase) + .map(function (type) { return _this.buildHooks(type); }) + .reduce(unnestR, []) + .filter(identity); + }; + /** + * Returns an array of newly built TransitionHook objects. + * + * - Finds all RegisteredHooks registered for the given `hookType` which matched the transition's [[TreeChanges]]. + * - Finds [[PathNode]] (or `PathNode[]`) to use as the TransitionHook context(s) + * - For each of the [[PathNode]]s, creates a TransitionHook + * + * @param hookType the type of the hook registration function, e.g., 'onEnter', 'onFinish'. + */ + HookBuilder.prototype.buildHooks = function (hookType) { + var transition = this.transition; + var treeChanges = transition.treeChanges(); + // Find all the matching registered hooks for a given hook type + var matchingHooks = this.getMatchingHooks(hookType, treeChanges); + if (!matchingHooks) + return []; + var baseHookOptions = { + transition: transition, + current: transition.options().current + }; + var makeTransitionHooks = function (hook) { + // Fetch the Nodes that caused this hook to match. + var matches = hook.matches(treeChanges); + // Select the PathNode[] that will be used as TransitionHook context objects + var matchingNodes = matches[hookType.criteriaMatchPath.name]; + // Return an array of HookTuples + return matchingNodes.map(function (node) { + var _options = extend({ + bind: hook.bind, + traceData: { hookType: hookType.name, context: node } + }, baseHookOptions); + var state = hookType.criteriaMatchPath.scope === exports.TransitionHookScope.STATE ? node.state.self : null; + var transitionHook = new TransitionHook(transition, state, hook, _options); + return { hook: hook, node: node, transitionHook: transitionHook }; + }); + }; + return matchingHooks.map(makeTransitionHooks) + .reduce(unnestR, []) + .sort(tupleSort(hookType.reverseSort)) + .map(function (tuple) { return tuple.transitionHook; }); + }; + /** + * Finds all RegisteredHooks from: + * - The Transition object instance hook registry + * - The TransitionService ($transitions) global hook registry + * + * which matched: + * - the eventType + * - the matchCriteria (to, from, exiting, retained, entering) + * + * @returns an array of matched [[RegisteredHook]]s + */ + HookBuilder.prototype.getMatchingHooks = function (hookType, treeChanges) { + var isCreate = hookType.hookPhase === exports.TransitionHookPhase.CREATE; + // Instance and Global hook registries + var $transitions = this.transition.router.transitionService; + var registries = isCreate ? [$transitions] : [this.transition, $transitions]; + return registries.map(function (reg) { return reg.getHooks(hookType.name); }) // Get named hooks from registries + .filter(assertPredicate(isArray, "broken event named: " + hookType.name)) // Sanity check + .reduce(unnestR, []) // Un-nest RegisteredHook[][] to RegisteredHook[] array + .filter(function (hook) { return hook.matches(treeChanges); }); // Only those satisfying matchCriteria + }; + return HookBuilder; +}()); +/** + * A factory for a sort function for HookTuples. + * + * The sort function first compares the PathNode depth (how deep in the state tree a node is), then compares + * the EventHook priority. + * + * @param reverseDepthSort a boolean, when true, reverses the sort order for the node depth + * @returns a tuple sort function + */ +function tupleSort(reverseDepthSort) { + if (reverseDepthSort === void 0) { reverseDepthSort = false; } + return function nodeDepthThenPriority(l, r) { + var factor = reverseDepthSort ? -1 : 1; + var depthDelta = (l.node.state.path.length - r.node.state.path.length) * factor; + return depthDelta !== 0 ? depthDelta : r.hook.priority - l.hook.priority; + }; +} + +/** + * @coreapi + * @module params + */ +/** */ +/** + * An internal class which implements [[ParamTypeDefinition]]. + * + * A [[ParamTypeDefinition]] is a plain javascript object used to register custom parameter types. + * When a param type definition is registered, an instance of this class is created internally. + * + * This class has naive implementations for all the [[ParamTypeDefinition]] methods. + * + * Used by [[UrlMatcher]] when matching or formatting URLs, or comparing and validating parameter values. + * + * #### Example: + * ```js + * var paramTypeDef = { + * decode: function(val) { return parseInt(val, 10); }, + * encode: function(val) { return val && val.toString(); }, + * equals: function(a, b) { return this.is(a) && a === b; }, + * is: function(val) { return angular.isNumber(val) && isFinite(val) && val % 1 === 0; }, + * pattern: /\d+/ + * } + * + * var paramType = new ParamType(paramTypeDef); + * ``` + * @internalapi + */ +var ParamType = /** @class */ (function () { + /** + * @param def A configuration object which contains the custom type definition. The object's + * properties will override the default methods and/or pattern in `ParamType`'s public interface. + * @returns a new ParamType object + */ + function ParamType(def) { + /** @inheritdoc */ + this.pattern = /.*/; + /** @inheritdoc */ + this.inherit = true; + extend(this, def); + } + // consider these four methods to be "abstract methods" that should be overridden + /** @inheritdoc */ + ParamType.prototype.is = function (val, key) { return true; }; + /** @inheritdoc */ + ParamType.prototype.encode = function (val, key) { return val; }; + /** @inheritdoc */ + ParamType.prototype.decode = function (val, key) { return val; }; + /** @inheritdoc */ + ParamType.prototype.equals = function (a, b) { return a == b; }; + ParamType.prototype.$subPattern = function () { + var sub = this.pattern.toString(); + return sub.substr(1, sub.length - 2); + }; + ParamType.prototype.toString = function () { + return "{ParamType:" + this.name + "}"; + }; + /** Given an encoded string, or a decoded object, returns a decoded object */ + ParamType.prototype.$normalize = function (val) { + return this.is(val) ? val : this.decode(val); + }; + /** + * Wraps an existing custom ParamType as an array of ParamType, depending on 'mode'. + * e.g.: + * - urlmatcher pattern "/path?{queryParam[]:int}" + * - url: "/path?queryParam=1&queryParam=2 + * - $stateParams.queryParam will be [1, 2] + * if `mode` is "auto", then + * - url: "/path?queryParam=1 will create $stateParams.queryParam: 1 + * - url: "/path?queryParam=1&queryParam=2 will create $stateParams.queryParam: [1, 2] + */ + ParamType.prototype.$asArray = function (mode, isSearch) { + if (!mode) + return this; + if (mode === "auto" && !isSearch) + throw new Error("'auto' array mode is for query parameters only"); + return new ArrayType(this, mode); + }; + return ParamType; +}()); +/** + * Wraps up a `ParamType` object to handle array values. + * @internalapi + */ +function ArrayType(type, mode) { + var _this = this; + // Wrap non-array value as array + function arrayWrap(val) { + return isArray(val) ? val : (isDefined(val) ? [val] : []); + } + // Unwrap array value for "auto" mode. Return undefined for empty array. + function arrayUnwrap(val) { + switch (val.length) { + case 0: return undefined; + case 1: return mode === "auto" ? val[0] : val; + default: return val; + } + } + // Wraps type (.is/.encode/.decode) functions to operate on each value of an array + function arrayHandler(callback, allTruthyMode) { + return function handleArray(val) { + if (isArray(val) && val.length === 0) + return val; + var arr = arrayWrap(val); + var result = map(arr, callback); + return (allTruthyMode === true) ? filter(result, function (x) { return !x; }).length === 0 : arrayUnwrap(result); + }; + } + // Wraps type (.equals) functions to operate on each value of an array + function arrayEqualsHandler(callback) { + return function handleArray(val1, val2) { + var left = arrayWrap(val1), right = arrayWrap(val2); + if (left.length !== right.length) + return false; + for (var i = 0; i < left.length; i++) { + if (!callback(left[i], right[i])) + return false; + } + return true; + }; + } + ['encode', 'decode', 'equals', '$normalize'].forEach(function (name) { + var paramTypeFn = type[name].bind(type); + var wrapperFn = name === 'equals' ? arrayEqualsHandler : arrayHandler; + _this[name] = wrapperFn(paramTypeFn); + }); + extend(this, { + dynamic: type.dynamic, + name: type.name, + pattern: type.pattern, + inherit: type.inherit, + is: arrayHandler(type.is.bind(type), true), + $arrayMode: mode + }); +} + +/** + * @coreapi + * @module params + */ /** for typedoc */ +/** @hidden */ var hasOwn = Object.prototype.hasOwnProperty; +/** @hidden */ var isShorthand = function (cfg) { + return ["value", "type", "squash", "array", "dynamic"].filter(hasOwn.bind(cfg || {})).length === 0; +}; +/** @internalapi */ + +(function (DefType) { + DefType[DefType["PATH"] = 0] = "PATH"; + DefType[DefType["SEARCH"] = 1] = "SEARCH"; + DefType[DefType["CONFIG"] = 2] = "CONFIG"; +})(exports.DefType || (exports.DefType = {})); +/** @hidden */ +function unwrapShorthand(cfg) { + cfg = isShorthand(cfg) && { value: cfg } || cfg; + getStaticDefaultValue['__cacheable'] = true; + function getStaticDefaultValue() { + return cfg.value; + } + return extend(cfg, { + $$fn: isInjectable(cfg.value) ? cfg.value : getStaticDefaultValue, + }); +} +/** @hidden */ +function getType(cfg, urlType, location, id, paramTypes) { + if (cfg.type && urlType && urlType.name !== 'string') + throw new Error("Param '" + id + "' has two type configurations."); + if (cfg.type && urlType && urlType.name === 'string' && paramTypes.type(cfg.type)) + return paramTypes.type(cfg.type); + if (urlType) + return urlType; + if (!cfg.type) { + var type = location === exports.DefType.CONFIG ? "any" : + location === exports.DefType.PATH ? "path" : + location === exports.DefType.SEARCH ? "query" : "string"; + return paramTypes.type(type); + } + return cfg.type instanceof ParamType ? cfg.type : paramTypes.type(cfg.type); +} +/** + * @internalapi + * returns false, true, or the squash value to indicate the "default parameter url squash policy". + */ +function getSquashPolicy(config, isOptional, defaultPolicy) { + var squash = config.squash; + if (!isOptional || squash === false) + return false; + if (!isDefined(squash) || squash == null) + return defaultPolicy; + if (squash === true || isString(squash)) + return squash; + throw new Error("Invalid squash policy: '" + squash + "'. Valid policies: false, true, or arbitrary string"); +} +/** @internalapi */ +function getReplace(config, arrayMode, isOptional, squash) { + var replace, configuredKeys, defaultPolicy = [ + { from: "", to: (isOptional || arrayMode ? undefined : "") }, + { from: null, to: (isOptional || arrayMode ? undefined : "") }, + ]; + replace = isArray(config.replace) ? config.replace : []; + if (isString(squash)) + replace.push({ from: squash, to: undefined }); + configuredKeys = map(replace, prop("from")); + return filter(defaultPolicy, function (item) { return configuredKeys.indexOf(item.from) === -1; }).concat(replace); +} +/** @internalapi */ +var Param = /** @class */ (function () { + function Param(id, type, config, location, urlMatcherFactory) { + config = unwrapShorthand(config); + type = getType(config, type, location, id, urlMatcherFactory.paramTypes); + var arrayMode = getArrayMode(); + type = arrayMode ? type.$asArray(arrayMode, location === exports.DefType.SEARCH) : type; + var isOptional = config.value !== undefined || location === exports.DefType.SEARCH; + var dynamic = isDefined(config.dynamic) ? !!config.dynamic : !!type.dynamic; + var raw = isDefined(config.raw) ? !!config.raw : !!type.raw; + var squash = getSquashPolicy(config, isOptional, urlMatcherFactory.defaultSquashPolicy()); + var replace = getReplace(config, arrayMode, isOptional, squash); + var inherit$$1 = isDefined(config.inherit) ? !!config.inherit : !!type.inherit; + // array config: param name (param[]) overrides default settings. explicit config overrides param name. + function getArrayMode() { + var arrayDefaults = { array: (location === exports.DefType.SEARCH ? "auto" : false) }; + var arrayParamNomenclature = id.match(/\[\]$/) ? { array: true } : {}; + return extend(arrayDefaults, arrayParamNomenclature, config).array; + } + extend(this, { id: id, type: type, location: location, isOptional: isOptional, dynamic: dynamic, raw: raw, squash: squash, replace: replace, inherit: inherit$$1, array: arrayMode, config: config }); + } + Param.prototype.isDefaultValue = function (value) { + return this.isOptional && this.type.equals(this.value(), value); + }; + /** + * [Internal] Gets the decoded representation of a value if the value is defined, otherwise, returns the + * default value, which may be the result of an injectable function. + */ + Param.prototype.value = function (value) { + var _this = this; + /** + * [Internal] Get the default value of a parameter, which may be an injectable function. + */ + var getDefaultValue = function () { + if (_this._defaultValueCache) + return _this._defaultValueCache.defaultValue; + if (!services.$injector) + throw new Error("Injectable functions cannot be called at configuration time"); + var defaultValue = services.$injector.invoke(_this.config.$$fn); + if (defaultValue !== null && defaultValue !== undefined && !_this.type.is(defaultValue)) + throw new Error("Default value (" + defaultValue + ") for parameter '" + _this.id + "' is not an instance of ParamType (" + _this.type.name + ")"); + if (_this.config.$$fn['__cacheable']) { + _this._defaultValueCache = { defaultValue: defaultValue }; + } + return defaultValue; + }; + var replaceSpecialValues = function (val$$1) { + for (var _i = 0, _a = _this.replace; _i < _a.length; _i++) { + var tuple = _a[_i]; + if (tuple.from === val$$1) + return tuple.to; + } + return val$$1; + }; + value = replaceSpecialValues(value); + return isUndefined(value) ? getDefaultValue() : this.type.$normalize(value); + }; + Param.prototype.isSearch = function () { + return this.location === exports.DefType.SEARCH; + }; + Param.prototype.validates = function (value) { + // There was no parameter value, but the param is optional + if ((isUndefined(value) || value === null) && this.isOptional) + return true; + // The value was not of the correct ParamType, and could not be decoded to the correct ParamType + var normalized = this.type.$normalize(value); + if (!this.type.is(normalized)) + return false; + // The value was of the correct type, but when encoded, did not match the ParamType's regexp + var encoded = this.type.encode(normalized); + return !(isString(encoded) && !this.type.pattern.exec(encoded)); + }; + Param.prototype.toString = function () { + return "{Param:" + this.id + " " + this.type + " squash: '" + this.squash + "' optional: " + this.isOptional + "}"; + }; + Param.values = function (params, values$$1) { + if (values$$1 === void 0) { values$$1 = {}; } + var paramValues = {}; + for (var _i = 0, params_1 = params; _i < params_1.length; _i++) { + var param = params_1[_i]; + paramValues[param.id] = param.value(values$$1[param.id]); + } + return paramValues; + }; + /** + * Finds [[Param]] objects which have different param values + * + * Filters a list of [[Param]] objects to only those whose parameter values differ in two param value objects + * + * @param params: The list of Param objects to filter + * @param values1: The first set of parameter values + * @param values2: the second set of parameter values + * + * @returns any Param objects whose values were different between values1 and values2 + */ + Param.changed = function (params, values1, values2) { + if (values1 === void 0) { values1 = {}; } + if (values2 === void 0) { values2 = {}; } + return params.filter(function (param) { return !param.type.equals(values1[param.id], values2[param.id]); }); + }; + /** + * Checks if two param value objects are equal (for a set of [[Param]] objects) + * + * @param params The list of [[Param]] objects to check + * @param values1 The first set of param values + * @param values2 The second set of param values + * + * @returns true if the param values in values1 and values2 are equal + */ + Param.equals = function (params, values1, values2) { + if (values1 === void 0) { values1 = {}; } + if (values2 === void 0) { values2 = {}; } + return Param.changed(params, values1, values2).length === 0; + }; + /** Returns true if a the parameter values are valid, according to the Param definitions */ + Param.validates = function (params, values$$1) { + if (values$$1 === void 0) { values$$1 = {}; } + return params.map(function (param) { return param.validates(values$$1[param.id]); }).reduce(allTrueR, true); + }; + return Param; +}()); + +/** @module path */ /** for typedoc */ +/** + * @internalapi + * + * A node in a [[TreeChanges]] path + * + * For a [[TreeChanges]] path, this class holds the stateful information for a single node in the path. + * Each PathNode corresponds to a state being entered, exited, or retained. + * The stateful information includes parameter values and resolve data. + */ +var PathNode = /** @class */ (function () { + function PathNode(stateOrNode) { + if (stateOrNode instanceof PathNode) { + var node = stateOrNode; + this.state = node.state; + this.paramSchema = node.paramSchema.slice(); + this.paramValues = extend({}, node.paramValues); + this.resolvables = node.resolvables.slice(); + this.views = node.views && node.views.slice(); + } + else { + var state = stateOrNode; + this.state = state; + this.paramSchema = state.parameters({ inherit: false }); + this.paramValues = {}; + this.resolvables = state.resolvables.map(function (res) { return res.clone(); }); + } + } + /** Sets [[paramValues]] for the node, from the values of an object hash */ + PathNode.prototype.applyRawParams = function (params) { + var getParamVal = function (paramDef) { return [paramDef.id, paramDef.value(params[paramDef.id])]; }; + this.paramValues = this.paramSchema.reduce(function (memo, pDef) { return applyPairs(memo, getParamVal(pDef)); }, {}); + return this; + }; + /** Gets a specific [[Param]] metadata that belongs to the node */ + PathNode.prototype.parameter = function (name) { + return find(this.paramSchema, propEq("id", name)); + }; + /** + * @returns true if the state and parameter values for another PathNode are + * equal to the state and param values for this PathNode + */ + PathNode.prototype.equals = function (node, paramsFn) { + var diff = this.diff(node, paramsFn); + return diff && diff.length === 0; + }; + /** + * Finds Params with different parameter values on another PathNode. + * + * Given another node (of the same state), finds the parameter values which differ. + * Returns the [[Param]] (schema objects) whose parameter values differ. + * + * Given another node for a different state, returns `false` + * + * @param node The node to compare to + * @param paramsFn A function that returns which parameters should be compared. + * @returns The [[Param]]s which differ, or null if the two nodes are for different states + */ + PathNode.prototype.diff = function (node, paramsFn) { + if (this.state !== node.state) + return false; + var params = paramsFn ? paramsFn(this) : this.paramSchema; + return Param.changed(params, this.paramValues, node.paramValues); + }; + /** Returns a clone of the PathNode */ + PathNode.clone = function (node) { + return new PathNode(node); + }; + return PathNode; +}()); + +/** @module path */ /** for typedoc */ +/** + * This class contains functions which convert TargetStates, Nodes and paths from one type to another. + */ +var PathUtils = /** @class */ (function () { + function PathUtils() { + } + /** Given a PathNode[], create an TargetState */ + PathUtils.makeTargetState = function (registry, path) { + var state = tail(path).state; + return new TargetState(registry, state, path.map(prop("paramValues")).reduce(mergeR, {}), {}); + }; + PathUtils.buildPath = function (targetState) { + var toParams = targetState.params(); + return targetState.$state().path.map(function (state) { return new PathNode(state).applyRawParams(toParams); }); + }; + /** Given a fromPath: PathNode[] and a TargetState, builds a toPath: PathNode[] */ + PathUtils.buildToPath = function (fromPath, targetState) { + var toPath = PathUtils.buildPath(targetState); + if (targetState.options().inherit) { + return PathUtils.inheritParams(fromPath, toPath, Object.keys(targetState.params())); + } + return toPath; + }; + /** + * Creates ViewConfig objects and adds to nodes. + * + * On each [[PathNode]], creates ViewConfig objects from the views: property of the node's state + */ + PathUtils.applyViewConfigs = function ($view, path, states) { + // Only apply the viewConfigs to the nodes for the given states + path.filter(function (node) { return inArray(states, node.state); }).forEach(function (node) { + var viewDecls = values(node.state.views || {}); + var subPath = PathUtils.subPath(path, function (n) { return n === node; }); + var viewConfigs = viewDecls.map(function (view) { return $view.createViewConfig(subPath, view); }); + node.views = viewConfigs.reduce(unnestR, []); + }); + }; + /** + * Given a fromPath and a toPath, returns a new to path which inherits parameters from the fromPath + * + * For a parameter in a node to be inherited from the from path: + * - The toPath's node must have a matching node in the fromPath (by state). + * - The parameter name must not be found in the toKeys parameter array. + * + * Note: the keys provided in toKeys are intended to be those param keys explicitly specified by some + * caller, for instance, $state.transitionTo(..., toParams). If a key was found in toParams, + * it is not inherited from the fromPath. + */ + PathUtils.inheritParams = function (fromPath, toPath, toKeys) { + if (toKeys === void 0) { toKeys = []; } + function nodeParamVals(path, state) { + var node = find(path, propEq('state', state)); + return extend({}, node && node.paramValues); + } + var noInherit = fromPath.map(function (node) { return node.paramSchema; }) + .reduce(unnestR, []) + .filter(function (param) { return !param.inherit; }) + .map(prop('id')); + /** + * Given an [[PathNode]] "toNode", return a new [[PathNode]] with param values inherited from the + * matching node in fromPath. Only inherit keys that aren't found in "toKeys" from the node in "fromPath"" + */ + function makeInheritedParamsNode(toNode) { + // All param values for the node (may include default key/vals, when key was not found in toParams) + var toParamVals = extend({}, toNode && toNode.paramValues); + // limited to only those keys found in toParams + var incomingParamVals = pick(toParamVals, toKeys); + toParamVals = omit(toParamVals, toKeys); + var fromParamVals = omit(nodeParamVals(fromPath, toNode.state) || {}, noInherit); + // extend toParamVals with any fromParamVals, then override any of those those with incomingParamVals + var ownParamVals = extend(toParamVals, fromParamVals, incomingParamVals); + return new PathNode(toNode.state).applyRawParams(ownParamVals); + } + // The param keys specified by the incoming toParams + return toPath.map(makeInheritedParamsNode); + }; + /** + * Computes the tree changes (entering, exiting) between a fromPath and toPath. + */ + PathUtils.treeChanges = function (fromPath, toPath, reloadState) { + var keep = 0, max = Math.min(fromPath.length, toPath.length); + var nodesMatch = function (node1, node2) { + return node1.equals(node2, PathUtils.nonDynamicParams); + }; + while (keep < max && fromPath[keep].state !== reloadState && nodesMatch(fromPath[keep], toPath[keep])) { + keep++; + } + /** Given a retained node, return a new node which uses the to node's param values */ + function applyToParams(retainedNode, idx) { + var cloned = PathNode.clone(retainedNode); + cloned.paramValues = toPath[idx].paramValues; + return cloned; + } + var from, retained, exiting, entering, to; + from = fromPath; + retained = from.slice(0, keep); + exiting = from.slice(keep); + // Create a new retained path (with shallow copies of nodes) which have the params of the toPath mapped + var retainedWithToParams = retained.map(applyToParams); + entering = toPath.slice(keep); + to = (retainedWithToParams).concat(entering); + return { from: from, to: to, retained: retained, exiting: exiting, entering: entering }; + }; + /** + * Returns a new path which is: the subpath of the first path which matches the second path. + * + * The new path starts from root and contains any nodes that match the nodes in the second path. + * It stops before the first non-matching node. + * + * Nodes are compared using their state property and their parameter values. + * If a `paramsFn` is provided, only the [[Param]] returned by the function will be considered when comparing nodes. + * + * @param pathA the first path + * @param pathB the second path + * @param paramsFn a function which returns the parameters to consider when comparing + * + * @returns an array of PathNodes from the first path which match the nodes in the second path + */ + PathUtils.matching = function (pathA, pathB, paramsFn) { + var done = false; + var tuples = arrayTuples(pathA, pathB); + return tuples.reduce(function (matching, _a) { + var nodeA = _a[0], nodeB = _a[1]; + done = done || !nodeA.equals(nodeB, paramsFn); + return done ? matching : matching.concat(nodeA); + }, []); + }; + /** + * Returns true if two paths are identical. + * + * @param pathA + * @param pathB + * @param paramsFn a function which returns the parameters to consider when comparing + * @returns true if the the states and parameter values for both paths are identical + */ + PathUtils.equals = function (pathA, pathB, paramsFn) { + return pathA.length === pathB.length && + PathUtils.matching(pathA, pathB, paramsFn).length === pathA.length; + }; + /** + * Return a subpath of a path, which stops at the first matching node + * + * Given an array of nodes, returns a subset of the array starting from the first node, + * stopping when the first node matches the predicate. + * + * @param path a path of [[PathNode]]s + * @param predicate a [[Predicate]] fn that matches [[PathNode]]s + * @returns a subpath up to the matching node, or undefined if no match is found + */ + PathUtils.subPath = function (path, predicate) { + var node = find(path, predicate); + var elementIdx = path.indexOf(node); + return elementIdx === -1 ? undefined : path.slice(0, elementIdx + 1); + }; + PathUtils.nonDynamicParams = function (node) { + return node.state.parameters({ inherit: false }) + .filter(function (param) { return !param.dynamic; }); + }; + /** Gets the raw parameter values from a path */ + PathUtils.paramValues = function (path) { + return path.reduce(function (acc, node) { return extend(acc, node.paramValues); }, {}); + }; + return PathUtils; +}()); + +/** + * @coreapi + * @module resolve + */ /** for typedoc */ +// TODO: explicitly make this user configurable +var defaultResolvePolicy = { + when: "LAZY", + async: "WAIT" +}; +/** + * The basic building block for the resolve system. + * + * Resolvables encapsulate a state's resolve's resolveFn, the resolveFn's declared dependencies, the wrapped (.promise), + * and the unwrapped-when-complete (.data) result of the resolveFn. + * + * Resolvable.get() either retrieves the Resolvable's existing promise, or else invokes resolve() (which invokes the + * resolveFn) and returns the resulting promise. + * + * Resolvable.get() and Resolvable.resolve() both execute within a context path, which is passed as the first + * parameter to those fns. + */ +var Resolvable = /** @class */ (function () { + function Resolvable(arg1, resolveFn, deps, policy, data) { + this.resolved = false; + this.promise = undefined; + if (arg1 instanceof Resolvable) { + extend(this, arg1); + } + else if (isFunction(resolveFn)) { + if (isNullOrUndefined(arg1)) + throw new Error("new Resolvable(): token argument is required"); + if (!isFunction(resolveFn)) + throw new Error("new Resolvable(): resolveFn argument must be a function"); + this.token = arg1; + this.policy = policy; + this.resolveFn = resolveFn; + this.deps = deps || []; + this.data = data; + this.resolved = data !== undefined; + this.promise = this.resolved ? services.$q.when(this.data) : undefined; + } + else if (isObject(arg1) && arg1.token && isFunction(arg1.resolveFn)) { + var literal = arg1; + return new Resolvable(literal.token, literal.resolveFn, literal.deps, literal.policy, literal.data); + } + } + Resolvable.prototype.getPolicy = function (state) { + var thisPolicy = this.policy || {}; + var statePolicy = state && state.resolvePolicy || {}; + return { + when: thisPolicy.when || statePolicy.when || defaultResolvePolicy.when, + async: thisPolicy.async || statePolicy.async || defaultResolvePolicy.async, + }; + }; + /** + * Asynchronously resolve this Resolvable's data + * + * Given a ResolveContext that this Resolvable is found in: + * Wait for this Resolvable's dependencies, then invoke this Resolvable's function + * and update the Resolvable's state + */ + Resolvable.prototype.resolve = function (resolveContext, trans) { + var _this = this; + var $q = services.$q; + // Gets all dependencies from ResolveContext and wait for them to be resolved + var getResolvableDependencies = function () { + return $q.all(resolveContext.getDependencies(_this).map(function (resolvable) { + return resolvable.get(resolveContext, trans); + })); + }; + // Invokes the resolve function passing the resolved dependencies as arguments + var invokeResolveFn = function (resolvedDeps) { + return _this.resolveFn.apply(null, resolvedDeps); + }; + /** + * For RXWAIT policy: + * + * Given an observable returned from a resolve function: + * - enables .cache() mode (this allows multicast subscribers) + * - then calls toPromise() (this triggers subscribe() and thus fetches) + * - Waits for the promise, then return the cached observable (not the first emitted value). + */ + var waitForRx = function (observable$) { + var cached = observable$.cache(1); + return cached.take(1).toPromise().then(function () { return cached; }); + }; + // If the resolve policy is RXWAIT, wait for the observable to emit something. otherwise pass through. + var node = resolveContext.findNode(this); + var state = node && node.state; + var maybeWaitForRx = this.getPolicy(state).async === "RXWAIT" ? waitForRx : identity; + // After the final value has been resolved, update the state of the Resolvable + var applyResolvedValue = function (resolvedValue) { + _this.data = resolvedValue; + _this.resolved = true; + trace.traceResolvableResolved(_this, trans); + return _this.data; + }; + // Sets the promise property first, then getsResolvableDependencies in the context of the promise chain. Always waits one tick. + return this.promise = $q.when() + .then(getResolvableDependencies) + .then(invokeResolveFn) + .then(maybeWaitForRx) + .then(applyResolvedValue); + }; + /** + * Gets a promise for this Resolvable's data. + * + * Fetches the data and returns a promise. + * Returns the existing promise if it has already been fetched once. + */ + Resolvable.prototype.get = function (resolveContext, trans) { + return this.promise || this.resolve(resolveContext, trans); + }; + Resolvable.prototype.toString = function () { + return "Resolvable(token: " + stringify(this.token) + ", requires: [" + this.deps.map(stringify) + "])"; + }; + Resolvable.prototype.clone = function () { + return new Resolvable(this); + }; + Resolvable.fromData = function (token, data) { + return new Resolvable(token, function () { return data; }, null, null, data); + }; + return Resolvable; +}()); + +/** @internalapi */ +var resolvePolicies = { + when: { + LAZY: "LAZY", + EAGER: "EAGER" + }, + async: { + WAIT: "WAIT", + NOWAIT: "NOWAIT", + RXWAIT: "RXWAIT" + } +}; + +/** @module resolve */ +/** for typedoc */ +var whens = resolvePolicies.when; +var ALL_WHENS = [whens.EAGER, whens.LAZY]; +var EAGER_WHENS = [whens.EAGER]; +var NATIVE_INJECTOR_TOKEN = "Native Injector"; +/** + * Encapsulates Dependency Injection for a path of nodes + * + * UI-Router states are organized as a tree. + * A nested state has a path of ancestors to the root of the tree. + * When a state is being activated, each element in the path is wrapped as a [[PathNode]]. + * A `PathNode` is a stateful object that holds things like parameters and resolvables for the state being activated. + * + * The ResolveContext closes over the [[PathNode]]s, and provides DI for the last node in the path. + */ +var ResolveContext = /** @class */ (function () { + function ResolveContext(_path) { + this._path = _path; + } + /** Gets all the tokens found in the resolve context, de-duplicated */ + ResolveContext.prototype.getTokens = function () { + return this._path.reduce(function (acc, node) { return acc.concat(node.resolvables.map(function (r) { return r.token; })); }, []).reduce(uniqR, []); + }; + /** + * Gets the Resolvable that matches the token + * + * Gets the last Resolvable that matches the token in this context, or undefined. + * Throws an error if it doesn't exist in the ResolveContext + */ + ResolveContext.prototype.getResolvable = function (token) { + var matching = this._path.map(function (node) { return node.resolvables; }) + .reduce(unnestR, []) + .filter(function (r) { return r.token === token; }); + return tail(matching); + }; + /** Returns the [[ResolvePolicy]] for the given [[Resolvable]] */ + ResolveContext.prototype.getPolicy = function (resolvable) { + var node = this.findNode(resolvable); + return resolvable.getPolicy(node.state); + }; + /** + * Returns a ResolveContext that includes a portion of this one + * + * Given a state, this method creates a new ResolveContext from this one. + * The new context starts at the first node (root) and stops at the node for the `state` parameter. + * + * #### Why + * + * When a transition is created, the nodes in the "To Path" are injected from a ResolveContext. + * A ResolveContext closes over a path of [[PathNode]]s and processes the resolvables. + * The "To State" can inject values from its own resolvables, as well as those from all its ancestor state's (node's). + * This method is used to create a narrower context when injecting ancestor nodes. + * + * @example + * `let ABCD = new ResolveContext([A, B, C, D]);` + * + * Given a path `[A, B, C, D]`, where `A`, `B`, `C` and `D` are nodes for states `a`, `b`, `c`, `d`: + * When injecting `D`, `D` should have access to all resolvables from `A`, `B`, `C`, `D`. + * However, `B` should only be able to access resolvables from `A`, `B`. + * + * When resolving for the `B` node, first take the full "To Path" Context `[A,B,C,D]` and limit to the subpath `[A,B]`. + * `let AB = ABCD.subcontext(a)` + */ + ResolveContext.prototype.subContext = function (state) { + return new ResolveContext(PathUtils.subPath(this._path, function (node) { return node.state === state; })); + }; + /** + * Adds Resolvables to the node that matches the state + * + * This adds a [[Resolvable]] (generally one created on the fly; not declared on a [[StateDeclaration.resolve]] block). + * The resolvable is added to the node matching the `state` parameter. + * + * These new resolvables are not automatically fetched. + * The calling code should either fetch them, fetch something that depends on them, + * or rely on [[resolvePath]] being called when some state is being entered. + * + * Note: each resolvable's [[ResolvePolicy]] is merged with the state's policy, and the global default. + * + * @param newResolvables the new Resolvables + * @param state Used to find the node to put the resolvable on + */ + ResolveContext.prototype.addResolvables = function (newResolvables, state) { + var node = find(this._path, propEq('state', state)); + var keys = newResolvables.map(function (r) { return r.token; }); + node.resolvables = node.resolvables.filter(function (r) { return keys.indexOf(r.token) === -1; }).concat(newResolvables); + }; + /** + * Returns a promise for an array of resolved path Element promises + * + * @param when + * @param trans + * @returns {Promise|any} + */ + ResolveContext.prototype.resolvePath = function (when, trans) { + var _this = this; + if (when === void 0) { when = "LAZY"; } + // This option determines which 'when' policy Resolvables we are about to fetch. + var whenOption = inArray(ALL_WHENS, when) ? when : "LAZY"; + // If the caller specified EAGER, only the EAGER Resolvables are fetched. + // if the caller specified LAZY, both EAGER and LAZY Resolvables are fetched.` + var matchedWhens = whenOption === resolvePolicies.when.EAGER ? EAGER_WHENS : ALL_WHENS; + // get the subpath to the state argument, if provided + trace.traceResolvePath(this._path, when, trans); + var matchesPolicy = function (acceptedVals, whenOrAsync) { + return function (resolvable) { + return inArray(acceptedVals, _this.getPolicy(resolvable)[whenOrAsync]); + }; + }; + // Trigger all the (matching) Resolvables in the path + // Reduce all the "WAIT" Resolvables into an array + var promises = this._path.reduce(function (acc, node) { + var nodeResolvables = node.resolvables.filter(matchesPolicy(matchedWhens, 'when')); + var nowait = nodeResolvables.filter(matchesPolicy(['NOWAIT'], 'async')); + var wait = nodeResolvables.filter(not(matchesPolicy(['NOWAIT'], 'async'))); + // For the matching Resolvables, start their async fetch process. + var subContext = _this.subContext(node.state); + var getResult = function (r) { return r.get(subContext, trans) + .then(function (value) { return ({ token: r.token, value: value }); }); }; + nowait.forEach(getResult); + return acc.concat(wait.map(getResult)); + }, []); + // Wait for all the "WAIT" resolvables + return services.$q.all(promises); + }; + ResolveContext.prototype.injector = function () { + return this._injector || (this._injector = new UIInjectorImpl(this)); + }; + ResolveContext.prototype.findNode = function (resolvable) { + return find(this._path, function (node) { return inArray(node.resolvables, resolvable); }); + }; + /** + * Gets the async dependencies of a Resolvable + * + * Given a Resolvable, returns its dependencies as a Resolvable[] + */ + ResolveContext.prototype.getDependencies = function (resolvable) { + var _this = this; + var node = this.findNode(resolvable); + // Find which other resolvables are "visible" to the `resolvable` argument + // subpath stopping at resolvable's node, or the whole path (if the resolvable isn't in the path) + var subPath = PathUtils.subPath(this._path, function (x) { return x === node; }) || this._path; + var availableResolvables = subPath + .reduce(function (acc, _node) { return acc.concat(_node.resolvables); }, []) //all of subpath's resolvables + .filter(function (res) { return res !== resolvable; }); // filter out the `resolvable` argument + var getDependency = function (token) { + var matching = availableResolvables.filter(function (r) { return r.token === token; }); + if (matching.length) + return tail(matching); + var fromInjector = _this.injector().getNative(token); + if (isUndefined(fromInjector)) { + throw new Error("Could not find Dependency Injection token: " + stringify(token)); + } + return new Resolvable(token, function () { return fromInjector; }, [], fromInjector); + }; + return resolvable.deps.map(getDependency); + }; + return ResolveContext; +}()); +var UIInjectorImpl = /** @class */ (function () { + function UIInjectorImpl(context) { + this.context = context; + this.native = this.get(NATIVE_INJECTOR_TOKEN) || services.$injector; + } + UIInjectorImpl.prototype.get = function (token) { + var resolvable = this.context.getResolvable(token); + if (resolvable) { + if (this.context.getPolicy(resolvable).async === 'NOWAIT') { + return resolvable.get(this.context); + } + if (!resolvable.resolved) { + throw new Error("Resolvable async .get() not complete:" + stringify(resolvable.token)); + } + return resolvable.data; + } + return this.getNative(token); + }; + UIInjectorImpl.prototype.getAsync = function (token) { + var resolvable = this.context.getResolvable(token); + if (resolvable) + return resolvable.get(this.context); + return services.$q.when(this.native.get(token)); + }; + UIInjectorImpl.prototype.getNative = function (token) { + return this.native && this.native.get(token); + }; + return UIInjectorImpl; +}()); + +/** + * @coreapi + * @module transition + */ +/** for typedoc */ +/** @hidden */ +var stateSelf = prop("self"); +/** + * Represents a transition between two states. + * + * When navigating to a state, we are transitioning **from** the current state **to** the new state. + * + * This object contains all contextual information about the to/from states, parameters, resolves. + * It has information about all states being entered and exited as a result of the transition. + */ +var Transition = /** @class */ (function () { + /** + * Creates a new Transition object. + * + * If the target state is not valid, an error is thrown. + * + * @internalapi + * + * @param fromPath The path of [[PathNode]]s from which the transition is leaving. The last node in the `fromPath` + * encapsulates the "from state". + * @param targetState The target state and parameters being transitioned to (also, the transition options) + * @param router The [[UIRouter]] instance + */ + function Transition(fromPath, targetState, router) { + var _this = this; + /** @hidden */ + this._deferred = services.$q.defer(); + /** + * This promise is resolved or rejected based on the outcome of the Transition. + * + * When the transition is successful, the promise is resolved + * When the transition is unsuccessful, the promise is rejected with the [[Rejection]] or javascript error + */ + this.promise = this._deferred.promise; + /** @hidden Holds the hook registration functions such as those passed to Transition.onStart() */ + this._registeredHooks = {}; + /** @hidden */ + this._hookBuilder = new HookBuilder(this); + /** Checks if this transition is currently active/running. */ + this.isActive = function () { + return _this.router.globals.transition === _this; + }; + this.router = router; + this._targetState = targetState; + if (!targetState.valid()) { + throw new Error(targetState.error()); + } + // current() is assumed to come from targetState.options, but provide a naive implementation otherwise. + this._options = extend({ current: val(this) }, targetState.options()); + this.$id = router.transitionService._transitionCount++; + var toPath = PathUtils.buildToPath(fromPath, targetState); + this._treeChanges = PathUtils.treeChanges(fromPath, toPath, this._options.reloadState); + this.createTransitionHookRegFns(); + var onCreateHooks = this._hookBuilder.buildHooksForPhase(exports.TransitionHookPhase.CREATE); + TransitionHook.invokeHooks(onCreateHooks, function () { return null; }); + this.applyViewConfigs(router); + } + /** @hidden */ + Transition.prototype.onBefore = function (criteria, callback, options) { return; }; + /** @inheritdoc */ + Transition.prototype.onStart = function (criteria, callback, options) { return; }; + /** @inheritdoc */ + Transition.prototype.onExit = function (criteria, callback, options) { return; }; + /** @inheritdoc */ + Transition.prototype.onRetain = function (criteria, callback, options) { return; }; + /** @inheritdoc */ + Transition.prototype.onEnter = function (criteria, callback, options) { return; }; + /** @inheritdoc */ + Transition.prototype.onFinish = function (criteria, callback, options) { return; }; + /** @inheritdoc */ + Transition.prototype.onSuccess = function (criteria, callback, options) { return; }; + /** @inheritdoc */ + Transition.prototype.onError = function (criteria, callback, options) { return; }; + /** @hidden + * Creates the transition-level hook registration functions + * (which can then be used to register hooks) + */ + Transition.prototype.createTransitionHookRegFns = function () { + var _this = this; + this.router.transitionService._pluginapi._getEvents() + .filter(function (type) { return type.hookPhase !== exports.TransitionHookPhase.CREATE; }) + .forEach(function (type) { return makeEvent(_this, _this.router.transitionService, type); }); + }; + /** @internalapi */ + Transition.prototype.getHooks = function (hookName) { + return this._registeredHooks[hookName]; + }; + Transition.prototype.applyViewConfigs = function (router) { + var enteringStates = this._treeChanges.entering.map(function (node) { return node.state; }); + PathUtils.applyViewConfigs(router.transitionService.$view, this._treeChanges.to, enteringStates); + }; + /** + * @internalapi + * + * @returns the internal from [State] object + */ + Transition.prototype.$from = function () { + return tail(this._treeChanges.from).state; + }; + /** + * @internalapi + * + * @returns the internal to [State] object + */ + Transition.prototype.$to = function () { + return tail(this._treeChanges.to).state; + }; + /** + * Returns the "from state" + * + * Returns the state that the transition is coming *from*. + * + * @returns The state declaration object for the Transition's ("from state"). + */ + Transition.prototype.from = function () { + return this.$from().self; + }; + /** + * Returns the "to state" + * + * Returns the state that the transition is going *to*. + * + * @returns The state declaration object for the Transition's target state ("to state"). + */ + Transition.prototype.to = function () { + return this.$to().self; + }; + /** + * Gets the Target State + * + * A transition's [[TargetState]] encapsulates the [[to]] state, the [[params]], and the [[options]] as a single object. + * + * @returns the [[TargetState]] of this Transition + */ + Transition.prototype.targetState = function () { + return this._targetState; + }; + /** + * Determines whether two transitions are equivalent. + * @deprecated + */ + Transition.prototype.is = function (compare) { + if (compare instanceof Transition) { + // TODO: Also compare parameters + return this.is({ to: compare.$to().name, from: compare.$from().name }); + } + return !((compare.to && !matchState(this.$to(), compare.to)) || + (compare.from && !matchState(this.$from(), compare.from))); + }; + Transition.prototype.params = function (pathname) { + if (pathname === void 0) { pathname = "to"; } + return Object.freeze(this._treeChanges[pathname].map(prop("paramValues")).reduce(mergeR, {})); + }; + /** + * Creates a [[UIInjector]] Dependency Injector + * + * Returns a Dependency Injector for the Transition's target state (to state). + * The injector provides resolve values which the target state has access to. + * + * The `UIInjector` can also provide values from the native root/global injector (ng1/ng2). + * + * #### Example: + * ```js + * .onEnter({ entering: 'myState' }, trans => { + * var myResolveValue = trans.injector().get('myResolve'); + * // Inject a global service from the global/native injector (if it exists) + * var MyService = trans.injector().get('MyService'); + * }) + * ``` + * + * In some cases (such as `onBefore`), you may need access to some resolve data but it has not yet been fetched. + * You can use [[UIInjector.getAsync]] to get a promise for the data. + * #### Example: + * ```js + * .onBefore({}, trans => { + * return trans.injector().getAsync('myResolve').then(myResolveValue => + * return myResolveValue !== 'ABORT'; + * }); + * }); + * ``` + * + * If a `state` is provided, the injector that is returned will be limited to resolve values that the provided state has access to. + * This can be useful if both a parent state `foo` and a child state `foo.bar` have both defined a resolve such as `data`. + * #### Example: + * ```js + * .onEnter({ to: 'foo.bar' }, trans => { + * // returns result of `foo` state's `data` resolve + * // even though `foo.bar` also has a `data` resolve + * var fooData = trans.injector('foo').get('data'); + * }); + * ``` + * + * If you need resolve data from the exiting states, pass `'from'` as `pathName`. + * The resolve data from the `from` path will be returned. + * #### Example: + * ```js + * .onExit({ exiting: 'foo.bar' }, trans => { + * // Gets the resolve value of `data` from the exiting state. + * var fooData = trans.injector(null, 'foo.bar').get('data'); + * }); + * ``` + * + * + * @param state Limits the resolves provided to only the resolves the provided state has access to. + * @param pathName Default: `'to'`: Chooses the path for which to create the injector. Use this to access resolves for `exiting` states. + * + * @returns a [[UIInjector]] + */ + Transition.prototype.injector = function (state, pathName) { + if (pathName === void 0) { pathName = "to"; } + var path = this._treeChanges[pathName]; + if (state) + path = PathUtils.subPath(path, function (node) { return node.state === state || node.state.name === state; }); + return new ResolveContext(path).injector(); + }; + /** + * Gets all available resolve tokens (keys) + * + * This method can be used in conjunction with [[injector]] to inspect the resolve values + * available to the Transition. + * + * This returns all the tokens defined on [[StateDeclaration.resolve]] blocks, for the states + * in the Transition's [[TreeChanges.to]] path. + * + * #### Example: + * This example logs all resolve values + * ```js + * let tokens = trans.getResolveTokens(); + * tokens.forEach(token => console.log(token + " = " + trans.injector().get(token))); + * ``` + * + * #### Example: + * This example creates promises for each resolve value. + * This triggers fetches of resolves (if any have not yet been fetched). + * When all promises have all settled, it logs the resolve values. + * ```js + * let tokens = trans.getResolveTokens(); + * let promise = tokens.map(token => trans.injector().getAsync(token)); + * Promise.all(promises).then(values => console.log("Resolved values: " + values)); + * ``` + * + * Note: Angular 1 users whould use `$q.all()` + * + * @param pathname resolve context's path name (e.g., `to` or `from`) + * + * @returns an array of resolve tokens (keys) + */ + Transition.prototype.getResolveTokens = function (pathname) { + if (pathname === void 0) { pathname = "to"; } + return new ResolveContext(this._treeChanges[pathname]).getTokens(); + }; + /** + * Dynamically adds a new [[Resolvable]] (i.e., [[StateDeclaration.resolve]]) to this transition. + * + * #### Example: + * ```js + * transitionService.onBefore({}, transition => { + * transition.addResolvable({ + * token: 'myResolve', + * deps: ['MyService'], + * resolveFn: myService => myService.getData() + * }); + * }); + * ``` + * + * @param resolvable a [[ResolvableLiteral]] object (or a [[Resolvable]]) + * @param state the state in the "to path" which should receive the new resolve (otherwise, the root state) + */ + Transition.prototype.addResolvable = function (resolvable, state) { + if (state === void 0) { state = ""; } + resolvable = is(Resolvable)(resolvable) ? resolvable : new Resolvable(resolvable); + var stateName = (typeof state === "string") ? state : state.name; + var topath = this._treeChanges.to; + var targetNode = find(topath, function (node) { return node.state.name === stateName; }); + var resolveContext = new ResolveContext(topath); + resolveContext.addResolvables([resolvable], targetNode.state); + }; + /** + * Gets the transition from which this transition was redirected. + * + * If the current transition is a redirect, this method returns the transition that was redirected. + * + * #### Example: + * ```js + * let transitionA = $state.go('A').transition + * transitionA.onStart({}, () => $state.target('B')); + * $transitions.onSuccess({ to: 'B' }, (trans) => { + * trans.to().name === 'B'; // true + * trans.redirectedFrom() === transitionA; // true + * }); + * ``` + * + * @returns The previous Transition, or null if this Transition is not the result of a redirection + */ + Transition.prototype.redirectedFrom = function () { + return this._options.redirectedFrom || null; + }; + /** + * Gets the original transition in a redirect chain + * + * A transition might belong to a long chain of multiple redirects. + * This method walks the [[redirectedFrom]] chain back to the original (first) transition in the chain. + * + * #### Example: + * ```js + * // states + * registry.register({ name: 'A', redirectTo: 'B' }); + * registry.register({ name: 'B', redirectTo: 'C' }); + * registry.register({ name: 'C', redirectTo: 'D' }); + * registry.register({ name: 'D' }); + * + * let transitionA = $state.go('A').transition + * + * $transitions.onSuccess({ to: 'D' }, (trans) => { + * trans.to().name === 'D'; // true + * trans.redirectedFrom().to().name === 'C'; // true + * trans.originalTransition() === transitionA; // true + * trans.originalTransition().to().name === 'A'; // true + * }); + * ``` + * + * @returns The original Transition that started a redirect chain + */ + Transition.prototype.originalTransition = function () { + var rf = this.redirectedFrom(); + return (rf && rf.originalTransition()) || this; + }; + /** + * Get the transition options + * + * @returns the options for this Transition. + */ + Transition.prototype.options = function () { + return this._options; + }; + /** + * Gets the states being entered. + * + * @returns an array of states that will be entered during this transition. + */ + Transition.prototype.entering = function () { + return map(this._treeChanges.entering, prop('state')).map(stateSelf); + }; + /** + * Gets the states being exited. + * + * @returns an array of states that will be exited during this transition. + */ + Transition.prototype.exiting = function () { + return map(this._treeChanges.exiting, prop('state')).map(stateSelf).reverse(); + }; + /** + * Gets the states being retained. + * + * @returns an array of states that are already entered from a previous Transition, that will not be + * exited during this Transition + */ + Transition.prototype.retained = function () { + return map(this._treeChanges.retained, prop('state')).map(stateSelf); + }; + /** + * Get the [[ViewConfig]]s associated with this Transition + * + * Each state can define one or more views (template/controller), which are encapsulated as `ViewConfig` objects. + * This method fetches the `ViewConfigs` for a given path in the Transition (e.g., "to" or "entering"). + * + * @param pathname the name of the path to fetch views for: + * (`'to'`, `'from'`, `'entering'`, `'exiting'`, `'retained'`) + * @param state If provided, only returns the `ViewConfig`s for a single state in the path + * + * @returns a list of ViewConfig objects for the given path. + */ + Transition.prototype.views = function (pathname, state) { + if (pathname === void 0) { pathname = "entering"; } + var path = this._treeChanges[pathname]; + path = !state ? path : path.filter(propEq('state', state)); + return path.map(prop("views")).filter(identity).reduce(unnestR, []); + }; + Transition.prototype.treeChanges = function (pathname) { + return pathname ? this._treeChanges[pathname] : this._treeChanges; + }; + /** + * Creates a new transition that is a redirection of the current one. + * + * This transition can be returned from a [[TransitionService]] hook to + * redirect a transition to a new state and/or set of parameters. + * + * @internalapi + * + * @returns Returns a new [[Transition]] instance. + */ + Transition.prototype.redirect = function (targetState) { + var redirects = 1, trans = this; + while ((trans = trans.redirectedFrom()) != null) { + if (++redirects > 20) + throw new Error("Too many consecutive Transition redirects (20+)"); + } + var redirectOpts = { redirectedFrom: this, source: "redirect" }; + // If the original transition was caused by URL sync, then use { location: 'replace' } + // on the new transition (unless the target state explicitly specifies location: false). + // This causes the original url to be replaced with the url for the redirect target + // so the original url disappears from the browser history. + if (this.options().source === 'url' && targetState.options().location !== false) { + redirectOpts.location = 'replace'; + } + var newOptions = extend({}, this.options(), targetState.options(), redirectOpts); + targetState = targetState.withOptions(newOptions, true); + var newTransition = this.router.transitionService.create(this._treeChanges.from, targetState); + var originalEnteringNodes = this._treeChanges.entering; + var redirectEnteringNodes = newTransition._treeChanges.entering; + // --- Re-use resolve data from original transition --- + // When redirecting from a parent state to a child state where the parent parameter values haven't changed + // (because of the redirect), the resolves fetched by the original transition are still valid in the + // redirected transition. + // + // This allows you to define a redirect on a parent state which depends on an async resolve value. + // You can wait for the resolve, then redirect to a child state based on the result. + // The redirected transition does not have to re-fetch the resolve. + // --------------------------------------------------------- + var nodeIsReloading = function (reloadState) { return function (node) { + return reloadState && node.state.includes[reloadState.name]; + }; }; + // Find any "entering" nodes in the redirect path that match the original path and aren't being reloaded + var matchingEnteringNodes = PathUtils.matching(redirectEnteringNodes, originalEnteringNodes, PathUtils.nonDynamicParams) + .filter(not(nodeIsReloading(targetState.options().reloadState))); + // Use the existing (possibly pre-resolved) resolvables for the matching entering nodes. + matchingEnteringNodes.forEach(function (node, idx) { + node.resolvables = originalEnteringNodes[idx].resolvables; + }); + return newTransition; + }; + /** @hidden If a transition doesn't exit/enter any states, returns any [[Param]] whose value changed */ + Transition.prototype._changedParams = function () { + var tc = this._treeChanges; + /** Return undefined if it's not a "dynamic" transition, for the following reasons */ + // If user explicitly wants a reload + if (this._options.reload) + return undefined; + // If any states are exiting or entering + if (tc.exiting.length || tc.entering.length) + return undefined; + // If to/from path lengths differ + if (tc.to.length !== tc.from.length) + return undefined; + // If the to/from paths are different + var pathsDiffer = arrayTuples(tc.to, tc.from) + .map(function (tuple) { return tuple[0].state !== tuple[1].state; }) + .reduce(anyTrueR, false); + if (pathsDiffer) + return undefined; + // Find any parameter values that differ + var nodeSchemas = tc.to.map(function (node) { return node.paramSchema; }); + var _a = [tc.to, tc.from].map(function (path) { return path.map(function (x) { return x.paramValues; }); }), toValues = _a[0], fromValues = _a[1]; + var tuples = arrayTuples(nodeSchemas, toValues, fromValues); + return tuples.map(function (_a) { + var schema = _a[0], toVals = _a[1], fromVals = _a[2]; + return Param.changed(schema, toVals, fromVals); + }).reduce(unnestR, []); + }; + /** + * Returns true if the transition is dynamic. + * + * A transition is dynamic if no states are entered nor exited, but at least one dynamic parameter has changed. + * + * @returns true if the Transition is dynamic + */ + Transition.prototype.dynamic = function () { + var changes = this._changedParams(); + return !changes ? false : changes.map(function (x) { return x.dynamic; }).reduce(anyTrueR, false); + }; + /** + * Returns true if the transition is ignored. + * + * A transition is ignored if no states are entered nor exited, and no parameter values have changed. + * + * @returns true if the Transition is ignored. + */ + Transition.prototype.ignored = function () { + return !!this._ignoredReason(); + }; + /** @hidden */ + Transition.prototype._ignoredReason = function () { + var pending = this.router.globals.transition; + var reloadState = this._options.reloadState; + var same = function (pathA, pathB) { + if (pathA.length !== pathB.length) + return false; + var matching = PathUtils.matching(pathA, pathB); + return pathA.length === matching.filter(function (node) { return !reloadState || !node.state.includes[reloadState.name]; }).length; + }; + var newTC = this.treeChanges(); + var pendTC = pending && pending.treeChanges(); + if (pendTC && same(pendTC.to, newTC.to) && same(pendTC.exiting, newTC.exiting)) + return "SameAsPending"; + if (newTC.exiting.length === 0 && newTC.entering.length === 0 && same(newTC.from, newTC.to)) + return "SameAsCurrent"; + }; + /** + * Runs the transition + * + * This method is generally called from the [[StateService.transitionTo]] + * + * @internalapi + * + * @returns a promise for a successful transition. + */ + Transition.prototype.run = function () { + var _this = this; + var runAllHooks = TransitionHook.runAllHooks; + // Gets transition hooks array for the given phase + var getHooksFor = function (phase) { + return _this._hookBuilder.buildHooksForPhase(phase); + }; + // When the chain is complete, then resolve or reject the deferred + var transitionSuccess = function () { + trace.traceSuccess(_this.$to(), _this); + _this.success = true; + _this._deferred.resolve(_this.to()); + runAllHooks(getHooksFor(exports.TransitionHookPhase.SUCCESS)); + }; + var transitionError = function (reason) { + trace.traceError(reason, _this); + _this.success = false; + _this._deferred.reject(reason); + _this._error = reason; + runAllHooks(getHooksFor(exports.TransitionHookPhase.ERROR)); + }; + var runTransition = function () { + // Wait to build the RUN hook chain until the BEFORE hooks are done + // This allows a BEFORE hook to dynamically add additional RUN hooks via the Transition object. + var allRunHooks = getHooksFor(exports.TransitionHookPhase.RUN); + var done = function () { return services.$q.when(undefined); }; + return TransitionHook.invokeHooks(allRunHooks, done); + }; + var startTransition = function () { + var globals = _this.router.globals; + globals.lastStartedTransitionId = _this.$id; + globals.transition = _this; + globals.transitionHistory.enqueue(_this); + trace.traceTransitionStart(_this); + return services.$q.when(undefined); + }; + var allBeforeHooks = getHooksFor(exports.TransitionHookPhase.BEFORE); + TransitionHook.invokeHooks(allBeforeHooks, startTransition) + .then(runTransition) + .then(transitionSuccess, transitionError); + return this.promise; + }; + /** + * Checks if the Transition is valid + * + * @returns true if the Transition is valid + */ + Transition.prototype.valid = function () { + return !this.error() || this.success !== undefined; + }; + /** + * Aborts this transition + * + * Imperative API to abort a Transition. + * This only applies to Transitions that are not yet complete. + */ + Transition.prototype.abort = function () { + // Do not set flag if the transition is already complete + if (isUndefined(this.success)) { + this._aborted = true; + } + }; + /** + * The Transition error reason. + * + * If the transition is invalid (and could not be run), returns the reason the transition is invalid. + * If the transition was valid and ran, but was not successful, returns the reason the transition failed. + * + * @returns an error message explaining why the transition is invalid, or the reason the transition failed. + */ + Transition.prototype.error = function () { + var state = this.$to(); + if (state.self.abstract) + return "Cannot transition to abstract state '" + state.name + "'"; + var paramDefs = state.parameters(), values$$1 = this.params(); + var invalidParams = paramDefs.filter(function (param) { return !param.validates(values$$1[param.id]); }); + if (invalidParams.length) { + return "Param values not valid for state '" + state.name + "'. Invalid params: [ " + invalidParams.map(function (param) { return param.id; }).join(', ') + " ]"; + } + if (this.success === false) + return this._error; + }; + /** + * A string representation of the Transition + * + * @returns A string representation of the Transition + */ + Transition.prototype.toString = function () { + var fromStateOrName = this.from(); + var toStateOrName = this.to(); + var avoidEmptyHash = function (params) { + return (params["#"] !== null && params["#"] !== undefined) ? params : omit(params, ["#"]); + }; + // (X) means the to state is invalid. + var id = this.$id, from = isObject(fromStateOrName) ? fromStateOrName.name : fromStateOrName, fromParams = stringify(avoidEmptyHash(this._treeChanges.from.map(prop('paramValues')).reduce(mergeR, {}))), toValid = this.valid() ? "" : "(X) ", to = isObject(toStateOrName) ? toStateOrName.name : toStateOrName, toParams = stringify(avoidEmptyHash(this.params())); + return "Transition#" + id + "( '" + from + "'" + fromParams + " -> " + toValid + "'" + to + "'" + toParams + " )"; + }; + /** @hidden */ + Transition.diToken = Transition; + return Transition; +}()); + +/** + * Functions that manipulate strings + * + * Although these functions are exported, they are subject to change without notice. + * + * @module common_strings + */ /** */ +/** + * Returns a string shortened to a maximum length + * + * If the string is already less than the `max` length, return the string. + * Else return the string, shortened to `max - 3` and append three dots ("..."). + * + * @param max the maximum length of the string to return + * @param str the input string + */ +function maxLength(max, str) { + if (str.length <= max) + return str; + return str.substr(0, max - 3) + "..."; +} +/** + * Returns a string, with spaces added to the end, up to a desired str length + * + * If the string is already longer than the desired length, return the string. + * Else returns the string, with extra spaces on the end, such that it reaches `length` characters. + * + * @param length the desired length of the string to return + * @param str the input string + */ +function padString(length, str) { + while (str.length < length) + str += " "; + return str; +} +function kebobString(camelCase) { + return camelCase + .replace(/^([A-Z])/, function ($1) { return $1.toLowerCase(); }) // replace first char + .replace(/([A-Z])/g, function ($1) { return "-" + $1.toLowerCase(); }); // replace rest +} +function functionToString(fn) { + var fnStr = fnToString(fn); + var namedFunctionMatch = fnStr.match(/^(function [^ ]+\([^)]*\))/); + var toStr = namedFunctionMatch ? namedFunctionMatch[1] : fnStr; + var fnName = fn['name'] || ""; + if (fnName && toStr.match(/function \(/)) { + return 'function ' + fnName + toStr.substr(9); + } + return toStr; +} +function fnToString(fn) { + var _fn = isArray(fn) ? fn.slice(-1)[0] : fn; + return _fn && _fn.toString() || "undefined"; +} +var stringifyPatternFn = null; +var stringifyPattern = function (value) { + var isRejection = Rejection.isRejectionPromise; + stringifyPatternFn = stringifyPatternFn || pattern([ + [not(isDefined), val("undefined")], + [isNull, val("null")], + [isPromise, val("[Promise]")], + [isRejection, function (x) { return x._transitionRejection.toString(); }], + [is(Rejection), invoke("toString")], + [is(Transition), invoke("toString")], + [is(Resolvable), invoke("toString")], + [isInjectable, functionToString], + [val(true), identity] + ]); + return stringifyPatternFn(value); +}; +function stringify(o) { + var seen = []; + function format(val$$1) { + if (isObject(val$$1)) { + if (seen.indexOf(val$$1) !== -1) + return '[circular ref]'; + seen.push(val$$1); + } + return stringifyPattern(val$$1); + } + return JSON.stringify(o, function (key, val$$1) { return format(val$$1); }).replace(/\\"/g, '"'); +} +/** Returns a function that splits a string on a character or substring */ +var beforeAfterSubstr = function (char) { return function (str) { + if (!str) + return ["", ""]; + var idx = str.indexOf(char); + if (idx === -1) + return [str, ""]; + return [str.substr(0, idx), str.substr(idx + 1)]; +}; }; +var hostRegex = new RegExp('^(?:[a-z]+:)?//[^/]+/'); +var stripFile = function (str) { return str.replace(/\/[^/]*$/, ''); }; +var splitHash = beforeAfterSubstr("#"); +var splitQuery = beforeAfterSubstr("?"); +var splitEqual = beforeAfterSubstr("="); +var trimHashVal = function (str) { return str ? str.replace(/^#/, "") : ""; }; +/** + * Splits on a delimiter, but returns the delimiters in the array + * + * #### Example: + * ```js + * var splitOnSlashes = splitOnDelim('/'); + * splitOnSlashes("/foo"); // ["/", "foo"] + * splitOnSlashes("/foo/"); // ["/", "foo", "/"] + * ``` + */ +function splitOnDelim(delim) { + var re = new RegExp("(" + delim + ")", "g"); + return function (str) { + return str.split(re).filter(identity); + }; +} + +/** + * Reduce fn that joins neighboring strings + * + * Given an array of strings, returns a new array + * where all neighboring strings have been joined. + * + * #### Example: + * ```js + * let arr = ["foo", "bar", 1, "baz", "", "qux" ]; + * arr.reduce(joinNeighborsR, []) // ["foobar", 1, "bazqux" ] + * ``` + */ +function joinNeighborsR(acc, x) { + if (isString(tail(acc)) && isString(x)) + return acc.slice(0, -1).concat(tail(acc) + x); + return pushR(acc, x); +} + +/** @module common */ /** for typedoc */ + +/** + * @coreapi + * @module params + */ +/** */ +/** + * A registry for parameter types. + * + * This registry manages the built-in (and custom) parameter types. + * + * The built-in parameter types are: + * + * - [[string]] + * - [[path]] + * - [[query]] + * - [[hash]] + * - [[int]] + * - [[bool]] + * - [[date]] + * - [[json]] + * - [[any]] + */ +var ParamTypes = /** @class */ (function () { + /** @internalapi */ + function ParamTypes() { + /** @hidden */ + this.enqueue = true; + /** @hidden */ + this.typeQueue = []; + /** @internalapi */ + this.defaultTypes = pick(ParamTypes.prototype, ["hash", "string", "query", "path", "int", "bool", "date", "json", "any"]); + // Register default types. Store them in the prototype of this.types. + var makeType = function (definition, name) { + return new ParamType(extend({ name: name }, definition)); + }; + this.types = inherit(map(this.defaultTypes, makeType), {}); + } + /** @internalapi */ + ParamTypes.prototype.dispose = function () { + this.types = {}; + }; + /** + * Registers a parameter type + * + * End users should call [[UrlMatcherFactory.type]], which delegates to this method. + */ + ParamTypes.prototype.type = function (name, definition, definitionFn) { + if (!isDefined(definition)) + return this.types[name]; + if (this.types.hasOwnProperty(name)) + throw new Error("A type named '" + name + "' has already been defined."); + this.types[name] = new ParamType(extend({ name: name }, definition)); + if (definitionFn) { + this.typeQueue.push({ name: name, def: definitionFn }); + if (!this.enqueue) + this._flushTypeQueue(); + } + return this; + }; + /** @internalapi */ + ParamTypes.prototype._flushTypeQueue = function () { + while (this.typeQueue.length) { + var type = this.typeQueue.shift(); + if (type.pattern) + throw new Error("You cannot override a type's .pattern at runtime."); + extend(this.types[type.name], services.$injector.invoke(type.def)); + } + }; + return ParamTypes; +}()); +/** @hidden */ +function initDefaultTypes() { + var makeDefaultType = function (def) { + var valToString = function (val$$1) { + return val$$1 != null ? val$$1.toString() : val$$1; + }; + var defaultTypeBase = { + encode: valToString, + decode: valToString, + is: is(String), + pattern: /.*/, + equals: function (a, b) { return a == b; }, + }; + return extend({}, defaultTypeBase, def); + }; + // Default Parameter Type Definitions + extend(ParamTypes.prototype, { + string: makeDefaultType({}), + path: makeDefaultType({ + pattern: /[^/]*/, + }), + query: makeDefaultType({}), + hash: makeDefaultType({ + inherit: false, + }), + int: makeDefaultType({ + decode: function (val$$1) { return parseInt(val$$1, 10); }, + is: function (val$$1) { + return !isNullOrUndefined(val$$1) && this.decode(val$$1.toString()) === val$$1; + }, + pattern: /-?\d+/, + }), + bool: makeDefaultType({ + encode: function (val$$1) { return val$$1 && 1 || 0; }, + decode: function (val$$1) { return parseInt(val$$1, 10) !== 0; }, + is: is(Boolean), + pattern: /0|1/, + }), + date: makeDefaultType({ + encode: function (val$$1) { + return !this.is(val$$1) ? undefined : [ + val$$1.getFullYear(), + ('0' + (val$$1.getMonth() + 1)).slice(-2), + ('0' + val$$1.getDate()).slice(-2), + ].join("-"); + }, + decode: function (val$$1) { + if (this.is(val$$1)) + return val$$1; + var match = this.capture.exec(val$$1); + return match ? new Date(match[1], match[2] - 1, match[3]) : undefined; + }, + is: function (val$$1) { return val$$1 instanceof Date && !isNaN(val$$1.valueOf()); }, + equals: function (l, r) { + return ['getFullYear', 'getMonth', 'getDate'] + .reduce(function (acc, fn) { return acc && l[fn]() === r[fn](); }, true); + }, + pattern: /[0-9]{4}-(?:0[1-9]|1[0-2])-(?:0[1-9]|[1-2][0-9]|3[0-1])/, + capture: /([0-9]{4})-(0[1-9]|1[0-2])-(0[1-9]|[1-2][0-9]|3[0-1])/, + }), + json: makeDefaultType({ + encode: toJson, + decode: fromJson, + is: is(Object), + equals: equals, + pattern: /[^/]*/, + }), + // does not encode/decode + any: makeDefaultType({ + encode: identity, + decode: identity, + is: function () { return true; }, + equals: equals, + }), + }); +} +initDefaultTypes(); + +/** + * @coreapi + * @module params + */ +/** */ +/** @internalapi */ +var StateParams = /** @class */ (function () { + function StateParams(params) { + if (params === void 0) { params = {}; } + extend(this, params); + } + /** + * Merges a set of parameters with all parameters inherited between the common parents of the + * current state and a given destination state. + * + * @param {Object} newParams The set of parameters which will be composited with inherited params. + * @param {Object} $current Internal definition of object representing the current state. + * @param {Object} $to Internal definition of object representing state to transition to. + */ + StateParams.prototype.$inherit = function (newParams, $current, $to) { + var parents = ancestors($current, $to), parentParams, inherited = {}, inheritList = []; + for (var i in parents) { + if (!parents[i] || !parents[i].params) + continue; + parentParams = Object.keys(parents[i].params); + if (!parentParams.length) + continue; + for (var j in parentParams) { + if (inheritList.indexOf(parentParams[j]) >= 0) + continue; + inheritList.push(parentParams[j]); + inherited[parentParams[j]] = this[parentParams[j]]; + } + } + return extend({}, inherited, newParams); + }; + + return StateParams; +}()); + +/** @module path */ /** for typedoc */ + +/** @module resolve */ /** for typedoc */ + +/** @module state */ /** for typedoc */ +var parseUrl = function (url) { + if (!isString(url)) + return false; + var root$$1 = url.charAt(0) === '^'; + return { val: root$$1 ? url.substring(1) : url, root: root$$1 }; +}; +function nameBuilder(state) { + return state.name; +} +function selfBuilder(state) { + state.self.$$state = function () { return state; }; + return state.self; +} +function dataBuilder(state) { + if (state.parent && state.parent.data) { + state.data = state.self.data = inherit(state.parent.data, state.data); + } + return state.data; +} +var getUrlBuilder = function ($urlMatcherFactoryProvider, root$$1) { + return function urlBuilder(state) { + var stateDec = state; + // For future states, i.e., states whose name ends with `.**`, + // match anything that starts with the url prefix + if (stateDec && stateDec.url && stateDec.name && stateDec.name.match(/\.\*\*$/)) { + stateDec.url += "{remainder:any}"; // match any path (.*) + } + var parsed = parseUrl(stateDec.url), parent = state.parent; + var url = !parsed ? stateDec.url : $urlMatcherFactoryProvider.compile(parsed.val, { + params: state.params || {}, + paramMap: function (paramConfig, isSearch) { + if (stateDec.reloadOnSearch === false && isSearch) + paramConfig = extend(paramConfig || {}, { dynamic: true }); + return paramConfig; + } + }); + if (!url) + return null; + if (!$urlMatcherFactoryProvider.isMatcher(url)) + throw new Error("Invalid url '" + url + "' in state '" + state + "'"); + return (parsed && parsed.root) ? url : ((parent && parent.navigable) || root$$1()).url.append(url); + }; +}; +var getNavigableBuilder = function (isRoot) { + return function navigableBuilder(state) { + return !isRoot(state) && state.url ? state : (state.parent ? state.parent.navigable : null); + }; +}; +var getParamsBuilder = function (paramFactory) { + return function paramsBuilder(state) { + var makeConfigParam = function (config, id) { return paramFactory.fromConfig(id, null, config); }; + var urlParams = (state.url && state.url.parameters({ inherit: false })) || []; + var nonUrlParams = values(mapObj(omit(state.params || {}, urlParams.map(prop('id'))), makeConfigParam)); + return urlParams.concat(nonUrlParams).map(function (p) { return [p.id, p]; }).reduce(applyPairs, {}); + }; +}; +function pathBuilder(state) { + return state.parent ? state.parent.path.concat(state) : /*root*/ [state]; +} +function includesBuilder(state) { + var includes = state.parent ? extend({}, state.parent.includes) : {}; + includes[state.name] = true; + return includes; +} +/** + * This is a [[StateBuilder.builder]] function for the `resolve:` block on a [[StateDeclaration]]. + * + * When the [[StateBuilder]] builds a [[StateObject]] object from a raw [[StateDeclaration]], this builder + * validates the `resolve` property and converts it to a [[Resolvable]] array. + * + * resolve: input value can be: + * + * { + * // analyzed but not injected + * myFooResolve: function() { return "myFooData"; }, + * + * // function.toString() parsed, "DependencyName" dep as string (not min-safe) + * myBarResolve: function(DependencyName) { return DependencyName.fetchSomethingAsPromise() }, + * + * // Array split; "DependencyName" dep as string + * myBazResolve: [ "DependencyName", function(dep) { return dep.fetchSomethingAsPromise() }, + * + * // Array split; DependencyType dep as token (compared using ===) + * myQuxResolve: [ DependencyType, function(dep) { return dep.fetchSometingAsPromise() }, + * + * // val.$inject used as deps + * // where: + * // corgeResolve.$inject = ["DependencyName"]; + * // function corgeResolve(dep) { dep.fetchSometingAsPromise() } + * // then "DependencyName" dep as string + * myCorgeResolve: corgeResolve, + * + * // inject service by name + * // When a string is found, desugar creating a resolve that injects the named service + * myGraultResolve: "SomeService" + * } + * + * or: + * + * [ + * new Resolvable("myFooResolve", function() { return "myFooData" }), + * new Resolvable("myBarResolve", function(dep) { return dep.fetchSomethingAsPromise() }, [ "DependencyName" ]), + * { provide: "myBazResolve", useFactory: function(dep) { dep.fetchSomethingAsPromise() }, deps: [ "DependencyName" ] } + * ] + */ +function resolvablesBuilder(state) { + /** convert resolve: {} and resolvePolicy: {} objects to an array of tuples */ + var objects2Tuples = function (resolveObj, resolvePolicies) { + return Object.keys(resolveObj || {}).map(function (token) { return ({ token: token, val: resolveObj[token], deps: undefined, policy: resolvePolicies[token] }); }); + }; + /** fetch DI annotations from a function or ng1-style array */ + var annotate = function (fn) { + var $injector = services.$injector; + // ng1 doesn't have an $injector until runtime. + // If the $injector doesn't exist, use "deferred" literal as a + // marker indicating they should be annotated when runtime starts + return fn['$inject'] || ($injector && $injector.annotate(fn, $injector.strictDi)) || "deferred"; + }; + /** true if the object has both `token` and `resolveFn`, and is probably a [[ResolveLiteral]] */ + var isResolveLiteral = function (obj) { return !!(obj.token && obj.resolveFn); }; + /** true if the object looks like a provide literal, or a ng2 Provider */ + var isLikeNg2Provider = function (obj) { return !!((obj.provide || obj.token) && (obj.useValue || obj.useFactory || obj.useExisting || obj.useClass)); }; + /** true if the object looks like a tuple from obj2Tuples */ + var isTupleFromObj = function (obj) { return !!(obj && obj.val && (isString(obj.val) || isArray(obj.val) || isFunction(obj.val))); }; + /** extracts the token from a Provider or provide literal */ + var token = function (p) { return p.provide || p.token; }; + /** Given a literal resolve or provider object, returns a Resolvable */ + var literal2Resolvable = pattern([ + [prop('resolveFn'), function (p) { return new Resolvable(token(p), p.resolveFn, p.deps, p.policy); }], + [prop('useFactory'), function (p) { return new Resolvable(token(p), p.useFactory, (p.deps || p.dependencies), p.policy); }], + [prop('useClass'), function (p) { return new Resolvable(token(p), function () { return new p.useClass(); }, [], p.policy); }], + [prop('useValue'), function (p) { return new Resolvable(token(p), function () { return p.useValue; }, [], p.policy, p.useValue); }], + [prop('useExisting'), function (p) { return new Resolvable(token(p), identity, [p.useExisting], p.policy); }], + ]); + var tuple2Resolvable = pattern([ + [pipe(prop("val"), isString), function (tuple) { return new Resolvable(tuple.token, identity, [tuple.val], tuple.policy); }], + [pipe(prop("val"), isArray), function (tuple) { return new Resolvable(tuple.token, tail(tuple.val), tuple.val.slice(0, -1), tuple.policy); }], + [pipe(prop("val"), isFunction), function (tuple) { return new Resolvable(tuple.token, tuple.val, annotate(tuple.val), tuple.policy); }], + ]); + var item2Resolvable = pattern([ + [is(Resolvable), function (r) { return r; }], + [isResolveLiteral, literal2Resolvable], + [isLikeNg2Provider, literal2Resolvable], + [isTupleFromObj, tuple2Resolvable], + [val(true), function (obj) { throw new Error("Invalid resolve value: " + stringify(obj)); }] + ]); + // If resolveBlock is already an array, use it as-is. + // Otherwise, assume it's an object and convert to an Array of tuples + var decl = state.resolve; + var items = isArray(decl) ? decl : objects2Tuples(decl, state.resolvePolicy || {}); + return items.map(item2Resolvable); +} +/** + * @internalapi A internal global service + * + * StateBuilder is a factory for the internal [[StateObject]] objects. + * + * When you register a state with the [[StateRegistry]], you register a plain old javascript object which + * conforms to the [[StateDeclaration]] interface. This factory takes that object and builds the corresponding + * [[StateObject]] object, which has an API and is used internally. + * + * Custom properties or API may be added to the internal [[StateObject]] object by registering a decorator function + * using the [[builder]] method. + */ +var StateBuilder = /** @class */ (function () { + function StateBuilder(matcher, urlMatcherFactory) { + this.matcher = matcher; + var self = this; + var root$$1 = function () { return matcher.find(""); }; + var isRoot = function (state) { return state.name === ""; }; + function parentBuilder(state) { + if (isRoot(state)) + return null; + return matcher.find(self.parentName(state)) || root$$1(); + } + this.builders = { + name: [nameBuilder], + self: [selfBuilder], + parent: [parentBuilder], + data: [dataBuilder], + // Build a URLMatcher if necessary, either via a relative or absolute URL + url: [getUrlBuilder(urlMatcherFactory, root$$1)], + // Keep track of the closest ancestor state that has a URL (i.e. is navigable) + navigable: [getNavigableBuilder(isRoot)], + params: [getParamsBuilder(urlMatcherFactory.paramFactory)], + // Each framework-specific ui-router implementation should define its own `views` builder + // e.g., src/ng1/statebuilders/views.ts + views: [], + // Keep a full path from the root down to this state as this is needed for state activation. + path: [pathBuilder], + // Speed up $state.includes() as it's used a lot + includes: [includesBuilder], + resolvables: [resolvablesBuilder] + }; + } + /** + * Registers a [[BuilderFunction]] for a specific [[StateObject]] property (e.g., `parent`, `url`, or `path`). + * More than one BuilderFunction can be registered for a given property. + * + * The BuilderFunction(s) will be used to define the property on any subsequently built [[StateObject]] objects. + * + * @param name The name of the State property being registered for. + * @param fn The BuilderFunction which will be used to build the State property + * @returns a function which deregisters the BuilderFunction + */ + StateBuilder.prototype.builder = function (name, fn) { + var builders = this.builders; + var array = builders[name] || []; + // Backwards compat: if only one builder exists, return it, else return whole arary. + if (isString(name) && !isDefined(fn)) + return array.length > 1 ? array : array[0]; + if (!isString(name) || !isFunction(fn)) + return; + builders[name] = array; + builders[name].push(fn); + return function () { return builders[name].splice(builders[name].indexOf(fn, 1)) && null; }; + }; + /** + * Builds all of the properties on an essentially blank State object, returning a State object which has all its + * properties and API built. + * + * @param state an uninitialized State object + * @returns the built State object + */ + StateBuilder.prototype.build = function (state) { + var _a = this, matcher = _a.matcher, builders = _a.builders; + var parent = this.parentName(state); + if (parent && !matcher.find(parent, undefined, false)) { + return null; + } + for (var key in builders) { + if (!builders.hasOwnProperty(key)) + continue; + var chain = builders[key].reduce(function (parentFn, step) { return function (_state) { return step(_state, parentFn); }; }, noop$1); + state[key] = chain(state); + } + return state; + }; + StateBuilder.prototype.parentName = function (state) { + // name = 'foo.bar.baz.**' + var name = state.name || ""; + // segments = ['foo', 'bar', 'baz', '.**'] + var segments = name.split('.'); + // segments = ['foo', 'bar', 'baz'] + var lastSegment = segments.pop(); + // segments = ['foo', 'bar'] (ignore .** segment for future states) + if (lastSegment === '**') + segments.pop(); + if (segments.length) { + if (state.parent) { + throw new Error("States that specify the 'parent:' property should not have a '.' in their name (" + name + ")"); + } + // 'foo.bar' + return segments.join("."); + } + if (!state.parent) + return ""; + return isString(state.parent) ? state.parent : state.parent.name; + }; + StateBuilder.prototype.name = function (state) { + var name = state.name; + if (name.indexOf('.') !== -1 || !state.parent) + return name; + var parentName = isString(state.parent) ? state.parent : state.parent.name; + return parentName ? parentName + "." + name : name; + }; + return StateBuilder; +}()); + +/** @module state */ /** for typedoc */ +var StateMatcher = /** @class */ (function () { + function StateMatcher(_states) { + this._states = _states; + } + StateMatcher.prototype.isRelative = function (stateName) { + stateName = stateName || ""; + return stateName.indexOf(".") === 0 || stateName.indexOf("^") === 0; + }; + StateMatcher.prototype.find = function (stateOrName, base, matchGlob) { + if (matchGlob === void 0) { matchGlob = true; } + if (!stateOrName && stateOrName !== "") + return undefined; + var isStr = isString(stateOrName); + var name = isStr ? stateOrName : stateOrName.name; + if (this.isRelative(name)) + name = this.resolvePath(name, base); + var state = this._states[name]; + if (state && (isStr || (!isStr && (state === stateOrName || state.self === stateOrName)))) { + return state; + } + else if (isStr && matchGlob) { + var _states = values(this._states); + var matches = _states.filter(function (state) { + return state.__stateObjectCache.nameGlob && + state.__stateObjectCache.nameGlob.matches(name); + }); + if (matches.length > 1) { + console.log("stateMatcher.find: Found multiple matches for " + name + " using glob: ", matches.map(function (match) { return match.name; })); + } + return matches[0]; + } + return undefined; + }; + StateMatcher.prototype.resolvePath = function (name, base) { + if (!base) + throw new Error("No reference point given for path '" + name + "'"); + var baseState = this.find(base); + var splitName = name.split("."), i = 0, pathLength = splitName.length, current = baseState; + for (; i < pathLength; i++) { + if (splitName[i] === "" && i === 0) { + current = baseState; + continue; + } + if (splitName[i] === "^") { + if (!current.parent) + throw new Error("Path '" + name + "' not valid for state '" + baseState.name + "'"); + current = current.parent; + continue; + } + break; + } + var relName = splitName.slice(i).join("."); + return current.name + (current.name && relName ? "." : "") + relName; + }; + return StateMatcher; +}()); + +/** @module state */ /** for typedoc */ +/** @internalapi */ +var StateQueueManager = /** @class */ (function () { + function StateQueueManager($registry, $urlRouter, states, builder, listeners) { + this.$registry = $registry; + this.$urlRouter = $urlRouter; + this.states = states; + this.builder = builder; + this.listeners = listeners; + this.queue = []; + this.matcher = $registry.matcher; + } + /** @internalapi */ + StateQueueManager.prototype.dispose = function () { + this.queue = []; + }; + StateQueueManager.prototype.register = function (stateDecl) { + var queue = this.queue; + var state = StateObject.create(stateDecl); + var name = state.name; + if (!isString(name)) + throw new Error("State must have a valid name"); + if (this.states.hasOwnProperty(name) || inArray(queue.map(prop('name')), name)) + throw new Error("State '" + name + "' is already defined"); + queue.push(state); + this.flush(); + return state; + }; + StateQueueManager.prototype.flush = function () { + var _this = this; + var _a = this, queue = _a.queue, states = _a.states, builder = _a.builder; + var registered = [], // states that got registered + orphans = [], // states that don't yet have a parent registered + previousQueueLength = {}; // keep track of how long the queue when an orphan was first encountered + var getState = function (name) { + return _this.states.hasOwnProperty(name) && _this.states[name]; + }; + while (queue.length > 0) { + var state = queue.shift(); + var name_1 = state.name; + var result = builder.build(state); + var orphanIdx = orphans.indexOf(state); + if (result) { + var existingState = getState(name_1); + if (existingState && existingState.name === name_1) { + throw new Error("State '" + name_1 + "' is already defined"); + } + var existingFutureState = getState(name_1 + ".**"); + if (existingFutureState) { + // Remove future state of the same name + this.$registry.deregister(existingFutureState); + } + states[name_1] = state; + this.attachRoute(state); + if (orphanIdx >= 0) + orphans.splice(orphanIdx, 1); + registered.push(state); + continue; + } + var prev = previousQueueLength[name_1]; + previousQueueLength[name_1] = queue.length; + if (orphanIdx >= 0 && prev === queue.length) { + // Wait until two consecutive iterations where no additional states were dequeued successfully. + // throw new Error(`Cannot register orphaned state '${name}'`); + queue.push(state); + return states; + } + else if (orphanIdx < 0) { + orphans.push(state); + } + queue.push(state); + } + if (registered.length) { + this.listeners.forEach(function (listener) { return listener("registered", registered.map(function (s) { return s.self; })); }); + } + return states; + }; + StateQueueManager.prototype.attachRoute = function (state) { + if (state.abstract || !state.url) + return; + this.$urlRouter.rule(this.$urlRouter.urlRuleFactory.create(state)); + }; + return StateQueueManager; +}()); + +/** + * @coreapi + * @module state + */ /** for typedoc */ +var StateRegistry = /** @class */ (function () { + /** @internalapi */ + function StateRegistry(_router) { + this._router = _router; + this.states = {}; + this.listeners = []; + this.matcher = new StateMatcher(this.states); + this.builder = new StateBuilder(this.matcher, _router.urlMatcherFactory); + this.stateQueue = new StateQueueManager(this, _router.urlRouter, this.states, this.builder, this.listeners); + this._registerRoot(); + } + /** @internalapi */ + StateRegistry.prototype._registerRoot = function () { + var rootStateDef = { + name: '', + url: '^', + views: null, + params: { + '#': { value: null, type: 'hash', dynamic: true } + }, + abstract: true + }; + var _root = this._root = this.stateQueue.register(rootStateDef); + _root.navigable = null; + }; + /** @internalapi */ + StateRegistry.prototype.dispose = function () { + var _this = this; + this.stateQueue.dispose(); + this.listeners = []; + this.get().forEach(function (state) { return _this.get(state) && _this.deregister(state); }); + }; + /** + * Listen for a State Registry events + * + * Adds a callback that is invoked when states are registered or deregistered with the StateRegistry. + * + * #### Example: + * ```js + * let allStates = registry.get(); + * + * // Later, invoke deregisterFn() to remove the listener + * let deregisterFn = registry.onStatesChanged((event, states) => { + * switch(event) { + * case: 'registered': + * states.forEach(state => allStates.push(state)); + * break; + * case: 'deregistered': + * states.forEach(state => { + * let idx = allStates.indexOf(state); + * if (idx !== -1) allStates.splice(idx, 1); + * }); + * break; + * } + * }); + * ``` + * + * @param listener a callback function invoked when the registered states changes. + * The function receives two parameters, `event` and `state`. + * See [[StateRegistryListener]] + * @return a function that deregisters the listener + */ + StateRegistry.prototype.onStatesChanged = function (listener) { + this.listeners.push(listener); + return function deregisterListener() { + removeFrom(this.listeners)(listener); + }.bind(this); + }; + /** + * Gets the implicit root state + * + * Gets the root of the state tree. + * The root state is implicitly created by UI-Router. + * Note: this returns the internal [[StateObject]] representation, not a [[StateDeclaration]] + * + * @return the root [[StateObject]] + */ + StateRegistry.prototype.root = function () { + return this._root; + }; + /** + * Adds a state to the registry + * + * Registers a [[StateDeclaration]] or queues it for registration. + * + * Note: a state will be queued if the state's parent isn't yet registered. + * + * @param stateDefinition the definition of the state to register. + * @returns the internal [[StateObject]] object. + * If the state was successfully registered, then the object is fully built (See: [[StateBuilder]]). + * If the state was only queued, then the object is not fully built. + */ + StateRegistry.prototype.register = function (stateDefinition) { + return this.stateQueue.register(stateDefinition); + }; + /** @hidden */ + StateRegistry.prototype._deregisterTree = function (state) { + var _this = this; + var all$$1 = this.get().map(function (s) { return s.$$state(); }); + var getChildren = function (states) { + var children = all$$1.filter(function (s) { return states.indexOf(s.parent) !== -1; }); + return children.length === 0 ? children : children.concat(getChildren(children)); + }; + var children = getChildren([state]); + var deregistered = [state].concat(children).reverse(); + deregistered.forEach(function (state) { + var $ur = _this._router.urlRouter; + // Remove URL rule + $ur.rules().filter(propEq("state", state)).forEach($ur.removeRule.bind($ur)); + // Remove state from registry + delete _this.states[state.name]; + }); + return deregistered; + }; + /** + * Removes a state from the registry + * + * This removes a state from the registry. + * If the state has children, they are are also removed from the registry. + * + * @param stateOrName the state's name or object representation + * @returns {StateObject[]} a list of removed states + */ + StateRegistry.prototype.deregister = function (stateOrName) { + var _state = this.get(stateOrName); + if (!_state) + throw new Error("Can't deregister state; not found: " + stateOrName); + var deregisteredStates = this._deregisterTree(_state.$$state()); + this.listeners.forEach(function (listener) { return listener("deregistered", deregisteredStates.map(function (s) { return s.self; })); }); + return deregisteredStates; + }; + StateRegistry.prototype.get = function (stateOrName, base) { + var _this = this; + if (arguments.length === 0) + return Object.keys(this.states).map(function (name) { return _this.states[name].self; }); + var found = this.matcher.find(stateOrName, base); + return found && found.self || null; + }; + StateRegistry.prototype.decorator = function (name, func) { + return this.builder.builder(name, func); + }; + return StateRegistry; +}()); + +/** + * @coreapi + * @module url + */ +/** for typedoc */ +/** @hidden */ +function quoteRegExp(string, param) { + var surroundPattern = ['', ''], result = string.replace(/[\\\[\]\^$*+?.()|{}]/g, "\\$&"); + if (!param) + return result; + switch (param.squash) { + case false: + surroundPattern = ['(', ')' + (param.isOptional ? '?' : '')]; + break; + case true: + result = result.replace(/\/$/, ''); + surroundPattern = ['(?:\/(', ')|\/)?']; + break; + default: + surroundPattern = ["(" + param.squash + "|", ')?']; + break; + } + return result + surroundPattern[0] + param.type.pattern.source + surroundPattern[1]; +} +/** @hidden */ +var memoizeTo = function (obj, prop$$1, fn) { + return obj[prop$$1] = obj[prop$$1] || fn(); +}; +/** @hidden */ +var splitOnSlash = splitOnDelim('/'); +/** + * Matches URLs against patterns. + * + * Matches URLs against patterns and extracts named parameters from the path or the search + * part of the URL. + * + * A URL pattern consists of a path pattern, optionally followed by '?' and a list of search (query) + * parameters. Multiple search parameter names are separated by '&'. Search parameters + * do not influence whether or not a URL is matched, but their values are passed through into + * the matched parameters returned by [[UrlMatcher.exec]]. + * + * - *Path parameters* are defined using curly brace placeholders (`/somepath/{param}`) + * or colon placeholders (`/somePath/:param`). + * + * - *A parameter RegExp* may be defined for a param after a colon + * (`/somePath/{param:[a-zA-Z0-9]+}`) in a curly brace placeholder. + * The regexp must match for the url to be matched. + * Should the regexp itself contain curly braces, they must be in matched pairs or escaped with a backslash. + * + * Note: a RegExp parameter will encode its value using either [[ParamTypes.path]] or [[ParamTypes.query]]. + * + * - *Custom parameter types* may also be specified after a colon (`/somePath/{param:int}`) in curly brace parameters. + * See [[UrlMatcherFactory.type]] for more information. + * + * - *Catch-all parameters* are defined using an asterisk placeholder (`/somepath/*catchallparam`). + * A catch-all * parameter value will contain the remainder of the URL. + * + * --- + * + * Parameter names may contain only word characters (latin letters, digits, and underscore) and + * must be unique within the pattern (across both path and search parameters). + * A path parameter matches any number of characters other than '/'. For catch-all + * placeholders the path parameter matches any number of characters. + * + * Examples: + * + * * `'/hello/'` - Matches only if the path is exactly '/hello/'. There is no special treatment for + * trailing slashes, and patterns have to match the entire path, not just a prefix. + * * `'/user/:id'` - Matches '/user/bob' or '/user/1234!!!' or even '/user/' but not '/user' or + * '/user/bob/details'. The second path segment will be captured as the parameter 'id'. + * * `'/user/{id}'` - Same as the previous example, but using curly brace syntax. + * * `'/user/{id:[^/]*}'` - Same as the previous example. + * * `'/user/{id:[0-9a-fA-F]{1,8}}'` - Similar to the previous example, but only matches if the id + * parameter consists of 1 to 8 hex digits. + * * `'/files/{path:.*}'` - Matches any URL starting with '/files/' and captures the rest of the + * path into the parameter 'path'. + * * `'/files/*path'` - ditto. + * * `'/calendar/{start:date}'` - Matches "/calendar/2014-11-12" (because the pattern defined + * in the built-in `date` ParamType matches `2014-11-12`) and provides a Date object in $stateParams.start + * + */ +var UrlMatcher = /** @class */ (function () { + /** + * @param pattern The pattern to compile into a matcher. + * @param paramTypes The [[ParamTypes]] registry + * @param config A configuration object + * - `caseInsensitive` - `true` if URL matching should be case insensitive, otherwise `false`, the default value (for backward compatibility) is `false`. + * - `strict` - `false` if matching against a URL with a trailing slash should be treated as equivalent to a URL without a trailing slash, the default value is `true`. + */ + function UrlMatcher(pattern$$1, paramTypes, paramFactory, config) { + var _this = this; + this.config = config; + /** @hidden */ + this._cache = { path: [this] }; + /** @hidden */ + this._children = []; + /** @hidden */ + this._params = []; + /** @hidden */ + this._segments = []; + /** @hidden */ + this._compiled = []; + this.pattern = pattern$$1; + this.config = defaults(this.config, { + params: {}, + strict: true, + caseInsensitive: false, + paramMap: identity + }); + // Find all placeholders and create a compiled pattern, using either classic or curly syntax: + // '*' name + // ':' name + // '{' name '}' + // '{' name ':' regexp '}' + // The regular expression is somewhat complicated due to the need to allow curly braces + // inside the regular expression. The placeholder regexp breaks down as follows: + // ([:*])([\w\[\]]+) - classic placeholder ($1 / $2) (search version has - for snake-case) + // \{([\w\[\]]+)(?:\:\s*( ... ))?\} - curly brace placeholder ($3) with optional regexp/type ... ($4) (search version has - for snake-case + // (?: ... | ... | ... )+ - the regexp consists of any number of atoms, an atom being either + // [^{}\\]+ - anything other than curly braces or backslash + // \\. - a backslash escape + // \{(?:[^{}\\]+|\\.)*\} - a matched set of curly braces containing other atoms + var placeholder = /([:*])([\w\[\]]+)|\{([\w\[\]]+)(?:\:\s*((?:[^{}\\]+|\\.|\{(?:[^{}\\]+|\\.)*\})+))?\}/g, searchPlaceholder = /([:]?)([\w\[\].-]+)|\{([\w\[\].-]+)(?:\:\s*((?:[^{}\\]+|\\.|\{(?:[^{}\\]+|\\.)*\})+))?\}/g, last = 0, m, patterns = []; + var checkParamErrors = function (id) { + if (!UrlMatcher.nameValidator.test(id)) + throw new Error("Invalid parameter name '" + id + "' in pattern '" + pattern$$1 + "'"); + if (find(_this._params, propEq('id', id))) + throw new Error("Duplicate parameter name '" + id + "' in pattern '" + pattern$$1 + "'"); + }; + // Split into static segments separated by path parameter placeholders. + // The number of segments is always 1 more than the number of parameters. + var matchDetails = function (m, isSearch) { + // IE[78] returns '' for unmatched groups instead of null + var id = m[2] || m[3]; + var regexp = isSearch ? m[4] : m[4] || (m[1] === '*' ? '[\\s\\S]*' : null); + var makeRegexpType = function (regexp) { return inherit(paramTypes.type(isSearch ? "query" : "path"), { + pattern: new RegExp(regexp, _this.config.caseInsensitive ? 'i' : undefined) + }); }; + return { + id: id, + regexp: regexp, + cfg: _this.config.params[id], + segment: pattern$$1.substring(last, m.index), + type: !regexp ? null : paramTypes.type(regexp) || makeRegexpType(regexp) + }; + }; + var p, segment; + while ((m = placeholder.exec(pattern$$1))) { + p = matchDetails(m, false); + if (p.segment.indexOf('?') >= 0) + break; // we're into the search part + checkParamErrors(p.id); + this._params.push(paramFactory.fromPath(p.id, p.type, this.config.paramMap(p.cfg, false))); + this._segments.push(p.segment); + patterns.push([p.segment, tail(this._params)]); + last = placeholder.lastIndex; + } + segment = pattern$$1.substring(last); + // Find any search parameter names and remove them from the last segment + var i = segment.indexOf('?'); + if (i >= 0) { + var search = segment.substring(i); + segment = segment.substring(0, i); + if (search.length > 0) { + last = 0; + while ((m = searchPlaceholder.exec(search))) { + p = matchDetails(m, true); + checkParamErrors(p.id); + this._params.push(paramFactory.fromSearch(p.id, p.type, this.config.paramMap(p.cfg, true))); + last = placeholder.lastIndex; + // check if ?& + } + } + } + this._segments.push(segment); + this._compiled = patterns.map(function (pattern$$1) { return quoteRegExp.apply(null, pattern$$1); }).concat(quoteRegExp(segment)); + } + /** + * Creates a new concatenated UrlMatcher + * + * Builds a new UrlMatcher by appending another UrlMatcher to this one. + * + * @param url A `UrlMatcher` instance to append as a child of the current `UrlMatcher`. + */ + UrlMatcher.prototype.append = function (url) { + this._children.push(url); + url._cache = { + path: this._cache.path.concat(url), + parent: this, + pattern: null, + }; + return url; + }; + /** @hidden */ + UrlMatcher.prototype.isRoot = function () { + return this._cache.path[0] === this; + }; + /** Returns the input pattern string */ + UrlMatcher.prototype.toString = function () { + return this.pattern; + }; + /** + * Tests the specified url/path against this matcher. + * + * Tests if the given url matches this matcher's pattern, and returns an object containing the captured + * parameter values. Returns null if the path does not match. + * + * The returned object contains the values + * of any search parameters that are mentioned in the pattern, but their value may be null if + * they are not present in `search`. This means that search parameters are always treated + * as optional. + * + * #### Example: + * ```js + * new UrlMatcher('/user/{id}?q&r').exec('/user/bob', { + * x: '1', q: 'hello' + * }); + * // returns { id: 'bob', q: 'hello', r: null } + * ``` + * + * @param path The URL path to match, e.g. `$location.path()`. + * @param search URL search parameters, e.g. `$location.search()`. + * @param hash URL hash e.g. `$location.hash()`. + * @param options + * + * @returns The captured parameter values. + */ + UrlMatcher.prototype.exec = function (path, search, hash, options) { + var _this = this; + if (search === void 0) { search = {}; } + if (options === void 0) { options = {}; } + var match = memoizeTo(this._cache, 'pattern', function () { + return new RegExp([ + '^', + unnest(_this._cache.path.map(prop('_compiled'))).join(''), + _this.config.strict === false ? '\/?' : '', + '$' + ].join(''), _this.config.caseInsensitive ? 'i' : undefined); + }).exec(path); + if (!match) + return null; + //options = defaults(options, { isolate: false }); + var allParams = this.parameters(), pathParams = allParams.filter(function (param) { return !param.isSearch(); }), searchParams = allParams.filter(function (param) { return param.isSearch(); }), nPathSegments = this._cache.path.map(function (urlm) { return urlm._segments.length - 1; }).reduce(function (a, x) { return a + x; }), values$$1 = {}; + if (nPathSegments !== match.length - 1) + throw new Error("Unbalanced capture group in route '" + this.pattern + "'"); + function decodePathArray(string) { + var reverseString = function (str) { return str.split("").reverse().join(""); }; + var unquoteDashes = function (str) { return str.replace(/\\-/g, "-"); }; + var split = reverseString(string).split(/-(?!\\)/); + var allReversed = map(split, reverseString); + return map(allReversed, unquoteDashes).reverse(); + } + for (var i = 0; i < nPathSegments; i++) { + var param = pathParams[i]; + var value = match[i + 1]; + // if the param value matches a pre-replace pair, replace the value before decoding. + for (var j = 0; j < param.replace.length; j++) { + if (param.replace[j].from === value) + value = param.replace[j].to; + } + if (value && param.array === true) + value = decodePathArray(value); + if (isDefined(value)) + value = param.type.decode(value); + values$$1[param.id] = param.value(value); + } + searchParams.forEach(function (param) { + var value = search[param.id]; + for (var j = 0; j < param.replace.length; j++) { + if (param.replace[j].from === value) + value = param.replace[j].to; + } + if (isDefined(value)) + value = param.type.decode(value); + values$$1[param.id] = param.value(value); + }); + if (hash) + values$$1["#"] = hash; + return values$$1; + }; + /** + * @hidden + * Returns all the [[Param]] objects of all path and search parameters of this pattern in order of appearance. + * + * @returns {Array.} An array of [[Param]] objects. Must be treated as read-only. If the + * pattern has no parameters, an empty array is returned. + */ + UrlMatcher.prototype.parameters = function (opts) { + if (opts === void 0) { opts = {}; } + if (opts.inherit === false) + return this._params; + return unnest(this._cache.path.map(function (matcher) { return matcher._params; })); + }; + /** + * @hidden + * Returns a single parameter from this UrlMatcher by id + * + * @param id + * @param opts + * @returns {T|Param|any|boolean|UrlMatcher|null} + */ + UrlMatcher.prototype.parameter = function (id, opts) { + var _this = this; + if (opts === void 0) { opts = {}; } + var findParam = function () { + for (var _i = 0, _a = _this._params; _i < _a.length; _i++) { + var param = _a[_i]; + if (param.id === id) + return param; + } + }; + var parent = this._cache.parent; + return findParam() || (opts.inherit !== false && parent && parent.parameter(id, opts)) || null; + }; + /** + * Validates the input parameter values against this UrlMatcher + * + * Checks an object hash of parameters to validate their correctness according to the parameter + * types of this `UrlMatcher`. + * + * @param params The object hash of parameters to validate. + * @returns Returns `true` if `params` validates, otherwise `false`. + */ + UrlMatcher.prototype.validates = function (params) { + var validParamVal = function (param, val$$1) { + return !param || param.validates(val$$1); + }; + params = params || {}; + // I'm not sure why this checks only the param keys passed in, and not all the params known to the matcher + var paramSchema = this.parameters().filter(function (paramDef) { return params.hasOwnProperty(paramDef.id); }); + return paramSchema.map(function (paramDef) { return validParamVal(paramDef, params[paramDef.id]); }).reduce(allTrueR, true); + }; + /** + * Given a set of parameter values, creates a URL from this UrlMatcher. + * + * Creates a URL that matches this pattern by substituting the specified values + * for the path and search parameters. + * + * #### Example: + * ```js + * new UrlMatcher('/user/{id}?q').format({ id:'bob', q:'yes' }); + * // returns '/user/bob?q=yes' + * ``` + * + * @param values the values to substitute for the parameters in this pattern. + * @returns the formatted URL (path and optionally search part). + */ + UrlMatcher.prototype.format = function (values$$1) { + if (values$$1 === void 0) { values$$1 = {}; } + // Build the full path of UrlMatchers (including all parent UrlMatchers) + var urlMatchers = this._cache.path; + // Extract all the static segments and Params (processed as ParamDetails) + // into an ordered array + var pathSegmentsAndParams = urlMatchers.map(UrlMatcher.pathSegmentsAndParams) + .reduce(unnestR, []) + .map(function (x) { return isString(x) ? x : getDetails(x); }); + // Extract the query params into a separate array + var queryParams = urlMatchers.map(UrlMatcher.queryParams) + .reduce(unnestR, []) + .map(getDetails); + var isInvalid = function (param) { return param.isValid === false; }; + if (pathSegmentsAndParams.concat(queryParams).filter(isInvalid).length) { + return null; + } + /** + * Given a Param, applies the parameter value, then returns detailed information about it + */ + function getDetails(param) { + // Normalize to typed value + var value = param.value(values$$1[param.id]); + var isValid = param.validates(value); + var isDefaultValue = param.isDefaultValue(value); + // Check if we're in squash mode for the parameter + var squash = isDefaultValue ? param.squash : false; + // Allow the Parameter's Type to encode the value + var encoded = param.type.encode(value); + return { param: param, value: value, isValid: isValid, isDefaultValue: isDefaultValue, squash: squash, encoded: encoded }; + } + // Build up the path-portion from the list of static segments and parameters + var pathString = pathSegmentsAndParams.reduce(function (acc, x) { + // The element is a static segment (a raw string); just append it + if (isString(x)) + return acc + x; + // Otherwise, it's a ParamDetails. + var squash = x.squash, encoded = x.encoded, param = x.param; + // If squash is === true, try to remove a slash from the path + if (squash === true) + return (acc.match(/\/$/)) ? acc.slice(0, -1) : acc; + // If squash is a string, use the string for the param value + if (isString(squash)) + return acc + squash; + if (squash !== false) + return acc; // ? + if (encoded == null) + return acc; + // If this parameter value is an array, encode the value using encodeDashes + if (isArray(encoded)) + return acc + map(encoded, UrlMatcher.encodeDashes).join("-"); + // If the parameter type is "raw", then do not encodeURIComponent + if (param.raw) + return acc + encoded; + // Encode the value + return acc + encodeURIComponent(encoded); + }, ""); + // Build the query string by applying parameter values (array or regular) + // then mapping to key=value, then flattening and joining using "&" + var queryString = queryParams.map(function (paramDetails) { + var param = paramDetails.param, squash = paramDetails.squash, encoded = paramDetails.encoded, isDefaultValue = paramDetails.isDefaultValue; + if (encoded == null || (isDefaultValue && squash !== false)) + return; + if (!isArray(encoded)) + encoded = [encoded]; + if (encoded.length === 0) + return; + if (!param.raw) + encoded = map(encoded, encodeURIComponent); + return encoded.map(function (val$$1) { return param.id + "=" + val$$1; }); + }).filter(identity).reduce(unnestR, []).join("&"); + // Concat the pathstring with the queryString (if exists) and the hashString (if exists) + return pathString + (queryString ? "?" + queryString : "") + (values$$1["#"] ? "#" + values$$1["#"] : ""); + }; + /** @hidden */ + UrlMatcher.encodeDashes = function (str) { + return encodeURIComponent(str).replace(/-/g, function (c) { return "%5C%" + c.charCodeAt(0).toString(16).toUpperCase(); }); + }; + /** @hidden Given a matcher, return an array with the matcher's path segments and path params, in order */ + UrlMatcher.pathSegmentsAndParams = function (matcher) { + var staticSegments = matcher._segments; + var pathParams = matcher._params.filter(function (p) { return p.location === exports.DefType.PATH; }); + return arrayTuples(staticSegments, pathParams.concat(undefined)) + .reduce(unnestR, []) + .filter(function (x) { return x !== "" && isDefined(x); }); + }; + /** @hidden Given a matcher, return an array with the matcher's query params */ + UrlMatcher.queryParams = function (matcher) { + return matcher._params.filter(function (p) { return p.location === exports.DefType.SEARCH; }); + }; + /** + * Compare two UrlMatchers + * + * This comparison function converts a UrlMatcher into static and dynamic path segments. + * Each static path segment is a static string between a path separator (slash character). + * Each dynamic segment is a path parameter. + * + * The comparison function sorts static segments before dynamic ones. + */ + UrlMatcher.compare = function (a, b) { + /** + * Turn a UrlMatcher and all its parent matchers into an array + * of slash literals '/', string literals, and Param objects + * + * This example matcher matches strings like "/foo/:param/tail": + * var matcher = $umf.compile("/foo").append($umf.compile("/:param")).append($umf.compile("/")).append($umf.compile("tail")); + * var result = segments(matcher); // [ '/', 'foo', '/', Param, '/', 'tail' ] + * + * Caches the result as `matcher._cache.segments` + */ + var segments = function (matcher) { + return matcher._cache.segments = matcher._cache.segments || + matcher._cache.path.map(UrlMatcher.pathSegmentsAndParams) + .reduce(unnestR, []) + .reduce(joinNeighborsR, []) + .map(function (x) { return isString(x) ? splitOnSlash(x) : x; }) + .reduce(unnestR, []); + }; + /** + * Gets the sort weight for each segment of a UrlMatcher + * + * Caches the result as `matcher._cache.weights` + */ + var weights = function (matcher) { + return matcher._cache.weights = matcher._cache.weights || + segments(matcher).map(function (segment) { + // Sort slashes first, then static strings, the Params + if (segment === '/') + return 1; + if (isString(segment)) + return 2; + if (segment instanceof Param) + return 3; + }); + }; + /** + * Pads shorter array in-place (mutates) + */ + var padArrays = function (l, r, padVal) { + var len = Math.max(l.length, r.length); + while (l.length < len) + l.push(padVal); + while (r.length < len) + r.push(padVal); + }; + var weightsA = weights(a), weightsB = weights(b); + padArrays(weightsA, weightsB, 0); + var cmp, i, pairs$$1 = arrayTuples(weightsA, weightsB); + for (i = 0; i < pairs$$1.length; i++) { + cmp = pairs$$1[i][0] - pairs$$1[i][1]; + if (cmp !== 0) + return cmp; + } + return 0; + }; + /** @hidden */ + UrlMatcher.nameValidator = /^\w+([-.]+\w+)*(?:\[\])?$/; + return UrlMatcher; +}()); + +/** + * @internalapi + * @module url + */ /** for typedoc */ +/** + * Factory for [[UrlMatcher]] instances. + * + * The factory is available to ng1 services as + * `$urlMatcherFactory` or ng1 providers as `$urlMatcherFactoryProvider`. + */ +var UrlMatcherFactory = /** @class */ (function () { + function UrlMatcherFactory() { + var _this = this; + /** @hidden */ this.paramTypes = new ParamTypes(); + /** @hidden */ this._isCaseInsensitive = false; + /** @hidden */ this._isStrictMode = true; + /** @hidden */ this._defaultSquashPolicy = false; + /** @hidden */ + this._getConfig = function (config) { + return extend({ strict: _this._isStrictMode, caseInsensitive: _this._isCaseInsensitive }, config); + }; + /** @internalapi Creates a new [[Param]] for a given location (DefType) */ + this.paramFactory = { + /** Creates a new [[Param]] from a CONFIG block */ + fromConfig: function (id, type, config) { + return new Param(id, type, config, exports.DefType.CONFIG, _this); + }, + /** Creates a new [[Param]] from a url PATH */ + fromPath: function (id, type, config) { + return new Param(id, type, config, exports.DefType.PATH, _this); + }, + /** Creates a new [[Param]] from a url SEARCH */ + fromSearch: function (id, type, config) { + return new Param(id, type, config, exports.DefType.SEARCH, _this); + }, + }; + extend(this, { UrlMatcher: UrlMatcher, Param: Param }); + } + /** @inheritdoc */ + UrlMatcherFactory.prototype.caseInsensitive = function (value) { + return this._isCaseInsensitive = isDefined(value) ? value : this._isCaseInsensitive; + }; + /** @inheritdoc */ + UrlMatcherFactory.prototype.strictMode = function (value) { + return this._isStrictMode = isDefined(value) ? value : this._isStrictMode; + }; + /** @inheritdoc */ + UrlMatcherFactory.prototype.defaultSquashPolicy = function (value) { + if (isDefined(value) && value !== true && value !== false && !isString(value)) + throw new Error("Invalid squash policy: " + value + ". Valid policies: false, true, arbitrary-string"); + return this._defaultSquashPolicy = isDefined(value) ? value : this._defaultSquashPolicy; + }; + /** + * Creates a [[UrlMatcher]] for the specified pattern. + * + * @param pattern The URL pattern. + * @param config The config object hash. + * @returns The UrlMatcher. + */ + UrlMatcherFactory.prototype.compile = function (pattern, config) { + return new UrlMatcher(pattern, this.paramTypes, this.paramFactory, this._getConfig(config)); + }; + /** + * Returns true if the specified object is a [[UrlMatcher]], or false otherwise. + * + * @param object The object to perform the type check against. + * @returns `true` if the object matches the `UrlMatcher` interface, by + * implementing all the same methods. + */ + UrlMatcherFactory.prototype.isMatcher = function (object) { + // TODO: typeof? + if (!isObject(object)) + return false; + var result = true; + forEach(UrlMatcher.prototype, function (val, name) { + if (isFunction(val)) + result = result && (isDefined(object[name]) && isFunction(object[name])); + }); + return result; + }; + + /** + * Creates and registers a custom [[ParamType]] object + * + * A [[ParamType]] can be used to generate URLs with typed parameters. + * + * @param name The type name. + * @param definition The type definition. See [[ParamTypeDefinition]] for information on the values accepted. + * @param definitionFn A function that is injected before the app runtime starts. + * The result of this function should be a [[ParamTypeDefinition]]. + * The result is merged into the existing `definition`. + * See [[ParamType]] for information on the values accepted. + * + * @returns - if a type was registered: the [[UrlMatcherFactory]] + * - if only the `name` parameter was specified: the currently registered [[ParamType]] object, or undefined + * + * Note: Register custom types *before using them* in a state definition. + * + * See [[ParamTypeDefinition]] for examples + */ + UrlMatcherFactory.prototype.type = function (name, definition, definitionFn) { + var type = this.paramTypes.type(name, definition, definitionFn); + return !isDefined(definition) ? type : this; + }; + + /** @hidden */ + UrlMatcherFactory.prototype.$get = function () { + this.paramTypes.enqueue = false; + this.paramTypes._flushTypeQueue(); + return this; + }; + + /** @internalapi */ + UrlMatcherFactory.prototype.dispose = function () { + this.paramTypes.dispose(); + }; + return UrlMatcherFactory; +}()); + +/** + * @coreapi + * @module url + */ /** */ +/** + * Creates a [[UrlRule]] + * + * Creates a [[UrlRule]] from a: + * + * - `string` + * - [[UrlMatcher]] + * - `RegExp` + * - [[StateObject]] + * @internalapi + */ +var UrlRuleFactory = /** @class */ (function () { + function UrlRuleFactory(router) { + this.router = router; + } + UrlRuleFactory.prototype.compile = function (str) { + return this.router.urlMatcherFactory.compile(str); + }; + UrlRuleFactory.prototype.create = function (what, handler) { + var _this = this; + var makeRule = pattern([ + [isString, function (_what) { return makeRule(_this.compile(_what)); }], + [is(UrlMatcher), function (_what) { return _this.fromUrlMatcher(_what, handler); }], + [isState, function (_what) { return _this.fromState(_what, _this.router); }], + [is(RegExp), function (_what) { return _this.fromRegExp(_what, handler); }], + [isFunction, function (_what) { return new BaseUrlRule(_what, handler); }], + ]); + var rule = makeRule(what); + if (!rule) + throw new Error("invalid 'what' in when()"); + return rule; + }; + /** + * A UrlRule which matches based on a UrlMatcher + * + * The `handler` may be either a `string`, a [[UrlRuleHandlerFn]] or another [[UrlMatcher]] + * + * ## Handler as a function + * + * If `handler` is a function, the function is invoked with: + * + * - matched parameter values ([[RawParams]] from [[UrlMatcher.exec]]) + * - url: the current Url ([[UrlParts]]) + * - router: the router object ([[UIRouter]]) + * + * #### Example: + * ```js + * var urlMatcher = $umf.compile("/foo/:fooId/:barId"); + * var rule = factory.fromUrlMatcher(urlMatcher, match => "/home/" + match.fooId + "/" + match.barId); + * var match = rule.match('/foo/123/456'); // results in { fooId: '123', barId: '456' } + * var result = rule.handler(match); // '/home/123/456' + * ``` + * + * ## Handler as UrlMatcher + * + * If `handler` is a UrlMatcher, the handler matcher is used to create the new url. + * The `handler` UrlMatcher is formatted using the matched param from the first matcher. + * The url is replaced with the result. + * + * #### Example: + * ```js + * var urlMatcher = $umf.compile("/foo/:fooId/:barId"); + * var handler = $umf.compile("/home/:fooId/:barId"); + * var rule = factory.fromUrlMatcher(urlMatcher, handler); + * var match = rule.match('/foo/123/456'); // results in { fooId: '123', barId: '456' } + * var result = rule.handler(match); // '/home/123/456' + * ``` + */ + UrlRuleFactory.prototype.fromUrlMatcher = function (urlMatcher, handler) { + var _handler = handler; + if (isString(handler)) + handler = this.router.urlMatcherFactory.compile(handler); + if (is(UrlMatcher)(handler)) + _handler = function (match) { return handler.format(match); }; + function match(url) { + var match = urlMatcher.exec(url.path, url.search, url.hash); + return urlMatcher.validates(match) && match; + } + // Prioritize URLs, lowest to highest: + // - Some optional URL parameters, but none matched + // - No optional parameters in URL + // - Some optional parameters, some matched + // - Some optional parameters, all matched + function matchPriority(params) { + var optional = urlMatcher.parameters().filter(function (param) { return param.isOptional; }); + if (!optional.length) + return 0.000001; + var matched = optional.filter(function (param) { return params[param.id]; }); + return matched.length / optional.length; + } + var details = { urlMatcher: urlMatcher, matchPriority: matchPriority, type: "URLMATCHER" }; + return extend(new BaseUrlRule(match, _handler), details); + }; + /** + * A UrlRule which matches a state by its url + * + * #### Example: + * ```js + * var rule = factory.fromState($state.get('foo'), router); + * var match = rule.match('/foo/123/456'); // results in { fooId: '123', barId: '456' } + * var result = rule.handler(match); + * // Starts a transition to 'foo' with params: { fooId: '123', barId: '456' } + * ``` + */ + UrlRuleFactory.prototype.fromState = function (state, router) { + /** + * Handles match by transitioning to matched state + * + * First checks if the router should start a new transition. + * A new transition is not required if the current state's URL + * and the new URL are already identical + */ + var handler = function (match) { + var $state = router.stateService; + var globals = router.globals; + if ($state.href(state, match) !== $state.href(globals.current, globals.params)) { + $state.transitionTo(state, match, { inherit: true, source: "url" }); + } + }; + var details = { state: state, type: "STATE" }; + return extend(this.fromUrlMatcher(state.url, handler), details); + }; + /** + * A UrlRule which matches based on a regular expression + * + * The `handler` may be either a [[UrlRuleHandlerFn]] or a string. + * + * ## Handler as a function + * + * If `handler` is a function, the function is invoked with: + * + * - regexp match array (from `regexp`) + * - url: the current Url ([[UrlParts]]) + * - router: the router object ([[UIRouter]]) + * + * #### Example: + * ```js + * var rule = factory.fromRegExp(/^\/foo\/(bar|baz)$/, match => "/home/" + match[1]) + * var match = rule.match('/foo/bar'); // results in [ '/foo/bar', 'bar' ] + * var result = rule.handler(match); // '/home/bar' + * ``` + * + * ## Handler as string + * + * If `handler` is a string, the url is *replaced by the string* when the Rule is invoked. + * The string is first interpolated using `string.replace()` style pattern. + * + * #### Example: + * ```js + * var rule = factory.fromRegExp(/^\/foo\/(bar|baz)$/, "/home/$1") + * var match = rule.match('/foo/bar'); // results in [ '/foo/bar', 'bar' ] + * var result = rule.handler(match); // '/home/bar' + * ``` + */ + UrlRuleFactory.prototype.fromRegExp = function (regexp, handler) { + if (regexp.global || regexp.sticky) + throw new Error("Rule RegExp must not be global or sticky"); + /** + * If handler is a string, the url will be replaced by the string. + * If the string has any String.replace() style variables in it (like `$2`), + * they will be replaced by the captures from [[match]] + */ + var redirectUrlTo = function (match) { + // Interpolates matched values into $1 $2, etc using a String.replace()-style pattern + return handler.replace(/\$(\$|\d{1,2})/, function (m, what) { + return match[what === '$' ? 0 : Number(what)]; + }); + }; + var _handler = isString(handler) ? redirectUrlTo : handler; + var match = function (url) { + return regexp.exec(url.path); + }; + var details = { regexp: regexp, type: "REGEXP" }; + return extend(new BaseUrlRule(match, _handler), details); + }; + UrlRuleFactory.isUrlRule = function (obj) { + return obj && ['type', 'match', 'handler'].every(function (key) { return isDefined(obj[key]); }); + }; + return UrlRuleFactory; +}()); +/** + * A base rule which calls `match` + * + * The value from the `match` function is passed through to the `handler`. + * @internalapi + */ +var BaseUrlRule = /** @class */ (function () { + function BaseUrlRule(match, handler) { + var _this = this; + this.match = match; + this.type = "RAW"; + this.matchPriority = function (match) { return 0 - _this.$id; }; + this.handler = handler || identity; + } + return BaseUrlRule; +}()); + +/** + * @internalapi + * @module url + */ +/** for typedoc */ +/** @hidden */ +function appendBasePath(url, isHtml5, absolute, baseHref) { + if (baseHref === '/') + return url; + if (isHtml5) + return stripFile(baseHref) + url; + if (absolute) + return baseHref.slice(1) + url; + return url; +} +/** @hidden */ +var prioritySort = function (a, b) { + return (b.priority || 0) - (a.priority || 0); +}; +/** @hidden */ +var typeSort = function (a, b) { + var weights = { "STATE": 4, "URLMATCHER": 4, "REGEXP": 3, "RAW": 2, "OTHER": 1 }; + return (weights[a.type] || 0) - (weights[b.type] || 0); +}; +/** @hidden */ +var urlMatcherSort = function (a, b) { + return !a.urlMatcher || !b.urlMatcher ? 0 : UrlMatcher.compare(a.urlMatcher, b.urlMatcher); +}; +/** @hidden */ +var idSort = function (a, b) { + // Identically sorted STATE and URLMATCHER best rule will be chosen by `matchPriority` after each rule matches the URL + var useMatchPriority = { STATE: true, URLMATCHER: true }; + var equal = useMatchPriority[a.type] && useMatchPriority[b.type]; + return equal ? 0 : (a.$id || 0) - (b.$id || 0); +}; +/** + * Default rule priority sorting function. + * + * Sorts rules by: + * + * - Explicit priority (set rule priority using [[UrlRulesApi.when]]) + * - Rule type (STATE: 4, URLMATCHER: 4, REGEXP: 3, RAW: 2, OTHER: 1) + * - `UrlMatcher` specificity ([[UrlMatcher.compare]]): works for STATE and URLMATCHER types to pick the most specific rule. + * - Rule registration order (for rule types other than STATE and URLMATCHER) + * - Equally sorted State and UrlMatcher rules will each match the URL. + * Then, the *best* match is chosen based on how many parameter values were matched. + * + * @coreapi + */ +var defaultRuleSortFn; +defaultRuleSortFn = function (a, b) { + var cmp = prioritySort(a, b); + if (cmp !== 0) + return cmp; + cmp = typeSort(a, b); + if (cmp !== 0) + return cmp; + cmp = urlMatcherSort(a, b); + if (cmp !== 0) + return cmp; + return idSort(a, b); +}; +/** + * Updates URL and responds to URL changes + * + * ### Deprecation warning: + * This class is now considered to be an internal API + * Use the [[UrlService]] instead. + * For configuring URL rules, use the [[UrlRulesApi]] which can be found as [[UrlService.rules]]. + * + * This class updates the URL when the state changes. + * It also responds to changes in the URL. + */ +var UrlRouter = /** @class */ (function () { + /** @hidden */ + function UrlRouter(router) { + /** @hidden */ this._sortFn = defaultRuleSortFn; + /** @hidden */ this._rules = []; + /** @hidden */ this.interceptDeferred = false; + /** @hidden */ this._id = 0; + /** @hidden */ this._sorted = false; + this._router = router; + this.urlRuleFactory = new UrlRuleFactory(router); + createProxyFunctions(val(UrlRouter.prototype), this, val(this)); + } + /** @internalapi */ + UrlRouter.prototype.dispose = function () { + this.listen(false); + this._rules = []; + delete this._otherwiseFn; + }; + /** @inheritdoc */ + UrlRouter.prototype.sort = function (compareFn) { + this._rules = this.stableSort(this._rules, this._sortFn = compareFn || this._sortFn); + this._sorted = true; + }; + UrlRouter.prototype.ensureSorted = function () { + this._sorted || this.sort(); + }; + UrlRouter.prototype.stableSort = function (arr, compareFn) { + var arrOfWrapper = arr.map(function (elem, idx) { return ({ elem: elem, idx: idx }); }); + arrOfWrapper.sort(function (wrapperA, wrapperB) { + var cmpDiff = compareFn(wrapperA.elem, wrapperB.elem); + return cmpDiff === 0 + ? wrapperA.idx - wrapperB.idx + : cmpDiff; + }); + return arrOfWrapper.map(function (wrapper) { return wrapper.elem; }); + }; + /** + * Given a URL, check all rules and return the best [[MatchResult]] + * @param url + * @returns {MatchResult} + */ + UrlRouter.prototype.match = function (url) { + var _this = this; + this.ensureSorted(); + url = extend({ path: '', search: {}, hash: '' }, url); + var rules = this.rules(); + if (this._otherwiseFn) + rules.push(this._otherwiseFn); + // Checks a single rule. Returns { rule: rule, match: match, weight: weight } if it matched, or undefined + var checkRule = function (rule) { + var match = rule.match(url, _this._router); + return match && { match: match, rule: rule, weight: rule.matchPriority(match) }; + }; + // The rules are pre-sorted. + // - Find the first matching rule. + // - Find any other matching rule that sorted *exactly the same*, according to `.sort()`. + // - Choose the rule with the highest match weight. + var best; + for (var i = 0; i < rules.length; i++) { + // Stop when there is a 'best' rule and the next rule sorts differently than it. + if (best && this._sortFn(rules[i], best.rule) !== 0) + break; + var current = checkRule(rules[i]); + // Pick the best MatchResult + best = (!best || current && current.weight > best.weight) ? current : best; + } + return best; + }; + /** @inheritdoc */ + UrlRouter.prototype.sync = function (evt) { + if (evt && evt.defaultPrevented) + return; + var router = this._router, $url = router.urlService, $state = router.stateService; + var url = { + path: $url.path(), search: $url.search(), hash: $url.hash(), + }; + var best = this.match(url); + var applyResult = pattern([ + [isString, function (newurl) { return $url.url(newurl, true); }], + [TargetState.isDef, function (def) { return $state.go(def.state, def.params, def.options); }], + [is(TargetState), function (target) { return $state.go(target.state(), target.params(), target.options()); }], + ]); + applyResult(best && best.rule.handler(best.match, url, router)); + }; + /** @inheritdoc */ + UrlRouter.prototype.listen = function (enabled) { + var _this = this; + if (enabled === false) { + this._stopFn && this._stopFn(); + delete this._stopFn; + } + else { + return this._stopFn = this._stopFn || this._router.urlService.onChange(function (evt) { return _this.sync(evt); }); + } + }; + /** + * Internal API. + * @internalapi + */ + UrlRouter.prototype.update = function (read) { + var $url = this._router.locationService; + if (read) { + this.location = $url.path(); + return; + } + if ($url.path() === this.location) + return; + $url.url(this.location, true); + }; + /** + * Internal API. + * + * Pushes a new location to the browser history. + * + * @internalapi + * @param urlMatcher + * @param params + * @param options + */ + UrlRouter.prototype.push = function (urlMatcher, params, options) { + var replace = options && !!options.replace; + this._router.urlService.url(urlMatcher.format(params || {}), replace); + }; + /** + * Builds and returns a URL with interpolated parameters + * + * #### Example: + * ```js + * matcher = $umf.compile("/about/:person"); + * params = { person: "bob" }; + * $bob = $urlRouter.href(matcher, params); + * // $bob == "/about/bob"; + * ``` + * + * @param urlMatcher The [[UrlMatcher]] object which is used as the template of the URL to generate. + * @param params An object of parameter values to fill the matcher's required parameters. + * @param options Options object. The options are: + * + * - **`absolute`** - {boolean=false}, If true will generate an absolute url, e.g. "http://www.example.com/fullurl". + * + * @returns Returns the fully compiled URL, or `null` if `params` fail validation against `urlMatcher` + */ + UrlRouter.prototype.href = function (urlMatcher, params, options) { + var url = urlMatcher.format(params); + if (url == null) + return null; + options = options || { absolute: false }; + var cfg = this._router.urlService.config; + var isHtml5 = cfg.html5Mode(); + if (!isHtml5 && url !== null) { + url = "#" + cfg.hashPrefix() + url; + } + url = appendBasePath(url, isHtml5, options.absolute, cfg.baseHref()); + if (!options.absolute || !url) { + return url; + } + var slash = (!isHtml5 && url ? '/' : ''), port = cfg.port(); + port = (port === 80 || port === 443 ? '' : ':' + port); + return [cfg.protocol(), '://', cfg.host(), port, slash, url].join(''); + }; + /** + * Manually adds a URL Rule. + * + * Usually, a url rule is added using [[StateDeclaration.url]] or [[when]]. + * This api can be used directly for more control (to register a [[BaseUrlRule]], for example). + * Rules can be created using [[UrlRouter.urlRuleFactory]], or create manually as simple objects. + * + * A rule should have a `match` function which returns truthy if the rule matched. + * It should also have a `handler` function which is invoked if the rule is the best match. + * + * @return a function that deregisters the rule + */ + UrlRouter.prototype.rule = function (rule) { + var _this = this; + if (!UrlRuleFactory.isUrlRule(rule)) + throw new Error("invalid rule"); + rule.$id = this._id++; + rule.priority = rule.priority || 0; + this._rules.push(rule); + this._sorted = false; + return function () { return _this.removeRule(rule); }; + }; + /** @inheritdoc */ + UrlRouter.prototype.removeRule = function (rule) { + removeFrom(this._rules, rule); + }; + /** @inheritdoc */ + UrlRouter.prototype.rules = function () { + this.ensureSorted(); + return this._rules.slice(); + }; + /** @inheritdoc */ + UrlRouter.prototype.otherwise = function (handler) { + var handlerFn = getHandlerFn(handler); + this._otherwiseFn = this.urlRuleFactory.create(val(true), handlerFn); + this._sorted = false; + }; + + /** @inheritdoc */ + UrlRouter.prototype.initial = function (handler) { + var handlerFn = getHandlerFn(handler); + var matchFn = function (urlParts, router) { + return router.globals.transitionHistory.size() === 0 && !!/^\/?$/.exec(urlParts.path); + }; + this.rule(this.urlRuleFactory.create(matchFn, handlerFn)); + }; + + /** @inheritdoc */ + UrlRouter.prototype.when = function (matcher, handler, options) { + var rule = this.urlRuleFactory.create(matcher, handler); + if (isDefined(options && options.priority)) + rule.priority = options.priority; + this.rule(rule); + return rule; + }; + + /** @inheritdoc */ + UrlRouter.prototype.deferIntercept = function (defer) { + if (defer === undefined) + defer = true; + this.interceptDeferred = defer; + }; + + return UrlRouter; +}()); +function getHandlerFn(handler) { + if (!isFunction(handler) && !isString(handler) && !is(TargetState)(handler) && !TargetState.isDef(handler)) { + throw new Error("'handler' must be a string, function, TargetState, or have a state: 'newtarget' property"); + } + return isFunction(handler) ? handler : val(handler); +} + +/** + * @coreapi + * @module view + */ /** for typedoc */ +/** + * The View service + * + * This service pairs existing `ui-view` components (which live in the DOM) + * with view configs (from the state declaration objects: [[StateDeclaration.views]]). + * + * - After a successful Transition, the views from the newly entered states are activated via [[activateViewConfig]]. + * The views from exited states are deactivated via [[deactivateViewConfig]]. + * (See: the [[registerActivateViews]] Transition Hook) + * + * - As `ui-view` components pop in and out of existence, they register themselves using [[registerUIView]]. + * + * - When the [[sync]] function is called, the registered `ui-view`(s) ([[ActiveUIView]]) + * are configured with the matching [[ViewConfig]](s) + * + */ +var ViewService = /** @class */ (function () { + function ViewService() { + var _this = this; + this._uiViews = []; + this._viewConfigs = []; + this._viewConfigFactories = {}; + this._pluginapi = { + _rootViewContext: this._rootViewContext.bind(this), + _viewConfigFactory: this._viewConfigFactory.bind(this), + _registeredUIViews: function () { return _this._uiViews; }, + _activeViewConfigs: function () { return _this._viewConfigs; }, + }; + } + ViewService.prototype._rootViewContext = function (context) { + return this._rootContext = context || this._rootContext; + }; + + ViewService.prototype._viewConfigFactory = function (viewType, factory) { + this._viewConfigFactories[viewType] = factory; + }; + ViewService.prototype.createViewConfig = function (path, decl) { + var cfgFactory = this._viewConfigFactories[decl.$type]; + if (!cfgFactory) + throw new Error("ViewService: No view config factory registered for type " + decl.$type); + var cfgs = cfgFactory(path, decl); + return isArray(cfgs) ? cfgs : [cfgs]; + }; + /** + * Deactivates a ViewConfig. + * + * This function deactivates a `ViewConfig`. + * After calling [[sync]], it will un-pair from any `ui-view` with which it is currently paired. + * + * @param viewConfig The ViewConfig view to deregister. + */ + ViewService.prototype.deactivateViewConfig = function (viewConfig) { + trace.traceViewServiceEvent("<- Removing", viewConfig); + removeFrom(this._viewConfigs, viewConfig); + }; + ViewService.prototype.activateViewConfig = function (viewConfig) { + trace.traceViewServiceEvent("-> Registering", viewConfig); + this._viewConfigs.push(viewConfig); + }; + ViewService.prototype.sync = function () { + var _this = this; + var uiViewsByFqn = this._uiViews.map(function (uiv) { return [uiv.fqn, uiv]; }).reduce(applyPairs, {}); + // Return a weighted depth value for a uiView. + // The depth is the nesting depth of ui-views (based on FQN; times 10,000) + // plus the depth of the state that is populating the uiView + function uiViewDepth(uiView) { + var stateDepth = function (context) { + return context && context.parent ? stateDepth(context.parent) + 1 : 1; + }; + return (uiView.fqn.split(".").length * 10000) + stateDepth(uiView.creationContext); + } + // Return the ViewConfig's context's depth in the context tree. + function viewConfigDepth(config) { + var context = config.viewDecl.$context, count = 0; + while (++count && context.parent) + context = context.parent; + return count; + } + // Given a depth function, returns a compare function which can return either ascending or descending order + var depthCompare = curry(function (depthFn, posNeg, left, right) { return posNeg * (depthFn(left) - depthFn(right)); }); + var matchingConfigPair = function (uiView) { + var matchingConfigs = _this._viewConfigs.filter(ViewService.matches(uiViewsByFqn, uiView)); + if (matchingConfigs.length > 1) { + // This is OK. Child states can target a ui-view that the parent state also targets (the child wins) + // Sort by depth and return the match from the deepest child + // console.log(`Multiple matching view configs for ${uiView.fqn}`, matchingConfigs); + matchingConfigs.sort(depthCompare(viewConfigDepth, -1)); // descending + } + return [uiView, matchingConfigs[0]]; + }; + var configureUIView = function (_a) { + var uiView = _a[0], viewConfig = _a[1]; + // If a parent ui-view is reconfigured, it could destroy child ui-views. + // Before configuring a child ui-view, make sure it's still in the active uiViews array. + if (_this._uiViews.indexOf(uiView) !== -1) + uiView.configUpdated(viewConfig); + }; + // Sort views by FQN and state depth. Process uiviews nearest the root first. + var pairs$$1 = this._uiViews.sort(depthCompare(uiViewDepth, 1)).map(matchingConfigPair); + trace.traceViewSync(pairs$$1); + pairs$$1.forEach(configureUIView); + }; + + /** + * Registers a `ui-view` component + * + * When a `ui-view` component is created, it uses this method to register itself. + * After registration the [[sync]] method is used to ensure all `ui-view` are configured with the proper [[ViewConfig]]. + * + * Note: the `ui-view` component uses the `ViewConfig` to determine what view should be loaded inside the `ui-view`, + * and what the view's state context is. + * + * Note: There is no corresponding `deregisterUIView`. + * A `ui-view` should hang on to the return value of `registerUIView` and invoke it to deregister itself. + * + * @param uiView The metadata for a UIView + * @return a de-registration function used when the view is destroyed. + */ + ViewService.prototype.registerUIView = function (uiView) { + trace.traceViewServiceUIViewEvent("-> Registering", uiView); + var uiViews = this._uiViews; + var fqnAndTypeMatches = function (uiv) { return uiv.fqn === uiView.fqn && uiv.$type === uiView.$type; }; + if (uiViews.filter(fqnAndTypeMatches).length) + trace.traceViewServiceUIViewEvent("!!!! duplicate uiView named:", uiView); + uiViews.push(uiView); + this.sync(); + return function () { + var idx = uiViews.indexOf(uiView); + if (idx === -1) { + trace.traceViewServiceUIViewEvent("Tried removing non-registered uiView", uiView); + return; + } + trace.traceViewServiceUIViewEvent("<- Deregistering", uiView); + removeFrom(uiViews)(uiView); + }; + }; + + /** + * Returns the list of views currently available on the page, by fully-qualified name. + * + * @return {Array} Returns an array of fully-qualified view names. + */ + ViewService.prototype.available = function () { + return this._uiViews.map(prop("fqn")); + }; + /** + * Returns the list of views on the page containing loaded content. + * + * @return {Array} Returns an array of fully-qualified view names. + */ + ViewService.prototype.active = function () { + return this._uiViews.filter(prop("$config")).map(prop("name")); + }; + /** + * Normalizes a view's name from a state.views configuration block. + * + * This should be used by a framework implementation to calculate the values for + * [[_ViewDeclaration.$uiViewName]] and [[_ViewDeclaration.$uiViewContextAnchor]]. + * + * @param context the context object (state declaration) that the view belongs to + * @param rawViewName the name of the view, as declared in the [[StateDeclaration.views]] + * + * @returns the normalized uiViewName and uiViewContextAnchor that the view targets + */ + ViewService.normalizeUIViewTarget = function (context, rawViewName) { + if (rawViewName === void 0) { rawViewName = ""; } + // TODO: Validate incoming view name with a regexp to allow: + // ex: "view.name@foo.bar" , "^.^.view.name" , "view.name@^.^" , "" , + // "@" , "$default@^" , "!$default.$default" , "!foo.bar" + var viewAtContext = rawViewName.split("@"); + var uiViewName = viewAtContext[0] || "$default"; // default to unnamed view + var uiViewContextAnchor = isString(viewAtContext[1]) ? viewAtContext[1] : "^"; // default to parent context + // Handle relative view-name sugar syntax. + // Matches rawViewName "^.^.^.foo.bar" into array: ["^.^.^.foo.bar", "^.^.^", "foo.bar"], + var relativeViewNameSugar = /^(\^(?:\.\^)*)\.(.*$)/.exec(uiViewName); + if (relativeViewNameSugar) { + // Clobbers existing contextAnchor (rawViewName validation will fix this) + uiViewContextAnchor = relativeViewNameSugar[1]; // set anchor to "^.^.^" + uiViewName = relativeViewNameSugar[2]; // set view-name to "foo.bar" + } + if (uiViewName.charAt(0) === '!') { + uiViewName = uiViewName.substr(1); + uiViewContextAnchor = ""; // target absolutely from root + } + // handle parent relative targeting "^.^.^" + var relativeMatch = /^(\^(?:\.\^)*)$/; + if (relativeMatch.exec(uiViewContextAnchor)) { + var anchor = uiViewContextAnchor.split(".").reduce((function (anchor, x) { return anchor.parent; }), context); + uiViewContextAnchor = anchor.name; + } + else if (uiViewContextAnchor === '.') { + uiViewContextAnchor = context.name; + } + return { uiViewName: uiViewName, uiViewContextAnchor: uiViewContextAnchor }; + }; + /** + * Given a ui-view and a ViewConfig, determines if they "match". + * + * A ui-view has a fully qualified name (fqn) and a context object. The fqn is built from its overall location in + * the DOM, describing its nesting relationship to any parent ui-view tags it is nested inside of. + * + * A ViewConfig has a target ui-view name and a context anchor. The ui-view name can be a simple name, or + * can be a segmented ui-view path, describing a portion of a ui-view fqn. + * + * In order for a ui-view to match ViewConfig, ui-view's $type must match the ViewConfig's $type + * + * If the ViewConfig's target ui-view name is a simple name (no dots), then a ui-view matches if: + * - the ui-view's name matches the ViewConfig's target name + * - the ui-view's context matches the ViewConfig's anchor + * + * If the ViewConfig's target ui-view name is a segmented name (with dots), then a ui-view matches if: + * - There exists a parent ui-view where: + * - the parent ui-view's name matches the first segment (index 0) of the ViewConfig's target name + * - the parent ui-view's context matches the ViewConfig's anchor + * - And the remaining segments (index 1..n) of the ViewConfig's target name match the tail of the ui-view's fqn + * + * Example: + * + * DOM: + * + * + * + * + * + * + * + * + * + * uiViews: [ + * { fqn: "$default", creationContext: { name: "" } }, + * { fqn: "$default.foo", creationContext: { name: "A" } }, + * { fqn: "$default.foo.$default", creationContext: { name: "A.B" } } + * { fqn: "$default.foo.$default.bar", creationContext: { name: "A.B.C" } } + * ] + * + * These four view configs all match the ui-view with the fqn: "$default.foo.$default.bar": + * + * - ViewConfig1: { uiViewName: "bar", uiViewContextAnchor: "A.B.C" } + * - ViewConfig2: { uiViewName: "$default.bar", uiViewContextAnchor: "A.B" } + * - ViewConfig3: { uiViewName: "foo.$default.bar", uiViewContextAnchor: "A" } + * - ViewConfig4: { uiViewName: "$default.foo.$default.bar", uiViewContextAnchor: "" } + * + * Using ViewConfig3 as an example, it matches the ui-view with fqn "$default.foo.$default.bar" because: + * - The ViewConfig's segmented target name is: [ "foo", "$default", "bar" ] + * - There exists a parent ui-view (which has fqn: "$default.foo") where: + * - the parent ui-view's name "foo" matches the first segment "foo" of the ViewConfig's target name + * - the parent ui-view's context "A" matches the ViewConfig's anchor context "A" + * - And the remaining segments [ "$default", "bar" ].join("."_ of the ViewConfig's target name match + * the tail of the ui-view's fqn "default.bar" + * + * @internalapi + */ + ViewService.matches = function (uiViewsByFqn, uiView) { return function (viewConfig) { + // Don't supply an ng1 ui-view with an ng2 ViewConfig, etc + if (uiView.$type !== viewConfig.viewDecl.$type) + return false; + // Split names apart from both viewConfig and uiView into segments + var vc = viewConfig.viewDecl; + var vcSegments = vc.$uiViewName.split("."); + var uivSegments = uiView.fqn.split("."); + // Check if the tails of the segment arrays match. ex, these arrays' tails match: + // vc: ["foo", "bar"], uiv fqn: ["$default", "foo", "bar"] + if (!equals(vcSegments, uivSegments.slice(0 - vcSegments.length))) + return false; + // Now check if the fqn ending at the first segment of the viewConfig matches the context: + // ["$default", "foo"].join(".") == "$default.foo", does the ui-view $default.foo context match? + var negOffset = (1 - vcSegments.length) || undefined; + var fqnToFirstSegment = uivSegments.slice(0, negOffset).join("."); + var uiViewContext = uiViewsByFqn[fqnToFirstSegment].creationContext; + return vc.$uiViewContextAnchor === (uiViewContext && uiViewContext.name); + }; }; + return ViewService; +}()); + +/** + * @coreapi + * @module core + */ /** */ +/** + * Global router state + * + * This is where we hold the global mutable state such as current state, current + * params, current transition, etc. + */ +var UIRouterGlobals = /** @class */ (function () { + function UIRouterGlobals() { + /** + * Current parameter values + * + * The parameter values from the latest successful transition + */ + this.params = new StateParams(); + /** @internalapi */ + this.lastStartedTransitionId = -1; + /** @internalapi */ + this.transitionHistory = new Queue([], 1); + /** @internalapi */ + this.successfulTransitions = new Queue([], 1); + } + UIRouterGlobals.prototype.dispose = function () { + this.transitionHistory.clear(); + this.successfulTransitions.clear(); + this.transition = null; + }; + return UIRouterGlobals; +}()); + +/** + * @coreapi + * @module url + */ /** */ +/** @hidden */ +var makeStub = function (keys) { + return keys.reduce(function (acc, key) { return (acc[key] = notImplemented(key), acc); }, { dispose: noop$1 }); +}; +/** @hidden */ var locationServicesFns = ["url", "path", "search", "hash", "onChange"]; +/** @hidden */ var locationConfigFns = ["port", "protocol", "host", "baseHref", "html5Mode", "hashPrefix"]; +/** @hidden */ var umfFns = ["type", "caseInsensitive", "strictMode", "defaultSquashPolicy"]; +/** @hidden */ var rulesFns = ["sort", "when", "initial", "otherwise", "rules", "rule", "removeRule"]; +/** @hidden */ var syncFns = ["deferIntercept", "listen", "sync", "match"]; +/** + * API for URL management + */ +var UrlService = /** @class */ (function () { + /** @hidden */ + function UrlService(router, lateBind) { + if (lateBind === void 0) { lateBind = true; } + this.router = router; + this.rules = {}; + this.config = {}; + // proxy function calls from UrlService to the LocationService/LocationConfig + var locationServices = function () { return router.locationService; }; + createProxyFunctions(locationServices, this, locationServices, locationServicesFns, lateBind); + var locationConfig = function () { return router.locationConfig; }; + createProxyFunctions(locationConfig, this.config, locationConfig, locationConfigFns, lateBind); + var umf = function () { return router.urlMatcherFactory; }; + createProxyFunctions(umf, this.config, umf, umfFns); + var urlRouter = function () { return router.urlRouter; }; + createProxyFunctions(urlRouter, this.rules, urlRouter, rulesFns); + createProxyFunctions(urlRouter, this, urlRouter, syncFns); + } + UrlService.prototype.url = function (newurl, replace, state) { return; }; + + /** @inheritdoc */ + UrlService.prototype.path = function () { return; }; + + /** @inheritdoc */ + UrlService.prototype.search = function () { return; }; + + /** @inheritdoc */ + UrlService.prototype.hash = function () { return; }; + + /** @inheritdoc */ + UrlService.prototype.onChange = function (callback) { return; }; + + /** + * Returns the current URL parts + * + * This method returns the current URL components as a [[UrlParts]] object. + * + * @returns the current url parts + */ + UrlService.prototype.parts = function () { + return { path: this.path(), search: this.search(), hash: this.hash() }; + }; + UrlService.prototype.dispose = function () { }; + /** @inheritdoc */ + UrlService.prototype.sync = function (evt) { return; }; + /** @inheritdoc */ + UrlService.prototype.listen = function (enabled) { return; }; + + /** @inheritdoc */ + UrlService.prototype.deferIntercept = function (defer) { return; }; + /** @inheritdoc */ + UrlService.prototype.match = function (urlParts) { return; }; + /** @hidden */ + UrlService.locationServiceStub = makeStub(locationServicesFns); + /** @hidden */ + UrlService.locationConfigStub = makeStub(locationConfigFns); + return UrlService; +}()); + +/** + * @coreapi + * @module core + */ /** */ +/** @hidden */ +var _routerInstance = 0; +/** + * The master class used to instantiate an instance of UI-Router. + * + * UI-Router (for each specific framework) will create an instance of this class during bootstrap. + * This class instantiates and wires the UI-Router services together. + * + * After a new instance of the UIRouter class is created, it should be configured for your app. + * For instance, app states should be registered with the [[UIRouter.stateRegistry]]. + * + * --- + * + * Normally the framework code will bootstrap UI-Router. + * If you are bootstrapping UIRouter manually, tell it to monitor the URL by calling + * [[UrlService.listen]] then [[UrlService.sync]]. + */ +var UIRouter = /** @class */ (function () { + /** + * Creates a new `UIRouter` object + * + * @param locationService a [[LocationServices]] implementation + * @param locationConfig a [[LocationConfig]] implementation + * @internalapi + */ + function UIRouter(locationService, locationConfig) { + if (locationService === void 0) { locationService = UrlService.locationServiceStub; } + if (locationConfig === void 0) { locationConfig = UrlService.locationConfigStub; } + this.locationService = locationService; + this.locationConfig = locationConfig; + /** @hidden */ this.$id = _routerInstance++; + /** @hidden */ this._disposed = false; + /** @hidden */ this._disposables = []; + /** Provides trace information to the console */ + this.trace = trace; + /** Provides services related to ui-view synchronization */ + this.viewService = new ViewService(); + /** Provides services related to Transitions */ + this.transitionService = new TransitionService(this); + /** Global router state */ + this.globals = new UIRouterGlobals(); + /** + * Deprecated for public use. Use [[urlService]] instead. + * @deprecated Use [[urlService]] instead + */ + this.urlMatcherFactory = new UrlMatcherFactory(); + /** + * Deprecated for public use. Use [[urlService]] instead. + * @deprecated Use [[urlService]] instead + */ + this.urlRouter = new UrlRouter(this); + /** Provides a registry for states, and related registration services */ + this.stateRegistry = new StateRegistry(this); + /** Provides services related to states */ + this.stateService = new StateService(this); + /** Provides services related to the URL */ + this.urlService = new UrlService(this); + /** @hidden */ + this._plugins = {}; + this.viewService._pluginapi._rootViewContext(this.stateRegistry.root()); + this.globals.$current = this.stateRegistry.root(); + this.globals.current = this.globals.$current.self; + this.disposable(this.globals); + this.disposable(this.stateService); + this.disposable(this.stateRegistry); + this.disposable(this.transitionService); + this.disposable(this.urlRouter); + this.disposable(locationService); + this.disposable(locationConfig); + } + /** Registers an object to be notified when the router is disposed */ + UIRouter.prototype.disposable = function (disposable) { + this._disposables.push(disposable); + }; + /** + * Disposes this router instance + * + * When called, clears resources retained by the router by calling `dispose(this)` on all + * registered [[disposable]] objects. + * + * Or, if a `disposable` object is provided, calls `dispose(this)` on that object only. + * + * @param disposable (optional) the disposable to dispose + */ + UIRouter.prototype.dispose = function (disposable) { + var _this = this; + if (disposable && isFunction(disposable.dispose)) { + disposable.dispose(this); + return undefined; + } + this._disposed = true; + this._disposables.slice().forEach(function (d) { + try { + typeof d.dispose === 'function' && d.dispose(_this); + removeFrom(_this._disposables, d); + } + catch (ignored) { } + }); + }; + /** + * Adds a plugin to UI-Router + * + * This method adds a UI-Router Plugin. + * A plugin can enhance or change UI-Router behavior using any public API. + * + * #### Example: + * ```js + * import { MyCoolPlugin } from "ui-router-cool-plugin"; + * + * var plugin = router.addPlugin(MyCoolPlugin); + * ``` + * + * ### Plugin authoring + * + * A plugin is simply a class (or constructor function) which accepts a [[UIRouter]] instance and (optionally) an options object. + * + * The plugin can implement its functionality using any of the public APIs of [[UIRouter]]. + * For example, it may configure router options or add a Transition Hook. + * + * The plugin can then be published as a separate module. + * + * #### Example: + * ```js + * export class MyAuthPlugin implements UIRouterPlugin { + * constructor(router: UIRouter, options: any) { + * this.name = "MyAuthPlugin"; + * let $transitions = router.transitionService; + * let $state = router.stateService; + * + * let authCriteria = { + * to: (state) => state.data && state.data.requiresAuth + * }; + * + * function authHook(transition: Transition) { + * let authService = transition.injector().get('AuthService'); + * if (!authService.isAuthenticated()) { + * return $state.target('login'); + * } + * } + * + * $transitions.onStart(authCriteria, authHook); + * } + * } + * ``` + * + * @param plugin one of: + * - a plugin class which implements [[UIRouterPlugin]] + * - a constructor function for a [[UIRouterPlugin]] which accepts a [[UIRouter]] instance + * - a factory function which accepts a [[UIRouter]] instance and returns a [[UIRouterPlugin]] instance + * @param options options to pass to the plugin class/factory + * @returns the registered plugin instance + */ + UIRouter.prototype.plugin = function (plugin, options) { + if (options === void 0) { options = {}; } + var pluginInstance = new plugin(this, options); + if (!pluginInstance.name) + throw new Error("Required property `name` missing on plugin: " + pluginInstance); + this._disposables.push(pluginInstance); + return this._plugins[pluginInstance.name] = pluginInstance; + }; + UIRouter.prototype.getPlugin = function (pluginName) { + return pluginName ? this._plugins[pluginName] : values(this._plugins); + }; + return UIRouter; +}()); + +/** @module hooks */ /** */ +function addCoreResolvables(trans) { + trans.addResolvable({ token: UIRouter, deps: [], resolveFn: function () { return trans.router; }, data: trans.router }, ""); + trans.addResolvable({ token: Transition, deps: [], resolveFn: function () { return trans; }, data: trans }, ""); + trans.addResolvable({ token: '$transition$', deps: [], resolveFn: function () { return trans; }, data: trans }, ""); + trans.addResolvable({ token: '$stateParams', deps: [], resolveFn: function () { return trans.params(); }, data: trans.params() }, ""); + trans.entering().forEach(function (state) { + trans.addResolvable({ token: '$state$', deps: [], resolveFn: function () { return state; }, data: state }, state); + }); +} +var registerAddCoreResolvables = function (transitionService) { + return transitionService.onCreate({}, addCoreResolvables); +}; + +/** @module hooks */ /** */ +/** + * A [[TransitionHookFn]] that redirects to a different state or params + * + * Registered using `transitionService.onStart({ to: (state) => !!state.redirectTo }, redirectHook);` + * + * See [[StateDeclaration.redirectTo]] + */ +var redirectToHook = function (trans) { + var redirect = trans.to().redirectTo; + if (!redirect) + return; + var $state = trans.router.stateService; + function handleResult(result) { + if (!result) + return; + if (result instanceof TargetState) + return result; + if (isString(result)) + return $state.target(result, trans.params(), trans.options()); + if (result['state'] || result['params']) + return $state.target(result['state'] || trans.to(), result['params'] || trans.params(), trans.options()); + } + if (isFunction(redirect)) { + return services.$q.when(redirect(trans)).then(handleResult); + } + return handleResult(redirect); +}; +var registerRedirectToHook = function (transitionService) { + return transitionService.onStart({ to: function (state) { return !!state.redirectTo; } }, redirectToHook); +}; + +/** + * A factory which creates an onEnter, onExit or onRetain transition hook function + * + * The returned function invokes the (for instance) state.onEnter hook when the + * state is being entered. + * + * @hidden + */ +function makeEnterExitRetainHook(hookName) { + return function (transition, state) { + var _state = state.$$state(); + var hookFn = _state[hookName]; + return hookFn(transition, state); + }; +} +/** + * The [[TransitionStateHookFn]] for onExit + * + * When the state is being exited, the state's .onExit function is invoked. + * + * Registered using `transitionService.onExit({ exiting: (state) => !!state.onExit }, onExitHook);` + * + * See: [[IHookRegistry.onExit]] + */ +var onExitHook = makeEnterExitRetainHook('onExit'); +var registerOnExitHook = function (transitionService) { + return transitionService.onExit({ exiting: function (state) { return !!state.onExit; } }, onExitHook); +}; +/** + * The [[TransitionStateHookFn]] for onRetain + * + * When the state was already entered, and is not being exited or re-entered, the state's .onRetain function is invoked. + * + * Registered using `transitionService.onRetain({ retained: (state) => !!state.onRetain }, onRetainHook);` + * + * See: [[IHookRegistry.onRetain]] + */ +var onRetainHook = makeEnterExitRetainHook('onRetain'); +var registerOnRetainHook = function (transitionService) { + return transitionService.onRetain({ retained: function (state) { return !!state.onRetain; } }, onRetainHook); +}; +/** + * The [[TransitionStateHookFn]] for onEnter + * + * When the state is being entered, the state's .onEnter function is invoked. + * + * Registered using `transitionService.onEnter({ entering: (state) => !!state.onEnter }, onEnterHook);` + * + * See: [[IHookRegistry.onEnter]] + */ +var onEnterHook = makeEnterExitRetainHook('onEnter'); +var registerOnEnterHook = function (transitionService) { + return transitionService.onEnter({ entering: function (state) { return !!state.onEnter; } }, onEnterHook); +}; + +/** @module hooks */ +/** for typedoc */ +/** + * A [[TransitionHookFn]] which resolves all EAGER Resolvables in the To Path + * + * Registered using `transitionService.onStart({}, eagerResolvePath);` + * + * When a Transition starts, this hook resolves all the EAGER Resolvables, which the transition then waits for. + * + * See [[StateDeclaration.resolve]] + */ +var eagerResolvePath = function (trans) { + return new ResolveContext(trans.treeChanges().to) + .resolvePath("EAGER", trans) + .then(noop$1); +}; +var registerEagerResolvePath = function (transitionService) { + return transitionService.onStart({}, eagerResolvePath, { priority: 1000 }); +}; +/** + * A [[TransitionHookFn]] which resolves all LAZY Resolvables for the state (and all its ancestors) in the To Path + * + * Registered using `transitionService.onEnter({ entering: () => true }, lazyResolveState);` + * + * When a State is being entered, this hook resolves all the Resolvables for this state, which the transition then waits for. + * + * See [[StateDeclaration.resolve]] + */ +var lazyResolveState = function (trans, state) { + return new ResolveContext(trans.treeChanges().to) + .subContext(state.$$state()) + .resolvePath("LAZY", trans) + .then(noop$1); +}; +var registerLazyResolveState = function (transitionService) { + return transitionService.onEnter({ entering: val(true) }, lazyResolveState, { priority: 1000 }); +}; + +/** @module hooks */ /** for typedoc */ +/** + * A [[TransitionHookFn]] which waits for the views to load + * + * Registered using `transitionService.onStart({}, loadEnteringViews);` + * + * Allows the views to do async work in [[ViewConfig.load]] before the transition continues. + * In angular 1, this includes loading the templates. + */ +var loadEnteringViews = function (transition) { + var $q = services.$q; + var enteringViews = transition.views("entering"); + if (!enteringViews.length) + return; + return $q.all(enteringViews.map(function (view) { return $q.when(view.load()); })).then(noop$1); +}; +var registerLoadEnteringViews = function (transitionService) { + return transitionService.onFinish({}, loadEnteringViews); +}; +/** + * A [[TransitionHookFn]] which activates the new views when a transition is successful. + * + * Registered using `transitionService.onSuccess({}, activateViews);` + * + * After a transition is complete, this hook deactivates the old views from the previous state, + * and activates the new views from the destination state. + * + * See [[ViewService]] + */ +var activateViews = function (transition) { + var enteringViews = transition.views("entering"); + var exitingViews = transition.views("exiting"); + if (!enteringViews.length && !exitingViews.length) + return; + var $view = transition.router.viewService; + exitingViews.forEach(function (vc) { return $view.deactivateViewConfig(vc); }); + enteringViews.forEach(function (vc) { return $view.activateViewConfig(vc); }); + $view.sync(); +}; +var registerActivateViews = function (transitionService) { + return transitionService.onSuccess({}, activateViews); +}; + +/** + * A [[TransitionHookFn]] which updates global UI-Router state + * + * Registered using `transitionService.onBefore({}, updateGlobalState);` + * + * Before a [[Transition]] starts, updates the global value of "the current transition" ([[Globals.transition]]). + * After a successful [[Transition]], updates the global values of "the current state" + * ([[Globals.current]] and [[Globals.$current]]) and "the current param values" ([[Globals.params]]). + * + * See also the deprecated properties: + * [[StateService.transition]], [[StateService.current]], [[StateService.params]] + */ +var updateGlobalState = function (trans) { + var globals = trans.router.globals; + var transitionSuccessful = function () { + globals.successfulTransitions.enqueue(trans); + globals.$current = trans.$to(); + globals.current = globals.$current.self; + copy(trans.params(), globals.params); + }; + var clearCurrentTransition = function () { + // Do not clear globals.transition if a different transition has started in the meantime + if (globals.transition === trans) + globals.transition = null; + }; + trans.onSuccess({}, transitionSuccessful, { priority: 10000 }); + trans.promise.then(clearCurrentTransition, clearCurrentTransition); +}; +var registerUpdateGlobalState = function (transitionService) { + return transitionService.onCreate({}, updateGlobalState); +}; + +/** + * A [[TransitionHookFn]] which updates the URL after a successful transition + * + * Registered using `transitionService.onSuccess({}, updateUrl);` + */ +var updateUrl = function (transition) { + var options = transition.options(); + var $state = transition.router.stateService; + var $urlRouter = transition.router.urlRouter; + // Dont update the url in these situations: + // The transition was triggered by a URL sync (options.source === 'url') + // The user doesn't want the url to update (options.location === false) + // The destination state, and all parents have no navigable url + if (options.source !== 'url' && options.location && $state.$current.navigable) { + var urlOptions = { replace: options.location === 'replace' }; + $urlRouter.push($state.$current.navigable.url, $state.params, urlOptions); + } + $urlRouter.update(true); +}; +var registerUpdateUrl = function (transitionService) { + return transitionService.onSuccess({}, updateUrl, { priority: 9999 }); +}; + +/** + * A [[TransitionHookFn]] that performs lazy loading + * + * When entering a state "abc" which has a `lazyLoad` function defined: + * - Invoke the `lazyLoad` function (unless it is already in process) + * - Flag the hook function as "in process" + * - The function should return a promise (that resolves when lazy loading is complete) + * - Wait for the promise to settle + * - If the promise resolves to a [[LazyLoadResult]], then register those states + * - Flag the hook function as "not in process" + * - If the hook was successful + * - Remove the `lazyLoad` function from the state declaration + * - If all the hooks were successful + * - Retry the transition (by returning a TargetState) + * + * ``` + * .state('abc', { + * component: 'fooComponent', + * lazyLoad: () => System.import('./fooComponent') + * }); + * ``` + * + * See [[StateDeclaration.lazyLoad]] + */ +var lazyLoadHook = function (transition) { + var router = transition.router; + function retryTransition() { + if (transition.originalTransition().options().source !== 'url') { + // The original transition was not triggered via url sync + // The lazy state should be loaded now, so re-try the original transition + var orig = transition.targetState(); + return router.stateService.target(orig.identifier(), orig.params(), orig.options()); + } + // The original transition was triggered via url sync + // Run the URL rules and find the best match + var $url = router.urlService; + var result = $url.match($url.parts()); + var rule = result && result.rule; + // If the best match is a state, redirect the transition (instead + // of calling sync() which supersedes the current transition) + if (rule && rule.type === "STATE") { + var state = rule.state; + var params = result.match; + return router.stateService.target(state, params, transition.options()); + } + // No matching state found, so let .sync() choose the best non-state match/otherwise + router.urlService.sync(); + } + var promises = transition.entering() + .filter(function (state) { return !!state.$$state().lazyLoad; }) + .map(function (state) { return lazyLoadState(transition, state); }); + return services.$q.all(promises).then(retryTransition); +}; +var registerLazyLoadHook = function (transitionService) { + return transitionService.onBefore({ entering: function (state) { return !!state.lazyLoad; } }, lazyLoadHook); +}; +/** + * Invokes a state's lazy load function + * + * @param transition a Transition context + * @param state the state to lazy load + * @returns A promise for the lazy load result + */ +function lazyLoadState(transition, state) { + var lazyLoadFn = state.$$state().lazyLoad; + // Store/get the lazy load promise on/from the hookfn so it doesn't get re-invoked + var promise = lazyLoadFn['_promise']; + if (!promise) { + var success = function (result) { + delete state.lazyLoad; + delete state.$$state().lazyLoad; + delete lazyLoadFn['_promise']; + return result; + }; + var error = function (err) { + delete lazyLoadFn['_promise']; + return services.$q.reject(err); + }; + promise = lazyLoadFn['_promise'] = + services.$q.when(lazyLoadFn(transition, state)) + .then(updateStateRegistry) + .then(success, error); + } + /** Register any lazy loaded state definitions */ + function updateStateRegistry(result) { + if (result && Array.isArray(result.states)) { + result.states.forEach(function (state) { return transition.router.stateRegistry.register(state); }); + } + return result; + } + return promise; +} + +/** + * This class defines a type of hook, such as `onBefore` or `onEnter`. + * Plugins can define custom hook types, such as sticky states does for `onInactive`. + * + * @interalapi + */ +var TransitionEventType = /** @class */ (function () { + function TransitionEventType(name, hookPhase, hookOrder, criteriaMatchPath, reverseSort, getResultHandler, getErrorHandler, synchronous) { + if (reverseSort === void 0) { reverseSort = false; } + if (getResultHandler === void 0) { getResultHandler = TransitionHook.HANDLE_RESULT; } + if (getErrorHandler === void 0) { getErrorHandler = TransitionHook.REJECT_ERROR; } + if (synchronous === void 0) { synchronous = false; } + this.name = name; + this.hookPhase = hookPhase; + this.hookOrder = hookOrder; + this.criteriaMatchPath = criteriaMatchPath; + this.reverseSort = reverseSort; + this.getResultHandler = getResultHandler; + this.getErrorHandler = getErrorHandler; + this.synchronous = synchronous; + } + return TransitionEventType; +}()); + +/** @module hooks */ /** */ +/** + * A [[TransitionHookFn]] that skips a transition if it should be ignored + * + * This hook is invoked at the end of the onBefore phase. + * + * If the transition should be ignored (because no parameter or states changed) + * then the transition is ignored and not processed. + */ +function ignoredHook(trans) { + var ignoredReason = trans._ignoredReason(); + if (!ignoredReason) + return; + trace.traceTransitionIgnored(trans); + var pending = trans.router.globals.transition; + // The user clicked a link going back to the *current state* ('A') + // However, there is also a pending transition in flight (to 'B') + // Abort the transition to 'B' because the user now wants to be back at 'A'. + if (ignoredReason === 'SameAsCurrent' && pending) { + pending.abort(); + } + return Rejection.ignored().toPromise(); +} +var registerIgnoredTransitionHook = function (transitionService) { + return transitionService.onBefore({}, ignoredHook, { priority: -9999 }); +}; + +/** @module hooks */ /** */ +/** + * A [[TransitionHookFn]] that rejects the Transition if it is invalid + * + * This hook is invoked at the end of the onBefore phase. + * If the transition is invalid (for example, param values do not validate) + * then the transition is rejected. + */ +function invalidTransitionHook(trans) { + if (!trans.valid()) { + throw new Error(trans.error()); + } +} +var registerInvalidTransitionHook = function (transitionService) { + return transitionService.onBefore({}, invalidTransitionHook, { priority: -10000 }); +}; + +/** + * @coreapi + * @module transition + */ +/** for typedoc */ +/** + * The default [[Transition]] options. + * + * Include this object when applying custom defaults: + * let reloadOpts = { reload: true, notify: true } + * let options = defaults(theirOpts, customDefaults, defaultOptions); + */ +var defaultTransOpts = { + location: true, + relative: null, + inherit: false, + notify: true, + reload: false, + custom: {}, + current: function () { return null; }, + source: "unknown" +}; +/** + * This class provides services related to Transitions. + * + * - Most importantly, it allows global Transition Hooks to be registered. + * - It allows the default transition error handler to be set. + * - It also has a factory function for creating new [[Transition]] objects, (used internally by the [[StateService]]). + * + * At bootstrap, [[UIRouter]] creates a single instance (singleton) of this class. + */ +var TransitionService = /** @class */ (function () { + /** @hidden */ + function TransitionService(_router) { + /** @hidden */ + this._transitionCount = 0; + /** @hidden The transition hook types, such as `onEnter`, `onStart`, etc */ + this._eventTypes = []; + /** @hidden The registered transition hooks */ + this._registeredHooks = {}; + /** @hidden The paths on a criteria object */ + this._criteriaPaths = {}; + this._router = _router; + this.$view = _router.viewService; + this._deregisterHookFns = {}; + this._pluginapi = createProxyFunctions(val(this), {}, val(this), [ + '_definePathType', + '_defineEvent', + '_getPathTypes', + '_getEvents', + 'getHooks', + ]); + this._defineCorePaths(); + this._defineCoreEvents(); + this._registerCoreTransitionHooks(); + } + /** + * Registers a [[TransitionHookFn]], called *while a transition is being constructed*. + * + * Registers a transition lifecycle hook, which is invoked during transition construction. + * + * This low level hook should only be used by plugins. + * This can be a useful time for plugins to add resolves or mutate the transition as needed. + * The Sticky States plugin uses this hook to modify the treechanges. + * + * ### Lifecycle + * + * `onCreate` hooks are invoked *while a transition is being constructed*. + * + * ### Return value + * + * The hook's return value is ignored + * + * @internalapi + * @param criteria defines which Transitions the Hook should be invoked for. + * @param callback the hook function which will be invoked. + * @param options the registration options + * @returns a function which deregisters the hook. + */ + TransitionService.prototype.onCreate = function (criteria, callback, options) { return; }; + /** @inheritdoc */ + TransitionService.prototype.onBefore = function (criteria, callback, options) { return; }; + /** @inheritdoc */ + TransitionService.prototype.onStart = function (criteria, callback, options) { return; }; + /** @inheritdoc */ + TransitionService.prototype.onExit = function (criteria, callback, options) { return; }; + /** @inheritdoc */ + TransitionService.prototype.onRetain = function (criteria, callback, options) { return; }; + /** @inheritdoc */ + TransitionService.prototype.onEnter = function (criteria, callback, options) { return; }; + /** @inheritdoc */ + TransitionService.prototype.onFinish = function (criteria, callback, options) { return; }; + /** @inheritdoc */ + TransitionService.prototype.onSuccess = function (criteria, callback, options) { return; }; + /** @inheritdoc */ + TransitionService.prototype.onError = function (criteria, callback, options) { return; }; + /** + * dispose + * @internalapi + */ + TransitionService.prototype.dispose = function (router) { + values(this._registeredHooks).forEach(function (hooksArray) { return hooksArray.forEach(function (hook) { + hook._deregistered = true; + removeFrom(hooksArray, hook); + }); }); + }; + /** + * Creates a new [[Transition]] object + * + * This is a factory function for creating new Transition objects. + * It is used internally by the [[StateService]] and should generally not be called by application code. + * + * @param fromPath the path to the current state (the from state) + * @param targetState the target state (destination) + * @returns a Transition + */ + TransitionService.prototype.create = function (fromPath, targetState) { + return new Transition(fromPath, targetState, this._router); + }; + /** @hidden */ + TransitionService.prototype._defineCoreEvents = function () { + var Phase = exports.TransitionHookPhase; + var TH = TransitionHook; + var paths = this._criteriaPaths; + var NORMAL_SORT = false, REVERSE_SORT = true; + var ASYNCHRONOUS = false, SYNCHRONOUS = true; + this._defineEvent("onCreate", Phase.CREATE, 0, paths.to, NORMAL_SORT, TH.LOG_REJECTED_RESULT, TH.THROW_ERROR, SYNCHRONOUS); + this._defineEvent("onBefore", Phase.BEFORE, 0, paths.to); + this._defineEvent("onStart", Phase.RUN, 0, paths.to); + this._defineEvent("onExit", Phase.RUN, 100, paths.exiting, REVERSE_SORT); + this._defineEvent("onRetain", Phase.RUN, 200, paths.retained); + this._defineEvent("onEnter", Phase.RUN, 300, paths.entering); + this._defineEvent("onFinish", Phase.RUN, 400, paths.to); + this._defineEvent("onSuccess", Phase.SUCCESS, 0, paths.to, NORMAL_SORT, TH.LOG_REJECTED_RESULT, TH.LOG_ERROR, SYNCHRONOUS); + this._defineEvent("onError", Phase.ERROR, 0, paths.to, NORMAL_SORT, TH.LOG_REJECTED_RESULT, TH.LOG_ERROR, SYNCHRONOUS); + }; + /** @hidden */ + TransitionService.prototype._defineCorePaths = function () { + var STATE = exports.TransitionHookScope.STATE, TRANSITION = exports.TransitionHookScope.TRANSITION; + this._definePathType("to", TRANSITION); + this._definePathType("from", TRANSITION); + this._definePathType("exiting", STATE); + this._definePathType("retained", STATE); + this._definePathType("entering", STATE); + }; + /** @hidden */ + TransitionService.prototype._defineEvent = function (name, hookPhase, hookOrder, criteriaMatchPath, reverseSort, getResultHandler, getErrorHandler, synchronous) { + if (reverseSort === void 0) { reverseSort = false; } + if (getResultHandler === void 0) { getResultHandler = TransitionHook.HANDLE_RESULT; } + if (getErrorHandler === void 0) { getErrorHandler = TransitionHook.REJECT_ERROR; } + if (synchronous === void 0) { synchronous = false; } + var eventType = new TransitionEventType(name, hookPhase, hookOrder, criteriaMatchPath, reverseSort, getResultHandler, getErrorHandler, synchronous); + this._eventTypes.push(eventType); + makeEvent(this, this, eventType); + }; + + /** @hidden */ + TransitionService.prototype._getEvents = function (phase) { + var transitionHookTypes = isDefined(phase) ? + this._eventTypes.filter(function (type) { return type.hookPhase === phase; }) : + this._eventTypes.slice(); + return transitionHookTypes.sort(function (l, r) { + var cmpByPhase = l.hookPhase - r.hookPhase; + return cmpByPhase === 0 ? l.hookOrder - r.hookOrder : cmpByPhase; + }); + }; + /** + * Adds a Path to be used as a criterion against a TreeChanges path + * + * For example: the `exiting` path in [[HookMatchCriteria]] is a STATE scoped path. + * It was defined by calling `defineTreeChangesCriterion('exiting', TransitionHookScope.STATE)` + * Each state in the exiting path is checked against the criteria and returned as part of the match. + * + * Another example: the `to` path in [[HookMatchCriteria]] is a TRANSITION scoped path. + * It was defined by calling `defineTreeChangesCriterion('to', TransitionHookScope.TRANSITION)` + * Only the tail of the `to` path is checked against the criteria and returned as part of the match. + * + * @hidden + */ + TransitionService.prototype._definePathType = function (name, hookScope) { + this._criteriaPaths[name] = { name: name, scope: hookScope }; + }; + /** * @hidden */ + TransitionService.prototype._getPathTypes = function () { + return this._criteriaPaths; + }; + /** @hidden */ + TransitionService.prototype.getHooks = function (hookName) { + return this._registeredHooks[hookName]; + }; + /** @hidden */ + TransitionService.prototype._registerCoreTransitionHooks = function () { + var fns = this._deregisterHookFns; + fns.addCoreResolves = registerAddCoreResolvables(this); + fns.ignored = registerIgnoredTransitionHook(this); + fns.invalid = registerInvalidTransitionHook(this); + // Wire up redirectTo hook + fns.redirectTo = registerRedirectToHook(this); + // Wire up onExit/Retain/Enter state hooks + fns.onExit = registerOnExitHook(this); + fns.onRetain = registerOnRetainHook(this); + fns.onEnter = registerOnEnterHook(this); + // Wire up Resolve hooks + fns.eagerResolve = registerEagerResolvePath(this); + fns.lazyResolve = registerLazyResolveState(this); + // Wire up the View management hooks + fns.loadViews = registerLoadEnteringViews(this); + fns.activateViews = registerActivateViews(this); + // Updates global state after a transition + fns.updateGlobals = registerUpdateGlobalState(this); + // After globals.current is updated at priority: 10000 + fns.updateUrl = registerUpdateUrl(this); + // Lazy load state trees + fns.lazyLoad = registerLazyLoadHook(this); + }; + return TransitionService; +}()); + +/** + * @coreapi + * @module state + */ +/** */ +/** + * Provides state related service functions + * + * This class provides services related to ui-router states. + * An instance of this class is located on the global [[UIRouter]] object. + */ +var StateService = /** @class */ (function () { + /** @internalapi */ + function StateService(router) { + this.router = router; + /** @internalapi */ + this.invalidCallbacks = []; + /** @hidden */ + this._defaultErrorHandler = function $defaultErrorHandler($error$) { + if ($error$ instanceof Error && $error$.stack) { + console.error($error$); + console.error($error$.stack); + } + else if ($error$ instanceof Rejection) { + console.error($error$.toString()); + if ($error$.detail && $error$.detail.stack) + console.error($error$.detail.stack); + } + else { + console.error($error$); + } + }; + var getters = ['current', '$current', 'params', 'transition']; + var boundFns = Object.keys(StateService.prototype).filter(not(inArray(getters))); + createProxyFunctions(val(StateService.prototype), this, val(this), boundFns); + } + Object.defineProperty(StateService.prototype, "transition", { + /** + * The [[Transition]] currently in progress (or null) + * + * This is a passthrough through to [[UIRouterGlobals.transition]] + */ + get: function () { return this.router.globals.transition; }, + enumerable: true, + configurable: true + }); + Object.defineProperty(StateService.prototype, "params", { + /** + * The latest successful state parameters + * + * This is a passthrough through to [[UIRouterGlobals.params]] + */ + get: function () { return this.router.globals.params; }, + enumerable: true, + configurable: true + }); + Object.defineProperty(StateService.prototype, "current", { + /** + * The current [[StateDeclaration]] + * + * This is a passthrough through to [[UIRouterGlobals.current]] + */ + get: function () { return this.router.globals.current; }, + enumerable: true, + configurable: true + }); + Object.defineProperty(StateService.prototype, "$current", { + /** + * The current [[StateObject]] + * + * This is a passthrough through to [[UIRouterGlobals.$current]] + */ + get: function () { return this.router.globals.$current; }, + enumerable: true, + configurable: true + }); + /** @internalapi */ + StateService.prototype.dispose = function () { + this.defaultErrorHandler(noop$1); + this.invalidCallbacks = []; + }; + /** + * Handler for when [[transitionTo]] is called with an invalid state. + * + * Invokes the [[onInvalid]] callbacks, in natural order. + * Each callback's return value is checked in sequence until one of them returns an instance of TargetState. + * The results of the callbacks are wrapped in $q.when(), so the callbacks may return promises. + * + * If a callback returns an TargetState, then it is used as arguments to $state.transitionTo() and the result returned. + * + * @internalapi + */ + StateService.prototype._handleInvalidTargetState = function (fromPath, toState) { + var _this = this; + var fromState = PathUtils.makeTargetState(this.router.stateRegistry, fromPath); + var globals = this.router.globals; + var latestThing = function () { return globals.transitionHistory.peekTail(); }; + var latest = latestThing(); + var callbackQueue = new Queue(this.invalidCallbacks.slice()); + var injector = new ResolveContext(fromPath).injector(); + var checkForRedirect = function (result) { + if (!(result instanceof TargetState)) { + return; + } + var target = result; + // Recreate the TargetState, in case the state is now defined. + target = _this.target(target.identifier(), target.params(), target.options()); + if (!target.valid()) { + return Rejection.invalid(target.error()).toPromise(); + } + if (latestThing() !== latest) { + return Rejection.superseded().toPromise(); + } + return _this.transitionTo(target.identifier(), target.params(), target.options()); + }; + function invokeNextCallback() { + var nextCallback = callbackQueue.dequeue(); + if (nextCallback === undefined) + return Rejection.invalid(toState.error()).toPromise(); + var callbackResult = services.$q.when(nextCallback(toState, fromState, injector)); + return callbackResult.then(checkForRedirect).then(function (result) { return result || invokeNextCallback(); }); + } + return invokeNextCallback(); + }; + /** + * Registers an Invalid State handler + * + * Registers a [[OnInvalidCallback]] function to be invoked when [[StateService.transitionTo]] + * has been called with an invalid state reference parameter + * + * Example: + * ```js + * stateService.onInvalid(function(to, from, injector) { + * if (to.name() === 'foo') { + * let lazyLoader = injector.get('LazyLoadService'); + * return lazyLoader.load('foo') + * .then(() => stateService.target('foo')); + * } + * }); + * ``` + * + * @param {function} callback invoked when the toState is invalid + * This function receives the (invalid) toState, the fromState, and an injector. + * The function may optionally return a [[TargetState]] or a Promise for a TargetState. + * If one is returned, it is treated as a redirect. + * + * @returns a function which deregisters the callback + */ + StateService.prototype.onInvalid = function (callback) { + this.invalidCallbacks.push(callback); + return function deregisterListener() { + removeFrom(this.invalidCallbacks)(callback); + }.bind(this); + }; + /** + * Reloads the current state + * + * A method that force reloads the current state, or a partial state hierarchy. + * All resolves are re-resolved, and components reinstantiated. + * + * #### Example: + * ```js + * let app angular.module('app', ['ui.router']); + * + * app.controller('ctrl', function ($scope, $state) { + * $scope.reload = function(){ + * $state.reload(); + * } + * }); + * ``` + * + * Note: `reload()` is just an alias for: + * + * ```js + * $state.transitionTo($state.current, $state.params, { + * reload: true, inherit: false + * }); + * ``` + * + * @param reloadState A state name or a state object. + * If present, this state and all its children will be reloaded, but ancestors will not reload. + * + * #### Example: + * ```js + * //assuming app application consists of 3 states: 'contacts', 'contacts.detail', 'contacts.detail.item' + * //and current state is 'contacts.detail.item' + * let app angular.module('app', ['ui.router']); + * + * app.controller('ctrl', function ($scope, $state) { + * $scope.reload = function(){ + * //will reload 'contact.detail' and nested 'contact.detail.item' states + * $state.reload('contact.detail'); + * } + * }); + * ``` + * + * @returns A promise representing the state of the new transition. See [[StateService.go]] + */ + StateService.prototype.reload = function (reloadState) { + return this.transitionTo(this.current, this.params, { + reload: isDefined(reloadState) ? reloadState : true, + inherit: false, + notify: false, + }); + }; + + /** + * Transition to a different state and/or parameters + * + * Convenience method for transitioning to a new state. + * + * `$state.go` calls `$state.transitionTo` internally but automatically sets options to + * `{ location: true, inherit: true, relative: router.globals.$current, notify: true }`. + * This allows you to use either an absolute or relative `to` argument (because of `relative: router.globals.$current`). + * It also allows you to specify * only the parameters you'd like to update, while letting unspecified parameters + * inherit from the current parameter values (because of `inherit: true`). + * + * #### Example: + * ```js + * let app = angular.module('app', ['ui.router']); + * + * app.controller('ctrl', function ($scope, $state) { + * $scope.changeState = function () { + * $state.go('contact.detail'); + * }; + * }); + * ``` + * + * @param to Absolute state name, state object, or relative state path (relative to current state). + * + * Some examples: + * + * - `$state.go('contact.detail')` - will go to the `contact.detail` state + * - `$state.go('^')` - will go to the parent state + * - `$state.go('^.sibling')` - if current state is `home.child`, will go to the `home.sibling` state + * - `$state.go('.child.grandchild')` - if current state is home, will go to the `home.child.grandchild` state + * + * @param params A map of the parameters that will be sent to the state, will populate $stateParams. + * + * Any parameters that are not specified will be inherited from current parameter values (because of `inherit: true`). + * This allows, for example, going to a sibling state that shares parameters defined by a parent state. + * + * @param options Transition options + * + * @returns {promise} A promise representing the state of the new transition. + */ + StateService.prototype.go = function (to, params, options) { + var defautGoOpts = { relative: this.$current, inherit: true }; + var transOpts = defaults(options, defautGoOpts, defaultTransOpts); + return this.transitionTo(to, params, transOpts); + }; + + /** + * Creates a [[TargetState]] + * + * This is a factory method for creating a TargetState + * + * This may be returned from a Transition Hook to redirect a transition, for example. + */ + StateService.prototype.target = function (identifier, params, options) { + if (options === void 0) { options = {}; } + // If we're reloading, find the state object to reload from + if (isObject(options.reload) && !options.reload.name) + throw new Error('Invalid reload state object'); + var reg = this.router.stateRegistry; + options.reloadState = options.reload === true ? reg.root() : reg.matcher.find(options.reload, options.relative); + if (options.reload && !options.reloadState) + throw new Error("No such reload state '" + (isString(options.reload) ? options.reload : options.reload.name) + "'"); + return new TargetState(this.router.stateRegistry, identifier, params, options); + }; + + StateService.prototype.getCurrentPath = function () { + var _this = this; + var globals = this.router.globals; + var latestSuccess = globals.successfulTransitions.peekTail(); + var rootPath = function () { return [new PathNode(_this.router.stateRegistry.root())]; }; + return latestSuccess ? latestSuccess.treeChanges().to : rootPath(); + }; + /** + * Low-level method for transitioning to a new state. + * + * The [[go]] method (which uses `transitionTo` internally) is recommended in most situations. + * + * #### Example: + * ```js + * let app = angular.module('app', ['ui.router']); + * + * app.controller('ctrl', function ($scope, $state) { + * $scope.changeState = function () { + * $state.transitionTo('contact.detail'); + * }; + * }); + * ``` + * + * @param to State name or state object. + * @param toParams A map of the parameters that will be sent to the state, + * will populate $stateParams. + * @param options Transition options + * + * @returns A promise representing the state of the new transition. See [[go]] + */ + StateService.prototype.transitionTo = function (to, toParams, options) { + var _this = this; + if (toParams === void 0) { toParams = {}; } + if (options === void 0) { options = {}; } + var router = this.router; + var globals = router.globals; + options = defaults(options, defaultTransOpts); + var getCurrent = function () { + return globals.transition; + }; + options = extend(options, { current: getCurrent }); + var ref = this.target(to, toParams, options); + var currentPath = this.getCurrentPath(); + if (!ref.exists()) + return this._handleInvalidTargetState(currentPath, ref); + if (!ref.valid()) + return silentRejection(ref.error()); + /** + * Special handling for Ignored, Aborted, and Redirected transitions + * + * The semantics for the transition.run() promise and the StateService.transitionTo() + * promise differ. For instance, the run() promise may be rejected because it was + * IGNORED, but the transitionTo() promise is resolved because from the user perspective + * no error occurred. Likewise, the transition.run() promise may be rejected because of + * a Redirect, but the transitionTo() promise is chained to the new Transition's promise. + */ + var rejectedTransitionHandler = function (transition) { return function (error) { + if (error instanceof Rejection) { + var isLatest = router.globals.lastStartedTransitionId === transition.$id; + if (error.type === exports.RejectType.IGNORED) { + isLatest && router.urlRouter.update(); + // Consider ignored `Transition.run()` as a successful `transitionTo` + return services.$q.when(globals.current); + } + var detail = error.detail; + if (error.type === exports.RejectType.SUPERSEDED && error.redirected && detail instanceof TargetState) { + // If `Transition.run()` was redirected, allow the `transitionTo()` promise to resolve successfully + // by returning the promise for the new (redirect) `Transition.run()`. + var redirect = transition.redirect(detail); + return redirect.run().catch(rejectedTransitionHandler(redirect)); + } + if (error.type === exports.RejectType.ABORTED) { + isLatest && router.urlRouter.update(); + return services.$q.reject(error); + } + } + var errorHandler = _this.defaultErrorHandler(); + errorHandler(error); + return services.$q.reject(error); + }; }; + var transition = this.router.transitionService.create(currentPath, ref); + var transitionToPromise = transition.run().catch(rejectedTransitionHandler(transition)); + silenceUncaughtInPromise(transitionToPromise); // issue #2676 + // Return a promise for the transition, which also has the transition object on it. + return extend(transitionToPromise, { transition: transition }); + }; + + /** + * Checks if the current state *is* the provided state + * + * Similar to [[includes]] but only checks for the full state name. + * If params is supplied then it will be tested for strict equality against the current + * active params object, so all params must match with none missing and no extras. + * + * #### Example: + * ```js + * $state.$current.name = 'contacts.details.item'; + * + * // absolute name + * $state.is('contact.details.item'); // returns true + * $state.is(contactDetailItemStateObject); // returns true + * ``` + * + * // relative name (. and ^), typically from a template + * // E.g. from the 'contacts.details' template + * ```html + *
Item
+ * ``` + * + * @param stateOrName The state name (absolute or relative) or state object you'd like to check. + * @param params A param object, e.g. `{sectionId: section.id}`, that you'd like + * to test against the current active state. + * @param options An options object. The options are: + * - `relative`: If `stateOrName` is a relative state name and `options.relative` is set, .is will + * test relative to `options.relative` state (or name). + * + * @returns Returns true if it is the state. + */ + StateService.prototype.is = function (stateOrName, params, options) { + options = defaults(options, { relative: this.$current }); + var state = this.router.stateRegistry.matcher.find(stateOrName, options.relative); + if (!isDefined(state)) + return undefined; + if (this.$current !== state) + return false; + if (!params) + return true; + var schema = state.parameters({ inherit: true, matchingKeys: params }); + return Param.equals(schema, Param.values(schema, params), this.params); + }; + + /** + * Checks if the current state *includes* the provided state + * + * A method to determine if the current active state is equal to or is the child of the + * state stateName. If any params are passed then they will be tested for a match as well. + * Not all the parameters need to be passed, just the ones you'd like to test for equality. + * + * #### Example when `$state.$current.name === 'contacts.details.item'` + * ```js + * // Using partial names + * $state.includes("contacts"); // returns true + * $state.includes("contacts.details"); // returns true + * $state.includes("contacts.details.item"); // returns true + * $state.includes("contacts.list"); // returns false + * $state.includes("about"); // returns false + * ``` + * + * #### Glob Examples when `* $state.$current.name === 'contacts.details.item.url'`: + * ```js + * $state.includes("*.details.*.*"); // returns true + * $state.includes("*.details.**"); // returns true + * $state.includes("**.item.**"); // returns true + * $state.includes("*.details.item.url"); // returns true + * $state.includes("*.details.*.url"); // returns true + * $state.includes("*.details.*"); // returns false + * $state.includes("item.**"); // returns false + * ``` + * + * @param stateOrName A partial name, relative name, glob pattern, + * or state object to be searched for within the current state name. + * @param params A param object, e.g. `{sectionId: section.id}`, + * that you'd like to test against the current active state. + * @param options An options object. The options are: + * - `relative`: If `stateOrName` is a relative state name and `options.relative` is set, .is will + * test relative to `options.relative` state (or name). + * + * @returns {boolean} Returns true if it does include the state + */ + StateService.prototype.includes = function (stateOrName, params, options) { + options = defaults(options, { relative: this.$current }); + var glob = isString(stateOrName) && Glob.fromString(stateOrName); + if (glob) { + if (!glob.matches(this.$current.name)) + return false; + stateOrName = this.$current.name; + } + var state = this.router.stateRegistry.matcher.find(stateOrName, options.relative), include = this.$current.includes; + if (!isDefined(state)) + return undefined; + if (!isDefined(include[state.name])) + return false; + if (!params) + return true; + var schema = state.parameters({ inherit: true, matchingKeys: params }); + return Param.equals(schema, Param.values(schema, params), this.params); + }; + + /** + * Generates a URL for a state and parameters + * + * Returns the url for the given state populated with the given params. + * + * #### Example: + * ```js + * expect($state.href("about.person", { person: "bob" })).toEqual("/about/bob"); + * ``` + * + * @param stateOrName The state name or state object you'd like to generate a url from. + * @param params An object of parameter values to fill the state's required parameters. + * @param options Options object. The options are: + * + * @returns {string} compiled state url + */ + StateService.prototype.href = function (stateOrName, params, options) { + var defaultHrefOpts = { + lossy: true, + inherit: true, + absolute: false, + relative: this.$current, + }; + options = defaults(options, defaultHrefOpts); + params = params || {}; + var state = this.router.stateRegistry.matcher.find(stateOrName, options.relative); + if (!isDefined(state)) + return null; + if (options.inherit) + params = this.params.$inherit(params, this.$current, state); + var nav = (state && options.lossy) ? state.navigable : state; + if (!nav || nav.url === undefined || nav.url === null) { + return null; + } + return this.router.urlRouter.href(nav.url, params, { + absolute: options.absolute, + }); + }; + + /** + * Sets or gets the default [[transitionTo]] error handler. + * + * The error handler is called when a [[Transition]] is rejected or when any error occurred during the Transition. + * This includes errors caused by resolves and transition hooks. + * + * Note: + * This handler does not receive certain Transition rejections. + * Redirected and Ignored Transitions are not considered to be errors by [[StateService.transitionTo]]. + * + * The built-in default error handler logs the error to the console. + * + * You can provide your own custom handler. + * + * #### Example: + * ```js + * stateService.defaultErrorHandler(function() { + * // Do not log transitionTo errors + * }); + * ``` + * + * @param handler a global error handler function + * @returns the current global error handler + */ + StateService.prototype.defaultErrorHandler = function (handler) { + return this._defaultErrorHandler = handler || this._defaultErrorHandler; + }; + StateService.prototype.get = function (stateOrName, base) { + var reg = this.router.stateRegistry; + if (arguments.length === 0) + return reg.get(); + return reg.get(stateOrName, base || this.$current); + }; + /** + * Lazy loads a state + * + * Explicitly runs a state's [[StateDeclaration.lazyLoad]] function. + * + * @param stateOrName the state that should be lazy loaded + * @param transition the optional Transition context to use (if the lazyLoad function requires an injector, etc) + * Note: If no transition is provided, a noop transition is created using the from the current state to the current state. + * This noop transition is not actually run. + * + * @returns a promise to lazy load + */ + StateService.prototype.lazyLoad = function (stateOrName, transition) { + var state = this.get(stateOrName); + if (!state || !state.lazyLoad) + throw new Error("Can not lazy load " + stateOrName); + var currentPath = this.getCurrentPath(); + var target = PathUtils.makeTargetState(this.router.stateRegistry, currentPath); + transition = transition || this.router.transitionService.create(currentPath, target); + return lazyLoadState(transition, state); + }; + return StateService; +}()); + +/** + * # Transition subsystem + * + * This module contains APIs related to a Transition. + * + * See: + * - [[TransitionService]] + * - [[Transition]] + * - [[HookFn]], [[TransitionHookFn]], [[TransitionStateHookFn]], [[HookMatchCriteria]], [[HookResult]] + * + * @coreapi + * @preferred + * @module transition + */ /** for typedoc */ + +/** + * @internalapi + * @module vanilla + */ +/** */ +/** + * An angular1-like promise api + * + * This object implements four methods similar to the + * [angular 1 promise api](https://docs.angularjs.org/api/ng/service/$q) + * + * UI-Router evolved from an angular 1 library to a framework agnostic library. + * However, some of the `@uirouter/core` code uses these ng1 style APIs to support ng1 style dependency injection. + * + * This API provides native ES6 promise support wrapped as a $q-like API. + * Internally, UI-Router uses this $q object to perform promise operations. + * The `angular-ui-router` (ui-router for angular 1) uses the $q API provided by angular. + * + * $q-like promise api + */ +var $q = { + /** Normalizes a value as a promise */ + when: function (val) { return new Promise(function (resolve, reject) { return resolve(val); }); }, + /** Normalizes a value as a promise rejection */ + reject: function (val) { return new Promise(function (resolve, reject) { reject(val); }); }, + /** @returns a deferred object, which has `resolve` and `reject` functions */ + defer: function () { + var deferred = {}; + deferred.promise = new Promise(function (resolve, reject) { + deferred.resolve = resolve; + deferred.reject = reject; + }); + return deferred; + }, + /** Like Promise.all(), but also supports object key/promise notation like $q */ + all: function (promises) { + if (isArray(promises)) { + return Promise.all(promises); + } + if (isObject(promises)) { + // Convert promises map to promises array. + // When each promise resolves, map it to a tuple { key: key, val: val } + var chain = Object.keys(promises) + .map(function (key) { return promises[key].then(function (val) { return ({ key: key, val: val }); }); }); + // Then wait for all promises to resolve, and convert them back to an object + return $q.all(chain).then(function (values) { + return values.reduce(function (acc, tuple) { acc[tuple.key] = tuple.val; return acc; }, {}); + }); + } + } +}; + +/** + * @internalapi + * @module vanilla + */ +/** */ +// globally available injectables +var globals = {}; +var STRIP_COMMENTS = /((\/\/.*$)|(\/\*[\s\S]*?\*\/))/mg; +var ARGUMENT_NAMES = /([^\s,]+)/g; +/** + * A basic angular1-like injector api + * + * This object implements four methods similar to the + * [angular 1 dependency injector](https://docs.angularjs.org/api/auto/service/$injector) + * + * UI-Router evolved from an angular 1 library to a framework agnostic library. + * However, some of the `@uirouter/core` code uses these ng1 style APIs to support ng1 style dependency injection. + * + * This object provides a naive implementation of a globally scoped dependency injection system. + * It supports the following DI approaches: + * + * ### Function parameter names + * + * A function's `.toString()` is called, and the parameter names are parsed. + * This only works when the parameter names aren't "mangled" by a minifier such as UglifyJS. + * + * ```js + * function injectedFunction(FooService, BarService) { + * // FooService and BarService are injected + * } + * ``` + * + * ### Function annotation + * + * A function may be annotated with an array of dependency names as the `$inject` property. + * + * ```js + * injectedFunction.$inject = [ 'FooService', 'BarService' ]; + * function injectedFunction(fs, bs) { + * // FooService and BarService are injected as fs and bs parameters + * } + * ``` + * + * ### Array notation + * + * An array provides the names of the dependencies to inject (as strings). + * The function is the last element of the array. + * + * ```js + * [ 'FooService', 'BarService', function (fs, bs) { + * // FooService and BarService are injected as fs and bs parameters + * }] + * ``` + * + * @type {$InjectorLike} + */ +var $injector = { + /** Gets an object from DI based on a string token */ + get: function (name) { return globals[name]; }, + /** Returns true if an object named `name` exists in global DI */ + has: function (name) { return $injector.get(name) != null; }, + /** + * Injects a function + * + * @param fn the function to inject + * @param context the function's `this` binding + * @param locals An object with additional DI tokens and values, such as `{ someToken: { foo: 1 } }` + */ + invoke: function (fn, context, locals) { + var all = extend({}, globals, locals || {}); + var params = $injector.annotate(fn); + var ensureExist = assertPredicate(function (key) { return all.hasOwnProperty(key); }, function (key) { return "DI can't find injectable: '" + key + "'"; }); + var args = params.filter(ensureExist).map(function (x) { return all[x]; }); + if (isFunction(fn)) + return fn.apply(context, args); + else + return fn.slice(-1)[0].apply(context, args); + }, + /** + * Returns a function's dependencies + * + * Analyzes a function (or array) and returns an array of DI tokens that the function requires. + * @return an array of `string`s + */ + annotate: function (fn) { + if (!isInjectable(fn)) + throw new Error("Not an injectable function: " + fn); + if (fn && fn.$inject) + return fn.$inject; + if (isArray(fn)) + return fn.slice(0, -1); + var fnStr = fn.toString().replace(STRIP_COMMENTS, ''); + var result = fnStr.slice(fnStr.indexOf('(') + 1, fnStr.indexOf(')')).match(ARGUMENT_NAMES); + return result || []; + } +}; + +/** + * @internalapi + * @module vanilla + */ +/** */ +var keyValsToObjectR = function (accum, _a) { + var key = _a[0], val = _a[1]; + if (!accum.hasOwnProperty(key)) { + accum[key] = val; + } + else if (isArray(accum[key])) { + accum[key].push(val); + } + else { + accum[key] = [accum[key], val]; + } + return accum; +}; +var getParams = function (queryString) { + return queryString.split("&").filter(identity).map(splitEqual).reduce(keyValsToObjectR, {}); +}; +function parseUrl$1(url) { + var orEmptyString = function (x) { return x || ""; }; + var _a = splitHash(url).map(orEmptyString), beforehash = _a[0], hash = _a[1]; + var _b = splitQuery(beforehash).map(orEmptyString), path = _b[0], search = _b[1]; + return { path: path, search: search, hash: hash, url: url }; +} +var buildUrl = function (loc) { + var path = loc.path(); + var searchObject = loc.search(); + var hash = loc.hash(); + var search = Object.keys(searchObject).map(function (key) { + var param = searchObject[key]; + var vals = isArray(param) ? param : [param]; + return vals.map(function (val) { return key + "=" + val; }); + }).reduce(unnestR, []).join("&"); + return path + (search ? "?" + search : "") + (hash ? "#" + hash : ""); +}; +function locationPluginFactory(name, isHtml5, serviceClass, configurationClass) { + return function (router) { + var service = router.locationService = new serviceClass(router); + var configuration = router.locationConfig = new configurationClass(router, isHtml5); + function dispose(router) { + router.dispose(service); + router.dispose(configuration); + } + return { name: name, service: service, configuration: configuration, dispose: dispose }; + }; +} + +/** + * @internalapi + * @module vanilla + */ /** */ +/** A base `LocationServices` */ +var BaseLocationServices = /** @class */ (function () { + function BaseLocationServices(router, fireAfterUpdate) { + var _this = this; + this.fireAfterUpdate = fireAfterUpdate; + this._listener = function (evt) { return _this._listeners.forEach(function (cb) { return cb(evt); }); }; + this._listeners = []; + this.hash = function () { return parseUrl$1(_this._get()).hash; }; + this.path = function () { return parseUrl$1(_this._get()).path; }; + this.search = function () { return getParams(parseUrl$1(_this._get()).search); }; + this._location = root.location; + this._history = root.history; + } + BaseLocationServices.prototype.url = function (url, replace) { + if (replace === void 0) { replace = true; } + if (isDefined(url) && url !== this._get()) { + this._set(null, null, url, replace); + if (this.fireAfterUpdate) { + this._listeners.forEach(function (cb) { return cb({ url: url }); }); + } + } + return buildUrl(this); + }; + BaseLocationServices.prototype.onChange = function (cb) { + var _this = this; + this._listeners.push(cb); + return function () { return removeFrom(_this._listeners, cb); }; + }; + BaseLocationServices.prototype.dispose = function (router) { + deregAll(this._listeners); + }; + return BaseLocationServices; +}()); + +var __extends = (undefined && undefined.__extends) || (function () { + var extendStatics = Object.setPrototypeOf || + ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) || + function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; }; + return function (d, b) { + extendStatics(d, b); + function __() { this.constructor = d; } + d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __()); + }; +})(); +/** + * @internalapi + * @module vanilla + */ +/** */ +/** A `LocationServices` that uses the browser hash "#" to get/set the current location */ +var HashLocationService = /** @class */ (function (_super) { + __extends(HashLocationService, _super); + function HashLocationService(router) { + var _this = _super.call(this, router, false) || this; + root.addEventListener('hashchange', _this._listener, false); + return _this; + } + HashLocationService.prototype._get = function () { + return trimHashVal(this._location.hash); + }; + HashLocationService.prototype._set = function (state, title, url, replace) { + this._location.hash = url; + }; + HashLocationService.prototype.dispose = function (router) { + _super.prototype.dispose.call(this, router); + root.removeEventListener('hashchange', this._listener); + }; + return HashLocationService; +}(BaseLocationServices)); + +var __extends$1 = (undefined && undefined.__extends) || (function () { + var extendStatics = Object.setPrototypeOf || + ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) || + function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; }; + return function (d, b) { + extendStatics(d, b); + function __() { this.constructor = d; } + d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __()); + }; +})(); +/** + * @internalapi + * @module vanilla + */ +/** */ +/** A `LocationServices` that gets/sets the current location from an in-memory object */ +var MemoryLocationService = /** @class */ (function (_super) { + __extends$1(MemoryLocationService, _super); + function MemoryLocationService(router) { + return _super.call(this, router, true) || this; + } + MemoryLocationService.prototype._get = function () { + return this._url; + }; + MemoryLocationService.prototype._set = function (state, title, url, replace) { + this._url = url; + }; + return MemoryLocationService; +}(BaseLocationServices)); + +var __extends$2 = (undefined && undefined.__extends) || (function () { + var extendStatics = Object.setPrototypeOf || + ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) || + function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; }; + return function (d, b) { + extendStatics(d, b); + function __() { this.constructor = d; } + d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __()); + }; +})(); +/** + * A `LocationServices` that gets/sets the current location using the browser's `location` and `history` apis + * + * Uses `history.pushState` and `history.replaceState` + */ +var PushStateLocationService = /** @class */ (function (_super) { + __extends$2(PushStateLocationService, _super); + function PushStateLocationService(router) { + var _this = _super.call(this, router, true) || this; + _this._config = router.urlService.config; + root.addEventListener('popstate', _this._listener, false); + return _this; + } + + /** + * Gets the base prefix without: + * - trailing slash + * - trailing filename + * - protocol and hostname + * + * If , this returns '/base'. + * If , this returns '/base'. + * + * See: https://html.spec.whatwg.org/dev/semantics.html#the-base-element + */ + PushStateLocationService.prototype._getBasePrefix = function () { + return stripFile(this._config.baseHref()); + }; + PushStateLocationService.prototype._get = function () { + var _a = this._location, pathname = _a.pathname, hash = _a.hash, search = _a.search; + search = splitQuery(search)[1]; // strip ? if found + hash = splitHash(hash)[1]; // strip # if found + var basePrefix = this._getBasePrefix(); + var exactMatch = pathname === this._config.baseHref(); + var startsWith = pathname.startsWith(basePrefix); + pathname = exactMatch ? '/' : startsWith ? pathname.substring(basePrefix.length) : pathname; + return pathname + (search ? '?' + search : '') + (hash ? '#' + hash : ''); + }; + PushStateLocationService.prototype._set = function (state, title, url, replace) { + var fullUrl = this._getBasePrefix() + url; + if (replace) { + this._history.replaceState(state, title, fullUrl); + } + else { + this._history.pushState(state, title, fullUrl); + } + }; + PushStateLocationService.prototype.dispose = function (router) { + _super.prototype.dispose.call(this, router); + root.removeEventListener('popstate', this._listener); + }; + return PushStateLocationService; +}(BaseLocationServices)); + +/** A `LocationConfig` mock that gets/sets all config from an in-memory object */ +var MemoryLocationConfig = /** @class */ (function () { + function MemoryLocationConfig() { + var _this = this; + this._baseHref = ''; + this._port = 80; + this._protocol = "http"; + this._host = "localhost"; + this._hashPrefix = ""; + this.port = function () { return _this._port; }; + this.protocol = function () { return _this._protocol; }; + this.host = function () { return _this._host; }; + this.baseHref = function () { return _this._baseHref; }; + this.html5Mode = function () { return false; }; + this.hashPrefix = function (newval) { return isDefined(newval) ? _this._hashPrefix = newval : _this._hashPrefix; }; + this.dispose = noop$1; + } + return MemoryLocationConfig; +}()); + +/** + * @internalapi + * @module vanilla + */ +/** */ +/** A `LocationConfig` that delegates to the browser's `location` object */ +var BrowserLocationConfig = /** @class */ (function () { + function BrowserLocationConfig(router, _isHtml5) { + if (_isHtml5 === void 0) { _isHtml5 = false; } + this._isHtml5 = _isHtml5; + this._baseHref = undefined; + this._hashPrefix = ""; + } + BrowserLocationConfig.prototype.port = function () { + if (location.port) { + return Number(location.port); + } + return this.protocol() === 'https' ? 443 : 80; + }; + BrowserLocationConfig.prototype.protocol = function () { + return location.protocol.replace(/:/g, ''); + }; + BrowserLocationConfig.prototype.host = function () { + return location.hostname; + }; + BrowserLocationConfig.prototype.html5Mode = function () { + return this._isHtml5; + }; + BrowserLocationConfig.prototype.hashPrefix = function (newprefix) { + return isDefined(newprefix) ? this._hashPrefix = newprefix : this._hashPrefix; + }; + + BrowserLocationConfig.prototype.baseHref = function (href) { + return isDefined(href) ? this._baseHref = href : + isDefined(this._baseHref) ? this._baseHref : this.applyDocumentBaseHref(); + }; + BrowserLocationConfig.prototype.applyDocumentBaseHref = function () { + var baseTag = document.getElementsByTagName("base")[0]; + return this._baseHref = baseTag ? baseTag.href.substr(location.origin.length) : ""; + }; + BrowserLocationConfig.prototype.dispose = function () { }; + return BrowserLocationConfig; +}()); + +/** + * @internalapi + * @module vanilla + */ +/** */ +function servicesPlugin(router) { + services.$injector = $injector; + services.$q = $q; + return { name: "vanilla.services", $q: $q, $injector: $injector, dispose: function () { return null; } }; +} +/** A `UIRouterPlugin` uses the browser hash to get/set the current location */ +var hashLocationPlugin = locationPluginFactory('vanilla.hashBangLocation', false, HashLocationService, BrowserLocationConfig); +/** A `UIRouterPlugin` that gets/sets the current location using the browser's `location` and `history` apis */ +var pushStateLocationPlugin = locationPluginFactory("vanilla.pushStateLocation", true, PushStateLocationService, BrowserLocationConfig); +/** A `UIRouterPlugin` that gets/sets the current location from an in-memory object */ +var memoryLocationPlugin = locationPluginFactory("vanilla.memoryLocation", false, MemoryLocationService, MemoryLocationConfig); + +/** + * @internalapi + * @module vanilla + */ +/** */ + +/** + * # Core classes and interfaces + * + * The classes and interfaces that are core to ui-router and do not belong + * to a more specific subsystem (such as resolve). + * + * @coreapi + * @preferred + * @module core + */ /** for typedoc */ +/** @internalapi */ +var UIRouterPluginBase = /** @class */ (function () { + function UIRouterPluginBase() { + } + UIRouterPluginBase.prototype.dispose = function (router) { }; + return UIRouterPluginBase; +}()); + +/** + * @coreapi + * @module common + */ /** */ + + + +var index$1 = Object.freeze({ + root: root, + fromJson: fromJson, + toJson: toJson, + forEach: forEach, + extend: extend, + equals: equals, + identity: identity, + noop: noop$1, + createProxyFunctions: createProxyFunctions, + inherit: inherit, + inArray: inArray, + _inArray: _inArray, + removeFrom: removeFrom, + _removeFrom: _removeFrom, + pushTo: pushTo, + _pushTo: _pushTo, + deregAll: deregAll, + defaults: defaults, + mergeR: mergeR, + ancestors: ancestors, + pick: pick, + omit: omit, + pluck: pluck, + filter: filter, + find: find, + mapObj: mapObj, + map: map, + values: values, + allTrueR: allTrueR, + anyTrueR: anyTrueR, + unnestR: unnestR, + flattenR: flattenR, + pushR: pushR, + uniqR: uniqR, + unnest: unnest, + flatten: flatten, + assertPredicate: assertPredicate, + assertMap: assertMap, + assertFn: assertFn, + pairs: pairs, + arrayTuples: arrayTuples, + applyPairs: applyPairs, + tail: tail, + copy: copy, + _extend: _extend, + silenceUncaughtInPromise: silenceUncaughtInPromise, + silentRejection: silentRejection, + notImplemented: notImplemented, + services: services, + Glob: Glob, + curry: curry, + compose: compose, + pipe: pipe, + prop: prop, + propEq: propEq, + parse: parse, + not: not, + and: and, + or: or, + all: all, + any: any, + is: is, + eq: eq, + val: val, + invoke: invoke, + pattern: pattern, + isUndefined: isUndefined, + isDefined: isDefined, + isNull: isNull, + isNullOrUndefined: isNullOrUndefined, + isFunction: isFunction, + isNumber: isNumber, + isString: isString, + isObject: isObject, + isArray: isArray, + isDate: isDate, + isRegExp: isRegExp, + isState: isState, + isInjectable: isInjectable, + isPromise: isPromise, + Queue: Queue, + maxLength: maxLength, + padString: padString, + kebobString: kebobString, + functionToString: functionToString, + fnToString: fnToString, + stringify: stringify, + beforeAfterSubstr: beforeAfterSubstr, + hostRegex: hostRegex, + stripFile: stripFile, + splitHash: splitHash, + splitQuery: splitQuery, + splitEqual: splitEqual, + trimHashVal: trimHashVal, + splitOnDelim: splitOnDelim, + joinNeighborsR: joinNeighborsR, + get Category () { return exports.Category; }, + Trace: Trace, + trace: trace, + get DefType () { return exports.DefType; }, + Param: Param, + ParamTypes: ParamTypes, + StateParams: StateParams, + ParamType: ParamType, + PathNode: PathNode, + PathUtils: PathUtils, + resolvePolicies: resolvePolicies, + defaultResolvePolicy: defaultResolvePolicy, + Resolvable: Resolvable, + NATIVE_INJECTOR_TOKEN: NATIVE_INJECTOR_TOKEN, + ResolveContext: ResolveContext, + resolvablesBuilder: resolvablesBuilder, + StateBuilder: StateBuilder, + StateObject: StateObject, + StateMatcher: StateMatcher, + StateQueueManager: StateQueueManager, + StateRegistry: StateRegistry, + StateService: StateService, + TargetState: TargetState, + get TransitionHookPhase () { return exports.TransitionHookPhase; }, + get TransitionHookScope () { return exports.TransitionHookScope; }, + HookBuilder: HookBuilder, + matchState: matchState, + RegisteredHook: RegisteredHook, + makeEvent: makeEvent, + get RejectType () { return exports.RejectType; }, + Rejection: Rejection, + Transition: Transition, + TransitionHook: TransitionHook, + TransitionEventType: TransitionEventType, + defaultTransOpts: defaultTransOpts, + TransitionService: TransitionService, + UrlMatcher: UrlMatcher, + UrlMatcherFactory: UrlMatcherFactory, + UrlRouter: UrlRouter, + UrlRuleFactory: UrlRuleFactory, + BaseUrlRule: BaseUrlRule, + UrlService: UrlService, + ViewService: ViewService, + UIRouterGlobals: UIRouterGlobals, + UIRouter: UIRouter, + $q: $q, + $injector: $injector, + BaseLocationServices: BaseLocationServices, + HashLocationService: HashLocationService, + MemoryLocationService: MemoryLocationService, + PushStateLocationService: PushStateLocationService, + MemoryLocationConfig: MemoryLocationConfig, + BrowserLocationConfig: BrowserLocationConfig, + keyValsToObjectR: keyValsToObjectR, + getParams: getParams, + parseUrl: parseUrl$1, + buildUrl: buildUrl, + locationPluginFactory: locationPluginFactory, + servicesPlugin: servicesPlugin, + hashLocationPlugin: hashLocationPlugin, + pushStateLocationPlugin: pushStateLocationPlugin, + memoryLocationPlugin: memoryLocationPlugin, + UIRouterPluginBase: UIRouterPluginBase +}); + +function getNg1ViewConfigFactory() { + var templateFactory = null; + return function (path, view) { + templateFactory = templateFactory || services.$injector.get("$templateFactory"); + return [new Ng1ViewConfig(path, view, templateFactory)]; + }; +} +var hasAnyKey = function (keys, obj) { + return keys.reduce(function (acc, key) { return acc || isDefined(obj[key]); }, false); +}; +/** + * This is a [[StateBuilder.builder]] function for angular1 `views`. + * + * When the [[StateBuilder]] builds a [[StateObject]] object from a raw [[StateDeclaration]], this builder + * handles the `views` property with logic specific to @uirouter/angularjs (ng1). + * + * If no `views: {}` property exists on the [[StateDeclaration]], then it creates the `views` object + * and applies the state-level configuration to a view named `$default`. + */ +function ng1ViewsBuilder(state) { + // Do not process root state + if (!state.parent) + return {}; + var tplKeys = ['templateProvider', 'templateUrl', 'template', 'notify', 'async'], ctrlKeys = ['controller', 'controllerProvider', 'controllerAs', 'resolveAs'], compKeys = ['component', 'bindings', 'componentProvider'], nonCompKeys = tplKeys.concat(ctrlKeys), allViewKeys = compKeys.concat(nonCompKeys); + // Do not allow a state to have both state-level props and also a `views: {}` property. + // A state without a `views: {}` property can declare properties for the `$default` view as properties of the state. + // However, the `$default` approach should not be mixed with a separate `views: ` block. + if (isDefined(state.views) && hasAnyKey(allViewKeys, state)) { + throw new Error("State '" + state.name + "' has a 'views' object. " + + "It cannot also have \"view properties\" at the state level. " + + "Move the following properties into a view (in the 'views' object): " + + (" " + allViewKeys.filter(function (key) { return isDefined(state[key]); }).join(", "))); + } + var views = {}, viewsObject = state.views || { "$default": pick(state, allViewKeys) }; + forEach(viewsObject, function (config, name) { + // Account for views: { "": { template... } } + name = name || "$default"; + // Account for views: { header: "headerComponent" } + if (isString(config)) + config = { component: config }; + // Make a shallow copy of the config object + config = extend({}, config); + // Do not allow a view to mix props for component-style view with props for template/controller-style view + if (hasAnyKey(compKeys, config) && hasAnyKey(nonCompKeys, config)) { + throw new Error("Cannot combine: " + compKeys.join("|") + " with: " + nonCompKeys.join("|") + " in stateview: '" + name + "@" + state.name + "'"); + } + config.resolveAs = config.resolveAs || '$resolve'; + config.$type = "ng1"; + config.$context = state; + config.$name = name; + var normalized = ViewService.normalizeUIViewTarget(config.$context, config.$name); + config.$uiViewName = normalized.uiViewName; + config.$uiViewContextAnchor = normalized.uiViewContextAnchor; + views[name] = config; + }); + return views; +} +var id$1 = 0; +var Ng1ViewConfig = /** @class */ (function () { + function Ng1ViewConfig(path, viewDecl, factory) { + var _this = this; + this.path = path; + this.viewDecl = viewDecl; + this.factory = factory; + this.$id = id$1++; + this.loaded = false; + this.getTemplate = function (uiView, context) { + return _this.component ? _this.factory.makeComponentTemplate(uiView, context, _this.component, _this.viewDecl.bindings) : _this.template; + }; + } + Ng1ViewConfig.prototype.load = function () { + var _this = this; + var $q = services.$q; + var context = new ResolveContext(this.path); + var params = this.path.reduce(function (acc, node) { return extend(acc, node.paramValues); }, {}); + var promises = { + template: $q.when(this.factory.fromConfig(this.viewDecl, params, context)), + controller: $q.when(this.getController(context)) + }; + return $q.all(promises).then(function (results) { + trace.traceViewServiceEvent("Loaded", _this); + _this.controller = results.controller; + extend(_this, results.template); // Either { template: "tpl" } or { component: "cmpName" } + return _this; + }); + }; + /** + * Gets the controller for a view configuration. + * + * @returns {Function|Promise.} Returns a controller, or a promise that resolves to a controller. + */ + Ng1ViewConfig.prototype.getController = function (context) { + var provider = this.viewDecl.controllerProvider; + if (!isInjectable(provider)) + return this.viewDecl.controller; + var deps = services.$injector.annotate(provider); + var providerFn = isArray(provider) ? tail(provider) : provider; + var resolvable = new Resolvable("", providerFn, deps); + return resolvable.get(context); + }; + return Ng1ViewConfig; +}()); + +/** @module view */ +/** for typedoc */ +/** + * Service which manages loading of templates from a ViewConfig. + */ +var TemplateFactory = /** @class */ (function () { + function TemplateFactory() { + var _this = this; + /** @hidden */ this._useHttp = ng.version.minor < 3; + /** @hidden */ this.$get = ['$http', '$templateCache', '$injector', function ($http, $templateCache, $injector) { + _this.$templateRequest = $injector.has && $injector.has('$templateRequest') && $injector.get('$templateRequest'); + _this.$http = $http; + _this.$templateCache = $templateCache; + return _this; + }]; + } + /** @hidden */ + TemplateFactory.prototype.useHttpService = function (value) { + this._useHttp = value; + }; + + /** + * Creates a template from a configuration object. + * + * @param config Configuration object for which to load a template. + * The following properties are search in the specified order, and the first one + * that is defined is used to create the template: + * + * @param params Parameters to pass to the template function. + * @param context The resolve context associated with the template's view + * + * @return {string|object} The template html as a string, or a promise for + * that string,or `null` if no template is configured. + */ + TemplateFactory.prototype.fromConfig = function (config, params, context) { + var defaultTemplate = ""; + var asTemplate = function (result) { return services.$q.when(result).then(function (str) { return ({ template: str }); }); }; + var asComponent = function (result) { return services.$q.when(result).then(function (str) { return ({ component: str }); }); }; + return (isDefined(config.template) ? asTemplate(this.fromString(config.template, params)) : + isDefined(config.templateUrl) ? asTemplate(this.fromUrl(config.templateUrl, params)) : + isDefined(config.templateProvider) ? asTemplate(this.fromProvider(config.templateProvider, params, context)) : + isDefined(config.component) ? asComponent(config.component) : + isDefined(config.componentProvider) ? asComponent(this.fromComponentProvider(config.componentProvider, params, context)) : + asTemplate(defaultTemplate)); + }; + + /** + * Creates a template from a string or a function returning a string. + * + * @param template html template as a string or function that returns an html template as a string. + * @param params Parameters to pass to the template function. + * + * @return {string|object} The template html as a string, or a promise for that + * string. + */ + TemplateFactory.prototype.fromString = function (template, params) { + return isFunction(template) ? template(params) : template; + }; + + /** + * Loads a template from the a URL via `$http` and `$templateCache`. + * + * @param {string|Function} url url of the template to load, or a function + * that returns a url. + * @param {Object} params Parameters to pass to the url function. + * @return {string|Promise.} The template html as a string, or a promise + * for that string. + */ + TemplateFactory.prototype.fromUrl = function (url, params) { + if (isFunction(url)) + url = url(params); + if (url == null) + return null; + if (this._useHttp) { + return this.$http.get(url, { cache: this.$templateCache, headers: { Accept: 'text/html' } }) + .then(function (response) { + return response.data; + }); + } + return this.$templateRequest(url); + }; + + /** + * Creates a template by invoking an injectable provider function. + * + * @param provider Function to invoke via `locals` + * @param {Function} injectFn a function used to invoke the template provider + * @return {string|Promise.} The template html as a string, or a promise + * for that string. + */ + TemplateFactory.prototype.fromProvider = function (provider, params, context) { + var deps = services.$injector.annotate(provider); + var providerFn = isArray(provider) ? tail(provider) : provider; + var resolvable = new Resolvable("", providerFn, deps); + return resolvable.get(context); + }; + + /** + * Creates a component's template by invoking an injectable provider function. + * + * @param provider Function to invoke via `locals` + * @param {Function} injectFn a function used to invoke the template provider + * @return {string} The template html as a string: "". + */ + TemplateFactory.prototype.fromComponentProvider = function (provider, params, context) { + var deps = services.$injector.annotate(provider); + var providerFn = isArray(provider) ? tail(provider) : provider; + var resolvable = new Resolvable("", providerFn, deps); + return resolvable.get(context); + }; + + /** + * Creates a template from a component's name + * + * This implements route-to-component. + * It works by retrieving the component (directive) metadata from the injector. + * It analyses the component's bindings, then constructs a template that instantiates the component. + * The template wires input and output bindings to resolves or from the parent component. + * + * @param uiView {object} The parent ui-view (for binding outputs to callbacks) + * @param context The ResolveContext (for binding outputs to callbacks returned from resolves) + * @param component {string} Component's name in camel case. + * @param bindings An object defining the component's bindings: {foo: '<'} + * @return {string} The template as a string: "". + */ + TemplateFactory.prototype.makeComponentTemplate = function (uiView, context, component, bindings) { + bindings = bindings || {}; + // Bind once prefix + var prefix = ng.version.minor >= 3 ? "::" : ""; + // Convert to kebob name. Add x- prefix if the string starts with `x-` or `data-` + var kebob = function (camelCase) { + var kebobed = kebobString(camelCase); + return /^(x|data)-/.exec(kebobed) ? "x-" + kebobed : kebobed; + }; + var attributeTpl = function (input) { + var name = input.name, type = input.type; + var attrName = kebob(name); + // If the ui-view has an attribute which matches a binding on the routed component + // then pass that attribute through to the routed component template. + // Prefer ui-view wired mappings to resolve data, unless the resolve was explicitly bound using `bindings:` + if (uiView.attr(attrName) && !bindings[name]) + return attrName + "='" + uiView.attr(attrName) + "'"; + var resolveName = bindings[name] || name; + // Pre-evaluate the expression for "@" bindings by enclosing in {{ }} + // some-attr="{{ ::$resolve.someResolveName }}" + if (type === '@') + return attrName + "='{{" + prefix + "$resolve." + resolveName + "}}'"; + // Wire "&" callbacks to resolves that return a callback function + // Get the result of the resolve (should be a function) and annotate it to get its arguments. + // some-attr="$resolve.someResolveResultName(foo, bar)" + if (type === '&') { + var res = context.getResolvable(resolveName); + var fn = res && res.data; + var args = fn && services.$injector.annotate(fn) || []; + // account for array style injection, i.e., ['foo', function(foo) {}] + var arrayIdxStr = isArray(fn) ? "[" + (fn.length - 1) + "]" : ''; + return attrName + "='$resolve." + resolveName + arrayIdxStr + "(" + args.join(",") + ")'"; + } + // some-attr="::$resolve.someResolveName" + return attrName + "='" + prefix + "$resolve." + resolveName + "'"; + }; + var attrs = getComponentBindings(component).map(attributeTpl).join(" "); + var kebobName = kebob(component); + return "<" + kebobName + " " + attrs + ">"; + }; + + return TemplateFactory; +}()); +// Gets all the directive(s)' inputs ('@', '=', and '<') and outputs ('&') +function getComponentBindings(name) { + var cmpDefs = services.$injector.get(name + "Directive"); // could be multiple + if (!cmpDefs || !cmpDefs.length) + throw new Error("Unable to find component named '" + name + "'"); + return cmpDefs.map(getBindings).reduce(unnestR, []); +} +// Given a directive definition, find its object input attributes +// Use different properties, depending on the type of directive (component, bindToController, normal) +var getBindings = function (def) { + if (isObject(def.bindToController)) + return scopeBindings(def.bindToController); + return scopeBindings(def.scope); +}; +// for ng 1.2 style, process the scope: { input: "=foo" } +// for ng 1.3 through ng 1.5, process the component's bindToController: { input: "=foo" } object +var scopeBindings = function (bindingsObj) { return Object.keys(bindingsObj || {}) + .map(function (key) { return [key, /^([=<@&])[?]?(.*)/.exec(bindingsObj[key])]; }) + .filter(function (tuple) { return isDefined(tuple) && isArray(tuple[1]); }) + .map(function (tuple) { return ({ name: tuple[1][2] || tuple[0], type: tuple[1][1] }); }); }; + +/** @module ng1 */ /** for typedoc */ +/** + * The Angular 1 `StateProvider` + * + * The `$stateProvider` works similar to Angular's v1 router, but it focuses purely + * on state. + * + * A state corresponds to a "place" in the application in terms of the overall UI and + * navigation. A state describes (via the controller / template / view properties) what + * the UI looks like and does at that place. + * + * States often have things in common, and the primary way of factoring out these + * commonalities in this model is via the state hierarchy, i.e. parent/child states aka + * nested states. + * + * The `$stateProvider` provides interfaces to declare these states for your app. + */ +var StateProvider = /** @class */ (function () { + function StateProvider(stateRegistry, stateService) { + this.stateRegistry = stateRegistry; + this.stateService = stateService; + createProxyFunctions(val(StateProvider.prototype), this, val(this)); + } + /** + * Decorates states when they are registered + * + * Allows you to extend (carefully) or override (at your own peril) the + * `stateBuilder` object used internally by [[StateRegistry]]. + * This can be used to add custom functionality to ui-router, + * for example inferring templateUrl based on the state name. + * + * When passing only a name, it returns the current (original or decorated) builder + * function that matches `name`. + * + * The builder functions that can be decorated are listed below. Though not all + * necessarily have a good use case for decoration, that is up to you to decide. + * + * In addition, users can attach custom decorators, which will generate new + * properties within the state's internal definition. There is currently no clear + * use-case for this beyond accessing internal states (i.e. $state.$current), + * however, expect this to become increasingly relevant as we introduce additional + * meta-programming features. + * + * **Warning**: Decorators should not be interdependent because the order of + * execution of the builder functions in non-deterministic. Builder functions + * should only be dependent on the state definition object and super function. + * + * + * Existing builder functions and current return values: + * + * - **parent** `{object}` - returns the parent state object. + * - **data** `{object}` - returns state data, including any inherited data that is not + * overridden by own values (if any). + * - **url** `{object}` - returns a {@link ui.router.util.type:UrlMatcher UrlMatcher} + * or `null`. + * - **navigable** `{object}` - returns closest ancestor state that has a URL (aka is + * navigable). + * - **params** `{object}` - returns an array of state params that are ensured to + * be a super-set of parent's params. + * - **views** `{object}` - returns a views object where each key is an absolute view + * name (i.e. "viewName@stateName") and each value is the config object + * (template, controller) for the view. Even when you don't use the views object + * explicitly on a state config, one is still created for you internally. + * So by decorating this builder function you have access to decorating template + * and controller properties. + * - **ownParams** `{object}` - returns an array of params that belong to the state, + * not including any params defined by ancestor states. + * - **path** `{string}` - returns the full path from the root down to this state. + * Needed for state activation. + * - **includes** `{object}` - returns an object that includes every state that + * would pass a `$state.includes()` test. + * + * #### Example: + * Override the internal 'views' builder with a function that takes the state + * definition, and a reference to the internal function being overridden: + * ```js + * $stateProvider.decorator('views', function (state, parent) { + * let result = {}, + * views = parent(state); + * + * angular.forEach(views, function (config, name) { + * let autoName = (state.name + '.' + name).replace('.', '/'); + * config.templateUrl = config.templateUrl || '/partials/' + autoName + '.html'; + * result[name] = config; + * }); + * return result; + * }); + * + * $stateProvider.state('home', { + * views: { + * 'contact.list': { controller: 'ListController' }, + * 'contact.item': { controller: 'ItemController' } + * } + * }); + * ``` + * + * + * ```js + * // Auto-populates list and item views with /partials/home/contact/list.html, + * // and /partials/home/contact/item.html, respectively. + * $state.go('home'); + * ``` + * + * @param {string} name The name of the builder function to decorate. + * @param {object} func A function that is responsible for decorating the original + * builder function. The function receives two parameters: + * + * - `{object}` - state - The state config object. + * - `{object}` - super - The original builder function. + * + * @return {object} $stateProvider - $stateProvider instance + */ + StateProvider.prototype.decorator = function (name, func) { + return this.stateRegistry.decorator(name, func) || this; + }; + StateProvider.prototype.state = function (name, definition) { + if (isObject(name)) { + definition = name; + } + else { + definition.name = name; + } + this.stateRegistry.register(definition); + return this; + }; + /** + * Registers an invalid state handler + * + * This is a passthrough to [[StateService.onInvalid]] for ng1. + */ + StateProvider.prototype.onInvalid = function (callback) { + return this.stateService.onInvalid(callback); + }; + return StateProvider; +}()); + +/** @module ng1 */ /** */ +/** + * This is a [[StateBuilder.builder]] function for angular1 `onEnter`, `onExit`, + * `onRetain` callback hooks on a [[Ng1StateDeclaration]]. + * + * When the [[StateBuilder]] builds a [[StateObject]] object from a raw [[StateDeclaration]], this builder + * ensures that those hooks are injectable for @uirouter/angularjs (ng1). + */ +var getStateHookBuilder = function (hookName) { + return function stateHookBuilder(state, parentFn) { + var hook = state[hookName]; + var pathname = hookName === 'onExit' ? 'from' : 'to'; + function decoratedNg1Hook(trans, state) { + var resolveContext = new ResolveContext(trans.treeChanges(pathname)); + var locals = extend(getLocals(resolveContext), { $state$: state, $transition$: trans }); + return services.$injector.invoke(hook, this, locals); + } + return hook ? decoratedNg1Hook : undefined; + }; +}; + +/** + * Implements UI-Router LocationServices and LocationConfig using Angular 1's $location service + */ +var Ng1LocationServices = /** @class */ (function () { + function Ng1LocationServices($locationProvider) { + // .onChange() registry + this._urlListeners = []; + this.$locationProvider = $locationProvider; + var _lp = val($locationProvider); + createProxyFunctions(_lp, this, _lp, ['hashPrefix']); + } + Ng1LocationServices.prototype.dispose = function () { }; + Ng1LocationServices.prototype.onChange = function (callback) { + var _this = this; + this._urlListeners.push(callback); + return function () { return removeFrom(_this._urlListeners)(callback); }; + }; + Ng1LocationServices.prototype.html5Mode = function () { + var html5Mode = this.$locationProvider.html5Mode(); + html5Mode = isObject(html5Mode) ? html5Mode.enabled : html5Mode; + return html5Mode && this.$sniffer.history; + }; + Ng1LocationServices.prototype.url = function (newUrl, replace, state) { + if (replace === void 0) { replace = false; } + if (newUrl) + this.$location.url(newUrl); + if (replace) + this.$location.replace(); + if (state) + this.$location.state(state); + return this.$location.url(); + }; + Ng1LocationServices.prototype._runtimeServices = function ($rootScope, $location, $sniffer, $browser) { + var _this = this; + this.$location = $location; + this.$sniffer = $sniffer; + // Bind $locationChangeSuccess to the listeners registered in LocationService.onChange + $rootScope.$on("$locationChangeSuccess", function (evt) { return _this._urlListeners.forEach(function (fn) { return fn(evt); }); }); + var _loc = val($location); + var _browser = val($browser); + // Bind these LocationService functions to $location + createProxyFunctions(_loc, this, _loc, ["replace", "path", "search", "hash"]); + // Bind these LocationConfig functions to $location + createProxyFunctions(_loc, this, _loc, ['port', 'protocol', 'host']); + // Bind these LocationConfig functions to $browser + createProxyFunctions(_browser, this, _browser, ['baseHref']); + }; + /** + * Applys ng1-specific path parameter encoding + * + * The Angular 1 `$location` service is a bit weird. + * It doesn't allow slashes to be encoded/decoded bi-directionally. + * + * See the writeup at https://github.com/angular-ui/ui-router/issues/2598 + * + * This code patches the `path` parameter type so it encoded/decodes slashes as ~2F + * + * @param router + */ + Ng1LocationServices.monkeyPatchPathParameterType = function (router) { + var pathType = router.urlMatcherFactory.type('path'); + pathType.encode = function (val) { + return val != null ? val.toString().replace(/(~|\/)/g, function (m) { return ({ '~': '~~', '/': '~2F' }[m]); }) : val; + }; + pathType.decode = function (val) { + return val != null ? val.toString().replace(/(~~|~2F)/g, function (m) { return ({ '~~': '~', '~2F': '/' }[m]); }) : val; + }; + }; + return Ng1LocationServices; +}()); + +/** @module url */ /** */ +/** + * Manages rules for client-side URL + * + * ### Deprecation warning: + * This class is now considered to be an internal API + * Use the [[UrlService]] instead. + * For configuring URL rules, use the [[UrlRulesApi]] which can be found as [[UrlService.rules]]. + * + * This class manages the router rules for what to do when the URL changes. + * + * This provider remains for backwards compatibility. + * + * @deprecated + */ +var UrlRouterProvider = /** @class */ (function () { + /** @hidden */ + function UrlRouterProvider(router) { + this._router = router; + this._urlRouter = router.urlRouter; + } + /** @hidden */ + UrlRouterProvider.prototype.$get = function () { + var urlRouter = this._urlRouter; + urlRouter.update(true); + if (!urlRouter.interceptDeferred) + urlRouter.listen(); + return urlRouter; + }; + /** + * Registers a url handler function. + * + * Registers a low level url handler (a `rule`). + * A rule detects specific URL patterns and returns a redirect, or performs some action. + * + * If a rule returns a string, the URL is replaced with the string, and all rules are fired again. + * + * #### Example: + * ```js + * var app = angular.module('app', ['ui.router.router']); + * + * app.config(function ($urlRouterProvider) { + * // Here's an example of how you might allow case insensitive urls + * $urlRouterProvider.rule(function ($injector, $location) { + * var path = $location.path(), + * normalized = path.toLowerCase(); + * + * if (path !== normalized) { + * return normalized; + * } + * }); + * }); + * ``` + * + * @param ruleFn + * Handler function that takes `$injector` and `$location` services as arguments. + * You can use them to detect a url and return a different url as a string. + * + * @return [[UrlRouterProvider]] (`this`) + */ + UrlRouterProvider.prototype.rule = function (ruleFn) { + var _this = this; + if (!isFunction(ruleFn)) + throw new Error("'rule' must be a function"); + var match = function () { + return ruleFn(services.$injector, _this._router.locationService); + }; + var rule = new BaseUrlRule(match, identity); + this._urlRouter.rule(rule); + return this; + }; + + /** + * Defines the path or behavior to use when no url can be matched. + * + * #### Example: + * ```js + * var app = angular.module('app', ['ui.router.router']); + * + * app.config(function ($urlRouterProvider) { + * // if the path doesn't match any of the urls you configured + * // otherwise will take care of routing the user to the + * // specified url + * $urlRouterProvider.otherwise('/index'); + * + * // Example of using function rule as param + * $urlRouterProvider.otherwise(function ($injector, $location) { + * return '/a/valid/url'; + * }); + * }); + * ``` + * + * @param rule + * The url path you want to redirect to or a function rule that returns the url path or performs a `$state.go()`. + * The function version is passed two params: `$injector` and `$location` services, and should return a url string. + * + * @return {object} `$urlRouterProvider` - `$urlRouterProvider` instance + */ + UrlRouterProvider.prototype.otherwise = function (rule) { + var _this = this; + var urlRouter = this._urlRouter; + if (isString(rule)) { + urlRouter.otherwise(rule); + } + else if (isFunction(rule)) { + urlRouter.otherwise(function () { return rule(services.$injector, _this._router.locationService); }); + } + else { + throw new Error("'rule' must be a string or function"); + } + return this; + }; + + /** + * Registers a handler for a given url matching. + * + * If the handler is a string, it is + * treated as a redirect, and is interpolated according to the syntax of match + * (i.e. like `String.replace()` for `RegExp`, or like a `UrlMatcher` pattern otherwise). + * + * If the handler is a function, it is injectable. + * It gets invoked if `$location` matches. + * You have the option of inject the match object as `$match`. + * + * The handler can return + * + * - **falsy** to indicate that the rule didn't match after all, then `$urlRouter` + * will continue trying to find another one that matches. + * - **string** which is treated as a redirect and passed to `$location.url()` + * - **void** or any **truthy** value tells `$urlRouter` that the url was handled. + * + * #### Example: + * ```js + * var app = angular.module('app', ['ui.router.router']); + * + * app.config(function ($urlRouterProvider) { + * $urlRouterProvider.when($state.url, function ($match, $stateParams) { + * if ($state.$current.navigable !== state || + * !equalForKeys($match, $stateParams) { + * $state.transitionTo(state, $match, false); + * } + * }); + * }); + * ``` + * + * @param what A pattern string to match, compiled as a [[UrlMatcher]]. + * @param handler The path (or function that returns a path) that you want to redirect your user to. + * @param ruleCallback [optional] A callback that receives the `rule` registered with [[UrlMatcher.rule]] + * + * Note: the handler may also invoke arbitrary code, such as `$state.go()` + */ + UrlRouterProvider.prototype.when = function (what, handler) { + if (isArray(handler) || isFunction(handler)) { + handler = UrlRouterProvider.injectableHandler(this._router, handler); + } + this._urlRouter.when(what, handler); + return this; + }; + + UrlRouterProvider.injectableHandler = function (router, handler) { + return function (match) { + return services.$injector.invoke(handler, null, { $match: match, $stateParams: router.globals.params }); + }; + }; + /** + * Disables monitoring of the URL. + * + * Call this method before UI-Router has bootstrapped. + * It will stop UI-Router from performing the initial url sync. + * + * This can be useful to perform some asynchronous initialization before the router starts. + * Once the initialization is complete, call [[listen]] to tell UI-Router to start watching and synchronizing the URL. + * + * #### Example: + * ```js + * var app = angular.module('app', ['ui.router']); + * + * app.config(function ($urlRouterProvider) { + * // Prevent $urlRouter from automatically intercepting URL changes; + * $urlRouterProvider.deferIntercept(); + * }) + * + * app.run(function (MyService, $urlRouter, $http) { + * $http.get("/stuff").then(function(resp) { + * MyService.doStuff(resp.data); + * $urlRouter.listen(); + * $urlRouter.sync(); + * }); + * }); + * ``` + * + * @param defer Indicates whether to defer location change interception. + * Passing no parameter is equivalent to `true`. + */ + UrlRouterProvider.prototype.deferIntercept = function (defer) { + this._urlRouter.deferIntercept(defer); + }; + + return UrlRouterProvider; +}()); + +/** + * # Angular 1 types + * + * UI-Router core provides various Typescript types which you can use for code completion and validating parameter values, etc. + * The customizations to the core types for Angular UI-Router are documented here. + * + * The optional [[$resolve]] service is also documented here. + * + * @module ng1 + * @preferred + */ +/** for typedoc */ +ng.module("ui.router.angular1", []); +var mod_init = ng.module('ui.router.init', []); +var mod_util = ng.module('ui.router.util', ['ng', 'ui.router.init']); +var mod_rtr = ng.module('ui.router.router', ['ui.router.util']); +var mod_state = ng.module('ui.router.state', ['ui.router.router', 'ui.router.util', 'ui.router.angular1']); +var mod_main = ng.module('ui.router', ['ui.router.init', 'ui.router.state', 'ui.router.angular1']); +var mod_cmpt = ng.module('ui.router.compat', ['ui.router']); // tslint:disable-line +var router = null; +$uiRouter.$inject = ['$locationProvider']; +/** This angular 1 provider instantiates a Router and exposes its services via the angular injector */ +function $uiRouter($locationProvider) { + // Create a new instance of the Router when the $uiRouterProvider is initialized + router = this.router = new UIRouter(); + router.stateProvider = new StateProvider(router.stateRegistry, router.stateService); + // Apply ng1 specific StateBuilder code for `views`, `resolve`, and `onExit/Retain/Enter` properties + router.stateRegistry.decorator("views", ng1ViewsBuilder); + router.stateRegistry.decorator("onExit", getStateHookBuilder("onExit")); + router.stateRegistry.decorator("onRetain", getStateHookBuilder("onRetain")); + router.stateRegistry.decorator("onEnter", getStateHookBuilder("onEnter")); + router.viewService._pluginapi._viewConfigFactory('ng1', getNg1ViewConfigFactory()); + var ng1LocationService = router.locationService = router.locationConfig = new Ng1LocationServices($locationProvider); + Ng1LocationServices.monkeyPatchPathParameterType(router); + // backwards compat: also expose router instance as $uiRouterProvider.router + router['router'] = router; + router['$get'] = $get; + $get.$inject = ['$location', '$browser', '$sniffer', '$rootScope', '$http', '$templateCache']; + function $get($location, $browser, $sniffer, $rootScope, $http, $templateCache) { + ng1LocationService._runtimeServices($rootScope, $location, $sniffer, $browser); + delete router['router']; + delete router['$get']; + return router; + } + return router; +} +var getProviderFor = function (serviceName) { return ['$uiRouterProvider', function ($urp) { + var service = $urp.router[serviceName]; + service["$get"] = function () { return service; }; + return service; + }]; }; +// This effectively calls $get() on `$uiRouterProvider` to trigger init (when ng enters runtime) +runBlock.$inject = ['$injector', '$q', '$uiRouter']; +function runBlock($injector, $q, $uiRouter) { + services.$injector = $injector; + services.$q = $q; + // The $injector is now available. + // Find any resolvables that had dependency annotation deferred + $uiRouter.stateRegistry.get() + .map(function (x) { return x.$$state().resolvables; }) + .reduce(unnestR, []) + .filter(function (x) { return x.deps === "deferred"; }) + .forEach(function (resolvable) { return resolvable.deps = $injector.annotate(resolvable.resolveFn, $injector.strictDi); }); +} +// $urlRouter service and $urlRouterProvider +var getUrlRouterProvider = function (uiRouter) { + return uiRouter.urlRouterProvider = new UrlRouterProvider(uiRouter); +}; +// $state service and $stateProvider +// $urlRouter service and $urlRouterProvider +var getStateProvider = function () { + return extend(router.stateProvider, { $get: function () { return router.stateService; } }); +}; +watchDigests.$inject = ['$rootScope']; +function watchDigests($rootScope) { + $rootScope.$watch(function () { trace.approximateDigests++; }); +} +mod_init.provider("$uiRouter", $uiRouter); +mod_rtr.provider('$urlRouter', ['$uiRouterProvider', getUrlRouterProvider]); +mod_util.provider('$urlService', getProviderFor('urlService')); +mod_util.provider('$urlMatcherFactory', ['$uiRouterProvider', function () { return router.urlMatcherFactory; }]); +mod_util.provider('$templateFactory', function () { return new TemplateFactory(); }); +mod_state.provider('$stateRegistry', getProviderFor('stateRegistry')); +mod_state.provider('$uiRouterGlobals', getProviderFor('globals')); +mod_state.provider('$transitions', getProviderFor('transitionService')); +mod_state.provider('$state', ['$uiRouterProvider', getStateProvider]); +mod_state.factory('$stateParams', ['$uiRouter', function ($uiRouter) { return $uiRouter.globals.params; }]); +mod_main.factory('$view', function () { return router.viewService; }); +mod_main.service("$trace", function () { return trace; }); +mod_main.run(watchDigests); +mod_util.run(['$urlMatcherFactory', function ($urlMatcherFactory) { }]); +mod_state.run(['$state', function ($state) { }]); +mod_rtr.run(['$urlRouter', function ($urlRouter) { }]); +mod_init.run(runBlock); +/** @hidden TODO: find a place to move this */ +var getLocals = function (ctx) { + var tokens = ctx.getTokens().filter(isString); + var tuples = tokens.map(function (key) { + var resolvable = ctx.getResolvable(key); + var waitPolicy = ctx.getPolicy(resolvable).async; + return [key, waitPolicy === 'NOWAIT' ? resolvable.promise : resolvable.data]; + }); + return tuples.reduce(applyPairs, {}); +}; + +/** + * # Angular 1 injectable services + * + * This is a list of the objects which can be injected using angular's injector. + * + * There are three different kind of injectable objects: + * + * ## **Provider** objects + * #### injectable into a `.config()` block during configtime + * + * - [[$uiRouterProvider]]: The UI-Router instance + * - [[$stateProvider]]: State registration + * - [[$transitionsProvider]]: Transition hooks + * - [[$urlServiceProvider]]: All URL related public APIs + * + * - [[$uiViewScrollProvider]]: Disable ui-router view scrolling + * - [[$urlRouterProvider]]: (deprecated) Url matching rules + * - [[$urlMatcherFactoryProvider]]: (deprecated) Url parsing config + * + * ## **Service** objects + * #### injectable globally during runtime + * + * - [[$uiRouter]]: The UI-Router instance + * - [[$trace]]: Enable transition trace/debug + * - [[$transitions]]: Transition hooks + * - [[$state]]: Imperative state related APIs + * - [[$stateRegistry]]: State registration + * - [[$urlService]]: All URL related public APIs + * - [[$uiRouterGlobals]]: Global variables + * - [[$uiViewScroll]]: Scroll an element into view + * + * - [[$stateParams]]: (deprecated) Global state param values + * - [[$urlRouter]]: (deprecated) URL synchronization + * - [[$urlMatcherFactory]]: (deprecated) URL parsing config + * + * ## **Per-Transition** objects + * + * - These kind of objects are injectable into: + * - Resolves ([[Ng1StateDeclaration.resolve]]), + * - Transition Hooks ([[TransitionService.onStart]], etc), + * - Routed Controllers ([[Ng1ViewDeclaration.controller]]) + * + * #### Different instances are injected based on the [[Transition]] + * + * - [[$transition$]]: The current Transition object + * - [[$stateParams]]: State param values for pending Transition (deprecated) + * - Any resolve data defined using [[Ng1StateDeclaration.resolve]] + * + * @ng1api + * @preferred + * @module injectables + */ /** */ +/** + * The current (or pending) State Parameters + * + * An injectable global **Service Object** which holds the state parameters for the latest **SUCCESSFUL** transition. + * + * The values are not updated until *after* a `Transition` successfully completes. + * + * **Also:** an injectable **Per-Transition Object** object which holds the pending state parameters for the pending `Transition` currently running. + * + * ### Deprecation warning: + * + * The value injected for `$stateParams` is different depending on where it is injected. + * + * - When injected into an angular service, the object injected is the global **Service Object** with the parameter values for the latest successful `Transition`. + * - When injected into transition hooks, resolves, or view controllers, the object is the **Per-Transition Object** with the parameter values for the running `Transition`. + * + * Because of these confusing details, this service is deprecated. + * + * ### Instead of using the global `$stateParams` service object, + * inject [[$uiRouterGlobals]] and use [[UIRouterGlobals.params]] + * + * ```js + * MyService.$inject = ['$uiRouterGlobals']; + * function MyService($uiRouterGlobals) { + * return { + * paramValues: function () { + * return $uiRouterGlobals.params; + * } + * } + * } + * ``` + * + * ### Instead of using the per-transition `$stateParams` object, + * inject the current `Transition` (as [[$transition$]]) and use [[Transition.params]] + * + * ```js + * MyController.$inject = ['$transition$']; + * function MyController($transition$) { + * var username = $transition$.params().username; + * // .. do something with username + * } + * ``` + * + * --- + * + * This object can be injected into other services. + * + * #### Deprecated Example: + * ```js + * SomeService.$inject = ['$http', '$stateParams']; + * function SomeService($http, $stateParams) { + * return { + * getUser: function() { + * return $http.get('/api/users/' + $stateParams.username); + * } + * } + * }; + * angular.service('SomeService', SomeService); + * ``` + * @deprecated + */ + +/** + * # Angular 1 Directives + * + * These are the directives included in UI-Router for Angular 1. + * These directives are used in templates to create viewports and link/navigate to states. + * + * @ng1api + * @preferred + * @module directives + */ /** for typedoc */ +/** @hidden */ +function parseStateRef(ref) { + var paramsOnly = ref.match(/^\s*({[^}]*})\s*$/), parsed; + if (paramsOnly) + ref = '(' + paramsOnly[1] + ')'; + parsed = ref.replace(/\n/g, " ").match(/^\s*([^(]*?)\s*(\((.*)\))?\s*$/); + if (!parsed || parsed.length !== 4) + throw new Error("Invalid state ref '" + ref + "'"); + return { state: parsed[1] || null, paramExpr: parsed[3] || null }; +} +/** @hidden */ +function stateContext(el) { + var $uiView = el.parent().inheritedData('$uiView'); + var path = parse('$cfg.path')($uiView); + return path ? tail(path).state.name : undefined; +} +/** @hidden */ +function processedDef($state, $element, def) { + var uiState = def.uiState || $state.current.name; + var uiStateOpts = extend(defaultOpts($element, $state), def.uiStateOpts || {}); + var href = $state.href(uiState, def.uiStateParams, uiStateOpts); + return { uiState: uiState, uiStateParams: def.uiStateParams, uiStateOpts: uiStateOpts, href: href }; +} +/** @hidden */ +function getTypeInfo(el) { + // SVGAElement does not use the href attribute, but rather the 'xlinkHref' attribute. + var isSvg = Object.prototype.toString.call(el.prop('href')) === '[object SVGAnimatedString]'; + var isForm = el[0].nodeName === "FORM"; + return { + attr: isForm ? "action" : (isSvg ? 'xlink:href' : 'href'), + isAnchor: el.prop("tagName").toUpperCase() === "A", + clickable: !isForm + }; +} +/** @hidden */ +function clickHook(el, $state, $timeout, type, getDef) { + return function (e) { + var button = e.which || e.button, target = getDef(); + if (!(button > 1 || e.ctrlKey || e.metaKey || e.shiftKey || el.attr('target'))) { + // HACK: This is to allow ng-clicks to be processed before the transition is initiated: + var transition = $timeout(function () { + $state.go(target.uiState, target.uiStateParams, target.uiStateOpts); + }); + e.preventDefault(); + // if the state has no URL, ignore one preventDefault from the directive. + var ignorePreventDefaultCount = type.isAnchor && !target.href ? 1 : 0; + e.preventDefault = function () { + if (ignorePreventDefaultCount-- <= 0) + $timeout.cancel(transition); + }; + } + }; +} +/** @hidden */ +function defaultOpts(el, $state) { + return { + relative: stateContext(el) || $state.$current, + inherit: true, + source: "sref" + }; +} +/** @hidden */ +function bindEvents(element, scope, hookFn, uiStateOpts) { + var events; + if (uiStateOpts) { + events = uiStateOpts.events; + } + if (!isArray(events)) { + events = ['click']; + } + var on = element.on ? 'on' : 'bind'; + for (var _i = 0, events_1 = events; _i < events_1.length; _i++) { + var event_1 = events_1[_i]; + element[on](event_1, hookFn); + } + scope.$on('$destroy', function () { + var off = element.off ? 'off' : 'unbind'; + for (var _i = 0, events_2 = events; _i < events_2.length; _i++) { + var event_2 = events_2[_i]; + element[off](event_2, hookFn); + } + }); +} +/** + * `ui-sref`: A directive for linking to a state + * + * A directive which links to a state (and optionally, parameters). + * When clicked, this directive activates the linked state with the supplied parameter values. + * + * ### Linked State + * The attribute value of the `ui-sref` is the name of the state to link to. + * + * #### Example: + * This will activate the `home` state when the link is clicked. + * ```html + * Home + * ``` + * + * ### Relative Links + * You can also use relative state paths within `ui-sref`, just like a relative path passed to `$state.go()` ([[StateService.go]]). + * You just need to be aware that the path is relative to the state that *created* the link. + * This allows a state to create a relative `ui-sref` which always targets the same destination. + * + * #### Example: + * Both these links are relative to the parent state, even when a child state is currently active. + * ```html + * child 1 state + * child 2 state + * ``` + * + * This link activates the parent state. + * ```html + * Return + * ``` + * + * ### hrefs + * If the linked state has a URL, the directive will automatically generate and + * update the `href` attribute (using the [[StateService.href]] method). + * + * #### Example: + * Assuming the `users` state has a url of `/users/` + * ```html + * Users + * ``` + * + * ### Parameter Values + * In addition to the state name, a `ui-sref` can include parameter values which are applied when activating the state. + * Param values can be provided in the `ui-sref` value after the state name, enclosed by parentheses. + * The content inside the parentheses is an expression, evaluated to the parameter values. + * + * #### Example: + * This example renders a list of links to users. + * The state's `userId` parameter value comes from each user's `user.id` property. + * ```html + *
  • + * {{ user.displayName }} + *
  • + * ``` + * + * Note: + * The parameter values expression is `$watch`ed for updates. + * + * ### Transition Options + * You can specify [[TransitionOptions]] to pass to [[StateService.go]] by using the `ui-sref-opts` attribute. + * Options are restricted to `location`, `inherit`, and `reload`. + * + * #### Example: + * ```html + * Home + * ``` + * + * ### Other DOM Events + * + * You can also customize which DOM events to respond to (instead of `click`) by + * providing an `events` array in the `ui-sref-opts` attribute. + * + * #### Example: + * ```html + * + * ``` + * + * ### Highlighting the active link + * This directive can be used in conjunction with [[uiSrefActive]] to highlight the active link. + * + * ### Examples + * If you have the following template: + * + * ```html + * Home + * About + * Next page + * + * + * ``` + * + * Then (assuming the current state is `contacts`) the rendered html including hrefs would be: + * + * ```html + * Home + * About + * Next page + * + *
      + *
    • + * Joe + *
    • + *
    • + * Alice + *
    • + *
    • + * Bob + *
    • + *
    + * + * Home + * ``` + * + * ### Notes + * + * - You can use `ui-sref` to change **only the parameter values** by omitting the state name and parentheses. + * #### Example: + * Sets the `lang` parameter to `en` and remains on the same state. + * + * ```html + * English + * ``` + * + * - A middle-click, right-click, or ctrl-click is handled (natively) by the browser to open the href in a new window, for example. + * + * - Unlike the parameter values expression, the state name is not `$watch`ed (for performance reasons). + * If you need to dynamically update the state being linked to, use the fully dynamic [[uiState]] directive. + */ +var uiSref; +uiSref = ['$uiRouter', '$timeout', + function $StateRefDirective($uiRouter, $timeout) { + var $state = $uiRouter.stateService; + return { + restrict: 'A', + require: ['?^uiSrefActive', '?^uiSrefActiveEq'], + link: function (scope, element, attrs, uiSrefActive) { + var type = getTypeInfo(element); + var active = uiSrefActive[1] || uiSrefActive[0]; + var unlinkInfoFn = null; + var hookFn; + var rawDef = {}; + var getDef = function () { return processedDef($state, element, rawDef); }; + var ref = parseStateRef(attrs.uiSref); + rawDef.uiState = ref.state; + rawDef.uiStateOpts = attrs.uiSrefOpts ? scope.$eval(attrs.uiSrefOpts) : {}; + function update() { + var def = getDef(); + if (unlinkInfoFn) + unlinkInfoFn(); + if (active) + unlinkInfoFn = active.$$addStateInfo(def.uiState, def.uiStateParams); + if (def.href != null) + attrs.$set(type.attr, def.href); + } + if (ref.paramExpr) { + scope.$watch(ref.paramExpr, function (val) { + rawDef.uiStateParams = extend({}, val); + update(); + }, true); + rawDef.uiStateParams = extend({}, scope.$eval(ref.paramExpr)); + } + update(); + scope.$on('$destroy', $uiRouter.stateRegistry.onStatesChanged(update)); + scope.$on('$destroy', $uiRouter.transitionService.onSuccess({}, update)); + if (!type.clickable) + return; + hookFn = clickHook(element, $state, $timeout, type, getDef); + bindEvents(element, scope, hookFn, rawDef.uiStateOpts); + } + }; + }]; +/** + * `ui-state`: A fully dynamic directive for linking to a state + * + * A directive which links to a state (and optionally, parameters). + * When clicked, this directive activates the linked state with the supplied parameter values. + * + * **This directive is very similar to [[uiSref]], but it `$observe`s and `$watch`es/evaluates all its inputs.** + * + * A directive which links to a state (and optionally, parameters). + * When clicked, this directive activates the linked state with the supplied parameter values. + * + * ### Linked State + * The attribute value of `ui-state` is an expression which is `$watch`ed and evaluated as the state to link to. + * **This is in contrast with `ui-sref`, which takes a state name as a string literal.** + * + * #### Example: + * Create a list of links. + * ```html + *
  • + * {{ link.displayName }} + *
  • + * ``` + * + * ### Relative Links + * If the expression evaluates to a relative path, it is processed like [[uiSref]]. + * You just need to be aware that the path is relative to the state that *created* the link. + * This allows a state to create relative `ui-state` which always targets the same destination. + * + * ### hrefs + * If the linked state has a URL, the directive will automatically generate and + * update the `href` attribute (using the [[StateService.href]] method). + * + * ### Parameter Values + * In addition to the state name expression, a `ui-state` can include parameter values which are applied when activating the state. + * Param values should be provided using the `ui-state-params` attribute. + * The `ui-state-params` attribute value is `$watch`ed and evaluated as an expression. + * + * #### Example: + * This example renders a list of links with param values. + * The state's `userId` parameter value comes from each user's `user.id` property. + * ```html + *
  • + * {{ link.displayName }} + *
  • + * ``` + * + * ### Transition Options + * You can specify [[TransitionOptions]] to pass to [[StateService.go]] by using the `ui-state-opts` attribute. + * Options are restricted to `location`, `inherit`, and `reload`. + * The value of the `ui-state-opts` is `$watch`ed and evaluated as an expression. + * + * #### Example: + * ```html + * Home + * ``` + * + * ### Other DOM Events + * + * You can also customize which DOM events to respond to (instead of `click`) by + * providing an `events` array in the `ui-state-opts` attribute. + * + * #### Example: + * ```html + * + * ``` + * + * ### Highlighting the active link + * This directive can be used in conjunction with [[uiSrefActive]] to highlight the active link. + * + * ### Notes + * + * - You can use `ui-params` to change **only the parameter values** by omitting the state name and supplying only `ui-state-params`. + * However, it might be simpler to use [[uiSref]] parameter-only links. + * + * #### Example: + * Sets the `lang` parameter to `en` and remains on the same state. + * + * ```html + * English + * ``` + * + * - A middle-click, right-click, or ctrl-click is handled (natively) by the browser to open the href in a new window, for example. + * ``` + */ +var uiState; +uiState = ['$uiRouter', '$timeout', + function $StateRefDynamicDirective($uiRouter, $timeout) { + var $state = $uiRouter.stateService; + return { + restrict: 'A', + require: ['?^uiSrefActive', '?^uiSrefActiveEq'], + link: function (scope, element, attrs, uiSrefActive) { + var type = getTypeInfo(element); + var active = uiSrefActive[1] || uiSrefActive[0]; + var unlinkInfoFn = null; + var hookFn; + var rawDef = {}; + var getDef = function () { return processedDef($state, element, rawDef); }; + var inputAttrs = ['uiState', 'uiStateParams', 'uiStateOpts']; + var watchDeregFns = inputAttrs.reduce(function (acc, attr) { return (acc[attr] = noop$1, acc); }, {}); + function update() { + var def = getDef(); + if (unlinkInfoFn) + unlinkInfoFn(); + if (active) + unlinkInfoFn = active.$$addStateInfo(def.uiState, def.uiStateParams); + if (def.href != null) + attrs.$set(type.attr, def.href); + } + inputAttrs.forEach(function (field) { + rawDef[field] = attrs[field] ? scope.$eval(attrs[field]) : null; + attrs.$observe(field, function (expr) { + watchDeregFns[field](); + watchDeregFns[field] = scope.$watch(expr, function (newval) { + rawDef[field] = newval; + update(); + }, true); + }); + }); + update(); + scope.$on('$destroy', $uiRouter.stateRegistry.onStatesChanged(update)); + scope.$on('$destroy', $uiRouter.transitionService.onSuccess({}, update)); + if (!type.clickable) + return; + hookFn = clickHook(element, $state, $timeout, type, getDef); + bindEvents(element, scope, hookFn, rawDef.uiStateOpts); + } + }; + }]; +/** + * `ui-sref-active` and `ui-sref-active-eq`: A directive that adds a CSS class when a `ui-sref` is active + * + * A directive working alongside [[uiSref]] and [[uiState]] to add classes to an element when the + * related directive's state is active (and remove them when it is inactive). + * + * The primary use-case is to highlight the active link in navigation menus, + * distinguishing it from the inactive menu items. + * + * ### Linking to a `ui-sref` or `ui-state` + * `ui-sref-active` can live on the same element as `ui-sref`/`ui-state`, or it can be on a parent element. + * If a `ui-sref-active` is a parent to more than one `ui-sref`/`ui-state`, it will apply the CSS class when **any of the links are active**. + * + * ### Matching + * + * The `ui-sref-active` directive applies the CSS class when the `ui-sref`/`ui-state`'s target state **or any child state is active**. + * This is a "fuzzy match" which uses [[StateService.includes]]. + * + * The `ui-sref-active-eq` directive applies the CSS class when the `ui-sref`/`ui-state`'s target state is directly active (not when child states are active). + * This is an "exact match" which uses [[StateService.is]]. + * + * ### Parameter values + * If the `ui-sref`/`ui-state` includes parameter values, the current parameter values must match the link's values for the link to be highlighted. + * This allows a list of links to the same state with different parameters to be rendered, and the correct one highlighted. + * + * #### Example: + * ```html + *
  • + * {{ user.lastName }} + *
  • + * ``` + * + * ### Examples + * + * Given the following template: + * #### Example: + * ```html + * + * ``` + * + * When the app state is `app.user` (or any child state), + * and contains the state parameter "user" with value "bilbobaggins", + * the resulting HTML will appear as (note the 'active' class): + * + * ```html + * + * ``` + * + * ### Glob mode + * + * It is possible to pass `ui-sref-active` an expression that evaluates to an object. + * The objects keys represent active class names and values represent the respective state names/globs. + * `ui-sref-active` will match if the current active state **includes** any of + * the specified state names/globs, even the abstract ones. + * + * #### Example: + * Given the following template, with "admin" being an abstract state: + * ```html + *
    + * Roles + *
    + * ``` + * + * When the current state is "admin.roles" the "active" class will be applied to both the
    and elements. + * It is important to note that the state names/globs passed to `ui-sref-active` override any state provided by a linked `ui-sref`. + * + * ### Notes: + * + * - The class name is interpolated **once** during the directives link time (any further changes to the + * interpolated value are ignored). + * + * - Multiple classes may be specified in a space-separated format: `ui-sref-active='class1 class2 class3'` + */ +var uiSrefActive; +uiSrefActive = ['$state', '$stateParams', '$interpolate', '$uiRouter', + function $StateRefActiveDirective($state, $stateParams, $interpolate, $uiRouter) { + return { + restrict: "A", + controller: ['$scope', '$element', '$attrs', + function ($scope, $element, $attrs) { + var states = [], activeEqClass, uiSrefActive; + // There probably isn't much point in $observing this + // uiSrefActive and uiSrefActiveEq share the same directive object with some + // slight difference in logic routing + activeEqClass = $interpolate($attrs.uiSrefActiveEq || '', false)($scope); + try { + uiSrefActive = $scope.$eval($attrs.uiSrefActive); + } + catch (e) { + // Do nothing. uiSrefActive is not a valid expression. + // Fall back to using $interpolate below + } + uiSrefActive = uiSrefActive || $interpolate($attrs.uiSrefActive || '', false)($scope); + if (isObject(uiSrefActive)) { + forEach(uiSrefActive, function (stateOrName, activeClass) { + if (isString(stateOrName)) { + var ref = parseStateRef(stateOrName); + addState(ref.state, $scope.$eval(ref.paramExpr), activeClass); + } + }); + } + // Allow uiSref to communicate with uiSrefActive[Equals] + this.$$addStateInfo = function (newState, newParams) { + // we already got an explicit state provided by ui-sref-active, so we + // shadow the one that comes from ui-sref + if (isObject(uiSrefActive) && states.length > 0) { + return; + } + var deregister = addState(newState, newParams, uiSrefActive); + update(); + return deregister; + }; + function updateAfterTransition(trans) { + trans.promise.then(update, noop$1); + } + $scope.$on('$stateChangeSuccess', update); + $scope.$on('$destroy', $uiRouter.transitionService.onStart({}, updateAfterTransition)); + if ($uiRouter.globals.transition) { + updateAfterTransition($uiRouter.globals.transition); + } + function addState(stateName, stateParams, activeClass) { + var state = $state.get(stateName, stateContext($element)); + var stateInfo = { + state: state || { name: stateName }, + params: stateParams, + activeClass: activeClass + }; + states.push(stateInfo); + return function removeState() { + removeFrom(states)(stateInfo); + }; + } + // Update route state + function update() { + var splitClasses = function (str) { + return str.split(/\s/).filter(identity); + }; + var getClasses = function (stateList) { + return stateList.map(function (x) { return x.activeClass; }).map(splitClasses).reduce(unnestR, []); + }; + var allClasses = getClasses(states).concat(splitClasses(activeEqClass)).reduce(uniqR, []); + var fuzzyClasses = getClasses(states.filter(function (x) { return $state.includes(x.state.name, x.params); })); + var exactlyMatchesAny = !!states.filter(function (x) { return $state.is(x.state.name, x.params); }).length; + var exactClasses = exactlyMatchesAny ? splitClasses(activeEqClass) : []; + var addClasses = fuzzyClasses.concat(exactClasses).reduce(uniqR, []); + var removeClasses = allClasses.filter(function (cls) { return !inArray(addClasses, cls); }); + $scope.$evalAsync(function () { + addClasses.forEach(function (className) { return $element.addClass(className); }); + removeClasses.forEach(function (className) { return $element.removeClass(className); }); + }); + } + update(); + }] + }; + }]; +ng.module('ui.router.state') + .directive('uiSref', uiSref) + .directive('uiSrefActive', uiSrefActive) + .directive('uiSrefActiveEq', uiSrefActive) + .directive('uiState', uiState); + +/** @module ng1 */ /** for typedoc */ +/** + * `isState` Filter: truthy if the current state is the parameter + * + * Translates to [[StateService.is]] `$state.is("stateName")`. + * + * #### Example: + * ```html + *
    show if state is 'stateName'
    + * ``` + */ +$IsStateFilter.$inject = ['$state']; +function $IsStateFilter($state) { + var isFilter = function (state, params, options) { + return $state.is(state, params, options); + }; + isFilter.$stateful = true; + return isFilter; +} +/** + * `includedByState` Filter: truthy if the current state includes the parameter + * + * Translates to [[StateService.includes]]` $state.is("fullOrPartialStateName")`. + * + * #### Example: + * ```html + *
    show if state includes 'fullOrPartialStateName'
    + * ``` + */ +$IncludedByStateFilter.$inject = ['$state']; +function $IncludedByStateFilter($state) { + var includesFilter = function (state, params, options) { + return $state.includes(state, params, options); + }; + includesFilter.$stateful = true; + return includesFilter; +} +ng.module('ui.router.state') + .filter('isState', $IsStateFilter) + .filter('includedByState', $IncludedByStateFilter); + +/** + * @ng1api + * @module directives + */ /** for typedoc */ +/** + * `ui-view`: A viewport directive which is filled in by a view from the active state. + * + * ### Attributes + * + * - `name`: (Optional) A view name. + * The name should be unique amongst the other views in the same state. + * You can have views of the same name that live in different states. + * The ui-view can be targeted in a View using the name ([[Ng1StateDeclaration.views]]). + * + * - `autoscroll`: an expression. When it evaluates to true, the `ui-view` will be scrolled into view when it is activated. + * Uses [[$uiViewScroll]] to do the scrolling. + * + * - `onload`: Expression to evaluate whenever the view updates. + * + * #### Example: + * A view can be unnamed or named. + * ```html + * + *
    + * + * + *
    + * + * + * + * ``` + * + * You can only have one unnamed view within any template (or root html). If you are only using a + * single view and it is unnamed then you can populate it like so: + * + * ```html + *
    + * $stateProvider.state("home", { + * template: "

    HELLO!

    " + * }) + * ``` + * + * The above is a convenient shortcut equivalent to specifying your view explicitly with the + * [[Ng1StateDeclaration.views]] config property, by name, in this case an empty name: + * + * ```js + * $stateProvider.state("home", { + * views: { + * "": { + * template: "

    HELLO!

    " + * } + * } + * }) + * ``` + * + * But typically you'll only use the views property if you name your view or have more than one view + * in the same template. There's not really a compelling reason to name a view if its the only one, + * but you could if you wanted, like so: + * + * ```html + *
    + * ``` + * + * ```js + * $stateProvider.state("home", { + * views: { + * "main": { + * template: "

    HELLO!

    " + * } + * } + * }) + * ``` + * + * Really though, you'll use views to set up multiple views: + * + * ```html + *
    + *
    + *
    + * ``` + * + * ```js + * $stateProvider.state("home", { + * views: { + * "": { + * template: "

    HELLO!

    " + * }, + * "chart": { + * template: "" + * }, + * "data": { + * template: "" + * } + * } + * }) + * ``` + * + * #### Examples for `autoscroll`: + * ```html + * + * + * + * + * + * + * + * ``` + * + * Resolve data: + * + * The resolved data from the state's `resolve` block is placed on the scope as `$resolve` (this + * can be customized using [[Ng1ViewDeclaration.resolveAs]]). This can be then accessed from the template. + * + * Note that when `controllerAs` is being used, `$resolve` is set on the controller instance *after* the + * controller is instantiated. The `$onInit()` hook can be used to perform initialization code which + * depends on `$resolve` data. + * + * #### Example: + * ```js + * $stateProvider.state('home', { + * template: '', + * resolve: { + * user: function(UserService) { return UserService.fetchUser(); } + * } + * }); + * ``` + */ +var uiView; +uiView = ['$view', '$animate', '$uiViewScroll', '$interpolate', '$q', + function $ViewDirective($view, $animate, $uiViewScroll, $interpolate, $q) { + function getRenderer(attrs, scope) { + return { + enter: function (element, target, cb) { + if (ng.version.minor > 2) { + $animate.enter(element, null, target).then(cb); + } + else { + $animate.enter(element, null, target, cb); + } + }, + leave: function (element, cb) { + if (ng.version.minor > 2) { + $animate.leave(element).then(cb); + } + else { + $animate.leave(element, cb); + } + } + }; + } + function configsEqual(config1, config2) { + return config1 === config2; + } + var rootData = { + $cfg: { viewDecl: { $context: $view._pluginapi._rootViewContext() } }, + $uiView: {} + }; + var directive = { + count: 0, + restrict: 'ECA', + terminal: true, + priority: 400, + transclude: 'element', + compile: function (tElement, tAttrs, $transclude) { + return function (scope, $element, attrs) { + var previousEl, currentEl, currentScope, unregister, onloadExp = attrs['onload'] || '', autoScrollExp = attrs['autoscroll'], renderer = getRenderer(attrs, scope), viewConfig = undefined, inherited = $element.inheritedData('$uiView') || rootData, name = $interpolate(attrs['uiView'] || attrs['name'] || '')(scope) || '$default'; + var activeUIView = { + $type: 'ng1', + id: directive.count++, + name: name, + fqn: inherited.$uiView.fqn ? inherited.$uiView.fqn + "." + name : name, + config: null, + configUpdated: configUpdatedCallback, + get creationContext() { + var fromParentTagConfig = parse('$cfg.viewDecl.$context')(inherited); + // Allow + // See https://github.com/angular-ui/ui-router/issues/3355 + var fromParentTag = parse('$uiView.creationContext')(inherited); + return fromParentTagConfig || fromParentTag; + } + }; + trace.traceUIViewEvent("Linking", activeUIView); + function configUpdatedCallback(config) { + if (config && !(config instanceof Ng1ViewConfig)) + return; + if (configsEqual(viewConfig, config)) + return; + trace.traceUIViewConfigUpdated(activeUIView, config && config.viewDecl && config.viewDecl.$context); + viewConfig = config; + updateView(config); + } + $element.data('$uiView', { $uiView: activeUIView }); + updateView(); + unregister = $view.registerUIView(activeUIView); + scope.$on("$destroy", function () { + trace.traceUIViewEvent("Destroying/Unregistering", activeUIView); + unregister(); + }); + function cleanupLastView() { + if (previousEl) { + trace.traceUIViewEvent("Removing (previous) el", previousEl.data('$uiView')); + previousEl.remove(); + previousEl = null; + } + if (currentScope) { + trace.traceUIViewEvent("Destroying scope", activeUIView); + currentScope.$destroy(); + currentScope = null; + } + if (currentEl) { + var _viewData_1 = currentEl.data('$uiViewAnim'); + trace.traceUIViewEvent("Animate out", _viewData_1); + renderer.leave(currentEl, function () { + _viewData_1.$$animLeave.resolve(); + previousEl = null; + }); + previousEl = currentEl; + currentEl = null; + } + } + function updateView(config) { + var newScope = scope.$new(); + var animEnter = $q.defer(), animLeave = $q.defer(); + var $uiViewData = { + $cfg: config, + $uiView: activeUIView, + }; + var $uiViewAnim = { + $animEnter: animEnter.promise, + $animLeave: animLeave.promise, + $$animLeave: animLeave + }; + /** + * @ngdoc event + * @name ui.router.state.directive:ui-view#$viewContentLoading + * @eventOf ui.router.state.directive:ui-view + * @eventType emits on ui-view directive scope + * @description + * + * Fired once the view **begins loading**, *before* the DOM is rendered. + * + * @param {Object} event Event object. + * @param {string} viewName Name of the view. + */ + newScope.$emit('$viewContentLoading', name); + var cloned = $transclude(newScope, function (clone) { + clone.data('$uiViewAnim', $uiViewAnim); + clone.data('$uiView', $uiViewData); + renderer.enter(clone, $element, function onUIViewEnter() { + animEnter.resolve(); + if (currentScope) + currentScope.$emit('$viewContentAnimationEnded'); + if (isDefined(autoScrollExp) && !autoScrollExp || scope.$eval(autoScrollExp)) { + $uiViewScroll(clone); + } + }); + cleanupLastView(); + }); + currentEl = cloned; + currentScope = newScope; + /** + * @ngdoc event + * @name ui.router.state.directive:ui-view#$viewContentLoaded + * @eventOf ui.router.state.directive:ui-view + * @eventType emits on ui-view directive scope + * @description * + * Fired once the view is **loaded**, *after* the DOM is rendered. + * + * @param {Object} event Event object. + */ + currentScope.$emit('$viewContentLoaded', config || viewConfig); + currentScope.$eval(onloadExp); + } + }; + } + }; + return directive; + }]; +$ViewDirectiveFill.$inject = ['$compile', '$controller', '$transitions', '$view', '$q', '$timeout']; +/** @hidden */ +function $ViewDirectiveFill($compile, $controller, $transitions, $view, $q, $timeout) { + var getControllerAs = parse('viewDecl.controllerAs'); + var getResolveAs = parse('viewDecl.resolveAs'); + return { + restrict: 'ECA', + priority: -400, + compile: function (tElement) { + var initial = tElement.html(); + tElement.empty(); + return function (scope, $element) { + var data = $element.data('$uiView'); + if (!data) { + $element.html(initial); + $compile($element.contents())(scope); + return; + } + var cfg = data.$cfg || { viewDecl: {}, getTemplate: ng_from_import.noop }; + var resolveCtx = cfg.path && new ResolveContext(cfg.path); + $element.html(cfg.getTemplate($element, resolveCtx) || initial); + trace.traceUIViewFill(data.$uiView, $element.html()); + var link = $compile($element.contents()); + var controller = cfg.controller; + var controllerAs = getControllerAs(cfg); + var resolveAs = getResolveAs(cfg); + var locals = resolveCtx && getLocals(resolveCtx); + scope[resolveAs] = locals; + if (controller) { + var controllerInstance = $controller(controller, extend({}, locals, { $scope: scope, $element: $element })); + if (controllerAs) { + scope[controllerAs] = controllerInstance; + scope[controllerAs][resolveAs] = locals; + } + // TODO: Use $view service as a central point for registering component-level hooks + // Then, when a component is created, tell the $view service, so it can invoke hooks + // $view.componentLoaded(controllerInstance, { $scope: scope, $element: $element }); + // scope.$on('$destroy', () => $view.componentUnloaded(controllerInstance, { $scope: scope, $element: $element })); + $element.data('$ngControllerController', controllerInstance); + $element.children().data('$ngControllerController', controllerInstance); + registerControllerCallbacks($q, $transitions, controllerInstance, scope, cfg); + } + // Wait for the component to appear in the DOM + if (isString(cfg.viewDecl.component)) { + var cmp_1 = cfg.viewDecl.component; + var kebobName = kebobString(cmp_1); + var tagRegexp_1 = new RegExp("^(x-|data-)?" + kebobName + "$", "i"); + var getComponentController = function () { + var directiveEl = [].slice.call($element[0].children) + .filter(function (el) { return el && el.tagName && tagRegexp_1.exec(el.tagName); }); + return directiveEl && ng.element(directiveEl).data("$" + cmp_1 + "Controller"); + }; + var deregisterWatch_1 = scope.$watch(getComponentController, function (ctrlInstance) { + if (!ctrlInstance) + return; + registerControllerCallbacks($q, $transitions, ctrlInstance, scope, cfg); + deregisterWatch_1(); + }); + } + link(scope); + }; + } + }; +} +/** @hidden */ +var hasComponentImpl = typeof ng.module('ui.router')['component'] === 'function'; +/** @hidden incrementing id */ +var _uiCanExitId = 0; +/** @hidden TODO: move these callbacks to $view and/or `/hooks/components.ts` or something */ +function registerControllerCallbacks($q, $transitions, controllerInstance, $scope, cfg) { + // Call $onInit() ASAP + if (isFunction(controllerInstance.$onInit) && !(cfg.viewDecl.component && hasComponentImpl)) { + controllerInstance.$onInit(); + } + var viewState = tail(cfg.path).state.self; + var hookOptions = { bind: controllerInstance }; + // Add component-level hook for onParamsChange + if (isFunction(controllerInstance.uiOnParamsChanged)) { + var resolveContext = new ResolveContext(cfg.path); + var viewCreationTrans_1 = resolveContext.getResolvable('$transition$').data; + // Fire callback on any successful transition + var paramsUpdated = function ($transition$) { + // Exit early if the $transition$ is the same as the view was created within. + // Exit early if the $transition$ will exit the state the view is for. + if ($transition$ === viewCreationTrans_1 || $transition$.exiting().indexOf(viewState) !== -1) + return; + var toParams = $transition$.params("to"); + var fromParams = $transition$.params("from"); + var toSchema = $transition$.treeChanges().to.map(function (node) { return node.paramSchema; }).reduce(unnestR, []); + var fromSchema = $transition$.treeChanges().from.map(function (node) { return node.paramSchema; }).reduce(unnestR, []); + // Find the to params that have different values than the from params + var changedToParams = toSchema.filter(function (param) { + var idx = fromSchema.indexOf(param); + return idx === -1 || !fromSchema[idx].type.equals(toParams[param.id], fromParams[param.id]); + }); + // Only trigger callback if a to param has changed or is new + if (changedToParams.length) { + var changedKeys_1 = changedToParams.map(function (x) { return x.id; }); + // Filter the params to only changed/new to params. `$transition$.params()` may be used to get all params. + var newValues = filter(toParams, function (val, key) { return changedKeys_1.indexOf(key) !== -1; }); + controllerInstance.uiOnParamsChanged(newValues, $transition$); + } + }; + $scope.$on('$destroy', $transitions.onSuccess({}, paramsUpdated, hookOptions)); + } + // Add component-level hook for uiCanExit + if (isFunction(controllerInstance.uiCanExit)) { + var id_1 = _uiCanExitId++; + var cacheProp_1 = '_uiCanExitIds'; + // Returns true if a redirect transition already answered truthy + var prevTruthyAnswer_1 = function (trans) { + return !!trans && (trans[cacheProp_1] && trans[cacheProp_1][id_1] === true || prevTruthyAnswer_1(trans.redirectedFrom())); + }; + // If a user answered yes, but the transition was later redirected, don't also ask for the new redirect transition + var wrappedHook = function (trans) { + var promise, ids = trans[cacheProp_1] = trans[cacheProp_1] || {}; + if (!prevTruthyAnswer_1(trans)) { + promise = $q.when(controllerInstance.uiCanExit(trans)); + promise.then(function (val) { return ids[id_1] = (val !== false); }); + } + return promise; + }; + var criteria = { exiting: viewState.name }; + $scope.$on('$destroy', $transitions.onBefore(criteria, wrappedHook, hookOptions)); + } +} +ng.module('ui.router.state').directive('uiView', uiView); +ng.module('ui.router.state').directive('uiView', $ViewDirectiveFill); + +/** @module ng1 */ /** */ +/** @hidden */ +function $ViewScrollProvider() { + var useAnchorScroll = false; + this.useAnchorScroll = function () { + useAnchorScroll = true; + }; + this.$get = ['$anchorScroll', '$timeout', function ($anchorScroll, $timeout) { + if (useAnchorScroll) { + return $anchorScroll; + } + return function ($element) { + return $timeout(function () { + $element[0].scrollIntoView(); + }, 0, false); + }; + }]; +} +ng.module('ui.router.state').provider('$uiViewScroll', $ViewScrollProvider); + +/** + * Main entry point for angular 1.x build + * @module ng1 + */ /** */ +var index = "ui.router"; + +exports['default'] = index; +exports.core = index$1; +exports.watchDigests = watchDigests; +exports.getLocals = getLocals; +exports.getNg1ViewConfigFactory = getNg1ViewConfigFactory; +exports.ng1ViewsBuilder = ng1ViewsBuilder; +exports.Ng1ViewConfig = Ng1ViewConfig; +exports.StateProvider = StateProvider; +exports.UrlRouterProvider = UrlRouterProvider; +exports.root = root; +exports.fromJson = fromJson; +exports.toJson = toJson; +exports.forEach = forEach; +exports.extend = extend; +exports.equals = equals; +exports.identity = identity; +exports.noop = noop$1; +exports.createProxyFunctions = createProxyFunctions; +exports.inherit = inherit; +exports.inArray = inArray; +exports._inArray = _inArray; +exports.removeFrom = removeFrom; +exports._removeFrom = _removeFrom; +exports.pushTo = pushTo; +exports._pushTo = _pushTo; +exports.deregAll = deregAll; +exports.defaults = defaults; +exports.mergeR = mergeR; +exports.ancestors = ancestors; +exports.pick = pick; +exports.omit = omit; +exports.pluck = pluck; +exports.filter = filter; +exports.find = find; +exports.mapObj = mapObj; +exports.map = map; +exports.values = values; +exports.allTrueR = allTrueR; +exports.anyTrueR = anyTrueR; +exports.unnestR = unnestR; +exports.flattenR = flattenR; +exports.pushR = pushR; +exports.uniqR = uniqR; +exports.unnest = unnest; +exports.flatten = flatten; +exports.assertPredicate = assertPredicate; +exports.assertMap = assertMap; +exports.assertFn = assertFn; +exports.pairs = pairs; +exports.arrayTuples = arrayTuples; +exports.applyPairs = applyPairs; +exports.tail = tail; +exports.copy = copy; +exports._extend = _extend; +exports.silenceUncaughtInPromise = silenceUncaughtInPromise; +exports.silentRejection = silentRejection; +exports.notImplemented = notImplemented; +exports.services = services; +exports.Glob = Glob; +exports.curry = curry; +exports.compose = compose; +exports.pipe = pipe; +exports.prop = prop; +exports.propEq = propEq; +exports.parse = parse; +exports.not = not; +exports.and = and; +exports.or = or; +exports.all = all; +exports.any = any; +exports.is = is; +exports.eq = eq; +exports.val = val; +exports.invoke = invoke; +exports.pattern = pattern; +exports.isUndefined = isUndefined; +exports.isDefined = isDefined; +exports.isNull = isNull; +exports.isNullOrUndefined = isNullOrUndefined; +exports.isFunction = isFunction; +exports.isNumber = isNumber; +exports.isString = isString; +exports.isObject = isObject; +exports.isArray = isArray; +exports.isDate = isDate; +exports.isRegExp = isRegExp; +exports.isState = isState; +exports.isInjectable = isInjectable; +exports.isPromise = isPromise; +exports.Queue = Queue; +exports.maxLength = maxLength; +exports.padString = padString; +exports.kebobString = kebobString; +exports.functionToString = functionToString; +exports.fnToString = fnToString; +exports.stringify = stringify; +exports.beforeAfterSubstr = beforeAfterSubstr; +exports.hostRegex = hostRegex; +exports.stripFile = stripFile; +exports.splitHash = splitHash; +exports.splitQuery = splitQuery; +exports.splitEqual = splitEqual; +exports.trimHashVal = trimHashVal; +exports.splitOnDelim = splitOnDelim; +exports.joinNeighborsR = joinNeighborsR; +exports.Trace = Trace; +exports.trace = trace; +exports.Param = Param; +exports.ParamTypes = ParamTypes; +exports.StateParams = StateParams; +exports.ParamType = ParamType; +exports.PathNode = PathNode; +exports.PathUtils = PathUtils; +exports.resolvePolicies = resolvePolicies; +exports.defaultResolvePolicy = defaultResolvePolicy; +exports.Resolvable = Resolvable; +exports.NATIVE_INJECTOR_TOKEN = NATIVE_INJECTOR_TOKEN; +exports.ResolveContext = ResolveContext; +exports.resolvablesBuilder = resolvablesBuilder; +exports.StateBuilder = StateBuilder; +exports.StateObject = StateObject; +exports.StateMatcher = StateMatcher; +exports.StateQueueManager = StateQueueManager; +exports.StateRegistry = StateRegistry; +exports.StateService = StateService; +exports.TargetState = TargetState; +exports.HookBuilder = HookBuilder; +exports.matchState = matchState; +exports.RegisteredHook = RegisteredHook; +exports.makeEvent = makeEvent; +exports.Rejection = Rejection; +exports.Transition = Transition; +exports.TransitionHook = TransitionHook; +exports.TransitionEventType = TransitionEventType; +exports.defaultTransOpts = defaultTransOpts; +exports.TransitionService = TransitionService; +exports.UrlMatcher = UrlMatcher; +exports.UrlMatcherFactory = UrlMatcherFactory; +exports.UrlRouter = UrlRouter; +exports.UrlRuleFactory = UrlRuleFactory; +exports.BaseUrlRule = BaseUrlRule; +exports.UrlService = UrlService; +exports.ViewService = ViewService; +exports.UIRouterGlobals = UIRouterGlobals; +exports.UIRouter = UIRouter; +exports.$q = $q; +exports.$injector = $injector; +exports.BaseLocationServices = BaseLocationServices; +exports.HashLocationService = HashLocationService; +exports.MemoryLocationService = MemoryLocationService; +exports.PushStateLocationService = PushStateLocationService; +exports.MemoryLocationConfig = MemoryLocationConfig; +exports.BrowserLocationConfig = BrowserLocationConfig; +exports.keyValsToObjectR = keyValsToObjectR; +exports.getParams = getParams; +exports.parseUrl = parseUrl$1; +exports.buildUrl = buildUrl; +exports.locationPluginFactory = locationPluginFactory; +exports.servicesPlugin = servicesPlugin; +exports.hashLocationPlugin = hashLocationPlugin; +exports.pushStateLocationPlugin = pushStateLocationPlugin; +exports.memoryLocationPlugin = memoryLocationPlugin; +exports.UIRouterPluginBase = UIRouterPluginBase; + +Object.defineProperty(exports, '__esModule', { value: true }); + +}))); +//# sourceMappingURL=angular-ui-router.js.map diff --git a/feature-toggles/main/main.html b/feature-toggles/main/main.html new file mode 100644 index 000000000..d95a86e36 --- /dev/null +++ b/feature-toggles/main/main.html @@ -0,0 +1,3 @@ +
    +
    +
    \ No newline at end of file diff --git a/feature-toggles/main/mainControler.js b/feature-toggles/main/mainControler.js new file mode 100644 index 000000000..c94cb7b40 --- /dev/null +++ b/feature-toggles/main/mainControler.js @@ -0,0 +1,10 @@ +(function() { + 'use strict'; + const app = angular.module('app'); + + app.controller('MainController', function() { + const mainCtrl = this; + + + }); +})(); \ No newline at end of file diff --git a/feature-toggles/manage/manage-toggles.html b/feature-toggles/manage/manage-toggles.html new file mode 100644 index 000000000..b71f4b9a8 --- /dev/null +++ b/feature-toggles/manage/manage-toggles.html @@ -0,0 +1 @@ +
    fkhjsdlhfkjlsdakjflhskdhflkha
    \ No newline at end of file diff --git a/feature-toggles/manage/manageToggles.js b/feature-toggles/manage/manageToggles.js new file mode 100644 index 000000000..b286ed157 --- /dev/null +++ b/feature-toggles/manage/manageToggles.js @@ -0,0 +1,9 @@ +(function() { + 'use strict'; + + const app = angular.module('app'); + + app.controller('ManageTogglesController', function() { + const ManageTogglesCtrl = this; + }); +})(); \ No newline at end of file From 49a84efe010e3ea74f613737e0ca3c69c4ac477f Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Tue, 8 Jan 2019 15:34:18 -0300 Subject: [PATCH 029/606] Create manage toggles layout --- feature-toggles/app.js | 14 ++++++------ feature-toggles/images/logo.png | Bin 0 -> 18003 bytes feature-toggles/index.html | 4 ++-- feature-toggles/main/main.html | 3 --- feature-toggles/manage/manage-toggles.html | 20 +++++++++++++++++- feature-toggles/signin/signin.html | 3 +++ .../signinController.js} | 4 ++-- 7 files changed, 34 insertions(+), 14 deletions(-) create mode 100644 feature-toggles/images/logo.png delete mode 100644 feature-toggles/main/main.html create mode 100644 feature-toggles/signin/signin.html rename feature-toggles/{main/mainControler.js => signin/signinController.js} (50%) diff --git a/feature-toggles/app.js b/feature-toggles/app.js index 4d6b3020b..25609e70c 100644 --- a/feature-toggles/app.js +++ b/feature-toggles/app.js @@ -20,19 +20,20 @@ $urlMatcherFactoryProvider.caseInsensitive(true); $stateProvider - .state("main", { - abstract: true, + .state("signin", { + url: "/signin", views: { main: { - templateUrl: "app/main/main.html", - controller: "MainController as mainCtrl" + templateUrl: "app/signin/signin.html", + controller: "SigninController as signinCtrl" } } }).state("manage-features", { url: "/", views: { - content: { - templateUrl: "app/manage/manage-toggles.html" + main: { + templateUrl: "app/manage/manage-toggles.html", + constroller: "ManageTogglesController as ManageTogglesCtrl" } } }); @@ -40,5 +41,6 @@ $urlRouterProvider.otherwise("/"); $locationProvider.html5Mode(true); + }); })(); \ No newline at end of file diff --git a/feature-toggles/images/logo.png b/feature-toggles/images/logo.png new file mode 100644 index 0000000000000000000000000000000000000000..24e5bcf1ff32bcd3f07cafbbf339d17f5573caeb GIT binary patch literal 18003 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 literal 0 HcmV?d00001 diff --git a/feature-toggles/index.html b/feature-toggles/index.html index bebb9401d..6f10f7cbf 100644 --- a/feature-toggles/index.html +++ b/feature-toggles/index.html @@ -17,7 +17,7 @@ -
    +
    @@ -40,7 +40,7 @@ - + \ No newline at end of file diff --git a/feature-toggles/main/main.html b/feature-toggles/main/main.html deleted file mode 100644 index d95a86e36..000000000 --- a/feature-toggles/main/main.html +++ /dev/null @@ -1,3 +0,0 @@ -
    -
    -
    \ No newline at end of file diff --git a/feature-toggles/manage/manage-toggles.html b/feature-toggles/manage/manage-toggles.html index b71f4b9a8..199436966 100644 --- a/feature-toggles/manage/manage-toggles.html +++ b/feature-toggles/manage/manage-toggles.html @@ -1 +1,19 @@ -
    fkhjsdlhfkjlsdakjflhskdhflkha
    \ No newline at end of file +
    + +
    +
    +
    + + Gerenciar instituição + + + ALL + ADMIN + SUPER USER + +
    +
    \ No newline at end of file diff --git a/feature-toggles/signin/signin.html b/feature-toggles/signin/signin.html new file mode 100644 index 000000000..6518a910b --- /dev/null +++ b/feature-toggles/signin/signin.html @@ -0,0 +1,3 @@ +
    + fsdlkhjfkl +
    \ No newline at end of file diff --git a/feature-toggles/main/mainControler.js b/feature-toggles/signin/signinController.js similarity index 50% rename from feature-toggles/main/mainControler.js rename to feature-toggles/signin/signinController.js index c94cb7b40..ee1475993 100644 --- a/feature-toggles/main/mainControler.js +++ b/feature-toggles/signin/signinController.js @@ -2,8 +2,8 @@ 'use strict'; const app = angular.module('app'); - app.controller('MainController', function() { - const mainCtrl = this; + app.controller('SigninController', function() { + const signinCtrl = this; }); From 4a4ac539c1486f82fc7d7c20da1406f7441e10fe Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Tue, 8 Jan 2019 16:37:30 -0300 Subject: [PATCH 030/606] Create manageToggleService --- feature-toggles/index.html | 2 ++ feature-toggles/manage/manage-toggles.css | 18 ++++++++++++++++++ feature-toggles/manage/manage-toggles.html | 12 ++++-------- feature-toggles/manage/manageTogglesService.js | 16 ++++++++++++++++ feature-toggles/signin/signin.html | 2 +- feature-toggles/signin/signinController.js | 2 -- 6 files changed, 41 insertions(+), 11 deletions(-) create mode 100644 feature-toggles/manage/manage-toggles.css create mode 100644 feature-toggles/manage/manageTogglesService.js diff --git a/feature-toggles/index.html b/feature-toggles/index.html index 6f10f7cbf..58329e445 100644 --- a/feature-toggles/index.html +++ b/feature-toggles/index.html @@ -15,6 +15,8 @@ + +
    diff --git a/feature-toggles/manage/manage-toggles.css b/feature-toggles/manage/manage-toggles.css new file mode 100644 index 000000000..f4fd234c4 --- /dev/null +++ b/feature-toggles/manage/manage-toggles.css @@ -0,0 +1,18 @@ +.content { + padding: 16px; +} + +.content > img { + height: 35px; +} + +.toggle-content { + display: grid; + grid-template-columns: auto auto; + justify-content: center; + grid-gap: 10px; +} + +.toggle-content > md-select { + margin: 16px 0; +} \ No newline at end of file diff --git a/feature-toggles/manage/manage-toggles.html b/feature-toggles/manage/manage-toggles.html index 199436966..8a7ecb7da 100644 --- a/feature-toggles/manage/manage-toggles.html +++ b/feature-toggles/manage/manage-toggles.html @@ -1,16 +1,12 @@ -
    - +
    +
    -
    +
    Gerenciar instituição - + ALL ADMIN SUPER USER diff --git a/feature-toggles/manage/manageTogglesService.js b/feature-toggles/manage/manageTogglesService.js new file mode 100644 index 000000000..014ea0a21 --- /dev/null +++ b/feature-toggles/manage/manageTogglesService.js @@ -0,0 +1,16 @@ +(function() { + 'use strict'; + + const app = angular.module('app'); + + app.service('ManageTogglesService', function($http) { + const service = this; + const uri = 'api/feature-toggles'; + + service.getFeatureToggles = function() { + return $http.get(uri).then(function(response) { + return response.data; + }); + }; + }); +})(); \ No newline at end of file diff --git a/feature-toggles/signin/signin.html b/feature-toggles/signin/signin.html index 6518a910b..d0cbe3325 100644 --- a/feature-toggles/signin/signin.html +++ b/feature-toggles/signin/signin.html @@ -1,3 +1,3 @@
    - fsdlkhjfkl + signin
    \ No newline at end of file diff --git a/feature-toggles/signin/signinController.js b/feature-toggles/signin/signinController.js index ee1475993..634560ca3 100644 --- a/feature-toggles/signin/signinController.js +++ b/feature-toggles/signin/signinController.js @@ -4,7 +4,5 @@ app.controller('SigninController', function() { const signinCtrl = this; - - }); })(); \ No newline at end of file From 1719e2df72d671e6e064fd5269ac5e9cdb9f1d11 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Tue, 8 Jan 2019 16:55:42 -0300 Subject: [PATCH 031/606] Fix issues --- ecis | 6 +++--- feature-toggles/manage/manageTogglesService.js | 2 +- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/ecis b/ecis index fe869d50f..da6192337 100755 --- a/ecis +++ b/ecis @@ -227,7 +227,7 @@ case "$1" in set_landingpage_url local $LANDINGPAGE_LOCAL set_support_url local $SUPPORT_LOCAL set_frontend_url local $FRONTEND_LOCAL - set_feature_url local $FRONTEND_LOCAL + set_feature_url local $FEATURE_LOCAL if [[ -n $2 ]] && [ $2 = "--enable_datastore_emulator" ] ; then gcloud beta emulators datastore start --host-port=0.0.0.0:8586 & @@ -311,14 +311,14 @@ case "$1" in APP_NAME="development-cis" SUPPORT_DOMAIN="support-dot-$APP_NAME.appspot.com" FRONTEND_DOMAIN="frontend-dot-$APP_NAME.appspot.com" - FEATURE_DOMAIN="frontend-dot-$APP_NAME.appspot.com" + FEATURE_DOMAIN="feature-dot-$APP_NAME.appspot.com" break ;; "eciis-splab") APP_NAME="eciis-splab" SUPPORT_DOMAIN="support.plataformacis.org" FRONTEND_DOMAIN="frontend.plataformacis.org" - FEATURE_DOMAIN="frontend.plataformacis.org" + FEATURE_DOMAIN="feature.plataformacis.org" break ;; "Other") diff --git a/feature-toggles/manage/manageTogglesService.js b/feature-toggles/manage/manageTogglesService.js index 014ea0a21..eff6433ac 100644 --- a/feature-toggles/manage/manageTogglesService.js +++ b/feature-toggles/manage/manageTogglesService.js @@ -7,7 +7,7 @@ const service = this; const uri = 'api/feature-toggles'; - service.getFeatureToggles = function() { + service.getAllFeatureToggles = function() { return $http.get(uri).then(function(response) { return response.data; }); From a8d074861160dc8fe784028ac4e90135b98c404c Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Tue, 8 Jan 2019 17:03:22 -0300 Subject: [PATCH 032/606] Fix issues --- feature-toggles/manage/manageToggles.js | 2 +- feature-toggles/manage/manageTogglesService.js | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/feature-toggles/manage/manageToggles.js b/feature-toggles/manage/manageToggles.js index b286ed157..15e8eb2ea 100644 --- a/feature-toggles/manage/manageToggles.js +++ b/feature-toggles/manage/manageToggles.js @@ -4,6 +4,6 @@ const app = angular.module('app'); app.controller('ManageTogglesController', function() { - const ManageTogglesCtrl = this; + const manageTogglesCtrl = this; }); })(); \ No newline at end of file diff --git a/feature-toggles/manage/manageTogglesService.js b/feature-toggles/manage/manageTogglesService.js index eff6433ac..a0a005a5c 100644 --- a/feature-toggles/manage/manageTogglesService.js +++ b/feature-toggles/manage/manageTogglesService.js @@ -5,10 +5,10 @@ app.service('ManageTogglesService', function($http) { const service = this; - const uri = 'api/feature-toggles'; + const URI = 'api/feature-toggles'; service.getAllFeatureToggles = function() { - return $http.get(uri).then(function(response) { + return $http.get(URI).then(function(response) { return response.data; }); }; From a8777381cf6f420a66ed47bf6c0cfa46fdf0e64a Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Wed, 9 Jan 2019 08:26:30 -0300 Subject: [PATCH 033/606] Create login page --- feature-toggles/app.js | 4 +- feature-toggles/auth/login.css | 117 ++++++++++++++++++ feature-toggles/auth/login.html | 48 +++++++ .../loginController.js} | 4 +- feature-toggles/firebase-config.js | 8 ++ feature-toggles/icons/google-icon.svg | 72 +++++++++++ feature-toggles/images/logowithname.png | Bin 0 -> 82243 bytes feature-toggles/index.html | 13 +- feature-toggles/signin/signin.html | 3 - 9 files changed, 258 insertions(+), 11 deletions(-) create mode 100644 feature-toggles/auth/login.css create mode 100644 feature-toggles/auth/login.html rename feature-toggles/{signin/signinController.js => auth/loginController.js} (50%) create mode 100644 feature-toggles/firebase-config.js create mode 100644 feature-toggles/icons/google-icon.svg create mode 100644 feature-toggles/images/logowithname.png delete mode 100644 feature-toggles/signin/signin.html diff --git a/feature-toggles/app.js b/feature-toggles/app.js index 25609e70c..ace4cc808 100644 --- a/feature-toggles/app.js +++ b/feature-toggles/app.js @@ -24,8 +24,8 @@ url: "/signin", views: { main: { - templateUrl: "app/signin/signin.html", - controller: "SigninController as signinCtrl" + templateUrl: "app/auth/login.html", + controller: "LoginController as loginCtrl" } } }).state("manage-features", { diff --git a/feature-toggles/auth/login.css b/feature-toggles/auth/login.css new file mode 100644 index 000000000..9f0d00d2f --- /dev/null +++ b/feature-toggles/auth/login.css @@ -0,0 +1,117 @@ +.login-btn { + margin: 0 0 1em auto; + width: 100%; +} + +.login-btn md-icon { + color: white; + margin-right: 0.5em; + text-align: center; +} + +#login-invite-icon { + font-size: 1.3em; + line-height: 24px; + margin-left: 0.3em; +} + +#login-google-icon { + margin-left: 0.5em; +} + +#login-google-icon svg { + height: auto; + width: 1.3em; + line-height: auto; + margin: 0.155em 0.2em; +} + +.login-text { + margin-top: 4px; + text-align: justify; +} + +.login-form { + width: 100%; + margin: 1em 0; +} + +.login-card { + box-shadow: 0 1px 3px 0 rgba(0,0,0,.2), 0 1px 1px 0 rgba(0,0,0,.14), 0 2px 1px -1px rgba(0,0,0,.12); + background-color: white; +} + +.login-card-content { + display: grid; + justify-items: center; + margin: 0; + width: 100%; +} + +.login-card-footer { + background-color: #E6E6E6; +} + +.login-content { + height: 100%; + width: 100%; + display: grid; + overflow: auto; +} + +#login-container { + display: grid; + padding: 8px; + justify-content: center; + align-content: center; + color: black; +} + +#form-content { + padding: 0 35px; +} + +.custom-card-logo { + margin: 1em; + width: 90%; +} + +.custom-card-back-btn { + margin: 0 -2.5em 0 0; +} + +@media screen and (max-width: 599px) { + #login-logo { + width: 50%; + margin: 1em 1em 7px 1em; + } + + #login-container { + grid-template-columns: 100%; + } + + #form-content { + padding: 0 16px; + } +} + +@media screen and (min-width: 600px) and (max-width: 1024px) { + #login-container { + grid-template-columns: 29px 65%; + align-items: center; + } +} + +@media screen and (min-width: 1025px) and (max-width: 1366px) { + #login-container { + grid-template-columns: 29px 35%; + align-items: center; + } +} + +@media screen and (min-width: 1367px) { + #login-container { + grid-template-columns: 29px 30%; + align-items: center; + } +} diff --git a/feature-toggles/auth/login.html b/feature-toggles/auth/login.html new file mode 100644 index 000000000..48974549a --- /dev/null +++ b/feature-toggles/auth/login.html @@ -0,0 +1,48 @@ + \ No newline at end of file diff --git a/feature-toggles/signin/signinController.js b/feature-toggles/auth/loginController.js similarity index 50% rename from feature-toggles/signin/signinController.js rename to feature-toggles/auth/loginController.js index 634560ca3..a6ce88298 100644 --- a/feature-toggles/signin/signinController.js +++ b/feature-toggles/auth/loginController.js @@ -2,7 +2,7 @@ 'use strict'; const app = angular.module('app'); - app.controller('SigninController', function() { - const signinCtrl = this; + app.controller('LoginController', function() { + const loginCtrl = this; }); })(); \ No newline at end of file diff --git a/feature-toggles/firebase-config.js b/feature-toggles/firebase-config.js new file mode 100644 index 000000000..b96d245b5 --- /dev/null +++ b/feature-toggles/firebase-config.js @@ -0,0 +1,8 @@ +var FIREBASE_CONFIG = { + apiKey: "AIzaSyBlpZudOyqMsSDIkZcPMmLCBxY5DWkoz14", + authDomain: "development-cis.firebaseapp.com", + databaseURL: "https://development-cis.firebaseio.com", + projectId: "development-cis", + storageBucket: "development-cis.appspot.com", + messagingSenderId: "531467954503" +}; diff --git a/feature-toggles/icons/google-icon.svg b/feature-toggles/icons/google-icon.svg new file mode 100644 index 000000000..800402771 --- /dev/null +++ b/feature-toggles/icons/google-icon.svg @@ -0,0 +1,72 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/feature-toggles/images/logowithname.png b/feature-toggles/images/logowithname.png new file mode 100644 index 0000000000000000000000000000000000000000..f27b230718ac9cc754154cddd71fb71a6f06fd7c GIT binary patch literal 82243 zcmeFYc{r4B)BrrTkRquhTaid1vSb~}5)vY^7umP$24j$tY$f|PWZ$#zL?l_pzAstF z*akC9hMD=E(eHh)>-+z^zQ5k*k9y2=?)yIHKKnWM9j^0K?fltmXF(v)dG*K2&p;sB zK@f<_o9+zoOP1KhWZ*wGPn8#*dagE}KCj%ZK@VTMT3TOMcYbAS{mlB+YlvHqwH$~- zaaCRUp}z0*)*M5enO8bLzKt%FE_U)>^NQ}?!!5>`>shoP$aHM$BdiIkWSR`JjT;F&^dZ4WaRAQ$zdv^E#o`n?)N1Bl zsHH8NKmz;=?94hpdzH2SG7=t${W2p#PsT|RO*a^vA-i9sNn4rP?j(IjD|9uY=Ir-s z9>o|PLah@U0i_0(E>b_mzfmoS5vwdO-X>s@VBH&Zh2q!td#9seZ7{xvU7Y50I$Qd| z;4&FtI^KIRe*$zIL7w`NPus*y#tGPLIL8FX={(yknN()FOatYQRa2?39Vt!?ooT3l zUzjtgIK5V$C;kjrV@_joI3JYpaaT?WE0ID;{(haCLS|5uOw-1eO?(0<={Z2q5qbTn ze*l>hl%AQ+WB^5P4`}V(@oS)W7qW*8iNNZn`L|EkNa0tcoqi@V)v92!=6F7+!^mf* z4T@9g=7ycH{F~*a4W;MxI^Oa|&zns~#eATTj_nU=!LXy-uE)iX);Qj5?{#yNB!o^; z`eqBjvMx$1LXb{IR^rRzr;BI4>w_FQaoaCii8_;j( zNuDp$0^ls$7*gIMiw{(^h)PGGG4J|)T3wh(&^NWF_a!B3jWtivB&Y^4id&W50u?0q z*McQ&3A{`LbG>H7H%?AljZV@p9REQn1u<@oeKs_^#*3h>rdB*KJwT+cA>JekpoEoA zHx2(D#5tJ_a6vpI{y9KxO09g=@|cu-^}%M)Tz9N9Va09>kyMc;4@cAu8+B~wn32Uw? z(5^FB9HR*3_WNa`lyhn8WB^5LfNc4r%v+#!CY%$@Y2cA;#(Z5*liCL&()c_XqJW1& zX7?;0a`t`xtxq&T*q9kE$a>I~oxlNr8{IY6$Ur)PT)vcfn*z#g3n19yxF?q$8AL}# z2@SiGzbeK9N{C9!r3c}A$MZ8L(LG4GV~wkC)Y0H6e*^)o7-1Ww1`&`0aM#q4rAN^N zS8``hzx}=dd^0Kmh9bb=qrZXDT!&#A{}FdXk=EM-Adb<2CWwe#&UdF-e-%G|N_dfC z5aSbxOkmiEo-7(KAD60QDNodA*YB=(E}K*hh|49 zowDfnQ(zXav|FI-`CtG&GAYTxeifb(rvOQJnXX5OpDw0jKnci@45}rC3<46kLjkZQ z{ifpS92%NIj37mRPB}?Q?`WG-fZvbFKjo)qotmg1v5T2hph0^;p2=Tx$>TCdqPf`F zWsUEj!dvr_e`3F4f2xt0PSgPFJl}#i$sYwc$&1asX}om``Wp~VkfIa^uwlRxR0-@X zXz;v7{lBTn^PE0Gr;BL*{P}ZWknD>9y68@M0lZ!(@4@MlNuKQAqyE2d0H^=^?eYJ< z6_A%XeH6*G{CkV~ham7r?Ek(wkWoKmfz5On#v{_^&_0)!LmrFm7|E-j4b-r|t~0YeS2E@;398x|fTts2f^-z7eri z3Vk*$|7L
    %u=%qyIDKQ$S{G=daQ!u8QZc-|D&W*ICa-=7qCN`s9XxnOasYr~2*! zPkj3Yf2})rj>^MEvMrqWm{+GwX#4DJwzgQ=(fXNx#wb@G{A%ujQOkqpJ~tB+jKd(&*Pr+1#o?0`le0rNx!b}L?t#r>|n{5G#)QY7$7$%y2?Dc(tbB zrlonOSKAzlD~0y7`&lVZI;$5jDngDqqXJcDGvcXNRis|DMJeb2{`HZO`U~@0Q5Q?zei^q8-N-KFDw2hqS8T^~Q0d`0-WunM1A;Ct3cDtjbep47exlX7;=yFmWughqq&?^PoVHGl&Z(duF8UqvYG3Q4vC(?nB$0 zR{1Ao87$CPy0JRPTmQBa+ZXp;&V|)6zV-eK3j?jQqpnX_f*U`-1bJp77J20J?l<^7 z%|1E)S$*6UNO_>}&l6@A--T zGAx@piOG)H&&dcH-l~pOa|reumuh%8qW|C8rnlnNy(m-7?IJ-ec~n?Km{s7k*JpFA zynbz#z*iD`?!4h>eioZnlOQtbeO~9k-4QQU*eGUBkqz#5N)~=%fqX5YXpLw8^2j51 z>h{~8lI)sK9(P7Y!hy3QHJ2kJe+Qb9NX03IfO^wCLaqJc0NOAE3c;M_KW%d+gNChLLDUZ z127h$I^6l8d`O%*C&nSR>`mm2ayKoVa0N9EN3#OjBfNkchqxd#;a|Lf|315z_0taX zu#`dsz~>uRPB%x6lwE8zrw50lUD#1jZxfm+CmeJSSomx3iKWFyU6iggKb~I@iT!jc zf|(r~*=?4cM?vF3^4FfQJjDB^Jb2jh^vm}x(do)KLR7FH%u3l&QB5o34phA1PdWcT zbT8|{7Xnno@C?QhpM)H%paM$QANu*9k)V)fgPNBqsBLW)Z(#(kz>*KuZEpPwf#>e*{EQ9R{3!mOZ zZuoVm>6bn`MV3!rcJq^<(d{=Mgndon$WgOc)qutyZe^ePgx`E27$>7tLcIh%A}i4n zHj}aqE3WDir}vl2r2o~!vEH@e(b@HAvx?h_ZInmHZb-^V7hPO3X4s?s55w2<)@?a<*nzj50H=ogFrOmKE8HFwVmo)cVho={LU0bu8V1X zUbClkpDv@fsX;7S$`{nC9&ixi+5Yp4AG6KR$|#X#DVK#muN-l>%w9RO3V6qb@>iL$ zhbz|9ey+zHR8#%NP{wsxssrM0JMSj3&GH9_PhU}!Ls~_OCn+9S+MaMHfu^ftH8lz} zkHBIf@y%Ql)bg7?B~RC+!va~4VNV`ajf;yn{ZrY#N_9+HjNSQy$Op`vn6DNKVf?vK ziu~oeT7w4)-G6f2w?TzO&K$VNi9gM$dH;U55Z)Qhj07gzaEMezM>MEz@@3r+@yPUX zU5M0>8d`V(zx7*+>dAQhu83+5)1klBKV9*2`mOH7E0Kl2=8=ti6h2B(oZ8&?W2Poh zZCfeBNwo1&l6ruY(E{3fj^=a&VrxE&i%!_Zi=?^fUVub7hD76}mqJ=TH@CCE-kI|eW28*TT0j>bLHi~s^-fnnc0Ns_6OrR;TO(b zN89>EDqAeL>Q(>4tFeSDu1kH(+L)>!0-{=8{t_};RbuS^thL31!03@ql`fuiOPFXf zC*u0zRM-aj4s?-jl(J7>v&Yjvxy(AIUz*co#R-uwqo3aK0raH)B|=D@(g19=-zkG zFfeh8S2TF7izC}nGw_7{+qG8)cuvOy*&wc_ufILA8BX&lnb$kJI&oI>DScjyEBBv? zL>Z4?aQZHIvz$|^oO|r75Kh+erd^~A_t3TPcdtGfEBcBNJLRQfD%tpCV6H-3u&jg8 zs`jH|39q=m;eXjT-Lh*yGqY~qo?`5){ky~R+_Ow%k0z-6pN+Nqkuzj*r~1s`4JtR& zLfY5X{&SzYRka46Em(QE4{_<-9kLB)0MQoj+8ov>Py_{5tPlPePFCYg9r%^-_b6#% zz`qO1<-5OEeKh2ZiVj0-*V$^GoCAI12Vy)Paf|r0?`LF0qVI^NI37i(PL$v2aa5Rc z;LUf_8q{ygmK?sHEOR7X;@A|a&-{ba&rDlWtK(CGYG(CCBU^H`lY6?w<*{>{Y1KQ)a>ADVYYsP7-j zpBe5f@Zg&X{%Ud=nz(g%QbKz6`?gHn8(NUy?GQ|aw!aBJY~lEE<7?EE);r`#cz4(G zGGys!2|^J>G!U60zP%|&>t}mY_OL{#eUqcQ=$AS@ZVk`X;gNawtM<2=5ZH~?2FL#&CSdAKKcfT7Qy)#kg?s$E_|HX1wER(=IfF{Sz0!}A zdx>=xNb-Y*n_e!4S6#G`Qab*73{sRB)82K`QZ=CLde(UX`rY9MKL*}bNn&n4Ix9q> zUrnkkSr^d<)GN*)PR9as=rfuyC75p>Isp_T)LNVzBIOJvFAIGAslFZ@(wwnea?xFk z5{e_Lg(7*4DGLhzx_Ci{kSeTzxz5$E9$Ki9ILifEXSCJ}j{R_fKudEwOy;H6rxCN1 zKdZN`LXnDXA0R^$SG>%z&{A$trV~#Iadx=MBj#`Gku3Xg{{q3g@t7a{)W@|2x4HnY z{-cG;k!lttG*uzA+T1#Ngut%6qol3(CL>-Zl19299i;=72X6a*$>Sgmo@f)%i_msJ zY7uLX&GpPaP1FyohpNuUJV&pI~5wT+e}ggSBf96^!H*GWavfC(uBl ztPD?|@&F=qLQ!NXspLs<9TIiu#{Jsa`@S6uXtIBL>uA~i1YNEz3FC*;ic3k)jk&tM@MRo~`#g1Sh^%a{F6F*HeiG+N z8|3b>bZwp2ZU5*a*9j-zY)^5yYNqY+@>%GGpai^VOGZK-xyW@%iCME2p$s4;koaKl zBLB-+mI+0e(Pz%%M(Av6d2)}vRa^5AKG65wbH>5gb|)0%gD`-ajoIh_85V}8PozW8 z)|?p(8@K6hFXJFpA|(K85Q$R<-#l!qF_2Oe9-3IU|2Iy{NaMfdYSkATKGCk*vU;H) zQ40KZWN(0e8~7LtE~~4hm>wRvUkM93FF?5&!% z!U7+MB89~%>?{xf6J$Vx9@fHB3M^!yc@T;ONo;!0HWTBVrsi{z@ci%|dw=fc?)9*9 zFUCNeBVSIZ$3A)fA3jM?^7M3wM4uzq2LrJa&X>^>sT1<^O0-;v~bG zhaz7o>V46(R?D=7e*U~=(_qi540}B6hPh6EdD?Ebs?ZbpjF<1xtH0{QyA+( zhF^bMQsYO_C@9=vJXOk1HV$SZB78HpUvp$Z*!Q~}nR&L05P5;n=AU<_SsUhyXx5A9 z5MhyCJT|xd?-&b0AypE5__z@{Rfv z!)IXUYL;1zE^-9Gl!QG7zc4X&UKT(Gu5HE?(x589REl7h464z)3jmg0DZv-n%3r{4 z;%^wfCj~Z3yL_iuB+_^bzW?FDz z`|=Ir*#Y0h4)N`|H!igb{1MmXDGz73{jSf6DdcV2Yj7})pKG~4;dX;^6$f3csu`Lh z+!Vg@ff+XuxZzoL{+Vtda&EZf2Bcs zu%}ofmw(x^m7@4krVJt7(B9)^CAlY~!ObX1=5dh5x&K+|1igUA<74g;588QXk$Ms#*xokW_V1D~ zG1KpZ7lO^=Vs?!jK2cdoR7K<~Od#II#;u zB|mRdiS7p^*Un+Swx>ILnUtzmTThJl#kTw$TWHQnw!&4diIg4m<}MqWo8 z!lk~Z^mF+oc%H)pOu zFXqlrG}?}fT$$~b0h#-ua>wky&JMmzUQml1C(ih^fT4S{KFDjAWmA_9${@$G4&`l; zi;xG)1y&+>Fri$u;Wn$QrOqS z8Gd(Dlm?Vn6oyolp(GUPEJD&N$5tfF_Z8Hgj*Aw7z~_eGHr-%WOjiNMDV#;8xCq;N zElW3YP57;EI%(IRyGQ0b&=I`Y!CH(rJNsT9oi zQEyBXLxd&lWKdmG{3a;Z0IbWI-5e^R2o|*TufM7y*zj5xYaT2PK9*2!ab-{&$rlZn!O6l{1@U@2l)5FqyUQeZiaMA{z%Fe%A(cUkAfJUd!m-JED#5U-4A0 z)hZXZZ+m^O@B`K0`OQki`&AW)D9{BIVV4(KV-LT>Me8;F(#BQZ@4@+&y^d%F+t6H5 z+~2kC+FV(##x_ZA$ld$lL!T=;3&vmYyoiDM46nhV_e)885~!(kZB`r3zA4;yY?%sN zqq`5BwT$50w8U7&a%HW2g+GRmG zrrh(VBrD{6-SP}Ys-`~rGGOMbYwX!MBfD?P{M)?FdPMn+f|g>cVQ7Qyd-)D*{dOo_ zL|N13$UC|rUb(hED+uISzI>M8BcWw0)L|IcV;{Kym>#R}mio5R`$vrBHU4o4X#Gs{ zoWuMc=Nqz5JKsrGwA2b})LQHwAx4!v8?6A7o7>jBt_?(}M}jXpRa6#WyEfY4>dC2j&6yF+*{9n&k}Fmti zFm#UPbCtKv0K4Z*Q`7(P9wDSIElK!=uhISX7!$lFmuk2`F_AeqX~{R z)Pv+a(|mNdIWUsz$?q3m^lF_Gg%f|(9tCgR$`iVm#FXiAmMtS}I6%it($75Lp30ln zjzj8FM=80OKPMYnK6zwgy^Wl@@NlOqU+V|f3CI+n#Fr_!*W@6K_{xT%|D55HUQ z%6siEeH+D?dpY}e*$E)m#S%2l!+;q|$`A@+<1Y*jWU(l+^xQcN;m%i!jY#`7#WYZh zo{0bEYN^2%G$82Y%ca4b3238vfLp6VvI(`jJ-?iY(i?^6)F`-#pZCsnA4#QI?;n`2 zC1G|LrgTqZay~hWxS=d^i*Ke^OWjAH=7D-(>(Kf)&6c~nOledge6@wnUx4ls^YpmN zTks(#c5XqV1wftZL4)e=7!f~;Bj!UAqcWso=T6F3iadVhrtiwFjSySPPqa#v|5k}K z>m=c-k1Cu|B*G%oaGD`H^{M|1I53yX6s%?L~dewrc(6YOPF@v=*KW=p#${%i~ zV(dNmD=at~T4GI#^q)H}enI)`JO?z>^RsxZGBzRL-77z(OE>3+pPc zYM|rmMW41-m)`odptwVm`s)J}cgHU(!f$K&nu852;^vyW9C*Wq(yf-I5=vYzW&>fI zOi;8yfG^is&eN@qb0n45^8G@VsxjqTfDh$`z{08zNIi9}83xYv67qYqRqp~^nOC9} zzDdmO?NXs4C%9AB`lT~-yVX9h_qN19>x#?ocl2C^WaQuU?h_@2)tSP}T8{^m%)aZ) zs8((-{q!dldW-%T6m|VIk8BmHBb7I=ts8$B*b3hk6W;l-Tm77QRJdv73##0Okn;ZS zC)nh^Hfjy+^tcVOVQ)`7jdLdX7u+XZb7;BS_~i++-cj@|mpPN^hS10OMxnKi%yK}| z8+n?J%!DYg$7IJ>4+mka(oo#?pe$)@~^gSXo1reCo~wwmwXEe7p;xPs_-%>*m2 zKPL2x=mrVb>TcHPbBth^BktjO9)mG}Lwr?kUyAh!^`hPK*KzK=)502md+F-ud@-ds zC-5dX@NSg6Z`=F3nQD~;F=3|{zSxSI+{)0xMKx3i$onL=aDBdxpvtDxSOik zq&jYwC*2%!B0@6Q-2^WGm3D@aHn|oF!bu|2l$*bAfKFu#KF%}jS;Pf`?{wSc;njHaRUqy!)c>4Sg`($HTm%qhbVq2nu0wl12hkZEv8V^g4C#HTXaI43k%qw$fw%7?)1 zi3#Tl%n;D9={r!r{W7ynnSKP*>&eqVmWrp^N-Fy%MhV*rGvnDXORTh_zlM=B^0|ti zC}0IxxmCftxorCp5JcF#?<>ui@Ljo(nWB22xMSZxvoJJlwFAXAk-gdaq~l+o{4#s{ zc}$WIksLcV5*7y%JIy`UU)5aPbYr*PVVWleO(UY(nvN>e9KZ|2T`GG_cFbtVZe<;K z#EDj|xy9S*1M_<6$JlMw0OI|c3+e7DK7SsfL`3jk!~^idrZEFg*;5F^nNV7lOWYDKo$WCg0Y%0=A!516_S1aTVBJeyP)47V1?9lyA=JW z8M8MJJ{%tzNy+lbV@j{Sd7kt5y)oo~Zjc}GT)}n$>ev>+r*!Dkf-)STag`P&>;ehb z(W?IxQKP{pEOwikZ!Rk?`d2G=%f{wrZKT85A}0Xjk2&Y=4O(b$W`q&02>+eivKcJ8 zd1{s{1be()PWix|5!OMyPG&=T8ENK_`Y63=_iKWxyrHO^>xTK4)iVMDiTzROYW8}z4oC<-aZNHznS>QM*l zi{fgBR>ira<@jcjH2`ms-lfr?XMF%cM7^ZOtD}BkzAbdna7PAy98+gx2b9jjBt+r& z<7b?-nWf86K{p`vy*=n?WbGev5}gcF@*Bw_HKLAP13gXfKz;1a+^s|ydPXRoH-`B{ zRHKd)dAGTsX2KWwe)`?H?T>#>nn*LSg7~K~Lo4YQvL~1_?(^f8%K=N7FmDnWr*N>2 z+3MrG$TuVM;0PmRray9lZ)U-Lk#@LF(s}gLWl!9={|Jf|T7M;{PB8k5sEW!HR!VGV zz`~b;B6Um#6*fX66zN4}z%M=bJa(dxiSd&KK#$G`P>TXSU{I3g=zQ%xG{mZPyS}vX z@@Ct!rWct;u2908GU>06LeZt6@#JtwXvI}6bQOjx!}*9maQ(1*0%g#4+LT`6i87cb zMJ@-{T^amr;nb!Z#gia6;|`cLq~4u=A9J^gzy;hpL=L&E=Pn;8aNHNZaRw5TsHmx0 ziPoe_2|nA%L;o^}sGGYhcNO~7d|3q+d@zK>3xJOFdE}T>*n)`gyISj7%7<{VyS6~+nO7? zP7ir$=2fIjjFnO`kB&T$gp|{)V`E6}BMW1y*$Z{swnAXZqKp$_p`-hAX^^qxSNejB z*1Ew8Rh5>Dkb5HHkf_8z64#)NpR}!ky2>U6-Z~rEkLzNa9?|%7m9O9Awxc zuGAfb0M=xjx>bgeGsF-H&KMq@HIt z;fyll(-mx)(#4fVE4kKq`ZuwPDmNSgdEeX@?@j&OkR+y-_-8x(9B#GgP3ju8BlLg; zTs&_o>6h`WP+z6b-(`hokcmgm!mu5;diALbADw}4Wdg|fO84Pah=O^p)bmgf;qJ_p z72NoQ6gSqq1vSu`gbUC%0Fq3t1)rhC{`l|F3|EbX=H>@f&M{|L46Ze=l&-(733<-$ z;{2*fGNpzMSk)d_HLGDaX*g9@u39Epiglgo`FX^bMC7`06Q2H~Hs>Ir_>BHNs@Wa}S_sAiG^mBu* zu8T)soiU*u1mzy@MioG5@3L~~IUIZ$Ub{qte;EG*^Cs>26BRZn<6z3~_sCZUR9O58 zlB!npSN_zloGG#8{I!xU?-!nb=X*bN&tgjZ#yG9}FlUCD(=3`_+asoL|1K42TkYk9 z%^Rp@wV^(ZIDZ?lt9$Jqw{B3Go-BZ3d?#K&FHj5w_Iqbk}V;ytX27Q4E)OP9WaMk@CN7 z0(0}f>3WT6f@CqzK`#J*f%4!?!^m~)@n`tkOklX(ePw(H(h@C@UM4#zz4hn-j?6V{SO`KyYH%a&0498jad2DilU`(ii#OJz4>}+` zL0r$sqny1!F$koW9(4<=x97MpPqpBmk*Z zcfq_IuV3~hTv+*uUiLswE>42yNh4LJ4!i{S(64ks%MCBHT$>)?|h z{H;9S`}G<@ysd zPJ;a4!i!aKb4e9)zYRo`%uL|lyE}LXuB*Q2*(n23OXzwI-~D-X!Gy4eNbByiY2;XW z5uqtDcLpK1F_~w+>}gqj%ssomI@=Dd>++uJ^Oq%Bq-V6&u95;82*+A+4l$4%lZKbs zuXuJ(obcm$QQp2Xsz7PqILA|#@KRhW?fcGA&*m-d$KCGk?n;g9c;^+!n^$MHIjVj# z;D;6LwspQD(*HKMi7Jp}MJt;@-S1Q_Y&mXty6_Yk?GxiRu0UME#=+d^djsA5oZyAlj&0@jsgt@OZB#LA1xHYtoliyMo zE#8Gaib2A1)P)NMLztmL5HMB1R44&>ret}_UY5+b_0(MLrTOHSiIMbA8DhUTOL9CN zWbvYM+O=%cI=`T~fu^Q52RgfmRIjQ1+(3L^&-~`7LfS8NU|J^k{Jw#ovoUxs8t=-;3BhcE!HZpG38;PRYEOaCJ2XP{Wop8V`;V zcyjbxp$MK?1)vFR+q}SxSx7Pu*iTMa%gl%x-nOPkV5&)zf1T|~ROV(!&d++hf0J2x zsEG0|%G~E|-3Z2?Kge|xlhI%ksbWHVHLXwV%F*5awOJd|D!e*#d)l@yeuyNn8%L{x z`XNqvFlI!u&Ybs=zEdbLbRNoN1fL==00ka%0m+EZC9y0`DSk40lF@o+T-i+NENScmSfHw|Xtk{pIq9NGU0+&%GH_lCjMH~~KR=~!geg>LS{$E+ zRjqHsMo*;jY~hd_occrCIz8S+iogclYEWX)dA3=(dXcyF))V>>cK)x&5Pieyu1_x? z1wA!6))~55sKIO+ON~t5Y7-ZvD@_#_Bk;g28n9lzT-572N|=BRp?qfV_!>OPNukx4Dl z1L%?D4Lf)D*Pcqgx!+!HHRgzQr1f}PuMlbkpbGFZh$sn!8eY%l6Zef0sgeHA;jjas zF}VI&+>NIZzIN_PMt-8^$s==>Sy-H_tz=YQc+fLawm%`!ZszANY&%{7{M9;T7)Nem zJffFiudTUu!2+o74tHsq8wKJbjeKL6r=>h7zRSnB6wiq$t;k_ox~-PJ20slK+h`a^ zrAgNeAxju|yKs_z9`(bJSQxu~=^4L9Ol3TEDy_FfYtd>`Nqk+`h!msQ5OHy)eDPz%Xx0lHrvyGYE2&Q08~Gln;CKVE*sL@ z;p`cU$Ecd)_&|$@*lp2`lcXd56~7a13k++GZmZ7g?F3Koa{Ypq8bUTp@|u~t&*W+* zlu1^kDfw`tu%e=ZZUi0qg_2B-mT`p9Hv<-(i*6l$xqq7F4_JyfHy-*uN3vbcKtG*) zY_1|VFSsz^bhSSL11;-wAn{GNiVC*q&O>#2o1ge9^eCi&e&VGY{c!#2fCd6rP_sw=p2Tx&YS#MiTRfAtxG=$(kMk6 z%4o8D(i{3+r?y}{=#Ujg_;huw$yo1FEmM`K$yd@ve)#x-{rygw&yG&#$)~_jp^un~ zN004SLxT7-9NSJBhPRJ^5R^)2j`S{lmE+w!9U?oxzxXQ<0(M9)NL!B!ptxo{iY?oi zzdmvSf?#>z_DM!wW`rf<>+FlSi-w$}yVN)53u7L~M4P025Viufl$qx$`Vuvw4_1g8 zFTPD=2gtTug!4U8JxIu7x<8-)5Z{PDwmAz3q;zSdqN|~7?hM#-IMy_GrhKNduw0`~ z`|!kLm0=L`9;s2QrvlMzWEjd4(w!z;ydg7p57?5mA7)v60+|n9wN#o5dYO7)&E@ZY z7?|PX^P75l9i0wz;vB1Ama9q7_>HlK2mVsk(rr|D0;WOX75ssFKEqVn3g(5lFSg1) zT%wdG}f8pl#Vc9DPxs+D!RB;VglCKgb~Syh`!x$+dUL zRdn|g#$mNPf3F-sZGdA{!(QOV{AXxekFYffc+aX&pe(M7mK9zrZaCPspK4L6j5>U8 zkk2%EG|Q0d2eakE5fjw<1gD;YlT3$O;+^F?FjcGpcjZZ24A+?V!!Hm7`W5+sj^s7< zHKttO@lm7<_r#?xT`&bFt6Mt!qtW;r{YvtAJId(?XD;1RE-^%ia#GCIn);ge1U|X@ z+^8I{panFSfwlm;Twi_H1Tl{-wcYsq6LmAjH%OiUNnr6x!*zz(i;(h2bfbSZ91&ZN>FX5 z$jV*f&oH#7KYLSYDyhydH&39K%M5;X1^Zn0#`begp?kw7TRQ z39{JhFJB9c;;*iExrV-@+HBb0e#|{+O1s*yAN7+fVELWaf|Q0)-8CMi=bhf%GcdsO z!jPjec@_$LrhZEmw!@kttvSqD66DKG33g|uFJUs6?9QD1s#`hNvvIK?(JO|5)AG{9 z-@9!b{GU>7JB7D7vLiZAke(yw^FE;>TKgLHo5?6Q0yoGmk2yeg>E11PEL%*3ELNUD zMM?LV+;*FH3X(Mf6aZ6X3@}_pE8z_ZB@$>_ggeN4MDiNRogrAO~Zvb0>mY{%)A=_IXYwBEJPGPz|Wv(Fl@lI2?u4$FmKBR%= zmyom(*+N7k-Wt%GUF400mSMHQd2SD1Vy%Ba8#nWv{|c3$h(oX-oHS{$;KsP) ztx+d6i<#sVnrkb{ov}40okzsI!5*1dz|8e<$H)FV8r-e>+MJ`A!iQlHTCTYF+7>d) z&_v+?AAC1FKxjLF*k$}d0X;}HTzm+LR}Ee36u@Rj1GhiJBxt0SQQo}Fw?iZ}Y7cXj4mTSa-te!H|_W!VTWMbLTfhPqf5a16nhp(^MH7zDz`~I5dva!r;=0 z|8T)>vlpeOKvs9^xk27cCke%cRJ#0BdiUOohqc!->P#A;`e>_L8+f1j-9&QAIk`!Q4wz#f*<_UCAuOTbV@g>5 zK%$e7GHuG0Tn|!{P3GkT76h+x2<3$!gU6Aoh-eYr4oBw=;z?(1dI0$Zw{tb;2(79X z-v|zxEMa7^_TJAT&#pV=0VCX1sOBlXk3bd-;^3%D4Bd;OhDl*1RSX=Nx9=8MucoKdH*wId)WoAMGO$;A15}{2o&L zy11(j%cmcLT8b2Oigyak&q_SqDXZST)H}O)BO76__E~i$g_KqXZMCDc)W~vTE5ccS zbl-~D4S4Jo8bFTcV!$(|^t<l7a0oah3-8}gT&pB!HOCAH>(eoYhn zRl2H(0)HbP;R`>mC6u5VGP1a#jw3(Gr92=&XJ=f13jXBOmbYLS4$iwfN?i@uO`XWI zoTC@}zCPIV`3S0>wxl*pNz(Rg{-G@-hj6U`0v@PewW5d`_`S;qNN%?T)d<$w=J+qa zoa+}Li&}5r#)*5i*{J^kULzml*q#Q?z&ex5v8MxlvLtBv0 zq?#+!hGXp>YvL9i1Xi5E4uRCA1+%3MWfS-K+yi7-*pCbH2NAXsT;j-@l%Oks$xl*o zTXHQ>K-|1Mk)$ndBJ-Tl(04wALQ9dodq*irR&GVY=d7?*h67-ZDv?R1Vg^D{_Bj)hS|24tsjPBjxPg&d3b zCHc4#D$*}BR>PZjJ2X*K7CTQo%(GJ&25$om2-R57yD45d^rHioR-|n={PJxQjocnb zZ@=b@WH)SIj3oVBstDF`j*y(h4i(qxi<&QHfP^ABEvaSVMya%ba+-IXSDSsU+0qgf z>-9c!poqKc9Ys^Um8hM>z?rGH;jwTHI^e{dfot$%8_gQUKOX;C%PZ%)&;wk}`03oO zKbt5{p@p@U96uJ@^xH)OQ{M!N2#vT3^eF@Ihp9j)^k5m<_`I=Yq3ajPh6-CcvopQD zaJ+-GVn{NIRD$iWd1O&}K8_@&m9r($ba*&ycnGMeu%^`9nd2!MSNlN! z*^@u-=fKtG1i|xk;;RjuvJ#Z!^Uny8;fsA2DFj3!8kF@svg<;T=~4yszwDb8%T3*jpaMVcHZuSM|4du1sua~QxPVJw5ud!tEo0vkiPLRtkzERjW=gNm zDLf|bwx6$&9^}7I$!RxU1>C;%3>iZ9C9G2F=X z`30Qw6^flF(?A(3cM)8N-I`Wm&@%)4LL+d5h#W`FCps_uUdLF-5gt2J<6XF4T)qY+ zvh$~Pp(Y39no}TSYqg+?2@k&5^RthOMu?G7uEgQ`!+? z*WABPXEMyxAvk0zI7ORia(qD$J#-Qer z?>rUYIU3MlPrVUb3vd=3I%jadg>;pTICZ{m&2Ojv!7HFYCb~@EQH(w&9~qM3yk+<> zvd`6K=jZ|^1LohGXt8%pU=mORE{j<$B=0G7Je+K@5fM9Dq+%DpqUcjHOpSe!cEXY| zQu)BmX#i0s^AzyepAUN}ir+G_Pf&Z-gIQ+uO)vRA`qFK<2w@UfWaPtrKlC5w)h6EL zBu-29`7Wq_fE>*E9Gc0heCPD|jPC#K8Yp3iI;86;x8sq6-!<_V8$@S%DH#QMmn@8`@0OBKUT=#g5ht|iall- zV#4hde)KN{4-*gXN{3*>OJZV;Y#DJO;!;^G@t?C zvUHm!mJA$lt4r{DN27)8);8*VSi0OYls#+PmDoSsY7a*U$@`a6_?SL*x~*01D70fW zSJQ+L#E(LW^=9J>=zPS;VZ|TfxHGJ@?Q+FTM9megnql}=YG&i=3pdIH65lXiJa_X( zVvjrt0{asF<#lXdnzZd&+I@5)0^%;3+!p}+@oFB4IZ9)>opRz+o&>FMf-Z%f8w!7a z@@Kxy5E1d%N%mVZgQ{}QeFL~2v@4Ll;Q7p46HnO!pjd__I8h7tJ%%8A`9Wa z%`VvHs4$21XGZbarZ^7Oq#&Aq0rJ9{;x$tBz)lWDHpE?>b%3_ z+d3vDvUzw8v2zY{F4Y($LK6mqDT5<+ITjbSx&HE>LB;7`NQ05)+Y(8EYe;7B;ZSY* z7*bgo?_p?=S-B`#M%_g}4>IkwzAh$VuoG*;(f?36r?=G;_A<49S59=YBmLK{SvLFq zCR||7bbS-YXPRibMD3bdJH}7_=i0YjU5@af;4kar+shAW@DX+Mg&j~+@SOqG7;mV8 zyuU$8O3KfGg;n>${*Qisek-#L9%5b_qaR?jV|A0!@Y2L|T8B}!YHt{6^0PV&=liGy z58=J>r*CL4^!=LTZ0(Qt=C;(;Rc9kSo0@$~?w4UMduUMMpiS0g9yep6wjv`JKXGuo zvAdaMAN&mNXEuaut0#A`(kUx{_Sd}Qmlx46A^UM5mzo>N&lhla|N zCspzBA8tGS_KyCDR?a_uxwq$5$ zLTqh){~wmVGA^p``}(UQB`KmH(jpBC(hWmPGawDp-Cfe4GDvrWG)OlHh=9b(Njv_Ts@{oLd`o!u$}g3q{Ysj-++XGP!a}0mR;ut zIziTm14;OW?J~N>N~D+Qv_Nk1Uz^<|LN-5YRZ)v$zG0pB5r}ipu zKAgyOIjbW?8Z~vc-PD7Pzw)T(t_pKpqmEdg$?GUyYmeWPm}>nMNw=Ow3yE@}@M8If z`8DEAZa%Jejl&0Tzn*r8kBx-p^MbI*gz#{hjxFaev=dF~{h}tMbh;0}82KximZvf)&(T8O5Y$m*;HH78KU3YA0a@_(|wir&-y)&^fQfIe+ za=o?@eNf$6CU`CQNKQ0Z-%oikYXlMpWvWRAOWz+d3A7fy?)So*JD)hf!?(SdXsb8>6c3Wl|Z@c?y z^%YIK-a7ppb})yt{j^kyM+f2Cs$DOk@7K1Qk=j8pzklv5E%}W4+w02(W_L;AjDi*3 zB=Dp(ISi%9r^>IgT(6(AJ&NW6Kl)Kmwz813@04L#PYk^gCNFB@4{qP4H#WI^cB;+Y zr6eVSqRjfoV4TS9=`EHelmfTrNz{+i7Gjf;k z2z9a7ID)U^Y!+nx$_uHE-_;u7=$GUhDn$bJ!jAR=;Tesek!ZRWYcb z#k>@OP$avQ=X~jGlP}E%$@dK_7a^T`fme8aLzVaKW2vdBYtxm+OdlvKurnD*sabj= z3<~-ESAVS+`Y$faLCFXSC3jN6TYpq%M_-lU#P-0lW)U)#AMB1ndOGc-sc`_IrAamR zm#0a^4#k{L_dQz4*e}^s6JPYkj;5m`8OKkCeWbo6gQ `G&Zr(sY`4DrBh6xy zkK=gR`0j9%uOKFzFlUw9s_bXfHf z1)ETGB(;o%@OENbTR@Vyi)gBg4d$r-)m?K+mjx@q(1LgOCoS@eYwbWAfm*vv%;FKv zc->9eBAIlZUX zmtu&$?Ut&6@f-f)+qR|?fp>arfwMhFn!U#TnuTv~)5qvd+zq>s+7DfWE02${*HXTk z6*F12ntAn%Hr>uZ4Y*ym&mw;3z7(3BJmccx0zDo!Ld{PoiBA8;bd8^GzONBUEwY5JZlJM9#w-Yd(D z@H_o-?S`5*wKm9-1K=Bb@`MVK4lLo}UXSeh>}6OT0g&Rj_`yU@A4%bo;M>vR~d_ zg#7G2UqChA{ukp5ZTj{!&N52UIz`Nh^rFm=W7vE5C2vmQYyJsVPwEbVJIb~K9<+__ zu^+PAAx}o<=2#$go*jIW?Xh{vu6t)czS481Kaq#z;m#^laj}zH#C{TcbI#$j*iZ;kdu>-E-u2!Nwk2#m0g;2r8GJ-MN`&lo?nCvuEF*F zrv8@K-8`=d9uY26eX9GAgJ9^P?>ErDEK6tU%C;KN;_@oW>YYi$cfuaPav-VHJM6vVxMz@n9vNZkn?qqst>czmNN- z9CBy4bPAVZdmP<*Fz=D`nh0kEq00r>UuP2y)9^%v6@>+hmZ^?E?P zoo-8ynecKq?S7sCkuhOMWhjC8$q4625R7}gxBd3c-gG1_zG~&DxjRwCq7k8q@X%0G zwyB@o&Fy}fCEY9xzi;n>Dhr<-(#NFlQ>gd~(;Zl*Na(m)+Pi1s;6To*AKgBpZDLP? zT0p%IR#(~!zwZnPQdiM_W4-v0@DCFvq=`{CF*@q3J>*KMV&r8!MacG?Fj{Kw@0XZg zVO+Xc9K*(U#am{e%&t{FO#fe7!rfAg7|9wQ0&kC#gK|5fwZa_21 zx&;SQ0QFX93CCsBN+9ub42gE52$_eF3w~l?F9kq5bbM&ZN$9sgx z5#cmlXS*hOwYm>!w#l|M#37K`Qmh^6#TD5Q$l0&|(!Q6@slV$J*KtWZsHYe7kr0Ar5hv&Qw3NF#p2k>8-a);|`G zbD-;`Lfk?p5CA9F{W5-R`N#M{^pVX{ZT(vaeU&i!1|*MaZs4zT(SPJ(UJX_!S;KG1 zhu2$q3)FoUXh5p`t@(wF*7*8Jd}?e^I{5}!1xxhM&P$`Z@Dr}RUe*sDfN{`Vg zAL-^hwg=I+aSm8Axn^t+M&5>QK)6v?Cgsq8UNRz}?frXkl4xWf4bc)6O5mGP;WD=< zVJNIW&98lkmPmw6yfLB;<(R5Fr=P78xmdSO@#C4D+}(IaAXr2*w~9?hs3M5g<)WUG zdD6}-jc}8bUi4(cA*9RCc5A731Xly_Mpfcd6llwo7k&Rw{3PbrcAo#5DuEX0JTxyK zO>B_P;Bl2St%}#XxSd!vy;4sIdor0D4Jv?Ix-%5&{aR6Srpae zbvG!-cK&WK?rY6Ek$-?xj+O@SOX$iTz0&6Bc`lw^1a-nK(2|QxQ^g-h1 z_rO^t9eFA2^n^7pLgG3}*>_(1bP68{vPwQEl6kS5glcPa<(^%L+JS28;9u zF|Qhelo!oxiA6B^-|;DBZ7(e4FD!H#64zeLp51{OR#pu^wVQ_#fNYAc8Y}4QOq<{k z*?z8(Poz?1bnV80jrw(Z6d^&5d`b{f`{t+KW_js_>XNG(d16p6KS>t-ukK4td97t^ z;xvIVj;C%y1X-Cj!VF>X-RRjX9?zcP2Js-5;0?W9q0uRxtnEe27b*Drg`+eGXtthb zO@czIQU5hd!sM>$<>RC2BVxjPL7jS{==flb{LCE>O~tEoIaD#iA)JY6u6kI8kOzQ^ zKqHfAMu`tx$)_joIfr_r(`S$MQx{hO^yyrqszOKKa18$CGtq+ax*uxzWjpAA!brHz zw0bXYix%Fy{72*nmg7h*b8hmZq^!AGsp+cpB8{;4*7TbM?A)=kbADaM9PI5SuV zUKu91G@CzqTBynIF+z<<0q1&p5&f(db9moK-ks*y_%j7PasK&V=oqtt_btKJqIR7k zG^gEfgK@#=ebMmWm4ysH;yiFT1N5sN{PD3LywH3;ZuNAf*K~A}x@?Iat@^7J*W79T zC`h{&B7yP#cCSlbxPWPxwi^{r7@jcIzJAttSlb)E@hglWS$Y~B5^V8FRI;NYI+M8( zEiYbHsaPE9!?#b)LFy9Dgx{jkqk97CAkFco&lDJi{Ort43s0wrr-+QMo^VAS!E$@u zUvDy6IeU*yOx^lE8l6H^1fB*wh+fzeW^~bi(>>I0T6`SyGcn-j92nx^>1>Hs2nw6| zJ-6n@UP9F}M2W$~*$tfid)N#1s0*YBZ|&DIL`1KKV39Bvy|4ayB+h6=FP`1%)9+#* zeHBfVz@$9dTN+=(+G5?C=a92q4O)@?Uqdc0^2Fd?ACfFG94Jx)3m!CVML~RQv8kB% zSi2QeWw|!@(18P%XwtoIt92Gq{N=_3tpftkwGH;M7Mp~);!!_3OOi(vtswY|jw(nH zbl<9cREGb)L_%icFzSrK4@Cw(WGLcZ*mHKg??X{b`Ab24oO%lFJ&OC zTcI-^ubgLywq;#<`dtd=HvH4Ep`i|u2X53%6B%h9QL+Sf;xFEQvw6$SZU4yb3X$#? zokt;j?!CB=x7DYA9SF9q%r}wi@LH^9?wNofe7QMx#y{hCqBY~cey%!IV=&uXc4{v+ zAY+i=-p&GVHxMjpY>|lQ#h!G)>ZM8p7i+)=F1GMb-c$+99aqbSvPJmkF5uG+Y6YXQ zem(>uj&Qy9_SwkrU!f^J`5||5B)QzGh@zl8i=HiPiLc-C)@{|ya(=Jx12Q%+yi+_J z#>&tYO{>N!_qqql;@O?N;<}!E$4w>Ep6!tS%674cY1I4bw#7K2x3sN~j_T18P6I3G z)re2Xr=!KNsjo6TbR4{tyg0+Sfd?~qW@ao2QQLPqY31sUUZWqig$ zlw2Q<5p;qKE}M^7fjwvvGcz%;`-*ti>=1zSYMLdTyw%FtbM!SWqT@S@?3#eQX{dJu zHhfS};TeCBF(gze^71)^*dbm13 zcVL-xTy9|K=qDs(jI|I&^s-QZs1W0J#0uZ()6-HJf@YEI@NAwIuj)x~E7Js(%~O)3 zM7Q-BKnHucRbGME#MEv;tUPvUa{oPrIpXbo1FuDd zHpQJ0dEw-qDNMcC^r_NxKegTSmQJAz;aJrXy4_B42(Nba{oiO`qTIb(&d*x}awvi29e&sa@!p)p213XwV4ze9uH`-?f5 z?KgkJADtZHoZ$C{IH$*`Ev=JVa2{sZQ!oyQxi{0d3tRM13<^4^+pi)u2itzDo{#lS zKKDGDujjgD)2^3RbG=^}5&7xb+;N~QL2Ui@El-y$Gvx5C>ECCv&S=daaWYI8wULxnBo>%=cJ^bmC2HOMDtRVwvD~f}(u14{J zGv*wt7to>R2pjCE-J|b(n>!?sJX-XPJko{Xv-;D*{~}BAJ1z}R)A$g{xgp-K6D*U= z?YjT--A%3N&Uf_&gzL>7{GRacznBaf0D*)(KY0vAmUGb4sLb1@WxTHyv&e$6)`*iE z@z1sxca@%eJx(nCdjjc_wcuq$pV&x?RD?MG0jM~*rvVL8r`ZW+AS>{%AJbtNh-Ulf zj^Yzm-RC*eZ$%4|TxvK(>-noOkYaj-q1{OVu4u zREAVfQ2mw&NrN&O=m@0bR|{zB;_ZWv5CLilyF2#~uys}MzWm*RM3`@%>4oXX$%VY8?xHN8?%Y_|?b5DKMVG-X>K{BAvPa!sH# zv?Zek>!mI^8>q{3z4j)}(!mULR!^fH;Jh_1fwT9dp7lqpr~K-9c^P{C4h-o>vi|tH z)490AWvU?*@X1~KSJSWVenn?+yst>=6q{hoFyIuFA4ft=(ij;tp#@=2#vFHkC2S{` z2cAG`-Ft+WLjzr_>Mdj6yOA%GOGBhkxUo|WOgrA7u>d;ylZyG#c73Kh;fMqDD)d*6 z0>|`9Ivh7g7InaoAsV`4@$X?wyv%4wOaXL><69;O{uLl82{8cS?T8UP#=hWHU^ztx zFyg)OC3c-1kMd7e0i9|hI>m;!t<|}jwVoo2j&g1k2$pNbcBUr<^Ksv)$jKQJ*5=^} zdk_=MI(zz{9b{?=L^hxYe{|qsS@yOEJ7MJXqDVOF!9GUIABnH$tTu;=&nO&78Vuf+Zw zuD>KsTRrZ}qPYrEYo;j{Z$JOit%9-f({GBKi@AlR{nz{LV7V&{0J!Fwe2u#m2=8wi zpO~BU!!qTxrIH>=#njv3r<=o~ASSqsMQY=1Pg0BJq`xmDPgD7t{dwB=fiX((Qs z&)oI3%bROI+J8ke`fV4{M^!3HBS(SrT(+d1#Yu=Xi%1gv{>R54)&qN-7QDIFX^D!m z-vHsSK*cL<0UP|bo2Oy~lRJ(gbB{vg2tBn}^>aSIU2*aN`!;|8n0l6#JBnbTz6!E$ z`{B9Ar(JLSbsWWO`nVjyx17=B@yv^h=YqfO{Ii^=z!^06NRR)Bg`xS=g{(ya+k>TQjaAG)v94ZJ5uibL~si2Br zsZ-8`;{p_-@+3V`|HVc47ne{VgOX&8y)$)yF&*Q-T!;hY<))LT#p<2WtzDzjx`9F)+dK7adVUE0oo`t@5(r85 zwzl!d1p=Y%e=I^_lXi#eyma%)Om-EmI`y3I>E#r&t4Evpy&hF);`Lio+9}6x6v~|Q zALF#+VUAQJ`z5ycwivwhIxq5N15_<4daKRqc3;ae{SUS?iOrt^VHq4J?8!Y~Pj z6jH2#)aUTC;m(jga!DE2O!Skk@8_TKQMMwhleN7~)%{4GRdy23=N?W!VP3o)f_-dQ z>-oBRkTxKpS&gC;k0n^$3duuS!8`1`>JA|-BglU{W}GI zW~!_E(lnnQUR{gLb!>65D01SsvP%LJPS3D1zE?G{`Y!-3iyf$1HV8lNI z0{%-0Dc|vg+k|9&@1%rMQL)Pxioo^|;3`Bia7~=@FFIQB@=i%uZ9Q}J9D_Pg03t~o zsa3>x8OoC6*p-(^So-_VD@RtqfJa_!_XxxI<0e=7+}-*BAx0sp*R&^9C#dIX4mff4+-7}0ZtrY zpmoUNZU2}^jJ;kZq|Np6(|pj$e1TG7)}4v9{D;J#x>FErB3}P`52i?AuaZ;!BgeVb z<(p`sk7e)9^ttG#|H4zju!b`j!O-N}qx^kg;;_i()+1Lp#en(=dz04VIVtgL4_95@ zQ_j92_jeHboW8j~bX4gRj&*b@s*K2|-zQf?y35&Nwsb9saGb8U={A6rru)XpDxopx zEjlPEGIt@(*87?ug;tQKKh3$uQrZFb#aU>!hL%OTU%SR4>NU&8hJ756{=}XSHonX0 zmoB`~jD-Jrrkwkdd}3L|*S`XL)IO8P;PGUoF2!uWNTa*vK!@irkO-Z|%MjAdukDLo z*)OnLY6U?z$X_&duaV9?{&5St4ehg)cDBK?Dcip6?`ea%9IRti6u3!=4Q$QX7hh_%rp2eX5zas-G0Fn z(6sVt^h@tKzenHc}EqS{b z-^k3`R+Ml>4V%gSo0iQ-Ncw=@6W4=3?`pigBzCqui%sh+$qa9hlFJopYV7rPeTFzX|D7vNk42vHRKO?%}T++R5$fsYYzarMWx z%OID=x25>RT`}|R&Yf6z6su}3kV%{xb*p}EROGd7FFEM=4Lsc2+fh4Dp25*bfooO| z4vuDvvGdBV4TMDfgE&XdC0*kt?`1GU;&^$@jJPl9g__qUx}^W5Sh~xWn>QKxvyBL zzdKC}+I&5uYHMq&kxspYoZOdxYseD#3w@W^w9A9)#CD!pf7@ytzvD7QZSVjKM|t5Ky7ww#;OHU0kmS`UnW=f#7>CHU2IHWf)$}4Z?nqFN zFb$2n>Enk{PUUF-_c-u)zzkS4OIC0ih!7@Ay!}8~{^48`dR)tpjJ+5@JC>BFLtXF^ zALK6Gk75&M35IZ;C_{84rZQVv28E#yT(IKVbP&{6*o*Q57B@CI<2zzYb`T0(#~d2u7!?ID z)k-~{Z}KHvh^8u(afpI%`q0{Pho46C#eJ=je;Cj|LsW(UqX6)U%2P1Q$2kx)plg(p z_$F6mhkwAJN3ZC3&dWB@w}h1ruVj9j{9w_ScB~=Ku;|9^{9YrI)CJ?%qHPVs!t zpId?>C)h(HKgg>G9SvT3iU$kODr}L#{?max_aI^jD`Nk{90t=p+lJf6& z#6j6VA&MxER$64A@t^&K*`Ba=)ZD}sg_LD;OS6U{X8=wsvcHuH0l-;rnvxr5CzfY> zdu^-5`!%i5RMM2}vqlkXZ% zsJOwaI(x_4=uD*3qLu@EBVz}q&+z;gW=5Fx(tTG3-KN|AGA(K`X|B&jA8?O@7h+-k zwfMke-X4?f;CzAirmM^M`E!JO$hyiD^Z*%a-p>vf&$iu{={p8IL)}6PuqK& zYfzb|vfHxCN_xV0C(nRi!ym#I?*F{ny#yJ{w7R(FFeW-r+$FFRlG z1p@Fgm7sJu#ZaduPo#&>650z|eh8XQ4gAn=f!ld-CMBJr5r}hZWlH`GPn`GO?p(kK zZgqQPU1=2MwB+s&-33|zt?_fn&YmEb_!N-sSY@RX z%)IUms?f(CG=axd>e;zEV%BP@DPb^zdE)iT_Zch&YWgjwikF+$`+sBi+iYj8PN3E7 z(UHVFMcZU$_*T|E>E5u(-RVE8uIrJ7;uvXjKmi1cX>Z4>r=PRVHecU3wOHO8TKFx~ zo^lb=W%>@b$HCf;LnF;DSMu@utSRr5hYWc0rG7ZP;a3?&HPj)XMl$gIPnRt98Ssz? z3Yr#F)=(#EP*006Eq$+u<%y0L*1ps?-ybYk7PW$G_r@%rm6qIv|K(+Xp( zVvsKhCD)a=Df4nGWXaE67wPkw){d*&i^NYR-`pQKeh-g9M`p$@T>PO`ary*EzKKtV z#tYRgPa;_Mxosq%n@U*b`LPc!6v?!tbkq_rz6d6?-kE{3bT`fi9xEpc7rakD?Rl1i zp_8{S>?@uz*UX%4JMXARp;A7a|V^U}!Cam-VXT?zfxR2akWNs(3O%2`suo1X1 zo*+ge$swYP1a6_t9agpnDyl0$+WcEZyn#6#%A{$(Ixl1Lgf74~$cLNxmeEavru9IN zyXspTP*3-E$nauII{`V4c}bAb!x1zCunIvrnf7gcy$g7VN5!oMEv`SXM@MFbGKe?5 z(2ze-t#$i$u@B1U9(sGPU*X;fROOHo@u+`%$$%+YH}TOQ|+Z@*)I0)|EbZQ7vf(fWQv?{fv+r(b}*u_|upj6(kLMnpNkwEU)T zW($+~L(1AtFK_3xA9zF3H?+g4!moDzyv%tN=wd^!qoW>v2mDd0KoCFQjs)_i+v3bAiq9^E1KA+Y07&c zKns$S1ER_xFdJ%!xc!V2h;EHH2E>PIk>*Vy3+Q_(&1~aHFu#Q% z&=E#;jnUbcadC_*F1U4OO73}u6%h=4B&0bN4gdB{B=GqBnG=Z$< za>jw0_v3-;O4}c%Y_qBur{2R~<8DAXy3h*i7mo(crUZ^*kt}Y`cE)dR6Y@^`gi6Ki zd;VnAgVtmV-z;&Gpj#nT1c5C`*%rxpJQiVIMkLv2;C?Ylz!aw+{irJU?bt{qs3~cTJRWr>&XJ4 z#RFyBfq~&Crt0B)GOqnd^F`JQ-v+Vt>EKmLaTl7Z#B8_bQj*uLl>iwyF)ah3#Vwh2 zfO`+p_vjwxMSxKLen%4I@&ygVS{hK=#zLD5`Fj#D0pR!-Mekq6cO%goMvCl^`_#Wb zH>&oUd8$YpP1N)Z)pCwJ4H8mc`0AXuhjH&`Q?Bj;2Ik+Cw_lc>Id7qO=2jZB+VPi( zo#T&aCBs2RS9G586G)99B=$q+x#z9sDHsyYf|=8D$>HMEkVQm|Y{-Te)88}!L45Lx z_}P@E-`mbk9T7YEY4W1{2)J7UMArI$x^H#g?k_Bp!ViPXdUA2^j$>fV ze$;InaQn-Rv{Ytsh%S5N0`^kn+ZWyA39_ZbjPqAEQ!qKsGG)iTA>cGh8#iH!1ynhzM3_n8(PdtxpO z@7CTY3i9HUUkkswR(*8Mm@zJ4Lbe8A?u;AT1Hvdqh{~*e zF5)N+0IHS4w?E&>6gYrp*KG*sRu1~r*@pZ1tH~&np31=#M6EGiHBV3S`v8Lm0&?ZX zTR_hDe(B#&kaV`I8QAT>`VZuxfA2_WO1K}zGT-dQ8GsWO*F(XPq~Sc@xOjc+>aPm7 z1IB(03-&{)z^ENy3;Qsb#5z>-ZTUWIpGa$>CQbEn5?2Y&Ka$A*C%XXo^ufEIVTylU z#lkc88RCJZq)G2hQ*c)NH4tFD4yN7h94MO7l@e&>(9`@C(Jj`=f+4cLAE-xtZhrRT zb~wI0@F{%aqutpY&#)Q)B*^7u2;jcZEoRzYyGNQ7D1F^tl;oNjGKO2?p}k8a6)V;< z?hc)1D5i+n4+dixpm( ze@CtF;MCU>>8;+)m%-jd&Vq|K_(IR+t8SWq?NwSYYFSL`ieq$Y+u{D0s2`V+r)?zJ5VqBBYs&wcA4MUCR3*WZo1ZZA^N(i4%Gu~fT@SU z@-UcXbRaZ)RE>l839u_%?6aY7=7Bf&MKwpe4g*nZjrDj}5GnYQ*O5wKF~9hbfFbxL zc^>#(vy{obq#4*Yv1PaCfv~g(%u+H~|i?lUKlz_Ghxzb$(; z^#Jm`=MtG$Uk_9a*^*%37O_z?)S;Z7P7VkckMB1ZyAlFEC4M&9BcsoX&q5$IHqulL z$J1tm)u|(G0zgchEbB-;QeL&T>hArLr^f1n|z0iC4YVSR98W|TSiS10{V3XS85Xk%I zP45h^kBgW*kr(X1BQk$hPfE`NmTmG``kM{viB1B`<{Xd6U24t>iTZf)(|fD_$LO)@ zKFrC3qFBi~#4b9p)TIK)BJZ2><#~C?>W#q8M?;ZHS#;*n?S|KVLBI%z_H?33D52Q7 zMGUqiNQ$UcLZ>Inif8@bo2YZa@AO$XSO44!1?DS`0*3%yZr^0=vB=^N()pLb@mnNR zYi;qy;Y3+4-0l%k`rDBacYRPEhvp(*d}HSw(WRy2*_+(uSS3us$lkh{ta7!PP<-|v z%_NZOc5RkECg){qKC-45a^L@J8@H7u%`JA;V8ul6)ix-rD*zKJD5^@> zouv)q68E-K&pmfJjl6a){Lg>i;`U(Y@RLE&hhx1`;bOS6TrS=OVFFJrS8wuG|S4FWXeNOL^Ni5=pAX^2c zmt3t|ee=~&kkvqF#=M=d$hDp4ZdXUNU#rmV*6(*Mic~c6|HJ7h5i&*_3uO(eT#f9M zHIdsXDYmfIccSzu|1JEILY#Tbx?fE!yOoTH6Ko7RzT~Kw zDgUXF^mSokgxQvL-l$|mTEf+%0oyTnW6URi^^W4aN~pcjXr`LN*hnDqgVM& zuC0uJ)0k0;76$%C!*JPIh|QX!2_twXUaL3GIsg8YzX2R;q&>p96PUYbg7A#UfI-cp z`o-}`y|e621c)w@LT-bkYW{LDS|;m7-W%gVC^A{te2&R-7c+oq627=lqTMY0G%qT& zZog6^M*7k5E@6g}TPIo|Ac&=@jNLqLO0155+@?sq2o@U7FDSfzvCTCji!LRKa8u9d z>ALR0JMR17=ze44t9ys1DDIhkJd$UDY&*Ys(`dJ)J>RsbE3|6e^`OaXutQ@HqS2KV zbW)cNw3~4!Fs!Sj>)De$d<&*@SkzieY6OEgT|gQC&;9AA7YzxtF({wx2t!1YJ4Ju# zv5c9%nn@^P(LAsFSsoed=#o#Uc?9;5P?4}&1#s)Vf9-ZScqp6aGn$3tA;MR7*B3o0+UnAr$0<2PFZ0~p z&x*c3Fp{8IzH>9nD>;~B160rinCd--xH55=%xZ&?0{D}?ut(R>gUh>0cxPYo;}DLL zEE!*z)p5cZTpWTnBBhbFT9qj5konK!^od(!+fd=RKdx<`6gR_bM^|PqpNy~V3Mw00 zbh{q#YqIH;19$AXb6yoE$GkgBIW;pvyRa%esdDHp!C42sBDqDrM;WQ>U9Ju;0H)OJw0H&W{-TK*#E@eZ@dajY3~lE;cmoo z3Du%PFNAGH-lVtMIu&1Lv1;#p@*TM%BYWrq{#4W1xtzhG=*?XlGuMjfH1_{4*J1>Q zM4RmWkWO-N)4+Sb74A>*0NLJtL{>{m>h`De3+oCY)i$BWYH+BBzPTXYCups?W+w$1 z(e3V%w(#9or26+e{CXcr+h~n>0}j`s3zPEkdSAn9>@7eL#E@ajx$;%Qqqh>Oa)`UY z`2BNqa*j^RldbJPIu0HnsJ4QLA&g>%Xxk{oAo`eGXue2B_)JdTQZ#3Jn7HE%bBTL2 z*i5FYYPbl!kdW?f%O)LsOxy1NV>2AAkj1osjRR_bccUS44zP?^-!CsKOPC3#=i}qs zI#$LUq2}Syylb~?|30g>-;3hl_+@{gCfdgpO&bY6I}A6@39{}l!Ji|jyHoJqVHs0< z&5EEV4Jd237j3#hnfd5_g>IW3Z(;f$x;xy{;QT?xiWLTKL4b$AOEcBDOjm-Q2U4XN z@pd58n`d zu|e7EtUrA&@2H9qSWD4zcc5Ou?6^XyVdW{#xi`zX{;_9mQhIDJmRGAuDfm zv~xgNLB;AtKOvhuwrMlM}b##%+q_4l6v$r$ zwp$}B(Hjg1LNSz_LP{65E=GlQIr`DTxF(V z|D@!oT0|sIJmq7TvmgfW2muALu$If)1CQ77^-~;pgjDD81F*Tvl6h`=N=VNR_7nQ# z+-fT}HJkJ%=`N?jr;>SYxD*dM!=KV{P`aB+M!emi^4&#+g%E|NOi7-lgy1tL+kbkB z5Pn8{o9l=7ZQkGIjaYGpb3Y%Xu_B1@|nJp@;_h8#}t;OUA5A2NmYYHgw(7mr9t zxYQI%U8M9*7H>2hL`sqwYws`lnjr~{7MCHmH#DtOytBa7SV+OIlEU)&fmd*Oihke0 z9E6~E0r*+R_R%F|S|7>;uXCCYsZ%#msLP)x_d^{v6qa-~(bh3rPq!LAU)z{IERF))?Um3^@aksVJxmkb=$kv z@|4jZ>@_+k&sJ_qpomkT`ntBml}8k#E43XcTx1tU3;i+lo|k^Kr5GB=EQT+NxP2@> z+gS-pvv}pEAm~ec356Q%SaSwfs_eUm_{yO;JzynBrFz2A0 zea`_+fm#56_ex?b=72jlQXncFV)Fa&q8`Z71JcOKncv{dh9f6{_)=Ua8jQdNgthX6 zCW;pQih&FvojE(>S&P-fgm4(fMKcNX^Oj#O#j&QIdLPE#YcO`RlBVH{h@^RP^w%4X zW=gm}9*C5#lnBN!yubCihin($xT(GU$UG?Wk$Y z0^DnO4>3@Z;|8dYv=?q6Cf~U@K1L4Vy?fLBpU^{(py&ZC+du41tg2A8Ti@l9AH&7B z?9NlYIPjI}%(sk;#Vaih4Ih2&^4fbVl&MAC&3%|QmC0en&bdL}XveYh=6xtBqyKnc zGfI;g9hq&m{*5xLAiqH72b|Hfo1(AR?PC)-E%4f1e5U8v@C@|a&?7&?4aBc&DeYea zL_hPnGU~MWZ4u(=OE=Cmf7byABrEV_4p`uX4vc6iAA%!4CB+$`{HL0P|L8=PDyv&+ zoohlZCvg4vlzen@oQoz-(xZ)K-w!%M#yr3)-S0La7!{VYqe{Br_7~XsMxQmnhTW)P@(H@Dmpw^cD8U_-N2y{;nwfUgS13_5VYd*Sr^`2#2rY&V6i@PF4*R(6 zALSFKHJ|lon}+&^j8jb!%aq$rp3Ach?$=r~GLXZOnFV76VshFQQqOeKFBQr3iI_jy z@<4%TR4pB~)7vi^ck;iZjb;WD^TO<~e5QYgJtXX2{;0PLmLI$FWdzTLCuWMJDPzbn z25wV6tuw_=&==s&|`Tud@D_oKc&j6cmjEp02*dLsuFQeYBsjds73vbSaPK7SRblQq9 zg`bFrmzU>8BD~>!4H_Xx%#pzrFsK&+uPLwu3*EHt3Hwh!-x*0-)5&6clt}!n6x+2y z)$jf9aw5Fs@%XxVsPE4s9l$e%ymDN%`l#TEuChoJn|LE#LQs5zY$Y(voTX;iCx zOKC*=JDWXaK?ioSI0$ANM96sE43HJYZDIY&cfrklC9m2RkJ( zevWbRV~xt7&{P*KT(GQ(bP4T&(6Yic?8$tcr|zer%cGlXQ>}&D0J!gu2E~r0o7!J) zZw@?py&i8Q{$nRUZY>{b%ge0r?%q+QrV&>9lt0CCQq@^WKRL7c@q>w2>C@M(D42IR zjUq?38qhKiHKD6XS0tK&;VXH%=sHcuycz!i=KuK@y~KW?tH3bfL9?^7z^f;1A?&>V zfjDRTLD@E82BnCO{oL2Oy1tTO!lcEvCC5gyN%E$f!6n%T_jiBoahmmj+dQE{Tsoew zc+j3fw?O`@o)FcuOcq{zah>yFw>E8J(;;YCfCROmwDp?F*?GNBh=3jH#2TQ?cwODT zGr&6=Ys}cBZ3~Mg1N*+ZCON;CQfc#eH1M7hmQ2Z%@UwNu0A9*}tuj7*d|SjE#)EyNjiwlZ2+{+BQ=VBvp8wP%Bg~=5lE@WWSm~4DR{}LmE(j();dQfb0j8{5A23W}|={E&BlsTDuPB7}b0sO^eR($dh z!8!uVjx635b}aKQB9)sZNP{o%&T#2|xd6;sLSn*7%87Xszd)DY;`Zs^otf(H-AVH| zTxJ2e{N6nQ7cF%V*N+Wyw&vQQr77p1=KxgkrC_eH?7?xbv7Kr*z4j>mcpA!woKo=Z zeeXOZ5#-D|@vkTKlr)~VpNCs?zs`~DyercG+Xm9lVSiEHEi_TUF-PT~<<~LoQ_3_i z`pv0r3xs0&*@Ur@Nu7i=3C&lO0s$lOAt3zue~Xjf)+&n2V>?hQR~Z?uni_(=rs4R& zfc4DFTUtXod+QZ3vb(!`>p00kwywU`W6FPeb(ttajl_bYvGib^N*ICLj6nsTi5N^wu;HHovX* zj7RoyX_}2N?Bm2ZrR26p zB5nefL9x+Z(dlB1s^9_y5WFfT)4Z!2cmN2)@TVpcON)_kC(>^{|E0&^nWg2L3YcYV zQ}Ytt0B8T3n>tqJMzyv|i`*3Nat!sf3f+3{J^IHVWq;~j15^-f@Iol?n2}NK_pUv2fexV{jY>DD zC`d?1cY}h8bSfYvIWu%hN(u@{4IR=sbO;L4B_Q1lFm(6a!~5OuyZ0CTo98*tIeV|Y z*4q2{XSlB!)ZUf9Z)r=NNf!-VVif;}PI*!E3nHrs_Pf{zHzeG6Z@pYX49S`3=4bFC@chNjP%jj&~3`Uw=%F94hGFAw$0fi!hc>Tg)RmIoEnKGN`51?af0X< z&Gxk+q|<;^6%xF5+sseBYfoB3<+T>dYN`YeceRkwTNwzWE>72PMk~A(`@M{~Ni)v6 z;iNyvQafaIqXwsmwkFXAQn7QswK7PC>vBNDu=5*vGY}N}A6YF$AzMK{ovIUb%+^}- z+jSh!l!{3c-_-y_OsIx{Dy>`Nl2df2uK;^Y*|}HI*Xw_UgXL-5{V8Ir7a){}?UclT zGibw@G^8ru70GTHqb(jR&+J&F5GT2sQK_F6x){k)?>`J$a$PkE39^fdUJMT(y4RXf z3a_TJ*B94;3O2d)9QhQKy4(p=DNBqydMNET%Sk*yR$^~lr2l!=tYI9-?;ovy%b>b} zf4C!XH7g_1)8)AMnaR##QC)$)B~sS~JLN5|WbrCADs=H<-(YP#4{_?CoG1t?$KH;4IbA8kVAYLC@xR~HPr2XW z#+#gR4qn&3p*&fy5_+cG%M|T>FB?+ek|Y{7B~K5FO3fZC^2P6lJ~s1UXX^=$#6B zvPI96usffcyhz~ZEGxZU`fL(?IxCIwgSj4mZjbB6V~N?LPXCGkv^BXQU?6T1QjA=K zC+H@hY|SgW{V>ywlWFqA>$6%06Vj30pu4yCn?g7hni7F$`6-!2yk%j(VTGw>*B3@O zz~sJatygEy)qdcVp0rc{!}cC60xtmCD*`OG8e;|CtmoBEq!^#F)Tvt6BVSl@2;^uPfp8gSq+z6+I^~3&GHM zHyhF`o>$I0cT?&9=tj#|f85zV=xH7)EttPt>X=F5W{aN}zpszeNl*!gN{n=Rwd55J zj78@VrdqB9B`MKl-un&7IMYB%Jbk$p!xV0$LYihPacP4X^hIrQ(6T z3RWi`xXL*$ku3|OwFOO$9Kdyic+4SO>i+h|74fpARL7Aq@cFg}zD^K0j#i`}ErCDq z$HhI+ldX{Vqe&(RWi`k6BoI};eNy!JUx$RYLL#)&i|1cQYA6v{qr;B*fX7|La3PlA zuu7QVzK#~gcE0^l2`>3Vh#V}{@wVth+uu*vPdGU>LCFoO(AW#axpCW6Qp0f5{%B70 zpG*;<4QKaSiCUvtZvo&Z;Dl}~TRBG2Ecl9_*^VNO4wgE;J_7O0sGY6KY1|fRU;7oC ztovWn44>>_-;TuNn)u?cpa4D#WD9E@suX-(Z5J|sN4|@|Zu*(e1Ipie&~_gtyJ-=; z)IYC?M3go9t_y5#Gs*w@TB%&w>SQja!;*W{&}pRSuQBc@82Y%MEOTiN$F}bNB+w{& zC&;-srE z$ub4!6Eslq$&Ze=MjPba=|N)pby*VLkE@(Cg57_Epr?9Rn5)wbD*P~ht^f_mOkT3@ zEr3D02@!u=(8sxSNTvu}^)HDVsazBNvPaKm#jPdxvR6C2M0}-=q&hYWHFICY{5|ZV z)AgQod7)oVyZrV*2EGr5Ol{uO?4e*H8SV`;63YH=-4D1D9}WiiUupCW>fRX!-Znjq zOpzw2e-cZ5E=}%AqQ$=%FPs5P+NXzFQihF+aP$q87@;Ge%C5Pppl#pHG#lI)VYMWA ze*7L;BJ`Qj)mKvMVfB$iw}ptw7mEZhl^T?2%qR1Iu(@h2%|M1chD=r zhQH|r<<->cukY*IbxFQLRXw*Ab^Zbe79P6BDdN`9@=bvRn!MzHnGx;@zsCl1=pOqc z7z*U)+Hyic^GhbAS46122->ue?RW9qC#|x-GiD9YYAE2~>UevqK=4!K6RTH5NJk0V z@W@WW;eFQ^37X&ByS_shD&en#7)z(#-_BRWjIg)Fredz>^aL&x^#m+)?q8^${mRY` zJX-~39F84^i-6pD`oa1cG_>Q$125p9A`|V-u3n~ilXUHb-Pcl=T3%g9u@c;#D%_Wt{H~{CH%yXYpPS$ zcIrPXzVZuQ8g$CR-@hQce715op3~#x7Y{e9k67$=`ha$Mq~k|Y+jK=EQLspv%#)?; zH7h%V>Hj!@`jsHr{$*;{+B5YpVuo=hJX!$Sn?D9MniaWLTi@QHzn&zobugmwN~HLl zi@a@Z6Ww&7rd^_`l@2-*4nQF&62?M6ZQ>92tUJCVGU4@wQ1*TsXb7jN&K7ExQI!AE z4&t;MYkWWOdA-IqA-sHEhS8(`N9rW5bpN!U`(bf`M8k4(UbAI`A59Hj4Pi}mN78~G zE3k@vt_#yA#08V^?Z(7W^rLa#w~|ATf5}A7D7_~cZ0Z*mB)szK{6Ri0) zH%C@p;?u8Jze{=)&YOnspK3V3L9@Wlk2%F7vGu68dtd_JZ>fh1wsRYsUpOn8i+U`G zVWKNp!LEwpCeYk^^X_j?WINia!`%>ml|bPu5*@D}Q0w{fzglHv1#>33gALjG`I|^J zX?d&_L+nMTU*b0}_URiga-$XbMUA}fJ~!6GWYgA0Kp#(Xi_d{%$=!5xMoem!CgI!(aFy9NU5;My-lk7wEIoN#n{R8fbKI|8 znv4lp5E+m8!wepD7!2mQ83!gWUw&Dcre^x@**T&$B+~E>Ygi>G5W4}OWxWBk1Rwl) zyyqp)CXmI*2D@nSgQ_CKns~0M_CK!i-Ed8&Sl02&T8v$l?cAG-+Rck)*CZb%inuz3 z8?Gup{qYbl4p66Iy$PBE>U1aO#j<#~e)!l!nI@!(*0-D$juVLMO7sEQ9M!jK6(Y$_ zf8)Tf_!LGy2P`&m{{C1jX6Z%^Ji95MEvH<9n+-KsaV$V9meLeRrpus)Msd6%5+?X1 z4eGdRf6^1jhaj+y;Tkg)E%3<$fccQP7YCRg>QF9s#h*a<2ds+tBPW5Zi@VOxv*6;#4~HkyG`arCd#dDk%DTr{!wPj7t1CAOa6#oapZ z*%k7ueyZz5nX@?qjQUK=XGSgAZ_+n8;eQGx0F_<9x7zb3n*o_msQ6j(clIW-*W#uo z2UD%hmMj|aNz}lLCfJMJ*$8d8)%#OqI|a_js9(+ge$3q)tyHn}@54(9lwa%Lr^S=( zjj5vjGE5BVS=i*2gG&>!6GgA|&_^PpX4otrbxhkQUwkNW4Jugun{3n$m8LkSA8PTgO*Q% z@4rE0qt5>o14dr8fQcRa3y#kvHN03d-L5X60gm5(1p=(f#bSEh&#y7%Ja*O}An1It zG-ok-CcJ7nM+=3+s$N+~(V(h{9c-q^^YZf3$q)LAvF)bKinlxI4;^iTMb?PaCHW2E zbTSaJ_DXrmRWGd*wBJ1jb;KM?odNwOm1ef{ka)_wlKima>&Uc-T7Je z#`776uKMc~^O*9cO{WBVy$J9T@@)Xth`I26uKYsA5J$LTNX{CD(<|p|Yn6f<^fU0& zM|R8;Efu)>DBb;3vWH@A{J2r zQ+n7y^!6&#f2FqoDl;ps{#`F;;jVP#-e#Z_6m%@HHQU>+M_Ivc#Srfj6An{Y(%c;) z>(_jAGj8mTL{wG|-xSBa*Nw)U;yo7lpmv|Y#y-%BmdwV6HON>sJI}~*$mf;6asr(F zb#1>Gsdnm;S=AfOJY|<=wLaHsXE93&jzKKwVEa1?4;bZY5v4|f#(0msRrAt6sLDMA z9@Dr-3jN(G*y^~V*UaSK-*|<;K~YvnxqVLtE~xOKqPiaRNt2so-EBHD%sW>$hu$1- zaxmuwe@Ujhj^LNsXE<^7TjR&#!0r{H!{|1%%bf7;>FqUhcR(8d*w|#i+dP^iv76M9 z+A|tOrrsbF(`H^2iQs{_@%v&?KrH;ySuXE;p(F9BV|ov;-j$2IZ-hT*{}D7f=@jJ{ zc<18o z{y%O>+RXh8RBQAM@HDkoeYyqxQVaV#{oMgk zXH@v9-7Pd{gPv=Wzk;E_!c%qQ{?rT=KCRhr0@&6k=)e*3;NNP#`u%cEH%8cQTMq!Q z<0}>PKkUygHyc7)6b0>+Y)P3~DW3_NEfT=A)09e)+#kQf>|kfbLJ{%lXi4g z!1@gpPm|*wpE%`KQV5UmJ-Wnvt!B$B=Z0CM+XiPQI#I+EdN+srSGNo+_RY{o2d6u| zKuH8j7nY^F!-5T2jLdV;ydr%213q{>qW`6lTmlBYR_a55in#D@Rs_4mT>K6GK$B#i-T3MM+j-IA zt*gxGn!-HX4eU{`(fcRPTCJCRWaCo6&#MSKebI^ z{uHLqS5q>3dw-km@=vZVi*kKS;KE(97_ieAl|J@?1$=!&UBq&4#Wi^37J&x>|le22hId-DR zRANgk=vT7+3plJ+-hDoa{_UWkO#@kVv z=CtZ4wGs{x2*!VC=A5TCVIr{S?Y!0_sRf_mM!65nZJTiKVNQ({ar}O3^W);Km6gw@ z%ny9@8ik{CDNsVn^pHLt+L9a&Q2c}!?zS!+V^=&tJ3^tm68j@9?uonH2<2e+%=Kh_ zbV#Gugu)0-_Tw30TjM`H$p*3-mhIwejx4eyx<5X%K$zvWbg(q*H#XdPoMxfNp&0o9 zSiumMZqB5R0W^FZ&mY2oJj^<35p08|o=m?aVUYSzl+f=U_4#F&{?i01?}=fKsn7Ic z7yjZkODt!9d})b^$qvH!B=q2tu@uns!r09{x%_71$y&VipATr$4Xo$7xH|5xKR0=m z^zg}#*~wW!@$s|4;wtA^DH0}uOs*f+YIY_{`f{$!wcbD9Cn9uOr4_6u{?f2?7B0PWMxz~{-%p~F_LF#@a2F|tyPzh#fPV}>9 z!~;@^Cp0R__c4@1Q%NQm5x2HWb-lrkCRX3;TGR3P}1 zAYZffE$h*<+>Si9W*>W*;GyQ3a1xh~73N~3e5-P4u0E15{ko~~#O-?>ce`B^O3l+* z)1hgl&By#1CbsM@ysvgyquQ__yR+C0rnF=spk3N8h{L;Du&_}ag1ZL!2(%NUMXrAp z-cunwN>>dNXEcbSHsar59Eb0j4$?z1<51e_mSia>;%}O#VFOIO6~(_NwYMqU4?<9K zigJp{P(ZQ8{O5fn~g9;(X%4_!2>?E4v92^ z!_zAJxxc61s!meE*6L~We#B~iSvW&wKi+glDxkT-fqlGUDucSY>Ca^mzC5Pb`NT zr14wF%AA@!D1yo6bJ5ERM&VeUZ(WLvrf^Qz>i1(hAvKsNq%jF?c?RN|WWsrE{9Wlu z>q^hcS@Mp#`XM>2|3J!XucU0DZnBoM;>p7|BW^sw=vNQ%Il-BQ1VR{A1$l9|hqZL( zmQ5sgFy+AmcOWn7I}17cahoBrH8@G%o+MEEsO%4SFM3GPez(0*HQo(oe^cpb9O#8J zCQ(qV*{IH5I*H*}rGJk}+UP1zO|6twwaTS60yR+)pG>Y>poAtjm`~3-+*T*U1 zEObf7UGb!(Y6cE9d%lHFpK^Wu6!BEe9{p$HxmkJG?bQxpLjgJaq7+N-H_OzPTb%s(N3k zihf&8bq-u+5wSGfAQUZi^Tg>lC)&{^HI10*%ohdyY}s0M<+q#aCOfV+gHUu>9`PsG zhns)j_ZZZmZCQp%h&*=jr!YalN?*K5luofW-glvtB^PYJqyHJ9w$f5;MR6Y7r157< zD_RleO&k@ud%C|;k#klyeLh_J)bpum!s)Q|D~7_(xtXWY4p(0-qPJ><=4g&~U?<2P zhnh>~OmwbgnFcllf}a)3;-@F-`aEm?9Cb==V%H~1lQix|!G_H&TR)(Ss*YPVcnrCd zkiTol*@5-GInE0w?09~_6I zQa#Eoo{om^AP=8&?yvTQ>6tw90nPK4k}x{a;d+A&(#uc&W&a#zul=iF^xifE&9Z>M&V7BRaKp{;kn{2_f}wr%5D3jUT0~Fvi+ruMu!y z$J9(e1V3txN%=(de1jKREy?K}U=%;Q+Blq}nC$r5Ywh?Dl?AUr{X-K4D)?L+lP>cp zfs0VQd0g1AAdt|zc8Tw_@5#+`dCaH!4v}4PBYYv`ZvSjK!L4|$@aKY(413}OGPJZd zqas>0t10Wz-i(LS5qq42c(?nz>V85Iv;#LGbHZ+2=7H}trZTjo5HHFRqGB!}J~vs> zzdzOF+(8w|Y*d}Pc|48W!JX_jqWYd^H-%a13t>w-5my#7D|5K{d}!KW$RiMIvtY(% z7``6NlIxWkCmXgYcAPOfQxQCjpul42P9Ckr zjYibO4~6&{gf^n)zpypozTQZZ=~#jCd&X2hS`~Nkt?^9Jt%OB%Q>kz;ZEEjJ%DUuF zQ~CAz&kXyR>v1VXEqD9PwEgu6ES=0g?xMe~WDqir!FR@Va`SegQuUYz)Y&+#Zrgjq z-TPc35S>BFQCftO*j$|tOPWcnf|fx<743MK%NQ+fR3|5vhl@>lZ@=EVi7f_@AfKhm zJUEoTYho`-2mK}>Ej8-dx<6$kvcr@k?3q-)Ra0}wJNP^snkjWP+f=bKSV|LtZ#7zn z{AhB~iN}gK@LKkSt8(m&I6jmmf%uII3ln|-A!@Cl(18I=jx&>oL^y?)B6Ej1In}^I z?Fl@%yt8wvX>y!>N#fK z3j66P2m-sC@wqIWv=@AIROpN$9yWq9L(?Lsbn+ zsQ_Y)p{v+t6CCd6Us?QuW=-)S1BK@JrrJ72-Z)u)PDsK}Fo>D=C?}^Xc>i(%=IQ(I zg6zl+Qaht5xTc}}VVx~o2(0nwxwgw>t_H5uZfTTcbRg0WWg7c$9qYlAep>aVi&F5K zrGbYye<~Ik(SS%T9=A>x83f{H;!j+~hl?KsUhB^Q9mrS)jb)h+lp*vg0SC%BZRtgu z0)TdA<6QOc?|GWITLBH&E|yJ0cn96j)~Mb*o)x)F4lPMY68${(hf;L7ZnUM*_Pimje)%Zx|B=gyLpjmsPQ*A zIg6;_%ZB$>EAFZs7mpyJZ|=!K+>9yYAhU8`@XbCQ3d&O9wA$M5jgH-~_kMV15KA~X z7}es733(la`bk-qIAKj>ngSLUw-;a?8ZJM!g_iG+7LQ_$Xc`Ih_HI`ao10} zK-;Mii!xs0uOqwZo8Ok@6(6rlWuqV(xO3-TXV{?QO)4FbLbVmWDjX1$t_f4OO)rg8 zHm0kxk4|g;$vrW9ZoBl?<|mnily|}F&+4HhHGg(<_I!wWSl1vB-5*I{O{*N0UI3VD zaMi=ccR$u4MUO^mgV(55xF5;mK_C!jW2>yN+DsejcZrZ3$fH}gO>JrK&Q$EOSnr5D+-k~O&Z@kKNYj8b(g?jB=vMy!5}8sxZ`h<3>d&kBE>5J;;`{{6sF zu!7btgl+~8p>Q|iIpEk&WTt=A;NfZpCAxUEk0b{&p2gY$R=lJeD_1S z!RgcjA6SQ<&o65x6E6hMc&C_SGYg8y=RZ7@RE=r31V4r%)?Xf^c$=}t-4AFz1f{%GNblGvzdlSQ-(dW z-=~z88fm+Hc1sAv`X*#%W8*-g^SL5y<|ca&KA0K&q$Y+=mJ=4c~ztsmT2SF#_w`hnvZr438|46S=UXU zQVf42NeEe=NL3ceeu&rk<%J1}V)`n_9##JAp{~sNTeHN}%C=KqUS+AVCM8ikb$;pR zSqHNk%hMvL(zLiy6j1Xb$umP3MvC5cGP%uD?vV^tvvo!u?0l6Hl;A3^NFZZ?Kpd>z zgIH{4L$=F^t(3*V;8qrQc4@ z-r}e(YM}Y6L7TCLfjf%wL&dvEH0RH`g_8CzrTy!;>`!9 z020WtYz{M|^)A?iHi}r!{`Hf!mUO($u?;PWTaSw%G8G%O_maveWryHq=bc9IqBVaP zE|+p%kLJIMu8y4yKa>u1quFGFZPq>jG?E}Z_S9KeyT3$0UTXNv-f%|_rGP=x%*9M! zClU!m{t@2n|5?p1Fi=qyM_|*Jr9MAP5WKdC$FzXBAb1^QM^zSWJ6-z*WMN(s;*n~O ztE8@z9{36=O z?fG~`Zwu)`&i!W~YD4@y2ki+ykc+krfuLh@$m4qE`?nh=;T`iDD>m~LKn_!?5_KCBsr$)s4<7nJ@^^1!%yN;`B|0K&`i zZ_y}0vk7lm`r?tuSW~k99$Jd(f`0Ac#E7Bbqxw``mf?+q!``=M{*{~do?XHa1=8Wq zC)YP0Il_AkIcAyTP!|64CSjz6Q*QGXerEBTvii;~XE!r`W@Afml~;3`kc^WDS7IE-Bs85S92#haS@#%8p0SNV60cDmRT`muP-=Mu?*el<%ur+l_lgUAuE zA5{wnOF5Ix0@vsvxDdaroAMu_eUoS(1w-tgKYCI6EraZ;u+3jdTKsK;T@auMAe2LO z!_~zXa0^n8`lgF61ddbF6_g|y+goP#D1N$sy+7-1nrC3}nuFJ6dmw%2y&}c&-vu}l z;2j_jf&tu2dM!VHUiJ67RI5Y{V=`>KSODQx)%6Go}y)VVB>0p z%^@CFvi;5PK};QJAM+9HZXkC|825N+vu)Fx`&+8`_@sLcM`r*}CEW%;M0ST*&L%{Q zC1qD%d@me_dmOLeu2YUsZmYl3I&$|McrxMYto9FWVUeezy7A!o-Go6hBhH~C0rJCn zrY%Hy<00LfFOqTW5XhqKBRL49Oj;0xc_crsA9=3B3D|;E#eQ`?EscNjA-Q!N(v~DJ z$er{H-a=Wv!@oibVf)?7hW*oD;{K>-%G7h`d653uMkd^)41E}0 zk~WSbEMc1Yz%1%vO?9zITiw6`|G%HL5lE+T;P7f2@>XVDbMXY6z=~<-qY2OB>cFa4 zJP2ulIX(nJ`PdZDU8BfWYH;e&=7(PEC%bhFp{jAo+QAS8KdwV!7#+pBYT?L;!J8GH zExF=+D8JeRo-BEeeUw_cL((^M4gP*;rYSEx`@o3p`v$%6RyV2=W4P&`EZM2=E)lP( zUpvrWolEh26!mIK-q@JD+2+pa^S^nJ;ffO zD;oL@n1n5z817reekee*(|LPNG0W3>v2pQY+i;C4 zrf+~eTC$Xo7@GZF)~N@olvqE_nwxYT++?h%%`u8%%)mMZ(m?W4PL)mZ6%P}SG~>iz zP|89aLsda7`wUN%EU2X3Lt@R`T&w=}dJQjMtCi?2+4S&$B@n!ng^AC->ejRR(K`|* zTGMk5C2hjg`8y)l^?Sa=e=YB85#ymBV2>RSmf8kY+3!6OY#ZU8I^qps zI)mzBg>KG~SKv7zE~vud3~#abLjabqM&zrOeml@w4+6O=AJ*eZgeB@1NXH}1#3{kA z1&*ghy6>z0R63ifVyFpvJss2N@4tIsiiZQwIj|mhN?-6|oB%F19A?2r*3+bqtht`# zFOKz=Of$E%R8v$8c1ntkeV-!rauJA)hiwX3>Hz|}U-Xs`H(NGtULZ4DB*`33zzM<2I{%L4--I%`=HLw zinms~a{cSM6VwX@uS>Pol?+@jY7!$~Yvm90*Gs@>3 zn1p&BD!6{%J|`PN2QvfI1W}H+`H$^jD0`LObRlZwRV_%HJ=ZQt>nMp0liv)++fkeJ z-uXD4_V>c~!%|@fwkM4ioDQE{@YOi81a&@oU3c=X=YMQE(FiKe*rLIGM3AVVM$+1xvcmP z1>tu`QM|I~K6YQ57q>Fh_{g$xz(e|(fx=hX3pP61vL@vo<37CbZoJk4EHPGXn0B#c zc5YOGE)z}e$x&3v8}Juzs*&eu%h>4zw*Xl(sDFn_^a@|5GE*PhnwErXr-mnv>^A?(c7&>a;c(gr2wQI8M($AYgV|&0(*xQi4GIL*GJ$lJqw)5)FGcoJX2U zSeYlEJaIp;uwU2WD~0{w8k3o-_bqUXBahm$H*i32^j{Y7C5lz6uD+f(|JhwnP<`ElVED>v|&+`=*2yM_SMOGz%@%$KZS|yzX;(w*p|PVT%9| zbkbkmtd}r8G`;;WEs6;13$?%d@Xr3|n3oRz@B;Cuy&ie9JKns=PXKVE#%vB}x3SjO zH<+|iOGaDj8=vx8)v{d%!*XgZjO1`-6!|p>M|L$w$ z`w4Nq!#90{BL{gvDI+>B4iKTZ-J9-L6b@yvI_EF7&!Lm00esD){Po3RyN z)U}?w&iUPUpL~muaw|d9LrtmV!{crJ33re=W3?l>6!oxT|nq(_h?UAocSY_VB6T#R;{#?;Pg{cOnZ7-cDm_7WE8Z9HA zQFU~v>*1@G*I%}YB;e!c^8TnKhr6anQ`gAsKLpSZM4IR(4~=T+4A8Gh@0)}W^9d+< zRAUZRj?sQs00Fh)Dh97hq+Tf|yAqGWOAcQ~YS3-XP-zzl6b$pf@m6t<=*m|dc+nHn zlRIPHixqVTks$pAk(S28Eh<6qIq80-;K7oGP38F8C8g~EAF^N_esUT6^hwE|cu<8Td;^2&qCE zFJEZXSe#V%wrxO{$_?n#cO+@fP5qxJR5=0sT!xQD_&<-nUQlSVBsE0oil!#6_2VQt z%)(}XWPXVi?ZVH1t)pppIYt`9LOvaWjQyyZ)kyAfs+pVWnC!&ZdC?)UKGm9S!!aGz zv{|@&zw7hu1Y{J+5=yQ!x+~)zl{P=+^0MmaW?eg`#chm!snPt&)29LjL(xo9+*?~) z_lzAt=zZs)HdTXDzS;huMUFPf_vu$nOtPWKRd`3$eV5l?B((?#?4OI;(xYd&nBY6l z)QhLn^z59~*C@9=)%A7!8KcAYf^sb4B^2!z7`r$3Ic{{aMlw)gy>}%Bu+wUe zs|!=on7pe8?QoSjnuG8lU#srwCTw03fq{kzCCBy#xa$;p;mh?EA{EsZsVSQxWCvU8 z^e92-Hpx`(KUF)1t%}M$zs5S-2J$aVl1|g7BHB~~Nt{7-brtVl$L$^T?mI6DkXK=Q z12e0lUy025{QP`=4{7x#{?u=211HP7>FwBn4;dai@oy|Ud8Y@eG?mma*1yd!sA9xE z(&eIkUW-8k_q7k+^=vzM_|Rh?#ya<4A3CB%8;ZiHm>|FKOiHk4c?oCO;SJM8Dr3OH2D^?AM!tx`CY5a%U#jPUN$?Hx2# zF6?g)l8e7Q8XMw zKU&+{$Cs2G$jo1BzD;BM$%H+knVl}6CTZpS^4fV3e)hL&%G`MUje+FYUfZFj1YS0D z9`=Swr}iofa33}kf7jzn1GLk7jBV}J#!p^oL@!%@i1MsMI;L4M9s)|j`(!cb4rG-q z8*8@RpPx6J0M^O^YE>q{qBgE!_Ee599I16+-3!TXkg_k@c<9Wn8L@$5TA6y}${3jh z&CkD|8qtx1*64Nk>sv1^>MEC4gGgzMr`04*WHL4~O~hMUEd$ z`y+2WNB3)|^R%-xlApg+Pm|=zQ%~jN7^udUpidQrlqK2g1T;hU`;{j>4S_H~7Nrdu zs7@0yn47eKV#Eb1I&Pi5)AwxwPf|5c0q;O_kZJqpeiZWKc=v%De( ze^OO@-^n*^;pcP3jWW%fCX25VjSUT-N~URCmu)46BA=LD+c1;Brn-5$VrREW+~ISy zPkIpS35JJy2EbKtmCWr?_66y3Lt&{OGb$?=o~#FjM?CyS9Lc2X;@h)GhM>E+9Olb< zZ^+qMT$bTv@&|HNp*P;cVIo9HJX)NKbskob95L3BG1$!T`AbY3N05c2 zb@E^H&C=_hmXvMrjfksaUnwDp3+KA*U6M0&m~WizqdA^%5KIb2kN66?Qr#{8^i-$T z#c@qL;FGI%Zgi-kNxuCXJe2vvE6K^GBk8&>W#o9>+A1Of)Yn&9Vqd!Z41fMgCQqw@9&19bA1?7_UY(<4e?DtJisXuPpI3G}=t}u;;ws z`}C5T9Ol{6ua>YF_EBWiwe_EgHC3cM)9F3V;umRvb`*BA`k0PL)u$h^LpSpo9-U35 z?&#M`3U%;H*X$yJErd+sTh9j*scjiK@20mI?V6{5*X@BmnG_vK`F=SszdkfEdA2kioK8|*t*U#P9nB^N~1LH`5^xDUB4`Jtq;$+RQLx$xW!~&8QMePi$8yDN)L7ZtG2SRs^=!N*;4D76%UT0J^H#>$I={xWpU&9R``-6BM|!Q zni`(g6|-4RMNAPK)mkQKb+PAhCA82>Fd&$XN5hcQ&Jd!@3-uE+JV*6jR4%ROnpY6oKv+PQi=ocL69zd?m7Qz#2KG0fvHj(AF&yXN*-E4+1Xe z)07ZVT8X>D^x$VL{wx63Y`rHAha;fHZ5?D1U!(&LfxKYWT+CDI^@>uv=o=_yZc`$s zO~7ErsOFLV{)M?eP%%b6>?1cO+;2o6j_)5F$QfIz9lUb~8+zTA$^=mcuR3pLoI7XF zVwL)~LfFi#M}wlP0iqi<=)ax1(UG4Kl>NIK44JPc2|T+TYJChT@%Fx#i9^W-p0Hu- zyC(bR(zTS!v?x(`IEGd&;4yJu-iyS$P`8o=UMihQB~?i<0sqfgO5x& zgi|zh*(a&!cyIO^YKryAmvSgI*YMcb*s}4u@x_e;3rd}!kgU<|k%J;GQ!lrJjxPze zZr62c80)|8sg@p*F593osg(xO7lc}aS-vGY22vygXxisFP-0h0hM4iF%hp&M7;#_q zzBssHE_dpVwyq|e5pjDBL`~mbt{Fl3(%u%?Ndph+xBrI!F&kkpLldOtOV?>M@6M5{ zI`l?yC2=Tyf5xzWPsWl-QGHUwa`6SU&5^gg)wJc?)ez^2O|*sTP|5;^lqltE2}uEn z+^S#)&fkf<2_`@hKN%4jx?*>Ic2|0Hm>S#GNZghr0HVS1r|TWrqO@Fo)uzwH)K!I- zmsQKt+Gxp zZ+W@vQ5R3Y0Ft;h>f$N!u(ybHhvC@Fmbao~^HzI8q=o9NSE7MOTra?29wzZbpi`fP zXZ)<0$@fGfeij9g12gp@_p94UvdIZZEwy!jw0(bfsgJ z`Bh-6!9{ZP!O__2`(jcx-&{;!FZ#WK+tg zGxNhPs-=0~>FoNPM{TN0{mJ7#R{0RSb>w=_OCieC0sVGZ>Xd%rFns82SWrGq$0P3T za$n6+J-80QF;#4A+XLJ#SbR9M!ub{1Z(DK|@tp-F;W;}6u+m1f~m{=@%;#+i?o zCI+>(oF_nsHGbvD-~1;y2h?4oy$*!6#y5g0(DnRWPo?h7u3$dq@JQ_T&wM>fN1+M8m-#M=UZh5ZXXq)a+e~^DA`8Y3|`eN`^)tigG;xX+s zDaYS=TxsFrwEAbwy^9;J#j>QXU;MQ*qVJ3E-f)`j__C(JCj@cpM+F7AV2$KUlP<}7 zvIRhNi}r-^nmS~sCKwsq)YO?I9;@|WjGc$s?YWAyP+!?Ko_m|nCfT|%qUmSL!ETHz zn1DuF^BtlJsYtyo$u2Q&&$k#S`)|yp^tkEhii3Io?17mQyWPU3O{E%jl> z*Cv#)yFFP*-y`kp`s#R`(gbk`H}T%GT<*5?`7p6%?N$rC#$qVTJq7 zS->yRfnyba-v%th0wB};E(j8_$Ex8p$8SZEKtSVXp~HW~4(6AofZrN@Jb%02Y>)rKw{Gc>;l2eL{Y5-wo zv%O0ZUZ9wId;J{LXdxdbn~XMQXH&jO8_69CU7eJI2}iXbkB^4k-iI{0?rO=xgrgX7 zvk$V-!&4H>GYsAxKPoi8m*6o6p=Zm1W$q)06zf-#j%Rt1WD*lq*v1AWwrF@^(M;W_ zZ*sV-_P9ySDd)cNlI`$}*x_sz#hNIw^oQrZ$ksTLXSik5P&ORy+z|m@g3zBX!H``9 zVATb$Pok48)}+-SujiBV@h{KXkDQl2c~axIV&iFSeu>s6;U%M7Pq23{-Fjlp)MyMC z38@Xf)Dv;Tm1X%@a)hl_$w7dvmAwmMH-EdcMcm~?C=vqlF;l6pWX8LCr1Llq&h=TY z!ON>_6!%pyy?C=6pee{R+spYX-b7kB&QIqTzy$eT3@9JbKywEk7sWm~JBY+|8a$X0 zYc_K_s=M7|(s1fLB`!5*(ER4PfVb(y<+g`dw8PBD-f~Iu)B;(#!-Tnuu3dNFWwdAC zT!Jtc0SxRYSTOJo*eN9JQm}UFds+rcs9BCNdvlm6jmv46Eu6pk zDx4`9anDFDcPiee=Tr|RFEusXPKeciB~6>J%2V)%5y9zgu1lN3?+sEw7r$?_$qiOe zEMqKnAAw`m4}wV6tNFliqH@)u2hE zE13s)mm@TQm^%Ed$q zm6a`&z4y8#qpV6O<60Tn%3jw9WnGeyi)&w(%xm3yx%YRxKfmuE{&Zij^E&4|&hz%vOV(t~3pSrKYUr;1@Wy z9pT@^TU(sW%a`5@a=-;L@tw2vPr3e)yev-L2aj?mA z+?y=WV*P^o;C=aY3fM@go8cJ_O8!j0aSy(IvD2wr;;`rOexB4z_eam|J>VSsy_Lw_ zCN=xP!s6)tIiV~{p^qSdU)m{$E!=`0 zg&%{Ni6_mR3^trWzA*?M^IBAcC?~|2z~fTl5YG+r#h8%D%*}aC3cX<+AJ_Hr6fDT` ze+?nu)$9!mdis)l%I{5x?pqhPog4-ILLyo4!Di_!{n0(4j7^&3o6}Cz*j#hD=jg&3 z%GRRq*&UG?+flM*T2Z2N>TOdzmZ(@wu-7lnU5|#uGA4iNQ3E$Bq^t%?Kf1dA^5(Ve zmX%bh)EXz`UUl$Wt!JAzo|$;M{y}g)7y~z9>>mH6MPS~}g$P>+mfxKyISJqk$f<9B zKJuYhBdGk(@!l)RpMV!}J+{)gTK;EX=mF*oUE{jP5VQ zKaapyc&0>`LhX=N66^%iK&v~MjSHbMulp_+1v7JtdsF<-MV!~Wzx-gw!2sEa`m=>? z!Ms?RVb(UQT@S&y6}n9##f)rkC7Hg>7}G1zw`9(=xH{@a^3p(giG^d}s$A6zJm# zFf*&~o@Nd2qDz}jXfZm1eDW`;u^skyIiz2X{Rv-L)dStOB>sBuQd2qK9;Q+Z>fYR8rG3vrWA2ED<4RUlv@>t zw+VMm0~zh4cJw;Io7`BzFm!lMS?U2scS19P@sFziU)88%i}H_!1ZY?d{YkwXk}fa{ z99R!dJ@RC41|xYFsMDxAo_+od2CPuf}Ag_~v|TB#wd$y$?dPvf+*Fn^?`b+w{*=S5ewjXG#G_J<-aV0q+SkIOKP z&!qG#@k?m#rx#y;^P6baicLHUD8*MjeiKG~)eY9567_&Y<%h^#e=hO_5;bKnR396; zIT!GVkzyg|?InnL43^t^oR^Uz{3Z_<7&9DY7b;3*(l5C(7`= zC`AgM5?uBChtC59zLbyGo|{=?opBL6tTtTbk-n1m zqJSdq3K+mZFhhg(X|59>36UCU1F9Xm2#B&kBoEz4vmzk@&;3=4_G+P_?NptGm0YYY zK(D7`7&NI{??x!!YYfIQmh?{pom{$!Z?fkgO^K`Z`oDOlX@EGyp0RqT3YO6kSrO^& zei&|N_lWQI8I2w5?trRSP;)VlfPf4ZKIqL^**sh|ZP(s=Rg|M(Q`G{PmA}73Cg}<{ zUf4K(J|-quy~74#iaP&XJc;O66|b)#7yMHQ?0k2JDws`Gm=YIO%Q6#L?6? zg?f|w`dNcKD&0S)okFGMHWyi0#?&1s8(s;pc7diEa9VO9nWG^shp&6&9(m?%OxLE5 zrzO9jzMyPzB0Vg3jf(0uHavQT=hJ%H3^iuOB535%WAaepj1pib$3v3}J0#}u4m>`z zVk6~@Bl~Jm4mSkZb4nA(1%4I(lokCInX>qLDVB4lP)AxTo3rf6M4zvPPn&PKrC#)! zf(l zzGNdPV)6)7J01yb_JF_>I}bAOI5%O4`Y{vb_AEOkMNlS|Gmz)p6Ar3&ohdJQ5b6iC zwXG0Sm7_~JCEqCi*?$}Naxg`3`s9%z>O3@xM@EXI@DWK>%@PR02nxcB743A|ks~7?-pM!X|e{+cR)-7qV z;KuU>$aiIbC>@<$Q#v~Cz%Nd|^;7=@gvzIK)l^WRa!V(Wf663p$&pl*iErnPc8tEB zeHvPkGVE8b!omRu;x)yY5mY1yuDnsd-4h51-HhA!ibUim0QM|&7@KLtGB;3hte&a? zAyNWaI&UAVWi}KxQvfboe%Su%RJ|&WP?|bJ_GURLWy=$~U9LXi|89K#s-7*q-`O_<0}Ur2`josN z&t~N&WLd06I~_sq^J%Py6+)LAaY>EucH@5vly;j7sg>~r|F9ZTed=^QO{XHK+~p#U z)u7=B4v<6N0Hx2zCqsX(Kb4Tz>%t8U>B9Ns;WTTe1*FqrO;k|T2JtS(+Ef$BwEqLQ zgB%a<@6M~oE}zz1MTV)kYuMGUX(}xDgsckr!Z#M9C`2x+X}1`Fu;r5yw}*;luT*Ko zC7^?xyi{LCclw0}rhkj&2ls+CMQ2%_BmN8Af32*%Y7G!-pgw;d4|9pH;DDtfMFIl@ zIcJT;W}eThbkuDLiT&9p2etWanSeO$CmIdlh;0mYIF68Z1M)O{`8o`net>iKcawRw zr(a=q%<-#Fp4|3@V-*MrTDay;j|D>E0Pn4ZsC+95T2Qqzkfx8N$d-ga-i$A$a)1LG zLVI(x?c{U9_&{jb{=$`AAi&-h{$y>x-1Zsc?e#Yb^$-SAHC**jF4v!U3&bFZjb!qz zfBOhq+#2p86Oz0pr`Ok7NuH%U_8sZNp8|-G)6VAkpo`&kUM!lY(Vq72#$jJwpLUyK zb?uhn9)6EHkYc|8G%gO`2}0I%b@CHEe@+P9*I(jd>VPt3mag9`DTeiyK(fX+sz)MO z%8J#|e;d#yc{c-hmpqYM4(^yY5h6<~D{qmUL5X`F@|tLJ7tXq*=F}xw2nX}G6^eNQyT8aIgGjbgeON%PR`0BEI7FN{Q0J$ z;#~in2fF?O+`hA4hw^*yLRjA*VslSx0?6IpZTDJaDnF8b`dw09Rqp!_6RoVY$2S6n zhWZB^wiUSmaeg_<2CU!zrI637Ty#r^-ODm1@t;4lTVg;WDW252SKme60zw5)kGYS0 zXb-Uo2FsNRL@_j!J9*x9vkT}UOS@~3_FJoN?D?$o_-$e-W7MkujsQbZm=@^ns=g{$ z4n&9$`5$|)x<^h5d=@@k*+7v2U&I~4FlJe6P9OGO%$2{|FbvV?to?_A5Kt2p?CxG^ zXJ|>1MjLO3g#?6Xt(Fc=z2<%T&3A18t@vF26UF3m~# z=!cJ)`)*(OZ)i)`Q1xbyLoMGWJ5#q+>3Tf0b*&`tDrMv=HeBk*B-hl`cw(RvvYXu9 z-DjVX-*YT<=(p4(yyewYP)dq?+Iw08`ck_X*F|$V6TCo~4`9mfzlq+Ip!<94)~ze5 zs;W!Ab+-jmnG%Rl#NcU<@RqpMb)aVAQqeOjTyNR-qyIgzIW1Z*X(Bv|qrciR$ zyd~3ib*5E=_WJ%l<2}8cbNn8Vlf32+Hfos$><*V7AETy3$-kV)SpO5$G>_JgJ^3fx zu_iz(c(e83@>!&I(39q~Y(83hq8h25hp`)9VEV8mC1>RoVA#sk|{GjT( zQn8wi&RcknKl<^8VQA~$zuOYuJFC^Ki6eh=^`$38v#h7#(PC(kVjn^JKSd9;()He+ z6m=Zz5WZW!cxf6cFyvpXBG-w-xD$S#DIYnF4;(&|zLJl>x~ltx@=f)H4%TLW-c|%+ zEv?v7fAamK z5dB!HRj^bJPY9OwsG1Ur7v`%f#|cCRICzcrZ9h5(kym^6aSWql7SdaC(5H9n$TO#W z;$^DgkLl;I(M|rxgL>I%X>xAi6^7U*ho~TVA*x;L7frguikCY> z-`2&kM6Y+ZIYxLW<5Sif8ygn@|91o~_XPYO9F|zS8c>JRQ&T(g8eKNC7mWMB4L4_Y}EM5=?$paUvzp!NG@2t8_Bx; zDuRIFzV!4|u=XQtb~@}v>NfNq3tu`GR@oZI7Bx7sOr7d&Rxd{B1MvuVK-D`-@vifs zYTT0~`B5OHP$@(WgAj>w@6E~`W78-B7yt50fqJmC%uPi_DEjDnxiHJXoy`&!b?kNN z>-N8kaZjF#+}APf`!SLIak6(3Eq4twWD!8JCg-p*Mk`6euhLJ*++{H;5fL(x6c=X- zY&4R78Y-bBwx@rm=C_WI(wWn)23WB7YHuirpMFiLs(6r+8V*!;mcLkR2hqAjH|nst z6Tnq3q-C~fN+Uqaulm&ny^3?+u?l-&Y;h77Z}10ql0IY)p>K zqzTxZaWW0AQ9uFFPC@^zFTaU26iNHM4Oz-*d~@5g2CyQZs7`C0`NT*mG3O)9_@?b| zxnXfZ|J(Pq-QyTT{#+0JT*SxyA&If0HyVxQ&X3#%{U5#MRyUFYJ?1hQ_gI+8h|0rb z#!x@9b6QZ~sn_ZmZ;XQ7Fsrljp7EI@PVZ zw+|b79!5hhc+Yg00b%GIWRyNHgAARr_J#6nUk)oP-b^}BdvoJdjR2~xpeNl$!)Lu&h z3h)phCNy$uswP8R$~q8b&%wq8l8|$zh7uYZe;I_PHyxJUHuvd2dB$Ff4Y0b91lF{> zcLrI-3Fxt~^pNKHI4cv^7|_TAluG9JFg=ODpkc=!1VdW9exoq9j87UCm^<0m)V8+x zKi0$_(LU(dRtiv*=JJ0Y)sV4@V+&!v8YJ=RPE+)HkK-Kq1*(^}bPRM)N{7IWNIt%r{|DYA(SpL{=|K0r^ zFzQeQ|DMbF@x{(kq$K~GFBnmc+T>mpBY!L#lk2kO26~f+t2@^47N~z#$$ABO=s_dz z&b)Mr4NHTO>l5dr6K4C*4tE6Was6-F81hTi++E$1@G6pUnY4(c?;r_#b6w3isOs)* zTn_Qu?$wqu1UelO>H^_)$=8+NWERo@eWwiBqt?Y08w-*1LEA0&63PuGQn$2<;c`g% zF|t+FOhFOv21?`aIfs)3oR7rODbhBiO#5Yvg>4L#@78vw#gwU&r-58ZSzGRb|JE=u zPuF-Dr7X2s$1r)9N+X;)hn~Oi$jN!H|G=d4Dloe^OY463_BasaU1;gK9GrzE{Y#~u z$=W2B91{^luE5FDo}1Ff>QR~#28#Bm=dSY=DI>86yP99wDuJ`F)3BM7@9AiYpZ#_G zkd=K2f908$=F49nU9PuY>pgmw23$;RN7m25gJb+I3}ei)iJ7nB=Up{8PZTPzJq>Bp zKxuw)JwgTRa&xR@@MNcA{77-ttH>a@Wg*ZAcg-@7VK}F}cVAC8kz23$v{wB9mW%@-49ZILW*Zb z(?1p8>G2NyBxWB7G9(xt^n;VXuiJ`ytjFTeH|r%aJ`xJyh2ixFr7jNb^%KWP%M#I) zxnLqi?Tgo@OHMw}+>;#XQ_h>XW=(G)(78DQihZ=$HUqOJ7@SJv{oQ&n)K6lEFTvWh z_Wtas@Yn+Oaxvo5*7u7dX%TwiU_K8M*mH_re&O+cx1Pt*`PK1ym-h#%oePZPy#krJ z<#jUm1XmdZOOS}wZ|@FsZK}*lknWi>dIAl{qwF+2igFDVrJs%V3C&INgP_3M=kDI- z-P!LevG;e`&};h$|Fr-F>OazwFL-iNU+kD=rSuW*5e1eY>>&|JXUTrWMJG;=kjC8`m@cB?DOeB{p>1)O8)-6GPSMJsD23*7RuQg1wH-XZMCP zExNopIO8Nn70HZHji0C>zby%Enl2$M6ip6W#>N&ate){wU);V=-K^wH4eTWV5HFgg`csbfhgMy z*x;thnI01B*&zP&^cyIf)~oe?9REp=GAtY9n-CwFHvx#f{FriUTR;YIh_v^C)lAf_ z)R>kbKsqK?s#rb6{e1^yJ|pN6))|IQWh992c7Hi?x`_b{?E#Fg@Nic|V8y)6*>4HZGX|Q5G@Le0Nb1aDtd0_!SPg`#!!h%2P^k66~&sSbSeR^ipp%(PV zT6OG`M{460mCxm<*83a(>=$v`N6*amGMx1TmLwNDO3q5R zirfc}8RTY~Yz5^A)4%A!!D;-c4XG#I}ovTNue~;uSd+ zi7JzitOfRrv?$LWNOffS3s-AMNnIZ~0!Ma&=fq_PZBMa4_CIUBfk`0$5#C==8 z{3X})b=FQ7_F#joqu9XgsdtW{2sL-{kC?a!(nDzIb{0by?tO_pFvVh1?7pa@NB=bPVi?bK zixh5wvK23#`uhVAY!PEMOKSL@xy&H zr^@GCnb)_cWdQu}H_QaKKB@R1&UH1=i+Qh|H1vQLy{fVhrDo=T!1-y__p!JJtvO_8 z_XEoK9oTIy6-BB|7tmk>u;W4n6vN^p^xm0Z)U2wDMN8J}YXn~o&-2(fve$d~YY_b` zXvIoaYek9WFWjkDs||7v4`os5k@7yULDG_`s7@EdC2we8rEoYz%!mpCU9Hk zdtVnZ9#pJ~%hwgQ>1-`IQ%K32TxWlLUS~PD3Xm{EX=IjbY@fl)%eD=vAwoCIeu;Z= zdaA>l;gD{u^Q8E}iYn}QMzuSH*V(_dzvmnUFapv(W!nnwih-{!bk`r~$e~i?%4p0O zvsIQEbQ=DNm9aX+&0%*kOb?v_U-~R(Iw2>msz_PMFhgTGcS7wsz7%!HcbZz|ezxbh zd7|Q-BhoX^-kVG^r>d$48G3y8jdE*P$a*%{PdX;%Oj}lKZ6-zvmvGjZLP}hw>SYaF zW~J9(SxmPix%6*|&4;3?dUm*#+F{yMg<0fYFgNp7*wuHrRM^!QuL5vC3~c`k47E-D zwS7-MFIPvE)OhH{%lUVc4qa=IzTW*5Gy)!=KlvdCPo5Fkg!~1ULPXveR4m z_ft!g`~MBk0n;t;z$~ubx8~M6&kO6v7||((t!84UYNaDK1Pb|gH*(u-g8ZcJ?#-k| zn$6P?%i<-#}Weo^#V_Kt8vYrMTllHOd8U>SEm_$uVaBVN#@)gUc{b=I@ zn0kP~8y#-N?SzXnyp;^#;UDVU5ARN}w_O5+r~K3f`!h{vTwBr8n!JYralY}sZA1r5 zOVSt0&?kMuaK82z@vJ8WR=)tRsbY7nB2AgJ`J&T1nLsR7i#Y-9j?S9gz$D5p1@?79 zp`QEg4<5Em=YXY@ox>>#-j<4RR-)P^sN8p8N#wxb{iXT9H#fI2Ocq?Qmlv?&K6)Dt z`ri?Co2x5*Z2I zS!4W#9N=kl>5qw7F27dtheG~sohKs7COly^tp)|eg8qP0^$yq0Jfkk;I%8$O{ti zD(iqF2O5S);HAonx0C7&Vcuf6Y$~^0@u|UqxKj_MoTdJolL=wmWc9hYBO>L@IqOSF zv=|}m9|}o)eKk#*DmukwKlV;(oE2~B5uY74_^>tNq$I2@q;rF4l-72Yo~rUJbCqUZ zU}0!8P)S4N+ux7Rfcn;0obCE_J}zhd)lmRGYw#votA`GkvRic z%#5v$F>3zXd$6OHVNqG@#L~+!&yAFMjxU?*Q;Xb&R@@FLXBkKiliZl$$E6QrTr*hV zLJH7*x$Ec;fOXAuaF1Ujj2MX|0jn{qA8|u1c@h+KnG!$^!p#53u z=t6MMD~R$Gm2guqO2~O0{=SbL*g`O!nQExC6`q6eP&%FQ6}m$>qBOMYklo;cn>CvLI~pGrIA1b1QXaoQlmtQYPh`kG!@XtoL==qkb{#a7Y82>-4-ZHCBka zjm~P>YAs53quOL)!D#cfkW}fQCb7>@J-|(lU@0T}u=0h;!U0dQuzbv;a@-U>uxSd{ zaL_-$0bQ*%4pUqqjyXA&-;I$elfQUJiQ#Ur#}BK-yRzA#^c0GG!hcje6)BoTK=Dmz zX<2-ums?mKX#HC2Jl*Yu=cI%p_e5F^I%F^3o)E=d;HHl%`78V!*!ziDG*}w=cb3tl z{kydxHj=IzSiGP#M!JF4xPiEV3JdA(*sb?26xbkv!yj5xu2dQ;B!7^#{W-%!{e|Ed_$n-H8VU z1U}~Px1IAt4I#>m8sRY^ znUZ1Wvu~O_Bj9pzb)-Ni`jxk3+^_ud>6zc%;Tv=>6q@?dwx@Zwwjwa5^iY)1 z(Zi|PN%L=dZ#44X0yPV-+u!iWQRydvd2`nkX$inP_CvOer$ZB{&U>vBtolB~1!xAk zcW>QFz$i|)=_`3bBz9gi)=JV$7d{!PkK`xX2PZRz{+)dnOMA}YYk6E4UiFrT>EKg4 zv8V+&*g!E-{nZr0^Nk{hf5(K-4^B|w@Ks_{8+wb*1zG+ks4}Tb)<`(OiblkezB=mAUcuwwfL? zhmwEjJ7`6@R61OH86Vtl+~+UrE_~PjIukM*fql^9cnxSv6FOdHI54!1b#e}QAl|gO zu+XfpC9qAPB}NQwCAfDKOX!cyuZ7HS2m^NB5Ky#=F_qdJdRHDgc`Za0Wj>Qs=>M5Xp+17#R=6#y^%FtH--fn*7jLTyx?mzBvA z2~cWPSMpDL2I$sP0SFy0yxxZ|Y1KLp7~?XE^4&rF7_RF+SN!`7^cq)LoIVHh`>_b4 z&2naRA;m3bg$)XJc-PxQg4$1k+p`ot?w?xFB{xwo3Z#q`TGp@s&umxzUGyq2i%3}W zuc5cV+)PR_`i4LprS+Ik)9Lk^V^r%q<(z>w!2fcEiU>4DV}H;8XSQk?!?UQ0nsX;C zRxO~F4O}#Fd-K)>`Q+$jIisPp|I?n0YynaH&%%*=m(*guJMbPFGEzYp>%GV=krTP; zMQm3UN}`nJ6&CEy5)IqEm~^vrJ48=sU2QA0Vb&nL0{5xcG^zpt>qUgd7;Ul*JB9Ll zivg3sniTSTBXdzu+WKqu2=PJTim?trxT=XUCm)MH@4vfx{Xw)V6fSf8^dX-c3ts7% zSeo16e}Cfname6;Mf-^ik)gGxpiT<-6M!IhY<0>fn`5%7C6FMr=Z9wBfa**|V zDziq(`LB`BbiXnSLunz}j=Km%8IEvK|2YMOc^|_vp?#tR464V1EU5Ot`MG^JzN`2* zY)|Q?F_G4Bzj4S`P`tb;{xyx3fkh{FuRRar2eqs5q5bGLe+0T?zBJ2y3-x><+k2ra zkkZQO_RVt;>tSoRS}H;RD|-A7y?%6DevN_5$)oM*PRk9|J_0^W@p-xPc$Q64p?%W) zfi^kjpUX6v{M}zp%_{}dI)@%r9>paHOR4u*-;aKnsm3V$m8q?bIbA+DtJP5B;&qIh ziXRil!nA2o-&{a)$rrzfDIHl(bJ?Y9V$|s&fSaju@L6S-#aWA;A8LdFZK`SFYGs*a;u-S3X z#;;Mf7v}Naw5jz6gm-J+I?qW_t>7BByxn8Ht>X>*7>tccxUOd?5NNclmAuD(6+eT* zJYwb*?>ix2I=#a?-^6{dvAovk%)9a zk&OM(V|7I0-|U8fZs+V6_kR|5hzD%BY|Cty2rvCDQ4IyEq3gebjST_?w0dZ3-HU#@ z-9KYWKA*#&Vo~MWE$Q{s;c%szEh8l%t07;DVDd?|<)wgs@-2ip}^8-#@ zxZw)lVNT8F-7uCBB{q0(6mz)`Ljy4vDM2JVJ$v{$bP-rSCzH3$u%ur12gI-3rK|WtTps&8zGt8Jw;*YaXnY(h_Ep9G_~F^% z>y5=~2=`4KwlwM3kVknp)u%&YOUK>P-;$>yp{TxmLj1c!-nZkgp-HlCi>;exdb4S8 zrEQ=7LrzXMZ08+rzUekwH@hGw=ac@@c$)T76$`X1r?qNSe2tTYyV4&`6eIfdSxBdwfTPZYRZ2E8c`ZZba*>A8h+8k-NT;-OCP3LzP1Da8q)lRdblI zVd;3wrm?NFQ>S;|faxkkgy55S_-OK8>5JWOLZ{p0|Nk!66?OB7^|~1Di<$HId*hw*?_mA zP{y6d&Bb$vdXK^ojOwk?W`aT(<&ai#VOFx$d)HbOf$E+)FR7WAwi9O*%Da8R+R`h za;C0$T>j7Z3Dk4?dE8;+=n$71-$X+KA~FnI3o-X`ce2l7`(Eiv5PlAP&nE#@B;uk` zmxos$65>@DKac-?^452Ryo$?qEW^z>Sx+;urS^vT32i5NHel0P>h@Zb2Dfz3LA$f| zWY`$S!zpgw`9v&|;{1&7{s#FLBcyjyT3Xq6qhmwK?bnD(xvz=Ch}; z4ZUN1@}l%TME;JOY2(D9ZCqD}=)nFe#t8f$UztpYq3 z>z+Q(Y&#gPsEApKGvbABh5YLm0~d>Fi5O&4sUc$7*M&x(nkBRH&jkP?`YnODRA;#r zqj4ur;NJliJ(5CadEC-G*6=piE&}VPS35msBE9rKeq_s_N@V9O@^?|GWRrJ?I%@g+eJb0uVhINu5H+heC9lLj$oNqt z9%8{+^Endb|8IR>rOcDd;juBy4PbwPv@zQ4CZjvBFCW(E#K~RA^>=xoxg9csvUJ`j zjqksd5uLETLGs+nuOyu;BB3*MNSs~md|+8HDT>z3X}aRaWO_O$A7t+{U9?tVky|+| zuyq6?-}7ATP%OBCrSqQY{+OHcJMjH{DmHh{yp96Y=_=cNpy?P+Ma4k!6*>SJJbJo& zfcE-F6XtP5+qtfMDo2mvHBDMb=<3`7zg+x7y;cNi`D#9eN4!|jbl(h7x_s@$_9ZCo zwItX5KxDj)2^`4OS5@W%$@r{(m_q1+1_DvCeY_c?(!Ah{v>o&w!$5`)D{fIkME0>S zqFg0iq>pQg#W>FFPiXl#vcjo8YM6)k*4mh;fFm&NERy(udW9_$?K4Pg?5nCINc1A~ zX=GdM96l=Jb$Y!DQ6oDU84m%-W4{|PyKgT6#{K$3S&unqn3fO}(Wg)Ku_wW_zA_%d zpA+{Qb7Jq=igTE0<2l_hRv|!vM($Kn`Q+4PT2BRG0)5H4n^UK8fJN9qZFH8=DH2sA zA4-DJWql^R>=lo;u?JzO#bxN0BwM;R@C+h2{RqNj6Q}wZF4NG6Ir=;wj(|XT!1iF< zuyntD4k9wPVYF$+s35-PK=hkeP{TUR%w7fq3krU)l+96=!V=FGso$Q`G+U3zYz0x&KWI9 zG{|89i~aukfgu+V&Ss6W%m=>idF@S$4GwYT(HzW*0@<+S$2f* z|8}1bjHTbpsK8qj4vK(Ppu0V1%HhtjV8cGUE4ZgP13l&yU^%E~vzo!vCaAj{CWhjTJBK0@!`a zH8P4x+|gqriyS$F1ZAj|XvsI*;1Cj4%lQz|M66I3M9~eAw~f*0_iKb^L4pfGC)8Q9 zHoX8GBf8J7;3%mfsaM83c|XGm-GnqV5LukPZ1T@Q?C?=^nqf#-J-CAt%$#F4=7(>z z&3=$YgbIcgVlM<$9eLh;aHs;)&hHLBZoIdkpe++BVmhi_i{}(u|y8wyv!J*Wq>gJ%t#^wPdTK5tg$^EA6uP zB&9U6mu(yd7zdyP8|okD!sSMS!U!)(_KUIy&FmmBdPta_AWiJGVgy{VL=X`4;edgY z8a{!k1$!l5O-WPi5bOwtWs)b@^RHOO&_eZo)+fWY(Ww{82`{Tjw?bV(EcbEDcVo@q z6_^p%!lCA?7FF{YQxfT045l}@cLg{4Xd)9`d@l*ZcPGfDHHgIaLED7%ozrr9WH@A` z9^N7sIvzQ$H} z6Bce>AU7P2?A68yJcJ>Fjf0Lq7b!i-oedbM&zhhcSd=A!qxBOR;I{1)-ei5f18ffh zvT4f9N{Ze`zs3u_;O`yI?(*jCOv7Yx!XdUb)o8nt=2%KmSb#n4b4_HevXO zac7>YDthBLex+=xMKCeN7oZ8B^<>@52pdJnF5L24sZcb7WQ(G%XEV7yLhiV&zCyLM zdL_uBWj-b+D~; zz@%d?E)msqliqF~(fI`e@qmvRr+Wn(C=l{m}@n275?}S(I&nDDp19X;N_oojmb!sm%2h7gjZ|8DR>u!piQ7|Kbw~ltVAD9ol>^q z9aK1Y$Y~h&RE0=Ezux2rlif~-)2<5s_0kT}clW&(emPxnD?PPMFTgTvq~avmpX^3h zppugj9X7Vzt1;!Vk^`CY>Qi^_#-nHXYMd6IFHAuwJ<9azb6mNIp4=*Pu8*c!oDjB1^U z^SlUu9&+_Ix*nTjPJKF(-iIeIdgdW-m#kcl$5Pw2sOm{d^MPl@X)vb=)aX#}rs@7k zcMxWMJn74n$JiX@UiEkF*`0r2cSv%dZNdWNt^Skl1Og<@@W4gUt;Ne-_u^KXw(*j8 zX|MH!*!4;ISYFX4dgXUfoYyQ1p4HjMZ-uTQM?umuxa{o&4zKl4^j-n_p1)KBWO6--?RC#|!C$@wxAcCjv4Kobx55J73R?}Ft^Zovy*QreA zW6kQ0L?}r+cJs1gc*xsjXhq_(9HS;?`Kp?`QLel1z3|Hob7hBRN5Q19!-~Dr2udPb zuelRvvPEp9(krTj?)~*y%Naq}JWMX_LNleGRr)ARgXIyfDEp*}E zE;&8C*c(mL!^5#E#Rx}AdnO-za7^O<*k8;}-JknrK+Swdb~xGr|+ zstheHEnU&dSmyc+ZxU6AVBX7?f81J|uvK98!5dyvO!_{VoJ=#AjY1_eU?DoUL~H`R z0_#Zmx6+8qd_T*-;wtW|1`h4LZNm&Af&%mDmGy$y;DVB(qiDZzO-CFdj;B7bSeHEO zog_OoqJ z$zjS`Jv7PW%FVV10b%sjmV}{Rc$hEK4&9%8(UX6Kx<~t0h_>oUB8E0MlZ`J z`E6+AQ&~O)5r@dQIey0NHKyvxX-aJa@_O6<@_PN}BRsWF#h=J=TY|7*QK3C@U!i@a zhBeRXp9jPzVY?;9EBJsUWVYF`3BFEmn!3+LPDOL-^|r z17^({RpQW$BmP%562dc090LSS)0pXhu*xo6M2C(ku5jMxXgMD{#~@jhW(oaPwiQRe zGn_!bQwJL!&&K^c$^p-R3Ny0P(4MV6NHc^#J@2_|euhV1wjR(Pi%ra`Jy*LP#(8;4 ze+l&bl>*Wr&kB-g%$Xe9%TUCW)Sl6mEhVmv>2;4^o`&&@O;ea;hi6n4(%NOiK?9-wHduZE)phed%=)sHl z47w<4evMhsAtM=*9LCItRw&ul==>mx8Ub+jl*|8?wCkqX)|oY9!{Rh(&j0eqWdG_8 zTe=QaUC}5`FQ}HO_36pb=8Xt?J*3Gzy4V^1##bW&bITHupi0&LHB&C^@2F}$=}_RA zF)Co#L0=fGf=>?^QdZS{a(*r-|N9_Or)}Z<({aR{8AG|#)cC_3l2iT~qwwTug9NdsVVDIPyMdBXI>D%Jvy! zt4{$KK&J#ki`y*Tn|GY`c%pe0nZ%jvBcxvm<6kd8gslGTJor7b*EeDuq*(aaPyi`* zpckb8=_frLh)&E@_8R$)Nzx02h0G14c~w2I@;5Lkm-keWR_554_UB^CQFFsS=|3pQ zY_^lht|xl6xO?w6s+sE_?$}xQgd17RbSKK+cxuCG?@EI;Mh*>y`63zo$(^?(!-(3j zfTczzNrAtJ4x2OA+ycVB5#NZ9>#%4iwfv?VO}g!#B%m^N)Wm-IVx`e{1J}*ScE0eG z*!@AhO4p!y;7EnAPVnwK{UW!(H+Xz~$1$H`$^7*@&sRZaECMJ3rpBod;vLcBT&f~C znR4Nk4%V z32QzGCAE7HhQ6~~GM?`t&IXZ2{AP?@1LhiaYInaqF0d-&4(adca9O%nE* z^TQ!ce*Ts;fA>oO2=zQs@-t331#i>#pN(k@oJDZ z@7CZcgt>bjL3Bk|gr6$6R3+ILb5VvbEBPF$*_ypUHy)=o*TGJB;27M?g7nR4@49Wi z@Rpz-tix`DYi~*usNb>2iK8lqI(UAYHhfRdUd$V$Js7KRDA~La!!q|}YrPXk2}t0w zVRoWl?c;e|3F%IF#M{_+!_Clt@B5 zS=uB!Q$(pyN!hgsS!3+mNZKS4db5m?w2(DRWC^W`nHKvpykrJhYDhDd`Q6Vm>Fx8q zuHX5?A9I{@?sMTpP$zm^bJH^uGnD!C77^4yxaN4{ExQ}4d( zt9HK-;C$q<)ohD>y4wAy=#jxSRwUDx<9klfCjQk;g7?)Z7d>@-{0~xF^OotfnAbpn zN(`g@u(xZT1Mfwyj?v9nyabWwT}j8sER&`e1bCKjr5(3*U9;KZJHz&M;GRo#YiT>u zz~HC7PHV>f`5@MJu2oB~et6_s&vn6%*0w8jT8^>n?L)(ZdRt8IL)zGcxRu_f)CGe(L-e`)$ux>nJpBP!E4LuX90a59>U-fs1f?HuymV zLXt0);ttBNlw6%iS25&qj(ntL(Q=)b8U*649VUgtgL-nO;{*Ylu?lk+QZ{7QLk z;#7I<%BjwOWSz$a#G3yNf-{l|i+~{&W~s4V$@6d0D>gxVRuUq6%Y8y6hBvGGUI1)CNFl7zjHYM z&Rx{I=%euPt&q^RwTx?0SGp}ehU6?8A&m!29*b%_(HA)LL3WimOl{Cnc3Y}ZH99wR zvC^AERws=nFLRGQKW}Bhu+(UEBCE9+)_j^pDPF=SF;0Rw#$DN_mBJSJ=P;k}K)O=kx5-pOt)!;I1+? zT{+3EhjAwVktmrxK{AWTokjt|xQOl@8X8QVdR))0)*UyObsz9{cx*^$`DStCtODe5 zY|_ah;yE85vO<*zT}{XC4TrooOSq#-;p*7fqAw!h_N%<>+h<5sXpDw@%WKhHdx`eu zDn+}a6k=z2`O(gIx#|b(YRbLO1`T(B6{7q5xMU1hDoHfk490z0-)$*AD5x^Utlc!V z?aWQDDp9T=r7+L(3Im3i-Ic|7!tU$KbVL&A+o~t`vIe?q6tz8XgIWHe4PSmI5vp|o zZeM^~Rq=41rF!B;W6ZLuED`=b<_<`=B)qU+#eMyrrHi=SThDw? z>l=cTQQ*Kes-CE$ofE=aFe-Z6H1<{+9crd6I&nYhrfSNK#T35mrkMl)Tlm-O+nt|j zes(FKTdG}SnOzZ1pJ>&7p=VYQ%S?;i;alKl4%y@_spqmR&F|Se?IL{{9PUi~9<=RD z0C023yLIPQ;?HPIX=fDv^s=H`ok}H`bR?UnO)GP0;;wwCDNqabqm`h=w-I{3!O5^` z2-?GgXr5}>Wa~}ZdPgqNfQ7d&L&WY5`z+3B|DZG~h%DoZi5uZ*Vcwr> zjO$Cjmv;G^r&4DosQ1%?Yr4E&F>_@3>ll!Ltq$(-_C>^!VbN{emq2D78P-%++qxQ% zCf3s{^6rmz#6Aqw%1u_`g?Er&p0u@0S9Eh}T3NbkrNhegN;M^c5yGMH8Dt8dUck`I z(0I+vkYU!Z*s8C#LvP4TiK$Ur#p?8NooVMFx3Ci(%r@JY{LYNsHiuV?T)sp**fss~ zZk-R(&>N(WliZb~n}=rNj?Vr3eBij8*vCVm&kN^+h(npQZ+DEXRGkDZv`<`}PP)so z>B}Bm(`LhU@^M(j+gL}iefSX}|AP2W$s2~GeC$1Vd;8p-Y!iAae^%}~DrE{~#-2sZ zL&53o&Cg;evyW*qw!T@+@0X1PKK-g#C*X8VuxWg5>@Ghy;oeL%WM&>N<#Ajt_Xbj! z_m1bB$IIU=FJ6}FIa*H(yIW6Iuv07E4PYeaPBH6B8krZDSQ5Tg((JmOB$$Ouc{VZo zdIrq6qXIs=ml{*qrrREzxLF-96MXK(nUT^*yJhW<9n{av%d}E}EN8bc!Y5J74Bx#p z;T9}&h zzM@t4)D%7i|N2$OI5QgPw-*$HrHvj(G2Go3wLi4ehW*1ZldHTapIOnRq+KSiq8vw3E^&E^{F7z5 zQ=`j$RwiXmENTF=_W8g{O*33{gdguvq<-^F-6pzA37%!S!B*&7^!hqRr=RDcE5av) z_>q0nQ><|#_PgyAPe{t|I6-|dY;Tu#Ntm+4`gZSZfP@sc#rN{vXT&sj=84|7zhk%L zQ9Aox?If_9@x~S% zcEKr9x~lc|^CQCDp92~-Btws;Eag&OKI|7(y}fMX{?-oZ2KU!CQ5K1gTwNz6UMHsE zGMX>v#EjB!o%j^3r=4o28K5p5#-um&e1T%oGTj!n)(3M_`bmDL{j^(`X>%)!>qw}#x3`C%pI^F&I%%oG5e?lPZEbcMEMs^gY&7wN`1~BbFUqWHXTh>K zr}n*GGARm-uh%HrSOO4Pymc>wn$0$uyqZC1-4$7-2>e|7#WNq(i;JHar}{rHsqVga zc&5natf2G9)OB=LDiAJgpHEN0ySKP??uWPj$UMw%JZW<-Uh#cI{571mLW5ShVNi`I zD8lu1J^?(9zGs~vwUTFSa#dfEmzH~EKtD%DjaMf{Z!=(IB$V)t>6YWkZaKx*dv?7xmGSKoI5hP2yJA(z={v;=c3#8> z=&QJ#YNWVdzNb1hEzwMSa(S7&J>znK=?#d~R;iYrZx$8|ra=6sM8 zu5cmUC%alRBHstmkwO-)U&x7Dyiy{FW1)a&SR1s*ua z@``EyrA&H?>KW#sPSQ$x;=7uw?euZUgj$=Uwav;>LR`v6L+!el^(~WD$5|^IHC}Gs zz@tdaF>p$CA5<9FW2s1yS{D@J+z68kBq{1~v2H-*aPeB)>@8NQit?qw*j3BvX^?7p zxw`lSnxJ-U2N{@qcg^d%i)-ji{NlX4A9j0k{3I{dk!1n)vX?EDtF{e_oYIM95Xj;$uVE! zpn1M~=%HR~-b}W+0y}taa9!qNPrB&w!8p^#5@*k=?q`D(niOAN^BT2$_gtN;>SS%! zy^+r=7>R^}O9lvP4jB1NBdzKE3d@HHqc- zxL(-IPLxP34L+lZe`ZCLJO227!>^OZ&DlJM-ZY;txziEgFVadoWv`ZvB&A~Jjb zmHsXrw%oX$*RS;5l8dfCO6UtLNmf|VAF~)wznnimyG^J4_4AoTFlRF{3W#ILz{X$4 z4I^bB&mxzpmpQrxmq8P)!#C{D(uitlb?SF=a-tH&^TerfHmbrx-1zGXdl_WyU!}!t zibJ68ebA7x_(V82YXJvw6ikA)=`0dX+j-~GAmW&M(9LpZY};7_`Cr;!bN2dN zH__-^4hT7NtGaeSg*;ClpR1+CR{n_35|&R8gtI!}3m#-}Nspw72x&aSr=gP?{bM3v zU{>As!`xZeY`*;EgH1lylnpPXrnod6JNMz3{0c5juN!EssQ%@R^;;m`bBxJmxOq`( zv*e(>OVO>vTVx!mN_^rX?69@aEAMB~ea*Cxt8pO@G)(Un5;YuE;mK=eD{8E`ITjw!AaB)3~G#ndt%{Q>@I+i z!gsY^b$hZR&?dO$Syh{0V$*4d>VTact22<|GrBwOn8-Y`KWkFElO~sc`Rn(Rq;Caq zX%W}g_0(TKu#WNZR)Ua_#v=p?{q}5lvY%%v`u5_f^~neG`36N6f8e5jN?orTUb=Pe zIji#YK_A^h_53z?@ZaLY;nOBX?`L`K(qVIZul0t|jy;rMUps4=Ijk+U25-g%IppsV zLC(b4o5~6esk2QI&DO|3pxutovmq5(8V^sV*geO2o?*Bq4u9R*+Ev|=n)zYTd#vy- zW_XGAv7%~1A<b!0Yy^BQGhRVFk@I-xjvjcd2_aZ$FuO1Kyj$dg zuv|SZd?J?*cAfrSR~6G3`07<-O=hr9{4+1laG$X<(|sEFB-mY7ir)6rvG!n#zH}pt z9Mz6e9t8KWvgzIUs}w5hXhFwbltC`Fdq-e(}s#NrQKy%kHRp z@_l>%R9)o2!NCnaLnYDEgG)sx(XS!eL{H3?c1wNEsDI_*PmH;F-_7|g7au~BOS)92 z@`l|R?>BMkAj7Y)PATi(G+z6lok4lkI$>QtWri9^u~XO%pKRtnpPPj)??O&rxLLSa zI+%-^6fgAtU%sGdpLkojyXpT#sNxw1@4fK z7v$=WX$D6B>NmZS-(nM_%xhmfxe?#be)Nf((ABtO5Hd%e>TlFvX>V}TcokmrB=ULn zNg-m7!~7{>?oE)ZSbJyvF6z%wpJ-{~35CZybz4qOuR*{$f3~6zLDGm$A67|{IziY5 zOwgoKHQOvAr!=S8{O&4G^r!;bUaf^$PKeh-Ioi(LGPlLK6UYrs401LEoAS9nbnaLT zHxayO2Z0-|3dKqJk7;P>6xNUuI~R7gAMHit<>AO~ITYu{m?f{en8urqzkUOy^WTdB z$RyfW`}2%v|LHz#7^w}QoHXa9X{Bpc#mx|#&&QOGVm&wwI1x*~3p&U*SQNDjQmehu zuOJ}^$p9T=aH^i6R&-X_CVK{Z{ce)Qi(V=jW zQFn=!Xq204<3|&fDX-rB1jqzbVCiY4%jd|xOGifsc0PV7dIg9;WU-QK8#<9PVMyl4VN#k*^d3YLpx0L zWdGQI>>!T)Ua%fkPa18~aiVEy=bImO40%#Grh#?6Fa~U$Ou{hPI6Cjh!?PuNLi_z1 zC9IdU2?q9wZoD75sxk^=eZhX_tz zhcO^?2PgMrGiLq7SH|_X0O8^62jK*q6ucr#%U2EJrJ>iv(T2Yc9hU_geI$uamf28U z=90CW6)VdSF}`7{9Ro9?tp|@CX})<}FEVJs+bA78h+5 zkPW81flRmGd>y0kqai$#hBk^joZi!2iHSS2JwxeWBAf*3&BVS&j?jg+@Zn?|Y!?>) zz4b|el|(l1B0ikZmRhyn(wyjYHhRHO_)|O;Y$+6#Bww8P*lI?b8xMJV(1T3RQrs03j@8&afUBX6y#v zP{BW&j_}>RJ+A1HHhHH2hgfenLFVVcK4)UeM5Tr6lR(ZU#i5y}0dK5IXRp>P3;D_jlke^0G zB<%Ydn53~5>%(QBU_nsb&2SB_5Gia4Tv*)903G#q-`|U02L>_oLRDp1Utm%eu$An@ zEe`F(Qi^#l?3_!dF#HKj zncH%b0d@o$=@9SM*kt|TR zZ_BY1WJj6*-vThn0|ReP?b4Q&atFw;HV5n)cV_yVPR`69Y>0BGG$}D8h%@Ofr+q69 z4s-XHEe2KZ+-{f4VIc%Z1^F2;+g8VLb6D$mB#`LHAvi+01Cwh!Wjy};T=s7%VBJNe z*BWuKo2Ps3Klz)~0=P^J0%USKpKpGD6v!^}OP&Ls;L#O?H36sDC;jUg=T(p2a}1v2 zC?*0QkV6KwQy$^JQuB1|7Dy5)f{iS=gh-RF{7FIkE7YFJWh@YeumV(9&{=jy?bv17 znamh)$k9f)4L2d-L-B*1)uA5#b&TJfsVs~2_D=0ThEyo+WjsQyk@N|UMzRpn#(G~! zp=r4zYW07x!4Kb}g61Fz*sw)|vx*geV}o;ewCgu^-^fC>VnIAkN&-G4@SC*BMdsI% zMtOQVI0Ug;=g37bNHb0AVNRfaI1r8=g z_Cf(>*4&7u;Cc@HB^@t@0S%hH*XGWoiTtG;kWnxb&QBrTKJ@bnd8mu$EJrDN_F=Oj zqZ>oHoHPIU0HYsx!1iGb8YzVR2jv_?pS1BQ#W#mzt0J|Jj zI4p*ZBKZKKa>OzM&51@cKTWRo!pqT|1vTMOxq`*;+T>Fu;D8xI*?H`Df=65f@}y2D zm^^%FK}%W)?ZoMt`Dw+&Et44sUlDk`Tq`e5Lem_YAF@S#da z;gBkLr~pHAb}ff9-fn?H(gn=HdW7@5@&^qP7>sM*z?d# z&cAIHZU-PV8B0LD93Gc{f7J1GSI}<*!v|Vz{}RlM<i7JJGM2qK)ta~4=* zu;e1EMh6ZU31DsItA1B{tmFR|kRRaYXO#|PQu@6xL1ASi$Ecfm!=}CfZsE-HZuK_j zPP9~wwFC%7<#Pqe)7n}eKTV(0*WV%~z2hpUFVuh_oB(0ab6P80q&VUWE*-xp)jli-QI<9`gmO4%Ygitu}IbTDh-vTWGw^9v7Xuyc&0npA|z-@W)x4#&J?~l9C>iLE!B$VZbe@WnwfU+CFR7-+jGv>h)3l#?r z9L12ndK)F!SQa|d)e2tI|L_OyIj}Y{8cm;1%dXa*3eUDU<`|-sW&CPkPEp|1hgh%> zrqJH&km(-05clF90<^>mh;Qxnv}IQtx6`jiaMLd~_CKt06GPV~gaLMf1|e*CDEBj# z_=5_G6`+(D?oO(zc2FBfB-w?|u$86OxD~s{!V7Q};}1>k!{7e568& zJ$O;lKH1>SFMu92ey2%ioR70E`0knz-?>wDr(aHlILkv2&o8)Eu0HidNw~ z^R^+T%;i@Dgnn@6<12GsKl06ND1`MT$z0rTZ?XOAtwr$ky!7oA=vHtafqMOoxq~SZ zXy#%*`;WK|7@4Ps_s%XMj#Ikz94=w--2?dE1c#_nLuTq}GE!JF3R5I!xx5F%+h%{# zv;wB7yXrVI?=L3wySq+FjYF&xbec`i32)*TkU(n-NV!j6hl;ST^C)8<$FP&U?pwXs zp$muN1xD0V6z8h3P%?Ci+zu)jb6*s2O*e#=%FIPB31gKD>%N5S{cY3LF6dGuXI}b? zcQ!h4LbA4F;nm=QJnSN)7>hcA*70`d4$C-*5hl|mYwcq=&^_1S5y10 z1cpK7g*y}TgTy|<0TYwHyxInOwXD{ty=<;0xh7=tN@D0XJ&k~YS8mSdJlJie?cpJ+ zW*p6!>n|K)y^iWKF-P7BOEKZDZb{%1W)$K3|Od zpy^08&7A_Knz&<#SylO+%3Tqys#;nzO{_+^>BVN)i^#0{Rc||;UN{{^6Np)b)1Cdv zlo*>SIedE?{`)$pn5Bk1yu|_P5TE8m#b!CVOzye)))_rdNlC5BsZZ(QgRN?FbcW$l zQW-dd_Ooio$aoEH5m^-$hXo(w1bQgm*%S$#*;$`RcGyqZ*0G*xvc91(DfqNTjABz@ z=Mq_U_C-6J4`QzA?28;Z6N9Y$$)xu7_D4z~RBtw0n(aH@M*FE7(lHw?kYgx9dJ2Un zs`mJs^w^tgbvmN#UHp_V-7CL~OS3=;vi}dWqAbQ9<_^rP-_P!plaVgd)x-ruS&8ha zK7ECKu0OVtQs1i`atfHB{-z)bs~JBx-l6>*lp?HanC6O9soZXM{zDnr=j`-qwR;NE zB;PyLF7>tkY0N>1hb=W-9Ub-b$afms5LFxw#yQC-1$06>&f zWBHkAjZd@ZXv~afb3M@l{6yVf`VpFxvaOc>gBj0sg7M$>$67zFxm5#{s;m{ohBttJ z@}%wCw{Lt?DeQQQXEK^d)&(|HiSk!r5R%;BdG(Wzp*91D);+c_M?$yg^5pGUiY|?l z1bSKR1{SGiHNJtf`_uqkWwLH{#Ue~CXyIeD~8FID`aIMT3gFT)@1Pon=D z1~#PQf#f~5acJ--XGF}dU#DumXRHo08YEBWKH=hS*p-V*M%qkp3_q^Vz+t`+c2Tnn zGjg~ZRCPnqb0)t_z2L2_Qlgv7qIRBW~?p(gTjTyQx^Ee2(m8?sbgPQ~Dl4jUb8sVsc^aVU7YhL@Hy`kD%#6q z%*D5y^jtJ-<}8oR1`{}Cy48-P`qW^YapNT= + -
    +
    @@ -35,14 +36,18 @@ - + - + - + \ No newline at end of file diff --git a/feature-toggles/signin/signin.html b/feature-toggles/signin/signin.html deleted file mode 100644 index d0cbe3325..000000000 --- a/feature-toggles/signin/signin.html +++ /dev/null @@ -1,3 +0,0 @@ -
    - signin -
    \ No newline at end of file From d522e0c899af8583b7bfc79155c42c7db9716d3e Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Wed, 9 Jan 2019 08:28:28 -0300 Subject: [PATCH 034/606] fix identation --- feature-toggles/auth/login.html | 88 ++++++++++++++++----------------- 1 file changed, 44 insertions(+), 44 deletions(-) diff --git a/feature-toggles/auth/login.html b/feature-toggles/auth/login.html index 48974549a..37010e5f3 100644 --- a/feature-toggles/auth/login.html +++ b/feature-toggles/auth/login.html @@ -1,48 +1,48 @@
    - -
    \ No newline at end of file diff --git a/feature-toggles/auth/loginController.js b/feature-toggles/auth/loginController.js index 47ed91a8a..82f9d7e71 100644 --- a/feature-toggles/auth/loginController.js +++ b/feature-toggles/auth/loginController.js @@ -3,7 +3,7 @@ const app = angular.module('app'); app.controller('LoginController', function(AuthService, $state, - $stateParams, $window) { + $stateParams, $window, $mdToast) { const loginCtrl = this; loginCtrl.user = {}; @@ -51,10 +51,6 @@ redirectTo(redirectPath); }; - loginCtrl.resetPassword = function resetPassword(ev) { - $state.go(STATES.RESET_PASSWORD); - }; - loginCtrl.goToLandingPage = function goToLandingPage() { $window.open(Config.LANDINGPAGE_URL, '_self'); }; diff --git a/feature-toggles/index.html b/feature-toggles/index.html index 58c9a2a46..7b1ececd8 100644 --- a/feature-toggles/index.html +++ b/feature-toggles/index.html @@ -46,7 +46,9 @@ + + diff --git a/feature-toggles/user/user.js b/feature-toggles/user/user.js index 7b30b8518..f7f8fe2b9 100644 --- a/feature-toggles/user/user.js +++ b/feature-toggles/user/user.js @@ -31,22 +31,6 @@ User.prototype.isMember = function isMember(institutionKey){ return _.includes(_.map(this.institutions, getKeyObj), institutionKey); }; -User.prototype.isValid = function isValid() { - if (_.isUndefined(this.name) || _.isEmpty(this.name)) { - return false; - } - - if (_.isUndefined(this.email) || _.isEmpty(this.email)) { - return false; - } - - var cpfNotNull = this.cpf !== null; - if (cpfNotNull && (_.isUndefined(this.cpf) || _.isEmpty(this.cpf))) { - return false; - } - return true; -}; - User.prototype.isInactive = function isInactive() { var notActive = this.state != 'active'; return notActive; diff --git a/feature-toggles/utils/httpService.js b/feature-toggles/utils/httpService.js index 97868ab87..c5ea2b1bf 100644 --- a/feature-toggles/utils/httpService.js +++ b/feature-toggles/utils/httpService.js @@ -3,20 +3,9 @@ var app = angular.module('app'); - app.service('HttpService', function HttpService($http, $q) { + app.service('HttpService', function HttpService($http) { var service = this; - service.showToast = function showToast(message) { - $mdToast.show( - $mdToast.simple() - .textContent(message) - .action('FECHAR') - .highlightAction(true) - .hideDelay(5000) - .position('bottom right') - ); - }; - var POST = 'POST'; var GET = 'GET'; var PUT = 'PUT'; @@ -44,20 +33,11 @@ }; function request(method, url, data) { - var deferred = $q.defer(); - - $http({ - method: method, - url: url, - data: data - }).then(function success(response) { - deferred.resolve(response.data); - }, function error(response) { - service.showToast(response.data.msg); - deferred.reject(response); - }); - - return deferred.promise; + return $http({ + method, + url, + data + }).then(response => response.data); } }); })(); \ No newline at end of file diff --git a/feature-toggles/utils/utils.js b/feature-toggles/utils/utils.js new file mode 100644 index 000000000..271decafe --- /dev/null +++ b/feature-toggles/utils/utils.js @@ -0,0 +1,15 @@ +'use strict'; + +var Utils = { + /** + * Replaces the original backend domain by the local one + * @param {object} config configutarion object + */ + updateBackendUrl : function updateBackendUrl(config) { + var restApiUrl = Config.BACKEND_URL; + + var restApiRegex = new RegExp('^.*?/api/(.*)$'); + + config.url = config.url.replace(restApiRegex, restApiUrl + '/api/$1'); + } +}; \ No newline at end of file From f835c645951b1f97c1af3d9c480bc0e6037f51c0 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Wed, 9 Jan 2019 10:39:11 -0300 Subject: [PATCH 038/606] Refact index.html --- feature-toggles/index.html | 16 +++++++++++++--- 1 file changed, 13 insertions(+), 3 deletions(-) diff --git a/feature-toggles/index.html b/feature-toggles/index.html index 7b1ececd8..1d2f98f5e 100644 --- a/feature-toggles/index.html +++ b/feature-toggles/index.html @@ -45,15 +45,25 @@ firebase.initializeApp(firebaseConfig); - + + + + + + - - + + + + + + + \ No newline at end of file From 805978e1cf7b0892faeff7ea3ea2b676b1103208 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Wed, 9 Jan 2019 10:41:28 -0300 Subject: [PATCH 039/606] Remove unnecessary methods --- feature-toggles/user/userService.js | 5 ----- 1 file changed, 5 deletions(-) diff --git a/feature-toggles/user/userService.js b/feature-toggles/user/userService.js index bf87f286c..009e9074f 100644 --- a/feature-toggles/user/userService.js +++ b/feature-toggles/user/userService.js @@ -5,13 +5,8 @@ app.service("UserService", function UserService(HttpService) { var service = this; - var USER_URI = "/api/user"; - service.getUser = function getUser(userKey) { - return HttpService.get(USER_URI + "/" + userKey + "/profile"); - }; - service.load = function load() { return HttpService.get(USER_URI); }; From 03911ad00749561ee184c1c20faf165b59e37ee5 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Mon, 14 Jan 2019 09:22:06 -0300 Subject: [PATCH 040/606] fix issues --- feature-toggles/auth/authService.js | 21 ++++++++------------- feature-toggles/auth/loginController.js | 8 ++------ feature-toggles/utils/httpService.js | 14 +++++++------- 3 files changed, 17 insertions(+), 26 deletions(-) diff --git a/feature-toggles/auth/authService.js b/feature-toggles/auth/authService.js index 651ca6d08..c80b693f8 100644 --- a/feature-toggles/auth/authService.js +++ b/feature-toggles/auth/authService.js @@ -1,13 +1,13 @@ (function() { 'use strict'; - var app = angular.module("app"); + const app = angular.module("app"); app.service("AuthService", function AuthService($state, $window, UserService) { - var service = this; + const service = this; - var authObj = firebase.auth(); - var userInfo; + const authObj = firebase.auth(); + let userInfo; let tokenLoaded = false; service.resolveTokenPromise; let loadTokenPromise; @@ -101,16 +101,11 @@ return response.user; }); }).then(function(user) { - if (user.emailVerified) { - return user.getIdToken(true).then(function(idToken) { - return service.setupUser(idToken, user.emailVerified).then(function success(userInfo) { - return userInfo; - }); + return user.getIdToken(true).then(function(idToken) { + return service.setupUser(idToken, user.emailVerified).then(function success(userInfo) { + return userInfo; }); - } else { - service.sendEmailVerification(user); - throw "Error! Email not verified."; - } + }); }); } diff --git a/feature-toggles/auth/loginController.js b/feature-toggles/auth/loginController.js index 82f9d7e71..31fcbc706 100644 --- a/feature-toggles/auth/loginController.js +++ b/feature-toggles/auth/loginController.js @@ -33,10 +33,6 @@ return promise; }; - loginCtrl.limpar = function limpar() { - loginCtrl.user = {}; - }; - loginCtrl.loginWithEmailPassword = function loginWithEmailPassword() { AuthService.loginWithEmailAndPassword(loginCtrl.user.email, loginCtrl.user.password).then( function success() { @@ -67,10 +63,10 @@ } } - (function main() { + loginCtrl.$onInit = function main() { if (AuthService.isLoggedIn()) { $state.go("manage-features"); } - })(); + }; }); })(); \ No newline at end of file diff --git a/feature-toggles/utils/httpService.js b/feature-toggles/utils/httpService.js index c5ea2b1bf..9b1e472ef 100644 --- a/feature-toggles/utils/httpService.js +++ b/feature-toggles/utils/httpService.js @@ -1,16 +1,16 @@ (function () { 'use strict'; - var app = angular.module('app'); + const app = angular.module('app'); app.service('HttpService', function HttpService($http) { - var service = this; + const service = this; - var POST = 'POST'; - var GET = 'GET'; - var PUT = 'PUT'; - var DELETE = 'DELETE'; - var PATCH = 'PATCH'; + const POST = 'POST'; + const GET = 'GET'; + const PUT = 'PUT'; + const DELETE = 'DELETE'; + const PATCH = 'PATCH'; service.get = function getMethod(url) { return request(GET, url); From 563c2df752b0fc52273d6942aa3f3d5a81db10da Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Mon, 14 Jan 2019 09:36:52 -0300 Subject: [PATCH 041/606] Change var to const/let --- feature-toggles/app.js | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/feature-toggles/app.js b/feature-toggles/app.js index c46c021b6..5def0869f 100644 --- a/feature-toggles/app.js +++ b/feature-toggles/app.js @@ -22,8 +22,8 @@ $urlMatcherFactoryProvider.caseInsensitive(true); $stateProvider - .state("signin", { - url: "/signin", + .state("singin", { + url: "/singin", views: { main: { templateUrl: "app/auth/login.html", @@ -50,15 +50,15 @@ app.factory('BearerAuthInterceptor', function ($injector, $q, $state) { return { request: function(config) { - var AuthService = $injector.get('AuthService'); + const AuthService = $injector.get('AuthService'); config.headers = config.headers || {}; if (AuthService.isLoggedIn()) { return AuthService.getUserToken().then(token => { config.headers.Authorization = 'Bearer ' + token; - var API_URL = "/api/"; - var FIRST_POSITION = 0; - var requestToApi = config.url.indexOf(API_URL) == FIRST_POSITION; + const API_URL = "/api/"; + const FIRST_POSITION = 0; + const requestToApi = config.url.indexOf(API_URL) == FIRST_POSITION; if (!_.isEmpty(AuthService.getCurrentUser().institutions) && requestToApi) { config.headers['Institution-Authorization'] = AuthService.getCurrentUser().current_institution.key; @@ -73,13 +73,13 @@ return config || $q.when(config); }, responseError: function(rejection) { - var AuthService = $injector.get('AuthService'); + const AuthService = $injector.get('AuthService'); if (rejection.status === 401) { if (AuthService.isLoggedIn()) { AuthService.logout(); rejection.data.msg = "Sua sessão expirou!"; } else { - $state.go("signin"); + $state.go("singin"); } } else if(rejection.status === 403) { rejection.data.msg = "Você não tem permissão para realizar esta operação!"; From 262f28adc7cf87849d6b1d55f70eb0c73ddfd0f3 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Mon, 14 Jan 2019 10:37:40 -0300 Subject: [PATCH 042/606] Create user factory --- feature-toggles/auth/authService.js | 6 +-- feature-toggles/index.html | 6 +-- feature-toggles/user/user.js | 58 ------------------------ feature-toggles/user/userFactory.js | 69 +++++++++++++++++++++++++++++ 4 files changed, 75 insertions(+), 64 deletions(-) delete mode 100644 feature-toggles/user/user.js create mode 100644 feature-toggles/user/userFactory.js diff --git a/feature-toggles/auth/authService.js b/feature-toggles/auth/authService.js index c80b693f8..6cdb91b2b 100644 --- a/feature-toggles/auth/authService.js +++ b/feature-toggles/auth/authService.js @@ -3,7 +3,7 @@ const app = angular.module("app"); - app.service("AuthService", function AuthService($state, $window, UserService) { + app.service("AuthService", function AuthService($state, $window, UserService, UserFactory) { const service = this; const authObj = firebase.auth(); @@ -158,7 +158,7 @@ } function configUser(userLoaded, firebaseUser) { - userInfo = new User(userLoaded); + userInfo = new UserFactory.user(userLoaded); _.extend(userInfo, firebaseUser); $window.localStorage.userInfo = JSON.stringify(userInfo); } @@ -166,7 +166,7 @@ function init() { if ($window.localStorage.userInfo) { var parse = JSON.parse($window.localStorage.userInfo); - userInfo = new User(parse); + userInfo = new UserFactory.user(parse); } } diff --git a/feature-toggles/index.html b/feature-toggles/index.html index 1d2f98f5e..617129481 100644 --- a/feature-toggles/index.html +++ b/feature-toggles/index.html @@ -45,13 +45,13 @@ firebase.initializeApp(firebaseConfig); - - - + + + diff --git a/feature-toggles/user/user.js b/feature-toggles/user/user.js deleted file mode 100644 index f7f8fe2b9..000000000 --- a/feature-toggles/user/user.js +++ /dev/null @@ -1,58 +0,0 @@ -"use strict"; - -function User(data) { - data = data || {}; - _.extend(this, data); - - if (!this.current_institution) { - this.changeInstitution(); - } -} - -User.prototype.changeInstitution = function changeInstitution(institution) { - if (this.institutions && this.institutions.length > 0) { - institution = institution || this.institutions[0]; - this.current_institution = _.find(this.institutions, {'key': institution.key}); - window.localStorage.userInfo = JSON.stringify(this); - } -}; - -User.prototype.isAdmin = function isAdmin(keyInstitution) { - var managed_institution = _.find(this.institutions_admin, function(institution) { - return getKey(institution) == keyInstitution; }); - return managed_institution; -}; - -User.prototype.isAdminOfCurrentInst = function isAdminOfCurrentInst() { - return this.institutions_admin.map(getKey).includes(this.current_institution.key); -}; - -User.prototype.isMember = function isMember(institutionKey){ - return _.includes(_.map(this.institutions, getKeyObj), institutionKey); -}; - -User.prototype.isInactive = function isInactive() { - var notActive = this.state != 'active'; - return notActive; -}; - -User.prototype.hasPermission = function hasPermission(permissionType, entityKey) { - var key = entityKey || this.current_institution.key; - if (this.permissions[permissionType]) { - return this.permissions[permissionType][key]; - } - return false; -}; - -function getKeyObj(obj) { - if(obj.key){ - return obj.key; - } -} - -function getKey(obj){ - var key = obj.split("/"); - key = key[key.length -1]; - - return key; -} \ No newline at end of file diff --git a/feature-toggles/user/userFactory.js b/feature-toggles/user/userFactory.js new file mode 100644 index 000000000..bc6c3539a --- /dev/null +++ b/feature-toggles/user/userFactory.js @@ -0,0 +1,69 @@ +(function() { + 'use strict'; + + const app = angular.module('app'); + + app.factory('UserFactory', function() { + + function User(data) { + data = data || {}; + _.extend(this, data); + + if (!this.current_institution) { + this.changeInstitution(); + } + } + + User.prototype.changeInstitution = function changeInstitution(institution) { + if (this.institutions && this.institutions.length > 0) { + institution = institution || this.institutions[0]; + this.current_institution = _.find(this.institutions, {'key': institution.key}); + window.localStorage.userInfo = JSON.stringify(this); + } + }; + + User.prototype.isAdmin = function isAdmin(keyInstitution) { + var managed_institution = _.find(this.institutions_admin, function(institution) { + return getKey(institution) == keyInstitution; }); + return managed_institution; + }; + + User.prototype.isAdminOfCurrentInst = function isAdminOfCurrentInst() { + return this.institutions_admin.map(getKey).includes(this.current_institution.key); + }; + + User.prototype.isMember = function isMember(institutionKey){ + return _.includes(_.map(this.institutions, getKeyObj), institutionKey); + }; + + User.prototype.isInactive = function isInactive() { + var notActive = this.state != 'active'; + return notActive; + }; + + User.prototype.hasPermission = function hasPermission(permissionType, entityKey) { + var key = entityKey || this.current_institution.key; + if (this.permissions[permissionType]) { + return this.permissions[permissionType][key]; + } + return false; + }; + + function getKeyObj(obj) { + if(obj.key){ + return obj.key; + } + } + + function getKey(obj){ + var key = obj.split("/"); + key = key[key.length -1]; + + return key; + } + + return { + user: User + }; + }); +})(); \ No newline at end of file From f9b138cc4a2630f5f5c48ccc73a6b20bfee367d9 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Mon, 14 Jan 2019 10:40:25 -0300 Subject: [PATCH 043/606] Change user factory function to arrow function --- feature-toggles/user/userFactory.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/feature-toggles/user/userFactory.js b/feature-toggles/user/userFactory.js index bc6c3539a..ad723a231 100644 --- a/feature-toggles/user/userFactory.js +++ b/feature-toggles/user/userFactory.js @@ -3,7 +3,7 @@ const app = angular.module('app'); - app.factory('UserFactory', function() { + app.factory('UserFactory', () => { function User(data) { data = data || {}; From dea227e27287987b3ce8e474a907906a0eeed937 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Mon, 14 Jan 2019 14:20:34 -0300 Subject: [PATCH 044/606] Add loadCricle in login.html --- feature-toggles/auth/authService.js | 3 +++ feature-toggles/auth/login.html | 3 ++- feature-toggles/auth/loginController.js | 9 +++++++ feature-toggles/index.html | 1 + feature-toggles/utils/loadCircleDirective.js | 26 ++++++++++++++++++++ 5 files changed, 41 insertions(+), 1 deletion(-) create mode 100644 feature-toggles/utils/loadCircleDirective.js diff --git a/feature-toggles/auth/authService.js b/feature-toggles/auth/authService.js index 6cdb91b2b..412c99514 100644 --- a/feature-toggles/auth/authService.js +++ b/feature-toggles/auth/authService.js @@ -96,6 +96,7 @@ }; function login(loginMethodPromisse) { + service.isLoadingUser = true; return authObj.setPersistence(firebase.auth.Auth.Persistence.LOCAL).then(function() { return loginMethodPromisse.then(function(response) { return response.user; @@ -106,6 +107,8 @@ return userInfo; }); }); + }).finally(() => { + service.isLoadingUser = false; }); } diff --git a/feature-toggles/auth/login.html b/feature-toggles/auth/login.html index 5a83bff3d..78bc20069 100644 --- a/feature-toggles/auth/login.html +++ b/feature-toggles/auth/login.html @@ -21,11 +21,12 @@ -
    +
    Acessar
    +
    -
    +
    - Gerenciar instituição + {{feature.name}} - + + DISABLED + ALL + ADMIN + SUPER USER + + + DISABLED ALL ADMIN SUPER USER
    + {{manageTogglesCtrl.features}}
    \ No newline at end of file diff --git a/feature-toggles/manage/manageToggles.js b/feature-toggles/manage/manageToggles.js deleted file mode 100644 index 15e8eb2ea..000000000 --- a/feature-toggles/manage/manageToggles.js +++ /dev/null @@ -1,9 +0,0 @@ -(function() { - 'use strict'; - - const app = angular.module('app'); - - app.controller('ManageTogglesController', function() { - const manageTogglesCtrl = this; - }); -})(); \ No newline at end of file diff --git a/feature-toggles/manage/manageTogglesController.js b/feature-toggles/manage/manageTogglesController.js new file mode 100644 index 000000000..c3ec818b8 --- /dev/null +++ b/feature-toggles/manage/manageTogglesController.js @@ -0,0 +1,16 @@ +(function() { + 'use strict'; + + const app = angular.module('app'); + + app.controller('ManageTogglesController', function(ManageTogglesService) { + const manageTogglesCtrl = this; + manageTogglesCtrl.features = []; + + manageTogglesCtrl.$onInit = function() { + ManageTogglesService.getAllFeatureToggles().then(function(features) { + manageTogglesCtrl.features = features; + }); + }; + }); +})(); \ No newline at end of file diff --git a/feature-toggles/manage/manageTogglesService.js b/feature-toggles/manage/manageTogglesService.js index a0a005a5c..10026b687 100644 --- a/feature-toggles/manage/manageTogglesService.js +++ b/feature-toggles/manage/manageTogglesService.js @@ -5,7 +5,7 @@ app.service('ManageTogglesService', function($http) { const service = this; - const URI = 'api/feature-toggles'; + const URI = '/api/feature-toggles'; service.getAllFeatureToggles = function() { return $http.get(URI).then(function(response) { From fad8a81ed1ec212709e2a9e13b566c7f4f80cf5a Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Wed, 16 Jan 2019 09:04:46 -0300 Subject: [PATCH 046/606] change feature toggles page layout --- feature-toggles/auth/authService.js | 2 +- feature-toggles/manage/manage-toggles.css | 14 ++++- feature-toggles/manage/manage-toggles.html | 54 +++++++++++-------- .../manage/manageTogglesController.js | 20 ++++++- 4 files changed, 64 insertions(+), 26 deletions(-) diff --git a/feature-toggles/auth/authService.js b/feature-toggles/auth/authService.js index 412c99514..a0c131067 100644 --- a/feature-toggles/auth/authService.js +++ b/feature-toggles/auth/authService.js @@ -128,7 +128,7 @@ executeLogoutListeners(); - $state.go("signin"); + $state.go("singin"); }; service.getCurrentUser = function getCurrentUser() { diff --git a/feature-toggles/manage/manage-toggles.css b/feature-toggles/manage/manage-toggles.css index dc0988154..2060ca896 100644 --- a/feature-toggles/manage/manage-toggles.css +++ b/feature-toggles/manage/manage-toggles.css @@ -1,16 +1,26 @@ .content { + display: grid; padding: 16px; + grid-template-columns: auto auto; + justify-content: space-between; } .content > img { height: 35px; } +.content > button { + border-radius: 50%; + border: none; + background-color: transparent; + color: white; +} + .toggle-content { display: grid; - grid-template-columns: auto auto auto; - justify-content: center; + grid-template-columns: auto 154px 154px; grid-gap: 10px; + align-items: center; } .toggle-content > md-select { diff --git a/feature-toggles/manage/manage-toggles.html b/feature-toggles/manage/manage-toggles.html index 69110374a..98ecb52c7 100644 --- a/feature-toggles/manage/manage-toggles.html +++ b/feature-toggles/manage/manage-toggles.html @@ -1,23 +1,35 @@ -
    - -
    -
    -
    - - {{feature.name}} - - - DISABLED - ALL - ADMIN - SUPER USER - - - DISABLED - ALL - ADMIN - SUPER USER - +
    +
    + + +
    +
    +

    Gerenciar features do sistema

    +
    +
    + + {{feature.name}} + + + + + DESABILITADO + TODOS + TESTE + SUPER USÁRIO + + + + + + DESABILITADO + TODOS + TESTE + SUPER USÁRIO + + +
    + SALVAR +
    - {{manageTogglesCtrl.features}}
    \ No newline at end of file diff --git a/feature-toggles/manage/manageTogglesController.js b/feature-toggles/manage/manageTogglesController.js index c3ec818b8..280ad6559 100644 --- a/feature-toggles/manage/manageTogglesController.js +++ b/feature-toggles/manage/manageTogglesController.js @@ -3,13 +3,29 @@ const app = angular.module('app'); - app.controller('ManageTogglesController', function(ManageTogglesService) { + app.controller('ManageTogglesController', function(ManageTogglesService, AuthService) { const manageTogglesCtrl = this; manageTogglesCtrl.features = []; + manageTogglesCtrl.modifiedFeatures = []; + + manageTogglesCtrl.logout = function() { + AuthService.logout(); + }; + + manageTogglesCtrl.addModifiedFeature = function(feature) { + const featureFound = _.find(manageTogglesCtrl.modifiedFeatures, element => element.name === feature.name); + + if (!featureFound) { + manageTogglesCtrl.modifiedFeatures.push(feature); + } + + console.log(manageTogglesCtrl.modifiedFeatures); + }; manageTogglesCtrl.$onInit = function() { - ManageTogglesService.getAllFeatureToggles().then(function(features) { + return ManageTogglesService.getAllFeatureToggles().then(function(features) { manageTogglesCtrl.features = features; + return features; }); }; }); From 6c59a87fdf44a0ebeec09a8d93653cae30cbd65a Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Wed, 16 Jan 2019 09:41:47 -0300 Subject: [PATCH 047/606] Create method to save changes of feature toggles --- feature-toggles/auth/authService.js | 3 +++ feature-toggles/manage/manage-toggles.html | 7 ++++--- feature-toggles/manage/manageTogglesController.js | 6 +++++- feature-toggles/manage/manageTogglesService.js | 12 +++++++----- feature-toggles/user/userFactory.js | 8 ++------ 5 files changed, 21 insertions(+), 15 deletions(-) diff --git a/feature-toggles/auth/authService.js b/feature-toggles/auth/authService.js index a0c131067..aaab660bf 100644 --- a/feature-toggles/auth/authService.js +++ b/feature-toggles/auth/authService.js @@ -109,6 +109,9 @@ }); }).finally(() => { service.isLoadingUser = false; + if (!userInfo.hasPermission('analyze_request_inst')) { + service.logout(); + } }); } diff --git a/feature-toggles/manage/manage-toggles.html b/feature-toggles/manage/manage-toggles.html index 98ecb52c7..30f254985 100644 --- a/feature-toggles/manage/manage-toggles.html +++ b/feature-toggles/manage/manage-toggles.html @@ -16,7 +16,7 @@

    Gerenciar features do sistema

    DESABILITADO TODOS TESTE - SUPER USÁRIO + SUPER USÁRIO @@ -25,11 +25,12 @@

    Gerenciar features do sistema

    DESABILITADO TODOS TESTE - SUPER USÁRIO + SUPER USÁRIO
    - SALVAR + SALVAR
    \ No newline at end of file diff --git a/feature-toggles/manage/manageTogglesController.js b/feature-toggles/manage/manageTogglesController.js index 280ad6559..0d21a51e1 100644 --- a/feature-toggles/manage/manageTogglesController.js +++ b/feature-toggles/manage/manageTogglesController.js @@ -18,8 +18,12 @@ if (!featureFound) { manageTogglesCtrl.modifiedFeatures.push(feature); } + }; - console.log(manageTogglesCtrl.modifiedFeatures); + manageTogglesCtrl.save = function save() { + const promise = ManageTogglesService.saveFeatures(manageTogglesCtrl.modifiedFeatures); + manageTogglesCtrl.modifiedFeatures = []; + return promise; }; manageTogglesCtrl.$onInit = function() { diff --git a/feature-toggles/manage/manageTogglesService.js b/feature-toggles/manage/manageTogglesService.js index 10026b687..3c5b7bfab 100644 --- a/feature-toggles/manage/manageTogglesService.js +++ b/feature-toggles/manage/manageTogglesService.js @@ -3,14 +3,16 @@ const app = angular.module('app'); - app.service('ManageTogglesService', function($http) { + app.service('ManageTogglesService', function(HttpService) { const service = this; const URI = '/api/feature-toggles'; - service.getAllFeatureToggles = function() { - return $http.get(URI).then(function(response) { - return response.data; - }); + service.getAllFeatureToggles = function getAllFeatureToggles() { + return HttpService.get(URI); + }; + + service.saveFeatures = function saveFeatures(features) { + HttpService.put(URI, features); }; }); })(); \ No newline at end of file diff --git a/feature-toggles/user/userFactory.js b/feature-toggles/user/userFactory.js index ad723a231..2d6827c47 100644 --- a/feature-toggles/user/userFactory.js +++ b/feature-toggles/user/userFactory.js @@ -41,12 +41,8 @@ return notActive; }; - User.prototype.hasPermission = function hasPermission(permissionType, entityKey) { - var key = entityKey || this.current_institution.key; - if (this.permissions[permissionType]) { - return this.permissions[permissionType][key]; - } - return false; + User.prototype.hasPermission = function hasPermission(permissionType) { + return permissionType in this.permissions; }; function getKeyObj(obj) { From 21e5b7ad1d99392f38170df091cc1b8d53134bd8 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Wed, 16 Jan 2019 10:22:54 -0300 Subject: [PATCH 048/606] Create message service --- feature-toggles/auth/authService.js | 3 ++- feature-toggles/auth/loginController.js | 17 +++------------- feature-toggles/index.html | 1 + .../manage/manageTogglesController.js | 6 ++++-- feature-toggles/utils/messageService.js | 20 +++++++++++++++++++ 5 files changed, 30 insertions(+), 17 deletions(-) create mode 100644 feature-toggles/utils/messageService.js diff --git a/feature-toggles/auth/authService.js b/feature-toggles/auth/authService.js index aaab660bf..795b81bce 100644 --- a/feature-toggles/auth/authService.js +++ b/feature-toggles/auth/authService.js @@ -3,7 +3,7 @@ const app = angular.module("app"); - app.service("AuthService", function AuthService($state, $window, UserService, UserFactory) { + app.service("AuthService", function AuthService($state, $window, UserService, UserFactory, MessageService) { const service = this; const authObj = firebase.auth(); @@ -111,6 +111,7 @@ service.isLoadingUser = false; if (!userInfo.hasPermission('analyze_request_inst')) { service.logout(); + MessageService.showToast("Você não possui permissão para acessar esta página."); } }); } diff --git a/feature-toggles/auth/loginController.js b/feature-toggles/auth/loginController.js index 98397524a..661365fe9 100644 --- a/feature-toggles/auth/loginController.js +++ b/feature-toggles/auth/loginController.js @@ -3,7 +3,7 @@ const app = angular.module('app'); app.controller('LoginController', function(AuthService, $state, - $stateParams, $window, $mdToast) { + $stateParams, $window, MessageService) { const loginCtrl = this; loginCtrl.user = {}; @@ -12,23 +12,12 @@ var redirectPath = $stateParams.redirect; - loginCtrl.showToast = function showToast(message) { - $mdToast.show( - $mdToast.simple() - .textContent(message) - .action('FECHAR') - .highlightAction(true) - .hideDelay(5000) - .position('bottom right') - ); - }; - loginCtrl.loginWithGoogle = function loginWithGoogle() { var promise = AuthService.loginWithGoogle(); promise.then(function success() { redirectTo(redirectPath); }).catch(function(error) { - loginCtrl.showToast(error); + MessageService.showToast(error); }); return promise; }; @@ -48,7 +37,7 @@ redirectTo(redirectPath); } ).catch(function(error) { - loginCtrl.showToast(error); + MessageService.showToast(error); }); }; diff --git a/feature-toggles/index.html b/feature-toggles/index.html index 73fe9063d..1d9747d21 100644 --- a/feature-toggles/index.html +++ b/feature-toggles/index.html @@ -56,6 +56,7 @@ + diff --git a/feature-toggles/manage/manageTogglesController.js b/feature-toggles/manage/manageTogglesController.js index 0d21a51e1..b771dcae3 100644 --- a/feature-toggles/manage/manageTogglesController.js +++ b/feature-toggles/manage/manageTogglesController.js @@ -3,7 +3,7 @@ const app = angular.module('app'); - app.controller('ManageTogglesController', function(ManageTogglesService, AuthService) { + app.controller('ManageTogglesController', function(ManageTogglesService, AuthService, MessageService) { const manageTogglesCtrl = this; manageTogglesCtrl.features = []; manageTogglesCtrl.modifiedFeatures = []; @@ -21,7 +21,9 @@ }; manageTogglesCtrl.save = function save() { - const promise = ManageTogglesService.saveFeatures(manageTogglesCtrl.modifiedFeatures); + const promise = ManageTogglesService.saveFeatures(manageTogglesCtrl.modifiedFeatures).catch(response => { + MessageService.showToast(response.msg); + }); manageTogglesCtrl.modifiedFeatures = []; return promise; }; diff --git a/feature-toggles/utils/messageService.js b/feature-toggles/utils/messageService.js new file mode 100644 index 000000000..6a26686b9 --- /dev/null +++ b/feature-toggles/utils/messageService.js @@ -0,0 +1,20 @@ +(function() { + 'use strict'; + + const app = angular.module('app'); + + app.service('MessageService', function($mdToast) { + const service = this; + + service.showToast = function showToast(message) { + $mdToast.show( + $mdToast.simple() + .textContent(message) + .action('FECHAR') + .highlightAction(true) + .hideDelay(5000) + .position('bottom right') + ); + }; + }); +})(); \ No newline at end of file From 753502d898e22ce853d4d59edb4dcf97d2b4e2e9 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Wed, 16 Jan 2019 10:32:03 -0300 Subject: [PATCH 049/606] add load circle when save changes --- feature-toggles/manage/manage-toggles.html | 6 ++++-- feature-toggles/manage/manageTogglesController.js | 14 +++++++++++--- feature-toggles/manage/manageTogglesService.js | 2 +- 3 files changed, 16 insertions(+), 6 deletions(-) diff --git a/feature-toggles/manage/manage-toggles.html b/feature-toggles/manage/manage-toggles.html index 30f254985..e96dc1fe8 100644 --- a/feature-toggles/manage/manage-toggles.html +++ b/feature-toggles/manage/manage-toggles.html @@ -29,8 +29,10 @@

    Gerenciar features do sistema

    - SALVAR + SALVAR + \ No newline at end of file diff --git a/feature-toggles/manage/manageTogglesController.js b/feature-toggles/manage/manageTogglesController.js index b771dcae3..3fd7b26be 100644 --- a/feature-toggles/manage/manageTogglesController.js +++ b/feature-toggles/manage/manageTogglesController.js @@ -5,6 +5,7 @@ app.controller('ManageTogglesController', function(ManageTogglesService, AuthService, MessageService) { const manageTogglesCtrl = this; + manageTogglesCtrl.isLoading = false; manageTogglesCtrl.features = []; manageTogglesCtrl.modifiedFeatures = []; @@ -21,9 +22,16 @@ }; manageTogglesCtrl.save = function save() { - const promise = ManageTogglesService.saveFeatures(manageTogglesCtrl.modifiedFeatures).catch(response => { - MessageService.showToast(response.msg); - }); + manageTogglesCtrl.isLoading = true; + const promise = ManageTogglesService.saveFeatures(manageTogglesCtrl.modifiedFeatures) + .then(response => { + MessageService.showToast("Alterações salvas com sucesso."); + return response; + }).catch(response => { + MessageService.showToast(response.msg); + }).finally(function() { + manageTogglesCtrl.isLoading = false; + }); manageTogglesCtrl.modifiedFeatures = []; return promise; }; diff --git a/feature-toggles/manage/manageTogglesService.js b/feature-toggles/manage/manageTogglesService.js index 3c5b7bfab..f6d0eae08 100644 --- a/feature-toggles/manage/manageTogglesService.js +++ b/feature-toggles/manage/manageTogglesService.js @@ -12,7 +12,7 @@ }; service.saveFeatures = function saveFeatures(features) { - HttpService.put(URI, features); + return HttpService.put(URI, features); }; }); })(); \ No newline at end of file From 7f31013640c24ef0c649e63321f97b9b8482185c Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Wed, 16 Jan 2019 10:35:14 -0300 Subject: [PATCH 050/606] Remove unnecessary functions --- feature-toggles/user/userFactory.js | 30 ----------------------------- 1 file changed, 30 deletions(-) diff --git a/feature-toggles/user/userFactory.js b/feature-toggles/user/userFactory.js index 2d6827c47..8120b0315 100644 --- a/feature-toggles/user/userFactory.js +++ b/feature-toggles/user/userFactory.js @@ -22,41 +22,11 @@ } }; - User.prototype.isAdmin = function isAdmin(keyInstitution) { - var managed_institution = _.find(this.institutions_admin, function(institution) { - return getKey(institution) == keyInstitution; }); - return managed_institution; - }; - - User.prototype.isAdminOfCurrentInst = function isAdminOfCurrentInst() { - return this.institutions_admin.map(getKey).includes(this.current_institution.key); - }; - - User.prototype.isMember = function isMember(institutionKey){ - return _.includes(_.map(this.institutions, getKeyObj), institutionKey); - }; - - User.prototype.isInactive = function isInactive() { - var notActive = this.state != 'active'; - return notActive; - }; User.prototype.hasPermission = function hasPermission(permissionType) { return permissionType in this.permissions; }; - function getKeyObj(obj) { - if(obj.key){ - return obj.key; - } - } - - function getKey(obj){ - var key = obj.split("/"); - key = key[key.length -1]; - - return key; - } return { user: User From fbe74f08692fefff5e105b79be12fd90bc67ce0f Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Wed, 16 Jan 2019 10:58:00 -0300 Subject: [PATCH 051/606] Create translation off features --- backend/admin.py | 5 ++++- backend/handlers/feature_toggle_handler.py | 7 ++++--- backend/models/feature.py | 11 +++++++---- feature-toggles/manage/manage-toggles.html | 2 +- feature-toggles/manage/manageTogglesService.js | 2 +- 5 files changed, 17 insertions(+), 10 deletions(-) diff --git a/backend/admin.py b/backend/admin.py index ca32d57ee..fd565f48d 100644 --- a/backend/admin.py +++ b/backend/admin.py @@ -42,7 +42,10 @@ { "name": 'manage-inst-edit', "enable_mobile": "DISABLED", - "enable_desktop": "ALL" + "enable_desktop": "ALL", + "translation_dict": { + "pt-br": "Editar informações da instituição" + } } ] diff --git a/backend/handlers/feature_toggle_handler.py b/backend/handlers/feature_toggle_handler.py index 36aa9cdd8..a3ed10283 100644 --- a/backend/handlers/feature_toggle_handler.py +++ b/backend/handlers/feature_toggle_handler.py @@ -11,7 +11,7 @@ __all__ = ['FeatureToggleHandler'] -def to_json(feature_list): +def to_json(feature_list, language=None): """ Method to generate list of feature models in json format object. @@ -19,7 +19,7 @@ def to_json(feature_list): feature_list -- List of features objects """ - features = [feature.make() for feature in feature_list] + features = [feature.make(language) for feature in feature_list] return json.dumps(features) class FeatureToggleHandler(BaseHandler): @@ -33,13 +33,14 @@ def get(self, user): """ feature_name = self.request.get('name') + language = self.request.get('lang') if feature_name: features = [Feature.get_feature(feature_name)] else: features = Feature.get_all_features() - self.response.write(to_json(features)) + self.response.write(to_json(features, language)) @login_required @json_response diff --git a/backend/models/feature.py b/backend/models/feature.py index 127c396a8..ed4aa8aaa 100644 --- a/backend/models/feature.py +++ b/backend/models/feature.py @@ -1,4 +1,5 @@ """Feature model.""" +import json from google.appengine.ext import ndb __all__ = ['Feature'] @@ -13,10 +14,11 @@ class Feature(ndb.Model): choices=set(["SUPER_USER", "ADMIN", "ALL", "DISABLED"])) enable_desktop = ndb.StringProperty( choices=set(["SUPER_USER", "ADMIN", "ALL", "DISABLED"])) + translation = ndb.JsonProperty(default="{}") @staticmethod @ndb.transactional(xg=True) - def create(name, enable_mobile="ALL", enable_desktop="ALL"): + def create(name, translation_dict, enable_mobile="ALL", enable_desktop="ALL"): """ Method to create new feature. @@ -27,6 +29,7 @@ def create(name, enable_mobile="ALL", enable_desktop="ALL"): """ feature = Feature(id=name, name=name, enable_desktop=enable_desktop, enable_mobile=enable_mobile) + feature.translation = json.dumps(translation_dict) feature.put() return feature @@ -80,15 +83,15 @@ def get_feature(feature_name): else: raise Exception("Feature not found!") - def make(self): + def make(self, language="pt-br"): """ Method to make feature. """ - make_obj = { 'name': self.name, 'enable_mobile': self.enable_mobile, - 'enable_desktop': self.enable_desktop + 'enable_desktop': self.enable_desktop, + 'translation': json.loads(self.translation).get(language) } return make_obj diff --git a/feature-toggles/manage/manage-toggles.html b/feature-toggles/manage/manage-toggles.html index e96dc1fe8..a22b7bec4 100644 --- a/feature-toggles/manage/manage-toggles.html +++ b/feature-toggles/manage/manage-toggles.html @@ -8,7 +8,7 @@

    Gerenciar features do sistema

    - {{feature.name}} + {{feature.translation || feature.name}} diff --git a/feature-toggles/manage/manageTogglesService.js b/feature-toggles/manage/manageTogglesService.js index f6d0eae08..1dba95c8a 100644 --- a/feature-toggles/manage/manageTogglesService.js +++ b/feature-toggles/manage/manageTogglesService.js @@ -5,7 +5,7 @@ app.service('ManageTogglesService', function(HttpService) { const service = this; - const URI = '/api/feature-toggles'; + const URI = '/api/feature-toggles?lang=pt-br'; service.getAllFeatureToggles = function getAllFeatureToggles() { return HttpService.get(URI); From fc7cb9d61932f3cd94b562a3857d911524141317 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Wed, 16 Jan 2019 11:30:34 -0300 Subject: [PATCH 052/606] Fix backend tests --- backend/handlers/feature_toggle_handler.py | 2 +- .../feature_toggle_handler_test.py | 12 ++++----- backend/test/model_test/feature_test.py | 25 ++++++++++--------- 3 files changed, 19 insertions(+), 20 deletions(-) diff --git a/backend/handlers/feature_toggle_handler.py b/backend/handlers/feature_toggle_handler.py index a3ed10283..95b13c950 100644 --- a/backend/handlers/feature_toggle_handler.py +++ b/backend/handlers/feature_toggle_handler.py @@ -11,7 +11,7 @@ __all__ = ['FeatureToggleHandler'] -def to_json(feature_list, language=None): +def to_json(feature_list, language="pt-br"): """ Method to generate list of feature models in json format object. diff --git a/backend/test/feature_toggle_handler_tests/feature_toggle_handler_test.py b/backend/test/feature_toggle_handler_tests/feature_toggle_handler_test.py index 0ead984fb..144014cbd 100644 --- a/backend/test/feature_toggle_handler_tests/feature_toggle_handler_test.py +++ b/backend/test/feature_toggle_handler_tests/feature_toggle_handler_test.py @@ -25,15 +25,15 @@ def setUp(cls): ], debug=True) cls.testapp = cls.webtest.TestApp(app) - cls.feature = Feature.create('feature-test') - cls.other_feature = Feature.create('feature-test-other') + cls.feature = Feature.create('feature-test', {'pt-br': 'Feature Teste'}) + cls.other_feature = Feature.create('feature-test-other', {'pt-br': 'Feature Teste'}) @patch('util.login_service.verify_token', return_value=USER) def test_get_all(self, verify_token): """Test get all features.""" - features = self.testapp.get('/api/feature-toggles').json + features = self.testapp.get('/api/feature-toggles?lang=pt-br').json features_make = [self.feature.make(), self.other_feature.make()] self.assertEquals(len(features), 2) @@ -42,14 +42,12 @@ def test_get_all(self, verify_token): @patch('util.login_service.verify_token', return_value=USER) def test_get_by_query(self, verify_token): """Test get feature with query parameter.""" - - feature = self.testapp.get('/api/feature-toggles?name=feature-test').json + feature = self.testapp.get('/api/feature-toggles?name=feature-test&lang=pt-br').json feature_make = [self.feature.make()] self.assertListEqual(feature, feature_make) - - feature = self.testapp.get('/api/feature-toggles?name=feature-test-other').json + feature = self.testapp.get('/api/feature-toggles?name=feature-test-other&lang=pt-br').json feature_make = [self.other_feature.make()] self.assertListEqual(feature, feature_make) diff --git a/backend/test/model_test/feature_test.py b/backend/test/model_test/feature_test.py index 984696066..7573e2425 100644 --- a/backend/test/model_test/feature_test.py +++ b/backend/test/model_test/feature_test.py @@ -19,23 +19,23 @@ def setUp(cls): def test_create(self): """Teste create a new feature.""" - feature = Feature.create('feature-test') + feature = Feature.create('feature-test', {'pt-br': 'Feature Teste'}) self.assertEqual(feature.name, 'feature-test') self.assertEqual(feature.enable_desktop, 'ALL') self.assertEqual(feature.enable_mobile, 'ALL') - feature = Feature.create('feature-test2', 'DISABLED') + feature = Feature.create('feature-test2', {'pt-br': 'Feature Teste'}, 'DISABLED') self.assertEqual(feature.name, 'feature-test2') self.assertEqual(feature.enable_desktop, 'ALL') self.assertEqual(feature.enable_mobile, 'DISABLED') - feature = Feature.create('feature-test3', 'DISABLED', 'DISABLED') + feature = Feature.create('feature-test3', {'pt-br': 'Feature Teste'}, 'DISABLED', 'DISABLED') self.assertEqual(feature.name, 'feature-test3') self.assertEqual(feature.enable_desktop, 'DISABLED') self.assertEqual(feature.enable_mobile, 'DISABLED') with self.assertRaises(Exception) as raises_context: - Feature.create('feature-test', 'asjdkhd') + Feature.create('feature-test', {'pt-br': 'Feature Teste'}, 'asjdkhd') exception_message = raises_context.exception self.assertEquals( @@ -45,7 +45,7 @@ def test_create(self): def test_set_visibility(self): """Test set visibility.""" - Feature.create('feature-test') + Feature.create('feature-test', {'pt-br': 'Feature Teste'}) feature_dict = { 'name': 'feature-test', @@ -62,14 +62,14 @@ def test_set_visibility(self): def test_get_all_features(self): """Test get all features.""" - feature = Feature.create('feature-test') - feature2 = Feature.create('feature-test2') + feature = Feature.create('feature-test', {'pt-br': 'Feature Teste'}) + feature2 = Feature.create('feature-test2', {'pt-br': 'Feature Teste'}) features = Feature.get_all_features() self.assertIn(feature, features) self.assertIn(feature2, features) - feature3 = Feature.create('feature-test3') + feature3 = Feature.create('feature-test3', {'pt-br': 'Feature Teste'}) self.assertNotIn(feature3, features) features = Feature.get_all_features() @@ -77,8 +77,8 @@ def test_get_all_features(self): def test_get_feature(self): """Test get feature.""" - feature = Feature.create('feature-test') - feature2 = Feature.create('feature-test2') + feature = Feature.create('feature-test', {'pt-br': 'Feature Teste'}) + feature2 = Feature.create('feature-test2', {'pt-br': 'Feature Teste'}) self.assertEqual(feature, Feature.get_feature('feature-test')) self.assertEqual(feature2, Feature.get_feature('feature-test2')) @@ -94,11 +94,12 @@ def test_get_feature(self): def test_make(self): """Test make feature.""" - feature = Feature.create('feature-test') + feature = Feature.create('feature-test', {'pt-br': 'Feature Teste'}) make = { 'name': 'feature-test', 'enable_mobile': 'ALL', - 'enable_desktop': 'ALL' + 'enable_desktop': 'ALL', + 'translation': 'Feature Teste' } self.assertEqual(make, feature.make()) From a0898ca77fafda8a8658369a0fa3c321198ee84d Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Wed, 16 Jan 2019 11:41:35 -0300 Subject: [PATCH 053/606] fix ecis script --- ecis | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ecis b/ecis index da6192337..de613930d 100755 --- a/ecis +++ b/ecis @@ -122,7 +122,7 @@ function set_frontend_url { function set_feature_url { url=$(set_service_url $1 $2); service="FEATURE_URL"; - line=7; + line=8; service_name="feature"; set_config_file $line $service $url $service_name; } From befde0fba8e792902d2e0836cbeea05f8eefddcb Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Mon, 21 Jan 2019 14:50:07 -0300 Subject: [PATCH 054/606] Fix issues --- feature-toggles/app.js | 5 +- feature-toggles/manage/manage-toggles.html | 14 +--- .../manage/manageTogglesController.js | 77 +++++++++++++------ .../manage/manageTogglesService.js | 8 +- feature-toggles/utils/messageService.js | 8 +- 5 files changed, 74 insertions(+), 38 deletions(-) diff --git a/feature-toggles/app.js b/feature-toggles/app.js index 62f8f23a0..65c7ca210 100644 --- a/feature-toggles/app.js +++ b/feature-toggles/app.js @@ -94,8 +94,11 @@ }; }); + /** + * Auth interceptor to check if the usr is logged in, if not, redirect to login page. + */ app.run(function authInterceptor(AuthService, $transitions) { - var ignored_routes = [ + const ignored_routes = [ "singin" ]; diff --git a/feature-toggles/manage/manage-toggles.html b/feature-toggles/manage/manage-toggles.html index a22b7bec4..08d1b3646 100644 --- a/feature-toggles/manage/manage-toggles.html +++ b/feature-toggles/manage/manage-toggles.html @@ -12,20 +12,14 @@

    Gerenciar features do sistema

    - - DESABILITADO - TODOS - TESTE - SUPER USÁRIO + + {{option.name}} - - DESABILITADO - TODOS - TESTE - SUPER USÁRIO + + {{option.name}}
    diff --git a/feature-toggles/manage/manageTogglesController.js b/feature-toggles/manage/manageTogglesController.js index 3fd7b26be..84349b8fb 100644 --- a/feature-toggles/manage/manageTogglesController.js +++ b/feature-toggles/manage/manageTogglesController.js @@ -3,44 +3,75 @@ const app = angular.module('app'); - app.controller('ManageTogglesController', function(ManageTogglesService, AuthService, MessageService) { + /** + * This controller manages the view that displays all features for the user. + */ + app.controller('ManageTogglesController', ['ManageTogglesService' , 'AuthService', 'MessageService' , function(ManageTogglesService, AuthService, + MessageService) { + const manageTogglesCtrl = this; manageTogglesCtrl.isLoading = false; + manageTogglesCtrl.oddFeatures = []; manageTogglesCtrl.features = []; - manageTogglesCtrl.modifiedFeatures = []; + + manageTogglesCtrl.options = [ + { + value: 'DISABLED', + name: 'DESABILITADO' + }, { + value: 'ALL', + name: 'TODOS' + }, { + value: 'ADMIN', + name: 'TESTE' + }, { + value: 'SUPER_USER', + name: 'SUPER USÁRIO' + } + ]; manageTogglesCtrl.logout = function() { AuthService.logout(); }; - manageTogglesCtrl.addModifiedFeature = function(feature) { - const featureFound = _.find(manageTogglesCtrl.modifiedFeatures, element => element.name === feature.name); + function getChangedFeatures() { + return manageTogglesCtrl.features.filter((feature, index) => { + const oddFeature = manageTogglesCtrl.oddFeatures[index]; + const isEqualMobile = oddFeature.enable_mobile === feature.enable_mobile; + const isEqualDesktop = oddFeature.enable_desktop === feature.enable_desktop; + return !isEqualMobile || !isEqualDesktop; + }); + } - if (!featureFound) { - manageTogglesCtrl.modifiedFeatures.push(feature); + manageTogglesCtrl.save = function save() { + const modifiedFeatures = getChangedFeatures(); + if (modifiedFeatures.length !== 0) { + manageTogglesCtrl.isLoading = true; + return ManageTogglesService.saveFeatures(modifiedFeatures) + .then(() => { + return loadFeatures().then(features => { + MessageService.showToast("Alterações salvas com sucesso."); + return features; + }); + }).catch(response => { + MessageService.showToast(response.msg); + }).finally(function() { + manageTogglesCtrl.isLoading = false; + }); } - }; - manageTogglesCtrl.save = function save() { - manageTogglesCtrl.isLoading = true; - const promise = ManageTogglesService.saveFeatures(manageTogglesCtrl.modifiedFeatures) - .then(response => { - MessageService.showToast("Alterações salvas com sucesso."); - return response; - }).catch(response => { - MessageService.showToast(response.msg); - }).finally(function() { - manageTogglesCtrl.isLoading = false; - }); - manageTogglesCtrl.modifiedFeatures = []; - return promise; }; - manageTogglesCtrl.$onInit = function() { + function loadFeatures() { return ManageTogglesService.getAllFeatureToggles().then(function(features) { - manageTogglesCtrl.features = features; + manageTogglesCtrl.oddFeatures = features; + manageTogglesCtrl.features = _.cloneDeep(features); return features; }); + } + + manageTogglesCtrl.$onInit = function() { + return loadFeatures(); }; - }); + }]); })(); \ No newline at end of file diff --git a/feature-toggles/manage/manageTogglesService.js b/feature-toggles/manage/manageTogglesService.js index 1dba95c8a..09d4c8933 100644 --- a/feature-toggles/manage/manageTogglesService.js +++ b/feature-toggles/manage/manageTogglesService.js @@ -3,7 +3,11 @@ const app = angular.module('app'); - app.service('ManageTogglesService', function(HttpService) { + /** + * This service is responsible for loading and changing the + * features through requests to the backend. + */ + app.service('ManageTogglesService', ['HttpService', function(HttpService) { const service = this; const URI = '/api/feature-toggles?lang=pt-br'; @@ -14,5 +18,5 @@ service.saveFeatures = function saveFeatures(features) { return HttpService.put(URI, features); }; - }); + }]); })(); \ No newline at end of file diff --git a/feature-toggles/utils/messageService.js b/feature-toggles/utils/messageService.js index 6a26686b9..219dbd4a7 100644 --- a/feature-toggles/utils/messageService.js +++ b/feature-toggles/utils/messageService.js @@ -3,9 +3,13 @@ const app = angular.module('app'); - app.service('MessageService', function($mdToast) { + app.service('MessageService', ['$mdToast', function($mdToast) { const service = this; + /** + * This function displays a small dialog containing the received message per parameter. + * @param {String} message - Message to show + */ service.showToast = function showToast(message) { $mdToast.show( $mdToast.simple() @@ -16,5 +20,5 @@ .position('bottom right') ); }; - }); + }]); })(); \ No newline at end of file From a5269970e4336c80723cd532476c08da1fc444c3 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Tue, 22 Jan 2019 10:02:10 -0300 Subject: [PATCH 055/606] Add button save in toggles --- feature-toggles/manage/manage-toggles.css | 2 +- feature-toggles/manage/manage-toggles.html | 10 +++---- .../manage/manageTogglesController.js | 29 +++++-------------- feature-toggles/manage/permission-select.html | 9 ++++++ 4 files changed, 22 insertions(+), 28 deletions(-) create mode 100644 feature-toggles/manage/permission-select.html diff --git a/feature-toggles/manage/manage-toggles.css b/feature-toggles/manage/manage-toggles.css index 2060ca896..547e1db8b 100644 --- a/feature-toggles/manage/manage-toggles.css +++ b/feature-toggles/manage/manage-toggles.css @@ -18,7 +18,7 @@ .toggle-content { display: grid; - grid-template-columns: auto 154px 154px; + grid-template-columns: auto 154px 154px 105px; grid-gap: 10px; align-items: center; } diff --git a/feature-toggles/manage/manage-toggles.html b/feature-toggles/manage/manage-toggles.html index 08d1b3646..c63fdffe3 100644 --- a/feature-toggles/manage/manage-toggles.html +++ b/feature-toggles/manage/manage-toggles.html @@ -4,7 +4,7 @@
    -

    Gerenciar features do sistema

    +

    Gerenciar funcionalidades do sistema

    @@ -22,11 +22,11 @@

    Gerenciar features do sistema

    {{option.name}} + SALVAR +
    - SALVAR -
    \ No newline at end of file diff --git a/feature-toggles/manage/manageTogglesController.js b/feature-toggles/manage/manageTogglesController.js index 84349b8fb..1aaeb63ac 100644 --- a/feature-toggles/manage/manageTogglesController.js +++ b/feature-toggles/manage/manageTogglesController.js @@ -34,32 +34,17 @@ AuthService.logout(); }; - function getChangedFeatures() { - return manageTogglesCtrl.features.filter((feature, index) => { - const oddFeature = manageTogglesCtrl.oddFeatures[index]; - const isEqualMobile = oddFeature.enable_mobile === feature.enable_mobile; - const isEqualDesktop = oddFeature.enable_desktop === feature.enable_desktop; - return !isEqualMobile || !isEqualDesktop; - }); - } - - manageTogglesCtrl.save = function save() { - const modifiedFeatures = getChangedFeatures(); - if (modifiedFeatures.length !== 0) { - manageTogglesCtrl.isLoading = true; - return ManageTogglesService.saveFeatures(modifiedFeatures) - .then(() => { - return loadFeatures().then(features => { - MessageService.showToast("Alterações salvas com sucesso."); - return features; - }); + manageTogglesCtrl.save = function save(feature) { + feature.isLoading = true; + return ManageTogglesService.saveFeatures([feature]) + .then(response => { + MessageService.showToast("Alterações salvas com sucesso."); + return response; }).catch(response => { MessageService.showToast(response.msg); }).finally(function() { - manageTogglesCtrl.isLoading = false; + feature.isLoading = false; }); - } - }; function loadFeatures() { diff --git a/feature-toggles/manage/permission-select.html b/feature-toggles/manage/permission-select.html new file mode 100644 index 000000000..7108bdd88 --- /dev/null +++ b/feature-toggles/manage/permission-select.html @@ -0,0 +1,9 @@ + + + + DESABILITADO + TODOS + TESTE + SUPER USÁRIO + + From b491e585eacab839830a215199e71c674364637b Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Tue, 22 Jan 2019 15:02:04 -0300 Subject: [PATCH 056/606] Create permissionSelectComponent --- feature-toggles/index.html | 1 + feature-toggles/manage/manage-toggles.html | 20 +++++--------- .../manage/manageTogglesController.js | 16 ------------ feature-toggles/manage/permission-select.html | 4 +-- .../manage/permissionSelect.component.js | 26 +++++++++++++++++++ 5 files changed, 35 insertions(+), 32 deletions(-) create mode 100644 feature-toggles/manage/permissionSelect.component.js diff --git a/feature-toggles/index.html b/feature-toggles/index.html index 1d9747d21..e8e27d673 100644 --- a/feature-toggles/index.html +++ b/feature-toggles/index.html @@ -68,5 +68,6 @@ + \ No newline at end of file diff --git a/feature-toggles/manage/manage-toggles.html b/feature-toggles/manage/manage-toggles.html index c63fdffe3..6d54efc6d 100644 --- a/feature-toggles/manage/manage-toggles.html +++ b/feature-toggles/manage/manage-toggles.html @@ -10,21 +10,13 @@

    Gerenciar funcionalidades do sistema

    {{feature.translation || feature.name}} - - - - {{option.name}} - - - - - - {{option.name}} - - + + SALVAR + md-colors="{background: 'teal-500'}" + ng-click="manageTogglesCtrl.save(feature)"> + SALVAR + diff --git a/feature-toggles/manage/manageTogglesController.js b/feature-toggles/manage/manageTogglesController.js index 1aaeb63ac..eb4f82096 100644 --- a/feature-toggles/manage/manageTogglesController.js +++ b/feature-toggles/manage/manageTogglesController.js @@ -14,22 +14,6 @@ manageTogglesCtrl.oddFeatures = []; manageTogglesCtrl.features = []; - manageTogglesCtrl.options = [ - { - value: 'DISABLED', - name: 'DESABILITADO' - }, { - value: 'ALL', - name: 'TODOS' - }, { - value: 'ADMIN', - name: 'TESTE' - }, { - value: 'SUPER_USER', - name: 'SUPER USÁRIO' - } - ]; - manageTogglesCtrl.logout = function() { AuthService.logout(); }; diff --git a/feature-toggles/manage/permission-select.html b/feature-toggles/manage/permission-select.html index 7108bdd88..d8e12a966 100644 --- a/feature-toggles/manage/permission-select.html +++ b/feature-toggles/manage/permission-select.html @@ -1,6 +1,6 @@ - + - + DESABILITADO TODOS TESTE diff --git a/feature-toggles/manage/permissionSelect.component.js b/feature-toggles/manage/permissionSelect.component.js new file mode 100644 index 000000000..de09739de --- /dev/null +++ b/feature-toggles/manage/permissionSelect.component.js @@ -0,0 +1,26 @@ +(function() { + 'use strict'; + + const app = angular.module('app'); + + app.controller('PermissionSelectController', permissionSelectController) + .component('permissionSelect', { + templateUrl: 'app/manage/permission-select.html', + controller: "PermissionSelectController", + controllerAs: 'permissionSelectCtrl', + bindings: { + label: '@', + feature: '=', + attr: '@' + } + }); + + function permissionSelectController() { + const permissionSelectCtrl = this; + + Object.defineProperty(permissionSelectCtrl, 'enable', { + get: () => permissionSelectCtrl.feature[permissionSelectCtrl.attr], + set: value => permissionSelectCtrl.feature[permissionSelectCtrl.attr] = value + }); + } +})(); \ No newline at end of file From 0080edec6c40d1b10a3498d98bd2e88806aa66d9 Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Tue, 22 Jan 2019 15:14:14 -0300 Subject: [PATCH 057/606] create states constants --- feature-toggles/app.js | 14 +++++++------- feature-toggles/auth/authService.js | 4 ++-- feature-toggles/auth/loginController.js | 6 +++--- feature-toggles/index.html | 1 + feature-toggles/utils/statesConstants.js | 10 ++++++++++ 5 files changed, 23 insertions(+), 12 deletions(-) create mode 100644 feature-toggles/utils/statesConstants.js diff --git a/feature-toggles/app.js b/feature-toggles/app.js index 65c7ca210..6c67001be 100644 --- a/feature-toggles/app.js +++ b/feature-toggles/app.js @@ -10,7 +10,7 @@ app.config(function($mdIconProvider, $mdThemingProvider, $urlMatcherFactoryProvider, $urlRouterProvider, - $locationProvider, $stateProvider, $httpProvider) { + $locationProvider, $stateProvider, $httpProvider, STATES) { $mdIconProvider.fontSet('md', 'material-icons'); $mdThemingProvider.theme('docs-dark'); @@ -22,15 +22,15 @@ $urlMatcherFactoryProvider.caseInsensitive(true); $stateProvider - .state("singin", { - url: "/singin", + .state(STATES.SIGNIN, { + url: "/signin", views: { main: { templateUrl: "app/auth/login.html", controller: "LoginController as loginCtrl" } } - }).state("manage-features", { + }).state(STATES.MANAGE_FEATURES, { url: "/", views: { main: { @@ -97,9 +97,9 @@ /** * Auth interceptor to check if the usr is logged in, if not, redirect to login page. */ - app.run(function authInterceptor(AuthService, $transitions) { + app.run(function authInterceptor(AuthService, STATES, $transitions) { const ignored_routes = [ - "singin" + STATES.SIGNIN ]; $transitions.onBefore({ @@ -107,7 +107,7 @@ return !(_.includes(ignored_routes, state.name)) && !AuthService.isLoggedIn(); } }, function(transition) { - return transition.router.stateService.target("singin"); + return transition.router.stateService.target(STATES.SIGNIN); }); }); })(); \ No newline at end of file diff --git a/feature-toggles/auth/authService.js b/feature-toggles/auth/authService.js index 795b81bce..a8f55028e 100644 --- a/feature-toggles/auth/authService.js +++ b/feature-toggles/auth/authService.js @@ -3,7 +3,7 @@ const app = angular.module("app"); - app.service("AuthService", function AuthService($state, $window, UserService, UserFactory, MessageService) { + app.service("AuthService", function AuthService($state, $window, UserService, UserFactory, MessageService, STATES) { const service = this; const authObj = firebase.auth(); @@ -132,7 +132,7 @@ executeLogoutListeners(); - $state.go("singin"); + $state.go(STATES.SIGNIN); }; service.getCurrentUser = function getCurrentUser() { diff --git a/feature-toggles/auth/loginController.js b/feature-toggles/auth/loginController.js index 661365fe9..01b96b366 100644 --- a/feature-toggles/auth/loginController.js +++ b/feature-toggles/auth/loginController.js @@ -3,7 +3,7 @@ const app = angular.module('app'); app.controller('LoginController', function(AuthService, $state, - $stateParams, $window, MessageService) { + $stateParams, $window, MessageService, STATES) { const loginCtrl = this; loginCtrl.user = {}; @@ -57,13 +57,13 @@ if (path) { window.location.pathname = path; } else { - $state.go("manage-features"); + $state.go(STATES.MANAGE_FEATURES); } } loginCtrl.$onInit = function main() { if (AuthService.isLoggedIn()) { - $state.go("manage-features"); + $state.go(STATES.MANAGE_FEATURES); } }; }); diff --git a/feature-toggles/index.html b/feature-toggles/index.html index e8e27d673..a991fc740 100644 --- a/feature-toggles/index.html +++ b/feature-toggles/index.html @@ -57,6 +57,7 @@ + diff --git a/feature-toggles/utils/statesConstants.js b/feature-toggles/utils/statesConstants.js new file mode 100644 index 000000000..1360ccd23 --- /dev/null +++ b/feature-toggles/utils/statesConstants.js @@ -0,0 +1,10 @@ +(function() { + 'use strict'; + + const app = angular.module('app'); + + app.constant('STATES', { + MANAGE_FEATURES: "manage-features", + SIGNIN: 'signin' + }); +})(); \ No newline at end of file From d225e85d9a651c640d23cfbe8c5eec569a96816a Mon Sep 17 00:00:00 2001 From: Luiz-FS Date: Tue, 22 Jan 2019 15:29:27 -0300 Subject: [PATCH 058/606] Move style to css file --- feature-toggles/manage/manage-toggles.css | 17 +++++++++++++++++ feature-toggles/manage/manage-toggles.html | 6 +++--- 2 files changed, 20 insertions(+), 3 deletions(-) diff --git a/feature-toggles/manage/manage-toggles.css b/feature-toggles/manage/manage-toggles.css index 547e1db8b..fd35ccb9f 100644 --- a/feature-toggles/manage/manage-toggles.css +++ b/feature-toggles/manage/manage-toggles.css @@ -1,3 +1,9 @@ +.navbar-content { + height: 100%; + display: grid; + grid-template-rows: auto 1fr; +} + .content { display: grid; padding: 16px; @@ -16,6 +22,17 @@ color: white; } +#toggles-box { + max-height: 100%; + overflow-y: auto; +} + +#toggles-box > div { + display: grid; + justify-content: center; + margin: 16px; +} + .toggle-content { display: grid; grid-template-columns: auto 154px 154px 105px; diff --git a/feature-toggles/manage/manage-toggles.html b/feature-toggles/manage/manage-toggles.html index 6d54efc6d..592f260c6 100644 --- a/feature-toggles/manage/manage-toggles.html +++ b/feature-toggles/manage/manage-toggles.html @@ -1,11 +1,11 @@ -
    +