@ysmood/material-ui
Version:
Material Design UI components built with React
148 lines (124 loc) • 3.94 kB
JSX
var React = require('react/addons');
var TabTemplate = require('./tabTemplate');
var InkBar = require('../ink-bar');
var StylePropable = require('../mixins/style-propable.js');
var Events = require('../utils/events');
var Tabs = React.createClass({
mixins: [StylePropable],
contextTypes: {
muiTheme: React.PropTypes.object
},
propTypes: {
initialSelectedIndex: React.PropTypes.number,
onActive: React.PropTypes.func,
tabWidth: React.PropTypes.number
},
getInitialState: function(){
var selectedIndex = 0;
if (this.props.initialSelectedIndex && this.props.initialSelectedIndex < this.props.children.length) {
selectedIndex = this.props.initialSelectedIndex;
}
return {
selectedIndex: selectedIndex
};
},
getEvenWidth: function(){
return (
parseInt(window
.getComputedStyle(React.findDOMNode(this))
.getPropertyValue('width'), 10)
);
},
componentDidMount: function() {
this._updateTabWidth();
Events.on(window, 'resize', this._updateTabWidth);
},
componentWillUnmount: function() {
Events.off(window, 'resize', this._updateTabWidth);
},
componentWillReceiveProps: function(newProps) {
if (newProps.hasOwnProperty('style')) this._updateTabWidth();
},
handleTouchTap: function(tabIndex, tab){
if (this.props.onChange && this.state.selectedIndex !== tabIndex) {
this.props.onChange(tabIndex, tab);
}
this.setState({selectedIndex: tabIndex});
//default CB is _onActive. Can be updated in tab.jsx
if(tab.props.onActive) tab.props.onActive(tab);
},
getStyles: function() {
var themeVariables = this.context.muiTheme.component.tabs;
return {
tabItemContainer: {
margin: '0',
padding: '0',
width: '100%',
height: '48px',
backgroundColor: themeVariables.backgroundColor,
whiteSpace: 'nowrap',
display: 'table'
}
};
},
render: function(){
var styles = this.getStyles();
var tabContent = []
var width = this.state.fixedWidth ?
100 / this.props.children.length +'%' :
this.props.tabWidth + 'px';
var left = 'calc(' + width + '*' + this.state.selectedIndex + ')';
var tabs = React.Children.map(this.props.children, function(tab, index){
if(tab.type.displayName === "Tab") {
if(tab.props.children) {
tabContent.push(React.createElement(TabTemplate, {
key: index,
selected: this.state.selectedIndex === index
}, tab.props.children));
} else {
tabContent.push(undefined)
}
return React.addons.cloneWithProps(tab, {
key: index,
selected: this.state.selectedIndex === index,
tabIndex: index,
width: width,
handleTouchTap: this.handleTouchTap
});
} else {
var type = tab.type.displayName || tab.type;
throw 'Tabs only accepts Tab Components as children. Found ' +
type + ' as child number ' + (index + 1) + ' of Tabs';
}
}, this);
return (
<div style={this.mergeAndPrefix(this.props.style)}>
<div style={this.mergeAndPrefix(styles.tabItemContainer, this.props.tabItemContainerStyle)}>
{tabs}
</div>
<InkBar left={left} width={width} />
<div>
{tabContent}
</div>
</div>
)
},
_tabWidthPropIsValid: function() {
return this.props.tabWidth &&
(this.props.tabWidth * this.props.children.length <= this.getEvenWidth());
},
// Validates that the tabWidth can fit all tabs on the tab bar. If not, the
// tabWidth is recalculated and fixed.
_updateTabWidth: function() {
if(this._tabWidthPropIsValid()) {
this.setState({
fixedWidth: false
});
} else {
this.setState({
fixedWidth: true
});
}
}
});
module.exports = Tabs;