@ngrdt/router 0.0.12 → 0.0.14
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/lib/directives/rdt-any-route-active.directive.mjs +9 -4
- package/esm2022/lib/rdt-route/rdt-angular-route.mjs +4 -2
- package/esm2022/lib/rdt-route/rdt-route-builder.mjs +8 -2
- package/esm2022/lib/rdt-route/rdt-route.mjs +14 -6
- package/esm2022/lib/services/rdt-router.service.mjs +61 -29
- package/fesm2022/ngrdt-router.mjs +91 -38
- package/fesm2022/ngrdt-router.mjs.map +1 -1
- package/lib/directives/rdt-any-route-active.directive.d.ts +4 -1
- package/lib/rdt-route/rdt-route.d.ts +4 -3
- package/lib/services/rdt-router.service.d.ts +23 -4
- package/package.json +1 -1
|
@@ -6,6 +6,8 @@ import { filter, take } from 'rxjs';
|
|
|
6
6
|
import { RdtReadonlyParameters } from '../rdt-route/utils';
|
|
7
7
|
import { RDT_ROUTES_PROVIDER } from './rdt-routes.token';
|
|
8
8
|
import * as i0 from "@angular/core";
|
|
9
|
+
const DEFAULT_TARGET = '_self';
|
|
10
|
+
const RDT_STATE_PARAMS_KEY = 'RdtStateParams';
|
|
9
11
|
export class RdtRouterService {
|
|
10
12
|
allRoutes = inject(RDT_ROUTES_PROVIDER, { optional: true });
|
|
11
13
|
location = inject(Location);
|
|
@@ -37,41 +39,54 @@ export class RdtRouterService {
|
|
|
37
39
|
this._currentUrl = e.url;
|
|
38
40
|
});
|
|
39
41
|
}
|
|
40
|
-
|
|
42
|
+
/**
|
|
43
|
+
* @returns window.history.state extended by state parameters passed from parent window.
|
|
44
|
+
*/
|
|
45
|
+
getHistoryState() {
|
|
46
|
+
if ('RdtStateParams' in window &&
|
|
47
|
+
typeof window[RDT_STATE_PARAMS_KEY] === 'object') {
|
|
48
|
+
return { ...window[RDT_STATE_PARAMS_KEY], ...(history.state ?? {}) };
|
|
49
|
+
}
|
|
50
|
+
else {
|
|
51
|
+
return history.state ?? {};
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
/**
|
|
55
|
+
* Navigates to the specified route.
|
|
56
|
+
* @param link Route to navigate to.
|
|
57
|
+
* @param params Parameter of route passed as first argument.
|
|
58
|
+
* @param extras Allows you to specify additional parameters for navigation. These extras have higher priority than those defined in the route.
|
|
59
|
+
* @returns Promise that resolves when navigation is complete.
|
|
60
|
+
*/
|
|
61
|
+
navigate(link, params, extras) {
|
|
41
62
|
const url = params
|
|
42
63
|
? link.createAbsoluteUrl(params)
|
|
43
64
|
: link.createAbsoluteUrl();
|
|
65
|
+
const target = extras?.target ?? DEFAULT_TARGET;
|
|
66
|
+
const queryParams = extras?.query ?? link.queryParams;
|
|
67
|
+
const stateParams = extras?.state ?? link.stateParams;
|
|
44
68
|
if (target === '_self') {
|
|
45
69
|
return this.router.navigate([url], {
|
|
46
|
-
queryParams:
|
|
47
|
-
state:
|
|
70
|
+
queryParams: queryParams,
|
|
71
|
+
state: stateParams,
|
|
72
|
+
replaceUrl: extras?.replaceUrl,
|
|
48
73
|
});
|
|
49
74
|
}
|
|
50
75
|
const absolutePath = RdtStringUtils.createAbsoluteUrl(url, this.baseHref);
|
|
51
|
-
const pathWithParams = RdtStringUtils.appendQueryParams(absolutePath,
|
|
76
|
+
const pathWithParams = RdtStringUtils.appendQueryParams(absolutePath, queryParams);
|
|
52
77
|
const win = window.open(pathWithParams, target);
|
|
53
78
|
if (win) {
|
|
54
|
-
win
|
|
79
|
+
win[RDT_STATE_PARAMS_KEY] = stateParams;
|
|
55
80
|
}
|
|
56
81
|
return Promise.resolve(true);
|
|
57
82
|
}
|
|
58
|
-
navigateHome(
|
|
59
|
-
return this.router.navigateByUrl(RdtStringUtils.appendQueryParams('/',
|
|
60
|
-
state:
|
|
83
|
+
navigateHome(extras) {
|
|
84
|
+
return this.router.navigateByUrl(RdtStringUtils.appendQueryParams('/', extras?.query ?? {}), {
|
|
85
|
+
state: extras?.state,
|
|
86
|
+
replaceUrl: extras?.replaceUrl,
|
|
61
87
|
});
|
|
62
88
|
}
|
|
63
|
-
|
|
64
|
-
if (this.allRoutes) {
|
|
65
|
-
for (const route of this.allRoutes) {
|
|
66
|
-
const params = route.parseAbsoluteUrl(url);
|
|
67
|
-
if (params) {
|
|
68
|
-
return { route, params };
|
|
69
|
-
}
|
|
70
|
-
}
|
|
71
|
-
}
|
|
72
|
-
return null;
|
|
73
|
-
}
|
|
74
|
-
navigateBack(params) {
|
|
89
|
+
navigateBack(extras) {
|
|
75
90
|
const parsed = this.parseAbsoluteUrl();
|
|
76
91
|
if (parsed) {
|
|
77
92
|
let route = parsed.route.withStaticParams(parsed.params);
|
|
@@ -80,23 +95,40 @@ export class RdtRouterService {
|
|
|
80
95
|
} while (route && route.entryDisabled);
|
|
81
96
|
// In case route has no ancestor with allowed entry,
|
|
82
97
|
if (!route) {
|
|
83
|
-
return this.navigateHome(
|
|
98
|
+
return this.navigateHome(extras);
|
|
84
99
|
}
|
|
85
|
-
if (
|
|
86
|
-
if (
|
|
87
|
-
route = route.withQueryParams(
|
|
100
|
+
if (extras) {
|
|
101
|
+
if (extras.query) {
|
|
102
|
+
route = route.withQueryParams(extras.query);
|
|
88
103
|
}
|
|
89
|
-
if (
|
|
90
|
-
route = route.withStateParams(
|
|
104
|
+
if (extras.state) {
|
|
105
|
+
route = route.withStateParams(extras.state);
|
|
91
106
|
}
|
|
92
107
|
}
|
|
93
|
-
return this.navigate(route);
|
|
108
|
+
return this.navigate(route, extras);
|
|
94
109
|
}
|
|
95
110
|
else {
|
|
96
111
|
console.warn(`Cannot go back from ${this.location.path()} because no route matches current url`);
|
|
97
112
|
return null;
|
|
98
113
|
}
|
|
99
114
|
}
|
|
115
|
+
/**
|
|
116
|
+
* This method will try to find matching route for given absolute URL, extract its parameters and return them.
|
|
117
|
+
* If no route matches the URL, it returns null. Parameters are extracted only for the last matching route, not ancestors.
|
|
118
|
+
* @param url Absolute URL to parse (e.g. '/home/details/123?query=abc').
|
|
119
|
+
* @returns Object containing the matching route and its parameters, or null if no route is matching.
|
|
120
|
+
*/
|
|
121
|
+
parseAbsoluteUrl(url = this.location.path()) {
|
|
122
|
+
if (this.allRoutes) {
|
|
123
|
+
for (const route of this.allRoutes) {
|
|
124
|
+
const params = route.parseAbsoluteUrl(url);
|
|
125
|
+
if (params) {
|
|
126
|
+
return { route, params };
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
return null;
|
|
131
|
+
}
|
|
100
132
|
extractAllParams(currentRoute) {
|
|
101
133
|
if (!currentRoute) {
|
|
102
134
|
const parsed = this.parseAbsoluteUrl();
|
|
@@ -115,7 +147,7 @@ export class RdtRouterService {
|
|
|
115
147
|
}
|
|
116
148
|
removeQueryParams(...paramNames) {
|
|
117
149
|
const regex = new RegExp(`[?&](${paramNames.join('|')})=[^&]*`, 'g');
|
|
118
|
-
return history.replaceState(
|
|
150
|
+
return history.replaceState(history.state, '', location.pathname + location.search.replace(regex, '').replace(/^&/, '?'));
|
|
119
151
|
}
|
|
120
152
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: RdtRouterService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
121
153
|
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: RdtRouterService, providedIn: 'root' });
|
|
@@ -126,4 +158,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImpor
|
|
|
126
158
|
providedIn: 'root',
|
|
127
159
|
}]
|
|
128
160
|
}], ctorParameters: () => [] });
|
|
129
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rdt-router.service.js","sourceRoot":"","sources":["../../../../../../@ngrdt/router/src/lib/services/rdt-router.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Location, PlatformLocation } from '@angular/common';\r\nimport { inject, Injectable } from '@angular/core';\r\nimport { NavigationEnd, Params, Router } from '@angular/router';\r\nimport { RdtStringUtils } from '@ngrdt/utils';\r\nimport { filter, Observable, take } from 'rxjs';\r\nimport { RdtRoute } from '../rdt-route/rdt-route';\r\nimport { RdtReadonlyParameters } from '../rdt-route/utils';\r\nimport { RDT_ROUTES_PROVIDER } from './rdt-routes.token';\r\n\r\nexport interface RdtNavigateParams {\r\n  state?: Params;\r\n  query?: Params;\r\n}\r\n\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class RdtRouterService {\r\n  readonly allRoutes = inject(RDT_ROUTES_PROVIDER, { optional: true });\r\n\r\n  private readonly location = inject(Location);\r\n  private readonly router = inject(Router);\r\n  private readonly baseHref = inject(PlatformLocation).getBaseHrefFromDOM();\r\n\r\n  private _previousUrl: string | null = null;\r\n  private _currentUrl: string | null = null;\r\n\r\n  get previousUrl() {\r\n    return this._previousUrl;\r\n  }\r\n\r\n  get currentUrl() {\r\n    return this._currentUrl;\r\n  }\r\n\r\n  parsePreviousUrl() {\r\n    return this._previousUrl ? this.parseAbsoluteUrl(this._previousUrl) : null;\r\n  }\r\n\r\n  parseCurrentUrl() {\r\n    return this._currentUrl ? this.parseAbsoluteUrl(this._currentUrl) : null;\r\n  }\r\n\r\n  readonly navigationEnd$ = this.router.events.pipe(\r\n    filter((e) => e instanceof NavigationEnd)\r\n  ) as Observable<NavigationEnd>;\r\n\r\n  readonly nextNavigationEnd$ = this.navigationEnd$.pipe(take(1));\r\n\r\n  constructor() {\r\n    if (this.allRoutes === null) {\r\n      console.warn(\r\n        'All routes not provided. Make sure to provide RDT_ROUTES_PROVIDER with RdtRoute[].'\r\n      );\r\n      this.allRoutes = [];\r\n    }\r\n\r\n    this.navigationEnd$.subscribe((e) => {\r\n      this._previousUrl = this._currentUrl;\r\n      this._currentUrl = e.url;\r\n    });\r\n  }\r\n\r\n  navigate<T extends object>(\r\n    link: RdtRoute<T>,\r\n    params?: Partial<T>,\r\n    target: '_blank' | '_self' | '_parent' | '_top' = '_self'\r\n  ) {\r\n    const url = params\r\n      ? link.createAbsoluteUrl(params)\r\n      : link.createAbsoluteUrl();\r\n\r\n    if (target === '_self') {\r\n      return this.router.navigate([url], {\r\n        queryParams: link.queryParams,\r\n        state: link.stateParams,\r\n      });\r\n    }\r\n\r\n    const absolutePath = RdtStringUtils.createAbsoluteUrl(url, this.baseHref);\r\n    const pathWithParams = RdtStringUtils.appendQueryParams(\r\n      absolutePath,\r\n      link.queryParams\r\n    );\r\n    const win = window.open(pathWithParams, target);\r\n    if (win) {\r\n      (win as any).RdtStateParams = link.stateParams;\r\n    }\r\n    return Promise.resolve(true);\r\n  }\r\n\r\n  navigateHome(params?: RdtNavigateParams) {\r\n    return this.router.navigateByUrl(\r\n      RdtStringUtils.appendQueryParams('/', params?.query ?? {}),\r\n      {\r\n        state: params?.state,\r\n      }\r\n    );\r\n  }\r\n\r\n  parseAbsoluteUrl(url = this.location.path()) {\r\n    if (this.allRoutes) {\r\n      for (const route of this.allRoutes) {\r\n        const params = route.parseAbsoluteUrl(url);\r\n        if (params) {\r\n          return { route, params };\r\n        }\r\n      }\r\n    }\r\n    return null;\r\n  }\r\n\r\n  navigateBack(params?: RdtNavigateParams) {\r\n    const parsed = this.parseAbsoluteUrl();\r\n    if (parsed) {\r\n      let route: RdtRoute | null = parsed.route.withStaticParams(parsed.params);\r\n      do {\r\n        route = route.parent;\r\n      } while (route && route.entryDisabled);\r\n\r\n      // In case route has no ancestor with allowed entry,\r\n      if (!route) {\r\n        return this.navigateHome(params);\r\n      }\r\n      if (params) {\r\n        if (params.query) {\r\n          route = route.withQueryParams(params.query);\r\n        }\r\n        if (params.state) {\r\n          route = route.withStateParams(params.state);\r\n        }\r\n      }\r\n      return this.navigate(route);\r\n    } else {\r\n      console.warn(\r\n        `Cannot go back from ${this.location.path()} because no route matches current url`\r\n      );\r\n      return null;\r\n    }\r\n  }\r\n\r\n  extractAllParams(currentRoute?: RdtRoute): RdtReadonlyParameters | null {\r\n    if (!currentRoute) {\r\n      const parsed = this.parseAbsoluteUrl();\r\n      if (!parsed) {\r\n        console.warn('No route matches current url.');\r\n        return new RdtReadonlyParameters();\r\n      }\r\n      currentRoute = parsed.route;\r\n    }\r\n    const url = this.location.path();\r\n    return currentRoute.parseAbsoluteUrl(url);\r\n  }\r\n\r\n  isParentOfCurrentLocation(route: RdtRoute) {\r\n    console.log(this.parseAbsoluteUrl());\r\n    return true;\r\n  }\r\n\r\n  removeQueryParams(...paramNames: string[]) {\r\n    const regex = new RegExp(`[?&](${paramNames.join('|')})=[^&]*`, 'g');\r\n\r\n    return history.replaceState(\r\n      null,\r\n      '',\r\n      location.pathname + location.search.replace(regex, '').replace(/^&/, '?')\r\n    );\r\n  }\r\n}\r\n"]}
|
|
161
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rdt-router.service.js","sourceRoot":"","sources":["../../../../../../@ngrdt/router/src/lib/services/rdt-router.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,gBAAgB,EAAE,MAAM,iBAAiB,CAAC;AAC7D,OAAO,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AACnD,OAAO,EAAE,aAAa,EAAU,MAAM,EAAE,MAAM,iBAAiB,CAAC;AAChE,OAAO,EAAE,cAAc,EAAE,MAAM,cAAc,CAAC;AAC9C,OAAO,EAAE,MAAM,EAAc,IAAI,EAAE,MAAM,MAAM,CAAC;AAEhD,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAC3D,OAAO,EAAE,mBAAmB,EAAE,MAAM,oBAAoB,CAAC;;AASzD,MAAM,cAAc,GAAG,OAAO,CAAC;AAC/B,MAAM,oBAAoB,GAAG,gBAAgB,CAAC;AAK9C,MAAM,OAAO,gBAAgB;IAClB,SAAS,GAAG,MAAM,CAAC,mBAAmB,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC;IAEpD,QAAQ,GAAG,MAAM,CAAC,QAAQ,CAAC,CAAC;IAC5B,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC;IACxB,QAAQ,GAAG,MAAM,CAAC,gBAAgB,CAAC,CAAC,kBAAkB,EAAE,CAAC;IAElE,YAAY,GAAkB,IAAI,CAAC;IACnC,WAAW,GAAkB,IAAI,CAAC;IAE1C,IAAI,WAAW;QACb,OAAO,IAAI,CAAC,YAAY,CAAC;IAC3B,CAAC;IAED,IAAI,UAAU;QACZ,OAAO,IAAI,CAAC,WAAW,CAAC;IAC1B,CAAC;IAED,gBAAgB;QACd,OAAO,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;IAC7E,CAAC;IAED,eAAe;QACb,OAAO,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;IAC3E,CAAC;IAEQ,cAAc,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAC/C,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,YAAY,aAAa,CAAC,CACb,CAAC;IAEtB,kBAAkB,GAAG,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;IAEhE;QACE,IAAI,IAAI,CAAC,SAAS,KAAK,IAAI,EAAE,CAAC;YAC5B,OAAO,CAAC,IAAI,CACV,oFAAoF,CACrF,CAAC;YACF,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;QACtB,CAAC;QAED,IAAI,CAAC,cAAc,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE;YAClC,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,WAAW,CAAC;YACrC,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC,GAAG,CAAC;QAC3B,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACH,eAAe;QACb,IACE,gBAAgB,IAAI,MAAM;YAC1B,OAAO,MAAM,CAAC,oBAAoB,CAAC,KAAK,QAAQ,EAChD,CAAC;YACD,OAAO,EAAE,GAAG,MAAM,CAAC,oBAAoB,CAAC,EAAE,GAAG,CAAC,OAAO,CAAC,KAAK,IAAI,EAAE,CAAC,EAAE,CAAC;QACvE,CAAC;aAAM,CAAC;YACN,OAAO,OAAO,CAAC,KAAK,IAAI,EAAE,CAAC;QAC7B,CAAC;IACH,CAAC;IAED;;;;;;OAMG;IACH,QAAQ,CACN,IAAiB,EACjB,MAAmB,EACnB,MAA0B;QAE1B,MAAM,GAAG,GAAG,MAAM;YAChB,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC;YAChC,CAAC,CAAC,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAE7B,MAAM,MAAM,GAAG,MAAM,EAAE,MAAM,IAAI,cAAc,CAAC;QAChD,MAAM,WAAW,GAAG,MAAM,EAAE,KAAK,IAAI,IAAI,CAAC,WAAW,CAAC;QACtD,MAAM,WAAW,GAAG,MAAM,EAAE,KAAK,IAAI,IAAI,CAAC,WAAW,CAAC;QAEtD,IAAI,MAAM,KAAK,OAAO,EAAE,CAAC;YACvB,OAAO,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,GAAG,CAAC,EAAE;gBACjC,WAAW,EAAE,WAAW;gBACxB,KAAK,EAAE,WAAW;gBAClB,UAAU,EAAE,MAAM,EAAE,UAAU;aAC/B,CAAC,CAAC;QACL,CAAC;QAED,MAAM,YAAY,GAAG,cAAc,CAAC,iBAAiB,CAAC,GAAG,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;QAC1E,MAAM,cAAc,GAAG,cAAc,CAAC,iBAAiB,CACrD,YAAY,EACZ,WAAW,CACZ,CAAC;QACF,MAAM,GAAG,GAAG,MAAM,CAAC,IAAI,CAAC,cAAc,EAAE,MAAM,CAAC,CAAC;QAChD,IAAI,GAAG,EAAE,CAAC;YACP,GAAW,CAAC,oBAAoB,CAAC,GAAG,WAAW,CAAC;QACnD,CAAC;QACD,OAAO,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;IAC/B,CAAC;IAED,YAAY,CAAC,MAA0B;QACrC,OAAO,IAAI,CAAC,MAAM,CAAC,aAAa,CAC9B,cAAc,CAAC,iBAAiB,CAAC,GAAG,EAAE,MAAM,EAAE,KAAK,IAAI,EAAE,CAAC,EAC1D;YACE,KAAK,EAAE,MAAM,EAAE,KAAK;YACpB,UAAU,EAAE,MAAM,EAAE,UAAU;SAC/B,CACF,CAAC;IACJ,CAAC;IAED,YAAY,CAAC,MAA0B;QACrC,MAAM,MAAM,GAAG,IAAI,CAAC,gBAAgB,EAAE,CAAC;QACvC,IAAI,MAAM,EAAE,CAAC;YACX,IAAI,KAAK,GAAoB,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;YAC1E,GAAG,CAAC;gBACF,KAAK,GAAG,KAAK,CAAC,MAAM,CAAC;YACvB,CAAC,QAAQ,KAAK,IAAI,KAAK,CAAC,aAAa,EAAE;YAEvC,oDAAoD;YACpD,IAAI,CAAC,KAAK,EAAE,CAAC;gBACX,OAAO,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC;YACnC,CAAC;YACD,IAAI,MAAM,EAAE,CAAC;gBACX,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;oBACjB,KAAK,GAAG,KAAK,CAAC,eAAe,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;gBAC9C,CAAC;gBACD,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;oBACjB,KAAK,GAAG,KAAK,CAAC,eAAe,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;gBAC9C,CAAC;YACH,CAAC;YACD,OAAO,IAAI,CAAC,QAAQ,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC;QACtC,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,IAAI,CACV,uBAAuB,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,uCAAuC,CACnF,CAAC;YACF,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED;;;;;OAKG;IACH,gBAAgB,CAAC,GAAG,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE;QACzC,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACnB,KAAK,MAAM,KAAK,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;gBACnC,MAAM,MAAM,GAAG,KAAK,CAAC,gBAAgB,CAAC,GAAG,CAAC,CAAC;gBAC3C,IAAI,MAAM,EAAE,CAAC;oBACX,OAAO,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC;gBAC3B,CAAC;YACH,CAAC;QACH,CAAC;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAED,gBAAgB,CAAC,YAAuB;QACtC,IAAI,CAAC,YAAY,EAAE,CAAC;YAClB,MAAM,MAAM,GAAG,IAAI,CAAC,gBAAgB,EAAE,CAAC;YACvC,IAAI,CAAC,MAAM,EAAE,CAAC;gBACZ,OAAO,CAAC,IAAI,CAAC,+BAA+B,CAAC,CAAC;gBAC9C,OAAO,IAAI,qBAAqB,EAAE,CAAC;YACrC,CAAC;YACD,YAAY,GAAG,MAAM,CAAC,KAAK,CAAC;QAC9B,CAAC;QACD,MAAM,GAAG,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACjC,OAAO,YAAY,CAAC,gBAAgB,CAAC,GAAG,CAAC,CAAC;IAC5C,CAAC;IAED,yBAAyB,CAAC,KAAe;QACvC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC,CAAC;QACrC,OAAO,IAAI,CAAC;IACd,CAAC;IAED,iBAAiB,CAAC,GAAG,UAAoB;QACvC,MAAM,KAAK,GAAG,IAAI,MAAM,CAAC,QAAQ,UAAU,CAAC,IAAI,CAAC,GAAG,CAAC,SAAS,EAAE,GAAG,CAAC,CAAC;QAErE,OAAO,OAAO,CAAC,YAAY,CACzB,OAAO,CAAC,KAAK,EACb,EAAE,EACF,QAAQ,CAAC,QAAQ,GAAG,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,CAC1E,CAAC;IACJ,CAAC;uGAvLU,gBAAgB;2GAAhB,gBAAgB,cAFf,MAAM;;2FAEP,gBAAgB;kBAH5B,UAAU;mBAAC;oBACV,UAAU,EAAE,MAAM;iBACnB","sourcesContent":["import { Location, PlatformLocation } from '@angular/common';\r\nimport { inject, Injectable } from '@angular/core';\r\nimport { NavigationEnd, Params, Router } from '@angular/router';\r\nimport { RdtStringUtils } from '@ngrdt/utils';\r\nimport { filter, Observable, take } from 'rxjs';\r\nimport { RdtRoute } from '../rdt-route/rdt-route';\r\nimport { RdtReadonlyParameters } from '../rdt-route/utils';\r\nimport { RDT_ROUTES_PROVIDER } from './rdt-routes.token';\r\n\r\nexport interface RdtNavigateExtras {\r\n  state?: Params;\r\n  query?: Params;\r\n  target?: '_blank' | '_self' | '_parent' | '_top';\r\n  replaceUrl?: boolean;\r\n}\r\n\r\nconst DEFAULT_TARGET = '_self';\r\nconst RDT_STATE_PARAMS_KEY = 'RdtStateParams';\r\n\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class RdtRouterService {\r\n  readonly allRoutes = inject(RDT_ROUTES_PROVIDER, { optional: true });\r\n\r\n  private readonly location = inject(Location);\r\n  private readonly router = inject(Router);\r\n  private readonly baseHref = inject(PlatformLocation).getBaseHrefFromDOM();\r\n\r\n  private _previousUrl: string | null = null;\r\n  private _currentUrl: string | null = null;\r\n\r\n  get previousUrl() {\r\n    return this._previousUrl;\r\n  }\r\n\r\n  get currentUrl() {\r\n    return this._currentUrl;\r\n  }\r\n\r\n  parsePreviousUrl() {\r\n    return this._previousUrl ? this.parseAbsoluteUrl(this._previousUrl) : null;\r\n  }\r\n\r\n  parseCurrentUrl() {\r\n    return this._currentUrl ? this.parseAbsoluteUrl(this._currentUrl) : null;\r\n  }\r\n\r\n  readonly navigationEnd$ = this.router.events.pipe(\r\n    filter((e) => e instanceof NavigationEnd)\r\n  ) as Observable<NavigationEnd>;\r\n\r\n  readonly nextNavigationEnd$ = this.navigationEnd$.pipe(take(1));\r\n\r\n  constructor() {\r\n    if (this.allRoutes === null) {\r\n      console.warn(\r\n        'All routes not provided. Make sure to provide RDT_ROUTES_PROVIDER with RdtRoute[].'\r\n      );\r\n      this.allRoutes = [];\r\n    }\r\n\r\n    this.navigationEnd$.subscribe((e) => {\r\n      this._previousUrl = this._currentUrl;\r\n      this._currentUrl = e.url;\r\n    });\r\n  }\r\n\r\n  /**\r\n   * @returns window.history.state extended by state parameters passed from parent window.\r\n   */\r\n  getHistoryState() {\r\n    if (\r\n      'RdtStateParams' in window &&\r\n      typeof window[RDT_STATE_PARAMS_KEY] === 'object'\r\n    ) {\r\n      return { ...window[RDT_STATE_PARAMS_KEY], ...(history.state ?? {}) };\r\n    } else {\r\n      return history.state ?? {};\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Navigates to the specified route.\r\n   * @param link Route to navigate to.\r\n   * @param params Parameter of route passed as first argument.\r\n   * @param extras Allows you to specify additional parameters for navigation. These extras have higher priority than those defined in the route.\r\n   * @returns Promise that resolves when navigation is complete.\r\n   */\r\n  navigate<T extends object>(\r\n    link: RdtRoute<T>,\r\n    params?: Partial<T>,\r\n    extras?: RdtNavigateExtras\r\n  ) {\r\n    const url = params\r\n      ? link.createAbsoluteUrl(params)\r\n      : link.createAbsoluteUrl();\r\n\r\n    const target = extras?.target ?? DEFAULT_TARGET;\r\n    const queryParams = extras?.query ?? link.queryParams;\r\n    const stateParams = extras?.state ?? link.stateParams;\r\n\r\n    if (target === '_self') {\r\n      return this.router.navigate([url], {\r\n        queryParams: queryParams,\r\n        state: stateParams,\r\n        replaceUrl: extras?.replaceUrl,\r\n      });\r\n    }\r\n\r\n    const absolutePath = RdtStringUtils.createAbsoluteUrl(url, this.baseHref);\r\n    const pathWithParams = RdtStringUtils.appendQueryParams(\r\n      absolutePath,\r\n      queryParams\r\n    );\r\n    const win = window.open(pathWithParams, target);\r\n    if (win) {\r\n      (win as any)[RDT_STATE_PARAMS_KEY] = stateParams;\r\n    }\r\n    return Promise.resolve(true);\r\n  }\r\n\r\n  navigateHome(extras?: RdtNavigateExtras) {\r\n    return this.router.navigateByUrl(\r\n      RdtStringUtils.appendQueryParams('/', extras?.query ?? {}),\r\n      {\r\n        state: extras?.state,\r\n        replaceUrl: extras?.replaceUrl,\r\n      }\r\n    );\r\n  }\r\n\r\n  navigateBack(extras?: RdtNavigateExtras) {\r\n    const parsed = this.parseAbsoluteUrl();\r\n    if (parsed) {\r\n      let route: RdtRoute | null = parsed.route.withStaticParams(parsed.params);\r\n      do {\r\n        route = route.parent;\r\n      } while (route && route.entryDisabled);\r\n\r\n      // In case route has no ancestor with allowed entry,\r\n      if (!route) {\r\n        return this.navigateHome(extras);\r\n      }\r\n      if (extras) {\r\n        if (extras.query) {\r\n          route = route.withQueryParams(extras.query);\r\n        }\r\n        if (extras.state) {\r\n          route = route.withStateParams(extras.state);\r\n        }\r\n      }\r\n      return this.navigate(route, extras);\r\n    } else {\r\n      console.warn(\r\n        `Cannot go back from ${this.location.path()} because no route matches current url`\r\n      );\r\n      return null;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * This method will try to find matching route for given absolute URL, extract its parameters and return them.\r\n   * If no route matches the URL, it returns null. Parameters are extracted only for the last matching route, not ancestors.\r\n   * @param url Absolute URL to parse (e.g. '/home/details/123?query=abc').\r\n   * @returns Object containing the matching route and its parameters, or null if no route is matching.\r\n   */\r\n  parseAbsoluteUrl(url = this.location.path()) {\r\n    if (this.allRoutes) {\r\n      for (const route of this.allRoutes) {\r\n        const params = route.parseAbsoluteUrl(url);\r\n        if (params) {\r\n          return { route, params };\r\n        }\r\n      }\r\n    }\r\n    return null;\r\n  }\r\n\r\n  extractAllParams(currentRoute?: RdtRoute): RdtReadonlyParameters | null {\r\n    if (!currentRoute) {\r\n      const parsed = this.parseAbsoluteUrl();\r\n      if (!parsed) {\r\n        console.warn('No route matches current url.');\r\n        return new RdtReadonlyParameters();\r\n      }\r\n      currentRoute = parsed.route;\r\n    }\r\n    const url = this.location.path();\r\n    return currentRoute.parseAbsoluteUrl(url);\r\n  }\r\n\r\n  isParentOfCurrentLocation(route: RdtRoute) {\r\n    console.log(this.parseAbsoluteUrl());\r\n    return true;\r\n  }\r\n\r\n  removeQueryParams(...paramNames: string[]) {\r\n    const regex = new RegExp(`[?&](${paramNames.join('|')})=[^&]*`, 'g');\r\n\r\n    return history.replaceState(\r\n      history.state,\r\n      '',\r\n      location.pathname + location.search.replace(regex, '').replace(/^&/, '?')\r\n    );\r\n  }\r\n}\r\n"]}
|
|
@@ -40,6 +40,8 @@ class RdtParameters extends RdtReadonlyParameters {
|
|
|
40
40
|
|
|
41
41
|
const RDT_ROUTES_PROVIDER = new InjectionToken('RDT_ROUTES_PROVIDER');
|
|
42
42
|
|
|
43
|
+
const DEFAULT_TARGET = '_self';
|
|
44
|
+
const RDT_STATE_PARAMS_KEY = 'RdtStateParams';
|
|
43
45
|
class RdtRouterService {
|
|
44
46
|
allRoutes = inject(RDT_ROUTES_PROVIDER, { optional: true });
|
|
45
47
|
location = inject(Location);
|
|
@@ -71,41 +73,54 @@ class RdtRouterService {
|
|
|
71
73
|
this._currentUrl = e.url;
|
|
72
74
|
});
|
|
73
75
|
}
|
|
74
|
-
|
|
76
|
+
/**
|
|
77
|
+
* @returns window.history.state extended by state parameters passed from parent window.
|
|
78
|
+
*/
|
|
79
|
+
getHistoryState() {
|
|
80
|
+
if ('RdtStateParams' in window &&
|
|
81
|
+
typeof window[RDT_STATE_PARAMS_KEY] === 'object') {
|
|
82
|
+
return { ...window[RDT_STATE_PARAMS_KEY], ...(history.state ?? {}) };
|
|
83
|
+
}
|
|
84
|
+
else {
|
|
85
|
+
return history.state ?? {};
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
/**
|
|
89
|
+
* Navigates to the specified route.
|
|
90
|
+
* @param link Route to navigate to.
|
|
91
|
+
* @param params Parameter of route passed as first argument.
|
|
92
|
+
* @param extras Allows you to specify additional parameters for navigation. These extras have higher priority than those defined in the route.
|
|
93
|
+
* @returns Promise that resolves when navigation is complete.
|
|
94
|
+
*/
|
|
95
|
+
navigate(link, params, extras) {
|
|
75
96
|
const url = params
|
|
76
97
|
? link.createAbsoluteUrl(params)
|
|
77
98
|
: link.createAbsoluteUrl();
|
|
99
|
+
const target = extras?.target ?? DEFAULT_TARGET;
|
|
100
|
+
const queryParams = extras?.query ?? link.queryParams;
|
|
101
|
+
const stateParams = extras?.state ?? link.stateParams;
|
|
78
102
|
if (target === '_self') {
|
|
79
103
|
return this.router.navigate([url], {
|
|
80
|
-
queryParams:
|
|
81
|
-
state:
|
|
104
|
+
queryParams: queryParams,
|
|
105
|
+
state: stateParams,
|
|
106
|
+
replaceUrl: extras?.replaceUrl,
|
|
82
107
|
});
|
|
83
108
|
}
|
|
84
109
|
const absolutePath = RdtStringUtils.createAbsoluteUrl(url, this.baseHref);
|
|
85
|
-
const pathWithParams = RdtStringUtils.appendQueryParams(absolutePath,
|
|
110
|
+
const pathWithParams = RdtStringUtils.appendQueryParams(absolutePath, queryParams);
|
|
86
111
|
const win = window.open(pathWithParams, target);
|
|
87
112
|
if (win) {
|
|
88
|
-
win
|
|
113
|
+
win[RDT_STATE_PARAMS_KEY] = stateParams;
|
|
89
114
|
}
|
|
90
115
|
return Promise.resolve(true);
|
|
91
116
|
}
|
|
92
|
-
navigateHome(
|
|
93
|
-
return this.router.navigateByUrl(RdtStringUtils.appendQueryParams('/',
|
|
94
|
-
state:
|
|
117
|
+
navigateHome(extras) {
|
|
118
|
+
return this.router.navigateByUrl(RdtStringUtils.appendQueryParams('/', extras?.query ?? {}), {
|
|
119
|
+
state: extras?.state,
|
|
120
|
+
replaceUrl: extras?.replaceUrl,
|
|
95
121
|
});
|
|
96
122
|
}
|
|
97
|
-
|
|
98
|
-
if (this.allRoutes) {
|
|
99
|
-
for (const route of this.allRoutes) {
|
|
100
|
-
const params = route.parseAbsoluteUrl(url);
|
|
101
|
-
if (params) {
|
|
102
|
-
return { route, params };
|
|
103
|
-
}
|
|
104
|
-
}
|
|
105
|
-
}
|
|
106
|
-
return null;
|
|
107
|
-
}
|
|
108
|
-
navigateBack(params) {
|
|
123
|
+
navigateBack(extras) {
|
|
109
124
|
const parsed = this.parseAbsoluteUrl();
|
|
110
125
|
if (parsed) {
|
|
111
126
|
let route = parsed.route.withStaticParams(parsed.params);
|
|
@@ -114,23 +129,40 @@ class RdtRouterService {
|
|
|
114
129
|
} while (route && route.entryDisabled);
|
|
115
130
|
// In case route has no ancestor with allowed entry,
|
|
116
131
|
if (!route) {
|
|
117
|
-
return this.navigateHome(
|
|
132
|
+
return this.navigateHome(extras);
|
|
118
133
|
}
|
|
119
|
-
if (
|
|
120
|
-
if (
|
|
121
|
-
route = route.withQueryParams(
|
|
134
|
+
if (extras) {
|
|
135
|
+
if (extras.query) {
|
|
136
|
+
route = route.withQueryParams(extras.query);
|
|
122
137
|
}
|
|
123
|
-
if (
|
|
124
|
-
route = route.withStateParams(
|
|
138
|
+
if (extras.state) {
|
|
139
|
+
route = route.withStateParams(extras.state);
|
|
125
140
|
}
|
|
126
141
|
}
|
|
127
|
-
return this.navigate(route);
|
|
142
|
+
return this.navigate(route, extras);
|
|
128
143
|
}
|
|
129
144
|
else {
|
|
130
145
|
console.warn(`Cannot go back from ${this.location.path()} because no route matches current url`);
|
|
131
146
|
return null;
|
|
132
147
|
}
|
|
133
148
|
}
|
|
149
|
+
/**
|
|
150
|
+
* This method will try to find matching route for given absolute URL, extract its parameters and return them.
|
|
151
|
+
* If no route matches the URL, it returns null. Parameters are extracted only for the last matching route, not ancestors.
|
|
152
|
+
* @param url Absolute URL to parse (e.g. '/home/details/123?query=abc').
|
|
153
|
+
* @returns Object containing the matching route and its parameters, or null if no route is matching.
|
|
154
|
+
*/
|
|
155
|
+
parseAbsoluteUrl(url = this.location.path()) {
|
|
156
|
+
if (this.allRoutes) {
|
|
157
|
+
for (const route of this.allRoutes) {
|
|
158
|
+
const params = route.parseAbsoluteUrl(url);
|
|
159
|
+
if (params) {
|
|
160
|
+
return { route, params };
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
return null;
|
|
165
|
+
}
|
|
134
166
|
extractAllParams(currentRoute) {
|
|
135
167
|
if (!currentRoute) {
|
|
136
168
|
const parsed = this.parseAbsoluteUrl();
|
|
@@ -149,7 +181,7 @@ class RdtRouterService {
|
|
|
149
181
|
}
|
|
150
182
|
removeQueryParams(...paramNames) {
|
|
151
183
|
const regex = new RegExp(`[?&](${paramNames.join('|')})=[^&]*`, 'g');
|
|
152
|
-
return history.replaceState(
|
|
184
|
+
return history.replaceState(history.state, '', location.pathname + location.search.replace(regex, '').replace(/^&/, '?'));
|
|
153
185
|
}
|
|
154
186
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: RdtRouterService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
155
187
|
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: RdtRouterService, providedIn: 'root' });
|
|
@@ -315,7 +347,9 @@ class RdtAngularRoute {
|
|
|
315
347
|
return undefined;
|
|
316
348
|
}
|
|
317
349
|
const routeSegmentLength = this.route.path.split('/').length;
|
|
318
|
-
const regex =
|
|
350
|
+
const regex = this.route.regex
|
|
351
|
+
? new RegExp(`^${this.route.regex.source}$`)
|
|
352
|
+
: this.route.absoluteRegex;
|
|
319
353
|
return [
|
|
320
354
|
(route, segments) => {
|
|
321
355
|
const path = segments
|
|
@@ -361,7 +395,7 @@ function ALWAYS_TRUE() {
|
|
|
361
395
|
class RdtRoute {
|
|
362
396
|
orderedParams = [];
|
|
363
397
|
_paramMap = {};
|
|
364
|
-
_regex;
|
|
398
|
+
_regex = null;
|
|
365
399
|
paramMappings = [];
|
|
366
400
|
_staticParams = {};
|
|
367
401
|
_queryParams = {};
|
|
@@ -373,6 +407,7 @@ class RdtRoute {
|
|
|
373
407
|
_children = [];
|
|
374
408
|
_entryDisabled = false;
|
|
375
409
|
_canBeEntered = ALWAYS_TRUE;
|
|
410
|
+
_built = false;
|
|
376
411
|
_absoluteRegex;
|
|
377
412
|
/**
|
|
378
413
|
* Map of parameters and their types.
|
|
@@ -432,7 +467,13 @@ class RdtRoute {
|
|
|
432
467
|
if (this._absoluteRegex) {
|
|
433
468
|
return this._absoluteRegex;
|
|
434
469
|
}
|
|
435
|
-
if (this.
|
|
470
|
+
if (!this._regex) {
|
|
471
|
+
this._absoluteRegex = new RegExp('^/$');
|
|
472
|
+
if (this._parent) {
|
|
473
|
+
throw new Error('Nested route with empty path is not allowed.');
|
|
474
|
+
}
|
|
475
|
+
}
|
|
476
|
+
else if (this._parent) {
|
|
436
477
|
// Remove initial ^ and ending $
|
|
437
478
|
const parent = this._parent.absoluteRegex.source.slice(1, -1);
|
|
438
479
|
this._absoluteRegex = new RegExp('^' + RdtStringUtils.joinPaths(parent, this._regex.source) + '$');
|
|
@@ -569,7 +610,7 @@ class RdtRoute {
|
|
|
569
610
|
return path;
|
|
570
611
|
}
|
|
571
612
|
toAngularRoute() {
|
|
572
|
-
if (!this.
|
|
613
|
+
if (!this._built) {
|
|
573
614
|
throw new Error('You have to first call .build() on route!');
|
|
574
615
|
}
|
|
575
616
|
return new RdtAngularRoute(this);
|
|
@@ -644,9 +685,10 @@ class RdtRoute {
|
|
|
644
685
|
clone._entryDisabled = this._entryDisabled;
|
|
645
686
|
clone.orderedParams = [...this.orderedParams];
|
|
646
687
|
clone._paramMap = { ...this._paramMap };
|
|
647
|
-
clone._regex = new RegExp(this._regex);
|
|
688
|
+
clone._regex = this._regex ? new RegExp(this._regex) : null;
|
|
648
689
|
clone.paramMappings = this.paramMappings;
|
|
649
|
-
clone._staticParams = this._staticParams;
|
|
690
|
+
clone._staticParams = { ...this._staticParams };
|
|
691
|
+
clone._stateParams = { ...this._stateParams };
|
|
650
692
|
return clone;
|
|
651
693
|
}
|
|
652
694
|
}
|
|
@@ -734,6 +776,7 @@ class RdtRouteBuilder extends RdtRoute {
|
|
|
734
776
|
if (typeof this._path !== 'string') {
|
|
735
777
|
throw new Error('Please provide path for route. Empty string is acceptable.');
|
|
736
778
|
}
|
|
779
|
+
this._built = true;
|
|
737
780
|
this.setRegex();
|
|
738
781
|
return this;
|
|
739
782
|
}
|
|
@@ -748,7 +791,12 @@ class RdtRouteBuilder extends RdtRoute {
|
|
|
748
791
|
}
|
|
749
792
|
substituted = substituted.replace(`:${p}`, RdtRouteBuilder.groups[type]);
|
|
750
793
|
});
|
|
751
|
-
|
|
794
|
+
if (substituted === '') {
|
|
795
|
+
this._regex = null;
|
|
796
|
+
}
|
|
797
|
+
else {
|
|
798
|
+
this._regex = new RegExp(substituted);
|
|
799
|
+
}
|
|
752
800
|
}
|
|
753
801
|
static groups = {
|
|
754
802
|
string: '(.+)',
|
|
@@ -768,6 +816,9 @@ const EXACT_FALSE_OPTS = {
|
|
|
768
816
|
fragment: 'ignored',
|
|
769
817
|
matrixParams: 'ignored',
|
|
770
818
|
};
|
|
819
|
+
/**
|
|
820
|
+
* Directive that appends class to host if any of the watched routes is active.
|
|
821
|
+
*/
|
|
771
822
|
class RdtAnyRouteActiveDirective {
|
|
772
823
|
destroyRef = inject(DestroyRef);
|
|
773
824
|
router = inject(Router);
|
|
@@ -853,7 +904,7 @@ class RdtAnyRouteActiveDirective {
|
|
|
853
904
|
}
|
|
854
905
|
}
|
|
855
906
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: RdtAnyRouteActiveDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
856
|
-
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.7", type: RdtAnyRouteActiveDirective, isStandalone: true, selector: "[rdtAnyRouteActive]", inputs: { anyRouteActive: "anyRouteActive", watchedRoutes: "watchedRoutes", anyRouteActiveOptions: "anyRouteActiveOptions", ariaCurrentWhenActive: "ariaCurrentWhenActive" }, providers: [RouterModule], usesOnChanges: true, ngImport: i0 });
|
|
907
|
+
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.7", type: RdtAnyRouteActiveDirective, isStandalone: true, selector: "[rdtAnyRouteActive]", inputs: { anyRouteActive: ["rdtAnyRouteActive", "anyRouteActive"], watchedRoutes: "watchedRoutes", anyRouteActiveOptions: "anyRouteActiveOptions", ariaCurrentWhenActive: "ariaCurrentWhenActive" }, providers: [RouterModule], usesOnChanges: true, ngImport: i0 });
|
|
857
908
|
}
|
|
858
909
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: RdtAnyRouteActiveDirective, decorators: [{
|
|
859
910
|
type: Directive,
|
|
@@ -863,9 +914,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImpor
|
|
|
863
914
|
providers: [RouterModule],
|
|
864
915
|
}]
|
|
865
916
|
}], propDecorators: { anyRouteActive: [{
|
|
866
|
-
type: Input
|
|
917
|
+
type: Input,
|
|
918
|
+
args: [{ required: true, alias: 'rdtAnyRouteActive' }]
|
|
867
919
|
}], watchedRoutes: [{
|
|
868
|
-
type: Input
|
|
920
|
+
type: Input,
|
|
921
|
+
args: [{ required: true }]
|
|
869
922
|
}], anyRouteActiveOptions: [{
|
|
870
923
|
type: Input
|
|
871
924
|
}], ariaCurrentWhenActive: [{
|