react-float-anchor
Advanced tools
Comparing version 3.2.0 to 3.3.0
@@ -0,1 +1,5 @@ | ||
## 3.3.0 (2019-10-22) | ||
* The choice return value from contain-by-screen is now available if you pass a callback as the `float` prop. | ||
## 3.2.0 (2019-10-15) | ||
@@ -2,0 +6,0 @@ |
import * as React from 'react'; | ||
import {Options} from 'contain-by-screen'; | ||
import {Options, Choice} from 'contain-by-screen'; | ||
export {Options} from 'contain-by-screen'; | ||
export {Options, Choice} from 'contain-by-screen'; | ||
export type Props = { | ||
anchor: (anchorRef: React.Ref<any>) => React.ReactNode; | ||
float?: React.ReactNode | null | undefined; | ||
float?: React.ReactNode | ((choice: Choice | null) => React.ReactNode) | null | undefined; | ||
options?: Options | null | undefined; | ||
@@ -11,0 +11,0 @@ zIndex?: number | string | null | undefined; |
@@ -69,2 +69,6 @@ "use strict"; | ||
_this = (0, _possibleConstructorReturn2["default"])(this, (_getPrototypeOf2 = (0, _getPrototypeOf3["default"])(FloatAnchor)).call.apply(_getPrototypeOf2, [this].concat(args))); | ||
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "state", { | ||
choice: null, | ||
floatNode: null | ||
}); | ||
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_portalEl", void 0); | ||
@@ -162,3 +166,3 @@ (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_portalRemoval", (0, _kefirBus["default"])()); | ||
if (this.props["float"] != null) { | ||
if (this.state.floatNode != null) { | ||
// We need to reposition after the page has had its layout done. | ||
@@ -169,8 +173,20 @@ this.repositionAsync(); | ||
}, { | ||
key: "shouldComponentUpdate", | ||
value: function shouldComponentUpdate(nextProps, nextState) { | ||
// If the only thing changed is state.choice *and* typeof props.float !== 'function', don't re-render. | ||
// If nothing has changed, allow the re-render so we keep the same behavior on a plain forceUpdate of a parent. | ||
// TODO in next major version, don't re-render when nothing has changed. | ||
if (typeof nextProps["float"] !== 'function' && this.state.choice !== nextState.choice && this.props.anchor === nextProps.anchor && this.props.parentElement === nextProps.parentElement && this.props["float"] === nextProps["float"] && this.props.options === nextProps.options && this.props.zIndex === nextProps.zIndex && this.props.floatContainerClassName === nextProps.floatContainerClassName) { | ||
return false; | ||
} | ||
return true; | ||
} | ||
}, { | ||
key: "componentDidUpdate", | ||
value: function componentDidUpdate(prevProps) { | ||
value: function componentDidUpdate(prevProps, prevState) { | ||
var portalEl = this._portalEl; | ||
if (portalEl) { | ||
if (this.props["float"] == null) { | ||
if (this.state.floatNode == null) { | ||
this._portalRemoval.value(null); | ||
@@ -190,5 +206,6 @@ } else { | ||
portalEl.style.zIndex = String(this.props.zIndex); | ||
} | ||
} // If this re-render happened because of a change in position choice, don't reposition again now. | ||
if (prevProps["float"] !== this.props["float"] || !(0, _isEqual["default"])(prevProps.options, this.props.options)) { | ||
if (prevState.floatNode !== this.state.floatNode && prevState.choice === this.state.choice || !(0, _isEqual["default"])(prevProps.options, this.props.options)) { | ||
this.repositionAsync(); | ||
@@ -198,3 +215,3 @@ } | ||
} else { | ||
if (this.props["float"] != null) { | ||
if (this.state.floatNode != null) { | ||
throw new Error('Should not happen: portalEl was null after rendering with float prop'); | ||
@@ -261,4 +278,11 @@ } | ||
if (portalEl && portalEl.parentElement && anchorRef) { | ||
(0, _containByScreen["default"])(portalEl, anchorRef, this.props.options || {}); // Make any child FloatAnchors reposition | ||
var _choice = (0, _containByScreen["default"])(portalEl, anchorRef, this.props.options || {}); | ||
if (!(0, _isEqual["default"])(this.state.choice, _choice)) { | ||
this.setState({ | ||
choice: _choice | ||
}); | ||
} // Make any child FloatAnchors reposition | ||
this._childContext.repositionEvents.value(null); | ||
@@ -270,5 +294,4 @@ } | ||
value: function render() { | ||
var _this$props = this.props, | ||
anchor = _this$props.anchor, | ||
_float = _this$props["float"]; | ||
var anchor = this.props.anchor; | ||
var _float = this.state.floatNode; | ||
var floatPortal = null; | ||
@@ -313,2 +336,9 @@ | ||
}) | ||
}, { | ||
key: "getDerivedStateFromProps", | ||
value: function getDerivedStateFromProps(props, state) { | ||
return { | ||
floatNode: typeof props["float"] === 'function' ? props["float"](state.choice) : props["float"] | ||
}; | ||
} | ||
}]); | ||
@@ -322,3 +352,3 @@ return FloatAnchor; | ||
parentElement: _propTypes["default"].object, | ||
"float": _propTypes["default"].node, | ||
"float": _propTypes["default"].oneOfType([_propTypes["default"].node, _propTypes["default"].func]), | ||
options: _propTypes["default"].object, | ||
@@ -331,2 +361,2 @@ zIndex: _propTypes["default"].oneOfType([_propTypes["default"].number, _propTypes["default"].string]), | ||
module.exports.default = exports.default; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/index.js"],"names":["requestAnimationFrame","global","cb","Promise","resolve","then","FloatAnchorContext","React","createContext","FloatAnchor","repositionEvents","repositionAsyncQueued","repositionAsyncEvents","anchorRef","_anchorRef","portalEl","_portalEl","rfaAnchor","undefined","parentElement","Error","target","props","document","body","documentElement","appendChild","Kefir","merge","fromEvents","window","capture","passive","filter","event","contains","takeUntilBy","_portalRemoval","onValue","repositionAsync","onEnd","removeChild","portalEl_firstCheck","createElement","className","floatContainerClassName","style","zIndex","String","position","parentCtx","context","_unmount","reposition","_childContext","value","prevProps","_mountPortalEl","options","end","anchor","float","floatPortal","_getOrCreatePortalEl","_setAnchorRef","node","parentNode","Component","PropTypes","func","isRequired","object","oneOfType","number","string"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;AAEA;;AACA;;AACA;;AACA;;AAEA;;AACA;;AACA;;AACA;;AAEA;;AAEA,IAAMA,qBAAqB,GAAGC,MAAM,CAACD,qBAAP,IAAiC,UAAAE,EAAE;AAAA,SAAIC,OAAO,CAACC,OAAR,GAAkBC,IAAlB,CAAuBH,EAAvB,CAAJ;AAAA,CAAjE;;AAeA;AACA;AACA,IAAMI,kBAAkB,GAAGC,kBAAMC,aAAN,CAA6C,IAA7C,CAA3B;;IAYqBC,W;;;;;;;;;;;;;;;;;;uGAaS,2B;iGACN,2B;sGACN;AACdC,MAAAA,gBAAgB,EAAG,2BADL;AAEdC,MAAAA,qBAAqB,EAAE,KAFT;AAGdC,MAAAA,qBAAqB,EAAG;AAHV,K;mGAMW,I;sGAQX,UAACC,SAAD,EAA6B;AAC3C,YAAKC,UAAL,GAAkBD,SAAlB;AAEA,UAAME,QAAQ,GAAG,MAAKC,SAAtB;;AACA,UAAID,QAAJ,EAAc;AACXA,QAAAA,QAAD,CAAgBE,SAAhB,GAA4BJ,SAAS,GAAGA,SAAH,GAAeK,SAApD;AACD;AACF,K;uGA4HgB,YAAM;AACrB,UAAMH,QAAQ,GAAG,MAAKC,SAAtB;AACA;;AACA,UAAID,QAAQ,CAACI,aAAb,EAA4B;AAC1B,cAAM,IAAIC,KAAJ,CAAU,6CAAV,CAAN;AACD;;AAED,UAAMP,SAAS,GAAG,MAAKC,UAAvB;AACA,UAAI,CAACD,SAAL,EAAgB,MAAM,IAAIO,KAAJ,CAAU,4CAAV,CAAN;AACfL,MAAAA,QAAD,CAAgBE,SAAhB,GAA4BJ,SAA5B;AAEA,UAAMQ,MAAM,GAAG,MAAKC,KAAL,CAAWH,aAAX,IAA4BI,QAAQ,CAACC,IAArC,IAA6CD,QAAQ,CAACE,eAArE;AACA;;AACAJ,MAAAA,MAAM,CAACK,WAAP,CAAmBX,QAAnB;;AAEAY,wBAAMC,KAAN,CAAY,CACVD,kBAAME,UAAN,CAAiBC,MAAjB,EAAyB,QAAzB,CADU,EAEV,uCAAsBA,MAAtB,EAA8B,QAA9B,EAAwC;AACtCC,QAAAA,OAAO,EAAE,IAD6B;AAEtCC,QAAAA,OAAO,EAAE;AAF6B,OAAxC,EAGGC,MAHH,CAGU,UAAAC,KAAK,EAAI;AACjB,YAAMrB,SAAS,GAAG,MAAKC,UAAvB;AACA,eAAOD,SAAS,IAAIqB,KAAK,CAACb,MAAN,CAAac,QAAb,CAAsBtB,SAAtB,CAApB;AACD,OAND,CAFU,CAAZ,EAUGuB,WAVH,CAUe,MAAKC,cAVpB,EAWGC,OAXH,CAWW,YAAM;AACb,cAAKC,eAAL;AACD,OAbH,EAcGC,KAdH,CAcS,YAAM;AACVzB,QAAAA,QAAD,CAAgBE,SAAhB,GAA4BC,SAA5B;AACA;;AACAH,QAAAA,QAAQ,CAACI,aAAT,CAAuBsB,WAAvB,CAAmC1B,QAAnC;AACD,OAlBH;AAmBD,K;;;;;;2CA5JmC;AAClC,UAAM2B,mBAAmB,GAAG,KAAK1B,SAAjC;;AACA,UAAI0B,mBAAJ,EAAyB;AACvB,eAAOA,mBAAP;AACD;;AAED,UAAM3B,QAAQ,GAAG,KAAKC,SAAL,GAAiBO,QAAQ,CAACoB,aAAT,CAAuB,KAAvB,CAAlC;AACA5B,MAAAA,QAAQ,CAAC6B,SAAT,GAAqB,KAAKtB,KAAL,CAAWuB,uBAAX,IAAsC,EAA3D;AACA9B,MAAAA,QAAQ,CAAC+B,KAAT,CAAeC,MAAf,GAAwBC,MAAM,CAAC,KAAK1B,KAAL,CAAWyB,MAAZ,CAA9B;AACAhC,MAAAA,QAAQ,CAAC+B,KAAT,CAAeG,QAAf,GAA0B,OAA1B;AAEA,aAAOlC,QAAP;AACD;;;wCAEmB;AAAA;;AAClB,UAAMmC,SAAkC,GAAG,KAAKC,OAAhD;;AACA,UAAID,SAAJ,EAAe;AACbA,QAAAA,SAAS,CAACxC,gBAAV,CACG0B,WADH,CACe,KAAKgB,QADpB,EAEGd,OAFH,CAEW;AAAA,iBAAM,MAAI,CAACe,UAAL,EAAN;AAAA,SAFX;AAIAH,QAAAA,SAAS,CAACtC,qBAAV,CACGwB,WADH,CACe,KAAKgB,QADpB,EAEGd,OAFH,CAEW,YAAM;AACb,cAAI,CAAC,MAAI,CAACgB,aAAL,CAAmB3C,qBAAxB,EAA+C;AAC7C,YAAA,MAAI,CAAC2C,aAAL,CAAmB3C,qBAAnB,GAA2C,IAA3C;;AACA,YAAA,MAAI,CAAC2C,aAAL,CAAmB1C,qBAAnB,CAAyC2C,KAAzC,CAA+C,IAA/C;AACD;AACF,SAPH;;AASA,YAAIL,SAAS,CAACvC,qBAAd,EAAqC;AACnC,eAAK2C,aAAL,CAAmB3C,qBAAnB,GAA2C,IAA3C;;AACA,eAAK2C,aAAL,CAAmB1C,qBAAnB,CAAyC2C,KAAzC,CAA+C,IAA/C;AACD;AACF;;AAED,UAAI,KAAKjC,KAAL,aAAoB,IAAxB,EAA8B;AAC5B;AACA,aAAKiB,eAAL;AACD;AACF;;;uCAEkBiB,S,EAAkB;AACnC,UAAMzC,QAAQ,GAAG,KAAKC,SAAtB;;AACA,UAAID,QAAJ,EAAc;AACZ,YAAI,KAAKO,KAAL,aAAoB,IAAxB,EAA8B;AAC5B,eAAKe,cAAL,CAAoBkB,KAApB,CAA0B,IAA1B;AACD,SAFD,MAEO;AACL,cAAIC,SAAS,CAACrC,aAAV,KAA4B,KAAKG,KAAL,CAAWH,aAA3C,EAA0D;AACxD,iBAAKkB,cAAL,CAAoBkB,KAApB,CAA0B,IAA1B;;AACA,iBAAKE,cAAL;AACD;;AACD,cAAID,SAAS,CAACX,uBAAV,KAAsC,KAAKvB,KAAL,CAAWuB,uBAArD,EAA8E;AAC5E9B,YAAAA,QAAQ,CAAC6B,SAAT,GAAqB,KAAKtB,KAAL,CAAWuB,uBAAX,IAAsC,EAA3D;AACD;;AACD,cAAIW,SAAS,CAACT,MAAV,KAAqB,KAAKzB,KAAL,CAAWyB,MAApC,EAA4C;AAC1ChC,YAAAA,QAAQ,CAAC+B,KAAT,CAAeC,MAAf,GAAwBC,MAAM,CAAC,KAAK1B,KAAL,CAAWyB,MAAZ,CAA9B;AACD;;AAED,cACES,SAAS,SAAT,KAAoB,KAAKlC,KAAL,SAApB,IACA,CAAC,yBAAQkC,SAAS,CAACE,OAAlB,EAA2B,KAAKpC,KAAL,CAAWoC,OAAtC,CAFH,EAGE;AACA,iBAAKnB,eAAL;AACD;AACF;AACF,OAtBD,MAsBO;AACL,YAAI,KAAKjB,KAAL,aAAoB,IAAxB,EAA8B;AAC5B,gBAAM,IAAIF,KAAJ,CAAU,sEAAV,CAAN;AACD;AACF;AACF;;;2CAEsB;AACrB,WAAKiB,cAAL,CAAoBkB,KAApB,CAA0B,IAA1B;;AACA,WAAKH,QAAL,CAAcG,KAAd,CAAoB,IAApB;;AACA,WAAKD,aAAL,CAAmB5C,gBAAnB,CAAoCiD,GAApC;;AACA,WAAKL,aAAL,CAAmB1C,qBAAnB,CAAyC+C,GAAzC;AACD,K,CAED;AACA;;;;sCACkB;AAAA;;AAChB;AACA,UAAI,KAAKL,aAAL,CAAmB3C,qBAAvB,EAA8C;AAC5C;AACD;;AACD,WAAK2C,aAAL,CAAmB3C,qBAAnB,GAA2C,IAA3C;;AACA,WAAK2C,aAAL,CAAmB1C,qBAAnB,CAAyC2C,KAAzC,CAA+C,IAA/C;;AAEAvD,MAAAA,qBAAqB,CAAC,YAAM;AAC1B;AACA;AACA,YAAMkD,SAAkC,GAAG,MAAI,CAACC,OAAhD;;AACA,YAAI,CAACD,SAAD,IAAc,CAACA,SAAS,CAACvC,qBAA7B,EAAoD;AAClD;AACA;AACA,cAAI,MAAI,CAAC2C,aAAL,CAAmB3C,qBAAvB,EAA8C;AAC5C,YAAA,MAAI,CAAC2C,aAAL,CAAmB3C,qBAAnB,GAA2C,KAA3C;;AACA,YAAA,MAAI,CAAC0C,UAAL;AACD;AACF;AACF,OAZoB,CAArB;AAaD;;;iCAEY;AACX;AACA,UAAMH,SAAkC,GAAG,KAAKC,OAAhD;;AACA,UAAI,CAACD,SAAD,IAAc,CAACA,SAAS,CAACvC,qBAA7B,EAAoD;AAClD,aAAK2C,aAAL,CAAmB3C,qBAAnB,GAA2C,KAA3C;AACD;;AAED,UAAMI,QAAQ,GAAG,KAAKC,SAAtB;AACA,UAAMH,SAAS,GAAG,KAAKC,UAAvB;;AACA,UAAIC,QAAQ,IAAIA,QAAQ,CAACI,aAArB,IAAsCN,SAA1C,EAAqD;AACnD,yCAAgBE,QAAhB,EAA0BF,SAA1B,EAAqC,KAAKS,KAAL,CAAWoC,OAAX,IAAsB,EAA3D,EADmD,CAGnD;;AACA,aAAKJ,aAAL,CAAmB5C,gBAAnB,CAAoC6C,KAApC,CAA0C,IAA1C;AACD;AACF;;;6BAsCQ;AAAA,wBACiB,KAAKjC,KADtB;AAAA,UACAsC,MADA,eACAA,MADA;AAAA,UACQC,MADR;AAEP,UAAIC,WAAW,GAAG,IAAlB;;AACA,UAAID,MAAK,IAAI,IAAb,EAAmB;AACjB,YAAM9C,QAAQ,GAAG,KAAKgD,oBAAL,EAAjB;;AACAD,QAAAA,WAAW,GACT,gCAAC,kBAAD,CAAoB,QAApB;AAA6B,UAAA,KAAK,EAAG,KAAKR;AAA1C,WACE,gCAAC,2BAAD;AAAiB,UAAA,OAAO,EAAE,KAAKG;AAA/B,UADF,EAEG,4BAAaI,MAAb,EAAoB9C,QAApB,CAFH,CADF;AAMD;;AAED,aACE,kEACG6C,MAAM,CAAE,KAAKI,aAAP,CADT,EAEGF,WAFH,CADF;AAMD;;;;;sDAhMmBG,I;;;;;;AAEhB,qBAAOA,IAAP;;;kBACQA,IAAI,GAAIA,IAAD,CAAYhD,SAAZ,IAA0BgD,IAAD,CAAYC,U;;;;;;;;;;;;;;EA1BjB3D,kBAAM4D,S;;;iCAA1B1D,W,eACA;AACjBmD,EAAAA,MAAM,EAAEQ,sBAAUC,IAAV,CAAeC,UADN;AAEjBnD,EAAAA,aAAa,EAAEiD,sBAAUG,MAFR;AAGjB,WAAOH,sBAAUH,IAHA;AAIjBP,EAAAA,OAAO,EAAEU,sBAAUG,MAJF;AAKjBxB,EAAAA,MAAM,EAAEqB,sBAAUI,SAAV,CAAoB,CAACJ,sBAAUK,MAAX,EAAmBL,sBAAUM,MAA7B,CAApB,CALS;AAMjB7B,EAAAA,uBAAuB,EAAEuB,sBAAUM;AANlB,C;iCADAjE,W,iBAUEH,kB","sourcesContent":["/* @flow */\n\nimport fromEventsWithOptions from './lib/fromEventsWithOptions';\nimport LifecycleHelper from './LifecycleHelper';\nimport Kefir from 'kefir';\nimport kefirBus from 'kefir-bus';\nimport type {Bus} from 'kefir-bus';\nimport React from 'react';\nimport {createPortal} from 'react-dom';\nimport PropTypes from 'prop-types';\nimport containByScreen from 'contain-by-screen';\nimport type {Options} from 'contain-by-screen';\nimport isEqual from 'lodash/isEqual';\n\nconst requestAnimationFrame = global.requestAnimationFrame || (cb => Promise.resolve().then(cb));\n\ntype FloatAnchorContextType = {\n  // Emits an event after the component repositions, so the children can reposition themselves to match.\n  repositionEvents: Kefir.Observable<null>,\n\n  // Signifies this component has a repositionAsync queued up. Children components should ignore repositionAsync\n  // calls while this is true. Children should copy their parent whenever their parent sets theirs to true.\n  // Components should clear this flag when they reposition unless they have a parent with it still set.\n  repositionAsyncQueued: boolean,\n\n  // Emits every time repositionAsyncQueued becomes true.\n  repositionAsyncEvents: Kefir.Observable<null>\n};\n\n// Context is used so that when a FloatAnchor has reposition() called on it,\n// all of its descendant FloatAnchor elements reposition too.\nconst FloatAnchorContext = React.createContext<?FloatAnchorContextType>(null);\n\nexport type {Options} from 'contain-by-screen';\n\nexport type Props = {\n  anchor: (anchorRef: React$Ref<any>) => React$Node;\n  parentElement?: ?HTMLElement;\n  float?: ?React$Node;\n  options?: ?Options;\n  zIndex?: ?number|string;\n  floatContainerClassName?: ?string;\n};\nexport default class FloatAnchor extends React.Component<Props> {\n  static propTypes = {\n    anchor: PropTypes.func.isRequired,\n    parentElement: PropTypes.object,\n    float: PropTypes.node,\n    options: PropTypes.object,\n    zIndex: PropTypes.oneOfType([PropTypes.number, PropTypes.string]),\n    floatContainerClassName: PropTypes.string\n  };\n\n  static contextType = FloatAnchorContext;\n\n  _portalEl: ?HTMLElement;\n  _portalRemoval: Bus<null> = kefirBus();\n  _unmount: Bus<null> = kefirBus();\n  _childContext = {\n    repositionEvents: (kefirBus(): Bus<null>),\n    repositionAsyncQueued: false,\n    repositionAsyncEvents: (kefirBus(): Bus<null>)\n  };\n\n  _anchorRef: ?HTMLElement = null;\n\n  static *parentNodes(node: Node): Iterator<Node> {\n    do {\n      yield (node: Node);\n    } while ((node = (node: any).rfaAnchor || (node: any).parentNode));\n  }\n\n  _setAnchorRef = (anchorRef: ?HTMLElement) => {\n    this._anchorRef = anchorRef;\n\n    const portalEl = this._portalEl;\n    if (portalEl) {\n      (portalEl: any).rfaAnchor = anchorRef ? anchorRef : undefined;\n    }\n  };\n\n  _getOrCreatePortalEl(): HTMLElement {\n    const portalEl_firstCheck = this._portalEl;\n    if (portalEl_firstCheck) {\n      return portalEl_firstCheck;\n    }\n\n    const portalEl = this._portalEl = document.createElement('div');\n    portalEl.className = this.props.floatContainerClassName || '';\n    portalEl.style.zIndex = String(this.props.zIndex);\n    portalEl.style.position = 'fixed';\n\n    return portalEl;\n  }\n\n  componentDidMount() {\n    const parentCtx: ?FloatAnchorContextType = this.context;\n    if (parentCtx) {\n      parentCtx.repositionEvents\n        .takeUntilBy(this._unmount)\n        .onValue(() => this.reposition());\n\n      parentCtx.repositionAsyncEvents\n        .takeUntilBy(this._unmount)\n        .onValue(() => {\n          if (!this._childContext.repositionAsyncQueued) {\n            this._childContext.repositionAsyncQueued = true;\n            this._childContext.repositionAsyncEvents.value(null);\n          }\n        });\n\n      if (parentCtx.repositionAsyncQueued) {\n        this._childContext.repositionAsyncQueued = true;\n        this._childContext.repositionAsyncEvents.value(null);\n      }\n    }\n\n    if (this.props.float != null) {\n      // We need to reposition after the page has had its layout done.\n      this.repositionAsync();\n    }\n  }\n\n  componentDidUpdate(prevProps: Props) {\n    const portalEl = this._portalEl;\n    if (portalEl) {\n      if (this.props.float == null) {\n        this._portalRemoval.value(null);\n      } else {\n        if (prevProps.parentElement !== this.props.parentElement) {\n          this._portalRemoval.value(null);\n          this._mountPortalEl();\n        }\n        if (prevProps.floatContainerClassName !== this.props.floatContainerClassName) {\n          portalEl.className = this.props.floatContainerClassName || '';\n        }\n        if (prevProps.zIndex !== this.props.zIndex) {\n          portalEl.style.zIndex = String(this.props.zIndex);\n        }\n\n        if (\n          prevProps.float !== this.props.float ||\n          !isEqual(prevProps.options, this.props.options)\n        ) {\n          this.repositionAsync();\n        }\n      }\n    } else {\n      if (this.props.float != null) {\n        throw new Error('Should not happen: portalEl was null after rendering with float prop');\n      }\n    }\n  }\n\n  componentWillUnmount() {\n    this._portalRemoval.value(null);\n    this._unmount.value(null);\n    this._childContext.repositionEvents.end();\n    this._childContext.repositionAsyncEvents.end();\n  }\n\n  // Repositions on the next animation frame. Automatically batches with other repositionAsync calls\n  // in the same tree.\n  repositionAsync() {\n    // If we already have a repositionAsync queued up, there's no reason to queue another.\n    if (this._childContext.repositionAsyncQueued) {\n      return;\n    }\n    this._childContext.repositionAsyncQueued = true;\n    this._childContext.repositionAsyncEvents.value(null);\n\n    requestAnimationFrame(() => {\n      // If our parent still has a repositionAsync queued up, then don't fire.\n      // The parent may have queued up a repositionAsync in the time since this repositionAsync() was called.\n      const parentCtx: ?FloatAnchorContextType = this.context;\n      if (!parentCtx || !parentCtx.repositionAsyncQueued) {\n        // Make sure we still have a repositionAsync queued up. It could be that reposition() has been called\n        // in the time since repositionAsync().\n        if (this._childContext.repositionAsyncQueued) {\n          this._childContext.repositionAsyncQueued = false;\n          this.reposition();\n        }\n      }\n    });\n  }\n\n  reposition() {\n    // Only clear our repositionAsyncQueued flag if we're not reflecting our parent's true value.\n    const parentCtx: ?FloatAnchorContextType = this.context;\n    if (!parentCtx || !parentCtx.repositionAsyncQueued) {\n      this._childContext.repositionAsyncQueued = false;\n    }\n\n    const portalEl = this._portalEl;\n    const anchorRef = this._anchorRef;\n    if (portalEl && portalEl.parentElement && anchorRef) {\n      containByScreen(portalEl, anchorRef, this.props.options || {});\n\n      // Make any child FloatAnchors reposition\n      this._childContext.repositionEvents.value(null);\n    }\n  }\n\n  _mountPortalEl = () => {\n    const portalEl = this._portalEl;\n    /*:: if (!portalEl) throw new Error(); */\n    if (portalEl.parentElement) {\n      throw new Error('Should not happen: portalEl already in page');\n    }\n\n    const anchorRef = this._anchorRef;\n    if (!anchorRef) throw new Error('ReactFloatAnchor missing anchorRef element');\n    (portalEl: any).rfaAnchor = anchorRef;\n\n    const target = this.props.parentElement || document.body || document.documentElement;\n    /*:: if (!target) throw new Error(); */\n    target.appendChild(portalEl);\n\n    Kefir.merge([\n      Kefir.fromEvents(window, 'resize'),\n      fromEventsWithOptions(window, 'scroll', {\n        capture: true,\n        passive: true\n      }).filter(event => {\n        const anchorRef = this._anchorRef;\n        return anchorRef && event.target.contains(anchorRef);\n      })\n    ])\n      .takeUntilBy(this._portalRemoval)\n      .onValue(() => {\n        this.repositionAsync();\n      })\n      .onEnd(() => {\n        (portalEl: any).rfaAnchor = undefined;\n        /*:: if (!portalEl.parentElement) throw new Error(); */\n        portalEl.parentElement.removeChild(portalEl);\n      });\n  };\n\n  render() {\n    const {anchor, float} = this.props;\n    let floatPortal = null;\n    if (float != null) {\n      const portalEl = this._getOrCreatePortalEl();\n      floatPortal = (\n        <FloatAnchorContext.Provider value={(this._childContext: FloatAnchorContextType)}>\n          <LifecycleHelper onMount={this._mountPortalEl} />\n          {createPortal(float, portalEl)}\n        </FloatAnchorContext.Provider>\n      );\n    }\n\n    return (\n      <>\n        {anchor((this._setAnchorRef: any))}\n        {floatPortal}\n      </>\n    );\n  }\n}\n"]} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/index.js"],"names":["requestAnimationFrame","global","cb","Promise","resolve","then","FloatAnchorContext","React","createContext","FloatAnchor","choice","floatNode","repositionEvents","repositionAsyncQueued","repositionAsyncEvents","anchorRef","_anchorRef","portalEl","_portalEl","rfaAnchor","undefined","parentElement","Error","target","props","document","body","documentElement","appendChild","Kefir","merge","fromEvents","window","capture","passive","filter","event","contains","takeUntilBy","_portalRemoval","onValue","repositionAsync","onEnd","removeChild","portalEl_firstCheck","createElement","className","floatContainerClassName","style","zIndex","String","position","parentCtx","context","_unmount","reposition","_childContext","value","state","nextProps","nextState","anchor","options","prevProps","prevState","_mountPortalEl","end","setState","float","floatPortal","_getOrCreatePortalEl","_setAnchorRef","node","parentNode","Component","PropTypes","func","isRequired","object","oneOfType","number","string"],"mappings":"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","sourcesContent":["/* @flow */\n\nimport fromEventsWithOptions from './lib/fromEventsWithOptions';\nimport LifecycleHelper from './LifecycleHelper';\nimport Kefir from 'kefir';\nimport kefirBus from 'kefir-bus';\nimport type {Bus} from 'kefir-bus';\nimport React from 'react';\nimport {createPortal} from 'react-dom';\nimport PropTypes from 'prop-types';\nimport containByScreen from 'contain-by-screen';\nimport type {Options, Choice} from 'contain-by-screen';\nimport isEqual from 'lodash/isEqual';\n\nconst requestAnimationFrame = global.requestAnimationFrame || (cb => Promise.resolve().then(cb));\n\ntype FloatAnchorContextType = {\n  // Emits an event after the component repositions, so the children can reposition themselves to match.\n  repositionEvents: Kefir.Observable<null>,\n\n  // Signifies this component has a repositionAsync queued up. Children components should ignore repositionAsync\n  // calls while this is true. Children should copy their parent whenever their parent sets theirs to true.\n  // Components should clear this flag when they reposition unless they have a parent with it still set.\n  repositionAsyncQueued: boolean,\n\n  // Emits every time repositionAsyncQueued becomes true.\n  repositionAsyncEvents: Kefir.Observable<null>\n};\n\n// Context is used so that when a FloatAnchor has reposition() called on it,\n// all of its descendant FloatAnchor elements reposition too.\nconst FloatAnchorContext = React.createContext<?FloatAnchorContextType>(null);\n\nexport type {Options, Choice} from 'contain-by-screen';\n\nexport type Props = {\n  anchor: (anchorRef: React$Ref<any>) => React$Node;\n  parentElement?: ?HTMLElement;\n  float?: ?React$Node | ((choice: Choice | null) => React$Node);\n  options?: ?Options;\n  zIndex?: ?number|string;\n  floatContainerClassName?: ?string;\n};\ntype State = {\n  choice: Choice | null;\n  floatNode: ?React$Node;\n};\nexport default class FloatAnchor extends React.Component<Props, State> {\n  static propTypes = {\n    anchor: PropTypes.func.isRequired,\n    parentElement: PropTypes.object,\n    float: PropTypes.oneOfType([PropTypes.node, PropTypes.func]),\n    options: PropTypes.object,\n    zIndex: PropTypes.oneOfType([PropTypes.number, PropTypes.string]),\n    floatContainerClassName: PropTypes.string\n  };\n\n  static contextType = FloatAnchorContext;\n\n  state: State = {\n    choice: null,\n    floatNode: null\n  };\n\n  _portalEl: ?HTMLElement;\n  _portalRemoval: Bus<null> = kefirBus();\n  _unmount: Bus<null> = kefirBus();\n  _childContext = {\n    repositionEvents: (kefirBus(): Bus<null>),\n    repositionAsyncQueued: false,\n    repositionAsyncEvents: (kefirBus(): Bus<null>)\n  };\n\n  _anchorRef: ?HTMLElement = null;\n\n  static *parentNodes(node: Node): Iterator<Node> {\n    do {\n      yield (node: Node);\n    } while ((node = (node: any).rfaAnchor || (node: any).parentNode));\n  }\n\n  _setAnchorRef = (anchorRef: ?HTMLElement) => {\n    this._anchorRef = anchorRef;\n\n    const portalEl = this._portalEl;\n    if (portalEl) {\n      (portalEl: any).rfaAnchor = anchorRef ? anchorRef : undefined;\n    }\n  };\n\n  _getOrCreatePortalEl(): HTMLElement {\n    const portalEl_firstCheck = this._portalEl;\n    if (portalEl_firstCheck) {\n      return portalEl_firstCheck;\n    }\n\n    const portalEl = this._portalEl = document.createElement('div');\n    portalEl.className = this.props.floatContainerClassName || '';\n    portalEl.style.zIndex = String(this.props.zIndex);\n    portalEl.style.position = 'fixed';\n\n    return portalEl;\n  }\n\n  componentDidMount() {\n    const parentCtx: ?FloatAnchorContextType = this.context;\n    if (parentCtx) {\n      parentCtx.repositionEvents\n        .takeUntilBy(this._unmount)\n        .onValue(() => this.reposition());\n\n      parentCtx.repositionAsyncEvents\n        .takeUntilBy(this._unmount)\n        .onValue(() => {\n          if (!this._childContext.repositionAsyncQueued) {\n            this._childContext.repositionAsyncQueued = true;\n            this._childContext.repositionAsyncEvents.value(null);\n          }\n        });\n\n      if (parentCtx.repositionAsyncQueued) {\n        this._childContext.repositionAsyncQueued = true;\n        this._childContext.repositionAsyncEvents.value(null);\n      }\n    }\n\n    if (this.state.floatNode != null) {\n      // We need to reposition after the page has had its layout done.\n      this.repositionAsync();\n    }\n  }\n\n  static getDerivedStateFromProps(props: Props, state: State): $Shape<State> | null {\n    return {\n      floatNode: typeof props.float === 'function' ? props.float(state.choice) : props.float\n    };\n  }\n\n  shouldComponentUpdate(nextProps: Props, nextState: State) {\n    // If the only thing changed is state.choice *and* typeof props.float !== 'function', don't re-render.\n    // If nothing has changed, allow the re-render so we keep the same behavior on a plain forceUpdate of a parent.\n    // TODO in next major version, don't re-render when nothing has changed.\n    if (\n      typeof nextProps.float !== 'function' &&\n      this.state.choice !== nextState.choice &&\n      this.props.anchor === nextProps.anchor &&\n      this.props.parentElement === nextProps.parentElement &&\n      this.props.float === nextProps.float &&\n      this.props.options === nextProps.options &&\n      this.props.zIndex === nextProps.zIndex &&\n      this.props.floatContainerClassName === nextProps.floatContainerClassName\n    ) {\n      return false;\n    }\n    return true;\n  }\n\n  componentDidUpdate(prevProps: Props, prevState: State) {\n    const portalEl = this._portalEl;\n    if (portalEl) {\n      if (this.state.floatNode == null) {\n        this._portalRemoval.value(null);\n      } else {\n        if (prevProps.parentElement !== this.props.parentElement) {\n          this._portalRemoval.value(null);\n          this._mountPortalEl();\n        }\n        if (prevProps.floatContainerClassName !== this.props.floatContainerClassName) {\n          portalEl.className = this.props.floatContainerClassName || '';\n        }\n        if (prevProps.zIndex !== this.props.zIndex) {\n          portalEl.style.zIndex = String(this.props.zIndex);\n        }\n\n        // If this re-render happened because of a change in position choice, don't reposition again now.\n        if (\n          (prevState.floatNode !== this.state.floatNode && prevState.choice === this.state.choice) ||\n          !isEqual(prevProps.options, this.props.options)\n        ) {\n          this.repositionAsync();\n        }\n      }\n    } else {\n      if (this.state.floatNode != null) {\n        throw new Error('Should not happen: portalEl was null after rendering with float prop');\n      }\n    }\n  }\n\n  componentWillUnmount() {\n    this._portalRemoval.value(null);\n    this._unmount.value(null);\n    this._childContext.repositionEvents.end();\n    this._childContext.repositionAsyncEvents.end();\n  }\n\n  // Repositions on the next animation frame. Automatically batches with other repositionAsync calls\n  // in the same tree.\n  repositionAsync() {\n    // If we already have a repositionAsync queued up, there's no reason to queue another.\n    if (this._childContext.repositionAsyncQueued) {\n      return;\n    }\n    this._childContext.repositionAsyncQueued = true;\n    this._childContext.repositionAsyncEvents.value(null);\n\n    requestAnimationFrame(() => {\n      // If our parent still has a repositionAsync queued up, then don't fire.\n      // The parent may have queued up a repositionAsync in the time since this repositionAsync() was called.\n      const parentCtx: ?FloatAnchorContextType = this.context;\n      if (!parentCtx || !parentCtx.repositionAsyncQueued) {\n        // Make sure we still have a repositionAsync queued up. It could be that reposition() has been called\n        // in the time since repositionAsync().\n        if (this._childContext.repositionAsyncQueued) {\n          this._childContext.repositionAsyncQueued = false;\n          this.reposition();\n        }\n      }\n    });\n  }\n\n  reposition() {\n    // Only clear our repositionAsyncQueued flag if we're not reflecting our parent's true value.\n    const parentCtx: ?FloatAnchorContextType = this.context;\n    if (!parentCtx || !parentCtx.repositionAsyncQueued) {\n      this._childContext.repositionAsyncQueued = false;\n    }\n\n    const portalEl = this._portalEl;\n    const anchorRef = this._anchorRef;\n    if (portalEl && portalEl.parentElement && anchorRef) {\n      const choice = containByScreen(portalEl, anchorRef, this.props.options || {});\n      if (!isEqual(this.state.choice, choice)) {\n        this.setState({choice});\n      }\n\n      // Make any child FloatAnchors reposition\n      this._childContext.repositionEvents.value(null);\n    }\n  }\n\n  _mountPortalEl = () => {\n    const portalEl = this._portalEl;\n    /*:: if (!portalEl) throw new Error(); */\n    if (portalEl.parentElement) {\n      throw new Error('Should not happen: portalEl already in page');\n    }\n\n    const anchorRef = this._anchorRef;\n    if (!anchorRef) throw new Error('ReactFloatAnchor missing anchorRef element');\n    (portalEl: any).rfaAnchor = anchorRef;\n\n    const target = this.props.parentElement || document.body || document.documentElement;\n    /*:: if (!target) throw new Error(); */\n    target.appendChild(portalEl);\n\n    Kefir.merge([\n      Kefir.fromEvents(window, 'resize'),\n      fromEventsWithOptions(window, 'scroll', {\n        capture: true,\n        passive: true\n      }).filter(event => {\n        const anchorRef = this._anchorRef;\n        return anchorRef && event.target.contains(anchorRef);\n      })\n    ])\n      .takeUntilBy(this._portalRemoval)\n      .onValue(() => {\n        this.repositionAsync();\n      })\n      .onEnd(() => {\n        (portalEl: any).rfaAnchor = undefined;\n        /*:: if (!portalEl.parentElement) throw new Error(); */\n        portalEl.parentElement.removeChild(portalEl);\n      });\n  };\n\n  render() {\n    const {anchor} = this.props;\n    const float = this.state.floatNode;\n    let floatPortal = null;\n    if (float != null) {\n      const portalEl = this._getOrCreatePortalEl();\n      floatPortal = (\n        <FloatAnchorContext.Provider value={(this._childContext: FloatAnchorContextType)}>\n          <LifecycleHelper onMount={this._mountPortalEl} />\n          {createPortal(float, portalEl)}\n        </FloatAnchorContext.Provider>\n      );\n    }\n\n    return (\n      <>\n        {anchor((this._setAnchorRef: any))}\n        {floatPortal}\n      </>\n    );\n  }\n}\n"]} |
{ | ||
"name": "react-float-anchor", | ||
"version": "3.2.0", | ||
"version": "3.3.0", | ||
"description": "React component for positioning an element aligned to another", | ||
@@ -5,0 +5,0 @@ "main": "js/index.js", |
@@ -35,10 +35,15 @@ # react-float-anchor | ||
`FloatAnchor` element was used, with no added wrapper elements around it. | ||
* `float` must be null, or a React node. If null, then FloatAnchor won't do | ||
anything other than render `anchor` as-is. If non-null, `float` will be | ||
rendered in a container div which has `position:fixed` styling, is attached | ||
directly to the document body (or parentElement), and is positioned to line up | ||
with the anchorRef element. | ||
* `float` must be null, a React node, or a function that returns a React node. | ||
If null, then FloatAnchor won't do anything other than render `anchor` as-is. | ||
If non-null, the `float` React node will be rendered in a container div which | ||
has `position:fixed` styling, is attached directly to the document body (or | ||
parentElement), and is positioned to line up with the anchorRef element. | ||
If `float` is a function, then the function will receive the return value from | ||
the most recent call to [contain-by-screen](https://github.com/Macil/contain-by-screen), | ||
or null if it hasn't been called yet. The initial render will pass null, and | ||
then the component will be re-rendered after having been positioned by | ||
contain-by-screen. | ||
* `options` is an optional object of options to control how the float element's | ||
container is aligned to the anchor element. The options are the same as those | ||
supported by [contain-by-screen (version ^1.0)](https://github.com/AgentME/contain-by-screen#readme). | ||
supported by [contain-by-screen (version ^1.0)](https://github.com/Macil/contain-by-screen#readme). | ||
* `zIndex` is an optional number controlling the z-index CSS property of the | ||
@@ -45,0 +50,0 @@ float element's container. |
Sorry, the diff of this file is not supported yet
65318
351
86