remix-run / history Goto Github PK
View Code? Open in Web Editor NEWManage session history with JavaScript
License: MIT License
Manage session history with JavaScript
License: MIT License
It's pretty clear to me at this point that jsdom isn't going to work. The support for the HTML5 history API appears to be incomplete, and, given the quirks of the various history implementations, I'd rather test in real browsers anyway.
Stuff we need to be able to do in a single integration test:
hashchange
/popstate
events when the URL changeswindow.history.pushState
/window.history.replaceState
window.location.replace
window.location.href
Stuff we need to be able to do in a test suite:
There are a few options out there, none of which I'm very familiar with. If anyone has any experience with any of these stacks, please chime in.
We use karma to test React Router, so it would be great if we could stick with that. But it seems like I remember it complaining when we tried to change the URL (which is strange because I thought karma ran all your tests in an <iframe>
... ?)
For better ES6/7 module optimization! See here for more info on jsnext:main and an example here. At least With the source it will be possible to bundle only what is needed using tools like Rollup.
Another reason is mybabel
config targets ES6 platforms i.e. Atom Electron, nw.js, cordova with crosswalk webview, etc which all have good support for ES6 and thus don't need full transform to ES5.
The changes are tiny so didn't push a PR, basically:
add "jsnext:main": "modules/index.js",
and
remove modules
from .npmignore
(i.e. don't exclude the ES6 source).
If possible would be appreciated!
Hi,
I would like to be able to check where browser back button is taking me and modify if needed. Is that possible? Also is it possible to stop transition all together from the hook?
Cheers
I'm using the react-routers match
function both client side and server side (I have my reasons). At the moment I'm doing this:
import { createLocation, createHistory } from 'history';
const history = createHistory();
const appLocation = createLocation(
location.pathname + location.search
);
Which works but isn't very clean IMHO. It would be cool if getCurrentLocation
was exposed on the history instance so I can turn the above into:
import { createHistory } from 'history';
const history = createHistory();
const appLocation = history.getCurrentLocation();
As always, I'm willing to submit a PR if authors are interested in this.
I think it would be great to allow opting out of using beforeunload
for the following reasons:
localStorage
to restore it on the next visit rather than annoying the user with a modal when he tries to close the browser window. There's even a discussion about disabling it by default at bugzilla.Incidentally, using beforeunload
to disable BFCache fixes this bug which occurs when re-entering the cached page at a different URL. But as far as I can tell we can fix this in a better way by using the pageshow
event. Something like
function pageShowListener(event) {
if (event.persisted)
history.transitionTo(getCurrentLocation());
}
That way, when using a browser with BFCache, re-entering our app at the same URL wouldn't require any re-renders, and re-entering at a different URL would allow for render optimizations such as React's reconciliation (DOM diffing).
I think I'm not getting my head around the property "key" of the "location" object. In the docs it says: "A unique identifier for this location".
But I have the following scenario:
/home
/about
/home
Why History gives a different key for the same location?
When the user manipulates the URL (either by changing the value in the location bar or using the back/forward buttons) and then does not confirm the transition in the prompt, we should restore the URL to what it was before they changed it. There are a few TODOs in the code about this:
AFAICT this would require us to keep track of a "current" value that would be part of a location's state. This value would be a number that represents the index of that location in the history stack.
For example, if we knew that a given location had a current value of 3, and the previous location had a current value of 6, we could easily calculate that we need to go(3)
to restore the URL to what it was before.
@taurose has said before that he is not sure how reliable this would be.
Getting the following when I'm trying to install this package:
[email protected] postinstall E:\projects\eVision.ShiftVision\src\node_modules\history
node -e "require('fs').stat('lib', function (e, s) { process.exit(e || !s.isDirectory() ? 1 : 0) })" || npm run build
"require('fs').stat('lib',
^^^^^^^^^^^^^^^^^^^^^^^^^^
SyntaxError: Unexpected token ILLEGAL
at Object. (eval-wrapper:6:22)
at Module._compile (module.js:456:26)
at evalScript (node.js:532:25)
at startup (node.js:80:7)
at node.js:902:3
[email protected] build E:\projects\eVision.ShiftVision\src\node_modules\history
babel ./modules -d lib --ignore 'tests'
modules\Actions.js -> lib\Actions.js
modules\AsyncUtils.js -> lib\AsyncUtils.js
modules\createBrowserHistory.js -> lib\createBrowserHistory.js
modules\createDOMHistory.js -> lib\createDOMHistory.js
modules\createHashHistory.js -> lib\createHashHistory.js
modules\createHistory.js -> lib\createHistory.js
modules\createLocation.js -> lib\createLocation.js
modules\createMemoryHistory.js -> lib\createMemoryHistory.js
modules\DOMStateStorage.js -> lib\DOMStateStorage.js
modules\DOMUtils.js -> lib\DOMUtils.js
modules\enableBeforeUnload.js -> lib\enableBeforeUnload.js
modules\enableQueries.js -> lib\enableQueries.js
modules\ExecutionEnvironment.js -> lib\ExecutionEnvironment.js
SyntaxError: modules/index.js: Unexpected token (1:7)
1 | export createHistory from './createBrowserHistory'
| ^
2 | export createHashHistory from './createHashHistory'
3 | export createMemoryHistory from './createMemoryHistory'
4 | export createLocation from './createLocation
npm ERR! Failed at the [email protected] build script 'babel ./modules -d lib --ignore 'tests''.
npm ERR! This is most likely a problem with the history package,
npm ERR! not with npm itself.
npm ERR! Tell the author that this fails on your system:
npm ERR! babel ./modules -d lib --ignore 'tests'
npm ERR! You can get their info via:
npm ERR! npm owner ls history
I'm currently working on an application for HbbTV devices using React and React Router (and therefore this package). After upgrading to React Router v1.0.0-rc, our first history-dependent version, navigation no longer works due to the use of sessionStorage in DOMStateStorage, which fails silently but prevents further execution. Unfortunately when dealing with HbbTV we often find that APIs like this are available on window
but are unusable or crippled somehow, making polyfilling them impossible.
We're considering the following workarounds so far:
try {} catch () {}
Link
component and write our own onClick
handler which grabs the href
from the DOM node and manually navigates to itThe last two options are very quick fixes but sacrifice functionality which is not ideal. Do you have any advice for utilising this package on platforms where sessionStorage exists but throws errors when used?
Hey. It would be great to know the direction of the transition thus we can set a different animation (with react-router
) for forward/back actions. I first though that I can depend on action
but browser back/forward buttons are always POP
. I would appreciate if we can add this feature, thx?
When installing, the build step complains that
'babel' is not recognized as an internal or external command, operable program or batch file.
I installed babel as a dev dependency in my project and then history was able to install correctly.
Could the build step be changed to run babel from the node_modules instead?
(I'm on Windows btw)
Our app has an object detail page that has a back button. We want to set it up so if you visit the page from somewhere in our app the back button will take you back to that page. However if you visit the detail page directly the back button will take you to a default object list page. This currently working using the following:
if (window.history.length > 1) {
history.goBack();
}
However, it does break down in the case that you come to the detail page from an external site. In this case the browser will have history and the back button will take you off our app where we would actually like it to take you to the default object list page.
Is it possible to have a history stack that is only the history within the context of the app that is using history? Basically the first time the app loads that page is treated as the first entry in the stack.
I just submitted a pull request (see #51) to fix the fact that this library strips window.location.hash from the url (which breaks in-page offset navigation when using react-router. Ordinarily, when I find a problem with a library and make a fix, I can just point to my github version of the library and the branch name in my package.json
and I'm good to go. But because of the strange build system this library uses, this is not possible, forcing everyone to wait for a new release from the primary author.
Unless I'm missing something (which is entirely possible) ...
For some reason, I would need the SPA page which was based on [email protected] use HTML5 history API on development, but use page reload on production.
const location = process.env.NODE_ENV === 'production'
? RefreshLocation
: HashLocation;
But I saw [email protected] use 'history/lib/createBrowserHistory', so I assume the code could like that
let isSupported = !options.forceReload || supportsHistory();
https://github.com/rackt/history/blob/master/modules/createBrowserHistory.js#L24
Thanks!
Device: iPhone 5
OS: iOS v9.0.1
Browser: Chrome v45.0.2454.89
Bootstrap code:
import { createBrowserHistory } from 'history' // v1.11.0
const history = createBrowserHistory()
history.pushState(null, '/demo/path1')
history.pushState(null, '/demo/path2')
history.goBack()
The history goto our site /demo/path1
The history goto www.google.com
P.S.: It is work fine in Safari.
When createHistory()
is called, transition hooks are not invoked. This makes async transitionHooks unsuitable for the data-loading use case. An API like the following would solve this problem:
let history = createHistory();
history.addTransitionHook(...);
history.listen(...);
history.start();
The call to history.start()
would then invoke the middleware with the initial location
before notifying listeners.
See #50, #22, and #23 for other issues related to supporting data loading via async transition hooks.
We should support BFCache users using the pageshow
event.
See #31 (comment) for more discussion.
When using safari in incognito mode, navigation fails with the following exception:
QuotaExceededError: DOM Exception 22: An attempt was made to add something to storage that exceeded the quota.
Safari denies usage of local/session storages when in incognito mode, and raises this error when you try to.
I am using the BrowserHistory history and it seems like the error stems from history/DOMStateStorage.
When using createBrowserHistory IE9 throws an error. Object doesn't support property or method 'replaceState'
This is likely because of the window.history.replaceState({ ...historyState, key }, null, path)
call in getCurrentLocation method.
EDIT: this only tested with react-router
The docs indicate three types of histories:
In the code, there are two more:
createBrowserHistory seems to actually be what the docs claim createHistory is, so it's unclear what createHistory actually does.
Firefox on Travis is failing for some reason. Tests run fine locally. :/
Maybe just skip Firefox on Travis and run everything on Sauce Labs instead? They've got a nice selection of platforms. And karma seems to have pretty good support for running tests on Sauce Labs and/or BrowserStack. e.g. see the Angular.js karma.config.js
.
In issue #22, I proposed adding support for async transitions via transition middleware. I also created a POC implementation in #23. While the PR was not merged due to a lack of design consensus, similar functionality was subsequently added in ae8dd6f in the form of async transition hooks.
The current implementation has a couple of limitations, however, that make it unsuitable for my use case. I'm creating this issue to re-open the discussion around async transitions building on @mjackson's work. My hope is that the current transitionHook
interface can be extended in such a way that it becomes a suitable approach to achieving async data loading.
Limitation of current transition hook design:
When createHistory()
is called, transition hooks are not invoked
This makes transitionHooks unsuitable for the data-loading use case. An API like the following would solve this problem:
let history = createHistory();
history.addTransitionHook(...);
history.listen(...);
history.start();
The call to history.start()
would then invoke the middleware with the initial location
before notifying listeners.
Interrupted transitions aren't supported
Since transitions can now be async, it's possible for a user-initiated transition to occur while a previous transition is still pending.
In modules/createHistory.js#L134, we throw an invariant
error if this situation arises. Instead of throwing an error, I think we should simply cancel the pending transition in favor of the new one. This matches native browser behavior.
Since transitions can now be async, it's possible for a user-initiated transition to occur while a previous transition is still pending.
In modules/createHistory.js#L134, we throw an invariant
error if this situation arises. Instead of throwing an error, I think we should simply cancel the pending transition in favor of the new one. This matches native browser behavior.
See #50, #22, and #23 for other issues related to supporting data loading via async transition hooks.
When using nginx to serve your application at a url such as, http://mydomain.com/some/base
createHistory()
cannot determine the application root url.
You get the following error when trying to visit your application:
Warning: Location "/some/base" did not match any routes
same issue on react-router: remix-run/react-router#1960
Using react-router (1.0.0-rc1) and history (1.9.1), using the History mixin to call this.pushState. I get transitioned to the new state, but fragment identifier seems to be discarded. For example, if take this call to history.pushState:
this.history.pushState(null, '/home#list')
My new location becomes http://whatever/home
with no hash.
I'm not sure when to use either - seems like they both support pushState?
The most minimal test caseย I could come up with:
import {createHistory} from "history/lib/createBrowserHistory";
let history = createHistory();
history.listen(loc => console.log(loc.pathname));
history.pushState(null, window.location.href);
The browser goes to the right location, but the pathname printed in the console (after pushState
is called) is wrong.
From https://developer.mozilla.org/en-US/docs/Web/API/History/pushState:
The new URL does not need to be absolute; if it's relative, it's resolved relative to the current URL. The new URL must be of the same origin as the current URL; otherwise, pushState() will throw an exception. This parameter is optional; if it isn't specified, it's set to the document's current URL.
I often run apps on the same host but different base paths. I worked around this by parsing the pathname out of the URL before calling pushState
.
Does exists any functions can we compare equal between two locations?
Maybe should exist a function like this:
export function locationEquals(location1, location2) {
if (location1 && location2 && location1.key && location2.key) {
return location1.key === location2.key;
}
return false;
}
Would it make sense to use the path as an implicit queryKey
when queryKey
is disabled? It's not as good as having an actual query key, but might be an improvement in common use cases.
I'm having issues using HashHistory with React Router 1.0-rc1 (I'm also using React 0.13 and Redux 3.0.0).
My Router looks like this:
render((
<div>
<Provider store={store}>{() =>
<Router history={createHistory()}>
<Redirect from="/" to="/home" />
<Route component={App} path="/">
<Route component={Landing} path="/landing" />
<Route component={AuthApp} onEnter={Auth.requireAuth}>
<Route component={Home} path="/home" />
</Route>
</Route>
<Route component={FourOhFour} path="*" />
</Router>
}</Provider>
{debug}
</div>
), document.getElementById('app'));
But I get this error when running the app on the default "/" route.
Uncaught TypeError: Cannot read property 'indexOf' of nullcreatePath @ useQueries.js:61createLocationFromRedirectInfo @ useRoutes.js:102(anonymous function) @ useRoutes.js:117done @ AsyncUtils.js:13(anonymous function) @ TransitionUtils.js:56(anonymous function) @ TransitionUtils.js:16(anonymous function) @ TransitionUtils.js:54next @ AsyncUtils.js:20loopAsync @ AsyncUtils.js:26runEnterHooks @ TransitionUtils.js:53finishMatch @ useRoutes.js:113(anonymous function) @ useRoutes.js:86done @ AsyncUtils.js:13(anonymous function) @ matchRoutes.js:129(anonymous function) @ matchRoutes.js:98done @ AsyncUtils.js:13(anonymous function) @ matchRoutes.js:129(anonymous function) @ matchRoutes.js:98done @ AsyncUtils.js:13(anonymous function) @ matchRoutes.js:129(anonymous function) @ matchRoutes.js:80getIndexRoute @ matchRoutes.js:31matchRouteDeep @ matchRoutes.js:74(anonymous function) @ matchRoutes.js:127next @ AsyncUtils.js:20loopAsync @ AsyncUtils.js:26matchRoutes @ matchRoutes.js:126(anonymous function) @ matchRoutes.js:92getChildRoutes @ matchRoutes.js:13matchRouteDeep @ matchRoutes.js:87(anonymous function) @ matchRoutes.js:127next @ AsyncUtils.js:20(anonymous function) @ matchRoutes.js:131matchRouteDeep @ matchRoutes.js:108(anonymous function) @ matchRoutes.js:127next @ AsyncUtils.js:20loopAsync @ AsyncUtils.js:26matchRoutes @ matchRoutes.js:126(anonymous function) @ matchRoutes.js:92getChildRoutes @ matchRoutes.js:13matchRouteDeep @ matchRoutes.js:87(anonymous function) @ matchRoutes.js:127next @ AsyncUtils.js:20(anonymous function) @ matchRoutes.js:131(anonymous function) @ matchRoutes.js:104getChildRoutes @ matchRoutes.js:19matchRouteDeep @ matchRoutes.js:87(anonymous function) @ matchRoutes.js:127next @ AsyncUtils.js:20loopAsync @ AsyncUtils.js:26matchRoutes @ matchRoutes.js:126match @ useRoutes.js:82(anonymous function) @ useRoutes.js:248(anonymous function) @ useQueries.js:45(anonymous function) @ createHistory.js:80updateLocation @ createHistory.js:79listen @ createHistory.js:102listen @ createDOMHistory.js:31listen @ createHashHistory.js:153listen @ useQueries.js:42listen @ useRoutes.js:244componentWillMount @ Router.js:92ReactCompositeComponentMixin.mountComponent @ ReactCompositeComponent.js:228ReactPerf.measure.wrapper @ ReactPerf.js:70ReactReconciler.mountComponent @ ReactReconciler.js:38ReactCompositeComponentMixin.mountComponent @ ReactCompositeComponent.js:247ReactPerf.measure.wrapper @ ReactPerf.js:70ReactReconciler.mountComponent @ ReactReconciler.js:38ReactMultiChild.Mixin.mountChildren @ ReactMultiChild.js:192ReactDOMComponent.Mixin._createContentMarkup @ ReactDOMComponent.js:289ReactDOMComponent.Mixin.mountComponent @ ReactDOMComponent.js:199ReactReconciler.mountComponent @ ReactReconciler.js:38ReactCompositeComponentMixin.mountComponent @ ReactCompositeComponent.js:247ReactPerf.measure.wrapper @ ReactPerf.js:70ReactReconciler.mountComponent @ ReactReconciler.js:38mountComponentIntoNode @ ReactMount.js:248Mixin.perform @ Transaction.js:134batchedMountComponentIntoNode @ ReactMount.js:269Mixin.perform @ Transaction.js:134ReactDefaultBatchingStrategy.batchedUpdates @ ReactDefaultBatchingStrategy.js:66batchedUpdates @ ReactUpdates.js:110ReactMount._renderNewRootComponent @ ReactMount.js:404ReactPerf.measure.wrapper @ ReactPerf.js:70ReactMount.render @ ReactMount.js:493ReactPerf.measure.wrapper @ ReactPerf.js:70startApp @ app.js:77Channel.fire
The failure is coming from this line in the createPath function of the useQueries file.
return history.createPath(pathname + (pathname.indexOf('?') === -1 ? '?' : '&') + queryString);
Apparently, my <Redirect />
is causing the pathname to be null for some reason. Why would this be happening?
I have an app that is mounted at /myapp/section
.
I'd love to be able to specify a rootUrl in order to avoid the below code. If you're on board I can happily issue a PR.
I currently have to specify the baseUrl like this when using React-router:
let base = '/myapp/section';
render() {
return (
<div>
<Link to={`${base}/about`}>About</Link>
{this.props.children}
</div>
)
}
...
<Router history={history}>
<Route path={`${base}/`} component={MainContainer}>
<IndexRoute component={SecondPage} />
<Route path={`${base}/thirdPage`} component={thirdPage}/>
</Route>
</Router>
Current history stack
1: /1
2: /2
3: /3
4: /4
> 5: /5 - current
Now we call go(-2)
1: /1
2: /2
> 3: /3 - current
4: /4
5: /5
Now we call pushState(null, '/6')
1: /1
2: /2
3: /3
> 4: /6 - current
I'm using history in conjunction with react router and redux router. With this setup there is no way to set the queryKey option to false. I'm consistently blowing up my session storage quota.
To initialize the redux router you pass the createHistory function as an argument. createHistory is then called by react router with the default option for queryKey.
I realize this issue has more to do with how these three project are used together, but I figured I would start here.
I am using the latest history version
"history": "1.12.1",
With Express i am getting the following warning
Warning: createLocation is deprecated; use history.createLocation instead
In the clientside code this was easy to fix
const history = require('history');
const historyObj = history.createHistory();
historyObj.createLocation(....)
When you try to run this code with express you will get the following error
[1] [piping] error given was: Error: Invariant Violation: Browser history needs a DOM
[1] at Object.invariant [as default] (/var/www/webapp/node_modules/history/node_modules/invariant/invariant.js:44:15)
[1] at Object.createBrowserHistory [as createHistory] (/var/www/webapp/node_modules/history/lib/createBrowserHistory.js:35:25)
[1] at Object.<anonymous> (/var/www/webapp/src/server.js:5:28)
[1] at Module._compile (module.js:434:26)
[1] at normalLoader (/var/www/webapp/node_modules/babel-core/lib/api/register/node.js:199:5)
[1] at Object.require.extensions.(anonymous function) [as .js] (/var/www/webapp/node_modules/babel-core/lib/api/register/node.js:216:7)
[1] at Module.load (module.js:355:32)
[1] at Module._load (module.js:310:12)
[1] at Function.module._load (/var/www/webapp/node_modules/piping/lib/launcher.js:32:16)
[1] at Module.require (module.js:365:17)
[1] [piping] further repeats of this error will be suppressed...
So how can we use createLocation with express?
Why only npm and not bower support? I used react-router with bower, so it doesn't make any sense to me why you guys don't have it supported.
I am using createMemoryHistory()
for server routing(react-router) and createBrowserHistory()
for client routing.
Then I have url with queries, on client I can reach a parsed query string in this.props.location.query
. But how to reach the same result on server?
I create location object like this
match({location, routes}, (/_..args_/) => {
// do something
});
return false
in a transition hook cancels the transition.
After registering a transition hook, there are two cases when transitioning away from a location:
In the case of (1), the only thing we can do is provide a string to onbeforeunload
to get a confirmation from the user (https://developer.mozilla.org/en-US/docs/Web/API/WindowEventHandlers/onbeforeunload).
In the case of (2) we could allow the developer to do anything they want.
Currently history
only handles case (1) with a bit of indirection between creating the history and registering hooks.
A strawman API to allow for both, and remove the indirection could look something like:
// callback with a string and both cases get a confirmation dialog
history.registerTransitionHook((isUnload, cb) => {
cb("you sure you want to leave?");
});
// if you want to do something fancy for case (2)
history.registerTransitionHook((isUnload, cb) => {
// if its an unload event you must do things synchronously
if (isUnload) {
cb("you sure?")
}
else {
// do something fancier than window.confirm
cb(true) // allow the transition
cb(false) // cancel the transition
}
});
With this API there would be no need for getUserConfirmation
and we could allow devs more freedom when transitioning inside the app instead of restricting them to the constraints of onbeforeunload
in all cases.
We should introduce a createNativeHistory
method that creates a history object that can be used with React Native. Instead of sessionStorage
, a native history should use React Native's AsyncStorage module to persist state.
When using useBasename
together with react-router
, initial location that matches basename
exactly results in location=''
which cannot be matched by react-router
to any route. E.g:
<Route component={App}>
<Route path="/" component={MyComponent} />
</Route>
...
const history = useBasename(createBrowserHistory)({
basename: '/basename'
});
Entering URL /basename
in browser address bar results in
Warning: Location "" did not match any routes
Not sure whether issue is with react-router
not matching empty location "/" or with useBasename, mishandling this specific case... Problem can be fixed on useBasename
side by simply replacing empty pathname with /
We need to add a getScrollPosition
option to history objects so we can store the scroll position before navigating to a new page. Location objects should have this data in the scrollX
and scrollY
properties.
var history = createHistory({
getScrollPosition() {
return { x: 0, y: 0 };
}
});
history.listen(function (location) {
console.log(location.scrollX, location.scrollY);
});
I'd like to hook into history to do data loading. To do this, I need to be able to pause a transition while data loading is occurring.
I think this can be simulated today using registerTransitionHook()
and a custom getUserConfirmation()
handler that doesn't actually show a user confirmation dialog. This is of course a hack.
A better solution would look something like this:
let history = createBrowserHistory();
history.registerTransitionMiddleware(function(nextLocation, prevLocation, fnNext, fnAbort) {
fetchSomeData(location).then(fnNext);
});
The transition would simply freeze until fnNext
or fnAbort
are called.
If the direction sounds good, then I could submit a PR for this.
Like we did in remix-run/react-router@fbe4377
This would greatly simplify setting queries.
E.g.
history.pushState(null, '/path', { thisQueryParam: 'yes', notThisQueryParam: null })
->
/path?thisQueryParam=yes
In other words, include params that are null. In cases an empty param is needed an empty string would be sufficient. I think this is a logical behavior that most developers will understand.
This requires the following workaround which seems odd to me:
erikras/react-redux-universal-hot-example#246
Let me know if you want a PR.
We need to support IE 9. We need someone very brave to:
We already have a check around the HTML5 history API tests that skips them if the environment doesn't support HTML5 history.
I have the setup below for the client environment and server envirnment but get the error:
Uncaught Error: Invariant Violation: Server-side <Router>s need location, branch, params, and components props. Try using Router.run to get all the props you need
my client entry point:
import React from 'react';
import ReactDOM from 'react-dom';
import Root from 'containers/Root';
import configureStore from 'stores';
//import { history } from 'react-router/lib/BrowserHistory';
import {history} from 'utils/history';
const target = document.getElementById('root');
const store = configureStore(window.__INITIAL_STATE__);
ReactDOM.render(<Root routerHistory={history} store={store}/>, target);
my server entry point:
import React from 'react';
import ReactDOM from 'react-dom/server';
import Router from 'react-router';
//import Location from 'react-router/lib/Location';
import routes from '../routes';
import Root from 'containers/Root';
import configureStore from 'stores';
import Location from 'history/lib/createLocation';
let _store;
export function getStoreState () {
return new Promise((resolve) => resolve(_store.getState()));
}
export function render (initialRouterState, initialStoreState) {
return new Promise((resolve, reject) => {
try {
_store = configureStore(initialStoreState);
const rendered = ReactDOM.renderToString(
<Root initialRouterState={initialRouterState} store={_store} />
);
resolve(rendered);
} catch (e) {
reject(e);
}
});
}
export function route (request) {
return new Promise((resolve, reject) => {
const location = new Location(request.path, request.query);
Router.run(routes, location, (err, initialState) => {
if (err) {
reject(err);
} else if (!initialState) {
reject('No initial router state received from React Router.');
} else {
resolve(initialState);
}
});
});
}
location.key
should always be defined, even on the initial POP
, just like it is for PUSH
and REPLACE
. If there is no key, we should replaceState
and define one.
Is it possible to have an option that detects HTML5 History API or falls back to hash automatically?
I would think that would be what most people would prefer to do these days. I think ember has that type of functionality with its "auto" location option.
A declarative, efficient, and flexible JavaScript library for building user interfaces.
๐ Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.
TypeScript is a superset of JavaScript that compiles to clean JavaScript output.
An Open Source Machine Learning Framework for Everyone
The Web framework for perfectionists with deadlines.
A PHP framework for web artisans
Bring data to life with SVG, Canvas and HTML. ๐๐๐
JavaScript (JS) is a lightweight interpreted programming language with first-class functions.
Some thing interesting about web. New door for the world.
A server is a program made to process requests and deliver data to clients.
Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.
Some thing interesting about visualization, use data art
Some thing interesting about game, make everyone happy.
We are working to build community through open source technology. NB: members must have two-factor auth.
Open source projects and samples from Microsoft.
Google โค๏ธ Open Source for everyone.
Alibaba Open Source for everyone
Data-Driven Documents codes.
China tencent open source team.