unreal3的viewport和client

名字里带viewport/client的类不少,以及相关的类:

FViewportFrame、FViewport

FViewportClient/UScriptViewportClient/UGameViewportClient

UClient/UWindowsClient

UEngine/UGameEngine/UEditorEngine/UUnrealEdEngine

极易混淆,现整理如下:

首先从总体层级上看,UEngine最大,它包含了UClientUGameViewportClient

class UEngine : public USubsystem
{
public:
……
    class UClient* Client;
    class UGameViewportClient* GameViewport;
……

UEngine是己只是个抽象类,实际运行时根据是游戏还是编辑器模式,创建相应子类的实例,如是游戏则UGameEngine,如是编辑器则UUnrealEdEngine

它的实例创建出来后就存在全局变量GEngine上。

在UGameEngine::Init里,先后创建了其它对象:

1、先创建了UClient,实际的子类通过“engine-ini:Engine.Engine.Client”配置指定,在windows上是【WinDrv.WindowsClient】,在mac上则是【MacDrv.MacClient】,可见各子类就是对应各平台相关的实现;

2、然后创建UGameViewportClient,实际的类型是通过Engine上的GameViewportClientClass这个globalconfig属性指定的,所以它会自动从配置中读取,配置的默认值是【Engine.GameViewportClient】

3、接着创建FViewportFrame:

ViewportFrame = Client->CreateViewportFrame(
                ViewportClient,
                *AppName,
                GSystemSettings.ResX,
                GSystemSettings.ResY,
                GSystemSettings.bFullscreen
                );
FViewportFrame* UWindowsClient::CreateViewportFrame(FViewportClient* ViewportClient,const TCHAR* InName,UINT SizeX,UINT SizeY,UBOOL Fullscreen)
{
  return new FWindowsViewport(this,ViewportClient,InName,SizeX,SizeY,Fullscreen,NULL);
}

注意:创建函数(CreateViewportFrame)是UClient上的、并把UGameViewportClient做为参数,由此可略知其从属关系。

做为一般情形,这里创建的是FViewportFrame,所谓-Frame,也就是(操作系统层面的)外包窗口,因为通常程序总是要自己创建“主窗口”。但是在某些模式下(如内嵌于浏览器等),外包主窗口已经有了,就不需要再创建FViewportFrame,取而代之的是通过CreateWindowChildViewport接口来创建一个FViewport。

实际上FViewportFrame接口只有两个函数,一是获取其中的FViewport,二是Resize改变大小,这个Resize更能体现其做为外包窗口的特征。

此外,前面提到UClient的实际类型是UWindowsClient,它创建的FViewportFrame自然也是同一平台体系下的子类即FWindowsViewport。

FWindowsViewport是一个特定于Windows平台相关的实现类,它同时实现了FViewportFrame和FViewport两个接口,当然这只是实现的方便,并不能抹杀这两个接口的差异。

因为3中的创建方式,FViewportFrame在创建时传入了两个关键对象:UClient和UGameViewportClient,这当然都会被记在其成员变量上,以便后续使用。

下面细看UEngine的三大台柱子:(UClient、UGameViewportClient、FViewport)

台柱之一:UClient

看起来它像是对整个app的封装,其中的Tick函数每帧被调用,主要内容就是处理窗口和输入消息。

它的另外一个作用则是用来创建FViewport(Frame),这主要是给平台相关的子类机会,创建同一平台体系的FViewport子类。

UClient、UWindowsClient都是纯c++类(没有相关的脚本类),但是由于手动在声明中添加了DECLARE_ABSTRACT_CLASS_INTRINSIC,所以也遵照UObject/UClass系统规则,可以用StaticLoadClass/ConstructObject等规范函数统一创建。

台柱之二:UGameViewportClient

UGameViewportClient继承自->UScriptViewportClient->(UObject,FViewportClient),这里UScriptViewportClient没啥用,纯粹是打个酱油,把FViewportClient这种纯c++类转换到UClass体系链中而已。

先看FViewportClient基类接口的方法:

看起来它主要是用于消息处理。

但是在UEngine这种高层基类中声明的属性GameViewport,其类型并非FViewportClient基类,而是直接作为UGameViewportClient子类变量,这主要是因为脚本要用,GameViewport属性是在UEngine的脚本类里声明的,在脚本里只用当然只能用符合UObject/UClass体系的东西,而做为继承链中间过渡的UScriptViewportClient就是起了这个作用。

当然UGameViewportClient也添加了很多新方法和属性,如它也有一个Tick函数,也是每帧被调用。还有一个非要重要的成员GlobalInteractions,这个数组存的就是所有的输入消息处理器。

台柱之三:FViewport

这个,实在看不出有什么特点,只能说它是跟渲染器关联最紧密的一个类了,unreal3应该就是用FViewport来抽象一个可以渲染的区域吧。

三者之间彼此关联,纠缠万分。

首先,UEngine上存着由它创建的UClient和UGameViewportClient各一个,并且在Tick中又分别调用了它们的Tick

然后,UGameViewportClient上存着由UClient创建的FViewportFrame(以及其内的FViewport),(UGameViewportClient::SetViewportFrame)

还有,做为子类的UWindowsClient上存着所有由它创建的FWindowsViewport(FViewportFrame,FViewport),(static TArray<FWindowsViewport*> Viewports)

最后,作为最后被创建出来的FWindowsViewport,它当然也要记着创建它的UClient,以及用来初始化它的UGameViewportClient(这个是由基类FViewport记着的)。

在消息处理方面:

首先,系统窗口的消息处理函数是UWindowsClient::StaticWndProc

然后,大部份消息会交给该窗口(HWnd)对应的FWindowsViewport来处理:Viewports(i)->ViewportWndProc

然后,除了需要即时响应的消息被立即处理外,大部份输入类消息被缓存起来:Client->DeferMessage,这样一来消息处理流程又回到了UWindowsClient里

然后,在UWindowsClient::Tick里,会调用ProcessDeferredMessages和ProcessInput,来处理所有的输入消息,前者包括了键盘和鼠标点击,后者是鼠标移动,在Unreal3里比较特别的是,鼠标移动并不通过WM_XXX类的窗口消息来获取,而是用IDirectInputDevice8接口来获取。

ProcessDeferredMessages里面,每个消息又转给它的FWindowsViewport上的同名函数,最终都会交到它里面的FViewportClient(也就是UGameViewportClient)::InputKey/InputChar

ProcessInput里面,取当前活跃的FWindowsViewport(通常也就一个窗口),把鼠标事件都交给它里面的FViewportClient(也就是UGameViewportClient)::InputAxis

可见,键盘和鼠标操作,最终都是到了UGameViewportClient的InputXXX里,而它们里面的分派逻辑,也大致一样:

先是给自己身上的代理处理,UGameViewportClient上的HandleInput(Axis/Key/Char)

然后是给GlobalInteractions里的每一个去处理,这里的每个元素都是一个Interaction,Interaction是Unreal3里用来处理用户操作的基类,它身上也有一套Input(Axis/Key/Char)。

实际运行观察结果 ,GlobalInteractions共有4个元素,分别是:

Console,用来处理调试指令

GFxInteraction,给swf对象分派消息

UIInteraction,这个UI不知是啥了,可能是Unreal3自己的UI模块?

PlayerManagerInteraction:比较逻辑的一层,把输入映射给对应的Player处理,有点难以理解,但是ini里的[Engine.PlayerInput]节中,所有bindings绑定的消息处理函数,就是由它分派的

时间: 2024-11-05 00:39:23

unreal3的viewport和client的相关文章

AJAX MVC server返回Json数据,client获取Json数据

<> 控制器 Controller using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Web.Mvc; using System.Web.Script.Serialization; namespace MvcApplication2.Controllers { public class HomeController : Controller { pu

关于client浏览器界面文字内容溢出用省略号表示方法

在实际的项目中,因为client浏览器文字内容的长度不确定性和页面布局的固定性,难免会出现文字内容超过div(或其它标签,下同)区域的情况.此时比較好的做法就是当文字超过限定的div宽度后自己主动以省略号(-)显示,这样.依照习惯,人们都会知道这儿有文字被省略了. 使用CSS截断字符串方法 CSS中有个属性叫做text-overflow:ellipsis. 说明:长处是内容能够为不论什么HTML元素.包含超链接和图片等,在IE6中还会在结尾自己主动显示省略号.缺点是必须指定宽度数值.而且宽度不能

推断client手机类型,并跳转到对应的app下载页面

实现的原理,是检測浏览器的 USER-AGENT 这个header,然后依据正則表達式来确定client类型. 假设都不匹配,Fallback回退策略是显示相应的页面.让用户自己选择. 适合採用二维码扫描方式下载APP: JSP版本号的代码例如以下所看到的:其它服务端版本号请百度搜索. 原帖地址: [您也能够訪问向上交流论坛查看很多其它行业,开源,招聘信息] <%@page import="java.util.regex.Matcher"%> <%@page impo

adb server version (31) doesn’t match this client (36); killing… 的解决方法

今天折腾了一下Android studio,好不容易搞完了,结果在运行adb命令的时候出错了 不要着急,我们先看下这个错误是怎么出现的. 1.搞完Android studio之后也升级使用了新的sdk,并配置了环境变量 2.启动了第三方模拟器 3.之后用adb devices命令就出现了上图的错误 从报错的原因来看应该是版本不一致导致的,大概的意思是说你现在的adb版本是31,但是你client端的却是36(这里的client端我猜就是第三方的模拟器)不匹配啊. 尝试解决下,既然版本不一致,那就

swoole webSocket server or client example

server: $ws = new swoole_websocket_server('0.0.0.0', 9502); $ws->on('open', function ($ws, $request) {    var_dump($request->fd, $request->get, $request->server);    $ws->push($request->fd, "hello world\n");}); $ws->on('mess

elasticsearch java 客户端之Client简介

elasticsearch通过构造一个client体现对外提供了一套丰富的java调用接口.总体来说client分为两类cluster信息方面的client及数据(index)方面的client.这两个大类由可以分为普通操作和admin操作两类.以下是client的继承关系(1.5版本,其它版本可能不一样): 通过这个继承关系图可以很清楚的了解client的实现,及功能.总共有三类即client, indicesAdminClient和ClusterAdminClient.它都有自己的实现类,但

QDjango,tufao,C++ websocket client/server

QDjango, a Qt-based C++ web frameworkhttps://github.com/jlaine/qdjango/ An asynchronous web framework for C++ built on top of Qt http://vinipsmaker.github.io/tufao/https://github.com/vinipsmaker/tufao C++ websocket client/server library http://www.za

打开vsphere client提示Unable connect to MKS

问题描述:客户有3台服务器,突然发现用户的vsphere client打开虚拟机控制台的时候,提示Unable connect to MKS:Failed to connect to server IP地址:902 解决方法:通过测试发现,有一台是没问题的,另外两台无法均报错,通过对比发现,出现问题的两台服务器网关地址填写错误导致的,把网关地址修改为正确的,发现问题得到解决.

移动端布局(viewport)方法

viewport默认有6个属性 width: 设置viewport的宽度(即之前所提及到的,浏览器的宽度详),这里可以为一个整数,又或者是字符串"width-device" initial-scale: 页面初始的缩放值,为数字,可以是小数 minimum-scale: 允许用户的最小缩放值,为数字,可以是小数 maximum-scale: 允许用户的最大缩放值,为数字,可以是小数 height: 设置viewport的高度(我们一般而言并不能用到) user-scalable: 是否