diff --git a/src/waypoint.jsx b/src/waypoint.jsx index 0146444..f2ac7b6 100644 --- a/src/waypoint.jsx +++ b/src/waypoint.jsx @@ -42,9 +42,11 @@ export default class Waypoint extends React.Component { } // this._ref may occasionally not be set at this time. To help ensure that - // this works smoothly, we want to delay the initial execution until the - // next tick. - this.cancelInitialTimeout = onNextTick(() => { + // this works smoothly and to avoid layout thrashing, we want to delay the + // initial execution until the next tick. + this.cancelOnNextTick = onNextTick(() => { + this.cancelOnNextTick = null; + // Berofe doing anything, we want to check that this._ref is avaliable in Waypoint ensureRefIsUsedByChild(this.props.children, this._ref); @@ -87,8 +89,21 @@ export default class Waypoint extends React.Component { return; } - // The element may have moved. - this._handleScroll(null); + // The element may have moved, so we need to recompute its position on the + // page. This happens via handleScroll in a way that forces layout to be + // computed. + // + // We want this to be deferred to avoid forcing layout during render, which + // causes layout thrashing. And, if we already have this work enqueued, we + // can just wait for that to happen instead of enqueueing again. + if (this.cancelOnNextTick) { + return; + } + + this.cancelOnNextTick = onNextTick(() => { + this.cancelOnNextTick = null; + this._handleScroll(null); + }); } componentWillUnmount() { @@ -103,8 +118,8 @@ export default class Waypoint extends React.Component { this.resizeEventListenerUnsubscribe(); } - if (this.cancelInitialTimeout) { - this.cancelInitialTimeout(); + if (this.cancelOnNextTick) { + this.cancelOnNextTick(); } }