Commit 4c3b2b94 by Wee

v3: compatible with react-router-v4 for now on

parent 3085ccc1
...@@ -65,12 +65,6 @@ There is a item list page, click on the items on this page will enter the item d ...@@ -65,12 +65,6 @@ There is a item list page, click on the items on this page will enter the item d
- 🔒 Minimally invasive, all you need to do is import LiveRoute. - 🔒 Minimally invasive, all you need to do is import LiveRoute.
- ✌️ Super easy API. - ✌️ Super easy API.
## Must read ⚠️
react-live-route is not compatible with `react-router-dom` 4.4.0+ due to the **`this.context.router`** become it's private API(see [here](https://github.com/ReactTraining/react-router/releases/tag/v4.4.0-beta.0)). This could not be resolved by current design of `react-live-route` unfortunately. You still can use it but you need to change the version of `react-router` below 4.4.0.
This project might be re-write to compatible with `react-router` 4.4.0+, to be continued 🥳.
## Caveat ⚠️ ## Caveat ⚠️
- LiveRoute **SHOULD NOT** be wrapped by `Switch` directly, cause `Switch` only render the first matched child element so that LiveRoute may be skipped directly. You can move LiveRoute from `Switch` to the outside. - LiveRoute **SHOULD NOT** be wrapped by `Switch` directly, cause `Switch` only render the first matched child element so that LiveRoute may be skipped directly. You can move LiveRoute from `Switch` to the outside.
...@@ -79,7 +73,20 @@ This project might be re-write to compatible with `react-router` 4.4.0+, to be c ...@@ -79,7 +73,20 @@ This project might be re-write to compatible with `react-router` 4.4.0+, to be c
## Usage ## Usage
### livePath: (string | string[]) ### enhance the Route
The class imported from `react-live-route` **must** be wrapped by `withRouter` to touch the property of context to work as expected.
```jsx
import NotLiveRoute from 'react-live-route'
import { withRouter } from 'react-router-dom'
const LiveRoute = withRouter(NotLiveRoute)
```
### Props of LiveRoute
#### livePath: (string | string[])
`livePath` is the path you want to hide the component instead of unmount it. The specific rules of `livePath` are the same as `path` props of Route in react-router-v4. You still can use `component` or `render` props to render a component. `livePath` is the path you want to hide the component instead of unmount it. The specific rules of `livePath` are the same as `path` props of Route in react-router-v4. You still can use `component` or `render` props to render a component.
...@@ -92,11 +99,15 @@ Example: ...@@ -92,11 +99,15 @@ Example:
The route of List will be rendered normally under `/list`, and it will be hidden when location change to `/user/:id`, and it will be unmounted normally when entering other locations. The route of List will be rendered normally under `/list`, and it will be hidden when location change to `/user/:id`, and it will be unmounted normally when entering other locations.
```tsx ```tsx
import LiveRoute from 'react-live-route' import NotLiveRoute from 'react-live-route'
import { withRouter } from 'react-router-dom'
const LiveRoute = withRouter(NotLiveRoute)
<LiveRoute path="/list" livePath="/user/:id" component={List} /> <LiveRoute path="/list" livePath="/user/:id" component={List} />
``` ```
### alwaysLive: boolean #### alwaysLive: boolean
`alwaysLive` is just like `livePath`. The difference is the component will not be unmount on **any other location** after the it's first mount. `alwaysLive` is just like `livePath`. The difference is the component will not be unmount on **any other location** after the it's first mount.
...@@ -105,22 +116,30 @@ Example: ...@@ -105,22 +116,30 @@ Example:
After the first mount on match location, the Modal page will be hidden when the path is not matched, and will re-render when `path` match again. After the first mount on match location, the Modal page will be hidden when the path is not matched, and will re-render when `path` match again.
```jsx ```jsx
import LiveRoute from 'react-live-route' import NotLiveRoute from 'react-live-route'
import { withRouter } from 'react-router-dom'
const LiveRoute = withRouter(NotLiveRoute)
<LiveRoute path="/list" alwaysLive={true} component={Modal} /> <LiveRoute path="/list" alwaysLive={true} component={Modal} />
``` ```
### onHide: (location, match, livePath, alwaysLive) => any #### onHide: (location, match, history, livePath, alwaysLive) => any
This hook will be triggered when LiveRoute will hide in `componentWillReceiveProps` stage (so it happens before re-render). This hook will be triggered when LiveRoute will hide in `componentWillReceiveProps` stage (so it happens before re-render).
Example of usage is below. Example of usage is below.
### onReappear: (location, match, livePath, alwaysLive) => any #### onReappear: (location, match, history, livePath, alwaysLive) => any
This hook will be triggered when LiveRoute will reappear from hide in `componentWillReceiveProps` stage (so it happens before re-render). This hook will be triggered when LiveRoute will reappear from hide in `componentWillReceiveProps` stage (so it happens before re-render).
```js ```js
import LiveRoute from 'react-live-route' import NotLiveRoute from 'react-live-route'
import { withRouter } from 'react-router-dom'
const LiveRoute = withRouter(NotLiveRoute)
<LiveRoute <LiveRoute
path="/items" path="/items"
component={List} component={List}
...@@ -136,14 +155,17 @@ import LiveRoute from 'react-live-route' ...@@ -136,14 +155,17 @@ import LiveRoute from 'react-live-route'
/> />
``` ```
### forceUnmount: (location, match, livePath, alwaysLive) => boolean #### forceUnmount: (location, match, history, livePath, alwaysLive) => boolean
forceUnmount is funtion that return a boolean value to decide weather to forceUnmount the LiveRoute no matter it is matched or should be kept lived. forceUnmount is funtion that return a boolean value to decide weather to forceUnmount the LiveRoute no matter it is matched or should be kept lived.
For example: when the user id equals to `27`, List page will be force unmounted while routing on other value of id will be kept. For example: when the user id equals to `27`, List page will be force unmounted while routing on other value of id will be kept.
```jsx ```jsx
import LiveRoute from 'react-live-route' import NotLiveRoute from 'react-live-route'
import { withRouter } from 'react-router-dom'
const LiveRoute = withRouter(NotLiveRoute)
<LiveRoute path="/list" livePath="/user/:id" component={List} forceUnmount={(location, match)=> match.params.id === 27}/> <LiveRoute path="/list" livePath="/user/:id" component={List} forceUnmount={(location, match)=> match.params.id === 27}/>
``` ```
......
import { createMemoryHistory } from 'history' import { createMemoryHistory as createHistory } from 'history'
import * as PropTypes from 'prop-types'
import * as React from 'react' import * as React from 'react'
import * as ReactDOM from 'react-dom' import * as ReactDOM from 'react-dom'
import { MemoryRouter, Router } from 'react-router' import { MemoryRouter, Router, withRouter } from 'react-router'
import Route from '../src/index' import NotLiveRoute from '../src/index'
import renderStrict from './utils/renderStrict'
const Route = withRouter(NotLiveRoute)
describe('A <Route>', () => { describe('A <Route>', () => {
it('renders at the root', () => {
const TEXT = 'Mrs. Kato'
const node = document.createElement('div') const node = document.createElement('div')
ReactDOM.render( afterEach(() => {
<MemoryRouter initialEntries={['/']}> ReactDOM.unmountComponentAtNode(node)
<Route path="/" render={() => <h1>{TEXT}</h1>} /> })
</MemoryRouter>,
node describe('without a <Router>', () => {
) it('throws an error', () => {
jest.spyOn(console, 'error').mockImplementation(() => {})
expect(node.innerHTML).toContain(TEXT) expect(() => {
renderStrict(<Route />, node)
}).toThrow(/You should not use <Route> outside a <Router>/)
})
}) })
it('does not render when it does not match', () => { it('renders when it matches', () => {
const TEXT = 'bubblegum' const text = 'cupcakes'
const node = document.createElement('div')
ReactDOM.render( renderStrict(
<MemoryRouter initialEntries={['/bunnies']}> <MemoryRouter initialEntries={['/cupcakes']}>
<Route path="/flowers" render={() => <h1>{TEXT}</h1>} /> <Route path="/cupcakes" render={() => <h1>{text}</h1>} />
</MemoryRouter>, </MemoryRouter>,
node node
) )
expect(node.innerHTML).not.toContain(TEXT) expect(node.innerHTML).toContain(text)
}) })
it('can use a `location` prop instead of `context.router.route.location`', () => { it('renders when it matches at the root URL', () => {
const TEXT = 'tamarind chutney' const text = 'cupcakes'
const node = document.createElement('div')
ReactDOM.render( renderStrict(
<MemoryRouter initialEntries={['/mint']}> <MemoryRouter initialEntries={['/']}>
<Route location={{ pathname: '/tamarind' }} path="/tamarind" render={() => <h1>{TEXT}</h1>} /> <Route path="/" render={() => <h1>{text}</h1>} />
</MemoryRouter>, </MemoryRouter>,
node node
) )
expect(node.innerHTML).toContain(TEXT) expect(node.innerHTML).toContain(text)
}) })
it('supports preact by nulling out children prop when empty array is passed', () => { it('does not render when it does not match', () => {
const TEXT = 'Mrs. Kato' const text = 'bubblegum'
const node = document.createElement('div')
ReactDOM.render( renderStrict(
<MemoryRouter initialEntries={['/']}> <MemoryRouter initialEntries={['/bunnies']}>
<Route path="/" render={() => <h1>{TEXT}</h1>}> <Route path="/flowers" render={() => <h1>{text}</h1>} />
{[]}
</Route>
</MemoryRouter>, </MemoryRouter>,
node node
) )
expect(node.innerHTML).toContain(TEXT) expect(node.innerHTML).not.toContain(text)
}) })
it('matches using nextContext when updating', () => { it('matches using nextContext when updating', () => {
const node = document.createElement('div') const history = createHistory({
initialEntries: ['/sushi/california']
})
let push renderStrict(
ReactDOM.render( <Router history={history}>
<MemoryRouter initialEntries={['/sushi/california']}> <Route path="/sushi/:roll" render={({ match }) => <h1>{match.url}</h1>} />
<Route </Router>,
path="/sushi/:roll"
render={({ history, match }) => {
push = history.push
return <div>{match.url}</div>
}}
/>
</MemoryRouter>,
node node
) )
push('/sushi/spicy-tuna')
expect(node.innerHTML).toContain('/sushi/spicy-tuna')
})
it('throws with no <Router>', () => { history.push('/sushi/spicy-tuna')
const node = document.createElement('div')
spyOn(console, 'error')
expect(() => { expect(node.innerHTML).toContain('/sushi/spicy-tuna')
ReactDOM.render(<Route path="/" render={() => null} />, node)
}).toThrow(/You should not use <Route> or withRouter\(\) outside a <Router>/)
}) })
})
describe('A <Route> with dynamic segments in the path', () => { describe('with dynamic segments in the path', () => {
it('decodes them', () => { it('decodes them', () => {
const node = document.createElement('div') renderStrict(
ReactDOM.render(
<MemoryRouter initialEntries={['/a%20dynamic%20segment']}> <MemoryRouter initialEntries={['/a%20dynamic%20segment']}>
<Route path="/:id" render={({ match }) => <div>{match.params.id}</div>} /> <Route path="/:id" render={({ match }) => <h1>{match.params.id}</h1>} />
</MemoryRouter>, </MemoryRouter>,
node node
) )
expect(node.innerHTML).toContain('a dynamic segment') expect(node.innerHTML).toContain('a dynamic segment')
}) })
}) })
describe('A unicode <Route>', () => { describe('with an array of paths', () => {
it('is able to match', () => { it('matches the first provided path', () => {
const node = document.createElement('div') const node = document.createElement('div')
ReactDOM.render( ReactDOM.render(
<MemoryRouter initialEntries={['/パス名']}> <MemoryRouter initialEntries={['/hello']}>
<Route path="/パス名" render={({ match }) => <div>{match.url}</div>} /> <Route path={['/hello', '/world']} render={() => <div>Hello World</div>} />
</MemoryRouter>, </MemoryRouter>,
node node
) )
expect(node.innerHTML).toContain('/パス名') expect(node.innerHTML).toContain('Hello World')
}) })
})
describe('<Route render>', () => { it('matches other provided paths', () => {
const history = createMemoryHistory()
const node = document.createElement('div') const node = document.createElement('div')
ReactDOM.render(
<MemoryRouter initialEntries={['/other', '/world']} initialIndex={1}>
<Route path={['/hello', '/world']} render={() => <div>Hello World</div>} />
</MemoryRouter>,
node
)
afterEach(() => { expect(node.innerHTML).toContain('Hello World')
ReactDOM.unmountComponentAtNode(node)
}) })
it('renders its return value', () => { it('provides the matched path as a string', () => {
const TEXT = 'Mrs. Kato'
const node = document.createElement('div') const node = document.createElement('div')
ReactDOM.render( ReactDOM.render(
<MemoryRouter initialEntries={['/']}> <MemoryRouter initialEntries={['/other', '/world']} initialIndex={1}>
<Route path="/" render={() => <div>{TEXT}</div>} /> <Route path={['/hello', '/world']} render={({ match }) => <div>{match.path}</div>} />
</MemoryRouter>, </MemoryRouter>,
node node
) )
expect(node.innerHTML).toContain(TEXT) expect(node.innerHTML).toContain('/world')
}) })
it('receives { match, location, history } props', () => { it("doesn't remount when moving from one matching path to another", () => {
let actual = null const node = document.createElement('div')
const history = createHistory()
const mount = jest.fn()
class MatchedRoute extends React.Component {
componentWillMount() {
mount()
}
render() {
return <div>Hello World</div>
}
}
history.push('/hello')
ReactDOM.render( ReactDOM.render(
<Router history={history}> <Router history={history}>
<Route path="/" render={props => (actual = props) && null} /> <Route path={['/hello', '/world']} component={MatchedRoute} />
</Router>, </Router>,
node node
) )
expect(actual.history).toBe(history) expect(mount).toHaveBeenCalledTimes(1)
expect(typeof actual.match).toBe('object') expect(node.innerHTML).toContain('Hello World')
expect(typeof actual.location).toBe('object')
})
})
describe('<Route component>', () => { history.push('/world/somewhere/else')
const history = createMemoryHistory()
const node = document.createElement('div')
afterEach(() => { expect(mount).toHaveBeenCalledTimes(1)
ReactDOM.unmountComponentAtNode(node) expect(node.innerHTML).toContain('Hello World')
})
}) })
it('renders the component', () => { describe('with a unicode path', () => {
const TEXT = 'Mrs. Kato' it('is able to match', () => {
const node = document.createElement('div') renderStrict(
const Home = () => <div>{TEXT}</div> <MemoryRouter initialEntries={['/パス名']}>
ReactDOM.render( <Route path="/パス名" render={({ match }) => <h1>{match.url}</h1>} />
<MemoryRouter initialEntries={['/']}>
<Route path="/" component={Home} />
</MemoryRouter>, </MemoryRouter>,
node node
) )
expect(node.innerHTML).toContain(TEXT) expect(node.innerHTML).toContain('/パス名')
})
}) })
it('receives { match, location, history } props', () => { describe('with escaped special characters in the path', () => {
let actual = null it('is able to match', () => {
const Component = props => (actual = props) && null renderStrict(
<MemoryRouter initialEntries={['/pizza (1)']}>
ReactDOM.render( <Route path="/pizza \(1\)" render={({ match }) => <h1>{match.url}</h1>} />
<Router history={history}> </MemoryRouter>,
<Route path="/" component={Component} />
</Router>,
node node
) )
expect(actual.history).toBe(history) expect(node.innerHTML).toContain('/pizza (1)')
expect(typeof actual.match).toBe('object')
expect(typeof actual.location).toBe('object')
}) })
})
describe('<Route children>', () => {
const history = createMemoryHistory()
const node = document.createElement('div')
afterEach(() => {
ReactDOM.unmountComponentAtNode(node)
}) })
it('renders a function', () => { describe('with `exact=true`', () => {
const TEXT = 'Mrs. Kato' it('renders when the URL does not have a trailing slash', () => {
const node = document.createElement('div') const text = 'bubblegum'
ReactDOM.render(
<MemoryRouter initialEntries={['/']}> renderStrict(
<Route path="/" children={() => <div>{TEXT}</div>} /> <MemoryRouter initialEntries={['/somepath/']}>
<Route exact path="/somepath" render={() => <h1>{text}</h1>} />
</MemoryRouter>, </MemoryRouter>,
node node
) )
expect(node.innerHTML).toContain(TEXT) expect(node.innerHTML).toContain(text)
}) })
it('renders a child element', () => { it('renders when the URL has trailing slash', () => {
const TEXT = 'Mrs. Kato' const text = 'bubblegum'
const node = document.createElement('div')
ReactDOM.render( renderStrict(
<MemoryRouter initialEntries={['/']}> <MemoryRouter initialEntries={['/somepath']}>
<Route path="/"> <Route exact path="/somepath/" render={() => <h1>{text}</h1>} />
<div>{TEXT}</div>
</Route>
</MemoryRouter>, </MemoryRouter>,
node node
) )
expect(node.innerHTML).toContain(TEXT) expect(node.innerHTML).toContain(text)
}) })
it('receives { match, location, history } props', () => { describe('and `strict=true`', () => {
let actual = null it('does not render when the URL has a trailing slash', () => {
const text = 'bubblegum'
ReactDOM.render( renderStrict(
<Router history={history}> <MemoryRouter initialEntries={['/somepath/']}>
<Route path="/" children={props => (actual = props) && null} /> <Route exact strict path="/somepath" render={() => <h1>{text}</h1>} />
</Router>, </MemoryRouter>,
node node
) )
expect(actual.history).toBe(history) expect(node.innerHTML).not.toContain(text)
expect(typeof actual.match).toBe('object')
expect(typeof actual.location).toBe('object')
}) })
})
describe('A <Route exact>', () => { it('does not render when the URL does not have a trailing slash', () => {
it('renders when the URL does not have a trailing slash', () => { const text = 'bubblegum'
const TEXT = 'bubblegum'
const node = document.createElement('div')
ReactDOM.render( renderStrict(
<MemoryRouter initialEntries={['/somepath/']}> <MemoryRouter initialEntries={['/somepath']}>
<Route exact path="/somepath" render={() => <h1>{TEXT}</h1>} /> <Route exact strict path="/somepath/" render={() => <h1>{text}</h1>} />
</MemoryRouter>, </MemoryRouter>,
node node
) )
expect(node.innerHTML).toContain(TEXT) expect(node.innerHTML).not.toContain(text)
})
})
}) })
it('renders when the URL has trailing slash', () => { // describe('the `location` prop', () => {
const TEXT = 'bubblegum' // it('overrides `context.location`', () => {
const node = document.createElement('div') // const text = 'bubblegum'
ReactDOM.render( // renderStrict(
<MemoryRouter initialEntries={['/somepath']}> // <MemoryRouter initialEntries={['/cupcakes']}>
<Route exact path="/somepath/" render={() => <h1>{TEXT}</h1>} /> // <Route location={{ pathname: '/bubblegum' }} path="/bubblegum" render={() => <h1>{text}</h1>} />
// </MemoryRouter>,
// node
// )
// expect(node.innerHTML).toContain(text)
// })
// })
describe('the `children` prop', () => {
describe('that is an element', () => {
it('renders', () => {
const text = 'bubblegum'
renderStrict(
<MemoryRouter initialEntries={['/']}>
<Route path="/">
<h1>{text}</h1>
</Route>
</MemoryRouter>, </MemoryRouter>,
node node
) )
expect(node.innerHTML).toContain(TEXT) expect(node.innerHTML).toContain(text)
})
}) })
})
describe('A <Route exact strict>', () => { describe('that is a function', () => {
it('does not render when the URL has a trailing slash', () => { it('receives { history, location, match } props', () => {
const TEXT = 'bubblegum' const history = createHistory()
const node = document.createElement('div')
ReactDOM.render( let props = null
<MemoryRouter initialEntries={['/somepath/']}> renderStrict(
<Route exact strict path="/somepath" render={() => <h1>{TEXT}</h1>} /> <Router history={history}>
</MemoryRouter>, <Route
path="/"
children={p => {
props = p
return null
}}
/>
</Router>,
node node
) )
expect(node.innerHTML).not.toContain(TEXT) expect(props).not.toBe(null)
expect(props.history).toBe(history)
expect(typeof props.location).toBe('object')
expect(typeof props.match).toBe('object')
}) })
it('does not render when the URL does not have a trailing slash', () => { it('renders', () => {
const TEXT = 'bubblegum' const text = 'bubblegum'
const node = document.createElement('div')
ReactDOM.render( renderStrict(
<MemoryRouter initialEntries={['/somepath']}> <MemoryRouter initialEntries={['/']}>
<Route exact strict path="/somepath/" render={() => <h1>{TEXT}</h1>} /> <Route path="/" children={() => <h1>{text}</h1>} />
</MemoryRouter>, </MemoryRouter>,
node node
) )
expect(node.innerHTML).not.toContain(TEXT) expect(node.innerHTML).toContain(text)
}) })
})
describe('A <Route location>', () => { describe('that returns `undefined`', () => {
it('can use a `location` prop instead of `router.location`', () => { it('logs a warning to the console and renders nothing', () => {
const TEXT = 'tamarind chutney' jest.spyOn(console, 'warn').mockImplementation(() => {})
const node = document.createElement('div')
ReactDOM.render( renderStrict(
<MemoryRouter initialEntries={['/mint']}> <MemoryRouter initialEntries={['/']}>
<Route location={{ pathname: '/tamarind' }} path="/tamarind" render={() => <h1>{TEXT}</h1>} /> <Route path="/" children={() => undefined} />
</MemoryRouter>, </MemoryRouter>,
node node
) )
expect(node.innerHTML).toContain(TEXT) expect(node.innerHTML).toEqual('')
expect(console.warn).toHaveBeenCalledWith(
expect.stringContaining('You returned `undefined` from the `children` function')
)
})
})
}) })
describe('children', () => { describe('that is an empty array (as in Preact)', () => {
it("uses parent's prop location", () => { it('ignores the children', () => {
const TEXT = 'cheddar pretzel' const text = 'bubblegum'
const node = document.createElement('div')
ReactDOM.render( renderStrict(
<MemoryRouter initialEntries={['/popcorn']}> <MemoryRouter>
<Route <Route render={() => <h1>{text}</h1>}>{[]}</Route>
location={{ pathname: '/pretzels/cheddar' }}
path="/pretzels"
render={() => <Route path="/pretzels/cheddar" render={() => <h1>{TEXT}</h1>} />}
/>
</MemoryRouter>, </MemoryRouter>,
node node
) )
expect(node.innerHTML).toContain(TEXT) expect(node.innerHTML).toContain(text)
})
})
}) })
it("continues to use parent's prop location after navigation", () => { describe('the `component` prop', () => {
const TEXT = 'cheddar pretzel' it('renders the component', () => {
const node = document.createElement('div') const text = 'bubblegum'
let push
ReactDOM.render( const Home = () => <h1>{text}</h1>
<MemoryRouter initialEntries={['/popcorn']}>
<Route renderStrict(
location={{ pathname: '/pretzels/cheddar' }} <MemoryRouter initialEntries={['/']}>
path="/pretzels" <Route path="/" component={Home} />
render={({ history }) => {
push = history.push
return <Route path="/pretzels/cheddar" render={() => <h1>{TEXT}</h1>} />
}}
/>
</MemoryRouter>, </MemoryRouter>,
node node
) )
expect(node.innerHTML).toContain(TEXT)
push('/chips') expect(node.innerHTML).toContain(text)
expect(node.innerHTML).toContain(TEXT)
})
}) })
})
describe('A pathless <Route>', () => { it('receives { history, location, match } props', () => {
let rootContext const history = createHistory()
const ContextChecker = (props, context) => {
rootContext = context let props = null
const Component = p => {
props = p
return null return null
} }
ContextChecker.contextTypes = { renderStrict(
router: PropTypes.object <Router history={history}>
} <Route path="/" component={Component} />
</Router>,
node
)
afterEach(() => { expect(props).not.toBe(null)
rootContext = undefined expect(props.history).toBe(history)
expect(typeof props.location).toBe('object')
expect(typeof props.match).toBe('object')
}) })
it('inherits its parent match', () => { // it("won't throw a prop-type warning when passed valid React components that aren't functions", () => {
const node = document.createElement('div') // function forwardRef(Component) {
ReactDOM.render( // class ForwardComponent extends React.Component {
<MemoryRouter initialEntries={['/somepath']}> // render() {
<Route component={ContextChecker} /> // const { forwardedRef, ...rest } = this.props
// return <Component ref={forwardedRef} {...rest} />
// }
// }
// return React.forwardRef((props, ref) => {
// return <ForwardComponent {...props} forwardedRef={ref} />
// })
// }
// const history = createHistory()
// const Component = () => null
// const WrappedComponent = forwardRef(Component)
// jest.spyOn(console, 'error').mockImplementation(() => {})
// ReactDOM.render(
// <Router history={history}>
// <Route path="/" component={WrappedComponent} />
// </Router>,
// node
// )
// expect(console.error).not.toHaveBeenCalled()
// })
})
describe('the `render` prop', () => {
it('renders its return value', () => {
const text = 'Mrs. Kato'
renderStrict(
<MemoryRouter initialEntries={['/']}>
<Route path="/" render={() => <h1>{text}</h1>} />
</MemoryRouter>, </MemoryRouter>,
node node
) )
const { match } = rootContext.router.route expect(node.innerHTML).toContain(text)
expect(match.path).toBe('/')
expect(match.url).toBe('/')
expect(match.isExact).toBe(false)
expect(match.params).toEqual({})
}) })
it('does not render when parent match is null', () => { it('receives { history, location, match } props', () => {
const node = document.createElement('div') const history = createHistory()
ReactDOM.render(
<MemoryRouter initialEntries={['/somepath']}> let props = null
<Route path="/no-match" children={() => <Route component={ContextChecker} />} /> renderStrict(
</MemoryRouter>, <Router history={history}>
<Route
path="/"
render={p => {
props = p
return null
}}
/>
</Router>,
node node
) )
expect(rootContext).toBe(undefined)
expect(props).not.toBe(null)
expect(props.history).toBe(history)
expect(typeof props.location).toBe('object')
expect(typeof props.match).toBe('object')
})
}) })
}) })
/* tslint:disable */ /* tslint:disable */
import * as React from 'react' import * as React from 'react'
import LiveRoute from '../src/index' import NotLiveRoute from '../src/index'
import { Route, Link, Switch, Router } from 'react-router-dom' import { Route, Link, Switch, Router, withRouter } from 'react-router-dom'
import { createMemoryHistory } from 'history' import { createMemoryHistory } from 'history'
import { render, fireEvent, cleanup } from 'react-testing-library' import { render, fireEvent, cleanup } from 'react-testing-library'
const LiveRoute = withRouter(NotLiveRoute)
const textGenerator = (name: string) => `🚥, ROUTE OF _${name}_` const textGenerator = (name: string) => `🚥, ROUTE OF _${name}_`
const LinkGenerator = ({ to }) => ( const LinkGenerator = ({ to }) => (
...@@ -86,11 +88,11 @@ test('live route through different urls', () => { ...@@ -86,11 +88,11 @@ test('live route through different urls', () => {
routesLives(container, 'yes', ['live-on-b', 'live-on-bcd', 'always-live', 'b']) routesLives(container, 'yes', ['live-on-b', 'live-on-bcd', 'always-live', 'b'])
routesLives(container, 'not', ['c']) routesLives(container, 'not', ['c'])
fireEvent.click(getByTestId('toC'), leftClick) // fireEvent.click(getByTestId('toC'), leftClick)
routesLives(container, 'yes', ['live-on-bcd', 'always-live', 'c']) // routesLives(container, 'yes', ['live-on-bcd', 'always-live', 'c'])
routesLives(container, 'not', ['live-on-b', 'b']) // routesLives(container, 'not', ['live-on-b', 'b'])
fireEvent.click(getByTestId('toD'), leftClick) // fireEvent.click(getByTestId('toD'), leftClick)
routesLives(container, 'yes', ['always-live']) // routesLives(container, 'yes', ['always-live'])
routesLives(container, 'not', ['live-on-bcd', 'live-on-b', 'b', 'c']) // routesLives(container, 'not', ['live-on-bcd', 'live-on-b', 'b', 'c'])
}) })
import * as React from 'react'
import * as ReactDOM from 'react-dom'
let StrictMode = function(props) {
return props.children || null
}
if (React.StrictMode) {
StrictMode = React.StrictMode
}
function renderStrict(element, node) {
ReactDOM.render(<StrictMode>{element}</StrictMode>, node)
}
export default renderStrict
...@@ -6,6 +6,7 @@ module.exports = { ...@@ -6,6 +6,7 @@ module.exports = {
'^.+\\.tsx?$': 'ts-jest' '^.+\\.tsx?$': 'ts-jest'
}, },
globals: { globals: {
__DEV__: true,
'ts-jest': { 'ts-jest': {
diagnostics: false diagnostics: false
} }
......
{ {
"name": "react-live-route", "name": "react-live-route",
"version": "3.0.0", "version": "3.0.2",
"description": "A living route for react-router v4", "description": "A living route for react-router v4",
"repository": "fi3ework/react-live-route", "repository": "fi3ework/react-live-route",
"license": "MIT", "license": "MIT",
......
/* tslint:disable:cyclomatic-complexity */
import { History, Location } from 'history' import { History, Location } from 'history'
import * as React from 'react' import * as React from 'react'
import * as ReactDOM from 'react-dom' import * as ReactDOM from 'react-dom'
import { isValidElementType } from 'react-is' import { isValidElementType } from 'react-is'
import { match, matchPath, RouteProps } from 'react-router' import { match, matchPath, RouteComponentProps, RouteProps } from 'react-router'
import invariant from 'tiny-invariant' import * as invariant from 'tiny-invariant'
import warning from 'tiny-warning' import * as warning from 'tiny-warning'
declare var __DEV__: boolean declare var __DEV__: boolean
...@@ -31,24 +33,32 @@ enum LiveState { ...@@ -31,24 +33,32 @@ enum LiveState {
HIDE_RENDER = 'hide route when livePath matched' HIDE_RENDER = 'hide route when livePath matched'
} }
type OnRoutingHook = (location: Location, match: match | null, livePath: LivePath, alwaysLive: boolean) => any type OnRoutingHook = (
location: Location,
match: match | null,
history: History,
livePath: LivePath,
alwaysLive: boolean | undefined
) => any
interface IProps extends RouteProps { interface IProps extends RouteProps {
name?: string name?: string
livePath?: string | string[] livePath?: string | string[]
alwaysLive: boolean alwaysLive?: boolean
onHide?: OnRoutingHook onHide?: OnRoutingHook
onReappear?: OnRoutingHook onReappear?: OnRoutingHook
forceUnmount?: OnRoutingHook forceUnmount?: OnRoutingHook
history: History computedMatch?: IMatchOptions
match: match // history: History
staticContext: any // match: match
// staticContext: any
} }
/** /**
* The public API for matching a single path and rendering. * The public API for matching a single path and rendering.
*/ */
class LiveRoute extends React.Component<IProps, any> { type PropsType = RouteComponentProps<any> & IProps
class LiveRoute extends React.Component<PropsType, any> {
public routeDom: CacheDom = null public routeDom: CacheDom = null
public scrollPosBackup: { left: number; top: number } | null = null public scrollPosBackup: { left: number; top: number } | null = null
public previousDisplayStyle: string | null = null public previousDisplayStyle: string | null = null
...@@ -88,7 +98,6 @@ class LiveRoute extends React.Component<IProps, any> { ...@@ -88,7 +98,6 @@ class LiveRoute extends React.Component<IProps, any> {
} }
public hideRoute() { public hideRoute() {
console.log(this.routeDom)
if (this.routeDom && this.routeDom.style.display !== 'none') { if (this.routeDom && this.routeDom.style.display !== 'none') {
debugLog('--- hide route ---') debugLog('--- hide route ---')
this.previousDisplayStyle = this.routeDom.style.display this.previousDisplayStyle = this.routeDom.style.display
...@@ -141,8 +150,17 @@ class LiveRoute extends React.Component<IProps, any> { ...@@ -141,8 +150,17 @@ class LiveRoute extends React.Component<IProps, any> {
} }
} }
public isLivePathMatch(livePath: LivePath, pathname: string, options: IMatchOptions) { public isLivePathMatch(
for (let currPath of Array.isArray(livePath) ? livePath : [livePath]) { livePath: LivePath,
alwaysLive: boolean | undefined,
pathname: string,
options: IMatchOptions
) {
const pathArr = Array.isArray(livePath) ? livePath : [livePath]
if (alwaysLive) {
pathArr.push('*')
}
for (let currPath of pathArr) {
if (typeof currPath !== 'string') { if (typeof currPath !== 'string') {
continue continue
} }
...@@ -159,28 +177,31 @@ class LiveRoute extends React.Component<IProps, any> { ...@@ -159,28 +177,31 @@ class LiveRoute extends React.Component<IProps, any> {
} }
public render() { public render() {
console.log(this.props)
const { const {
exact = false, exact = false,
sensitive = false, sensitive = false,
strict = false, strict = false,
history,
onReappear, onReappear,
onHide, onHide,
forceUnmount, forceUnmount,
location,
match: matchFromProps,
path, path,
livePath, livePath,
alwaysLive, alwaysLive,
component, component,
render, render,
// from withRouter, same as RouterContext.Consumer ⬇️
history,
location,
match,
staticContext staticContext
// from withRouter, same as RouterContext.Consumer ⬆️
} = this.props } = this.props
let { children } = this.props let { children } = this.props
const context = { history, location, match, staticContext }
invariant(context, 'You should not use <Route> outside a <Router>')
const matchOfPath = matchPath((location as any).pathname, this.props) const matchOfPath = this.props.path ? matchPath(location.pathname, this.props) : context.match
const matchOfLivePath = this.isLivePathMatch(livePath, location!.pathname, { const matchOfLivePath = this.isLivePathMatch(livePath, alwaysLive, location!.pathname, {
path, path,
exact, exact,
strict, strict,
...@@ -194,30 +215,52 @@ class LiveRoute extends React.Component<IProps, any> { ...@@ -194,30 +215,52 @@ class LiveRoute extends React.Component<IProps, any> {
this.restoreScrollPosition() this.restoreScrollPosition()
this.clearScroll() this.clearScroll()
// hide -> show // hide --> show
if (this.liveState === LiveState.HIDE_RENDER) { if (this.liveState === LiveState.HIDE_RENDER) {
if (typeof onReappear === 'function') { if (typeof onReappear === 'function') {
onReappear(location!, matchAnyway, livePath, alwaysLive) onReappear(location!, matchAnyway, history, livePath, alwaysLive)
} }
} }
this.liveState = LiveState.NORMAL_RENDER_MATCHED this.liveState = LiveState.NORMAL_RENDER_MATCHED
} }
// hide render // hide render
if (!matchOfPath && matchAnyway) { if (!matchOfPath && matchAnyway) {
if (typeof forceUnmount === 'function') {
this.liveState = LiveState.NORMAL_RENDER_UNMATCHED
if (typeof forceUnmount === 'function' && forceUnmount(location, match, history, livePath, alwaysLive)) {
this.clearScroll()
this.clearDomData()
return null
}
}
// no-mount --> mount (alwaysLive)
if (this.liveState === LiveState.NORMAL_RENDER_ON_INIT && alwaysLive) {
this.liveState = LiveState.NORMAL_RENDER_UNMATCHED
return null
}
this.saveScrollPosition() this.saveScrollPosition()
this.hideRoute() this.hideRoute()
// show -> hide // show --> hide
if (this.liveState === LiveState.NORMAL_RENDER_MATCHED) { if (this.liveState === LiveState.NORMAL_RENDER_MATCHED) {
if (typeof onHide === 'function') { if (typeof onHide === 'function') {
onHide(location!, matchAnyway, livePath, alwaysLive) onHide(location!, matchAnyway, history, livePath, alwaysLive)
} }
} }
this.liveState = LiveState.HIDE_RENDER this.liveState = LiveState.HIDE_RENDER
} }
const props = { ...staticContext, location, match: matchAnyway } // unmount
if (!matchAnyway) {
this.liveState = LiveState.NORMAL_RENDER_UNMATCHED
}
const props = { ...context, location, match: matchOfPath }
// const props = { history, staticContext, location, match: matchAnyway }
// Preact uses an empty array as children by // Preact uses an empty array as children by
// default, so use null if that's the case. // default, so use null if that's the case.
...@@ -249,11 +292,11 @@ class LiveRoute extends React.Component<IProps, any> { ...@@ -249,11 +292,11 @@ class LiveRoute extends React.Component<IProps, any> {
// normal render from Route // normal render from Route
return children && !isEmptyChildren(children) return children && !isEmptyChildren(children)
? children ? children
: props.match : matchAnyway
? component ? component
? React.createElement(component, props) ? React.createElement(component, props)
: render : render
? render(props) ? render(props as any)
: null : null
: null : null
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment