7.7 KiB
lang |
---|
zh-CN |
系统说明
相比于旧样板中core
包揽一切,新样板使用了Mota
作为系统主接口,通过模块化管理,让 API 结构更加清晰。
Mota
Mota
作为新样板的主接口,一共包含三个功能函数,三个接口函数,以及两个接口对象。对于Mota
对象的具体内容,请参考 API 列表。
获取样板接口
你可以通过以下两个函数获取样板的接口。
function require(type: 'var' | 'fn' | 'class' | 'module', key: string): any;
function requireAll(type: 'var' | 'fn' | 'class' | 'module'): any;
在这里,type
描述的是要获取的接口类型,可以填写下面这四个内容:
var
: 获取变量fn
: 获取函数class
: 获取类module
: 获取模块
对于Mota.require
函数,还需要接受第二个参数,表示的是要获取的接口名称。
::: warning 对于不存在的接口,获取之后会直接报错! :::
示例
// ----- Mota.require
const getHeroStatusOn = Mota.require('fn', 'getHeroStatusOn'); // 获取勇士真实属性的接口
const KeyCode = Mota.require('var', 'KeyCode'); // 获取KeyCode接口
const DamageEnemy = Mota.require('class', 'DamageEnemy'); // 获取怪物接口
const Damage = Mota.require('module', 'Damage');
// 对于模块,每个模块都是对象,因此还可以使用对象解构语法
const { calDamageWith, getSingleEnemy } = Mota.require('module', 'Damage');
// ----- Mota.requireAll
// 它会获取到一种类型的所有接口,因此搭配对象解构语法最为合适
const { hook, KeyCode, loading, mainSetting } = Mota.requireAll('var');
// 它等价于
const hook = Mota.require('var', 'hook');
const KeyCode = Mota.require('var', 'KeyCode');
const loading = Mota.require('var', 'loading');
const mainSetting = Mota.require('vaar', 'mainSetting');
插件接口与第三方库接口
与获取样板接口类似,插件和第三方库也有对应的接口,它们分别是Mota.Plugin
和Mota.Package
。
Mota.Plugin
它用于获取与注册插件。对于获取,依然是require
与requireAll
两个函数:
function require(plugin: string): any;
function requireAll(): any;
与系统接口不同的是,插件不再拥有type
参数,直接填入插件名称或者获取全部即可。
示例
// 注意,这里的 _r 后缀表示该插件定义于渲染进程,_g 后缀表示定义于游戏进程
// 对于渲染进程与游戏进程的信息请查看本页的`进程分离`栏
const shadow = Mota.Plugin.require('shadow_r'); // 获取点光源插件
const remainEnemy = Mota.Plugin.require('remainEnemy_g'); // 获取漏怪检测插件
// 同样,你也可以使用requireAll
const { shadow_r: shadow, remainEnemy_g: remainEnemy } = Mota.Plugin.requireAll();
::: warning 与系统接口同样,获取不存在的插件时会报错 :::
Mota.Package
它与插件接口的用法完全相同,这里不再赘述。
注册插件
你可以使用register
函数来注册一个插件
function register<K extends string, D>(plugin: K, data: D, init?: (plugin: K, data: D) => void): void
function register<K extends string>(plugin: K, init: (plugin: K) => any): void
这个函数共有上述两种用法,对于第一种用法,第一个参数传入插件名称,第二个参数传入插件暴露出的内容,例如可以是一系列函数、类、变量等。第三参数可选,是插件的初始化函数,初始化函数会在游戏加载中初始化完毕。初始化函数共有两个参数,接受plugin
插件名,和data
插件内容。
对于第二种用法,第一个参数传入插件名称,第二个参数传入插件的初始化函数,同时要求必须有返回值,返回值作为插件的内容。初始化函数接受plugin
插件名作为参数。
如果你使用了第二种用法,同时没有返回值,那么如果通过require
获取插件,会获取到undefined
。
插件加载流程见游戏加载流程
示例
::: code-group
let cnt = 0;
function init() {
core.registerAnimationFrame('example', true, () => {
cnt++;
})
}
function getCnt() {
return cnt;
}
// 注册插件,内容使用对象内容简写语法,将 init 和 getCnt 函数作为插件内容,
// init 作为初始化函数
Mota.Plugin.register('example', { init, getCnt }, init);
function init() {
let cnt = 0;
// 直接在这里初始化即可
core.registerAnimationFrame('example', true, () => {
cnt++;
})
function getCnt() {
return cnt;
}
return { getCnt };
}
// 注册插件
Mota.Plugin.register('example', init);
:::
函数复写
新样板还提供了函数复写的接口,它是Mota.rewrite
:
// 类型已经经过了大幅简化
function rewrite(
base: any,
key: string,
type: 'full' | 'add' | 'front',
re: Function,
bind?: any,
rebind?: any
): Function
这个函数共有六个参数,后两个参数可选。
base
: 函数所在对象key
: 函数在对象中的名称type
: 复写类型,full
表示全量复写,add
表示在原函数之后新增内容,front
表示在原函数之前新增内容re
: 复写函数,在add
模式下,第一个参数会变成原函数的返回值,同时原参数会向后平移一位bind
: 原函数调用对象,即原函数的this
指向,默认是base
rebind
: 复写函数的调用对象,即复写函数中的this
指向,默认是base
::: warning
全量复写会覆盖之前的add
与front
模式复写!!!
:::
样板插件中已经自带了很多复写案例,可以查看插件来获取具体用法。
进程分离
新样板中对进程进行了分离,分为了渲染进程与游戏进程。
对于渲染进程,任何不会直接或间接影响到录像验证的内容都会放在渲染进程。在录像验证与编辑器中,渲染进程都不会执行,因此也无法获取渲染进程的任何插件与系统接口,同样所有的第三方库也无法获取。
对于游戏进程,不论在什么情况下都会执行,因此任何会影响录像的内容都应该放在游戏进程。对于构建版样板(即群文件版),你所编写的代码一般都是直接在游戏进程下执行的。因此,为了能够在游戏进程下能够执行渲染进程的内容,新样板提供了下面两个接口:
function r(fn: Function): void
function rf(fn: Function): Function
对于r
函数,它是将传入的函数在渲染进程下执行,对于rf
函数,它是将传入的函数包裹为渲染进程函数并输出,之后直接调用即可。
示例
Mota.r(() => {
const mainSetting = Mota.require('var', 'mainSetting');
mainSetting.setValue('screen.fontSize', 20);
});
// 它等价于
const f = Mota.rf(() => {
const mainSetting = Mota.require('var', 'mainSetting');
mainSetting.setValue('screen.fontSize', 20);
});
f();
游戏加载流程
新样板对游戏加载流程进行了部分改动,现在它的加载流程如下:
- 加载
project/data.js
- 加载游戏进程,同时注册游戏进程插件
- 加载渲染进程,同时注册渲染进程插件与第三方库
- 加载
plugins.js
,执行其包含的每一个函数 - 初始化
core
,开始加载游戏资源 - 初始化插件
- 等待游戏资源加载完毕
因此,如果直接在plugins.js
的函数中直接调用core
会报错。同时插件在初始化之前获取也会报错,即使没有初始化函数,因此plugins.js
顶层函数中只能获取系统接口与第三方库,不能获取插件。