跳至主要内容

View Transitions

在 ChatGPT 中打开
Java API
25.11 实验性

视图过渡在DOM变化时提供动画过渡,减少视觉上的突兀,并在导航或内容更新过程中保持空间上下文。webforJ与浏览器的视图过渡API集成,处理新旧状态之间动画协调的复杂性。

显示代码

实验性功能
此功能为实验性,未来版本可能会改变或删除。

基本用法

要创建视图过渡,请使用 Page.getCurrent().startViewTransition(),该方法返回一个用于配置过渡的构建器:

Page.getCurrent().startViewTransition()
.onUpdate(done -> {
container.remove(oldView);
container.add(newView);
done.run();
})
.start();

过渡过程捕获当前状态的快照,在 onUpdate 回调中应用您的DOM更改,然后从旧快照动画到新内容。您必须调用 done.run() 来信号您的更改已完成。

onUpdate 回调是必需的

如果在未设置更新回调的情况下调用 start(),将抛出 IllegalStateException

应用过渡

webforJ提供预定义的过渡类型,您可以将其应用于进入或退出DOM的组件:

常量效果
ViewTransition.NONE无动画
ViewTransition.FADE旧内容和新内容之间的交叉淡入淡出
ViewTransition.SLIDE_LEFT内容向左流动(类似向前导航)
ViewTransition.SLIDE_RIGHT内容向右流动(类似返回导航)
ViewTransition.SLIDE_UP内容向上流动
ViewTransition.SLIDE_DOWN内容向下流动
ViewTransition.ZOOM旧内容缩小,新内容放大
ViewTransition.ZOOM_OUT旧内容放大,新内容缩小

使用 enter() 来为添加的组件动画,使用 exit() 为被移除的组件动画:

// 为进入DOM的组件动画
Page.getCurrent().startViewTransition()
.enter(chatPanel, ViewTransition.ZOOM)
.onUpdate(done -> {
container.add(chatPanel);
done.run();
})
.start();

// 为退出DOM的组件动画
Page.getCurrent().startViewTransition()
.exit(chatPanel, ViewTransition.FADE)
.onUpdate(done -> {
container.remove(chatPanel);
done.run();
})
.start();

共享组件过渡

共享组件过渡创建一种变形效果,其中组件似乎从旧视图中的位置转变为新视图中的位置。这是通过使用 setViewTransitionName() 方法为组件提供相同的过渡名称来实现的,该方法在任何实现了 HasStyle 接口的组件上可用。

// 在卡片视图中
image.setViewTransitionName("blog-image");

// 在详情视图中 - 相同名称创建变形
image.setViewTransitionName("blog-image");

在这些视图之间过渡时,浏览器在位置之间动画化组件,创建一个连接的视觉体验。

使用唯一名称

在处理列表或重复组件时,在过渡名称中包含唯一标识符。每个组件都需要自己独特的名称,以便正确变形到新视图中的相应组件。对多个可见组件使用相同的名称会导致未定义行为。

显示代码

列表排序

共享组件过渡的一个常见用例是当其顺序更改时为列表项动画。通过为每个项目分配唯一的 view-transition-name,浏览器会自动将组件动画到其新位置:

// 每个卡片根据其ID获取一个唯一的过渡名称
card.setViewTransitionName("card-" + item.id());

// 当洗牌时,只需更新DOM - 浏览器处理动画
Page.getCurrent().startViewTransition()
.onUpdate(done -> {
renderList();
done.run();
})
.start();
显示代码

自定义CSS动画

为了对动画进行完全控制,您可以定义自定义CSS关键帧。webforJ在您的过渡名称后附加 -enter-exit 后缀,您可以使用这些后缀来定位视图过渡伪元素:

/* 定义进入组件的关键帧 */
@keyframes flip-enter {
from {
opacity: 0;
transform: perspective(1000px) rotateX(-90deg);
}
to {
opacity: 1;
transform: perspective(1000px) rotateX(0deg);
}
}

/* 应用于视图过渡伪元素 */
::view-transition-new(flip-in-enter) {
animation: flip-enter 450ms cubic-bezier(0.34, 1.56, 0.64, 1);
transform-origin: top center;
}

::view-transition-old(flip-in-enter) {
display: none;
}

通过将其名称(不带后缀)传递给 enter()exit() 来引用您的自定义动画:

// 使用 "flip-in" - webforJ 自动添加 "-enter" 后缀
Page.getCurrent().startViewTransition()
.enter(notification, "flip-in")
.onUpdate(done -> {
stage.add(notification);
done.run();
})
.start();

// 使用 "blur-out" 进行退出 - webforJ 自动添加 "-exit" 后缀
Page.getCurrent().startViewTransition()
.exit(notification, "blur-out")
.onUpdate(done -> {
stage.remove(notification);
done.run();
})
.start();
显示代码

CSS自定义

每种预定义的过渡类型都公开CSS自定义属性以进行微调:

变量默认值描述
--vt-fade-duration200ms动画持续时间
--vt-fade-easingcubic-bezier(0.4, 0, 0.2, 1)缓动函数
变量默认值描述
--vt-slide-left-duration200ms动画持续时间
--vt-slide-left-easingcubic-bezier(0.4, 0, 0.2, 1)缓动函数
--vt-slide-left-distance30%滑动距离
变量默认值描述
--vt-slide-right-duration200ms动画持续时间
--vt-slide-right-easingcubic-bezier(0.4, 0, 0.2, 1)缓动函数
--vt-slide-right-distance30%滑动距离
变量默认值描述
--vt-slide-up-duration200ms动画持续时间
--vt-slide-up-easingcubic-bezier(0.4, 0, 0.2, 1)缓动函数
--vt-slide-up-distance30%滑动距离
变量默认值描述
--vt-slide-down-duration200ms动画持续时间
--vt-slide-down-easingcubic-bezier(0.4, 0, 0.2, 1)缓动函数
--vt-slide-down-distance30%滑动距离
变量默认值描述
--vt-zoom-duration200ms动画持续时间
--vt-zoom-easingcubic-bezier(0.4, 0, 0.2, 1)缓动函数
--vt-zoom-scale0.8缩放因子(旧内容缩小到此,新内容从此放大)
变量默认值描述
--vt-zoom-out-duration200ms动画持续时间
--vt-zoom-out-easingcubic-bezier(0.4, 0, 0.2, 1)缓动函数
--vt-zoom-out-scale1.2缩放因子(旧内容放大到此,新内容从此缩小)

要自定义,请在您的CSS中重写这些变量:

:root {
--vt-fade-duration: 300ms;
--vt-slide-left-distance: 50%;
}

对于高级自定义,直接定位视图过渡伪元素:

::view-transition-old(vt-slide-left-exit) {
animation-duration: 400ms;
}

::view-transition-new(vt-slide-left-enter) {
animation-timing-function: ease-out;
}