1
0

index.tsx 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  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. // eslint-disable-next-line
  50. return findDOMNode(this.childNode || this);
  51. } catch (error) {
  52. // swallow error if findDOMNode is run on unmounted component.
  53. return null;
  54. }
  55. };
  56. observeElement(force = false) {
  57. const element = this.getElement();
  58. if (!(element && element instanceof Element)) {
  59. // stop everything if not defined
  60. this.observer.disconnect();
  61. return;
  62. }
  63. if (element === this.element && !force) {
  64. // abort if given same element -- nothing to update (unless forced)
  65. return;
  66. } else {
  67. // clear observer list if new element
  68. this.observer.disconnect();
  69. // remember element reference for next time
  70. this.element = element;
  71. }
  72. // observer callback is invoked immediately when observing new elements
  73. this.observer.observe(element);
  74. if (
  75. this.props.observeParent &&
  76. element.parentNode &&
  77. element.parentNode.ownerDocument &&
  78. element.parentNode.ownerDocument.defaultView &&
  79. element.parentNode instanceof element.parentNode.ownerDocument.defaultView.HTMLElement
  80. ) {
  81. this._parentNode = element.parentNode;
  82. this.observer.observe(this._parentNode);
  83. }
  84. }
  85. mergeRef = (ref: any, node: HTMLDivElement) => {
  86. this.childNode = node;
  87. if (typeof ref === 'function') {
  88. ref(node);
  89. } else if (typeof ref === 'object' && ref && 'current' in ref) {
  90. ref.current = node;
  91. }
  92. };
  93. render() {
  94. const child = React.Children.only(this.props.children);
  95. const { ref } = child as any;
  96. return React.cloneElement(child as React.ReactElement, {
  97. ref: (node: HTMLDivElement) => this.mergeRef(ref, node),
  98. });
  99. }
  100. }