主要实现蓝图节点中时间轴的功能。



目前UE提供了两种实现方式,一个是使用FTimeLine其是一个时间轴的结构体;另一种方式是使用UTimeLineComponent,其是一个时间轴组件类。两者内部定义的函数基本一样,组件类中使用这个结构体变量作为类中的成员变量。

声明一个时间轴变量/组件

FTimeLine结构体和UTimeLineComponent组件都存在于TimelineComponent.这个类里面所以需要

#include "Components/TimelineComponent.h"

参考1

  1. 利用UTimeLineComponent

    时间轴是利用委托绑定函数来实现在时间轴的范围内,实现动画切换或者其他功能的。

    所以我们一般需要定义两个委托事件,分别处理时间轴每帧的操作和时间轴结束的操作。UE定义了4种绑定事件委托的类型(也对应有静态委托)



    一般绑定事件使用FOnTimelineEvent,当如果我们想使用曲线或者向量来控制时间轴的变化我们需要使用对应的下面3个委托。这3个委托会返回对应曲线的每一帧的位置数值。



    如果利用C++编写就需要在UE中新建曲线,然后在使用时间轴的Actor类中,定义一个曲线变量用来接受设置的曲线,并且将该变量传入对应的动态委托中。



//.h

//设置一个时间轴组件变量
UPROPERTY(BlueprintReadWrite, EditDefaultsOnly)
UTimelineComponent* CWTimeline;
//设置一个曲线
UPROPERTY(BlueprintReadWrite, EditDefaultsOnly)
UCurveFloat* CWFloatCurve; //设置委托事件
//定义曲线更新事件,事件定义的选择要和我们使用的曲线选择对应
FOnTimelineFloat OnCWTimelineTickCallBack;
//定义曲线播发完成事件
FOnTimelineEvent OnCWTimelineFinishedCallBack; //设置委托绑定的函数
UFUNCTION()
void CWTimelineTickCallBack(float value);
UFUNCTION()
void CWTimelineFinishedCallBack();

接下来就需要开始定义和初始化这些声明设置。在BeginPlay()中进行委托的绑定和组件的初始化操作。(不知道在构造函数位置是否可以)

由于我们使用的组件,所以需要实例化来获取里面结构体,就如同相机组件或者弹簧臂组件一样。这是第一个与FTimeLine不同的地方。

CWTimeline = NewObject<UTimelineComponent>(this,"CrouchTimeLine");
//NewObject CreateDefaultObject的区别还不太明白。。。。。。

然后就直接开始委托绑定

OnCWTimelineTickCallBack.BindDynamic(this,&ClassName::CWTimelineTickCallBack)
//我们要把这个委托附加给时间轴,然后我们是根据曲线来进行时间轴的输出,我们利用委托可以获得曲线的每一帧的数值,然后传给被绑定的函数
CWTimeline->AddInterpFloat(CWFloatCurve,OnCWTimelineTickCallBack) //当时间轴事件结束的委托
OnCWTimelineFinishedCallBack.BindDynamic(this, &AJCharacter:CWTimelineFinishedCallBack);
CWTimeline->SetTimelineFinishedFunc(OnCWTimelineFinishedCallBack); //注册组件
CWTimeline->RegisterComponent();

在委托所绑定的函数上进行我们要进行的操作,比如将摄像机的位置发生变化

其中利用插值函数实现平滑的过渡

value的实参是通过委托获取曲线的每一帧数值。

void AJCharacter::CWTimelineTickCallBack(float value)
{
SpringArmComp->SocketOffset.X = 0.0f;
SpringArmComp->SocketOffset.Y = SpringArmComp->SocketOffset.Y;
SpringArmComp->SocketOffset.Z = FMath::Lerp(0.0f, -40.0f,value);
} void AJCharacter::CWTimelineFinishedCallBack()
{
}

最后我们可以在想要使用时间轴的函数中进行播放时间轴,比如在下蹲时摄像头的变化

void AJCharacter::JCrouch()
{
if (!GetCharacterMovement()->IsCrouching()) {
Crouch();
bIsCrouch = true;
GetCharacterMovement()->MaxWalkSpeedCrouched = 135.0f;
CWTimeline->Play();
}
else {
UnCrouch();
bIsCrouch = false;
GetCharacterMovement()->MaxWalkSpeed = 420.0f;
CWTimeline->Reverse();
} }

利用CWTimeline->Play()和CWTimeline->Reverse()实现正向和反向的播放。

注意:我们必须在Tick()函数中设置时间轴的TickComponent才能进行时间轴的播放

CWTimeline->TickComponent(DeltaTime, ELevelTick::LEVELTICK_TimeOnly, NULL);
  1. 利用FTimeLine结构体

    参考2

  2. FOnTimelineFloat委托

DECLARE_DYNAMIC_DELEGATE_OneParam( FOnTimelineFloat, float, Output );

通过定义可以看出,该委托有一个float类型的输出,该输出可以直接使用于该委托绑定的函数。

而这个值就是通过我们设置的CurveFloat插值得到的。

我们利用AddInetrpFloat(CurveFloat,FOnTimelineFloat)后,实际将CurveFloat做为其中一个参数传入到了元素类型为FTimelineFloatTrack的一个数组中。



而时间轴的更新是通过其里面的FTimeline::SetPlaybackPosition方法。

	/** Jump to a position in the timeline.
* @param bFireEvents If true, event functions that are between current position and new playback position will fire.
* @param bFireUpdate If true, the update output exec will fire after setting the new playback position.
*/
UFUNCTION(BlueprintCallable, Category="Components|Timeline", meta=(AdvancedDisplay="bFireUpdate"))
ENGINE_API void SetPlaybackPosition(float NewPosition, bool bFireEvents, bool bFireUpdate = true);

通过VecEntry.InterpFunc.ExecuteIfBound(Vec);将当前位置值传递出去,

FTimelineVectorTrack& VecEntry = InterpVectors[InterpIdx];
if ( VecEntry.VectorCurve && (VecEntry.InterpFunc.IsBound() || VecEntry.VectorPropertyName != NAME_None || VecEntry.InterpFuncStatic.IsBound()) )
{
// Get vector from curve
FVector const Vec = VecEntry.VectorCurve->GetVectorValue(Position); // Pass vec to specified function
VecEntry.InterpFunc.ExecuteIfBound(Vec);
//VecEntry.InterpFunc获取委托

通过AddInetrpFloat后我们得到的一个结构体变量,InterpFunc就是里面的委托,用来传递代码中委托时间的float值

	/** Function that the output from ValueCurve will be passed to */
UPROPERTY()
FOnTimelineFloat InterpFunc;

ExecuteIfBound为单播委托的调用方式,输出Vec,使得绑定函数可以使用。参考

  1. Play()和Reverse()
void FTimeline::Play()
{
bReversePlayback = false;//true
bPlaying = true;
}

调用之后,因为我们在Tick()使用了TickComponent,其将调用TickTimeLine



设置bPlaying为true后将调用SetPlaybackPosition方法,根据DeltaTime返回Position的值,从而获取曲线的对应值。从而得到输出

UE4_C++实现TimeLine的更多相关文章

  1. Chrome Developer Tools:Timeline Panel说明

    一.Timeline panel 概况 Timeline工具栏可以详细检测出Web应用在加载过程中,时间花费情况的概览.这些应用包括下载资源,处理DOM事件, 页面布局渲染或者向屏幕绘制元素. 如下图 ...

  2. Chrome 开发工具之Timeline

    之前有说到Element,Console,Sources大多运用于debug,Network可用于debug和查看性能,今天的主角Timeline更多的是用在性能优化方面,它的作用就是记录与分析应用程 ...

  3. iOS之TimeLine(时间轴)的实现

    这是一个关于OC时间轴的简单实现,我认为重要的是思路. 感谢作者:Cyandev 的文章<iOS 实现时间线列表效果>给的思路.这里先附上Objective-C的代码实现,有时间再去试试S ...

  4. 如何使用Goolge Timeline工具

    网上中文的资料版本比较老,找到一个新版本的英文介绍,翻一下,原文:https://developers.google.com/web/tools/chrome-devtools/profile/eva ...

  5. 【三石jQuery视频教程】03.创建垂直时间表(Timeline)

    视频地址:http://v.qq.com/page/g/i/o/g0150rvi6io.html 大家好,欢迎来到[三石jQuery视频教程],我是您的老朋友 - 三生石上. 今天,我们要通过基本的H ...

  6. WPF中的动画——(三)时间线(TimeLine)

    WPF中的动画——(三)时间线(TimeLine) 时间线(TimeLine)表示时间段. 它提供的属性可以让控制该时间段的长度.开始时间.重复次数.该时间段内时间进度的快慢等等.在WPF中内置了如下 ...

  7. 如何使用Chrome Timeline 工具(译)

    [原文地址]https://developers.google.com/web/tools/chrome-devtools/profile/evaluate-performance/timeline- ...

  8. 分享最新的博客到LinkedIn Timeline

    使用Octopress作为我的博客框架有两年了.使用起来一直很顺手,这个工具真正的把博客跟写代码等同起来,非常酷炫.再加上各种各样的定制化,简直是随心所欲.我针对自己的需求对Octopress框架进行 ...

  9. chrome dev debug network 的timeline说明

    在使用chrome的时候F12的开发者工具中有个network,其中对每个请求有个timeline的说明,当鼠标放上去会有下面的显示: 这里面的几个指标在说明在chrome使用文档有说明: 下面我用人 ...

  10. Chrome开发者工具详解(3)-Timeline面板

    Chrome开发者工具详解(3)-Timeline面板 注: 这一篇主要讲解面板Timeline,参考了Google的相关文档,主要用于公司内部技术分享.. Timeline面板 Timeline面板 ...

随机推荐

  1. springboot jpa自定义SQL查询

    说明 在使用JPA实现数据持久化过程中经常会遇到这种情况:我有2张表是一对多的关系,需要通过一个外键ID去关联查询到另外一张表的字段.例如,1张商品表food_info其中存有商品分类ID categ ...

  2. 如何在 libevent 中读取超过 4096 字节的数据

    如何在 libevent 中读取超过 4096 字节的数据 bufferevent 是 libevent 中相对高层的封装,较 event 使用起来方便很多. 之前有一个需求,需要从服务端读取数据进行 ...

  3. 【Android逆向】脱壳项目frida_dump 原理分析

    脱dex核心文件dump_dex.js 核心函数 function dump_dex() { var libart = Process.findModuleByName("libart.so ...

  4. Jenkins下载插件报错

    只要看日志报了什么错 下载超时 更新代理源https://mirrors.tuna.tsinghua.edu.cn/jenkins/updates/update-center.json 报unable ...

  5. 基于Vue(提供Vue2/Vue3版本)和.Net Core前后端分离、强大、跨平台的快速开发框架

    前言 今天大姚给大家推荐一款基于Vue(提供Vue2/Vue3版本)和.Net Core前后端分离.开源免费(MIT License).强大.跨平台的快速开发框架,并且框架内置代码生成器(解决重复性工 ...

  6. 【Azure 存储服务】Azure Storage Account 下的 Table 查询的性能调优

    问题描述 Azure Storage Account 下的 Table 查询的性能调优? 问题解答 因为Azure Storage Table服务(表服务) 与常规的关系型数据库不一样(例如:MySQ ...

  7. 【Azure 应用服务】Azure App Service能否使用Storage Account File Share

    问题描述 Azure App Service能否使用Storage Account File Share? 问题回答 如果部署的App Service为Linux环境,可以直接使用Mount stor ...

  8. [Python] 超简单的 超星学习通自动签到

    目录 概述 代码 其他的 文件编码问题 windows 和 linux下换行符不同的问题 概述 今天两节课的签到都错过了 /(ㄒoㄒ)/~~ 所以决定花点时间做一个自动签到的工具 经过观察发现超星的结 ...

  9. Java 重写 equals + toString 练习

    1 package com.bytezreo.objectclass2; 2 3 /** 4 * 5 * @Description 重写 equals + toString 6 * @author B ...

  10. XAF新手入门 - 数据字典示例

    前言 通过前面文章的介绍,大家应该对模块与类型信息子系统有所了解,再通过一个示例来加深大家对它的理解. 在准备写这个系列文章之前,就准备是概念+示例的组合,这样大家对概念的理解会更深刻.之前的规划是在 ...