UNPKG

@cisstech/nge

Version:

NG Essentials is a collection of libraries for Angular developers.

135 lines 19 kB
import { Directive, ElementRef, Input } from '@angular/core'; import { Scroll } from '@angular/router'; import * as i0 from "@angular/core"; import * as i1 from "@angular/router"; import * as i2 from "@angular/common"; export class NgeDocTocDirective { constructor(router, location, elementRef, activatedRoute) { this.router = router; this.location = location; this.elementRef = elementRef; this.activatedRoute = activatedRoute; this.subscriptions = []; this.observer = new MutationObserver(() => { this.observer?.disconnect(); this.build(); }); this.anchors = []; this.subscriptions.push(this.router.events.subscribe((event) => { if (event instanceof Scroll && event.anchor) { this.scroll(event.anchor); } })); } ngOnDestroy() { this.intersection?.disconnect(); this.subscriptions.forEach((s) => s.unsubscribe()); } ngOnChanges() { this.build(); } build() { this.clear(); if (!this.component) { return; } const componentNode = this.component.injector.get(ElementRef).nativeElement; const tocContainer = this.elementRef.nativeElement; const h2Nodes = Array.from(componentNode.children).filter((node) => { return node.tagName === 'H2' && node.parentNode?.isSameNode(componentNode); }); this.detectIntersection(); const ul = document.createElement('ul'); h2Nodes.forEach((h2) => { const id = this.dashify(h2.textContent || ''); const target = document.createElement('span'); target.id = id; h2.insertAdjacentElement('afterend', target); const li = document.createElement('li'); const anchor = document.createElement('a'); anchor.innerHTML = h2.innerHTML; // .substring(1) will remove the leading / (prevent errors when baseHref is defined in index.html) anchor.href = this.location.path().substring(1) + '#' + id; li.appendChild(anchor); ul.appendChild(li); h2.setAttribute('data-toc-id', id); li.setAttribute('data-toc-id', id); this.anchors.push(li); this.intersection?.observe(h2); }); tocContainer.appendChild(ul); const { fragment } = this.activatedRoute.snapshot; if (fragment) { this.scroll(fragment); } this.observer.observe(componentNode, { childList: true, subtree: true, }); } dashify(input) { return input .trim() .replace(/([a-z])([A-Z])/g, '$1-$2') .replace(/\W/g, (m) => (/[À-ž]/.test(m) ? m : '-')) .replace(/^-+|-+$/g, '') .replace(/-{2,}/g, (m) => '-') // Condense multiple consecutive dashes to one. .toLowerCase(); } detectIntersection() { const tocContainer = this.elementRef.nativeElement; const rect = tocContainer.getBoundingClientRect(); const bottom = -window.innerHeight + rect.y + 200; this.intersection?.disconnect(); this.intersection = new IntersectionObserver((entries) => { entries.forEach((entry) => { if (entry.isIntersecting) { this.anchors.forEach((anchor) => { anchor.classList.remove('active'); const a = anchor.getAttribute('data-toc-id'); const b = entry.target.getAttribute('data-toc-id'); if (a === b) { anchor.classList.add('active'); } }); } }); }, { // A BOX OF 200px STARTING AT THE POSITION OF THE TOC ELEMENT rootMargin: `0px 0px ${bottom}px 0px`, }); } clear() { const tocContainer = this.elementRef.nativeElement; tocContainer.innerHTML = ''; this.observer.disconnect(); this.intersection?.disconnect(); this.anchors = []; } scroll(query) { const targetElement = document.querySelector(`h2[data-toc-id="${query}"]`); if (!targetElement) { window.scrollTo(0, 0); } else if (!this.isInViewport(targetElement)) { targetElement.scrollIntoView(); } } isInViewport(elem) { const bounding = elem.getBoundingClientRect(); return (bounding.top >= 0 && bounding.left >= 0 && bounding.bottom <= (window.innerHeight || document.documentElement.clientHeight) && bounding.right <= (window.innerWidth || document.documentElement.clientWidth)); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.1", ngImport: i0, type: NgeDocTocDirective, deps: [{ token: i1.Router }, { token: i2.Location }, { token: i0.ElementRef }, { token: i1.ActivatedRoute }], target: i0.ɵɵFactoryTarget.Directive }); } static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.1", type: NgeDocTocDirective, selector: "[ngeDocToc]", inputs: { component: ["ngeDocToc", "component"] }, usesOnChanges: true, ngImport: i0 }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.1", ngImport: i0, type: NgeDocTocDirective, decorators: [{ type: Directive, args: [{ selector: '[ngeDocToc]' }] }], ctorParameters: () => [{ type: i1.Router }, { type: i2.Location }, { type: i0.ElementRef }, { type: i1.ActivatedRoute }], propDecorators: { component: [{ type: Input, args: ['ngeDocToc'] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"toc.directive.js","sourceRoot":"","sources":["../../../../../../projects/nge/doc/src/renderer/toc.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Location } from '@angular/common'\nimport { ComponentRef, Directive, ElementRef, Input, OnChanges, OnDestroy } from '@angular/core'\nimport { ActivatedRoute, Router, Scroll } from '@angular/router'\nimport { Subscription } from 'rxjs'\n\n@Directive({ selector: '[ngeDocToc]' })\nexport class NgeDocTocDirective implements OnDestroy, OnChanges {\n  private readonly subscriptions: Subscription[] = []\n  private readonly observer = new MutationObserver(() => {\n    this.observer?.disconnect()\n    this.build()\n  })\n\n  private intersection?: IntersectionObserver\n  private anchors: HTMLElement[] = []\n\n  @Input('ngeDocToc')\n  component?: ComponentRef<any>\n\n  constructor(\n    private readonly router: Router,\n    private readonly location: Location,\n    private readonly elementRef: ElementRef<HTMLElement>,\n    private readonly activatedRoute: ActivatedRoute\n  ) {\n    this.subscriptions.push(\n      this.router.events.subscribe((event) => {\n        if (event instanceof Scroll && event.anchor) {\n          this.scroll(event.anchor)\n        }\n      })\n    )\n  }\n\n  ngOnDestroy(): void {\n    this.intersection?.disconnect()\n    this.subscriptions.forEach((s) => s.unsubscribe())\n  }\n\n  ngOnChanges(): void {\n    this.build()\n  }\n\n  private build(): void {\n    this.clear()\n\n    if (!this.component) {\n      return\n    }\n\n    const componentNode = this.component.injector.get(ElementRef).nativeElement as HTMLElement\n\n    const tocContainer = this.elementRef.nativeElement\n\n    const h2Nodes = Array.from(componentNode.children).filter((node) => {\n      return node.tagName === 'H2' && node.parentNode?.isSameNode(componentNode)\n    })\n\n    this.detectIntersection()\n\n    const ul = document.createElement('ul')\n    h2Nodes.forEach((h2) => {\n      const id = this.dashify(h2.textContent || '')\n      const target = document.createElement('span')\n      target.id = id\n      h2.insertAdjacentElement('afterend', target)\n\n      const li = document.createElement('li')\n      const anchor = document.createElement('a')\n      anchor.innerHTML = h2.innerHTML\n      // .substring(1) will remove the leading / (prevent errors when baseHref is defined in index.html)\n      anchor.href = this.location.path().substring(1) + '#' + id\n\n      li.appendChild(anchor)\n      ul.appendChild(li)\n\n      h2.setAttribute('data-toc-id', id)\n      li.setAttribute('data-toc-id', id)\n\n      this.anchors.push(li)\n\n      this.intersection?.observe(h2)\n    })\n\n    tocContainer.appendChild(ul)\n\n    const { fragment } = this.activatedRoute.snapshot\n    if (fragment) {\n      this.scroll(fragment)\n    }\n\n    this.observer.observe(componentNode, {\n      childList: true,\n      subtree: true,\n    })\n  }\n\n  private dashify(input: string): string {\n    return input\n      .trim()\n      .replace(/([a-z])([A-Z])/g, '$1-$2')\n      .replace(/\\W/g, (m) => (/[À-ž]/.test(m) ? m : '-'))\n      .replace(/^-+|-+$/g, '')\n      .replace(/-{2,}/g, (m) => '-') // Condense multiple consecutive dashes to one.\n      .toLowerCase()\n  }\n\n  private detectIntersection(): void {\n    const tocContainer = this.elementRef.nativeElement\n    const rect = tocContainer.getBoundingClientRect()\n    const bottom = -window.innerHeight + rect.y + 200\n\n    this.intersection?.disconnect()\n    this.intersection = new IntersectionObserver(\n      (entries) => {\n        entries.forEach((entry) => {\n          if (entry.isIntersecting) {\n            this.anchors.forEach((anchor) => {\n              anchor.classList.remove('active')\n              const a = anchor.getAttribute('data-toc-id')\n              const b = entry.target.getAttribute('data-toc-id')\n              if (a === b) {\n                anchor.classList.add('active')\n              }\n            })\n          }\n        })\n      },\n      {\n        // A BOX OF 200px STARTING AT THE POSITION OF THE TOC ELEMENT\n        rootMargin: `0px 0px ${bottom}px 0px`,\n      }\n    )\n  }\n\n  private clear(): void {\n    const tocContainer = this.elementRef.nativeElement\n    tocContainer.innerHTML = ''\n    this.observer.disconnect()\n    this.intersection?.disconnect()\n    this.anchors = []\n  }\n\n  private scroll(query: string): void {\n    const targetElement = document.querySelector(`h2[data-toc-id=\"${query}\"]`)\n    if (!targetElement) {\n      window.scrollTo(0, 0)\n    } else if (!this.isInViewport(targetElement)) {\n      targetElement.scrollIntoView()\n    }\n  }\n\n  private isInViewport(elem: any): boolean {\n    const bounding = elem.getBoundingClientRect()\n    return (\n      bounding.top >= 0 &&\n      bounding.left >= 0 &&\n      bounding.bottom <= (window.innerHeight || document.documentElement.clientHeight) &&\n      bounding.right <= (window.innerWidth || document.documentElement.clientWidth)\n    )\n  }\n}\n"]}