Skip to content

Commit e965152

Browse files
committed
[added] Affix and AutoAffix
1 parent 87fad33 commit e965152

11 files changed

+767
-8
lines changed

examples/Affix.js

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
import React from 'react';
2+
import AutoAffix from 'react-overlays/lib/AutoAffix';
3+
4+
class AffixExample extends React.Component {
5+
render() {
6+
return (
7+
<div className='affix-example'>
8+
<AutoAffix viewportOffsetTop={15} container={this}>
9+
<div className='panel panel-default'>
10+
<div className='panel-body'>
11+
I am an affixed element
12+
</div>
13+
</div>
14+
</AutoAffix>
15+
</div>
16+
);
17+
}
18+
}
19+
20+
export default AffixExample;

examples/App.js

+24-1
Original file line numberDiff line numberDiff line change
@@ -5,24 +5,30 @@ import Editor from '@jquense/component-playground';
55

66
import PropTable from './PropTable';
77

8+
import AffixSource from '../webpack/example-loader!./Affix';
89
import ModalExample from '../webpack/example-loader!./Modal';
910
import OverlaySource from '../webpack/example-loader!./Overlay';
1011
import PortalSource from '../webpack/example-loader!./Portal';
1112
import PositionSource from '../webpack/example-loader!./Position';
1213
import TransitionSource from '../webpack/example-loader!./Transition';
1314

15+
import AffixMetadata from '../webpack/metadata-loader!react-overlays/Affix';
16+
import AutoAffixMetadata from '../webpack/metadata-loader!react-overlays/AutoAffix';
1417
import PortalMetadata from '../webpack/metadata-loader!react-overlays/Portal';
1518
import PositionMetadata from '../webpack/metadata-loader!react-overlays/Position';
1619
import OverlayMetadata from '../webpack/metadata-loader!react-overlays/Overlay';
1720
import ModalMetadata from '../webpack/metadata-loader!react-overlays/Modal';
1821
import TransitionMetadata from '../webpack/metadata-loader!react-overlays/Transition';
1922

2023
import * as ReactOverlays from 'react-overlays';
24+
import getOffset from 'dom-helpers/query/offset';
2125

2226
import './styles.less';
2327
import injectCss from './injectCss';
2428

25-
let scope = { React, findDOMNode, Button, injectCss, ...ReactOverlays };
29+
let scope = {
30+
React, findDOMNode, Button, injectCss, ...ReactOverlays, getOffset
31+
};
2632

