CocosCreator 之 Tween缓动系统的使用

版本: 3.4.0

语言: TypeScript

环境: Mac


简介


在CocosCreator 3.x版本后, Tween缓动系统代替了原有的Action动作。官方使用缓动系统的主要目的之一是用于解决离线动画无法满足需求时的动态动画问题。

简单的示例:

let duration = 2;
tween(this.tipNode)
  // 延迟两秒
  .delay(duration)
  // 2秒钟移动到(0, 300, 0)的位置,并缩放由1为0
  .to(duration, {position: new Vec3(0, 300, 0)})
  // 执行回调
  .call(() => {
    console.log("运行结束");
  })
  // 开始运行当前缓动对象
  .start();

使用缓动,主要通过tween函数来构建Tween实例化对象。它非Tween成员:

// tween 是一个工具函数,帮助实例化Tween实例,target用于设置缓动的目标
export function tween<T>(target?: T): Tween<T>;

在缓动中,主要通过byto修改某些属性, 其中:

  • by 使用 相对值 修改属性
  • to 使用 绝对值 修改属性

Tween提供的大部分接口返回的对象是this 或者一个新的Tween对象, 因此可以使用 链式调用

上面的示例等同于:

// tweenObj 对象就是新生成的Tween实例化对象
let duration = 2;
let tweenObj_1 = tween(this.tipNode);
let tweenObj_2 = tweenObj_1.delay(duration);
let tweenObj_3 = tweenObj_2.to(duration, {position: new Vec3(0, 300, 0)});
let tweenObj_4 = tweenObj_3.call(() => {
  console.log("运行结束");
});
let tweenObj_5 = tweenObj_4.start();
console.log(typeof(tweenObj_5), tweenObj_5);

本篇文章主要讲述下关于缓动系统的使用,理解可能有误,欢迎您的指出。


常用接口


Tween类的基本使用:

// 声明的tween函数,用于实例化Tween对象
export function tween<T>(target?: T): Tween<T>;
// 构造函数相关,关于target可以通过构造函数或target方法设置缓动目标
export class Tween<T> {
  constructor(target?: T | null);
  target(target: T): Tween<T | undefined>;
}

创建缓动的简单示例:

let duration = 2;
// 使用构造函数
tween(this.node)
	.to(duration, {position:new Vec3(0, 10, 0)})
  .start();
// 使用target
tween()
	.target(this.node)
	.to(duration, {position:new Vec3(0, 10, 0), scale: new Vec3(0, 0, 0)})
	.start();

注意: Tween 支持同时修改目标的多个属性,并非仅一个

Tween提供的接口较多,为了方便查看,主要为了如下几类:

  • 静态接口
  • 基础接口
  • 队列接口 主要用于顺序执行,并列执行, 执行重复次数等复杂动作

静态接口

接口说明
stopAll(): void停止所有缓动
stopAllByTag(tag: number, target?: object): void停止所有指定标签的缓动
stopAllByTarget(target?: object): void停止所有指定对象的缓动

简单示例:

tween(this.node)
	.tag(1000)
	.to(duration, {position:new Vec3(0, 10, 0)})
	.call(() => {
  	// 三种方式,任一即可
    Tween.stopAllByTarget(this.tipNode);
  	Tween.stopAllByTag(1000);
  	Tween.stopAll();
  })
  .start();

基础接口

接口说明
tag(tag: number): this设置缓动的标签
target(target: T): Tween<T>添加一个 直接设置缓动目标 的瞬时动作
start(): Tween<T>运行当前缓动
stop(): Tween<T>停止当前缓动
clone(target: T): Tween<T>克隆当前缓动
to(duration, props, opts): Tween<T>添加一个对属性进行 相对值 计算的间隔动作
by(duration, props, opts): Tween<T>添加一个对属性进行 绝对值 计算的间隔动作
set(props): Tween<T>添加一个 直接设置目标属性 的瞬时动作
delay(duration: number): Tween<T>添加一个延时 action
call(callback: Function): Tween<T>添加一个回调 action
show(): Tween<T>添加一个显示action,只适用target 是节点类型
hide(): Tween<T>添加一个隐藏 action,只适用target 是节点类型
removeSelf(): Tween<T>添加一个移除自己 action,只适用target是节点类型

