Huge News!Announcing our $40M Series B led by Abstract Ventures.Learn More
Socket
Sign inDemoInstall
Socket

ng-table-virtual-scroll

Package Overview
Dependencies
Maintainers
1
Versions
21
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

ng-table-virtual-scroll - npm Package Compare versions

Comparing version 1.3.6 to 1.3.7

90

bundles/ng-table-virtual-scroll.umd.js

@@ -233,6 +233,12 @@ (function (global, factory) {

}
function __spreadArray(to, from) {
for (var i = 0, il = from.length, j = to.length; i < il; i++, j++)
to[j] = from[i];
return to;
function __spreadArray(to, from, pack) {
if (pack || arguments.length === 2)
for (var i = 0, l = from.length, ar; i < l; i++) {
if (ar || !(i in from)) {
if (!ar)
ar = Array.prototype.slice.call(from, 0, i);
ar[i] = from[i];
}
}
return to.concat(ar || from);
}

@@ -303,14 +309,17 @@ function __await(v) {

}
function __classPrivateFieldGet(receiver, privateMap) {
if (!privateMap.has(receiver)) {
throw new TypeError("attempted to get private field on non-instance");
}
return privateMap.get(receiver);
function __classPrivateFieldGet(receiver, state, kind, f) {
if (kind === "a" && !f)
throw new TypeError("Private accessor was defined without a getter");
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver))
throw new TypeError("Cannot read private member from an object whose class did not declare it");
return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
}
function __classPrivateFieldSet(receiver, privateMap, value) {
if (!privateMap.has(receiver)) {
throw new TypeError("attempted to set private field on non-instance");
}
privateMap.set(receiver, value);
return value;
function __classPrivateFieldSet(receiver, state, value, kind, f) {
if (kind === "m")
throw new TypeError("Private method is not writable");
if (kind === "a" && !f)
throw new TypeError("Private accessor was defined without a setter");
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver))
throw new TypeError("Cannot write private member to an object whose class did not declare it");
return (kind === "a" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;
}

@@ -408,3 +417,2 @@

FixedSizeTableVirtualScrollStrategy.prototype.onContentRendered = function () {
// no-op
};

@@ -438,15 +446,36 @@ FixedSizeTableVirtualScrollStrategy.prototype.onRenderedOffsetChanged = function () {

}
var renderedOffset = this.viewport.getOffsetToRenderedContentStart();
var start = renderedOffset / this.rowHeight;
var itemsDisplayed = Math.ceil(this.viewport.getViewportSize() / this.rowHeight);
var bufferItems = Math.ceil(itemsDisplayed * this.bufferMultiplier);
var end = start + itemsDisplayed + 2 * bufferItems;
var bufferOffset = renderedOffset + bufferItems * this.rowHeight;
var scrollOffset = this.viewport.measureScrollOffset();
var amount = Math.ceil(this.viewport.getViewportSize() / this.rowHeight);
var offset = Math.max(scrollOffset - this.headerHeight, 0);
var buffer = Math.ceil(amount * this.bufferMultiplier);
var skip = Math.round(offset / this.rowHeight);
var index = Math.max(0, skip);
var start = Math.max(0, index - buffer);
var end = Math.min(this.dataLength, index + amount + buffer);
var renderedOffset = start * this.rowHeight;
this.viewport.setRenderedContentOffset(renderedOffset);
this.viewport.setRenderedRange({ start: start, end: end });
this.indexChange.next(index);
this.stickyChange.next(renderedOffset);
// How far the scroll offset is from the lower buffer, which is usually where items start being displayed
var relativeScrollOffset = scrollOffset - bufferOffset;
var rowsScrolled = relativeScrollOffset / this.rowHeight;
var displayed = scrollOffset / this.rowHeight;
this.indexChange.next(displayed);
// Only bother updating the displayed information if we've scrolled more than a row
var rowSensitivity = 1.0;
if (Math.abs(rowsScrolled) < rowSensitivity) {
this.viewport.setRenderedContentOffset(renderedOffset);
this.viewport.setRenderedRange({ start: start, end: end });
return;
}
// Special case for the start of the table.
// At the top of the table, the first few rows are first rendered because they're visible, and then still rendered
// Because they move into the buffer. So we only need to change what's rendered once the user scrolls far enough down.
if (renderedOffset === 0 && rowsScrolled < 0) {
this.viewport.setRenderedContentOffset(renderedOffset);
this.viewport.setRenderedRange({ start: start, end: end });
return;
}
var rowsToMove = Math.sign(rowsScrolled) * Math.floor(Math.abs(rowsScrolled));
var adjustedRenderedOffset = Math.max(0, renderedOffset + rowsToMove * this.rowHeight);
this.viewport.setRenderedContentOffset(adjustedRenderedOffset);
var adjustedStart = Math.max(0, start + rowsToMove);
var adjustedEnd = adjustedStart + itemsDisplayed + 2 * bufferItems;
this.viewport.setRenderedRange({ start: adjustedStart, end: adjustedEnd });
this.stickyChange.next(adjustedRenderedOffset);
};

@@ -505,3 +534,6 @@ return FixedSizeTableVirtualScrollStrategy;