2733
const Anchor = React.createClass({
2834
propTypes: {
@@ -72,6 +78,7 @@ const Example = React.createClass({
7278
<li><a href='#modals'>Modals</a></li>
7379
<li><a href='#position'>Position</a></li>
7480
<li><a href='#overlay'>Overlay</a></li>
81+
<li><a href='#affixes'>Affixes</a></li>
7582
</ul>
7683
</article>
7784
<main className='col-md-10'>
@@ -130,6 +137,22 @@ const Example = React.createClass({
130137
metadata={OverlayMetadata}
131138
/>
132139
</section>
140+
<section>
141+
<h2 className='page-header'>
142+
<Anchor>Affixes</Anchor>
143+
</h2>
144+
<p dangerouslySetInnerHTML={{__html: AffixMetadata.Affix.descHtml }}/>
145+
<p dangerouslySetInnerHTML={{__html: AutoAffixMetadata.AutoAffix.descHtml }}/>
146+
<ExampleEditor codeText={AffixSource} />
147+
<PropTable
148+
component='Affix'
149+
metadata={AffixMetadata}
150+
/>
151+
<PropTable
152+
component='AutoAffix'
153+
metadata={AutoAffixMetadata}
154+
/>
155+
</section>
133156
</main>
134157
</div>
135158
);

examples/PropTable.js

+4-3
Original file line numberDiff line numberDiff line change
@@ -40,16 +40,17 @@ const PropTable = React.createClass({
4040

4141
render(){
4242
let propsData = this.propsData;
43-
let composes = this.props.metadata[this.props.component].composes || [];
44-
4543
if ( !Object.keys(propsData).length ){
4644
return <span/>;
4745
}
4846

47+
let {component, metadata} = this.props;
48+
let composes = metadata[component].composes || [];
49+
4950
return (
5051
<div>
5152
<h3>
52-
Props
53+
{component} Props
5354
{ !!composes.length && [<br/>,
5455
<small>
5556
{'Also accepts the same props as: '}

examples/styles.less

+5-1
Original file line numberDiff line numberDiff line change
@@ -187,4 +187,8 @@ h4 a:focus .anchor-icon {
187187
button {
188188
margin-bottom: 10px;
189189
}
190-
}
190+
}
191+
192+
.affix-example {
193+
height: 500px;
194+
}

package.json

+2-2
Original file line numberDiff line numberDiff line change
@@ -76,9 +76,9 @@
7676
"mt-changelog": "^0.6.1",
7777
"node-libs-browser": "^0.5.2",
7878
"raw-loader": "^0.5.1",
79-
"react": "0.14.0",
79+
"react": "^0.14.0",
8080
"react-addons-test-utils": "^0.14.0",
81-
"react-bootstrap": "0.24.5-react-pre.0",
81+
"react-bootstrap": "^0.27.3",
8282
"react-component-metadata": "^1.2.2",
8383
"react-dom": "^0.14.0",
8484
"react-hot-loader": "^1.2.7",

src/Affix.js

+213
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,213 @@
1+
import classNames from 'classnames';
2+
import getHeight from 'dom-helpers/query/height';
3+
import getOffset from 'dom-helpers/query/offset';
4+
import getOffsetParent from 'dom-helpers/query/offsetParent';
5+
import getScrollTop from 'dom-helpers/query/scrollTop';
6+
import requestAnimationFrame from 'dom-helpers/util/requestAnimationFrame';
7+
import React from 'react';
8+
import ReactDOM from 'react-dom';
9+
10+
import addEventListener from './utils/addEventListener';
11+
import getDocumentHeight from './utils/getDocumentHeight';
12+
import ownerDocument from './utils/ownerDocument';
13+
import ownerWindow from './utils/ownerWindow';
14+
15+
/**
16+
* The `<Affix/>` component toggles `position: fixed;` on and off, emulating
17+
* the effect found with `position: sticky;`.
18+
*/
19+
class Affix extends React.Component {
20+
constructor(props, context) {
21+
super(props, context);
22+
23+
this.state = {
24+
affixed: 'top',
25+
position: null,
26+
top: null
27+
};
28+
29+
this._needPositionUpdate = false;
30+
}
31+
32+
componentDidMount() {
33+
this._isMounted = true;
34+
35+
this._windowScrollListener = addEventListener(
36+
ownerWindow(this), 'scroll', () => this.onWindowScroll()
37+
);
38+
this._documentClickListener = addEventListener(
39+
ownerDocument(this), 'click', () => this.onDocumentClick()
40+
);
41+
42+
this.onUpdate();
43+
}
44+
45+
componentWillReceiveProps() {
46+
this._needPositionUpdate = true;
47+
}
48+
49+
componentDidUpdate() {
50+
if (this._needPositionUpdate) {
51+
this._needPositionUpdate = false;
52+
this.onUpdate();
53+
}
54+
}
55+
56+
componentWillUnmount() {
57+
this._isMounted = false;
58+
59+
if (this._windowScrollListener) {
60+
this._windowScrollListener.remove();
61+
}
62+
if (this._documentClickListener) {
63+
this._documentClickListener.remove();
64+
}
65+
}
66+
67+
onWindowScroll() {
68+
this.onUpdate();
69+
}
70+
71+
onDocumentClick() {
72+
requestAnimationFrame(() => this.onUpdate());
73+
}
74+
75+
onUpdate() {
76+
if (!this._isMounted) {
77+
return;
78+
}
79+
80+
const {offsetTop, viewportOffsetTop} = this.props;
81+
const scrollTop = getScrollTop(ownerWindow(this));
82+
const positionTopMin = scrollTop + (viewportOffsetTop || 0);
83+
84+
if (positionTopMin <= offsetTop) {
85+
this.updateState('top', null, null);
86+
return;
87+
}
88+
89+
if (positionTopMin > this.getPositionTopMax()) {
90+
if (this.state.affixed === 'bottom') {
91+
this.updateStateAtBottom();
92+
} else {
93+
// Setting position away from `fixed` can change the offset parent of
94+
// the affix, so we can't calculate the correct position until after
95+
// we've updated its position.
96+
this.setState({
97+
affixed: 'bottom',
98+
position: 'absolute',
99+
top: null
100+
}, () => {
101+
if (!this._isMounted) {
102+
return;
103+
}
104+
105+
this.updateStateAtBottom();
106+
});
107+
}
108+
return;
109+
}
110+
111+
this.updateState('affix', 'fixed', viewportOffsetTop);
112+
}
113+
114+
getPositionTopMax() {
115+
const documentHeight = getDocumentHeight(ownerDocument(this));
116+
const height = getHeight(ReactDOM.findDOMNode(this));
117+
118+
return documentHeight - height - this.props.offsetBottom;
119+
}
120+
121+
updateState(affixed, position, top) {
122+
if (
123+
affixed === this.state.affixed &&
124+
position === this.state.position &&
125+
top === this.state.top
126+
) {
127+
return;
128+
}
129+
130+
this.setState({affixed, position, top});
131+
}
132+
133+
updateStateAtBottom() {
134+
const positionTopMax = this.getPositionTopMax();
135+
const offsetParent = getOffsetParent(ReactDOM.findDOMNode(this));
136+
const parentTop = getOffset(offsetParent).top;
137+
138+
this.updateState('bottom', 'absolute', positionTopMax - parentTop);
139+
}
140+
141+
render() {
142+
const child = React.Children.only(this.props.children);
143+
const {className, style} = child.props;
144+
145+
const {affixed, position, top} = this.state;
146+
const positionStyle = {position, top};
147+
148+
let affixClassName;
149+
let affixStyle;
150+
if (affixed === 'top') {
151+
affixClassName = this.props.topClassName;
152+
affixStyle = this.props.topStyle;
153+
} else if (affixed === 'bottom') {
154+
affixClassName = this.props.bottomClassName;
155+
affixStyle = this.props.bottomStyle;
156+
} else {
157+
affixClassName = this.props.affixClassName;
158+
affixStyle = this.props.affixStyle;
159+
}
160+
161+
return React.cloneElement(child, {
162+
className: classNames(affixClassName, className),
163+
style: {...positionStyle, ...affixStyle, ...style}
164+
});
165+
}
166+
}
167+
168+
Affix.propTypes = {
169+
/**
170+
* Pixels to offset from top of screen when calculating position
171+
*/
172+
offsetTop: React.PropTypes.number,
173+
/**
174+
* When affixed, pixels to offset from top of viewport
175+
*/
176+
viewportOffsetTop: React.PropTypes.number,
177+
/**
178+
* Pixels to offset from bottom of screen when calculating position
179+
*/
180+
offsetBottom: React.PropTypes.number,
181+
/**
182+
* CSS class or classes to apply when at top
183+
*/
184+
topClassName: React.PropTypes.string,
185+
/**
186+
* Style to apply when at top
187+
*/
188+
topStyle: React.PropTypes.object,
189+
/**
190+
* CSS class or classes to apply when affixed
191+
*/
192+
affixClassName: React.PropTypes.string,
193+
/**
194+
* Style to apply when affixed
195+
*/
196+
affixStyle: React.PropTypes.object,
197+
/**
198+
* CSS class or classes to apply when at bottom
199+
*/
200+
bottomClassName: React.PropTypes.string,
201+
/**
202+
* Style to apply when at bottom
203+
*/
204+
bottomStyle: React.PropTypes.object
205+
};
206+
207+
Affix.defaultProps = {
208+
offsetTop: 0,
209+
viewportOffsetTop: null,
210+
offsetBottom: 0
211+
};
212+
213+
export default Affix;

0 commit comments

Comments
 (0)