@nutui/nutui-react
Version:
京东风格的轻量级移动端 React 组件库,支持一套代码生成 H5 和小程序
166 lines (165 loc) • 7.45 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
Object.defineProperty(exports, "Sticky", {
enumerable: true,
get: function() {
return Sticky;
}
});
var _interop_require_default = require("@swc/helpers/_/_interop_require_default");
var _interop_require_wildcard = require("@swc/helpers/_/_interop_require_wildcard");
var _define_property = require("@swc/helpers/_/_define_property");
var _object_spread = require("@swc/helpers/_/_object_spread");
var _object_spread_props = require("@swc/helpers/_/_object_spread_props");
var _object_without_properties = require("@swc/helpers/_/_object_without_properties");
var _sliced_to_array = require("@swc/helpers/_/_sliced_to_array");
var _react = /*#__PURE__*/ _interop_require_wildcard._(require("react"));
var _classnames = /*#__PURE__*/ _interop_require_default._(require("classnames"));
var _getscrollparent = require("../../utils/get-scroll-parent");
var _getrect = require("../../utils/get-rect");
var _usewatch = require("../../hooks/use-watch");
var _typings = require("../../utils/typings");
var defaultProps = (0, _object_spread_props._)((0, _object_spread._)({}, _typings.ComponentDefaults), {
position: 'top',
threshold: 0,
zIndex: 900
});
var classPrefix = 'nut-sticky';
var Sticky = function Sticky(props) {
var _ref = (0, _object_spread._)({}, defaultProps, props), position = _ref.position, zIndex = _ref.zIndex, children = _ref.children, container = _ref.container, style = _ref.style, className = _ref.className, threshold = _ref.threshold, onChange = _ref.onChange, rest = (0, _object_without_properties._)(_ref, [
"position",
"zIndex",
"children",
"container",
"style",
"className",
"threshold",
"onChange"
]);
var stickyRef = (0, _react.useRef)(null);
var rootRef = (0, _react.useRef)(null);
var _useState = (0, _sliced_to_array._)((0, _react.useState)(false), 2), isFixed = _useState[0], setIsFixed = _useState[1];
var _obj;
var _useState1 = (0, _sliced_to_array._)((0, _react.useState)((_obj = {}, (0, _define_property._)(_obj, position, "".concat(threshold, "px")), (0, _define_property._)(_obj, "zIndex", zIndex), _obj)), 2), stickyStyle = _useState1[0], setStickyStyle = _useState1[1];
(0, _react.useEffect)(function() {
var _obj;
setStickyStyle((0, _object_spread_props._)((0, _object_spread._)({}, stickyStyle), (_obj = {}, (0, _define_property._)(_obj, position, "".concat(threshold, "px")), (0, _define_property._)(_obj, "zIndex", zIndex), _obj)));
}, [
threshold,
position,
zIndex
]);
var _useState2 = (0, _sliced_to_array._)((0, _react.useState)({}), 2), rootStyle = _useState2[0], setRootStyle = _useState2[1];
var getElement = (0, _react.useCallback)(function() {
return (0, _getscrollparent.getScrollParent)(rootRef.current);
}, []);
(0, _react.useEffect)(function() {
if (position === 'top') return;
var containerEle = container && container.current;
var rootEle = rootRef.current;
var stickyEle = stickyRef.current;
if (!rootEle && !containerEle) return;
var rootRect = (0, _getrect.getRect)(rootEle);
var containerRect = (0, _getrect.getRect)(containerEle);
var clientHeight = document.documentElement.clientHeight;
var stickyRect = (0, _getrect.getRect)(stickyEle);
var fixed = clientHeight - threshold < rootRect.bottom;
if (containerEle) {
fixed = containerRect.bottom > clientHeight - threshold - stickyRect.height && clientHeight - threshold - stickyRect.height > containerRect.top;
}
var defaultPostVal = fixed ? 'fixed' : 'inherit';
setStickyStyle(function(prev) {
return (0, _object_spread_props._)((0, _object_spread._)({}, prev), {
position: defaultPostVal
});
});
setIsFixed(fixed);
}, [
position,
container,
threshold
]);
var handleScroll = (0, _react.useCallback)(function() {
var containerEle = container && container.current;
var rootEle = rootRef.current;
var stickyEle = stickyRef.current;
if (!rootEle && !containerEle) return;
var rootRect = (0, _getrect.getRect)(rootEle);
var stickyRect = (0, _getrect.getRect)(stickyEle);
var containerRect = (0, _getrect.getRect)(containerEle);
if (rootRect.height) {
setRootStyle(function(prev) {
return (0, _object_spread_props._)((0, _object_spread._)({}, prev), {
height: rootRect.height
});
});
}
var getFixed = function getFixed() {
var fixed = false;
if (position === 'top') {
fixed = containerEle ? threshold > rootRect.top && containerRect.bottom > 0 : threshold > rootRect.top;
} else {
var clientHeight = document.documentElement.clientHeight;
fixed = containerEle ? containerRect.bottom > 0 && clientHeight - threshold - stickyRect.height > containerRect.top : clientHeight - threshold < rootRect.bottom;
}
return {
position: fixed ? 'fixed' : 'inherit',
fixed: fixed
};
};
var getTransform = function getTransform() {
if (position === 'top' && containerEle) {
var diff = containerRect.bottom - threshold - stickyRect.height;
var transform = diff < 0 ? diff : 0;
return {
transform: "translate3d(0, ".concat(transform, "px, 0)")
};
}
if (position === 'bottom' && containerEle) {
var clientHeight = document.documentElement.clientHeight;
var diff1 = containerRect.bottom - (clientHeight - threshold);
var transform1 = diff1 < 0 ? diff1 : 0;
return {
transform: "translate3d(0, ".concat(transform1, "px, 0)")
};
}
return {};
};
var fixed = getFixed();
setStickyStyle(function(prev) {
return (0, _object_spread_props._)((0, _object_spread._)({}, prev, getTransform()), {
position: fixed.position
});
});
setIsFixed(fixed.fixed);
}, [
position,
threshold,
container
]);
(0, _usewatch.useWatch)(isFixed, function() {
onChange && onChange(isFixed);
});
(0, _react.useEffect)(function() {
var el = getElement();
el === null || el === void 0 ? void 0 : el.addEventListener('scroll', handleScroll, false);
return function() {
el === null || el === void 0 ? void 0 : el.removeEventListener('scroll', handleScroll);
};
}, [
getElement,
handleScroll
]);
return /*#__PURE__*/ _react.default.createElement("div", (0, _object_spread._)({
ref: rootRef,
style: (0, _object_spread._)({}, style, rootStyle),
className: (0, _classnames.default)(classPrefix, className)
}, rest), /*#__PURE__*/ _react.default.createElement("div", {
className: "nut-sticky-box",
ref: stickyRef,
style: stickyStyle
}, children));
};
Sticky.displayName = 'NutSticky';