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 { ChangeDetectorRef, ContentChildren, Directive, ElementRef, Input, Optional, QueryList, Renderer2 } from '@angular/core';
|
---|
9 | import { from, of } from 'rxjs';
|
---|
10 | import { mergeAll } from 'rxjs/operators';
|
---|
11 | import { NavigationEnd } from '../events';
|
---|
12 | import { Router } from '../router';
|
---|
13 | import { RouterLink, RouterLinkWithHref } from './router_link';
|
---|
14 | /**
|
---|
15 | *
|
---|
16 | * @description
|
---|
17 | *
|
---|
18 | * Tracks whether the linked route of an element is currently active, and allows you
|
---|
19 | * to specify one or more CSS classes to add to the element when the linked route
|
---|
20 | * is active.
|
---|
21 | *
|
---|
22 | * Use this directive to create a visual distinction for elements associated with an active route.
|
---|
23 | * For example, the following code highlights the word "Bob" when the router
|
---|
24 | * activates the associated route:
|
---|
25 | *
|
---|
26 | * ```
|
---|
27 | * <a routerLink="/user/bob" routerLinkActive="active-link">Bob</a>
|
---|
28 | * ```
|
---|
29 | *
|
---|
30 | * Whenever the URL is either '/user' or '/user/bob', the "active-link" class is
|
---|
31 | * added to the anchor tag. If the URL changes, the class is removed.
|
---|
32 | *
|
---|
33 | * You can set more than one class using a space-separated string or an array.
|
---|
34 | * For example:
|
---|
35 | *
|
---|
36 | * ```
|
---|
37 | * <a routerLink="/user/bob" routerLinkActive="class1 class2">Bob</a>
|
---|
38 | * <a routerLink="/user/bob" [routerLinkActive]="['class1', 'class2']">Bob</a>
|
---|
39 | * ```
|
---|
40 | *
|
---|
41 | * To add the classes only when the URL matches the link exactly, add the option `exact: true`:
|
---|
42 | *
|
---|
43 | * ```
|
---|
44 | * <a routerLink="/user/bob" routerLinkActive="active-link" [routerLinkActiveOptions]="{exact:
|
---|
45 | * true}">Bob</a>
|
---|
46 | * ```
|
---|
47 | *
|
---|
48 | * To directly check the `isActive` status of the link, assign the `RouterLinkActive`
|
---|
49 | * instance to a template variable.
|
---|
50 | * For example, the following checks the status without assigning any CSS classes:
|
---|
51 | *
|
---|
52 | * ```
|
---|
53 | * <a routerLink="/user/bob" routerLinkActive #rla="routerLinkActive">
|
---|
54 | * Bob {{ rla.isActive ? '(already open)' : ''}}
|
---|
55 | * </a>
|
---|
56 | * ```
|
---|
57 | *
|
---|
58 | * You can apply the `RouterLinkActive` directive to an ancestor of linked elements.
|
---|
59 | * For example, the following sets the active-link class on the `<div>` parent tag
|
---|
60 | * when the URL is either '/user/jim' or '/user/bob'.
|
---|
61 | *
|
---|
62 | * ```
|
---|
63 | * <div routerLinkActive="active-link" [routerLinkActiveOptions]="{exact: true}">
|
---|
64 | * <a routerLink="/user/jim">Jim</a>
|
---|
65 | * <a routerLink="/user/bob">Bob</a>
|
---|
66 | * </div>
|
---|
67 | * ```
|
---|
68 | *
|
---|
69 | * @ngModule RouterModule
|
---|
70 | *
|
---|
71 | * @publicApi
|
---|
72 | */
|
---|
73 | export class RouterLinkActive {
|
---|
74 | constructor(router, element, renderer, cdr, link, linkWithHref) {
|
---|
75 | this.router = router;
|
---|
76 | this.element = element;
|
---|
77 | this.renderer = renderer;
|
---|
78 | this.cdr = cdr;
|
---|
79 | this.link = link;
|
---|
80 | this.linkWithHref = linkWithHref;
|
---|
81 | this.classes = [];
|
---|
82 | this.isActive = false;
|
---|
83 | /**
|
---|
84 | * Options to configure how to determine if the router link is active.
|
---|
85 | *
|
---|
86 | * These options are passed to the `Router.isActive()` function.
|
---|
87 | *
|
---|
88 | * @see Router.isActive
|
---|
89 | */
|
---|
90 | this.routerLinkActiveOptions = { exact: false };
|
---|
91 | this.routerEventsSubscription = router.events.subscribe((s) => {
|
---|
92 | if (s instanceof NavigationEnd) {
|
---|
93 | this.update();
|
---|
94 | }
|
---|
95 | });
|
---|
96 | }
|
---|
97 | /** @nodoc */
|
---|
98 | ngAfterContentInit() {
|
---|
99 | // `of(null)` is used to force subscribe body to execute once immediately (like `startWith`).
|
---|
100 | of(this.links.changes, this.linksWithHrefs.changes, of(null)).pipe(mergeAll()).subscribe(_ => {
|
---|
101 | this.update();
|
---|
102 | this.subscribeToEachLinkOnChanges();
|
---|
103 | });
|
---|
104 | }
|
---|
105 | subscribeToEachLinkOnChanges() {
|
---|
106 | var _a;
|
---|
107 | (_a = this.linkInputChangesSubscription) === null || _a === void 0 ? void 0 : _a.unsubscribe();
|
---|
108 | const allLinkChanges = [...this.links.toArray(), ...this.linksWithHrefs.toArray(), this.link, this.linkWithHref]
|
---|
109 | .filter((link) => !!link)
|
---|
110 | .map(link => link.onChanges);
|
---|
111 | this.linkInputChangesSubscription = from(allLinkChanges).pipe(mergeAll()).subscribe(link => {
|
---|
112 | if (this.isActive !== this.isLinkActive(this.router)(link)) {
|
---|
113 | this.update();
|
---|
114 | }
|
---|
115 | });
|
---|
116 | }
|
---|
117 | set routerLinkActive(data) {
|
---|
118 | const classes = Array.isArray(data) ? data : data.split(' ');
|
---|
119 | this.classes = classes.filter(c => !!c);
|
---|
120 | }
|
---|
121 | /** @nodoc */
|
---|
122 | ngOnChanges(changes) {
|
---|
123 | this.update();
|
---|
124 | }
|
---|
125 | /** @nodoc */
|
---|
126 | ngOnDestroy() {
|
---|
127 | var _a;
|
---|
128 | this.routerEventsSubscription.unsubscribe();
|
---|
129 | (_a = this.linkInputChangesSubscription) === null || _a === void 0 ? void 0 : _a.unsubscribe();
|
---|
130 | }
|
---|
131 | update() {
|
---|
132 | if (!this.links || !this.linksWithHrefs || !this.router.navigated)
|
---|
133 | return;
|
---|
134 | Promise.resolve().then(() => {
|
---|
135 | const hasActiveLinks = this.hasActiveLinks();
|
---|
136 | if (this.isActive !== hasActiveLinks) {
|
---|
137 | this.isActive = hasActiveLinks;
|
---|
138 | this.cdr.markForCheck();
|
---|
139 | this.classes.forEach((c) => {
|
---|
140 | if (hasActiveLinks) {
|
---|
141 | this.renderer.addClass(this.element.nativeElement, c);
|
---|
142 | }
|
---|
143 | else {
|
---|
144 | this.renderer.removeClass(this.element.nativeElement, c);
|
---|
145 | }
|
---|
146 | });
|
---|
147 | }
|
---|
148 | });
|
---|
149 | }
|
---|
150 | isLinkActive(router) {
|
---|
151 | const options = isActiveMatchOptions(this.routerLinkActiveOptions) ?
|
---|
152 | this.routerLinkActiveOptions :
|
---|
153 | // While the types should disallow `undefined` here, it's possible without strict inputs
|
---|
154 | (this.routerLinkActiveOptions.exact || false);
|
---|
155 | return (link) => router.isActive(link.urlTree, options);
|
---|
156 | }
|
---|
157 | hasActiveLinks() {
|
---|
158 | const isActiveCheckFn = this.isLinkActive(this.router);
|
---|
159 | return this.link && isActiveCheckFn(this.link) ||
|
---|
160 | this.linkWithHref && isActiveCheckFn(this.linkWithHref) ||
|
---|
161 | this.links.some(isActiveCheckFn) || this.linksWithHrefs.some(isActiveCheckFn);
|
---|
162 | }
|
---|
163 | }
|
---|
164 | RouterLinkActive.decorators = [
|
---|
165 | { type: Directive, args: [{
|
---|
166 | selector: '[routerLinkActive]',
|
---|
167 | exportAs: 'routerLinkActive',
|
---|
168 | },] }
|
---|
169 | ];
|
---|
170 | RouterLinkActive.ctorParameters = () => [
|
---|
171 | { type: Router },
|
---|
172 | { type: ElementRef },
|
---|
173 | { type: Renderer2 },
|
---|
174 | { type: ChangeDetectorRef },
|
---|
175 | { type: RouterLink, decorators: [{ type: Optional }] },
|
---|
176 | { type: RouterLinkWithHref, decorators: [{ type: Optional }] }
|
---|
177 | ];
|
---|
178 | RouterLinkActive.propDecorators = {
|
---|
179 | links: [{ type: ContentChildren, args: [RouterLink, { descendants: true },] }],
|
---|
180 | linksWithHrefs: [{ type: ContentChildren, args: [RouterLinkWithHref, { descendants: true },] }],
|
---|
181 | routerLinkActiveOptions: [{ type: Input }],
|
---|
182 | routerLinkActive: [{ type: Input }]
|
---|
183 | };
|
---|
184 | /**
|
---|
185 | * Use instead of `'paths' in options` to be compatible with property renaming
|
---|
186 | */
|
---|
187 | function isActiveMatchOptions(options) {
|
---|
188 | return !!options.paths;
|
---|
189 | }
|
---|
190 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"router_link_active.js","sourceRoot":"","sources":["../../../../../../../packages/router/src/directives/router_link_active.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,EAAmB,iBAAiB,EAAE,eAAe,EAAE,SAAS,EAAE,UAAU,EAAE,KAAK,EAAwB,QAAQ,EAAE,SAAS,EAAE,SAAS,EAAgB,MAAM,eAAe,CAAC;AACtL,OAAO,EAAC,IAAI,EAAE,EAAE,EAAe,MAAM,MAAM,CAAC;AAC5C,OAAO,EAAC,QAAQ,EAAC,MAAM,gBAAgB,CAAC;AAExC,OAAO,EAAQ,aAAa,EAAC,MAAM,WAAW,CAAC;AAC/C,OAAO,EAAC,MAAM,EAAC,MAAM,WAAW,CAAC;AAGjC,OAAO,EAAC,UAAU,EAAE,kBAAkB,EAAC,MAAM,eAAe,CAAC;AAG7D;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA0DG;AAKH,MAAM,OAAO,gBAAgB;IAoB3B,YACY,MAAc,EAAU,OAAmB,EAAU,QAAmB,EAC/D,GAAsB,EAAsB,IAAiB,EAC1D,YAAiC;QAF7C,WAAM,GAAN,MAAM,CAAQ;QAAU,YAAO,GAAP,OAAO,CAAY;QAAU,aAAQ,GAAR,QAAQ,CAAW;QAC/D,QAAG,GAAH,GAAG,CAAmB;QAAsB,SAAI,GAAJ,IAAI,CAAa;QAC1D,iBAAY,GAAZ,YAAY,CAAqB;QAlBjD,YAAO,GAAa,EAAE,CAAC;QAGf,aAAQ,GAAY,KAAK,CAAC;QAE1C;;;;;;WAMG;QACM,4BAAuB,GAA0C,EAAC,KAAK,EAAE,KAAK,EAAC,CAAC;QAOvF,IAAI,CAAC,wBAAwB,GAAG,MAAM,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,CAAQ,EAAE,EAAE;YACnE,IAAI,CAAC,YAAY,aAAa,EAAE;gBAC9B,IAAI,CAAC,MAAM,EAAE,CAAC;aACf;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED,aAAa;IACb,kBAAkB;QAChB,6FAA6F;QAC7F,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,EAAE,IAAI,CAAC,cAAc,CAAC,OAAO,EAAE,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE;YAC3F,IAAI,CAAC,MAAM,EAAE,CAAC;YACd,IAAI,CAAC,4BAA4B,EAAE,CAAC;QACtC,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,4BAA4B;;QAClC,MAAA,IAAI,CAAC,4BAA4B,0CAAE,WAAW,EAAE,CAAC;QACjD,MAAM,cAAc,GAChB,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,EAAE,EAAE,GAAG,IAAI,CAAC,cAAc,CAAC,OAAO,EAAE,EAAE,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,YAAY,CAAC;aACpF,MAAM,CAAC,CAAC,IAAI,EAAyC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC;aAC/D,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACrC,IAAI,CAAC,4BAA4B,GAAG,IAAI,CAAC,cAAc,CAAC,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE;YACzF,IAAI,IAAI,CAAC,QAAQ,KAAK,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,EAAE;gBAC1D,IAAI,CAAC,MAAM,EAAE,CAAC;aACf;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED,IACI,gBAAgB,CAAC,IAAqB;QACxC,MAAM,OAAO,GAAG,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAC7D,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IAC1C,CAAC;IAED,aAAa;IACb,WAAW,CAAC,OAAsB;QAChC,IAAI,CAAC,MAAM,EAAE,CAAC;IAChB,CAAC;IACD,aAAa;IACb,WAAW;;QACT,IAAI,CAAC,wBAAwB,CAAC,WAAW,EAAE,CAAC;QAC5C,MAAA,IAAI,CAAC,4BAA4B,0CAAE,WAAW,EAAE,CAAC;IACnD,CAAC;IAEO,MAAM;QACZ,IAAI,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,cAAc,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,SAAS;YAAE,OAAO;QAC1E,OAAO,CAAC,OAAO,EAAE,CAAC,IAAI,CAAC,GAAG,EAAE;YAC1B,MAAM,cAAc,GAAG,IAAI,CAAC,cAAc,EAAE,CAAC;YAC7C,IAAI,IAAI,CAAC,QAAQ,KAAK,cAAc,EAAE;gBACnC,IAAY,CAAC,QAAQ,GAAG,cAAc,CAAC;gBACxC,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;gBACxB,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;oBACzB,IAAI,cAAc,EAAE;wBAClB,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,aAAa,EAAE,CAAC,CAAC,CAAC;qBACvD;yBAAM;wBACL,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,OAAO,CAAC,aAAa,EAAE,CAAC,CAAC,CAAC;qBAC1D;gBACH,CAAC,CAAC,CAAC;aACJ;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,YAAY,CAAC,MAAc;QACjC,MAAM,OAAO,GACT,oBAAoB,CAAC,IAAI,CAAC,uBAAuB,CAAC,CAAC,CAAC;YACpD,IAAI,CAAC,uBAAuB,CAAC,CAAC;YAC9B,wFAAwF;YACxF,CAAC,IAAI,CAAC,uBAAuB,CAAC,KAAK,IAAI,KAAK,CAAC,CAAC;QAClD,OAAO,CAAC,IAAmC,EAAE,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;IACzF,CAAC;IAEO,cAAc;QACpB,MAAM,eAAe,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACvD,OAAO,IAAI,CAAC,IAAI,IAAI,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC;YAC1C,IAAI,CAAC,YAAY,IAAI,eAAe,CAAC,IAAI,CAAC,YAAY,CAAC;YACvD,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,eAAe,CAAC,IAAI,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;IACpF,CAAC;;;YAzGF,SAAS,SAAC;gBACT,QAAQ,EAAE,oBAAoB;gBAC9B,QAAQ,EAAE,kBAAkB;aAC7B;;;YApEO,MAAM;YAL2D,UAAU;YAAoD,SAAS;YAAtH,iBAAiB;YAQnC,UAAU,uBAwF8B,QAAQ;YAxFpC,kBAAkB,uBAyF/B,QAAQ;;;oBAtBZ,eAAe,SAAC,UAAU,EAAE,EAAC,WAAW,EAAE,IAAI,EAAC;6BAC/C,eAAe,SAAC,kBAAkB,EAAE,EAAC,WAAW,EAAE,IAAI,EAAC;sCAevD,KAAK;+BAoCL,KAAK;;AAmDR;;GAEG;AACH,SAAS,oBAAoB,CAAC,OACoB;IAChD,OAAO,CAAC,CAAE,OAAgC,CAAC,KAAK,CAAC;AACnD,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 {AfterContentInit, ChangeDetectorRef, ContentChildren, Directive, ElementRef, Input, OnChanges, OnDestroy, Optional, QueryList, Renderer2, SimpleChanges} from '@angular/core';\nimport {from, of, Subscription} from 'rxjs';\nimport {mergeAll} from 'rxjs/operators';\n\nimport {Event, NavigationEnd} from '../events';\nimport {Router} from '../router';\nimport {IsActiveMatchOptions} from '../url_tree';\n\nimport {RouterLink, RouterLinkWithHref} from './router_link';\n\n\n/**\n *\n * @description\n *\n * Tracks whether the linked route of an element is currently active, and allows you\n * to specify one or more CSS classes to add to the element when the linked route\n * is active.\n *\n * Use this directive to create a visual distinction for elements associated with an active route.\n * For example, the following code highlights the word \"Bob\" when the router\n * activates the associated route:\n *\n * ```\n * <a routerLink=\"/user/bob\" routerLinkActive=\"active-link\">Bob</a>\n * ```\n *\n * Whenever the URL is either '/user' or '/user/bob', the \"active-link\" class is\n * added to the anchor tag. If the URL changes, the class is removed.\n *\n * You can set more than one class using a space-separated string or an array.\n * For example:\n *\n * ```\n * <a routerLink=\"/user/bob\" routerLinkActive=\"class1 class2\">Bob</a>\n * <a routerLink=\"/user/bob\" [routerLinkActive]=\"['class1', 'class2']\">Bob</a>\n * ```\n *\n * To add the classes only when the URL matches the link exactly, add the option `exact: true`:\n *\n * ```\n * <a routerLink=\"/user/bob\" routerLinkActive=\"active-link\" [routerLinkActiveOptions]=\"{exact:\n * true}\">Bob</a>\n * ```\n *\n * To directly check the `isActive` status of the link, assign the `RouterLinkActive`\n * instance to a template variable.\n * For example, the following checks the status without assigning any CSS classes:\n *\n * ```\n * <a routerLink=\"/user/bob\" routerLinkActive #rla=\"routerLinkActive\">\n *   Bob {{ rla.isActive ? '(already open)' : ''}}\n * </a>\n * ```\n *\n * You can apply the `RouterLinkActive` directive to an ancestor of linked elements.\n * For example, the following sets the active-link class on the `<div>`  parent tag\n * when the URL is either '/user/jim' or '/user/bob'.\n *\n * ```\n * <div routerLinkActive=\"active-link\" [routerLinkActiveOptions]=\"{exact: true}\">\n *   <a routerLink=\"/user/jim\">Jim</a>\n *   <a routerLink=\"/user/bob\">Bob</a>\n * </div>\n * ```\n *\n * @ngModule RouterModule\n *\n * @publicApi\n */\n@Directive({\n  selector: '[routerLinkActive]',\n  exportAs: 'routerLinkActive',\n})\nexport class RouterLinkActive implements OnChanges, OnDestroy, AfterContentInit {\n  @ContentChildren(RouterLink, {descendants: true}) links!: QueryList<RouterLink>;\n  @ContentChildren(RouterLinkWithHref, {descendants: true})\n  linksWithHrefs!: QueryList<RouterLinkWithHref>;\n\n  private classes: string[] = [];\n  private routerEventsSubscription: Subscription;\n  private linkInputChangesSubscription?: Subscription;\n  public readonly isActive: boolean = false;\n\n  /**\n   * Options to configure how to determine if the router link is active.\n   *\n   * These options are passed to the `Router.isActive()` function.\n   *\n   * @see Router.isActive\n   */\n  @Input() routerLinkActiveOptions: {exact: boolean}|IsActiveMatchOptions = {exact: false};\n\n\n  constructor(\n      private router: Router, private element: ElementRef, private renderer: Renderer2,\n      private readonly cdr: ChangeDetectorRef, @Optional() private link?: RouterLink,\n      @Optional() private linkWithHref?: RouterLinkWithHref) {\n    this.routerEventsSubscription = router.events.subscribe((s: Event) => {\n      if (s instanceof NavigationEnd) {\n        this.update();\n      }\n    });\n  }\n\n  /** @nodoc */\n  ngAfterContentInit(): void {\n    // `of(null)` is used to force subscribe body to execute once immediately (like `startWith`).\n    of(this.links.changes, this.linksWithHrefs.changes, of(null)).pipe(mergeAll()).subscribe(_ => {\n      this.update();\n      this.subscribeToEachLinkOnChanges();\n    });\n  }\n\n  private subscribeToEachLinkOnChanges() {\n    this.linkInputChangesSubscription?.unsubscribe();\n    const allLinkChanges =\n        [...this.links.toArray(), ...this.linksWithHrefs.toArray(), this.link, this.linkWithHref]\n            .filter((link): link is RouterLink|RouterLinkWithHref => !!link)\n            .map(link => link.onChanges);\n    this.linkInputChangesSubscription = from(allLinkChanges).pipe(mergeAll()).subscribe(link => {\n      if (this.isActive !== this.isLinkActive(this.router)(link)) {\n        this.update();\n      }\n    });\n  }\n\n  @Input()\n  set routerLinkActive(data: string[]|string) {\n    const classes = Array.isArray(data) ? data : data.split(' ');\n    this.classes = classes.filter(c => !!c);\n  }\n\n  /** @nodoc */\n  ngOnChanges(changes: SimpleChanges): void {\n    this.update();\n  }\n  /** @nodoc */\n  ngOnDestroy(): void {\n    this.routerEventsSubscription.unsubscribe();\n    this.linkInputChangesSubscription?.unsubscribe();\n  }\n\n  private update(): void {\n    if (!this.links || !this.linksWithHrefs || !this.router.navigated) return;\n    Promise.resolve().then(() => {\n      const hasActiveLinks = this.hasActiveLinks();\n      if (this.isActive !== hasActiveLinks) {\n        (this as any).isActive = hasActiveLinks;\n        this.cdr.markForCheck();\n        this.classes.forEach((c) => {\n          if (hasActiveLinks) {\n            this.renderer.addClass(this.element.nativeElement, c);\n          } else {\n            this.renderer.removeClass(this.element.nativeElement, c);\n          }\n        });\n      }\n    });\n  }\n\n  private isLinkActive(router: Router): (link: (RouterLink|RouterLinkWithHref)) => boolean {\n    const options: boolean|IsActiveMatchOptions =\n        isActiveMatchOptions(this.routerLinkActiveOptions) ?\n        this.routerLinkActiveOptions :\n        // While the types should disallow `undefined` here, it's possible without strict inputs\n        (this.routerLinkActiveOptions.exact || false);\n    return (link: RouterLink|RouterLinkWithHref) => router.isActive(link.urlTree, options);\n  }\n\n  private hasActiveLinks(): boolean {\n    const isActiveCheckFn = this.isLinkActive(this.router);\n    return this.link && isActiveCheckFn(this.link) ||\n        this.linkWithHref && isActiveCheckFn(this.linkWithHref) ||\n        this.links.some(isActiveCheckFn) || this.linksWithHrefs.some(isActiveCheckFn);\n  }\n}\n\n/**\n * Use instead of `'paths' in options` to be compatible with property renaming\n */\nfunction isActiveMatchOptions(options: {exact: boolean}|\n                              IsActiveMatchOptions): options is IsActiveMatchOptions {\n  return !!(options as IsActiveMatchOptions).paths;\n}\n"]} |
---|