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.2-master.d0df2f1

4

CHANGELOG.md

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

## [Unreleased]
### Changed
- `*`: bumped to `babel-jest@24.8.0`, `lerna@3.15.0`, and `jest@24.8.0`, in PR [#22](https://github.com/compulim/react-scroll-to-bottom/pull/22)
### Fixed
- `Composer`: fix [#22](https://github.com/compulim/react-scroll-to-bottom/issue/22), synthetic `scroll` events crafted by Chrome should not cause stickiness to lose, in PR [#23](https://github.com/compulim/react-scroll-to-bottom/issue/23)

@@ -10,0 +14,0 @@ ## [1.3.1] - 2019-02-13

51

lib/ScrollToBottom/Composer.js

@@ -126,2 +126,4 @@ "use strict";

internalContext: {
offsetHeight: 0,
scrollHeight: 0,
setTarget: function setTarget(target) {

@@ -228,3 +230,4 @@ return _this.setState(function () {

if (target) {
var scrollTop = state.scrollTop,
var internalContext = state.internalContext,
scrollTop = state.scrollTop,
stateContext = state.stateContext;

@@ -238,2 +241,3 @@

var nextInternalContext = internalContext;
var nextStateContext = stateContext;

@@ -251,11 +255,36 @@ nextStateContext = (0, _simpleUpdateIn.default)(nextStateContext, ['atBottom'], function () {

return atTop;
}); // Sticky means:
}); // Chrome will emit "synthetic" scroll event if the container is resized or an element is added
// We need to ignore these "synthetic" events
// Repro: In playground, press 4-1-5-1-1 (small, add one, normal, add one, add one)
// Nomatter how fast or slow the sequence is being presssed, it should still stick to the bottom
var offsetHeight = target.offsetHeight,
scrollHeight = target.scrollHeight;
var resized = offsetHeight !== internalContext.offsetHeight;
var elementChanged = scrollHeight !== internalContext.scrollHeight;
if (resized) {
nextInternalContext = (0, _simpleUpdateIn.default)(nextInternalContext, ['offsetHeight'], function () {
return offsetHeight;
});
}
if (elementChanged) {
nextInternalContext = (0, _simpleUpdateIn.default)(nextInternalContext, ['scrollHeight'], function () {
return scrollHeight;
});
} // Sticky means:
// - If it is scrolled programatically, we are still in sticky mode
// - If it is scrolled by the user, then sticky means if we are at the end
// Only update stickiness if the scroll event is not due to synthetic scroll done by Chrome
nextStateContext = (0, _simpleUpdateIn.default)(nextStateContext, ['sticky'], function () {
return stateContext.animating ? true : atEnd;
}); // If no scrollTop is set (not in programmatic scrolling mode), we should set "animating" to false
if (!resized && !elementChanged) {
nextStateContext = (0, _simpleUpdateIn.default)(nextStateContext, ['sticky'], function () {
return stateContext.animating ? true : atEnd;
});
} // If no scrollTop is set (not in programmatic scrolling mode), we should set "animating" to false
// "animating" is used to calculate the "sticky" property
if (scrollTop === null) {

@@ -267,7 +296,7 @@ nextStateContext = (0, _simpleUpdateIn.default)(nextStateContext, ['animating'], function () {

if (stateContext !== nextStateContext) {
return {
stateContext: nextStateContext
};
}
return _objectSpread({}, internalContext === nextInternalContext ? {} : {
internalContext: nextInternalContext
}, stateContext === nextStateContext ? {} : {
stateContext: nextStateContext
});
}

@@ -335,2 +364,2 @@ }, function () {

};
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/ScrollToBottom/Composer.js"],"names":["MIN_CHECK_INTERVAL","NEAR_END_THRESHOLD","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 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 NEAR_END_THRESHOLD = 1;\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 < NEAR_END_THRESHOLD;\n  const atTop = scrollTop < NEAR_END_THRESHOLD;\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"]}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/ScrollToBottom/Composer.js"],"names":["MIN_CHECK_INTERVAL","NEAR_END_THRESHOLD","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","nextInternalContext","nextStateContext","resized","elementChanged","children","debounce","React","Component","defaultProps","propTypes","PropTypes","number"],"mappings":"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","sourcesContent":["import 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 NEAR_END_THRESHOLD = 1;\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 < NEAR_END_THRESHOLD;\n  const atTop = scrollTop < NEAR_END_THRESHOLD;\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        offsetHeight: 0,\n        scrollHeight: 0,\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 { internalContext, scrollTop, stateContext } = state;\n        const { atBottom, atEnd, atStart, atTop } = computeViewState(state);\n        let nextInternalContext = internalContext;\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        // Chrome will emit \"synthetic\" scroll event if the container is resized or an element is added\n        // We need to ignore these \"synthetic\" events\n        // Repro: In playground, press 4-1-5-1-1 (small, add one, normal, add one, add one)\n        //        Nomatter how fast or slow the sequence is being presssed, it should still stick to the bottom\n        const { offsetHeight, scrollHeight } = target;\n        const resized = offsetHeight !== internalContext.offsetHeight;\n        const elementChanged = scrollHeight !== internalContext.scrollHeight;\n\n        if (resized) {\n          nextInternalContext = updateIn(nextInternalContext, ['offsetHeight'], () => offsetHeight);\n        }\n\n        if (elementChanged) {\n          nextInternalContext = updateIn(nextInternalContext, ['scrollHeight'], () => scrollHeight);\n        }\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\n        // Only update stickiness if the scroll event is not due to synthetic scroll done by Chrome\n        if (!resized && !elementChanged) {\n          nextStateContext = updateIn(nextStateContext, ['sticky'], () => stateContext.animating ? true : atEnd);\n        }\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        return {\n          ...internalContext === nextInternalContext ? {} : { internalContext: nextInternalContext },\n          ...stateContext === nextStateContext ? {} : { 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"]}

@@ -22,2 +22,5 @@ "use strict";

},
scrollToStart: function scrollToStart() {
return 0;
},
scrollToTop: function scrollToTop() {

@@ -31,2 +34,2 @@ return 0;

exports.default = _default;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9TY3JvbGxUb0JvdHRvbS9GdW5jdGlvbkNvbnRleHQuanMiXSwibmFtZXMiOlsiY29udGV4dCIsIlJlYWN0IiwiY3JlYXRlQ29udGV4dCIsInNjcm9sbFRvIiwic2Nyb2xsVG9Cb3R0b20iLCJzY3JvbGxUb0VuZCIsInNjcm9sbFRvVG9wIiwiZGlzcGxheU5hbWUiXSwibWFwcGluZ3MiOiI7Ozs7Ozs7QUFBQTs7OztBQUVBLElBQU1BLE9BQU8sR0FBR0MsZUFBTUMsYUFBTixDQUFvQjtBQUNsQ0MsRUFBQUEsUUFBUSxFQUFFO0FBQUEsV0FBTSxDQUFOO0FBQUEsR0FEd0I7QUFFbENDLEVBQUFBLGNBQWMsRUFBRTtBQUFBLFdBQU0sQ0FBTjtBQUFBLEdBRmtCO0FBR2xDQyxFQUFBQSxXQUFXLEVBQUU7QUFBQSxXQUFNLENBQU47QUFBQSxHQUhxQjtBQUlsQ0MsRUFBQUEsV0FBVyxFQUFFO0FBQUEsV0FBTSxDQUFOO0FBQUE7QUFKcUIsQ0FBcEIsQ0FBaEI7O0FBT0FOLE9BQU8sQ0FBQ08sV0FBUixHQUFzQiwrQkFBdEI7ZUFFZVAsTyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCc7XG5cbmNvbnN0IGNvbnRleHQgPSBSZWFjdC5jcmVhdGVDb250ZXh0KHtcbiAgc2Nyb2xsVG86ICgpID0+IDAsXG4gIHNjcm9sbFRvQm90dG9tOiAoKSA9PiAwLFxuICBzY3JvbGxUb0VuZDogKCkgPT4gMCxcbiAgc2Nyb2xsVG9Ub3A6ICgpID0+IDBcbn0pO1xuXG5jb250ZXh0LmRpc3BsYXlOYW1lID0gJ1Njcm9sbFRvQm90dG9tRnVuY3Rpb25Db250ZXh0JztcblxuZXhwb3J0IGRlZmF1bHQgY29udGV4dDtcbiJdfQ==
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9TY3JvbGxUb0JvdHRvbS9GdW5jdGlvbkNvbnRleHQuanMiXSwibmFtZXMiOlsiY29udGV4dCIsIlJlYWN0IiwiY3JlYXRlQ29udGV4dCIsInNjcm9sbFRvIiwic2Nyb2xsVG9Cb3R0b20iLCJzY3JvbGxUb0VuZCIsInNjcm9sbFRvU3RhcnQiLCJzY3JvbGxUb1RvcCIsImRpc3BsYXlOYW1lIl0sIm1hcHBpbmdzIjoiOzs7Ozs7O0FBQUE7Ozs7QUFFQSxJQUFNQSxPQUFPLEdBQUdDLGVBQU1DLGFBQU4sQ0FBb0I7QUFDbENDLEVBQUFBLFFBQVEsRUFBRTtBQUFBLFdBQU0sQ0FBTjtBQUFBLEdBRHdCO0FBRWxDQyxFQUFBQSxjQUFjLEVBQUU7QUFBQSxXQUFNLENBQU47QUFBQSxHQUZrQjtBQUdsQ0MsRUFBQUEsV0FBVyxFQUFFO0FBQUEsV0FBTSxDQUFOO0FBQUEsR0FIcUI7QUFJbENDLEVBQUFBLGFBQWEsRUFBRTtBQUFBLFdBQU0sQ0FBTjtBQUFBLEdBSm1CO0FBS2xDQyxFQUFBQSxXQUFXLEVBQUU7QUFBQSxXQUFNLENBQU47QUFBQTtBQUxxQixDQUFwQixDQUFoQjs7QUFRQVAsT0FBTyxDQUFDUSxXQUFSLEdBQXNCLCtCQUF0QjtlQUVlUixPIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFJlYWN0IGZyb20gJ3JlYWN0JztcblxuY29uc3QgY29udGV4dCA9IFJlYWN0LmNyZWF0ZUNvbnRleHQoe1xuICBzY3JvbGxUbzogKCkgPT4gMCxcbiAgc2Nyb2xsVG9Cb3R0b206ICgpID0+IDAsXG4gIHNjcm9sbFRvRW5kOiAoKSA9PiAwLFxuICBzY3JvbGxUb1N0YXJ0OiAoKSA9PiAwLFxuICBzY3JvbGxUb1RvcDogKCkgPT4gMFxufSk7XG5cbmNvbnRleHQuZGlzcGxheU5hbWUgPSAnU2Nyb2xsVG9Cb3R0b21GdW5jdGlvbkNvbnRleHQnO1xuXG5leHBvcnQgZGVmYXVsdCBjb250ZXh0O1xuIl19

@@ -13,7 +13,6 @@ "use strict";

var context = _react.default.createContext({
_handleUpdate: function _handleUpdate() {
offsetHeight: 0,
scrollHeight: 0,
setTarget: function setTarget() {
return 0;
},
_setTarget: function _setTarget() {
return 0;
}

@@ -25,2 +24,2 @@ });

exports.default = _default;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9TY3JvbGxUb0JvdHRvbS9JbnRlcm5hbENvbnRleHQuanMiXSwibmFtZXMiOlsiY29udGV4dCIsIlJlYWN0IiwiY3JlYXRlQ29udGV4dCIsIl9oYW5kbGVVcGRhdGUiLCJfc2V0VGFyZ2V0IiwiZGlzcGxheU5hbWUiXSwibWFwcGluZ3MiOiI7Ozs7Ozs7QUFBQTs7OztBQUVBLElBQU1BLE9BQU8sR0FBR0MsZUFBTUMsYUFBTixDQUFvQjtBQUNsQ0MsRUFBQUEsYUFBYSxFQUFFO0FBQUEsV0FBTSxDQUFOO0FBQUEsR0FEbUI7QUFFbENDLEVBQUFBLFVBQVUsRUFBRTtBQUFBLFdBQU0sQ0FBTjtBQUFBO0FBRnNCLENBQXBCLENBQWhCOztBQUtBSixPQUFPLENBQUNLLFdBQVIsR0FBc0IsK0JBQXRCO2VBRWVMLE8iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnO1xuXG5jb25zdCBjb250ZXh0ID0gUmVhY3QuY3JlYXRlQ29udGV4dCh7XG4gIF9oYW5kbGVVcGRhdGU6ICgpID0+IDAsXG4gIF9zZXRUYXJnZXQ6ICgpID0+IDBcbn0pO1xuXG5jb250ZXh0LmRpc3BsYXlOYW1lID0gJ1Njcm9sbFRvQm90dG9tSW50ZXJuYWxDb250ZXh0JztcblxuZXhwb3J0IGRlZmF1bHQgY29udGV4dDtcbiJdfQ==
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9TY3JvbGxUb0JvdHRvbS9JbnRlcm5hbENvbnRleHQuanMiXSwibmFtZXMiOlsiY29udGV4dCIsIlJlYWN0IiwiY3JlYXRlQ29udGV4dCIsIm9mZnNldEhlaWdodCIsInNjcm9sbEhlaWdodCIsInNldFRhcmdldCIsImRpc3BsYXlOYW1lIl0sIm1hcHBpbmdzIjoiOzs7Ozs7O0FBQUE7Ozs7QUFFQSxJQUFNQSxPQUFPLEdBQUdDLGVBQU1DLGFBQU4sQ0FBb0I7QUFDbENDLEVBQUFBLFlBQVksRUFBRSxDQURvQjtBQUVsQ0MsRUFBQUEsWUFBWSxFQUFFLENBRm9CO0FBR2xDQyxFQUFBQSxTQUFTLEVBQUU7QUFBQSxXQUFNLENBQU47QUFBQTtBQUh1QixDQUFwQixDQUFoQjs7QUFNQUwsT0FBTyxDQUFDTSxXQUFSLEdBQXNCLCtCQUF0QjtlQUVlTixPIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFJlYWN0IGZyb20gJ3JlYWN0JztcblxuY29uc3QgY29udGV4dCA9IFJlYWN0LmNyZWF0ZUNvbnRleHQoe1xuICBvZmZzZXRIZWlnaHQ6IDAsXG4gIHNjcm9sbEhlaWdodDogMCxcbiAgc2V0VGFyZ2V0OiAoKSA9PiAwXG59KTtcblxuY29udGV4dC5kaXNwbGF5TmFtZSA9ICdTY3JvbGxUb0JvdHRvbUludGVybmFsQ29udGV4dCc7XG5cbmV4cG9ydCBkZWZhdWx0IGNvbnRleHQ7XG4iXX0=

@@ -13,6 +13,8 @@ "use strict";

var context = _react.default.createContext({
animating: false,
atBottom: true,
atEnd: true,
atTop: true,
mode: 'bottom'
mode: 'bottom',
sticky: true
});

@@ -23,2 +25,2 @@

exports.default = _default;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9TY3JvbGxUb0JvdHRvbS9TdGF0ZUNvbnRleHQuanMiXSwibmFtZXMiOlsiY29udGV4dCIsIlJlYWN0IiwiY3JlYXRlQ29udGV4dCIsImF0Qm90dG9tIiwiYXRFbmQiLCJhdFRvcCIsIm1vZGUiLCJkaXNwbGF5TmFtZSJdLCJtYXBwaW5ncyI6Ijs7Ozs7OztBQUFBOzs7O0FBRUEsSUFBTUEsT0FBTyxHQUFHQyxlQUFNQyxhQUFOLENBQW9CO0FBQ2xDQyxFQUFBQSxRQUFRLEVBQUUsSUFEd0I7QUFFbENDLEVBQUFBLEtBQUssRUFBRSxJQUYyQjtBQUdsQ0MsRUFBQUEsS0FBSyxFQUFFLElBSDJCO0FBSWxDQyxFQUFBQSxJQUFJLEVBQUU7QUFKNEIsQ0FBcEIsQ0FBaEI7O0FBT0FOLE9BQU8sQ0FBQ08sV0FBUixHQUFzQiw0QkFBdEI7ZUFFZVAsTyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCc7XG5cbmNvbnN0IGNvbnRleHQgPSBSZWFjdC5jcmVhdGVDb250ZXh0KHtcbiAgYXRCb3R0b206IHRydWUsXG4gIGF0RW5kOiB0cnVlLFxuICBhdFRvcDogdHJ1ZSxcbiAgbW9kZTogJ2JvdHRvbSdcbn0pO1xuXG5jb250ZXh0LmRpc3BsYXlOYW1lID0gJ1Njcm9sbFRvQm90dG9tU3RhdGVDb250ZXh0JztcblxuZXhwb3J0IGRlZmF1bHQgY29udGV4dFxuIl19
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9TY3JvbGxUb0JvdHRvbS9TdGF0ZUNvbnRleHQuanMiXSwibmFtZXMiOlsiY29udGV4dCIsIlJlYWN0IiwiY3JlYXRlQ29udGV4dCIsImFuaW1hdGluZyIsImF0Qm90dG9tIiwiYXRFbmQiLCJhdFRvcCIsIm1vZGUiLCJzdGlja3kiLCJkaXNwbGF5TmFtZSJdLCJtYXBwaW5ncyI6Ijs7Ozs7OztBQUFBOzs7O0FBRUEsSUFBTUEsT0FBTyxHQUFHQyxlQUFNQyxhQUFOLENBQW9CO0FBQ2xDQyxFQUFBQSxTQUFTLEVBQUUsS0FEdUI7QUFFbENDLEVBQUFBLFFBQVEsRUFBRSxJQUZ3QjtBQUdsQ0MsRUFBQUEsS0FBSyxFQUFFLElBSDJCO0FBSWxDQyxFQUFBQSxLQUFLLEVBQUUsSUFKMkI7QUFLbENDLEVBQUFBLElBQUksRUFBRSxRQUw0QjtBQU1sQ0MsRUFBQUEsTUFBTSxFQUFFO0FBTjBCLENBQXBCLENBQWhCOztBQVNBUixPQUFPLENBQUNTLFdBQVIsR0FBc0IsNEJBQXRCO2VBRWVULE8iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnO1xuXG5jb25zdCBjb250ZXh0ID0gUmVhY3QuY3JlYXRlQ29udGV4dCh7XG4gIGFuaW1hdGluZzogZmFsc2UsXG4gIGF0Qm90dG9tOiB0cnVlLFxuICBhdEVuZDogdHJ1ZSxcbiAgYXRUb3A6IHRydWUsXG4gIG1vZGU6ICdib3R0b20nLFxuICBzdGlja3k6IHRydWVcbn0pO1xuXG5jb250ZXh0LmRpc3BsYXlOYW1lID0gJ1Njcm9sbFRvQm90dG9tU3RhdGVDb250ZXh0JztcblxuZXhwb3J0IGRlZmF1bHQgY29udGV4dFxuIl19
{
"name": "react-scroll-to-bottom",
"version": "1.3.2-master.a1d5777",
"version": "1.3.2-master.d0df2f1",
"description": "React container that will auto scroll to bottom",

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