简单示例:

let duration = 2;
// 示例1: 显示和隐藏节点
tween(this.node)        
  .hide()        
  .delay(1.0)
  .show()
  .start();
// 示例2: 移除节点
tween(this.node)        
  .delay(1.0)
  .removeSelf()        
  .start();
// 示例3: 设置目标属性、可选选项及销毁缓动
tween()
	.tag(1000)
	.target(this.tipNode)
	.set({position: new Vec3(0, 0, 0), scale: new Vec3(1, 1, 1)})
	.delay(duration)
	// 设置位置属性及缓动的可选选项
	.to(duration, {position: new Vec3(0, 300, 0), scale: new Vec3(0, 0, 0)}, {
  	// 设置缓动方式
  	easing: "smooth",
  	onComplete: (target) => {
    	console.log("当缓动动作完成时触发");
  	},
	})
	.call(() => {
  	Tween.stopAllByTarget(this.tipNode);
	})
	.start();

队列接口

接口说明
union(): Tween<T>将之前所有的 action 整合为一个 action
then(other: Tween<T>): Tween<T>插入一个 tween 到队列中
sequence(...args: Tween<T>[]): Tween<T>添加一个顺序执行的缓动
parallel(...args: Tween<T>[]): Tween<T>添加一个同时进行的缓动
repeat(repeatTimes, embedTween?): Tween<T>执行几次
repeatForever(embedTween?): Tween<T>一直重复执行

简单的示例:

let duration: number = 1.0;

// 整合       
tween(this.node)
  .to(duration, {position:new Vec3(0, 10, 0)})
	// 此时 Tween 内的动作数量为 2
  .to(duration, {position:new Vec3(0, -10, 0)})
	// 这里会将上述的两个缓动整合成一个,此时 Tween 内的动作数量为 1
  .union()
  .start(); 

// 插入
let t2 = tween(this.node)
	.to(duration, { position: new Vec3(0, -10, 0) })

tween(this.node)
  .by(duration, { position: new Vec3(0, 10, 0) })
  .then(t2)
  .start();

// 使用sequence按照顺序执行
let t1 = tween(this.node)
	.to(duration, {position: new Vec3(0, 10, 0)})
let t2 = tween(this.node)
	.to(duration, {position: new Vec3(0, -10, 0)})
tween(this.node).sequence(t1, t2).start();

// 使用parallel同时执行
let t1 = tween(this.node)
	.to(duration, {position: new Vec3(0, 10, 0)})
let t2 = tween(this.node)
	.to(duration, {position: new Vec3(0, -10, 0)})
tween(this.node).parallel(t1, t2).start();

// 重复执行
// 方式1: 
let tweenDuration: number = 1.0;
tween(this.node)
  .to(tweenDuration, { position: new Vec3(0, 10, 0) })
  .by(tweenDuration, { position: new Vec3(0, -10, 0) })
  .repeat(3) // 注意这里会重复 by 这个缓动 3 次
  .start()
// 方式2:
let embedTween = tween(this.node)
	.by(tweenDuration, {position: new Vec3(0, -10, 0)})

tween(this.node)
    .to(tweenDuration, {position: new Vec3(0, 10, 0)})
    .repeat(3, embedTween)  // 这里会重复 embedTween
    .start();

ITweenOption 缓动选项


ITweenOption主要应用于Tweento和by方法中, 它作为可选参数,用于设定缓动的方式和回调相关。

export class Tween<T> {
  // 添加相对值属性
  to(duration: number, props: __private.cocos_tween_tween_ConstructorType<T>, opts?: ITweenOption): Tween<T>;
  // 添加绝对值属性
  by(duration: number, props: __private.cocos_tween_tween_ConstructorType<T>, opts?: ITweenOption): Tween<T>;
}

