You're Invited:Meet the Socket Team at BlackHat and DEF CON in Las Vegas, Aug 4-6.RSVP
Socket
Book a DemoInstallSign in
Socket

react-scroll-to-bottom

Package Overview
Dependencies
Maintainers
1
Versions
83
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

react-scroll-to-bottom - npm Package Compare versions

Comparing version

to
1.3.1-master.1fa3dd1

6

CHANGELOG.md

@@ -8,2 +8,4 @@ # Changelog

## [Unreleased]
### Changed
- `Composer`: fix #13, user scrolling in Firefox may have the scroll position locked occasionally, in PR [#12](https://github.com/compulim/react-scroll-to-bottom/issues/12)

@@ -13,4 +15,4 @@ ## [1.3.0] - 2019-01-21

- Playground: bumped to `react@16.6.0`, `react-dom@16.6.0`, and `react-scripts@2.1.6`
- Update algorithm, instead of using `componentDidUpdate`, we now use `setInterval` to check if the panel is sticky or not, this help to track content update that happen outside of React lifecycle, for example, `HTMLImageElement.onload` event
- `scrollTo()` now accepts `"100%"` instead of `"bottom"`
- `*`: Update algorithm, instead of using `componentDidUpdate`, we now use `setInterval` to check if the panel is sticky or not, this help to track content update that happen outside of React lifecycle, for example, `HTMLImageElement.onload` event
- `Composer`: `scrollTo()` now accepts `"100%"` instead of `"bottom"`

@@ -17,0 +19,0 @@ ### Removed

@@ -50,4 +50,6 @@ "use strict";

