@ng-bootstrap/ng-bootstrap
Version:
Angular powered Bootstrap
301 lines • 32 kB
JavaScript
import { ChangeDetectorRef, ContentChildren, DestroyRef, Directive, ElementRef, inject, Input, Output, } from '@angular/core';
import { NgbScrollSpyService } from './scrollspy.service';
import { isString } from '../util/util';
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
import * as i0 from "@angular/core";
/**
* A helper directive to that links menu items and fragments together.
*
* It will automatically add the `.active` class to the menu item when the associated fragment becomes active.
*
* @since 15.1.0
*/
class NgbScrollSpyItem {
constructor() {
this._changeDetector = inject(ChangeDetectorRef);
this._scrollSpyMenu = inject(NgbScrollSpyMenu, { optional: true });
this._scrollSpyAPI = this._scrollSpyMenu ?? inject(NgbScrollSpyService);
this._destroyRef = inject(DestroyRef);
this._isActive = false;
}
/**
* References the scroll spy directive, the id of the associated fragment and the parent menu item.
*
* Can be used like:
* - `ngbScrollSpyItem="fragmentId"`
* - `[ngbScrollSpyItem]="scrollSpy" fragment="fragmentId"
* - `[ngbScrollSpyItem]="[scrollSpy, 'fragmentId']"` parent="parentId"`
* - `[ngbScrollSpyItem]="[scrollSpy, 'fragmentId', 'parentId']"`
*
* As well as together with `[fragment]` and `[parent]` inputs.
*/
set data(data) {
if (Array.isArray(data)) {
this._scrollSpyAPI = data[0];
this.fragment = data[1];
this.parent ??= data[2];
}
else if (data instanceof NgbScrollSpy) {
this._scrollSpyAPI = data;
}
else if (isString(data)) {
this.fragment = data;
}
}
ngOnInit() {
// if it is not a part of a bigger menu, it should handle activation itself
if (!this._scrollSpyMenu) {
this._scrollSpyAPI.active$.pipe(takeUntilDestroyed(this._destroyRef)).subscribe((active) => {
if (active === this.fragment) {
this._activate();
}
else {
this._deactivate();
}
this._changeDetector.markForCheck();
});
}
}
/**
* @internal
*/
_activate() {
this._isActive = true;
if (this._scrollSpyMenu) {
this._scrollSpyMenu.getItem(this.parent ?? '')?._activate();
}
}
/**
* @internal
*/
_deactivate() {
this._isActive = false;
if (this._scrollSpyMenu) {
this._scrollSpyMenu.getItem(this.parent ?? '')?._deactivate();
}
}
/**
* Returns `true`, if the associated fragment is active.
*/
isActive() {
return this._isActive;
}
/**
* Scrolls to the associated fragment.
*/
scrollTo(options) {
this._scrollSpyAPI.scrollTo(this.fragment, options);
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.6", ngImport: i0, type: NgbScrollSpyItem, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.0.6", type: NgbScrollSpyItem, isStandalone: true, selector: "[ngbScrollSpyItem]", inputs: { data: ["ngbScrollSpyItem", "data"], fragment: "fragment", parent: "parent" }, host: { listeners: { "click": "scrollTo();" }, properties: { "class.active": "isActive()" } }, exportAs: ["ngbScrollSpyItem"], ngImport: i0 }); }
}
export { NgbScrollSpyItem };
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.6", ngImport: i0, type: NgbScrollSpyItem, decorators: [{
type: Directive,
args: [{
selector: '[ngbScrollSpyItem]',
standalone: true,
exportAs: 'ngbScrollSpyItem',
host: {
'[class.active]': 'isActive()',
'(click)': 'scrollTo();',
},
}]
}], propDecorators: { data: [{
type: Input,
args: ['ngbScrollSpyItem']
}], fragment: [{
type: Input
}], parent: [{
type: Input
}] } });
/**
* An optional scroll spy menu directive to build hierarchical menus
* and simplify the [`NgbScrollSpyItem`](#/components/scrollspy/api#NgbScrollSpyItem) configuration.
*
* @since 15.1.0
*/
class NgbScrollSpyMenu {
constructor() {
this._scrollSpyRef = inject(NgbScrollSpyService);
this._destroyRef = inject(DestroyRef);
this._map = new Map();
this._lastActiveItem = null;
}
set scrollSpy(scrollSpy) {
this._scrollSpyRef = scrollSpy;
}
get active() {
return this._scrollSpyRef.active;
}
get active$() {
return this._scrollSpyRef.active$;
}
scrollTo(fragment, options) {
this._scrollSpyRef.scrollTo(fragment, options);
}
getItem(id) {
return this._map.get(id);
}
ngAfterViewInit() {
this._items.changes.pipe(takeUntilDestroyed(this._destroyRef)).subscribe(() => this._rebuildMap());
this._rebuildMap();
this._scrollSpyRef.active$.pipe(takeUntilDestroyed(this._destroyRef)).subscribe((activeId) => {
this._lastActiveItem?._deactivate();
const item = this._map.get(activeId);
if (item) {
item._activate();
this._lastActiveItem = item;
}
});
}
_rebuildMap() {
this._map.clear();
for (let item of this._items) {
this._map.set(item.fragment, item);
}
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.6", ngImport: i0, type: NgbScrollSpyMenu, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.0.6", type: NgbScrollSpyMenu, isStandalone: true, selector: "[ngbScrollSpyMenu]", inputs: { scrollSpy: ["ngbScrollSpyMenu", "scrollSpy"] }, queries: [{ propertyName: "_items", predicate: NgbScrollSpyItem, descendants: true }], ngImport: i0 }); }
}
export { NgbScrollSpyMenu };
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.6", ngImport: i0, type: NgbScrollSpyMenu, decorators: [{
type: Directive,
args: [{
selector: '[ngbScrollSpyMenu]',
standalone: true,
}]
}], propDecorators: { _items: [{
type: ContentChildren,
args: [NgbScrollSpyItem, { descendants: true }]
}], scrollSpy: [{
type: Input,
args: ['ngbScrollSpyMenu']
}] } });
/**
* A directive to put on a scrollable container.
*
* It will instantiate a [`NgbScrollSpyService`](#/components/scrollspy/api#NgbScrollSpyService).
*
* @since 15.1.0
*/
class NgbScrollSpy {
constructor() {
this._initialFragment = null;
this._service = inject(NgbScrollSpyService);
this._nativeElement = inject(ElementRef).nativeElement;
/**
* An event raised when the active section changes.
*
* Payload is the id of the new active section, empty string if none.
*/
this.activeChange = this._service.active$;
}
set active(fragment) {
this._initialFragment = fragment;
this.scrollTo(fragment);
}
/**
* Getter/setter for the currently active fragment id.
*/
get active() {
return this._service.active;
}
/**
* Returns an observable that emits currently active section id.
*/
get active$() {
return this._service.active$;
}
ngAfterViewInit() {
this._service.start({
processChanges: this.processChanges,
root: this._nativeElement,
rootMargin: this.rootMargin,
threshold: this.threshold,
...(this._initialFragment && { initialFragment: this._initialFragment }),
});
}
/**
* @internal
*/
_registerFragment(fragment) {
this._service.observe(fragment.id);
}
/**
* @internal
*/
_unregisterFragment(fragment) {
this._service.unobserve(fragment.id);
}
/**
* Scrolls to a fragment that is identified by the `ngbScrollSpyFragment` directive.
* An id or an element reference can be passed.
*/
scrollTo(fragment, options) {
this._service.scrollTo(fragment, {
...(this.scrollBehavior && { behavior: this.scrollBehavior }),
...options,
});
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.6", ngImport: i0, type: NgbScrollSpy, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.0.6", type: NgbScrollSpy, isStandalone: true, selector: "[ngbScrollSpy]", inputs: { processChanges: "processChanges", rootMargin: "rootMargin", scrollBehavior: "scrollBehavior", threshold: "threshold", active: "active" }, outputs: { activeChange: "activeChange" }, host: { attributes: { "tabindex": "0" }, styleAttribute: "overflow-y: auto" }, providers: [NgbScrollSpyService], exportAs: ["ngbScrollSpy"], ngImport: i0 }); }
}
export { NgbScrollSpy };
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.6", ngImport: i0, type: NgbScrollSpy, decorators: [{
type: Directive,
args: [{
selector: '[ngbScrollSpy]',
standalone: true,
exportAs: 'ngbScrollSpy',
host: {
tabindex: '0',
style: 'overflow-y: auto',
},
providers: [NgbScrollSpyService],
}]
}], propDecorators: { processChanges: [{
type: Input
}], rootMargin: [{
type: Input
}], scrollBehavior: [{
type: Input
}], threshold: [{
type: Input
}], active: [{
type: Input
}], activeChange: [{
type: Output
}] } });
/**
* A directive to put on a fragment observed inside a scrollspy container.
*
* @since 15.1.0
*/
class NgbScrollSpyFragment {
constructor() {
this._destroyRef = inject(DestroyRef);
this._scrollSpy = inject(NgbScrollSpy);
}
ngAfterViewInit() {
this._scrollSpy._registerFragment(this);
this._destroyRef.onDestroy(() => this._scrollSpy._unregisterFragment(this));
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.6", ngImport: i0, type: NgbScrollSpyFragment, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.0.6", type: NgbScrollSpyFragment, isStandalone: true, selector: "[ngbScrollSpyFragment]", inputs: { id: ["ngbScrollSpyFragment", "id"] }, host: { properties: { "id": "id" } }, ngImport: i0 }); }
}
export { NgbScrollSpyFragment };
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.6", ngImport: i0, type: NgbScrollSpyFragment, decorators: [{
type: Directive,
args: [{
selector: '[ngbScrollSpyFragment]',
standalone: true,
host: {
'[id]': 'id',
},
}]
}], propDecorators: { id: [{
type: Input,
args: ['ngbScrollSpyFragment']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scrollspy.js","sourceRoot":"","sources":["../../../../src/scrollspy/scrollspy.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n\tAfterViewInit,\n\tChangeDetectorRef,\n\tContentChildren,\n\tDestroyRef,\n\tDirective,\n\tElementRef,\n\tinject,\n\tInput,\n\tOnInit,\n\tOutput,\n\tQueryList,\n} from '@angular/core';\nimport { NgbScrollSpyProcessChanges, NgbScrollSpyService, NgbScrollToOptions } from './scrollspy.service';\nimport { Observable } from 'rxjs';\nimport { isString } from '../util/util';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\n\n/**\n * Common interface for the scroll spy API.\n *\n * @internal\n */\nexport interface NgbScrollSpyRef {\n\tget active(): string;\n\tget active$(): Observable<string>;\n\tscrollTo(fragment: string | HTMLElement, options?: NgbScrollToOptions): void;\n}\n\n/**\n * A helper directive to that links menu items and fragments together.\n *\n * It will automatically add the `.active` class to the menu item when the associated fragment becomes active.\n *\n * @since 15.1.0\n */\n@Directive({\n\tselector: '[ngbScrollSpyItem]',\n\tstandalone: true,\n\texportAs: 'ngbScrollSpyItem',\n\thost: {\n\t\t'[class.active]': 'isActive()',\n\t\t'(click)': 'scrollTo();',\n\t},\n})\nexport class NgbScrollSpyItem implements OnInit {\n\tprivate _changeDetector = inject(ChangeDetectorRef);\n\tprivate _scrollSpyMenu = inject(NgbScrollSpyMenu, { optional: true });\n\tprivate _scrollSpyAPI: NgbScrollSpyRef = this._scrollSpyMenu ?? inject(NgbScrollSpyService);\n\tprivate _destroyRef = inject(DestroyRef);\n\n\tprivate _isActive = false;\n\n\t/**\n\t * References the scroll spy directive, the id of the associated fragment and the parent menu item.\n\t *\n\t * Can be used like:\n\t *  - `ngbScrollSpyItem=\"fragmentId\"`\n\t *  - `[ngbScrollSpyItem]=\"scrollSpy\" fragment=\"fragmentId\"\n\t *  - `[ngbScrollSpyItem]=\"[scrollSpy, 'fragmentId']\"` parent=\"parentId\"`\n\t *  - `[ngbScrollSpyItem]=\"[scrollSpy, 'fragmentId', 'parentId']\"`\n\t *\n\t *  As well as together with `[fragment]` and `[parent]` inputs.\n\t */\n\t@Input('ngbScrollSpyItem') set data(data: NgbScrollSpy | string | [NgbScrollSpy, string, string?]) {\n\t\tif (Array.isArray(data)) {\n\t\t\tthis._scrollSpyAPI = data[0];\n\t\t\tthis.fragment = data[1];\n\t\t\tthis.parent ??= data[2];\n\t\t} else if (data instanceof NgbScrollSpy) {\n\t\t\tthis._scrollSpyAPI = data;\n\t\t} else if (isString(data)) {\n\t\t\tthis.fragment = data;\n\t\t}\n\t}\n\n\t/**\n\t * The id of the associated fragment.\n\t */\n\t@Input() fragment: string;\n\n\t/**\n\t * The id of the parent scroll spy menu item.\n\t */\n\t@Input() parent: string | undefined;\n\n\tngOnInit(): void {\n\t\t// if it is not a part of a bigger menu, it should handle activation itself\n\t\tif (!this._scrollSpyMenu) {\n\t\t\tthis._scrollSpyAPI.active$.pipe(takeUntilDestroyed(this._destroyRef)).subscribe((active: string) => {\n\t\t\t\tif (active === this.fragment) {\n\t\t\t\t\tthis._activate();\n\t\t\t\t} else {\n\t\t\t\t\tthis._deactivate();\n\t\t\t\t}\n\t\t\t\tthis._changeDetector.markForCheck();\n\t\t\t});\n\t\t}\n\t}\n\n\t/**\n\t * @internal\n\t */\n\t_activate(): void {\n\t\tthis._isActive = true;\n\t\tif (this._scrollSpyMenu) {\n\t\t\tthis._scrollSpyMenu.getItem(this.parent ?? '')?._activate();\n\t\t}\n\t}\n\n\t/**\n\t * @internal\n\t */\n\t_deactivate(): void {\n\t\tthis._isActive = false;\n\t\tif (this._scrollSpyMenu) {\n\t\t\tthis._scrollSpyMenu.getItem(this.parent ?? '')?._deactivate();\n\t\t}\n\t}\n\n\t/**\n\t * Returns `true`, if the associated fragment is active.\n\t */\n\tisActive(): boolean {\n\t\treturn this._isActive;\n\t}\n\n\t/**\n\t * Scrolls to the associated fragment.\n\t */\n\tscrollTo(options?: NgbScrollToOptions): void {\n\t\tthis._scrollSpyAPI.scrollTo(this.fragment, options);\n\t}\n}\n\n/**\n * An optional scroll spy menu directive to build hierarchical menus\n * and simplify the [`NgbScrollSpyItem`](#/components/scrollspy/api#NgbScrollSpyItem) configuration.\n *\n * @since 15.1.0\n */\n@Directive({\n\tselector: '[ngbScrollSpyMenu]',\n\tstandalone: true,\n})\nexport class NgbScrollSpyMenu implements NgbScrollSpyRef, AfterViewInit {\n\tprivate _scrollSpyRef: NgbScrollSpyRef = inject(NgbScrollSpyService);\n\tprivate _destroyRef = inject(DestroyRef);\n\tprivate _map = new Map<string, NgbScrollSpyItem>();\n\tprivate _lastActiveItem: NgbScrollSpyItem | null = null;\n\n\t@ContentChildren(NgbScrollSpyItem, { descendants: true }) private _items: QueryList<NgbScrollSpyItem>;\n\n\t@Input('ngbScrollSpyMenu') set scrollSpy(scrollSpy: NgbScrollSpy) {\n\t\tthis._scrollSpyRef = scrollSpy;\n\t}\n\n\tget active(): string {\n\t\treturn this._scrollSpyRef.active;\n\t}\n\tget active$(): Observable<string> {\n\t\treturn this._scrollSpyRef.active$;\n\t}\n\tscrollTo(fragment: string, options?: NgbScrollToOptions): void {\n\t\tthis._scrollSpyRef.scrollTo(fragment, options);\n\t}\n\n\tgetItem(id: string): NgbScrollSpyItem | undefined {\n\t\treturn this._map.get(id);\n\t}\n\n\tngAfterViewInit() {\n\t\tthis._items.changes.pipe(takeUntilDestroyed(this._destroyRef)).subscribe(() => this._rebuildMap());\n\t\tthis._rebuildMap();\n\n\t\tthis._scrollSpyRef.active$.pipe(takeUntilDestroyed(this._destroyRef)).subscribe((activeId) => {\n\t\t\tthis._lastActiveItem?._deactivate();\n\t\t\tconst item = this._map.get(activeId);\n\t\t\tif (item) {\n\t\t\t\titem._activate();\n\t\t\t\tthis._lastActiveItem = item;\n\t\t\t}\n\t\t});\n\t}\n\n\tprivate _rebuildMap() {\n\t\tthis._map.clear();\n\t\tfor (let item of this._items) {\n\t\t\tthis._map.set(item.fragment, item);\n\t\t}\n\t}\n}\n\n/**\n * A directive to put on a scrollable container.\n *\n * It will instantiate a [`NgbScrollSpyService`](#/components/scrollspy/api#NgbScrollSpyService).\n *\n * @since 15.1.0\n */\n@Directive({\n\tselector: '[ngbScrollSpy]',\n\tstandalone: true,\n\texportAs: 'ngbScrollSpy',\n\thost: {\n\t\ttabindex: '0',\n\t\tstyle: 'overflow-y: auto',\n\t},\n\tproviders: [NgbScrollSpyService],\n})\nexport class NgbScrollSpy implements NgbScrollSpyRef, AfterViewInit {\n\tstatic ngAcceptInputType_scrollBehavior: string;\n\n\tprivate _initialFragment: string | null = null;\n\tprivate _service = inject(NgbScrollSpyService);\n\tprivate _nativeElement = inject<ElementRef<HTMLElement>>(ElementRef).nativeElement;\n\n\t/**\n\t * A function that is called when the `IntersectionObserver` detects a change.\n\t *\n\t * See [`NgbScrollSpyOptions`](#/components/scrollspy/api#NgbScrollSpyOptions) for more details.\n\t */\n\t@Input() processChanges: NgbScrollSpyProcessChanges;\n\n\t/**\n\t * An `IntersectionObserver` root margin.\n\t */\n\t@Input() rootMargin: string;\n\n\t/**\n\t * The scroll behavior for the `.scrollTo()` method.\n\t */\n\t@Input() scrollBehavior: 'auto' | 'smooth';\n\n\t/**\n\t * An `IntersectionObserver` threshold.\n\t */\n\t@Input() threshold: number | number[];\n\n\t@Input() set active(fragment: string) {\n\t\tthis._initialFragment = fragment;\n\t\tthis.scrollTo(fragment);\n\t}\n\n\t/**\n\t * An event raised when the active section changes.\n\t *\n\t * Payload is the id of the new active section, empty string if none.\n\t */\n\t@Output() activeChange = this._service.active$;\n\n\t/**\n\t * Getter/setter for the currently active fragment id.\n\t */\n\tget active(): string {\n\t\treturn this._service.active;\n\t}\n\n\t/**\n\t * Returns an observable that emits currently active section id.\n\t */\n\tget active$(): Observable<string> {\n\t\treturn this._service.active$;\n\t}\n\n\tngAfterViewInit(): void {\n\t\tthis._service.start({\n\t\t\tprocessChanges: this.processChanges,\n\t\t\troot: this._nativeElement,\n\t\t\trootMargin: this.rootMargin,\n\t\t\tthreshold: this.threshold,\n\t\t\t...(this._initialFragment && { initialFragment: this._initialFragment }),\n\t\t});\n\t}\n\n\t/**\n\t * @internal\n\t */\n\t_registerFragment(fragment: NgbScrollSpyFragment): void {\n\t\tthis._service.observe(fragment.id);\n\t}\n\n\t/**\n\t * @internal\n\t */\n\t_unregisterFragment(fragment: NgbScrollSpyFragment): void {\n\t\tthis._service.unobserve(fragment.id);\n\t}\n\n\t/**\n\t * Scrolls to a fragment that is identified by the `ngbScrollSpyFragment` directive.\n\t * An id or an element reference can be passed.\n\t */\n\tscrollTo(fragment: string | HTMLElement, options?: NgbScrollToOptions): void {\n\t\tthis._service.scrollTo(fragment, {\n\t\t\t...(this.scrollBehavior && { behavior: this.scrollBehavior }),\n\t\t\t...options,\n\t\t});\n\t}\n}\n\n/**\n * A directive to put on a fragment observed inside a scrollspy container.\n *\n * @since 15.1.0\n */\n@Directive({\n\tselector: '[ngbScrollSpyFragment]',\n\tstandalone: true,\n\thost: {\n\t\t'[id]': 'id',\n\t},\n})\nexport class NgbScrollSpyFragment implements AfterViewInit {\n\tprivate _destroyRef = inject(DestroyRef);\n\tprivate _scrollSpy = inject(NgbScrollSpy);\n\n\t/**\n\t * The unique id of the fragment.\n\t * It must be a string unique to the document, as it will be set as the id of the element.\n\t */\n\t@Input('ngbScrollSpyFragment') id: string;\n\n\tngAfterViewInit() {\n\t\tthis._scrollSpy._registerFragment(this);\n\t\tthis._destroyRef.onDestroy(() => this._scrollSpy._unregisterFragment(this));\n\t}\n}\n"]}