ant-design/components/affix/index.tsx

235 lines
6.3 KiB
TypeScript
Raw Normal View History

2016-06-22 13:18:43 +08:00
import * as React from 'react';
import * as ReactDOM from 'react-dom';
2016-06-16 22:34:54 +08:00
import addEventListener from 'rc-util/lib/Dom/addEventListener';
import classNames from 'classnames';
2016-07-05 10:46:00 +08:00
import assign from 'object-assign';
2016-07-09 16:00:05 +08:00
import shallowequal from 'shallowequal';
2015-08-03 16:49:42 +08:00
function getScroll(target, top) {
const prop = top ? 'pageYOffset' : 'pageXOffset';
const method = top ? 'scrollTop' : 'scrollLeft';
const isWindow = target === window;
let ret = isWindow ? target[prop] : target[method];
// ie6,7,8 standard mode
if (isWindow && typeof ret !== 'number') {
ret = window.document.documentElement[method];
2015-08-04 14:57:18 +08:00
}
2015-08-04 14:57:18 +08:00
return ret;
}
2016-08-22 17:18:26 +08:00
function getTargetRect(target): any {
return target !== window ?
target.getBoundingClientRect() :
{ top: 0, left: 0, bottom: 0 };
}
function getOffset(element, target) {
const elemRect = element.getBoundingClientRect();
const targetRect = getTargetRect(target);
const scrollTop = getScroll(target, true);
const scrollLeft = getScroll(target, false);
const docElem = window.document.body;
const clientTop = docElem.clientTop || 0;
const clientLeft = docElem.clientLeft || 0;
2015-08-04 14:57:18 +08:00
return {
top: elemRect.top - targetRect.top +
scrollTop - clientTop,
left: elemRect.left - targetRect.left +
scrollLeft - clientLeft,
2015-08-04 14:57:18 +08:00
};
}
2016-06-22 13:18:43 +08:00
// Affix
export interface AffixProps {
/**
*
*/
2016-07-13 11:14:24 +08:00
offsetTop?: number;
offset?: number;
offsetBottom?: number;
style?: React.CSSProperties;
onChange?: (affixed?: boolean) => any;
2016-08-22 17:18:26 +08:00
target?: () => Window | HTMLElement;
2016-06-22 13:18:43 +08:00
}
export default class Affix extends React.Component<AffixProps, any> {
static propTypes = {
2016-02-29 16:44:38 +08:00
offsetTop: React.PropTypes.number,
offsetBottom: React.PropTypes.number,
target: React.PropTypes.func,
2016-08-23 21:00:35 +08:00
};
2015-09-01 16:18:46 +08:00
2016-05-26 16:06:15 +08:00
static defaultProps = {
target() {
return window;
},
2016-05-26 16:06:15 +08:00
onChange() {},
2016-07-13 11:14:24 +08:00
};
2016-05-26 16:06:15 +08:00
2016-07-13 17:22:23 +08:00
scrollEvent: any;
resizeEvent: any;
refs: {
2016-08-01 13:30:13 +08:00
[key: string]: any;
fixedNode: HTMLElement;
2016-07-13 17:22:23 +08:00
};
constructor(props) {
super(props);
this.state = {
2016-02-29 12:10:36 +08:00
affixStyle: null,
2016-07-11 14:52:21 +08:00
placeholderStyle: null,
2015-08-03 16:49:42 +08:00
};
}
2015-08-03 16:49:42 +08:00
2016-07-11 14:52:21 +08:00
setAffixStyle(e, affixStyle) {
const { onChange, target } = this.props;
2016-07-09 16:00:05 +08:00
const originalAffixStyle = this.state.affixStyle;
const isWindow = target() === window;
if (e.type === 'scroll' && originalAffixStyle && affixStyle && isWindow) {
2016-07-11 14:52:21 +08:00
return;
}
2016-07-09 16:00:05 +08:00
if (shallowequal(affixStyle, originalAffixStyle)) {
return;
}
this.setState({ affixStyle }, () => {
const affixed = !!this.state.affixStyle;
if ((affixStyle && !originalAffixStyle) ||
(!affixStyle && originalAffixStyle)) {
onChange(affixed);
2016-07-09 16:00:05 +08:00
}
});
}
2016-07-11 14:52:21 +08:00
setPlaceholderStyle(e, placeholderStyle) {
const originalPlaceholderStyle = this.state.placeholderStyle;
if (e.type === 'resize') {
return;
}
if (shallowequal(placeholderStyle, originalPlaceholderStyle)) {
return;
}
this.setState({ placeholderStyle });
}
updatePosition = (e) => {
let { offsetTop, offsetBottom, offset, target } = this.props;
const targetNode = target();
// Backwards support
offsetTop = offsetTop || offset;
const scrollTop = getScroll(targetNode, true);
2016-08-22 17:18:26 +08:00
const affixNode = ReactDOM.findDOMNode(this) as HTMLElement;
const elemOffset = getOffset(affixNode, targetNode);
2016-02-29 16:44:38 +08:00
const elemSize = {
width: this.refs.fixedNode.offsetWidth,
height: this.refs.fixedNode.offsetHeight,
2016-02-29 16:44:38 +08:00
};
2015-08-03 16:49:42 +08:00
2016-07-12 16:07:12 +08:00
const offsetMode = {
top: null as boolean,
bottom: null as boolean,
};
// Default to `offsetTop=0`.
2016-02-29 16:44:38 +08:00
if (typeof offsetTop !== 'number' && typeof offsetBottom !== 'number') {
offsetMode.top = true;
offsetTop = 0;
} else {
offsetMode.top = typeof offsetTop === 'number';
offsetMode.bottom = typeof offsetBottom === 'number';
2015-08-03 16:49:42 +08:00
}
const targetRect = getTargetRect(targetNode);
2016-08-24 16:09:55 +08:00
const targetInnerHeight =
(targetNode as Window).innerHeight || (targetNode as HTMLElement).clientHeight;
2016-02-29 16:44:38 +08:00
if (scrollTop > elemOffset.top - offsetTop && offsetMode.top) {
// Fixed Top
2016-07-11 14:52:21 +08:00
this.setAffixStyle(e, {
2016-07-09 16:00:05 +08:00
position: 'fixed',
top: targetRect.top + offsetTop,
left: targetRect.left + elemOffset.left,
2016-08-22 17:18:26 +08:00
width: affixNode.offsetWidth,
2016-07-09 16:00:05 +08:00
});
2016-07-11 14:52:21 +08:00
this.setPlaceholderStyle(e, {
2016-07-12 16:07:12 +08:00
width: affixNode.offsetWidth,
height: affixNode.offsetHeight,
2016-07-11 14:52:21 +08:00
});
} else if (
scrollTop < elemOffset.top + elemSize.height + offsetBottom - targetInnerHeight &&
offsetMode.bottom
) {
2016-02-29 16:44:38 +08:00
// Fixed Bottom
const targetBottomOffet = targetNode === window ? 0 : (window.innerHeight - targetRect.bottom);
2016-07-11 14:52:21 +08:00
this.setAffixStyle(e, {
2016-07-09 16:00:05 +08:00
position: 'fixed',
bottom: targetBottomOffet + offsetBottom,
left: targetRect.left + elemOffset.left,
2016-08-22 17:18:26 +08:00
width: affixNode.offsetWidth,
2016-07-09 16:00:05 +08:00
});
2016-07-11 14:52:21 +08:00
this.setPlaceholderStyle(e, {
2016-07-12 16:07:12 +08:00
width: affixNode.offsetWidth,
height: affixNode.offsetHeight,
2016-07-11 14:52:21 +08:00
});
2016-07-09 16:00:05 +08:00
} else {
2016-07-11 14:52:21 +08:00
this.setAffixStyle(e, null);
this.setPlaceholderStyle(e, null);
2015-08-03 16:49:42 +08:00
}
}
2015-08-03 16:49:42 +08:00
componentDidMount() {
const target = this.props.target;
this.setTargetEventListeners(target);
}
2015-08-03 16:49:42 +08:00
componentWillReceiveProps(nextProps) {
if (this.props.target !== nextProps.target) {
this.clearScrollEventListeners();
this.setTargetEventListeners(nextProps.target);
// Mock Event object.
this.updatePosition({});
2015-08-04 15:03:00 +08:00
}
}
2015-08-03 16:49:42 +08:00
componentWillUnmount() {
this.clearScrollEventListeners();
}
setTargetEventListeners(getTarget) {
const target = getTarget();
this.scrollEvent = addEventListener(target, 'scroll', this.updatePosition);
this.resizeEvent = addEventListener(target, 'resize', this.updatePosition);
}
clearScrollEventListeners() {
['scrollEvent', 'resizeEvent'].forEach((name) => {
if (this[name]) {
this[name].remove();
}
});
}
2015-08-03 16:49:42 +08:00
render() {
const className = classNames({
2016-02-29 16:44:38 +08:00
'ant-affix': this.state.affixStyle,
});
2015-08-03 16:49:42 +08:00
2016-07-05 10:46:00 +08:00
const props = assign({}, this.props);
delete props.offsetTop;
delete props.offsetBottom;
delete props.target;
2015-08-03 16:49:42 +08:00
return (
2016-07-11 14:52:21 +08:00
<div {...props} style={this.state.placeholderStyle}>
2016-02-29 16:44:38 +08:00
<div className={className} ref="fixedNode" style={this.state.affixStyle}>
2015-08-04 14:57:18 +08:00
{this.props.children}
</div>
2015-08-03 16:49:42 +08:00
</div>
);
}
}