UNPKG

gentics-ui-core

Version:

This is the common core framework for the Gentics CMS and Mesh UI, and other Angular applications.

259 lines 52 kB
import { ChangeDetectionStrategy, Component, ElementRef, EventEmitter, Input, Output, QueryList, ViewChild, ViewChildren, ChangeDetectorRef } from '@angular/core'; import { RouterLinkWithHref } from '@angular/router'; import { BehaviorSubject, Subscription, timer } from 'rxjs'; import { debounceTime } from 'rxjs/operators'; import { UserAgentRef } from '../modal/user-agent-ref'; import * as i0 from "@angular/core"; import * as i1 from "../modal/user-agent-ref"; import * as i2 from "@angular/common"; import * as i3 from "@angular/router"; import * as i4 from "../icon/icon.directive"; /** The width configured in the .ellipsis CSS class. */ const ELLIPSIS_WIDTH = 13; /** * A Breadcrumbs navigation component. * * ```html * <gtx-breadcrumbs></gtx-breadcrumbs> * ``` */ export class Breadcrumbs { constructor(changeDetector, elementRef, userAgent) { this.changeDetector = changeDetector; this.elementRef = elementRef; this.userAgent = userAgent; /** * Fires when a link is clicked */ this.linkClick = new EventEmitter(); /** * Fires when the expand button is clicked */ this.multilineExpandedChange = new EventEmitter(); this.isMultiline = false; this.isMultilineExpanded = false; this.isDisabled = false; this.isOverflowing = false; this.showArrow = false; this.subscriptions = new Subscription(); this.resizeEvents = new BehaviorSubject(null); this.preventClicksWhenDisabled = (ev) => { if (this.isDisabled) { let target = ev.target; if (target.tagName.toLowerCase() === 'a' && target.classList.contains('breadcrumb')) { ev.preventDefault(); ev.stopImmediatePropagation(); } } }; } /** * If true the first folder and all the folder names from the end of the breadcrumbs, which fit into one line are shown * and an ellipsis in between. */ get multiline() { return this.isMultiline; } set multiline(val) { this.isMultiline = val != undefined && val !== false; } /** * If true the breadcrumbs are always expanded */ get multilineExpanded() { return this.isMultilineExpanded; } set multilineExpanded(val) { this.isMultilineExpanded = val != undefined && val !== false; } /** * Controls whether the navigation is disabled. */ get disabled() { return this.isDisabled; } set disabled(val) { this.isDisabled = val != undefined && val !== false; } ngAfterViewInit() { let element = this.elementRef.nativeElement; if (element) { // Listen in the "capture" phase to prevent routerLinks when disabled element.firstElementChild.addEventListener('click', this.preventClicksWhenDisabled, true); element.style.setProperty('--collapsedColor', this.collapsedColor); } const timerSub = timer(500, 500) .subscribe(() => this.resizeEvents.next()); this.subscriptions.add(timerSub); this.setUpResizeSub(); this.preventDisabledRouterLinks(); this.routerLinkChildren.changes.subscribe(() => this.preventDisabledRouterLinks()); this.resizeEvents.next(null); } ngOnChanges(changes) { if (changes['links'] || changes['routerLinks']) { let allLinks = (this.links || []).concat(this.routerLinks || []); this.backLink = allLinks[allLinks.length - 2]; this.resizeEvents.next(null); } if (changes['multiline'] || changes['multilineExpanded']) { this.resizeEvents.next(null); } } ngOnDestroy() { let element = this.elementRef.nativeElement; element.firstElementChild.removeEventListener('click', this.preventClicksWhenDisabled, true); this.subscriptions.unsubscribe(); } onLinkClicked(link, event) { if (this.isDisabled) { event.preventDefault(); event.stopImmediatePropagation(); } else { this.linkClick.emit(link); } } toggleMultilineExpanded() { this.multilineExpanded = !this.multilineExpanded; this.multilineExpandedChange.emit(this.multilineExpanded); this.resizeEvents.next(null); this.changeDetector.markForCheck(); } setUpResizeSub() { let prevLinks; let prevRouterLinks; let prevIsExpanded; let prevNavWidth = -1; const resizeSub = this.resizeEvents .pipe(debounceTime(5)) .subscribe(() => { if (!this.lastPart || !this.navWrapper) { return; } // If neither the links, nor isMultilineExpanded, nor the navWrapper element's clientWidth has changed, we don't need to do anything. const currNavWidth = this.navWrapper.nativeElement.clientWidth; if (prevLinks === this.links && prevRouterLinks === this.routerLinks && prevIsExpanded === this.isMultilineExpanded && prevNavWidth === currNavWidth) { return; } prevLinks = this.links; prevRouterLinks = this.routerLinks; prevIsExpanded = this.isMultilineExpanded; prevNavWidth = currNavWidth; const elements = this.lastPart.nativeElement.querySelectorAll('a.breadcrumb'); if (elements.length > 0) { const firstOffsetBottom = elements[0].offsetTop + elements[0].offsetHeight; const lastOffsetBottom = elements[elements.length - 1].offsetTop + elements[elements.length - 1].offsetHeight; this.showArrow = firstOffsetBottom !== lastOffsetBottom; } else { this.showArrow = false; } this.shortenTexts(); this.changeDetector.markForCheck(); }); this.subscriptions.add(resizeSub); } shortenTexts() { const navWrapper = this.navWrapper.nativeElement; const lastPart = this.lastPart.nativeElement; const innerElements = lastPart.querySelectorAll('a.breadcrumb'); const defaultElements = this.getCuttableBreadcrumbsTexts(); this.isOverflowing = false; // Reset all elements to their default states. const offset = this.multilineExpanded ? 0 : 1; for (let i = 0; i < innerElements.length; i++) { const innerElement = innerElements[i]; innerElement.classList.remove('without'); innerElement.classList.remove('hidden'); innerElement.textContent = defaultElements[i + offset]; } if (this.multilineExpanded) { return; } for (let i = 0; i < innerElements.length; ++i) { const innerElement = innerElements[i]; while (lastPart.offsetLeft + lastPart.scrollWidth + ELLIPSIS_WIDTH > navWrapper.clientWidth) { this.isOverflowing = true; if (innerElement.textContent.length === 0) { innerElement.classList.add('hidden'); const nextInnerElement = innerElements[i + 1]; if (nextInnerElement) { nextInnerElement.classList.add('without'); } break; } else { innerElement.textContent = innerElement.textContent.substring(1); } } } } getCuttableBreadcrumbsTexts() { let defaultBreadcrumbs = []; if (this.links) { for (let i = 0; i < this.links.length; i++) { defaultBreadcrumbs.push(this.links[i].text); } } if (this.routerLinks) { for (let i = 0; i < this.routerLinks.length; i++) { defaultBreadcrumbs.push(this.routerLinks[i].text); } } return defaultBreadcrumbs; } onResize(event) { this.resizeEvents.next(null); } /** * Workaround/Hack for the native angular "RouterLink" having no way to disable navigation on click. */ preventDisabledRouterLinks() { const thisComponent = this; const createsCompileErrorIfRouterLinkAPIChanges = 'onClick'; for (const link of this.routerLinkChildren.filter(link => !link.hasOwnProperty('onClick'))) { const originalOnClick = link.onClick; link.onClick = function interceptedOnClick(...args) { if (thisComponent.isDisabled) { return true; } else { return originalOnClick.apply(this, args); } }; } } } /** @nocollapse */ Breadcrumbs.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: Breadcrumbs, deps: [{ token: i0.ChangeDetectorRef }, { token: i0.ElementRef }, { token: i1.UserAgentRef }], target: i0.ɵɵFactoryTarget.Component }); /** @nocollapse */ Breadcrumbs.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.8", type: Breadcrumbs, selector: "gtx-breadcrumbs", inputs: { links: "links", routerLinks: "routerLinks", collapsedColor: "collapsedColor", multiline: "multiline", multilineExpanded: "multilineExpanded", disabled: "disabled" }, outputs: { linkClick: "linkClick", multilineExpandedChange: "multilineExpandedChange" }, viewQueries: [{ propertyName: "navWrapper", first: true, predicate: ["navWrapper"], descendants: true }, { propertyName: "lastPart", first: true, predicate: ["lastPart"], descendants: true }, { propertyName: "routerLinkChildren", predicate: RouterLinkWithHref, descendants: true }], usesOnChanges: true, ngImport: i0, template: "<nav [class.disabled]=\"isDisabled\" [class.multiline]=\"multiline\">\n <div #navWrapper class=\"nav-wrapper\" [class.is-overflowing]=\"isOverflowing\" [class.multilineExpanded]=\"multilineExpanded\" [class.multiline]=\"multiline\" (window:resize)=\"onResize($event)\">\n <div class=\"inner-wrapper\">\n <a class=\"back-button\" *ngIf=\"backLink && backLink.route\"\n (click)=\"onLinkClicked(backLink, $event)\"\n [routerLink]=\"backLink.route\"\n [title]=\"backLink.text\"></a>\n <a class=\"back-button\" *ngIf=\"backLink && !backLink.route\"\n (click)=\"onLinkClicked(backLink, $event)\"\n [attr.href]=\"isDisabled ? null : backLink?.href\"\n [title]=\"backLink.text\"></a>\n\n <div class=\"other-content\">\n <ng-content></ng-content>\n </div>\n\n <ng-template [ngIf]=\"links\">\n <a *ngIf=\"!multilineExpanded && links[0]\" class=\"breadcrumb\"\n [attr.href]=\"isDisabled ? null : links[0].href\"\n (click)=\"onLinkClicked(links[0], $event)\"\n [title]=\"links[0].tooltip || links[0].text\"\n >{{ links[0].text }}</a>\n <div class=\"ellipsis\" *ngIf=\"isOverflowing && !multilineExpanded\" [class.multilineExpanded]=\"multilineExpanded\" (click)=\"toggleMultilineExpanded()\">\n ...\n </div>\n <div #lastPart class=\"lastPart\">\n <ng-container *ngFor=\"let link of links; let i = index\">\n <a *ngIf=\"!multilineExpanded && i > 0\"\n class=\"breadcrumb last\"\n [attr.href]=\"isDisabled ? null : link?.href\"\n (click)=\"onLinkClicked(link, $event)\"\n [title]=\"link.tooltip || link.text\"\n >{{ link.text }}</a>\n <a *ngIf=\"multilineExpanded\"\n class=\"breadcrumb last\"\n [attr.href]=\"isDisabled ? null : link?.href\"\n (click)=\"onLinkClicked(link, $event)\"\n [title]=\"link.tooltip || link.text\"\n >{{ link.text }}\n </a>\n </ng-container>\n <span *ngIf=\"multiline && multilineExpanded && !isOverflowing && showArrow\" class=\"back_arrow\" [class.multilineExpanded]=\"multilineExpanded\" (click)=\"toggleMultilineExpanded()\">\n <icon>arrow_back</icon>\n </span>\n </div>\n </ng-template>\n\n <ng-template [ngIf]=\"routerLinks\">\n <a *ngIf=\"!multilineExpanded && routerLinks[0]\" class=\"breadcrumb\"\n [routerLink]=\"routerLinks[0].route\"\n (click)=\"onLinkClicked(routerLinks[0], $event)\"\n [title]=\"routerLinks[0].tooltip || routerLinks[0].text\"\n >{{ routerLinks[0].text }}</a>\n <div class=\"ellipsis\" *ngIf=\"isOverflowing && !multilineExpanded\" [class.multilineExpanded]=\"multilineExpanded\" (click)=\"toggleMultilineExpanded()\">\n ...\n </div>\n <div #lastPart class=\"lastPart\">\n <ng-container *ngFor=\"let routerLink of routerLinks; let i = index\">\n <a *ngIf=\"!multilineExpanded && i > 0\"\n class=\"breadcrumb last\"\n [routerLink]=\"routerLink.route\"\n (click)=\"onLinkClicked(routerLink, $event)\"\n [title]=\"routerLink.tooltip || routerLink.text\"\n >{{ routerLink.text }}</a>\n <a *ngIf=\"multilineExpanded\"\n class=\"breadcrumb last\"\n [routerLink]=\"routerLink.route\"\n (click)=\"onLinkClicked(routerLink, $event)\"\n [title]=\"routerLink.tooltip || routerLink.text\"\n >{{ routerLink.text }}\n </a>\n </ng-container>\n <span *ngIf=\"multiline && multilineExpanded && !isOverflowing && showArrow\" class=\"back_arrow\" [class.multilineExpanded]=\"multilineExpanded\" (click)=\"toggleMultilineExpanded()\">\n <icon>arrow_back</icon>\n </span>\n </div>\n </ng-template>\n </div>\n </div>\n</nav>\n", directives: [{ type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.RouterLinkWithHref, selector: "a[routerLink],area[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "preserveFragment", "skipLocationChange", "replaceUrl", "state", "relativeTo", "routerLink"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i4.Icon, selector: "icon" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: Breadcrumbs, decorators: [{ type: Component, args: [{ selector: 'gtx-breadcrumbs', changeDetection: ChangeDetectionStrategy.OnPush, template: "<nav [class.disabled]=\"isDisabled\" [class.multiline]=\"multiline\">\n <div #navWrapper class=\"nav-wrapper\" [class.is-overflowing]=\"isOverflowing\" [class.multilineExpanded]=\"multilineExpanded\" [class.multiline]=\"multiline\" (window:resize)=\"onResize($event)\">\n <div class=\"inner-wrapper\">\n <a class=\"back-button\" *ngIf=\"backLink && backLink.route\"\n (click)=\"onLinkClicked(backLink, $event)\"\n [routerLink]=\"backLink.route\"\n [title]=\"backLink.text\"></a>\n <a class=\"back-button\" *ngIf=\"backLink && !backLink.route\"\n (click)=\"onLinkClicked(backLink, $event)\"\n [attr.href]=\"isDisabled ? null : backLink?.href\"\n [title]=\"backLink.text\"></a>\n\n <div class=\"other-content\">\n <ng-content></ng-content>\n </div>\n\n <ng-template [ngIf]=\"links\">\n <a *ngIf=\"!multilineExpanded && links[0]\" class=\"breadcrumb\"\n [attr.href]=\"isDisabled ? null : links[0].href\"\n (click)=\"onLinkClicked(links[0], $event)\"\n [title]=\"links[0].tooltip || links[0].text\"\n >{{ links[0].text }}</a>\n <div class=\"ellipsis\" *ngIf=\"isOverflowing && !multilineExpanded\" [class.multilineExpanded]=\"multilineExpanded\" (click)=\"toggleMultilineExpanded()\">\n ...\n </div>\n <div #lastPart class=\"lastPart\">\n <ng-container *ngFor=\"let link of links; let i = index\">\n <a *ngIf=\"!multilineExpanded && i > 0\"\n class=\"breadcrumb last\"\n [attr.href]=\"isDisabled ? null : link?.href\"\n (click)=\"onLinkClicked(link, $event)\"\n [title]=\"link.tooltip || link.text\"\n >{{ link.text }}</a>\n <a *ngIf=\"multilineExpanded\"\n class=\"breadcrumb last\"\n [attr.href]=\"isDisabled ? null : link?.href\"\n (click)=\"onLinkClicked(link, $event)\"\n [title]=\"link.tooltip || link.text\"\n >{{ link.text }}\n </a>\n </ng-container>\n <span *ngIf=\"multiline && multilineExpanded && !isOverflowing && showArrow\" class=\"back_arrow\" [class.multilineExpanded]=\"multilineExpanded\" (click)=\"toggleMultilineExpanded()\">\n <icon>arrow_back</icon>\n </span>\n </div>\n </ng-template>\n\n <ng-template [ngIf]=\"routerLinks\">\n <a *ngIf=\"!multilineExpanded && routerLinks[0]\" class=\"breadcrumb\"\n [routerLink]=\"routerLinks[0].route\"\n (click)=\"onLinkClicked(routerLinks[0], $event)\"\n [title]=\"routerLinks[0].tooltip || routerLinks[0].text\"\n >{{ routerLinks[0].text }}</a>\n <div class=\"ellipsis\" *ngIf=\"isOverflowing && !multilineExpanded\" [class.multilineExpanded]=\"multilineExpanded\" (click)=\"toggleMultilineExpanded()\">\n ...\n </div>\n <div #lastPart class=\"lastPart\">\n <ng-container *ngFor=\"let routerLink of routerLinks; let i = index\">\n <a *ngIf=\"!multilineExpanded && i > 0\"\n class=\"breadcrumb last\"\n [routerLink]=\"routerLink.route\"\n (click)=\"onLinkClicked(routerLink, $event)\"\n [title]=\"routerLink.tooltip || routerLink.text\"\n >{{ routerLink.text }}</a>\n <a *ngIf=\"multilineExpanded\"\n class=\"breadcrumb last\"\n [routerLink]=\"routerLink.route\"\n (click)=\"onLinkClicked(routerLink, $event)\"\n [title]=\"routerLink.tooltip || routerLink.text\"\n >{{ routerLink.text }}\n </a>\n </ng-container>\n <span *ngIf=\"multiline && multilineExpanded && !isOverflowing && showArrow\" class=\"back_arrow\" [class.multilineExpanded]=\"multilineExpanded\" (click)=\"toggleMultilineExpanded()\">\n <icon>arrow_back</icon>\n </span>\n </div>\n </ng-template>\n </div>\n </div>\n</nav>\n" }] }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i0.ElementRef }, { type: i1.UserAgentRef }]; }, propDecorators: { links: [{ type: Input }], routerLinks: [{ type: Input }], collapsedColor: [{ type: Input }], multiline: [{ type: Input }], multilineExpanded: [{ type: Input }], disabled: [{ type: Input }], linkClick: [{ type: Output }], multilineExpandedChange: [{ type: Output }], routerLinkChildren: [{ type: ViewChildren, args: [RouterLinkWithHref] }], navWrapper: [{ type: ViewChild, args: ['navWrapper'] }], lastPart: [{ type: ViewChild, args: ['lastPart'] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"breadcrumbs.component.js","sourceRoot":"","sources":["../../../../../src/components/breadcrumbs/breadcrumbs.component.ts","../../../../../src/components/breadcrumbs/breadcrumbs.tpl.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    Component,\n    ElementRef,\n    EventEmitter,\n    Input,\n    OnChanges,\n    OnDestroy,\n    Output,\n    QueryList,\n    SimpleChanges,\n    ViewChild,\n    ViewChildren,\n    ChangeDetectorRef,\n    AfterViewInit\n} from '@angular/core';\nimport {RouterLinkWithHref} from '@angular/router';\nimport {BehaviorSubject, Subscription, timer} from 'rxjs';\nimport {debounceTime} from 'rxjs/operators';\n\nimport {UserAgentRef} from '../modal/user-agent-ref';\n\nexport interface IBreadcrumbLink {\n    href?: string;\n    route?: any;\n    text: string;\n    tooltip?: string;\n    [key: string]: any;\n}\n\nexport interface IBreadcrumbRouterLink {\n    route: any[];\n    text: string;\n    tooltip?: string;\n    [key: string]: any;\n}\n\n/** The width configured in the .ellipsis CSS class. */\nconst ELLIPSIS_WIDTH = 13;\n\n/**\n * A Breadcrumbs navigation component.\n *\n * ```html\n * <gtx-breadcrumbs></gtx-breadcrumbs>\n * ```\n */\n@Component({\n    selector: 'gtx-breadcrumbs',\n    templateUrl: './breadcrumbs.tpl.html',\n    changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class Breadcrumbs implements OnChanges, OnDestroy, AfterViewInit {\n\n    /**\n     * A list of links to display\n     */\n    @Input() links: IBreadcrumbLink[];\n\n    /**\n     * A list of RouterLinks to display\n     */\n    @Input() routerLinks: IBreadcrumbRouterLink[];\n\n    /**\n     * A color that is used for collapsed state background.\n     */\n    @Input() collapsedColor: string;\n\n    /**\n     * If true the first folder and all the folder names from the end of the breadcrumbs, which fit into one line are shown\n     * and an ellipsis in between.\n     */\n    @Input() get multiline(): boolean {\n        return this.isMultiline;\n    }\n    set multiline(val: boolean) {\n        this.isMultiline = val != undefined && val !== false;\n    }\n\n    /**\n     * If true the breadcrumbs are always expanded\n     */\n    @Input() get multilineExpanded(): boolean {\n        return this.isMultilineExpanded;\n    }\n    set multilineExpanded(val: boolean) {\n        this.isMultilineExpanded = val != undefined && val !== false;\n    }\n\n    /**\n     * Controls whether the navigation is disabled.\n     */\n    @Input() get disabled(): boolean {\n        return this.isDisabled;\n    }\n    set disabled(val: boolean) {\n        this.isDisabled = val != undefined && val !== false;\n    }\n\n    /**\n     * Fires when a link is clicked\n     */\n    @Output() linkClick = new EventEmitter<IBreadcrumbLink | IBreadcrumbRouterLink>();\n\n    /**\n     * Fires when the expand button is clicked\n     */\n    @Output() multilineExpandedChange = new EventEmitter<boolean>();\n\n    isMultiline: boolean = false;\n    isMultilineExpanded: boolean = false;\n    isDisabled: boolean = false;\n    isOverflowing: boolean = false;\n\n    showArrow: boolean = false;\n\n    backLink: IBreadcrumbLink | IBreadcrumbRouterLink;\n    @ViewChildren(RouterLinkWithHref) routerLinkChildren: QueryList<RouterLinkWithHref>;\n\n    @ViewChild('navWrapper')\n    navWrapper: ElementRef;\n\n    @ViewChild('lastPart')\n    lastPart: ElementRef;\n\n    private subscriptions = new Subscription();\n    private resizeEvents = new BehaviorSubject<void>(null);\n\n    constructor(private changeDetector: ChangeDetectorRef,\n                private elementRef: ElementRef,\n                private userAgent: UserAgentRef) { }\n\n    ngAfterViewInit(): void {\n        let element: HTMLElement = this.elementRef.nativeElement;\n        if (element) {\n            // Listen in the \"capture\" phase to prevent routerLinks when disabled\n            element.firstElementChild.addEventListener('click', this.preventClicksWhenDisabled, true);\n            element.style.setProperty('--collapsedColor', this.collapsedColor);\n        }\n\n        const timerSub = timer(500, 500)\n            .subscribe(() => this.resizeEvents.next());\n        this.subscriptions.add(timerSub);\n        this.setUpResizeSub();\n\n        this.preventDisabledRouterLinks();\n        this.routerLinkChildren.changes.subscribe(() => this.preventDisabledRouterLinks());\n        this.resizeEvents.next(null);\n    }\n\n    ngOnChanges(changes: SimpleChanges): void {\n        if (changes['links'] || changes['routerLinks']) {\n            let allLinks = (this.links || []).concat(this.routerLinks || []);\n            this.backLink = allLinks[allLinks.length - 2];\n            this.resizeEvents.next(null);\n        }\n        if (changes['multiline'] || changes['multilineExpanded']) {\n            this.resizeEvents.next(null);\n        }\n    }\n\n    ngOnDestroy(): void {\n        let element: HTMLElement = this.elementRef.nativeElement;\n        element.firstElementChild.removeEventListener('click', this.preventClicksWhenDisabled, true);\n        this.subscriptions.unsubscribe();\n    }\n\n    onLinkClicked(link: IBreadcrumbLink | IBreadcrumbRouterLink, event: Event): void {\n        if (this.isDisabled) {\n            event.preventDefault();\n            event.stopImmediatePropagation();\n        } else {\n            this.linkClick.emit(link);\n        }\n    }\n\n    toggleMultilineExpanded(): void {\n        this.multilineExpanded = !this.multilineExpanded;\n        this.multilineExpandedChange.emit(this.multilineExpanded);\n        this.resizeEvents.next(null);\n        this.changeDetector.markForCheck();\n    }\n\n    private setUpResizeSub() {\n        let prevLinks: IBreadcrumbLink[];\n        let prevRouterLinks: IBreadcrumbRouterLink[];\n        let prevIsExpanded: boolean;\n        let prevNavWidth = -1;\n\n        const resizeSub = this.resizeEvents\n            .pipe(debounceTime(5))\n            .subscribe(() => {\n                if (!this.lastPart || !this.navWrapper) {\n                    return;\n                }\n                // If neither the links, nor isMultilineExpanded, nor the navWrapper element's clientWidth has changed, we don't need to do anything.\n                const currNavWidth = this.navWrapper.nativeElement.clientWidth;\n                if (prevLinks === this.links && prevRouterLinks === this.routerLinks && prevIsExpanded === this.isMultilineExpanded && prevNavWidth === currNavWidth) {\n                    return;\n                }\n                prevLinks = this.links;\n                prevRouterLinks = this.routerLinks;\n                prevIsExpanded = this.isMultilineExpanded;\n                prevNavWidth = currNavWidth;\n\n                const elements = this.lastPart.nativeElement.querySelectorAll('a.breadcrumb');\n                if (elements.length > 0) {\n                    const firstOffsetBottom = elements[0].offsetTop + elements[0].offsetHeight;\n                    const lastOffsetBottom = elements[elements.length - 1].offsetTop + elements[elements.length - 1].offsetHeight;\n                    this.showArrow = firstOffsetBottom !== lastOffsetBottom;\n                } else {\n                    this.showArrow = false;\n                }\n                this.shortenTexts();\n                this.changeDetector.markForCheck();\n            });\n\n        this.subscriptions.add(resizeSub);\n    }\n\n    private shortenTexts() {\n        const navWrapper = this.navWrapper.nativeElement as HTMLElement;\n        const lastPart = this.lastPart.nativeElement as HTMLElement;\n        const innerElements = lastPart.querySelectorAll('a.breadcrumb');\n        const defaultElements = this.getCuttableBreadcrumbsTexts();\n\n        this.isOverflowing = false;\n\n        // Reset all elements to their default states.\n        const offset = this.multilineExpanded ? 0 : 1;\n        for (let i = 0; i < innerElements.length; i++) {\n            const innerElement = innerElements[i];\n            innerElement.classList.remove('without');\n            innerElement.classList.remove('hidden');\n            innerElement.textContent = defaultElements[i + offset];\n        }\n\n        if (this.multilineExpanded) {\n            return;\n        }\n\n        for (let i = 0; i < innerElements.length; ++i) {\n            const innerElement = innerElements[i];\n            while (lastPart.offsetLeft + lastPart.scrollWidth + ELLIPSIS_WIDTH > navWrapper.clientWidth) {\n                this.isOverflowing = true;\n                if (innerElement.textContent.length === 0) {\n                    innerElement.classList.add('hidden');\n                    const nextInnerElement = innerElements[i + 1];\n                    if (nextInnerElement) {\n                        nextInnerElement.classList.add('without');\n                    }\n                    break;\n                } else {\n                    innerElement.textContent = innerElement.textContent.substring(1);\n                }\n            }\n        }\n    }\n\n    private getCuttableBreadcrumbsTexts(): string[] {\n        let defaultBreadcrumbs: string[] = [];\n        if (this.links) {\n            for (let i = 0; i < this.links.length; i++) {\n                defaultBreadcrumbs.push(this.links[i].text);\n            }\n        }\n        if (this.routerLinks) {\n            for (let i = 0; i < this.routerLinks.length; i++) {\n                defaultBreadcrumbs.push(this.routerLinks[i].text);\n            }\n        }\n        return defaultBreadcrumbs;\n    }\n\n    onResize(event: any): void {\n        this.resizeEvents.next(null);\n    }\n\n    private preventClicksWhenDisabled = (ev: Event): void => {\n        if (this.isDisabled) {\n            let target = ev.target as HTMLElement;\n            if (target.tagName.toLowerCase() === 'a' && target.classList.contains('breadcrumb')) {\n                ev.preventDefault();\n                ev.stopImmediatePropagation();\n            }\n        }\n    }\n\n    /**\n     * Workaround/Hack for the native angular \"RouterLink\" having no way to disable navigation on click.\n     */\n    private preventDisabledRouterLinks(): void {\n        const thisComponent = this;\n        const createsCompileErrorIfRouterLinkAPIChanges: keyof RouterLinkWithHref = 'onClick';\n\n        for (const link of this.routerLinkChildren.filter(link => !link.hasOwnProperty('onClick'))) {\n            const originalOnClick = link.onClick;\n            link.onClick = function interceptedOnClick(...args: any[]): boolean {\n                if (thisComponent.isDisabled) {\n                    return true;\n                } else {\n                    return originalOnClick.apply(this, args);\n                }\n            };\n        }\n    }\n}\n","<nav [class.disabled]=\"isDisabled\" [class.multiline]=\"multiline\">\n    <div #navWrapper class=\"nav-wrapper\" [class.is-overflowing]=\"isOverflowing\" [class.multilineExpanded]=\"multilineExpanded\" [class.multiline]=\"multiline\" (window:resize)=\"onResize($event)\">\n        <div class=\"inner-wrapper\">\n            <a class=\"back-button\" *ngIf=\"backLink && backLink.route\"\n                (click)=\"onLinkClicked(backLink, $event)\"\n                [routerLink]=\"backLink.route\"\n                [title]=\"backLink.text\"></a>\n            <a class=\"back-button\" *ngIf=\"backLink && !backLink.route\"\n                (click)=\"onLinkClicked(backLink, $event)\"\n                [attr.href]=\"isDisabled ? null : backLink?.href\"\n                [title]=\"backLink.text\"></a>\n\n            <div class=\"other-content\">\n                <ng-content></ng-content>\n            </div>\n\n            <ng-template [ngIf]=\"links\">\n                <a *ngIf=\"!multilineExpanded && links[0]\" class=\"breadcrumb\"\n                    [attr.href]=\"isDisabled ? null : links[0].href\"\n                    (click)=\"onLinkClicked(links[0], $event)\"\n                    [title]=\"links[0].tooltip || links[0].text\"\n                >{{ links[0].text }}</a>\n                <div class=\"ellipsis\" *ngIf=\"isOverflowing && !multilineExpanded\" [class.multilineExpanded]=\"multilineExpanded\" (click)=\"toggleMultilineExpanded()\">\n                    ...\n                </div>\n                <div #lastPart class=\"lastPart\">\n                    <ng-container *ngFor=\"let link of links; let i = index\">\n                        <a *ngIf=\"!multilineExpanded && i > 0\"\n                            class=\"breadcrumb last\"\n                            [attr.href]=\"isDisabled ? null : link?.href\"\n                            (click)=\"onLinkClicked(link, $event)\"\n                            [title]=\"link.tooltip || link.text\"\n                        >{{ link.text }}</a>\n                        <a *ngIf=\"multilineExpanded\"\n                            class=\"breadcrumb last\"\n                            [attr.href]=\"isDisabled ? null : link?.href\"\n                            (click)=\"onLinkClicked(link, $event)\"\n                            [title]=\"link.tooltip || link.text\"\n                        >{{ link.text }}\n                       </a>\n                    </ng-container>\n                    <span *ngIf=\"multiline && multilineExpanded && !isOverflowing && showArrow\" class=\"back_arrow\" [class.multilineExpanded]=\"multilineExpanded\" (click)=\"toggleMultilineExpanded()\">\n                        <icon>arrow_back</icon>\n                    </span>\n                </div>\n            </ng-template>\n\n            <ng-template [ngIf]=\"routerLinks\">\n                <a *ngIf=\"!multilineExpanded && routerLinks[0]\" class=\"breadcrumb\"\n                    [routerLink]=\"routerLinks[0].route\"\n                    (click)=\"onLinkClicked(routerLinks[0], $event)\"\n                    [title]=\"routerLinks[0].tooltip || routerLinks[0].text\"\n                >{{ routerLinks[0].text }}</a>\n                <div class=\"ellipsis\" *ngIf=\"isOverflowing && !multilineExpanded\" [class.multilineEx