用ECMAScript4 ( ActionScript3) 实现Unity的热更新 -- 使用原型链和EventTrigger
原型链是JS的必备,作为ECMAScript4,原型链也是支持的。
特别说明,ActionScript3是支持完整的面向对象继承支持的,原型链只在某些非常特殊的情况下使用。
本文旨在介绍如何使用原型链。
任意对象类型都有一个prototype属性,包括导入的Unity的API也有。我们可以和JS一样的对这个prototype进行操作,比如动态的添加成员,添加方法等。
- 如果您决定使用原型链,那么必须舍弃编译时类型检查。如果使用强类型,则编译器会认为动态添加的成员不存在,而提示编译错误。解决办法为去掉变量的类型说明,或者将变量强制转换成Object。
- 和JS引擎一样,原型链必须在运行时顺着prototype链查找。因此会影响性能。
本文说明,如何在脚本中,动态监听 EventTrigger。我们将这篇文章中介绍的使用不可热更的C#代码做的监听,改写为ActionScript3实现。
工程说明
- 新建一个Untiy工程,并且创建AS3热更新项目。如果不会创建,请参考这里
- 在HotFixDemoScene1上添加一个Image。

- 本次代码需要使用 RectTransformUtility 这个类。如果是Unity2017.2版本以后,由于这个类型被分拆到了其他dll (UnityEngine.UIModule.dll),因此需要将它加入到配置表里。
- 打开热更新工程的genapi.config.xml ,将如下配置加入配置节中:
<!--Configure DLLs to export-->
<buildassemblys> <assembly value="D:\Program Files\Unity\Editor\Data\Mono\lib\mono\2.0\System.dll"></assembly>
<assembly value="D:\Program Files\Unity\Editor\Data\Managed\UnityEngine\UnityEngine.CoreModule.dll"></assembly>
<assembly value="D:\Program Files\Unity\Editor\Data\Managed\UnityEngine\UnityEngine.UIModule.dll"></assembly>
<assembly value="D:\Program Files\Unity\Editor\Data\UnityExtensions\Unity\GUISystem\UnityEngine.UI.dll"></assembly> <assembly value="F:/ASTool_UnityTest/HotFix_EventTrigger\Library\ScriptAssemblies\Assembly-CSharp.dll"></assembly> </buildassemblys> - 打开热更新项目的Main.as文件,将内容改为如下代码:
package
{ [Doc]
/**
* ...
* @author
*/
public class Main
{ public function Main()
{
} public function update():void
{
} } }
import as3runtime.RefOutStore;
import system.collections.generic.List_Of_EventTrigger_Entry;
import unityengine.GameObject;
import unityengine.MonoBehaviour;
import unityengine.RectTransform;
import unityengine.RectTransformUtility;
import unityengine.Vector3;
import unityengine.events.UnityAction_Of_BaseEventData;
import unityengine.eventsystems.BaseEventData;
import unityengine.eventsystems.EventTrigger;
import unityengine.eventsystems.EventTriggerType;
import unityengine.eventsystems.EventTrigger_Entry;
import unityengine.eventsystems.PointerEventData; class UGUIEventTriggerTool
{
/**
* 静态方法。检查某个GameObject上是否包含EventTrigger组件。如果有则返回它,否则创建一个并返回。
* @param go 输入的GameObject
* @return 返回EventTrigger
*/
public static function Get(go:GameObject):EventTrigger
{
var trigger:EventTrigger = go.getComponent(EventTrigger) as EventTrigger;
if (null == trigger)
{
trigger = EventTrigger(go.addComponent(EventTrigger));
}
return trigger;
}
} /* *
* 在EventTrigger的原型链上定义AddEventListener方法。
* EventTrigger类型没有公开构造函数,因此API导出时,不会为他创建在脚本中继承的接口。
* 这里我们可以用原型链来对它进行扩展。
* 包外代码只会执行一次。所以只会在原型链上定义一次AddEventListener方法。
* */
EventTrigger.prototype.AddEventListener =
function (eventTriggerType:EventTriggerType, action:UnityAction_Of_BaseEventData):void
{
/**
* EventTrigger有一个嵌套内部类 UnityEngine.EventSystems.EventTrigger.Entry。
* 由于ActionScript3并不支持嵌套类,因此它被导出成为EventTrigger_Entry。
*/
var entry:EventTrigger_Entry = new EventTrigger_Entry();
entry.eventID = eventTriggerType;
entry.callback.addListener(action); this.triggers.add(entry); } /**
* 扩展MonoBehaviour,让Image组件可以被拖拽。
*/
class UGUIEventTriggerTest extends MonoBehaviour
{
function Start()
{
/**
* 此处使用 * 类型来代表任意类型。
* 这样即可跳过编译时类型检查,使用原型链。
* 否则将会提示编译错误。
*/
var t:* = UGUIEventTriggerTool.Get(gameObject);
/**
* 调用在原型链上定义的方法,给Drag事件添加处理函数。
* 我们可以直接使用匿名函数来作为处理函数。
*/
t.AddEventListener(EventTriggerType.Drag, function (baseData:BaseEventData):void
{
var data:PointerEventData = baseData as PointerEventData;
var rt:RectTransform = data.pointerPress.getComponent(RectTransform) as RectTransform;
var globalMousePos:Vector3; /**
* RectTransformUtility.screenPointToWorldPointInRectangle的参数
* worldPoint : (Out)UnityEngine.Vector3
* 是一个Ref Out参数。ActionScript3是没有ref和out关键字的,
* 因此这里使用RefOutStore来接收返回的参数。
*/
var store:RefOutStore = new RefOutStore();
if (RectTransformUtility.screenPointToWorldPointInRectangle(rt, data.position, data.pressEventCamera, globalMousePos,store))
{
//传入形参名,来提取值。
globalMousePos = store.getValue("worldPoint") as Vector3;
rt.position = globalMousePos;
}
} );
} }
//将脚本挂载到Image上。
GameObject.find("Image").addComponent(UGUIEventTriggerTest); - 点击编译,然后在Unity中点击播放。我们现在即可拖动这个Image。

