UNPKG

@dbg-riskit/angular-view

Version:

91 lines 13.8 kB
import { ContentChildren, Directive, Input } from '@angular/core'; import { NavigationEnd, RouterLink, RouterLinkWithHref } from '@angular/router'; import { SubLinkDirective } from './sub.link.directive'; import * as i0 from "@angular/core"; import * as i1 from "@angular/router"; export class LinkActiveDirective { constructor(router, element, renderer, changeDetectorRef) { this.router = router; this.element = element; this.renderer = renderer; this.changeDetectorRef = changeDetectorRef; this.classes = []; this.isActive = false; this.routerLinkActiveOptions = { exact: false }; this.subscription = router.events.subscribe((s) => { if (s instanceof NavigationEnd) { this.update(); } }); } ngAfterContentInit() { this.links.changes.subscribe((_) => this.update()); this.linksWithHrefs.changes.subscribe((_) => this.update()); this.subLinks.changes.subscribe((_) => this.update()); this.update(); } set menuLinkActive(data) { const classes = Array.isArray(data) ? data : data.split(' '); this.classes = classes.filter((c) => !!c); } ngOnChanges(changes) { this.update(); } ngOnDestroy() { this.subscription.unsubscribe(); } update() { if (!this.links || !this.linksWithHrefs || !this.subLinks || !this.router.navigated) { return; } const hasActiveLinks = this.hasActiveLinks(); // react only when status has changed to prevent unnecessary dom updates if (this.isActive !== hasActiveLinks) { this.classes.forEach((c) => { if (hasActiveLinks) { this.renderer.addClass(this.element.nativeElement, c); } else { this.renderer.removeClass(this.element.nativeElement, c); } }); Promise.resolve(hasActiveLinks).then((active) => { this.isActive = active; this.changeDetectorRef.detectChanges(); }); } } isLinkActive(router) { return (link) => link.urlTree ? router.isActive(link.urlTree, this.routerLinkActiveOptions.exact) : false; } hasActiveLinks() { return this.links.some(this.isLinkActive(this.router)) || this.linksWithHrefs.some(this.isLinkActive(this.router)) || this.subLinks.some(this.isLinkActive(this.router)); } } LinkActiveDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.2.6", ngImport: i0, type: LinkActiveDirective, deps: [{ token: i1.Router }, { token: i0.ElementRef }, { token: i0.Renderer2 }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Directive }); LinkActiveDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.2.6", type: LinkActiveDirective, selector: "[riskLinkActive]", inputs: { routerLinkActiveOptions: ["riskLinkActiveOptions", "routerLinkActiveOptions"], menuLinkActive: ["riskLinkActive", "menuLinkActive"] }, queries: [{ propertyName: "links", predicate: RouterLink, descendants: true }, { propertyName: "linksWithHrefs", predicate: RouterLinkWithHref, descendants: true }, { propertyName: "subLinks", predicate: SubLinkDirective, descendants: true }], exportAs: ["riskLinkActive"], usesOnChanges: true, ngImport: i0 }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.2.6", ngImport: i0, type: LinkActiveDirective, decorators: [{ type: Directive, args: [{ selector: '[riskLinkActive]', exportAs: 'riskLinkActive' }] }], ctorParameters: function () { return [{ type: i1.Router }, { type: i0.ElementRef }, { type: i0.Renderer2 }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { links: [{ type: ContentChildren, args: [RouterLink, { descendants: true }] }], linksWithHrefs: [{ type: ContentChildren, args: [RouterLinkWithHref, { descendants: true }] }], subLinks: [{ type: ContentChildren, args: [SubLinkDirective, { descendants: true }] }], routerLinkActiveOptions: [{ type: Input, args: ['riskLinkActiveOptions'] }], menuLinkActive: [{ type: Input, args: ['riskLinkActive'] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"link.active.directive.js","sourceRoot":"","sources":["../../../../../pkg/dbg-riskit/angular-view/src/lib/link.active.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    AfterContentInit,\n    ChangeDetectorRef,\n    ContentChildren,\n    Directive,\n    ElementRef,\n    Input,\n    OnChanges,\n    OnDestroy,\n    QueryList,\n    Renderer2,\n    SimpleChanges\n} from '@angular/core';\nimport {NavigationEnd, Router, RouterLink, RouterLinkWithHref} from '@angular/router';\nimport {Subscription} from 'rxjs';\nimport {SubLinkDirective} from './sub.link.directive';\n\n@Directive({\n    selector: '[riskLinkActive]',\n    exportAs: 'riskLinkActive'\n})\nexport class LinkActiveDirective implements OnChanges, OnDestroy, AfterContentInit {\n\n    @ContentChildren(RouterLink, {descendants: true})\n    public links?: QueryList<RouterLink>;\n\n    @ContentChildren(RouterLinkWithHref, {descendants: true})\n    public linksWithHrefs?: QueryList<RouterLinkWithHref>;\n\n    @ContentChildren(SubLinkDirective, {descendants: true})\n    public subLinks?: QueryList<SubLinkDirective>;\n\n    private classes: string[] = [];\n    private readonly subscription: Subscription;\n    public readonly isActive: boolean = false;\n\n    @Input('riskLinkActiveOptions')\n    public routerLinkActiveOptions: { exact: boolean } = {exact: false};\n\n    public constructor(private readonly router: Router,\n                       private readonly element: ElementRef,\n                       private readonly renderer: Renderer2,\n                       private readonly changeDetectorRef: ChangeDetectorRef) {\n        this.subscription = router.events.subscribe((s) => {\n            if (s instanceof NavigationEnd) {\n                this.update();\n            }\n        });\n    }\n\n    public ngAfterContentInit(): void {\n        this.links!.changes.subscribe((_) => this.update());\n        this.linksWithHrefs!.changes.subscribe((_) => this.update());\n        this.subLinks!.changes.subscribe((_) => this.update());\n        this.update();\n    }\n\n    @Input('riskLinkActive')\n    public set menuLinkActive(data: string[] | string) {\n        const classes = Array.isArray(data) ? data : data.split(' ');\n        this.classes = classes.filter((c) => !!c);\n    }\n\n    public ngOnChanges(changes: SimpleChanges): void {\n        this.update();\n    }\n\n    public ngOnDestroy(): void {\n        this.subscription.unsubscribe();\n    }\n\n    private update(): void {\n        if (!this.links || !this.linksWithHrefs || !this.subLinks || !this.router.navigated) {\n            return;\n        }\n        const hasActiveLinks = this.hasActiveLinks();\n\n        // react only when status has changed to prevent unnecessary dom updates\n        if (this.isActive !== hasActiveLinks) {\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            Promise.resolve(hasActiveLinks).then((active) => {\n                (this as {\n                    isActive: boolean\n                }).isActive = active;\n                this.changeDetectorRef.detectChanges();\n            });\n        }\n    }\n\n    private isLinkActive(router: Router): (link: (SubLinkDirective | RouterLink | RouterLinkWithHref)) => boolean {\n        return (link: SubLinkDirective | RouterLink | RouterLinkWithHref) =>\n            link.urlTree ? router.isActive(link.urlTree, this.routerLinkActiveOptions.exact) : false;\n    }\n\n    private hasActiveLinks(): boolean {\n        return this.links!.some(this.isLinkActive(this.router))\n               || this.linksWithHrefs!.some(this.isLinkActive(this.router))\n               || this.subLinks!.some(this.isLinkActive(this.router));\n    }\n\n}\n"]}