1
0

index.tsx 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. import React from 'react';
  2. import { findDOMNode } from 'react-dom';
  3. import PropTypes from 'prop-types';
  4. import BaseComponent, { BaseProps } from '../_base/baseComponent';
  5. import ResizeObserver from 'resize-observer-polyfill';
  6. /** A parallel type to `ResizeObserverEntry` (from resize-observer-polyfill). */
  7. export interface ResizeEntry {
  8. contentRect: DOMRectReadOnly;
  9. target: Element;
  10. }
  11. export interface ReactResizeObserverProps extends BaseProps {
  12. onResize?: (entries: ResizeEntry[]) => void;
  13. observeParent?: boolean;
  14. }
  15. export default class ReactResizeObserver extends BaseComponent<ReactResizeObserverProps> {
  16. static propTypes = {
  17. onResize: PropTypes.func,
  18. observeParent: PropTypes.bool,
  19. };
  20. static defaultProps = {
  21. onResize: () => {}, // eslint-disable-line
  22. observeParent: false,
  23. };
  24. observer: ResizeObserver;
  25. childNode: any;
  26. element: Element;
  27. _parentNode: HTMLElement;
  28. constructor(props: ReactResizeObserverProps) {
  29. super(props);
  30. this.observer = new ResizeObserver(props.onResize);
  31. }
  32. componentDidMount() {
  33. this.observeElement();
  34. }
  35. componentDidUpdate(prevProps: ReactResizeObserverProps) {
  36. this.observeElement(this.props.observeParent !== prevProps.observeParent);
  37. }
  38. componentWillUnmount() {
  39. if (this.observer) {
  40. this.observer.disconnect();
  41. this.observer = null;
  42. }
  43. }
  44. getElement = () => {
  45. try {
  46. // using findDOMNode for two reasons:
  47. // 1. cloning to insert a ref is unwieldy and not performant.
  48. // 2. ensure that we resolve to an actual DOM node (instead of any JSX ref instance).
  49. return findDOMNode(this.childNode || this);
  50. } catch (error) {
  51. // swallow error if findDOMNode is run on unmounted component.
  52. return null;
  53. }
  54. };
  55. observeElement(force = false) {
  56. const element = this.getElement();
  57. if (!(element && element instanceof Element)) {
  58. // stop everything if not defined
  59. this.observer.disconnect();
  60. return;
  61. }
  62. if (element === this.element && !force) {
  63. // abort if given same element -- nothing to update (unless forced)
  64. return;
  65. } else {
  66. // clear observer list if new element
  67. this.observer.disconnect();
  68. // remember element reference for next time
  69. this.element = element;
  70. }
  71. // observer callback is invoked immediately when observing new elements
  72. this.observer.observe(element);
  73. if (
  74. this.props.observeParent &&
  75. element.parentNode &&
  76. element.parentNode.ownerDocument &&
  77. element.parentNode.ownerDocument.defaultView &&
  78. element.parentNode instanceof element.parentNode.ownerDocument.defaultView.HTMLElement
  79. ) {
  80. this._parentNode = element.parentNode;
  81. this.observer.observe(this._parentNode);
  82. }
  83. }
  84. mergeRef = (ref: any, node: HTMLDivElement) => {
  85. this.childNode = node;
  86. if (typeof ref === 'function') {
  87. ref(node);
  88. } else if (typeof ref === 'object' && ref && 'current' in ref) {
  89. ref.current = node;
  90. }
  91. };
  92. render() {
  93. const child = React.Children.only(this.props.children);
  94. const { ref } = child as React.ComponentPropsWithRef<any>;
  95. return React.cloneElement(child as React.ReactElement, {
  96. ref: (node: HTMLDivElement) => this.mergeRef(ref, node),
  97. });
  98. }
  99. }