这样可以更灵活的控制或修改目标的属性, 主要定义如下:

// 缓动方式
export type TweenEasing =
'linear'    | 'smooth'     | 'fade'         | 'constant'     |
'quadIn'    | 'quadOut'    | 'quadInOut'    | 'quadOutIn'    |
'cubicIn'   | 'cubicOut'   | 'cubicInOut'   | 'cubicOutIn'   |
'quartIn'   | 'quartOut'   | 'quartInOut'   | 'quartOutIn'   | 
'quintIn'   | 'quintOut'   | 'quintInOut'   | 'quintOutIn'   |
'sineIn'    | 'sineOut'    | 'sineInOut'    | 'sineOutIn'    |
'expoIn'    | 'expoOut'    | 'expoInOut'    | 'expoOutIn'    |
'circIn'    | 'circOut'    | 'circInOut'    | 'circOutIn'    |
'elasticIn' | 'elasticOut' | 'elasticInOut' | 'elasticOutIn' |
'backIn'    | 'backOut'    | 'backInOut'    | 'backOutIn'    |
'bounceIn'  | 'bounceOut'  | 'bounceInOut'  | 'bounceOutIn';

// 缓动选项
export interface ITweenOption {
  // 设置缓动方式
  easing?: TweenEasing | ((k: number) => number);
  // 插值函数,参数的意义 start:起始值,end:目标值,current:当前值,ratio:当前进度
  progress?: (start: number, end: number, current: number, ratio: number) => number;
  // 回调,当缓动动作启动时触发
  onStart?: (target?: object) => void;
  // 回调,当缓动动作更新时触发
  onUpdate?: (target?: object, ratio?: number) => void;
  // 回调,当缓动动作完成时触发
  onComplete?: (target?: object) => void;
}

官方提供的示例:

let duration: number = 1.0;
tween(this.node)
  .to(duration, {position: new Vec3(0, 10, 0)}, {
  	// 设置缓动函数
  	easing: "backIn",
  	onStart: (target?: object) => {
			console.log("缓动动作启动时触发");
  	},
  	onUpdate: (target: Vec3, ratio: number) => {
      console.log("当缓动动作更新时触发, 进度:", ratio);
      // 将缓动系统计算出的结果赋予 node 的位置
    	this.node.position = target;                    
  	},
  	onComplete: (target?: object) => {
			console.log("当缓动动作完成时触发");
  	},
  	progress: (start, end, current, ratio): number => {
    	// 返回自定义插值进度
      return 0.0;
  	}
	})
  .start();   

这里做下延伸,在CocosCreator 3.x以后,节点透明度的参数opacity修改为了组件: UIOpactiy

节点的基本属性仅包含: 位置、缩放和旋转。

如果想实现一个提示功能,在节点上移的过程中逐渐透明, UI如下:
请添加图片描述

代码如下:

let duration = 2;
let tweenObj = tween(this.tipNode)
	.tag(1000)
	.delay(duration)
	.to(duration, {position: new Vec3(0, 300, 0)}, {
  	easing: "smooth",
  	onUpdate: (target, ratio) => {
      // 更新时触发的回调,使用进度修改透明度
    	let uiOpacity = this.tipNode.getComponent(UIOpacity);
    	uiOpacity.opacity = (1 - ratio) * 255;
  	},
	})
	.removeSelf()
	.start();
console.log("tweenObj:", tweenObj);

最后看下tween对象的构成:
请添加图片描述

主要组成部分:

  • _actions 用于创建一组对象的缓冲
  • _finalAction 最终使用的Action对象
  • _tag 标记
  • _target 目标

至此结束。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/194151.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

R语言期末考试复习二

上篇文章的后续&#xff01;&#xff01;&#xff01;&#xff01; http://t.csdnimg.cn/sqvYD 1.给向量vec1设置名为"A","B","C","D","E","F","G"。 2.将矩阵mat1的行名设置为"Row1"&#…

