praxis 0.19.0 → 0.20.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/.gitignore +5 -0
- data/CHANGELOG.md +16 -0
- data/lib/api_browser/Gruntfile.js +125 -16
- data/lib/api_browser/app/index.html +5 -1
- data/lib/api_browser/app/js/directives/fixed_if_fits.js +28 -17
- data/lib/api_browser/app/js/directives/highlight.js +14 -0
- data/lib/api_browser/app/js/directives/request_examples.js +29 -0
- data/lib/api_browser/app/js/factories/Documentation.js +6 -0
- data/lib/api_browser/app/js/factories/Example.js +47 -0
- data/lib/api_browser/app/js/factories/prepare_template.js +15 -0
- data/lib/api_browser/app/js/factories/template_for.js +2 -12
- data/lib/api_browser/app/sass/modules/_sidebar.scss +2 -0
- data/lib/api_browser/app/views/action.html +6 -24
- data/lib/api_browser/app/views/examples/general.html +26 -0
- data/lib/api_browser/{bower.json → bower_template.json} +13 -3
- data/lib/api_browser/package.json +3 -1
- data/lib/praxis/application.rb +2 -0
- data/lib/praxis/docs/generator.rb +4 -2
- data/lib/praxis/extensions/field_selection/field_selector.rb +1 -0
- data/lib/praxis/media_type.rb +1 -1
- data/lib/praxis/plugin.rb +4 -0
- data/lib/praxis/plugin_concern.rb +2 -1
- data/lib/praxis/tasks/api_docs.rb +9 -2
- data/lib/praxis/version.rb +1 -1
- data/praxis.gemspec +2 -2
- data/spec/praxis/action_definition_spec.rb +26 -1
- data/spec/praxis/extensions/field_selection/field_selector_spec.rb +4 -0
- data/spec/praxis/media_type_spec.rb +4 -3
- data/spec/support/spec_media_types.rb +5 -1
- metadata +12 -242
- data/lib/api_browser/app/bower_components/angular-mocks/.bower.json +0 -19
- data/lib/api_browser/app/bower_components/angular-mocks/README.md +0 -63
- data/lib/api_browser/app/bower_components/angular-mocks/angular-mocks.js +0 -2452
- data/lib/api_browser/app/bower_components/angular-mocks/bower.json +0 -9
- data/lib/api_browser/app/bower_components/angular-mocks/ngAnimateMock.js +0 -2
- data/lib/api_browser/app/bower_components/angular-mocks/ngMock.js +0 -2
- data/lib/api_browser/app/bower_components/angular-mocks/ngMockE2E.js +0 -2
- data/lib/api_browser/app/bower_components/angular-mocks/package.json +0 -27
- data/lib/api_browser/app/bower_components/angular-sanitize/.bower.json +0 -19
- data/lib/api_browser/app/bower_components/angular-sanitize/README.md +0 -68
- data/lib/api_browser/app/bower_components/angular-sanitize/angular-sanitize.js +0 -683
- data/lib/api_browser/app/bower_components/angular-sanitize/angular-sanitize.min.js +0 -16
- data/lib/api_browser/app/bower_components/angular-sanitize/angular-sanitize.min.js.map +0 -8
- data/lib/api_browser/app/bower_components/angular-sanitize/bower.json +0 -9
- data/lib/api_browser/app/bower_components/angular-sanitize/index.js +0 -2
- data/lib/api_browser/app/bower_components/angular-sanitize/package.json +0 -26
- data/lib/api_browser/app/bower_components/angular-ui-bootstrap-bower/.bower.json +0 -31
- data/lib/api_browser/app/bower_components/angular-ui-bootstrap-bower/bower.json +0 -19
- data/lib/api_browser/app/bower_components/angular-ui-bootstrap-bower/ui-bootstrap-csp.css +0 -6
- data/lib/api_browser/app/bower_components/angular-ui-bootstrap-bower/ui-bootstrap-tpls.js +0 -4840
- data/lib/api_browser/app/bower_components/angular-ui-bootstrap-bower/ui-bootstrap-tpls.min.js +0 -10
- data/lib/api_browser/app/bower_components/angular-ui-bootstrap-bower/ui-bootstrap.js +0 -4461
- data/lib/api_browser/app/bower_components/angular-ui-bootstrap-bower/ui-bootstrap.min.js +0 -9
- data/lib/api_browser/app/bower_components/angular-ui-router/.bower.json +0 -33
- data/lib/api_browser/app/bower_components/angular-ui-router/CHANGELOG.md +0 -228
- data/lib/api_browser/app/bower_components/angular-ui-router/CONTRIBUTING.md +0 -65
- data/lib/api_browser/app/bower_components/angular-ui-router/LICENSE +0 -21
- data/lib/api_browser/app/bower_components/angular-ui-router/README.md +0 -245
- data/lib/api_browser/app/bower_components/angular-ui-router/api/angular-ui-router.d.ts +0 -126
- data/lib/api_browser/app/bower_components/angular-ui-router/bower.json +0 -23
- data/lib/api_browser/app/bower_components/angular-ui-router/release/angular-ui-router.js +0 -4370
- data/lib/api_browser/app/bower_components/angular-ui-router/release/angular-ui-router.min.js +0 -7
- data/lib/api_browser/app/bower_components/angular-ui-router/src/common.js +0 -292
- data/lib/api_browser/app/bower_components/angular-ui-router/src/resolve.js +0 -252
- data/lib/api_browser/app/bower_components/angular-ui-router/src/state.js +0 -1465
- data/lib/api_browser/app/bower_components/angular-ui-router/src/stateDirectives.js +0 -285
- data/lib/api_browser/app/bower_components/angular-ui-router/src/stateFilters.js +0 -39
- data/lib/api_browser/app/bower_components/angular-ui-router/src/templateFactory.js +0 -110
- data/lib/api_browser/app/bower_components/angular-ui-router/src/urlMatcherFactory.js +0 -1050
- data/lib/api_browser/app/bower_components/angular-ui-router/src/urlRouter.js +0 -427
- data/lib/api_browser/app/bower_components/angular-ui-router/src/view.js +0 -71
- data/lib/api_browser/app/bower_components/angular-ui-router/src/viewDirective.js +0 -303
- data/lib/api_browser/app/bower_components/angular-ui-router/src/viewScroll.js +0 -52
- data/lib/api_browser/app/bower_components/angular/.bower.json +0 -17
- data/lib/api_browser/app/bower_components/angular/README.md +0 -64
- data/lib/api_browser/app/bower_components/angular/angular-csp.css +0 -21
- data/lib/api_browser/app/bower_components/angular/angular.js +0 -28133
- data/lib/api_browser/app/bower_components/angular/angular.min.js +0 -289
- 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 +0 -8
- data/lib/api_browser/app/bower_components/angular/bower.json +0 -8
- data/lib/api_browser/app/bower_components/angular/index.js +0 -2
- data/lib/api_browser/app/bower_components/angular/package.json +0 -25
- data/lib/api_browser/app/bower_components/bootstrap-sass/.bower.json +0 -41
- data/lib/api_browser/app/bower_components/bootstrap-sass/CHANGELOG.md +0 -108
- data/lib/api_browser/app/bower_components/bootstrap-sass/CONTRIBUTING.md +0 -79
- data/lib/api_browser/app/bower_components/bootstrap-sass/README.md +0 -218
- data/lib/api_browser/app/bower_components/bootstrap-sass/bower.json +0 -22
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/fonts/bootstrap/glyphicons-halflings-regular.eot +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/fonts/bootstrap/glyphicons-halflings-regular.svg +0 -229
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/fonts/bootstrap/glyphicons-halflings-regular.ttf +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/fonts/bootstrap/glyphicons-halflings-regular.woff +0 -0
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/javascripts/bootstrap.js +0 -12
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/javascripts/bootstrap/affix.js +0 -126
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/javascripts/bootstrap/alert.js +0 -98
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/javascripts/bootstrap/button.js +0 -115
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/javascripts/bootstrap/carousel.js +0 -217
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/javascripts/bootstrap/collapse.js +0 -179
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/javascripts/bootstrap/dropdown.js +0 -154
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/javascripts/bootstrap/modal.js +0 -246
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/javascripts/bootstrap/popover.js +0 -117
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/javascripts/bootstrap/scrollspy.js +0 -158
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/javascripts/bootstrap/tab.js +0 -135
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/javascripts/bootstrap/tooltip.js +0 -386
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/javascripts/bootstrap/transition.js +0 -56
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap.scss +0 -1
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_alerts.scss +0 -67
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_badges.scss +0 -51
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_breadcrumbs.scss +0 -23
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_button-groups.scss +0 -227
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_buttons.scss +0 -155
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_carousel.scss +0 -232
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_close.scss +0 -35
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_code.scss +0 -53
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_component-animations.scss +0 -29
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_dropdowns.scss +0 -188
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_forms.scss +0 -374
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_glyphicons.scss +0 -237
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_grid.scss +0 -79
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_input-groups.scss +0 -136
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_jumbotron.scss +0 -46
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_labels.scss +0 -64
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_list-group.scss +0 -88
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_media.scss +0 -56
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_mixins.scss +0 -848
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_modals.scss +0 -129
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_navbar.scss +0 -616
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_navs.scss +0 -242
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_normalize.scss +0 -406
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_pager.scss +0 -55
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_pagination.scss +0 -85
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_panels.scss +0 -182
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_popovers.scss +0 -133
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_print.scss +0 -105
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_progress-bars.scss +0 -80
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_responsive-utilities.scss +0 -198
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_scaffolding.scss +0 -119
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_tables.scss +0 -231
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_theme.scss +0 -247
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_thumbnails.scss +0 -38
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_tooltip.scss +0 -95
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_type.scss +0 -281
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_utilities.scss +0 -56
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_variables.scss +0 -646
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/_wells.scss +0 -29
- data/lib/api_browser/app/bower_components/bootstrap-sass/vendor/assets/stylesheets/bootstrap/bootstrap.scss +0 -49
- data/lib/api_browser/app/bower_components/jquery/.bower.json +0 -38
- data/lib/api_browser/app/bower_components/jquery/MIT-LICENSE.txt +0 -21
- data/lib/api_browser/app/bower_components/jquery/bower.json +0 -27
- data/lib/api_browser/app/bower_components/jquery/dist/jquery.js +0 -9190
- data/lib/api_browser/app/bower_components/jquery/dist/jquery.min.js +0 -5
- data/lib/api_browser/app/bower_components/jquery/dist/jquery.min.map +0 -1
- data/lib/api_browser/app/bower_components/jquery/src/ajax.js +0 -806
- data/lib/api_browser/app/bower_components/jquery/src/ajax/jsonp.js +0 -89
- data/lib/api_browser/app/bower_components/jquery/src/ajax/load.js +0 -75
- data/lib/api_browser/app/bower_components/jquery/src/ajax/parseJSON.js +0 -13
- data/lib/api_browser/app/bower_components/jquery/src/ajax/parseXML.js +0 -28
- data/lib/api_browser/app/bower_components/jquery/src/ajax/script.js +0 -64
- data/lib/api_browser/app/bower_components/jquery/src/ajax/var/nonce.js +0 -5
- data/lib/api_browser/app/bower_components/jquery/src/ajax/var/rquery.js +0 -3
- data/lib/api_browser/app/bower_components/jquery/src/ajax/xhr.js +0 -135
- data/lib/api_browser/app/bower_components/jquery/src/attributes.js +0 -11
- data/lib/api_browser/app/bower_components/jquery/src/attributes/attr.js +0 -143
- data/lib/api_browser/app/bower_components/jquery/src/attributes/classes.js +0 -158
- data/lib/api_browser/app/bower_components/jquery/src/attributes/prop.js +0 -96
- data/lib/api_browser/app/bower_components/jquery/src/attributes/support.js +0 -35
- data/lib/api_browser/app/bower_components/jquery/src/attributes/val.js +0 -163
- data/lib/api_browser/app/bower_components/jquery/src/callbacks.js +0 -205
- data/lib/api_browser/app/bower_components/jquery/src/core.js +0 -498
- data/lib/api_browser/app/bower_components/jquery/src/core/access.js +0 -60
- data/lib/api_browser/app/bower_components/jquery/src/core/init.js +0 -123
- data/lib/api_browser/app/bower_components/jquery/src/core/parseHTML.js +0 -39
- data/lib/api_browser/app/bower_components/jquery/src/core/ready.js +0 -97
- data/lib/api_browser/app/bower_components/jquery/src/core/var/rsingleTag.js +0 -4
- data/lib/api_browser/app/bower_components/jquery/src/css.js +0 -451
- data/lib/api_browser/app/bower_components/jquery/src/css/addGetHookIf.js +0 -24
- data/lib/api_browser/app/bower_components/jquery/src/css/curCSS.js +0 -57
- data/lib/api_browser/app/bower_components/jquery/src/css/defaultDisplay.js +0 -70
- data/lib/api_browser/app/bower_components/jquery/src/css/hiddenVisibleSelectors.js +0 -15
- data/lib/api_browser/app/bower_components/jquery/src/css/support.js +0 -91
- data/lib/api_browser/app/bower_components/jquery/src/css/swap.js +0 -28
- data/lib/api_browser/app/bower_components/jquery/src/css/var/cssExpand.js +0 -3
- data/lib/api_browser/app/bower_components/jquery/src/css/var/getStyles.js +0 -5
- data/lib/api_browser/app/bower_components/jquery/src/css/var/isHidden.js +0 -13
- data/lib/api_browser/app/bower_components/jquery/src/css/var/rmargin.js +0 -3
- data/lib/api_browser/app/bower_components/jquery/src/css/var/rnumnonpx.js +0 -5
- data/lib/api_browser/app/bower_components/jquery/src/data.js +0 -179
- data/lib/api_browser/app/bower_components/jquery/src/data/Data.js +0 -181
- data/lib/api_browser/app/bower_components/jquery/src/data/accepts.js +0 -20
- data/lib/api_browser/app/bower_components/jquery/src/data/var/data_priv.js +0 -5
- data/lib/api_browser/app/bower_components/jquery/src/data/var/data_user.js +0 -5
- data/lib/api_browser/app/bower_components/jquery/src/deferred.js +0 -149
- data/lib/api_browser/app/bower_components/jquery/src/deprecated.js +0 -13
- data/lib/api_browser/app/bower_components/jquery/src/dimensions.js +0 -50
- data/lib/api_browser/app/bower_components/jquery/src/effects.js +0 -649
- data/lib/api_browser/app/bower_components/jquery/src/effects/Tween.js +0 -114
- data/lib/api_browser/app/bower_components/jquery/src/effects/animatedSelector.js +0 -13
- data/lib/api_browser/app/bower_components/jquery/src/event.js +0 -868
- data/lib/api_browser/app/bower_components/jquery/src/event/alias.js +0 -39
- data/lib/api_browser/app/bower_components/jquery/src/event/support.js +0 -9
- data/lib/api_browser/app/bower_components/jquery/src/exports/amd.js +0 -24
- data/lib/api_browser/app/bower_components/jquery/src/exports/global.js +0 -32
- data/lib/api_browser/app/bower_components/jquery/src/intro.js +0 -44
- data/lib/api_browser/app/bower_components/jquery/src/jquery.js +0 -36
- data/lib/api_browser/app/bower_components/jquery/src/manipulation.js +0 -582
- data/lib/api_browser/app/bower_components/jquery/src/manipulation/_evalUrl.js +0 -18
- data/lib/api_browser/app/bower_components/jquery/src/manipulation/support.js +0 -31
- data/lib/api_browser/app/bower_components/jquery/src/manipulation/var/rcheckableType.js +0 -3
- data/lib/api_browser/app/bower_components/jquery/src/offset.js +0 -204
- data/lib/api_browser/app/bower_components/jquery/src/outro.js +0 -1
- data/lib/api_browser/app/bower_components/jquery/src/queue.js +0 -142
- data/lib/api_browser/app/bower_components/jquery/src/queue/delay.js +0 -22
- data/lib/api_browser/app/bower_components/jquery/src/selector-native.js +0 -172
- data/lib/api_browser/app/bower_components/jquery/src/selector-sizzle.js +0 -14
- data/lib/api_browser/app/bower_components/jquery/src/selector.js +0 -1
- data/lib/api_browser/app/bower_components/jquery/src/serialize.js +0 -111
- data/lib/api_browser/app/bower_components/jquery/src/sizzle/dist/sizzle.js +0 -2044
- data/lib/api_browser/app/bower_components/jquery/src/sizzle/dist/sizzle.min.js +0 -3
- data/lib/api_browser/app/bower_components/jquery/src/sizzle/dist/sizzle.min.map +0 -1
- data/lib/api_browser/app/bower_components/jquery/src/traversing.js +0 -200
- data/lib/api_browser/app/bower_components/jquery/src/traversing/findFilter.js +0 -100
- data/lib/api_browser/app/bower_components/jquery/src/traversing/var/rneedsContext.js +0 -6
- data/lib/api_browser/app/bower_components/jquery/src/var/arr.js +0 -3
- data/lib/api_browser/app/bower_components/jquery/src/var/class2type.js +0 -4
- data/lib/api_browser/app/bower_components/jquery/src/var/concat.js +0 -5
- data/lib/api_browser/app/bower_components/jquery/src/var/hasOwn.js +0 -5
- data/lib/api_browser/app/bower_components/jquery/src/var/indexOf.js +0 -5
- data/lib/api_browser/app/bower_components/jquery/src/var/pnum.js +0 -3
- data/lib/api_browser/app/bower_components/jquery/src/var/push.js +0 -5
- data/lib/api_browser/app/bower_components/jquery/src/var/rnotwhite.js +0 -3
- data/lib/api_browser/app/bower_components/jquery/src/var/slice.js +0 -5
- data/lib/api_browser/app/bower_components/jquery/src/var/strundefined.js +0 -3
- data/lib/api_browser/app/bower_components/jquery/src/var/support.js +0 -4
- data/lib/api_browser/app/bower_components/jquery/src/var/toString.js +0 -5
- data/lib/api_browser/app/bower_components/jquery/src/wrap.js +0 -78
- data/lib/api_browser/app/bower_components/lodash/.bower.json +0 -30
- data/lib/api_browser/app/bower_components/lodash/LICENSE.txt +0 -22
- data/lib/api_browser/app/bower_components/lodash/bower.json +0 -20
- data/lib/api_browser/app/bower_components/lodash/lodash.js +0 -12235
- data/lib/api_browser/app/bower_components/lodash/lodash.min.js +0 -98
- data/lib/api_browser/app/bower_components/showdown/.bower.json +0 -39
- data/lib/api_browser/app/bower_components/showdown/.jshintignore +0 -2
- data/lib/api_browser/app/bower_components/showdown/.travis.yml +0 -8
- data/lib/api_browser/app/bower_components/showdown/Gruntfile.js +0 -100
- data/lib/api_browser/app/bower_components/showdown/README.md +0 -317
- data/lib/api_browser/app/bower_components/showdown/bower.json +0 -26
- data/lib/api_browser/app/bower_components/showdown/compressed/Showdown.js +0 -1606
- data/lib/api_browser/app/bower_components/showdown/compressed/Showdown.js.map +0 -1
- data/lib/api_browser/app/bower_components/showdown/compressed/Showdown.min.js +0 -2
- data/lib/api_browser/app/bower_components/showdown/compressed/extensions/github.min.js +0 -2
- data/lib/api_browser/app/bower_components/showdown/compressed/extensions/github.min.js.map +0 -1
- data/lib/api_browser/app/bower_components/showdown/compressed/extensions/prettify.min.js +0 -2
- data/lib/api_browser/app/bower_components/showdown/compressed/extensions/prettify.min.js.map +0 -1
- data/lib/api_browser/app/bower_components/showdown/compressed/extensions/table.min.js +0 -2
- data/lib/api_browser/app/bower_components/showdown/compressed/extensions/table.min.js.map +0 -1
- data/lib/api_browser/app/bower_components/showdown/compressed/extensions/twitter.min.js +0 -2
- data/lib/api_browser/app/bower_components/showdown/compressed/extensions/twitter.min.js.map +0 -1
- data/lib/api_browser/app/bower_components/showdown/license.txt +0 -34
- data/lib/api_browser/app/bower_components/showdown/package.json +0 -47
- data/lib/api_browser/app/bower_components/showdown/src/extensions/github.js +0 -25
- data/lib/api_browser/app/bower_components/showdown/src/extensions/prettify.js +0 -29
- data/lib/api_browser/app/bower_components/showdown/src/extensions/table.js +0 -106
- data/lib/api_browser/app/bower_components/showdown/src/extensions/twitter.js +0 -42
- data/lib/api_browser/app/bower_components/showdown/src/ng-showdown.js +0 -150
- data/lib/api_browser/app/bower_components/showdown/src/showdown.js +0 -1454
@@ -1,285 +0,0 @@
|
|
1
|
-
function parseStateRef(ref, current) {
|
2
|
-
var preparsed = ref.match(/^\s*({[^}]*})\s*$/), parsed;
|
3
|
-
if (preparsed) ref = current + '(' + preparsed[1] + ')';
|
4
|
-
parsed = ref.replace(/\n/g, " ").match(/^([^(]+?)\s*(\((.*)\))?$/);
|
5
|
-
if (!parsed || parsed.length !== 4) throw new Error("Invalid state ref '" + ref + "'");
|
6
|
-
return { state: parsed[1], paramExpr: parsed[3] || null };
|
7
|
-
}
|
8
|
-
|
9
|
-
function stateContext(el) {
|
10
|
-
var stateData = el.parent().inheritedData('$uiView');
|
11
|
-
|
12
|
-
if (stateData && stateData.state && stateData.state.name) {
|
13
|
-
return stateData.state;
|
14
|
-
}
|
15
|
-
}
|
16
|
-
|
17
|
-
/**
|
18
|
-
* @ngdoc directive
|
19
|
-
* @name ui.router.state.directive:ui-sref
|
20
|
-
*
|
21
|
-
* @requires ui.router.state.$state
|
22
|
-
* @requires $timeout
|
23
|
-
*
|
24
|
-
* @restrict A
|
25
|
-
*
|
26
|
-
* @description
|
27
|
-
* A directive that binds a link (`<a>` tag) to a state. If the state has an associated
|
28
|
-
* URL, the directive will automatically generate & update the `href` attribute via
|
29
|
-
* the {@link ui.router.state.$state#methods_href $state.href()} method. Clicking
|
30
|
-
* the link will trigger a state transition with optional parameters.
|
31
|
-
*
|
32
|
-
* Also middle-clicking, right-clicking, and ctrl-clicking on the link will be
|
33
|
-
* handled natively by the browser.
|
34
|
-
*
|
35
|
-
* You can also use relative state paths within ui-sref, just like the relative
|
36
|
-
* paths passed to `$state.go()`. You just need to be aware that the path is relative
|
37
|
-
* to the state that the link lives in, in other words the state that loaded the
|
38
|
-
* template containing the link.
|
39
|
-
*
|
40
|
-
* You can specify options to pass to {@link ui.router.state.$state#go $state.go()}
|
41
|
-
* using the `ui-sref-opts` attribute. Options are restricted to `location`, `inherit`,
|
42
|
-
* and `reload`.
|
43
|
-
*
|
44
|
-
* @example
|
45
|
-
* Here's an example of how you'd use ui-sref and how it would compile. If you have the
|
46
|
-
* following template:
|
47
|
-
* <pre>
|
48
|
-
* <a ui-sref="home">Home</a> | <a ui-sref="about">About</a> | <a ui-sref="{page: 2}">Next page</a>
|
49
|
-
*
|
50
|
-
* <ul>
|
51
|
-
* <li ng-repeat="contact in contacts">
|
52
|
-
* <a ui-sref="contacts.detail({ id: contact.id })">{{ contact.name }}</a>
|
53
|
-
* </li>
|
54
|
-
* </ul>
|
55
|
-
* </pre>
|
56
|
-
*
|
57
|
-
* Then the compiled html would be (assuming Html5Mode is off and current state is contacts):
|
58
|
-
* <pre>
|
59
|
-
* <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>
|
60
|
-
*
|
61
|
-
* <ul>
|
62
|
-
* <li ng-repeat="contact in contacts">
|
63
|
-
* <a href="#/contacts/1" ui-sref="contacts.detail({ id: contact.id })">Joe</a>
|
64
|
-
* </li>
|
65
|
-
* <li ng-repeat="contact in contacts">
|
66
|
-
* <a href="#/contacts/2" ui-sref="contacts.detail({ id: contact.id })">Alice</a>
|
67
|
-
* </li>
|
68
|
-
* <li ng-repeat="contact in contacts">
|
69
|
-
* <a href="#/contacts/3" ui-sref="contacts.detail({ id: contact.id })">Bob</a>
|
70
|
-
* </li>
|
71
|
-
* </ul>
|
72
|
-
*
|
73
|
-
* <a ui-sref="home" ui-sref-opts="{reload: true}">Home</a>
|
74
|
-
* </pre>
|
75
|
-
*
|
76
|
-
* @param {string} ui-sref 'stateName' can be any valid absolute or relative state
|
77
|
-
* @param {Object} ui-sref-opts options to pass to {@link ui.router.state.$state#go $state.go()}
|
78
|
-
*/
|
79
|
-
$StateRefDirective.$inject = ['$state', '$timeout'];
|
80
|
-
function $StateRefDirective($state, $timeout) {
|
81
|
-
var allowedOptions = ['location', 'inherit', 'reload', 'absolute'];
|
82
|
-
|
83
|
-
return {
|
84
|
-
restrict: 'A',
|
85
|
-
require: ['?^uiSrefActive', '?^uiSrefActiveEq'],
|
86
|
-
link: function(scope, element, attrs, uiSrefActive) {
|
87
|
-
var ref = parseStateRef(attrs.uiSref, $state.current.name);
|
88
|
-
var params = null, url = null, base = stateContext(element) || $state.$current;
|
89
|
-
// SVGAElement does not use the href attribute, but rather the 'xlinkHref' attribute.
|
90
|
-
var hrefKind = Object.prototype.toString.call(element.prop('href')) === '[object SVGAnimatedString]' ?
|
91
|
-
'xlink:href' : 'href';
|
92
|
-
var newHref = null, isAnchor = element.prop("tagName").toUpperCase() === "A";
|
93
|
-
var isForm = element[0].nodeName === "FORM";
|
94
|
-
var attr = isForm ? "action" : hrefKind, nav = true;
|
95
|
-
|
96
|
-
var options = { relative: base, inherit: true };
|
97
|
-
var optionsOverride = scope.$eval(attrs.uiSrefOpts) || {};
|
98
|
-
|
99
|
-
angular.forEach(allowedOptions, function(option) {
|
100
|
-
if (option in optionsOverride) {
|
101
|
-
options[option] = optionsOverride[option];
|
102
|
-
}
|
103
|
-
});
|
104
|
-
|
105
|
-
var update = function(newVal) {
|
106
|
-
if (newVal) params = angular.copy(newVal);
|
107
|
-
if (!nav) return;
|
108
|
-
|
109
|
-
newHref = $state.href(ref.state, params, options);
|
110
|
-
|
111
|
-
var activeDirective = uiSrefActive[1] || uiSrefActive[0];
|
112
|
-
if (activeDirective) {
|
113
|
-
activeDirective.$$addStateInfo(ref.state, params);
|
114
|
-
}
|
115
|
-
if (newHref === null) {
|
116
|
-
nav = false;
|
117
|
-
return false;
|
118
|
-
}
|
119
|
-
attrs.$set(attr, newHref);
|
120
|
-
};
|
121
|
-
|
122
|
-
if (ref.paramExpr) {
|
123
|
-
scope.$watch(ref.paramExpr, function(newVal, oldVal) {
|
124
|
-
if (newVal !== params) update(newVal);
|
125
|
-
}, true);
|
126
|
-
params = angular.copy(scope.$eval(ref.paramExpr));
|
127
|
-
}
|
128
|
-
update();
|
129
|
-
|
130
|
-
if (isForm) return;
|
131
|
-
|
132
|
-
element.bind("click", function(e) {
|
133
|
-
var button = e.which || e.button;
|
134
|
-
if ( !(button > 1 || e.ctrlKey || e.metaKey || e.shiftKey || element.attr('target')) ) {
|
135
|
-
// HACK: This is to allow ng-clicks to be processed before the transition is initiated:
|
136
|
-
var transition = $timeout(function() {
|
137
|
-
$state.go(ref.state, params, options);
|
138
|
-
});
|
139
|
-
e.preventDefault();
|
140
|
-
|
141
|
-
// if the state has no URL, ignore one preventDefault from the <a> directive.
|
142
|
-
var ignorePreventDefaultCount = isAnchor && !newHref ? 1: 0;
|
143
|
-
e.preventDefault = function() {
|
144
|
-
if (ignorePreventDefaultCount-- <= 0)
|
145
|
-
$timeout.cancel(transition);
|
146
|
-
};
|
147
|
-
}
|
148
|
-
});
|
149
|
-
}
|
150
|
-
};
|
151
|
-
}
|
152
|
-
|
153
|
-
/**
|
154
|
-
* @ngdoc directive
|
155
|
-
* @name ui.router.state.directive:ui-sref-active
|
156
|
-
*
|
157
|
-
* @requires ui.router.state.$state
|
158
|
-
* @requires ui.router.state.$stateParams
|
159
|
-
* @requires $interpolate
|
160
|
-
*
|
161
|
-
* @restrict A
|
162
|
-
*
|
163
|
-
* @description
|
164
|
-
* A directive working alongside ui-sref to add classes to an element when the
|
165
|
-
* related ui-sref directive's state is active, and removing them when it is inactive.
|
166
|
-
* The primary use-case is to simplify the special appearance of navigation menus
|
167
|
-
* relying on `ui-sref`, by having the "active" state's menu button appear different,
|
168
|
-
* distinguishing it from the inactive menu items.
|
169
|
-
*
|
170
|
-
* ui-sref-active can live on the same element as ui-sref or on a parent element. The first
|
171
|
-
* ui-sref-active found at the same level or above the ui-sref will be used.
|
172
|
-
*
|
173
|
-
* Will activate when the ui-sref's target state or any child state is active. If you
|
174
|
-
* need to activate only when the ui-sref target state is active and *not* any of
|
175
|
-
* it's children, then you will use
|
176
|
-
* {@link ui.router.state.directive:ui-sref-active-eq ui-sref-active-eq}
|
177
|
-
*
|
178
|
-
* @example
|
179
|
-
* Given the following template:
|
180
|
-
* <pre>
|
181
|
-
* <ul>
|
182
|
-
* <li ui-sref-active="active" class="item">
|
183
|
-
* <a href ui-sref="app.user({user: 'bilbobaggins'})">@bilbobaggins</a>
|
184
|
-
* </li>
|
185
|
-
* </ul>
|
186
|
-
* </pre>
|
187
|
-
*
|
188
|
-
*
|
189
|
-
* When the app state is "app.user" (or any children states), and contains the state parameter "user" with value "bilbobaggins",
|
190
|
-
* the resulting HTML will appear as (note the 'active' class):
|
191
|
-
* <pre>
|
192
|
-
* <ul>
|
193
|
-
* <li ui-sref-active="active" class="item active">
|
194
|
-
* <a ui-sref="app.user({user: 'bilbobaggins'})" href="/users/bilbobaggins">@bilbobaggins</a>
|
195
|
-
* </li>
|
196
|
-
* </ul>
|
197
|
-
* </pre>
|
198
|
-
*
|
199
|
-
* The class name is interpolated **once** during the directives link time (any further changes to the
|
200
|
-
* interpolated value are ignored).
|
201
|
-
*
|
202
|
-
* Multiple classes may be specified in a space-separated format:
|
203
|
-
* <pre>
|
204
|
-
* <ul>
|
205
|
-
* <li ui-sref-active='class1 class2 class3'>
|
206
|
-
* <a ui-sref="app.user">link</a>
|
207
|
-
* </li>
|
208
|
-
* </ul>
|
209
|
-
* </pre>
|
210
|
-
*/
|
211
|
-
|
212
|
-
/**
|
213
|
-
* @ngdoc directive
|
214
|
-
* @name ui.router.state.directive:ui-sref-active-eq
|
215
|
-
*
|
216
|
-
* @requires ui.router.state.$state
|
217
|
-
* @requires ui.router.state.$stateParams
|
218
|
-
* @requires $interpolate
|
219
|
-
*
|
220
|
-
* @restrict A
|
221
|
-
*
|
222
|
-
* @description
|
223
|
-
* The same as {@link ui.router.state.directive:ui-sref-active ui-sref-active} but will only activate
|
224
|
-
* when the exact target state used in the `ui-sref` is active; no child states.
|
225
|
-
*
|
226
|
-
*/
|
227
|
-
$StateRefActiveDirective.$inject = ['$state', '$stateParams', '$interpolate'];
|
228
|
-
function $StateRefActiveDirective($state, $stateParams, $interpolate) {
|
229
|
-
return {
|
230
|
-
restrict: "A",
|
231
|
-
controller: ['$scope', '$element', '$attrs', function ($scope, $element, $attrs) {
|
232
|
-
var states = [], activeClass;
|
233
|
-
|
234
|
-
// There probably isn't much point in $observing this
|
235
|
-
// uiSrefActive and uiSrefActiveEq share the same directive object with some
|
236
|
-
// slight difference in logic routing
|
237
|
-
activeClass = $interpolate($attrs.uiSrefActiveEq || $attrs.uiSrefActive || '', false)($scope);
|
238
|
-
|
239
|
-
// Allow uiSref to communicate with uiSrefActive[Equals]
|
240
|
-
this.$$addStateInfo = function (newState, newParams) {
|
241
|
-
var state = $state.get(newState, stateContext($element));
|
242
|
-
|
243
|
-
states.push({
|
244
|
-
state: state || { name: newState },
|
245
|
-
params: newParams
|
246
|
-
});
|
247
|
-
|
248
|
-
update();
|
249
|
-
};
|
250
|
-
|
251
|
-
$scope.$on('$stateChangeSuccess', update);
|
252
|
-
|
253
|
-
// Update route state
|
254
|
-
function update() {
|
255
|
-
if (anyMatch()) {
|
256
|
-
$element.addClass(activeClass);
|
257
|
-
} else {
|
258
|
-
$element.removeClass(activeClass);
|
259
|
-
}
|
260
|
-
}
|
261
|
-
|
262
|
-
function anyMatch() {
|
263
|
-
for (var i = 0; i < states.length; i++) {
|
264
|
-
if (isMatch(states[i].state, states[i].params)) {
|
265
|
-
return true;
|
266
|
-
}
|
267
|
-
}
|
268
|
-
return false;
|
269
|
-
}
|
270
|
-
|
271
|
-
function isMatch(state, params) {
|
272
|
-
if (typeof $attrs.uiSrefActiveEq !== 'undefined') {
|
273
|
-
return $state.is(state.name, params);
|
274
|
-
} else {
|
275
|
-
return $state.includes(state.name, params);
|
276
|
-
}
|
277
|
-
}
|
278
|
-
}]
|
279
|
-
};
|
280
|
-
}
|
281
|
-
|
282
|
-
angular.module('ui.router.state')
|
283
|
-
.directive('uiSref', $StateRefDirective)
|
284
|
-
.directive('uiSrefActive', $StateRefActiveDirective)
|
285
|
-
.directive('uiSrefActiveEq', $StateRefActiveDirective);
|
@@ -1,39 +0,0 @@
|
|
1
|
-
/**
|
2
|
-
* @ngdoc filter
|
3
|
-
* @name ui.router.state.filter:isState
|
4
|
-
*
|
5
|
-
* @requires ui.router.state.$state
|
6
|
-
*
|
7
|
-
* @description
|
8
|
-
* Translates to {@link ui.router.state.$state#methods_is $state.is("stateName")}.
|
9
|
-
*/
|
10
|
-
$IsStateFilter.$inject = ['$state'];
|
11
|
-
function $IsStateFilter($state) {
|
12
|
-
var isFilter = function (state) {
|
13
|
-
return $state.is(state);
|
14
|
-
};
|
15
|
-
isFilter.$stateful = true;
|
16
|
-
return isFilter;
|
17
|
-
}
|
18
|
-
|
19
|
-
/**
|
20
|
-
* @ngdoc filter
|
21
|
-
* @name ui.router.state.filter:includedByState
|
22
|
-
*
|
23
|
-
* @requires ui.router.state.$state
|
24
|
-
*
|
25
|
-
* @description
|
26
|
-
* Translates to {@link ui.router.state.$state#methods_includes $state.includes('fullOrPartialStateName')}.
|
27
|
-
*/
|
28
|
-
$IncludedByStateFilter.$inject = ['$state'];
|
29
|
-
function $IncludedByStateFilter($state) {
|
30
|
-
var includesFilter = function (state) {
|
31
|
-
return $state.includes(state);
|
32
|
-
};
|
33
|
-
includesFilter.$stateful = true;
|
34
|
-
return includesFilter;
|
35
|
-
}
|
36
|
-
|
37
|
-
angular.module('ui.router.state')
|
38
|
-
.filter('isState', $IsStateFilter)
|
39
|
-
.filter('includedByState', $IncludedByStateFilter);
|
@@ -1,110 +0,0 @@
|
|
1
|
-
/**
|
2
|
-
* @ngdoc object
|
3
|
-
* @name ui.router.util.$templateFactory
|
4
|
-
*
|
5
|
-
* @requires $http
|
6
|
-
* @requires $templateCache
|
7
|
-
* @requires $injector
|
8
|
-
*
|
9
|
-
* @description
|
10
|
-
* Service. Manages loading of templates.
|
11
|
-
*/
|
12
|
-
$TemplateFactory.$inject = ['$http', '$templateCache', '$injector'];
|
13
|
-
function $TemplateFactory( $http, $templateCache, $injector) {
|
14
|
-
|
15
|
-
/**
|
16
|
-
* @ngdoc function
|
17
|
-
* @name ui.router.util.$templateFactory#fromConfig
|
18
|
-
* @methodOf ui.router.util.$templateFactory
|
19
|
-
*
|
20
|
-
* @description
|
21
|
-
* Creates a template from a configuration object.
|
22
|
-
*
|
23
|
-
* @param {object} config Configuration object for which to load a template.
|
24
|
-
* The following properties are search in the specified order, and the first one
|
25
|
-
* that is defined is used to create the template:
|
26
|
-
*
|
27
|
-
* @param {string|object} config.template html string template or function to
|
28
|
-
* load via {@link ui.router.util.$templateFactory#fromString fromString}.
|
29
|
-
* @param {string|object} config.templateUrl url to load or a function returning
|
30
|
-
* the url to load via {@link ui.router.util.$templateFactory#fromUrl fromUrl}.
|
31
|
-
* @param {Function} config.templateProvider function to invoke via
|
32
|
-
* {@link ui.router.util.$templateFactory#fromProvider fromProvider}.
|
33
|
-
* @param {object} params Parameters to pass to the template function.
|
34
|
-
* @param {object} locals Locals to pass to `invoke` if the template is loaded
|
35
|
-
* via a `templateProvider`. Defaults to `{ params: params }`.
|
36
|
-
*
|
37
|
-
* @return {string|object} The template html as a string, or a promise for
|
38
|
-
* that string,or `null` if no template is configured.
|
39
|
-
*/
|
40
|
-
this.fromConfig = function (config, params, locals) {
|
41
|
-
return (
|
42
|
-
isDefined(config.template) ? this.fromString(config.template, params) :
|
43
|
-
isDefined(config.templateUrl) ? this.fromUrl(config.templateUrl, params) :
|
44
|
-
isDefined(config.templateProvider) ? this.fromProvider(config.templateProvider, params, locals) :
|
45
|
-
null
|
46
|
-
);
|
47
|
-
};
|
48
|
-
|
49
|
-
/**
|
50
|
-
* @ngdoc function
|
51
|
-
* @name ui.router.util.$templateFactory#fromString
|
52
|
-
* @methodOf ui.router.util.$templateFactory
|
53
|
-
*
|
54
|
-
* @description
|
55
|
-
* Creates a template from a string or a function returning a string.
|
56
|
-
*
|
57
|
-
* @param {string|object} template html template as a string or function that
|
58
|
-
* returns an html template as a string.
|
59
|
-
* @param {object} params Parameters to pass to the template function.
|
60
|
-
*
|
61
|
-
* @return {string|object} The template html as a string, or a promise for that
|
62
|
-
* string.
|
63
|
-
*/
|
64
|
-
this.fromString = function (template, params) {
|
65
|
-
return isFunction(template) ? template(params) : template;
|
66
|
-
};
|
67
|
-
|
68
|
-
/**
|
69
|
-
* @ngdoc function
|
70
|
-
* @name ui.router.util.$templateFactory#fromUrl
|
71
|
-
* @methodOf ui.router.util.$templateFactory
|
72
|
-
*
|
73
|
-
* @description
|
74
|
-
* Loads a template from the a URL via `$http` and `$templateCache`.
|
75
|
-
*
|
76
|
-
* @param {string|Function} url url of the template to load, or a function
|
77
|
-
* that returns a url.
|
78
|
-
* @param {Object} params Parameters to pass to the url function.
|
79
|
-
* @return {string|Promise.<string>} The template html as a string, or a promise
|
80
|
-
* for that string.
|
81
|
-
*/
|
82
|
-
this.fromUrl = function (url, params) {
|
83
|
-
if (isFunction(url)) url = url(params);
|
84
|
-
if (url == null) return null;
|
85
|
-
else return $http
|
86
|
-
.get(url, { cache: $templateCache, headers: { Accept: 'text/html' }})
|
87
|
-
.then(function(response) { return response.data; });
|
88
|
-
};
|
89
|
-
|
90
|
-
/**
|
91
|
-
* @ngdoc function
|
92
|
-
* @name ui.router.util.$templateFactory#fromProvider
|
93
|
-
* @methodOf ui.router.util.$templateFactory
|
94
|
-
*
|
95
|
-
* @description
|
96
|
-
* Creates a template by invoking an injectable provider function.
|
97
|
-
*
|
98
|
-
* @param {Function} provider Function to invoke via `$injector.invoke`
|
99
|
-
* @param {Object} params Parameters for the template.
|
100
|
-
* @param {Object} locals Locals to pass to `invoke`. Defaults to
|
101
|
-
* `{ params: params }`.
|
102
|
-
* @return {string|Promise.<string>} The template html as a string, or a promise
|
103
|
-
* for that string.
|
104
|
-
*/
|
105
|
-
this.fromProvider = function (provider, params, locals) {
|
106
|
-
return $injector.invoke(provider, null, locals || { params: params });
|
107
|
-
};
|
108
|
-
}
|
109
|
-
|
110
|
-
angular.module('ui.router.util').service('$templateFactory', $TemplateFactory);
|
@@ -1,1050 +0,0 @@
|
|
1
|
-
var $$UMFP; // reference to $UrlMatcherFactoryProvider
|
2
|
-
|
3
|
-
/**
|
4
|
-
* @ngdoc object
|
5
|
-
* @name ui.router.util.type:UrlMatcher
|
6
|
-
*
|
7
|
-
* @description
|
8
|
-
* Matches URLs against patterns and extracts named parameters from the path or the search
|
9
|
-
* part of the URL. A URL pattern consists of a path pattern, optionally followed by '?' and a list
|
10
|
-
* of search parameters. Multiple search parameter names are separated by '&'. Search parameters
|
11
|
-
* do not influence whether or not a URL is matched, but their values are passed through into
|
12
|
-
* the matched parameters returned by {@link ui.router.util.type:UrlMatcher#methods_exec exec}.
|
13
|
-
*
|
14
|
-
* Path parameter placeholders can be specified using simple colon/catch-all syntax or curly brace
|
15
|
-
* syntax, which optionally allows a regular expression for the parameter to be specified:
|
16
|
-
*
|
17
|
-
* * `':'` name - colon placeholder
|
18
|
-
* * `'*'` name - catch-all placeholder
|
19
|
-
* * `'{' name '}'` - curly placeholder
|
20
|
-
* * `'{' name ':' regexp|type '}'` - curly placeholder with regexp or type name. Should the
|
21
|
-
* regexp itself contain curly braces, they must be in matched pairs or escaped with a backslash.
|
22
|
-
*
|
23
|
-
* Parameter names may contain only word characters (latin letters, digits, and underscore) and
|
24
|
-
* must be unique within the pattern (across both path and search parameters). For colon
|
25
|
-
* placeholders or curly placeholders without an explicit regexp, a path parameter matches any
|
26
|
-
* number of characters other than '/'. For catch-all placeholders the path parameter matches
|
27
|
-
* any number of characters.
|
28
|
-
*
|
29
|
-
* Examples:
|
30
|
-
*
|
31
|
-
* * `'/hello/'` - Matches only if the path is exactly '/hello/'. There is no special treatment for
|
32
|
-
* trailing slashes, and patterns have to match the entire path, not just a prefix.
|
33
|
-
* * `'/user/:id'` - Matches '/user/bob' or '/user/1234!!!' or even '/user/' but not '/user' or
|
34
|
-
* '/user/bob/details'. The second path segment will be captured as the parameter 'id'.
|
35
|
-
* * `'/user/{id}'` - Same as the previous example, but using curly brace syntax.
|
36
|
-
* * `'/user/{id:[^/]*}'` - Same as the previous example.
|
37
|
-
* * `'/user/{id:[0-9a-fA-F]{1,8}}'` - Similar to the previous example, but only matches if the id
|
38
|
-
* parameter consists of 1 to 8 hex digits.
|
39
|
-
* * `'/files/{path:.*}'` - Matches any URL starting with '/files/' and captures the rest of the
|
40
|
-
* path into the parameter 'path'.
|
41
|
-
* * `'/files/*path'` - ditto.
|
42
|
-
* * `'/calendar/{start:date}'` - Matches "/calendar/2014-11-12" (because the pattern defined
|
43
|
-
* in the built-in `date` Type matches `2014-11-12`) and provides a Date object in $stateParams.start
|
44
|
-
*
|
45
|
-
* @param {string} pattern The pattern to compile into a matcher.
|
46
|
-
* @param {Object} config A configuration object hash:
|
47
|
-
* @param {Object=} parentMatcher Used to concatenate the pattern/config onto
|
48
|
-
* an existing UrlMatcher
|
49
|
-
*
|
50
|
-
* * `caseInsensitive` - `true` if URL matching should be case insensitive, otherwise `false`, the default value (for backward compatibility) is `false`.
|
51
|
-
* * `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`.
|
52
|
-
*
|
53
|
-
* @property {string} prefix A static prefix of this pattern. The matcher guarantees that any
|
54
|
-
* URL matching this matcher (i.e. any string for which {@link ui.router.util.type:UrlMatcher#methods_exec exec()} returns
|
55
|
-
* non-null) will start with this prefix.
|
56
|
-
*
|
57
|
-
* @property {string} source The pattern that was passed into the constructor
|
58
|
-
*
|
59
|
-
* @property {string} sourcePath The path portion of the source property
|
60
|
-
*
|
61
|
-
* @property {string} sourceSearch The search portion of the source property
|
62
|
-
*
|
63
|
-
* @property {string} regex The constructed regex that will be used to match against the url when
|
64
|
-
* it is time to determine which url will match.
|
65
|
-
*
|
66
|
-
* @returns {Object} New `UrlMatcher` object
|
67
|
-
*/
|
68
|
-
function UrlMatcher(pattern, config, parentMatcher) {
|
69
|
-
config = extend({ params: {} }, isObject(config) ? config : {});
|
70
|
-
|
71
|
-
// Find all placeholders and create a compiled pattern, using either classic or curly syntax:
|
72
|
-
// '*' name
|
73
|
-
// ':' name
|
74
|
-
// '{' name '}'
|
75
|
-
// '{' name ':' regexp '}'
|
76
|
-
// The regular expression is somewhat complicated due to the need to allow curly braces
|
77
|
-
// inside the regular expression. The placeholder regexp breaks down as follows:
|
78
|
-
// ([:*])([\w\[\]]+) - classic placeholder ($1 / $2) (search version has - for snake-case)
|
79
|
-
// \{([\w\[\]]+)(?:\:( ... ))?\} - curly brace placeholder ($3) with optional regexp/type ... ($4) (search version has - for snake-case
|
80
|
-
// (?: ... | ... | ... )+ - the regexp consists of any number of atoms, an atom being either
|
81
|
-
// [^{}\\]+ - anything other than curly braces or backslash
|
82
|
-
// \\. - a backslash escape
|
83
|
-
// \{(?:[^{}\\]+|\\.)*\} - a matched set of curly braces containing other atoms
|
84
|
-
var placeholder = /([:*])([\w\[\]]+)|\{([\w\[\]]+)(?:\:((?:[^{}\\]+|\\.|\{(?:[^{}\\]+|\\.)*\})+))?\}/g,
|
85
|
-
searchPlaceholder = /([:]?)([\w\[\]-]+)|\{([\w\[\]-]+)(?:\:((?:[^{}\\]+|\\.|\{(?:[^{}\\]+|\\.)*\})+))?\}/g,
|
86
|
-
compiled = '^', last = 0, m,
|
87
|
-
segments = this.segments = [],
|
88
|
-
parentParams = parentMatcher ? parentMatcher.params : {},
|
89
|
-
params = this.params = parentMatcher ? parentMatcher.params.$$new() : new $$UMFP.ParamSet(),
|
90
|
-
paramNames = [];
|
91
|
-
|
92
|
-
function addParameter(id, type, config, location) {
|
93
|
-
paramNames.push(id);
|
94
|
-
if (parentParams[id]) return parentParams[id];
|
95
|
-
if (!/^\w+(-+\w+)*(?:\[\])?$/.test(id)) throw new Error("Invalid parameter name '" + id + "' in pattern '" + pattern + "'");
|
96
|
-
if (params[id]) throw new Error("Duplicate parameter name '" + id + "' in pattern '" + pattern + "'");
|
97
|
-
params[id] = new $$UMFP.Param(id, type, config, location);
|
98
|
-
return params[id];
|
99
|
-
}
|
100
|
-
|
101
|
-
function quoteRegExp(string, pattern, squash, optional) {
|
102
|
-
var surroundPattern = ['',''], result = string.replace(/[\\\[\]\^$*+?.()|{}]/g, "\\$&");
|
103
|
-
if (!pattern) return result;
|
104
|
-
switch(squash) {
|
105
|
-
case false: surroundPattern = ['(', ')' + (optional ? "?" : "")]; break;
|
106
|
-
case true: surroundPattern = ['?(', ')?']; break;
|
107
|
-
default: surroundPattern = ['(' + squash + "|", ')?']; break;
|
108
|
-
}
|
109
|
-
return result + surroundPattern[0] + pattern + surroundPattern[1];
|
110
|
-
}
|
111
|
-
|
112
|
-
this.source = pattern;
|
113
|
-
|
114
|
-
// Split into static segments separated by path parameter placeholders.
|
115
|
-
// The number of segments is always 1 more than the number of parameters.
|
116
|
-
function matchDetails(m, isSearch) {
|
117
|
-
var id, regexp, segment, type, cfg, arrayMode;
|
118
|
-
id = m[2] || m[3]; // IE[78] returns '' for unmatched groups instead of null
|
119
|
-
cfg = config.params[id];
|
120
|
-
segment = pattern.substring(last, m.index);
|
121
|
-
regexp = isSearch ? m[4] : m[4] || (m[1] == '*' ? '.*' : null);
|
122
|
-
type = $$UMFP.type(regexp || "string") || inherit($$UMFP.type("string"), { pattern: new RegExp(regexp, config.caseInsensitive ? 'i' : undefined) });
|
123
|
-
return {
|
124
|
-
id: id, regexp: regexp, segment: segment, type: type, cfg: cfg
|
125
|
-
};
|
126
|
-
}
|
127
|
-
|
128
|
-
var p, param, segment;
|
129
|
-
while ((m = placeholder.exec(pattern))) {
|
130
|
-
p = matchDetails(m, false);
|
131
|
-
if (p.segment.indexOf('?') >= 0) break; // we're into the search part
|
132
|
-
|
133
|
-
param = addParameter(p.id, p.type, p.cfg, "path");
|
134
|
-
compiled += quoteRegExp(p.segment, param.type.pattern.source, param.squash, param.isOptional);
|
135
|
-
segments.push(p.segment);
|
136
|
-
last = placeholder.lastIndex;
|
137
|
-
}
|
138
|
-
segment = pattern.substring(last);
|
139
|
-
|
140
|
-
// Find any search parameter names and remove them from the last segment
|
141
|
-
var i = segment.indexOf('?');
|
142
|
-
|
143
|
-
if (i >= 0) {
|
144
|
-
var search = this.sourceSearch = segment.substring(i);
|
145
|
-
segment = segment.substring(0, i);
|
146
|
-
this.sourcePath = pattern.substring(0, last + i);
|
147
|
-
|
148
|
-
if (search.length > 0) {
|
149
|
-
last = 0;
|
150
|
-
while ((m = searchPlaceholder.exec(search))) {
|
151
|
-
p = matchDetails(m, true);
|
152
|
-
param = addParameter(p.id, p.type, p.cfg, "search");
|
153
|
-
last = placeholder.lastIndex;
|
154
|
-
// check if ?&
|
155
|
-
}
|
156
|
-
}
|
157
|
-
} else {
|
158
|
-
this.sourcePath = pattern;
|
159
|
-
this.sourceSearch = '';
|
160
|
-
}
|
161
|
-
|
162
|
-
compiled += quoteRegExp(segment) + (config.strict === false ? '\/?' : '') + '$';
|
163
|
-
segments.push(segment);
|
164
|
-
|
165
|
-
this.regexp = new RegExp(compiled, config.caseInsensitive ? 'i' : undefined);
|
166
|
-
this.prefix = segments[0];
|
167
|
-
this.$$paramNames = paramNames;
|
168
|
-
}
|
169
|
-
|
170
|
-
/**
|
171
|
-
* @ngdoc function
|
172
|
-
* @name ui.router.util.type:UrlMatcher#concat
|
173
|
-
* @methodOf ui.router.util.type:UrlMatcher
|
174
|
-
*
|
175
|
-
* @description
|
176
|
-
* Returns a new matcher for a pattern constructed by appending the path part and adding the
|
177
|
-
* search parameters of the specified pattern to this pattern. The current pattern is not
|
178
|
-
* modified. This can be understood as creating a pattern for URLs that are relative to (or
|
179
|
-
* suffixes of) the current pattern.
|
180
|
-
*
|
181
|
-
* @example
|
182
|
-
* The following two matchers are equivalent:
|
183
|
-
* <pre>
|
184
|
-
* new UrlMatcher('/user/{id}?q').concat('/details?date');
|
185
|
-
* new UrlMatcher('/user/{id}/details?q&date');
|
186
|
-
* </pre>
|
187
|
-
*
|
188
|
-
* @param {string} pattern The pattern to append.
|
189
|
-
* @param {Object} config An object hash of the configuration for the matcher.
|
190
|
-
* @returns {UrlMatcher} A matcher for the concatenated pattern.
|
191
|
-
*/
|
192
|
-
UrlMatcher.prototype.concat = function (pattern, config) {
|
193
|
-
// Because order of search parameters is irrelevant, we can add our own search
|
194
|
-
// parameters to the end of the new pattern. Parse the new pattern by itself
|
195
|
-
// and then join the bits together, but it's much easier to do this on a string level.
|
196
|
-
var defaultConfig = {
|
197
|
-
caseInsensitive: $$UMFP.caseInsensitive(),
|
198
|
-
strict: $$UMFP.strictMode(),
|
199
|
-
squash: $$UMFP.defaultSquashPolicy()
|
200
|
-
};
|
201
|
-
return new UrlMatcher(this.sourcePath + pattern + this.sourceSearch, extend(defaultConfig, config), this);
|
202
|
-
};
|
203
|
-
|
204
|
-
UrlMatcher.prototype.toString = function () {
|
205
|
-
return this.source;
|
206
|
-
};
|
207
|
-
|
208
|
-
/**
|
209
|
-
* @ngdoc function
|
210
|
-
* @name ui.router.util.type:UrlMatcher#exec
|
211
|
-
* @methodOf ui.router.util.type:UrlMatcher
|
212
|
-
*
|
213
|
-
* @description
|
214
|
-
* Tests the specified path against this matcher, and returns an object containing the captured
|
215
|
-
* parameter values, or null if the path does not match. The returned object contains the values
|
216
|
-
* of any search parameters that are mentioned in the pattern, but their value may be null if
|
217
|
-
* they are not present in `searchParams`. This means that search parameters are always treated
|
218
|
-
* as optional.
|
219
|
-
*
|
220
|
-
* @example
|
221
|
-
* <pre>
|
222
|
-
* new UrlMatcher('/user/{id}?q&r').exec('/user/bob', {
|
223
|
-
* x: '1', q: 'hello'
|
224
|
-
* });
|
225
|
-
* // returns { id: 'bob', q: 'hello', r: null }
|
226
|
-
* </pre>
|
227
|
-
*
|
228
|
-
* @param {string} path The URL path to match, e.g. `$location.path()`.
|
229
|
-
* @param {Object} searchParams URL search parameters, e.g. `$location.search()`.
|
230
|
-
* @returns {Object} The captured parameter values.
|
231
|
-
*/
|
232
|
-
UrlMatcher.prototype.exec = function (path, searchParams) {
|
233
|
-
var m = this.regexp.exec(path);
|
234
|
-
if (!m) return null;
|
235
|
-
searchParams = searchParams || {};
|
236
|
-
|
237
|
-
var paramNames = this.parameters(), nTotal = paramNames.length,
|
238
|
-
nPath = this.segments.length - 1,
|
239
|
-
values = {}, i, j, cfg, paramName;
|
240
|
-
|
241
|
-
if (nPath !== m.length - 1) throw new Error("Unbalanced capture group in route '" + this.source + "'");
|
242
|
-
|
243
|
-
function decodePathArray(string) {
|
244
|
-
function reverseString(str) { return str.split("").reverse().join(""); }
|
245
|
-
function unquoteDashes(str) { return str.replace(/\\-/g, "-"); }
|
246
|
-
|
247
|
-
var split = reverseString(string).split(/-(?!\\)/);
|
248
|
-
var allReversed = map(split, reverseString);
|
249
|
-
return map(allReversed, unquoteDashes).reverse();
|
250
|
-
}
|
251
|
-
|
252
|
-
for (i = 0; i < nPath; i++) {
|
253
|
-
paramName = paramNames[i];
|
254
|
-
var param = this.params[paramName];
|
255
|
-
var paramVal = m[i+1];
|
256
|
-
// if the param value matches a pre-replace pair, replace the value before decoding.
|
257
|
-
for (j = 0; j < param.replace; j++) {
|
258
|
-
if (param.replace[j].from === paramVal) paramVal = param.replace[j].to;
|
259
|
-
}
|
260
|
-
if (paramVal && param.array === true) paramVal = decodePathArray(paramVal);
|
261
|
-
values[paramName] = param.value(paramVal);
|
262
|
-
}
|
263
|
-
for (/**/; i < nTotal; i++) {
|
264
|
-
paramName = paramNames[i];
|
265
|
-
values[paramName] = this.params[paramName].value(searchParams[paramName]);
|
266
|
-
}
|
267
|
-
|
268
|
-
return values;
|
269
|
-
};
|
270
|
-
|
271
|
-
/**
|
272
|
-
* @ngdoc function
|
273
|
-
* @name ui.router.util.type:UrlMatcher#parameters
|
274
|
-
* @methodOf ui.router.util.type:UrlMatcher
|
275
|
-
*
|
276
|
-
* @description
|
277
|
-
* Returns the names of all path and search parameters of this pattern in an unspecified order.
|
278
|
-
*
|
279
|
-
* @returns {Array.<string>} An array of parameter names. Must be treated as read-only. If the
|
280
|
-
* pattern has no parameters, an empty array is returned.
|
281
|
-
*/
|
282
|
-
UrlMatcher.prototype.parameters = function (param) {
|
283
|
-
if (!isDefined(param)) return this.$$paramNames;
|
284
|
-
return this.params[param] || null;
|
285
|
-
};
|
286
|
-
|
287
|
-
/**
|
288
|
-
* @ngdoc function
|
289
|
-
* @name ui.router.util.type:UrlMatcher#validate
|
290
|
-
* @methodOf ui.router.util.type:UrlMatcher
|
291
|
-
*
|
292
|
-
* @description
|
293
|
-
* Checks an object hash of parameters to validate their correctness according to the parameter
|
294
|
-
* types of this `UrlMatcher`.
|
295
|
-
*
|
296
|
-
* @param {Object} params The object hash of parameters to validate.
|
297
|
-
* @returns {boolean} Returns `true` if `params` validates, otherwise `false`.
|
298
|
-
*/
|
299
|
-
UrlMatcher.prototype.validates = function (params) {
|
300
|
-
return this.params.$$validates(params);
|
301
|
-
};
|
302
|
-
|
303
|
-
/**
|
304
|
-
* @ngdoc function
|
305
|
-
* @name ui.router.util.type:UrlMatcher#format
|
306
|
-
* @methodOf ui.router.util.type:UrlMatcher
|
307
|
-
*
|
308
|
-
* @description
|
309
|
-
* Creates a URL that matches this pattern by substituting the specified values
|
310
|
-
* for the path and search parameters. Null values for path parameters are
|
311
|
-
* treated as empty strings.
|
312
|
-
*
|
313
|
-
* @example
|
314
|
-
* <pre>
|
315
|
-
* new UrlMatcher('/user/{id}?q').format({ id:'bob', q:'yes' });
|
316
|
-
* // returns '/user/bob?q=yes'
|
317
|
-
* </pre>
|
318
|
-
*
|
319
|
-
* @param {Object} values the values to substitute for the parameters in this pattern.
|
320
|
-
* @returns {string} the formatted URL (path and optionally search part).
|
321
|
-
*/
|
322
|
-
UrlMatcher.prototype.format = function (values) {
|
323
|
-
values = values || {};
|
324
|
-
var segments = this.segments, params = this.parameters(), paramset = this.params;
|
325
|
-
if (!this.validates(values)) return null;
|
326
|
-
|
327
|
-
var i, search = false, nPath = segments.length - 1, nTotal = params.length, result = segments[0];
|
328
|
-
|
329
|
-
function encodeDashes(str) { // Replace dashes with encoded "\-"
|
330
|
-
return encodeURIComponent(str).replace(/-/g, function(c) { return '%5C%' + c.charCodeAt(0).toString(16).toUpperCase(); });
|
331
|
-
}
|
332
|
-
|
333
|
-
for (i = 0; i < nTotal; i++) {
|
334
|
-
var isPathParam = i < nPath;
|
335
|
-
var name = params[i], param = paramset[name], value = param.value(values[name]);
|
336
|
-
var isDefaultValue = param.isOptional && param.type.equals(param.value(), value);
|
337
|
-
var squash = isDefaultValue ? param.squash : false;
|
338
|
-
var encoded = param.type.encode(value);
|
339
|
-
|
340
|
-
if (isPathParam) {
|
341
|
-
var nextSegment = segments[i + 1];
|
342
|
-
if (squash === false) {
|
343
|
-
if (encoded != null) {
|
344
|
-
if (isArray(encoded)) {
|
345
|
-
result += map(encoded, encodeDashes).join("-");
|
346
|
-
} else {
|
347
|
-
result += encodeURIComponent(encoded);
|
348
|
-
}
|
349
|
-
}
|
350
|
-
result += nextSegment;
|
351
|
-
} else if (squash === true) {
|
352
|
-
var capture = result.match(/\/$/) ? /\/?(.*)/ : /(.*)/;
|
353
|
-
result += nextSegment.match(capture)[1];
|
354
|
-
} else if (isString(squash)) {
|
355
|
-
result += squash + nextSegment;
|
356
|
-
}
|
357
|
-
} else {
|
358
|
-
if (encoded == null || (isDefaultValue && squash !== false)) continue;
|
359
|
-
if (!isArray(encoded)) encoded = [ encoded ];
|
360
|
-
encoded = map(encoded, encodeURIComponent).join('&' + name + '=');
|
361
|
-
result += (search ? '&' : '?') + (name + '=' + encoded);
|
362
|
-
search = true;
|
363
|
-
}
|
364
|
-
}
|
365
|
-
|
366
|
-
return result;
|
367
|
-
};
|
368
|
-
|
369
|
-
/**
|
370
|
-
* @ngdoc object
|
371
|
-
* @name ui.router.util.type:Type
|
372
|
-
*
|
373
|
-
* @description
|
374
|
-
* Implements an interface to define custom parameter types that can be decoded from and encoded to
|
375
|
-
* string parameters matched in a URL. Used by {@link ui.router.util.type:UrlMatcher `UrlMatcher`}
|
376
|
-
* objects when matching or formatting URLs, or comparing or validating parameter values.
|
377
|
-
*
|
378
|
-
* See {@link ui.router.util.$urlMatcherFactory#methods_type `$urlMatcherFactory#type()`} for more
|
379
|
-
* information on registering custom types.
|
380
|
-
*
|
381
|
-
* @param {Object} config A configuration object which contains the custom type definition. The object's
|
382
|
-
* properties will override the default methods and/or pattern in `Type`'s public interface.
|
383
|
-
* @example
|
384
|
-
* <pre>
|
385
|
-
* {
|
386
|
-
* decode: function(val) { return parseInt(val, 10); },
|
387
|
-
* encode: function(val) { return val && val.toString(); },
|
388
|
-
* equals: function(a, b) { return this.is(a) && a === b; },
|
389
|
-
* is: function(val) { return angular.isNumber(val) isFinite(val) && val % 1 === 0; },
|
390
|
-
* pattern: /\d+/
|
391
|
-
* }
|
392
|
-
* </pre>
|
393
|
-
*
|
394
|
-
* @property {RegExp} pattern The regular expression pattern used to match values of this type when
|
395
|
-
* coming from a substring of a URL.
|
396
|
-
*
|
397
|
-
* @returns {Object} Returns a new `Type` object.
|
398
|
-
*/
|
399
|
-
function Type(config) {
|
400
|
-
extend(this, config);
|
401
|
-
}
|
402
|
-
|
403
|
-
/**
|
404
|
-
* @ngdoc function
|
405
|
-
* @name ui.router.util.type:Type#is
|
406
|
-
* @methodOf ui.router.util.type:Type
|
407
|
-
*
|
408
|
-
* @description
|
409
|
-
* Detects whether a value is of a particular type. Accepts a native (decoded) value
|
410
|
-
* and determines whether it matches the current `Type` object.
|
411
|
-
*
|
412
|
-
* @param {*} val The value to check.
|
413
|
-
* @param {string} key Optional. If the type check is happening in the context of a specific
|
414
|
-
* {@link ui.router.util.type:UrlMatcher `UrlMatcher`} object, this is the name of the
|
415
|
-
* parameter in which `val` is stored. Can be used for meta-programming of `Type` objects.
|
416
|
-
* @returns {Boolean} Returns `true` if the value matches the type, otherwise `false`.
|
417
|
-
*/
|
418
|
-
Type.prototype.is = function(val, key) {
|
419
|
-
return true;
|
420
|
-
};
|
421
|
-
|
422
|
-
/**
|
423
|
-
* @ngdoc function
|
424
|
-
* @name ui.router.util.type:Type#encode
|
425
|
-
* @methodOf ui.router.util.type:Type
|
426
|
-
*
|
427
|
-
* @description
|
428
|
-
* Encodes a custom/native type value to a string that can be embedded in a URL. Note that the
|
429
|
-
* return value does *not* need to be URL-safe (i.e. passed through `encodeURIComponent()`), it
|
430
|
-
* only needs to be a representation of `val` that has been coerced to a string.
|
431
|
-
*
|
432
|
-
* @param {*} val The value to encode.
|
433
|
-
* @param {string} key The name of the parameter in which `val` is stored. Can be used for
|
434
|
-
* meta-programming of `Type` objects.
|
435
|
-
* @returns {string} Returns a string representation of `val` that can be encoded in a URL.
|
436
|
-
*/
|
437
|
-
Type.prototype.encode = function(val, key) {
|
438
|
-
return val;
|
439
|
-
};
|
440
|
-
|
441
|
-
/**
|
442
|
-
* @ngdoc function
|
443
|
-
* @name ui.router.util.type:Type#decode
|
444
|
-
* @methodOf ui.router.util.type:Type
|
445
|
-
*
|
446
|
-
* @description
|
447
|
-
* Converts a parameter value (from URL string or transition param) to a custom/native value.
|
448
|
-
*
|
449
|
-
* @param {string} val The URL parameter value to decode.
|
450
|
-
* @param {string} key The name of the parameter in which `val` is stored. Can be used for
|
451
|
-
* meta-programming of `Type` objects.
|
452
|
-
* @returns {*} Returns a custom representation of the URL parameter value.
|
453
|
-
*/
|
454
|
-
Type.prototype.decode = function(val, key) {
|
455
|
-
return val;
|
456
|
-
};
|
457
|
-
|
458
|
-
/**
|
459
|
-
* @ngdoc function
|
460
|
-
* @name ui.router.util.type:Type#equals
|
461
|
-
* @methodOf ui.router.util.type:Type
|
462
|
-
*
|
463
|
-
* @description
|
464
|
-
* Determines whether two decoded values are equivalent.
|
465
|
-
*
|
466
|
-
* @param {*} a A value to compare against.
|
467
|
-
* @param {*} b A value to compare against.
|
468
|
-
* @returns {Boolean} Returns `true` if the values are equivalent/equal, otherwise `false`.
|
469
|
-
*/
|
470
|
-
Type.prototype.equals = function(a, b) {
|
471
|
-
return a == b;
|
472
|
-
};
|
473
|
-
|
474
|
-
Type.prototype.$subPattern = function() {
|
475
|
-
var sub = this.pattern.toString();
|
476
|
-
return sub.substr(1, sub.length - 2);
|
477
|
-
};
|
478
|
-
|
479
|
-
Type.prototype.pattern = /.*/;
|
480
|
-
|
481
|
-
Type.prototype.toString = function() { return "{Type:" + this.name + "}"; };
|
482
|
-
|
483
|
-
/** Given an encoded string, or a decoded object, returns a decoded object */
|
484
|
-
Type.prototype.$normalize = function(val) {
|
485
|
-
return this.is(val) ? val : this.decode(val);
|
486
|
-
};
|
487
|
-
|
488
|
-
/*
|
489
|
-
* Wraps an existing custom Type as an array of Type, depending on 'mode'.
|
490
|
-
* e.g.:
|
491
|
-
* - urlmatcher pattern "/path?{queryParam[]:int}"
|
492
|
-
* - url: "/path?queryParam=1&queryParam=2
|
493
|
-
* - $stateParams.queryParam will be [1, 2]
|
494
|
-
* if `mode` is "auto", then
|
495
|
-
* - url: "/path?queryParam=1 will create $stateParams.queryParam: 1
|
496
|
-
* - url: "/path?queryParam=1&queryParam=2 will create $stateParams.queryParam: [1, 2]
|
497
|
-
*/
|
498
|
-
Type.prototype.$asArray = function(mode, isSearch) {
|
499
|
-
if (!mode) return this;
|
500
|
-
if (mode === "auto" && !isSearch) throw new Error("'auto' array mode is for query parameters only");
|
501
|
-
|
502
|
-
function ArrayType(type, mode) {
|
503
|
-
function bindTo(type, callbackName) {
|
504
|
-
return function() {
|
505
|
-
return type[callbackName].apply(type, arguments);
|
506
|
-
};
|
507
|
-
}
|
508
|
-
|
509
|
-
// Wrap non-array value as array
|
510
|
-
function arrayWrap(val) { return isArray(val) ? val : (isDefined(val) ? [ val ] : []); }
|
511
|
-
// Unwrap array value for "auto" mode. Return undefined for empty array.
|
512
|
-
function arrayUnwrap(val) {
|
513
|
-
switch(val.length) {
|
514
|
-
case 0: return undefined;
|
515
|
-
case 1: return mode === "auto" ? val[0] : val;
|
516
|
-
default: return val;
|
517
|
-
}
|
518
|
-
}
|
519
|
-
function falsey(val) { return !val; }
|
520
|
-
|
521
|
-
// Wraps type (.is/.encode/.decode) functions to operate on each value of an array
|
522
|
-
function arrayHandler(callback, allTruthyMode) {
|
523
|
-
return function handleArray(val) {
|
524
|
-
val = arrayWrap(val);
|
525
|
-
var result = map(val, callback);
|
526
|
-
if (allTruthyMode === true)
|
527
|
-
return filter(result, falsey).length === 0;
|
528
|
-
return arrayUnwrap(result);
|
529
|
-
};
|
530
|
-
}
|
531
|
-
|
532
|
-
// Wraps type (.equals) functions to operate on each value of an array
|
533
|
-
function arrayEqualsHandler(callback) {
|
534
|
-
return function handleArray(val1, val2) {
|
535
|
-
var left = arrayWrap(val1), right = arrayWrap(val2);
|
536
|
-
if (left.length !== right.length) return false;
|
537
|
-
for (var i = 0; i < left.length; i++) {
|
538
|
-
if (!callback(left[i], right[i])) return false;
|
539
|
-
}
|
540
|
-
return true;
|
541
|
-
};
|
542
|
-
}
|
543
|
-
|
544
|
-
this.encode = arrayHandler(bindTo(type, 'encode'));
|
545
|
-
this.decode = arrayHandler(bindTo(type, 'decode'));
|
546
|
-
this.is = arrayHandler(bindTo(type, 'is'), true);
|
547
|
-
this.equals = arrayEqualsHandler(bindTo(type, 'equals'));
|
548
|
-
this.pattern = type.pattern;
|
549
|
-
this.$normalize = arrayHandler(bindTo(type, '$normalize'));
|
550
|
-
this.name = type.name;
|
551
|
-
this.$arrayMode = mode;
|
552
|
-
}
|
553
|
-
|
554
|
-
return new ArrayType(this, mode);
|
555
|
-
};
|
556
|
-
|
557
|
-
|
558
|
-
|
559
|
-
/**
|
560
|
-
* @ngdoc object
|
561
|
-
* @name ui.router.util.$urlMatcherFactory
|
562
|
-
*
|
563
|
-
* @description
|
564
|
-
* Factory for {@link ui.router.util.type:UrlMatcher `UrlMatcher`} instances. The factory
|
565
|
-
* is also available to providers under the name `$urlMatcherFactoryProvider`.
|
566
|
-
*/
|
567
|
-
function $UrlMatcherFactory() {
|
568
|
-
$$UMFP = this;
|
569
|
-
|
570
|
-
var isCaseInsensitive = false, isStrictMode = true, defaultSquashPolicy = false;
|
571
|
-
|
572
|
-
function valToString(val) { return val != null ? val.toString().replace(/\//g, "%2F") : val; }
|
573
|
-
function valFromString(val) { return val != null ? val.toString().replace(/%2F/g, "/") : val; }
|
574
|
-
|
575
|
-
var $types = {}, enqueue = true, typeQueue = [], injector, defaultTypes = {
|
576
|
-
string: {
|
577
|
-
encode: valToString,
|
578
|
-
decode: valFromString,
|
579
|
-
// TODO: in 1.0, make string .is() return false if value is undefined/null by default.
|
580
|
-
// In 0.2.x, string params are optional by default for backwards compat
|
581
|
-
is: function(val) { return val == null || !isDefined(val) || typeof val === "string"; },
|
582
|
-
pattern: /[^/]*/
|
583
|
-
},
|
584
|
-
int: {
|
585
|
-
encode: valToString,
|
586
|
-
decode: function(val) { return parseInt(val, 10); },
|
587
|
-
is: function(val) { return isDefined(val) && this.decode(val.toString()) === val; },
|
588
|
-
pattern: /\d+/
|
589
|
-
},
|
590
|
-
bool: {
|
591
|
-
encode: function(val) { return val ? 1 : 0; },
|
592
|
-
decode: function(val) { return parseInt(val, 10) !== 0; },
|
593
|
-
is: function(val) { return val === true || val === false; },
|
594
|
-
pattern: /0|1/
|
595
|
-
},
|
596
|
-
date: {
|
597
|
-
encode: function (val) {
|
598
|
-
if (!this.is(val))
|
599
|
-
return undefined;
|
600
|
-
return [ val.getFullYear(),
|
601
|
-
('0' + (val.getMonth() + 1)).slice(-2),
|
602
|
-
('0' + val.getDate()).slice(-2)
|
603
|
-
].join("-");
|
604
|
-
},
|
605
|
-
decode: function (val) {
|
606
|
-
if (this.is(val)) return val;
|
607
|
-
var match = this.capture.exec(val);
|
608
|
-
return match ? new Date(match[1], match[2] - 1, match[3]) : undefined;
|
609
|
-
},
|
610
|
-
is: function(val) { return val instanceof Date && !isNaN(val.valueOf()); },
|
611
|
-
equals: function (a, b) { return this.is(a) && this.is(b) && a.toISOString() === b.toISOString(); },
|
612
|
-
pattern: /[0-9]{4}-(?:0[1-9]|1[0-2])-(?:0[1-9]|[1-2][0-9]|3[0-1])/,
|
613
|
-
capture: /([0-9]{4})-(0[1-9]|1[0-2])-(0[1-9]|[1-2][0-9]|3[0-1])/
|
614
|
-
},
|
615
|
-
json: {
|
616
|
-
encode: angular.toJson,
|
617
|
-
decode: angular.fromJson,
|
618
|
-
is: angular.isObject,
|
619
|
-
equals: angular.equals,
|
620
|
-
pattern: /[^/]*/
|
621
|
-
},
|
622
|
-
any: { // does not encode/decode
|
623
|
-
encode: angular.identity,
|
624
|
-
decode: angular.identity,
|
625
|
-
equals: angular.equals,
|
626
|
-
pattern: /.*/
|
627
|
-
}
|
628
|
-
};
|
629
|
-
|
630
|
-
function getDefaultConfig() {
|
631
|
-
return {
|
632
|
-
strict: isStrictMode,
|
633
|
-
caseInsensitive: isCaseInsensitive
|
634
|
-
};
|
635
|
-
}
|
636
|
-
|
637
|
-
function isInjectable(value) {
|
638
|
-
return (isFunction(value) || (isArray(value) && isFunction(value[value.length - 1])));
|
639
|
-
}
|
640
|
-
|
641
|
-
/**
|
642
|
-
* [Internal] Get the default value of a parameter, which may be an injectable function.
|
643
|
-
*/
|
644
|
-
$UrlMatcherFactory.$$getDefaultValue = function(config) {
|
645
|
-
if (!isInjectable(config.value)) return config.value;
|
646
|
-
if (!injector) throw new Error("Injectable functions cannot be called at configuration time");
|
647
|
-
return injector.invoke(config.value);
|
648
|
-
};
|
649
|
-
|
650
|
-
/**
|
651
|
-
* @ngdoc function
|
652
|
-
* @name ui.router.util.$urlMatcherFactory#caseInsensitive
|
653
|
-
* @methodOf ui.router.util.$urlMatcherFactory
|
654
|
-
*
|
655
|
-
* @description
|
656
|
-
* Defines whether URL matching should be case sensitive (the default behavior), or not.
|
657
|
-
*
|
658
|
-
* @param {boolean} value `false` to match URL in a case sensitive manner; otherwise `true`;
|
659
|
-
* @returns {boolean} the current value of caseInsensitive
|
660
|
-
*/
|
661
|
-
this.caseInsensitive = function(value) {
|
662
|
-
if (isDefined(value))
|
663
|
-
isCaseInsensitive = value;
|
664
|
-
return isCaseInsensitive;
|
665
|
-
};
|
666
|
-
|
667
|
-
/**
|
668
|
-
* @ngdoc function
|
669
|
-
* @name ui.router.util.$urlMatcherFactory#strictMode
|
670
|
-
* @methodOf ui.router.util.$urlMatcherFactory
|
671
|
-
*
|
672
|
-
* @description
|
673
|
-
* Defines whether URLs should match trailing slashes, or not (the default behavior).
|
674
|
-
*
|
675
|
-
* @param {boolean=} value `false` to match trailing slashes in URLs, otherwise `true`.
|
676
|
-
* @returns {boolean} the current value of strictMode
|
677
|
-
*/
|
678
|
-
this.strictMode = function(value) {
|
679
|
-
if (isDefined(value))
|
680
|
-
isStrictMode = value;
|
681
|
-
return isStrictMode;
|
682
|
-
};
|
683
|
-
|
684
|
-
/**
|
685
|
-
* @ngdoc function
|
686
|
-
* @name ui.router.util.$urlMatcherFactory#defaultSquashPolicy
|
687
|
-
* @methodOf ui.router.util.$urlMatcherFactory
|
688
|
-
*
|
689
|
-
* @description
|
690
|
-
* Sets the default behavior when generating or matching URLs with default parameter values.
|
691
|
-
*
|
692
|
-
* @param {string} value A string that defines the default parameter URL squashing behavior.
|
693
|
-
* `nosquash`: When generating an href with a default parameter value, do not squash the parameter value from the URL
|
694
|
-
* `slash`: When generating an href with a default parameter value, squash (remove) the parameter value, and, if the
|
695
|
-
* parameter is surrounded by slashes, squash (remove) one slash from the URL
|
696
|
-
* any other string, e.g. "~": When generating an href with a default parameter value, squash (remove)
|
697
|
-
* the parameter value from the URL and replace it with this string.
|
698
|
-
*/
|
699
|
-
this.defaultSquashPolicy = function(value) {
|
700
|
-
if (!isDefined(value)) return defaultSquashPolicy;
|
701
|
-
if (value !== true && value !== false && !isString(value))
|
702
|
-
throw new Error("Invalid squash policy: " + value + ". Valid policies: false, true, arbitrary-string");
|
703
|
-
defaultSquashPolicy = value;
|
704
|
-
return value;
|
705
|
-
};
|
706
|
-
|
707
|
-
/**
|
708
|
-
* @ngdoc function
|
709
|
-
* @name ui.router.util.$urlMatcherFactory#compile
|
710
|
-
* @methodOf ui.router.util.$urlMatcherFactory
|
711
|
-
*
|
712
|
-
* @description
|
713
|
-
* Creates a {@link ui.router.util.type:UrlMatcher `UrlMatcher`} for the specified pattern.
|
714
|
-
*
|
715
|
-
* @param {string} pattern The URL pattern.
|
716
|
-
* @param {Object} config The config object hash.
|
717
|
-
* @returns {UrlMatcher} The UrlMatcher.
|
718
|
-
*/
|
719
|
-
this.compile = function (pattern, config) {
|
720
|
-
return new UrlMatcher(pattern, extend(getDefaultConfig(), config));
|
721
|
-
};
|
722
|
-
|
723
|
-
/**
|
724
|
-
* @ngdoc function
|
725
|
-
* @name ui.router.util.$urlMatcherFactory#isMatcher
|
726
|
-
* @methodOf ui.router.util.$urlMatcherFactory
|
727
|
-
*
|
728
|
-
* @description
|
729
|
-
* Returns true if the specified object is a `UrlMatcher`, or false otherwise.
|
730
|
-
*
|
731
|
-
* @param {Object} object The object to perform the type check against.
|
732
|
-
* @returns {Boolean} Returns `true` if the object matches the `UrlMatcher` interface, by
|
733
|
-
* implementing all the same methods.
|
734
|
-
*/
|
735
|
-
this.isMatcher = function (o) {
|
736
|
-
if (!isObject(o)) return false;
|
737
|
-
var result = true;
|
738
|
-
|
739
|
-
forEach(UrlMatcher.prototype, function(val, name) {
|
740
|
-
if (isFunction(val)) {
|
741
|
-
result = result && (isDefined(o[name]) && isFunction(o[name]));
|
742
|
-
}
|
743
|
-
});
|
744
|
-
return result;
|
745
|
-
};
|
746
|
-
|
747
|
-
/**
|
748
|
-
* @ngdoc function
|
749
|
-
* @name ui.router.util.$urlMatcherFactory#type
|
750
|
-
* @methodOf ui.router.util.$urlMatcherFactory
|
751
|
-
*
|
752
|
-
* @description
|
753
|
-
* Registers a custom {@link ui.router.util.type:Type `Type`} object that can be used to
|
754
|
-
* generate URLs with typed parameters.
|
755
|
-
*
|
756
|
-
* @param {string} name The type name.
|
757
|
-
* @param {Object|Function} definition The type definition. See
|
758
|
-
* {@link ui.router.util.type:Type `Type`} for information on the values accepted.
|
759
|
-
* @param {Object|Function} definitionFn (optional) A function that is injected before the app
|
760
|
-
* runtime starts. The result of this function is merged into the existing `definition`.
|
761
|
-
* See {@link ui.router.util.type:Type `Type`} for information on the values accepted.
|
762
|
-
*
|
763
|
-
* @returns {Object} Returns `$urlMatcherFactoryProvider`.
|
764
|
-
*
|
765
|
-
* @example
|
766
|
-
* This is a simple example of a custom type that encodes and decodes items from an
|
767
|
-
* array, using the array index as the URL-encoded value:
|
768
|
-
*
|
769
|
-
* <pre>
|
770
|
-
* var list = ['John', 'Paul', 'George', 'Ringo'];
|
771
|
-
*
|
772
|
-
* $urlMatcherFactoryProvider.type('listItem', {
|
773
|
-
* encode: function(item) {
|
774
|
-
* // Represent the list item in the URL using its corresponding index
|
775
|
-
* return list.indexOf(item);
|
776
|
-
* },
|
777
|
-
* decode: function(item) {
|
778
|
-
* // Look up the list item by index
|
779
|
-
* return list[parseInt(item, 10)];
|
780
|
-
* },
|
781
|
-
* is: function(item) {
|
782
|
-
* // Ensure the item is valid by checking to see that it appears
|
783
|
-
* // in the list
|
784
|
-
* return list.indexOf(item) > -1;
|
785
|
-
* }
|
786
|
-
* });
|
787
|
-
*
|
788
|
-
* $stateProvider.state('list', {
|
789
|
-
* url: "/list/{item:listItem}",
|
790
|
-
* controller: function($scope, $stateParams) {
|
791
|
-
* console.log($stateParams.item);
|
792
|
-
* }
|
793
|
-
* });
|
794
|
-
*
|
795
|
-
* // ...
|
796
|
-
*
|
797
|
-
* // Changes URL to '/list/3', logs "Ringo" to the console
|
798
|
-
* $state.go('list', { item: "Ringo" });
|
799
|
-
* </pre>
|
800
|
-
*
|
801
|
-
* This is a more complex example of a type that relies on dependency injection to
|
802
|
-
* interact with services, and uses the parameter name from the URL to infer how to
|
803
|
-
* handle encoding and decoding parameter values:
|
804
|
-
*
|
805
|
-
* <pre>
|
806
|
-
* // Defines a custom type that gets a value from a service,
|
807
|
-
* // where each service gets different types of values from
|
808
|
-
* // a backend API:
|
809
|
-
* $urlMatcherFactoryProvider.type('dbObject', {}, function(Users, Posts) {
|
810
|
-
*
|
811
|
-
* // Matches up services to URL parameter names
|
812
|
-
* var services = {
|
813
|
-
* user: Users,
|
814
|
-
* post: Posts
|
815
|
-
* };
|
816
|
-
*
|
817
|
-
* return {
|
818
|
-
* encode: function(object) {
|
819
|
-
* // Represent the object in the URL using its unique ID
|
820
|
-
* return object.id;
|
821
|
-
* },
|
822
|
-
* decode: function(value, key) {
|
823
|
-
* // Look up the object by ID, using the parameter
|
824
|
-
* // name (key) to call the correct service
|
825
|
-
* return services[key].findById(value);
|
826
|
-
* },
|
827
|
-
* is: function(object, key) {
|
828
|
-
* // Check that object is a valid dbObject
|
829
|
-
* return angular.isObject(object) && object.id && services[key];
|
830
|
-
* }
|
831
|
-
* equals: function(a, b) {
|
832
|
-
* // Check the equality of decoded objects by comparing
|
833
|
-
* // their unique IDs
|
834
|
-
* return a.id === b.id;
|
835
|
-
* }
|
836
|
-
* };
|
837
|
-
* });
|
838
|
-
*
|
839
|
-
* // In a config() block, you can then attach URLs with
|
840
|
-
* // type-annotated parameters:
|
841
|
-
* $stateProvider.state('users', {
|
842
|
-
* url: "/users",
|
843
|
-
* // ...
|
844
|
-
* }).state('users.item', {
|
845
|
-
* url: "/{user:dbObject}",
|
846
|
-
* controller: function($scope, $stateParams) {
|
847
|
-
* // $stateParams.user will now be an object returned from
|
848
|
-
* // the Users service
|
849
|
-
* },
|
850
|
-
* // ...
|
851
|
-
* });
|
852
|
-
* </pre>
|
853
|
-
*/
|
854
|
-
this.type = function (name, definition, definitionFn) {
|
855
|
-
if (!isDefined(definition)) return $types[name];
|
856
|
-
if ($types.hasOwnProperty(name)) throw new Error("A type named '" + name + "' has already been defined.");
|
857
|
-
|
858
|
-
$types[name] = new Type(extend({ name: name }, definition));
|
859
|
-
if (definitionFn) {
|
860
|
-
typeQueue.push({ name: name, def: definitionFn });
|
861
|
-
if (!enqueue) flushTypeQueue();
|
862
|
-
}
|
863
|
-
return this;
|
864
|
-
};
|
865
|
-
|
866
|
-
// `flushTypeQueue()` waits until `$urlMatcherFactory` is injected before invoking the queued `definitionFn`s
|
867
|
-
function flushTypeQueue() {
|
868
|
-
while(typeQueue.length) {
|
869
|
-
var type = typeQueue.shift();
|
870
|
-
if (type.pattern) throw new Error("You cannot override a type's .pattern at runtime.");
|
871
|
-
angular.extend($types[type.name], injector.invoke(type.def));
|
872
|
-
}
|
873
|
-
}
|
874
|
-
|
875
|
-
// Register default types. Store them in the prototype of $types.
|
876
|
-
forEach(defaultTypes, function(type, name) { $types[name] = new Type(extend({name: name}, type)); });
|
877
|
-
$types = inherit($types, {});
|
878
|
-
|
879
|
-
/* No need to document $get, since it returns this */
|
880
|
-
this.$get = ['$injector', function ($injector) {
|
881
|
-
injector = $injector;
|
882
|
-
enqueue = false;
|
883
|
-
flushTypeQueue();
|
884
|
-
|
885
|
-
forEach(defaultTypes, function(type, name) {
|
886
|
-
if (!$types[name]) $types[name] = new Type(type);
|
887
|
-
});
|
888
|
-
return this;
|
889
|
-
}];
|
890
|
-
|
891
|
-
this.Param = function Param(id, type, config, location) {
|
892
|
-
var self = this;
|
893
|
-
config = unwrapShorthand(config);
|
894
|
-
type = getType(config, type, location);
|
895
|
-
var arrayMode = getArrayMode();
|
896
|
-
type = arrayMode ? type.$asArray(arrayMode, location === "search") : type;
|
897
|
-
if (type.name === "string" && !arrayMode && location === "path" && config.value === undefined)
|
898
|
-
config.value = ""; // for 0.2.x; in 0.3.0+ do not automatically default to ""
|
899
|
-
var isOptional = config.value !== undefined;
|
900
|
-
var squash = getSquashPolicy(config, isOptional);
|
901
|
-
var replace = getReplace(config, arrayMode, isOptional, squash);
|
902
|
-
|
903
|
-
function unwrapShorthand(config) {
|
904
|
-
var keys = isObject(config) ? objectKeys(config) : [];
|
905
|
-
var isShorthand = indexOf(keys, "value") === -1 && indexOf(keys, "type") === -1 &&
|
906
|
-
indexOf(keys, "squash") === -1 && indexOf(keys, "array") === -1;
|
907
|
-
if (isShorthand) config = { value: config };
|
908
|
-
config.$$fn = isInjectable(config.value) ? config.value : function () { return config.value; };
|
909
|
-
return config;
|
910
|
-
}
|
911
|
-
|
912
|
-
function getType(config, urlType, location) {
|
913
|
-
if (config.type && urlType) throw new Error("Param '"+id+"' has two type configurations.");
|
914
|
-
if (urlType) return urlType;
|
915
|
-
if (!config.type) return (location === "config" ? $types.any : $types.string);
|
916
|
-
return config.type instanceof Type ? config.type : new Type(config.type);
|
917
|
-
}
|
918
|
-
|
919
|
-
// array config: param name (param[]) overrides default settings. explicit config overrides param name.
|
920
|
-
function getArrayMode() {
|
921
|
-
var arrayDefaults = { array: (location === "search" ? "auto" : false) };
|
922
|
-
var arrayParamNomenclature = id.match(/\[\]$/) ? { array: true } : {};
|
923
|
-
return extend(arrayDefaults, arrayParamNomenclature, config).array;
|
924
|
-
}
|
925
|
-
|
926
|
-
/**
|
927
|
-
* returns false, true, or the squash value to indicate the "default parameter url squash policy".
|
928
|
-
*/
|
929
|
-
function getSquashPolicy(config, isOptional) {
|
930
|
-
var squash = config.squash;
|
931
|
-
if (!isOptional || squash === false) return false;
|
932
|
-
if (!isDefined(squash) || squash == null) return defaultSquashPolicy;
|
933
|
-
if (squash === true || isString(squash)) return squash;
|
934
|
-
throw new Error("Invalid squash policy: '" + squash + "'. Valid policies: false, true, or arbitrary string");
|
935
|
-
}
|
936
|
-
|
937
|
-
function getReplace(config, arrayMode, isOptional, squash) {
|
938
|
-
var replace, configuredKeys, defaultPolicy = [
|
939
|
-
{ from: "", to: (isOptional || arrayMode ? undefined : "") },
|
940
|
-
{ from: null, to: (isOptional || arrayMode ? undefined : "") }
|
941
|
-
];
|
942
|
-
replace = isArray(config.replace) ? config.replace : [];
|
943
|
-
if (isString(squash))
|
944
|
-
replace.push({ from: squash, to: undefined });
|
945
|
-
configuredKeys = map(replace, function(item) { return item.from; } );
|
946
|
-
return filter(defaultPolicy, function(item) { return indexOf(configuredKeys, item.from) === -1; }).concat(replace);
|
947
|
-
}
|
948
|
-
|
949
|
-
/**
|
950
|
-
* [Internal] Get the default value of a parameter, which may be an injectable function.
|
951
|
-
*/
|
952
|
-
function $$getDefaultValue() {
|
953
|
-
if (!injector) throw new Error("Injectable functions cannot be called at configuration time");
|
954
|
-
var defaultValue = injector.invoke(config.$$fn);
|
955
|
-
if (defaultValue !== null && defaultValue !== undefined && !self.type.is(defaultValue))
|
956
|
-
throw new Error("Default value (" + defaultValue + ") for parameter '" + self.id + "' is not an instance of Type (" + self.type.name + ")");
|
957
|
-
return defaultValue;
|
958
|
-
}
|
959
|
-
|
960
|
-
/**
|
961
|
-
* [Internal] Gets the decoded representation of a value if the value is defined, otherwise, returns the
|
962
|
-
* default value, which may be the result of an injectable function.
|
963
|
-
*/
|
964
|
-
function $value(value) {
|
965
|
-
function hasReplaceVal(val) { return function(obj) { return obj.from === val; }; }
|
966
|
-
function $replace(value) {
|
967
|
-
var replacement = map(filter(self.replace, hasReplaceVal(value)), function(obj) { return obj.to; });
|
968
|
-
return replacement.length ? replacement[0] : value;
|
969
|
-
}
|
970
|
-
value = $replace(value);
|
971
|
-
return !isDefined(value) ? $$getDefaultValue() : self.type.$normalize(value);
|
972
|
-
}
|
973
|
-
|
974
|
-
function toString() { return "{Param:" + id + " " + type + " squash: '" + squash + "' optional: " + isOptional + "}"; }
|
975
|
-
|
976
|
-
extend(this, {
|
977
|
-
id: id,
|
978
|
-
type: type,
|
979
|
-
location: location,
|
980
|
-
array: arrayMode,
|
981
|
-
squash: squash,
|
982
|
-
replace: replace,
|
983
|
-
isOptional: isOptional,
|
984
|
-
value: $value,
|
985
|
-
dynamic: undefined,
|
986
|
-
config: config,
|
987
|
-
toString: toString
|
988
|
-
});
|
989
|
-
};
|
990
|
-
|
991
|
-
function ParamSet(params) {
|
992
|
-
extend(this, params || {});
|
993
|
-
}
|
994
|
-
|
995
|
-
ParamSet.prototype = {
|
996
|
-
$$new: function() {
|
997
|
-
return inherit(this, extend(new ParamSet(), { $$parent: this}));
|
998
|
-
},
|
999
|
-
$$keys: function () {
|
1000
|
-
var keys = [], chain = [], parent = this,
|
1001
|
-
ignore = objectKeys(ParamSet.prototype);
|
1002
|
-
while (parent) { chain.push(parent); parent = parent.$$parent; }
|
1003
|
-
chain.reverse();
|
1004
|
-
forEach(chain, function(paramset) {
|
1005
|
-
forEach(objectKeys(paramset), function(key) {
|
1006
|
-
if (indexOf(keys, key) === -1 && indexOf(ignore, key) === -1) keys.push(key);
|
1007
|
-
});
|
1008
|
-
});
|
1009
|
-
return keys;
|
1010
|
-
},
|
1011
|
-
$$values: function(paramValues) {
|
1012
|
-
var values = {}, self = this;
|
1013
|
-
forEach(self.$$keys(), function(key) {
|
1014
|
-
values[key] = self[key].value(paramValues && paramValues[key]);
|
1015
|
-
});
|
1016
|
-
return values;
|
1017
|
-
},
|
1018
|
-
$$equals: function(paramValues1, paramValues2) {
|
1019
|
-
var equal = true, self = this;
|
1020
|
-
forEach(self.$$keys(), function(key) {
|
1021
|
-
var left = paramValues1 && paramValues1[key], right = paramValues2 && paramValues2[key];
|
1022
|
-
if (!self[key].type.equals(left, right)) equal = false;
|
1023
|
-
});
|
1024
|
-
return equal;
|
1025
|
-
},
|
1026
|
-
$$validates: function $$validate(paramValues) {
|
1027
|
-
var keys = this.$$keys(), i, param, rawVal, normalized, encoded;
|
1028
|
-
for (i = 0; i < keys.length; i++) {
|
1029
|
-
param = this[keys[i]];
|
1030
|
-
rawVal = paramValues[keys[i]];
|
1031
|
-
if ((rawVal === undefined || rawVal === null) && param.isOptional)
|
1032
|
-
break; // There was no parameter value, but the param is optional
|
1033
|
-
normalized = param.type.$normalize(rawVal);
|
1034
|
-
if (!param.type.is(normalized))
|
1035
|
-
return false; // The value was not of the correct Type, and could not be decoded to the correct Type
|
1036
|
-
encoded = param.type.encode(normalized);
|
1037
|
-
if (angular.isString(encoded) && !param.type.pattern.exec(encoded))
|
1038
|
-
return false; // The value was of the correct type, but when encoded, did not match the Type's regexp
|
1039
|
-
}
|
1040
|
-
return true;
|
1041
|
-
},
|
1042
|
-
$$parent: undefined
|
1043
|
-
};
|
1044
|
-
|
1045
|
-
this.ParamSet = ParamSet;
|
1046
|
-
}
|
1047
|
-
|
1048
|
-
// Register as a provider so it's available to other providers
|
1049
|
-
angular.module('ui.router.util').provider('$urlMatcherFactory', $UrlMatcherFactory);
|
1050
|
-
angular.module('ui.router.util').run(['$urlMatcherFactory', function($urlMatcherFactory) { }]);
|