The original address
1. Introduction
In daily development, the transition animation during the page switch is a relatively basic scene. In react projects, we usually use the React-router to manage routes. However, the React-router does not provide the corresponding transition animation function. Instead, it simply replaces components. In some ways, the experience is not so friendly.
To animate react, there are many options: react-transition-group, react-Motion, Animated, etc. However, since the react-transition-group adds enter, enter-active, exit, and exit-active checkmarks to the element, it is designed for our page entry and exit. Based on this, this paper chooses react-transition-group to achieve animation effects.
Next, this article will combine the two to provide a way to realize the animation of the route transition, just to throw a brick to attract jade ~
2. Requirements
Let’s figure out what the transition looks like. As shown below:
3. react-router
First, let’s take a look at the basic usage of the React-router.
Here we will use the BrowserRouter, Switch, and Route components provided by React-Router.
- BrowserRouter: A route implemented in the form of the history API provided by HTML5 (as well as a hash route).
- Switch: If multiple Route components match at the same time, this is displayed by default. However, the Route component wrapped in the Switch displays only the first Route that is matched.
- Route: indicates the Route component. Path specifies the matched Route. Component specifies the component displayed during Route matching.
// src/App1/index.js
export default class App1 extends React.PureComponent {
render() {
return( <BrowserRouter> <Switch> <Route exact path={'/'} component={HomePage}/> <Route exact path={'/about'} component={AboutPage}/> <Route exact path={'/list'} component={ListPage}/> <Route exact path={'/detail'} component={DetailPage}/> </Switch> </BrowserRouter> ); }}Copy the code
As shown above, this is the key implementation part of routing. We created the home page, about page, list page and details page. The jump relation is:
- Home ↔ about page
- Home ↔ list page ↔ details page
Take a look at the current default route switching effect:
4. react-transition-group
As can be seen from the above illustration, the React-Router has no transition effect during route switching. Instead, it is replaced directly, which is very stiff.
We need to learn how to use react-transition-group before we introduce how to use react-transition-group. With this in mind, I’ll take a brief look at the CSSTransition and TransitionGroup components it provides.
4.1 CSSTransition
CSSTransition is a component provided by the React-Transition-group. Here is a brief introduction to its working principle.
When the in prop is set to true, the child component will first receive the class example-enter, then the example-enter-active will be added in the next tick. CSSTransition forces a reflow between before adding the example-enter-active. This is an important trick because it allows us to transition between example-enter and example-enter-active even though they were added immediately one after another. Most notably, this is what makes it possible for us to animate appearance.
When the IN attribute of the CSSTransition is set to true, the CSSTransition will first add the class of xxx-Enter to its child components and then add the class of xxx-Enter to the next tick. So we can take advantage of this by using the TRANSITION property of our CSS to animate elements smoothly between two states.
Conversely, when the IN property is set to false, CSSTransition adds the classes of XXX-exit and XXX-exit-Active to the child components. (More details can be found on the official website)
Based on the above two points, should we just write the CSS style corresponding to the class in advance? Try a small demo, as shown below:
// src/App2/index.js
export default class App2 extends React.PureComponent {
state = {show: true};
onToggle = (a)= > this.setState({show:!this.state.show});
render() {
const {show} = this.state;
return (
<div className={'container'} >
<div className={'square-wrapper'} >
<CSSTransition
in={show}
timeout={500}
classNames={'fade'}
unmountOnExit={true}
>
<div className={'square'} / >
</CSSTransition>
</div>
<Button onClick={this.onToggle}>toggle</Button>
</div>); }}Copy the code
/* src/App2/index.css */
.fade-enter {
opacity: 0;
transform: translateX(100%);
}
.fade-enter-active {
opacity: 1;
transform: translateX(0);
transition: all 500ms;
}
.fade-exit {
opacity: 1;
transform: translateX(0);
}
.fade-exit-active {
opacity: 0;
transform: translateX(100%);transition: all 500ms;
}
Copy the code
Take a look at the effect. Is it similar to the entry and exit effect of the page?
4.2 TransitionGroup
While CSSTransition is handy for handling animations, it’s still a bit thin to manage multiple pages of animations directly. React-transition-group provides TransitionGroup components.
The component manages a set of transition components ( and ) in a list. Like with the transition components, is a state machine for managing the mounting and unmounting of components over time.
The TransitionGroup component is used to manage the mounting and unmounting of a bunch of nodes, and is ideal for handling multiple pages. This may seem confusing, but let’s take a look at some code to explain how TransitionGroup works.
// src/App3/index.js
export default class App3 extends React.PureComponent {
state = {num: 0};
onToggle = (a)= > this.setState({num: (this.state.num + 1) % 2});
render() {
const {num} = this.state;
return (
<div className={'container'} >
<TransitionGroup className={'square-wrapper'} >
<CSSTransition
key={num}
timeout={500}
classNames={'fade'}
>
<div className={'square'} >{num}</div>
</CSSTransition>
</TransitionGroup>
<Button onClick={this.onToggle}>toggle</Button>
</div>); }}Copy the code
Let’s look at the effect first, and then explain:
Comparing the code for App3 and App2, we can see that CSSTransition does not have the in attribute this time, but uses the key attribute. But why does it still work?
Before we answer that question, let’s consider a question:
Because the React DOM diff mechanism uses the key property, react unmounts the old node and mounts the new node if the keys are different. In the above code, shouldn’t the old node disappear immediately because the key has changed, but why do we still see it fade out?
The key is the TransitionGroup, because it saves the node to be removed when it senses that its children are changing, and does not remove the node until the end of the animation.
So in the example above, when we press the toggle button, the process of change can be understood as follows:
0 < TransitionGroup > < div > < / div > < / TransitionGroup > ⬇ ️ ⬇ ️ ⬇ ️ ⬇ ️ ⬇ ️ ⬇ ️ ⬇ ️ ⬇ ️ < TransitionGroup > < div > 0 < / div > < div > 1 < / div > < / TransitionGroup > ⬇ ️ ⬇ ️ ⬇ ️ ⬇ ️ ⬇ ️ ⬇ ️ ⬇ ️ ⬇ ️ < TransitionGroup > < div > 1 < / div > < / TransitionGroup >Copy the code
As explained above, we can subtly use key changes to let the TransitionGroup take over the creation and destruction of the page during the transition, just by choosing the right key and what CSS styles are needed to animate it.
5. Page transition animation
React-router and react-transition-group (router, router, router, router, router, router, router, router)
As mentioned at the end of the previous section, with TransitionGroup our problem becomes selecting the appropriate key value. So in a routing system, what is the appropriate key value?
Since we are triggering the transition animation during a page switch, it is natural that the value associated with the route should be the key value. The Location object in the React-Router has a key property that changes as the address in the browser changes. However, this might not be appropriate in a real-world scenario where a query parameter or hash change can also cause location.key to change, but often these scenarios do not need to trigger the transition animation.
Therefore, I think the selection of key value depends on different projects. In most cases, location. pathName is recommended as the key because it is the route between our different pages.
React-transition-group (router) {react-router (router) {react-transition-group (router)}
// src/App4/index.js
const Routes = withRouter(({location}) = >( <TransitionGroup className={'router-wrapper'}> <CSSTransition timeout={5000} classNames={'fade'} key={location.pathname} > <Switch location={location}> <Route exact path={'/'} component={HomePage} /> <Route exact path={'/about'} component={AboutPage} /> <Route exact path={'/list'} component={ListPage} /> <Route exact path={'/detail'} component={DetailPage} /> </Switch> </CSSTransition> </TransitionGroup> )); export default class App4 extends React.PureComponent { render() { return ( <BrowserRouter> <Routes/> </BrowserRouter> ); }}Copy the code
This is the effect:
App4 uses much the same code as App3, except that the original div is replaced with a Switch component and withRouter is also used.
WithRouter is a high-order component of the React-Router that provides location, history, and other objects for your component. Because we’re going to use location. pathName as the CSSTransition key, we use it.
In addition, there is a hole in the location property of the Switch.
A location object to be used for matching children elements instead of the current history location (usually the current browser URL).
The Switch component uses this object to match the route in its children, using the current browser URL by default. If we didn’t specify it in the above example, then something strange would happen in the transition animation, where there are two identical nodes moving at the same time… Like this:
This is because the TransitionGroup component retains the Switch node to be removed, but when the location changes, the old Switch node matches the route in its children with the changed location. Since the location is up to date, both switches match the same page. Fortunately, we can change the Location property of the Switch, as shown in the code above, so that it doesn’t always match with the current location.
6. Page dynamic transition animation
Although react-transition-group and react-router are used to implement a simple transition animation, there is a serious problem. Looking carefully at the schematic diagram in the previous section, it is not difficult to find that the animation effect of entering the next page is in line with expectations, but the animation effect of retreating is what the hell… The previous page should fade in from the left and the current page fades out from the right. The current page fades out on the left and the next page fades in on the right, just like entering the next page. The reason for the error is simple:
First, we divide route changes into forward and back operations. In the forward operation, the exit effect of the current page fades out to the left; During the back operation, the exit effect of the current page fades out to the right. So we just use the fade-exit and fade-exit-Active classes, and obviously the animation will be the same.
Therefore, the solution is simple, we use two classes to manage the animation of the forward and back operations, respectively.
/* src/App5/index.css */
/* Route forward entry/exit animation */
.forward-enter {
opacity: 0;
transform: translateX(100%);
}
.forward-enter-active {
opacity: 1;
transform: translateX(0);
transition: all 500ms;
}
.forward-exit {
opacity: 1;
transform: translateX(0);
}
.forward-exit-active {
opacity: 0;
transform: translateX(100%);transition: all 500ms;
}
/* Entry/exit animation */ when the route backs up
.back-enter {
opacity: 0;
transform: translateX(100%); }.back-enter-active {
opacity: 1;
transform: translateX(0);
transition: all 500ms;
}
.back-exit {
opacity: 1;
transform: translateX(0);
}
.back-exit-active {
opacity: 0;
transform: translate(100%).transition: all 500ms;
}
Copy the code
However, CSS support alone is not enough, we need to add appropriate classes for different routing operations. Once a component is mounted, its exit animation is already determined. See this issue on the web site. That is, even if we could dynamically add different ClassNames attributes to CSSTransition to specify the animation effect, it would not work.
The solution is given at the bottom of the issue. We can override the className of the TransitionGroup with the ChildFactory property and the react. cloneElement method. Such as:
<TransitionGroup childFactory={child => React.cloneElement(child, {
classNames: 'your-animation-class-name'
})}>
<CSSTransition>
...
</CSSTransition>
</TransitionGroup>
Copy the code
All that remains is how to choose the right animation class. The essence of this problem is how to determine whether the current route change is a forward or back operation. To react – the router already close to ready to us, it provides the history of the object has an action attribute, represent the current routing change type, its value is’ PUSH ‘|’ POP ‘|’ REPLACE ‘. So, let’s tweak the code again:
// src/App5/index.js
const ANIMATION_MAP = {
PUSH: 'forward'.POP: 'back'
}
const Routes = withRouter(({location, history}) = > (
<TransitionGroup
className={'router-wrapper'}
childFactory={child => React.cloneElement(
child,
{classNames: ANIMATION_MAP[history.action]}
)}
>
<CSSTransition
timeout={500}
key={location.pathname}
>
<Switch location={location}>
<Route exact path={'/'} component={HomePage} />
<Route exact path={'/about'} component={AboutPage} />
<Route exact path={'/list'} component={ListPage} />
<Route exact path={'/detail'} component={DetailPage} />
</Switch>
</CSSTransition>
</TransitionGroup>
));
Copy the code
Take a look at the modified animation:
7. Optimize
In fact, this section is not an optimization, the idea of transition animations is basically over here, and you can use your imagination to make transitions even cooler by adding CSS. However, here is how to write our route more configuration (personal preference, don’t spray).
As we know, the React – Router received a major overhaul with the V4 upgrade. Dynamic routing is preferred over static routing. However, on a case-by-case basis, in some projects individuals prefer centralized routing management. In the case of the above example, a RouteConfig would be desirable, like the following:
// src/App6/RouteConfig.js
export const RouterConfig = [
{
path: '/',
component: HomePage
},
{
path: '/about',
component: AboutPage,
sceneConfig: {
enter: 'from-bottom'.exit: 'to-bottom'
}
},
{
path: '/list',
component: ListPage,
sceneConfig: {
enter: 'from-right'.exit: 'to-right'
}
},
{
path: '/detail',
component: DetailPage,
sceneConfig: {
enter: 'from-right'.exit: 'to-right'}}];Copy the code
From the above RouterConfig, we can clearly see which component is assigned to each page and what the transitions look like, such as the About page entering from the bottom and the list and details pages entering from the right. All in all, we can get a lot of useful information directly from this static route configuration table without having to dig into the code to get the information.
So, for the above requirement, how should our corresponding routing code be adjusted? Take a look below:
// src/App6/index.js
const DEFAULT_SCENE_CONFIG = {
enter: 'from-right'.exit: 'to-exit'
};
const getSceneConfig = location= > {
const matchedRoute = RouterConfig.find(config= > new RegExp(` ^${config.path}$`).test(location.pathname));
return (matchedRoute && matchedRoute.sceneConfig) || DEFAULT_SCENE_CONFIG;
};
let oldLocation = null;
const Routes = withRouter(({location, history}) = > {
// The transition animation should use the current page's sceneConfig, so:
// For push, use the route sceneConfig that matches the new location
// For pop operations, use the route sceneConfig matched by the old location
let classNames = ' ';
if(history.action === 'PUSH') {
classNames = 'forward-' + getSceneConfig(location).enter;
} else if(history.action === 'POP' && oldLocation) {
classNames = 'back-' + getSceneConfig(oldLocation).exit;
}
// Update the old location
oldLocation = location;
return (
<TransitionGroup
className={'router-wrapper'}
childFactory={child= > React.cloneElement(child, {classNames})}
>
<CSSTransition timeout={500} key={location.pathname}>
<Switch location={location}>
{RouterConfig.map((config, index) => (
<Route exact key={index} {. config} / >
))}
</Switch>
</CSSTransition>
</TransitionGroup>
);
});
Copy the code
Because CSS code is a bit too much, I won’t post it here, but it is nothing more than the corresponding transition animation configuration, the complete code can be seen on Github repository. Let’s take a look at the results so far:
8. Summarize
React-router and react-transition-group are introduced in this paper. The working principle of CSSTransition and TransitionGroup animation is also analyzed. Then the react-router and react-transition-group are combined to complete a transition animation. The childFactory property of TransitionGroup is used to solve the problem of dynamic transition animation. Finally, routes are configured to achieve unified route management and animation configuration, and a react-router + react-transition-group transition animation exploration is completed.
9. Reference
- A shallow dive into router v4 animated transitions
- Dynamic transitions with react router and react transition group
- Issue#182 of react-transition-group
- StackOverflow: react-transition-group and react clone element do not send updated props
All the code for this article is hosted here, if you think it is good, you can give it a STAR.