性能优先!给 Next.js 添加页面切换过渡效果
?本文为稀土掘金技术社区首发签约文章,30天内禁止转载,30天后未获授权禁止转载,侵权必究!
?点击关注公众号,“技术干货”及时达!群友发了一个页面跳转有过渡动画的网站,问:Next.js 项目怎么做页面切换的过渡动画?
作为「辛苦优化一个月,生产上线两用户」的受害者之一,我本来觉得这都不是现在该考虑的问题。
不过转念一想,作为 Next.js 手艺人,写文章都只写 Next.js 技巧的开发者,我还是有必要了解一下这个问题的解决方案。而且,万一将来页面过渡的手艺有用武之地呢?
实测了几种实现方案后,我选择了一种性能最佳的方案——结合 HTML View Transition API 和 React startTransition 来实现,这种方案的好处是使用了浏览器的原生 API,性能更好,而且 React startTransition 可以优化 UI 的响应。
读完本文你将学会:
HTML View Transition API 的用法React startTransition 的用法手写页面切换过渡动画的高性能方案分析一个同类方案里的最佳实践HTML View Transition API 的用法介绍View Transition API 是一个发布没多久的 HTML 新特性,它可以让网页在不同状态或页面之间切换时,创建流畅的过渡动画效果,这种效果可以让用户体验更加丝滑。
在 View Transitions API 发布之前,只能通过 JavaScript 和 CSS 来实现过渡效果,有了这个新特性,开发者可以直接使用浏览器的原生能力实现过渡效果,这样性能更佳,代码也更简洁。
View Transitions API 的基本用法View Transitions API 用法分为两种情况:单页应用(SPA)和多页应用(MPA)。
在单页应用(SPA)中的使用:
if(document.startViewTransition){
document.startViewTransition(()={
//在这里更新DOM
updateDOM();
}else{
//对于不支持的浏览器,直接更新DOM
updateDOM();
}
这段代码会在支持 View Transitions 的浏览器中创建一个平滑的过渡效果,而在不支持的浏览器中则没有过渡效果,但不影响其他功能。
在多页应用(MPA)中的使用:
@view-transition{
navigation:auto;
}
这样就可以在页面导航时自动应用视图转换效果。
这是 View Transitions API 的基本用法,除此之外,开发者还可以实现自定义转换效果、为不同元素设置不同的动画和控制视图转换流程,这些不是本文的要点,你可以到 MDN 文档进行学习。
React startTransition 的用法介绍startTransition 是 React 18 发布的新特性。它允许开发者将某些状态更新标记为“过渡”(Transition),被标记后这些更新会被标记为“低优先级”,它们是非阻塞的,如果有其他更高优先级的更新(比如用户输入),React会先处理这些高优先级更新,高优先级更新完成后再重启 startTransitions 里面的状态更新,以此保障 UI 的响应性。
startTransition 的基本用法StartTransition 的用法很简单,下面是一个 tab 切换的示例:
importReact,{useState,startTransition}from'react';
functionTabContainer(){
const[tab,setTab]=useState('about');
functionselectTab(nextTab){
startTransition(()={
//在这里进行状态更新
setTab(nextTab);
}
return(
div
buttononClick={()=selectTab('about')}About/button
buttononClick={()=selectTab('posts')}Posts/button
buttononClick={()=selectTab('contact')}Contact/button
{tab==='about'AboutTab/}
{tab==='posts'PostsTab/}
{tab==='contact'ContactTab/}
/div
}
手写页面切换过渡动画的高性能方案我们先来分析一下,实现页面切换过渡效果,要考虑哪些场景?
Link 组件跳转页面浏览器前进后退跳转页面现在就来依次实现这两个场景的过渡效果。
覆盖 Link 组件跳转行为读过 Next.js 源码的朋友都知道,Next.js 的 Link 标签是对 HTML 标签的封装,其中跳转行为不是使用 标签的 href 属性,而是使用 onClick 拦截了跳转,并且在点击事件里用浏览器的 history API 来完成跳转。
我们想要在 Link 标签上实现过渡效果,可以依葫芦画瓢,用相似的思路来做——写一个自定义 onClick 事件,自己实现跳转的行为。
新建一个文件 components/TransitionLink.tsx:
"useclient";
importLinkfrom"next/link";
import{useRouter}from"next/navigation";
importReactfrom"react";
constTransitionLink=({href,children,...props})={
constrouter=useRouter();
consthandleClick=(e)={
e.preventDefault();
console.log('拦截跳转行为')
return(
Linkhref={href}onClick={handleClick}{...props}
{children}
/Link
};
exportdefaultTransitionLink;
这段代码里,我们自定义了 onClick 事件,并把其他 props 参数原样传递。
现在把代码里原有的 Link 用 TransitionLink 替换,点击会发现页面没有跳转,但是浏览器控制台打印出 拦截跳转行为,这就说明被我们拦截成功了。
接下来使用 View Transition 和 history API 加入跳转代码:
"useclient";
importLinkfrom"next/link";
import{useRouter}from"next/navigation";
importReactfrom"react";
constTransitionLink=({href,children,...props})={
constrouter=useRouter();
consthandleClick=(e)={
e.preventDefault();
//ViewTransition过渡
if(document.startViewTransition){
document.startViewTransition(()={
router.push(href);//示例只考虑push的情况,正式封装需要考虑replace
}else{
router.push(href);
}
return(
Linkhref={href}onClick={handleClick}{...props}
{children}
/Link
};
exportdefaultTransitionLink;
相比上一段代码,这里只增加了 document.startViewTransition 方法。
现在到页面上尝试跳转页面,会看到轻微的过渡效果。因为 View Transition 默认过渡时间是 0.3 秒,肉眼可能辨别不出来有过渡效果。
为了让过渡效果更明显,可以在全局样式中修改过渡时间,修改 styles/globals.css:
:root{
--view-transition-duration:1s;
}
::view-transition-old(root),
::view-transition-new(root){
animation-duration:var(--view-transition-duration);//覆盖默认过渡时间
}
再去尝试跳转,就会看到非常明显的过渡效果了。
为了让这里的过渡效果不阻塞 UI 的响应,最好再加上 React startTransition 方法:
//……
if(document.startViewTransition){
document.startViewTransition(()={
React.startTransition(()={//新增startTransition
router.push(href);
}else{
router.push(href);
}
//……
一个简单的 Link 跳转过渡效果就大功告成了。
覆盖浏览器前进后退过渡效果浏览器前进后退和 Link 标签导航有所不同,后者可以通过封装组件精准修改跳转行为,但是前者在页面内没有指定的载体,我们只能通过构建一个 Context 来包裹页面或需要过渡动画的组件来实现页面或指定页面区域的过渡效果。
同时,熟悉 HMTL history API 的朋友都知道,浏览器历史堆栈的变化会触发 popstate 事件。那么我们就可以尝试通过改写 popstate 事件来实现视图过渡的效果。
所以,要实现浏览器前进后退的跳转过渡,需要两步骤:
第一步:创建 Context,包裹页面:
//components/TransitionProvider.tsx
"useclient";
importReact,{createContext,useContext}from"react";
constTransitionContext=createContext(null);
exportconstuseTransitionContext=()=useContext(TransitionContext);
exportconstTransitionProvider=({children})={
returnTransitionContext.Provider{children}/TransitionContext.Provider;
};
在 layout.tsx 中引入 provider
//app/layout.tsx
//……
body
TransitionProvider
//……
/TransitionProvider
/body
//……
第二步,创建一个独立的 hook,用于改写 window 的 popstate 事件,让它执行 document.startViewTransition 方法,并在 TransitionProvider.tsx 中调用
//components/TransitionProvider.tsx
"useclient";
importReact,{createContext,useContext}from"react";
importuseSimplifiedViewTransitionfrom'@/hooks/useSimplifiedViewTransition'
constTransitionContext=createContext(null);
exportconstuseTransitionContext=()=useContext(TransitionContext);
exportconstTransitionProvider=({children})={
useSimplifiedViewTransition()//新增
returnTransitionContext.Provider{children}/TransitionContext.Provider;
};
在 useSimplifiedViewTransition 里面,我们在 useEffect 里定义修改的 popstate 事件:
import{useEffect}from'react';
functionuseSimplifiedViewTransition(){
useEffect(()={
if(!document.startViewTransition){
console.log('浏览器不支持ViewTransitionsAPI');
return;
}
consthandlePopState=()={
console.log('视图切换');
document.startViewTransition(()={
//浏览器自动处理视图转换
console.log('startViewTransition');
//添加popstate事件监听器
window.addEventListener('popstate',handlePopState);
//清理函数
return()={
window.removeEventListener('popstate',handlePopState);
},
}
exportdefaultuseSimplifiedViewTransition;
在这段代码里,我们只在首次渲染的时候定义 popstate 事件,后面每一次点击浏览器的前进后退按钮,都会触发 handlePopState 方法。
不过,这样实现存在一个问题,虽然每一次点击前进后退 handlePopState 都会触发,但是过渡效果可能只在第一次点击的时候会出现。这是因为这段代码里,执行到 document.startViewTransition 的时候,DOM 已经完成更新了。
那么应该如何修改才能实现每一次点击浏览器前进后退都有过渡效果呢?下一节的最佳实践就来解决这个问题。
最佳实践源码分析根据上面实现的想法去搜索类似的库,我找到了next-view-transitions这个仓库,它更完整地实现了Link跳转页面和点击浏览器前进后退按钮跳转页面的过渡效果,通知我们就来学习一下这个仓库的源码。
仓库核心文件有4个,其中index.js只是导出必要的方法,可以忽略掉。
Link组件被封装在link.tsx文件夹中,下面代码非必填项,如果要看完整代码请到开源仓库查看:
import NextLink from 'next/link'
import { useRouter } from 'next/navigation'
import { startTransition, useCallback } from 'react'
import { useSetFinishViewTransition } from './transition-context'
export function Link(props: React.ComponentPropstypeof NextLink) {
const router = useRouter()
const finishViewTransition = useSetFinishViewTransition()
const { href, as, replace, scroll } = props
const onClick = useCallback(
(e: React.MouseEventHTMLAnchorElement) = {
// 如果提供了自定义的 onClick 处理函数,先执行它
if (props.onClick) {
props.onClick(e)
}
// 检查浏览器是否支持 startViewTransition API
if ('startViewTransition' in document) {
// 阻止默认的链接点击行为
e.preventDefault()
// @ts-ignore
document.startViewTransition(
() =
new Promisevoid((resolve) = {
// 使用 React 的 startTransition 来标记低优先级更新
startTransition(() = {
// 根据 replace 属性决定使用 router.replace 还是 router.push
router[replace ? 'replace' : 'push'](as || href, {
scroll: scroll ?? true,
})
// 设置完成视图转换的回调,该回调会在适当的时机解析 Promise
finishViewTransition(() = resolve)
})
})
)
}
},
// 依赖数组,当这些值变化时会重新创建 onClick 函数
[props.onClick, href, as, replace, scroll]
)
// 返回增强的 NextLink 组件,传递所有原始 props 和新的 onClick 处理函数
return NextLink {...props} onClick={onClick} /
}
这个Link组件与我们之前实现的TransitionLink组件一样,但增加了API的可用性判断和调用方法的判断。
第三份文件是transition-context.tsx,
import type { Dispatch, SetStateAction } from 'react'
import { createContext, use, useEffect, useState } from 'react'
import { useBrowserNativeTransitions } from './browser-native-events'
// 创建一个 Context 用于管理视图转换的完成函数
// 默认值是一个返回空函数的函数,确保在 Provider 外使用时不会出错
const ViewTransitionsContext = createContext
DispatchSetStateAction(() = void) | null
(() = () = {})
export function ViewTransitions({
children,
}: Readonly{
children: React.ReactNode
}) {
// 状态用于存储当前的视图转换完成函数
const [finishViewTransition, setFinishViewTransition] = useState
null | (() = void)
(null)
// 当 finishViewTransition 函数被设置时,执行它并重置状态
useEffect(() = {
if (finishViewTransition) {
finishViewTransition()
setFinishViewTransition(null)
}
}, [finishViewTransition])
// 使用自定义 hook 来处理浏览器原生的转换事件
useBrowserNativeTransitions()
// 提供 Context,允许子组件访问和设置 finishViewTransition
return (
ViewTransitionsContext.Provider value={setFinishViewTransition}
{children}
/ViewTransitionsContext.Provider
)
}
// 自定义 hook,用于获取设置 finishViewTransition 的函数
export function useSetFinishViewTransition() {
return use(ViewTransitionsContext)
}
这部分逻辑也不难理解,关键在于调用useBrowserNativeTransitions()这个hook,我们来看看这个hook是怎么实现浏览器前进后退过渡效果的。
//./browser-native-events.ts
import{useEffect,useRef,useState,use}from'react'
import{usePathname}from'next/navigation'
exportfunctionuseBrowserNativeTransitions(){
//获取当前路径名
constpathname=usePathname()
//使用ref存储当前路径名,以便在不触发重渲染的情况下更新
constcurrentPathname=useRef(pathname)
//创建一个状态来跟踪视图转换
//状态为null或包含两个元素的元组:
//1.Promise:等待视图转换开始
//2.函数:用于完成视图转换
const[currentViewTransition,setCurrentViewTransition]=useState
null|[Promisevoid,()=void]
(null)
useEffect(()={
//检查浏览器是否支持startViewTransitionAPI
if(!('startViewTransition'indocument)){
return()={}
}
//定义popstate事件处理函数
constonPopState=()={
letpendingViewTransitionResolve:()=void
//创建一个Promise,用于控制视图转换的结束
constpendingViewTransition=newPromisevoid((resolve)={
pendingViewTransitionResolve=resolve
})
//创建一个Promise,用于等待视图转换开始
constpendingStartViewTransition=newPromisevoid((resolve)={
//@ts-ignore
document.startViewTransition(()={
resolve()//解析Promise,表示转换已开始
returnpendingViewTransition//返回控制转换结束的Promise
})
})
//更新状态,存储转换相关的Promise和解析函数
setCurrentViewTransition([
pendingStartViewTransition,
pendingViewTransitionResolve!,
])
}
//添加popstate事件监听器
window.addEventListener('popstate',onPopState)
//清理函数:移除事件监听器
return()={
window.removeEventListener('popstate',onPopState)
}
},[])
//如果存在当前视图转换且路径发生变化
if(currentViewTransitioncurrentPathname.current!==pathname){
//使用React的use函数阻塞渲染,直到视图转换开始
//这确保在DOM被截图之前,新的内容不会被渲染
use(currentViewTransition[0])
}
//使用ref保持转换引用的最新状态
consttransitionRef=useRef(currentViewTransition)
useEffect(()={
transitionRef.current=currentViewTransition
},[currentViewTransition])
//确保在新路由组件挂载后完成视图转换
useEffect(()={
//当新路由组件实际挂载时
currentPathname.current=pathname//更新当前路径
//如果存在转换,完成它
if(transitionRef.current){
transitionRef.current[1]()//调用解析函数完成转换
transitionRef.current=null//重置转换状态
}
},[pathname])
}
此处代码要点:
if(currentViewTransitioncurrentPathname.current!==pathname){
use(currentViewTransition[0])
}
使用 React 的函数抛出一个 promise 来“挂”组件的渲染,直到视图转换开始,这样确保新的布局内容不会在渲染之前,从而实现平滑转换。
结语使用 HTML View Transition API 和 React startTransition 结合的方案实现页面切换过渡效果,既不需要引入额外的JS包,还因为可以使用原始API更节约性能,这大概就是目前性能最优的过渡效果方案了。
如果想知道代码最终实现的过渡效果如何,可以分别到weijunext.com和gapis.money对比一下,用肉眼感受一下添加过渡效果和没添加过渡效果的区别。
关于我我是一位前端工程师,Next.js 手艺人,AI 领袖。
今年致力于 Next.js 和 Node.js 领域的开源项目开发和知识分享。
欢迎关注我的掘金和Github
点击关注公众号,“技术干货”及时达!
阅读原文
网站开发网络凭借多年的网站建设经验,坚持以“帮助中小企业实现网络营销化”为宗旨,累计为4000多家客户提供品质建站服务,得到了客户的一致好评。如果您有网站建设、网站改版、域名注册、主机空间、手机网站建设、网站备案等方面的需求...
请立即点击咨询我们或拨打咨询热线:13245491521 13245491521 ,我们会详细为你一一解答你心中的疑难。 项目经理在线