用ECMAScript4 ( ActionScript3) 实现Unity的热更新 -- 使用原型链和EventTrigger的更多相关文章
- 用ECMAScript4 ( ActionScript3) 实现Unity的热更新
unity热更新是一个经久不衰的话题.除了最常见的lua之外,还有如JSBinding,C#等等.这里介绍一个使用ECMAScript4进行热更新的方案.它吸收了各家的优点,特色很鲜明. 项目地址: ...
- 用ECMAScript4 ( ActionScript3) 实现Unity的热更新 -- Demo分析
如何创建工程 下载最新的Unity发布插件包. 打开Unity,新建一个项目 将插件包导入 在菜单中点击ASRuntime/Create ActionScript3 FlashDevelop HotF ...
- 用ECMAScript4 ( ActionScript3) 实现Unity的热更新 -- 使用第三方组件
Unity开发中,常常会用到一些第三方组件.本文以实例介绍如何在热更新脚本中使用这些第三方组件. 首先说明几个基本步骤: 第三方组件通常是以dll或者源码方式提供的,它们本身往往无法热更. 我们在脚本 ...
- 用ECMAScript4 ( ActionScript3) 实现Unity的热更新 -- CustomYieldInstruction 自定义中断指令
ActionScript3脚本引擎为了方便热更新逻辑开发,提供的从脚本继承Unity类库功能在一些情况下可以提供开发的便利. 这次来建立一个示例,演示一下如何在脚本中自定义协程中断指令 Unity中的 ...
- 用ECMAScript4 ( ActionScript3) 实现Unity的热更新 -- 使用FairyGUI (一)
我们的热更新脚本在实际使用中,当然也要支持常用的第三方组件,例如这里介绍一个非常实用的第三方UI库:FairyGUI. 什么是FairyGUI 这里照搬FaiyGUI官网的介绍: 重新定义 UI 制作 ...
- 用ECMAScript4 ( ActionScript3) 实现Unity的热更新 -- 使用FairyGUI (二)
上次讲解了FairyGUI的最简单的热更新办法,并对其中一个Demo进行了修改并做成了热更新的方式. 这次我们来一个更加复杂一些的情况:Emoji. FairyGUI的 Example 04 - ...
- 用ECMAScript4 ( ActionScript3) 实现Unity的热更新 -- 热更新Live2D
live2D是一个很强大的2D动画组件.我们可以使用AS3脚本对它进行热更新. live2D在Unity中的使用请看这里: 如何获取Live2D 总得来说,我们可以先去live2D官网下载它的Unit ...
- 用ECMAScript4 ( ActionScript3) 实现Unity的热更新 -- 在脚本中使用MonoBehaviour
继上次分析了热更新的Demo后,这次来介绍如何在热更新代码中使用MonoBehaviour. MonoBehaviour挂载到GameObject对象上的脚本的基类.平常Unity开发时,简单的做法就 ...
- 用ECMAScript4 ( ActionScript3) 实现Unity的热更新 -- 操作符重载和隐式类型转换
C#中,某些类型会定义隐式类型转换和操作符重载.Unity中,有些对象也定义了隐式类型转换和操作符重载.典型情况有:UnityEngine.Object.UnityEngine.Object的销毁是调 ...
随机推荐
- Spring Boot入门教程2-1、使用Spring Boot+MyBatis访问数据库(CURD)注解版
一.前言 什么是MyBatis?MyBatis是目前Java平台最为流行的ORM框架https://baike.baidu.com/item/MyBatis/2824918 本篇开发环境1.操作系统: ...
- [转] 关于VS中区分debug与release,32位与64位编译的宏定义
在vs编程中,常常涉及到32位和64位程序的编译,怎么判断当前编译是32位编译还是64位编译?如何判断是debug下编译还是release下编译?因为之前用到,这里记录一下,省的忘了又要疯狂的goog ...
- Java 线程锁机制 -Synchronized Lock 互斥锁 读写锁
(1)synchronized 是互斥锁: (2)ReentrantLock 顾名思义 :可重入锁 (3)ReadWriteLock :读写锁 读写锁特点: a)多个读者可以同时进行读b)写者必须互斥 ...
- java之静态属性和静态方法
前言 静态属性和方法必须用static修饰符 静态属性和非静态属性的区别: 1.在内存中存放位置不同 所有带static修饰符的属性或者方法都存放在内存中的方法区 而非静态属性存放在内存中的堆区 ...
- EM算法的直观描述
解决含有隐变量的问题有三种方法,其中第三种方法就是通常所说的em算法.下面以统计学习方法中给出的三硬币问题为例来分别描述这三种方法.(a,b,c三硬币抛出来为正的概率分别为pai,p,q,每轮抛硬币先 ...
- 听翁恺老师mooc笔记(1)--为何选择学习C
知识点1:众多编程语言,为何选择C? 现在我们的同学喜欢java,也参加很多java的培训班,java是比较热门,但是C语言在工业界依然有重要的地位,在很多领域无可替代,几乎所有和硬件打交道的地方都得 ...
- 实验MyOD
实验MyOD 编写MyOD.java 用java MyOD XXX实现Linux下od -tx -tc XXX的功能 提交测试代码和运行结果截图,加上学号水印,提交码云代码链接. 代码如下: (刚开始 ...
- 个人作业2:QQ音乐APP案例分析
APP案例分析 QQ音乐 选择理由:毕竟作为QQ音乐九年的资深老用户以及音乐爱好者 第一部分 调研 1.第一次上手的体验 我算是很早期的QQ音乐的用户,用QQ音乐七八年,除了体验各方面还不错之外 ...
- GitChat招募IT类写作作者
GitChat是一个移动端的IT知识.技术分享平台,于2017.10和CSDN合并,成为其旗下独立品牌. 我们正在寻求有互联网基因的人来一起分享IT人员的关切,诚挚邀请您来做一次分享(让IT类文章变现 ...
- C 函数指针与回调函数
函数指针是指向函数的指针变量. 通常我们说的指针变量是指向一个整型.字符型或数组等变量,而函数指针是指向函数. 函数指针可以像一般函数一样,用于调用函数.传递参数. 函数指针变量的声明: #inclu ...