WaveEffect.js 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. "use strict";
  2. "use client";
  3. var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault").default;
  4. var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard").default;
  5. Object.defineProperty(exports, "__esModule", {
  6. value: true
  7. });
  8. exports.default = void 0;
  9. var React = _interopRequireWildcard(require("react"));
  10. var _classnames = _interopRequireDefault(require("classnames"));
  11. var _rcMotion = _interopRequireDefault(require("rc-motion"));
  12. var _raf = _interopRequireDefault(require("rc-util/lib/raf"));
  13. var _ref = require("rc-util/lib/ref");
  14. var _UnstableContext = require("../../config-provider/UnstableContext");
  15. var _interface = require("./interface");
  16. var _util = require("./util");
  17. function validateNum(value) {
  18. return Number.isNaN(value) ? 0 : value;
  19. }
  20. const WaveEffect = props => {
  21. const {
  22. className,
  23. target,
  24. component,
  25. registerUnmount
  26. } = props;
  27. const divRef = React.useRef(null);
  28. // ====================== Refs ======================
  29. const unmountRef = React.useRef(null);
  30. React.useEffect(() => {
  31. unmountRef.current = registerUnmount();
  32. }, []);
  33. // ===================== Effect =====================
  34. const [color, setWaveColor] = React.useState(null);
  35. const [borderRadius, setBorderRadius] = React.useState([]);
  36. const [left, setLeft] = React.useState(0);
  37. const [top, setTop] = React.useState(0);
  38. const [width, setWidth] = React.useState(0);
  39. const [height, setHeight] = React.useState(0);
  40. const [enabled, setEnabled] = React.useState(false);
  41. const waveStyle = {
  42. left,
  43. top,
  44. width,
  45. height,
  46. borderRadius: borderRadius.map(radius => `${radius}px`).join(' ')
  47. };
  48. if (color) {
  49. waveStyle['--wave-color'] = color;
  50. }
  51. function syncPos() {
  52. const nodeStyle = getComputedStyle(target);
  53. // Get wave color from target
  54. setWaveColor((0, _util.getTargetWaveColor)(target));
  55. const isStatic = nodeStyle.position === 'static';
  56. // Rect
  57. const {
  58. borderLeftWidth,
  59. borderTopWidth
  60. } = nodeStyle;
  61. setLeft(isStatic ? target.offsetLeft : validateNum(-parseFloat(borderLeftWidth)));
  62. setTop(isStatic ? target.offsetTop : validateNum(-parseFloat(borderTopWidth)));
  63. setWidth(target.offsetWidth);
  64. setHeight(target.offsetHeight);
  65. // Get border radius
  66. const {
  67. borderTopLeftRadius,
  68. borderTopRightRadius,
  69. borderBottomLeftRadius,
  70. borderBottomRightRadius
  71. } = nodeStyle;
  72. setBorderRadius([borderTopLeftRadius, borderTopRightRadius, borderBottomRightRadius, borderBottomLeftRadius].map(radius => validateNum(parseFloat(radius))));
  73. }
  74. React.useEffect(() => {
  75. if (target) {
  76. // We need delay to check position here
  77. // since UI may change after click
  78. const id = (0, _raf.default)(() => {
  79. syncPos();
  80. setEnabled(true);
  81. });
  82. // Add resize observer to follow size
  83. let resizeObserver;
  84. if (typeof ResizeObserver !== 'undefined') {
  85. resizeObserver = new ResizeObserver(syncPos);
  86. resizeObserver.observe(target);
  87. }
  88. return () => {
  89. _raf.default.cancel(id);
  90. resizeObserver === null || resizeObserver === void 0 ? void 0 : resizeObserver.disconnect();
  91. };
  92. }
  93. }, []);
  94. if (!enabled) {
  95. return null;
  96. }
  97. const isSmallComponent = (component === 'Checkbox' || component === 'Radio') && (target === null || target === void 0 ? void 0 : target.classList.contains(_interface.TARGET_CLS));
  98. return /*#__PURE__*/React.createElement(_rcMotion.default, {
  99. visible: true,
  100. motionAppear: true,
  101. motionName: "wave-motion",
  102. motionDeadline: 5000,
  103. onAppearEnd: (_, event) => {
  104. var _a, _b;
  105. if (event.deadline || event.propertyName === 'opacity') {
  106. const holder = (_a = divRef.current) === null || _a === void 0 ? void 0 : _a.parentElement;
  107. (_b = unmountRef.current) === null || _b === void 0 ? void 0 : _b.call(unmountRef).then(() => {
  108. holder === null || holder === void 0 ? void 0 : holder.remove();
  109. });
  110. }
  111. return false;
  112. }
  113. }, ({
  114. className: motionClassName
  115. }, ref) => (/*#__PURE__*/React.createElement("div", {
  116. ref: (0, _ref.composeRef)(divRef, ref),
  117. className: (0, _classnames.default)(className, motionClassName, {
  118. 'wave-quick': isSmallComponent
  119. }),
  120. style: waveStyle
  121. })));
  122. };
  123. const showWaveEffect = (target, info) => {
  124. var _a;
  125. const {
  126. component
  127. } = info;
  128. // Skip for unchecked checkbox
  129. if (component === 'Checkbox' && !((_a = target.querySelector('input')) === null || _a === void 0 ? void 0 : _a.checked)) {
  130. return;
  131. }
  132. // Create holder
  133. const holder = document.createElement('div');
  134. holder.style.position = 'absolute';
  135. holder.style.left = '0px';
  136. holder.style.top = '0px';
  137. target === null || target === void 0 ? void 0 : target.insertBefore(holder, target === null || target === void 0 ? void 0 : target.firstChild);
  138. const reactRender = (0, _UnstableContext.unstableSetRender)();
  139. let unmountCallback = null;
  140. function registerUnmount() {
  141. return unmountCallback;
  142. }
  143. unmountCallback = reactRender(/*#__PURE__*/React.createElement(WaveEffect, Object.assign({}, info, {
  144. target: target,
  145. registerUnmount: registerUnmount
  146. })), holder);
  147. };
  148. var _default = exports.default = showWaveEffect;