antd
Version:
An enterprise-class UI design language and React-based implementation
261 lines (208 loc) • 9.44 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", {
value: true
});
exports["default"] = undefined;
var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; };
var _class, _temp;
var _react = require('react');
var _react2 = _interopRequireDefault(_react);
var _reactDom = require('react-dom');
var _reactDom2 = _interopRequireDefault(_reactDom);
var _addEventListener = require('rc-util/lib/Dom/addEventListener');
var _addEventListener2 = _interopRequireDefault(_addEventListener);
var _classnames = require('classnames');
var _classnames2 = _interopRequireDefault(_classnames);
var _warning = require('warning');
var _warning2 = _interopRequireDefault(_warning);
var _shallowequal = require('shallowequal');
var _shallowequal2 = _interopRequireDefault(_shallowequal);
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; }
function _defaults(obj, defaults) { var keys = Object.getOwnPropertyNames(defaults); for (var i = 0; i < keys.length; i++) { var key = keys[i]; var value = Object.getOwnPropertyDescriptor(defaults, key); if (value && value.configurable && obj[key] === undefined) { Object.defineProperty(obj, key, value); } } return obj; }
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : _defaults(subClass, superClass); }
function getScroll(target, top) {
var prop = top ? 'pageYOffset' : 'pageXOffset';
var method = top ? 'scrollTop' : 'scrollLeft';
var isWindow = target === window;
var ret = isWindow ? target[prop] : target[method];
// ie6,7,8 standard mode
if (isWindow && typeof ret !== 'number') {
ret = window.document.documentElement[method];
}
return ret;
}
function getTargetRect(target) {
return target !== window ? target.getBoundingClientRect() : { top: 0, left: 0, bottom: 0 };
}
function getOffset(element, target) {
var elemRect = element.getBoundingClientRect();
var targetRect = getTargetRect(target);
var scrollTop = getScroll(target, true);
var scrollLeft = getScroll(target, false);
var docElem = window.document.body;
var clientTop = docElem.clientTop || 0;
var clientLeft = docElem.clientLeft || 0;
return {
top: elemRect.top - targetRect.top + scrollTop - clientTop,
left: elemRect.left - targetRect.left + scrollLeft - clientLeft
};
}
var Affix = (_temp = _class = function (_React$Component) {
_inherits(Affix, _React$Component);
function Affix(props) {
_classCallCheck(this, Affix);
var _this = _possibleConstructorReturn(this, _React$Component.call(this, props));
_this.updatePosition = function (e) {
var _this$props = _this.props,
offsetTop = _this$props.offsetTop,
offsetBottom = _this$props.offsetBottom,
offset = _this$props.offset,
target = _this$props.target;
var targetNode = target();
// Backwards support
offsetTop = offsetTop || offset;
var scrollTop = getScroll(targetNode, true);
var elemOffset = getOffset(_reactDom2["default"].findDOMNode(_this), targetNode);
var elemSize = {
width: _this.refs.fixedNode.offsetWidth,
height: _this.refs.fixedNode.offsetHeight
};
var offsetMode = {};
// Default to `offsetTop=0`.
if (typeof offsetTop !== 'number' && typeof offsetBottom !== 'number') {
offsetMode.top = true;
offsetTop = 0;
} else {
offsetMode.top = typeof offsetTop === 'number';
offsetMode.bottom = typeof offsetBottom === 'number';
}
var targetRect = getTargetRect(targetNode);
var targetInnerHeight = targetNode.innerHeight || targetNode.clientHeight;
if (scrollTop > elemOffset.top - offsetTop && offsetMode.top) {
// Fixed Top
_this.setAffixStyle(e, {
position: 'fixed',
top: targetRect.top + offsetTop,
left: targetRect.left + elemOffset.left,
width: _reactDom2["default"].findDOMNode(_this).offsetWidth
});
_this.setPlaceholderStyle(e, {
width: _reactDom2["default"].findDOMNode(_this).offsetWidth,
height: _reactDom2["default"].findDOMNode(_this).offsetHeight
});
} else if (scrollTop < elemOffset.top + elemSize.height + offsetBottom - targetInnerHeight && offsetMode.bottom) {
// Fixed Bottom
var targetBottomOffet = targetNode === window ? 0 : window.innerHeight - targetRect.bottom;
_this.setAffixStyle(e, {
position: 'fixed',
bottom: targetBottomOffet + offsetBottom,
left: targetRect.left + elemOffset.left,
width: _reactDom2["default"].findDOMNode(_this).offsetWidth
});
_this.setPlaceholderStyle(e, {
width: _reactDom2["default"].findDOMNode(_this).offsetWidth,
height: _reactDom2["default"].findDOMNode(_this).offsetHeight
});
} else {
_this.setAffixStyle(e, null);
_this.setPlaceholderStyle(e, null);
}
};
_this.state = {
affixStyle: null,
placeholderStyle: null
};
return _this;
}
Affix.prototype.setAffixStyle = function setAffixStyle(e, affixStyle) {
var _this2 = this;
var _props = this.props,
onChange = _props.onChange,
target = _props.target;
var originalAffixStyle = this.state.affixStyle;
var isWindow = target() === window;
if (e.type === 'scroll' && originalAffixStyle && affixStyle && isWindow) {
return;
}
if ((0, _shallowequal2["default"])(affixStyle, originalAffixStyle)) {
return;
}
this.setState({ affixStyle: affixStyle }, function () {
var affixed = !!_this2.state.affixStyle;
if (affixStyle && !originalAffixStyle || !affixStyle && originalAffixStyle) {
onChange(affixed);
}
});
};
Affix.prototype.setPlaceholderStyle = function setPlaceholderStyle(e, placeholderStyle) {
var originalPlaceholderStyle = this.state.placeholderStyle;
if (e.type === 'resize') {
return;
}
if ((0, _shallowequal2["default"])(placeholderStyle, originalPlaceholderStyle)) {
return;
}
this.setState({ placeholderStyle: placeholderStyle });
};
Affix.prototype.componentDidMount = function componentDidMount() {
(0, _warning2["default"])(!('offset' in this.props), '`offset` prop of Affix is deprecated, use `offsetTop` instead.');
var target = this.props.target;
this.setTargetEventListeners(target);
};
Affix.prototype.componentWillReceiveProps = function componentWillReceiveProps(nextProps) {
if (this.props.target !== nextProps.target) {
this.clearScrollEventListeners();
this.setTargetEventListeners(nextProps.target);
// Mock Event object.
this.updatePosition({});
}
};
Affix.prototype.componentWillUnmount = function componentWillUnmount() {
this.clearScrollEventListeners();
};
Affix.prototype.setTargetEventListeners = function setTargetEventListeners(getTarget) {
var target = getTarget();
this.scrollEvent = (0, _addEventListener2["default"])(target, 'scroll', this.updatePosition);
this.resizeEvent = (0, _addEventListener2["default"])(target, 'resize', this.updatePosition);
};
Affix.prototype.clearScrollEventListeners = function clearScrollEventListeners() {
var _this3 = this;
['scrollEvent', 'resizeEvent'].forEach(function (name) {
if (_this3[name]) {
_this3[name].remove();
}
});
};
Affix.prototype.render = function render() {
var className = (0, _classnames2["default"])({
'ant-affix': this.state.affixStyle
});
var props = _extends({}, this.props);
delete props.offsetTop;
delete props.offsetBottom;
delete props.target;
return _react2["default"].createElement(
'div',
_extends({}, props, { style: this.state.placeholderStyle }),
_react2["default"].createElement(
'div',
{ className: className, ref: 'fixedNode', style: this.state.affixStyle },
this.props.children
)
);
};
return Affix;
}(_react2["default"].Component), _class.propTypes = {
offsetTop: _react2["default"].PropTypes.number,
offsetBottom: _react2["default"].PropTypes.number,
target: _react2["default"].PropTypes.func
}, _class.defaultProps = {
target: function target() {
return window;
},
onChange: function onChange() {}
}, _temp);
exports["default"] = Affix;
module.exports = exports['default'];