praxis 0.16.1 → 0.17.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/.rspec +0 -1
- data/.ruby-version +1 -1
- data/.travis.yml +2 -1
- data/CHANGELOG.md +41 -0
- data/CONTRIBUTING.md +3 -0
- data/lib/api_browser/Gruntfile.js +20 -4
- data/lib/api_browser/app/bower_components/angular-mocks/.bower.json +6 -6
- data/lib/api_browser/app/bower_components/angular-mocks/README.md +11 -5
- data/lib/api_browser/app/bower_components/angular-mocks/angular-mocks.js +475 -216
- data/lib/api_browser/app/bower_components/angular-mocks/bower.json +2 -2
- data/lib/api_browser/app/bower_components/angular-mocks/ngAnimateMock.js +2 -0
- data/lib/api_browser/app/bower_components/angular-mocks/ngMock.js +2 -0
- data/lib/api_browser/app/bower_components/angular-mocks/ngMockE2E.js +2 -0
- data/lib/api_browser/app/bower_components/angular-mocks/package.json +1 -1
- data/lib/api_browser/app/bower_components/angular-sanitize/.bower.json +8 -8
- data/lib/api_browser/app/bower_components/angular-sanitize/README.md +19 -5
- data/lib/api_browser/app/bower_components/angular-sanitize/angular-sanitize.js +186 -127
- data/lib/api_browser/app/bower_components/angular-sanitize/angular-sanitize.min.js +12 -10
- data/lib/api_browser/app/bower_components/angular-sanitize/angular-sanitize.min.js.map +3 -3
- data/lib/api_browser/app/bower_components/angular-sanitize/bower.json +3 -2
- data/lib/api_browser/app/bower_components/angular-sanitize/index.js +2 -0
- data/lib/api_browser/app/bower_components/angular-sanitize/package.json +26 -0
- data/lib/api_browser/app/bower_components/angular-ui-bootstrap-bower/.bower.json +15 -8
- data/lib/api_browser/app/bower_components/angular-ui-bootstrap-bower/bower.json +11 -3
- data/lib/api_browser/app/bower_components/angular-ui-bootstrap-bower/ui-bootstrap-csp.css +6 -0
- data/lib/api_browser/app/bower_components/angular-ui-bootstrap-bower/ui-bootstrap-tpls.js +1177 -453
- data/lib/api_browser/app/bower_components/angular-ui-bootstrap-bower/ui-bootstrap-tpls.min.js +4 -4
- data/lib/api_browser/app/bower_components/angular-ui-bootstrap-bower/ui-bootstrap.js +1066 -404
- data/lib/api_browser/app/bower_components/angular-ui-bootstrap-bower/ui-bootstrap.min.js +3 -3
- data/lib/api_browser/app/bower_components/angular-ui-router/.bower.json +5 -6
- data/lib/api_browser/app/bower_components/angular-ui-router/CHANGELOG.md +208 -3
- data/lib/api_browser/app/bower_components/angular-ui-router/CONTRIBUTING.md +65 -0
- data/lib/api_browser/app/bower_components/angular-ui-router/LICENSE +1 -1
- data/lib/api_browser/app/bower_components/angular-ui-router/README.md +36 -71
- data/lib/api_browser/app/bower_components/angular-ui-router/api/angular-ui-router.d.ts +126 -0
- data/lib/api_browser/app/bower_components/angular-ui-router/bower.json +1 -1
- data/lib/api_browser/app/bower_components/angular-ui-router/release/angular-ui-router.js +1902 -755
- data/lib/api_browser/app/bower_components/angular-ui-router/release/angular-ui-router.min.js +2 -2
- data/lib/api_browser/app/bower_components/angular-ui-router/src/common.js +69 -23
- data/lib/api_browser/app/bower_components/angular-ui-router/src/resolve.js +15 -5
- data/lib/api_browser/app/bower_components/angular-ui-router/src/state.js +556 -295
- data/lib/api_browser/app/bower_components/angular-ui-router/src/stateDirectives.js +101 -42
- data/lib/api_browser/app/bower_components/angular-ui-router/src/stateFilters.js +6 -2
- data/lib/api_browser/app/bower_components/angular-ui-router/src/templateFactory.js +2 -2
- data/lib/api_browser/app/bower_components/angular-ui-router/src/urlMatcherFactory.js +822 -97
- data/lib/api_browser/app/bower_components/angular-ui-router/src/urlRouter.js +274 -120
- data/lib/api_browser/app/bower_components/angular-ui-router/src/viewDirective.js +33 -20
- data/lib/api_browser/app/bower_components/angular-ui-router/src/viewScroll.js +1 -1
- data/lib/api_browser/app/bower_components/angular/.bower.json +5 -5
- data/lib/api_browser/app/bower_components/angular/README.md +2 -5
- data/lib/api_browser/app/bower_components/angular/angular-csp.css +5 -8
- data/lib/api_browser/app/bower_components/angular/angular.js +12975 -6996
- data/lib/api_browser/app/bower_components/angular/angular.min.js +285 -213
- data/lib/api_browser/app/bower_components/angular/angular.min.js.gzip +0 -0
- data/lib/api_browser/app/bower_components/angular/angular.min.js.map +3 -3
- data/lib/api_browser/app/bower_components/angular/bower.json +1 -1
- data/lib/api_browser/app/bower_components/angular/index.js +2 -0
- data/lib/api_browser/app/bower_components/angular/package.json +2 -2
- data/lib/api_browser/app/bower_components/bootstrap-sass/.bower.json +31 -16
- data/lib/api_browser/app/bower_components/bootstrap-sass/CHANGELOG.md +108 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/CONTRIBUTING.md +55 -37
- data/lib/api_browser/app/bower_components/bootstrap-sass/README.md +147 -206
- data/lib/api_browser/app/bower_components/bootstrap-sass/bower.json +19 -8
- data/lib/api_browser/app/bower_components/bootstrap-sass/{dist/fonts → vendor/assets/fonts/bootstrap}/glyphicons-halflings-regular.eot +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{dist/fonts → vendor/assets/fonts/bootstrap}/glyphicons-halflings-regular.svg +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{dist/fonts → vendor/assets/fonts/bootstrap}/glyphicons-halflings-regular.ttf +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{dist/fonts → vendor/assets/fonts/bootstrap}/glyphicons-halflings-regular.woff +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/javascripts/bootstrap.js +12 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{js → vendor/assets/javascripts/bootstrap}/affix.js +1 -1
- data/lib/api_browser/app/bower_components/bootstrap-sass/{js → vendor/assets/javascripts/bootstrap}/alert.js +1 -1
- data/lib/api_browser/app/bower_components/bootstrap-sass/{js → vendor/assets/javascripts/bootstrap}/button.js +11 -5
- data/lib/api_browser/app/bower_components/bootstrap-sass/{js → vendor/assets/javascripts/bootstrap}/carousel.js +5 -5
- data/lib/api_browser/app/bower_components/bootstrap-sass/{js → vendor/assets/javascripts/bootstrap}/collapse.js +1 -1
- data/lib/api_browser/app/bower_components/bootstrap-sass/{js → vendor/assets/javascripts/bootstrap}/dropdown.js +5 -5
- data/lib/api_browser/app/bower_components/bootstrap-sass/{js → vendor/assets/javascripts/bootstrap}/modal.js +1 -1
- data/lib/api_browser/app/bower_components/bootstrap-sass/{js → vendor/assets/javascripts/bootstrap}/popover.js +1 -1
- data/lib/api_browser/app/bower_components/bootstrap-sass/{js → vendor/assets/javascripts/bootstrap}/scrollspy.js +2 -2
- data/lib/api_browser/app/bower_components/bootstrap-sass/{js → vendor/assets/javascripts/bootstrap}/tab.js +1 -1
- data/lib/api_browser/app/bower_components/bootstrap-sass/{js → vendor/assets/javascripts/bootstrap}/tooltip.js +1 -1
- data/lib/api_browser/app/bower_components/bootstrap-sass/{js → vendor/assets/javascripts/bootstrap}/transition.js +1 -1
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap.scss +1 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_alerts.scss +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_badges.scss +6 -6
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_breadcrumbs.scss +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_button-groups.scss +7 -33
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_buttons.scss +2 -5
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_carousel.scss +1 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_close.scss +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_code.scss +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_component-animations.scss +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_dropdowns.scss +3 -8
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_forms.scss +11 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_glyphicons.scss +5 -5
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_grid.scss +12 -26
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_input-groups.scss +1 -1
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_jumbotron.scss +8 -2
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_labels.scss +6 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_list-group.scss +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_media.scss +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_mixins.scss +38 -51
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_modals.scss +2 -5
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_navbar.scss +41 -53
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_navs.scss +0 -20
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_normalize.scss +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_pager.scss +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_pagination.scss +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_panels.scss +11 -1
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_popovers.scss +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_print.scss +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_progress-bars.scss +0 -12
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_responsive-utilities.scss +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_scaffolding.scss +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_tables.scss +5 -18
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_theme.scss +2 -2
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_thumbnails.scss +9 -3
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_tooltip.scss +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_type.scss +54 -52
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_utilities.scss +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_variables.scss +20 -11
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/_wells.scss +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/{lib → vendor/assets/stylesheets/bootstrap}/bootstrap.scss +0 -0
- data/lib/api_browser/app/bower_components/lodash/.bower.json +9 -13
- data/lib/api_browser/app/bower_components/lodash/LICENSE.txt +3 -3
- data/lib/api_browser/app/bower_components/lodash/bower.json +4 -7
- data/lib/api_browser/app/bower_components/lodash/lodash.js +12235 -0
- data/lib/api_browser/app/bower_components/lodash/lodash.min.js +98 -0
- data/lib/api_browser/app/index.html +0 -1
- data/lib/api_browser/app/js/app.js +2 -5
- data/lib/api_browser/app/js/controllers/action.js +21 -37
- data/lib/api_browser/app/js/controllers/controller.js +23 -1
- data/lib/api_browser/app/js/controllers/menu.js +46 -14
- data/lib/api_browser/app/js/controllers/type.js +2 -9
- data/lib/api_browser/app/js/directives/attribute_description.js +15 -5
- data/lib/api_browser/app/js/directives/attribute_table.js +6 -6
- data/lib/api_browser/app/js/directives/fixed_if_fits.js +20 -0
- data/lib/api_browser/app/js/directives/no_container.js +6 -6
- data/lib/api_browser/app/js/directives/type_placeholder.js +21 -0
- data/lib/api_browser/app/js/factories/Configuration.js +13 -0
- data/lib/api_browser/app/js/factories/Documentation.js +0 -3
- data/lib/api_browser/app/js/factories/normalize_attributes.js +19 -0
- data/lib/api_browser/app/js/factories/template_for.js +113 -0
- data/lib/api_browser/app/sass/modules/_body.scss +26 -4
- data/lib/api_browser/app/sass/modules/_sidebar.scss +68 -1
- data/lib/api_browser/app/sass/praxis.scss +1 -5
- data/lib/api_browser/app/sass/variables/_bootstrap-variables.scss +13 -4
- data/lib/api_browser/app/views/action.html +13 -17
- data/lib/api_browser/app/views/controller.html +32 -4
- data/lib/api_browser/app/views/directives/attribute_description.html +1 -1
- data/lib/api_browser/app/views/directives/attribute_description/{_default.html → default.html} +0 -0
- data/lib/api_browser/app/views/directives/attribute_description/example.html +13 -0
- data/lib/api_browser/app/views/directives/attribute_description/headers.html +8 -0
- data/lib/api_browser/app/views/directives/attribute_description/member_options.html +4 -0
- data/lib/api_browser/app/views/directives/attribute_description/values.html +14 -0
- data/lib/api_browser/app/views/directives/attribute_table.html +2 -2
- data/lib/api_browser/app/views/home.html +1 -3
- data/lib/api_browser/app/views/layout.html +3 -36
- data/lib/api_browser/app/views/menu.html +45 -0
- data/lib/api_browser/app/views/navbar.html +1 -1
- data/lib/api_browser/app/views/type.html +2 -2
- data/lib/api_browser/app/views/type/{_details.html → details.html} +6 -6
- data/lib/api_browser/app/views/types/embedded/default.html +10 -0
- data/lib/api_browser/app/views/types/embedded/links.html +11 -0
- data/lib/api_browser/app/views/types/embedded/struct.html +2 -0
- data/lib/api_browser/app/views/types/label/link.html +1 -0
- data/lib/api_browser/app/views/types/label/primitive.html +1 -0
- data/lib/api_browser/app/views/types/label/primitive_collection.html +1 -0
- data/lib/api_browser/app/views/types/label/type.html +1 -0
- data/lib/api_browser/app/views/types/label/type_collection.html +1 -0
- data/lib/api_browser/app/views/{directives/request_body/_default.html → types/standalone/default.html} +1 -1
- data/lib/api_browser/app/views/types/standalone/struct.html +1 -0
- data/lib/api_browser/bower.json +9 -9
- data/lib/api_browser/package.json +1 -1
- data/lib/praxis.rb +10 -4
- data/lib/praxis/action_definition.rb +16 -4
- data/lib/praxis/action_definition/headers_dsl_compiler.rb +5 -2
- data/lib/praxis/api_definition.rb +3 -1
- data/lib/praxis/api_general_info.rb +49 -5
- data/lib/praxis/application.rb +12 -4
- data/lib/praxis/bootloader.rb +1 -0
- data/lib/praxis/bootloader_stages/environment.rb +2 -0
- data/lib/praxis/bootloader_stages/routing.rb +1 -1
- data/lib/praxis/bootloader_stages/subgroup_loader.rb +1 -0
- data/lib/praxis/exceptions/validation.rb +7 -0
- data/lib/praxis/handlers/plain.rb +16 -0
- data/lib/praxis/handlers/xml.rb +4 -4
- data/lib/praxis/links.rb +13 -3
- data/lib/praxis/media_type_identifier.rb +3 -0
- data/lib/praxis/multipart/parser.rb +41 -48
- data/lib/praxis/multipart/part.rb +196 -3
- data/lib/praxis/request.rb +14 -11
- data/lib/praxis/request_stages/request_stage.rb +4 -0
- data/lib/praxis/request_stages/response.rb +10 -9
- data/lib/praxis/request_stages/validate.rb +1 -7
- data/lib/praxis/request_stages/validate_params_and_headers.rb +30 -5
- data/lib/praxis/request_stages/validate_payload.rb +14 -5
- data/lib/praxis/resource_definition.rb +117 -15
- data/lib/praxis/response.rb +6 -5
- data/lib/praxis/response_definition.rb +51 -5
- data/lib/praxis/responses/http.rb +5 -0
- data/lib/praxis/responses/multipart_ok.rb +51 -0
- data/lib/praxis/responses/validation_error.rb +7 -7
- data/lib/praxis/restful_doc_generator.rb +9 -4
- data/lib/praxis/route.rb +3 -2
- data/lib/praxis/router.rb +26 -16
- data/lib/praxis/router/rack.rb +51 -0
- data/lib/praxis/router/simple.rb +146 -0
- data/lib/praxis/routing_config.rb +2 -2
- data/lib/praxis/trait.rb +1 -1
- data/lib/praxis/types/fuzzy_hash.rb +49 -0
- data/lib/praxis/types/media_type_common.rb +1 -1
- data/lib/praxis/types/multipart.rb +47 -12
- data/lib/praxis/types/multipart_array.rb +320 -0
- data/lib/praxis/types/multipart_array/part_definition.rb +52 -0
- data/lib/praxis/validation_handler.rb +10 -0
- data/lib/praxis/version.rb +2 -2
- data/praxis.gemspec +3 -3
- data/spec/api_browser/directives/type_placeholder_spec.js +134 -0
- data/spec/api_browser/factories/normalize_attributes_spec.js +97 -0
- data/spec/api_browser/factories/template_for_spec.js +67 -0
- data/spec/functional_spec.rb +111 -45
- data/spec/praxis/action_definition_spec.rb +31 -7
- data/spec/praxis/api_definition_spec.rb +2 -2
- data/spec/praxis/api_general_info_spec.rb +25 -0
- data/spec/praxis/application_spec.rb +24 -11
- data/spec/praxis/handlers/xml_spec.rb +55 -33
- data/spec/praxis/links_spec.rb +18 -1
- data/spec/praxis/media_type_collection_spec.rb +1 -1
- data/spec/praxis/media_type_spec.rb +2 -2
- data/spec/praxis/multipart/parser_spec.rb +21 -13
- data/spec/praxis/plugins/praxis_mapper_plugin_spec.rb +1 -1
- data/spec/praxis/request_spec.rb +52 -24
- data/spec/praxis/{request_stages_action_spec.rb → request_stages/action_spec.rb} +1 -1
- data/spec/praxis/{request_stage_spec.rb → request_stages/request_stage_spec.rb} +0 -0
- data/spec/praxis/{request_stages_validate_spec.rb → request_stages/validate_spec.rb} +1 -1
- data/spec/praxis/resource_definition_spec.rb +30 -4
- data/spec/praxis/response_definition_spec.rb +60 -19
- data/spec/praxis/response_spec.rb +2 -2
- data/spec/praxis/responses/validation_error_spec.rb +33 -16
- data/spec/praxis/route_spec.rb +4 -2
- data/spec/praxis/router_spec.rb +28 -12
- data/spec/praxis/routing_config_spec.rb +11 -5
- data/spec/praxis/types/collection_spec.rb +1 -1
- data/spec/praxis/types/fuzzy_hash_spec.rb +20 -0
- data/spec/praxis/types/multipart_array/part_definition_spec.rb +5 -0
- data/spec/praxis/types/multipart_array_spec.rb +334 -0
- data/spec/praxis/types/multipart_spec.rb +14 -5
- data/spec/spec_app/app/controllers/instances.rb +20 -10
- data/spec/spec_app/app/controllers/volumes.rb +8 -4
- data/spec/spec_app/app/responses/bulk_response.rb +0 -6
- data/spec/spec_app/config/environment.rb +13 -0
- data/spec/spec_app/design/api.rb +7 -10
- data/spec/spec_app/design/media_types/instance.rb +3 -1
- data/spec/spec_app/design/resources/instances.rb +50 -41
- data/spec/spec_app/design/resources/volume_snapshots.rb +39 -0
- data/spec/spec_app/design/resources/volumes.rb +11 -6
- data/spec/spec_helper.rb +3 -1
- metadata +125 -218
- data/lib/api_browser/app/bower_components/angular-ui-router/src/compat.js +0 -146
- data/lib/api_browser/app/bower_components/bootstrap-sass/CNAME +0 -1
- data/lib/api_browser/app/bower_components/bootstrap-sass/DOCS-LICENSE +0 -319
- data/lib/api_browser/app/bower_components/bootstrap-sass/Gemfile +0 -5
- data/lib/api_browser/app/bower_components/bootstrap-sass/Gemfile.lock +0 -14
- data/lib/api_browser/app/bower_components/bootstrap-sass/Gruntfile.js +0 -244
- data/lib/api_browser/app/bower_components/bootstrap-sass/LICENSE +0 -176
- data/lib/api_browser/app/bower_components/bootstrap-sass/LICENSE-MIT +0 -21
- data/lib/api_browser/app/bower_components/bootstrap-sass/Rakefile +0 -44
- data/lib/api_browser/app/bower_components/bootstrap-sass/_config.yml +0 -25
- data/lib/api_browser/app/bower_components/bootstrap-sass/_includes/ads.html +0 -1
- data/lib/api_browser/app/bower_components/bootstrap-sass/_includes/footer.html +0 -34
- data/lib/api_browser/app/bower_components/bootstrap-sass/_includes/header.html +0 -42
- data/lib/api_browser/app/bower_components/bootstrap-sass/_includes/nav-about.html +0 -12
- data/lib/api_browser/app/bower_components/bootstrap-sass/_includes/nav-components.html +0 -137
- data/lib/api_browser/app/bower_components/bootstrap-sass/_includes/nav-css.html +0 -99
- data/lib/api_browser/app/bower_components/bootstrap-sass/_includes/nav-customize.html +0 -40
- data/lib/api_browser/app/bower_components/bootstrap-sass/_includes/nav-getting-started.html +0 -44
- data/lib/api_browser/app/bower_components/bootstrap-sass/_includes/nav-javascript.html +0 -88
- data/lib/api_browser/app/bower_components/bootstrap-sass/_includes/nav-main.html +0 -37
- data/lib/api_browser/app/bower_components/bootstrap-sass/_includes/old-bs-docs.html +0 -8
- data/lib/api_browser/app/bower_components/bootstrap-sass/_includes/social-buttons.html +0 -16
- data/lib/api_browser/app/bower_components/bootstrap-sass/_layouts/default.html +0 -79
- data/lib/api_browser/app/bower_components/bootstrap-sass/_layouts/home.html +0 -47
- data/lib/api_browser/app/bower_components/bootstrap-sass/about.html +0 -93
- data/lib/api_browser/app/bower_components/bootstrap-sass/browserstack.json +0 -37
- data/lib/api_browser/app/bower_components/bootstrap-sass/components.html +0 -3689
- data/lib/api_browser/app/bower_components/bootstrap-sass/composer.json +0 -28
- data/lib/api_browser/app/bower_components/bootstrap-sass/css.html +0 -2674
- data/lib/api_browser/app/bower_components/bootstrap-sass/customize.html +0 -1715
- data/lib/api_browser/app/bower_components/bootstrap-sass/dist/css/bootstrap-theme.css +0 -427
- data/lib/api_browser/app/bower_components/bootstrap-sass/dist/css/bootstrap-theme.min.css +0 -1
- data/lib/api_browser/app/bower_components/bootstrap-sass/dist/css/bootstrap.css +0 -6350
- data/lib/api_browser/app/bower_components/bootstrap-sass/dist/css/bootstrap.min.css +0 -1
- data/lib/api_browser/app/bower_components/bootstrap-sass/dist/js/bootstrap.js +0 -2002
- data/lib/api_browser/app/bower_components/bootstrap-sass/dist/js/bootstrap.min.js +0 -9
- data/lib/api_browser/app/bower_components/bootstrap-sass/docs-assets/css/docs.css +0 -1195
- data/lib/api_browser/app/bower_components/bootstrap-sass/docs-assets/css/pygments-manni.css +0 -66
- data/lib/api_browser/app/bower_components/bootstrap-sass/docs-assets/ico/apple-touch-icon-144-precomposed.png +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/docs-assets/ico/favicon.png +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/docs-assets/js/application.js +0 -103
- data/lib/api_browser/app/bower_components/bootstrap-sass/docs-assets/js/customizer.js +0 -332
- data/lib/api_browser/app/bower_components/bootstrap-sass/docs-assets/js/filesaver.js +0 -169
- data/lib/api_browser/app/bower_components/bootstrap-sass/docs-assets/js/holder.js +0 -404
- data/lib/api_browser/app/bower_components/bootstrap-sass/docs-assets/js/ie8-responsive-file-warning.js +0 -12
- data/lib/api_browser/app/bower_components/bootstrap-sass/docs-assets/js/jszip.js +0 -1467
- data/lib/api_browser/app/bower_components/bootstrap-sass/docs-assets/js/less.js +0 -9
- data/lib/api_browser/app/bower_components/bootstrap-sass/docs-assets/js/raw-files.js +0 -3
- data/lib/api_browser/app/bower_components/bootstrap-sass/docs-assets/js/uglify.js +0 -14
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/carousel/carousel.css +0 -148
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/carousel/index.html +0 -206
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/grid/grid.css +0 -28
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/grid/index.html +0 -148
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/jumbotron-narrow/index.html +0 -82
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/jumbotron-narrow/jumbotron-narrow.css +0 -79
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/jumbotron/index.html +0 -99
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/jumbotron/jumbotron.css +0 -5
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/justified-nav/index.html +0 -83
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/justified-nav/justified-nav.css +0 -88
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/navbar-fixed-top/index.html +0 -91
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/navbar-fixed-top/navbar-fixed-top.css +0 -4
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/navbar-static-top/index.html +0 -92
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/navbar-static-top/navbar-static-top.css +0 -7
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/navbar/index.html +0 -88
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/navbar/navbar.css +0 -8
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/non-responsive/index.html +0 -101
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/non-responsive/non-responsive.css +0 -116
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/offcanvas/index.html +0 -130
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/offcanvas/offcanvas.css +0 -50
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/offcanvas/offcanvas.js +0 -5
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/screenshots/carousel.jpg +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/screenshots/grid.jpg +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/screenshots/jumbotron-narrow.jpg +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/screenshots/jumbotron.jpg +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/screenshots/justified-nav.jpg +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/screenshots/navbar-fixed.jpg +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/screenshots/navbar-static.jpg +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/screenshots/navbar.jpg +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/screenshots/non-responsive.jpg +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/screenshots/offcanvas.jpg +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/screenshots/sign-in.jpg +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/screenshots/starter-template.jpg +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/screenshots/sticky-footer-navbar.jpg +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/screenshots/sticky-footer.jpg +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/screenshots/theme.jpg +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/signin/index.html +0 -50
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/signin/signin.css +0 -40
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/starter-template/index.html +0 -68
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/starter-template/starter-template.css +0 -7
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/sticky-footer-navbar/index.html +0 -91
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/sticky-footer-navbar/sticky-footer-navbar.css +0 -45
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/sticky-footer/index.html +0 -55
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/sticky-footer/sticky-footer.css +0 -38
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/theme/index.html +0 -384
- data/lib/api_browser/app/bower_components/bootstrap-sass/examples/theme/theme.css +0 -14
- data/lib/api_browser/app/bower_components/bootstrap-sass/fonts/glyphicons-halflings-regular.eot +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/fonts/glyphicons-halflings-regular.svg +0 -229
- data/lib/api_browser/app/bower_components/bootstrap-sass/fonts/glyphicons-halflings-regular.ttf +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/fonts/glyphicons-halflings-regular.woff +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/getting-started.html +0 -1021
- data/lib/api_browser/app/bower_components/bootstrap-sass/index.html +0 -16
- data/lib/api_browser/app/bower_components/bootstrap-sass/javascript.html +0 -1983
- data/lib/api_browser/app/bower_components/bootstrap-sass/js/tests/index.html +0 -52
- data/lib/api_browser/app/bower_components/bootstrap-sass/js/tests/unit/affix.js +0 -25
- data/lib/api_browser/app/bower_components/bootstrap-sass/js/tests/unit/alert.js +0 -62
- data/lib/api_browser/app/bower_components/bootstrap-sass/js/tests/unit/button.js +0 -116
- data/lib/api_browser/app/bower_components/bootstrap-sass/js/tests/unit/carousel.js +0 -87
- data/lib/api_browser/app/bower_components/bootstrap-sass/js/tests/unit/collapse.js +0 -164
- data/lib/api_browser/app/bower_components/bootstrap-sass/js/tests/unit/dropdown.js +0 -219
- data/lib/api_browser/app/bower_components/bootstrap-sass/js/tests/unit/modal.js +0 -196
- data/lib/api_browser/app/bower_components/bootstrap-sass/js/tests/unit/phantom.js +0 -69
- data/lib/api_browser/app/bower_components/bootstrap-sass/js/tests/unit/popover.js +0 -133
- data/lib/api_browser/app/bower_components/bootstrap-sass/js/tests/unit/scrollspy.js +0 -37
- data/lib/api_browser/app/bower_components/bootstrap-sass/js/tests/unit/tab.js +0 -86
- data/lib/api_browser/app/bower_components/bootstrap-sass/js/tests/unit/tooltip.js +0 -437
- data/lib/api_browser/app/bower_components/bootstrap-sass/js/tests/unit/transition.js +0 -13
- data/lib/api_browser/app/bower_components/bootstrap-sass/js/tests/vendor/jquery.js +0 -6
- data/lib/api_browser/app/bower_components/bootstrap-sass/js/tests/vendor/qunit.css +0 -232
- data/lib/api_browser/app/bower_components/bootstrap-sass/js/tests/vendor/qunit.js +0 -1510
- data/lib/api_browser/app/bower_components/bootstrap-sass/package.json +0 -40
- data/lib/api_browser/app/bower_components/lodash/dist/lodash.compat.js +0 -7157
- data/lib/api_browser/app/bower_components/lodash/dist/lodash.compat.min.js +0 -61
- data/lib/api_browser/app/bower_components/lodash/dist/lodash.js +0 -6785
- data/lib/api_browser/app/bower_components/lodash/dist/lodash.min.js +0 -56
- data/lib/api_browser/app/bower_components/lodash/dist/lodash.underscore.js +0 -4979
- data/lib/api_browser/app/bower_components/lodash/dist/lodash.underscore.min.js +0 -39
- data/lib/api_browser/app/js/directives/attribute_table_row.js +0 -17
- data/lib/api_browser/app/js/directives/request_body.js +0 -25
- data/lib/api_browser/app/js/directives/request_headers.js +0 -17
- data/lib/api_browser/app/js/directives/request_parameters.js +0 -17
- data/lib/api_browser/app/js/directives/type_label.js +0 -52
- data/lib/api_browser/app/js/factories/PayloadTemplates.js +0 -10
- data/lib/api_browser/app/js/factories/TemplateProvider.js +0 -45
- data/lib/api_browser/app/js/factories/TypeTemplates.js +0 -11
- data/lib/api_browser/app/views/directives/attribute_description/_example.html +0 -13
- data/lib/api_browser/app/views/directives/attribute_description/_headers.html +0 -8
- data/lib/api_browser/app/views/directives/attribute_table_row/_default.html +0 -10
- data/lib/api_browser/app/views/directives/attribute_table_row/_links.html +0 -11
- data/lib/api_browser/app/views/directives/attribute_table_row/_struct.html +0 -2
- data/lib/api_browser/app/views/directives/request_body/_struct.html +0 -1
- data/lib/api_browser/app/views/resource/_actions.html +0 -27
@@ -0,0 +1,126 @@
|
|
1
|
+
// Type definitions for Angular JS 1.1.5+ (ui.router module)
|
2
|
+
// Project: https://github.com/angular-ui/ui-router
|
3
|
+
// Definitions by: Michel Salib <https://github.com/michelsalib>
|
4
|
+
// Definitions: https://github.com/borisyankov/DefinitelyTyped
|
5
|
+
|
6
|
+
declare module ng.ui {
|
7
|
+
|
8
|
+
interface IState {
|
9
|
+
name?: string;
|
10
|
+
template?: string;
|
11
|
+
templateUrl?: any; // string || () => string
|
12
|
+
templateProvider?: any; // () => string || IPromise<string>
|
13
|
+
controller?: any;
|
14
|
+
controllerAs?: string;
|
15
|
+
controllerProvider?: any;
|
16
|
+
resolve?: {};
|
17
|
+
url?: string;
|
18
|
+
params?: any;
|
19
|
+
views?: {};
|
20
|
+
abstract?: boolean;
|
21
|
+
onEnter?: (...args: any[]) => void;
|
22
|
+
onExit?: (...args: any[]) => void;
|
23
|
+
data?: any;
|
24
|
+
reloadOnSearch?: boolean;
|
25
|
+
}
|
26
|
+
|
27
|
+
interface ITypedState<T> extends IState {
|
28
|
+
data?: T;
|
29
|
+
}
|
30
|
+
|
31
|
+
interface IStateProvider extends IServiceProvider {
|
32
|
+
state(name: string, config: IState): IStateProvider;
|
33
|
+
state(config: IState): IStateProvider;
|
34
|
+
decorator(name?: string, decorator?: (state: IState, parent: Function) => any): any;
|
35
|
+
}
|
36
|
+
|
37
|
+
interface IUrlMatcher {
|
38
|
+
concat(pattern: string): IUrlMatcher;
|
39
|
+
exec(path: string, searchParams: {}): {};
|
40
|
+
parameters(): string[];
|
41
|
+
format(values: {}): string;
|
42
|
+
}
|
43
|
+
|
44
|
+
interface IUrlMatcherFactory {
|
45
|
+
compile(pattern: string): IUrlMatcher;
|
46
|
+
isMatcher(o: any): boolean;
|
47
|
+
}
|
48
|
+
|
49
|
+
interface IUrlRouterProvider extends IServiceProvider {
|
50
|
+
when(whenPath: RegExp, handler: Function): IUrlRouterProvider;
|
51
|
+
when(whenPath: RegExp, handler: any[]): IUrlRouterProvider;
|
52
|
+
when(whenPath: RegExp, toPath: string): IUrlRouterProvider;
|
53
|
+
when(whenPath: IUrlMatcher, hanlder: Function): IUrlRouterProvider;
|
54
|
+
when(whenPath: IUrlMatcher, handler: any[]): IUrlRouterProvider;
|
55
|
+
when(whenPath: IUrlMatcher, toPath: string): IUrlRouterProvider;
|
56
|
+
when(whenPath: string, handler: Function): IUrlRouterProvider;
|
57
|
+
when(whenPath: string, handler: any[]): IUrlRouterProvider;
|
58
|
+
when(whenPath: string, toPath: string): IUrlRouterProvider;
|
59
|
+
otherwise(handler: Function): IUrlRouterProvider;
|
60
|
+
otherwise(handler: any[]): IUrlRouterProvider;
|
61
|
+
otherwise(path: string): IUrlRouterProvider;
|
62
|
+
rule(handler: Function): IUrlRouterProvider;
|
63
|
+
rule(handler: any[]): IUrlRouterProvider;
|
64
|
+
}
|
65
|
+
|
66
|
+
interface IStateOptions {
|
67
|
+
location?: any;
|
68
|
+
inherit?: boolean;
|
69
|
+
relative?: IState;
|
70
|
+
notify?: boolean;
|
71
|
+
reload?: boolean;
|
72
|
+
}
|
73
|
+
|
74
|
+
interface IHrefOptions {
|
75
|
+
lossy?: boolean;
|
76
|
+
inherit?: boolean;
|
77
|
+
relative?: IState;
|
78
|
+
absolute?: boolean;
|
79
|
+
}
|
80
|
+
|
81
|
+
interface IStateService {
|
82
|
+
go(to: string, params?: {}, options?: IStateOptions): IPromise<any>;
|
83
|
+
transitionTo(state: string, params?: {}, updateLocation?: boolean): void;
|
84
|
+
transitionTo(state: string, params?: {}, options?: IStateOptions): void;
|
85
|
+
includes(state: string, params?: {}): boolean;
|
86
|
+
is(state:string, params?: {}): boolean;
|
87
|
+
is(state: IState, params?: {}): boolean;
|
88
|
+
href(state: IState, params?: {}, options?: IHrefOptions): string;
|
89
|
+
href(state: string, params?: {}, options?: IHrefOptions): string;
|
90
|
+
get(state: string): IState;
|
91
|
+
get(): IState[];
|
92
|
+
current: IState;
|
93
|
+
params: any;
|
94
|
+
reload(): void;
|
95
|
+
}
|
96
|
+
|
97
|
+
interface IStateParamsService {
|
98
|
+
[key: string]: any;
|
99
|
+
}
|
100
|
+
|
101
|
+
interface IStateParams {
|
102
|
+
[key: string]: any;
|
103
|
+
}
|
104
|
+
|
105
|
+
interface IUrlRouterService {
|
106
|
+
/*
|
107
|
+
* Triggers an update; the same update that happens when the address bar
|
108
|
+
* url changes, aka $locationChangeSuccess.
|
109
|
+
*
|
110
|
+
* This method is useful when you need to use preventDefault() on the
|
111
|
+
* $locationChangeSuccess event, perform some custom logic (route protection,
|
112
|
+
* auth, config, redirection, etc) and then finally proceed with the transition
|
113
|
+
* by calling $urlRouter.sync().
|
114
|
+
*
|
115
|
+
*/
|
116
|
+
sync(): void;
|
117
|
+
}
|
118
|
+
|
119
|
+
interface IUiViewScrollProvider {
|
120
|
+
/*
|
121
|
+
* Reverts back to using the core $anchorScroll service for scrolling
|
122
|
+
* based on the url anchor.
|
123
|
+
*/
|
124
|
+
useAnchorScroll(): void;
|
125
|
+
}
|
126
|
+
}
|
@@ -1,6 +1,6 @@
|
|
1
1
|
/**
|
2
2
|
* State-based routing for AngularJS
|
3
|
-
* @version v0.2.
|
3
|
+
* @version v0.2.15
|
4
4
|
* @link http://angular-ui.github.com/
|
5
5
|
* @license MIT License, http://www.opensource.org/licenses/MIT
|
6
6
|
*/
|
@@ -62,13 +62,13 @@ function ancestors(first, second) {
|
|
62
62
|
* @param {Object} object A JavaScript object.
|
63
63
|
* @return {Array} Returns the keys of the object as an array.
|
64
64
|
*/
|
65
|
-
function
|
65
|
+
function objectKeys(object) {
|
66
66
|
if (Object.keys) {
|
67
67
|
return Object.keys(object);
|
68
68
|
}
|
69
69
|
var result = [];
|
70
70
|
|
71
|
-
|
71
|
+
forEach(object, function(val, key) {
|
72
72
|
result.push(key);
|
73
73
|
});
|
74
74
|
return result;
|
@@ -81,7 +81,7 @@ function keys(object) {
|
|
81
81
|
* @param {*} value A value to search the array for.
|
82
82
|
* @return {Number} Returns the array index value of `value`, or `-1` if not present.
|
83
83
|
*/
|
84
|
-
function
|
84
|
+
function indexOf(array, value) {
|
85
85
|
if (Array.prototype.indexOf) {
|
86
86
|
return array.indexOf(value, Number(arguments[2]) || 0);
|
87
87
|
}
|
@@ -109,11 +109,12 @@ function inheritParams(currentParams, newParams, $current, $to) {
|
|
109
109
|
var parents = ancestors($current, $to), parentParams, inherited = {}, inheritList = [];
|
110
110
|
|
111
111
|
for (var i in parents) {
|
112
|
-
if (!parents[i].params
|
113
|
-
parentParams = parents[i].params;
|
112
|
+
if (!parents[i].params) continue;
|
113
|
+
parentParams = objectKeys(parents[i].params);
|
114
|
+
if (!parentParams.length) continue;
|
114
115
|
|
115
116
|
for (var j in parentParams) {
|
116
|
-
if (
|
117
|
+
if (indexOf(inheritList, parentParams[j]) >= 0) continue;
|
117
118
|
inheritList.push(parentParams[j]);
|
118
119
|
inherited[parentParams[j]] = currentParams[parentParams[j]];
|
119
120
|
}
|
@@ -121,23 +122,6 @@ function inheritParams(currentParams, newParams, $current, $to) {
|
|
121
122
|
return extend({}, inherited, newParams);
|
122
123
|
}
|
123
124
|
|
124
|
-
/**
|
125
|
-
* Normalizes a set of values to string or `null`, filtering them by a list of keys.
|
126
|
-
*
|
127
|
-
* @param {Array} keys The list of keys to normalize/return.
|
128
|
-
* @param {Object} values An object hash of values to normalize.
|
129
|
-
* @return {Object} Returns an object hash of normalized string values.
|
130
|
-
*/
|
131
|
-
function normalize(keys, values) {
|
132
|
-
var normalized = {};
|
133
|
-
|
134
|
-
forEach(keys, function (name) {
|
135
|
-
var value = values[name];
|
136
|
-
normalized[name] = (value != null) ? String(value) : null;
|
137
|
-
});
|
138
|
-
return normalized;
|
139
|
-
}
|
140
|
-
|
141
125
|
/**
|
142
126
|
* Performs a non-strict comparison of the subset of two objects, defined by a list of keys.
|
143
127
|
*
|
@@ -175,6 +159,68 @@ function filterByKeys(keys, values) {
|
|
175
159
|
});
|
176
160
|
return filtered;
|
177
161
|
}
|
162
|
+
|
163
|
+
// like _.indexBy
|
164
|
+
// when you know that your index values will be unique, or you want last-one-in to win
|
165
|
+
function indexBy(array, propName) {
|
166
|
+
var result = {};
|
167
|
+
forEach(array, function(item) {
|
168
|
+
result[item[propName]] = item;
|
169
|
+
});
|
170
|
+
return result;
|
171
|
+
}
|
172
|
+
|
173
|
+
// extracted from underscore.js
|
174
|
+
// Return a copy of the object only containing the whitelisted properties.
|
175
|
+
function pick(obj) {
|
176
|
+
var copy = {};
|
177
|
+
var keys = Array.prototype.concat.apply(Array.prototype, Array.prototype.slice.call(arguments, 1));
|
178
|
+
forEach(keys, function(key) {
|
179
|
+
if (key in obj) copy[key] = obj[key];
|
180
|
+
});
|
181
|
+
return copy;
|
182
|
+
}
|
183
|
+
|
184
|
+
// extracted from underscore.js
|
185
|
+
// Return a copy of the object omitting the blacklisted properties.
|
186
|
+
function omit(obj) {
|
187
|
+
var copy = {};
|
188
|
+
var keys = Array.prototype.concat.apply(Array.prototype, Array.prototype.slice.call(arguments, 1));
|
189
|
+
for (var key in obj) {
|
190
|
+
if (indexOf(keys, key) == -1) copy[key] = obj[key];
|
191
|
+
}
|
192
|
+
return copy;
|
193
|
+
}
|
194
|
+
|
195
|
+
function pluck(collection, key) {
|
196
|
+
var result = isArray(collection) ? [] : {};
|
197
|
+
|
198
|
+
forEach(collection, function(val, i) {
|
199
|
+
result[i] = isFunction(key) ? key(val) : val[key];
|
200
|
+
});
|
201
|
+
return result;
|
202
|
+
}
|
203
|
+
|
204
|
+
function filter(collection, callback) {
|
205
|
+
var array = isArray(collection);
|
206
|
+
var result = array ? [] : {};
|
207
|
+
forEach(collection, function(val, i) {
|
208
|
+
if (callback(val, i)) {
|
209
|
+
result[array ? result.length : i] = val;
|
210
|
+
}
|
211
|
+
});
|
212
|
+
return result;
|
213
|
+
}
|
214
|
+
|
215
|
+
function map(collection, callback) {
|
216
|
+
var result = isArray(collection) ? [] : {};
|
217
|
+
|
218
|
+
forEach(collection, function(val, i) {
|
219
|
+
result[i] = callback(val, i);
|
220
|
+
});
|
221
|
+
return result;
|
222
|
+
}
|
223
|
+
|
178
224
|
/**
|
179
225
|
* @ngdoc overview
|
180
226
|
* @name ui.router.util
|
@@ -301,6 +347,7 @@ function $Resolve( $q, $injector) {
|
|
301
347
|
*/
|
302
348
|
this.study = function (invocables) {
|
303
349
|
if (!isObject(invocables)) throw new Error("'invocables' must be an object");
|
350
|
+
var invocableKeys = objectKeys(invocables || {});
|
304
351
|
|
305
352
|
// Perform a topological sort of invocables to build an ordered plan
|
306
353
|
var plan = [], cycle = [], visited = {};
|
@@ -309,7 +356,7 @@ function $Resolve( $q, $injector) {
|
|
309
356
|
|
310
357
|
cycle.push(key);
|
311
358
|
if (visited[key] === VISIT_IN_PROGRESS) {
|
312
|
-
cycle.splice(0,
|
359
|
+
cycle.splice(0, indexOf(cycle, key));
|
313
360
|
throw new Error("Cyclic dependency: " + cycle.join(" -> "));
|
314
361
|
}
|
315
362
|
visited[key] = VISIT_IN_PROGRESS;
|
@@ -361,7 +408,8 @@ function $Resolve( $q, $injector) {
|
|
361
408
|
if (!--wait) {
|
362
409
|
if (!merged) merge(values, parent.$$values);
|
363
410
|
result.$$values = values;
|
364
|
-
result.$$promises = true; // keep for isResolve()
|
411
|
+
result.$$promises = result.$$promises || true; // keep for isResolve()
|
412
|
+
delete result.$$inheritedValues;
|
365
413
|
resolution.resolve(values);
|
366
414
|
}
|
367
415
|
}
|
@@ -370,20 +418,28 @@ function $Resolve( $q, $injector) {
|
|
370
418
|
result.$$failure = reason;
|
371
419
|
resolution.reject(reason);
|
372
420
|
}
|
373
|
-
|
421
|
+
|
374
422
|
// Short-circuit if parent has already failed
|
375
423
|
if (isDefined(parent.$$failure)) {
|
376
424
|
fail(parent.$$failure);
|
377
425
|
return result;
|
378
426
|
}
|
379
427
|
|
428
|
+
if (parent.$$inheritedValues) {
|
429
|
+
merge(values, omit(parent.$$inheritedValues, invocableKeys));
|
430
|
+
}
|
431
|
+
|
380
432
|
// Merge parent values if the parent has already resolved, or merge
|
381
433
|
// parent promises and wait if the parent resolve is still in progress.
|
434
|
+
extend(promises, parent.$$promises);
|
382
435
|
if (parent.$$values) {
|
383
|
-
merged = merge(values, parent.$$values);
|
436
|
+
merged = merge(values, omit(parent.$$values, invocableKeys));
|
437
|
+
result.$$inheritedValues = omit(parent.$$values, invocableKeys);
|
384
438
|
done();
|
385
439
|
} else {
|
386
|
-
|
440
|
+
if (parent.$$inheritedValues) {
|
441
|
+
result.$$inheritedValues = omit(parent.$$inheritedValues, invocableKeys);
|
442
|
+
}
|
387
443
|
parent.then(done, fail);
|
388
444
|
}
|
389
445
|
|
@@ -586,13 +642,13 @@ function $TemplateFactory( $http, $templateCache, $injector) {
|
|
586
642
|
if (isFunction(url)) url = url(params);
|
587
643
|
if (url == null) return null;
|
588
644
|
else return $http
|
589
|
-
.get(url, { cache: $templateCache })
|
645
|
+
.get(url, { cache: $templateCache, headers: { Accept: 'text/html' }})
|
590
646
|
.then(function(response) { return response.data; });
|
591
647
|
};
|
592
648
|
|
593
649
|
/**
|
594
650
|
* @ngdoc function
|
595
|
-
* @name ui.router.util.$templateFactory#
|
651
|
+
* @name ui.router.util.$templateFactory#fromProvider
|
596
652
|
* @methodOf ui.router.util.$templateFactory
|
597
653
|
*
|
598
654
|
* @description
|
@@ -612,6 +668,8 @@ function $TemplateFactory( $http, $templateCache, $injector) {
|
|
612
668
|
|
613
669
|
angular.module('ui.router.util').service('$templateFactory', $TemplateFactory);
|
614
670
|
|
671
|
+
var $$UMFP; // reference to $UrlMatcherFactoryProvider
|
672
|
+
|
615
673
|
/**
|
616
674
|
* @ngdoc object
|
617
675
|
* @name ui.router.util.type:UrlMatcher
|
@@ -622,24 +680,24 @@ angular.module('ui.router.util').service('$templateFactory', $TemplateFactory);
|
|
622
680
|
* of search parameters. Multiple search parameter names are separated by '&'. Search parameters
|
623
681
|
* do not influence whether or not a URL is matched, but their values are passed through into
|
624
682
|
* the matched parameters returned by {@link ui.router.util.type:UrlMatcher#methods_exec exec}.
|
625
|
-
*
|
683
|
+
*
|
626
684
|
* Path parameter placeholders can be specified using simple colon/catch-all syntax or curly brace
|
627
685
|
* syntax, which optionally allows a regular expression for the parameter to be specified:
|
628
686
|
*
|
629
687
|
* * `':'` name - colon placeholder
|
630
688
|
* * `'*'` name - catch-all placeholder
|
631
689
|
* * `'{' name '}'` - curly placeholder
|
632
|
-
* * `'{' name ':' regexp '}'` - curly placeholder with regexp. Should the
|
633
|
-
* curly braces, they must be in matched pairs or escaped with a backslash.
|
690
|
+
* * `'{' name ':' regexp|type '}'` - curly placeholder with regexp or type name. Should the
|
691
|
+
* regexp itself contain curly braces, they must be in matched pairs or escaped with a backslash.
|
634
692
|
*
|
635
693
|
* Parameter names may contain only word characters (latin letters, digits, and underscore) and
|
636
|
-
* must be unique within the pattern (across both path and search parameters). For colon
|
694
|
+
* must be unique within the pattern (across both path and search parameters). For colon
|
637
695
|
* placeholders or curly placeholders without an explicit regexp, a path parameter matches any
|
638
696
|
* number of characters other than '/'. For catch-all placeholders the path parameter matches
|
639
697
|
* any number of characters.
|
640
|
-
*
|
698
|
+
*
|
641
699
|
* Examples:
|
642
|
-
*
|
700
|
+
*
|
643
701
|
* * `'/hello/'` - Matches only if the path is exactly '/hello/'. There is no special treatment for
|
644
702
|
* trailing slashes, and patterns have to match the entire path, not just a prefix.
|
645
703
|
* * `'/user/:id'` - Matches '/user/bob' or '/user/1234!!!' or even '/user/' but not '/user' or
|
@@ -651,25 +709,34 @@ angular.module('ui.router.util').service('$templateFactory', $TemplateFactory);
|
|
651
709
|
* * `'/files/{path:.*}'` - Matches any URL starting with '/files/' and captures the rest of the
|
652
710
|
* path into the parameter 'path'.
|
653
711
|
* * `'/files/*path'` - ditto.
|
712
|
+
* * `'/calendar/{start:date}'` - Matches "/calendar/2014-11-12" (because the pattern defined
|
713
|
+
* in the built-in `date` Type matches `2014-11-12`) and provides a Date object in $stateParams.start
|
654
714
|
*
|
655
|
-
* @param {string} pattern
|
715
|
+
* @param {string} pattern The pattern to compile into a matcher.
|
716
|
+
* @param {Object} config A configuration object hash:
|
717
|
+
* @param {Object=} parentMatcher Used to concatenate the pattern/config onto
|
718
|
+
* an existing UrlMatcher
|
719
|
+
*
|
720
|
+
* * `caseInsensitive` - `true` if URL matching should be case insensitive, otherwise `false`, the default value (for backward compatibility) is `false`.
|
721
|
+
* * `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`.
|
656
722
|
*
|
657
723
|
* @property {string} prefix A static prefix of this pattern. The matcher guarantees that any
|
658
724
|
* URL matching this matcher (i.e. any string for which {@link ui.router.util.type:UrlMatcher#methods_exec exec()} returns
|
659
725
|
* non-null) will start with this prefix.
|
660
726
|
*
|
661
|
-
* @property {string} source The pattern that was passed into the
|
727
|
+
* @property {string} source The pattern that was passed into the constructor
|
662
728
|
*
|
663
729
|
* @property {string} sourcePath The path portion of the source property
|
664
730
|
*
|
665
731
|
* @property {string} sourceSearch The search portion of the source property
|
666
732
|
*
|
667
|
-
* @property {string} regex The constructed regex that will be used to match against the url when
|
733
|
+
* @property {string} regex The constructed regex that will be used to match against the url when
|
668
734
|
* it is time to determine which url will match.
|
669
735
|
*
|
670
|
-
* @returns {Object} New UrlMatcher object
|
736
|
+
* @returns {Object} New `UrlMatcher` object
|
671
737
|
*/
|
672
|
-
function UrlMatcher(pattern) {
|
738
|
+
function UrlMatcher(pattern, config, parentMatcher) {
|
739
|
+
config = extend({ params: {} }, isObject(config) ? config : {});
|
673
740
|
|
674
741
|
// Find all placeholders and create a compiled pattern, using either classic or curly syntax:
|
675
742
|
// '*' name
|
@@ -678,63 +745,96 @@ function UrlMatcher(pattern) {
|
|
678
745
|
// '{' name ':' regexp '}'
|
679
746
|
// The regular expression is somewhat complicated due to the need to allow curly braces
|
680
747
|
// inside the regular expression. The placeholder regexp breaks down as follows:
|
681
|
-
// ([:*])(\w+)
|
682
|
-
// \{(\w+)(?:\:( ... ))?\}
|
683
|
-
// (?: ... | ... | ... )+
|
684
|
-
// [^{}\\]+
|
685
|
-
// \\.
|
686
|
-
// \{(?:[^{}\\]+|\\.)*\}
|
687
|
-
var placeholder
|
688
|
-
|
748
|
+
// ([:*])([\w\[\]]+) - classic placeholder ($1 / $2) (search version has - for snake-case)
|
749
|
+
// \{([\w\[\]]+)(?:\:( ... ))?\} - curly brace placeholder ($3) with optional regexp/type ... ($4) (search version has - for snake-case
|
750
|
+
// (?: ... | ... | ... )+ - the regexp consists of any number of atoms, an atom being either
|
751
|
+
// [^{}\\]+ - anything other than curly braces or backslash
|
752
|
+
// \\. - a backslash escape
|
753
|
+
// \{(?:[^{}\\]+|\\.)*\} - a matched set of curly braces containing other atoms
|
754
|
+
var placeholder = /([:*])([\w\[\]]+)|\{([\w\[\]]+)(?:\:((?:[^{}\\]+|\\.|\{(?:[^{}\\]+|\\.)*\})+))?\}/g,
|
755
|
+
searchPlaceholder = /([:]?)([\w\[\]-]+)|\{([\w\[\]-]+)(?:\:((?:[^{}\\]+|\\.|\{(?:[^{}\\]+|\\.)*\})+))?\}/g,
|
756
|
+
compiled = '^', last = 0, m,
|
689
757
|
segments = this.segments = [],
|
690
|
-
|
691
|
-
|
692
|
-
|
693
|
-
|
694
|
-
|
695
|
-
|
696
|
-
|
758
|
+
parentParams = parentMatcher ? parentMatcher.params : {},
|
759
|
+
params = this.params = parentMatcher ? parentMatcher.params.$$new() : new $$UMFP.ParamSet(),
|
760
|
+
paramNames = [];
|
761
|
+
|
762
|
+
function addParameter(id, type, config, location) {
|
763
|
+
paramNames.push(id);
|
764
|
+
if (parentParams[id]) return parentParams[id];
|
765
|
+
if (!/^\w+(-+\w+)*(?:\[\])?$/.test(id)) throw new Error("Invalid parameter name '" + id + "' in pattern '" + pattern + "'");
|
766
|
+
if (params[id]) throw new Error("Duplicate parameter name '" + id + "' in pattern '" + pattern + "'");
|
767
|
+
params[id] = new $$UMFP.Param(id, type, config, location);
|
768
|
+
return params[id];
|
697
769
|
}
|
698
770
|
|
699
|
-
function quoteRegExp(string) {
|
700
|
-
|
771
|
+
function quoteRegExp(string, pattern, squash, optional) {
|
772
|
+
var surroundPattern = ['',''], result = string.replace(/[\\\[\]\^$*+?.()|{}]/g, "\\$&");
|
773
|
+
if (!pattern) return result;
|
774
|
+
switch(squash) {
|
775
|
+
case false: surroundPattern = ['(', ')' + (optional ? "?" : "")]; break;
|
776
|
+
case true: surroundPattern = ['?(', ')?']; break;
|
777
|
+
default: surroundPattern = ['(' + squash + "|", ')?']; break;
|
778
|
+
}
|
779
|
+
return result + surroundPattern[0] + pattern + surroundPattern[1];
|
701
780
|
}
|
702
781
|
|
703
782
|
this.source = pattern;
|
704
783
|
|
705
784
|
// Split into static segments separated by path parameter placeholders.
|
706
785
|
// The number of segments is always 1 more than the number of parameters.
|
707
|
-
|
786
|
+
function matchDetails(m, isSearch) {
|
787
|
+
var id, regexp, segment, type, cfg, arrayMode;
|
788
|
+
id = m[2] || m[3]; // IE[78] returns '' for unmatched groups instead of null
|
789
|
+
cfg = config.params[id];
|
790
|
+
segment = pattern.substring(last, m.index);
|
791
|
+
regexp = isSearch ? m[4] : m[4] || (m[1] == '*' ? '.*' : null);
|
792
|
+
type = $$UMFP.type(regexp || "string") || inherit($$UMFP.type("string"), { pattern: new RegExp(regexp, config.caseInsensitive ? 'i' : undefined) });
|
793
|
+
return {
|
794
|
+
id: id, regexp: regexp, segment: segment, type: type, cfg: cfg
|
795
|
+
};
|
796
|
+
}
|
797
|
+
|
798
|
+
var p, param, segment;
|
708
799
|
while ((m = placeholder.exec(pattern))) {
|
709
|
-
|
710
|
-
|
711
|
-
|
712
|
-
|
713
|
-
compiled += quoteRegExp(segment
|
714
|
-
|
715
|
-
segments.push(segment);
|
800
|
+
p = matchDetails(m, false);
|
801
|
+
if (p.segment.indexOf('?') >= 0) break; // we're into the search part
|
802
|
+
|
803
|
+
param = addParameter(p.id, p.type, p.cfg, "path");
|
804
|
+
compiled += quoteRegExp(p.segment, param.type.pattern.source, param.squash, param.isOptional);
|
805
|
+
segments.push(p.segment);
|
716
806
|
last = placeholder.lastIndex;
|
717
807
|
}
|
718
808
|
segment = pattern.substring(last);
|
719
809
|
|
720
810
|
// Find any search parameter names and remove them from the last segment
|
721
811
|
var i = segment.indexOf('?');
|
812
|
+
|
722
813
|
if (i >= 0) {
|
723
814
|
var search = this.sourceSearch = segment.substring(i);
|
724
815
|
segment = segment.substring(0, i);
|
725
|
-
this.sourcePath = pattern.substring(0, last+i);
|
726
|
-
|
727
|
-
|
728
|
-
|
816
|
+
this.sourcePath = pattern.substring(0, last + i);
|
817
|
+
|
818
|
+
if (search.length > 0) {
|
819
|
+
last = 0;
|
820
|
+
while ((m = searchPlaceholder.exec(search))) {
|
821
|
+
p = matchDetails(m, true);
|
822
|
+
param = addParameter(p.id, p.type, p.cfg, "search");
|
823
|
+
last = placeholder.lastIndex;
|
824
|
+
// check if ?&
|
825
|
+
}
|
826
|
+
}
|
729
827
|
} else {
|
730
828
|
this.sourcePath = pattern;
|
731
829
|
this.sourceSearch = '';
|
732
830
|
}
|
733
831
|
|
734
|
-
compiled += quoteRegExp(segment) + '$';
|
832
|
+
compiled += quoteRegExp(segment) + (config.strict === false ? '\/?' : '') + '$';
|
735
833
|
segments.push(segment);
|
736
|
-
|
834
|
+
|
835
|
+
this.regexp = new RegExp(compiled, config.caseInsensitive ? 'i' : undefined);
|
737
836
|
this.prefix = segments[0];
|
837
|
+
this.$$paramNames = paramNames;
|
738
838
|
}
|
739
839
|
|
740
840
|
/**
|
@@ -750,19 +850,25 @@ function UrlMatcher(pattern) {
|
|
750
850
|
*
|
751
851
|
* @example
|
752
852
|
* The following two matchers are equivalent:
|
753
|
-
*
|
853
|
+
* <pre>
|
754
854
|
* new UrlMatcher('/user/{id}?q').concat('/details?date');
|
755
855
|
* new UrlMatcher('/user/{id}/details?q&date');
|
756
|
-
*
|
856
|
+
* </pre>
|
757
857
|
*
|
758
858
|
* @param {string} pattern The pattern to append.
|
759
|
-
* @
|
859
|
+
* @param {Object} config An object hash of the configuration for the matcher.
|
860
|
+
* @returns {UrlMatcher} A matcher for the concatenated pattern.
|
760
861
|
*/
|
761
|
-
UrlMatcher.prototype.concat = function (pattern) {
|
862
|
+
UrlMatcher.prototype.concat = function (pattern, config) {
|
762
863
|
// Because order of search parameters is irrelevant, we can add our own search
|
763
864
|
// parameters to the end of the new pattern. Parse the new pattern by itself
|
764
865
|
// and then join the bits together, but it's much easier to do this on a string level.
|
765
|
-
|
866
|
+
var defaultConfig = {
|
867
|
+
caseInsensitive: $$UMFP.caseInsensitive(),
|
868
|
+
strict: $$UMFP.strictMode(),
|
869
|
+
squash: $$UMFP.defaultSquashPolicy()
|
870
|
+
};
|
871
|
+
return new UrlMatcher(this.sourcePath + pattern + this.sourceSearch, extend(defaultConfig, config), this);
|
766
872
|
};
|
767
873
|
|
768
874
|
UrlMatcher.prototype.toString = function () {
|
@@ -782,10 +888,12 @@ UrlMatcher.prototype.toString = function () {
|
|
782
888
|
* as optional.
|
783
889
|
*
|
784
890
|
* @example
|
785
|
-
*
|
786
|
-
* new UrlMatcher('/user/{id}?q&r').exec('/user/bob', {
|
787
|
-
*
|
788
|
-
*
|
891
|
+
* <pre>
|
892
|
+
* new UrlMatcher('/user/{id}?q&r').exec('/user/bob', {
|
893
|
+
* x: '1', q: 'hello'
|
894
|
+
* });
|
895
|
+
* // returns { id: 'bob', q: 'hello', r: null }
|
896
|
+
* </pre>
|
789
897
|
*
|
790
898
|
* @param {string} path The URL path to match, e.g. `$location.path()`.
|
791
899
|
* @param {Object} searchParams URL search parameters, e.g. `$location.search()`.
|
@@ -794,15 +902,38 @@ UrlMatcher.prototype.toString = function () {
|
|
794
902
|
UrlMatcher.prototype.exec = function (path, searchParams) {
|
795
903
|
var m = this.regexp.exec(path);
|
796
904
|
if (!m) return null;
|
905
|
+
searchParams = searchParams || {};
|
797
906
|
|
798
|
-
var
|
799
|
-
nPath = this.segments.length-1,
|
800
|
-
values = {}, i;
|
907
|
+
var paramNames = this.parameters(), nTotal = paramNames.length,
|
908
|
+
nPath = this.segments.length - 1,
|
909
|
+
values = {}, i, j, cfg, paramName;
|
801
910
|
|
802
911
|
if (nPath !== m.length - 1) throw new Error("Unbalanced capture group in route '" + this.source + "'");
|
803
912
|
|
804
|
-
|
805
|
-
|
913
|
+
function decodePathArray(string) {
|
914
|
+
function reverseString(str) { return str.split("").reverse().join(""); }
|
915
|
+
function unquoteDashes(str) { return str.replace(/\\-/g, "-"); }
|
916
|
+
|
917
|
+
var split = reverseString(string).split(/-(?!\\)/);
|
918
|
+
var allReversed = map(split, reverseString);
|
919
|
+
return map(allReversed, unquoteDashes).reverse();
|
920
|
+
}
|
921
|
+
|
922
|
+
for (i = 0; i < nPath; i++) {
|
923
|
+
paramName = paramNames[i];
|
924
|
+
var param = this.params[paramName];
|
925
|
+
var paramVal = m[i+1];
|
926
|
+
// if the param value matches a pre-replace pair, replace the value before decoding.
|
927
|
+
for (j = 0; j < param.replace; j++) {
|
928
|
+
if (param.replace[j].from === paramVal) paramVal = param.replace[j].to;
|
929
|
+
}
|
930
|
+
if (paramVal && param.array === true) paramVal = decodePathArray(paramVal);
|
931
|
+
values[paramName] = param.value(paramVal);
|
932
|
+
}
|
933
|
+
for (/**/; i < nTotal; i++) {
|
934
|
+
paramName = paramNames[i];
|
935
|
+
values[paramName] = this.params[paramName].value(searchParams[paramName]);
|
936
|
+
}
|
806
937
|
|
807
938
|
return values;
|
808
939
|
};
|
@@ -814,12 +945,29 @@ UrlMatcher.prototype.exec = function (path, searchParams) {
|
|
814
945
|
*
|
815
946
|
* @description
|
816
947
|
* Returns the names of all path and search parameters of this pattern in an unspecified order.
|
817
|
-
*
|
948
|
+
*
|
818
949
|
* @returns {Array.<string>} An array of parameter names. Must be treated as read-only. If the
|
819
950
|
* pattern has no parameters, an empty array is returned.
|
820
951
|
*/
|
821
|
-
UrlMatcher.prototype.parameters = function () {
|
822
|
-
return this
|
952
|
+
UrlMatcher.prototype.parameters = function (param) {
|
953
|
+
if (!isDefined(param)) return this.$$paramNames;
|
954
|
+
return this.params[param] || null;
|
955
|
+
};
|
956
|
+
|
957
|
+
/**
|
958
|
+
* @ngdoc function
|
959
|
+
* @name ui.router.util.type:UrlMatcher#validate
|
960
|
+
* @methodOf ui.router.util.type:UrlMatcher
|
961
|
+
*
|
962
|
+
* @description
|
963
|
+
* Checks an object hash of parameters to validate their correctness according to the parameter
|
964
|
+
* types of this `UrlMatcher`.
|
965
|
+
*
|
966
|
+
* @param {Object} params The object hash of parameters to validate.
|
967
|
+
* @returns {boolean} Returns `true` if `params` validates, otherwise `false`.
|
968
|
+
*/
|
969
|
+
UrlMatcher.prototype.validates = function (params) {
|
970
|
+
return this.params.$$validates(params);
|
823
971
|
};
|
824
972
|
|
825
973
|
/**
|
@@ -833,31 +981,54 @@ UrlMatcher.prototype.parameters = function () {
|
|
833
981
|
* treated as empty strings.
|
834
982
|
*
|
835
983
|
* @example
|
836
|
-
*
|
984
|
+
* <pre>
|
837
985
|
* new UrlMatcher('/user/{id}?q').format({ id:'bob', q:'yes' });
|
838
986
|
* // returns '/user/bob?q=yes'
|
839
|
-
*
|
987
|
+
* </pre>
|
840
988
|
*
|
841
989
|
* @param {Object} values the values to substitute for the parameters in this pattern.
|
842
990
|
* @returns {string} the formatted URL (path and optionally search part).
|
843
991
|
*/
|
844
992
|
UrlMatcher.prototype.format = function (values) {
|
845
|
-
|
846
|
-
|
993
|
+
values = values || {};
|
994
|
+
var segments = this.segments, params = this.parameters(), paramset = this.params;
|
995
|
+
if (!this.validates(values)) return null;
|
847
996
|
|
848
|
-
var nPath = segments.length-1, nTotal = params.length,
|
849
|
-
result = segments[0], i, search, value;
|
997
|
+
var i, search = false, nPath = segments.length - 1, nTotal = params.length, result = segments[0];
|
850
998
|
|
851
|
-
|
852
|
-
|
853
|
-
// TODO: Maybe we should throw on null here? It's not really good style to use '' and null interchangeabley
|
854
|
-
if (value != null) result += encodeURIComponent(value);
|
855
|
-
result += segments[i+1];
|
999
|
+
function encodeDashes(str) { // Replace dashes with encoded "\-"
|
1000
|
+
return encodeURIComponent(str).replace(/-/g, function(c) { return '%5C%' + c.charCodeAt(0).toString(16).toUpperCase(); });
|
856
1001
|
}
|
857
|
-
|
858
|
-
|
859
|
-
|
860
|
-
|
1002
|
+
|
1003
|
+
for (i = 0; i < nTotal; i++) {
|
1004
|
+
var isPathParam = i < nPath;
|
1005
|
+
var name = params[i], param = paramset[name], value = param.value(values[name]);
|
1006
|
+
var isDefaultValue = param.isOptional && param.type.equals(param.value(), value);
|
1007
|
+
var squash = isDefaultValue ? param.squash : false;
|
1008
|
+
var encoded = param.type.encode(value);
|
1009
|
+
|
1010
|
+
if (isPathParam) {
|
1011
|
+
var nextSegment = segments[i + 1];
|
1012
|
+
if (squash === false) {
|
1013
|
+
if (encoded != null) {
|
1014
|
+
if (isArray(encoded)) {
|
1015
|
+
result += map(encoded, encodeDashes).join("-");
|
1016
|
+
} else {
|
1017
|
+
result += encodeURIComponent(encoded);
|
1018
|
+
}
|
1019
|
+
}
|
1020
|
+
result += nextSegment;
|
1021
|
+
} else if (squash === true) {
|
1022
|
+
var capture = result.match(/\/$/) ? /\/?(.*)/ : /(.*)/;
|
1023
|
+
result += nextSegment.match(capture)[1];
|
1024
|
+
} else if (isString(squash)) {
|
1025
|
+
result += squash + nextSegment;
|
1026
|
+
}
|
1027
|
+
} else {
|
1028
|
+
if (encoded == null || (isDefaultValue && squash !== false)) continue;
|
1029
|
+
if (!isArray(encoded)) encoded = [ encoded ];
|
1030
|
+
encoded = map(encoded, encodeURIComponent).join('&' + name + '=');
|
1031
|
+
result += (search ? '&' : '?') + (name + '=' + encoded);
|
861
1032
|
search = true;
|
862
1033
|
}
|
863
1034
|
}
|
@@ -865,6 +1036,194 @@ UrlMatcher.prototype.format = function (values) {
|
|
865
1036
|
return result;
|
866
1037
|
};
|
867
1038
|
|
1039
|
+
/**
|
1040
|
+
* @ngdoc object
|
1041
|
+
* @name ui.router.util.type:Type
|
1042
|
+
*
|
1043
|
+
* @description
|
1044
|
+
* Implements an interface to define custom parameter types that can be decoded from and encoded to
|
1045
|
+
* string parameters matched in a URL. Used by {@link ui.router.util.type:UrlMatcher `UrlMatcher`}
|
1046
|
+
* objects when matching or formatting URLs, or comparing or validating parameter values.
|
1047
|
+
*
|
1048
|
+
* See {@link ui.router.util.$urlMatcherFactory#methods_type `$urlMatcherFactory#type()`} for more
|
1049
|
+
* information on registering custom types.
|
1050
|
+
*
|
1051
|
+
* @param {Object} config A configuration object which contains the custom type definition. The object's
|
1052
|
+
* properties will override the default methods and/or pattern in `Type`'s public interface.
|
1053
|
+
* @example
|
1054
|
+
* <pre>
|
1055
|
+
* {
|
1056
|
+
* decode: function(val) { return parseInt(val, 10); },
|
1057
|
+
* encode: function(val) { return val && val.toString(); },
|
1058
|
+
* equals: function(a, b) { return this.is(a) && a === b; },
|
1059
|
+
* is: function(val) { return angular.isNumber(val) isFinite(val) && val % 1 === 0; },
|
1060
|
+
* pattern: /\d+/
|
1061
|
+
* }
|
1062
|
+
* </pre>
|
1063
|
+
*
|
1064
|
+
* @property {RegExp} pattern The regular expression pattern used to match values of this type when
|
1065
|
+
* coming from a substring of a URL.
|
1066
|
+
*
|
1067
|
+
* @returns {Object} Returns a new `Type` object.
|
1068
|
+
*/
|
1069
|
+
function Type(config) {
|
1070
|
+
extend(this, config);
|
1071
|
+
}
|
1072
|
+
|
1073
|
+
/**
|
1074
|
+
* @ngdoc function
|
1075
|
+
* @name ui.router.util.type:Type#is
|
1076
|
+
* @methodOf ui.router.util.type:Type
|
1077
|
+
*
|
1078
|
+
* @description
|
1079
|
+
* Detects whether a value is of a particular type. Accepts a native (decoded) value
|
1080
|
+
* and determines whether it matches the current `Type` object.
|
1081
|
+
*
|
1082
|
+
* @param {*} val The value to check.
|
1083
|
+
* @param {string} key Optional. If the type check is happening in the context of a specific
|
1084
|
+
* {@link ui.router.util.type:UrlMatcher `UrlMatcher`} object, this is the name of the
|
1085
|
+
* parameter in which `val` is stored. Can be used for meta-programming of `Type` objects.
|
1086
|
+
* @returns {Boolean} Returns `true` if the value matches the type, otherwise `false`.
|
1087
|
+
*/
|
1088
|
+
Type.prototype.is = function(val, key) {
|
1089
|
+
return true;
|
1090
|
+
};
|
1091
|
+
|
1092
|
+
/**
|
1093
|
+
* @ngdoc function
|
1094
|
+
* @name ui.router.util.type:Type#encode
|
1095
|
+
* @methodOf ui.router.util.type:Type
|
1096
|
+
*
|
1097
|
+
* @description
|
1098
|
+
* Encodes a custom/native type value to a string that can be embedded in a URL. Note that the
|
1099
|
+
* return value does *not* need to be URL-safe (i.e. passed through `encodeURIComponent()`), it
|
1100
|
+
* only needs to be a representation of `val` that has been coerced to a string.
|
1101
|
+
*
|
1102
|
+
* @param {*} val The value to encode.
|
1103
|
+
* @param {string} key The name of the parameter in which `val` is stored. Can be used for
|
1104
|
+
* meta-programming of `Type` objects.
|
1105
|
+
* @returns {string} Returns a string representation of `val` that can be encoded in a URL.
|
1106
|
+
*/
|
1107
|
+
Type.prototype.encode = function(val, key) {
|
1108
|
+
return val;
|
1109
|
+
};
|
1110
|
+
|
1111
|
+
/**
|
1112
|
+
* @ngdoc function
|
1113
|
+
* @name ui.router.util.type:Type#decode
|
1114
|
+
* @methodOf ui.router.util.type:Type
|
1115
|
+
*
|
1116
|
+
* @description
|
1117
|
+
* Converts a parameter value (from URL string or transition param) to a custom/native value.
|
1118
|
+
*
|
1119
|
+
* @param {string} val The URL parameter value to decode.
|
1120
|
+
* @param {string} key The name of the parameter in which `val` is stored. Can be used for
|
1121
|
+
* meta-programming of `Type` objects.
|
1122
|
+
* @returns {*} Returns a custom representation of the URL parameter value.
|
1123
|
+
*/
|
1124
|
+
Type.prototype.decode = function(val, key) {
|
1125
|
+
return val;
|
1126
|
+
};
|
1127
|
+
|
1128
|
+
/**
|
1129
|
+
* @ngdoc function
|
1130
|
+
* @name ui.router.util.type:Type#equals
|
1131
|
+
* @methodOf ui.router.util.type:Type
|
1132
|
+
*
|
1133
|
+
* @description
|
1134
|
+
* Determines whether two decoded values are equivalent.
|
1135
|
+
*
|
1136
|
+
* @param {*} a A value to compare against.
|
1137
|
+
* @param {*} b A value to compare against.
|
1138
|
+
* @returns {Boolean} Returns `true` if the values are equivalent/equal, otherwise `false`.
|
1139
|
+
*/
|
1140
|
+
Type.prototype.equals = function(a, b) {
|
1141
|
+
return a == b;
|
1142
|
+
};
|
1143
|
+
|
1144
|
+
Type.prototype.$subPattern = function() {
|
1145
|
+
var sub = this.pattern.toString();
|
1146
|
+
return sub.substr(1, sub.length - 2);
|
1147
|
+
};
|
1148
|
+
|
1149
|
+
Type.prototype.pattern = /.*/;
|
1150
|
+
|
1151
|
+
Type.prototype.toString = function() { return "{Type:" + this.name + "}"; };
|
1152
|
+
|
1153
|
+
/** Given an encoded string, or a decoded object, returns a decoded object */
|
1154
|
+
Type.prototype.$normalize = function(val) {
|
1155
|
+
return this.is(val) ? val : this.decode(val);
|
1156
|
+
};
|
1157
|
+
|
1158
|
+
/*
|
1159
|
+
* Wraps an existing custom Type as an array of Type, depending on 'mode'.
|
1160
|
+
* e.g.:
|
1161
|
+
* - urlmatcher pattern "/path?{queryParam[]:int}"
|
1162
|
+
* - url: "/path?queryParam=1&queryParam=2
|
1163
|
+
* - $stateParams.queryParam will be [1, 2]
|
1164
|
+
* if `mode` is "auto", then
|
1165
|
+
* - url: "/path?queryParam=1 will create $stateParams.queryParam: 1
|
1166
|
+
* - url: "/path?queryParam=1&queryParam=2 will create $stateParams.queryParam: [1, 2]
|
1167
|
+
*/
|
1168
|
+
Type.prototype.$asArray = function(mode, isSearch) {
|
1169
|
+
if (!mode) return this;
|
1170
|
+
if (mode === "auto" && !isSearch) throw new Error("'auto' array mode is for query parameters only");
|
1171
|
+
|
1172
|
+
function ArrayType(type, mode) {
|
1173
|
+
function bindTo(type, callbackName) {
|
1174
|
+
return function() {
|
1175
|
+
return type[callbackName].apply(type, arguments);
|
1176
|
+
};
|
1177
|
+
}
|
1178
|
+
|
1179
|
+
// Wrap non-array value as array
|
1180
|
+
function arrayWrap(val) { return isArray(val) ? val : (isDefined(val) ? [ val ] : []); }
|
1181
|
+
// Unwrap array value for "auto" mode. Return undefined for empty array.
|
1182
|
+
function arrayUnwrap(val) {
|
1183
|
+
switch(val.length) {
|
1184
|
+
case 0: return undefined;
|
1185
|
+
case 1: return mode === "auto" ? val[0] : val;
|
1186
|
+
default: return val;
|
1187
|
+
}
|
1188
|
+
}
|
1189
|
+
function falsey(val) { return !val; }
|
1190
|
+
|
1191
|
+
// Wraps type (.is/.encode/.decode) functions to operate on each value of an array
|
1192
|
+
function arrayHandler(callback, allTruthyMode) {
|
1193
|
+
return function handleArray(val) {
|
1194
|
+
val = arrayWrap(val);
|
1195
|
+
var result = map(val, callback);
|
1196
|
+
if (allTruthyMode === true)
|
1197
|
+
return filter(result, falsey).length === 0;
|
1198
|
+
return arrayUnwrap(result);
|
1199
|
+
};
|
1200
|
+
}
|
1201
|
+
|
1202
|
+
// Wraps type (.equals) functions to operate on each value of an array
|
1203
|
+
function arrayEqualsHandler(callback) {
|
1204
|
+
return function handleArray(val1, val2) {
|
1205
|
+
var left = arrayWrap(val1), right = arrayWrap(val2);
|
1206
|
+
if (left.length !== right.length) return false;
|
1207
|
+
for (var i = 0; i < left.length; i++) {
|
1208
|
+
if (!callback(left[i], right[i])) return false;
|
1209
|
+
}
|
1210
|
+
return true;
|
1211
|
+
};
|
1212
|
+
}
|
1213
|
+
|
1214
|
+
this.encode = arrayHandler(bindTo(type, 'encode'));
|
1215
|
+
this.decode = arrayHandler(bindTo(type, 'decode'));
|
1216
|
+
this.is = arrayHandler(bindTo(type, 'is'), true);
|
1217
|
+
this.equals = arrayEqualsHandler(bindTo(type, 'equals'));
|
1218
|
+
this.pattern = type.pattern;
|
1219
|
+
this.$normalize = arrayHandler(bindTo(type, '$normalize'));
|
1220
|
+
this.name = type.name;
|
1221
|
+
this.$arrayMode = mode;
|
1222
|
+
}
|
1223
|
+
|
1224
|
+
return new ArrayType(this, mode);
|
1225
|
+
};
|
1226
|
+
|
868
1227
|
|
869
1228
|
|
870
1229
|
/**
|
@@ -872,10 +1231,148 @@ UrlMatcher.prototype.format = function (values) {
|
|
872
1231
|
* @name ui.router.util.$urlMatcherFactory
|
873
1232
|
*
|
874
1233
|
* @description
|
875
|
-
* Factory for {@link ui.router.util.type:UrlMatcher} instances. The factory
|
876
|
-
* under the name `$urlMatcherFactoryProvider`.
|
1234
|
+
* Factory for {@link ui.router.util.type:UrlMatcher `UrlMatcher`} instances. The factory
|
1235
|
+
* is also available to providers under the name `$urlMatcherFactoryProvider`.
|
877
1236
|
*/
|
878
1237
|
function $UrlMatcherFactory() {
|
1238
|
+
$$UMFP = this;
|
1239
|
+
|
1240
|
+
var isCaseInsensitive = false, isStrictMode = true, defaultSquashPolicy = false;
|
1241
|
+
|
1242
|
+
function valToString(val) { return val != null ? val.toString().replace(/\//g, "%2F") : val; }
|
1243
|
+
function valFromString(val) { return val != null ? val.toString().replace(/%2F/g, "/") : val; }
|
1244
|
+
|
1245
|
+
var $types = {}, enqueue = true, typeQueue = [], injector, defaultTypes = {
|
1246
|
+
string: {
|
1247
|
+
encode: valToString,
|
1248
|
+
decode: valFromString,
|
1249
|
+
// TODO: in 1.0, make string .is() return false if value is undefined/null by default.
|
1250
|
+
// In 0.2.x, string params are optional by default for backwards compat
|
1251
|
+
is: function(val) { return val == null || !isDefined(val) || typeof val === "string"; },
|
1252
|
+
pattern: /[^/]*/
|
1253
|
+
},
|
1254
|
+
int: {
|
1255
|
+
encode: valToString,
|
1256
|
+
decode: function(val) { return parseInt(val, 10); },
|
1257
|
+
is: function(val) { return isDefined(val) && this.decode(val.toString()) === val; },
|
1258
|
+
pattern: /\d+/
|
1259
|
+
},
|
1260
|
+
bool: {
|
1261
|
+
encode: function(val) { return val ? 1 : 0; },
|
1262
|
+
decode: function(val) { return parseInt(val, 10) !== 0; },
|
1263
|
+
is: function(val) { return val === true || val === false; },
|
1264
|
+
pattern: /0|1/
|
1265
|
+
},
|
1266
|
+
date: {
|
1267
|
+
encode: function (val) {
|
1268
|
+
if (!this.is(val))
|
1269
|
+
return undefined;
|
1270
|
+
return [ val.getFullYear(),
|
1271
|
+
('0' + (val.getMonth() + 1)).slice(-2),
|
1272
|
+
('0' + val.getDate()).slice(-2)
|
1273
|
+
].join("-");
|
1274
|
+
},
|
1275
|
+
decode: function (val) {
|
1276
|
+
if (this.is(val)) return val;
|
1277
|
+
var match = this.capture.exec(val);
|
1278
|
+
return match ? new Date(match[1], match[2] - 1, match[3]) : undefined;
|
1279
|
+
},
|
1280
|
+
is: function(val) { return val instanceof Date && !isNaN(val.valueOf()); },
|
1281
|
+
equals: function (a, b) { return this.is(a) && this.is(b) && a.toISOString() === b.toISOString(); },
|
1282
|
+
pattern: /[0-9]{4}-(?:0[1-9]|1[0-2])-(?:0[1-9]|[1-2][0-9]|3[0-1])/,
|
1283
|
+
capture: /([0-9]{4})-(0[1-9]|1[0-2])-(0[1-9]|[1-2][0-9]|3[0-1])/
|
1284
|
+
},
|
1285
|
+
json: {
|
1286
|
+
encode: angular.toJson,
|
1287
|
+
decode: angular.fromJson,
|
1288
|
+
is: angular.isObject,
|
1289
|
+
equals: angular.equals,
|
1290
|
+
pattern: /[^/]*/
|
1291
|
+
},
|
1292
|
+
any: { // does not encode/decode
|
1293
|
+
encode: angular.identity,
|
1294
|
+
decode: angular.identity,
|
1295
|
+
equals: angular.equals,
|
1296
|
+
pattern: /.*/
|
1297
|
+
}
|
1298
|
+
};
|
1299
|
+
|
1300
|
+
function getDefaultConfig() {
|
1301
|
+
return {
|
1302
|
+
strict: isStrictMode,
|
1303
|
+
caseInsensitive: isCaseInsensitive
|
1304
|
+
};
|
1305
|
+
}
|
1306
|
+
|
1307
|
+
function isInjectable(value) {
|
1308
|
+
return (isFunction(value) || (isArray(value) && isFunction(value[value.length - 1])));
|
1309
|
+
}
|
1310
|
+
|
1311
|
+
/**
|
1312
|
+
* [Internal] Get the default value of a parameter, which may be an injectable function.
|
1313
|
+
*/
|
1314
|
+
$UrlMatcherFactory.$$getDefaultValue = function(config) {
|
1315
|
+
if (!isInjectable(config.value)) return config.value;
|
1316
|
+
if (!injector) throw new Error("Injectable functions cannot be called at configuration time");
|
1317
|
+
return injector.invoke(config.value);
|
1318
|
+
};
|
1319
|
+
|
1320
|
+
/**
|
1321
|
+
* @ngdoc function
|
1322
|
+
* @name ui.router.util.$urlMatcherFactory#caseInsensitive
|
1323
|
+
* @methodOf ui.router.util.$urlMatcherFactory
|
1324
|
+
*
|
1325
|
+
* @description
|
1326
|
+
* Defines whether URL matching should be case sensitive (the default behavior), or not.
|
1327
|
+
*
|
1328
|
+
* @param {boolean} value `false` to match URL in a case sensitive manner; otherwise `true`;
|
1329
|
+
* @returns {boolean} the current value of caseInsensitive
|
1330
|
+
*/
|
1331
|
+
this.caseInsensitive = function(value) {
|
1332
|
+
if (isDefined(value))
|
1333
|
+
isCaseInsensitive = value;
|
1334
|
+
return isCaseInsensitive;
|
1335
|
+
};
|
1336
|
+
|
1337
|
+
/**
|
1338
|
+
* @ngdoc function
|
1339
|
+
* @name ui.router.util.$urlMatcherFactory#strictMode
|
1340
|
+
* @methodOf ui.router.util.$urlMatcherFactory
|
1341
|
+
*
|
1342
|
+
* @description
|
1343
|
+
* Defines whether URLs should match trailing slashes, or not (the default behavior).
|
1344
|
+
*
|
1345
|
+
* @param {boolean=} value `false` to match trailing slashes in URLs, otherwise `true`.
|
1346
|
+
* @returns {boolean} the current value of strictMode
|
1347
|
+
*/
|
1348
|
+
this.strictMode = function(value) {
|
1349
|
+
if (isDefined(value))
|
1350
|
+
isStrictMode = value;
|
1351
|
+
return isStrictMode;
|
1352
|
+
};
|
1353
|
+
|
1354
|
+
/**
|
1355
|
+
* @ngdoc function
|
1356
|
+
* @name ui.router.util.$urlMatcherFactory#defaultSquashPolicy
|
1357
|
+
* @methodOf ui.router.util.$urlMatcherFactory
|
1358
|
+
*
|
1359
|
+
* @description
|
1360
|
+
* Sets the default behavior when generating or matching URLs with default parameter values.
|
1361
|
+
*
|
1362
|
+
* @param {string} value A string that defines the default parameter URL squashing behavior.
|
1363
|
+
* `nosquash`: When generating an href with a default parameter value, do not squash the parameter value from the URL
|
1364
|
+
* `slash`: When generating an href with a default parameter value, squash (remove) the parameter value, and, if the
|
1365
|
+
* parameter is surrounded by slashes, squash (remove) one slash from the URL
|
1366
|
+
* any other string, e.g. "~": When generating an href with a default parameter value, squash (remove)
|
1367
|
+
* the parameter value from the URL and replace it with this string.
|
1368
|
+
*/
|
1369
|
+
this.defaultSquashPolicy = function(value) {
|
1370
|
+
if (!isDefined(value)) return defaultSquashPolicy;
|
1371
|
+
if (value !== true && value !== false && !isString(value))
|
1372
|
+
throw new Error("Invalid squash policy: " + value + ". Valid policies: false, true, arbitrary-string");
|
1373
|
+
defaultSquashPolicy = value;
|
1374
|
+
return value;
|
1375
|
+
};
|
879
1376
|
|
880
1377
|
/**
|
881
1378
|
* @ngdoc function
|
@@ -883,13 +1380,14 @@ function $UrlMatcherFactory() {
|
|
883
1380
|
* @methodOf ui.router.util.$urlMatcherFactory
|
884
1381
|
*
|
885
1382
|
* @description
|
886
|
-
* Creates a {@link ui.router.util.type:UrlMatcher} for the specified pattern.
|
887
|
-
*
|
1383
|
+
* Creates a {@link ui.router.util.type:UrlMatcher `UrlMatcher`} for the specified pattern.
|
1384
|
+
*
|
888
1385
|
* @param {string} pattern The URL pattern.
|
889
|
-
* @
|
1386
|
+
* @param {Object} config The config object hash.
|
1387
|
+
* @returns {UrlMatcher} The UrlMatcher.
|
890
1388
|
*/
|
891
|
-
this.compile = function (pattern) {
|
892
|
-
return new UrlMatcher(pattern);
|
1389
|
+
this.compile = function (pattern, config) {
|
1390
|
+
return new UrlMatcher(pattern, extend(getDefaultConfig(), config));
|
893
1391
|
};
|
894
1392
|
|
895
1393
|
/**
|
@@ -898,29 +1396,335 @@ function $UrlMatcherFactory() {
|
|
898
1396
|
* @methodOf ui.router.util.$urlMatcherFactory
|
899
1397
|
*
|
900
1398
|
* @description
|
901
|
-
* Returns true if the specified object is a UrlMatcher
|
1399
|
+
* Returns true if the specified object is a `UrlMatcher`, or false otherwise.
|
902
1400
|
*
|
903
1401
|
* @param {Object} object The object to perform the type check against.
|
904
|
-
* @returns {Boolean} Returns `true` if the object
|
1402
|
+
* @returns {Boolean} Returns `true` if the object matches the `UrlMatcher` interface, by
|
1403
|
+
* implementing all the same methods.
|
905
1404
|
*/
|
906
1405
|
this.isMatcher = function (o) {
|
907
|
-
|
1406
|
+
if (!isObject(o)) return false;
|
1407
|
+
var result = true;
|
1408
|
+
|
1409
|
+
forEach(UrlMatcher.prototype, function(val, name) {
|
1410
|
+
if (isFunction(val)) {
|
1411
|
+
result = result && (isDefined(o[name]) && isFunction(o[name]));
|
1412
|
+
}
|
1413
|
+
});
|
1414
|
+
return result;
|
908
1415
|
};
|
909
|
-
|
1416
|
+
|
1417
|
+
/**
|
1418
|
+
* @ngdoc function
|
1419
|
+
* @name ui.router.util.$urlMatcherFactory#type
|
1420
|
+
* @methodOf ui.router.util.$urlMatcherFactory
|
1421
|
+
*
|
1422
|
+
* @description
|
1423
|
+
* Registers a custom {@link ui.router.util.type:Type `Type`} object that can be used to
|
1424
|
+
* generate URLs with typed parameters.
|
1425
|
+
*
|
1426
|
+
* @param {string} name The type name.
|
1427
|
+
* @param {Object|Function} definition The type definition. See
|
1428
|
+
* {@link ui.router.util.type:Type `Type`} for information on the values accepted.
|
1429
|
+
* @param {Object|Function} definitionFn (optional) A function that is injected before the app
|
1430
|
+
* runtime starts. The result of this function is merged into the existing `definition`.
|
1431
|
+
* See {@link ui.router.util.type:Type `Type`} for information on the values accepted.
|
1432
|
+
*
|
1433
|
+
* @returns {Object} Returns `$urlMatcherFactoryProvider`.
|
1434
|
+
*
|
1435
|
+
* @example
|
1436
|
+
* This is a simple example of a custom type that encodes and decodes items from an
|
1437
|
+
* array, using the array index as the URL-encoded value:
|
1438
|
+
*
|
1439
|
+
* <pre>
|
1440
|
+
* var list = ['John', 'Paul', 'George', 'Ringo'];
|
1441
|
+
*
|
1442
|
+
* $urlMatcherFactoryProvider.type('listItem', {
|
1443
|
+
* encode: function(item) {
|
1444
|
+
* // Represent the list item in the URL using its corresponding index
|
1445
|
+
* return list.indexOf(item);
|
1446
|
+
* },
|
1447
|
+
* decode: function(item) {
|
1448
|
+
* // Look up the list item by index
|
1449
|
+
* return list[parseInt(item, 10)];
|
1450
|
+
* },
|
1451
|
+
* is: function(item) {
|
1452
|
+
* // Ensure the item is valid by checking to see that it appears
|
1453
|
+
* // in the list
|
1454
|
+
* return list.indexOf(item) > -1;
|
1455
|
+
* }
|
1456
|
+
* });
|
1457
|
+
*
|
1458
|
+
* $stateProvider.state('list', {
|
1459
|
+
* url: "/list/{item:listItem}",
|
1460
|
+
* controller: function($scope, $stateParams) {
|
1461
|
+
* console.log($stateParams.item);
|
1462
|
+
* }
|
1463
|
+
* });
|
1464
|
+
*
|
1465
|
+
* // ...
|
1466
|
+
*
|
1467
|
+
* // Changes URL to '/list/3', logs "Ringo" to the console
|
1468
|
+
* $state.go('list', { item: "Ringo" });
|
1469
|
+
* </pre>
|
1470
|
+
*
|
1471
|
+
* This is a more complex example of a type that relies on dependency injection to
|
1472
|
+
* interact with services, and uses the parameter name from the URL to infer how to
|
1473
|
+
* handle encoding and decoding parameter values:
|
1474
|
+
*
|
1475
|
+
* <pre>
|
1476
|
+
* // Defines a custom type that gets a value from a service,
|
1477
|
+
* // where each service gets different types of values from
|
1478
|
+
* // a backend API:
|
1479
|
+
* $urlMatcherFactoryProvider.type('dbObject', {}, function(Users, Posts) {
|
1480
|
+
*
|
1481
|
+
* // Matches up services to URL parameter names
|
1482
|
+
* var services = {
|
1483
|
+
* user: Users,
|
1484
|
+
* post: Posts
|
1485
|
+
* };
|
1486
|
+
*
|
1487
|
+
* return {
|
1488
|
+
* encode: function(object) {
|
1489
|
+
* // Represent the object in the URL using its unique ID
|
1490
|
+
* return object.id;
|
1491
|
+
* },
|
1492
|
+
* decode: function(value, key) {
|
1493
|
+
* // Look up the object by ID, using the parameter
|
1494
|
+
* // name (key) to call the correct service
|
1495
|
+
* return services[key].findById(value);
|
1496
|
+
* },
|
1497
|
+
* is: function(object, key) {
|
1498
|
+
* // Check that object is a valid dbObject
|
1499
|
+
* return angular.isObject(object) && object.id && services[key];
|
1500
|
+
* }
|
1501
|
+
* equals: function(a, b) {
|
1502
|
+
* // Check the equality of decoded objects by comparing
|
1503
|
+
* // their unique IDs
|
1504
|
+
* return a.id === b.id;
|
1505
|
+
* }
|
1506
|
+
* };
|
1507
|
+
* });
|
1508
|
+
*
|
1509
|
+
* // In a config() block, you can then attach URLs with
|
1510
|
+
* // type-annotated parameters:
|
1511
|
+
* $stateProvider.state('users', {
|
1512
|
+
* url: "/users",
|
1513
|
+
* // ...
|
1514
|
+
* }).state('users.item', {
|
1515
|
+
* url: "/{user:dbObject}",
|
1516
|
+
* controller: function($scope, $stateParams) {
|
1517
|
+
* // $stateParams.user will now be an object returned from
|
1518
|
+
* // the Users service
|
1519
|
+
* },
|
1520
|
+
* // ...
|
1521
|
+
* });
|
1522
|
+
* </pre>
|
1523
|
+
*/
|
1524
|
+
this.type = function (name, definition, definitionFn) {
|
1525
|
+
if (!isDefined(definition)) return $types[name];
|
1526
|
+
if ($types.hasOwnProperty(name)) throw new Error("A type named '" + name + "' has already been defined.");
|
1527
|
+
|
1528
|
+
$types[name] = new Type(extend({ name: name }, definition));
|
1529
|
+
if (definitionFn) {
|
1530
|
+
typeQueue.push({ name: name, def: definitionFn });
|
1531
|
+
if (!enqueue) flushTypeQueue();
|
1532
|
+
}
|
1533
|
+
return this;
|
1534
|
+
};
|
1535
|
+
|
1536
|
+
// `flushTypeQueue()` waits until `$urlMatcherFactory` is injected before invoking the queued `definitionFn`s
|
1537
|
+
function flushTypeQueue() {
|
1538
|
+
while(typeQueue.length) {
|
1539
|
+
var type = typeQueue.shift();
|
1540
|
+
if (type.pattern) throw new Error("You cannot override a type's .pattern at runtime.");
|
1541
|
+
angular.extend($types[type.name], injector.invoke(type.def));
|
1542
|
+
}
|
1543
|
+
}
|
1544
|
+
|
1545
|
+
// Register default types. Store them in the prototype of $types.
|
1546
|
+
forEach(defaultTypes, function(type, name) { $types[name] = new Type(extend({name: name}, type)); });
|
1547
|
+
$types = inherit($types, {});
|
1548
|
+
|
910
1549
|
/* No need to document $get, since it returns this */
|
911
|
-
this.$get = function () {
|
1550
|
+
this.$get = ['$injector', function ($injector) {
|
1551
|
+
injector = $injector;
|
1552
|
+
enqueue = false;
|
1553
|
+
flushTypeQueue();
|
1554
|
+
|
1555
|
+
forEach(defaultTypes, function(type, name) {
|
1556
|
+
if (!$types[name]) $types[name] = new Type(type);
|
1557
|
+
});
|
912
1558
|
return this;
|
1559
|
+
}];
|
1560
|
+
|
1561
|
+
this.Param = function Param(id, type, config, location) {
|
1562
|
+
var self = this;
|
1563
|
+
config = unwrapShorthand(config);
|
1564
|
+
type = getType(config, type, location);
|
1565
|
+
var arrayMode = getArrayMode();
|
1566
|
+
type = arrayMode ? type.$asArray(arrayMode, location === "search") : type;
|
1567
|
+
if (type.name === "string" && !arrayMode && location === "path" && config.value === undefined)
|
1568
|
+
config.value = ""; // for 0.2.x; in 0.3.0+ do not automatically default to ""
|
1569
|
+
var isOptional = config.value !== undefined;
|
1570
|
+
var squash = getSquashPolicy(config, isOptional);
|
1571
|
+
var replace = getReplace(config, arrayMode, isOptional, squash);
|
1572
|
+
|
1573
|
+
function unwrapShorthand(config) {
|
1574
|
+
var keys = isObject(config) ? objectKeys(config) : [];
|
1575
|
+
var isShorthand = indexOf(keys, "value") === -1 && indexOf(keys, "type") === -1 &&
|
1576
|
+
indexOf(keys, "squash") === -1 && indexOf(keys, "array") === -1;
|
1577
|
+
if (isShorthand) config = { value: config };
|
1578
|
+
config.$$fn = isInjectable(config.value) ? config.value : function () { return config.value; };
|
1579
|
+
return config;
|
1580
|
+
}
|
1581
|
+
|
1582
|
+
function getType(config, urlType, location) {
|
1583
|
+
if (config.type && urlType) throw new Error("Param '"+id+"' has two type configurations.");
|
1584
|
+
if (urlType) return urlType;
|
1585
|
+
if (!config.type) return (location === "config" ? $types.any : $types.string);
|
1586
|
+
return config.type instanceof Type ? config.type : new Type(config.type);
|
1587
|
+
}
|
1588
|
+
|
1589
|
+
// array config: param name (param[]) overrides default settings. explicit config overrides param name.
|
1590
|
+
function getArrayMode() {
|
1591
|
+
var arrayDefaults = { array: (location === "search" ? "auto" : false) };
|
1592
|
+
var arrayParamNomenclature = id.match(/\[\]$/) ? { array: true } : {};
|
1593
|
+
return extend(arrayDefaults, arrayParamNomenclature, config).array;
|
1594
|
+
}
|
1595
|
+
|
1596
|
+
/**
|
1597
|
+
* returns false, true, or the squash value to indicate the "default parameter url squash policy".
|
1598
|
+
*/
|
1599
|
+
function getSquashPolicy(config, isOptional) {
|
1600
|
+
var squash = config.squash;
|
1601
|
+
if (!isOptional || squash === false) return false;
|
1602
|
+
if (!isDefined(squash) || squash == null) return defaultSquashPolicy;
|
1603
|
+
if (squash === true || isString(squash)) return squash;
|
1604
|
+
throw new Error("Invalid squash policy: '" + squash + "'. Valid policies: false, true, or arbitrary string");
|
1605
|
+
}
|
1606
|
+
|
1607
|
+
function getReplace(config, arrayMode, isOptional, squash) {
|
1608
|
+
var replace, configuredKeys, defaultPolicy = [
|
1609
|
+
{ from: "", to: (isOptional || arrayMode ? undefined : "") },
|
1610
|
+
{ from: null, to: (isOptional || arrayMode ? undefined : "") }
|
1611
|
+
];
|
1612
|
+
replace = isArray(config.replace) ? config.replace : [];
|
1613
|
+
if (isString(squash))
|
1614
|
+
replace.push({ from: squash, to: undefined });
|
1615
|
+
configuredKeys = map(replace, function(item) { return item.from; } );
|
1616
|
+
return filter(defaultPolicy, function(item) { return indexOf(configuredKeys, item.from) === -1; }).concat(replace);
|
1617
|
+
}
|
1618
|
+
|
1619
|
+
/**
|
1620
|
+
* [Internal] Get the default value of a parameter, which may be an injectable function.
|
1621
|
+
*/
|
1622
|
+
function $$getDefaultValue() {
|
1623
|
+
if (!injector) throw new Error("Injectable functions cannot be called at configuration time");
|
1624
|
+
var defaultValue = injector.invoke(config.$$fn);
|
1625
|
+
if (defaultValue !== null && defaultValue !== undefined && !self.type.is(defaultValue))
|
1626
|
+
throw new Error("Default value (" + defaultValue + ") for parameter '" + self.id + "' is not an instance of Type (" + self.type.name + ")");
|
1627
|
+
return defaultValue;
|
1628
|
+
}
|
1629
|
+
|
1630
|
+
/**
|
1631
|
+
* [Internal] Gets the decoded representation of a value if the value is defined, otherwise, returns the
|
1632
|
+
* default value, which may be the result of an injectable function.
|
1633
|
+
*/
|
1634
|
+
function $value(value) {
|
1635
|
+
function hasReplaceVal(val) { return function(obj) { return obj.from === val; }; }
|
1636
|
+
function $replace(value) {
|
1637
|
+
var replacement = map(filter(self.replace, hasReplaceVal(value)), function(obj) { return obj.to; });
|
1638
|
+
return replacement.length ? replacement[0] : value;
|
1639
|
+
}
|
1640
|
+
value = $replace(value);
|
1641
|
+
return !isDefined(value) ? $$getDefaultValue() : self.type.$normalize(value);
|
1642
|
+
}
|
1643
|
+
|
1644
|
+
function toString() { return "{Param:" + id + " " + type + " squash: '" + squash + "' optional: " + isOptional + "}"; }
|
1645
|
+
|
1646
|
+
extend(this, {
|
1647
|
+
id: id,
|
1648
|
+
type: type,
|
1649
|
+
location: location,
|
1650
|
+
array: arrayMode,
|
1651
|
+
squash: squash,
|
1652
|
+
replace: replace,
|
1653
|
+
isOptional: isOptional,
|
1654
|
+
value: $value,
|
1655
|
+
dynamic: undefined,
|
1656
|
+
config: config,
|
1657
|
+
toString: toString
|
1658
|
+
});
|
913
1659
|
};
|
1660
|
+
|
1661
|
+
function ParamSet(params) {
|
1662
|
+
extend(this, params || {});
|
1663
|
+
}
|
1664
|
+
|
1665
|
+
ParamSet.prototype = {
|
1666
|
+
$$new: function() {
|
1667
|
+
return inherit(this, extend(new ParamSet(), { $$parent: this}));
|
1668
|
+
},
|
1669
|
+
$$keys: function () {
|
1670
|
+
var keys = [], chain = [], parent = this,
|
1671
|
+
ignore = objectKeys(ParamSet.prototype);
|
1672
|
+
while (parent) { chain.push(parent); parent = parent.$$parent; }
|
1673
|
+
chain.reverse();
|
1674
|
+
forEach(chain, function(paramset) {
|
1675
|
+
forEach(objectKeys(paramset), function(key) {
|
1676
|
+
if (indexOf(keys, key) === -1 && indexOf(ignore, key) === -1) keys.push(key);
|
1677
|
+
});
|
1678
|
+
});
|
1679
|
+
return keys;
|
1680
|
+
},
|
1681
|
+
$$values: function(paramValues) {
|
1682
|
+
var values = {}, self = this;
|
1683
|
+
forEach(self.$$keys(), function(key) {
|
1684
|
+
values[key] = self[key].value(paramValues && paramValues[key]);
|
1685
|
+
});
|
1686
|
+
return values;
|
1687
|
+
},
|
1688
|
+
$$equals: function(paramValues1, paramValues2) {
|
1689
|
+
var equal = true, self = this;
|
1690
|
+
forEach(self.$$keys(), function(key) {
|
1691
|
+
var left = paramValues1 && paramValues1[key], right = paramValues2 && paramValues2[key];
|
1692
|
+
if (!self[key].type.equals(left, right)) equal = false;
|
1693
|
+
});
|
1694
|
+
return equal;
|
1695
|
+
},
|
1696
|
+
$$validates: function $$validate(paramValues) {
|
1697
|
+
var keys = this.$$keys(), i, param, rawVal, normalized, encoded;
|
1698
|
+
for (i = 0; i < keys.length; i++) {
|
1699
|
+
param = this[keys[i]];
|
1700
|
+
rawVal = paramValues[keys[i]];
|
1701
|
+
if ((rawVal === undefined || rawVal === null) && param.isOptional)
|
1702
|
+
break; // There was no parameter value, but the param is optional
|
1703
|
+
normalized = param.type.$normalize(rawVal);
|
1704
|
+
if (!param.type.is(normalized))
|
1705
|
+
return false; // The value was not of the correct Type, and could not be decoded to the correct Type
|
1706
|
+
encoded = param.type.encode(normalized);
|
1707
|
+
if (angular.isString(encoded) && !param.type.pattern.exec(encoded))
|
1708
|
+
return false; // The value was of the correct type, but when encoded, did not match the Type's regexp
|
1709
|
+
}
|
1710
|
+
return true;
|
1711
|
+
},
|
1712
|
+
$$parent: undefined
|
1713
|
+
};
|
1714
|
+
|
1715
|
+
this.ParamSet = ParamSet;
|
914
1716
|
}
|
915
1717
|
|
916
1718
|
// Register as a provider so it's available to other providers
|
917
1719
|
angular.module('ui.router.util').provider('$urlMatcherFactory', $UrlMatcherFactory);
|
1720
|
+
angular.module('ui.router.util').run(['$urlMatcherFactory', function($urlMatcherFactory) { }]);
|
918
1721
|
|
919
1722
|
/**
|
920
1723
|
* @ngdoc object
|
921
1724
|
* @name ui.router.router.$urlRouterProvider
|
922
1725
|
*
|
923
1726
|
* @requires ui.router.util.$urlMatcherFactoryProvider
|
1727
|
+
* @requires $locationProvider
|
924
1728
|
*
|
925
1729
|
* @description
|
926
1730
|
* `$urlRouterProvider` has the responsibility of watching `$location`.
|
@@ -931,10 +1735,9 @@ angular.module('ui.router.util').provider('$urlMatcherFactory', $UrlMatcherFacto
|
|
931
1735
|
* There are several methods on `$urlRouterProvider` that make it useful to use directly
|
932
1736
|
* in your module config.
|
933
1737
|
*/
|
934
|
-
$UrlRouterProvider.$inject = ['$urlMatcherFactoryProvider'];
|
935
|
-
function $UrlRouterProvider(
|
936
|
-
var rules = [],
|
937
|
-
otherwise = null;
|
1738
|
+
$UrlRouterProvider.$inject = ['$locationProvider', '$urlMatcherFactoryProvider'];
|
1739
|
+
function $UrlRouterProvider( $locationProvider, $urlMatcherFactory) {
|
1740
|
+
var rules = [], otherwise = null, interceptDeferred = false, listener;
|
938
1741
|
|
939
1742
|
// Returns a string that is a prefix of all strings matching the RegExp
|
940
1743
|
function regExpPrefix(re) {
|
@@ -955,7 +1758,7 @@ function $UrlRouterProvider( $urlMatcherFactory) {
|
|
955
1758
|
* @methodOf ui.router.router.$urlRouterProvider
|
956
1759
|
*
|
957
1760
|
* @description
|
958
|
-
* Defines rules that are used by `$urlRouterProvider to find matches for
|
1761
|
+
* Defines rules that are used by `$urlRouterProvider` to find matches for
|
959
1762
|
* specific URLs.
|
960
1763
|
*
|
961
1764
|
* @example
|
@@ -978,14 +1781,13 @@ function $UrlRouterProvider( $urlMatcherFactory) {
|
|
978
1781
|
* @param {object} rule Handler function that takes `$injector` and `$location`
|
979
1782
|
* services as arguments. You can use them to return a valid path as a string.
|
980
1783
|
*
|
981
|
-
* @return {object}
|
1784
|
+
* @return {object} `$urlRouterProvider` - `$urlRouterProvider` instance
|
982
1785
|
*/
|
983
|
-
this.rule =
|
984
|
-
|
985
|
-
|
986
|
-
|
987
|
-
|
988
|
-
};
|
1786
|
+
this.rule = function (rule) {
|
1787
|
+
if (!isFunction(rule)) throw new Error("'rule' must be a function");
|
1788
|
+
rules.push(rule);
|
1789
|
+
return this;
|
1790
|
+
};
|
989
1791
|
|
990
1792
|
/**
|
991
1793
|
* @ngdoc object
|
@@ -993,7 +1795,7 @@ function $UrlRouterProvider( $urlMatcherFactory) {
|
|
993
1795
|
* @methodOf ui.router.router.$urlRouterProvider
|
994
1796
|
*
|
995
1797
|
* @description
|
996
|
-
* Defines a path that is used when an
|
1798
|
+
* Defines a path that is used when an invalid route is requested.
|
997
1799
|
*
|
998
1800
|
* @example
|
999
1801
|
* <pre>
|
@@ -1007,27 +1809,26 @@ function $UrlRouterProvider( $urlMatcherFactory) {
|
|
1007
1809
|
*
|
1008
1810
|
* // Example of using function rule as param
|
1009
1811
|
* $urlRouterProvider.otherwise(function ($injector, $location) {
|
1010
|
-
*
|
1812
|
+
* return '/a/valid/url';
|
1011
1813
|
* });
|
1012
1814
|
* });
|
1013
1815
|
* </pre>
|
1014
1816
|
*
|
1015
1817
|
* @param {string|object} rule The url path you want to redirect to or a function
|
1016
1818
|
* rule that returns the url path. The function version is passed two params:
|
1017
|
-
* `$injector` and `$location` services.
|
1819
|
+
* `$injector` and `$location` services, and must return a url string.
|
1018
1820
|
*
|
1019
|
-
* @return {object}
|
1821
|
+
* @return {object} `$urlRouterProvider` - `$urlRouterProvider` instance
|
1020
1822
|
*/
|
1021
|
-
this.otherwise =
|
1022
|
-
|
1023
|
-
|
1024
|
-
|
1025
|
-
|
1026
|
-
|
1027
|
-
|
1028
|
-
|
1029
|
-
|
1030
|
-
};
|
1823
|
+
this.otherwise = function (rule) {
|
1824
|
+
if (isString(rule)) {
|
1825
|
+
var redirect = rule;
|
1826
|
+
rule = function () { return redirect; };
|
1827
|
+
}
|
1828
|
+
else if (!isFunction(rule)) throw new Error("'rule' must be a function");
|
1829
|
+
otherwise = rule;
|
1830
|
+
return this;
|
1831
|
+
};
|
1031
1832
|
|
1032
1833
|
|
1033
1834
|
function handleIfMatch($injector, handler, match) {
|
@@ -1043,8 +1844,8 @@ function $UrlRouterProvider( $urlMatcherFactory) {
|
|
1043
1844
|
*
|
1044
1845
|
* @description
|
1045
1846
|
* Registers a handler for a given url matching. if handle is a string, it is
|
1046
|
-
* treated as a redirect, and is interpolated according to the
|
1047
|
-
* (i.e. like String.replace() for RegExp
|
1847
|
+
* treated as a redirect, and is interpolated according to the syntax of match
|
1848
|
+
* (i.e. like `String.replace()` for `RegExp`, or like a `UrlMatcher` pattern otherwise).
|
1048
1849
|
*
|
1049
1850
|
* If the handler is a function, it is injectable. It gets invoked if `$location`
|
1050
1851
|
* matches. You have the option of inject the match object as `$match`.
|
@@ -1073,52 +1874,102 @@ function $UrlRouterProvider( $urlMatcherFactory) {
|
|
1073
1874
|
* @param {string|object} what The incoming path that you want to redirect.
|
1074
1875
|
* @param {string|object} handler The path you want to redirect your user to.
|
1075
1876
|
*/
|
1076
|
-
this.when =
|
1077
|
-
|
1078
|
-
|
1079
|
-
|
1080
|
-
|
1081
|
-
|
1082
|
-
|
1083
|
-
|
1084
|
-
|
1085
|
-
|
1086
|
-
|
1087
|
-
|
1088
|
-
handler = ['$match', function ($match) { return redirect.format($match); }];
|
1089
|
-
}
|
1090
|
-
return extend(function ($injector, $location) {
|
1091
|
-
return handleIfMatch($injector, handler, what.exec($location.path(), $location.search()));
|
1092
|
-
}, {
|
1093
|
-
prefix: isString(what.prefix) ? what.prefix : ''
|
1094
|
-
});
|
1095
|
-
},
|
1096
|
-
regex: function (what, handler) {
|
1097
|
-
if (what.global || what.sticky) throw new Error("when() RegExp must not be global or sticky");
|
1098
|
-
|
1099
|
-
if (handlerIsString) {
|
1100
|
-
redirect = handler;
|
1101
|
-
handler = ['$match', function ($match) { return interpolate(redirect, $match); }];
|
1102
|
-
}
|
1103
|
-
return extend(function ($injector, $location) {
|
1104
|
-
return handleIfMatch($injector, handler, what.exec($location.path()));
|
1105
|
-
}, {
|
1106
|
-
prefix: regExpPrefix(what)
|
1107
|
-
});
|
1877
|
+
this.when = function (what, handler) {
|
1878
|
+
var redirect, handlerIsString = isString(handler);
|
1879
|
+
if (isString(what)) what = $urlMatcherFactory.compile(what);
|
1880
|
+
|
1881
|
+
if (!handlerIsString && !isFunction(handler) && !isArray(handler))
|
1882
|
+
throw new Error("invalid 'handler' in when()");
|
1883
|
+
|
1884
|
+
var strategies = {
|
1885
|
+
matcher: function (what, handler) {
|
1886
|
+
if (handlerIsString) {
|
1887
|
+
redirect = $urlMatcherFactory.compile(handler);
|
1888
|
+
handler = ['$match', function ($match) { return redirect.format($match); }];
|
1108
1889
|
}
|
1109
|
-
|
1110
|
-
|
1111
|
-
|
1890
|
+
return extend(function ($injector, $location) {
|
1891
|
+
return handleIfMatch($injector, handler, what.exec($location.path(), $location.search()));
|
1892
|
+
}, {
|
1893
|
+
prefix: isString(what.prefix) ? what.prefix : ''
|
1894
|
+
});
|
1895
|
+
},
|
1896
|
+
regex: function (what, handler) {
|
1897
|
+
if (what.global || what.sticky) throw new Error("when() RegExp must not be global or sticky");
|
1112
1898
|
|
1113
|
-
|
1114
|
-
|
1115
|
-
return
|
1899
|
+
if (handlerIsString) {
|
1900
|
+
redirect = handler;
|
1901
|
+
handler = ['$match', function ($match) { return interpolate(redirect, $match); }];
|
1116
1902
|
}
|
1903
|
+
return extend(function ($injector, $location) {
|
1904
|
+
return handleIfMatch($injector, handler, what.exec($location.path()));
|
1905
|
+
}, {
|
1906
|
+
prefix: regExpPrefix(what)
|
1907
|
+
});
|
1117
1908
|
}
|
1118
|
-
|
1119
|
-
throw new Error("invalid 'what' in when()");
|
1120
1909
|
};
|
1121
1910
|
|
1911
|
+
var check = { matcher: $urlMatcherFactory.isMatcher(what), regex: what instanceof RegExp };
|
1912
|
+
|
1913
|
+
for (var n in check) {
|
1914
|
+
if (check[n]) return this.rule(strategies[n](what, handler));
|
1915
|
+
}
|
1916
|
+
|
1917
|
+
throw new Error("invalid 'what' in when()");
|
1918
|
+
};
|
1919
|
+
|
1920
|
+
/**
|
1921
|
+
* @ngdoc function
|
1922
|
+
* @name ui.router.router.$urlRouterProvider#deferIntercept
|
1923
|
+
* @methodOf ui.router.router.$urlRouterProvider
|
1924
|
+
*
|
1925
|
+
* @description
|
1926
|
+
* Disables (or enables) deferring location change interception.
|
1927
|
+
*
|
1928
|
+
* If you wish to customize the behavior of syncing the URL (for example, if you wish to
|
1929
|
+
* defer a transition but maintain the current URL), call this method at configuration time.
|
1930
|
+
* Then, at run time, call `$urlRouter.listen()` after you have configured your own
|
1931
|
+
* `$locationChangeSuccess` event handler.
|
1932
|
+
*
|
1933
|
+
* @example
|
1934
|
+
* <pre>
|
1935
|
+
* var app = angular.module('app', ['ui.router.router']);
|
1936
|
+
*
|
1937
|
+
* app.config(function ($urlRouterProvider) {
|
1938
|
+
*
|
1939
|
+
* // Prevent $urlRouter from automatically intercepting URL changes;
|
1940
|
+
* // this allows you to configure custom behavior in between
|
1941
|
+
* // location changes and route synchronization:
|
1942
|
+
* $urlRouterProvider.deferIntercept();
|
1943
|
+
*
|
1944
|
+
* }).run(function ($rootScope, $urlRouter, UserService) {
|
1945
|
+
*
|
1946
|
+
* $rootScope.$on('$locationChangeSuccess', function(e) {
|
1947
|
+
* // UserService is an example service for managing user state
|
1948
|
+
* if (UserService.isLoggedIn()) return;
|
1949
|
+
*
|
1950
|
+
* // Prevent $urlRouter's default handler from firing
|
1951
|
+
* e.preventDefault();
|
1952
|
+
*
|
1953
|
+
* UserService.handleLogin().then(function() {
|
1954
|
+
* // Once the user has logged in, sync the current URL
|
1955
|
+
* // to the router:
|
1956
|
+
* $urlRouter.sync();
|
1957
|
+
* });
|
1958
|
+
* });
|
1959
|
+
*
|
1960
|
+
* // Configures $urlRouter's listener *after* your custom listener
|
1961
|
+
* $urlRouter.listen();
|
1962
|
+
* });
|
1963
|
+
* </pre>
|
1964
|
+
*
|
1965
|
+
* @param {boolean} defer Indicates whether to defer location change interception. Passing
|
1966
|
+
no parameter is equivalent to `true`.
|
1967
|
+
*/
|
1968
|
+
this.deferIntercept = function (defer) {
|
1969
|
+
if (defer === undefined) defer = true;
|
1970
|
+
interceptDeferred = defer;
|
1971
|
+
};
|
1972
|
+
|
1122
1973
|
/**
|
1123
1974
|
* @ngdoc object
|
1124
1975
|
* @name ui.router.router.$urlRouter
|
@@ -1126,66 +1977,172 @@ function $UrlRouterProvider( $urlMatcherFactory) {
|
|
1126
1977
|
* @requires $location
|
1127
1978
|
* @requires $rootScope
|
1128
1979
|
* @requires $injector
|
1980
|
+
* @requires $browser
|
1129
1981
|
*
|
1130
1982
|
* @description
|
1131
1983
|
*
|
1132
1984
|
*/
|
1133
|
-
this.$get =
|
1134
|
-
|
1135
|
-
|
1136
|
-
|
1137
|
-
|
1138
|
-
|
1139
|
-
|
1140
|
-
|
1141
|
-
|
1142
|
-
|
1143
|
-
|
1144
|
-
|
1145
|
-
|
1985
|
+
this.$get = $get;
|
1986
|
+
$get.$inject = ['$location', '$rootScope', '$injector', '$browser'];
|
1987
|
+
function $get( $location, $rootScope, $injector, $browser) {
|
1988
|
+
|
1989
|
+
var baseHref = $browser.baseHref(), location = $location.url(), lastPushedUrl;
|
1990
|
+
|
1991
|
+
function appendBasePath(url, isHtml5, absolute) {
|
1992
|
+
if (baseHref === '/') return url;
|
1993
|
+
if (isHtml5) return baseHref.slice(0, -1) + url;
|
1994
|
+
if (absolute) return baseHref.slice(1) + url;
|
1995
|
+
return url;
|
1996
|
+
}
|
1997
|
+
|
1998
|
+
// TODO: Optimize groups of rules with non-empty prefix into some sort of decision tree
|
1999
|
+
function update(evt) {
|
2000
|
+
if (evt && evt.defaultPrevented) return;
|
2001
|
+
var ignoreUpdate = lastPushedUrl && $location.url() === lastPushedUrl;
|
2002
|
+
lastPushedUrl = undefined;
|
2003
|
+
// TODO: Re-implement this in 1.0 for https://github.com/angular-ui/ui-router/issues/1573
|
2004
|
+
//if (ignoreUpdate) return true;
|
2005
|
+
|
2006
|
+
function check(rule) {
|
2007
|
+
var handled = rule($injector, $location);
|
2008
|
+
|
2009
|
+
if (!handled) return false;
|
2010
|
+
if (isString(handled)) $location.replace().url(handled);
|
2011
|
+
return true;
|
2012
|
+
}
|
2013
|
+
var n = rules.length, i;
|
2014
|
+
|
2015
|
+
for (i = 0; i < n; i++) {
|
2016
|
+
if (check(rules[i])) return;
|
2017
|
+
}
|
2018
|
+
// always check otherwise last to allow dynamic updates to the set of rules
|
2019
|
+
if (otherwise) check(otherwise);
|
2020
|
+
}
|
2021
|
+
|
2022
|
+
function listen() {
|
2023
|
+
listener = listener || $rootScope.$on('$locationChangeSuccess', update);
|
2024
|
+
return listener;
|
2025
|
+
}
|
2026
|
+
|
2027
|
+
if (!interceptDeferred) listen();
|
2028
|
+
|
2029
|
+
return {
|
2030
|
+
/**
|
2031
|
+
* @ngdoc function
|
2032
|
+
* @name ui.router.router.$urlRouter#sync
|
2033
|
+
* @methodOf ui.router.router.$urlRouter
|
2034
|
+
*
|
2035
|
+
* @description
|
2036
|
+
* Triggers an update; the same update that happens when the address bar url changes, aka `$locationChangeSuccess`.
|
2037
|
+
* This method is useful when you need to use `preventDefault()` on the `$locationChangeSuccess` event,
|
2038
|
+
* perform some custom logic (route protection, auth, config, redirection, etc) and then finally proceed
|
2039
|
+
* with the transition by calling `$urlRouter.sync()`.
|
2040
|
+
*
|
2041
|
+
* @example
|
2042
|
+
* <pre>
|
2043
|
+
* angular.module('app', ['ui.router'])
|
2044
|
+
* .run(function($rootScope, $urlRouter) {
|
2045
|
+
* $rootScope.$on('$locationChangeSuccess', function(evt) {
|
2046
|
+
* // Halt state change from even starting
|
2047
|
+
* evt.preventDefault();
|
2048
|
+
* // Perform custom logic
|
2049
|
+
* var meetsRequirement = ...
|
2050
|
+
* // Continue with the update and state transition if logic allows
|
2051
|
+
* if (meetsRequirement) $urlRouter.sync();
|
2052
|
+
* });
|
2053
|
+
* });
|
2054
|
+
* </pre>
|
2055
|
+
*/
|
2056
|
+
sync: function() {
|
2057
|
+
update();
|
2058
|
+
},
|
2059
|
+
|
2060
|
+
listen: function() {
|
2061
|
+
return listen();
|
2062
|
+
},
|
2063
|
+
|
2064
|
+
update: function(read) {
|
2065
|
+
if (read) {
|
2066
|
+
location = $location.url();
|
2067
|
+
return;
|
1146
2068
|
}
|
1147
|
-
|
1148
|
-
|
1149
|
-
|
2069
|
+
if ($location.url() === location) return;
|
2070
|
+
|
2071
|
+
$location.url(location);
|
2072
|
+
$location.replace();
|
2073
|
+
},
|
2074
|
+
|
2075
|
+
push: function(urlMatcher, params, options) {
|
2076
|
+
var url = urlMatcher.format(params || {});
|
2077
|
+
|
2078
|
+
// Handle the special hash param, if needed
|
2079
|
+
if (url !== null && params && params['#']) {
|
2080
|
+
url += '#' + params['#'];
|
1150
2081
|
}
|
1151
|
-
// always check otherwise last to allow dynamic updates to the set of rules
|
1152
|
-
if (otherwise) check(otherwise);
|
1153
|
-
}
|
1154
2082
|
|
1155
|
-
|
2083
|
+
$location.url(url);
|
2084
|
+
lastPushedUrl = options && options.$$avoidResync ? $location.url() : undefined;
|
2085
|
+
if (options && options.replace) $location.replace();
|
2086
|
+
},
|
1156
2087
|
|
1157
|
-
|
1158
|
-
|
1159
|
-
|
1160
|
-
|
1161
|
-
|
1162
|
-
|
1163
|
-
|
1164
|
-
|
1165
|
-
|
1166
|
-
|
1167
|
-
|
1168
|
-
|
1169
|
-
|
1170
|
-
|
1171
|
-
|
1172
|
-
|
1173
|
-
|
1174
|
-
|
1175
|
-
|
1176
|
-
|
1177
|
-
|
1178
|
-
|
1179
|
-
|
1180
|
-
|
1181
|
-
|
1182
|
-
|
1183
|
-
|
1184
|
-
|
1185
|
-
|
2088
|
+
/**
|
2089
|
+
* @ngdoc function
|
2090
|
+
* @name ui.router.router.$urlRouter#href
|
2091
|
+
* @methodOf ui.router.router.$urlRouter
|
2092
|
+
*
|
2093
|
+
* @description
|
2094
|
+
* A URL generation method that returns the compiled URL for a given
|
2095
|
+
* {@link ui.router.util.type:UrlMatcher `UrlMatcher`}, populated with the provided parameters.
|
2096
|
+
*
|
2097
|
+
* @example
|
2098
|
+
* <pre>
|
2099
|
+
* $bob = $urlRouter.href(new UrlMatcher("/about/:person"), {
|
2100
|
+
* person: "bob"
|
2101
|
+
* });
|
2102
|
+
* // $bob == "/about/bob";
|
2103
|
+
* </pre>
|
2104
|
+
*
|
2105
|
+
* @param {UrlMatcher} urlMatcher The `UrlMatcher` object which is used as the template of the URL to generate.
|
2106
|
+
* @param {object=} params An object of parameter values to fill the matcher's required parameters.
|
2107
|
+
* @param {object=} options Options object. The options are:
|
2108
|
+
*
|
2109
|
+
* - **`absolute`** - {boolean=false}, If true will generate an absolute url, e.g. "http://www.example.com/fullurl".
|
2110
|
+
*
|
2111
|
+
* @returns {string} Returns the fully compiled URL, or `null` if `params` fail validation against `urlMatcher`
|
2112
|
+
*/
|
2113
|
+
href: function(urlMatcher, params, options) {
|
2114
|
+
if (!urlMatcher.validates(params)) return null;
|
2115
|
+
|
2116
|
+
var isHtml5 = $locationProvider.html5Mode();
|
2117
|
+
if (angular.isObject(isHtml5)) {
|
2118
|
+
isHtml5 = isHtml5.enabled;
|
1186
2119
|
}
|
1187
|
-
|
1188
|
-
|
2120
|
+
|
2121
|
+
var url = urlMatcher.format(params);
|
2122
|
+
options = options || {};
|
2123
|
+
|
2124
|
+
if (!isHtml5 && url !== null) {
|
2125
|
+
url = "#" + $locationProvider.hashPrefix() + url;
|
2126
|
+
}
|
2127
|
+
|
2128
|
+
// Handle special hash param, if needed
|
2129
|
+
if (url !== null && params && params['#']) {
|
2130
|
+
url += '#' + params['#'];
|
2131
|
+
}
|
2132
|
+
|
2133
|
+
url = appendBasePath(url, isHtml5, options.absolute);
|
2134
|
+
|
2135
|
+
if (!options.absolute || !url) {
|
2136
|
+
return url;
|
2137
|
+
}
|
2138
|
+
|
2139
|
+
var slash = (!isHtml5 && url ? '/' : ''), port = $location.port();
|
2140
|
+
port = (port === 80 || port === 443 ? '' : ':' + port);
|
2141
|
+
|
2142
|
+
return [$location.protocol(), '://', $location.host(), port, slash, url].join('');
|
2143
|
+
}
|
2144
|
+
};
|
2145
|
+
}
|
1189
2146
|
}
|
1190
2147
|
|
1191
2148
|
angular.module('ui.router.router').provider('$urlRouter', $UrlRouterProvider);
|
@@ -1196,7 +2153,6 @@ angular.module('ui.router.router').provider('$urlRouter', $UrlRouterProvider);
|
|
1196
2153
|
*
|
1197
2154
|
* @requires ui.router.router.$urlRouterProvider
|
1198
2155
|
* @requires ui.router.util.$urlMatcherFactoryProvider
|
1199
|
-
* @requires $locationProvider
|
1200
2156
|
*
|
1201
2157
|
* @description
|
1202
2158
|
* The new `$stateProvider` works similar to Angular's v1 router, but it focuses purely
|
@@ -1212,8 +2168,8 @@ angular.module('ui.router.router').provider('$urlRouter', $UrlRouterProvider);
|
|
1212
2168
|
*
|
1213
2169
|
* The `$stateProvider` provides interfaces to declare these states for your app.
|
1214
2170
|
*/
|
1215
|
-
$StateProvider.$inject = ['$urlRouterProvider', '$urlMatcherFactoryProvider'
|
1216
|
-
function $StateProvider( $urlRouterProvider, $urlMatcherFactory
|
2171
|
+
$StateProvider.$inject = ['$urlRouterProvider', '$urlMatcherFactoryProvider'];
|
2172
|
+
function $StateProvider( $urlRouterProvider, $urlMatcherFactory) {
|
1217
2173
|
|
1218
2174
|
var root, states = {}, $state, queue = {}, abstractKey = 'abstract';
|
1219
2175
|
|
@@ -1241,18 +2197,14 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1241
2197
|
|
1242
2198
|
// Build a URLMatcher if necessary, either via a relative or absolute URL
|
1243
2199
|
url: function(state) {
|
1244
|
-
var url = state.url;
|
2200
|
+
var url = state.url, config = { params: state.params || {} };
|
1245
2201
|
|
1246
2202
|
if (isString(url)) {
|
1247
|
-
if (url.charAt(0) == '^')
|
1248
|
-
|
1249
|
-
}
|
1250
|
-
return (state.parent.navigable || root).url.concat(url);
|
2203
|
+
if (url.charAt(0) == '^') return $urlMatcherFactory.compile(url.substring(1), config);
|
2204
|
+
return (state.parent.navigable || root).url.concat(url, config);
|
1251
2205
|
}
|
1252
2206
|
|
1253
|
-
if ($urlMatcherFactory.isMatcher(url)
|
1254
|
-
return url;
|
1255
|
-
}
|
2207
|
+
if (!url || $urlMatcherFactory.isMatcher(url)) return url;
|
1256
2208
|
throw new Error("Invalid url '" + url + "' in state '" + state + "'");
|
1257
2209
|
},
|
1258
2210
|
|
@@ -1261,14 +2213,18 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1261
2213
|
return state.url ? state : (state.parent ? state.parent.navigable : null);
|
1262
2214
|
},
|
1263
2215
|
|
2216
|
+
// Own parameters for this state. state.url.params is already built at this point. Create and add non-url params
|
2217
|
+
ownParams: function(state) {
|
2218
|
+
var params = state.url && state.url.params || new $$UMFP.ParamSet();
|
2219
|
+
forEach(state.params || {}, function(config, id) {
|
2220
|
+
if (!params[id]) params[id] = new $$UMFP.Param(id, null, config, "config");
|
2221
|
+
});
|
2222
|
+
return params;
|
2223
|
+
},
|
2224
|
+
|
1264
2225
|
// Derive parameters for this state and ensure they're a super-set of parent's parameters
|
1265
2226
|
params: function(state) {
|
1266
|
-
|
1267
|
-
return state.url ? state.url.parameters() : state.parent.params;
|
1268
|
-
}
|
1269
|
-
if (!isArray(state.params)) throw new Error("Invalid params in state '" + state + "'");
|
1270
|
-
if (state.url) throw new Error("Both params and url specicified in state '" + state + "'");
|
1271
|
-
return state.params;
|
2227
|
+
return state.parent && state.parent.params ? extend(state.parent.params.$$new(), state.ownParams) : new $$UMFP.ParamSet();
|
1272
2228
|
},
|
1273
2229
|
|
1274
2230
|
// If there is no explicit multi-view configuration, make one up so we don't have
|
@@ -1286,26 +2242,6 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1286
2242
|
return views;
|
1287
2243
|
},
|
1288
2244
|
|
1289
|
-
ownParams: function(state) {
|
1290
|
-
if (!state.parent) {
|
1291
|
-
return state.params;
|
1292
|
-
}
|
1293
|
-
var paramNames = {}; forEach(state.params, function (p) { paramNames[p] = true; });
|
1294
|
-
|
1295
|
-
forEach(state.parent.params, function (p) {
|
1296
|
-
if (!paramNames[p]) {
|
1297
|
-
throw new Error("Missing required parameter '" + p + "' in state '" + state.name + "'");
|
1298
|
-
}
|
1299
|
-
paramNames[p] = false;
|
1300
|
-
});
|
1301
|
-
var ownParams = [];
|
1302
|
-
|
1303
|
-
forEach(paramNames, function (own, p) {
|
1304
|
-
if (own) ownParams.push(p);
|
1305
|
-
});
|
1306
|
-
return ownParams;
|
1307
|
-
},
|
1308
|
-
|
1309
2245
|
// Keep a full path from the root down to this state as this is needed for state activation.
|
1310
2246
|
path: function(state) {
|
1311
2247
|
return state.parent ? state.parent.path.concat(state) : []; // exclude root from path
|
@@ -1326,12 +2262,16 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1326
2262
|
}
|
1327
2263
|
|
1328
2264
|
function findState(stateOrName, base) {
|
2265
|
+
if (!stateOrName) return undefined;
|
2266
|
+
|
1329
2267
|
var isStr = isString(stateOrName),
|
1330
2268
|
name = isStr ? stateOrName : stateOrName.name,
|
1331
2269
|
path = isRelative(name);
|
1332
2270
|
|
1333
2271
|
if (path) {
|
1334
2272
|
if (!base) throw new Error("No reference point given for path '" + name + "'");
|
2273
|
+
base = findState(base);
|
2274
|
+
|
1335
2275
|
var rel = name.split("."), i = 0, pathLength = rel.length, current = base;
|
1336
2276
|
|
1337
2277
|
for (; i < pathLength; i++) {
|
@@ -1364,6 +2304,13 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1364
2304
|
queue[parentName].push(state);
|
1365
2305
|
}
|
1366
2306
|
|
2307
|
+
function flushQueuedChildren(parentName) {
|
2308
|
+
var queued = queue[parentName] || [];
|
2309
|
+
while(queued.length) {
|
2310
|
+
registerState(queued.shift());
|
2311
|
+
}
|
2312
|
+
}
|
2313
|
+
|
1367
2314
|
function registerState(state) {
|
1368
2315
|
// Wrap a new object around the state so we can store our private details easily.
|
1369
2316
|
state = inherit(state, {
|
@@ -1379,6 +2326,7 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1379
2326
|
// Get parent name
|
1380
2327
|
var parentName = (name.indexOf('.') !== -1) ? name.substring(0, name.lastIndexOf('.'))
|
1381
2328
|
: (isString(state.parent)) ? state.parent
|
2329
|
+
: (isObject(state.parent) && isString(state.parent.name)) ? state.parent.name
|
1382
2330
|
: '';
|
1383
2331
|
|
1384
2332
|
// If parent is not registered yet, add state to queue and register later
|
@@ -1395,17 +2343,13 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1395
2343
|
if (!state[abstractKey] && state.url) {
|
1396
2344
|
$urlRouterProvider.when(state.url, ['$match', '$stateParams', function ($match, $stateParams) {
|
1397
2345
|
if ($state.$current.navigable != state || !equalForKeys($match, $stateParams)) {
|
1398
|
-
$state.transitionTo(state, $match, { location: false });
|
2346
|
+
$state.transitionTo(state, $match, { inherit: true, location: false });
|
1399
2347
|
}
|
1400
2348
|
}]);
|
1401
2349
|
}
|
1402
2350
|
|
1403
2351
|
// Register any queued children
|
1404
|
-
|
1405
|
-
for (var i = 0; i < queue[name].length; i++) {
|
1406
|
-
registerState(queue[name][i]);
|
1407
|
-
}
|
1408
|
-
}
|
2352
|
+
flushQueuedChildren(name);
|
1409
2353
|
|
1410
2354
|
return state;
|
1411
2355
|
}
|
@@ -1420,14 +2364,21 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1420
2364
|
var globSegments = glob.split('.'),
|
1421
2365
|
segments = $state.$current.name.split('.');
|
1422
2366
|
|
2367
|
+
//match single stars
|
2368
|
+
for (var i = 0, l = globSegments.length; i < l; i++) {
|
2369
|
+
if (globSegments[i] === '*') {
|
2370
|
+
segments[i] = '*';
|
2371
|
+
}
|
2372
|
+
}
|
2373
|
+
|
1423
2374
|
//match greedy starts
|
1424
2375
|
if (globSegments[0] === '**') {
|
1425
|
-
segments = segments.slice(
|
2376
|
+
segments = segments.slice(indexOf(segments, globSegments[1]));
|
1426
2377
|
segments.unshift('**');
|
1427
2378
|
}
|
1428
2379
|
//match greedy ends
|
1429
2380
|
if (globSegments[globSegments.length - 1] === '**') {
|
1430
|
-
segments.splice(
|
2381
|
+
segments.splice(indexOf(segments, globSegments[globSegments.length - 2]) + 1, Number.MAX_VALUE);
|
1431
2382
|
segments.push('**');
|
1432
2383
|
}
|
1433
2384
|
|
@@ -1435,13 +2386,6 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1435
2386
|
return false;
|
1436
2387
|
}
|
1437
2388
|
|
1438
|
-
//match single stars
|
1439
|
-
for (var i = 0, l = globSegments.length; i < l; i++) {
|
1440
|
-
if (globSegments[i] === '*') {
|
1441
|
-
segments[i] = '*';
|
1442
|
-
}
|
1443
|
-
}
|
1444
|
-
|
1445
2389
|
return segments.join('') === globSegments.join('');
|
1446
2390
|
}
|
1447
2391
|
|
@@ -1489,7 +2433,8 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1489
2433
|
* - **parent** `{object}` - returns the parent state object.
|
1490
2434
|
* - **data** `{object}` - returns state data, including any inherited data that is not
|
1491
2435
|
* overridden by own values (if any).
|
1492
|
-
* - **url** `{object}` - returns a {link ui.router.util.type:UrlMatcher}
|
2436
|
+
* - **url** `{object}` - returns a {@link ui.router.util.type:UrlMatcher UrlMatcher}
|
2437
|
+
* or `null`.
|
1493
2438
|
* - **navigable** `{object}` - returns closest ancestor state that has a URL (aka is
|
1494
2439
|
* navigable).
|
1495
2440
|
* - **params** `{object}` - returns an array of state params that are ensured to
|
@@ -1505,17 +2450,17 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1505
2450
|
* - **path** `{string}` - returns the full path from the root down to this state.
|
1506
2451
|
* Needed for state activation.
|
1507
2452
|
* - **includes** `{object}` - returns an object that includes every state that
|
1508
|
-
* would pass a
|
2453
|
+
* would pass a `$state.includes()` test.
|
1509
2454
|
*
|
1510
2455
|
* @example
|
1511
2456
|
* <pre>
|
1512
2457
|
* // Override the internal 'views' builder with a function that takes the state
|
1513
2458
|
* // definition, and a reference to the internal function being overridden:
|
1514
|
-
* $stateProvider.decorator('views', function (
|
2459
|
+
* $stateProvider.decorator('views', function (state, parent) {
|
1515
2460
|
* var result = {},
|
1516
2461
|
* views = parent(state);
|
1517
2462
|
*
|
1518
|
-
* angular.forEach(
|
2463
|
+
* angular.forEach(views, function (config, name) {
|
1519
2464
|
* var autoName = (state.name + '.' + name).replace('.', '/');
|
1520
2465
|
* config.templateUrl = config.templateUrl || '/partials/' + autoName + '.html';
|
1521
2466
|
* result[name] = config;
|
@@ -1571,9 +2516,12 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1571
2516
|
* Registers a state configuration under a given state name. The stateConfig object
|
1572
2517
|
* has the following acceptable properties.
|
1573
2518
|
*
|
2519
|
+
* @param {string} name A unique state name, e.g. "home", "about", "contacts".
|
2520
|
+
* To create a parent/child state use a dot, e.g. "about.sales", "home.newest".
|
2521
|
+
* @param {object} stateConfig State configuration object.
|
2522
|
+
* @param {string|function=} stateConfig.template
|
1574
2523
|
* <a id='template'></a>
|
1575
|
-
*
|
1576
|
-
* - **`template`** - {string|function=} - html template as a string or a function that returns
|
2524
|
+
* html template as a string or a function that returns
|
1577
2525
|
* an html template as a string which should be used by the uiView directives. This property
|
1578
2526
|
* takes precedence over templateUrl.
|
1579
2527
|
*
|
@@ -1582,9 +2530,17 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1582
2530
|
* - {array.<object>} - state parameters extracted from the current $location.path() by
|
1583
2531
|
* applying the current state
|
1584
2532
|
*
|
2533
|
+
* <pre>template:
|
2534
|
+
* "<h1>inline template definition</h1>" +
|
2535
|
+
* "<div ui-view></div>"</pre>
|
2536
|
+
* <pre>template: function(params) {
|
2537
|
+
* return "<h1>generated template</h1>"; }</pre>
|
2538
|
+
* </div>
|
2539
|
+
*
|
2540
|
+
* @param {string|function=} stateConfig.templateUrl
|
1585
2541
|
* <a id='templateUrl'></a>
|
1586
2542
|
*
|
1587
|
-
*
|
2543
|
+
* path or function that returns a path to an html
|
1588
2544
|
* template that should be used by uiView.
|
1589
2545
|
*
|
1590
2546
|
* If `templateUrl` is a function, it will be called with the following parameters:
|
@@ -1592,82 +2548,261 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1592
2548
|
* - {array.<object>} - state parameters extracted from the current $location.path() by
|
1593
2549
|
* applying the current state
|
1594
2550
|
*
|
1595
|
-
* <
|
2551
|
+
* <pre>templateUrl: "home.html"</pre>
|
2552
|
+
* <pre>templateUrl: function(params) {
|
2553
|
+
* return myTemplates[params.pageId]; }</pre>
|
1596
2554
|
*
|
1597
|
-
*
|
1598
|
-
*
|
2555
|
+
* @param {function=} stateConfig.templateProvider
|
2556
|
+
* <a id='templateProvider'></a>
|
2557
|
+
* Provider function that returns HTML content string.
|
2558
|
+
* <pre> templateProvider:
|
2559
|
+
* function(MyTemplateService, params) {
|
2560
|
+
* return MyTemplateService.getTemplate(params.pageId);
|
2561
|
+
* }</pre>
|
1599
2562
|
*
|
2563
|
+
* @param {string|function=} stateConfig.controller
|
1600
2564
|
* <a id='controller'></a>
|
1601
2565
|
*
|
1602
|
-
*
|
2566
|
+
* Controller fn that should be associated with newly
|
1603
2567
|
* related scope or the name of a registered controller if passed as a string.
|
2568
|
+
* Optionally, the ControllerAs may be declared here.
|
2569
|
+
* <pre>controller: "MyRegisteredController"</pre>
|
2570
|
+
* <pre>controller:
|
2571
|
+
* "MyRegisteredController as fooCtrl"}</pre>
|
2572
|
+
* <pre>controller: function($scope, MyService) {
|
2573
|
+
* $scope.data = MyService.getData(); }</pre>
|
1604
2574
|
*
|
2575
|
+
* @param {function=} stateConfig.controllerProvider
|
1605
2576
|
* <a id='controllerProvider'></a>
|
1606
2577
|
*
|
1607
|
-
*
|
1608
|
-
*
|
2578
|
+
* Injectable provider function that returns the actual controller or string.
|
2579
|
+
* <pre>controllerProvider:
|
2580
|
+
* function(MyResolveData) {
|
2581
|
+
* if (MyResolveData.foo)
|
2582
|
+
* return "FooCtrl"
|
2583
|
+
* else if (MyResolveData.bar)
|
2584
|
+
* return "BarCtrl";
|
2585
|
+
* else return function($scope) {
|
2586
|
+
* $scope.baz = "Qux";
|
2587
|
+
* }
|
2588
|
+
* }</pre>
|
1609
2589
|
*
|
2590
|
+
* @param {string=} stateConfig.controllerAs
|
1610
2591
|
* <a id='controllerAs'></a>
|
1611
2592
|
*
|
1612
|
-
*
|
2593
|
+
* A controller alias name. If present the controller will be
|
1613
2594
|
* published to scope under the controllerAs name.
|
2595
|
+
* <pre>controllerAs: "myCtrl"</pre>
|
2596
|
+
*
|
2597
|
+
* @param {string|object=} stateConfig.parent
|
2598
|
+
* <a id='parent'></a>
|
2599
|
+
* Optionally specifies the parent state of this state.
|
1614
2600
|
*
|
2601
|
+
* <pre>parent: 'parentState'</pre>
|
2602
|
+
* <pre>parent: parentState // JS variable</pre>
|
2603
|
+
*
|
2604
|
+
* @param {object=} stateConfig.resolve
|
1615
2605
|
* <a id='resolve'></a>
|
1616
2606
|
*
|
1617
|
-
*
|
2607
|
+
* An optional map<string, function> of dependencies which
|
1618
2608
|
* should be injected into the controller. If any of these dependencies are promises,
|
1619
|
-
* the router will wait for them all to be resolved
|
1620
|
-
*
|
1621
|
-
* of the resolved promises are injected
|
1622
|
-
* of the promises are rejected the $stateChangeError event is fired.
|
2609
|
+
* the router will wait for them all to be resolved before the controller is instantiated.
|
2610
|
+
* If all the promises are resolved successfully, the $stateChangeSuccess event is fired
|
2611
|
+
* and the values of the resolved promises are injected into any controllers that reference them.
|
2612
|
+
* If any of the promises are rejected the $stateChangeError event is fired.
|
2613
|
+
*
|
2614
|
+
* The map object is:
|
1623
2615
|
*
|
1624
2616
|
* - key - {string}: name of dependency to be injected into controller
|
1625
2617
|
* - factory - {string|function}: If string then it is alias for service. Otherwise if function,
|
1626
2618
|
* it is injected and return value it treated as dependency. If result is a promise, it is
|
1627
2619
|
* resolved before its value is injected into controller.
|
1628
2620
|
*
|
1629
|
-
* <
|
2621
|
+
* <pre>resolve: {
|
2622
|
+
* myResolve1:
|
2623
|
+
* function($http, $stateParams) {
|
2624
|
+
* return $http.get("/api/foos/"+stateParams.fooID);
|
2625
|
+
* }
|
2626
|
+
* }</pre>
|
2627
|
+
*
|
2628
|
+
* @param {string=} stateConfig.url
|
2629
|
+
* <a id='url'></a>
|
2630
|
+
*
|
2631
|
+
* A url fragment with optional parameters. When a state is navigated or
|
2632
|
+
* transitioned to, the `$stateParams` service will be populated with any
|
2633
|
+
* parameters that were passed.
|
2634
|
+
*
|
2635
|
+
* (See {@link ui.router.util.type:UrlMatcher UrlMatcher} `UrlMatcher`} for
|
2636
|
+
* more details on acceptable patterns )
|
2637
|
+
*
|
2638
|
+
* examples:
|
2639
|
+
* <pre>url: "/home"
|
2640
|
+
* url: "/users/:userid"
|
2641
|
+
* url: "/books/{bookid:[a-zA-Z_-]}"
|
2642
|
+
* url: "/books/{categoryid:int}"
|
2643
|
+
* url: "/books/{publishername:string}/{categoryid:int}"
|
2644
|
+
* url: "/messages?before&after"
|
2645
|
+
* url: "/messages?{before:date}&{after:date}"
|
2646
|
+
* url: "/messages/:mailboxid?{before:date}&{after:date}"
|
2647
|
+
* </pre>
|
2648
|
+
*
|
2649
|
+
* @param {object=} stateConfig.views
|
2650
|
+
* <a id='views'></a>
|
2651
|
+
* an optional map<string, object> which defined multiple views, or targets views
|
2652
|
+
* manually/explicitly.
|
2653
|
+
*
|
2654
|
+
* Examples:
|
2655
|
+
*
|
2656
|
+
* Targets three named `ui-view`s in the parent state's template
|
2657
|
+
* <pre>views: {
|
2658
|
+
* header: {
|
2659
|
+
* controller: "headerCtrl",
|
2660
|
+
* templateUrl: "header.html"
|
2661
|
+
* }, body: {
|
2662
|
+
* controller: "bodyCtrl",
|
2663
|
+
* templateUrl: "body.html"
|
2664
|
+
* }, footer: {
|
2665
|
+
* controller: "footCtrl",
|
2666
|
+
* templateUrl: "footer.html"
|
2667
|
+
* }
|
2668
|
+
* }</pre>
|
2669
|
+
*
|
2670
|
+
* Targets named `ui-view="header"` from grandparent state 'top''s template, and named `ui-view="body" from parent state's template.
|
2671
|
+
* <pre>views: {
|
2672
|
+
* 'header@top': {
|
2673
|
+
* controller: "msgHeaderCtrl",
|
2674
|
+
* templateUrl: "msgHeader.html"
|
2675
|
+
* }, 'body': {
|
2676
|
+
* controller: "messagesCtrl",
|
2677
|
+
* templateUrl: "messages.html"
|
2678
|
+
* }
|
2679
|
+
* }</pre>
|
2680
|
+
*
|
2681
|
+
* @param {boolean=} [stateConfig.abstract=false]
|
2682
|
+
* <a id='abstract'></a>
|
2683
|
+
* An abstract state will never be directly activated,
|
2684
|
+
* but can provide inherited properties to its common children states.
|
2685
|
+
* <pre>abstract: true</pre>
|
2686
|
+
*
|
2687
|
+
* @param {function=} stateConfig.onEnter
|
2688
|
+
* <a id='onEnter'></a>
|
2689
|
+
*
|
2690
|
+
* Callback function for when a state is entered. Good way
|
2691
|
+
* to trigger an action or dispatch an event, such as opening a dialog.
|
2692
|
+
* If minifying your scripts, make sure to explictly annotate this function,
|
2693
|
+
* because it won't be automatically annotated by your build tools.
|
2694
|
+
*
|
2695
|
+
* <pre>onEnter: function(MyService, $stateParams) {
|
2696
|
+
* MyService.foo($stateParams.myParam);
|
2697
|
+
* }</pre>
|
2698
|
+
*
|
2699
|
+
* @param {function=} stateConfig.onExit
|
2700
|
+
* <a id='onExit'></a>
|
2701
|
+
*
|
2702
|
+
* Callback function for when a state is exited. Good way to
|
2703
|
+
* trigger an action or dispatch an event, such as opening a dialog.
|
2704
|
+
* If minifying your scripts, make sure to explictly annotate this function,
|
2705
|
+
* because it won't be automatically annotated by your build tools.
|
2706
|
+
*
|
2707
|
+
* <pre>onExit: function(MyService, $stateParams) {
|
2708
|
+
* MyService.cleanup($stateParams.myParam);
|
2709
|
+
* }</pre>
|
2710
|
+
*
|
2711
|
+
* @param {boolean=} [stateConfig.reloadOnSearch=true]
|
2712
|
+
* <a id='reloadOnSearch'></a>
|
2713
|
+
*
|
2714
|
+
* If `false`, will not retrigger the same state
|
2715
|
+
* just because a search/query parameter has changed (via $location.search() or $location.hash()).
|
2716
|
+
* Useful for when you'd like to modify $location.search() without triggering a reload.
|
2717
|
+
* <pre>reloadOnSearch: false</pre>
|
2718
|
+
*
|
2719
|
+
* @param {object=} stateConfig.data
|
2720
|
+
* <a id='data'></a>
|
2721
|
+
*
|
2722
|
+
* Arbitrary data object, useful for custom configuration. The parent state's `data` is
|
2723
|
+
* prototypally inherited. In other words, adding a data property to a state adds it to
|
2724
|
+
* the entire subtree via prototypal inheritance.
|
1630
2725
|
*
|
1631
|
-
*
|
1632
|
-
*
|
1633
|
-
*
|
2726
|
+
* <pre>data: {
|
2727
|
+
* requiredRole: 'foo'
|
2728
|
+
* } </pre>
|
1634
2729
|
*
|
2730
|
+
* @param {object=} stateConfig.params
|
1635
2731
|
* <a id='params'></a>
|
1636
2732
|
*
|
1637
|
-
*
|
1638
|
-
*
|
1639
|
-
*
|
1640
|
-
* $stateParams service will be populated with any parameters that were passed.
|
2733
|
+
* A map which optionally configures parameters declared in the `url`, or
|
2734
|
+
* defines additional non-url parameters. For each parameter being
|
2735
|
+
* configured, add a configuration object keyed to the name of the parameter.
|
1641
2736
|
*
|
1642
|
-
*
|
2737
|
+
* Each parameter configuration object may contain the following properties:
|
1643
2738
|
*
|
1644
|
-
*
|
1645
|
-
*
|
2739
|
+
* - ** value ** - {object|function=}: specifies the default value for this
|
2740
|
+
* parameter. This implicitly sets this parameter as optional.
|
1646
2741
|
*
|
1647
|
-
*
|
2742
|
+
* When UI-Router routes to a state and no value is
|
2743
|
+
* specified for this parameter in the URL or transition, the
|
2744
|
+
* default value will be used instead. If `value` is a function,
|
2745
|
+
* it will be injected and invoked, and the return value used.
|
1648
2746
|
*
|
1649
|
-
*
|
1650
|
-
*
|
2747
|
+
* *Note*: `undefined` is treated as "no default value" while `null`
|
2748
|
+
* is treated as "the default value is `null`".
|
1651
2749
|
*
|
1652
|
-
*
|
2750
|
+
* *Shorthand*: If you only need to configure the default value of the
|
2751
|
+
* parameter, you may use a shorthand syntax. In the **`params`**
|
2752
|
+
* map, instead mapping the param name to a full parameter configuration
|
2753
|
+
* object, simply set map it to the default parameter value, e.g.:
|
1653
2754
|
*
|
1654
|
-
*
|
1655
|
-
*
|
2755
|
+
* <pre>// define a parameter's default value
|
2756
|
+
* params: {
|
2757
|
+
* param1: { value: "defaultValue" }
|
2758
|
+
* }
|
2759
|
+
* // shorthand default values
|
2760
|
+
* params: {
|
2761
|
+
* param1: "defaultValue",
|
2762
|
+
* param2: "param2Default"
|
2763
|
+
* }</pre>
|
1656
2764
|
*
|
1657
|
-
*
|
2765
|
+
* - ** array ** - {boolean=}: *(default: false)* If true, the param value will be
|
2766
|
+
* treated as an array of values. If you specified a Type, the value will be
|
2767
|
+
* treated as an array of the specified Type. Note: query parameter values
|
2768
|
+
* default to a special `"auto"` mode.
|
1658
2769
|
*
|
1659
|
-
*
|
1660
|
-
*
|
2770
|
+
* For query parameters in `"auto"` mode, if multiple values for a single parameter
|
2771
|
+
* are present in the URL (e.g.: `/foo?bar=1&bar=2&bar=3`) then the values
|
2772
|
+
* are mapped to an array (e.g.: `{ foo: [ '1', '2', '3' ] }`). However, if
|
2773
|
+
* only one value is present (e.g.: `/foo?bar=1`) then the value is treated as single
|
2774
|
+
* value (e.g.: `{ foo: '1' }`).
|
1661
2775
|
*
|
1662
|
-
* <
|
2776
|
+
* <pre>params: {
|
2777
|
+
* param1: { array: true }
|
2778
|
+
* }</pre>
|
1663
2779
|
*
|
1664
|
-
*
|
1665
|
-
*
|
1666
|
-
*
|
2780
|
+
* - ** squash ** - {bool|string=}: `squash` configures how a default parameter value is represented in the URL when
|
2781
|
+
* the current parameter value is the same as the default value. If `squash` is not set, it uses the
|
2782
|
+
* configured default squash policy.
|
2783
|
+
* (See {@link ui.router.util.$urlMatcherFactory#methods_defaultSquashPolicy `defaultSquashPolicy()`})
|
1667
2784
|
*
|
1668
|
-
*
|
2785
|
+
* There are three squash settings:
|
2786
|
+
*
|
2787
|
+
* - false: The parameter's default value is not squashed. It is encoded and included in the URL
|
2788
|
+
* - true: The parameter's default value is omitted from the URL. If the parameter is preceeded and followed
|
2789
|
+
* by slashes in the state's `url` declaration, then one of those slashes are omitted.
|
2790
|
+
* This can allow for cleaner looking URLs.
|
2791
|
+
* - `"<arbitrary string>"`: The parameter's default value is replaced with an arbitrary placeholder of your choice.
|
2792
|
+
*
|
2793
|
+
* <pre>params: {
|
2794
|
+
* param1: {
|
2795
|
+
* value: "defaultId",
|
2796
|
+
* squash: true
|
2797
|
+
* } }
|
2798
|
+
* // squash "defaultValue" to "~"
|
2799
|
+
* params: {
|
2800
|
+
* param1: {
|
2801
|
+
* value: "defaultValue",
|
2802
|
+
* squash: "~"
|
2803
|
+
* } }
|
2804
|
+
* </pre>
|
1669
2805
|
*
|
1670
|
-
* - **`data`** - {object=} - Arbitrary data object, useful for custom configuration.
|
1671
2806
|
*
|
1672
2807
|
* @example
|
1673
2808
|
* <pre>
|
@@ -1676,7 +2811,7 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1676
2811
|
* // stateName can be a single top-level name (must be unique).
|
1677
2812
|
* $stateProvider.state("home", {});
|
1678
2813
|
*
|
1679
|
-
* // Or it can be a nested state name. This state is a child of the
|
2814
|
+
* // Or it can be a nested state name. This state is a child of the
|
1680
2815
|
* // above "home" state.
|
1681
2816
|
* $stateProvider.state("home.newest", {});
|
1682
2817
|
*
|
@@ -1690,9 +2825,6 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1690
2825
|
* .state("contacts", {});
|
1691
2826
|
* </pre>
|
1692
2827
|
*
|
1693
|
-
* @param {string} name A unique state name, e.g. "home", "about", "contacts".
|
1694
|
-
* To create a parent/child state use a dot, e.g. "about.sales", "home.newest".
|
1695
|
-
* @param {object} definition State configuration object.
|
1696
2828
|
*/
|
1697
2829
|
this.state = state;
|
1698
2830
|
function state(name, definition) {
|
@@ -1713,6 +2845,7 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1713
2845
|
* @requires $injector
|
1714
2846
|
* @requires ui.router.util.$resolve
|
1715
2847
|
* @requires ui.router.state.$stateParams
|
2848
|
+
* @requires ui.router.router.$urlRouter
|
1716
2849
|
*
|
1717
2850
|
* @property {object} params A param object, e.g. {sectionId: section.id)}, that
|
1718
2851
|
* you'd like to test against the current active state.
|
@@ -1726,26 +2859,82 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1726
2859
|
* between them. It also provides interfaces to ask for current state or even states
|
1727
2860
|
* you're coming from.
|
1728
2861
|
*/
|
1729
|
-
// $urlRouter is injected just to ensure it gets instantiated
|
1730
2862
|
this.$get = $get;
|
1731
|
-
$get.$inject = ['$rootScope', '$q', '$view', '$injector', '$resolve', '$stateParams', '$
|
1732
|
-
function $get( $rootScope, $q, $view, $injector, $resolve, $stateParams, $
|
2863
|
+
$get.$inject = ['$rootScope', '$q', '$view', '$injector', '$resolve', '$stateParams', '$urlRouter', '$location', '$urlMatcherFactory'];
|
2864
|
+
function $get( $rootScope, $q, $view, $injector, $resolve, $stateParams, $urlRouter, $location, $urlMatcherFactory) {
|
1733
2865
|
|
1734
2866
|
var TransitionSuperseded = $q.reject(new Error('transition superseded'));
|
1735
2867
|
var TransitionPrevented = $q.reject(new Error('transition prevented'));
|
1736
2868
|
var TransitionAborted = $q.reject(new Error('transition aborted'));
|
1737
2869
|
var TransitionFailed = $q.reject(new Error('transition failed'));
|
1738
|
-
var currentLocation = $location.url();
|
1739
|
-
var baseHref = $browser.baseHref();
|
1740
2870
|
|
1741
|
-
|
1742
|
-
|
1743
|
-
|
1744
|
-
|
2871
|
+
// Handles the case where a state which is the target of a transition is not found, and the user
|
2872
|
+
// can optionally retry or defer the transition
|
2873
|
+
function handleRedirect(redirect, state, params, options) {
|
2874
|
+
/**
|
2875
|
+
* @ngdoc event
|
2876
|
+
* @name ui.router.state.$state#$stateNotFound
|
2877
|
+
* @eventOf ui.router.state.$state
|
2878
|
+
* @eventType broadcast on root scope
|
2879
|
+
* @description
|
2880
|
+
* Fired when a requested state **cannot be found** using the provided state name during transition.
|
2881
|
+
* The event is broadcast allowing any handlers a single chance to deal with the error (usually by
|
2882
|
+
* lazy-loading the unfound state). A special `unfoundState` object is passed to the listener handler,
|
2883
|
+
* you can see its three properties in the example. You can use `event.preventDefault()` to abort the
|
2884
|
+
* transition and the promise returned from `go` will be rejected with a `'transition aborted'` value.
|
2885
|
+
*
|
2886
|
+
* @param {Object} event Event object.
|
2887
|
+
* @param {Object} unfoundState Unfound State information. Contains: `to, toParams, options` properties.
|
2888
|
+
* @param {State} fromState Current state object.
|
2889
|
+
* @param {Object} fromParams Current state params.
|
2890
|
+
*
|
2891
|
+
* @example
|
2892
|
+
*
|
2893
|
+
* <pre>
|
2894
|
+
* // somewhere, assume lazy.state has not been defined
|
2895
|
+
* $state.go("lazy.state", {a:1, b:2}, {inherit:false});
|
2896
|
+
*
|
2897
|
+
* // somewhere else
|
2898
|
+
* $scope.$on('$stateNotFound',
|
2899
|
+
* function(event, unfoundState, fromState, fromParams){
|
2900
|
+
* console.log(unfoundState.to); // "lazy.state"
|
2901
|
+
* console.log(unfoundState.toParams); // {a:1, b:2}
|
2902
|
+
* console.log(unfoundState.options); // {inherit:false} + default options
|
2903
|
+
* })
|
2904
|
+
* </pre>
|
2905
|
+
*/
|
2906
|
+
var evt = $rootScope.$broadcast('$stateNotFound', redirect, state, params);
|
2907
|
+
|
2908
|
+
if (evt.defaultPrevented) {
|
2909
|
+
$urlRouter.update();
|
2910
|
+
return TransitionAborted;
|
2911
|
+
}
|
2912
|
+
|
2913
|
+
if (!evt.retry) {
|
2914
|
+
return null;
|
2915
|
+
}
|
2916
|
+
|
2917
|
+
// Allow the handler to return a promise to defer state lookup retry
|
2918
|
+
if (options.$retry) {
|
2919
|
+
$urlRouter.update();
|
2920
|
+
return TransitionFailed;
|
1745
2921
|
}
|
2922
|
+
var retryTransition = $state.transition = $q.when(evt.retry);
|
2923
|
+
|
2924
|
+
retryTransition.then(function() {
|
2925
|
+
if (retryTransition !== $state.transition) return TransitionSuperseded;
|
2926
|
+
redirect.options.$retry = true;
|
2927
|
+
return $state.transitionTo(redirect.to, redirect.toParams, redirect.options);
|
2928
|
+
}, function() {
|
2929
|
+
return TransitionAborted;
|
2930
|
+
});
|
2931
|
+
$urlRouter.update();
|
2932
|
+
|
2933
|
+
return retryTransition;
|
1746
2934
|
}
|
1747
2935
|
|
1748
2936
|
root.locals = { resolve: null, globals: { $stateParams: {} } };
|
2937
|
+
|
1749
2938
|
$state = {
|
1750
2939
|
params: {},
|
1751
2940
|
current: root.self,
|
@@ -1759,8 +2948,8 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1759
2948
|
* @methodOf ui.router.state.$state
|
1760
2949
|
*
|
1761
2950
|
* @description
|
1762
|
-
* A method that force reloads the current state. All resolves are re-resolved,
|
1763
|
-
*
|
2951
|
+
* A method that force reloads the current state. All resolves are re-resolved,
|
2952
|
+
* controllers reinstantiated, and events re-fired.
|
1764
2953
|
*
|
1765
2954
|
* @example
|
1766
2955
|
* <pre>
|
@@ -1776,12 +2965,37 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1776
2965
|
* `reload()` is just an alias for:
|
1777
2966
|
* <pre>
|
1778
2967
|
* $state.transitionTo($state.current, $stateParams, {
|
1779
|
-
* reload: true, inherit: false, notify:
|
2968
|
+
* reload: true, inherit: false, notify: true
|
2969
|
+
* });
|
2970
|
+
* </pre>
|
2971
|
+
*
|
2972
|
+
* @param {string=|object=} state - A state name or a state object, which is the root of the resolves to be re-resolved.
|
2973
|
+
* @example
|
2974
|
+
* <pre>
|
2975
|
+
* //assuming app application consists of 3 states: 'contacts', 'contacts.detail', 'contacts.detail.item'
|
2976
|
+
* //and current state is 'contacts.detail.item'
|
2977
|
+
* var app angular.module('app', ['ui.router']);
|
2978
|
+
*
|
2979
|
+
* app.controller('ctrl', function ($scope, $state) {
|
2980
|
+
* $scope.reload = function(){
|
2981
|
+
* //will reload 'contact.detail' and 'contact.detail.item' states
|
2982
|
+
* $state.reload('contact.detail');
|
2983
|
+
* }
|
2984
|
+
* });
|
2985
|
+
* </pre>
|
2986
|
+
*
|
2987
|
+
* `reload()` is just an alias for:
|
2988
|
+
* <pre>
|
2989
|
+
* $state.transitionTo($state.current, $stateParams, {
|
2990
|
+
* reload: true, inherit: false, notify: true
|
1780
2991
|
* });
|
1781
2992
|
* </pre>
|
2993
|
+
|
2994
|
+
* @returns {promise} A promise representing the state of the new transition. See
|
2995
|
+
* {@link ui.router.state.$state#methods_go $state.go}.
|
1782
2996
|
*/
|
1783
|
-
$state.reload = function reload() {
|
1784
|
-
$state.transitionTo($state.current, $stateParams, { reload: true, inherit: false, notify:
|
2997
|
+
$state.reload = function reload(state) {
|
2998
|
+
return $state.transitionTo($state.current, $stateParams, { reload: state || true, inherit: false, notify: true});
|
1785
2999
|
};
|
1786
3000
|
|
1787
3001
|
/**
|
@@ -1851,7 +3065,7 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1851
3065
|
*
|
1852
3066
|
*/
|
1853
3067
|
$state.go = function go(to, params, options) {
|
1854
|
-
return
|
3068
|
+
return $state.transitionTo(to, params, extend({ inherit: true, relative: $state.$current }, options));
|
1855
3069
|
};
|
1856
3070
|
|
1857
3071
|
/**
|
@@ -1885,9 +3099,11 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1885
3099
|
* - **`relative`** - {object=}, When transitioning with relative path (e.g '^'),
|
1886
3100
|
* defines which state to be relative from.
|
1887
3101
|
* - **`notify`** - {boolean=true}, If `true` will broadcast $stateChangeStart and $stateChangeSuccess events.
|
1888
|
-
* - **`reload`** (v0.2.5) - {boolean=false}, If `true` will force transition even if the state or params
|
3102
|
+
* - **`reload`** (v0.2.5) - {boolean=false|string=|object=}, If `true` will force transition even if the state or params
|
1889
3103
|
* have not changed, aka a reload of the same state. It differs from reloadOnSearch because you'd
|
1890
3104
|
* use this when you want to force a reload when *everything* is the same, including search params.
|
3105
|
+
* if String, then will reload the state with the name given in reload, and any children.
|
3106
|
+
* if Object, then a stateObj is expected, will reload the state found in stateObj, and any children.
|
1891
3107
|
*
|
1892
3108
|
* @returns {promise} A promise representing the state of the new transition. See
|
1893
3109
|
* {@link ui.router.state.$state#methods_go $state.go}.
|
@@ -1901,64 +3117,15 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1901
3117
|
var from = $state.$current, fromParams = $state.params, fromPath = from.path;
|
1902
3118
|
var evt, toState = findState(to, options.relative);
|
1903
3119
|
|
3120
|
+
// Store the hash param for later (since it will be stripped out by various methods)
|
3121
|
+
var hash = toParams['#'];
|
3122
|
+
|
1904
3123
|
if (!isDefined(toState)) {
|
1905
|
-
// Broadcast not found event and abort the transition if prevented
|
1906
3124
|
var redirect = { to: to, toParams: toParams, options: options };
|
3125
|
+
var redirectResult = handleRedirect(redirect, from.self, fromParams, options);
|
1907
3126
|
|
1908
|
-
|
1909
|
-
|
1910
|
-
* @name ui.router.state.$state#$stateNotFound
|
1911
|
-
* @eventOf ui.router.state.$state
|
1912
|
-
* @eventType broadcast on root scope
|
1913
|
-
* @description
|
1914
|
-
* Fired when a requested state **cannot be found** using the provided state name during transition.
|
1915
|
-
* The event is broadcast allowing any handlers a single chance to deal with the error (usually by
|
1916
|
-
* lazy-loading the unfound state). A special `unfoundState` object is passed to the listener handler,
|
1917
|
-
* you can see its three properties in the example. You can use `event.preventDefault()` to abort the
|
1918
|
-
* transition and the promise returned from `go` will be rejected with a `'transition aborted'` value.
|
1919
|
-
*
|
1920
|
-
* @param {Object} event Event object.
|
1921
|
-
* @param {Object} unfoundState Unfound State information. Contains: `to, toParams, options` properties.
|
1922
|
-
* @param {State} fromState Current state object.
|
1923
|
-
* @param {Object} fromParams Current state params.
|
1924
|
-
*
|
1925
|
-
* @example
|
1926
|
-
*
|
1927
|
-
* <pre>
|
1928
|
-
* // somewhere, assume lazy.state has not been defined
|
1929
|
-
* $state.go("lazy.state", {a:1, b:2}, {inherit:false});
|
1930
|
-
*
|
1931
|
-
* // somewhere else
|
1932
|
-
* $scope.$on('$stateNotFound',
|
1933
|
-
* function(event, unfoundState, fromState, fromParams){
|
1934
|
-
* console.log(unfoundState.to); // "lazy.state"
|
1935
|
-
* console.log(unfoundState.toParams); // {a:1, b:2}
|
1936
|
-
* console.log(unfoundState.options); // {inherit:false} + default options
|
1937
|
-
* })
|
1938
|
-
* </pre>
|
1939
|
-
*/
|
1940
|
-
evt = $rootScope.$broadcast('$stateNotFound', redirect, from.self, fromParams);
|
1941
|
-
if (evt.defaultPrevented) {
|
1942
|
-
syncUrl();
|
1943
|
-
return TransitionAborted;
|
1944
|
-
}
|
1945
|
-
|
1946
|
-
// Allow the handler to return a promise to defer state lookup retry
|
1947
|
-
if (evt.retry) {
|
1948
|
-
if (options.$retry) {
|
1949
|
-
syncUrl();
|
1950
|
-
return TransitionFailed;
|
1951
|
-
}
|
1952
|
-
var retryTransition = $state.transition = $q.when(evt.retry);
|
1953
|
-
retryTransition.then(function() {
|
1954
|
-
if (retryTransition !== $state.transition) return TransitionSuperseded;
|
1955
|
-
redirect.options.$retry = true;
|
1956
|
-
return $state.transitionTo(redirect.to, redirect.toParams, redirect.options);
|
1957
|
-
}, function() {
|
1958
|
-
return TransitionAborted;
|
1959
|
-
});
|
1960
|
-
syncUrl();
|
1961
|
-
return retryTransition;
|
3127
|
+
if (redirectResult) {
|
3128
|
+
return redirectResult;
|
1962
3129
|
}
|
1963
3130
|
|
1964
3131
|
// Always retry once if the $stateNotFound was not prevented
|
@@ -1967,38 +3134,68 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
1967
3134
|
toParams = redirect.toParams;
|
1968
3135
|
options = redirect.options;
|
1969
3136
|
toState = findState(to, options.relative);
|
3137
|
+
|
1970
3138
|
if (!isDefined(toState)) {
|
1971
|
-
if (options.relative) throw new Error("
|
1972
|
-
throw new Error("
|
3139
|
+
if (!options.relative) throw new Error("No such state '" + to + "'");
|
3140
|
+
throw new Error("Could not resolve '" + to + "' from state '" + options.relative + "'");
|
1973
3141
|
}
|
1974
3142
|
}
|
1975
3143
|
if (toState[abstractKey]) throw new Error("Cannot transition to abstract state '" + to + "'");
|
1976
3144
|
if (options.inherit) toParams = inheritParams($stateParams, toParams || {}, $state.$current, toState);
|
3145
|
+
if (!toState.params.$$validates(toParams)) return TransitionFailed;
|
3146
|
+
|
3147
|
+
toParams = toState.params.$$values(toParams);
|
1977
3148
|
to = toState;
|
1978
3149
|
|
1979
3150
|
var toPath = to.path;
|
1980
3151
|
|
1981
3152
|
// Starting from the root of the path, keep all levels that haven't changed
|
1982
|
-
var keep, state, locals = root.locals, toLocals = [];
|
1983
|
-
|
1984
|
-
|
1985
|
-
|
1986
|
-
|
3153
|
+
var keep = 0, state = toPath[keep], locals = root.locals, toLocals = [];
|
3154
|
+
|
3155
|
+
if (!options.reload) {
|
3156
|
+
while (state && state === fromPath[keep] && state.ownParams.$$equals(toParams, fromParams)) {
|
3157
|
+
locals = toLocals[keep] = state.locals;
|
3158
|
+
keep++;
|
3159
|
+
state = toPath[keep];
|
3160
|
+
}
|
3161
|
+
} else if (isString(options.reload) || isObject(options.reload)) {
|
3162
|
+
if (isObject(options.reload) && !options.reload.name) {
|
3163
|
+
throw new Error('Invalid reload state object');
|
3164
|
+
}
|
3165
|
+
|
3166
|
+
var reloadState = options.reload === true ? fromPath[0] : findState(options.reload);
|
3167
|
+
if (options.reload && !reloadState) {
|
3168
|
+
throw new Error("No such reload state '" + (isString(options.reload) ? options.reload : options.reload.name) + "'");
|
3169
|
+
}
|
3170
|
+
|
3171
|
+
while (state && state === fromPath[keep] && state !== reloadState) {
|
3172
|
+
locals = toLocals[keep] = state.locals;
|
3173
|
+
keep++;
|
3174
|
+
state = toPath[keep];
|
3175
|
+
}
|
1987
3176
|
}
|
1988
3177
|
|
1989
3178
|
// If we're going to the same state and all locals are kept, we've got nothing to do.
|
1990
3179
|
// But clear 'transition', as we still want to cancel any other pending transitions.
|
1991
|
-
// TODO: We may not want to bump 'transition' if we're called from a location change
|
1992
|
-
// because we might accidentally abort a legitimate
|
1993
|
-
|
1994
|
-
|
1995
|
-
|
3180
|
+
// TODO: We may not want to bump 'transition' if we're called from a location change
|
3181
|
+
// that we've initiated ourselves, because we might accidentally abort a legitimate
|
3182
|
+
// transition initiated from code?
|
3183
|
+
if (shouldSkipReload(to, toParams, from, fromParams, locals, options)) {
|
3184
|
+
if (hash) toParams['#'] = hash;
|
3185
|
+
$state.params = toParams;
|
3186
|
+
copy($state.params, $stateParams);
|
3187
|
+
if (options.location && to.navigable && to.navigable.url) {
|
3188
|
+
$urlRouter.push(to.navigable.url, toParams, {
|
3189
|
+
$$avoidResync: true, replace: options.location === 'replace'
|
3190
|
+
});
|
3191
|
+
$urlRouter.update(true);
|
3192
|
+
}
|
1996
3193
|
$state.transition = null;
|
1997
3194
|
return $q.when($state.current);
|
1998
3195
|
}
|
1999
3196
|
|
2000
|
-
//
|
2001
|
-
toParams =
|
3197
|
+
// Filter parameters before we pass them to event handlers etc.
|
3198
|
+
toParams = filterByKeys(to.params.$$keys(), toParams || {});
|
2002
3199
|
|
2003
3200
|
// Broadcast start event and cancel the transition if requested
|
2004
3201
|
if (options.notify) {
|
@@ -2029,9 +3226,9 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
2029
3226
|
* })
|
2030
3227
|
* </pre>
|
2031
3228
|
*/
|
2032
|
-
|
2033
|
-
|
2034
|
-
|
3229
|
+
if ($rootScope.$broadcast('$stateChangeStart', to.self, toParams, from.self, fromParams).defaultPrevented) {
|
3230
|
+
$rootScope.$broadcast('$stateChangeCancel', to.self, toParams, from.self, fromParams);
|
3231
|
+
$urlRouter.update();
|
2035
3232
|
return TransitionPrevented;
|
2036
3233
|
}
|
2037
3234
|
}
|
@@ -2044,9 +3241,10 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
2044
3241
|
// empty and gets filled asynchronously. We need to keep track of the promise for the
|
2045
3242
|
// (fully resolved) current locals, and pass this down the chain.
|
2046
3243
|
var resolved = $q.when(locals);
|
2047
|
-
|
3244
|
+
|
3245
|
+
for (var l = keep; l < toPath.length; l++, state = toPath[l]) {
|
2048
3246
|
locals = toLocals[l] = inherit(locals);
|
2049
|
-
resolved = resolveState(state, toParams, state===to, resolved, locals);
|
3247
|
+
resolved = resolveState(state, toParams, state === to, resolved, locals, options);
|
2050
3248
|
}
|
2051
3249
|
|
2052
3250
|
// Once everything is resolved, we are ready to perform the actual transition
|
@@ -2059,7 +3257,7 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
2059
3257
|
if ($state.transition !== transition) return TransitionSuperseded;
|
2060
3258
|
|
2061
3259
|
// Exit 'from' states not kept
|
2062
|
-
for (l=fromPath.length-1; l>=keep; l--) {
|
3260
|
+
for (l = fromPath.length - 1; l >= keep; l--) {
|
2063
3261
|
exiting = fromPath[l];
|
2064
3262
|
if (exiting.self.onExit) {
|
2065
3263
|
$injector.invoke(exiting.self.onExit, exiting.self, exiting.locals.globals);
|
@@ -2068,7 +3266,7 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
2068
3266
|
}
|
2069
3267
|
|
2070
3268
|
// Enter 'to' states not kept
|
2071
|
-
for (l=keep; l<toPath.length; l++) {
|
3269
|
+
for (l = keep; l < toPath.length; l++) {
|
2072
3270
|
entering = toPath[l];
|
2073
3271
|
entering.locals = toLocals[l];
|
2074
3272
|
if (entering.self.onEnter) {
|
@@ -2076,6 +3274,9 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
2076
3274
|
}
|
2077
3275
|
}
|
2078
3276
|
|
3277
|
+
// Re-add the saved hash before we start returning things
|
3278
|
+
if (hash) toParams['#'] = hash;
|
3279
|
+
|
2079
3280
|
// Run it again, to catch any transitions in callbacks
|
2080
3281
|
if ($state.transition !== transition) return TransitionSuperseded;
|
2081
3282
|
|
@@ -2086,14 +3287,10 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
2086
3287
|
copy($state.params, $stateParams);
|
2087
3288
|
$state.transition = null;
|
2088
3289
|
|
2089
|
-
|
2090
|
-
|
2091
|
-
|
2092
|
-
|
2093
|
-
|
2094
|
-
if (options.location === 'replace') {
|
2095
|
-
$location.replace();
|
2096
|
-
}
|
3290
|
+
if (options.location && to.navigable) {
|
3291
|
+
$urlRouter.push(to.navigable.url, to.navigable.locals.globals.$stateParams, {
|
3292
|
+
$$avoidResync: true, replace: options.location === 'replace'
|
3293
|
+
});
|
2097
3294
|
}
|
2098
3295
|
|
2099
3296
|
if (options.notify) {
|
@@ -2113,7 +3310,7 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
2113
3310
|
*/
|
2114
3311
|
$rootScope.$broadcast('$stateChangeSuccess', to.self, toParams, from.self, fromParams);
|
2115
3312
|
}
|
2116
|
-
|
3313
|
+
$urlRouter.update(true);
|
2117
3314
|
|
2118
3315
|
return $state.current;
|
2119
3316
|
}, function (error) {
|
@@ -2138,8 +3335,11 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
2138
3335
|
* @param {Object} fromParams The params supplied to the `fromState`.
|
2139
3336
|
* @param {Error} error The resolve error object.
|
2140
3337
|
*/
|
2141
|
-
$rootScope.$broadcast('$stateChangeError', to.self, toParams, from.self, fromParams, error);
|
2142
|
-
|
3338
|
+
evt = $rootScope.$broadcast('$stateChangeError', to.self, toParams, from.self, fromParams, error);
|
3339
|
+
|
3340
|
+
if (!evt.defaultPrevented) {
|
3341
|
+
$urlRouter.update();
|
3342
|
+
}
|
2143
3343
|
|
2144
3344
|
return $q.reject(error);
|
2145
3345
|
});
|
@@ -2154,35 +3354,40 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
2154
3354
|
*
|
2155
3355
|
* @description
|
2156
3356
|
* Similar to {@link ui.router.state.$state#methods_includes $state.includes},
|
2157
|
-
* but only checks for the full state name. If params is supplied then it will be
|
2158
|
-
* tested for strict equality against the current active params object, so all params
|
3357
|
+
* but only checks for the full state name. If params is supplied then it will be
|
3358
|
+
* tested for strict equality against the current active params object, so all params
|
2159
3359
|
* must match with none missing and no extras.
|
2160
3360
|
*
|
2161
3361
|
* @example
|
2162
3362
|
* <pre>
|
3363
|
+
* $state.$current.name = 'contacts.details.item';
|
3364
|
+
*
|
3365
|
+
* // absolute name
|
2163
3366
|
* $state.is('contact.details.item'); // returns true
|
2164
3367
|
* $state.is(contactDetailItemStateObject); // returns true
|
2165
3368
|
*
|
2166
|
-
* //
|
3369
|
+
* // relative name (. and ^), typically from a template
|
3370
|
+
* // E.g. from the 'contacts.details' template
|
3371
|
+
* <div ng-class="{highlighted: $state.is('.item')}">Item</div>
|
2167
3372
|
* </pre>
|
2168
3373
|
*
|
2169
|
-
* @param {string|object}
|
2170
|
-
* @param {object=} params A param object, e.g. `{sectionId: section.id}`, that you'd like
|
3374
|
+
* @param {string|object} stateOrName The state name (absolute or relative) or state object you'd like to check.
|
3375
|
+
* @param {object=} params A param object, e.g. `{sectionId: section.id}`, that you'd like
|
2171
3376
|
* to test against the current active state.
|
3377
|
+
* @param {object=} options An options object. The options are:
|
3378
|
+
*
|
3379
|
+
* - **`relative`** - {string|object} - If `stateOrName` is a relative state name and `options.relative` is set, .is will
|
3380
|
+
* test relative to `options.relative` state (or name).
|
3381
|
+
*
|
2172
3382
|
* @returns {boolean} Returns true if it is the state.
|
2173
3383
|
*/
|
2174
|
-
$state.is = function is(stateOrName, params) {
|
2175
|
-
|
2176
|
-
|
2177
|
-
if (!isDefined(state)) {
|
2178
|
-
return undefined;
|
2179
|
-
}
|
2180
|
-
|
2181
|
-
if ($state.$current !== state) {
|
2182
|
-
return false;
|
2183
|
-
}
|
3384
|
+
$state.is = function is(stateOrName, params, options) {
|
3385
|
+
options = extend({ relative: $state.$current }, options || {});
|
3386
|
+
var state = findState(stateOrName, options.relative);
|
2184
3387
|
|
2185
|
-
|
3388
|
+
if (!isDefined(state)) { return undefined; }
|
3389
|
+
if ($state.$current !== state) { return false; }
|
3390
|
+
return params ? equalForKeys(state.params.$$values(params), $stateParams) : true;
|
2186
3391
|
};
|
2187
3392
|
|
2188
3393
|
/**
|
@@ -2191,25 +3396,28 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
2191
3396
|
* @methodOf ui.router.state.$state
|
2192
3397
|
*
|
2193
3398
|
* @description
|
2194
|
-
* A method to determine if the current active state is equal to or is the child of the
|
3399
|
+
* A method to determine if the current active state is equal to or is the child of the
|
2195
3400
|
* state stateName. If any params are passed then they will be tested for a match as well.
|
2196
3401
|
* Not all the parameters need to be passed, just the ones you'd like to test for equality.
|
2197
3402
|
*
|
2198
3403
|
* @example
|
3404
|
+
* Partial and relative names
|
2199
3405
|
* <pre>
|
2200
3406
|
* $state.$current.name = 'contacts.details.item';
|
2201
3407
|
*
|
3408
|
+
* // Using partial names
|
2202
3409
|
* $state.includes("contacts"); // returns true
|
2203
3410
|
* $state.includes("contacts.details"); // returns true
|
2204
3411
|
* $state.includes("contacts.details.item"); // returns true
|
2205
3412
|
* $state.includes("contacts.list"); // returns false
|
2206
3413
|
* $state.includes("about"); // returns false
|
2207
|
-
* </pre>
|
2208
3414
|
*
|
2209
|
-
*
|
2210
|
-
*
|
3415
|
+
* // Using relative names (. and ^), typically from a template
|
3416
|
+
* // E.g. from the 'contacts.details' template
|
3417
|
+
* <div ng-class="{highlighted: $state.includes('.item')}">Item</div>
|
3418
|
+
* </pre>
|
2211
3419
|
*
|
2212
|
-
*
|
3420
|
+
* Basic globbing patterns
|
2213
3421
|
* <pre>
|
2214
3422
|
* $state.$current.name = 'contacts.details.item.url';
|
2215
3423
|
*
|
@@ -2222,37 +3430,30 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
2222
3430
|
* $state.includes("item.**"); // returns false
|
2223
3431
|
* </pre>
|
2224
3432
|
*
|
2225
|
-
* @param {string} stateOrName A partial name
|
2226
|
-
*
|
3433
|
+
* @param {string} stateOrName A partial name, relative name, or glob pattern
|
3434
|
+
* to be searched for within the current state name.
|
3435
|
+
* @param {object=} params A param object, e.g. `{sectionId: section.id}`,
|
2227
3436
|
* that you'd like to test against the current active state.
|
3437
|
+
* @param {object=} options An options object. The options are:
|
3438
|
+
*
|
3439
|
+
* - **`relative`** - {string|object=} - If `stateOrName` is a relative state reference and `options.relative` is set,
|
3440
|
+
* .includes will test relative to `options.relative` state (or name).
|
3441
|
+
*
|
2228
3442
|
* @returns {boolean} Returns true if it does include the state
|
2229
3443
|
*/
|
2230
|
-
|
2231
|
-
|
3444
|
+
$state.includes = function includes(stateOrName, params, options) {
|
3445
|
+
options = extend({ relative: $state.$current }, options || {});
|
2232
3446
|
if (isString(stateOrName) && isGlob(stateOrName)) {
|
2233
|
-
if (doesStateMatchGlob(stateOrName)) {
|
2234
|
-
stateOrName = $state.$current.name;
|
2235
|
-
} else {
|
3447
|
+
if (!doesStateMatchGlob(stateOrName)) {
|
2236
3448
|
return false;
|
2237
3449
|
}
|
3450
|
+
stateOrName = $state.$current.name;
|
2238
3451
|
}
|
2239
3452
|
|
2240
|
-
var state = findState(stateOrName);
|
2241
|
-
if (!isDefined(state)) {
|
2242
|
-
|
2243
|
-
|
2244
|
-
|
2245
|
-
if (!isDefined($state.$current.includes[state.name])) {
|
2246
|
-
return false;
|
2247
|
-
}
|
2248
|
-
|
2249
|
-
var validParams = true;
|
2250
|
-
angular.forEach(params, function(value, key) {
|
2251
|
-
if (!isDefined($stateParams[key]) || $stateParams[key] !== value) {
|
2252
|
-
validParams = false;
|
2253
|
-
}
|
2254
|
-
});
|
2255
|
-
return validParams;
|
3453
|
+
var state = findState(stateOrName, options.relative);
|
3454
|
+
if (!isDefined(state)) { return undefined; }
|
3455
|
+
if (!isDefined($state.$current.includes[state.name])) { return false; }
|
3456
|
+
return params ? equalForKeys(state.params.$$values(params), $stateParams, objectKeys(params)) : true;
|
2256
3457
|
};
|
2257
3458
|
|
2258
3459
|
|
@@ -2276,7 +3477,7 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
2276
3477
|
* - **`lossy`** - {boolean=true} - If true, and if there is no url associated with the state provided in the
|
2277
3478
|
* first parameter, then the constructed href url will be built from the first navigable ancestor (aka
|
2278
3479
|
* ancestor with a valid url).
|
2279
|
-
* - **`inherit`** - {boolean=
|
3480
|
+
* - **`inherit`** - {boolean=true}, If `true` will inherit url parameters from current url.
|
2280
3481
|
* - **`relative`** - {object=$state.$current}, When transitioning with relative path (e.g '^'),
|
2281
3482
|
* defines which state to be relative from.
|
2282
3483
|
* - **`absolute`** - {boolean=false}, If true will generate an absolute url, e.g. "http://www.example.com/fullurl".
|
@@ -2284,33 +3485,26 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
2284
3485
|
* @returns {string} compiled state url
|
2285
3486
|
*/
|
2286
3487
|
$state.href = function href(stateOrName, params, options) {
|
2287
|
-
options = extend({
|
3488
|
+
options = extend({
|
3489
|
+
lossy: true,
|
3490
|
+
inherit: true,
|
3491
|
+
absolute: false,
|
3492
|
+
relative: $state.$current
|
3493
|
+
}, options || {});
|
3494
|
+
|
2288
3495
|
var state = findState(stateOrName, options.relative);
|
2289
|
-
if (!isDefined(state)) return null;
|
2290
3496
|
|
2291
|
-
|
3497
|
+
if (!isDefined(state)) return null;
|
3498
|
+
if (options.inherit) params = inheritParams($stateParams, params || {}, $state.$current, state);
|
3499
|
+
|
2292
3500
|
var nav = (state && options.lossy) ? state.navigable : state;
|
2293
|
-
var url = (nav && nav.url) ? nav.url.format(normalize(state.params, params || {})) : null;
|
2294
|
-
if (!$locationProvider.html5Mode() && url) {
|
2295
|
-
url = "#" + $locationProvider.hashPrefix() + url;
|
2296
|
-
}
|
2297
|
-
|
2298
|
-
if (baseHref !== '/') {
|
2299
|
-
if ($locationProvider.html5Mode()) {
|
2300
|
-
url = baseHref.slice(0, -1) + url;
|
2301
|
-
} else if (options.absolute){
|
2302
|
-
url = baseHref.slice(1) + url;
|
2303
|
-
}
|
2304
|
-
}
|
2305
3501
|
|
2306
|
-
if (
|
2307
|
-
|
2308
|
-
$location.host() +
|
2309
|
-
($location.port() == 80 || $location.port() == 443 ? '' : ':' + $location.port()) +
|
2310
|
-
(!$locationProvider.html5Mode() && url ? '/' : '') +
|
2311
|
-
url;
|
3502
|
+
if (!nav || nav.url === undefined || nav.url === null) {
|
3503
|
+
return null;
|
2312
3504
|
}
|
2313
|
-
return url
|
3505
|
+
return $urlRouter.href(nav.url, filterByKeys(state.params.$$keys().concat('#'), params || {}), {
|
3506
|
+
absolute: options.absolute
|
3507
|
+
});
|
2314
3508
|
};
|
2315
3509
|
|
2316
3510
|
/**
|
@@ -2321,26 +3515,23 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
2321
3515
|
* @description
|
2322
3516
|
* Returns the state configuration object for any specific state or all states.
|
2323
3517
|
*
|
2324
|
-
* @param {string|object=} stateOrName If provided, will only get the config for
|
3518
|
+
* @param {string|object=} stateOrName (absolute or relative) If provided, will only get the config for
|
2325
3519
|
* the requested state. If not provided, returns an array of ALL state configs.
|
2326
|
-
* @
|
3520
|
+
* @param {string|object=} context When stateOrName is a relative state reference, the state will be retrieved relative to context.
|
3521
|
+
* @returns {Object|Array} State configuration object or array of all objects.
|
2327
3522
|
*/
|
2328
3523
|
$state.get = function (stateOrName, context) {
|
2329
|
-
if (
|
2330
|
-
|
2331
|
-
forEach(states, function(state) { list.push(state.self); });
|
2332
|
-
return list;
|
2333
|
-
}
|
2334
|
-
var state = findState(stateOrName, context);
|
3524
|
+
if (arguments.length === 0) return map(objectKeys(states), function(name) { return states[name].self; });
|
3525
|
+
var state = findState(stateOrName, context || $state.$current);
|
2335
3526
|
return (state && state.self) ? state.self : null;
|
2336
3527
|
};
|
2337
3528
|
|
2338
|
-
function resolveState(state, params, paramsAreFiltered, inherited, dst) {
|
3529
|
+
function resolveState(state, params, paramsAreFiltered, inherited, dst, options) {
|
2339
3530
|
// Make a restricted $stateParams with only the parameters that apply to this state if
|
2340
3531
|
// necessary. In addition to being available to the controller and onEnter/onExit callbacks,
|
2341
3532
|
// we also need $stateParams to be available for any $injector calls we make during the
|
2342
3533
|
// dependency resolution process.
|
2343
|
-
var $stateParams = (paramsAreFiltered) ? params : filterByKeys(state.params, params);
|
3534
|
+
var $stateParams = (paramsAreFiltered) ? params : filterByKeys(state.params.$$keys(), params);
|
2344
3535
|
var locals = { $stateParams: $stateParams };
|
2345
3536
|
|
2346
3537
|
// Resolve 'global' dependencies for the state, i.e. those not specific to a view.
|
@@ -2348,35 +3539,43 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
2348
3539
|
// to the set that should be visible to the state, and are independent of when we update
|
2349
3540
|
// the global $state and $stateParams values.
|
2350
3541
|
dst.resolve = $resolve.resolve(state.resolve, locals, dst.resolve, state);
|
2351
|
-
var promises = [
|
3542
|
+
var promises = [dst.resolve.then(function (globals) {
|
2352
3543
|
dst.globals = globals;
|
2353
|
-
})
|
3544
|
+
})];
|
2354
3545
|
if (inherited) promises.push(inherited);
|
2355
3546
|
|
2356
|
-
|
2357
|
-
|
2358
|
-
|
2359
|
-
|
2360
|
-
|
2361
|
-
|
2362
|
-
|
2363
|
-
|
2364
|
-
|
2365
|
-
|
2366
|
-
|
2367
|
-
|
2368
|
-
|
2369
|
-
|
2370
|
-
|
2371
|
-
|
2372
|
-
|
2373
|
-
|
2374
|
-
|
2375
|
-
|
2376
|
-
|
3547
|
+
function resolveViews() {
|
3548
|
+
var viewsPromises = [];
|
3549
|
+
|
3550
|
+
// Resolve template and dependencies for all views.
|
3551
|
+
forEach(state.views, function (view, name) {
|
3552
|
+
var injectables = (view.resolve && view.resolve !== state.resolve ? view.resolve : {});
|
3553
|
+
injectables.$template = [ function () {
|
3554
|
+
return $view.load(name, { view: view, locals: dst.globals, params: $stateParams, notify: options.notify }) || '';
|
3555
|
+
}];
|
3556
|
+
|
3557
|
+
viewsPromises.push($resolve.resolve(injectables, dst.globals, dst.resolve, state).then(function (result) {
|
3558
|
+
// References to the controller (only instantiated at link time)
|
3559
|
+
if (isFunction(view.controllerProvider) || isArray(view.controllerProvider)) {
|
3560
|
+
var injectLocals = angular.extend({}, injectables, dst.globals);
|
3561
|
+
result.$$controller = $injector.invoke(view.controllerProvider, null, injectLocals);
|
3562
|
+
} else {
|
3563
|
+
result.$$controller = view.controller;
|
3564
|
+
}
|
3565
|
+
// Provide access to the state itself for internal use
|
3566
|
+
result.$$state = state;
|
3567
|
+
result.$$controllerAs = view.controllerAs;
|
3568
|
+
dst[name] = result;
|
3569
|
+
}));
|
3570
|
+
});
|
3571
|
+
|
3572
|
+
return $q.all(viewsPromises).then(function(){
|
3573
|
+
return dst.globals;
|
3574
|
+
});
|
3575
|
+
}
|
2377
3576
|
|
2378
3577
|
// Wait for all the promises and then return the activation object
|
2379
|
-
return $q.all(promises).then(function (values) {
|
3578
|
+
return $q.all(promises).then(resolveViews).then(function (values) {
|
2380
3579
|
return dst;
|
2381
3580
|
});
|
2382
3581
|
}
|
@@ -2384,8 +3583,27 @@ function $StateProvider( $urlRouterProvider, $urlMatcherFactory, $
|
|
2384
3583
|
return $state;
|
2385
3584
|
}
|
2386
3585
|
|
2387
|
-
function
|
2388
|
-
|
3586
|
+
function shouldSkipReload(to, toParams, from, fromParams, locals, options) {
|
3587
|
+
// Return true if there are no differences in non-search (path/object) params, false if there are differences
|
3588
|
+
function nonSearchParamsEqual(fromAndToState, fromParams, toParams) {
|
3589
|
+
// Identify whether all the parameters that differ between `fromParams` and `toParams` were search params.
|
3590
|
+
function notSearchParam(key) {
|
3591
|
+
return fromAndToState.params[key].location != "search";
|
3592
|
+
}
|
3593
|
+
var nonQueryParamKeys = fromAndToState.params.$$keys().filter(notSearchParam);
|
3594
|
+
var nonQueryParams = pick.apply({}, [fromAndToState.params].concat(nonQueryParamKeys));
|
3595
|
+
var nonQueryParamSet = new $$UMFP.ParamSet(nonQueryParams);
|
3596
|
+
return nonQueryParamSet.$$equals(fromParams, toParams);
|
3597
|
+
}
|
3598
|
+
|
3599
|
+
// If reload was not explicitly requested
|
3600
|
+
// and we're transitioning to the same state we're already in
|
3601
|
+
// and the locals didn't change
|
3602
|
+
// or they changed in a way that doesn't merit reloading
|
3603
|
+
// (reloadOnParams:false, or reloadOnSearch.false and only search params changed)
|
3604
|
+
// Then return true.
|
3605
|
+
if (!options.reload && to === from &&
|
3606
|
+
(locals === from.locals || (to.self.reloadOnSearch === false && nonSearchParamsEqual(from, fromParams, toParams)))) {
|
2389
3607
|
return true;
|
2390
3608
|
}
|
2391
3609
|
}
|
@@ -2511,7 +3729,7 @@ function $ViewScrollProvider() {
|
|
2511
3729
|
}
|
2512
3730
|
|
2513
3731
|
return function ($element) {
|
2514
|
-
$timeout(function () {
|
3732
|
+
return $timeout(function () {
|
2515
3733
|
$element[0].scrollIntoView();
|
2516
3734
|
}, 0, false);
|
2517
3735
|
};
|
@@ -2536,7 +3754,7 @@ angular.module('ui.router.state').provider('$uiViewScroll', $ViewScrollProvider)
|
|
2536
3754
|
* @description
|
2537
3755
|
* The ui-view directive tells $state where to place your templates.
|
2538
3756
|
*
|
2539
|
-
* @param {string=}
|
3757
|
+
* @param {string=} name A view name. The name should be unique amongst the other views in the
|
2540
3758
|
* same state. You can have views of the same name that live in different states.
|
2541
3759
|
*
|
2542
3760
|
* @param {string=} autoscroll It allows you to set the scroll behavior of the browser window
|
@@ -2633,8 +3851,8 @@ angular.module('ui.router.state').provider('$uiViewScroll', $ViewScrollProvider)
|
|
2633
3851
|
* <ui-view autoscroll='scopeVariable'/>
|
2634
3852
|
* </pre>
|
2635
3853
|
*/
|
2636
|
-
$ViewDirective.$inject = ['$state', '$injector', '$uiViewScroll'];
|
2637
|
-
function $ViewDirective( $state, $injector, $uiViewScroll) {
|
3854
|
+
$ViewDirective.$inject = ['$state', '$injector', '$uiViewScroll', '$interpolate'];
|
3855
|
+
function $ViewDirective( $state, $injector, $uiViewScroll, $interpolate) {
|
2638
3856
|
|
2639
3857
|
function getService() {
|
2640
3858
|
return ($injector.has) ? function(service) {
|
@@ -2664,8 +3882,14 @@ function $ViewDirective( $state, $injector, $uiViewScroll) {
|
|
2664
3882
|
|
2665
3883
|
if ($animate) {
|
2666
3884
|
return {
|
2667
|
-
enter: function(element, target, cb) {
|
2668
|
-
|
3885
|
+
enter: function(element, target, cb) {
|
3886
|
+
var promise = $animate.enter(element, null, target, cb);
|
3887
|
+
if (promise && promise.then) promise.then(cb);
|
3888
|
+
},
|
3889
|
+
leave: function(element, cb) {
|
3890
|
+
var promise = $animate.leave(element, cb);
|
3891
|
+
if (promise && promise.then) promise.then(cb);
|
3892
|
+
}
|
2669
3893
|
};
|
2670
3894
|
}
|
2671
3895
|
|
@@ -2724,14 +3948,20 @@ function $ViewDirective( $state, $injector, $uiViewScroll) {
|
|
2724
3948
|
}
|
2725
3949
|
|
2726
3950
|
function updateView(firstTime) {
|
2727
|
-
var newScope
|
2728
|
-
name =
|
3951
|
+
var newScope,
|
3952
|
+
name = getUiViewName(scope, attrs, $element, $interpolate),
|
2729
3953
|
previousLocals = name && $state.$current && $state.$current.locals[name];
|
2730
3954
|
|
2731
3955
|
if (!firstTime && previousLocals === latestLocals) return; // nothing to do
|
3956
|
+
newScope = scope.$new();
|
3957
|
+
latestLocals = $state.$current.locals[name];
|
2732
3958
|
|
2733
3959
|
var clone = $transclude(newScope, function(clone) {
|
2734
3960
|
renderer.enter(clone, $element, function onUiViewEnter() {
|
3961
|
+
if(currentScope) {
|
3962
|
+
currentScope.$emit('$viewContentAnimationEnded');
|
3963
|
+
}
|
3964
|
+
|
2735
3965
|
if (angular.isDefined(autoScrollExp) && !autoScrollExp || scope.$eval(autoScrollExp)) {
|
2736
3966
|
$uiViewScroll(clone);
|
2737
3967
|
}
|
@@ -2739,8 +3969,6 @@ function $ViewDirective( $state, $injector, $uiViewScroll) {
|
|
2739
3969
|
cleanupLastView();
|
2740
3970
|
});
|
2741
3971
|
|
2742
|
-
latestLocals = $state.$current.locals[clone.data('$uiViewName')];
|
2743
|
-
|
2744
3972
|
currentEl = clone;
|
2745
3973
|
currentScope = newScope;
|
2746
3974
|
/**
|
@@ -2763,24 +3991,16 @@ function $ViewDirective( $state, $injector, $uiViewScroll) {
|
|
2763
3991
|
return directive;
|
2764
3992
|
}
|
2765
3993
|
|
2766
|
-
$ViewDirectiveFill.$inject = ['$compile', '$controller', '$state'];
|
2767
|
-
function $ViewDirectiveFill ($compile,
|
3994
|
+
$ViewDirectiveFill.$inject = ['$compile', '$controller', '$state', '$interpolate'];
|
3995
|
+
function $ViewDirectiveFill ( $compile, $controller, $state, $interpolate) {
|
2768
3996
|
return {
|
2769
3997
|
restrict: 'ECA',
|
2770
3998
|
priority: -400,
|
2771
3999
|
compile: function (tElement) {
|
2772
4000
|
var initial = tElement.html();
|
2773
4001
|
return function (scope, $element, attrs) {
|
2774
|
-
var name = attrs.uiView || attrs.name || '',
|
2775
|
-
inherited = $element.inheritedData('$uiView');
|
2776
|
-
|
2777
|
-
if (name.indexOf('@') < 0) {
|
2778
|
-
name = name + '@' + (inherited ? inherited.state.name : '');
|
2779
|
-
}
|
2780
|
-
|
2781
|
-
$element.data('$uiViewName', name);
|
2782
|
-
|
2783
4002
|
var current = $state.$current,
|
4003
|
+
name = getUiViewName(scope, attrs, $element, $interpolate),
|
2784
4004
|
locals = current && current.locals[name];
|
2785
4005
|
|
2786
4006
|
if (! locals) {
|
@@ -2794,6 +4014,7 @@ function $ViewDirectiveFill ($compile, $controller, $state) {
|
|
2794
4014
|
|
2795
4015
|
if (locals.$$controller) {
|
2796
4016
|
locals.$scope = scope;
|
4017
|
+
locals.$element = $element;
|
2797
4018
|
var controller = $controller(locals.$$controller, locals);
|
2798
4019
|
if (locals.$$controllerAs) {
|
2799
4020
|
scope[locals.$$controllerAs] = controller;
|
@@ -2808,11 +4029,23 @@ function $ViewDirectiveFill ($compile, $controller, $state) {
|
|
2808
4029
|
};
|
2809
4030
|
}
|
2810
4031
|
|
4032
|
+
/**
|
4033
|
+
* Shared ui-view code for both directives:
|
4034
|
+
* Given scope, element, and its attributes, return the view's name
|
4035
|
+
*/
|
4036
|
+
function getUiViewName(scope, attrs, element, $interpolate) {
|
4037
|
+
var name = $interpolate(attrs.uiView || attrs.name || '')(scope);
|
4038
|
+
var inherited = element.inheritedData('$uiView');
|
4039
|
+
return name.indexOf('@') >= 0 ? name : (name + '@' + (inherited ? inherited.state.name : ''));
|
4040
|
+
}
|
4041
|
+
|
2811
4042
|
angular.module('ui.router.state').directive('uiView', $ViewDirective);
|
2812
4043
|
angular.module('ui.router.state').directive('uiView', $ViewDirectiveFill);
|
2813
4044
|
|
2814
|
-
function parseStateRef(ref) {
|
2815
|
-
var
|
4045
|
+
function parseStateRef(ref, current) {
|
4046
|
+
var preparsed = ref.match(/^\s*({[^}]*})\s*$/), parsed;
|
4047
|
+
if (preparsed) ref = current + '(' + preparsed[1] + ')';
|
4048
|
+
parsed = ref.replace(/\n/g, " ").match(/^([^(]+?)\s*(\((.*)\))?$/);
|
2816
4049
|
if (!parsed || parsed.length !== 4) throw new Error("Invalid state ref '" + ref + "'");
|
2817
4050
|
return { state: parsed[1], paramExpr: parsed[3] || null };
|
2818
4051
|
}
|
@@ -2856,7 +4089,7 @@ function stateContext(el) {
|
|
2856
4089
|
* Here's an example of how you'd use ui-sref and how it would compile. If you have the
|
2857
4090
|
* following template:
|
2858
4091
|
* <pre>
|
2859
|
-
* <a ui-sref="home">Home</a> | <a ui-sref="about">About</a>
|
4092
|
+
* <a ui-sref="home">Home</a> | <a ui-sref="about">About</a> | <a ui-sref="{page: 2}">Next page</a>
|
2860
4093
|
*
|
2861
4094
|
* <ul>
|
2862
4095
|
* <li ng-repeat="contact in contacts">
|
@@ -2865,9 +4098,9 @@ function stateContext(el) {
|
|
2865
4098
|
* </ul>
|
2866
4099
|
* </pre>
|
2867
4100
|
*
|
2868
|
-
* Then the compiled html would be (assuming Html5Mode is off):
|
4101
|
+
* Then the compiled html would be (assuming Html5Mode is off and current state is contacts):
|
2869
4102
|
* <pre>
|
2870
|
-
* <a href="#/home" ui-sref="home">Home</a> | <a href="#/about" ui-sref="about">About</a>
|
4103
|
+
* <a href="#/home" ui-sref="home">Home</a> | <a href="#/about" ui-sref="about">About</a> | <a href="#/contacts?page=2" ui-sref="{page: 2}">Next page</a>
|
2871
4104
|
*
|
2872
4105
|
* <ul>
|
2873
4106
|
* <li ng-repeat="contact in contacts">
|
@@ -2889,21 +4122,24 @@ function stateContext(el) {
|
|
2889
4122
|
*/
|
2890
4123
|
$StateRefDirective.$inject = ['$state', '$timeout'];
|
2891
4124
|
function $StateRefDirective($state, $timeout) {
|
2892
|
-
var allowedOptions = ['location', 'inherit', 'reload'];
|
4125
|
+
var allowedOptions = ['location', 'inherit', 'reload', 'absolute'];
|
2893
4126
|
|
2894
4127
|
return {
|
2895
4128
|
restrict: 'A',
|
2896
|
-
require: '?^uiSrefActive',
|
4129
|
+
require: ['?^uiSrefActive', '?^uiSrefActiveEq'],
|
2897
4130
|
link: function(scope, element, attrs, uiSrefActive) {
|
2898
|
-
var ref = parseStateRef(attrs.uiSref);
|
4131
|
+
var ref = parseStateRef(attrs.uiSref, $state.current.name);
|
2899
4132
|
var params = null, url = null, base = stateContext(element) || $state.$current;
|
4133
|
+
// SVGAElement does not use the href attribute, but rather the 'xlinkHref' attribute.
|
4134
|
+
var hrefKind = Object.prototype.toString.call(element.prop('href')) === '[object SVGAnimatedString]' ?
|
4135
|
+
'xlink:href' : 'href';
|
4136
|
+
var newHref = null, isAnchor = element.prop("tagName").toUpperCase() === "A";
|
2900
4137
|
var isForm = element[0].nodeName === "FORM";
|
2901
|
-
var attr = isForm ? "action" :
|
4138
|
+
var attr = isForm ? "action" : hrefKind, nav = true;
|
2902
4139
|
|
2903
|
-
var options = {
|
2904
|
-
relative: base
|
2905
|
-
};
|
4140
|
+
var options = { relative: base, inherit: true };
|
2906
4141
|
var optionsOverride = scope.$eval(attrs.uiSrefOpts) || {};
|
4142
|
+
|
2907
4143
|
angular.forEach(allowedOptions, function(option) {
|
2908
4144
|
if (option in optionsOverride) {
|
2909
4145
|
options[option] = optionsOverride[option];
|
@@ -2911,26 +4147,27 @@ function $StateRefDirective($state, $timeout) {
|
|
2911
4147
|
});
|
2912
4148
|
|
2913
4149
|
var update = function(newVal) {
|
2914
|
-
if (newVal) params = newVal;
|
4150
|
+
if (newVal) params = angular.copy(newVal);
|
2915
4151
|
if (!nav) return;
|
2916
4152
|
|
2917
|
-
|
4153
|
+
newHref = $state.href(ref.state, params, options);
|
2918
4154
|
|
2919
|
-
|
2920
|
-
|
4155
|
+
var activeDirective = uiSrefActive[1] || uiSrefActive[0];
|
4156
|
+
if (activeDirective) {
|
4157
|
+
activeDirective.$$addStateInfo(ref.state, params);
|
2921
4158
|
}
|
2922
|
-
if (
|
4159
|
+
if (newHref === null) {
|
2923
4160
|
nav = false;
|
2924
4161
|
return false;
|
2925
4162
|
}
|
2926
|
-
|
4163
|
+
attrs.$set(attr, newHref);
|
2927
4164
|
};
|
2928
4165
|
|
2929
4166
|
if (ref.paramExpr) {
|
2930
4167
|
scope.$watch(ref.paramExpr, function(newVal, oldVal) {
|
2931
4168
|
if (newVal !== params) update(newVal);
|
2932
4169
|
}, true);
|
2933
|
-
params = scope.$eval(ref.paramExpr);
|
4170
|
+
params = angular.copy(scope.$eval(ref.paramExpr));
|
2934
4171
|
}
|
2935
4172
|
update();
|
2936
4173
|
|
@@ -2940,10 +4177,17 @@ function $StateRefDirective($state, $timeout) {
|
|
2940
4177
|
var button = e.which || e.button;
|
2941
4178
|
if ( !(button > 1 || e.ctrlKey || e.metaKey || e.shiftKey || element.attr('target')) ) {
|
2942
4179
|
// HACK: This is to allow ng-clicks to be processed before the transition is initiated:
|
2943
|
-
$timeout(function() {
|
4180
|
+
var transition = $timeout(function() {
|
2944
4181
|
$state.go(ref.state, params, options);
|
2945
4182
|
});
|
2946
4183
|
e.preventDefault();
|
4184
|
+
|
4185
|
+
// if the state has no URL, ignore one preventDefault from the <a> directive.
|
4186
|
+
var ignorePreventDefaultCount = isAnchor && !newHref ? 1: 0;
|
4187
|
+
e.preventDefault = function() {
|
4188
|
+
if (ignorePreventDefaultCount-- <= 0)
|
4189
|
+
$timeout.cancel(transition);
|
4190
|
+
};
|
2947
4191
|
}
|
2948
4192
|
});
|
2949
4193
|
}
|
@@ -2961,12 +4205,20 @@ function $StateRefDirective($state, $timeout) {
|
|
2961
4205
|
* @restrict A
|
2962
4206
|
*
|
2963
4207
|
* @description
|
2964
|
-
* A directive working alongside ui-sref to add classes to an element when the
|
4208
|
+
* A directive working alongside ui-sref to add classes to an element when the
|
2965
4209
|
* related ui-sref directive's state is active, and removing them when it is inactive.
|
2966
|
-
* The primary use-case is to simplify the special appearance of navigation menus
|
4210
|
+
* The primary use-case is to simplify the special appearance of navigation menus
|
2967
4211
|
* relying on `ui-sref`, by having the "active" state's menu button appear different,
|
2968
4212
|
* distinguishing it from the inactive menu items.
|
2969
4213
|
*
|
4214
|
+
* ui-sref-active can live on the same element as ui-sref or on a parent element. The first
|
4215
|
+
* ui-sref-active found at the same level or above the ui-sref will be used.
|
4216
|
+
*
|
4217
|
+
* Will activate when the ui-sref's target state or any child state is active. If you
|
4218
|
+
* need to activate only when the ui-sref target state is active and *not* any of
|
4219
|
+
* it's children, then you will use
|
4220
|
+
* {@link ui.router.state.directive:ui-sref-active-eq ui-sref-active-eq}
|
4221
|
+
*
|
2970
4222
|
* @example
|
2971
4223
|
* Given the following template:
|
2972
4224
|
* <pre>
|
@@ -2976,8 +4228,9 @@ function $StateRefDirective($state, $timeout) {
|
|
2976
4228
|
* </li>
|
2977
4229
|
* </ul>
|
2978
4230
|
* </pre>
|
2979
|
-
*
|
2980
|
-
*
|
4231
|
+
*
|
4232
|
+
*
|
4233
|
+
* When the app state is "app.user" (or any children states), and contains the state parameter "user" with value "bilbobaggins",
|
2981
4234
|
* the resulting HTML will appear as (note the 'active' class):
|
2982
4235
|
* <pre>
|
2983
4236
|
* <ul>
|
@@ -2986,10 +4239,10 @@ function $StateRefDirective($state, $timeout) {
|
|
2986
4239
|
* </li>
|
2987
4240
|
* </ul>
|
2988
4241
|
* </pre>
|
2989
|
-
*
|
2990
|
-
* The class name is interpolated **once** during the directives link time (any further changes to the
|
2991
|
-
* interpolated value are ignored).
|
2992
|
-
*
|
4242
|
+
*
|
4243
|
+
* The class name is interpolated **once** during the directives link time (any further changes to the
|
4244
|
+
* interpolated value are ignored).
|
4245
|
+
*
|
2993
4246
|
* Multiple classes may be specified in a space-separated format:
|
2994
4247
|
* <pre>
|
2995
4248
|
* <ul>
|
@@ -2999,20 +4252,43 @@ function $StateRefDirective($state, $timeout) {
|
|
2999
4252
|
* </ul>
|
3000
4253
|
* </pre>
|
3001
4254
|
*/
|
3002
|
-
|
3003
|
-
|
3004
|
-
|
4255
|
+
|
4256
|
+
/**
|
4257
|
+
* @ngdoc directive
|
4258
|
+
* @name ui.router.state.directive:ui-sref-active-eq
|
4259
|
+
*
|
4260
|
+
* @requires ui.router.state.$state
|
4261
|
+
* @requires ui.router.state.$stateParams
|
4262
|
+
* @requires $interpolate
|
4263
|
+
*
|
4264
|
+
* @restrict A
|
4265
|
+
*
|
4266
|
+
* @description
|
4267
|
+
* The same as {@link ui.router.state.directive:ui-sref-active ui-sref-active} but will only activate
|
4268
|
+
* when the exact target state used in the `ui-sref` is active; no child states.
|
4269
|
+
*
|
4270
|
+
*/
|
4271
|
+
$StateRefActiveDirective.$inject = ['$state', '$stateParams', '$interpolate'];
|
4272
|
+
function $StateRefActiveDirective($state, $stateParams, $interpolate) {
|
4273
|
+
return {
|
3005
4274
|
restrict: "A",
|
3006
|
-
controller: ['$scope', '$element', '$attrs', function($scope, $element, $attrs) {
|
3007
|
-
var
|
4275
|
+
controller: ['$scope', '$element', '$attrs', function ($scope, $element, $attrs) {
|
4276
|
+
var states = [], activeClass;
|
3008
4277
|
|
3009
4278
|
// There probably isn't much point in $observing this
|
3010
|
-
|
4279
|
+
// uiSrefActive and uiSrefActiveEq share the same directive object with some
|
4280
|
+
// slight difference in logic routing
|
4281
|
+
activeClass = $interpolate($attrs.uiSrefActiveEq || $attrs.uiSrefActive || '', false)($scope);
|
4282
|
+
|
4283
|
+
// Allow uiSref to communicate with uiSrefActive[Equals]
|
4284
|
+
this.$$addStateInfo = function (newState, newParams) {
|
4285
|
+
var state = $state.get(newState, stateContext($element));
|
4286
|
+
|
4287
|
+
states.push({
|
4288
|
+
state: state || { name: newState },
|
4289
|
+
params: newParams
|
4290
|
+
});
|
3011
4291
|
|
3012
|
-
// Allow uiSref to communicate with uiSrefActive
|
3013
|
-
this.$$setStateInfo = function(newState, newParams) {
|
3014
|
-
state = $state.get(newState, stateContext($element));
|
3015
|
-
params = newParams;
|
3016
4292
|
update();
|
3017
4293
|
};
|
3018
4294
|
|
@@ -3020,15 +4296,28 @@ function $StateActiveDirective($state, $stateParams, $interpolate) {
|
|
3020
4296
|
|
3021
4297
|
// Update route state
|
3022
4298
|
function update() {
|
3023
|
-
if (
|
4299
|
+
if (anyMatch()) {
|
3024
4300
|
$element.addClass(activeClass);
|
3025
4301
|
} else {
|
3026
4302
|
$element.removeClass(activeClass);
|
3027
4303
|
}
|
3028
4304
|
}
|
3029
4305
|
|
3030
|
-
function
|
3031
|
-
|
4306
|
+
function anyMatch() {
|
4307
|
+
for (var i = 0; i < states.length; i++) {
|
4308
|
+
if (isMatch(states[i].state, states[i].params)) {
|
4309
|
+
return true;
|
4310
|
+
}
|
4311
|
+
}
|
4312
|
+
return false;
|
4313
|
+
}
|
4314
|
+
|
4315
|
+
function isMatch(state, params) {
|
4316
|
+
if (typeof $attrs.uiSrefActiveEq !== 'undefined') {
|
4317
|
+
return $state.is(state.name, params);
|
4318
|
+
} else {
|
4319
|
+
return $state.includes(state.name, params);
|
4320
|
+
}
|
3032
4321
|
}
|
3033
4322
|
}]
|
3034
4323
|
};
|
@@ -3036,7 +4325,8 @@ function $StateActiveDirective($state, $stateParams, $interpolate) {
|
|
3036
4325
|
|
3037
4326
|
angular.module('ui.router.state')
|
3038
4327
|
.directive('uiSref', $StateRefDirective)
|
3039
|
-
.directive('uiSrefActive', $
|
4328
|
+
.directive('uiSrefActive', $StateRefActiveDirective)
|
4329
|
+
.directive('uiSrefActiveEq', $StateRefActiveDirective);
|
3040
4330
|
|
3041
4331
|
/**
|
3042
4332
|
* @ngdoc filter
|
@@ -3049,9 +4339,11 @@ angular.module('ui.router.state')
|
|
3049
4339
|
*/
|
3050
4340
|
$IsStateFilter.$inject = ['$state'];
|
3051
4341
|
function $IsStateFilter($state) {
|
3052
|
-
|
4342
|
+
var isFilter = function (state) {
|
3053
4343
|
return $state.is(state);
|
3054
4344
|
};
|
4345
|
+
isFilter.$stateful = true;
|
4346
|
+
return isFilter;
|
3055
4347
|
}
|
3056
4348
|
|
3057
4349
|
/**
|
@@ -3065,159 +4357,14 @@ function $IsStateFilter($state) {
|
|
3065
4357
|
*/
|
3066
4358
|
$IncludedByStateFilter.$inject = ['$state'];
|
3067
4359
|
function $IncludedByStateFilter($state) {
|
3068
|
-
|
4360
|
+
var includesFilter = function (state) {
|
3069
4361
|
return $state.includes(state);
|
3070
4362
|
};
|
4363
|
+
includesFilter.$stateful = true;
|
4364
|
+
return includesFilter;
|
3071
4365
|
}
|
3072
4366
|
|
3073
4367
|
angular.module('ui.router.state')
|
3074
4368
|
.filter('isState', $IsStateFilter)
|
3075
4369
|
.filter('includedByState', $IncludedByStateFilter);
|
3076
|
-
|
3077
|
-
/*
|
3078
|
-
* @ngdoc object
|
3079
|
-
* @name ui.router.compat.$routeProvider
|
3080
|
-
*
|
3081
|
-
* @requires ui.router.state.$stateProvider
|
3082
|
-
* @requires ui.router.router.$urlRouterProvider
|
3083
|
-
*
|
3084
|
-
* @description
|
3085
|
-
* `$routeProvider` of the `ui.router.compat` module overwrites the existing
|
3086
|
-
* `routeProvider` from the core. This is done to provide compatibility between
|
3087
|
-
* the UI Router and the core router.
|
3088
|
-
*
|
3089
|
-
* It also provides a `when()` method to register routes that map to certain urls.
|
3090
|
-
* Behind the scenes it actually delegates either to
|
3091
|
-
* {@link ui.router.router.$urlRouterProvider $urlRouterProvider} or to the
|
3092
|
-
* {@link ui.router.state.$stateProvider $stateProvider} to postprocess the given
|
3093
|
-
* router definition object.
|
3094
|
-
*/
|
3095
|
-
$RouteProvider.$inject = ['$stateProvider', '$urlRouterProvider'];
|
3096
|
-
function $RouteProvider( $stateProvider, $urlRouterProvider) {
|
3097
|
-
|
3098
|
-
var routes = [];
|
3099
|
-
|
3100
|
-
onEnterRoute.$inject = ['$$state'];
|
3101
|
-
function onEnterRoute( $$state) {
|
3102
|
-
/*jshint validthis: true */
|
3103
|
-
this.locals = $$state.locals.globals;
|
3104
|
-
this.params = this.locals.$stateParams;
|
3105
|
-
}
|
3106
|
-
|
3107
|
-
function onExitRoute() {
|
3108
|
-
/*jshint validthis: true */
|
3109
|
-
this.locals = null;
|
3110
|
-
this.params = null;
|
3111
|
-
}
|
3112
|
-
|
3113
|
-
this.when = when;
|
3114
|
-
/*
|
3115
|
-
* @ngdoc function
|
3116
|
-
* @name ui.router.compat.$routeProvider#when
|
3117
|
-
* @methodOf ui.router.compat.$routeProvider
|
3118
|
-
*
|
3119
|
-
* @description
|
3120
|
-
* Registers a route with a given route definition object. The route definition
|
3121
|
-
* object has the same interface the angular core route definition object has.
|
3122
|
-
*
|
3123
|
-
* @example
|
3124
|
-
* <pre>
|
3125
|
-
* var app = angular.module('app', ['ui.router.compat']);
|
3126
|
-
*
|
3127
|
-
* app.config(function ($routeProvider) {
|
3128
|
-
* $routeProvider.when('home', {
|
3129
|
-
* controller: function () { ... },
|
3130
|
-
* templateUrl: 'path/to/template'
|
3131
|
-
* });
|
3132
|
-
* });
|
3133
|
-
* </pre>
|
3134
|
-
*
|
3135
|
-
* @param {string} url URL as string
|
3136
|
-
* @param {object} route Route definition object
|
3137
|
-
*
|
3138
|
-
* @return {object} $routeProvider - $routeProvider instance
|
3139
|
-
*/
|
3140
|
-
function when(url, route) {
|
3141
|
-
/*jshint validthis: true */
|
3142
|
-
if (route.redirectTo != null) {
|
3143
|
-
// Redirect, configure directly on $urlRouterProvider
|
3144
|
-
var redirect = route.redirectTo, handler;
|
3145
|
-
if (isString(redirect)) {
|
3146
|
-
handler = redirect; // leave $urlRouterProvider to handle
|
3147
|
-
} else if (isFunction(redirect)) {
|
3148
|
-
// Adapt to $urlRouterProvider API
|
3149
|
-
handler = function (params, $location) {
|
3150
|
-
return redirect(params, $location.path(), $location.search());
|
3151
|
-
};
|
3152
|
-
} else {
|
3153
|
-
throw new Error("Invalid 'redirectTo' in when()");
|
3154
|
-
}
|
3155
|
-
$urlRouterProvider.when(url, handler);
|
3156
|
-
} else {
|
3157
|
-
// Regular route, configure as state
|
3158
|
-
$stateProvider.state(inherit(route, {
|
3159
|
-
parent: null,
|
3160
|
-
name: 'route:' + encodeURIComponent(url),
|
3161
|
-
url: url,
|
3162
|
-
onEnter: onEnterRoute,
|
3163
|
-
onExit: onExitRoute
|
3164
|
-
}));
|
3165
|
-
}
|
3166
|
-
routes.push(route);
|
3167
|
-
return this;
|
3168
|
-
}
|
3169
|
-
|
3170
|
-
/*
|
3171
|
-
* @ngdoc object
|
3172
|
-
* @name ui.router.compat.$route
|
3173
|
-
*
|
3174
|
-
* @requires ui.router.state.$state
|
3175
|
-
* @requires $rootScope
|
3176
|
-
* @requires $routeParams
|
3177
|
-
*
|
3178
|
-
* @property {object} routes - Array of registered routes.
|
3179
|
-
* @property {object} params - Current route params as object.
|
3180
|
-
* @property {string} current - Name of the current route.
|
3181
|
-
*
|
3182
|
-
* @description
|
3183
|
-
* The `$route` service provides interfaces to access defined routes. It also let's
|
3184
|
-
* you access route params through `$routeParams` service, so you have fully
|
3185
|
-
* control over all the stuff you would actually get from angular's core `$route`
|
3186
|
-
* service.
|
3187
|
-
*/
|
3188
|
-
this.$get = $get;
|
3189
|
-
$get.$inject = ['$state', '$rootScope', '$routeParams'];
|
3190
|
-
function $get( $state, $rootScope, $routeParams) {
|
3191
|
-
|
3192
|
-
var $route = {
|
3193
|
-
routes: routes,
|
3194
|
-
params: $routeParams,
|
3195
|
-
current: undefined
|
3196
|
-
};
|
3197
|
-
|
3198
|
-
function stateAsRoute(state) {
|
3199
|
-
return (state.name !== '') ? state : undefined;
|
3200
|
-
}
|
3201
|
-
|
3202
|
-
$rootScope.$on('$stateChangeStart', function (ev, to, toParams, from, fromParams) {
|
3203
|
-
$rootScope.$broadcast('$routeChangeStart', stateAsRoute(to), stateAsRoute(from));
|
3204
|
-
});
|
3205
|
-
|
3206
|
-
$rootScope.$on('$stateChangeSuccess', function (ev, to, toParams, from, fromParams) {
|
3207
|
-
$route.current = stateAsRoute(to);
|
3208
|
-
$rootScope.$broadcast('$routeChangeSuccess', stateAsRoute(to), stateAsRoute(from));
|
3209
|
-
copy(toParams, $route.params);
|
3210
|
-
});
|
3211
|
-
|
3212
|
-
$rootScope.$on('$stateChangeError', function (ev, to, toParams, from, fromParams, error) {
|
3213
|
-
$rootScope.$broadcast('$routeChangeError', stateAsRoute(to), stateAsRoute(from), error);
|
3214
|
-
});
|
3215
|
-
|
3216
|
-
return $route;
|
3217
|
-
}
|
3218
|
-
}
|
3219
|
-
|
3220
|
-
angular.module('ui.router.compat')
|
3221
|
-
.provider('$route', $RouteProvider)
|
3222
|
-
.directive('ngView', $ViewDirective);
|
3223
4370
|
})(window, window.angular);
|