8 个适用于电脑的顶级免费分区恢复软件

Windows PC 上的数据管理有时可能会带来压力&#xff0c;尤其是当您有多个分区时。大多数时候&#xff0c;磁盘管理工具使分析磁盘、释放空间甚至创建分区变得非常容易。但有时会发生不可预见的事件&#xff0c;可能导致分区丢失&#xff0c;从而造成潜在的数据灾难。嗯&#x…

ATA-7030高压放大器在等离子体实验中的应用有哪些

高压放大器在等离子体实验中有多种重要应用。等离子体是一种带电粒子与电中性粒子混合的物质&#xff0c;其具有多种独特的物理性质&#xff0c;因此在许多领域具有广泛的应用&#xff0c;例如聚变能源、等离子体医学、材料加工等。下面安泰电子将介绍高压放大器在等离子体实验…

pycharm安装PyQt5及其工具

PyCharm安装PyQt5及其工具&#xff08;Qt Designer、PyUIC、PyRcc&#xff09;详细教程_pycharm pyqt5-CSDN博客 上面是原文链接&#xff0c;根据原文链接&#xff0c;我重新记录一下。IDE&#xff1a;pycharm 2023.2.5 一共需要安装5个。 在PyCharm中如何完整优雅地安装配置…

Spring-SpringFramework特性以及IOC相关知识

SpringFramework五大模块 特性 IOC思想和DI IOC是容器&#xff0c;用于管理资源 IOC&#xff1a;Inversion of Control 反转控制 DI&#xff1a;Dependecy Injection 依赖注入 组件以预先定义好的方式接受来自容器的资源注入 IOC在Spring中的实现 spring提供两种方式&…

2023.11.27如何使用内网穿透工具实现Java远程连接操作本地Elasticsearch搜索引擎

文章目录 前言1. Windows 安装 Cpolar2. 创建Elasticsearch公网连接地址3. 远程连接Elasticsearch4. 设置固定二级子域名 前言 简单几步,结合Cpolar内网穿透工具实现Java远程连接操作本地Elasticsearch。 什么是elasticsearch&#xff1f;一个开源的分布式搜索引擎&#xff0…

Vue 3 面试经验分享

&#x1f337;&#x1f341; 博主猫头虎&#xff08;&#x1f405;&#x1f43e;&#xff09;带您 Go to New World✨&#x1f341; &#x1f984; 博客首页——&#x1f405;&#x1f43e;猫头虎的博客&#x1f390; &#x1f433; 《面试题大全专栏》 &#x1f995; 文章图文…

powershell获取微软o365 21v日志

0x00 背景 o365 21v为o365的大陆版本&#xff0c;主要给国内用户使用。微软提供了powershell工具和接口获取云上日志。微软o365国内的代理目前是世纪互联。本文介绍如何用powershell和配置证书拉取云上日志。 0x01 实践 第一步&#xff0c;ip权限开通&#xff1a; 由世纪互联…

正则表达式 通配符 awk文本处理工具

目录 什么是正则表达式 概念 正则表达式的结构 正则表达式的组成 元字符 元字符点&#xff08;.&#xff09; 代表字符. 点值表示点需要转义 \ r..t 代表r到t之间任意两个字符 过滤出小写 过滤出非小写 space空格 [[:space:]] 表示次数 位置锚定 例&#xff1a…

小航助学题库蓝桥杯题库stem选拔赛(21年1月)(含题库教师学生账号)

需要在线模拟训练的题库账号请点击 小航助学编程在线模拟试卷系统&#xff08;含题库答题软件账号&#xff09;_程序猿下山的博客-CSDN博客 需要在线模拟训练的题库账号请点击 小航助学编程在线模拟试卷系统&#xff08;含题库答题软件账号&#xff09;_程序猿下山的博客-CSD…

小航助学题库蓝桥杯题库stem选拔赛(21年3月)(含题库教师学生账号)

