新聞中心
大家好,我卡頌。

要說React有什么其他框架沒有的、獨一無二的特性,那一定是「并發(fā)更新」。圍繞并發(fā)更新,存在兩個很有意思的現(xiàn)象:
- 很多開發(fā)者聽說過他
- 很少開發(fā)者直接使用過他
這兩個現(xiàn)象看似矛盾,其實很好解釋 —— React18之后的新特性,主要是面向上層框架的(主要是Next.js)。
換句話說,這些新特性(比如并發(fā)更新)主要是供框架集成,而不是開發(fā)者直接使用。
比如,并發(fā)更新的兩個核心API —— useTransition和useDeferredValue,都是針對「視圖切換」的場景。
而在前端交互中,最主要的「視圖切換」場景就是「路由切換」,所以包含路由功能的前端框架就會集成這兩個API。
而現(xiàn)在,一個試驗性瀏覽器API —— View Transitions API將原生實現(xiàn)「視圖切換」功能。
他到底有什么用?如果其他框架使用它,是不是能獲得React同樣的并發(fā)更新能力?
什么是視圖切換?
不管是View Transitions API,還是React的useTransition,都是為了提高「視圖切換」場景下的用戶體驗。
什么是「視圖切換」?以view-transitions Demo[1]舉例。這是個簡單的相冊Demo,點擊左邊圖片縮略圖,右邊會顯示大圖:
整個過程簡單來說包括3個步驟:
- 點擊縮略圖。
- 請求大圖數(shù)據(jù)。
- 大圖請求成功后,顯示大圖。
從步驟1到3的過程就是個典型的「視圖切換」。整個過程有很多可以優(yōu)化體驗的地方,比如:
- 從舊圖到新圖的漸變過渡效果。
- 點擊縮略圖發(fā)起圖片請求后,大圖區(qū)域可以先顯示舊圖(而不是立刻顯示loading效果),待新圖請求成功后再過渡到新圖。
這里解釋下第二點,對于切換類的交互,相比于「當視圖切換時立刻顯示loading效果,待新視圖加載完成后過渡到新視圖」,「當視圖切換時先顯示舊視圖,待新視圖加載完成后過渡到視圖」在延遲不高的情況下體驗會更好。
除了上述這些「體驗優(yōu)化的點」,視圖切換的實現(xiàn)還有很多細節(jié)需要考慮,比如:
- 如何處理新舊視圖切換時的過渡效果?
- 如何處理新視圖加載時的loading效果?
- 當正在請求新視圖數(shù)據(jù)時(此時視圖處在舊視圖中),用戶又對舊視圖產(chǎn)生交互怎么辦?
- 視圖切換時如何處理頁面滾動位置、光標聚焦(focus)位置?
- 對于使用屏幕閱讀器的盲人,視圖切換時閱讀器會朗讀什么?
除此之外,不同場景下的「視圖切換」實現(xiàn)細節(jié)也不同。比如,如何在切換頁面時優(yōu)化視圖切換效果?
在SPA(單頁應(yīng)用)出現(xiàn)之前,網(wǎng)站通常是由多個頁面組成。比如下面網(wǎng)站的每個Tab欄,對應(yīng)一個獨立網(wǎng)頁,其中:
- bramus Tab對應(yīng) https://http203-playlist.netlify.app/with-bramus/
- cassie Tab對應(yīng)https://http203-playlist.netlify.app/with-cassie/
在Tab之間切換,瀏覽器會:
- 卸載之前的頁面
- 請求新頁面數(shù)據(jù)
- 加載新頁面
從「頁面卸載」到「頁面加載」之間的白屏間隙會造成屏幕閃爍。
要優(yōu)化這種場景下優(yōu)化視圖切換效果,當前唯一做法是利用history API接管路由操作,將網(wǎng)頁變成一個SPA。
既然「視圖切換」是如此常見的需求,且有這么多需要考慮的因素,那瀏覽器為什么不原生實現(xiàn)呢?
于是,View Transitions API應(yīng)運而生。
當前View Transitions API不支持跨頁面的視圖切換,但未來會支持。
View Transitions的使用
View Transitions API[2]的使用很簡單,只需要用document.startViewTransition包裹視圖切換后的回調(diào)函數(shù)即可。
對于上述相冊示例,回調(diào)函數(shù)的邏輯是「將img標簽src屬性更新為新圖片地址」:
const transition = document.startViewTransition(() => {
galleryImg.src = /* 新圖片地址 */;
});剩下所有跟過渡相關(guān)的實現(xiàn),都由Transitions API解決。
View Transitions實現(xiàn)原理
在視圖切換時,存在2個概念:
- 切換前的舊視圖
- 切換后的新視圖
當使用View Transitions后,會依次做:
- 對頁面進行截圖,作為舊視圖。
- 執(zhí)行傳遞給document.startViewTransition的回調(diào)。
- DOM更新后,對更新后的頁面進行截圖,作為新視圖。
構(gòu)造一棵代表過渡效果的偽元素樹,掛載在根元素(html元素)下,結(jié)構(gòu)類似如下:
::view-transition
└─ ::view-transition-group(root)
└─ ::view-transition-image-pair(root)
├─ ::view-transition-old(root)
└─ ::view-transition-new(root)其中:
- 舊視圖保存在::view-transition-old(root)中。
- 新視圖保存在::view-transition-new(root)中。
對于上述相冊示例,掛載的偽元素樹結(jié)構(gòu)如下:
之所以要掛載一棵偽元素樹,主要是因為兩個原因:
- 開發(fā)者可以對微元素應(yīng)用CSS規(guī)則。
比如,上述兩個「保存了新/舊視圖的截圖」的偽元素,類似于img標簽,開發(fā)者可以對他們應(yīng)用CSS動畫,當新/舊視圖切換時,實現(xiàn)自定義的過渡效果。
- 方便對整個頁面中不同「視圖切換」分組。
比如,在上述相冊示例中,視圖切換的元素包括兩部分:
- 新/舊視圖之間的切換(下圖紅框部分)
- 新/舊圖片名稱的切換(下圖綠框部分)
相冊對應(yīng)的HTML結(jié)構(gòu)如下:
- img標簽對應(yīng)視圖部分(下圖紅框部分)。
- figcaption標簽對應(yīng)圖片名稱部分(下圖綠框部分)。
當我們?yōu)閒igcaption元素設(shè)置不同的view-transition-name:
figcaption {
view-transition-name: figure-caption;
}會得到一棵新的偽元素樹,其中「視圖部分」和「圖片名稱部分」偽元素是分離開的:
通過給頁面中不同HTML元素定義不同的view-transition-name屬性,就能獨立控制這個元素是視圖切換時的過渡效果。
與 React 的區(qū)別
瀏覽器原生的View Transitions API與React中的useTransition相比,誰更強大呢?
毫無疑問,前者更強大。
這是因為,對于View Transitions API,通過操作偽元素樹,開發(fā)者可以自定義過渡效果(就像對img元素使用CSS過渡動畫一樣簡單)。即使不使用CSS Transition,使用JS Transition也完全沒問題。
document.startViewTransition方法會返回一個transition對象實例:
const transition = document.startViewTransition(() => {
galleryImg.src = /* 新圖片地址 */;
});該實例包含了一系列視圖切換生命周期對應(yīng)的promise,比如:
- ViewTransition.ready:偽元素樹構(gòu)造完成,準備開始過渡時。
- ViewTransition.finished:過渡效果完成后,此時新視圖已經(jīng)可以響應(yīng)用戶交互。
而在React中,使用useTransition后,與其說完成的是「視圖切換」,不如說完成的是:
- 首先,完成狀態(tài)的切換。
- React內(nèi)部再將狀態(tài)變化映射到視圖變化。
本質(zhì)來說,操作視圖的是React,而不是開發(fā)者。所以,開發(fā)者很難控制過渡效果。
動效庫Framer的作者認為,由于開發(fā)者很難控制并發(fā)更新的完整生命周期,所以很難在并發(fā)更新時表達animation效果:
簡單來說就是,開發(fā)者很難為并發(fā)更新定制過渡效果(用CSS或JS)。
總結(jié)
可以認為,View Transitions API是比useTransition抽象程度更高、開發(fā)者可控性更高的API。useTransition能實現(xiàn)的,他可以。useTransition不能實現(xiàn)的,他也可以。
要說缺點,View Transitions API是Web平臺獨有的,而useTransition依賴React核心的并發(fā)更新能力,是跨端的。
當前,View Transitions API的兼容性并不好:
但是,一旦他變成可以大規(guī)模使用的API,那么其他前端框架只要接入他,就能輕松獲得比React耗費大量精力實現(xiàn)的useTransition(以及底層的并發(fā)更新特性)更強大的視圖切換能力。
參考資料
[1]view-transitions Demo:https://mdn.github.io/dom-examples/view-transitions/#。
[2]View Transitions API:https://developer.mozilla.org/en-US/docs/Web/API/View_Transitions_API。
本文標題:瀏覽器要原生實現(xiàn)React的并發(fā)更新了?
網(wǎng)頁鏈接:http://fisionsoft.com.cn/article/cddicdo.html


咨詢
建站咨詢
