HumanBreak/docs/guide/system.md
2024-02-08 16:50:58 +08:00

7.7 KiB
Raw Blame History

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.PluginMota.Package

Mota.Plugin

它用于获取与注册插件。对于获取,依然是requirerequireAll两个函数:

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 全量复写会覆盖之前的addfront模式复写!!! :::

样板插件中已经自带了很多复写案例,可以查看插件来获取具体用法。

进程分离

新样板中对进程进行了分离,分为了渲染进程与游戏进程。

对于渲染进程,任何不会直接或间接影响到录像验证的内容都会放在渲染进程。在录像验证与编辑器中,渲染进程都不会执行,因此也无法获取渲染进程的任何插件与系统接口,同样所有的第三方库也无法获取。

对于游戏进程,不论在什么情况下都会执行,因此任何会影响录像的内容都应该放在游戏进程。对于构建版样板(即群文件版),你所编写的代码一般都是直接在游戏进程下执行的。因此,为了能够在游戏进程下能够执行渲染进程的内容,新样板提供了下面两个接口:

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();

游戏加载流程

新样板对游戏加载流程进行了部分改动,现在它的加载流程如下:

  1. 加载project/data.js
  2. 加载游戏进程,同时注册游戏进程插件
  3. 加载渲染进程,同时注册渲染进程插件与第三方库
  4. 加载plugins.js,执行其包含的每一个函数
  5. 初始化core,开始加载游戏资源
  6. 初始化插件
  7. 等待游戏资源加载完毕

因此,如果直接在plugins.js的函数中直接调用core会报错。同时插件在初始化之前获取也会报错,即使没有初始化函数,因此plugins.js顶层函数中只能获取系统接口与第三方库,不能获取插件。