需要在线模拟训练的题库账号请点击 小航助学编程在线模拟试卷系统&#xff08;含题库答题软件账号&#xff09;_程序猿下山的博客-CSDN博客 需要在线模拟训练的题库账号请点击 小航助学编程在线模拟试卷系统&#xff08;含题库答题软件账号&#xff09;_程序猿下山的博客-CSD…

6.Spring源码解析-loadBeanDefinitions(String location)

这里resourceLoader其实就是ClassPathXmlApplicationContext 1.ClassPathXmlApplicationContext 在上文中图例就能看出来 获取资源组可能存在多个bean.xml 循环单独加载资源组 创建一个编码资源并解析 获取当前正在加载的资源发现是空 创建了一个字节输入流&#xff0c…

HTML5+CSS3+JS小实例:九宫格图片鼠标移入移出方向感知特效

实例:九宫格图片鼠标移入移出方向感知特效 技术栈:HTML+CSS+JS 效果: 源码: 【HTML】 <!DOCTYPE html> <html><head><meta http-equiv="content-type" content="text/html; charset=utf-8"><meta name="viewport&…

单车模型及其线性化

文章目录 1 单车模型2 线性化3 实现效果4 参考资料 1 单车模型 这里讨论的是以后轴为中心的单车运动学模型&#xff0c;由下式表达&#xff1a; S ˙ [ x ˙ y ˙ ψ ˙ ] [ v c o s ( ψ ) v s i n ( ψ ) v t a n ( ψ ) L ] \dot S \begin{bmatrix} \dot x\\ \dot y\\ \d…

【vue_3】关于超链接的问题

1、需求2、修改前的代码3、修改之后&#xff08;1&#xff09;第一次&#xff08;2&#xff09;第二次&#xff08;3&#xff09;第三次&#xff08;4&#xff09;第四次&#xff08;5&#xff09;第五次 1、需求 需求&#xff1a;要给没有超链接的列表添加软超链接 2、修改前…

单片机的串口通信

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、串口是什么&#xff1f;二、单片机结构讲解2.1 串口发送2.2串口接收2.3 还差点什么&#xff1f;2.3.1控制寄存器2.3.1.1 配置方式2.3.1.1 波特率 三、测试通…

Scanner常用知识点

在Java中&#xff0c;Scanner类是用于读取用户输入的工具类&#xff0c;可以从多种输入源读取数据&#xff0c;如标准输入流、文件或字符串。以下是一些Scanner类的常用知识点&#xff1a; Scanner的初始化&#xff1a;在使用Scanner类之前&#xff0c;需要先将其导入到你的Ja…

机器学习——多元线性回归升维

机器学习升维 升维使用sklearn库实现特征升维实现天猫年度销量预测实现中国人寿保险预测 升维 定义&#xff1a;将原始的数据表示从低维空间映射到高维空间。在线性回归中&#xff0c;升维通常是通过引入额外的特征来实现的&#xff0c;目的是为了更好地捕捉数据的复杂性&#…

二十九、微服务案例完善(数据聚合、自动补全、数据同步)

目录 一、定义 二、分类 1、桶(Bucket)聚合: 2、度量(Metric&#xff09;聚合: 3、管道聚合&#xff08;Pipeline Aggregation&#xff09;&#xff1a; 4、注意&#xff1a; 参与聚合的字段类型必须是: 三、使用DSL实现聚合 聚合所必须的三要素&#xff1a; 聚合可配…

【读懂AUTOSAR】DoIP模块(1)-- 使用场景和链接的建立规范

引子 --什么是?为什么使用DoIP? DoIP就是通过IP进行诊断的意思(Diagnostic Over IP)。我们熟悉的诊断都是通过CAN总线的啊,为什么要通过IP?IP是什么? IP就是Internet Protocol,就是”互联网协议“啦! 那DoIP就是通过互联网进行的诊断喽,也可以叫做“基于以太网的诊…