CocosCreator 之 Tween缓动系统的使用

news2024/11/24 19:17:51

版本: 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 目标

至此结束。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Linux文件操作应用及open和fork

1.文件操作的应用: 1).打开一个文件并往里面写入hello: #include <stdio.h> #include <unistd.h> #include <stdlib.h> #include <fcntl.h> #include <assert.h> int main() { int fdopen("file.txt",O_WRONLY|O_CREAT,0600); …

oracle免费资源 终止实例 以及新建一台实例的折腾记录

事情的背景是这样的&#xff0c;我的一台oracle小鸡&#xff0c;不太好用的样子&#xff0c;有时候SSH连不上&#xff0c;有时候莫名其妙卡住。所以我就想把它重新安装一下系统&#xff0c;恢复成最初的样子。 然后在网上查资料&#xff0c;是有办法把系统重装一下的。但是略微…

pdf加密文件解密(pdf文件解密小工具)

工具放在文章末尾&#xff01; 1.pdf文件加密后会有很多使用权限的限制很不方便&#xff0c;只要是为了pdf的数据不被二次利用&#xff0c;未加密的pdf功能都是可以正常使用的 2.加密后的pdf使用权限会被限制部分 3.工具只能解决pdf编辑等加密情况&#xff0c;不能解决文件打…

深度学习及其基本原理

深度学习的 Ups and Downs概念区分神经网络的构成深度学习基本原理深度学习的普遍近似定理扩展&#xff1a;反卷积网络——可视化每一层提取的特征 深度学习的 Ups and Downs 1958&#xff1a;感知机&#xff08;线性模型&#xff09;1969&#xff1a;感知机有局限性1980s&…

stream流和方法引用

1.Stream流 1.1体验Stream流【理解】 案例需求 按照下面的要求完成集合的创建和遍历 创建一个集合&#xff0c;存储多个字符串元素把集合中所有以"张"开头的元素存储到一个新的集合把"张"开头的集合中的长度为3的元素存储到一个新的集合遍历上一步得到的集…

Go 基本语法

一、​​​​变量定义方法 var 定义变量 var 变量名 类型 表达式 var name string "Snail" var age int 21 var isOK bool bool 2.类型推导方式定义变量 a 在函数内部&#xff0c;可以使用更简略的: 方式声明并初始化变量**注意&#xff1a;**短变量只能用于声…

MySQL数据库如何实现跨服务器访问数据

点击上方蓝字关注我 在使用MySQL数据库时&#xff0c;很多同学经常会问&#xff0c;我能跨服务器访问另一库的数据么&#xff1f;得到的答案很多时候是让人失望的。那么如果真的需要访问&#xff0c;又不想使用拷贝表及数据的方式&#xff0c;可以实现么&#xff0c;又该如何实…

【jupyter notebook中插件 nbextensions 安装失败分析与解决方法】

文章目录 问题描述分析与解决总结 问题描述 一开始在安装 notebook 中的插件 nbextensions 时根本没有注意到版本的适配问题&#xff0c;都是进行默认的安装&#xff0c;结果安装是最新版本的 notebook7.x&#xff0c;恰好 notebook7.x 版本不再适应插件 nbextensions&#xf…

【计算方法与科学建模】矩阵特征值与特征向量的计算(五):乘幂法的加速(带有原点移位的乘幂法)

文章目录 一、Jacobi 旋转法二、Jacobi 过关法三、Householder 方法四、乘幂法四、乘幂法的加速 矩阵的特征值&#xff08;eigenvalue&#xff09;和特征向量&#xff08;eigenvector&#xff09;在很多应用中都具有重要的数学和物理意义。 本文将详细介绍乘幂法的基本原理和步…

尚硅谷大数据项目《在线教育之实时数仓》笔记008

视频地址&#xff1a;尚硅谷大数据项目《在线教育之实时数仓》_哔哩哔哩_bilibili 目录 第10章 数仓开发之DWS层 P066 P067 P068 P069 P070 P071 P072 P073 P074 P075 P076 P077 P078 P079 P080 P081 P082 第10章 数仓开发之DWS层 P066 第10章 数仓开发之DW…

互联网金融智能风险防控技术要求

《互联网金融智能风险防控技术要求》 8月6日&#xff0c;国家市场监督管理总局和国家标准化管理委员会发布《互联网金融智能风险防控技术要求》&#xff08;GB/T 42929-2023&#xff09;&#xff08;以下简称“《要求》”&#xff09;&#xff0c;将于2023年12月1日实施。 《要…

MySQL数据库:外键、唯一键、唯一索引

目录 说明 一、如果要使用外键&#xff0c;表的存储引擎选择哪个&#xff1f; 1.1 答 1.2 示范 1.2.1 主表 &#xff08;1&#xff09;MyISAM的表&#xff1a;masterTable2 &#xff08;2&#xff09;InnoDB的表&#xff1a;masterTable1 1.2.2 从表 &#xff08;1&am…

图书管理系统源码,图书管理系统开发,图书借阅系统源码四TuShuManager应用程序MVC视图View

Asp.net web应用程序MVC之View视图 .ASP.NET MVC页面也就是要说的视图基本被放在Views文件夹下&#xff1b; 2.利用APS.NET MVC模板生成框架&#xff0c;Views文件夹下的默认页面为.cshtml页面&#xff1b; 3.ASP.NET MVC默认页面为Razor格式的页面&#xff0c;因此默认页面为.…

无人机光伏巡检代替人工,贵州电站运维升级

无人机光伏巡检如何做到降本增效&#xff1f;贵州省光伏电站有新招&#xff01;某70MWp的光伏电站通过引入复亚智能无人机光伏巡检系统&#xff0c;专注于使用无人机对区域内的光伏面板进行自动巡航巡查&#xff0c;利用自动化巡检和故障识别技术&#xff0c;显著提升了光伏电站…

UniPro集成华为云WeLink 为企业客户构建互为联接的协作平台

华为云WeLink是华为开启数字化办公体验、帮助企业实现数字化转型的实践&#xff0c;类似钉钉。UniPro的客户企业中&#xff0c;有使用WeLink作为协作工具的&#xff0c;基于客户的实际业务需求&#xff0c;UniPro实现了与WeLink集成的能力&#xff0c;以帮助客户企业丰富和扩展…

【触想智能】无风扇工控电脑一体机使用优势分析

无风扇工控电脑一体机是属于工控一体机分类中的其中一种&#xff0c;看名字&#xff0c;很明显就是没有散热风扇的工控电脑一体机&#xff0c;而平常我们使用的电脑主机是带有电源风扇、CPU散热风扇的。 无风扇工控电脑一体机的配置组成和商用电脑主机的配置基本一样&#xff0…

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

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

JMeter之压力测试——混合场景并发

在实际的压力测试场景中&#xff0c;有时会遇到多个场景混合并发的情况&#xff0c;这时就需要设置不同的并发比例对不同场景请求数量的控制&#xff0c;下面提供两种方案。 一、多线程组方案 1.业务场景设计如下&#xff1a;场景A、场景B、场景C&#xff0c;三个场景按照并发…

FPGA模块——AD高速转换模块(并行输出转换的数据)

FPGA模块——AD高速转换模块&#xff08;并行输出转换的数据&#xff09; &#xff08;1&#xff09;AD9280/3PA9280芯片&#xff08;2&#xff09;代码 &#xff08;1&#xff09;AD9280/3PA9280芯片 AD9280/3PA9280芯片的引脚功能&#xff1a; 工作电压2.7到5.5v 数据对应&a…

<Linux> 文件理解与操作

目录 前言&#xff1a; 一、关于文件的预备知识 二、C语言文件操作 1. fope 2. fclose 3. 文件写入 3.1 fprintf 3.2 snprintf 三、系统文件操作 1. open 2. close 3. write 4. read 四、C文件接口与系统文件IO的关系 五、文件描述符 1. 理解文件描述符 2. 文…