Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/xxsnakerxx/react-native-ya-navigator

Yet another RN navigator component
https://github.com/xxsnakerxx/react-native-ya-navigator

Last synced: about 1 month ago
JSON representation

Yet another RN navigator component

Awesome Lists containing this project

README

        

# The project is not maintained any more!!!

# React Native Yet Another Navigator

![preview_ios](images/ya_navigator_ios.gif)
![preview_android](images/ya_navigator_android.gif)

## Table of contents
- [Main goals](#main-goals)
- [Dependencies](#dependencies)
- [Installation](#installation)
- [Usage](#usage)
- [Contributing](#contributing)
- [Copyright and license](#copyright-and-license)

## Main goals
- the scene can handle navigation bar items events
- the scene can change navigation bar items dynamically
- the scene can show/hide navigation bar dynamically
- the scene itself defines a configuration of the navigation bar

## Dependencies
- React Native >= `0.25.1`
- [react-native-vector-icons](https://github.com/oblador/react-native-vector-icons#readme)

## Installation

First of all, this component uses awesome [react-native-vector-icons](https://github.com/oblador/react-native-vector-icons#readme), so you need to install it (it's simple)...

then,

```javascript
npm install react-native-ya-navigator --save
```

## Usage

### YANavigator component

```javascript
import YANavigator from 'react-native-ya-navigator';

class App extends React.Component {
render() {
return (

)
}
}
```

#### YANavigator [propTypes](https://github.com/xxsnakerxx/react-native-ya-navigator/blob/master/Navigator.js#L436):
- `style`
- `navBarStyle`
- `sceneStyle`
- `initialRoute`
- `initialRouteStack`
- `defaultSceneConfig` (default value is __Navigator.SceneConfigs.PushFromRight__ for `iOS`
and __Navigator.SceneConfigs.FadeAndroid__ for `Android`).
- `useNavigationBar` (useful if you want to render your navBar component on each scene ([ToolbarAndroid](https://facebook.github.io/react-native/docs/toolbarandroid.html#content) for example) instead of the embedded navBar)
- `navBarUnderlay` (the view that will be rendered under all navBar items ([react-native-blur](https://github.com/react-native-fellowship/react-native-blur) for example))
- `navBarBackBtn`
- `icon`
- `iconWidth` (if you provide custom icon, set this for properly title animations on iOS)
- `textStyle`
- `eachSceneProps` (these props will be passed to each scene, for example, if you are using YANavigator inside tabs, you can to pass 'selected' prop to each scene, so each scene can decide should it updated via shouldComponentUpdate if it was hidden)
- `customEventedProps` (you can pass here array of prop names that you need for link your custom components rendered in navigation bar)
- `navBarFixedHeight` (use this to set custom fixed nav bar height)
- `navBarCrossPlatformUI` (this prop means that title on android will be in center)

Also `YANavigator` class has static property `navBarHeight` (you can use it in your styles)

### Navigation bar configuration in a scene

Your scene component should define `static` property `navigationDelegate`

```javascript
class MyScene extends React.Component {
render() {
return {this.props.children}
}

static navigationDelegate = {
/**
* if you want to listen nav bar items press events
* you must to provide id key
* @type {Something unique}
*/
id: 'myScene',
sceneConfig: myCustomSceneConfig,
/**
* false by default
* @type {bool}
*/
navBarIsHidden: true|false,
/**
* @type {String}
*/
navBarBackgroundColor: 'red',
/**
* @param {object} props [route props]
* @return {Class|JSX}
*/
renderTitle(props) {
return MyTitleComponent
// or
return
},
/**
* @param {object} props [route props]
* @return {Class|JSX}
*/
renderNavBarLeftPart(props) {
return MyButtonComponent
// or
return
},
/**
* @param {object} props [route props]
* @return {Class|JSX}
*/
renderNavBarRightPart(props) {
return MyButtonComponent
// or
return
},
/**
* will be called first on back android button press
* @param {object} navigator [navigator instance]
*/
onAndroidBackPress(navigator) {
navigator.popToPop();
}
/**
* If it's true, 'onNavBarBackBtnPress' method will be called on backBtnPress instead
* of navigator.pop()
* false by default
* @type {bool}
*/
overrideBackBtnPress: true|false,
/**
* Tint color of backBtn (applies to icon and text)
* @type {String}
*/
navBarBackBtnColor: 'white',
}
}
```

### Listening navigation bar items events

You should wrap your scene component with `YANavigator.Scene` component and set __this__ to `delegate` prop.
__Don't forget to define `id` in the `navigationDelegate`__

```javascript
class MyScene extends React.Component {
render() {
return (

{this.props.children}

)
}
```

Also `YANavigator.Scene` has `style` prop and `paddingTop` (if it's true(__default value__) then scene will have top padding equals height of the navigation bar, also you can use `YANavigator.navBarHeight` in your styles)

And one more thing... ;-)

You can listen when a scene will lose focus via route prop `onSceneWillBlur`

```javascript
...
onLinkPress = (link) => {
tabBar.hide(),

this.props.navigator.push({
component: Browser,
props: {
url: link,
/**
* @param {Boolean} true means the scene was popped, false means a new scene was pushed
*/
onSceneWillBlur: (isBack) => tabBar.show(),
},
})
}
...
```

### How to handle navigation bar items events?

There are a few simple rules

- if you pass as navBar item just a `class`, it should have `propTypes` with prop that you want to listen, then you should define method that will be called (`onNavBarTitlePress`, `onNavBarLeftPartPress`, `onNavBarRightPartPress`, `onNavBarTitleChange`, `onNavBarTitleValueChange`, etc...)
- if you pass as navBar item `JSX`, then props that you want to listen should return just a string - __name of the delegate method that will be called__
- currently supported props `onPress`, `onChange`, `onValueChange`, `onSelection`, `onBlur`, `onFocus`, `onSelectionChange`, `onSubmitEditing`

```javascript
class MyNavBarTitle extends React.Component {
render() {
return (

{'Default Text'}

)
}

static propTypes = {
onPress: React.PropTypes.func, // required
}
}

class MyScene extends React.Component {
onNavBarTitlePress(e) {
// press event
console.log(e)
}

onFirstBtnPress(e) {
alert('Right side - first btn press');
}

onSecondBtnPress(e) {
alert('Right side - second btn press');
}

onSceneWillFocus() {
console.log('Scene will focus');
}

onSceneDidFocus() {
console.log('Scene did focus');
}

render() {
return (

{this.props.children}

)
}

static navigationDelegate = {
id: 'myScene',
renderTitle() {
return MyNavBarTitle;
},
renderNavBarLeftPart() {
return (

'onFirstBtnPress'}>
{'1'}

'onSecondBtnPress'}>
{'2'}


)
}
}
```

### How to change navigation bar items dynamically?

There are two options:

1. Each scene can access to navBar items via `ref` and modify its state using standard `setState` method, or can call other methods provided by your component.

__ref__ generated from the template

```javascript
const ref = `${navigationDelegate.id || `${navigator.state.presentedIndex + 1}_scene`}_leftPart|rightPart|title`;

// usage
this.props.navigator.navBarParts['navDelegateId_rightPart'].doSmth()
// or if navigationDelegate id is not defined
this.props.navigator.navBarParts['1_scene_rightPart'].doSmth()
```

2. If you want re-render your navBar component with new props or just re-render use the template

```javascript
YourComponentClass.navigationDelegate.renderTitle = () => // return component here with the new props

this.props.navigator.forceUpdateNavBar();
```

Also NavBar component has some helpful methods
- `show`('fade'|'slide') __default behavior is `fade`__
- `hide`('fade'|'slide') __default behavior is `fade`__

```javascript
this.props.navigator.showNavBar('slide');
this.props.navigator.hideNavBar('fade');
```

```javascript
class MyNavBarTitle extends React.Component {
constructor(props) {
super(props)

this.state = {
text: props.text
}
}
render() {
return (

{this.state.text}

)
}

static propTypes = {
onPress: React.PropTypes.func,
text: React.PropTypes.string,
}

static defualtProps = {
text: 'Default Text',
}
}

class MyScene extends React.Component {
onBtnPress() {
MyScene.navigationDelegate.renderTitle = () =>
MyScene.navigationDelegate.renderNavBarRightPart = () => (
'onBtnPress'}>
{'Updated btn'}

);

this.props.navigator.forceUpdateNavBar();
}

onNavBarTitlePress() {
this.props.navigator.navBarParts.myScene_title.setState({
text: 'Other title',
})
}

render() {
return (

{this.props.children}

)
}

static navigationDelegate = {
id: 'myScene',
renderTitle() {
return MyNavBarTitle;
},
renderNavBarRightPart() {
return (
'onBtnPress'}>
{'Btn'}

)
}
}
```

#### Feel free to go to [example](example) and explore it for more details

## Contributing

Just submit a pull request!

## Copyright and license

Code and documentation copyright 2015 Dmitriy Kolesnikov. Code released under the [MIT license](LICENSE).