var MIN_CHECK_INTERVAL = 17;
var MIN_CHECK_INTERVAL = 17; // 1 frame
var SCROLL_DECISION_DURATION = 34; // 2 frames
function setImmediateInterval(fn, ms) {

@@ -164,2 +166,3 @@ fn();

clearInterval(this._stickyCheckTimeout);
var stickyButNotAtEndSince = false;
this._stickyCheckTimeout = setImmediateInterval(function () {

@@ -170,7 +173,17 @@ var state = _this2.state;

if (sticky && target) {
var _computeViewState = computeViewState(state),
atEnd = _computeViewState.atEnd;
!atEnd && state.functionContext.scrollToEnd();
if (sticky && target && !computeViewState(state).atEnd) {
if (!stickyButNotAtEndSince) {
stickyButNotAtEndSince = Date.now();
} else if (Date.now() - stickyButNotAtEndSince > SCROLL_DECISION_DURATION) {
// Quirks: In Firefox, after user scroll down, Firefox do two things:
// 1. Set to a new "scrollTop"
// 2. Fire "scroll" event
// For what we observed, #1 is fired about 20ms before #2. There is a chance that this stickyCheckTimeout is being scheduled between 1 and 2.
// That means, if we just look at #1 to decide if we should scroll, we will always scroll, in oppose to the user's intention.
// Repro: Open Firefox, set checkInterval to a lower number, and try to scroll by dragging the scroll handler. It will jump back.
state.functionContext.scrollToEnd();
stickyButNotAtEndSince = false;
}
} else {
stickyButNotAtEndSince = false;
}

@@ -221,7 +234,7 @@ }, Math.max(MIN_CHECK_INTERVAL, this.props.checkInterval) || MIN_CHECK_INTERVAL);

var _computeViewState2 = computeViewState(state),
atBottom = _computeViewState2.atBottom,
atEnd = _computeViewState2.atEnd,
atStart = _computeViewState2.atStart,
atTop = _computeViewState2.atTop;
var _computeViewState = computeViewState(state),
atBottom = _computeViewState.atBottom,
atEnd = _computeViewState.atEnd,
atStart = _computeViewState.atStart,
atTop = _computeViewState.atTop;

@@ -315,3 +328,3 @@ var nextStateContext = stateContext;

Composer.defaultProps = {
checkInterval: 150,
checkInterval: 100,
debounce: 17

@@ -323,2 +336,2 @@ };

};
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/ScrollToBottom/Composer.js"],"names":["MIN_CHECK_INTERVAL","setImmediateInterval","fn","ms","setInterval","computeViewState","mode","stateContext","target","offsetHeight","scrollHeight","scrollTop","atBottom","atTop","atEnd","atStart","Composer","props","handleScroll","bind","handleScrollEnd","_ignoreScrollEventBefore","state","functionContext","scrollTo","setState","scrollToBottom","scrollToEnd","scrollToTop","scrollToStart","internalContext","setTarget","animating","sticky","enableWorker","clearInterval","_stickyCheckTimeout","Math","max","checkInterval","disableWorker","nextProps","timeStampLow","nextStateContext","Date","now","children","debounce","React","Component","defaultProps","propTypes","PropTypes","number"],"mappings":"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","sourcesContent":["import memoize from 'memoize-one';\nimport PropTypes from 'prop-types';\nimport React from 'react';\nimport updateIn from 'simple-update-in';\n\nimport EventSpy from '../EventSpy';\nimport FunctionContext from './FunctionContext';\nimport InternalContext from './InternalContext';\nimport SpineTo from '../SpineTo';\nimport StateContext from './StateContext';\n\nconst MIN_CHECK_INTERVAL = 17;\n\nfunction setImmediateInterval(fn, ms) {\n  fn();\n\n  return setInterval(fn, ms);\n}\n\nfunction computeViewState({ stateContext: { mode }, target: { offsetHeight, scrollHeight, scrollTop } }) {\n  const atBottom = scrollHeight - scrollTop - offsetHeight <= 0;\n  const atTop = scrollTop <= 0;\n  const atEnd = mode === 'top' ? atTop : atBottom;\n\n  return {\n    atBottom,\n    atEnd,\n    atStart: !atEnd,\n    atTop\n  };\n}\n\nexport default class Composer extends React.Component {\n  constructor(props) {\n    super(props);\n\n    this.handleScroll = this.handleScroll.bind(this);\n    this.handleScrollEnd = this.handleScrollEnd.bind(this);\n\n    this._ignoreScrollEventBefore = 0;\n\n    this.state = {\n      functionContext: {\n        scrollTo: scrollTop => this.setState(({ stateContext }) => ({\n          scrollTop,\n          stateContext: updateIn(stateContext, ['animating'], () => true)\n        })),\n        scrollToBottom: () => this.state.functionContext.scrollTo('100%'),\n        scrollToEnd: () => {\n          const { state: { functionContext, stateContext } } = this;\n\n          stateContext.mode === 'top' ? functionContext.scrollToTop() : functionContext.scrollToBottom();\n        },\n        scrollToStart: () => {\n          const { state: { functionContext, stateContext } } = this;\n\n          stateContext.mode === 'top' ? functionContext.scrollToBottom() : functionContext.scrollToTop();\n        },\n        scrollToTop: () => this.state.functionContext.scrollTo(0)\n      },\n      internalContext: {\n        setTarget: target => this.setState(() => ({ target }))\n      },\n      scrollTop: props.mode === 'top' ? 0 : '100%',\n      stateContext: {\n        animating: false,\n        atBottom: true,\n        atEnd: true,\n        atTop: true,\n        mode: props.mode,\n        sticky: true\n      },\n      target: null\n    };\n  }\n\n  componentDidMount() {\n    this.enableWorker();\n  }\n\n  disableWorker() {\n    clearInterval(this._stickyCheckTimeout);\n  }\n\n  enableWorker() {\n    clearInterval(this._stickyCheckTimeout);\n\n    this._stickyCheckTimeout = setImmediateInterval(\n      () => {\n        const { state } = this;\n        const { stateContext: { sticky }, target } = state;\n\n        if (sticky && target) {\n          const { atEnd } = computeViewState(state);\n\n          !atEnd && state.functionContext.scrollToEnd();\n        }\n      },\n      Math.max(MIN_CHECK_INTERVAL, this.props.checkInterval) || MIN_CHECK_INTERVAL\n    );\n  }\n\n  componentWillUnmount() {\n    this.disableWorker();\n  }\n\n  componentWillReceiveProps(nextProps) {\n    this.setState(({ stateContext }) => ({\n      stateContext: {\n        ...stateContext,\n        mode: nextProps.mode === 'top' ? 'top' : 'bottom'\n      }\n    }));\n  }\n\n  handleScroll({ timeStampLow }) {\n    // Currently, there are no reliable way to check if the \"scroll\" event is trigger due to\n    // user gesture, programmatic scrolling, or Chrome-synthesized \"scroll\" event to compensate size change.\n    // Thus, we use our best-effort to guess if it is triggered by user gesture, and disable sticky if it is heading towards the start direction.\n\n    if (timeStampLow <= this._ignoreScrollEventBefore) {\n      // Since we debounce \"scroll\" event, this handler might be called after spineTo.onEnd (a.k.a. artificial scrolling).\n      // We should ignore debounced event fired after scrollEnd, because without skipping them, the userInitiatedScroll calculated below will not be accurate.\n      // Thus, on a fast machine, adding elements super fast will lose the \"stickiness\".\n\n      return;\n    }\n\n    this.disableWorker();\n\n    this.setState(state => {\n      const { target } = state;\n\n      if (target) {\n        const { scrollTop, stateContext } = state;\n        const { atBottom, atEnd, atStart, atTop } = computeViewState(state);\n        let nextStateContext = stateContext;\n\n        nextStateContext = updateIn(nextStateContext, ['atBottom'], () => atBottom);\n        nextStateContext = updateIn(nextStateContext, ['atEnd'], () => atEnd);\n        nextStateContext = updateIn(nextStateContext, ['atStart'], () => atStart);\n        nextStateContext = updateIn(nextStateContext, ['atTop'], () => atTop);\n\n        // Sticky means:\n        // - If it is scrolled programatically, we are still in sticky mode\n        // - If it is scrolled by the user, then sticky means if we are at the end\n        nextStateContext = updateIn(nextStateContext, ['sticky'], () => stateContext.animating ? true : atEnd);\n\n        // If no scrollTop is set (not in programmatic scrolling mode), we should set \"animating\" to false\n        // \"animating\" is used to calculate the \"sticky\" property\n        if (scrollTop === null) {\n          nextStateContext = updateIn(nextStateContext, ['animating'], () => false);\n        }\n\n        if (stateContext !== nextStateContext) {\n          return { stateContext: nextStateContext };\n        }\n      }\n    }, () => {\n      this.state.stateContext.sticky && this.enableWorker();\n    });\n  }\n\n  handleScrollEnd() {\n    // We should ignore debouncing handleScroll that emit before this time\n    this._ignoreScrollEventBefore = Date.now();\n\n    this.setState(() => ({ scrollTop: null }));\n  }\n\n  render() {\n    const {\n      handleScroll,\n      handleScrollEnd,\n      props: { children, debounce },\n      state: { functionContext, internalContext, scrollTop, stateContext, target }\n    } = this;\n\n    return (\n      <InternalContext.Provider value={ internalContext }>\n        <FunctionContext.Provider value={ functionContext }>\n          <StateContext.Provider value={ stateContext }>\n            { children }\n            {\n              target &&\n                <EventSpy\n                  debounce={ debounce }\n                  name=\"scroll\"\n                  onEvent={ handleScroll }\n                  target={ target }\n                />\n            }\n            {\n              target && scrollTop !== null &&\n                <SpineTo\n                  name=\"scrollTop\"\n                  onEnd={ handleScrollEnd }\n                  target={ target }\n                  value={ scrollTop }\n                />\n            }\n          </StateContext.Provider>\n        </FunctionContext.Provider>\n      </InternalContext.Provider>\n    );\n  }\n}\n\nComposer.defaultProps = {\n  checkInterval: 150,\n  debounce: 17\n};\n\nComposer.propTypes = {\n  checkInterval: PropTypes.number,\n  debounce: PropTypes.number\n};\n"]}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/ScrollToBottom/Composer.js"],"names":["MIN_CHECK_INTERVAL","SCROLL_DECISION_DURATION","setImmediateInterval","fn","ms","setInterval","computeViewState","mode","stateContext","target","offsetHeight","scrollHeight","scrollTop","atBottom","atTop","atEnd","atStart","Composer","props","handleScroll","bind","handleScrollEnd","_ignoreScrollEventBefore","state","functionContext","scrollTo","setState","scrollToBottom","scrollToEnd","scrollToTop","scrollToStart","internalContext","setTarget","animating","sticky","enableWorker","clearInterval","_stickyCheckTimeout","stickyButNotAtEndSince","Date","now","Math","max","checkInterval","disableWorker","nextProps","timeStampLow","nextStateContext","children","debounce","React","Component","defaultProps","propTypes","PropTypes","number"],"mappings":"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","sourcesContent":["import memoize from 'memoize-one';\nimport PropTypes from 'prop-types';\nimport React from 'react';\nimport updateIn from 'simple-update-in';\n\nimport EventSpy from '../EventSpy';\nimport FunctionContext from './FunctionContext';\nimport InternalContext from './InternalContext';\nimport SpineTo from '../SpineTo';\nimport StateContext from './StateContext';\n\nconst MIN_CHECK_INTERVAL = 17;       // 1 frame\nconst SCROLL_DECISION_DURATION = 34; // 2 frames\n\nfunction setImmediateInterval(fn, ms) {\n  fn();\n\n  return setInterval(fn, ms);\n}\n\nfunction computeViewState({ stateContext: { mode }, target: { offsetHeight, scrollHeight, scrollTop } }) {\n  const atBottom = scrollHeight - scrollTop - offsetHeight <= 0;\n  const atTop = scrollTop <= 0;\n  const atEnd = mode === 'top' ? atTop : atBottom;\n\n  return {\n    atBottom,\n    atEnd,\n    atStart: !atEnd,\n    atTop\n  };\n}\n\nexport default class Composer extends React.Component {\n  constructor(props) {\n    super(props);\n\n    this.handleScroll = this.handleScroll.bind(this);\n    this.handleScrollEnd = this.handleScrollEnd.bind(this);\n\n    this._ignoreScrollEventBefore = 0;\n\n    this.state = {\n      functionContext: {\n        scrollTo: scrollTop => this.setState(({ stateContext }) => ({\n          scrollTop,\n          stateContext: updateIn(stateContext, ['animating'], () => true)\n        })),\n        scrollToBottom: () => this.state.functionContext.scrollTo('100%'),\n        scrollToEnd: () => {\n          const { state: { functionContext, stateContext } } = this;\n\n          stateContext.mode === 'top' ? functionContext.scrollToTop() : functionContext.scrollToBottom();\n        },\n        scrollToStart: () => {\n          const { state: { functionContext, stateContext } } = this;\n\n          stateContext.mode === 'top' ? functionContext.scrollToBottom() : functionContext.scrollToTop();\n        },\n        scrollToTop: () => this.state.functionContext.scrollTo(0)\n      },\n      internalContext: {\n        setTarget: target => this.setState(() => ({ target }))\n      },\n      scrollTop: props.mode === 'top' ? 0 : '100%',\n      stateContext: {\n        animating: false,\n        atBottom: true,\n        atEnd: true,\n        atTop: true,\n        mode: props.mode,\n        sticky: true\n      },\n      target: null\n    };\n  }\n\n  componentDidMount() {\n    this.enableWorker();\n  }\n\n  disableWorker() {\n    clearInterval(this._stickyCheckTimeout);\n  }\n\n  enableWorker() {\n    clearInterval(this._stickyCheckTimeout);\n\n    let stickyButNotAtEndSince = false;\n\n    this._stickyCheckTimeout = setImmediateInterval(\n      () => {\n        const { state } = this;\n        const { stateContext: { sticky }, target } = state;\n\n        if (\n          sticky\n          && target\n          && !computeViewState(state).atEnd\n        ) {\n          if (!stickyButNotAtEndSince) {\n            stickyButNotAtEndSince = Date.now();\n          } else if (Date.now() - stickyButNotAtEndSince > SCROLL_DECISION_DURATION) {\n            // Quirks: In Firefox, after user scroll down, Firefox do two things:\n            //         1. Set to a new \"scrollTop\"\n            //         2. Fire \"scroll\" event\n            //         For what we observed, #1 is fired about 20ms before #2. There is a chance that this stickyCheckTimeout is being scheduled between 1 and 2.\n            //         That means, if we just look at #1 to decide if we should scroll, we will always scroll, in oppose to the user's intention.\n            // Repro: Open Firefox, set checkInterval to a lower number, and try to scroll by dragging the scroll handler. It will jump back.\n\n            state.functionContext.scrollToEnd();\n            stickyButNotAtEndSince = false;\n          }\n        } else {\n          stickyButNotAtEndSince = false;\n        }\n      },\n      Math.max(MIN_CHECK_INTERVAL, this.props.checkInterval) || MIN_CHECK_INTERVAL\n    );\n  }\n\n  componentWillUnmount() {\n    this.disableWorker();\n  }\n\n  componentWillReceiveProps(nextProps) {\n    this.setState(({ stateContext }) => ({\n      stateContext: {\n        ...stateContext,\n        mode: nextProps.mode === 'top' ? 'top' : 'bottom'\n      }\n    }));\n  }\n\n  handleScroll({ timeStampLow }) {\n    // Currently, there are no reliable way to check if the \"scroll\" event is trigger due to\n    // user gesture, programmatic scrolling, or Chrome-synthesized \"scroll\" event to compensate size change.\n    // Thus, we use our best-effort to guess if it is triggered by user gesture, and disable sticky if it is heading towards the start direction.\n\n    if (timeStampLow <= this._ignoreScrollEventBefore) {\n      // Since we debounce \"scroll\" event, this handler might be called after spineTo.onEnd (a.k.a. artificial scrolling).\n      // We should ignore debounced event fired after scrollEnd, because without skipping them, the userInitiatedScroll calculated below will not be accurate.\n      // Thus, on a fast machine, adding elements super fast will lose the \"stickiness\".\n\n      return;\n    }\n\n    this.disableWorker();\n\n    this.setState(state => {\n      const { target } = state;\n\n      if (target) {\n        const { scrollTop, stateContext } = state;\n        const { atBottom, atEnd, atStart, atTop } = computeViewState(state);\n        let nextStateContext = stateContext;\n\n        nextStateContext = updateIn(nextStateContext, ['atBottom'], () => atBottom);\n        nextStateContext = updateIn(nextStateContext, ['atEnd'], () => atEnd);\n        nextStateContext = updateIn(nextStateContext, ['atStart'], () => atStart);\n        nextStateContext = updateIn(nextStateContext, ['atTop'], () => atTop);\n\n        // Sticky means:\n        // - If it is scrolled programatically, we are still in sticky mode\n        // - If it is scrolled by the user, then sticky means if we are at the end\n        nextStateContext = updateIn(nextStateContext, ['sticky'], () => stateContext.animating ? true : atEnd);\n\n        // If no scrollTop is set (not in programmatic scrolling mode), we should set \"animating\" to false\n        // \"animating\" is used to calculate the \"sticky\" property\n        if (scrollTop === null) {\n          nextStateContext = updateIn(nextStateContext, ['animating'], () => false);\n        }\n\n        if (stateContext !== nextStateContext) {\n          return { stateContext: nextStateContext };\n        }\n      }\n    }, () => {\n      this.state.stateContext.sticky && this.enableWorker();\n    });\n  }\n\n  handleScrollEnd() {\n    // We should ignore debouncing handleScroll that emit before this time\n    this._ignoreScrollEventBefore = Date.now();\n\n    this.setState(() => ({ scrollTop: null }));\n  }\n\n  render() {\n    const {\n      handleScroll,\n      handleScrollEnd,\n      props: { children, debounce },\n      state: { functionContext, internalContext, scrollTop, stateContext, target }\n    } = this;\n\n    return (\n      <InternalContext.Provider value={ internalContext }>\n        <FunctionContext.Provider value={ functionContext }>\n          <StateContext.Provider value={ stateContext }>\n            { children }\n            {\n              target &&\n                <EventSpy\n                  debounce={ debounce }\n                  name=\"scroll\"\n                  onEvent={ handleScroll }\n                  target={ target }\n                />\n            }\n            {\n              target && scrollTop !== null &&\n                <SpineTo\n                  name=\"scrollTop\"\n                  onEnd={ handleScrollEnd }\n                  target={ target }\n                  value={ scrollTop }\n                />\n            }\n          </StateContext.Provider>\n        </FunctionContext.Provider>\n      </InternalContext.Provider>\n    );\n  }\n}\n\nComposer.defaultProps = {\n  checkInterval: 100,\n  debounce: 17\n};\n\nComposer.propTypes = {\n  checkInterval: PropTypes.number,\n  debounce: PropTypes.number\n};\n"]}
{
"name": "react-scroll-to-bottom",
"version": "1.3.0",
"version": "1.3.1-master.1fa3dd1",
"description": "React container that will auto scroll to bottom",

@@ -5,0 +5,0 @@ "keywords": [