this.scrollStrategy.stickyChange
.pipe(operators.filter(function () { return _this.isStickyEnabled(); }), operators.delayWhen(function () { return !_this.stickyPositions ? rxjs.timer(0) : rxjs.of(); }), operators.tap(function () {
.pipe(operators.filter(function () { return _this.isStickyEnabled(); }),
// breaks sticky header on the top. needs investigation
// delayWhen(() => !this.stickyPositions ? timer(0) : of()),
operators.tap(function () {
if (!_this.stickyPositions) {

@@ -508,0 +540,0 @@ _this.initStickyPositions();

@@ -15,3 +15,3 @@ !function(t,e){"object"==typeof exports&&"undefined"!=typeof module?e(exports,require("@angular/core"),require("@angular/cdk/scrolling"),require("rxjs/operators"),require("rxjs"),require("@angular/material/table")):"function"==typeof define&&define.amd?define("ng-table-virtual-scroll",["exports","@angular/core","@angular/cdk/scrolling","rxjs/operators","rxjs","@angular/material/table"],e):e((t="undefined"!=typeof globalThis?globalThis:t||self)["ng-table-virtual-scroll"]={},t.ng.core,t.ng.cdk.scrolling,t.rxjs.operators,t.rxjs,t.ng.material.table)}(this,(function(t,e,r,i,n,o){"use strict";

PERFORMANCE OF THIS SOFTWARE.
***************************************************************************** */var a=function(t,e){return(a=Object.setPrototypeOf||{__proto__:[]}instanceof Array&&function(t,e){t.__proto__=e}||function(t,e){for(var r in e)Object.prototype.hasOwnProperty.call(e,r)&&(t[r]=e[r])})(t,e)};Object.create;function s(t,e){var r="function"==typeof Symbol&&t[Symbol.iterator];if(!r)return t;var i,n,o=r.call(t),a=[];try{for(;(void 0===e||e-- >0)&&!(i=o.next()).done;)a.push(i.value)}catch(t){n={error:t}}finally{try{i&&!i.done&&(r=o.return)&&r.call(o)}finally{if(n)throw n.error}}return a}Object.create;var h=function(t){function e(){return null!==t&&t.apply(this,arguments)||this}return function(t,e){if("function"!=typeof e&&null!==e)throw new TypeError("Class extends value "+String(e)+" is not a constructor or null");function r(){this.constructor=t}a(t,e),t.prototype=null===e?Object.create(e):(r.prototype=e.prototype,new r)}(e,t),e.prototype._updateChangeSubscription=function(){var t,e=this;this.initStreams();var r=this._sort,o=this._paginator,a=this._internalPageChanges,h=this._filter,c=this._renderData,l=r?n.merge(r.sortChange,r.initialized):n.of(null),u=o?n.merge(o.page,a,o.initialized):n.of(null),p=this._data,f=n.combineLatest([p,h]).pipe(i.map((function(t){var r=s(t,1)[0];return e._filterData(r)}))),d=n.combineLatest([f,l]).pipe(i.map((function(t){var r=s(t,1)[0];return e._orderData(r)}))),g=n.combineLatest([d,u]).pipe(i.map((function(t){var r=s(t,1)[0];return e._pageData(r)})));null===(t=this._renderChangesSubscription)||void 0===t||t.unsubscribe(),this._renderChangesSubscription=new n.Subscription,this._renderChangesSubscription.add(g.subscribe((function(t){return e.dataToRender$.next(t)}))),this._renderChangesSubscription.add(this.dataOfRange$.subscribe((function(t){return c.next(t)})))},e.prototype.initStreams=function(){this.streamsReady||(this.dataToRender$=new n.ReplaySubject(1),this.dataOfRange$=new n.ReplaySubject(1),this.streamsReady=!0)},e}(o.MatTableDataSource),c=function(){function t(){this.indexChange=new n.Subject,this.stickyChange=new n.Subject,this.renderedRangeStream=new n.BehaviorSubject({start:0,end:0}),this.scrolledIndexChange=this.indexChange.pipe(i.distinctUntilChanged()),this._dataLength=0}return Object.defineProperty(t.prototype,"dataLength",{get:function(){return this._dataLength},set:function(t){this._dataLength=t,this.onDataLengthChanged()},enumerable:!1,configurable:!0}),t.prototype.attach=function(t){this.viewport=t,this.viewport.renderedRangeStream.subscribe(this.renderedRangeStream),this.onDataLengthChanged()},t.prototype.detach=function(){this.indexChange.complete(),this.stickyChange.complete(),this.renderedRangeStream.complete()},t.prototype.onContentScrolled=function(){this.updateContent()},t.prototype.onDataLengthChanged=function(){this.viewport&&this.viewport.setTotalContentSize(this.dataLength*this.rowHeight+this.headerHeight+this.footerHeight),this.updateContent()},t.prototype.onContentRendered=function(){},t.prototype.onRenderedOffsetChanged=function(){},t.prototype.scrollToIndex=function(t,e){this.viewport&&this.rowHeight&&this.viewport.scrollToOffset((t-1)*this.rowHeight+this.headerHeight,e)},t.prototype.setConfig=function(t){var e=t.rowHeight,r=t.headerHeight,i=t.footerHeight,n=t.bufferMultiplier;this.rowHeight===e&&this.headerHeight===r&&this.footerHeight===i&&this.bufferMultiplier===n||(this.rowHeight=e,this.headerHeight=r,this.footerHeight=i,this.bufferMultiplier=n,this.onDataLengthChanged())},t.prototype.updateContent=function(){if(this.viewport&&this.rowHeight){var t=this.viewport.measureScrollOffset(),e=Math.ceil(this.viewport.getViewportSize()/this.rowHeight),r=Math.max(t-this.headerHeight,0),i=Math.ceil(e*this.bufferMultiplier),n=Math.round(r/this.rowHeight),o=Math.max(0,n),a=Math.max(0,o-i),s=Math.min(this.dataLength,o+e+i),h=a*this.rowHeight;this.viewport.setRenderedContentOffset(h),this.viewport.setRenderedRange({start:a,end:s}),this.indexChange.next(o),this.stickyChange.next(h)}},t}();function l(t){return t.scrollStrategy}c.decorators=[{type:e.Injectable}];var u=".mat-header-row .mat-table-sticky",p=48,f=56,d=!0,g=48,y=!1,b=.7,S=function(){function t(t){this.zone=t,this.destroyed$=new n.Subject,this.rowHeight=p,this.headerEnabled=d,this.headerHeight=f,this.footerEnabled=y,this.footerHeight=g,this.bufferMultiplier=b,this.scrollStrategy=new c,this.dataSourceChanges=new n.Subject}return t.prototype.ngOnDestroy=function(){this.destroyed$.next(),this.destroyed$.complete(),this.dataSourceChanges.complete()},t.prototype.isStickyEnabled=function(){return!!this.scrollStrategy.viewport&&this.table._headerRowDefs.map((function(t){return t.sticky})).reduce((function(t,e){return t&&e}),!0)},t.prototype.ngAfterContentInit=function(){var t=this,e=this.table._switchDataSource;this.table._switchDataSource=function(r){e.call(t.table,r),t.connectDataSource(r)},this.connectDataSource(this.table.dataSource),this.scrollStrategy.stickyChange.pipe(i.filter((function(){return t.isStickyEnabled()})),i.delayWhen((function(){return t.stickyPositions?n.of():n.timer(0)})),i.tap((function(){t.stickyPositions||t.initStickyPositions()})),i.takeUntil(this.destroyed$)).subscribe((function(e){t.setSticky(e)}))},t.prototype.connectDataSource=function(t){var e=this;if(this.dataSourceChanges.next(),!(t instanceof h))throw new Error("[tvsItemSize] requires TableVirtualScrollDataSource be set as [dataSource] of [mat-table]");t.dataToRender$.pipe(i.distinctUntilChanged(),i.takeUntil(this.dataSourceChanges),i.takeUntil(this.destroyed$),i.tap((function(t){return e.scrollStrategy.dataLength=t.length})),i.switchMap((function(t){return e.scrollStrategy.renderedRangeStream.pipe(i.map((function(e){var r=e.start,i=e.end;return"number"!=typeof r||"number"!=typeof i?t:t.slice(r,i)})))}))).subscribe((function(r){e.zone.run((function(){t.dataOfRange$.next(r)}))}))},t.prototype.ngOnChanges=function(){var t={rowHeight:+this.rowHeight||p,headerHeight:this.headerEnabled?+this.headerHeight||f:0,footerHeight:this.footerEnabled?+this.footerHeight||g:0,bufferMultiplier:+this.bufferMultiplier||b};this.scrollStrategy.setConfig(t)},t.prototype.setSticky=function(t){var e=this;this.scrollStrategy.viewport.elementRef.nativeElement.querySelectorAll(u).forEach((function(r){var i=r.parentElement,n=0;e.stickyPositions.has(i)&&(n=e.stickyPositions.get(i)),r.style.top=n-t+"px"})),this.scrollStrategy.viewport.elementRef.nativeElement.querySelectorAll(".mat-footer-row .mat-table-sticky").forEach((function(r){var i=r.parentElement,n=0;e.stickyPositions.has(i)&&(n=e.stickyPositions.get(i)),r.style.bottom=-n+t+"px"}))},t.prototype.initStickyPositions=function(){var t=this;this.stickyPositions=new Map,this.scrollStrategy.viewport.elementRef.nativeElement.querySelectorAll(u).forEach((function(e){var r=e.parentElement;t.stickyPositions.has(r)||t.stickyPositions.set(r,r.offsetTop)}))},t}();S.decorators=[{type:e.Directive,args:[{selector:"cdk-virtual-scroll-viewport[tvsItemSize]",providers:[{provide:r.VIRTUAL_SCROLL_STRATEGY,useFactory:l,deps:[e.forwardRef((function(){return S}))]}]}]}],S.ctorParameters=function(){return[{type:e.NgZone}]},S.propDecorators={rowHeight:[{type:e.Input,args:["tvsItemSize"]}],headerEnabled:[{type:e.Input}],headerHeight:[{type:e.Input}],footerEnabled:[{type:e.Input}],footerHeight:[{type:e.Input}],bufferMultiplier:[{type:e.Input}],table:[{type:e.ContentChild,args:[o.MatTable,{static:!1}]}]};var m=function(){};m.decorators=[{type:e.NgModule,args:[{declarations:[S],imports:[],exports:[S]}]}],t.FixedSizeTableVirtualScrollStrategy=c,t.TableItemSizeDirective=S,t.TableVirtualScrollDataSource=h,t.TableVirtualScrollModule=m,t._tableVirtualScrollDirectiveStrategyFactory=l,Object.defineProperty(t,"__esModule",{value:!0})}));
***************************************************************************** */var a=function(t,e){return(a=Object.setPrototypeOf||{__proto__:[]}instanceof Array&&function(t,e){t.__proto__=e}||function(t,e){for(var r in e)Object.prototype.hasOwnProperty.call(e,r)&&(t[r]=e[r])})(t,e)};Object.create;function s(t,e){var r="function"==typeof Symbol&&t[Symbol.iterator];if(!r)return t;var i,n,o=r.call(t),a=[];try{for(;(void 0===e||e-- >0)&&!(i=o.next()).done;)a.push(i.value)}catch(t){n={error:t}}finally{try{i&&!i.done&&(r=o.return)&&r.call(o)}finally{if(n)throw n.error}}return a}Object.create;var h=function(t){function e(){return null!==t&&t.apply(this,arguments)||this}return function(t,e){if("function"!=typeof e&&null!==e)throw new TypeError("Class extends value "+String(e)+" is not a constructor or null");function r(){this.constructor=t}a(t,e),t.prototype=null===e?Object.create(e):(r.prototype=e.prototype,new r)}(e,t),e.prototype._updateChangeSubscription=function(){var t,e=this;this.initStreams();var r=this._sort,o=this._paginator,a=this._internalPageChanges,h=this._filter,c=this._renderData,l=r?n.merge(r.sortChange,r.initialized):n.of(null),u=o?n.merge(o.page,a,o.initialized):n.of(null),p=this._data,d=n.combineLatest([p,h]).pipe(i.map((function(t){var r=s(t,1)[0];return e._filterData(r)}))),f=n.combineLatest([d,l]).pipe(i.map((function(t){var r=s(t,1)[0];return e._orderData(r)}))),g=n.combineLatest([f,u]).pipe(i.map((function(t){var r=s(t,1)[0];return e._pageData(r)})));null===(t=this._renderChangesSubscription)||void 0===t||t.unsubscribe(),this._renderChangesSubscription=new n.Subscription,this._renderChangesSubscription.add(g.subscribe((function(t){return e.dataToRender$.next(t)}))),this._renderChangesSubscription.add(this.dataOfRange$.subscribe((function(t){return c.next(t)})))},e.prototype.initStreams=function(){this.streamsReady||(this.dataToRender$=new n.ReplaySubject(1),this.dataOfRange$=new n.ReplaySubject(1),this.streamsReady=!0)},e}(o.MatTableDataSource),c=function(){function t(){this.indexChange=new n.Subject,this.stickyChange=new n.Subject,this.renderedRangeStream=new n.BehaviorSubject({start:0,end:0}),this.scrolledIndexChange=this.indexChange.pipe(i.distinctUntilChanged()),this._dataLength=0}return Object.defineProperty(t.prototype,"dataLength",{get:function(){return this._dataLength},set:function(t){this._dataLength=t,this.onDataLengthChanged()},enumerable:!1,configurable:!0}),t.prototype.attach=function(t){this.viewport=t,this.viewport.renderedRangeStream.subscribe(this.renderedRangeStream),this.onDataLengthChanged()},t.prototype.detach=function(){this.indexChange.complete(),this.stickyChange.complete(),this.renderedRangeStream.complete()},t.prototype.onContentScrolled=function(){this.updateContent()},t.prototype.onDataLengthChanged=function(){this.viewport&&this.viewport.setTotalContentSize(this.dataLength*this.rowHeight+this.headerHeight+this.footerHeight),this.updateContent()},t.prototype.onContentRendered=function(){},t.prototype.onRenderedOffsetChanged=function(){},t.prototype.scrollToIndex=function(t,e){this.viewport&&this.rowHeight&&this.viewport.scrollToOffset((t-1)*this.rowHeight+this.headerHeight,e)},t.prototype.setConfig=function(t){var e=t.rowHeight,r=t.headerHeight,i=t.footerHeight,n=t.bufferMultiplier;this.rowHeight===e&&this.headerHeight===r&&this.footerHeight===i&&this.bufferMultiplier===n||(this.rowHeight=e,this.headerHeight=r,this.footerHeight=i,this.bufferMultiplier=n,this.onDataLengthChanged())},t.prototype.updateContent=function(){if(this.viewport&&this.rowHeight){var t=this.viewport.getOffsetToRenderedContentStart(),e=t/this.rowHeight,r=Math.ceil(this.viewport.getViewportSize()/this.rowHeight),i=Math.ceil(r*this.bufferMultiplier),n=e+r+2*i,o=t+i*this.rowHeight,a=this.viewport.measureScrollOffset(),s=(a-o)/this.rowHeight,h=a/this.rowHeight;this.indexChange.next(h);if(Math.abs(s)<1)return this.viewport.setRenderedContentOffset(t),void this.viewport.setRenderedRange({start:e,end:n});if(0===t&&s<0)return this.viewport.setRenderedContentOffset(t),void this.viewport.setRenderedRange({start:e,end:n});var c=Math.sign(s)*Math.floor(Math.abs(s)),l=Math.max(0,t+c*this.rowHeight);this.viewport.setRenderedContentOffset(l);var u=Math.max(0,e+c),p=u+r+2*i;this.viewport.setRenderedRange({start:u,end:p}),this.stickyChange.next(l)}},t}();function l(t){return t.scrollStrategy}c.decorators=[{type:e.Injectable}];var u=".mat-header-row .mat-table-sticky",p=48,d=56,f=!0,g=48,y=!1,b=.7,S=function(){function t(t){this.zone=t,this.destroyed$=new n.Subject,this.rowHeight=p,this.headerEnabled=f,this.headerHeight=d,this.footerEnabled=y,this.footerHeight=g,this.bufferMultiplier=b,this.scrollStrategy=new c,this.dataSourceChanges=new n.Subject}return t.prototype.ngOnDestroy=function(){this.destroyed$.next(),this.destroyed$.complete(),this.dataSourceChanges.complete()},t.prototype.isStickyEnabled=function(){return!!this.scrollStrategy.viewport&&this.table._headerRowDefs.map((function(t){return t.sticky})).reduce((function(t,e){return t&&e}),!0)},t.prototype.ngAfterContentInit=function(){var t=this,e=this.table._switchDataSource;this.table._switchDataSource=function(r){e.call(t.table,r),t.connectDataSource(r)},this.connectDataSource(this.table.dataSource),this.scrollStrategy.stickyChange.pipe(i.filter((function(){return t.isStickyEnabled()})),i.tap((function(){t.stickyPositions||t.initStickyPositions()})),i.takeUntil(this.destroyed$)).subscribe((function(e){t.setSticky(e)}))},t.prototype.connectDataSource=function(t){var e=this;if(this.dataSourceChanges.next(),!(t instanceof h))throw new Error("[tvsItemSize] requires TableVirtualScrollDataSource be set as [dataSource] of [mat-table]");t.dataToRender$.pipe(i.distinctUntilChanged(),i.takeUntil(this.dataSourceChanges),i.takeUntil(this.destroyed$),i.tap((function(t){return e.scrollStrategy.dataLength=t.length})),i.switchMap((function(t){return e.scrollStrategy.renderedRangeStream.pipe(i.map((function(e){var r=e.start,i=e.end;return"number"!=typeof r||"number"!=typeof i?t:t.slice(r,i)})))}))).subscribe((function(r){e.zone.run((function(){t.dataOfRange$.next(r)}))}))},t.prototype.ngOnChanges=function(){var t={rowHeight:+this.rowHeight||p,headerHeight:this.headerEnabled?+this.headerHeight||d:0,footerHeight:this.footerEnabled?+this.footerHeight||g:0,bufferMultiplier:+this.bufferMultiplier||b};this.scrollStrategy.setConfig(t)},t.prototype.setSticky=function(t){var e=this;this.scrollStrategy.viewport.elementRef.nativeElement.querySelectorAll(u).forEach((function(r){var i=r.parentElement,n=0;e.stickyPositions.has(i)&&(n=e.stickyPositions.get(i)),r.style.top=n-t+"px"})),this.scrollStrategy.viewport.elementRef.nativeElement.querySelectorAll(".mat-footer-row .mat-table-sticky").forEach((function(r){var i=r.parentElement,n=0;e.stickyPositions.has(i)&&(n=e.stickyPositions.get(i)),r.style.bottom=-n+t+"px"}))},t.prototype.initStickyPositions=function(){var t=this;this.stickyPositions=new Map,this.scrollStrategy.viewport.elementRef.nativeElement.querySelectorAll(u).forEach((function(e){var r=e.parentElement;t.stickyPositions.has(r)||t.stickyPositions.set(r,r.offsetTop)}))},t}();S.decorators=[{type:e.Directive,args:[{selector:"cdk-virtual-scroll-viewport[tvsItemSize]",providers:[{provide:r.VIRTUAL_SCROLL_STRATEGY,useFactory:l,deps:[e.forwardRef((function(){return S}))]}]}]}],S.ctorParameters=function(){return[{type:e.NgZone}]},S.propDecorators={rowHeight:[{type:e.Input,args:["tvsItemSize"]}],headerEnabled:[{type:e.Input}],headerHeight:[{type:e.Input}],footerEnabled:[{type:e.Input}],footerHeight:[{type:e.Input}],bufferMultiplier:[{type:e.Input}],table:[{type:e.ContentChild,args:[o.MatTable,{static:!1}]}]};var v=function(){};v.decorators=[{type:e.NgModule,args:[{declarations:[S],imports:[],exports:[S]}]}],t.FixedSizeTableVirtualScrollStrategy=c,t.TableItemSizeDirective=S,t.TableVirtualScrollDataSource=h,t.TableVirtualScrollModule=v,t._tableVirtualScrollDirectiveStrategyFactory=l,Object.defineProperty(t,"__esModule",{value:!0})}));
//# sourceMappingURL=ng-table-virtual-scroll.umd.min.js.map

@@ -39,3 +39,2 @@ import { Injectable } from '@angular/core';

onContentRendered() {
// no-op
}

@@ -69,15 +68,36 @@ onRenderedOffsetChanged() {

}
const renderedOffset = this.viewport.getOffsetToRenderedContentStart();
const start = renderedOffset / this.rowHeight;
const itemsDisplayed = Math.ceil(this.viewport.getViewportSize() / this.rowHeight);
const bufferItems = Math.ceil(itemsDisplayed * this.bufferMultiplier);
const end = start + itemsDisplayed + 2 * bufferItems;
const bufferOffset = renderedOffset + bufferItems * this.rowHeight;
const scrollOffset = this.viewport.measureScrollOffset();
const amount = Math.ceil(this.viewport.getViewportSize() / this.rowHeight);
const offset = Math.max(scrollOffset - this.headerHeight, 0);
const buffer = Math.ceil(amount * this.bufferMultiplier);
const skip = Math.round(offset / this.rowHeight);
const index = Math.max(0, skip);
const start = Math.max(0, index - buffer);
const end = Math.min(this.dataLength, index + amount + buffer);
const renderedOffset = start * this.rowHeight;
this.viewport.setRenderedContentOffset(renderedOffset);
this.viewport.setRenderedRange({ start, end });
this.indexChange.next(index);
this.stickyChange.next(renderedOffset);
// How far the scroll offset is from the lower buffer, which is usually where items start being displayed
const relativeScrollOffset = scrollOffset - bufferOffset;
const rowsScrolled = relativeScrollOffset / this.rowHeight;
const displayed = scrollOffset / this.rowHeight;
this.indexChange.next(displayed);
// Only bother updating the displayed information if we've scrolled more than a row
const rowSensitivity = 1.0;
if (Math.abs(rowsScrolled) < rowSensitivity) {
this.viewport.setRenderedContentOffset(renderedOffset);
this.viewport.setRenderedRange({ start, end });
return;
}
// Special case for the start of the table.
// At the top of the table, the first few rows are first rendered because they're visible, and then still rendered
// Because they move into the buffer. So we only need to change what's rendered once the user scrolls far enough down.
if (renderedOffset === 0 && rowsScrolled < 0) {
this.viewport.setRenderedContentOffset(renderedOffset);
this.viewport.setRenderedRange({ start, end });
return;
}
const rowsToMove = Math.sign(rowsScrolled) * Math.floor(Math.abs(rowsScrolled));
const adjustedRenderedOffset = Math.max(0, renderedOffset + rowsToMove * this.rowHeight);
this.viewport.setRenderedContentOffset(adjustedRenderedOffset);
const adjustedStart = Math.max(0, start + rowsToMove);
const adjustedEnd = adjustedStart + itemsDisplayed + 2 * bufferItems;
this.viewport.setRenderedRange({ start: adjustedStart, end: adjustedEnd });
this.stickyChange.next(adjustedRenderedOffset);
}

@@ -88,2 +108,2 @@ }

];
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fixed-size-table-virtual-scroll-strategy.js","sourceRoot":"","sources":["../../../../projects/ng-table-virtual-scroll/src/lib/fixed-size-table-virtual-scroll-strategy.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { distinctUntilChanged } from 'rxjs/operators';\nimport { BehaviorSubject, Subject } from 'rxjs';\nimport { CdkVirtualScrollViewport, VirtualScrollStrategy } from '@angular/cdk/scrolling';\nimport { ListRange } from '@angular/cdk/collections';\n\nexport interface TSVStrategyConfigs {\n  rowHeight: number;\n  headerHeight: number;\n  footerHeight: number;\n  bufferMultiplier: number;\n}\n\n@Injectable()\nexport class FixedSizeTableVirtualScrollStrategy implements VirtualScrollStrategy {\n  private rowHeight!: number;\n  private headerHeight!: number;\n  private footerHeight!: number;\n  private bufferMultiplier!: number;\n  private indexChange = new Subject<number>();\n  public stickyChange = new Subject<number>();\n\n  public viewport: CdkVirtualScrollViewport;\n\n  public renderedRangeStream = new BehaviorSubject<ListRange>({start: 0, end: 0});\n\n  public scrolledIndexChange = this.indexChange.pipe(distinctUntilChanged());\n\n  get dataLength(): number {\n    return this._dataLength;\n  }\n\n  set dataLength(value: number) {\n    this._dataLength = value;\n    this.onDataLengthChanged();\n  }\n\n  private _dataLength = 0;\n\n  public attach(viewport: CdkVirtualScrollViewport): void {\n    this.viewport = viewport;\n    this.viewport.renderedRangeStream.subscribe(this.renderedRangeStream);\n    this.onDataLengthChanged();\n  }\n\n  public detach(): void {\n    this.indexChange.complete();\n    this.stickyChange.complete();\n    this.renderedRangeStream.complete();\n  }\n\n  public onContentScrolled(): void {\n    this.updateContent();\n  }\n\n  public onDataLengthChanged(): void {\n    if (this.viewport) {\n      this.viewport.setTotalContentSize(this.dataLength * this.rowHeight + this.headerHeight + this.footerHeight);\n    }\n    this.updateContent();\n  }\n\n  public onContentRendered(): void {\n    // no-op\n  }\n\n  public onRenderedOffsetChanged(): void {\n    // no-op\n  }\n\n  public scrollToIndex(index: number, behavior?: ScrollBehavior): void {\n    if (!this.viewport || !this.rowHeight) {\n      return;\n    }\n    this.viewport.scrollToOffset((index - 1 ) * this.rowHeight + this.headerHeight, behavior);\n  }\n\n  public setConfig(configs: TSVStrategyConfigs) {\n    const {rowHeight, headerHeight, footerHeight, bufferMultiplier} = configs;\n    if (\n      this.rowHeight === rowHeight\n      && this.headerHeight === headerHeight\n      && this.footerHeight === footerHeight\n      && this.bufferMultiplier === bufferMultiplier\n    ) {\n      return;\n    }\n    this.rowHeight = rowHeight;\n    this.headerHeight = headerHeight;\n    this.footerHeight = footerHeight;\n    this.bufferMultiplier = bufferMultiplier;\n    this.onDataLengthChanged();\n  }\n\n  private updateContent() {\n    if (!this.viewport || !this.rowHeight) {\n      return;\n    }\n    const scrollOffset = this.viewport.measureScrollOffset();\n    const amount = Math.ceil(this.viewport.getViewportSize() / this.rowHeight);\n    const offset = Math.max(scrollOffset - this.headerHeight, 0);\n    const buffer = Math.ceil(amount * this.bufferMultiplier);\n\n    const skip = Math.round(offset / this.rowHeight);\n    const index = Math.max(0, skip);\n    const start = Math.max(0, index - buffer);\n    const end = Math.min(this.dataLength, index + amount + buffer);\n    const renderedOffset = start * this.rowHeight;\n    this.viewport.setRenderedContentOffset(renderedOffset);\n    this.viewport.setRenderedRange({start, end});\n    this.indexChange.next(index);\n    this.stickyChange.next(renderedOffset);\n  }\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fixed-size-table-virtual-scroll-strategy.js","sourceRoot":"","sources":["../../../../projects/ng-table-virtual-scroll/src/lib/fixed-size-table-virtual-scroll-strategy.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { distinctUntilChanged } from 'rxjs/operators';\nimport { BehaviorSubject, Subject } from 'rxjs';\nimport { CdkVirtualScrollViewport, VirtualScrollStrategy } from '@angular/cdk/scrolling';\nimport { ListRange } from '@angular/cdk/collections';\n\nexport interface TSVStrategyConfigs {\n  rowHeight: number;\n  headerHeight: number;\n  footerHeight: number;\n  bufferMultiplier: number;\n}\n\n@Injectable()\nexport class FixedSizeTableVirtualScrollStrategy implements VirtualScrollStrategy {\n  private rowHeight!: number;\n  private headerHeight!: number;\n  private footerHeight!: number;\n  private bufferMultiplier!: number;\n  private indexChange = new Subject<number>();\n  public stickyChange = new Subject<number>();\n\n  public viewport: CdkVirtualScrollViewport;\n\n  public renderedRangeStream = new BehaviorSubject<ListRange>({start: 0, end: 0});\n\n  public scrolledIndexChange = this.indexChange.pipe(distinctUntilChanged());\n\n  get dataLength(): number {\n    return this._dataLength;\n  }\n\n  set dataLength(value: number) {\n    this._dataLength = value;\n    this.onDataLengthChanged();\n  }\n\n  private _dataLength = 0;\n\n  public attach(viewport: CdkVirtualScrollViewport): void {\n    this.viewport = viewport;\n    this.viewport.renderedRangeStream.subscribe(this.renderedRangeStream);\n    this.onDataLengthChanged();\n  }\n\n  public detach(): void {\n    this.indexChange.complete();\n    this.stickyChange.complete();\n    this.renderedRangeStream.complete();\n  }\n\n  public onContentScrolled(): void {\n    this.updateContent();\n  }\n\n  public onDataLengthChanged(): void {\n    if (this.viewport) {\n      this.viewport.setTotalContentSize(this.dataLength * this.rowHeight + this.headerHeight + this.footerHeight);\n    }\n    this.updateContent();\n  }\n\n  public onContentRendered(): void {\n  }\n\n  public onRenderedOffsetChanged(): void {\n    // no-op\n  }\n\n  public scrollToIndex(index: number, behavior?: ScrollBehavior): void {\n    if (!this.viewport || !this.rowHeight) {\n      return;\n    }\n    this.viewport.scrollToOffset((index - 1 ) * this.rowHeight + this.headerHeight, behavior);\n  }\n\n  public setConfig(configs: TSVStrategyConfigs) {\n    const {rowHeight, headerHeight, footerHeight, bufferMultiplier} = configs;\n    if (\n      this.rowHeight === rowHeight\n      && this.headerHeight === headerHeight\n      && this.footerHeight === footerHeight\n      && this.bufferMultiplier === bufferMultiplier\n    ) {\n      return;\n    }\n    this.rowHeight = rowHeight;\n    this.headerHeight = headerHeight;\n    this.footerHeight = footerHeight;\n    this.bufferMultiplier = bufferMultiplier;\n    this.onDataLengthChanged();\n  }\n\n  private updateContent() {\n    if (!this.viewport || !this.rowHeight) {\n      return;\n    }\n\n    const renderedOffset = this.viewport.getOffsetToRenderedContentStart();\n    const start = renderedOffset / this.rowHeight;\n    const itemsDisplayed = Math.ceil(this.viewport.getViewportSize() / this.rowHeight);\n    const bufferItems = Math.ceil(itemsDisplayed * this.bufferMultiplier);\n    const end = start + itemsDisplayed + 2 * bufferItems;\n\n\n    const bufferOffset = renderedOffset + bufferItems * this.rowHeight;\n    const scrollOffset = this.viewport.measureScrollOffset();\n\n    // How far the scroll offset is from the lower buffer, which is usually where items start being displayed\n    const relativeScrollOffset = scrollOffset - bufferOffset;\n    const rowsScrolled = relativeScrollOffset / this.rowHeight;\n\n    const displayed = scrollOffset / this.rowHeight;\n    this.indexChange.next(displayed);\n\n    // Only bother updating the displayed information if we've scrolled more than a row\n    const rowSensitivity = 1.0;\n    if (Math.abs(rowsScrolled) < rowSensitivity) {\n      this.viewport.setRenderedContentOffset(renderedOffset);\n      this.viewport.setRenderedRange({start, end});\n      return;\n    }\n\n    // Special case for the start of the table.\n    // At the top of the table, the first few rows are first rendered because they're visible, and then still rendered\n    // Because they move into the buffer. So we only need to change what's rendered once the user scrolls far enough down.\n    if (renderedOffset === 0 && rowsScrolled < 0) {\n      this.viewport.setRenderedContentOffset(renderedOffset);\n      this.viewport.setRenderedRange({start, end});\n      return;\n    }\n\n    const rowsToMove = Math.sign(rowsScrolled) * Math.floor(Math.abs(rowsScrolled));\n    const adjustedRenderedOffset = Math.max(0, renderedOffset + rowsToMove * this.rowHeight);\n    this.viewport.setRenderedContentOffset(adjustedRenderedOffset);\n\n    const adjustedStart = Math.max(0, start + rowsToMove);\n    const adjustedEnd = adjustedStart + itemsDisplayed + 2 * bufferItems;\n    this.viewport.setRenderedRange({start: adjustedStart, end: adjustedEnd});\n\n    this.stickyChange.next(adjustedRenderedOffset);\n  }\n}\n"]}
import { ContentChild, Directive, forwardRef, Input, NgZone } from '@angular/core';
import { VIRTUAL_SCROLL_STRATEGY } from '@angular/cdk/scrolling';
import { delayWhen, distinctUntilChanged, filter, map, switchMap, takeUntil, tap } from 'rxjs/operators';
import { distinctUntilChanged, filter, map, switchMap, takeUntil, tap } from 'rxjs/operators';
import { TableVirtualScrollDataSource } from './table-data-source';
import { MatTable } from '@angular/material/table';
import { FixedSizeTableVirtualScrollStrategy } from './fixed-size-table-virtual-scroll-strategy';
import { of, Subject, timer } from 'rxjs';
import { Subject } from 'rxjs';
export function _tableVirtualScrollDirectiveStrategyFactory(tableDir) {

@@ -53,3 +53,6 @@ return tableDir.scrollStrategy;

this.scrollStrategy.stickyChange
.pipe(filter(() => this.isStickyEnabled()), delayWhen(() => !this.stickyPositions ? timer(0) : of()), tap(() => {
.pipe(filter(() => this.isStickyEnabled()),
// breaks sticky header on the top. needs investigation
// delayWhen(() => !this.stickyPositions ? timer(0) : of()),
tap(() => {
if (!this.stickyPositions) {

@@ -143,2 +146,2 @@ this.initStickyPositions();

};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-item-size.directive.js","sourceRoot":"","sources":["../../../../projects/ng-table-virtual-scroll/src/lib/table-item-size.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterContentInit,\n  ContentChild,\n  Directive,\n  forwardRef,\n  Input,\n  NgZone,\n  OnChanges,\n  OnDestroy\n} from '@angular/core';\nimport { VIRTUAL_SCROLL_STRATEGY } from '@angular/cdk/scrolling';\nimport { delayWhen, distinctUntilChanged, filter, map, switchMap, takeUntil, tap } from 'rxjs/operators';\nimport { TableVirtualScrollDataSource } from './table-data-source';\nimport { MatTable } from '@angular/material/table';\nimport { FixedSizeTableVirtualScrollStrategy } from './fixed-size-table-virtual-scroll-strategy';\nimport { CdkHeaderRowDef } from '@angular/cdk/table';\nimport { of, Subject, timer } from 'rxjs';\n\nexport function _tableVirtualScrollDirectiveStrategyFactory(tableDir: TableItemSizeDirective) {\n  return tableDir.scrollStrategy;\n}\n\nconst stickyHeaderSelector = '.mat-header-row .mat-table-sticky';\nconst stickyFooterSelector = '.mat-footer-row .mat-table-sticky';\n\nconst defaults = {\n  rowHeight: 48,\n  headerHeight: 56,\n  headerEnabled: true,\n  footerHeight: 48,\n  footerEnabled: false,\n  bufferMultiplier: 0.7\n};\n\n@Directive({\n  selector: 'cdk-virtual-scroll-viewport[tvsItemSize]',\n  providers: [{\n    provide: VIRTUAL_SCROLL_STRATEGY,\n    useFactory: _tableVirtualScrollDirectiveStrategyFactory,\n    deps: [forwardRef(() => TableItemSizeDirective)]\n  }]\n})\nexport class TableItemSizeDirective implements OnChanges, AfterContentInit, OnDestroy {\n  private destroyed$ = new Subject();\n\n  // tslint:disable-next-line:no-input-rename\n  @Input('tvsItemSize')\n  rowHeight: string | number = defaults.rowHeight;\n\n  @Input()\n  headerEnabled: boolean = defaults.headerEnabled;\n\n  @Input()\n  headerHeight: string | number = defaults.headerHeight;\n\n  @Input()\n  footerEnabled: boolean = defaults.footerEnabled;\n\n  @Input()\n  footerHeight: string | number = defaults.footerHeight;\n\n  @Input()\n  bufferMultiplier: string | number = defaults.bufferMultiplier;\n\n  @ContentChild(MatTable, { static: false })\n  table: MatTable<any>;\n\n  scrollStrategy = new FixedSizeTableVirtualScrollStrategy();\n\n  dataSourceChanges = new Subject<void>();\n\n  private stickyPositions: Map<HTMLElement, number>;\n\n  constructor(private zone: NgZone) {\n  }\n\n  ngOnDestroy() {\n    this.destroyed$.next();\n    this.destroyed$.complete();\n    this.dataSourceChanges.complete();\n  }\n\n  private isStickyEnabled(): boolean {\n    return !!this.scrollStrategy.viewport && (this.table['_headerRowDefs'] as CdkHeaderRowDef[])\n      .map(def => def.sticky)\n      .reduce((prevState, state) => prevState && state, true);\n  }\n\n  ngAfterContentInit() {\n    const switchDataSourceOrigin = this.table['_switchDataSource'];\n    this.table['_switchDataSource'] = (dataSource: any) => {\n      switchDataSourceOrigin.call(this.table, dataSource);\n      this.connectDataSource(dataSource);\n    };\n\n    this.connectDataSource(this.table.dataSource);\n\n    this.scrollStrategy.stickyChange\n      .pipe(\n        filter(() => this.isStickyEnabled()),\n        delayWhen(() => !this.stickyPositions ? timer(0) : of()),\n        tap(() => {\n          if (!this.stickyPositions) {\n            this.initStickyPositions();\n          }\n        }),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe((stickyOffset) => {\n        this.setSticky(stickyOffset);\n      });\n  }\n\n  connectDataSource(dataSource: any) {\n    this.dataSourceChanges.next();\n    if (dataSource instanceof TableVirtualScrollDataSource) {\n      dataSource\n        .dataToRender$\n        .pipe(\n          distinctUntilChanged(),\n          takeUntil(this.dataSourceChanges),\n          takeUntil(this.destroyed$),\n          tap(data => this.scrollStrategy.dataLength = data.length),\n          switchMap(data =>\n            this.scrollStrategy\n              .renderedRangeStream\n              .pipe(\n                map(({\n                       start,\n                       end\n                     }) => typeof start !== 'number' || typeof end !== 'number' ? data : data.slice(start, end))\n              )\n          )\n        )\n        .subscribe(data => {\n          this.zone.run(() => {\n            dataSource.dataOfRange$.next(data);\n          });\n        });\n    } else {\n      throw new Error('[tvsItemSize] requires TableVirtualScrollDataSource be set as [dataSource] of [mat-table]');\n    }\n  }\n\n  ngOnChanges() {\n    const config = {\n      rowHeight: +this.rowHeight || defaults.rowHeight,\n      headerHeight: this.headerEnabled ? +this.headerHeight || defaults.headerHeight : 0,\n      footerHeight: this.footerEnabled ? +this.footerHeight || defaults.footerHeight : 0,\n      bufferMultiplier: +this.bufferMultiplier || defaults.bufferMultiplier\n    };\n    this.scrollStrategy.setConfig(config);\n  }\n\n\n  setSticky(offset: number) {\n    this.scrollStrategy.viewport.elementRef.nativeElement.querySelectorAll(stickyHeaderSelector)\n      .forEach((el: HTMLElement) => {\n        const parent = el.parentElement;\n        let baseOffset = 0;\n        if (this.stickyPositions.has(parent)) {\n          baseOffset = this.stickyPositions.get(parent);\n        }\n        el.style.top = `${baseOffset - offset}px`;\n      });\n    this.scrollStrategy.viewport.elementRef.nativeElement.querySelectorAll(stickyFooterSelector)\n      .forEach((el: HTMLElement) => {\n        const parent = el.parentElement;\n        let baseOffset = 0;\n        if (this.stickyPositions.has(parent)) {\n          baseOffset = this.stickyPositions.get(parent);\n        }\n        el.style.bottom = `${-baseOffset + offset}px`;\n      });\n  }\n\n  private initStickyPositions() {\n    this.stickyPositions = new Map<HTMLElement, number>();\n    this.scrollStrategy.viewport.elementRef.nativeElement.querySelectorAll(stickyHeaderSelector)\n      .forEach(el => {\n        const parent = el.parentElement;\n        if (!this.stickyPositions.has(parent)) {\n          this.stickyPositions.set(parent, parent.offsetTop);\n        }\n      });\n  }\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-item-size.directive.js","sourceRoot":"","sources":["../../../../projects/ng-table-virtual-scroll/src/lib/table-item-size.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterContentInit,\n  ContentChild,\n  Directive,\n  forwardRef,\n  Input,\n  NgZone,\n  OnChanges,\n  OnDestroy\n} from '@angular/core';\nimport { VIRTUAL_SCROLL_STRATEGY } from '@angular/cdk/scrolling';\nimport { delayWhen, distinctUntilChanged, filter, map, switchMap, takeUntil, tap } from 'rxjs/operators';\nimport { TableVirtualScrollDataSource } from './table-data-source';\nimport { MatTable } from '@angular/material/table';\nimport { FixedSizeTableVirtualScrollStrategy } from './fixed-size-table-virtual-scroll-strategy';\nimport { CdkHeaderRowDef } from '@angular/cdk/table';\nimport { of, Subject, timer } from 'rxjs';\n\nexport function _tableVirtualScrollDirectiveStrategyFactory(tableDir: TableItemSizeDirective) {\n  return tableDir.scrollStrategy;\n}\n\nconst stickyHeaderSelector = '.mat-header-row .mat-table-sticky';\nconst stickyFooterSelector = '.mat-footer-row .mat-table-sticky';\n\nconst defaults = {\n  rowHeight: 48,\n  headerHeight: 56,\n  headerEnabled: true,\n  footerHeight: 48,\n  footerEnabled: false,\n  bufferMultiplier: 0.7\n};\n\n@Directive({\n  selector: 'cdk-virtual-scroll-viewport[tvsItemSize]',\n  providers: [{\n    provide: VIRTUAL_SCROLL_STRATEGY,\n    useFactory: _tableVirtualScrollDirectiveStrategyFactory,\n    deps: [forwardRef(() => TableItemSizeDirective)]\n  }]\n})\nexport class TableItemSizeDirective implements OnChanges, AfterContentInit, OnDestroy {\n  private destroyed$ = new Subject();\n\n  // tslint:disable-next-line:no-input-rename\n  @Input('tvsItemSize')\n  rowHeight: string | number = defaults.rowHeight;\n\n  @Input()\n  headerEnabled: boolean = defaults.headerEnabled;\n\n  @Input()\n  headerHeight: string | number = defaults.headerHeight;\n\n  @Input()\n  footerEnabled: boolean = defaults.footerEnabled;\n\n  @Input()\n  footerHeight: string | number = defaults.footerHeight;\n\n  @Input()\n  bufferMultiplier: string | number = defaults.bufferMultiplier;\n\n  @ContentChild(MatTable, { static: false })\n  table: MatTable<any>;\n\n  scrollStrategy = new FixedSizeTableVirtualScrollStrategy();\n\n  dataSourceChanges = new Subject<void>();\n\n  private stickyPositions: Map<HTMLElement, number>;\n\n  constructor(private zone: NgZone) {\n  }\n\n  ngOnDestroy() {\n    this.destroyed$.next();\n    this.destroyed$.complete();\n    this.dataSourceChanges.complete();\n  }\n\n  private isStickyEnabled(): boolean {\n    return !!this.scrollStrategy.viewport && (this.table['_headerRowDefs'] as CdkHeaderRowDef[])\n      .map(def => def.sticky)\n      .reduce((prevState, state) => prevState && state, true);\n  }\n\n  ngAfterContentInit() {\n    const switchDataSourceOrigin = this.table['_switchDataSource'];\n    this.table['_switchDataSource'] = (dataSource: any) => {\n      switchDataSourceOrigin.call(this.table, dataSource);\n      this.connectDataSource(dataSource);\n    };\n\n    this.connectDataSource(this.table.dataSource);\n\n    this.scrollStrategy.stickyChange\n      .pipe(\n        filter(() => this.isStickyEnabled()),\n        // breaks sticky header on the top. needs investigation\n        // delayWhen(() => !this.stickyPositions ? timer(0) : of()),\n        tap(() => {\n          if (!this.stickyPositions) {\n            this.initStickyPositions();\n          }\n        }),\n        takeUntil(this.destroyed$)\n      )\n      .subscribe((stickyOffset) => {\n        this.setSticky(stickyOffset);\n      });\n  }\n\n  connectDataSource(dataSource: any) {\n    this.dataSourceChanges.next();\n    if (dataSource instanceof TableVirtualScrollDataSource) {\n      dataSource\n        .dataToRender$\n        .pipe(\n          distinctUntilChanged(),\n          takeUntil(this.dataSourceChanges),\n          takeUntil(this.destroyed$),\n          tap(data => this.scrollStrategy.dataLength = data.length),\n          switchMap(data =>\n            this.scrollStrategy\n              .renderedRangeStream\n              .pipe(\n                map(({\n                       start,\n                       end\n                     }) => typeof start !== 'number' || typeof end !== 'number' ? data : data.slice(start, end))\n              )\n          )\n        )\n        .subscribe(data => {\n          this.zone.run(() => {\n            dataSource.dataOfRange$.next(data);\n          });\n        });\n    } else {\n      throw new Error('[tvsItemSize] requires TableVirtualScrollDataSource be set as [dataSource] of [mat-table]');\n    }\n  }\n\n  ngOnChanges() {\n    const config = {\n      rowHeight: +this.rowHeight || defaults.rowHeight,\n      headerHeight: this.headerEnabled ? +this.headerHeight || defaults.headerHeight : 0,\n      footerHeight: this.footerEnabled ? +this.footerHeight || defaults.footerHeight : 0,\n      bufferMultiplier: +this.bufferMultiplier || defaults.bufferMultiplier\n    };\n    this.scrollStrategy.setConfig(config);\n  }\n\n\n  setSticky(offset: number) {\n    this.scrollStrategy.viewport.elementRef.nativeElement.querySelectorAll(stickyHeaderSelector)\n      .forEach((el: HTMLElement) => {\n        const parent = el.parentElement;\n        let baseOffset = 0;\n        if (this.stickyPositions.has(parent)) {\n          baseOffset = this.stickyPositions.get(parent);\n        }\n        el.style.top = `${baseOffset - offset}px`;\n      });\n    this.scrollStrategy.viewport.elementRef.nativeElement.querySelectorAll(stickyFooterSelector)\n      .forEach((el: HTMLElement) => {\n        const parent = el.parentElement;\n        let baseOffset = 0;\n        if (this.stickyPositions.has(parent)) {\n          baseOffset = this.stickyPositions.get(parent);\n        }\n        el.style.bottom = `${-baseOffset + offset}px`;\n      });\n  }\n\n  private initStickyPositions() {\n    this.stickyPositions = new Map<HTMLElement, number>();\n    this.scrollStrategy.viewport.elementRef.nativeElement.querySelectorAll(stickyHeaderSelector)\n      .forEach(el => {\n        const parent = el.parentElement;\n        if (!this.stickyPositions.has(parent)) {\n          this.stickyPositions.set(parent, parent.offsetTop);\n        }\n      });\n  }\n}\n"]}
import { Injectable, Directive, forwardRef, NgZone, Input, ContentChild, NgModule } from '@angular/core';
import { VIRTUAL_SCROLL_STRATEGY } from '@angular/cdk/scrolling';
import { map, distinctUntilChanged, filter, delayWhen, tap, takeUntil, switchMap } from 'rxjs/operators';
import { merge, of, combineLatest, Subscription, ReplaySubject, Subject, BehaviorSubject, timer } from 'rxjs';
import { map, distinctUntilChanged, filter, tap, takeUntil, switchMap } from 'rxjs/operators';
import { merge, of, combineLatest, Subscription, ReplaySubject, Subject, BehaviorSubject } from 'rxjs';
import { MatTableDataSource, MatTable } from '@angular/material/table';

@@ -78,3 +78,2 @@

onContentRendered() {
// no-op
}

@@ -108,15 +107,36 @@ onRenderedOffsetChanged() {

}
const renderedOffset = this.viewport.getOffsetToRenderedContentStart();
const start = renderedOffset / this.rowHeight;
const itemsDisplayed = Math.ceil(this.viewport.getViewportSize() / this.rowHeight);
const bufferItems = Math.ceil(itemsDisplayed * this.bufferMultiplier);
const end = start + itemsDisplayed + 2 * bufferItems;
const bufferOffset = renderedOffset + bufferItems * this.rowHeight;
const scrollOffset = this.viewport.measureScrollOffset();
const amount = Math.ceil(this.viewport.getViewportSize() / this.rowHeight);
const offset = Math.max(scrollOffset - this.headerHeight, 0);
const buffer = Math.ceil(amount * this.bufferMultiplier);
const skip = Math.round(offset / this.rowHeight);
const index = Math.max(0, skip);
const start = Math.max(0, index - buffer);
const end = Math.min(this.dataLength, index + amount + buffer);
const renderedOffset = start * this.rowHeight;
this.viewport.setRenderedContentOffset(renderedOffset);
this.viewport.setRenderedRange({ start, end });
this.indexChange.next(index);
this.stickyChange.next(renderedOffset);
// How far the scroll offset is from the lower buffer, which is usually where items start being displayed
const relativeScrollOffset = scrollOffset - bufferOffset;
const rowsScrolled = relativeScrollOffset / this.rowHeight;
const displayed = scrollOffset / this.rowHeight;
this.indexChange.next(displayed);
// Only bother updating the displayed information if we've scrolled more than a row
const rowSensitivity = 1.0;
if (Math.abs(rowsScrolled) < rowSensitivity) {
this.viewport.setRenderedContentOffset(renderedOffset);
this.viewport.setRenderedRange({ start, end });
return;
}
// Special case for the start of the table.
// At the top of the table, the first few rows are first rendered because they're visible, and then still rendered
// Because they move into the buffer. So we only need to change what's rendered once the user scrolls far enough down.
if (renderedOffset === 0 && rowsScrolled < 0) {
this.viewport.setRenderedContentOffset(renderedOffset);
this.viewport.setRenderedRange({ start, end });
return;
}
const rowsToMove = Math.sign(rowsScrolled) * Math.floor(Math.abs(rowsScrolled));
const adjustedRenderedOffset = Math.max(0, renderedOffset + rowsToMove * this.rowHeight);
this.viewport.setRenderedContentOffset(adjustedRenderedOffset);
const adjustedStart = Math.max(0, start + rowsToMove);
const adjustedEnd = adjustedStart + itemsDisplayed + 2 * bufferItems;
this.viewport.setRenderedRange({ start: adjustedStart, end: adjustedEnd });
this.stickyChange.next(adjustedRenderedOffset);
}

@@ -173,3 +193,6 @@ }

this.scrollStrategy.stickyChange
.pipe(filter(() => this.isStickyEnabled()), delayWhen(() => !this.stickyPositions ? timer(0) : of()), tap(() => {
.pipe(filter(() => this.isStickyEnabled()),
// breaks sticky header on the top. needs investigation
// delayWhen(() => !this.stickyPositions ? timer(0) : of()),
tap(() => {
if (!this.stickyPositions) {

@@ -176,0 +199,0 @@ this.initStickyPositions();

{
"name": "ng-table-virtual-scroll",
"version": "1.3.6",
"version": "1.3.7",
"description": "Virtual scroll for for Angular Material Table",

@@ -5,0 +5,0 @@ "homepage": "https://github.com/diprokon/ng-table-virtual-scroll",

Sorry, the diff of this file is not supported yet

Sorry, the diff of this file is not supported yet

Sorry, the diff of this file is not supported yet

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc