还在用 Transition 和 Animation?View Transition出炉了!
zhezhongyun 2025-08-05 22:21 31 浏览
大家好,很高兴又见面了,我是"高级前端进阶",由我带着大家一起关注前端前沿、深入前端底层技术,大家一起进步,也欢迎大家关注、点赞、收藏、转发,您的支持是我不断创作的动力。
今天给大家带来的主题是新API介绍,即 View Transitions API。话不多说,直接进入正题!
1.什么是 View Transitions
View Transition 是一种流行的设计选择,可减少用户的认知负荷,帮助用户留在当前上下文中,并减少在应用程序的状态或视图之间移动时感知到的加载延迟。
然而,在 Web 上创建视图转换历来都很困难。 单页应用程序 (SPA) 中的状态间转换往往涉及编写大量的 CSS 和 JavaScript ,因为本身的流程就非常复杂:
- 处理新旧内容的加载和定位
- 对旧状态和新状态进行动画处理以创建过渡
- 防止用户与旧内容的意外交互导致问题
- 转换完成后删除旧内容
由于新旧内容同时出现在 DOM 中,还存在可访问性问题,例如:阅读位置丢失、焦点混乱和奇怪的实时区域公告行为。 并且跨文档视图转换(即跨常规网站不同页面)是不可能的。
不可否认,CSS Transitions 和 Animations 在过去十年中彻底改变了网页动效,但并非一切都很容易。 而 View Transitions API 使用以下过程简化了这个流程:
- API 获取当前页面状态的快照。
- 根据需要更新 DOM ,例如:添加或删除元素。
- API 获取新页面状态的快照。
- API 使用默认淡入淡出或开发者自定义的任何 CSS 动画在两种状态之间进行动画处理。
开发者只需要像以前一样更新 DOM, 当 View Transitions API 可用于创建类似演示的效果时,通过几行附加代码可以逐步增强页面。
但是值得注意的是,当前该 API 是实验性的,但最近基于 Chromium 的浏览器支持页内(in-Page)、基于单文档(single-document) DOM 的效果。
Chrome 115+ 中还提供了用于导航的 ViewTransition API,并在各个页面加载之间提供动画,容易使用并且不需要任何 JavaScript。
Mozilla 和 Apple 尚未透露在 Firefox 和 Safari 中实现该 API 的意图。 任何没有 View Transitions API 的浏览器都将继续工作,因此该功能属于渐进增强。
2.新旧技术
老牌的开发者可能会对这种技术比较熟悉。 Microsoft 在 Internet Explorer 4.0(1997 年发布)中添加了元素和整个页面转换,并在 IE5.5(2000 年发布)中进行了进一步更新。 开发者可以使用标签添加类似 PowerPoint 的框、圆圈、擦拭、溶解、百叶窗、幻灯片、条带和螺旋:
<meta http-equiv="Page-Enter" content="progid:DXImageTransform.Microsoft.Iris(Motion='in', IrisStyle='circle')">
<meta http-equiv="Page-Exit" content="progid:DXImageTransform.Microsoft.Iris(Motion='out', IrisStyle='circle')">奇怪的是,这项技术从未被广泛采用,非 Web 标准,可能当时 W3C 正处于起步阶段,开发人员很乐意使用大量其他 IE 特定技术!
3.创建页内过渡
接下来一起来看看如何使用 View Transitions API 技术来创建页面内的过渡效果。
假设 HTML 页面有两个 <article> 元素, ID 分别为 article1 和 article2:
<main>
<div id="articleroot">
<article id="article1">
<h2>Article 1 content</h2>
<figure>
<img src="image1.jpg" width="800" height="500" alt="image" />
</figure>
<p>Lorem ipsum dolor sit amet...</p>
</article>
<article id="article2">
<h2>Article 2 content</h2>
<figure>
<img src="image2.jpg" width="800" height="500" alt="image" />
</figure>
<p>Ut pretium ac orci nec dictum...</p>
</article>
</div>
</main>switchArticle() 函数处理所有 DOM 更新。 它通过添加或删除 hidden 属性来显示或隐藏 Article 内容。 页面加载时,Article 由页面 URL 的 location.hash 确定,如果未设置,则根据第一个 <article> 元素确定:
// 获取所有article元素
const article = document.getElementsByTagName("article");
// 页面加载完成显示一个aticle
switchArticle();
// 显示当前article
function switchArticle(e) {
const hash = e?.target?.hash?.slice(1) || location?.hash?.slice(1);
Array.from(article).forEach((a, i) => {
if (a.id === hash || (!hash && !i)) {
a.removeAttribute("hidden");
} else {
a.setAttribute("hidden", "");
}
});
}事件处理函数监听所有页面点击,并在用户点击带有 #hash 的链接时调用 switchArticle():
// 监听click点击事件
document.body.addEventListener("click", (e) => {
if (!e?.target?.hash) return;
switchArticle(e);
});可以通过将 switchArticle() 函数作为回调传递给
document.startViewTransition() 来更新此处理程序以使用视图转换(但是需要首先检查 API 是否可用):
View Transitions API 的 startViewTransition() 方法:启动一个新的视图转换并返回一个 ViewTransition 对象来表示它。调用 startViewTransition() 时,将执行一系列步骤,如:视图转换过程中所述。
document.body.addEventListener("click", (e) => {
if (!e?.target?.hash) return;
if (document.startViewTransition) {
// 使用 View Transition effect
document.startViewTransition(() => switchArticle(e));
} else {
// View Transition 不可用
switchArticle(e);
}
});
document.startViewTransition() 获取初始状态的快照,运行 switchArticle(),获取新状态的新快照,并在两者之间创建默认的半秒淡入淡出。CSS 中提供以下选择器来定位旧状态和新状态:
::view-transition-old(root) {
/* 淡出效果 */
}
::view-transition-new(root) {
/* 淡入效果 */
}
- ::view-transition:视图转换覆盖的根,包含所有视图转换并位于所有其他页面内容顶部
- ::view-transition-old:旧页面视图的屏幕截图
- ::view-transition-new:新页面视图的实时表示,它们都以与 <img> 或 <video> 相同的方式渲染为替换内容,这意味着可以使用方便的属性(如对象适合和对象位置)进行样式设置。
比如,上面的示例将动画持续时间增加到一秒,以便淡入淡出效果更加明显:
::view-transition-old(root),
::view-transition-new(root) {
animation-duration: 1s;
}视图转换组(根)可以同时将效果应用于 old 状态和 new 状态,尽管在大多数情况下开发者不太可能应用相同的动画。
4.异步 DOM 更新
传递给
document.startViewTransition() 的回调可以返回一个 Promise,因此异步更新是可能的。例如:
document.startViewTransition(async () => {
const response = await fetch("/some-data");
const json = await response.json();
doDOMUpdates(json);
// 更新dom
await sendAnalyticsEvent();
});上面代码示例会冻结页面,直到 Promise resolve,因此延迟可能会影响用户体验。在 .startViewTransition() 调用之外运行尽可能多的代码会更有效。比如下面的示例:
const response = await fetch("/some-data");
const json = await response.json();
document.startViewTransition(() => doDOMUpdates(json));
await sendAnalyticsEvent();5.使用 Web Animations API
虽然 CSS 足以满足大多数效果,但 Web Animations API 允许在 JavaScript 中进行进一步的计时和效果控制。
document.startViewTransition() 返回一个运行 .ready Promise 的对象,当旧和新的伪元素可用时,该 Promise 会 resolve(注意第二个 .animate() 参数中的 pseudoElement 属性)。在以下示例中,ready 用于触发自定义循环显示视图转换,该转换从用户点击时光标的位置发出,动画由 Web 动画 API 提供。
// Store the last click event
let lastClick;
addEventListener("click", (event) => (lastClick = event));
function spaNavigate(data) {
// 不支持此 API 的浏览器
if (!document.startViewTransition) {
updateTheDOMSomehow(data);
return;
}
// 获取点击位置,或者回退到屏幕中间
const x = lastClick?.clientX ?? innerWidth / 2;
const y = lastClick?.clientY ?? innerHeight / 2;
// 获取到最远角的距离
const endRadius = Math.hypot(
Math.max(x, innerWidth - x),
Math.max(y, innerHeight - y),
);
// 创建 transition:
const transition = document.startViewTransition(() => {
updateTheDOMSomehow(data);
});
// 等待创建伪元素
transition.ready.then(() => {
// 为根的新视图设置动画
document.documentElement.animate(
{
clipPath: [
`circle(0 at ${x}px ${y}px)`,
`circle(${endRadius}px at ${x}px ${y}px)`,
],
},
{
duration: 500,
easing: "ease-in",
// Specify which pseudo-element to animate
pseudoElement: "::view-transition-new(root)",
},
);
});
}如上面代码所示,ViewTransition 接口的 read-only 属性是一个 Promise,一旦伪元素树创建完毕并且过渡动画即将开始,该 Promise 就会 resolve。
如果转换无法开始,ready 将 reject ,当然此时可能是由于配置错误造成的,例如:重复的视图转换名称,或者传递给
Document.startViewTransition() 的回调抛出或返回 reject 的Promise 。
6.创建多页面导航过渡
当用户在多页面应用程序 (MPA)上的页面加载之间导航时,开发者还可以使用视图转换,即被称为用于导航的 ViewTransition API,但是开发者必须在 Chrome 115 中的 chrome://flags/ 中启用它(目前是针对开发人员的 Canary nightly 版本)。 该标志在浏览器的早期版本中也可用,但 API 可能丢失或不稳定。
该过程比页内转换更容易,因为它是通过 HTML <head> 中的单个 meta 标记启用的:
<meta name="view-transition" content="same-origin" />开发者可以按照与上面所示相同的方式定义 ::view-transition-old 和 ::view-transition-new CSS 选择器。 除非开发者想使用 Web Animations API,否则依然不需任何 JavaScript
::view-transition-image-pair(root) {
isolation: auto;
}
::view-transition-old(root),
::view-transition-new(root) {
animation: none;
mix-blend-mode: normal;
display: block;
}Chrome 115 最终版发布时,导航 API 可能默认启用,也可能不启用。 但是开发者今天已经可以使用该技术,因为不支持该 API 的浏览器将回退到标准的非动画页面加载。
7.禁用动画
动画可能会让一些用户感到不适, 大多数操作系统提供用户首选项设置来禁用。 开发者也可以使用 CSS Preferred-reduced-motion 媒体查询来检测这一点并相应地关闭动画。
Preferred-reduced-motion CSS 媒体功能用于检测用户是否在其设备上启用了一项设置,以最大限度地减少不必要的运动量。 该设置用于向设备上的浏览器传达用户更喜欢删除、减少或替换基于运动的动画的界面。
比如下面的是:
@media (prefers-reduced-motion) {
::view-transition-group(*),
::view-transition-old(*),
::view-transition-new(*) {
animation: none !important;
}
}8.总结
View Transitions API 简化了在页面内和页面间更改元素状态时的动画。 这种类型的转换以前需要通过大量的 JavaScript 实现,而且还可能破坏浏览器导航,例如:后退按钮。
现在借助于 View Transitions API 一切变得非常容易。但是 View Transitions API 还比较新, 无法保证它会保持不变、成为 W3C 标准或在 Firefox 和 Safari 中得到实现。 然而,今天开发者已经可以使用该 API,因为它是一个渐进增强的特性,在低版本的浏览器中能保证最基础的用户体验。
因为篇幅有限,关于 View Transitions API 的用法和特性文章并没有过多展开,如果有兴趣,可以在我的主页继续阅读,同时文末的参考资料提供了大量优秀文档以供学习。最后,欢迎大家点赞、评论、转发、收藏,您的支持是我不断创作的动力。
参考资料
https://www.sitepoint.com/view-transitions-api-introduction/
https://developer.mozilla.org/en-US/docs/Web/API/View_Transitions_API
https://developer.mozilla.org/en-US/docs/Web/API/View_Transitions_API
https://developer.mozilla.org/en-US/docs/Web/API/ViewTransition/ready
https://developer.mozilla.org/en-US/docs/Web/CSS/@media/prefers-reduced-motion
https://12daysofweb.dev/2022/view-transitions-api/
https://developer.chrome.com/blog/spa-view-transitions-land/
https://ali-akhtar.medium.com/uikit-animation-part-1-8a74221d5dc0
相关推荐
- Python入门学习记录之一:变量_python怎么用变量
-
写这个,主要是对自己学习python知识的一个总结,也是加深自己的印象。变量(英文:variable),也叫标识符。在python中,变量的命名规则有以下三点:>变量名只能包含字母、数字和下划线...
- python变量命名规则——来自小白的总结
-
python是一个动态编译类编程语言,所以程序在运行前不需要如C语言的先行编译动作,因此也只有在程序运行过程中才能发现程序的问题。基于此,python的变量就有一定的命名规范。python作为当前热门...
- Python入门学习教程:第 2 章 变量与数据类型
-
2.1什么是变量?在编程中,变量就像一个存放数据的容器,它可以存储各种信息,并且这些信息可以被读取和修改。想象一下,变量就如同我们生活中的盒子,你可以把东西放进去,也可以随时拿出来看看,甚至可以换成...
- 绘制学术论文中的“三线表”具体指导
-
在科研过程中,大家用到最多的可能就是“三线表”。“三线表”,一般主要由三条横线构成,当然在变量名栏里也可以拆分单元格,出现更多的线。更重要的是,“三线表”也是一种数据记录规范,以“三线表”形式记录的数...
- Python基础语法知识--变量和数据类型
-
学习Python中的变量和数据类型至关重要,因为它们构成了Python编程的基石。以下是帮助您了解Python中的变量和数据类型的分步指南:1.变量:变量在Python中用于存储数据值。它们充...
- 一文搞懂 Python 中的所有标点符号
-
反引号`无任何作用。传说Python3中它被移除是因为和单引号字符'太相似。波浪号~(按位取反符号)~被称为取反或补码运算符。它放在我们想要取反的对象前面。如果放在一个整数n...
- Python变量类型和运算符_python中变量的含义
-
别再被小名词坑哭了:Python新手常犯的那些隐蔽错误,我用同事的真实bug拆给你看我记得有一次和同事张姐一起追查一个看似随机崩溃的脚本,最后发现罪魁祸首竟然是她把变量命名成了list。说实话...
- 从零开始:深入剖析 Spring Boot3 中配置文件的加载顺序
-
在当今的互联网软件开发领域,SpringBoot无疑是最为热门和广泛应用的框架之一。它以其强大的功能、便捷的开发体验,极大地提升了开发效率,成为众多开发者构建Web应用程序的首选。而在Spr...
- Python中下划线 ‘_’ 的用法,你知道几种
-
Python中下划线()是一个有特殊含义和用途的符号,它可以用来表示以下几种情况:1在解释器中,下划线(_)表示上一个表达式的值,可以用来进行快速计算或测试。例如:>>>2+...
- 解锁Shell编程:变量_shell $变量
-
引言:开启Shell编程大门Shell作为用户与Linux内核之间的桥梁,为我们提供了强大的命令行交互方式。它不仅能执行简单的文件操作、进程管理,还能通过编写脚本实现复杂的自动化任务。无论是...
- 一文学会Python的变量命名规则!_python的变量命名有哪些要求
-
目录1.变量的命名原则3.内置函数尽量不要做变量4.删除变量和垃圾回收机制5.结语1.变量的命名原则①由英文字母、_(下划线)、或中文开头②变量名称只能由英文字母、数字、下画线或中文字所组成。③英文字...
- 更可靠的Rust-语法篇-区分语句/表达式,略览if/loop/while/for
-
src/main.rs://函数定义fnadd(a:i32,b:i32)->i32{a+b//末尾表达式}fnmain(){leta:i3...
- C++第五课:变量的命名规则_c++中变量的命名规则
-
变量的命名不是想怎么起就怎么起的,而是有一套固定的规则的。具体规则:1.名字要合法:变量名必须是由字母、数字或下划线组成。例如:a,a1,a_1。2.开头不能是数字。例如:可以a1,但不能起1a。3....
- Rust编程-核心篇-不安全编程_rust安全性
-
Unsafe的必要性Rust的所有权系统和类型系统为我们提供了强大的安全保障,但在某些情况下,我们需要突破这些限制来:与C代码交互实现底层系统编程优化性能关键代码实现某些编译器无法验证的安全操作Rus...
- 探秘 Python 内存管理:背后的神奇机制
-
在编程的世界里,内存管理就如同幕后的精密操控者,确保程序的高效运行。Python作为一种广泛使用的编程语言,其内存管理机制既巧妙又复杂,为开发者们提供了便利的同时,也展现了强大的底层控制能力。一、P...
- 一周热门
- 最近发表
- 标签列表
-
- HTML 教程 (33)
- HTML 简介 (35)
- HTML 实例/测验 (32)
- HTML 测验 (32)
- JavaScript 和 HTML DOM 参考手册 (32)
- HTML 拓展阅读 (30)
- HTML文本框样式 (31)
- HTML滚动条样式 (34)
- HTML5 浏览器支持 (33)
- HTML5 新元素 (33)
- HTML5 WebSocket (30)
- HTML5 代码规范 (32)
- HTML5 标签 (717)
- HTML5 标签 (已废弃) (75)
- HTML5电子书 (32)
- HTML5开发工具 (34)
- HTML5小游戏源码 (34)
- HTML5模板下载 (30)
- HTTP 状态消息 (33)
- HTTP 方法:GET 对比 POST (33)
- 键盘快捷键 (35)
- 标签 (226)
- opacity 属性 (32)
- transition 属性 (33)
- 1-1. 变量声明 (31)
