UNPKG

@taiga-ui/core

Version:

Core library for creating Angular components and applications using Taiga UI

93 lines 14.6 kB
import { Directive, inject, Input } from '@angular/core'; import { takeUntilDestroyed } from '@angular/core/rxjs-interop'; import { WA_ANIMATION_FRAME } from '@ng-web-apis/common'; import { tuiScrollFrom, tuiZonefree, tuiZonefreeScheduler, } from '@taiga-ui/cdk/observables'; import { tuiInjectElement } from '@taiga-ui/cdk/utils/dom'; import { TUI_SCROLL_REF } from '@taiga-ui/core/tokens'; import { merge, throttleTime } from 'rxjs'; import { TuiScrollbarService } from './scrollbar.service'; import * as i0 from "@angular/core"; const MIN_WIDTH = 24; class TuiScrollbarDirective { constructor() { this.el = inject(TUI_SCROLL_REF).nativeElement; this.style = tuiInjectElement().style; this.scrollSub = inject(TuiScrollbarService) .pipe(takeUntilDestroyed()) .subscribe(([top, left]) => { this.el.style.scrollBehavior = 'auto'; if (this.tuiScrollbar === 'horizontal') { this.el.scrollLeft = left; } else { this.el.scrollTop = top; } this.el.style.scrollBehavior = ''; }); this.styleSub = merge(inject(WA_ANIMATION_FRAME).pipe(throttleTime(100, tuiZonefreeScheduler())), tuiScrollFrom(this.el)) .pipe(tuiZonefree(), takeUntilDestroyed()) .subscribe(() => { const dimension = { scrollTop: this.el.scrollTop, scrollHeight: this.el.scrollHeight, clientHeight: this.el.clientHeight, scrollLeft: this.el.scrollLeft, scrollWidth: this.el.scrollWidth, clientWidth: this.el.clientWidth, }; const thumb = `${this.getThumb(dimension) * 100}%`; const view = `${this.getView(dimension) * 100}%`; if (this.tuiScrollbar === 'vertical') { this.style.top = thumb; this.style.height = view; } else { this.style.left = thumb; this.style.insetInlineStart = thumb; this.style.width = view; } }); this.tuiScrollbar = 'vertical'; } getScrolled(dimension) { return this.tuiScrollbar === 'vertical' ? dimension.scrollTop / (dimension.scrollHeight - dimension.clientHeight) : dimension.scrollLeft / (dimension.scrollWidth - dimension.clientWidth); } getCompensation(dimension) { if (((dimension.clientHeight * dimension.clientHeight) / dimension.scrollHeight > MIN_WIDTH && this.tuiScrollbar === 'vertical') || ((dimension.clientWidth * dimension.clientWidth) / dimension.scrollWidth > MIN_WIDTH && this.tuiScrollbar === 'horizontal')) { return 0; } return this.tuiScrollbar === 'vertical' ? MIN_WIDTH / dimension.clientHeight : MIN_WIDTH / dimension.clientWidth; } getThumb(dimension) { const compensation = this.getCompensation(dimension) || this.getView(dimension); return Math.abs(this.getScrolled(dimension) * (1 - compensation)); } getView(dimension) { return this.tuiScrollbar === 'vertical' ? Math.ceil((dimension.clientHeight / dimension.scrollHeight) * 100) / 100 : Math.ceil((dimension.clientWidth / dimension.scrollWidth) * 100) / 100; } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiScrollbarDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); } static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: TuiScrollbarDirective, isStandalone: true, selector: "[tuiScrollbar]", inputs: { tuiScrollbar: "tuiScrollbar" }, providers: [TuiScrollbarService], ngImport: i0 }); } } export { TuiScrollbarDirective }; i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiScrollbarDirective, decorators: [{ type: Directive, args: [{ standalone: true, selector: '[tuiScrollbar]', providers: [TuiScrollbarService], }] }], propDecorators: { tuiScrollbar: [{ type: Input }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scrollbar.directive.js","sourceRoot":"","sources":["../../../../../projects/core/components/scrollbar/scrollbar.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {Directive, inject, Input} from '@angular/core';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {WA_ANIMATION_FRAME} from '@ng-web-apis/common';\nimport {\n    tuiScrollFrom,\n    tuiZonefree,\n    tuiZonefreeScheduler,\n} from '@taiga-ui/cdk/observables';\nimport {tuiInjectElement} from '@taiga-ui/cdk/utils/dom';\nimport {TUI_SCROLL_REF} from '@taiga-ui/core/tokens';\nimport {merge, throttleTime} from 'rxjs';\n\nimport {TuiScrollbarService} from './scrollbar.service';\n\nconst MIN_WIDTH = 24;\n\ninterface ComputedDimension {\n    scrollTop: number;\n    scrollHeight: number;\n    clientHeight: number;\n    scrollLeft: number;\n    scrollWidth: number;\n    clientWidth: number;\n}\n\n@Directive({\n    standalone: true,\n    selector: '[tuiScrollbar]',\n    providers: [TuiScrollbarService],\n})\nexport class TuiScrollbarDirective {\n    private readonly el = inject(TUI_SCROLL_REF).nativeElement;\n    private readonly style = tuiInjectElement().style;\n\n    protected readonly scrollSub = inject(TuiScrollbarService)\n        .pipe(takeUntilDestroyed())\n        .subscribe(([top, left]) => {\n            this.el.style.scrollBehavior = 'auto';\n\n            if (this.tuiScrollbar === 'horizontal') {\n                this.el.scrollLeft = left;\n            } else {\n                this.el.scrollTop = top;\n            }\n\n            this.el.style.scrollBehavior = '';\n        });\n\n    protected readonly styleSub = merge(\n        inject(WA_ANIMATION_FRAME).pipe(throttleTime(100, tuiZonefreeScheduler())),\n        tuiScrollFrom(this.el),\n    )\n        .pipe(tuiZonefree(), takeUntilDestroyed())\n        .subscribe(() => {\n            const dimension: ComputedDimension = {\n                scrollTop: this.el.scrollTop,\n                scrollHeight: this.el.scrollHeight,\n                clientHeight: this.el.clientHeight,\n                scrollLeft: this.el.scrollLeft,\n                scrollWidth: this.el.scrollWidth,\n                clientWidth: this.el.clientWidth,\n            };\n\n            const thumb = `${this.getThumb(dimension) * 100}%`;\n            const view = `${this.getView(dimension) * 100}%`;\n\n            if (this.tuiScrollbar === 'vertical') {\n                this.style.top = thumb;\n                this.style.height = view;\n            } else {\n                this.style.left = thumb;\n                this.style.insetInlineStart = thumb;\n                this.style.width = view;\n            }\n        });\n\n    @Input()\n    public tuiScrollbar: 'horizontal' | 'vertical' = 'vertical';\n\n    private getScrolled(dimension: ComputedDimension): number {\n        return this.tuiScrollbar === 'vertical'\n            ? dimension.scrollTop / (dimension.scrollHeight - dimension.clientHeight)\n            : dimension.scrollLeft / (dimension.scrollWidth - dimension.clientWidth);\n    }\n\n    private getCompensation(dimension: ComputedDimension): number {\n        if (\n            ((dimension.clientHeight * dimension.clientHeight) / dimension.scrollHeight >\n                MIN_WIDTH &&\n                this.tuiScrollbar === 'vertical') ||\n            ((dimension.clientWidth * dimension.clientWidth) / dimension.scrollWidth >\n                MIN_WIDTH &&\n                this.tuiScrollbar === 'horizontal')\n        ) {\n            return 0;\n        }\n\n        return this.tuiScrollbar === 'vertical'\n            ? MIN_WIDTH / dimension.clientHeight\n            : MIN_WIDTH / dimension.clientWidth;\n    }\n\n    private getThumb(dimension: ComputedDimension): number {\n        const compensation = this.getCompensation(dimension) || this.getView(dimension);\n\n        return Math.abs(this.getScrolled(dimension) * (1 - compensation));\n    }\n\n    private getView(dimension: ComputedDimension): number {\n        return this.tuiScrollbar === 'vertical'\n            ? Math.ceil((dimension.clientHeight / dimension.scrollHeight) * 100) / 100\n            : Math.ceil((dimension.clientWidth / dimension.scrollWidth) * 100) / 100;\n    }\n}\n"]}