diff --git a/README.md b/README.md index 507effda..ee488408 100644 --- a/README.md +++ b/README.md @@ -53,6 +53,22 @@ You can even mix-and-match URL parameters and normal `props`. ``` +### Nesting routers + +Routers will append the parent Routers' URLs together to come up with the matching route for children. + +```js + + //will route '/' + //will route '/app/*' (could also use default here) + //will route '/app/b' + //will route '/app/c' + + //will route '/d' + //will route anything not listed above + +``` + ### Lazy Loading Lazy loading (code splitting) with `preact-router` can be implemented easily using the [AsyncRoute](https://www.npmjs.com/package/preact-async-route) module: diff --git a/src/index.js b/src/index.js index 868b190c..e76badf7 100644 --- a/src/index.js +++ b/src/index.js @@ -1,5 +1,5 @@ import { cloneElement, h, Component } from 'preact'; -import { exec, pathRankSort, assign } from './util'; +import { exec, pathRankSort, assign, segmentize } from './util'; let customHistory = null; @@ -146,12 +146,23 @@ function initEventListeners() { class Router extends Component { - constructor(props) { + constructor(props, context) { super(props); + this.baseUrl = this.props.base || ''; + if (props.path) { + let segments = segmentize(props.path); + segments.forEach(segment => { + if (segment.indexOf(':') == -1) { + this.baseUrl = this.baseUrl + '/' + segment; + } + }); + } if (props.history) { customHistory = props.history; } - + if (context && context['preact-router-base'] && !this.props.base) { + this.baseUrl = context['preact-router-base'] + this.baseUrl; + } this.state = { url: props.url || getCurrentUrl() }; @@ -159,6 +170,11 @@ class Router extends Component { initEventListeners(); } + getChildContext() { + let result = {['preact-router-base']: this.baseUrl}; + return result; + } + shouldComponentUpdate(props) { if (props.static!==true) return true; return props.url!==this.props.url || props.onChange!==this.props.onChange; @@ -211,7 +227,7 @@ class Router extends Component { getMatchingChildren(children, url, invoke) { return children.slice().sort(pathRankSort).map( vnode => { let attrs = vnode.attributes || {}, - path = attrs.path, + path = this.baseUrl + attrs.path, matches = exec(url, path, attrs); if (matches) { if (invoke!==false) { diff --git a/src/match.js b/src/match.js index ae3a00ca..24ae5179 100644 --- a/src/match.js +++ b/src/match.js @@ -1,7 +1,23 @@ -import { h, Component } from 'preact'; +import { h, Component, cloneElement } from 'preact'; import { subscribers, getCurrentUrl, Link as StaticLink } from 'preact-router'; +import { exec, segmentize } from './util'; export class Match extends Component { + constructor(props, context) { + super(props); + this.baseUrl = this.props.base || ''; + if (props.path) { + let segments = segmentize(props.path); + segments.forEach(segment => { + if (segment.indexOf(':') == -1) { + this.baseUrl = this.baseUrl + '/' + segment; + } + }); + } + if (context && context['preact-router-base']) { + this.baseUrl = context['preact-router-base'] + this.baseUrl; + } + } update = url => { this.nextUrl = url; this.setState({}); @@ -12,15 +28,22 @@ export class Match extends Component { componentWillUnmount() { subscribers.splice(subscribers.indexOf(this.update)>>>0, 1); } - render(props) { + getChildContext() { + let result = {['preact-router-base']: this.baseUrl}; + return result; + } + render(props, state, context) { let url = this.nextUrl || getCurrentUrl(), path = url.replace(/\?.+$/,''); this.nextUrl = null; - return props.children[0] && props.children[0]({ + const newProps = { url, path, - matches: path===props.path - }); + matches: path===props.path || exec(path, context['preact-router-base'] + props.path, {}) + }; + return props.children[0] && + (typeof props.children[0] === 'function' ? + props.children[0](newProps) : cloneElement(props.children[0], newProps)); } } diff --git a/test/dom.js b/test/dom.js index 5687a832..67842a84 100644 --- a/test/dom.js +++ b/test/dom.js @@ -121,6 +121,193 @@ describe('dom', () => { expect(A.prototype.componentWillUnmount).to.have.been.calledOnce; }); + it('should support nested routers with default', () => { + class X { + componentWillMount() {} + componentWillUnmount() {} + render(){ return
; } + } + sinon.spy(X.prototype, 'componentWillMount'); + sinon.spy(X.prototype, 'componentWillUnmount'); + class Y { + componentWillMount() {} + componentWillUnmount() {} + render(){ return
; } + } + sinon.spy(Y.prototype, 'componentWillMount'); + sinon.spy(Y.prototype, 'componentWillUnmount'); + mount( + + + + + + + ); + expect(X.prototype.componentWillMount).not.to.have.been.called; + expect(Y.prototype.componentWillMount).not.to.have.been.called; + route('/app/x'); + expect(X.prototype.componentWillMount).to.have.been.calledOnce; + expect(X.prototype.componentWillUnmount).not.to.have.been.called; + expect(Y.prototype.componentWillMount).not.to.have.been.called; + expect(Y.prototype.componentWillUnmount).not.to.have.been.called; + route('/app/y'); + expect(X.prototype.componentWillMount).to.have.been.calledOnce; + expect(X.prototype.componentWillUnmount).to.have.been.calledOnce; + expect(Y.prototype.componentWillMount).to.have.been.calledOnce; + expect(Y.prototype.componentWillUnmount).not.to.have.been.called; + }); + + it('should support nested routers with path', () => { + class X { + componentWillMount() {} + componentWillUnmount() {} + render(){ return
; } + } + sinon.spy(X.prototype, 'componentWillMount'); + sinon.spy(X.prototype, 'componentWillUnmount'); + class Y { + componentWillMount() {} + componentWillUnmount() {} + render(){ return
; } + } + sinon.spy(Y.prototype, 'componentWillMount'); + sinon.spy(Y.prototype, 'componentWillUnmount'); + mount( + + + + + + + ); + expect(X.prototype.componentWillMount).not.to.have.been.called; + expect(Y.prototype.componentWillMount).not.to.have.been.called; + route('/baz/j'); + expect(X.prototype.componentWillMount).to.have.been.calledOnce; + expect(X.prototype.componentWillUnmount).not.to.have.been.called; + expect(Y.prototype.componentWillMount).not.to.have.been.called; + expect(Y.prototype.componentWillUnmount).not.to.have.been.called; + route('/baz/box/k'); + expect(X.prototype.componentWillMount).to.have.been.calledOnce; + expect(X.prototype.componentWillUnmount).to.have.been.calledOnce; + expect(Y.prototype.componentWillMount).to.have.been.calledOnce; + expect(Y.prototype.componentWillUnmount).not.to.have.been.called; + }); + + it('should support deeply nested routers', () => { + class X { + componentWillMount() {} + componentWillUnmount() {} + render(){ return
; } + } + sinon.spy(X.prototype, 'componentWillMount'); + sinon.spy(X.prototype, 'componentWillUnmount'); + class Y { + componentWillMount() {} + componentWillUnmount() {} + render(){ return
; } + } + sinon.spy(Y.prototype, 'componentWillMount'); + sinon.spy(Y.prototype, 'componentWillUnmount'); + mount( + + + + + + + + + ); + expect(X.prototype.componentWillMount).not.to.have.been.called; + expect(Y.prototype.componentWillMount).not.to.have.been.called; + route('/baz/j'); + expect(X.prototype.componentWillMount).to.have.been.calledOnce; + expect(X.prototype.componentWillUnmount).not.to.have.been.called; + expect(Y.prototype.componentWillMount).not.to.have.been.called; + expect(Y.prototype.componentWillUnmount).not.to.have.been.called; + route('/baz/box/k'); + expect(X.prototype.componentWillMount).to.have.been.calledOnce; + expect(X.prototype.componentWillUnmount).to.have.been.calledOnce; + expect(Y.prototype.componentWillMount).to.have.been.calledOnce; + expect(Y.prototype.componentWillUnmount).not.to.have.been.called; + }); + + it('should support nested routers and Match(s)', () => { + class X { + componentWillMount() {} + componentWillUnmount() {} + render(){ return
; } + } + sinon.spy(X.prototype, 'componentWillMount'); + sinon.spy(X.prototype, 'componentWillUnmount'); + class Y { + componentWillMount() {} + componentWillUnmount() {} + render(){ return
; } + } + sinon.spy(Y.prototype, 'componentWillMount'); + sinon.spy(Y.prototype, 'componentWillUnmount'); + mount( + + + + + + + ); + expect(X.prototype.componentWillMount, 'X1').not.to.have.been.called; + expect(Y.prototype.componentWillMount, 'Y1').not.to.have.been.called; + route('/ccc/jjj'); + expect(X.prototype.componentWillMount, 'X2').to.have.been.calledOnce; + expect(X.prototype.componentWillUnmount, 'X3').not.to.have.been.called; + expect(Y.prototype.componentWillMount, 'Y2').not.to.have.been.called; + expect(Y.prototype.componentWillUnmount, 'Y3').not.to.have.been.called; + route('/ccc/xxx/kkk'); + expect(X.prototype.componentWillMount, 'X4').to.have.been.calledOnce; + expect(X.prototype.componentWillUnmount, 'X5').to.have.been.calledOnce; + expect(Y.prototype.componentWillMount, 'Y4').to.have.been.calledOnce; + expect(Y.prototype.componentWillUnmount, 'Y5').not.to.have.been.called; + }); + + it('should support nested router reset via base attr', () => { + class X { + componentWillMount() {} + componentWillUnmount() {} + render(){ return
; } + } + sinon.spy(X.prototype, 'componentWillMount'); + sinon.spy(X.prototype, 'componentWillUnmount'); + class Y { + componentWillMount() {} + componentWillUnmount() {} + render(){ return
; } + } + sinon.spy(Y.prototype, 'componentWillMount'); + sinon.spy(Y.prototype, 'componentWillUnmount'); + mount( + + + + + + + ); + expect(X.prototype.componentWillMount).not.to.have.been.called; + expect(Y.prototype.componentWillMount).not.to.have.been.called; + route('/baz/j'); + expect(X.prototype.componentWillMount).to.have.been.calledOnce; + expect(X.prototype.componentWillUnmount).not.to.have.been.called; + expect(Y.prototype.componentWillMount).not.to.have.been.called; + expect(Y.prototype.componentWillUnmount).not.to.have.been.called; + route('/baz/foo/k'); + expect(X.prototype.componentWillMount).to.have.been.calledOnce; + expect(X.prototype.componentWillUnmount).to.have.been.calledOnce; + expect(Y.prototype.componentWillMount).to.have.been.calledOnce; + expect(Y.prototype.componentWillUnmount).not.to.have.been.called; + }); + it('should support re-routing', done => { class A { componentWillMount() {