@angular/router 16.0.0-next.4 → 16.0.0-next.6
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.
- package/esm2022/src/apply_redirects.mjs +112 -0
- package/{esm2020 → esm2022}/src/components/empty_outlet.mjs +4 -4
- package/esm2022/src/create_url_tree.mjs +418 -0
- package/{esm2020 → esm2022}/src/directives/router_link.mjs +4 -4
- package/{esm2020 → esm2022}/src/directives/router_link_active.mjs +4 -4
- package/esm2022/src/directives/router_outlet.mjs +336 -0
- package/esm2022/src/index.mjs +30 -0
- package/esm2022/src/navigation_transition.mjs +397 -0
- package/esm2022/src/operators/activate_routes.mjs +192 -0
- package/{esm2020 → esm2022}/src/operators/check_guards.mjs +1 -1
- package/esm2022/src/operators/recognize.mjs +16 -0
- package/{esm2020 → esm2022}/src/page_title_strategy.mjs +7 -7
- package/esm2022/src/provide_router.mjs +507 -0
- package/esm2022/src/recognize.mjs +360 -0
- package/{esm2020 → esm2022}/src/route_reuse_strategy.mjs +7 -7
- package/esm2022/src/router.mjs +724 -0
- package/esm2022/src/router_config.mjs +18 -0
- package/esm2022/src/router_config_loader.mjs +132 -0
- package/esm2022/src/router_module.mjs +220 -0
- package/{esm2020 → esm2022}/src/router_outlet_context.mjs +4 -4
- package/esm2022/src/router_preloader.mjs +167 -0
- package/{esm2020 → esm2022}/src/router_scroller.mjs +4 -4
- package/esm2022/src/router_state.mjs +408 -0
- package/{esm2020 → esm2022}/src/url_handling_strategy.mjs +7 -7
- package/esm2022/src/url_tree.mjs +642 -0
- package/esm2022/src/utils/navigations.mjs +42 -0
- package/esm2022/src/utils/type_guards.mjs +54 -0
- package/{esm2020 → esm2022}/src/version.mjs +1 -1
- package/{esm2020 → esm2022}/testing/src/router_testing_harness.mjs +7 -7
- package/{esm2020 → esm2022}/testing/src/router_testing_module.mjs +10 -10
- package/{fesm2020 → fesm2022}/router.mjs +799 -787
- package/fesm2022/router.mjs.map +1 -0
- package/{fesm2020 → fesm2022}/testing.mjs +16 -16
- package/{fesm2020 → fesm2022}/testing.mjs.map +1 -1
- package/{fesm2020 → fesm2022}/upgrade.mjs +1 -1
- package/index.d.ts +92 -5
- package/package.json +14 -24
- package/testing/index.d.ts +1 -1
- package/upgrade/index.d.ts +1 -1
- package/esm2020/src/apply_redirects.mjs +0 -331
- package/esm2020/src/create_url_tree.mjs +0 -417
- package/esm2020/src/directives/router_outlet.mjs +0 -259
- package/esm2020/src/index.mjs +0 -30
- package/esm2020/src/navigation_transition.mjs +0 -399
- package/esm2020/src/operators/activate_routes.mjs +0 -177
- package/esm2020/src/operators/apply_redirects.mjs +0 -14
- package/esm2020/src/operators/recognize.mjs +0 -14
- package/esm2020/src/provide_router.mjs +0 -477
- package/esm2020/src/recognize.mjs +0 -270
- package/esm2020/src/router.mjs +0 -710
- package/esm2020/src/router_config.mjs +0 -19
- package/esm2020/src/router_config_loader.mjs +0 -131
- package/esm2020/src/router_module.mjs +0 -216
- package/esm2020/src/router_preloader.mjs +0 -167
- package/esm2020/src/router_state.mjs +0 -409
- package/esm2020/src/url_tree.mjs +0 -631
- package/esm2020/src/utils/navigations.mjs +0 -42
- package/esm2020/src/utils/type_guards.mjs +0 -54
- package/fesm2015/router.mjs +0 -7001
- package/fesm2015/router.mjs.map +0 -1
- package/fesm2015/testing.mjs +0 -270
- package/fesm2015/testing.mjs.map +0 -1
- package/fesm2015/upgrade.mjs +0 -146
- package/fesm2015/upgrade.mjs.map +0 -1
- package/fesm2020/router.mjs.map +0 -1
- /package/{esm2020 → esm2022}/index.mjs +0 -0
- /package/{esm2020 → esm2022}/public_api.mjs +0 -0
- /package/{esm2020 → esm2022}/router.mjs +0 -0
- /package/{esm2020 → esm2022}/src/create_router_state.mjs +0 -0
- /package/{esm2020 → esm2022}/src/errors.mjs +0 -0
- /package/{esm2020 → esm2022}/src/events.mjs +0 -0
- /package/{esm2020 → esm2022}/src/models.mjs +0 -0
- /package/{esm2020 → esm2022}/src/models_deprecated.mjs +0 -0
- /package/{esm2020 → esm2022}/src/navigation_canceling_error.mjs +0 -0
- /package/{esm2020 → esm2022}/src/operators/prioritized_guard_value.mjs +0 -0
- /package/{esm2020 → esm2022}/src/operators/resolve_data.mjs +0 -0
- /package/{esm2020 → esm2022}/src/operators/switch_tap.mjs +0 -0
- /package/{esm2020 → esm2022}/src/private_export.mjs +0 -0
- /package/{esm2020 → esm2022}/src/shared.mjs +0 -0
- /package/{esm2020 → esm2022}/src/utils/collection.mjs +0 -0
- /package/{esm2020 → esm2022}/src/utils/config.mjs +0 -0
- /package/{esm2020 → esm2022}/src/utils/config_matching.mjs +0 -0
- /package/{esm2020 → esm2022}/src/utils/functional_guards.mjs +0 -0
- /package/{esm2020 → esm2022}/src/utils/preactivation.mjs +0 -0
- /package/{esm2020 → esm2022}/src/utils/tree.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/index.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/public_api.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/src/testing.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/testing.mjs +0 -0
- /package/{esm2020 → esm2022}/upgrade/index.mjs +0 -0
- /package/{esm2020 → esm2022}/upgrade/public_api.mjs +0 -0
- /package/{esm2020 → esm2022}/upgrade/src/upgrade.mjs +0 -0
- /package/{esm2020 → esm2022}/upgrade/upgrade.mjs +0 -0
- /package/{fesm2020 → fesm2022}/upgrade.mjs.map +0 -0
|
@@ -0,0 +1,360 @@
|
|
|
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 { ɵRuntimeError as RuntimeError } from '@angular/core';
|
|
9
|
+
import { from, of } from 'rxjs';
|
|
10
|
+
import { catchError, concatMap, defaultIfEmpty, first, last as rxjsLast, map, mergeMap, scan, switchMap, tap } from 'rxjs/operators';
|
|
11
|
+
import { absoluteRedirect, AbsoluteRedirect, ApplyRedirects, canLoadFails, noMatch, NoMatch } from './apply_redirects';
|
|
12
|
+
import { createUrlTreeFromSnapshot } from './create_url_tree';
|
|
13
|
+
import { runCanLoadGuards } from './operators/check_guards';
|
|
14
|
+
import { ActivatedRouteSnapshot, inheritedParamsDataResolve, RouterStateSnapshot } from './router_state';
|
|
15
|
+
import { PRIMARY_OUTLET } from './shared';
|
|
16
|
+
import { UrlSegmentGroup } from './url_tree';
|
|
17
|
+
import { last } from './utils/collection';
|
|
18
|
+
import { getOutlet, sortByMatchingOutlets } from './utils/config';
|
|
19
|
+
import { isImmediateMatch, match, matchWithChecks, noLeftoversInUrl, split } from './utils/config_matching';
|
|
20
|
+
import { TreeNode } from './utils/tree';
|
|
21
|
+
import { isEmptyError } from './utils/type_guards';
|
|
22
|
+
export function recognize(injector, configLoader, rootComponentType, config, urlTree, urlSerializer, paramsInheritanceStrategy = 'emptyOnly') {
|
|
23
|
+
return new Recognizer(injector, configLoader, rootComponentType, config, urlTree, paramsInheritanceStrategy, urlSerializer)
|
|
24
|
+
.recognize();
|
|
25
|
+
}
|
|
26
|
+
export class Recognizer {
|
|
27
|
+
constructor(injector, configLoader, rootComponentType, config, urlTree, paramsInheritanceStrategy, urlSerializer) {
|
|
28
|
+
this.injector = injector;
|
|
29
|
+
this.configLoader = configLoader;
|
|
30
|
+
this.rootComponentType = rootComponentType;
|
|
31
|
+
this.config = config;
|
|
32
|
+
this.urlTree = urlTree;
|
|
33
|
+
this.paramsInheritanceStrategy = paramsInheritanceStrategy;
|
|
34
|
+
this.urlSerializer = urlSerializer;
|
|
35
|
+
this.allowRedirects = true;
|
|
36
|
+
this.applyRedirects = new ApplyRedirects(this.urlSerializer, this.urlTree);
|
|
37
|
+
}
|
|
38
|
+
noMatchError(e) {
|
|
39
|
+
return new RuntimeError(4002 /* RuntimeErrorCode.NO_MATCH */, (typeof ngDevMode === 'undefined' || ngDevMode) &&
|
|
40
|
+
`Cannot match any routes. URL Segment: '${e.segmentGroup}'`);
|
|
41
|
+
}
|
|
42
|
+
recognize() {
|
|
43
|
+
const rootSegmentGroup = split(this.urlTree.root, [], [], this.config).segmentGroup;
|
|
44
|
+
return this.processSegmentGroup(this.injector, this.config, rootSegmentGroup, PRIMARY_OUTLET)
|
|
45
|
+
.pipe(catchError((e) => {
|
|
46
|
+
if (e instanceof AbsoluteRedirect) {
|
|
47
|
+
// After an absolute redirect we do not apply any more redirects!
|
|
48
|
+
// If this implementation changes, update the documentation note in `redirectTo`.
|
|
49
|
+
this.allowRedirects = false;
|
|
50
|
+
this.urlTree = e.urlTree;
|
|
51
|
+
return this.match(e.urlTree);
|
|
52
|
+
}
|
|
53
|
+
if (e instanceof NoMatch) {
|
|
54
|
+
throw this.noMatchError(e);
|
|
55
|
+
}
|
|
56
|
+
throw e;
|
|
57
|
+
}), map(children => {
|
|
58
|
+
// Use Object.freeze to prevent readers of the Router state from modifying it outside
|
|
59
|
+
// of a navigation, resulting in the router being out of sync with the browser.
|
|
60
|
+
const root = new ActivatedRouteSnapshot([], Object.freeze({}), Object.freeze({ ...this.urlTree.queryParams }), this.urlTree.fragment, {}, PRIMARY_OUTLET, this.rootComponentType, null, {});
|
|
61
|
+
const rootNode = new TreeNode(root, children);
|
|
62
|
+
const routeState = new RouterStateSnapshot('', rootNode);
|
|
63
|
+
const tree = createUrlTreeFromSnapshot(root, [], this.urlTree.queryParams, this.urlTree.fragment);
|
|
64
|
+
// https://github.com/angular/angular/issues/47307
|
|
65
|
+
// Creating the tree stringifies the query params
|
|
66
|
+
// We don't want to do this here so reassign them to the original.
|
|
67
|
+
tree.queryParams = this.urlTree.queryParams;
|
|
68
|
+
routeState.url = this.urlSerializer.serialize(tree);
|
|
69
|
+
this.inheritParamsAndData(routeState._root);
|
|
70
|
+
return { state: routeState, tree };
|
|
71
|
+
}));
|
|
72
|
+
}
|
|
73
|
+
match(tree) {
|
|
74
|
+
const expanded$ = this.processSegmentGroup(this.injector, this.config, tree.root, PRIMARY_OUTLET);
|
|
75
|
+
return expanded$.pipe(catchError((e) => {
|
|
76
|
+
if (e instanceof NoMatch) {
|
|
77
|
+
throw this.noMatchError(e);
|
|
78
|
+
}
|
|
79
|
+
throw e;
|
|
80
|
+
}));
|
|
81
|
+
}
|
|
82
|
+
inheritParamsAndData(routeNode) {
|
|
83
|
+
const route = routeNode.value;
|
|
84
|
+
const i = inheritedParamsDataResolve(route, this.paramsInheritanceStrategy);
|
|
85
|
+
route.params = Object.freeze(i.params);
|
|
86
|
+
route.data = Object.freeze(i.data);
|
|
87
|
+
routeNode.children.forEach(n => this.inheritParamsAndData(n));
|
|
88
|
+
}
|
|
89
|
+
processSegmentGroup(injector, config, segmentGroup, outlet) {
|
|
90
|
+
if (segmentGroup.segments.length === 0 && segmentGroup.hasChildren()) {
|
|
91
|
+
return this.processChildren(injector, config, segmentGroup);
|
|
92
|
+
}
|
|
93
|
+
return this.processSegment(injector, config, segmentGroup, segmentGroup.segments, outlet, true);
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Matches every child outlet in the `segmentGroup` to a `Route` in the config. Returns `null` if
|
|
97
|
+
* we cannot find a match for _any_ of the children.
|
|
98
|
+
*
|
|
99
|
+
* @param config - The `Routes` to match against
|
|
100
|
+
* @param segmentGroup - The `UrlSegmentGroup` whose children need to be matched against the
|
|
101
|
+
* config.
|
|
102
|
+
*/
|
|
103
|
+
processChildren(injector, config, segmentGroup) {
|
|
104
|
+
// Expand outlets one at a time, starting with the primary outlet. We need to do it this way
|
|
105
|
+
// because an absolute redirect from the primary outlet takes precedence.
|
|
106
|
+
const childOutlets = [];
|
|
107
|
+
for (const child of Object.keys(segmentGroup.children)) {
|
|
108
|
+
if (child === 'primary') {
|
|
109
|
+
childOutlets.unshift(child);
|
|
110
|
+
}
|
|
111
|
+
else {
|
|
112
|
+
childOutlets.push(child);
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
return from(childOutlets)
|
|
116
|
+
.pipe(concatMap(childOutlet => {
|
|
117
|
+
const child = segmentGroup.children[childOutlet];
|
|
118
|
+
// Sort the config so that routes with outlets that match the one being activated
|
|
119
|
+
// appear first, followed by routes for other outlets, which might match if they have
|
|
120
|
+
// an empty path.
|
|
121
|
+
const sortedConfig = sortByMatchingOutlets(config, childOutlet);
|
|
122
|
+
return this.processSegmentGroup(injector, sortedConfig, child, childOutlet);
|
|
123
|
+
}), scan((children, outletChildren) => {
|
|
124
|
+
children.push(...outletChildren);
|
|
125
|
+
return children;
|
|
126
|
+
}), defaultIfEmpty(null), rxjsLast(), mergeMap(children => {
|
|
127
|
+
if (children === null)
|
|
128
|
+
return noMatch(segmentGroup);
|
|
129
|
+
// Because we may have matched two outlets to the same empty path segment, we can have
|
|
130
|
+
// multiple activated results for the same outlet. We should merge the children of
|
|
131
|
+
// these results so the final return value is only one `TreeNode` per outlet.
|
|
132
|
+
const mergedChildren = mergeEmptyPathMatches(children);
|
|
133
|
+
if (typeof ngDevMode === 'undefined' || ngDevMode) {
|
|
134
|
+
// This should really never happen - we are only taking the first match for each
|
|
135
|
+
// outlet and merge the empty path matches.
|
|
136
|
+
checkOutletNameUniqueness(mergedChildren);
|
|
137
|
+
}
|
|
138
|
+
sortActivatedRouteSnapshots(mergedChildren);
|
|
139
|
+
return of(mergedChildren);
|
|
140
|
+
}));
|
|
141
|
+
}
|
|
142
|
+
processSegment(injector, routes, segmentGroup, segments, outlet, allowRedirects) {
|
|
143
|
+
return from(routes).pipe(concatMap(r => {
|
|
144
|
+
return this
|
|
145
|
+
.processSegmentAgainstRoute(r._injector ?? injector, routes, r, segmentGroup, segments, outlet, allowRedirects)
|
|
146
|
+
.pipe(catchError((e) => {
|
|
147
|
+
if (e instanceof NoMatch) {
|
|
148
|
+
return of(null);
|
|
149
|
+
}
|
|
150
|
+
throw e;
|
|
151
|
+
}));
|
|
152
|
+
}), first((x) => !!x), catchError(e => {
|
|
153
|
+
if (isEmptyError(e)) {
|
|
154
|
+
if (noLeftoversInUrl(segmentGroup, segments, outlet)) {
|
|
155
|
+
return of([]);
|
|
156
|
+
}
|
|
157
|
+
return noMatch(segmentGroup);
|
|
158
|
+
}
|
|
159
|
+
throw e;
|
|
160
|
+
}));
|
|
161
|
+
}
|
|
162
|
+
processSegmentAgainstRoute(injector, routes, route, rawSegment, segments, outlet, allowRedirects) {
|
|
163
|
+
if (!isImmediateMatch(route, rawSegment, segments, outlet))
|
|
164
|
+
return noMatch(rawSegment);
|
|
165
|
+
if (route.redirectTo === undefined) {
|
|
166
|
+
return this.matchSegmentAgainstRoute(injector, rawSegment, route, segments, outlet, allowRedirects);
|
|
167
|
+
}
|
|
168
|
+
if (allowRedirects && this.allowRedirects) {
|
|
169
|
+
return this.expandSegmentAgainstRouteUsingRedirect(injector, rawSegment, routes, route, segments, outlet);
|
|
170
|
+
}
|
|
171
|
+
return noMatch(rawSegment);
|
|
172
|
+
}
|
|
173
|
+
expandSegmentAgainstRouteUsingRedirect(injector, segmentGroup, routes, route, segments, outlet) {
|
|
174
|
+
if (route.path === '**') {
|
|
175
|
+
return this.expandWildCardWithParamsAgainstRouteUsingRedirect(injector, routes, route, outlet);
|
|
176
|
+
}
|
|
177
|
+
return this.expandRegularSegmentAgainstRouteUsingRedirect(injector, segmentGroup, routes, route, segments, outlet);
|
|
178
|
+
}
|
|
179
|
+
expandWildCardWithParamsAgainstRouteUsingRedirect(injector, routes, route, outlet) {
|
|
180
|
+
const newTree = this.applyRedirects.applyRedirectCommands([], route.redirectTo, {});
|
|
181
|
+
if (route.redirectTo.startsWith('/')) {
|
|
182
|
+
return absoluteRedirect(newTree);
|
|
183
|
+
}
|
|
184
|
+
return this.applyRedirects.lineralizeSegments(route, newTree)
|
|
185
|
+
.pipe(mergeMap((newSegments) => {
|
|
186
|
+
const group = new UrlSegmentGroup(newSegments, {});
|
|
187
|
+
return this.processSegment(injector, routes, group, newSegments, outlet, false);
|
|
188
|
+
}));
|
|
189
|
+
}
|
|
190
|
+
expandRegularSegmentAgainstRouteUsingRedirect(injector, segmentGroup, routes, route, segments, outlet) {
|
|
191
|
+
const { matched, consumedSegments, remainingSegments, positionalParamSegments } = match(segmentGroup, route, segments);
|
|
192
|
+
if (!matched)
|
|
193
|
+
return noMatch(segmentGroup);
|
|
194
|
+
const newTree = this.applyRedirects.applyRedirectCommands(consumedSegments, route.redirectTo, positionalParamSegments);
|
|
195
|
+
if (route.redirectTo.startsWith('/')) {
|
|
196
|
+
return absoluteRedirect(newTree);
|
|
197
|
+
}
|
|
198
|
+
return this.applyRedirects.lineralizeSegments(route, newTree)
|
|
199
|
+
.pipe(mergeMap((newSegments) => {
|
|
200
|
+
return this.processSegment(injector, routes, segmentGroup, newSegments.concat(remainingSegments), outlet, false);
|
|
201
|
+
}));
|
|
202
|
+
}
|
|
203
|
+
matchSegmentAgainstRoute(injector, rawSegment, route, segments, outlet, allowRedirects) {
|
|
204
|
+
let matchResult;
|
|
205
|
+
if (route.path === '**') {
|
|
206
|
+
const params = segments.length > 0 ? last(segments).parameters : {};
|
|
207
|
+
const snapshot = new ActivatedRouteSnapshot(segments, params, Object.freeze({ ...this.urlTree.queryParams }), this.urlTree.fragment, getData(route), getOutlet(route), route.component ?? route._loadedComponent ?? null, route, getResolve(route));
|
|
208
|
+
matchResult = of({
|
|
209
|
+
snapshot,
|
|
210
|
+
consumedSegments: [],
|
|
211
|
+
remainingSegments: [],
|
|
212
|
+
});
|
|
213
|
+
// Prior versions of the route matching algorithm would stop matching at the wildcard route.
|
|
214
|
+
// We should investigate a better strategy for any existing children. Otherwise, these
|
|
215
|
+
// child segments are silently dropped from the navigation.
|
|
216
|
+
// https://github.com/angular/angular/issues/40089
|
|
217
|
+
rawSegment.children = {};
|
|
218
|
+
}
|
|
219
|
+
else {
|
|
220
|
+
matchResult =
|
|
221
|
+
matchWithChecks(rawSegment, route, segments, injector, this.urlSerializer)
|
|
222
|
+
.pipe(map(({ matched, consumedSegments, remainingSegments, parameters }) => {
|
|
223
|
+
if (!matched) {
|
|
224
|
+
return null;
|
|
225
|
+
}
|
|
226
|
+
const snapshot = new ActivatedRouteSnapshot(consumedSegments, parameters, Object.freeze({ ...this.urlTree.queryParams }), this.urlTree.fragment, getData(route), getOutlet(route), route.component ?? route._loadedComponent ?? null, route, getResolve(route));
|
|
227
|
+
return { snapshot, consumedSegments, remainingSegments };
|
|
228
|
+
}));
|
|
229
|
+
}
|
|
230
|
+
return matchResult.pipe(switchMap((result) => {
|
|
231
|
+
if (result === null) {
|
|
232
|
+
return noMatch(rawSegment);
|
|
233
|
+
}
|
|
234
|
+
// If the route has an injector created from providers, we should start using that.
|
|
235
|
+
injector = route._injector ?? injector;
|
|
236
|
+
return this.getChildConfig(injector, route, segments)
|
|
237
|
+
.pipe(switchMap(({ routes: childConfig }) => {
|
|
238
|
+
const childInjector = route._loadedInjector ?? injector;
|
|
239
|
+
const { snapshot, consumedSegments, remainingSegments } = result;
|
|
240
|
+
const { segmentGroup, slicedSegments } = split(rawSegment, consumedSegments, remainingSegments, childConfig);
|
|
241
|
+
if (slicedSegments.length === 0 && segmentGroup.hasChildren()) {
|
|
242
|
+
return this.processChildren(childInjector, childConfig, segmentGroup)
|
|
243
|
+
.pipe(map(children => {
|
|
244
|
+
if (children === null) {
|
|
245
|
+
return null;
|
|
246
|
+
}
|
|
247
|
+
return [new TreeNode(snapshot, children)];
|
|
248
|
+
}));
|
|
249
|
+
}
|
|
250
|
+
if (childConfig.length === 0 && slicedSegments.length === 0) {
|
|
251
|
+
return of([new TreeNode(snapshot, [])]);
|
|
252
|
+
}
|
|
253
|
+
const matchedOnOutlet = getOutlet(route) === outlet;
|
|
254
|
+
// If we matched a config due to empty path match on a different outlet, we need to
|
|
255
|
+
// continue passing the current outlet for the segment rather than switch to PRIMARY.
|
|
256
|
+
// Note that we switch to primary when we have a match because outlet configs look like
|
|
257
|
+
// this: {path: 'a', outlet: 'a', children: [
|
|
258
|
+
// {path: 'b', component: B},
|
|
259
|
+
// {path: 'c', component: C},
|
|
260
|
+
// ]}
|
|
261
|
+
// Notice that the children of the named outlet are configured with the primary outlet
|
|
262
|
+
return this
|
|
263
|
+
.processSegment(childInjector, childConfig, segmentGroup, slicedSegments, matchedOnOutlet ? PRIMARY_OUTLET : outlet, true)
|
|
264
|
+
.pipe(map(children => {
|
|
265
|
+
return [new TreeNode(snapshot, children)];
|
|
266
|
+
}));
|
|
267
|
+
}));
|
|
268
|
+
}));
|
|
269
|
+
}
|
|
270
|
+
getChildConfig(injector, route, segments) {
|
|
271
|
+
if (route.children) {
|
|
272
|
+
// The children belong to the same module
|
|
273
|
+
return of({ routes: route.children, injector });
|
|
274
|
+
}
|
|
275
|
+
if (route.loadChildren) {
|
|
276
|
+
// lazy children belong to the loaded module
|
|
277
|
+
if (route._loadedRoutes !== undefined) {
|
|
278
|
+
return of({ routes: route._loadedRoutes, injector: route._loadedInjector });
|
|
279
|
+
}
|
|
280
|
+
return runCanLoadGuards(injector, route, segments, this.urlSerializer)
|
|
281
|
+
.pipe(mergeMap((shouldLoadResult) => {
|
|
282
|
+
if (shouldLoadResult) {
|
|
283
|
+
return this.configLoader.loadChildren(injector, route)
|
|
284
|
+
.pipe(tap((cfg) => {
|
|
285
|
+
route._loadedRoutes = cfg.routes;
|
|
286
|
+
route._loadedInjector = cfg.injector;
|
|
287
|
+
}));
|
|
288
|
+
}
|
|
289
|
+
return canLoadFails(route);
|
|
290
|
+
}));
|
|
291
|
+
}
|
|
292
|
+
return of({ routes: [], injector });
|
|
293
|
+
}
|
|
294
|
+
}
|
|
295
|
+
function sortActivatedRouteSnapshots(nodes) {
|
|
296
|
+
nodes.sort((a, b) => {
|
|
297
|
+
if (a.value.outlet === PRIMARY_OUTLET)
|
|
298
|
+
return -1;
|
|
299
|
+
if (b.value.outlet === PRIMARY_OUTLET)
|
|
300
|
+
return 1;
|
|
301
|
+
return a.value.outlet.localeCompare(b.value.outlet);
|
|
302
|
+
});
|
|
303
|
+
}
|
|
304
|
+
function hasEmptyPathConfig(node) {
|
|
305
|
+
const config = node.value.routeConfig;
|
|
306
|
+
return config && config.path === '';
|
|
307
|
+
}
|
|
308
|
+
/**
|
|
309
|
+
* Finds `TreeNode`s with matching empty path route configs and merges them into `TreeNode` with
|
|
310
|
+
* the children from each duplicate. This is necessary because different outlets can match a
|
|
311
|
+
* single empty path route config and the results need to then be merged.
|
|
312
|
+
*/
|
|
313
|
+
function mergeEmptyPathMatches(nodes) {
|
|
314
|
+
const result = [];
|
|
315
|
+
// The set of nodes which contain children that were merged from two duplicate empty path nodes.
|
|
316
|
+
const mergedNodes = new Set();
|
|
317
|
+
for (const node of nodes) {
|
|
318
|
+
if (!hasEmptyPathConfig(node)) {
|
|
319
|
+
result.push(node);
|
|
320
|
+
continue;
|
|
321
|
+
}
|
|
322
|
+
const duplicateEmptyPathNode = result.find(resultNode => node.value.routeConfig === resultNode.value.routeConfig);
|
|
323
|
+
if (duplicateEmptyPathNode !== undefined) {
|
|
324
|
+
duplicateEmptyPathNode.children.push(...node.children);
|
|
325
|
+
mergedNodes.add(duplicateEmptyPathNode);
|
|
326
|
+
}
|
|
327
|
+
else {
|
|
328
|
+
result.push(node);
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
// For each node which has children from multiple sources, we need to recompute a new `TreeNode`
|
|
332
|
+
// by also merging those children. This is necessary when there are multiple empty path configs
|
|
333
|
+
// in a row. Put another way: whenever we combine children of two nodes, we need to also check
|
|
334
|
+
// if any of those children can be combined into a single node as well.
|
|
335
|
+
for (const mergedNode of mergedNodes) {
|
|
336
|
+
const mergedChildren = mergeEmptyPathMatches(mergedNode.children);
|
|
337
|
+
result.push(new TreeNode(mergedNode.value, mergedChildren));
|
|
338
|
+
}
|
|
339
|
+
return result.filter(n => !mergedNodes.has(n));
|
|
340
|
+
}
|
|
341
|
+
function checkOutletNameUniqueness(nodes) {
|
|
342
|
+
const names = {};
|
|
343
|
+
nodes.forEach(n => {
|
|
344
|
+
const routeWithSameOutletName = names[n.value.outlet];
|
|
345
|
+
if (routeWithSameOutletName) {
|
|
346
|
+
const p = routeWithSameOutletName.url.map(s => s.toString()).join('/');
|
|
347
|
+
const c = n.value.url.map(s => s.toString()).join('/');
|
|
348
|
+
throw new RuntimeError(4006 /* RuntimeErrorCode.TWO_SEGMENTS_WITH_SAME_OUTLET */, (typeof ngDevMode === 'undefined' || ngDevMode) &&
|
|
349
|
+
`Two segments cannot have the same outlet name: '${p}' and '${c}'.`);
|
|
350
|
+
}
|
|
351
|
+
names[n.value.outlet] = n.value;
|
|
352
|
+
});
|
|
353
|
+
}
|
|
354
|
+
function getData(route) {
|
|
355
|
+
return route.data || {};
|
|
356
|
+
}
|
|
357
|
+
function getResolve(route) {
|
|
358
|
+
return route.resolve || {};
|
|
359
|
+
}
|
|
360
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"recognize.js","sourceRoot":"","sources":["../../../../../../packages/router/src/recognize.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,EAA4B,aAAa,IAAI,YAAY,EAAC,MAAM,eAAe,CAAC;AACvF,OAAO,EAAC,IAAI,EAAc,EAAE,EAAC,MAAM,MAAM,CAAC;AAC1C,OAAO,EAAC,UAAU,EAAE,SAAS,EAAE,cAAc,EAAE,KAAK,EAAE,IAAI,IAAI,QAAQ,EAAE,GAAG,EAAE,QAAQ,EAAE,IAAI,EAAE,SAAS,EAAE,GAAG,EAAC,MAAM,gBAAgB,CAAC;AAEnI,OAAO,EAAC,gBAAgB,EAAE,gBAAgB,EAAE,cAAc,EAAE,YAAY,EAAE,OAAO,EAAE,OAAO,EAAC,MAAM,mBAAmB,CAAC;AACrH,OAAO,EAAC,yBAAyB,EAAC,MAAM,mBAAmB,CAAC;AAG5D,OAAO,EAAC,gBAAgB,EAAC,MAAM,0BAA0B,CAAC;AAE1D,OAAO,EAAC,sBAAsB,EAAE,0BAA0B,EAA6B,mBAAmB,EAAC,MAAM,gBAAgB,CAAC;AAClI,OAAO,EAAC,cAAc,EAAC,MAAM,UAAU,CAAC;AACxC,OAAO,EAAa,eAAe,EAAyB,MAAM,YAAY,CAAC;AAC/E,OAAO,EAAC,IAAI,EAAC,MAAM,oBAAoB,CAAC;AACxC,OAAO,EAAC,SAAS,EAAE,qBAAqB,EAAC,MAAM,gBAAgB,CAAC;AAChE,OAAO,EAAC,gBAAgB,EAAE,KAAK,EAAE,eAAe,EAAE,gBAAgB,EAAE,KAAK,EAAC,MAAM,yBAAyB,CAAC;AAC1G,OAAO,EAAC,QAAQ,EAAC,MAAM,cAAc,CAAC;AACtC,OAAO,EAAC,YAAY,EAAC,MAAM,qBAAqB,CAAC;AAGjD,MAAM,UAAU,SAAS,CACrB,QAA6B,EAAE,YAAgC,EAC/D,iBAAiC,EAAE,MAAc,EAAE,OAAgB,EACnE,aAA4B,EAC5B,4BACI,WAAW;IACjB,OAAO,IAAI,UAAU,CACV,QAAQ,EAAE,YAAY,EAAE,iBAAiB,EAAE,MAAM,EAAE,OAAO,EAAE,yBAAyB,EACrF,aAAa,CAAC;SACpB,SAAS,EAAE,CAAC;AACnB,CAAC;AAED,MAAM,OAAO,UAAU;IAIrB,YACY,QAA6B,EAAU,YAAgC,EACvE,iBAAiC,EAAU,MAAc,EAAU,OAAgB,EACnF,yBAAoD,EAC3C,aAA4B;QAHrC,aAAQ,GAAR,QAAQ,CAAqB;QAAU,iBAAY,GAAZ,YAAY,CAAoB;QACvE,sBAAiB,GAAjB,iBAAiB,CAAgB;QAAU,WAAM,GAAN,MAAM,CAAQ;QAAU,YAAO,GAAP,OAAO,CAAS;QACnF,8BAAyB,GAAzB,yBAAyB,CAA2B;QAC3C,kBAAa,GAAb,aAAa,CAAe;QAPjD,mBAAc,GAAG,IAAI,CAAC;QACd,mBAAc,GAAG,IAAI,cAAc,CAAC,IAAI,CAAC,aAAa,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;IAM1B,CAAC;IAE7C,YAAY,CAAC,CAAU;QAC7B,OAAO,IAAI,YAAY,uCAEnB,CAAC,OAAO,SAAS,KAAK,WAAW,IAAI,SAAS,CAAC;YAC3C,0CAA0C,CAAC,CAAC,YAAY,GAAG,CAAC,CAAC;IACvE,CAAC;IAED,SAAS;QACP,MAAM,gBAAgB,GAAG,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,EAAE,EAAE,EAAE,EAAE,IAAI,CAAC,MAAM,CAAC,CAAC,YAAY,CAAC;QAEpF,OAAO,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,MAAM,EAAE,gBAAgB,EAAE,cAAc,CAAC;aACxF,IAAI,CACD,UAAU,CAAC,CAAC,CAAM,EAAE,EAAE;YACpB,IAAI,CAAC,YAAY,gBAAgB,EAAE;gBACjC,iEAAiE;gBACjE,iFAAiF;gBACjF,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC;gBAC5B,IAAI,CAAC,OAAO,GAAG,CAAC,CAAC,OAAO,CAAC;gBACzB,OAAO,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC;aAC9B;YAED,IAAI,CAAC,YAAY,OAAO,EAAE;gBACxB,MAAM,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;aAC5B;YAED,MAAM,CAAC,CAAC;QACV,CAAC,CAAC,EACF,GAAG,CAAC,QAAQ,CAAC,EAAE;YACb,qFAAqF;YACrF,+EAA+E;YAC/E,MAAM,IAAI,GAAG,IAAI,sBAAsB,CACnC,EAAE,EAAE,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,EAAE,MAAM,CAAC,MAAM,CAAC,EAAC,GAAG,IAAI,CAAC,OAAO,CAAC,WAAW,EAAC,CAAC,EACnE,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,EAAE,EAAE,cAAc,EAAE,IAAI,CAAC,iBAAiB,EAAE,IAAI,EAAE,EAAE,CAAC,CAAC;YAEjF,MAAM,QAAQ,GAAG,IAAI,QAAQ,CAAyB,IAAI,EAAE,QAAQ,CAAC,CAAC;YACtE,MAAM,UAAU,GAAG,IAAI,mBAAmB,CAAC,EAAE,EAAE,QAAQ,CAAC,CAAC;YACzD,MAAM,IAAI,GAAG,yBAAyB,CAClC,IAAI,EAAE,EAAE,EAAE,IAAI,CAAC,OAAO,CAAC,WAAW,EAAE,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;YAC/D,kDAAkD;YAClD,iDAAiD;YACjD,kEAAkE;YAClE,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC;YAC5C,UAAU,CAAC,GAAG,GAAG,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;YACpD,IAAI,CAAC,oBAAoB,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;YAC5C,OAAO,EAAC,KAAK,EAAE,UAAU,EAAE,IAAI,EAAC,CAAC;QACnC,CAAC,CAAC,CAAC,CAAC;IACd,CAAC;IAGO,KAAK,CAAC,IAAa;QACzB,MAAM,SAAS,GACX,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,IAAI,EAAE,cAAc,CAAC,CAAC;QACpF,OAAO,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAM,EAAE,EAAE;YAC1C,IAAI,CAAC,YAAY,OAAO,EAAE;gBACxB,MAAM,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;aAC5B;YAED,MAAM,CAAC,CAAC;QACV,CAAC,CAAC,CAAC,CAAC;IACN,CAAC;IAED,oBAAoB,CAAC,SAA2C;QAC9D,MAAM,KAAK,GAAG,SAAS,CAAC,KAAK,CAAC;QAE9B,MAAM,CAAC,GAAG,0BAA0B,CAAC,KAAK,EAAE,IAAI,CAAC,yBAAyB,CAAC,CAAC;QAC5E,KAAK,CAAC,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC;QACvC,KAAK,CAAC,IAAI,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;QAEnC,SAAS,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC,CAAC,CAAC,CAAC;IAChE,CAAC;IAED,mBAAmB,CACf,QAA6B,EAAE,MAAe,EAAE,YAA6B,EAC7E,MAAc;QAChB,IAAI,YAAY,CAAC,QAAQ,CAAC,MAAM,KAAK,CAAC,IAAI,YAAY,CAAC,WAAW,EAAE,EAAE;YACpE,OAAO,IAAI,CAAC,eAAe,CAAC,QAAQ,EAAE,MAAM,EAAE,YAAY,CAAC,CAAC;SAC7D;QAED,OAAO,IAAI,CAAC,cAAc,CAAC,QAAQ,EAAE,MAAM,EAAE,YAAY,EAAE,YAAY,CAAC,QAAQ,EAAE,MAAM,EAAE,IAAI,CAAC,CAAC;IAClG,CAAC;IAED;;;;;;;OAOG;IACH,eAAe,CAAC,QAA6B,EAAE,MAAe,EAAE,YAA6B;QAE3F,4FAA4F;QAC5F,yEAAyE;QACzE,MAAM,YAAY,GAAa,EAAE,CAAC;QAClC,KAAK,MAAM,KAAK,IAAI,MAAM,CAAC,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,EAAE;YACtD,IAAI,KAAK,KAAK,SAAS,EAAE;gBACvB,YAAY,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;aAC7B;iBAAM;gBACL,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;aAC1B;SACF;QACD,OAAO,IAAI,CAAC,YAAY,CAAC;aACpB,IAAI,CACD,SAAS,CAAC,WAAW,CAAC,EAAE;YACtB,MAAM,KAAK,GAAG,YAAY,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC;YACjD,iFAAiF;YACjF,qFAAqF;YACrF,iBAAiB;YACjB,MAAM,YAAY,GAAG,qBAAqB,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC;YAChE,OAAO,IAAI,CAAC,mBAAmB,CAAC,QAAQ,EAAE,YAAY,EAAE,KAAK,EAAE,WAAW,CAAC,CAAC;QAC9E,CAAC,CAAC,EACF,IAAI,CAAC,CAAC,QAAQ,EAAE,cAAc,EAAE,EAAE;YAChC,QAAQ,CAAC,IAAI,CAAC,GAAG,cAAc,CAAC,CAAC;YACjC,OAAO,QAAQ,CAAC;QAClB,CAAC,CAAC,EACF,cAAc,CAAC,IAAiD,CAAC,EACjE,QAAQ,EAAE,EACV,QAAQ,CAAC,QAAQ,CAAC,EAAE;YAClB,IAAI,QAAQ,KAAK,IAAI;gBAAE,OAAO,OAAO,CAAC,YAAY,CAAC,CAAC;YACpD,sFAAsF;YACtF,kFAAkF;YAClF,6EAA6E;YAC7E,MAAM,cAAc,GAAG,qBAAqB,CAAC,QAAQ,CAAC,CAAC;YACvD,IAAI,OAAO,SAAS,KAAK,WAAW,IAAI,SAAS,EAAE;gBACjD,gFAAgF;gBAChF,2CAA2C;gBAC3C,yBAAyB,CAAC,cAAc,CAAC,CAAC;aAC3C;YACD,2BAA2B,CAAC,cAAc,CAAC,CAAC;YAC5C,OAAO,EAAE,CAAC,cAAc,CAAC,CAAC;QAC5B,CAAC,CAAC,CACL,CAAC;IACR,CAAC;IAED,cAAc,CACV,QAA6B,EAAE,MAAe,EAAE,YAA6B,EAC7E,QAAsB,EAAE,MAAc,EACtC,cAAuB;QACzB,OAAO,IAAI,CAAC,MAAM,CAAC,CAAC,IAAI,CACpB,SAAS,CAAC,CAAC,CAAC,EAAE;YACZ,OAAO,IAAI;iBACN,0BAA0B,CACvB,CAAC,CAAC,SAAS,IAAI,QAAQ,EAAE,MAAM,EAAE,CAAC,EAAE,YAAY,EAAE,QAAQ,EAAE,MAAM,EAClE,cAAc,CAAC;iBAClB,IAAI,CAAC,UAAU,CAAC,CAAC,CAAM,EAAE,EAAE;gBAC1B,IAAI,CAAC,YAAY,OAAO,EAAE;oBACxB,OAAO,EAAE,CAAC,IAAI,CAAC,CAAC;iBACjB;gBACD,MAAM,CAAC,CAAC;YACV,CAAC,CAAC,CAAC,CAAC;QACV,CAAC,CAAC,EACF,KAAK,CAAC,CAAC,CAAC,EAA2C,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,UAAU,CAAC,CAAC,CAAC,EAAE;YACzE,IAAI,YAAY,CAAC,CAAC,CAAC,EAAE;gBACnB,IAAI,gBAAgB,CAAC,YAAY,EAAE,QAAQ,EAAE,MAAM,CAAC,EAAE;oBACpD,OAAO,EAAE,CAAC,EAAE,CAAC,CAAC;iBACf;gBACD,OAAO,OAAO,CAAC,YAAY,CAAC,CAAC;aAC9B;YACD,MAAM,CAAC,CAAC;QACV,CAAC,CAAC,CAAC,CAAC;IACV,CAAC;IAED,0BAA0B,CACtB,QAA6B,EAAE,MAAe,EAAE,KAAY,EAAE,UAA2B,EACzF,QAAsB,EAAE,MAAc,EACtC,cAAuB;QACzB,IAAI,CAAC,gBAAgB,CAAC,KAAK,EAAE,UAAU,EAAE,QAAQ,EAAE,MAAM,CAAC;YAAE,OAAO,OAAO,CAAC,UAAU,CAAC,CAAC;QAEvF,IAAI,KAAK,CAAC,UAAU,KAAK,SAAS,EAAE;YAClC,OAAO,IAAI,CAAC,wBAAwB,CAChC,QAAQ,EAAE,UAAU,EAAE,KAAK,EAAE,QAAQ,EAAE,MAAM,EAAE,cAAc,CAAC,CAAC;SACpE;QAED,IAAI,cAAc,IAAI,IAAI,CAAC,cAAc,EAAE;YACzC,OAAO,IAAI,CAAC,sCAAsC,CAC9C,QAAQ,EAAE,UAAU,EAAE,MAAM,EAAE,KAAK,EAAE,QAAQ,EAAE,MAAM,CAAC,CAAC;SAC5D;QAED,OAAO,OAAO,CAAC,UAAU,CAAC,CAAC;IAC7B,CAAC;IAEO,sCAAsC,CAC1C,QAA6B,EAAE,YAA6B,EAAE,MAAe,EAAE,KAAY,EAC3F,QAAsB,EAAE,MAAc;QACxC,IAAI,KAAK,CAAC,IAAI,KAAK,IAAI,EAAE;YACvB,OAAO,IAAI,CAAC,iDAAiD,CACzD,QAAQ,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,CAAC,CAAC;SACtC;QAED,OAAO,IAAI,CAAC,6CAA6C,CACrD,QAAQ,EAAE,YAAY,EAAE,MAAM,EAAE,KAAK,EAAE,QAAQ,EAAE,MAAM,CAAC,CAAC;IAC/D,CAAC;IAEO,iDAAiD,CACrD,QAA6B,EAAE,MAAe,EAAE,KAAY,EAC5D,MAAc;QAChB,MAAM,OAAO,GAAG,IAAI,CAAC,cAAc,CAAC,qBAAqB,CAAC,EAAE,EAAE,KAAK,CAAC,UAAW,EAAE,EAAE,CAAC,CAAC;QACrF,IAAI,KAAK,CAAC,UAAW,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE;YACrC,OAAO,gBAAgB,CAAC,OAAO,CAAC,CAAC;SAClC;QAED,OAAO,IAAI,CAAC,cAAc,CAAC,kBAAkB,CAAC,KAAK,EAAE,OAAO,CAAC;aACxD,IAAI,CAAC,QAAQ,CAAC,CAAC,WAAyB,EAAE,EAAE;YAC3C,MAAM,KAAK,GAAG,IAAI,eAAe,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;YACnD,OAAO,IAAI,CAAC,cAAc,CAAC,QAAQ,EAAE,MAAM,EAAE,KAAK,EAAE,WAAW,EAAE,MAAM,EAAE,KAAK,CAAC,CAAC;QAClF,CAAC,CAAC,CAAC,CAAC;IACV,CAAC;IAEO,6CAA6C,CACjD,QAA6B,EAAE,YAA6B,EAAE,MAAe,EAAE,KAAY,EAC3F,QAAsB,EAAE,MAAc;QACxC,MAAM,EAAC,OAAO,EAAE,gBAAgB,EAAE,iBAAiB,EAAE,uBAAuB,EAAC,GACzE,KAAK,CAAC,YAAY,EAAE,KAAK,EAAE,QAAQ,CAAC,CAAC;QACzC,IAAI,CAAC,OAAO;YAAE,OAAO,OAAO,CAAC,YAAY,CAAC,CAAC;QAE3C,MAAM,OAAO,GAAG,IAAI,CAAC,cAAc,CAAC,qBAAqB,CACrD,gBAAgB,EAAE,KAAK,CAAC,UAAW,EAAE,uBAAuB,CAAC,CAAC;QAClE,IAAI,KAAK,CAAC,UAAW,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE;YACrC,OAAO,gBAAgB,CAAC,OAAO,CAAC,CAAC;SAClC;QAED,OAAO,IAAI,CAAC,cAAc,CAAC,kBAAkB,CAAC,KAAK,EAAE,OAAO,CAAC;aACxD,IAAI,CAAC,QAAQ,CAAC,CAAC,WAAyB,EAAE,EAAE;YAC3C,OAAO,IAAI,CAAC,cAAc,CACtB,QAAQ,EAAE,MAAM,EAAE,YAAY,EAAE,WAAW,CAAC,MAAM,CAAC,iBAAiB,CAAC,EAAE,MAAM,EAAE,KAAK,CAAC,CAAC;QAC5F,CAAC,CAAC,CAAC,CAAC;IACV,CAAC;IAED,wBAAwB,CACpB,QAA6B,EAAE,UAA2B,EAAE,KAAY,EACxE,QAAsB,EAAE,MAAc,EACtC,cAAuB;QACzB,IAAI,WAIG,CAAC;QAER,IAAI,KAAK,CAAC,IAAI,KAAK,IAAI,EAAE;YACvB,MAAM,MAAM,GAAG,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAE,CAAC,UAAU,CAAC,CAAC,CAAC,EAAE,CAAC;YACrE,MAAM,QAAQ,GAAG,IAAI,sBAAsB,CACvC,QAAQ,EAAE,MAAM,EAAE,MAAM,CAAC,MAAM,CAAC,EAAC,GAAG,IAAI,CAAC,OAAO,CAAC,WAAW,EAAC,CAAC,EAAE,IAAI,CAAC,OAAO,CAAC,QAAQ,EACrF,OAAO,CAAC,KAAK,CAAC,EAAE,SAAS,CAAC,KAAK,CAAC,EAAE,KAAK,CAAC,SAAS,IAAI,KAAK,CAAC,gBAAgB,IAAI,IAAI,EACnF,KAAK,EAAE,UAAU,CAAC,KAAK,CAAC,CAAC,CAAC;YAC9B,WAAW,GAAG,EAAE,CAAC;gBACf,QAAQ;gBACR,gBAAgB,EAAE,EAAE;gBACpB,iBAAiB,EAAE,EAAE;aACtB,CAAC,CAAC;YACH,4FAA4F;YAC5F,sFAAsF;YACtF,2DAA2D;YAC3D,kDAAkD;YAClD,UAAU,CAAC,QAAQ,GAAG,EAAE,CAAC;SAC1B;aAAM;YACL,WAAW;gBACP,eAAe,CAAC,UAAU,EAAE,KAAK,EAAE,QAAQ,EAAE,QAAQ,EAAE,IAAI,CAAC,aAAa,CAAC;qBACrE,IAAI,CAAC,GAAG,CAAC,CAAC,EAAC,OAAO,EAAE,gBAAgB,EAAE,iBAAiB,EAAE,UAAU,EAAC,EAAE,EAAE;oBACvE,IAAI,CAAC,OAAO,EAAE;wBACZ,OAAO,IAAI,CAAC;qBACb;oBAED,MAAM,QAAQ,GAAG,IAAI,sBAAsB,CACvC,gBAAgB,EAAE,UAAU,EAAE,MAAM,CAAC,MAAM,CAAC,EAAC,GAAG,IAAI,CAAC,OAAO,CAAC,WAAW,EAAC,CAAC,EAC1E,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,OAAO,CAAC,KAAK,CAAC,EAAE,SAAS,CAAC,KAAK,CAAC,EACvD,KAAK,CAAC,SAAS,IAAI,KAAK,CAAC,gBAAgB,IAAI,IAAI,EAAE,KAAK,EAAE,UAAU,CAAC,KAAK,CAAC,CAAC,CAAC;oBACjF,OAAO,EAAC,QAAQ,EAAE,gBAAgB,EAAE,iBAAiB,EAAC,CAAC;gBACzD,CAAC,CAAC,CAAC,CAAC;SACb;QAED,OAAO,WAAW,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,MAAM,EAAE,EAAE;YAC3C,IAAI,MAAM,KAAK,IAAI,EAAE;gBACnB,OAAO,OAAO,CAAC,UAAU,CAAC,CAAC;aAC5B;YAED,mFAAmF;YACnF,QAAQ,GAAG,KAAK,CAAC,SAAS,IAAI,QAAQ,CAAC;YACvC,OAAO,IAAI,CAAC,cAAc,CAAC,QAAQ,EAAE,KAAK,EAAE,QAAQ,CAAC;iBAChD,IAAI,CAAC,SAAS,CAAC,CAAC,EAAC,MAAM,EAAE,WAAW,EAAC,EAAE,EAAE;gBACxC,MAAM,aAAa,GAAG,KAAK,CAAC,eAAe,IAAI,QAAQ,CAAC;gBAExD,MAAM,EAAC,QAAQ,EAAE,gBAAgB,EAAE,iBAAiB,EAAC,GAAG,MAAM,CAAC;gBAE/D,MAAM,EAAC,YAAY,EAAE,cAAc,EAAC,GAChC,KAAK,CAAC,UAAU,EAAE,gBAAgB,EAAE,iBAAiB,EAAE,WAAW,CAAC,CAAC;gBAExE,IAAI,cAAc,CAAC,MAAM,KAAK,CAAC,IAAI,YAAY,CAAC,WAAW,EAAE,EAAE;oBAC7D,OAAO,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE,WAAW,EAAE,YAAY,CAAC;yBAChE,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE;wBACnB,IAAI,QAAQ,KAAK,IAAI,EAAE;4BACrB,OAAO,IAAI,CAAC;yBACb;wBACD,OAAO,CAAC,IAAI,QAAQ,CAAyB,QAAQ,EAAE,QAAQ,CAAC,CAAC,CAAC;oBACpE,CAAC,CAAC,CAAC,CAAC;iBACT;gBAED,IAAI,WAAW,CAAC,MAAM,KAAK,CAAC,IAAI,cAAc,CAAC,MAAM,KAAK,CAAC,EAAE;oBAC3D,OAAO,EAAE,CAAC,CAAC,IAAI,QAAQ,CAAyB,QAAQ,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC;iBACjE;gBAED,MAAM,eAAe,GAAG,SAAS,CAAC,KAAK,CAAC,KAAK,MAAM,CAAC;gBACpD,mFAAmF;gBACnF,qFAAqF;gBACrF,uFAAuF;gBACvF,6CAA6C;gBAC7C,8BAA8B;gBAC9B,8BAA8B;gBAC9B,KAAK;gBACL,sFAAsF;gBACtF,OAAO,IAAI;qBACN,cAAc,CACX,aAAa,EAAE,WAAW,EAAE,YAAY,EAAE,cAAc,EACxD,eAAe,CAAC,CAAC,CAAC,cAAc,CAAC,CAAC,CAAC,MAAM,EAAE,IAAI,CAAC;qBACnD,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE;oBACnB,OAAO,CAAC,IAAI,QAAQ,CAAyB,QAAQ,EAAE,QAAQ,CAAC,CAAC,CAAC;gBACpE,CAAC,CAAC,CAAC,CAAC;YACV,CAAC,CAAC,CAAC,CAAC;QACV,CAAC,CAAC,CAAC,CAAC;IACN,CAAC;IACO,cAAc,CAAC,QAA6B,EAAE,KAAY,EAAE,QAAsB;QAExF,IAAI,KAAK,CAAC,QAAQ,EAAE;YAClB,yCAAyC;YACzC,OAAO,EAAE,CAAC,EAAC,MAAM,EAAE,KAAK,CAAC,QAAQ,EAAE,QAAQ,EAAC,CAAC,CAAC;SAC/C;QAED,IAAI,KAAK,CAAC,YAAY,EAAE;YACtB,4CAA4C;YAC5C,IAAI,KAAK,CAAC,aAAa,KAAK,SAAS,EAAE;gBACrC,OAAO,EAAE,CAAC,EAAC,MAAM,EAAE,KAAK,CAAC,aAAa,EAAE,QAAQ,EAAE,KAAK,CAAC,eAAe,EAAC,CAAC,CAAC;aAC3E;YAED,OAAO,gBAAgB,CAAC,QAAQ,EAAE,KAAK,EAAE,QAAQ,EAAE,IAAI,CAAC,aAAa,CAAC;iBACjE,IAAI,CAAC,QAAQ,CAAC,CAAC,gBAAyB,EAAE,EAAE;gBAC3C,IAAI,gBAAgB,EAAE;oBACpB,OAAO,IAAI,CAAC,YAAY,CAAC,YAAY,CAAC,QAAQ,EAAE,KAAK,CAAC;yBACjD,IAAI,CAAC,GAAG,CAAC,CAAC,GAAuB,EAAE,EAAE;wBACpC,KAAK,CAAC,aAAa,GAAG,GAAG,CAAC,MAAM,CAAC;wBACjC,KAAK,CAAC,eAAe,GAAG,GAAG,CAAC,QAAQ,CAAC;oBACvC,CAAC,CAAC,CAAC,CAAC;iBACT;gBACD,OAAO,YAAY,CAAC,KAAK,CAAC,CAAC;YAC7B,CAAC,CAAC,CAAC,CAAC;SACT;QAED,OAAO,EAAE,CAAC,EAAC,MAAM,EAAE,EAAE,EAAE,QAAQ,EAAC,CAAC,CAAC;IACpC,CAAC;CACF;AAED,SAAS,2BAA2B,CAAC,KAAyC;IAC5E,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE;QAClB,IAAI,CAAC,CAAC,KAAK,CAAC,MAAM,KAAK,cAAc;YAAE,OAAO,CAAC,CAAC,CAAC;QACjD,IAAI,CAAC,CAAC,KAAK,CAAC,MAAM,KAAK,cAAc;YAAE,OAAO,CAAC,CAAC;QAChD,OAAO,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;IACtD,CAAC,CAAC,CAAC;AACL,CAAC;AAED,SAAS,kBAAkB,CAAC,IAAsC;IAChE,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC;IACtC,OAAO,MAAM,IAAI,MAAM,CAAC,IAAI,KAAK,EAAE,CAAC;AACtC,CAAC;AAED;;;;GAIG;AACH,SAAS,qBAAqB,CAAC,KAA8C;IAE3E,MAAM,MAAM,GAA4C,EAAE,CAAC;IAC3D,gGAAgG;IAChG,MAAM,WAAW,GAA0C,IAAI,GAAG,EAAE,CAAC;IAErE,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE;QACxB,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,EAAE;YAC7B,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAClB,SAAS;SACV;QAED,MAAM,sBAAsB,GACxB,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,KAAK,UAAU,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;QACvF,IAAI,sBAAsB,KAAK,SAAS,EAAE;YACxC,sBAAsB,CAAC,QAAQ,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,QAAQ,CAAC,CAAC;YACvD,WAAW,CAAC,GAAG,CAAC,sBAAsB,CAAC,CAAC;SACzC;aAAM;YACL,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;SACnB;KACF;IACD,gGAAgG;IAChG,+FAA+F;IAC/F,8FAA8F;IAC9F,uEAAuE;IACvE,KAAK,MAAM,UAAU,IAAI,WAAW,EAAE;QACpC,MAAM,cAAc,GAAG,qBAAqB,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;QAClE,MAAM,CAAC,IAAI,CAAC,IAAI,QAAQ,CAAC,UAAU,CAAC,KAAK,EAAE,cAAc,CAAC,CAAC,CAAC;KAC7D;IACD,OAAO,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACjD,CAAC;AAED,SAAS,yBAAyB,CAAC,KAAyC;IAC1E,MAAM,KAAK,GAA0C,EAAE,CAAC;IACxD,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE;QAChB,MAAM,uBAAuB,GAAG,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QACtD,IAAI,uBAAuB,EAAE;YAC3B,MAAM,CAAC,GAAG,uBAAuB,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YACvE,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YACvD,MAAM,IAAI,YAAY,4DAElB,CAAC,OAAO,SAAS,KAAK,WAAW,IAAI,SAAS,CAAC;gBAC3C,mDAAmD,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;SAC9E;QACD,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC;IAClC,CAAC,CAAC,CAAC;AACL,CAAC;AAED,SAAS,OAAO,CAAC,KAAY;IAC3B,OAAO,KAAK,CAAC,IAAI,IAAI,EAAE,CAAC;AAC1B,CAAC;AAED,SAAS,UAAU,CAAC,KAAY;IAC9B,OAAO,KAAK,CAAC,OAAO,IAAI,EAAE,CAAC;AAC7B,CAAC","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 {EnvironmentInjector, Type, ɵRuntimeError as RuntimeError} from '@angular/core';\nimport {from, Observable, of} from 'rxjs';\nimport {catchError, concatMap, defaultIfEmpty, first, last as rxjsLast, map, mergeMap, scan, switchMap, tap} from 'rxjs/operators';\n\nimport {absoluteRedirect, AbsoluteRedirect, ApplyRedirects, canLoadFails, noMatch, NoMatch} from './apply_redirects';\nimport {createUrlTreeFromSnapshot} from './create_url_tree';\nimport {RuntimeErrorCode} from './errors';\nimport {Data, LoadedRouterConfig, ResolveData, Route, Routes} from './models';\nimport {runCanLoadGuards} from './operators/check_guards';\nimport {RouterConfigLoader} from './router_config_loader';\nimport {ActivatedRouteSnapshot, inheritedParamsDataResolve, ParamsInheritanceStrategy, RouterStateSnapshot} from './router_state';\nimport {PRIMARY_OUTLET} from './shared';\nimport {UrlSegment, UrlSegmentGroup, UrlSerializer, UrlTree} from './url_tree';\nimport {last} from './utils/collection';\nimport {getOutlet, sortByMatchingOutlets} from './utils/config';\nimport {isImmediateMatch, match, matchWithChecks, noLeftoversInUrl, split} from './utils/config_matching';\nimport {TreeNode} from './utils/tree';\nimport {isEmptyError} from './utils/type_guards';\n\n\nexport function recognize(\n    injector: EnvironmentInjector, configLoader: RouterConfigLoader,\n    rootComponentType: Type<any>|null, config: Routes, urlTree: UrlTree,\n    urlSerializer: UrlSerializer,\n    paramsInheritanceStrategy: ParamsInheritanceStrategy =\n        'emptyOnly'): Observable<{state: RouterStateSnapshot, tree: UrlTree}> {\n  return new Recognizer(\n             injector, configLoader, rootComponentType, config, urlTree, paramsInheritanceStrategy,\n             urlSerializer)\n      .recognize();\n}\n\nexport class Recognizer {\n  allowRedirects = true;\n  private applyRedirects = new ApplyRedirects(this.urlSerializer, this.urlTree);\n\n  constructor(\n      private injector: EnvironmentInjector, private configLoader: RouterConfigLoader,\n      private rootComponentType: Type<any>|null, private config: Routes, private urlTree: UrlTree,\n      private paramsInheritanceStrategy: ParamsInheritanceStrategy,\n      private readonly urlSerializer: UrlSerializer) {}\n\n  private noMatchError(e: NoMatch): any {\n    return new RuntimeError(\n        RuntimeErrorCode.NO_MATCH,\n        (typeof ngDevMode === 'undefined' || ngDevMode) &&\n            `Cannot match any routes. URL Segment: '${e.segmentGroup}'`);\n  }\n\n  recognize(): Observable<{state: RouterStateSnapshot, tree: UrlTree}> {\n    const rootSegmentGroup = split(this.urlTree.root, [], [], this.config).segmentGroup;\n\n    return this.processSegmentGroup(this.injector, this.config, rootSegmentGroup, PRIMARY_OUTLET)\n        .pipe(\n            catchError((e: any) => {\n              if (e instanceof AbsoluteRedirect) {\n                // After an absolute redirect we do not apply any more redirects!\n                // If this implementation changes, update the documentation note in `redirectTo`.\n                this.allowRedirects = false;\n                this.urlTree = e.urlTree;\n                return this.match(e.urlTree);\n              }\n\n              if (e instanceof NoMatch) {\n                throw this.noMatchError(e);\n              }\n\n              throw e;\n            }),\n            map(children => {\n              // Use Object.freeze to prevent readers of the Router state from modifying it outside\n              // of a navigation, resulting in the router being out of sync with the browser.\n              const root = new ActivatedRouteSnapshot(\n                  [], Object.freeze({}), Object.freeze({...this.urlTree.queryParams}),\n                  this.urlTree.fragment, {}, PRIMARY_OUTLET, this.rootComponentType, null, {});\n\n              const rootNode = new TreeNode<ActivatedRouteSnapshot>(root, children);\n              const routeState = new RouterStateSnapshot('', rootNode);\n              const tree = createUrlTreeFromSnapshot(\n                  root, [], this.urlTree.queryParams, this.urlTree.fragment);\n              // https://github.com/angular/angular/issues/47307\n              // Creating the tree stringifies the query params\n              // We don't want to do this here so reassign them to the original.\n              tree.queryParams = this.urlTree.queryParams;\n              routeState.url = this.urlSerializer.serialize(tree);\n              this.inheritParamsAndData(routeState._root);\n              return {state: routeState, tree};\n            }));\n  }\n\n\n  private match(tree: UrlTree) {\n    const expanded$ =\n        this.processSegmentGroup(this.injector, this.config, tree.root, PRIMARY_OUTLET);\n    return expanded$.pipe(catchError((e: any) => {\n      if (e instanceof NoMatch) {\n        throw this.noMatchError(e);\n      }\n\n      throw e;\n    }));\n  }\n\n  inheritParamsAndData(routeNode: TreeNode<ActivatedRouteSnapshot>): void {\n    const route = routeNode.value;\n\n    const i = inheritedParamsDataResolve(route, this.paramsInheritanceStrategy);\n    route.params = Object.freeze(i.params);\n    route.data = Object.freeze(i.data);\n\n    routeNode.children.forEach(n => this.inheritParamsAndData(n));\n  }\n\n  processSegmentGroup(\n      injector: EnvironmentInjector, config: Route[], segmentGroup: UrlSegmentGroup,\n      outlet: string): Observable<TreeNode<ActivatedRouteSnapshot>[]> {\n    if (segmentGroup.segments.length === 0 && segmentGroup.hasChildren()) {\n      return this.processChildren(injector, config, segmentGroup);\n    }\n\n    return this.processSegment(injector, config, segmentGroup, segmentGroup.segments, outlet, true);\n  }\n\n  /**\n   * Matches every child outlet in the `segmentGroup` to a `Route` in the config. Returns `null` if\n   * we cannot find a match for _any_ of the children.\n   *\n   * @param config - The `Routes` to match against\n   * @param segmentGroup - The `UrlSegmentGroup` whose children need to be matched against the\n   *     config.\n   */\n  processChildren(injector: EnvironmentInjector, config: Route[], segmentGroup: UrlSegmentGroup):\n      Observable<TreeNode<ActivatedRouteSnapshot>[]> {\n    // Expand outlets one at a time, starting with the primary outlet. We need to do it this way\n    // because an absolute redirect from the primary outlet takes precedence.\n    const childOutlets: string[] = [];\n    for (const child of Object.keys(segmentGroup.children)) {\n      if (child === 'primary') {\n        childOutlets.unshift(child);\n      } else {\n        childOutlets.push(child);\n      }\n    }\n    return from(childOutlets)\n        .pipe(\n            concatMap(childOutlet => {\n              const child = segmentGroup.children[childOutlet];\n              // Sort the config so that routes with outlets that match the one being activated\n              // appear first, followed by routes for other outlets, which might match if they have\n              // an empty path.\n              const sortedConfig = sortByMatchingOutlets(config, childOutlet);\n              return this.processSegmentGroup(injector, sortedConfig, child, childOutlet);\n            }),\n            scan((children, outletChildren) => {\n              children.push(...outletChildren);\n              return children;\n            }),\n            defaultIfEmpty(null as TreeNode<ActivatedRouteSnapshot>[] | null),\n            rxjsLast(),\n            mergeMap(children => {\n              if (children === null) return noMatch(segmentGroup);\n              // Because we may have matched two outlets to the same empty path segment, we can have\n              // multiple activated results for the same outlet. We should merge the children of\n              // these results so the final return value is only one `TreeNode` per outlet.\n              const mergedChildren = mergeEmptyPathMatches(children);\n              if (typeof ngDevMode === 'undefined' || ngDevMode) {\n                // This should really never happen - we are only taking the first match for each\n                // outlet and merge the empty path matches.\n                checkOutletNameUniqueness(mergedChildren);\n              }\n              sortActivatedRouteSnapshots(mergedChildren);\n              return of(mergedChildren);\n            }),\n        );\n  }\n\n  processSegment(\n      injector: EnvironmentInjector, routes: Route[], segmentGroup: UrlSegmentGroup,\n      segments: UrlSegment[], outlet: string,\n      allowRedirects: boolean): Observable<TreeNode<ActivatedRouteSnapshot>[]> {\n    return from(routes).pipe(\n        concatMap(r => {\n          return this\n              .processSegmentAgainstRoute(\n                  r._injector ?? injector, routes, r, segmentGroup, segments, outlet,\n                  allowRedirects)\n              .pipe(catchError((e: any) => {\n                if (e instanceof NoMatch) {\n                  return of(null);\n                }\n                throw e;\n              }));\n        }),\n        first((x): x is TreeNode<ActivatedRouteSnapshot>[] => !!x), catchError(e => {\n          if (isEmptyError(e)) {\n            if (noLeftoversInUrl(segmentGroup, segments, outlet)) {\n              return of([]);\n            }\n            return noMatch(segmentGroup);\n          }\n          throw e;\n        }));\n  }\n\n  processSegmentAgainstRoute(\n      injector: EnvironmentInjector, routes: Route[], route: Route, rawSegment: UrlSegmentGroup,\n      segments: UrlSegment[], outlet: string,\n      allowRedirects: boolean): Observable<TreeNode<ActivatedRouteSnapshot>[]> {\n    if (!isImmediateMatch(route, rawSegment, segments, outlet)) return noMatch(rawSegment);\n\n    if (route.redirectTo === undefined) {\n      return this.matchSegmentAgainstRoute(\n          injector, rawSegment, route, segments, outlet, allowRedirects);\n    }\n\n    if (allowRedirects && this.allowRedirects) {\n      return this.expandSegmentAgainstRouteUsingRedirect(\n          injector, rawSegment, routes, route, segments, outlet);\n    }\n\n    return noMatch(rawSegment);\n  }\n\n  private expandSegmentAgainstRouteUsingRedirect(\n      injector: EnvironmentInjector, segmentGroup: UrlSegmentGroup, routes: Route[], route: Route,\n      segments: UrlSegment[], outlet: string): Observable<TreeNode<ActivatedRouteSnapshot>[]> {\n    if (route.path === '**') {\n      return this.expandWildCardWithParamsAgainstRouteUsingRedirect(\n          injector, routes, route, outlet);\n    }\n\n    return this.expandRegularSegmentAgainstRouteUsingRedirect(\n        injector, segmentGroup, routes, route, segments, outlet);\n  }\n\n  private expandWildCardWithParamsAgainstRouteUsingRedirect(\n      injector: EnvironmentInjector, routes: Route[], route: Route,\n      outlet: string): Observable<TreeNode<ActivatedRouteSnapshot>[]> {\n    const newTree = this.applyRedirects.applyRedirectCommands([], route.redirectTo!, {});\n    if (route.redirectTo!.startsWith('/')) {\n      return absoluteRedirect(newTree);\n    }\n\n    return this.applyRedirects.lineralizeSegments(route, newTree)\n        .pipe(mergeMap((newSegments: UrlSegment[]) => {\n          const group = new UrlSegmentGroup(newSegments, {});\n          return this.processSegment(injector, routes, group, newSegments, outlet, false);\n        }));\n  }\n\n  private expandRegularSegmentAgainstRouteUsingRedirect(\n      injector: EnvironmentInjector, segmentGroup: UrlSegmentGroup, routes: Route[], route: Route,\n      segments: UrlSegment[], outlet: string): Observable<TreeNode<ActivatedRouteSnapshot>[]> {\n    const {matched, consumedSegments, remainingSegments, positionalParamSegments} =\n        match(segmentGroup, route, segments);\n    if (!matched) return noMatch(segmentGroup);\n\n    const newTree = this.applyRedirects.applyRedirectCommands(\n        consumedSegments, route.redirectTo!, positionalParamSegments);\n    if (route.redirectTo!.startsWith('/')) {\n      return absoluteRedirect(newTree);\n    }\n\n    return this.applyRedirects.lineralizeSegments(route, newTree)\n        .pipe(mergeMap((newSegments: UrlSegment[]) => {\n          return this.processSegment(\n              injector, routes, segmentGroup, newSegments.concat(remainingSegments), outlet, false);\n        }));\n  }\n\n  matchSegmentAgainstRoute(\n      injector: EnvironmentInjector, rawSegment: UrlSegmentGroup, route: Route,\n      segments: UrlSegment[], outlet: string,\n      allowRedirects: boolean): Observable<TreeNode<ActivatedRouteSnapshot>[]> {\n    let matchResult: Observable<{\n      snapshot: ActivatedRouteSnapshot,\n      consumedSegments: UrlSegment[],\n      remainingSegments: UrlSegment[],\n    }|null>;\n\n    if (route.path === '**') {\n      const params = segments.length > 0 ? last(segments)!.parameters : {};\n      const snapshot = new ActivatedRouteSnapshot(\n          segments, params, Object.freeze({...this.urlTree.queryParams}), this.urlTree.fragment,\n          getData(route), getOutlet(route), route.component ?? route._loadedComponent ?? null,\n          route, getResolve(route));\n      matchResult = of({\n        snapshot,\n        consumedSegments: [],\n        remainingSegments: [],\n      });\n      // Prior versions of the route matching algorithm would stop matching at the wildcard route.\n      // We should investigate a better strategy for any existing children. Otherwise, these\n      // child segments are silently dropped from the navigation.\n      // https://github.com/angular/angular/issues/40089\n      rawSegment.children = {};\n    } else {\n      matchResult =\n          matchWithChecks(rawSegment, route, segments, injector, this.urlSerializer)\n              .pipe(map(({matched, consumedSegments, remainingSegments, parameters}) => {\n                if (!matched) {\n                  return null;\n                }\n\n                const snapshot = new ActivatedRouteSnapshot(\n                    consumedSegments, parameters, Object.freeze({...this.urlTree.queryParams}),\n                    this.urlTree.fragment, getData(route), getOutlet(route),\n                    route.component ?? route._loadedComponent ?? null, route, getResolve(route));\n                return {snapshot, consumedSegments, remainingSegments};\n              }));\n    }\n\n    return matchResult.pipe(switchMap((result) => {\n      if (result === null) {\n        return noMatch(rawSegment);\n      }\n\n      // If the route has an injector created from providers, we should start using that.\n      injector = route._injector ?? injector;\n      return this.getChildConfig(injector, route, segments)\n          .pipe(switchMap(({routes: childConfig}) => {\n            const childInjector = route._loadedInjector ?? injector;\n\n            const {snapshot, consumedSegments, remainingSegments} = result;\n\n            const {segmentGroup, slicedSegments} =\n                split(rawSegment, consumedSegments, remainingSegments, childConfig);\n\n            if (slicedSegments.length === 0 && segmentGroup.hasChildren()) {\n              return this.processChildren(childInjector, childConfig, segmentGroup)\n                  .pipe(map(children => {\n                    if (children === null) {\n                      return null;\n                    }\n                    return [new TreeNode<ActivatedRouteSnapshot>(snapshot, children)];\n                  }));\n            }\n\n            if (childConfig.length === 0 && slicedSegments.length === 0) {\n              return of([new TreeNode<ActivatedRouteSnapshot>(snapshot, [])]);\n            }\n\n            const matchedOnOutlet = getOutlet(route) === outlet;\n            // If we matched a config due to empty path match on a different outlet, we need to\n            // continue passing the current outlet for the segment rather than switch to PRIMARY.\n            // Note that we switch to primary when we have a match because outlet configs look like\n            // this: {path: 'a', outlet: 'a', children: [\n            //  {path: 'b', component: B},\n            //  {path: 'c', component: C},\n            // ]}\n            // Notice that the children of the named outlet are configured with the primary outlet\n            return this\n                .processSegment(\n                    childInjector, childConfig, segmentGroup, slicedSegments,\n                    matchedOnOutlet ? PRIMARY_OUTLET : outlet, true)\n                .pipe(map(children => {\n                  return [new TreeNode<ActivatedRouteSnapshot>(snapshot, children)];\n                }));\n          }));\n    }));\n  }\n  private getChildConfig(injector: EnvironmentInjector, route: Route, segments: UrlSegment[]):\n      Observable<LoadedRouterConfig> {\n    if (route.children) {\n      // The children belong to the same module\n      return of({routes: route.children, injector});\n    }\n\n    if (route.loadChildren) {\n      // lazy children belong to the loaded module\n      if (route._loadedRoutes !== undefined) {\n        return of({routes: route._loadedRoutes, injector: route._loadedInjector});\n      }\n\n      return runCanLoadGuards(injector, route, segments, this.urlSerializer)\n          .pipe(mergeMap((shouldLoadResult: boolean) => {\n            if (shouldLoadResult) {\n              return this.configLoader.loadChildren(injector, route)\n                  .pipe(tap((cfg: LoadedRouterConfig) => {\n                    route._loadedRoutes = cfg.routes;\n                    route._loadedInjector = cfg.injector;\n                  }));\n            }\n            return canLoadFails(route);\n          }));\n    }\n\n    return of({routes: [], injector});\n  }\n}\n\nfunction sortActivatedRouteSnapshots(nodes: TreeNode<ActivatedRouteSnapshot>[]): void {\n  nodes.sort((a, b) => {\n    if (a.value.outlet === PRIMARY_OUTLET) return -1;\n    if (b.value.outlet === PRIMARY_OUTLET) return 1;\n    return a.value.outlet.localeCompare(b.value.outlet);\n  });\n}\n\nfunction hasEmptyPathConfig(node: TreeNode<ActivatedRouteSnapshot>) {\n  const config = node.value.routeConfig;\n  return config && config.path === '';\n}\n\n/**\n * Finds `TreeNode`s with matching empty path route configs and merges them into `TreeNode` with\n * the children from each duplicate. This is necessary because different outlets can match a\n * single empty path route config and the results need to then be merged.\n */\nfunction mergeEmptyPathMatches(nodes: Array<TreeNode<ActivatedRouteSnapshot>>):\n    Array<TreeNode<ActivatedRouteSnapshot>> {\n  const result: Array<TreeNode<ActivatedRouteSnapshot>> = [];\n  // The set of nodes which contain children that were merged from two duplicate empty path nodes.\n  const mergedNodes: Set<TreeNode<ActivatedRouteSnapshot>> = new Set();\n\n  for (const node of nodes) {\n    if (!hasEmptyPathConfig(node)) {\n      result.push(node);\n      continue;\n    }\n\n    const duplicateEmptyPathNode =\n        result.find(resultNode => node.value.routeConfig === resultNode.value.routeConfig);\n    if (duplicateEmptyPathNode !== undefined) {\n      duplicateEmptyPathNode.children.push(...node.children);\n      mergedNodes.add(duplicateEmptyPathNode);\n    } else {\n      result.push(node);\n    }\n  }\n  // For each node which has children from multiple sources, we need to recompute a new `TreeNode`\n  // by also merging those children. This is necessary when there are multiple empty path configs\n  // in a row. Put another way: whenever we combine children of two nodes, we need to also check\n  // if any of those children can be combined into a single node as well.\n  for (const mergedNode of mergedNodes) {\n    const mergedChildren = mergeEmptyPathMatches(mergedNode.children);\n    result.push(new TreeNode(mergedNode.value, mergedChildren));\n  }\n  return result.filter(n => !mergedNodes.has(n));\n}\n\nfunction checkOutletNameUniqueness(nodes: TreeNode<ActivatedRouteSnapshot>[]): void {\n  const names: {[k: string]: ActivatedRouteSnapshot} = {};\n  nodes.forEach(n => {\n    const routeWithSameOutletName = names[n.value.outlet];\n    if (routeWithSameOutletName) {\n      const p = routeWithSameOutletName.url.map(s => s.toString()).join('/');\n      const c = n.value.url.map(s => s.toString()).join('/');\n      throw new RuntimeError(\n          RuntimeErrorCode.TWO_SEGMENTS_WITH_SAME_OUTLET,\n          (typeof ngDevMode === 'undefined' || ngDevMode) &&\n              `Two segments cannot have the same outlet name: '${p}' and '${c}'.`);\n    }\n    names[n.value.outlet] = n.value;\n  });\n}\n\nfunction getData(route: Route): Data {\n  return route.data || {};\n}\n\nfunction getResolve(route: Route): ResolveData {\n  return route.resolve || {};\n}\n"]}
|
|
@@ -15,11 +15,11 @@ import * as i0 from "@angular/core";
|
|
|
15
15
|
* @publicApi
|
|
16
16
|
*/
|
|
17
17
|
class RouteReuseStrategy {
|
|
18
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0-next.6", ngImport: i0, type: RouteReuseStrategy, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
19
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.0.0-next.6", ngImport: i0, type: RouteReuseStrategy, providedIn: 'root', useFactory: () => inject(DefaultRouteReuseStrategy) }); }
|
|
18
20
|
}
|
|
19
|
-
RouteReuseStrategy.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0-next.4", ngImport: i0, type: RouteReuseStrategy, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
20
|
-
RouteReuseStrategy.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.0.0-next.4", ngImport: i0, type: RouteReuseStrategy, providedIn: 'root', useFactory: () => inject(DefaultRouteReuseStrategy) });
|
|
21
21
|
export { RouteReuseStrategy };
|
|
22
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0-next.
|
|
22
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0-next.6", ngImport: i0, type: RouteReuseStrategy, decorators: [{
|
|
23
23
|
type: Injectable,
|
|
24
24
|
args: [{ providedIn: 'root', useFactory: () => inject(DefaultRouteReuseStrategy) }]
|
|
25
25
|
}] });
|
|
@@ -70,12 +70,12 @@ export class BaseRouteReuseStrategy {
|
|
|
70
70
|
}
|
|
71
71
|
}
|
|
72
72
|
class DefaultRouteReuseStrategy extends BaseRouteReuseStrategy {
|
|
73
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0-next.6", ngImport: i0, type: DefaultRouteReuseStrategy, deps: null, target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
74
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.0.0-next.6", ngImport: i0, type: DefaultRouteReuseStrategy, providedIn: 'root' }); }
|
|
73
75
|
}
|
|
74
|
-
DefaultRouteReuseStrategy.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0-next.4", ngImport: i0, type: DefaultRouteReuseStrategy, deps: null, target: i0.ɵɵFactoryTarget.Injectable });
|
|
75
|
-
DefaultRouteReuseStrategy.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.0.0-next.4", ngImport: i0, type: DefaultRouteReuseStrategy, providedIn: 'root' });
|
|
76
76
|
export { DefaultRouteReuseStrategy };
|
|
77
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0-next.
|
|
77
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0-next.6", ngImport: i0, type: DefaultRouteReuseStrategy, decorators: [{
|
|
78
78
|
type: Injectable,
|
|
79
79
|
args: [{ providedIn: 'root' }]
|
|
80
80
|
}] });
|
|
81
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
81
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicm91dGVfcmV1c2Vfc3RyYXRlZ3kuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9yb3V0ZXIvc3JjL3JvdXRlX3JldXNlX3N0cmF0ZWd5LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7R0FNRztBQUVILE9BQU8sRUFBZSxNQUFNLEVBQUUsVUFBVSxFQUFDLE1BQU0sZUFBZSxDQUFDOztBQXlCL0Q7Ozs7OztHQU1HO0FBQ0gsTUFDc0Isa0JBQWtCO3lIQUFsQixrQkFBa0I7NkhBQWxCLGtCQUFrQixjQURmLE1BQU0sY0FBYyxHQUFHLEVBQUUsQ0FBQyxNQUFNLENBQUMseUJBQXlCLENBQUM7O1NBQzlELGtCQUFrQjtzR0FBbEIsa0JBQWtCO2tCQUR2QyxVQUFVO21CQUFDLEVBQUMsVUFBVSxFQUFFLE1BQU0sRUFBRSxVQUFVLEVBQUUsR0FBRyxFQUFFLENBQUMsTUFBTSxDQUFDLHlCQUF5QixDQUFDLEVBQUM7O0FBc0JyRjs7Ozs7Ozs7Ozs7Ozs7OztHQWdCRztBQUNILE1BQU0sT0FBZ0Isc0JBQXNCO0lBQzFDOzs7U0FHSztJQUNMLFlBQVksQ0FBQyxLQUE2QjtRQUN4QyxPQUFPLEtBQUssQ0FBQztJQUNmLENBQUM7SUFFRDs7T0FFRztJQUNILEtBQUssQ0FBQyxLQUE2QixFQUFFLFlBQWlDLElBQVMsQ0FBQztJQUVoRiwrRUFBK0U7SUFDL0UsWUFBWSxDQUFDLEtBQTZCO1FBQ3hDLE9BQU8sS0FBSyxDQUFDO0lBQ2YsQ0FBQztJQUVELG1GQUFtRjtJQUNuRixRQUFRLENBQUMsS0FBNkI7UUFDcEMsT0FBTyxJQUFJLENBQUM7SUFDZCxDQUFDO0lBRUQ7Ozs7T0FJRztJQUNILGdCQUFnQixDQUFDLE1BQThCLEVBQUUsSUFBNEI7UUFDM0UsT0FBTyxNQUFNLENBQUMsV0FBVyxLQUFLLElBQUksQ0FBQyxXQUFXLENBQUM7SUFDakQsQ0FBQztDQUNGO0FBRUQsTUFDYSx5QkFBMEIsU0FBUSxzQkFBc0I7eUhBQXhELHlCQUF5Qjs2SEFBekIseUJBQXlCLGNBRGIsTUFBTTs7U0FDbEIseUJBQXlCO3NHQUF6Qix5QkFBeUI7a0JBRHJDLFVBQVU7bUJBQUMsRUFBQyxVQUFVLEVBQUUsTUFBTSxFQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBAbGljZW5zZVxuICogQ29weXJpZ2h0IEdvb2dsZSBMTEMgQWxsIFJpZ2h0cyBSZXNlcnZlZC5cbiAqXG4gKiBVc2Ugb2YgdGhpcyBzb3VyY2UgY29kZSBpcyBnb3Zlcm5lZCBieSBhbiBNSVQtc3R5bGUgbGljZW5zZSB0aGF0IGNhbiBiZVxuICogZm91bmQgaW4gdGhlIExJQ0VOU0UgZmlsZSBhdCBodHRwczovL2FuZ3VsYXIuaW8vbGljZW5zZVxuICovXG5cbmltcG9ydCB7Q29tcG9uZW50UmVmLCBpbmplY3QsIEluamVjdGFibGV9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5pbXBvcnQge091dGxldENvbnRleHR9IGZyb20gJy4vcm91dGVyX291dGxldF9jb250ZXh0JztcbmltcG9ydCB7QWN0aXZhdGVkUm91dGUsIEFjdGl2YXRlZFJvdXRlU25hcHNob3R9IGZyb20gJy4vcm91dGVyX3N0YXRlJztcbmltcG9ydCB7VHJlZU5vZGV9IGZyb20gJy4vdXRpbHMvdHJlZSc7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uXG4gKlxuICogUmVwcmVzZW50cyB0aGUgZGV0YWNoZWQgcm91dGUgdHJlZS5cbiAqXG4gKiBUaGlzIGlzIGFuIG9wYXF1ZSB2YWx1ZSB0aGUgcm91dGVyIHdpbGwgZ2l2ZSB0byBhIGN1c3RvbSByb3V0ZSByZXVzZSBzdHJhdGVneVxuICogdG8gc3RvcmUgYW5kIHJldHJpZXZlIGxhdGVyIG9uLlxuICpcbiAqIEBwdWJsaWNBcGlcbiAqL1xuZXhwb3J0IHR5cGUgRGV0YWNoZWRSb3V0ZUhhbmRsZSA9IHt9O1xuXG4vKiogQGludGVybmFsICovXG5leHBvcnQgdHlwZSBEZXRhY2hlZFJvdXRlSGFuZGxlSW50ZXJuYWwgPSB7XG4gIGNvbnRleHRzOiBNYXA8c3RyaW5nLCBPdXRsZXRDb250ZXh0PixcbiAgY29tcG9uZW50UmVmOiBDb21wb25lbnRSZWY8YW55PixcbiAgcm91dGU6IFRyZWVOb2RlPEFjdGl2YXRlZFJvdXRlPixcbn07XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uXG4gKlxuICogUHJvdmlkZXMgYSB3YXkgdG8gY3VzdG9taXplIHdoZW4gYWN0aXZhdGVkIHJvdXRlcyBnZXQgcmV1c2VkLlxuICpcbiAqIEBwdWJsaWNBcGlcbiAqL1xuQEluamVjdGFibGUoe3Byb3ZpZGVkSW46ICdyb290JywgdXNlRmFjdG9yeTogKCkgPT4gaW5qZWN0KERlZmF1bHRSb3V0ZVJldXNlU3RyYXRlZ3kpfSlcbmV4cG9ydCBhYnN0cmFjdCBjbGFzcyBSb3V0ZVJldXNlU3RyYXRlZ3kge1xuICAvKiogRGV0ZXJtaW5lcyBpZiB0aGlzIHJvdXRlIChhbmQgaXRzIHN1YnRyZWUpIHNob3VsZCBiZSBkZXRhY2hlZCB0byBiZSByZXVzZWQgbGF0ZXIgKi9cbiAgYWJzdHJhY3Qgc2hvdWxkRGV0YWNoKHJvdXRlOiBBY3RpdmF0ZWRSb3V0ZVNuYXBzaG90KTogYm9vbGVhbjtcblxuICAvKipcbiAgICogU3RvcmVzIHRoZSBkZXRhY2hlZCByb3V0ZS5cbiAgICpcbiAgICogU3RvcmluZyBhIGBudWxsYCB2YWx1ZSBzaG91bGQgZXJhc2UgdGhlIHByZXZpb3VzbHkgc3RvcmVkIHZhbHVlLlxuICAgKi9cbiAgYWJzdHJhY3Qgc3RvcmUocm91dGU6IEFjdGl2YXRlZFJvdXRlU25hcHNob3QsIGhhbmRsZTogRGV0YWNoZWRSb3V0ZUhhbmRsZXxudWxsKTogdm9pZDtcblxuICAvKiogRGV0ZXJtaW5lcyBpZiB0aGlzIHJvdXRlIChhbmQgaXRzIHN1YnRyZWUpIHNob3VsZCBiZSByZWF0dGFjaGVkICovXG4gIGFic3RyYWN0IHNob3VsZEF0dGFjaChyb3V0ZTogQWN0aXZhdGVkUm91dGVTbmFwc2hvdCk6IGJvb2xlYW47XG5cbiAgLyoqIFJldHJpZXZlcyB0aGUgcHJldmlvdXNseSBzdG9yZWQgcm91dGUgKi9cbiAgYWJzdHJhY3QgcmV0cmlldmUocm91dGU6IEFjdGl2YXRlZFJvdXRlU25hcHNob3QpOiBEZXRhY2hlZFJvdXRlSGFuZGxlfG51bGw7XG5cbiAgLyoqIERldGVybWluZXMgaWYgYSByb3V0ZSBzaG91bGQgYmUgcmV1c2VkICovXG4gIGFic3RyYWN0IHNob3VsZFJldXNlUm91dGUoZnV0dXJlOiBBY3RpdmF0ZWRSb3V0ZVNuYXBzaG90LCBjdXJyOiBBY3RpdmF0ZWRSb3V0ZVNuYXBzaG90KTogYm9vbGVhbjtcbn1cblxuLyoqXG4gKiBAZGVzY3JpcHRpb25cbiAqXG4gKiBUaGlzIGJhc2Ugcm91dGUgcmV1c2Ugc3RyYXRlZ3kgb25seSByZXVzZXMgcm91dGVzIHdoZW4gdGhlIG1hdGNoZWQgcm91dGVyIGNvbmZpZ3MgYXJlXG4gKiBpZGVudGljYWwuIFRoaXMgcHJldmVudHMgY29tcG9uZW50cyBmcm9tIGJlaW5nIGRlc3Ryb3llZCBhbmQgcmVjcmVhdGVkXG4gKiB3aGVuIGp1c3QgdGhlIHJvdXRlIHBhcmFtZXRlcnMsIHF1ZXJ5IHBhcmFtZXRlcnMgb3IgZnJhZ21lbnQgY2hhbmdlXG4gKiAodGhhdCBpcywgdGhlIGV4aXN0aW5nIGNvbXBvbmVudCBpcyBfcmV1c2VkXykuXG4gKlxuICogVGhpcyBzdHJhdGVneSBkb2VzIG5vdCBzdG9yZSBhbnkgcm91dGVzIGZvciBsYXRlciByZXVzZS5cbiAqXG4gKiBBbmd1bGFyIHVzZXMgdGhpcyBzdHJhdGVneSBieSBkZWZhdWx0LlxuICpcbiAqXG4gKiBJdCBjYW4gYmUgdXNlZCBhcyBhIGJhc2UgY2xhc3MgZm9yIGN1c3RvbSByb3V0ZSByZXVzZSBzdHJhdGVnaWVzLCBpLmUuIHlvdSBjYW4gY3JlYXRlIHlvdXIgb3duXG4gKiBjbGFzcyB0aGF0IGV4dGVuZHMgdGhlIGBCYXNlUm91dGVSZXVzZVN0cmF0ZWd5YCBvbmUuXG4gKiBAcHVibGljQXBpXG4gKi9cbmV4cG9ydCBhYnN0cmFjdCBjbGFzcyBCYXNlUm91dGVSZXVzZVN0cmF0ZWd5IGltcGxlbWVudHMgUm91dGVSZXVzZVN0cmF0ZWd5IHtcbiAgLyoqXG4gICAqIFdoZXRoZXIgdGhlIGdpdmVuIHJvdXRlIHNob3VsZCBkZXRhY2ggZm9yIGxhdGVyIHJldXNlLlxuICAgKiBBbHdheXMgcmV0dXJucyBmYWxzZSBmb3IgYEJhc2VSb3V0ZVJldXNlU3RyYXRlZ3lgLlxuICAgKiAqL1xuICBzaG91bGREZXRhY2gocm91dGU6IEFjdGl2YXRlZFJvdXRlU25hcHNob3QpOiBib29sZWFuIHtcbiAgICByZXR1cm4gZmFsc2U7XG4gIH1cblxuICAvKipcbiAgICogQSBuby1vcDsgdGhlIHJvdXRlIGlzIG5ldmVyIHN0b3JlZCBzaW5jZSB0aGlzIHN0cmF0ZWd5IG5ldmVyIGRldGFjaGVzIHJvdXRlcyBmb3IgbGF0ZXIgcmUtdXNlLlxuICAgKi9cbiAgc3RvcmUocm91dGU6IEFjdGl2YXRlZFJvdXRlU25hcHNob3QsIGRldGFjaGVkVHJlZTogRGV0YWNoZWRSb3V0ZUhhbmRsZSk6IHZvaWQge31cblxuICAvKiogUmV0dXJucyBgZmFsc2VgLCBtZWFuaW5nIHRoZSByb3V0ZSAoYW5kIGl0cyBzdWJ0cmVlKSBpcyBuZXZlciByZWF0dGFjaGVkICovXG4gIHNob3VsZEF0dGFjaChyb3V0ZTogQWN0aXZhdGVkUm91dGVTbmFwc2hvdCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiBmYWxzZTtcbiAgfVxuXG4gIC8qKiBSZXR1cm5zIGBudWxsYCBiZWNhdXNlIHRoaXMgc3RyYXRlZ3kgZG9lcyBub3Qgc3RvcmUgcm91dGVzIGZvciBsYXRlciByZS11c2UuICovXG4gIHJldHJpZXZlKHJvdXRlOiBBY3RpdmF0ZWRSb3V0ZVNuYXBzaG90KTogRGV0YWNoZWRSb3V0ZUhhbmRsZXxudWxsIHtcbiAgICByZXR1cm4gbnVsbDtcbiAgfVxuXG4gIC8qKlxuICAgKiBEZXRlcm1pbmVzIGlmIGEgcm91dGUgc2hvdWxkIGJlIHJldXNlZC5cbiAgICogVGhpcyBzdHJhdGVneSByZXR1cm5zIGB0cnVlYCB3aGVuIHRoZSBmdXR1cmUgcm91dGUgY29uZmlnIGFuZCBjdXJyZW50IHJvdXRlIGNvbmZpZyBhcmVcbiAgICogaWRlbnRpY2FsLlxuICAgKi9cbiAgc2hvdWxkUmV1c2VSb3V0ZShmdXR1cmU6IEFjdGl2YXRlZFJvdXRlU25hcHNob3QsIGN1cnI6IEFjdGl2YXRlZFJvdXRlU25hcHNob3QpOiBib29sZWFuIHtcbiAgICByZXR1cm4gZnV0dXJlLnJvdXRlQ29uZmlnID09PSBjdXJyLnJvdXRlQ29uZmlnO1xuICB9XG59XG5cbkBJbmplY3RhYmxlKHtwcm92aWRlZEluOiAncm9vdCd9KVxuZXhwb3J0IGNsYXNzIERlZmF1bHRSb3V0ZVJldXNlU3RyYXRlZ3kgZXh0ZW5kcyBCYXNlUm91dGVSZXVzZVN0cmF0ZWd5IHtcbn1cbiJdfQ==
|