[6a3a178] | 1 | /**
|
---|
| 2 | * @license
|
---|
| 3 | * Copyright Google LLC All Rights Reserved.
|
---|
| 4 | *
|
---|
| 5 | * Use of this source code is governed by an MIT-style license that can be
|
---|
| 6 | * found in the LICENSE file at https://angular.io/license
|
---|
| 7 | */
|
---|
| 8 | import { convertToParamMap, PRIMARY_OUTLET } from './shared';
|
---|
| 9 | import { equalArraysOrString, forEach, shallowEqual } from './utils/collection';
|
---|
| 10 | export function createEmptyUrlTree() {
|
---|
| 11 | return new UrlTree(new UrlSegmentGroup([], {}), {}, null);
|
---|
| 12 | }
|
---|
| 13 | const pathCompareMap = {
|
---|
| 14 | 'exact': equalSegmentGroups,
|
---|
| 15 | 'subset': containsSegmentGroup,
|
---|
| 16 | };
|
---|
| 17 | const paramCompareMap = {
|
---|
| 18 | 'exact': equalParams,
|
---|
| 19 | 'subset': containsParams,
|
---|
| 20 | 'ignored': () => true,
|
---|
| 21 | };
|
---|
| 22 | export function containsTree(container, containee, options) {
|
---|
| 23 | return pathCompareMap[options.paths](container.root, containee.root, options.matrixParams) &&
|
---|
| 24 | paramCompareMap[options.queryParams](container.queryParams, containee.queryParams) &&
|
---|
| 25 | !(options.fragment === 'exact' && container.fragment !== containee.fragment);
|
---|
| 26 | }
|
---|
| 27 | function equalParams(container, containee) {
|
---|
| 28 | // TODO: This does not handle array params correctly.
|
---|
| 29 | return shallowEqual(container, containee);
|
---|
| 30 | }
|
---|
| 31 | function equalSegmentGroups(container, containee, matrixParams) {
|
---|
| 32 | if (!equalPath(container.segments, containee.segments))
|
---|
| 33 | return false;
|
---|
| 34 | if (!matrixParamsMatch(container.segments, containee.segments, matrixParams)) {
|
---|
| 35 | return false;
|
---|
| 36 | }
|
---|
| 37 | if (container.numberOfChildren !== containee.numberOfChildren)
|
---|
| 38 | return false;
|
---|
| 39 | for (const c in containee.children) {
|
---|
| 40 | if (!container.children[c])
|
---|
| 41 | return false;
|
---|
| 42 | if (!equalSegmentGroups(container.children[c], containee.children[c], matrixParams))
|
---|
| 43 | return false;
|
---|
| 44 | }
|
---|
| 45 | return true;
|
---|
| 46 | }
|
---|
| 47 | function containsParams(container, containee) {
|
---|
| 48 | return Object.keys(containee).length <= Object.keys(container).length &&
|
---|
| 49 | Object.keys(containee).every(key => equalArraysOrString(container[key], containee[key]));
|
---|
| 50 | }
|
---|
| 51 | function containsSegmentGroup(container, containee, matrixParams) {
|
---|
| 52 | return containsSegmentGroupHelper(container, containee, containee.segments, matrixParams);
|
---|
| 53 | }
|
---|
| 54 | function containsSegmentGroupHelper(container, containee, containeePaths, matrixParams) {
|
---|
| 55 | if (container.segments.length > containeePaths.length) {
|
---|
| 56 | const current = container.segments.slice(0, containeePaths.length);
|
---|
| 57 | if (!equalPath(current, containeePaths))
|
---|
| 58 | return false;
|
---|
| 59 | if (containee.hasChildren())
|
---|
| 60 | return false;
|
---|
| 61 | if (!matrixParamsMatch(current, containeePaths, matrixParams))
|
---|
| 62 | return false;
|
---|
| 63 | return true;
|
---|
| 64 | }
|
---|
| 65 | else if (container.segments.length === containeePaths.length) {
|
---|
| 66 | if (!equalPath(container.segments, containeePaths))
|
---|
| 67 | return false;
|
---|
| 68 | if (!matrixParamsMatch(container.segments, containeePaths, matrixParams))
|
---|
| 69 | return false;
|
---|
| 70 | for (const c in containee.children) {
|
---|
| 71 | if (!container.children[c])
|
---|
| 72 | return false;
|
---|
| 73 | if (!containsSegmentGroup(container.children[c], containee.children[c], matrixParams)) {
|
---|
| 74 | return false;
|
---|
| 75 | }
|
---|
| 76 | }
|
---|
| 77 | return true;
|
---|
| 78 | }
|
---|
| 79 | else {
|
---|
| 80 | const current = containeePaths.slice(0, container.segments.length);
|
---|
| 81 | const next = containeePaths.slice(container.segments.length);
|
---|
| 82 | if (!equalPath(container.segments, current))
|
---|
| 83 | return false;
|
---|
| 84 | if (!matrixParamsMatch(container.segments, current, matrixParams))
|
---|
| 85 | return false;
|
---|
| 86 | if (!container.children[PRIMARY_OUTLET])
|
---|
| 87 | return false;
|
---|
| 88 | return containsSegmentGroupHelper(container.children[PRIMARY_OUTLET], containee, next, matrixParams);
|
---|
| 89 | }
|
---|
| 90 | }
|
---|
| 91 | function matrixParamsMatch(containerPaths, containeePaths, options) {
|
---|
| 92 | return containeePaths.every((containeeSegment, i) => {
|
---|
| 93 | return paramCompareMap[options](containerPaths[i].parameters, containeeSegment.parameters);
|
---|
| 94 | });
|
---|
| 95 | }
|
---|
| 96 | /**
|
---|
| 97 | * @description
|
---|
| 98 | *
|
---|
| 99 | * Represents the parsed URL.
|
---|
| 100 | *
|
---|
| 101 | * Since a router state is a tree, and the URL is nothing but a serialized state, the URL is a
|
---|
| 102 | * serialized tree.
|
---|
| 103 | * UrlTree is a data structure that provides a lot of affordances in dealing with URLs
|
---|
| 104 | *
|
---|
| 105 | * @usageNotes
|
---|
| 106 | * ### Example
|
---|
| 107 | *
|
---|
| 108 | * ```
|
---|
| 109 | * @Component({templateUrl:'template.html'})
|
---|
| 110 | * class MyComponent {
|
---|
| 111 | * constructor(router: Router) {
|
---|
| 112 | * const tree: UrlTree =
|
---|
| 113 | * router.parseUrl('/team/33/(user/victor//support:help)?debug=true#fragment');
|
---|
| 114 | * const f = tree.fragment; // return 'fragment'
|
---|
| 115 | * const q = tree.queryParams; // returns {debug: 'true'}
|
---|
| 116 | * const g: UrlSegmentGroup = tree.root.children[PRIMARY_OUTLET];
|
---|
| 117 | * const s: UrlSegment[] = g.segments; // returns 2 segments 'team' and '33'
|
---|
| 118 | * g.children[PRIMARY_OUTLET].segments; // returns 2 segments 'user' and 'victor'
|
---|
| 119 | * g.children['support'].segments; // return 1 segment 'help'
|
---|
| 120 | * }
|
---|
| 121 | * }
|
---|
| 122 | * ```
|
---|
| 123 | *
|
---|
| 124 | * @publicApi
|
---|
| 125 | */
|
---|
| 126 | export class UrlTree {
|
---|
| 127 | /** @internal */
|
---|
| 128 | constructor(
|
---|
| 129 | /** The root segment group of the URL tree */
|
---|
| 130 | root,
|
---|
| 131 | /** The query params of the URL */
|
---|
| 132 | queryParams,
|
---|
| 133 | /** The fragment of the URL */
|
---|
| 134 | fragment) {
|
---|
| 135 | this.root = root;
|
---|
| 136 | this.queryParams = queryParams;
|
---|
| 137 | this.fragment = fragment;
|
---|
| 138 | }
|
---|
| 139 | get queryParamMap() {
|
---|
| 140 | if (!this._queryParamMap) {
|
---|
| 141 | this._queryParamMap = convertToParamMap(this.queryParams);
|
---|
| 142 | }
|
---|
| 143 | return this._queryParamMap;
|
---|
| 144 | }
|
---|
| 145 | /** @docsNotRequired */
|
---|
| 146 | toString() {
|
---|
| 147 | return DEFAULT_SERIALIZER.serialize(this);
|
---|
| 148 | }
|
---|
| 149 | }
|
---|
| 150 | /**
|
---|
| 151 | * @description
|
---|
| 152 | *
|
---|
| 153 | * Represents the parsed URL segment group.
|
---|
| 154 | *
|
---|
| 155 | * See `UrlTree` for more information.
|
---|
| 156 | *
|
---|
| 157 | * @publicApi
|
---|
| 158 | */
|
---|
| 159 | export class UrlSegmentGroup {
|
---|
| 160 | constructor(
|
---|
| 161 | /** The URL segments of this group. See `UrlSegment` for more information */
|
---|
| 162 | segments,
|
---|
| 163 | /** The list of children of this group */
|
---|
| 164 | children) {
|
---|
| 165 | this.segments = segments;
|
---|
| 166 | this.children = children;
|
---|
| 167 | /** The parent node in the url tree */
|
---|
| 168 | this.parent = null;
|
---|
| 169 | forEach(children, (v, k) => v.parent = this);
|
---|
| 170 | }
|
---|
| 171 | /** Whether the segment has child segments */
|
---|
| 172 | hasChildren() {
|
---|
| 173 | return this.numberOfChildren > 0;
|
---|
| 174 | }
|
---|
| 175 | /** Number of child segments */
|
---|
| 176 | get numberOfChildren() {
|
---|
| 177 | return Object.keys(this.children).length;
|
---|
| 178 | }
|
---|
| 179 | /** @docsNotRequired */
|
---|
| 180 | toString() {
|
---|
| 181 | return serializePaths(this);
|
---|
| 182 | }
|
---|
| 183 | }
|
---|
| 184 | /**
|
---|
| 185 | * @description
|
---|
| 186 | *
|
---|
| 187 | * Represents a single URL segment.
|
---|
| 188 | *
|
---|
| 189 | * A UrlSegment is a part of a URL between the two slashes. It contains a path and the matrix
|
---|
| 190 | * parameters associated with the segment.
|
---|
| 191 | *
|
---|
| 192 | * @usageNotes
|
---|
| 193 | * ### Example
|
---|
| 194 | *
|
---|
| 195 | * ```
|
---|
| 196 | * @Component({templateUrl:'template.html'})
|
---|
| 197 | * class MyComponent {
|
---|
| 198 | * constructor(router: Router) {
|
---|
| 199 | * const tree: UrlTree = router.parseUrl('/team;id=33');
|
---|
| 200 | * const g: UrlSegmentGroup = tree.root.children[PRIMARY_OUTLET];
|
---|
| 201 | * const s: UrlSegment[] = g.segments;
|
---|
| 202 | * s[0].path; // returns 'team'
|
---|
| 203 | * s[0].parameters; // returns {id: 33}
|
---|
| 204 | * }
|
---|
| 205 | * }
|
---|
| 206 | * ```
|
---|
| 207 | *
|
---|
| 208 | * @publicApi
|
---|
| 209 | */
|
---|
| 210 | export class UrlSegment {
|
---|
| 211 | constructor(
|
---|
| 212 | /** The path part of a URL segment */
|
---|
| 213 | path,
|
---|
| 214 | /** The matrix parameters associated with a segment */
|
---|
| 215 | parameters) {
|
---|
| 216 | this.path = path;
|
---|
| 217 | this.parameters = parameters;
|
---|
| 218 | }
|
---|
| 219 | get parameterMap() {
|
---|
| 220 | if (!this._parameterMap) {
|
---|
| 221 | this._parameterMap = convertToParamMap(this.parameters);
|
---|
| 222 | }
|
---|
| 223 | return this._parameterMap;
|
---|
| 224 | }
|
---|
| 225 | /** @docsNotRequired */
|
---|
| 226 | toString() {
|
---|
| 227 | return serializePath(this);
|
---|
| 228 | }
|
---|
| 229 | }
|
---|
| 230 | export function equalSegments(as, bs) {
|
---|
| 231 | return equalPath(as, bs) && as.every((a, i) => shallowEqual(a.parameters, bs[i].parameters));
|
---|
| 232 | }
|
---|
| 233 | export function equalPath(as, bs) {
|
---|
| 234 | if (as.length !== bs.length)
|
---|
| 235 | return false;
|
---|
| 236 | return as.every((a, i) => a.path === bs[i].path);
|
---|
| 237 | }
|
---|
| 238 | export function mapChildrenIntoArray(segment, fn) {
|
---|
| 239 | let res = [];
|
---|
| 240 | forEach(segment.children, (child, childOutlet) => {
|
---|
| 241 | if (childOutlet === PRIMARY_OUTLET) {
|
---|
| 242 | res = res.concat(fn(child, childOutlet));
|
---|
| 243 | }
|
---|
| 244 | });
|
---|
| 245 | forEach(segment.children, (child, childOutlet) => {
|
---|
| 246 | if (childOutlet !== PRIMARY_OUTLET) {
|
---|
| 247 | res = res.concat(fn(child, childOutlet));
|
---|
| 248 | }
|
---|
| 249 | });
|
---|
| 250 | return res;
|
---|
| 251 | }
|
---|
| 252 | /**
|
---|
| 253 | * @description
|
---|
| 254 | *
|
---|
| 255 | * Serializes and deserializes a URL string into a URL tree.
|
---|
| 256 | *
|
---|
| 257 | * The url serialization strategy is customizable. You can
|
---|
| 258 | * make all URLs case insensitive by providing a custom UrlSerializer.
|
---|
| 259 | *
|
---|
| 260 | * See `DefaultUrlSerializer` for an example of a URL serializer.
|
---|
| 261 | *
|
---|
| 262 | * @publicApi
|
---|
| 263 | */
|
---|
| 264 | export class UrlSerializer {
|
---|
| 265 | }
|
---|
| 266 | /**
|
---|
| 267 | * @description
|
---|
| 268 | *
|
---|
| 269 | * A default implementation of the `UrlSerializer`.
|
---|
| 270 | *
|
---|
| 271 | * Example URLs:
|
---|
| 272 | *
|
---|
| 273 | * ```
|
---|
| 274 | * /inbox/33(popup:compose)
|
---|
| 275 | * /inbox/33;open=true/messages/44
|
---|
| 276 | * ```
|
---|
| 277 | *
|
---|
| 278 | * DefaultUrlSerializer uses parentheses to serialize secondary segments (e.g., popup:compose), the
|
---|
| 279 | * colon syntax to specify the outlet, and the ';parameter=value' syntax (e.g., open=true) to
|
---|
| 280 | * specify route specific parameters.
|
---|
| 281 | *
|
---|
| 282 | * @publicApi
|
---|
| 283 | */
|
---|
| 284 | export class DefaultUrlSerializer {
|
---|
| 285 | /** Parses a url into a `UrlTree` */
|
---|
| 286 | parse(url) {
|
---|
| 287 | const p = new UrlParser(url);
|
---|
| 288 | return new UrlTree(p.parseRootSegment(), p.parseQueryParams(), p.parseFragment());
|
---|
| 289 | }
|
---|
| 290 | /** Converts a `UrlTree` into a url */
|
---|
| 291 | serialize(tree) {
|
---|
| 292 | const segment = `/${serializeSegment(tree.root, true)}`;
|
---|
| 293 | const query = serializeQueryParams(tree.queryParams);
|
---|
| 294 | const fragment = typeof tree.fragment === `string` ? `#${encodeUriFragment(tree.fragment)}` : '';
|
---|
| 295 | return `${segment}${query}${fragment}`;
|
---|
| 296 | }
|
---|
| 297 | }
|
---|
| 298 | const DEFAULT_SERIALIZER = new DefaultUrlSerializer();
|
---|
| 299 | export function serializePaths(segment) {
|
---|
| 300 | return segment.segments.map(p => serializePath(p)).join('/');
|
---|
| 301 | }
|
---|
| 302 | function serializeSegment(segment, root) {
|
---|
| 303 | if (!segment.hasChildren()) {
|
---|
| 304 | return serializePaths(segment);
|
---|
| 305 | }
|
---|
| 306 | if (root) {
|
---|
| 307 | const primary = segment.children[PRIMARY_OUTLET] ?
|
---|
| 308 | serializeSegment(segment.children[PRIMARY_OUTLET], false) :
|
---|
| 309 | '';
|
---|
| 310 | const children = [];
|
---|
| 311 | forEach(segment.children, (v, k) => {
|
---|
| 312 | if (k !== PRIMARY_OUTLET) {
|
---|
| 313 | children.push(`${k}:${serializeSegment(v, false)}`);
|
---|
| 314 | }
|
---|
| 315 | });
|
---|
| 316 | return children.length > 0 ? `${primary}(${children.join('//')})` : primary;
|
---|
| 317 | }
|
---|
| 318 | else {
|
---|
| 319 | const children = mapChildrenIntoArray(segment, (v, k) => {
|
---|
| 320 | if (k === PRIMARY_OUTLET) {
|
---|
| 321 | return [serializeSegment(segment.children[PRIMARY_OUTLET], false)];
|
---|
| 322 | }
|
---|
| 323 | return [`${k}:${serializeSegment(v, false)}`];
|
---|
| 324 | });
|
---|
| 325 | // use no parenthesis if the only child is a primary outlet route
|
---|
| 326 | if (Object.keys(segment.children).length === 1 && segment.children[PRIMARY_OUTLET] != null) {
|
---|
| 327 | return `${serializePaths(segment)}/${children[0]}`;
|
---|
| 328 | }
|
---|
| 329 | return `${serializePaths(segment)}/(${children.join('//')})`;
|
---|
| 330 | }
|
---|
| 331 | }
|
---|
| 332 | /**
|
---|
| 333 | * Encodes a URI string with the default encoding. This function will only ever be called from
|
---|
| 334 | * `encodeUriQuery` or `encodeUriSegment` as it's the base set of encodings to be used. We need
|
---|
| 335 | * a custom encoding because encodeURIComponent is too aggressive and encodes stuff that doesn't
|
---|
| 336 | * have to be encoded per https://url.spec.whatwg.org.
|
---|
| 337 | */
|
---|
| 338 | function encodeUriString(s) {
|
---|
| 339 | return encodeURIComponent(s)
|
---|
| 340 | .replace(/%40/g, '@')
|
---|
| 341 | .replace(/%3A/gi, ':')
|
---|
| 342 | .replace(/%24/g, '$')
|
---|
| 343 | .replace(/%2C/gi, ',');
|
---|
| 344 | }
|
---|
| 345 | /**
|
---|
| 346 | * This function should be used to encode both keys and values in a query string key/value. In
|
---|
| 347 | * the following URL, you need to call encodeUriQuery on "k" and "v":
|
---|
| 348 | *
|
---|
| 349 | * http://www.site.org/html;mk=mv?k=v#f
|
---|
| 350 | */
|
---|
| 351 | export function encodeUriQuery(s) {
|
---|
| 352 | return encodeUriString(s).replace(/%3B/gi, ';');
|
---|
| 353 | }
|
---|
| 354 | /**
|
---|
| 355 | * This function should be used to encode a URL fragment. In the following URL, you need to call
|
---|
| 356 | * encodeUriFragment on "f":
|
---|
| 357 | *
|
---|
| 358 | * http://www.site.org/html;mk=mv?k=v#f
|
---|
| 359 | */
|
---|
| 360 | export function encodeUriFragment(s) {
|
---|
| 361 | return encodeURI(s);
|
---|
| 362 | }
|
---|
| 363 | /**
|
---|
| 364 | * This function should be run on any URI segment as well as the key and value in a key/value
|
---|
| 365 | * pair for matrix params. In the following URL, you need to call encodeUriSegment on "html",
|
---|
| 366 | * "mk", and "mv":
|
---|
| 367 | *
|
---|
| 368 | * http://www.site.org/html;mk=mv?k=v#f
|
---|
| 369 | */
|
---|
| 370 | export function encodeUriSegment(s) {
|
---|
| 371 | return encodeUriString(s).replace(/\(/g, '%28').replace(/\)/g, '%29').replace(/%26/gi, '&');
|
---|
| 372 | }
|
---|
| 373 | export function decode(s) {
|
---|
| 374 | return decodeURIComponent(s);
|
---|
| 375 | }
|
---|
| 376 | // Query keys/values should have the "+" replaced first, as "+" in a query string is " ".
|
---|
| 377 | // decodeURIComponent function will not decode "+" as a space.
|
---|
| 378 | export function decodeQuery(s) {
|
---|
| 379 | return decode(s.replace(/\+/g, '%20'));
|
---|
| 380 | }
|
---|
| 381 | export function serializePath(path) {
|
---|
| 382 | return `${encodeUriSegment(path.path)}${serializeMatrixParams(path.parameters)}`;
|
---|
| 383 | }
|
---|
| 384 | function serializeMatrixParams(params) {
|
---|
| 385 | return Object.keys(params)
|
---|
| 386 | .map(key => `;${encodeUriSegment(key)}=${encodeUriSegment(params[key])}`)
|
---|
| 387 | .join('');
|
---|
| 388 | }
|
---|
| 389 | function serializeQueryParams(params) {
|
---|
| 390 | const strParams = Object.keys(params)
|
---|
| 391 | .map((name) => {
|
---|
| 392 | const value = params[name];
|
---|
| 393 | return Array.isArray(value) ?
|
---|
| 394 | value.map(v => `${encodeUriQuery(name)}=${encodeUriQuery(v)}`).join('&') :
|
---|
| 395 | `${encodeUriQuery(name)}=${encodeUriQuery(value)}`;
|
---|
| 396 | })
|
---|
| 397 | .filter(s => !!s);
|
---|
| 398 | return strParams.length ? `?${strParams.join('&')}` : '';
|
---|
| 399 | }
|
---|
| 400 | const SEGMENT_RE = /^[^\/()?;=#]+/;
|
---|
| 401 | function matchSegments(str) {
|
---|
| 402 | const match = str.match(SEGMENT_RE);
|
---|
| 403 | return match ? match[0] : '';
|
---|
| 404 | }
|
---|
| 405 | const QUERY_PARAM_RE = /^[^=?&#]+/;
|
---|
| 406 | // Return the name of the query param at the start of the string or an empty string
|
---|
| 407 | function matchQueryParams(str) {
|
---|
| 408 | const match = str.match(QUERY_PARAM_RE);
|
---|
| 409 | return match ? match[0] : '';
|
---|
| 410 | }
|
---|
| 411 | const QUERY_PARAM_VALUE_RE = /^[^?&#]+/;
|
---|
| 412 | // Return the value of the query param at the start of the string or an empty string
|
---|
| 413 | function matchUrlQueryParamValue(str) {
|
---|
| 414 | const match = str.match(QUERY_PARAM_VALUE_RE);
|
---|
| 415 | return match ? match[0] : '';
|
---|
| 416 | }
|
---|
| 417 | class UrlParser {
|
---|
| 418 | constructor(url) {
|
---|
| 419 | this.url = url;
|
---|
| 420 | this.remaining = url;
|
---|
| 421 | }
|
---|
| 422 | parseRootSegment() {
|
---|
| 423 | this.consumeOptional('/');
|
---|
| 424 | if (this.remaining === '' || this.peekStartsWith('?') || this.peekStartsWith('#')) {
|
---|
| 425 | return new UrlSegmentGroup([], {});
|
---|
| 426 | }
|
---|
| 427 | // The root segment group never has segments
|
---|
| 428 | return new UrlSegmentGroup([], this.parseChildren());
|
---|
| 429 | }
|
---|
| 430 | parseQueryParams() {
|
---|
| 431 | const params = {};
|
---|
| 432 | if (this.consumeOptional('?')) {
|
---|
| 433 | do {
|
---|
| 434 | this.parseQueryParam(params);
|
---|
| 435 | } while (this.consumeOptional('&'));
|
---|
| 436 | }
|
---|
| 437 | return params;
|
---|
| 438 | }
|
---|
| 439 | parseFragment() {
|
---|
| 440 | return this.consumeOptional('#') ? decodeURIComponent(this.remaining) : null;
|
---|
| 441 | }
|
---|
| 442 | parseChildren() {
|
---|
| 443 | if (this.remaining === '') {
|
---|
| 444 | return {};
|
---|
| 445 | }
|
---|
| 446 | this.consumeOptional('/');
|
---|
| 447 | const segments = [];
|
---|
| 448 | if (!this.peekStartsWith('(')) {
|
---|
| 449 | segments.push(this.parseSegment());
|
---|
| 450 | }
|
---|
| 451 | while (this.peekStartsWith('/') && !this.peekStartsWith('//') && !this.peekStartsWith('/(')) {
|
---|
| 452 | this.capture('/');
|
---|
| 453 | segments.push(this.parseSegment());
|
---|
| 454 | }
|
---|
| 455 | let children = {};
|
---|
| 456 | if (this.peekStartsWith('/(')) {
|
---|
| 457 | this.capture('/');
|
---|
| 458 | children = this.parseParens(true);
|
---|
| 459 | }
|
---|
| 460 | let res = {};
|
---|
| 461 | if (this.peekStartsWith('(')) {
|
---|
| 462 | res = this.parseParens(false);
|
---|
| 463 | }
|
---|
| 464 | if (segments.length > 0 || Object.keys(children).length > 0) {
|
---|
| 465 | res[PRIMARY_OUTLET] = new UrlSegmentGroup(segments, children);
|
---|
| 466 | }
|
---|
| 467 | return res;
|
---|
| 468 | }
|
---|
| 469 | // parse a segment with its matrix parameters
|
---|
| 470 | // ie `name;k1=v1;k2`
|
---|
| 471 | parseSegment() {
|
---|
| 472 | const path = matchSegments(this.remaining);
|
---|
| 473 | if (path === '' && this.peekStartsWith(';')) {
|
---|
| 474 | throw new Error(`Empty path url segment cannot have parameters: '${this.remaining}'.`);
|
---|
| 475 | }
|
---|
| 476 | this.capture(path);
|
---|
| 477 | return new UrlSegment(decode(path), this.parseMatrixParams());
|
---|
| 478 | }
|
---|
| 479 | parseMatrixParams() {
|
---|
| 480 | const params = {};
|
---|
| 481 | while (this.consumeOptional(';')) {
|
---|
| 482 | this.parseParam(params);
|
---|
| 483 | }
|
---|
| 484 | return params;
|
---|
| 485 | }
|
---|
| 486 | parseParam(params) {
|
---|
| 487 | const key = matchSegments(this.remaining);
|
---|
| 488 | if (!key) {
|
---|
| 489 | return;
|
---|
| 490 | }
|
---|
| 491 | this.capture(key);
|
---|
| 492 | let value = '';
|
---|
| 493 | if (this.consumeOptional('=')) {
|
---|
| 494 | const valueMatch = matchSegments(this.remaining);
|
---|
| 495 | if (valueMatch) {
|
---|
| 496 | value = valueMatch;
|
---|
| 497 | this.capture(value);
|
---|
| 498 | }
|
---|
| 499 | }
|
---|
| 500 | params[decode(key)] = decode(value);
|
---|
| 501 | }
|
---|
| 502 | // Parse a single query parameter `name[=value]`
|
---|
| 503 | parseQueryParam(params) {
|
---|
| 504 | const key = matchQueryParams(this.remaining);
|
---|
| 505 | if (!key) {
|
---|
| 506 | return;
|
---|
| 507 | }
|
---|
| 508 | this.capture(key);
|
---|
| 509 | let value = '';
|
---|
| 510 | if (this.consumeOptional('=')) {
|
---|
| 511 | const valueMatch = matchUrlQueryParamValue(this.remaining);
|
---|
| 512 | if (valueMatch) {
|
---|
| 513 | value = valueMatch;
|
---|
| 514 | this.capture(value);
|
---|
| 515 | }
|
---|
| 516 | }
|
---|
| 517 | const decodedKey = decodeQuery(key);
|
---|
| 518 | const decodedVal = decodeQuery(value);
|
---|
| 519 | if (params.hasOwnProperty(decodedKey)) {
|
---|
| 520 | // Append to existing values
|
---|
| 521 | let currentVal = params[decodedKey];
|
---|
| 522 | if (!Array.isArray(currentVal)) {
|
---|
| 523 | currentVal = [currentVal];
|
---|
| 524 | params[decodedKey] = currentVal;
|
---|
| 525 | }
|
---|
| 526 | currentVal.push(decodedVal);
|
---|
| 527 | }
|
---|
| 528 | else {
|
---|
| 529 | // Create a new value
|
---|
| 530 | params[decodedKey] = decodedVal;
|
---|
| 531 | }
|
---|
| 532 | }
|
---|
| 533 | // parse `(a/b//outlet_name:c/d)`
|
---|
| 534 | parseParens(allowPrimary) {
|
---|
| 535 | const segments = {};
|
---|
| 536 | this.capture('(');
|
---|
| 537 | while (!this.consumeOptional(')') && this.remaining.length > 0) {
|
---|
| 538 | const path = matchSegments(this.remaining);
|
---|
| 539 | const next = this.remaining[path.length];
|
---|
| 540 | // if is is not one of these characters, then the segment was unescaped
|
---|
| 541 | // or the group was not closed
|
---|
| 542 | if (next !== '/' && next !== ')' && next !== ';') {
|
---|
| 543 | throw new Error(`Cannot parse url '${this.url}'`);
|
---|
| 544 | }
|
---|
| 545 | let outletName = undefined;
|
---|
| 546 | if (path.indexOf(':') > -1) {
|
---|
| 547 | outletName = path.substr(0, path.indexOf(':'));
|
---|
| 548 | this.capture(outletName);
|
---|
| 549 | this.capture(':');
|
---|
| 550 | }
|
---|
| 551 | else if (allowPrimary) {
|
---|
| 552 | outletName = PRIMARY_OUTLET;
|
---|
| 553 | }
|
---|
| 554 | const children = this.parseChildren();
|
---|
| 555 | segments[outletName] = Object.keys(children).length === 1 ? children[PRIMARY_OUTLET] :
|
---|
| 556 | new UrlSegmentGroup([], children);
|
---|
| 557 | this.consumeOptional('//');
|
---|
| 558 | }
|
---|
| 559 | return segments;
|
---|
| 560 | }
|
---|
| 561 | peekStartsWith(str) {
|
---|
| 562 | return this.remaining.startsWith(str);
|
---|
| 563 | }
|
---|
| 564 | // Consumes the prefix when it is present and returns whether it has been consumed
|
---|
| 565 | consumeOptional(str) {
|
---|
| 566 | if (this.peekStartsWith(str)) {
|
---|
| 567 | this.remaining = this.remaining.substring(str.length);
|
---|
| 568 | return true;
|
---|
| 569 | }
|
---|
| 570 | return false;
|
---|
| 571 | }
|
---|
| 572 | capture(str) {
|
---|
| 573 | if (!this.consumeOptional(str)) {
|
---|
| 574 | throw new Error(`Expected "${str}".`);
|
---|
| 575 | }
|
---|
| 576 | }
|
---|
| 577 | }
|
---|
| 578 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"url_tree.js","sourceRoot":"","sources":["../../../../../../packages/router/src/url_tree.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {convertToParamMap, ParamMap, Params, PRIMARY_OUTLET} from './shared';\nimport {equalArraysOrString, forEach, shallowEqual} from './utils/collection';\n\nexport function createEmptyUrlTree() {\n  return new UrlTree(new UrlSegmentGroup([], {}), {}, null);\n}\n\n/**\n * A set of options which specify how to determine if a `UrlTree` is active, given the `UrlTree`\n * for the current router state.\n *\n * @publicApi\n * @see Router.isActive\n */\nexport interface IsActiveMatchOptions {\n  /**\n   * Defines the strategy for comparing the matrix parameters of two `UrlTree`s.\n   *\n   * The matrix parameter matching is dependent on the strategy for matching the\n   * segments. That is, if the `paths` option is set to `'subset'`, only\n   * the matrix parameters of the matching segments will be compared.\n   *\n   * - `'exact'`: Requires that matching segments also have exact matrix parameter\n   * matches.\n   * - `'subset'`: The matching segments in the router's active `UrlTree` may contain\n   * extra matrix parameters, but those that exist in the `UrlTree` in question must match.\n   * - `'ignored'`: When comparing `UrlTree`s, matrix params will be ignored.\n   */\n  matrixParams: 'exact'|'subset'|'ignored';\n  /**\n   * Defines the strategy for comparing the query parameters of two `UrlTree`s.\n   *\n   * - `'exact'`: the query parameters must match exactly.\n   * - `'subset'`: the active `UrlTree` may contain extra parameters,\n   * but must match the key and value of any that exist in the `UrlTree` in question.\n   * - `'ignored'`: When comparing `UrlTree`s, query params will be ignored.\n   */\n  queryParams: 'exact'|'subset'|'ignored';\n  /**\n   * Defines the strategy for comparing the `UrlSegment`s of the `UrlTree`s.\n   *\n   * - `'exact'`: all segments in each `UrlTree` must match.\n   * - `'subset'`: a `UrlTree` will be determined to be active if it\n   * is a subtree of the active route. That is, the active route may contain extra\n   * segments, but must at least have all the segements of the `UrlTree` in question.\n   */\n  paths: 'exact'|'subset';\n  /**\n   * - 'exact'`: indicates that the `UrlTree` fragments must be equal.\n   * - `'ignored'`: the fragments will not be compared when determining if a\n   * `UrlTree` is active.\n   */\n  fragment: 'exact'|'ignored';\n}\n\ntype ParamMatchOptions = 'exact'|'subset'|'ignored';\n\ntype PathCompareFn =\n    (container: UrlSegmentGroup, containee: UrlSegmentGroup, matrixParams: ParamMatchOptions) =>\n        boolean;\ntype ParamCompareFn = (container: Params, containee: Params) => boolean;\n\nconst pathCompareMap: Record<IsActiveMatchOptions['paths'], PathCompareFn> = {\n  'exact': equalSegmentGroups,\n  'subset': containsSegmentGroup,\n};\nconst paramCompareMap: Record<ParamMatchOptions, ParamCompareFn> = {\n  'exact': equalParams,\n  'subset': containsParams,\n  'ignored': () => true,\n};\n\nexport function containsTree(\n    container: UrlTree, containee: UrlTree, options: IsActiveMatchOptions): boolean {\n  return pathCompareMap[options.paths](container.root, containee.root, options.matrixParams) &&\n      paramCompareMap[options.queryParams](container.queryParams, containee.queryParams) &&\n      !(options.fragment === 'exact' && container.fragment !== containee.fragment);\n}\n\nfunction equalParams(container: Params, containee: Params): boolean {\n  // TODO: This does not handle array params correctly.\n  return shallowEqual(container, containee);\n}\n\nfunction equalSegmentGroups(\n    container: UrlSegmentGroup, containee: UrlSegmentGroup,\n    matrixParams: ParamMatchOptions): boolean {\n  if (!equalPath(container.segments, containee.segments)) return false;\n  if (!matrixParamsMatch(container.segments, containee.segments, matrixParams)) {\n    return false;\n  }\n  if (container.numberOfChildren !== containee.numberOfChildren) return false;\n  for (const c in containee.children) {\n    if (!container.children[c]) return false;\n    if (!equalSegmentGroups(container.children[c], containee.children[c], matrixParams))\n      return false;\n  }\n  return true;\n}\n\nfunction containsParams(container: Params, containee: Params): boolean {\n  return Object.keys(containee).length <= Object.keys(container).length &&\n      Object.keys(containee).every(key => equalArraysOrString(container[key], containee[key]));\n}\n\nfunction containsSegmentGroup(\n    container: UrlSegmentGroup, containee: UrlSegmentGroup,\n    matrixParams: ParamMatchOptions): boolean {\n  return containsSegmentGroupHelper(container, containee, containee.segments, matrixParams);\n}\n\nfunction containsSegmentGroupHelper(\n    container: UrlSegmentGroup, containee: UrlSegmentGroup, containeePaths: UrlSegment[],\n    matrixParams: ParamMatchOptions): boolean {\n  if (container.segments.length > containeePaths.length) {\n    const current = container.segments.slice(0, containeePaths.length);\n    if (!equalPath(current, containeePaths)) return false;\n    if (containee.hasChildren()) return false;\n    if (!matrixParamsMatch(current, containeePaths, matrixParams)) return false;\n    return true;\n\n  } else if (container.segments.length === containeePaths.length) {\n    if (!equalPath(container.segments, containeePaths)) return false;\n    if (!matrixParamsMatch(container.segments, containeePaths, matrixParams)) return false;\n    for (const c in containee.children) {\n      if (!container.children[c]) return false;\n      if (!containsSegmentGroup(container.children[c], containee.children[c], matrixParams)) {\n        return false;\n      }\n    }\n    return true;\n\n  } else {\n    const current = containeePaths.slice(0, container.segments.length);\n    const next = containeePaths.slice(container.segments.length);\n    if (!equalPath(container.segments, current)) return false;\n    if (!matrixParamsMatch(container.segments, current, matrixParams)) return false;\n    if (!container.children[PRIMARY_OUTLET]) return false;\n    return containsSegmentGroupHelper(\n        container.children[PRIMARY_OUTLET], containee, next, matrixParams);\n  }\n}\n\nfunction matrixParamsMatch(\n    containerPaths: UrlSegment[], containeePaths: UrlSegment[], options: ParamMatchOptions) {\n  return containeePaths.every((containeeSegment, i) => {\n    return paramCompareMap[options](containerPaths[i].parameters, containeeSegment.parameters);\n  });\n}\n\n/**\n * @description\n *\n * Represents the parsed URL.\n *\n * Since a router state is a tree, and the URL is nothing but a serialized state, the URL is a\n * serialized tree.\n * UrlTree is a data structure that provides a lot of affordances in dealing with URLs\n *\n * @usageNotes\n * ### Example\n *\n * ```\n * @Component({templateUrl:'template.html'})\n * class MyComponent {\n *   constructor(router: Router) {\n *     const tree: UrlTree =\n *       router.parseUrl('/team/33/(user/victor//support:help)?debug=true#fragment');\n *     const f = tree.fragment; // return 'fragment'\n *     const q = tree.queryParams; // returns {debug: 'true'}\n *     const g: UrlSegmentGroup = tree.root.children[PRIMARY_OUTLET];\n *     const s: UrlSegment[] = g.segments; // returns 2 segments 'team' and '33'\n *     g.children[PRIMARY_OUTLET].segments; // returns 2 segments 'user' and 'victor'\n *     g.children['support'].segments; // return 1 segment 'help'\n *   }\n * }\n * ```\n *\n * @publicApi\n */\nexport class UrlTree {\n  /** @internal */\n  // TODO(issue/24571): remove '!'.\n  _queryParamMap!: ParamMap;\n\n  /** @internal */\n  constructor(\n      /** The root segment group of the URL tree */\n      public root: UrlSegmentGroup,\n      /** The query params of the URL */\n      public queryParams: Params,\n      /** The fragment of the URL */\n      public fragment: string|null) {}\n\n  get queryParamMap(): ParamMap {\n    if (!this._queryParamMap) {\n      this._queryParamMap = convertToParamMap(this.queryParams);\n    }\n    return this._queryParamMap;\n  }\n\n  /** @docsNotRequired */\n  toString(): string {\n    return DEFAULT_SERIALIZER.serialize(this);\n  }\n}\n\n/**\n * @description\n *\n * Represents the parsed URL segment group.\n *\n * See `UrlTree` for more information.\n *\n * @publicApi\n */\nexport class UrlSegmentGroup {\n  /** @internal */\n  _sourceSegment?: UrlSegmentGroup;\n  /** @internal */\n  _segmentIndexShift?: number;\n  /** The parent node in the url tree */\n  parent: UrlSegmentGroup|null = null;\n\n  constructor(\n      /** The URL segments of this group. See `UrlSegment` for more information */\n      public segments: UrlSegment[],\n      /** The list of children of this group */\n      public children: {[key: string]: UrlSegmentGroup}) {\n    forEach(children, (v: any, k: any) => v.parent = this);\n  }\n\n  /** Whether the segment has child segments */\n  hasChildren(): boolean {\n    return this.numberOfChildren > 0;\n  }\n\n  /** Number of child segments */\n  get numberOfChildren(): number {\n    return Object.keys(this.children).length;\n  }\n\n  /** @docsNotRequired */\n  toString(): string {\n    return serializePaths(this);\n  }\n}\n\n\n/**\n * @description\n *\n * Represents a single URL segment.\n *\n * A UrlSegment is a part of a URL between the two slashes. It contains a path and the matrix\n * parameters associated with the segment.\n *\n * @usageNotes\n * ### Example\n *\n * ```\n * @Component({templateUrl:'template.html'})\n * class MyComponent {\n *   constructor(router: Router) {\n *     const tree: UrlTree = router.parseUrl('/team;id=33');\n *     const g: UrlSegmentGroup = tree.root.children[PRIMARY_OUTLET];\n *     const s: UrlSegment[] = g.segments;\n *     s[0].path; // returns 'team'\n *     s[0].parameters; // returns {id: 33}\n *   }\n * }\n * ```\n *\n * @publicApi\n */\nexport class UrlSegment {\n  /** @internal */\n  // TODO(issue/24571): remove '!'.\n  _parameterMap!: ParamMap;\n\n  constructor(\n      /** The path part of a URL segment */\n      public path: string,\n\n      /** The matrix parameters associated with a segment */\n      public parameters: {[name: string]: string}) {}\n\n  get parameterMap() {\n    if (!this._parameterMap) {\n      this._parameterMap = convertToParamMap(this.parameters);\n    }\n    return this._parameterMap;\n  }\n\n  /** @docsNotRequired */\n  toString(): string {\n    return serializePath(this);\n  }\n}\n\nexport function equalSegments(as: UrlSegment[], bs: UrlSegment[]): boolean {\n  return equalPath(as, bs) && as.every((a, i) => shallowEqual(a.parameters, bs[i].parameters));\n}\n\nexport function equalPath(as: UrlSegment[], bs: UrlSegment[]): boolean {\n  if (as.length !== bs.length) return false;\n  return as.every((a, i) => a.path === bs[i].path);\n}\n\nexport function mapChildrenIntoArray<T>(\n    segment: UrlSegmentGroup, fn: (v: UrlSegmentGroup, k: string) => T[]): T[] {\n  let res: T[] = [];\n  forEach(segment.children, (child: UrlSegmentGroup, childOutlet: string) => {\n    if (childOutlet === PRIMARY_OUTLET) {\n      res = res.concat(fn(child, childOutlet));\n    }\n  });\n  forEach(segment.children, (child: UrlSegmentGroup, childOutlet: string) => {\n    if (childOutlet !== PRIMARY_OUTLET) {\n      res = res.concat(fn(child, childOutlet));\n    }\n  });\n  return res;\n}\n\n\n/**\n * @description\n *\n * Serializes and deserializes a URL string into a URL tree.\n *\n * The url serialization strategy is customizable. You can\n * make all URLs case insensitive by providing a custom UrlSerializer.\n *\n * See `DefaultUrlSerializer` for an example of a URL serializer.\n *\n * @publicApi\n */\nexport abstract class UrlSerializer {\n  /** Parse a url into a `UrlTree` */\n  abstract parse(url: string): UrlTree;\n\n  /** Converts a `UrlTree` into a url */\n  abstract serialize(tree: UrlTree): string;\n}\n\n/**\n * @description\n *\n * A default implementation of the `UrlSerializer`.\n *\n * Example URLs:\n *\n * ```\n * /inbox/33(popup:compose)\n * /inbox/33;open=true/messages/44\n * ```\n *\n * DefaultUrlSerializer uses parentheses to serialize secondary segments (e.g., popup:compose), the\n * colon syntax to specify the outlet, and the ';parameter=value' syntax (e.g., open=true) to\n * specify route specific parameters.\n *\n * @publicApi\n */\nexport class DefaultUrlSerializer implements UrlSerializer {\n  /** Parses a url into a `UrlTree` */\n  parse(url: string): UrlTree {\n    const p = new UrlParser(url);\n    return new UrlTree(p.parseRootSegment(), p.parseQueryParams(), p.parseFragment());\n  }\n\n  /** Converts a `UrlTree` into a url */\n  serialize(tree: UrlTree): string {\n    const segment = `/${serializeSegment(tree.root, true)}`;\n    const query = serializeQueryParams(tree.queryParams);\n    const fragment =\n        typeof tree.fragment === `string` ? `#${encodeUriFragment(tree.fragment)}` : '';\n\n    return `${segment}${query}${fragment}`;\n  }\n}\n\nconst DEFAULT_SERIALIZER = new DefaultUrlSerializer();\n\nexport function serializePaths(segment: UrlSegmentGroup): string {\n  return segment.segments.map(p => serializePath(p)).join('/');\n}\n\nfunction serializeSegment(segment: UrlSegmentGroup, root: boolean): string {\n  if (!segment.hasChildren()) {\n    return serializePaths(segment);\n  }\n\n  if (root) {\n    const primary = segment.children[PRIMARY_OUTLET] ?\n        serializeSegment(segment.children[PRIMARY_OUTLET], false) :\n        '';\n    const children: string[] = [];\n\n    forEach(segment.children, (v: UrlSegmentGroup, k: string) => {\n      if (k !== PRIMARY_OUTLET) {\n        children.push(`${k}:${serializeSegment(v, false)}`);\n      }\n    });\n\n    return children.length > 0 ? `${primary}(${children.join('//')})` : primary;\n\n  } else {\n    const children = mapChildrenIntoArray(segment, (v: UrlSegmentGroup, k: string) => {\n      if (k === PRIMARY_OUTLET) {\n        return [serializeSegment(segment.children[PRIMARY_OUTLET], false)];\n      }\n\n      return [`${k}:${serializeSegment(v, false)}`];\n    });\n\n    // use no parenthesis if the only child is a primary outlet route\n    if (Object.keys(segment.children).length === 1 && segment.children[PRIMARY_OUTLET] != null) {\n      return `${serializePaths(segment)}/${children[0]}`;\n    }\n\n    return `${serializePaths(segment)}/(${children.join('//')})`;\n  }\n}\n\n/**\n * Encodes a URI string with the default encoding. This function will only ever be called from\n * `encodeUriQuery` or `encodeUriSegment` as it's the base set of encodings to be used. We need\n * a custom encoding because encodeURIComponent is too aggressive and encodes stuff that doesn't\n * have to be encoded per https://url.spec.whatwg.org.\n */\nfunction encodeUriString(s: string): string {\n  return encodeURIComponent(s)\n      .replace(/%40/g, '@')\n      .replace(/%3A/gi, ':')\n      .replace(/%24/g, '$')\n      .replace(/%2C/gi, ',');\n}\n\n/**\n * This function should be used to encode both keys and values in a query string key/value. In\n * the following URL, you need to call encodeUriQuery on \"k\" and \"v\":\n *\n * http://www.site.org/html;mk=mv?k=v#f\n */\nexport function encodeUriQuery(s: string): string {\n  return encodeUriString(s).replace(/%3B/gi, ';');\n}\n\n/**\n * This function should be used to encode a URL fragment. In the following URL, you need to call\n * encodeUriFragment on \"f\":\n *\n * http://www.site.org/html;mk=mv?k=v#f\n */\nexport function encodeUriFragment(s: string): string {\n  return encodeURI(s);\n}\n\n/**\n * This function should be run on any URI segment as well as the key and value in a key/value\n * pair for matrix params. In the following URL, you need to call encodeUriSegment on \"html\",\n * \"mk\", and \"mv\":\n *\n * http://www.site.org/html;mk=mv?k=v#f\n */\nexport function encodeUriSegment(s: string): string {\n  return encodeUriString(s).replace(/\\(/g, '%28').replace(/\\)/g, '%29').replace(/%26/gi, '&');\n}\n\nexport function decode(s: string): string {\n  return decodeURIComponent(s);\n}\n\n// Query keys/values should have the \"+\" replaced first, as \"+\" in a query string is \" \".\n// decodeURIComponent function will not decode \"+\" as a space.\nexport function decodeQuery(s: string): string {\n  return decode(s.replace(/\\+/g, '%20'));\n}\n\nexport function serializePath(path: UrlSegment): string {\n  return `${encodeUriSegment(path.path)}${serializeMatrixParams(path.parameters)}`;\n}\n\nfunction serializeMatrixParams(params: {[key: string]: string}): string {\n  return Object.keys(params)\n      .map(key => `;${encodeUriSegment(key)}=${encodeUriSegment(params[key])}`)\n      .join('');\n}\n\nfunction serializeQueryParams(params: {[key: string]: any}): string {\n  const strParams: string[] =\n      Object.keys(params)\n          .map((name) => {\n            const value = params[name];\n            return Array.isArray(value) ?\n                value.map(v => `${encodeUriQuery(name)}=${encodeUriQuery(v)}`).join('&') :\n                `${encodeUriQuery(name)}=${encodeUriQuery(value)}`;\n          })\n          .filter(s => !!s);\n\n  return strParams.length ? `?${strParams.join('&')}` : '';\n}\n\nconst SEGMENT_RE = /^[^\\/()?;=#]+/;\nfunction matchSegments(str: string): string {\n  const match = str.match(SEGMENT_RE);\n  return match ? match[0] : '';\n}\n\nconst QUERY_PARAM_RE = /^[^=?&#]+/;\n// Return the name of the query param at the start of the string or an empty string\nfunction matchQueryParams(str: string): string {\n  const match = str.match(QUERY_PARAM_RE);\n  return match ? match[0] : '';\n}\n\nconst QUERY_PARAM_VALUE_RE = /^[^?&#]+/;\n// Return the value of the query param at the start of the string or an empty string\nfunction matchUrlQueryParamValue(str: string): string {\n  const match = str.match(QUERY_PARAM_VALUE_RE);\n  return match ? match[0] : '';\n}\n\nclass UrlParser {\n  private remaining: string;\n\n  constructor(private url: string) {\n    this.remaining = url;\n  }\n\n  parseRootSegment(): UrlSegmentGroup {\n    this.consumeOptional('/');\n\n    if (this.remaining === '' || this.peekStartsWith('?') || this.peekStartsWith('#')) {\n      return new UrlSegmentGroup([], {});\n    }\n\n    // The root segment group never has segments\n    return new UrlSegmentGroup([], this.parseChildren());\n  }\n\n  parseQueryParams(): Params {\n    const params: Params = {};\n    if (this.consumeOptional('?')) {\n      do {\n        this.parseQueryParam(params);\n      } while (this.consumeOptional('&'));\n    }\n    return params;\n  }\n\n  parseFragment(): string|null {\n    return this.consumeOptional('#') ? decodeURIComponent(this.remaining) : null;\n  }\n\n  private parseChildren(): {[outlet: string]: UrlSegmentGroup} {\n    if (this.remaining === '') {\n      return {};\n    }\n\n    this.consumeOptional('/');\n\n    const segments: UrlSegment[] = [];\n    if (!this.peekStartsWith('(')) {\n      segments.push(this.parseSegment());\n    }\n\n    while (this.peekStartsWith('/') && !this.peekStartsWith('//') && !this.peekStartsWith('/(')) {\n      this.capture('/');\n      segments.push(this.parseSegment());\n    }\n\n    let children: {[outlet: string]: UrlSegmentGroup} = {};\n    if (this.peekStartsWith('/(')) {\n      this.capture('/');\n      children = this.parseParens(true);\n    }\n\n    let res: {[outlet: string]: UrlSegmentGroup} = {};\n    if (this.peekStartsWith('(')) {\n      res = this.parseParens(false);\n    }\n\n    if (segments.length > 0 || Object.keys(children).length > 0) {\n      res[PRIMARY_OUTLET] = new UrlSegmentGroup(segments, children);\n    }\n\n    return res;\n  }\n\n  // parse a segment with its matrix parameters\n  // ie `name;k1=v1;k2`\n  private parseSegment(): UrlSegment {\n    const path = matchSegments(this.remaining);\n    if (path === '' && this.peekStartsWith(';')) {\n      throw new Error(`Empty path url segment cannot have parameters: '${this.remaining}'.`);\n    }\n\n    this.capture(path);\n    return new UrlSegment(decode(path), this.parseMatrixParams());\n  }\n\n  private parseMatrixParams(): {[key: string]: string} {\n    const params: {[key: string]: string} = {};\n    while (this.consumeOptional(';')) {\n      this.parseParam(params);\n    }\n    return params;\n  }\n\n  private parseParam(params: {[key: string]: string}): void {\n    const key = matchSegments(this.remaining);\n    if (!key) {\n      return;\n    }\n    this.capture(key);\n    let value: any = '';\n    if (this.consumeOptional('=')) {\n      const valueMatch = matchSegments(this.remaining);\n      if (valueMatch) {\n        value = valueMatch;\n        this.capture(value);\n      }\n    }\n\n    params[decode(key)] = decode(value);\n  }\n\n  // Parse a single query parameter `name[=value]`\n  private parseQueryParam(params: Params): void {\n    const key = matchQueryParams(this.remaining);\n    if (!key) {\n      return;\n    }\n    this.capture(key);\n    let value: any = '';\n    if (this.consumeOptional('=')) {\n      const valueMatch = matchUrlQueryParamValue(this.remaining);\n      if (valueMatch) {\n        value = valueMatch;\n        this.capture(value);\n      }\n    }\n\n    const decodedKey = decodeQuery(key);\n    const decodedVal = decodeQuery(value);\n\n    if (params.hasOwnProperty(decodedKey)) {\n      // Append to existing values\n      let currentVal = params[decodedKey];\n      if (!Array.isArray(currentVal)) {\n        currentVal = [currentVal];\n        params[decodedKey] = currentVal;\n      }\n      currentVal.push(decodedVal);\n    } else {\n      // Create a new value\n      params[decodedKey] = decodedVal;\n    }\n  }\n\n  // parse `(a/b//outlet_name:c/d)`\n  private parseParens(allowPrimary: boolean): {[outlet: string]: UrlSegmentGroup} {\n    const segments: {[key: string]: UrlSegmentGroup} = {};\n    this.capture('(');\n\n    while (!this.consumeOptional(')') && this.remaining.length > 0) {\n      const path = matchSegments(this.remaining);\n\n      const next = this.remaining[path.length];\n\n      // if is is not one of these characters, then the segment was unescaped\n      // or the group was not closed\n      if (next !== '/' && next !== ')' && next !== ';') {\n        throw new Error(`Cannot parse url '${this.url}'`);\n      }\n\n      let outletName: string = undefined!;\n      if (path.indexOf(':') > -1) {\n        outletName = path.substr(0, path.indexOf(':'));\n        this.capture(outletName);\n        this.capture(':');\n      } else if (allowPrimary) {\n        outletName = PRIMARY_OUTLET;\n      }\n\n      const children = this.parseChildren();\n      segments[outletName] = Object.keys(children).length === 1 ? children[PRIMARY_OUTLET] :\n                                                                  new UrlSegmentGroup([], children);\n      this.consumeOptional('//');\n    }\n\n    return segments;\n  }\n\n  private peekStartsWith(str: string): boolean {\n    return this.remaining.startsWith(str);\n  }\n\n  // Consumes the prefix when it is present and returns whether it has been consumed\n  private consumeOptional(str: string): boolean {\n    if (this.peekStartsWith(str)) {\n      this.remaining = this.remaining.substring(str.length);\n      return true;\n    }\n    return false;\n  }\n\n  private capture(str: string): void {\n    if (!this.consumeOptional(str)) {\n      throw new Error(`Expected \"${str}\".`);\n    }\n  }\n}\n"]} |
---|