Skip to content
This repository has been archived by the owner on Feb 15, 2019. It is now read-only.

Commit

Permalink
Allow DrawerNavigationItem props to be overwritten in subclasses usin…
Browse files Browse the repository at this point in the history
…g getters or methods on the class
  • Loading branch information
dantman committed Mar 3, 2017
1 parent 55e5c4b commit 8cfc4dc
Show file tree
Hide file tree
Showing 2 changed files with 92 additions and 46 deletions.
108 changes: 65 additions & 43 deletions example/components/DrawerNavigationExample.js
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import React, { Component } from 'react';
import {
StyleSheet,
View,
Text,
Image,
} from 'react-native';
Expand All @@ -21,72 +22,93 @@ class Heading extends DrawerNavigationChild {
}
}

export default class DrawerNavigationExample extends Component {

_renderHeader = () => {
class DrawerItem extends DrawerNavigationItem {
renderIcon = (isSelected: bool) => {
let extraStyle = {marginTop: 2};
if (this.props.icon === 'md-alert') {
extraStyle = {...extraStyle, marginLeft: -3};
}
return (
<View style={{height: 180, width: 300}}>
<Image source={require('../assets/sparkles.jpg')} style={styles.header} />
</View>
<Ionicons
style={[styles.icon, isSelected ? styles.selectedText : null, extraStyle]}
name={this.props.icon}
size={24}
/>
);
};

_renderTitle = (text: string, isSelected: bool) => {
renderTitle = (isSelected: bool) => {
return (
<Text style={[styles.buttonTitleText, isSelected ? styles.selectedText : null]}>
{text}
{this.props.title}
</Text>
);
};

_renderIcon = (name: string, isSelected: bool) => {
let extraStyle = {marginTop: 2};
if (name === 'md-alert') {
extraStyle = {...extraStyle, marginLeft: -3};
}
get selectedItemStyle() {
return styles.selectedItemStyle;
}

get children() {
let { defaultRouteConfig } = this.props;

return (
<Ionicons
style={[styles.icon, isSelected ? styles.selectedText : null, extraStyle]}
name={name}
size={24}
<StackNavigation
id="root"
defaultRouteConfig={defaultRouteConfig}
initialRoute={Router.getRoute('home')}
/>
);
}
}

export default class DrawerNavigationExample extends Component {
renderHeader = () => {
return (
<View style={{height: 180, width: 300}}>
<Image source={require('../assets/sparkles.jpg')} style={styles.header} />
</View>
);
};

render() {
return (
<DrawerNavigation
drawerPosition="right"
renderHeader={this._renderHeader}
renderHeader={this.renderHeader}
drawerWidth={300}
initialItem="home">
<DrawerNavigationItem
<DrawerItem
id="home"
selectedStyle={styles.selectedItemStyle}
renderTitle={isSelected => this._renderTitle('Examples', isSelected)}
renderIcon={isSelected => this._renderIcon('md-apps', isSelected)}>
<StackNavigation
id="root"
defaultRouteConfig={{
navigationBar: {
backgroundColor: '#0084FF',
tintColor: '#fff',
},
}}
initialRoute={Router.getRoute('home')}
/>
</DrawerNavigationItem>
<Heading title='Meta' />
<DrawerNavigationItem
icon="md-apps"
title="Examples"
defaultRouteConfig={{
navigationBar: {
backgroundColor: '#0084FF',
tintColor: '#fff',
},
}}
stack={{
id: 'root',
initialRoute: Router.getRoute('home'),
}}
/>
<Heading title="Meta" />
<DrawerItem
id="another"
selectedStyle={styles.selectedItemStyle}
renderTitle={isSelected => this._renderTitle('About', isSelected)}
renderIcon={isSelected => this._renderIcon('md-alert', isSelected)}>
<StackNavigation
id="about"
initialRoute={Router.getRoute('about')}
/>
</DrawerNavigationItem>
icon="md-alert"
title="About"
defaultRouteConfig={{
navigationBar: {
backgroundColor: '#0084FF',
tintColor: '#fff',
},
}}
stack={{
id: 'root',
initialRoute: Router.getRoute('about'),
}}
/>
</DrawerNavigation>
);
}
Expand Down
30 changes: 27 additions & 3 deletions src/drawer/ExNavigationDrawerItem.js
Original file line number Diff line number Diff line change
Expand Up @@ -8,11 +8,34 @@ import TouchableNativeFeedbackSafe from '@exponent/react-native-touchable-native
import ExNavigationDrawerChild from './ExNavigationDrawerChild';
import type { Props } from './ExNavigationDrawerChild';

function _set(self, name, fn) {
// Stop use of `get` from interfering with `renderTitle => ...` in subclasses
Object.defineProperty(self, name, {
value: fn,
});
}

export default class ExNavigationDrawerItem extends ExNavigationDrawerChild {
props: Props;

get showsTouches() { return this.props.showsTouches; }
get renderIcon() { return this.props.renderIcon; }
get renderTitle() { return this.props.renderTitle; }
get renderRight() { return this.props.renderRight; }
get onPress() { return this.props.onPress; }
get onLongPress() { return this.props.onLongPress; }
get children() { return this.props.children; }
set showsTouches(fn) { _set(this, 'showsTouches', fn); }
set renderIcon(fn) { _set(this, 'renderIcon', fn); }
set renderTitle(fn) { _set(this, 'renderTitle', fn); }
set renderRight(fn) { _set(this, 'renderRight', fn); }
set onPress(fn) { _set(this, 'onPress', fn); }
set onLongPress(fn) { _set(this, 'onLongPress', fn); }
set children(fn) { _set(this, 'children', fn); }

renderDrawerItem() {
let { showsTouches, isSelected, renderIcon, renderTitle, renderRight, onPress, onLongPress } = this.props;
let { isSelected } = this.props;
let { showsTouches, renderIcon, renderTitle, renderRight, onPress, onLongPress } = this;
const icon = renderIcon && renderIcon(isSelected);
const title = renderTitle && renderTitle(isSelected);
const rightElement = renderRight && renderRight(isSelected);
Expand Down Expand Up @@ -60,8 +83,9 @@ export default class ExNavigationDrawerItem extends ExNavigationDrawerChild {
}

renderContent() {
if (React.Children.count(this.props.children) > 0) {
return React.Children.only(this.props.children);
const children = this.children;
if (React.Children.count(children) > 0) {
return React.Children.only(children);
}

return null;
Expand Down

0 comments on commit 8cfc4dc

Please sign in to comment.