diff --git a/LANGS.md b/LANGS.md new file mode 100644 index 0000000..1d7c5ae --- /dev/null +++ b/LANGS.md @@ -0,0 +1,2 @@ +* [English](en_us/) +* [简体中文](zh_cn/) \ No newline at end of file diff --git a/README.md b/README.md new file mode 100644 index 0000000..56df9cd --- /dev/null +++ b/README.md @@ -0,0 +1,29 @@ +--- +description: Welcome to the public TrMenu documentation +--- + +# First Page +> This Wiki is following the original [Wiki](https://trmenu.trixey.cc/) with some new features and points. + +## Introduction + +TrMenu is an advanced dynamic menu system developed based on [TabooLib](https://docs.tabooproject.org/) and published on **2019-10-4** + + + +* Unique efficient, readable & visual configuration experience +* Powerful, flexible and expandable +* Open source code + + + +Up to now, TrMenu has gone through three major version iterations + +This document is only for TrMenu v3.x (old versions are not supported and are strongly not recommended) + +## Note + +This document may be more technical description, beginners can try to view the following **community documents** to facilitate understanding and get started + +{% embed url="https://wiki.ptms.ink/index.php?title=%E7%A4%BE%E5%8C%BA:TrMenu" caption="TrMenu community tutorial" %} + diff --git a/SUMMARY.md b/SUMMARY.md new file mode 100644 index 0000000..f6c60cd --- /dev/null +++ b/SUMMARY.md @@ -0,0 +1,71 @@ +# Table of contents + +* [First Page](README.md) +* [Start](start/README.md) + * [Purchase](start/purchase.md) + * [Install](start/install.md) + * [Configuration](start/configurate.md) +* [Migrate](migrate.md) + +## MENU + +* [Create](menu/create.md) +* [Configuration](menu/configuration/README.md) + * [Title](menu/configuration/title.md) + * [Layout](menu/configuration/layout.md) + * [Option](menu/configuration/option.md) + * [Bindings](menu/configuration/bindings.md) + * [Event](menu/configuration/event.md) + * [Built-in function](menu/configuration/internal-functions.md) + * [Tasks](menu/configuration/tasks.md) +* [Icon](menu/icon/README.md) + * [Settings](menu/icon/settings.md) + * [Display](menu/icon/display/README.md) + * [Material](menu/icon/display/material.md) + * [Name](menu/icon/display/name.md) + * [Lore](menu/icon/display/lore.md) + * [Property](menu/icon/display/property.md) + * [Handler](menu/icon/handler.md) +* [Action](menu/action/README.md) + * [Type](menu/action/types.md) + * [Option](menu/action/option.md) + * [Reaction \(action group\)](menu/action/reactions.md) +* [Condition](menu/condition.md) + +## DATABASE + +* [LOCAL](database/local.md) +* [MONGODB](database/mongodb.md) +* [Custom](database/custom.md) + +## SCRIPT + +* [Kether](script/kether.md) +* [JavaScript](script/javascript.md) + +## USAGE + +* [Command](usage/command.md) +* [Arguments](usage/arguments.md) +* [Item Matchers](usage/item-matchers.md) +* [Functions](usage/functions.md) +* [Shortcuts](usage/shortcuts.md) +* [Command register](usage/register-commands.md) +* [Node Variable](usage/node-getter.md) + +## HOOK + +* [PAPI Variables](hook/placeholders.md) +* [Support Plugins](hook/hookplugins.md) + +## DEVELOP + +* [API](develop/api/README.md) + * [Events](develop/api/events.md) + * [Receptacle](develop/api/receptacle.md) +* [Example](develop/example.md) + +## appendix + +* [Upgrade Wizard](appendix/v3-guide.md) + diff --git a/appendix/v3-guide.md b/appendix/v3-guide.md new file mode 100644 index 0000000..a92919c --- /dev/null +++ b/appendix/v3-guide.md @@ -0,0 +1,145 @@ +--- +description: This document is taking you to quickly understand TrMenu 3.0 and upgrade steps +--- + +# Upgrade Wizard + +## Update Content + +### Bottom Layer + +* Configuration file & language file format are all redesigned +* 90% of the code is completely rewritten, aiming at the ultimate performance experience +* The virtual container structure of the package is completely remade, faster & more flexible + +#### Function + +* Added `Free-Slot` item to unlock items in the target slot (unstable) + +#### Material + +```text +head:%player_name% head:BlackSky +head:eyJ0ZXh0dXJlcyI6eyJTS0lOIjp7InVybCI6Imh0dHA6Ly90ZXh0dXJlcy5taW5lY3JhZnQubmV0L3RleHR1cmUvNDRmNDUyZDk5OGVhYmFjNDY0MmM2YjBmZTVhOGY0ZTJlNjczZWRjYWUyYTZkZmQ5ZTZhMmU4NmU3ODZlZGFjMCJ9fX0= +head:44f452d998eabac4642c6b0fe5a8f4e2e673edcae2a6dfd9e6a2e86e786edac0 + +repo:myCustomItem +repo:%custom_variable_whichreturnstheid% + +source:HDB:random +source:ORAXEN:itemId + +{json Content} +``` + +* Special items are no longer needed { } 、<> etc. +* HeadDatabase / Oraxen / **ItemsAdder**\(新\) ****Other item hooks need to be in ItemSource format +* SkinsRestorer hook and heads of custom materials are all integrated into the player’s head, and will be automatically detected and processed + +#### Condition + +Conditional detection is the biggest change part of this update + +Now the condition abandoned the "smart expression" provided by TrMenu 2,and only supports **Kether** and **JavaScript** statements. Kether statement is used by default + +* **Kether** + * Kether official documentation will provide instructions for use +* **JavaScript** + * Js prefix is required, for example `condition: 'js: player.getHealth() > 20.0'` + * Full cache, does not support dynamic compilation of variables, use of variables requires a set of functions + * **Objects** is provided by default + * `player` + * `bukkitServer` + * `utils` + * **Function** is provided by default + * `vars("STRING")` Compile variables and return String + * `varInt("STRING")` Compile variable, return integer + +> Reasons for deprecating v2: +> +> ```text +> 1. Bloated and inflexible, many errors caused by imperfections +> 2. No standard caching, variable processing +> ``` + +#### Function + +{% page-ref page="../usage/functions.md" %} + +#### Action + +* **ActionBound** The \_\|\|\_ left over from v1 is discarded, and at this stage, it is unified as `&&&` +* **Title** The unified specification format of the action is + * `title: [Title] [Subtitle] [Fade In] [Stay] [Fade Out]` + * Title text application that contains spaces · Enclose +* **Refresh** Actions can now specify icon slots to refresh a single icon +* **ActionOption** Action parameters are now recommended to use the format definition of `{[TYPE]=[VALUE]}` +* **InputCatcher** The standard parameters of the catcher now include `type`, `start`, `cancel`, `end`. The special parameters are `display`, `itemLeft`, `itemRight`, `content` & add BOOK type catcher + +#### Data + +* Meta & Data function module is rewritten, and GlobalData is added to set global variables +* Can be called by function + +#### Command + +* Item repo and item conversion are combined into one command +* Sound preview function is rewritten based on ReceptacleAPI +* Rewrite the performance loss statistics function +* Menu list formatting +* Debug function rewrite Mirror & Dump function integrated under Debug + +{% page-ref page="../usage/command.md" %} + +#### API + +* Provide `ReceptacleAPI` to create custom virtual containers, use cases can refer to `me.arasple.mc.trmenu.module.internal.command.impl.CommandSounds` + +## Start + +Because the configuration & language structure of v3.x and v2.x are completely incompatible, the menu is partially compatible + +Therefore, in order to better complete the upgrade, please rename the old TrMenu folder for future use. + +Then replace the plugin to generate a new TrMenu folder + +## Configuration + +Quickly understand the configuration structure according to this Wiki & configure as needed + +## Menu + +Point of incompatibility + +* Built-in functions & conditions & scripts +* Part of the hook material writing +* \(v1.x \) TITLE action writing +* \(v1.x\) Action option binding `_||_` + +Solution + +* Make sure to read the new version of the document completely +* Rewrite conditions according to the new format (for JavaScript, string related expressions need to be processed) + +## Q & A + +#### Why not provide one-click migrate + +This update discards most of the bloated design of v2 & rewrites almost all modules + +Especially when it comes to scripts and conditions, the migration function is not easy to do + +If you only use some simple conditions or JavaScript forms, it is relatively easier to upgrade + +For example, application replacement function `hasPerm.` -> `perm *` + +#### Is it necessary to upgrade + +For a large-scale server that is running and a large menu volume, it is not necessary in the short term + +New servers or menus with low demand (less conditional expressions) are recommended to upgrade as soon as possible + +The old version has ceased maintenance and support, and no longer deals with remaining issues + + + diff --git a/database/custom.md b/database/custom.md new file mode 100644 index 0000000..69a6fe8 --- /dev/null +++ b/database/custom.md @@ -0,0 +1,3 @@ +# Custom + +Refer to: https://trmenu.docs.insinuate.cn/develop/api/events#customdatabaseevent \ No newline at end of file diff --git a/database/local.md b/database/local.md new file mode 100644 index 0000000..a4de16e --- /dev/null +++ b/database/local.md @@ -0,0 +1,18 @@ +--- +description: Local database +--- + +# LOCAL + +## Configuration + +```yaml +Database: + Method: LOCAL +``` + +## Located + +```text +./plugins/TrMenu/data/data.db +``` \ No newline at end of file diff --git a/database/mongodb.md b/database/mongodb.md new file mode 100644 index 0000000..54b3446 --- /dev/null +++ b/database/mongodb.md @@ -0,0 +1,16 @@ +--- +description: MongoDB database +--- + +# MONGODB + +## Configuration + +```yaml +Database: + Method: MONGODB + Url: + Client: 'mongodb://localhost:3307' + Database: trixey + Collection: menu +``` \ No newline at end of file diff --git a/develop/api/README.md b/develop/api/README.md new file mode 100644 index 0000000..7ff0182 --- /dev/null +++ b/develop/api/README.md @@ -0,0 +1,31 @@ +# API + +Maven: +```xml + + + roselle-public + https://repo.iroselle.com/repository/maven-public/ + + + + + + me.arasple + trmenu + {LATEST-VERSION} + provided + + +``` + +Gradle Kotlin DSL +```kotlin +repositories { + maven("https://repo.iroselle.com/repository/maven-public/") +} +dependencies { + compileOnly("me.arasple:TrMenu:{LATEST-VERSION}") +} + +``` \ No newline at end of file diff --git a/develop/api/events.md b/develop/api/events.md new file mode 100644 index 0000000..6909c2e --- /dev/null +++ b/develop/api/events.md @@ -0,0 +1,94 @@ +--- +description: TrMenu built-in events +--- + +# Events + +## CustomDatabaseEvent +* Custom Database +```kotlin +package me.arasple.mc.trmenu.api.event + +import me.arasple.mc.trmenu.module.internal.database.Database +import taboolib.platform.type.BukkitProxyEvent + +/** + * @Author sky + * @Since 2020-08-14 14:52 + */ +class CustomDatabaseEvent(val name: String, var database: Database? = null) : BukkitProxyEvent() { + + override val allowCancelled: Boolean + get() = false +} +``` + + +## MenuOpenEvent +* Menu Open Event +```kotlin +package me.arasple.mc.trmenu.api.event + +import me.arasple.mc.trmenu.module.display.Menu +import me.arasple.mc.trmenu.module.display.MenuSession +import taboolib.platform.type.BukkitProxyEvent + +/** + * @author Arasple + * @date 2021/1/29 17:34 + */ +class MenuOpenEvent(val session: MenuSession, val menu: Menu, val page: Int, val reason: Reason = Reason.UNKNOWN) : + BukkitProxyEvent() { + + enum class Reason { + + RELOAD, + + PLAYER_COMMAND, + + CONSOLE, + + BINDING_COMMANDS, + + BINDING_ITEMS, + + BINDING_SHORTCUT, + + UNKNOWN + + } + +} +``` + + +## MenuPageChangeEvent +* Menu Page Change Event +```kotlin +package me.arasple.mc.trmenu.api.event + +import me.arasple.mc.trmenu.module.display.MenuSession +import taboolib.platform.type.BukkitProxyEvent + +/** + * @author Arasple + * @date 2021/2/4 11:33 + */ +class MenuPageChangeEvent(val session: MenuSession, val from: Int, val to: Int, val override: Boolean) : BukkitProxyEvent() +``` + + +## MenuCloseEvent +* Menu Close Event +```kotlin +package me.arasple.mc.trmenu.api.event + +import me.arasple.mc.trmenu.module.display.MenuSession +import taboolib.platform.type.BukkitProxyEvent + +/** + * @author Arasple + * @date 2021/1/29 17:34 + */ +class MenuCloseEvent(val session: MenuSession) : BukkitProxyEvent() +``` \ No newline at end of file diff --git a/develop/api/receptacle.md b/develop/api/receptacle.md new file mode 100644 index 0000000..337d7b0 --- /dev/null +++ b/develop/api/receptacle.md @@ -0,0 +1,3 @@ +# Receptacle + +> This API has been moved to TabooLib 6, so please focus on the [ReceptacleAPI](https://docs.tabooproject.org/plugin/module/ui-receptacle/index.html) document of TabooLib Docs. \ No newline at end of file diff --git a/develop/example.md b/develop/example.md new file mode 100644 index 0000000..db08a62 --- /dev/null +++ b/develop/example.md @@ -0,0 +1,23 @@ +# Example + +> The following code is demonstrated in accordance with the TabooLib plugin method +```kotlin +package com.github.username + +import taboolib.common.platform.event.SubscribeEvent + +/** + * @project Example + * + * @author Score2 + * @since 2021/10/23 22:35 + */ +object Demo { + + @SubscribeEvent + fun e(e: MenuOpenEvent) { + println("玩家: ${e.session.viewer.name} 打开了菜单: ${e.menu.id}") + } + +} +``` \ No newline at end of file diff --git a/hook/hookplugins.md b/hook/hookplugins.md new file mode 100644 index 0000000..6dfb4ea --- /dev/null +++ b/hook/hookplugins.md @@ -0,0 +1,13 @@ +# Support Plugins + +| Plugin | Use for | +| :--- | :--- | +| HeadDatabase | Item material | +| Oraxen | Item material | +| PlayerPoints | Actions and variables | +| SkinsRestorer | Custom player skull textures | +| ItemsAdder | Item material | +| Floodgate | Script util | +| Vault | Actions and variables | +| FastScript | Custom script (not currently supported) | +| Zaphkiel | Item material | \ No newline at end of file diff --git a/hook/placeholders.md b/hook/placeholders.md new file mode 100644 index 0000000..9fc3d52 --- /dev/null +++ b/hook/placeholders.md @@ -0,0 +1,34 @@ +# PAPI Variables + +## Total Menu + +> %trmenu\_menus% + +## Pass Argument + +> %trmenu\_args\_<Index>% + +## Metadata Data + +> %trmenu\_meta\_<key> +> +> %trmenu\_data\_<key>% + +## Menu Data + +> %trmenu\_menu\_page% \# The page number currently being viewed +> +> %trmenu\_menu\_pages% \# The total number of pages in the menu +> +> %trmenu\_menu\_next% \# Next page number +> +> %trmenu\_menu\_prev% \# Previous page number +> +> %trmenu\_menu\_title% \# Title of the currently menu + +## JavaScript Operation + +> %trmenu\_js\_<Context>% + + + diff --git a/menu/action/README.md b/menu/action/README.md new file mode 100644 index 0000000..c201240 --- /dev/null +++ b/menu/action/README.md @@ -0,0 +1,2 @@ +# 动作 + diff --git a/menu/action/option.md b/menu/action/option.md new file mode 100644 index 0000000..bd8a065 --- /dev/null +++ b/menu/action/option.md @@ -0,0 +1,45 @@ +--- +description: 单条动作支持的参数 +--- + +# 参数 + +## 延时 + +> {delay=\[TICKS\]} + +```yaml +# 延时 1sec 发送消息 +- 'tell: Delayed Message {delay=20}' +``` + +## 概率 + +> {chance=\[rate\]} + +```yaml +- 'give-item: material:DIAMOND {chance=0.8}' +``` + +## 条件 + +> {condition=<Expression>} + +```yaml +- 'tell: VIP User Message {condition=perm *user.vip}' +``` + +## 遍历 + +> {players=<Expression>} + +```yaml +- 'tell: A Broadcast Message {players}' + +- 'tell: An admin broadcast message {players: perm *admin}' +``` + +* 循环为所有满足条件的在线玩家执行该动作 + + + diff --git a/menu/action/reactions.md b/menu/action/reactions.md new file mode 100644 index 0000000..f1c4d32 --- /dev/null +++ b/menu/action/reactions.md @@ -0,0 +1,35 @@ +# 反应 \(动作组\) + +## 常规 + +```yaml +actions: + - 'tell: Hello, %player_name%' +``` + +```yaml +actions: 'tell: Hello, %player_name%' +``` + +## 三元动作组 + +```yaml + actions: + all: + - condition: 'js: utils.chance(0.1)' + actions: + - 'tell: Prize 1' + - 'return' + - condition: 'js: utils.chance(0.5)' + actions: + - 'tell: Prize 2' + - 'return' + - condition: '' + actions: + - 'tell: Prize 3' +``` + +## 注意 + +* 动作做的执行结果取决于是否有 return 动作返回,若有返回则为 false + diff --git a/menu/action/types.md b/menu/action/types.md new file mode 100644 index 0000000..54f2ac4 --- /dev/null +++ b/menu/action/types.md @@ -0,0 +1,367 @@ +# 类型 + +## 发送文本 + +> 向玩家发送一条普通消息 + +```yaml +- 'tell: Hello There' +``` + +```yaml +- 'tell: Hello There;There is a message' +- 'tell: First Line\nSecond Line!' +``` + +## 强制聊天 + +> 让玩家发送一条消息到聊天框 + +```yaml +- 'chat: Hello Everyone!' +``` + +## 发送音效 + +> 向玩家播放目标音效 + +```yaml +- 'sound: BLOCK_ANVIL_HIT-1-2' +``` + +```yaml +- 'sound: BLOCK_BEEHIVE_ENTER-1-2;BLOCK_BUBBLE_COLUMN_WHIRLPOOL_INSIDE-1-0' +``` + +## 发送 Title + +> 向玩家展示一条 Title 信息 + +```yaml +# 用 \s 代替标题中需要使用的空格 +- 'title: WELCOME %player_name%~ 20 20 10' + +# 用 ` 括起内容,支持直接用空格 +- 'title: `&c&lPermission Required` `&7&lYou need permission &6<rmenu.use &7<o open this menu` 15 20 15' +``` + +* TITLE 后对应的三个整数型参数分别为 FadeIn, Stay, FadeOut \(渐入、停留、渐出\) 的时间,可以不设置 + +## 发送 Actionbar + +> 向玩家展示一条 Actionbar 信息 + +```yaml +- 'actionbar: Hello There' +``` + +## 发送 Tellraw + +> 向玩家发送一条原版 Json 消息,或利用 TrMenu 轻松构建一条 + +{% tabs %} +{% tab title="原版 JSON" %} +```yaml +- 'json: {"text":"Hello World!"}' +``` +{% endtab %} + +{% tab title="构建消息 1" %} +```yaml +- 'json: Hello World! <--- Click That' +``` +{% endtab %} + +{% tab title="构建消息 2" %} +``` +- 'json: &3Hello, &b%player_name%&3, Buy Ranks on our <&2&nstore@url=https://store.example.net> &3site.' +``` +{% endtab %} +{% endtabs %} + +## 执行命令 + +> 调用对象执行命令 + +### 玩家身份 + +```yaml +- 'command: spawn' +``` + +```yaml +# 以 ; 分隔,可以执行多个命令 +# 下方动作等同于 +# - 'command: spawn' +# - 'command: say I am Back' + +- 'command: spawn ; say I am Back' +``` + +### Operator 身份 + +```yaml +- 'op: kick {meta:input}' +``` + +```yaml +- 'op: kick {meta:input} ; broadcast I just kicked {meta:input}' +``` + +{% hint style="danger" %} +实现以 Op 执行命令的方法是 **授予 OP,执行操作,撤销 OP** + +在可以用 **CONSOLE** 控制台执行命令实现需求的情况下,**不推荐**使用此动作 +{% endhint %} + +### Console 控制台 + +```yaml +- 'console: lp perm add %player_name% vip.user; give %player_name% diamond 64' +``` + +## 跨服传送 + +> BungeeCord 群组下,跨服传送玩家到目标服务器 + +```yaml +- 'server: Lobby' +``` + +## 菜单 + +### 关闭菜单 + +```yaml +- 'close' +``` + +### 关闭菜单 \(静默\) + +> 通过此形式关闭菜单,将不会执行 Events.Close 的动作组 + +```yaml +- 'force-close' +``` + +### 开启菜单 + +```yaml +- 'open: Example' +``` + +```yaml +# 打开菜单 Shop-Handler, +# 指定参数为 [APPLE, 10, 1] + +- 'open: Shop-Handler APPLE 10 1' +``` + +```yaml +# 打开菜单 Browser 的指定页码 3 + +- 'open: Browser:3' +``` + +### 开启菜单 \(静默\) + +> 通过此形式开启菜单,将不会执行 Events.Open 的动作组 + +```yaml +- 'force-open: :[Page]' +``` + +### 切换页码 + +```yaml +- 'page: 0' +``` + +```yaml +- 'page: %trmenu_menu_next%' +``` + +### 设置容器标题 + +```yaml +- 'set-title: CUSTOM_TITLE' +``` + +* 会被动态标题覆盖 + +### 设置菜单参数 + +```yaml +- 'set-arguments: Custom\sArgument 10 Hello' +``` + +### 刷新图标 + +```yaml +# 刷新菜单所有活动图标 +- 'refresh' + +# 刷新菜单 ID 为 Bee 图标 +- 'refresh: Bee' + +# 刷新菜单 ID 为 A, B, C 的三个图标 +- 'refresh: A;B;C' +``` + +* 这里的 “刷新” 是重新计算筛选子图标,而非更新动画 + +### 更新图标 + +> 使用方法同上,能够更新指定图标 Lore、名称 等显示属性 & 函数变量 + +```yaml +- 'update: Bee' +``` + +### 重置菜单 + +> 重置所有图标的动画属性缓存 + +```yaml +- 'reset' +``` + +### 数据操作 + +> TrMenu 所提供的数据功能分三类 +> +> Meta - 临时数据,关服后不会存储 +> +> Data - 存储数据(可通过配置 TabooLib 实现跨服) +> +> GlobalData - 全局本地文件存储的数据,不针对玩家 + +* 操作类型 SET \[key\] \[value\]、DEL \[key\] +* 下面以 Meta 为例 + +```yaml +- 'set-meta: customMeta customValue' +- 'set-meta: customMeta customValue; customKey2 customValue 2' + +- 'del-meta: customMeta' +- 'del-meta: customMeta;customKey2' + +- 'del-meta: (?i)customs?-?(meta|key)' +``` + +## 功能 + +### 扣除和给予物品 + +```yaml +# 扣除一组钻石 +- 'take-item: material:DIAMOND,amount:64' + +# 扣除一组铁锭和半组金锭 +- 'take-item: material:IRON_INGOT,amount:64;material:GOLD_INGOT,amount:32' +``` + +```yaml +# 给予物品 +- 'give-item: material:DIAMOND,amount:32,name:&bCustom Diamond' +``` + +### 控制经济货币 + +```yaml +- 'give-money: %server_time_s%' +- 'give-money: 100' +``` + +```yaml +- 'take-money: 100' +``` + +```yaml +- 'set-money: 500' +``` + +* Money 控制仅支持和 Vault 挂钩的经济系统 +* 除 Money 以外,还支持 Points (挂钩 PlayerPoints),操作同理,这里不再赘述 + +### 捕获器 + +> TrMenu 提供的强大的捕获器功能 & 实现可输入 GUI 的设计 + +#### 类型 + +> CHAT 聊天框 +> +> SIGN 告示牌 +> +> ANVIL 铁砧 +> +> BOOK 书 + +#### 变量 + +> {meta: input} 最后一次输入的内容 + +#### 配置 + +```yaml +- catcher: + amount: + type: SIGN + start: 'actionbar: &b&lPlease input a valid number' + cancel: 'tell: &8Cancelled' + end: + - 'set-args: {0} {1} {2} `${js: Math.min(Math.max(varInt("{meta:input}"), 1), 64)}`' + - 'menu: Shop-Handler-Purchase' +``` + +## 逻辑 + +### 延时 + +> 延时执行该动作之后的所有动作 + +```yaml +- 'tell: The diamond reward will be distributed in 3 seconds' +- 'delay: 60' +- 'give-item: material:DIAMOND' +``` + +### 中断 + +> 中断之后所有动作的执行,并使该反应组结果为 FALSE + +```yaml +- 'return' +``` + +## 脚本 + +### JavaScript 脚本执行 + +> 预编译执行 JavaScript 脚本 + +```yaml +- 'js: player.sendMessage("Hello World!")' +``` + +### Kether 脚本执行 + +```yaml +- 'ke: tell *"Hello World!"' +- 'tell color *"Hello &bTrMenu"' +``` + +* 我们对 Kether 语法完整支持,且强烈推荐使用,若不指定 TrMenu 动作名称 +* 将默认采用 Kether 脚本执行!如下 + +```yaml +Events: + Open: |- + if perm *trmenu.use then { + tell *"You have permission" + } else { + tell *"You dont have permission" + } +``` + diff --git a/menu/chuang-jian.md b/menu/chuang-jian.md new file mode 100644 index 0000000..cb16cd9 --- /dev/null +++ b/menu/chuang-jian.md @@ -0,0 +1,7 @@ +--- +description: Create a menu +--- + +# Create + +Format diff --git a/menu/condition.md b/menu/condition.md new file mode 100644 index 0000000..f3aad15 --- /dev/null +++ b/menu/condition.md @@ -0,0 +1,35 @@ +# Condition + +## Introduce + +"Conditional expression" is a single string that can be used in the following scenarios + +* Action arguments +* Reaction \(action group\) conditions +* 子图标筛选条件 + +The execution result should return a Boolean value \(Boolean\), namely `true` / `false` + +Currently, TrMenu supports two types of expressions, Kether and JavaScript, as conditional compilation. + +Among them, Kether is uniformly called by default \*\(recommended\) + +## Use + +Kether expressions are used directly, JavaScript expressions need to be prefixed with `js:` and standard variable processing + +For example, create a condition that only holds when the player has the permission of `vip.user` or the balance is greater than or equal to `1000` + +* Kether: `any [ perm *vip.user money *1000 ]` +* JavaSctipt: `js: player.hasPermission("vip.user") || utils.hasMoney(player, 1000.0)` + +The format of the two grammars can be further studied through the chapter `SCRIPT` + +## Advanced + +{% page-ref page="../script/kether.md" %} + +{% page-ref page="../script/javascript.md" %} + +> In-depth understanding of the structure of the two grammars and application scenarios in TrMenu to make a more powerful menu + diff --git a/menu/configuration/README.md b/menu/configuration/README.md new file mode 100644 index 0000000..4281459 --- /dev/null +++ b/menu/configuration/README.md @@ -0,0 +1,8 @@ +# 配置 + +## 引导 + +本节中将带你了解 TrMenu 菜单配置结构中的各个项目以及配置规则 + +其中大部分内容都是可选的,事实上你仅需 布局 & 图标 两个项目就可以创建一个虚拟容器 + diff --git a/menu/configuration/bindings.md b/menu/configuration/bindings.md new file mode 100644 index 0000000..b3e84f2 --- /dev/null +++ b/menu/configuration/bindings.md @@ -0,0 +1,29 @@ +--- +description: 绑定菜单到快捷命令打开 & 物品特征 +--- + +# 绑定 + +## 示例 + +```yaml +# +# 菜单绑定的快捷打开方式 +# +Bindings: + # 绑定命令, 支持正则 + Commands: + - 'tester' + # 绑定到物品特征 + Items: + - 'material:compass' +``` + +## 注意 + +* 绑定命令支持**正则匹配**,例如 添加 \(?i\) 前缀可以忽略大小写 +* 绑定命令支持**空格 & 其它参数**,插件将自动匹配并读取玩家提供的真实参数 +* 了解绑定 **物品特征** 的详细写法,请查看下面章节 + +{% page-ref page="../../usage/item-matchers.md" %} + diff --git a/menu/configuration/event.md b/menu/configuration/event.md new file mode 100644 index 0000000..a7052f4 --- /dev/null +++ b/menu/configuration/event.md @@ -0,0 +1,33 @@ +--- +description: 对菜单的一些事件执行条件动作反应 +--- + +# 事件 + +## 示例 + +```yaml +# +# 菜单的事件处理 +# +Events: + # 开启菜单 + Open: + - condition: 'perm *trmenu.use' + actions: + - 'sound: BLOCK_CHEST_OPEN-1-0' + deny: + - 'sound: ENTITY_ITEM_BREAK-1-0' + - 'title: `&c&lPermission Required` `&7&lYou need permission &6<rmenu.use &7<o open this menu` 15 20 15' + - 'return' + # 关闭菜单 + Close: + - 'sound: BLOCK_CHEST_CLOSE-1-0' + +``` + +## 注意 + +* 事件配置的对象均是反应(动作组) +* 若执行 RETURN 动作,则事件将取消 + diff --git a/menu/configuration/internal-functions.md b/menu/configuration/internal-functions.md new file mode 100644 index 0000000..0384de3 --- /dev/null +++ b/menu/configuration/internal-functions.md @@ -0,0 +1,59 @@ +--- +description: 内置 JavaScript 函数是菜单配置 Functions 节点下的自定义脚本 +--- + +# 内置函数 + +{% hint style="info" %} +使用本功能,需要有一些 JavaScript 语法基础 +{% endhint %} + +## 示例 + +```yaml +Functions: + flash: |- + function flash() { + var display = new Date().getSeconds() % 2 == 0 + return display ? args[0] : " " + } + flash() +``` + +* 在该例中,我们在 Functions 下定义了一个 `flash` 函数,本函数根据判断当前系统的时间(秒)是否为偶数,决定是返回传入参数还是一个空格 +* 该函数可以实现闪烁的动画效果,且支持自定义参数参数作为闪烁符号 + +## 注意 + +* 调用格式为 `${[funcName]_[Arg1]_[Arg2]}` 例如 ${flash\_>} +* 默认函数上方将自动添加行 `var args = new Array()` 并写入参数,因此在内置函数中传入的参数可以直接通过 args 数组的形式使用 + + + +#### 拓展例子 + +```yaml +bStats: + servers: 'vars("${bStats.query_servers_&a_&7 servers}")' + players: 'vars("${bStats.query_players_&6_&7 players}")' + menus: 'vars("${bStats.query_menus_&2_&7 menus}")' + opens: |- + function opens() { + var data = utils.query("https://bstats.org/api/v1/plugins/5742/charts/menu_open_counts/data?maxElements=1") + if (data.has()){ + return "&b" + data.asJson().getAsJsonArray().get(0).getAsJsonArray().get(1) + "&7" + } + return "&8Loading." + vars("${flash_.}") + "&7" + } + opens() + query: |- + function query() { + var data = utils.query("https://bstats.org/api/v1/plugins/5742/charts/" + args[0] + "/data?maxElements=1") + if (data.has()){ + return args[1] + data.asJson().getAsJsonArray().get(0).getAsJsonArray().get(1) + args[2] + } + return "&8Loading." + vars("${flash_.}") + } + query() +``` + diff --git a/menu/configuration/layout.md b/menu/configuration/layout.md new file mode 100644 index 0000000..199e72e --- /dev/null +++ b/menu/configuration/layout.md @@ -0,0 +1,60 @@ +--- +description: 通过布局用字符快速排版菜单,提高编辑效率和可视化 +--- + +# 布局 + +## 示例 + +```yaml +# +# 菜单的布局功能 +# 你可以用 ` ` 来标记设置多个字符名称图标的位置 +# +Layout: + - '########`Close`' + - ' ' + - ' ' + - ' ' + - '########`Next`' + +# +# 玩家容器菜单的布局 +# +PlayerInventory: + - ' ' + - ' ' + - ' ' + - ' ' +``` + +* **每个字符**代表一个图标在菜单中的位置,你也可以用 \` 将字符串包裹起来,从而不受限于单个字符 +* 字符即是图标的唯一 ID,通过布局实现可视化编辑从而不再需要为图标单独设置槽位 +* 布局的行数将同时定义箱子容器的大小 +* 玩家容器菜单布局仅限 4\*9 大小 + +## 多页 + +```yaml +Layout: + - - '########`Close`' + - ' ' + - ' ' + - ' ' + - '########`Next`' + + - - '########`Close`' + - ' ' + - ' * ' + - ' ' + - '`Pre`########' +``` + +* 布局功能可以支持多个布局,从而在同一个菜单中快速实现多页功能 +* PlayerInventory 同理对应 + +## 注意 + +* 若无 **玩家容器菜单布局** 与 **菜单布局** 对应,则在该菜单页玩家容器为空 +* 标识图标的字符支持汉字 + diff --git a/menu/configuration/option.md b/menu/configuration/option.md new file mode 100644 index 0000000..9294aac --- /dev/null +++ b/menu/configuration/option.md @@ -0,0 +1,38 @@ +--- +description: 配置菜单的一些选项属性 +--- + +# 选项 + +## 示例 + +```yaml +# +# 菜单的选项设置 +# +Options: + # 是否启用菜单传参功能 (默认开启) + Arguments: true + # 默认填充参数 + Default-Arguments: [] + # 默认布局页码 + Default-Layout: 0 + # 自由槽位 + Free-Slots: + - '71-73' + # 是否隐藏玩家容器 + Hide-Player-Inventory: true + # 防频繁点击的间隔 + Min-Click-Delay: 200 + # 强制需要依赖的 PlaceholderAPI 拓展变量 + Depend-Expansions: ['server', 'player', 'progress'] +``` + +## 注意 + +* 若设置 **默认填充参数** ,则将自动补全玩家未提供的参数,例如 + * Default-Arguments: \["TabooLib", "Virus"\] + * 玩家提供的参数为 \["TabooLib"\], 则第二个参数将自动补全为 Virus +* **默认菜单布局页码** 即在未指定页码的情况下默认打开菜单后初次显示的页码 +* **依赖PAPI拓展** 菜单开启时将检测是否安装完整,否则拒绝开启并自动请求下载 + diff --git a/menu/configuration/tasks.md b/menu/configuration/tasks.md new file mode 100644 index 0000000..ff2499a --- /dev/null +++ b/menu/configuration/tasks.md @@ -0,0 +1,29 @@ +--- +description: 可自定义的周期任务,将在打开菜单后执行,关闭菜单后取消 +--- + +# 周期任务 + +## 示例 + +```yaml +# +# 菜单自定义周期性任务 +# +Tasks: + # 任务 Id + tikTok: + # 周期 ( Ticks ) + period: 20 + # 反应 + task: + - condition: '$ sender.isOp()' + actions: + - 'sound: BLOCK_NOTE_BLOCK_BIT-1-2' +``` + +## 注意 + +* 每个周期任务由 **任务 ID、周期、反应** 三部分组成 +* 切换页码不会终止周期任务 + diff --git a/menu/configuration/title.md b/menu/configuration/title.md new file mode 100644 index 0000000..717faf0 --- /dev/null +++ b/menu/configuration/title.md @@ -0,0 +1,44 @@ +--- +description: 容器显示标题 +--- + +# 标题 + +## 示例 + +```yaml +# +# 菜单的标题, 支持动态标题周期切换 +# 标题支持配置颜色 & 函数变量 +# +Title: + - 'Hello, TrMenu!' + - 'Hello, %player_name%!' + - 'Support Animated Titles' + +# +# 若有多个标题, 动态切换的周期 (ticks) +# +Title-Update: 40 +``` + +```yaml +# +# 静态标题 +# +Title: 'TrMenu Menu' + +# +# 此项可以不用写,默认值即是 -1 +# +# Title-Update: -1 +``` + +## 注意 + +* 容器标题本身是不支持修改的,动态标题实现的原理是通过发送 `PacketWindowOpen` 包再打开一个新的容器,并重新发送容器内所有物品的数据包 +* 动态标题的周期性更新是全局菜单适用,不支持分页独立配置 +* 若需不同页面不同标题,有两种实现方式 + * 动态标题配置函数变量,对页码进行判断 + * 配置静态标题,切页后调用动作二次修改标题 + diff --git a/menu/create.md b/menu/create.md new file mode 100644 index 0000000..623e1b5 --- /dev/null +++ b/menu/create.md @@ -0,0 +1,136 @@ +# Create + +## Base + +The menu of TrMenu is configured, read and loaded in **YAML** \(.yml\) file format + +Therefore, to create a new menu, you can start by creating a new YAML file + +## Path + +The menu file can be placed in the default menu directory \(menus\) or in a custom path \(you need to configure yourself\) + +The menu file name \(excluding the extension suffix\) is the unique ID of the menu. + +Menus with duplicate IDs will still be loaded, but it will affect the normal operation of the open command + +## Configuration + +The configuration items of the menu will be analyzed in detail in the following chapters + +Please browse the sample mode provided below to quickly understand the basic structure of the menu + +```yaml +# Title displayed by the inventory +Title: 'TrMenu' + +# Refresh time of inventory title +Title-Update: 40 + +# Menu layout +Layout: [] + +# Menu layout - player inventory +PlayerInventory: [] + +# Menu options +Options: + # Whether to enable the argument + Arguments: false + # Default argument filling + Default-Arguments: [ ] + # Free slot + Free-Slots: + - 71 + - 72 + # Default page number + Default-Layout: 0 + # Whether to hide player inventory items + Hide-Player-Inventory: false + # Minimum click delay + Min-Click-Delay: 200 + # Menu dependent PlaceholderAPI extension + Depend-Expansions: [ 'server', 'player', 'progress', 'animations' ] + +# Menu binding +Bindings: + # Command + Commands: + - '(?i)example(-)?(gui)?(s)?' + # Items + Items: + - 'material:compass' + - 'material:clock,lore:OPEN_MENU' + - 'texture:eyJ0ZXh0dXJlcyI6eyJTS0lOIjp7InVybCI6Imh0dHA6Ly90ZXh0dXJlcy5taW5lY3JhZnQubmV0L3RleHR1cmUvNDRmNDUyZDk5OGVhYmFjNDY0MmM2YjBmZTVhOGY0ZTJlNjczZWRjYWUyYTZkZmQ5ZTZhMmU4NmU3ODZlZGFjMCJ9fX0=' + +# Menu event reaction +Events: + # Open event + Open: + - condition: 'perm *trmenu.use' + actions: + - 'sound: BLOCK_CHEST_OPEN-1-0' + deny: + - 'sound: ENTITY_ITEM_BREAK-1-0' + - 'title: `&c&lPermission Required` `&7&lYou need permission &6<rmenu.use &7<o open this menu` 15 20 15' + - 'return' + # Close event + Close: + - 'sound: BLOCK_CHEST_CLOSE-1-0' + +# Icon body of the menu +Icons: + # Icon Id + 'Close': + # Icon update frequency + update: [] + # Icon refresh frequency + refresh: -1 + # Display part + display: [] + # Action part + actions: [] + +# Timed task +Tasks: + # Task ID + tikTok: + # Task period (in ticks) + period: 80 + # Task response (reactions) + task: + - condition: '$ sender.isOp()' + actions: + - 'sound: BLOCK_NOTE_BLOCK_BIT-1-2' + +# Built-in custom JavaScript functions +Functions: + id: 'content' + +``` + +### Structure + +* Title + * Single or multiple titles + * Title update cycle +* Layout + * Menu layout + * Player inventory layout +* Options + * Default completion arguments + * Default layout page number + * Whether to hide player inventory items + * Free slot + * Anti-frequent click interval + * Depend PlaceholderAPI extensions +* Binding + * Binding regular commands + * Bound items +* Event + * Open the menu to perform actions + * Close the menu to perform an action +* Icon +* Built-in script +* Periodic task + diff --git a/menu/icon/README.md b/menu/icon/README.md new file mode 100644 index 0000000..c715583 --- /dev/null +++ b/menu/icon/README.md @@ -0,0 +1,32 @@ +--- +description: 无图标,不菜单 +--- + +# 图标 + +## 结构 + +#### 图标 ID + +* 更新 周期 +* 重新筛选计算子图标 周期 +* 是否继承 \* +* 条件 \* +* 优先级 \* +* 槽位 +* **显示部分** + * 材质 + * 属性 + * NBT + * 标签 + * 发光 + * 数量 + * 名称 + * 描述 +* **交互处理** + * <类型>:反应 + +\* 表示该属性为子图标设计 + +更新/刷新 周期 和 槽位是图标全局属性,不支持子图标另行设置 + diff --git a/menu/icon/display/README.md b/menu/icon/display/README.md new file mode 100644 index 0000000..d254487 --- /dev/null +++ b/menu/icon/display/README.md @@ -0,0 +1,33 @@ +--- +description: 图标的显示属性(子图标 & 默认图标 通用) +--- + +# 显示 + +## 示例 + +```yaml + # 图标 Id + 'Close': + # 显示属性更新频率 + update: [] + # 显示部分 + display: + # 材质,支持多个,动态更新 + material: '' + # 名称,支持多个,动态更新 + name: '' + # Lore 描述,支持多组,动态更新 + lore: [] + # NBT + nbt: + key: value + # 物品标签, List 形式 + flags: [] + # 物品数量 + amount: '' + # 发光效果 + shiny: 'true' + +``` + diff --git a/menu/icon/display/lore.md b/menu/icon/display/lore.md new file mode 100644 index 0000000..86d96a5 --- /dev/null +++ b/menu/icon/display/lore.md @@ -0,0 +1,37 @@ +--- +description: 物品的 Lore 描述 +--- + +# 描述 + +## 示例 + +```yaml +# 嵌套 List,多组动态 Lore +lores: + - - '&7Thanks &f:> &7for using!' + - - '&7Thanks &f:) &7for using!' + +# 静态 Lore 描述 +lore: + - 'Hello There' + - 'Hello TrMenu!' +``` + +## 条件 + +{% hint style="info" %} +此功能将在每次更新 Lore 时进行计算条件,可能影响插件性能表现 +{% endhint %} + +```yaml +lore: + - 'Line 1' + - 'Line 2' + # 当玩家有 vip.user 权限时显示此行 + - 'You are VIP User {condition=perm *vip.user}' + - 'Line 3' +``` + + + diff --git a/menu/icon/display/material.md b/menu/icon/display/material.md new file mode 100644 index 0000000..e81e627 --- /dev/null +++ b/menu/icon/display/material.md @@ -0,0 +1,115 @@ +# 材质 + +## 普通材质 + +```yaml +material: 'Red Stained Glass Pane' +``` + +```yaml +material: 'Wool:3' +``` + +* **材质实现动画的方式就是写成集合形式,将在设定周期内循环切换** +* 普通材质也支持使用函数变量 + +## 物品仓库 + +* TrMenu 内置的物品存储调用功能,通过子命令 Item 实现 + +```yaml +material: 'repo:myCustomItem' + +material: 'repo:%custom_variable_whichreturnstheid%' +``` + +## 转化格式 JSON + +* 通过 `/trmenu item toJson` 转化得到的物品文本格式 +* 该方法能支持一切形式的物品 + +```yaml +material: '{"type":"DIAMOND_SWORD","data":0,"amount":1,"meta":{"Damage":{"type":"INT","data":0}}}' +``` + +## 头颅材质 + +### 玩家头颅 + +```yaml +material: 'head:%player_name%' + +material: 'head:BlackSky' +``` + +* TrMenu 采用本地 NMS 读取在线玩家的皮肤 + 异步联网读取离线正版玩家 皮肤材质的形式,不卡线程 +* 已对 SkinsRestorer 进行完整支持,能够获取到玩家显示的皮肤! + +### 自定义纹理头颅 + +```yaml +material: 'head:eyJ0ZXh0dXJlcyI6eyJTS0lOIjp7InVybCI6Imh0dHA6Ly90ZXh0dXJlcy5taW5lY3JhZnQubmV0L3RleHR1cmUvNDRmNDUyZDk5OGVhYmFjNDY0MmM2YjBmZTVhOGY0ZTJlNjczZWRjYWUyYTZkZmQ5ZTZhMmU4NmU3ODZlZGFjMCJ9fX0=' + +material: 'head:44f452d998eabac4642c6b0fe5a8f4e2e673edcae2a6dfd9e6a2e86e786edac0' +``` + +## 物品源 + +* TrMenu 对挂钩插件物品支持的实现 +* 统一格式为 `source::` + +### HeadDatabase + +```yaml +material: 'source:HeadDatabase:myHead' + +material: 'source:HDB:random' +``` + +### Oraxen + +```yaml +material: 'source:ORAXEN:itemId' +``` + +### ItemsAdder + +```yaml +material: 'source:ITEMADDER:itemId' +material: 'source:IA:anotherItemId' +``` + +### Zaphkiel + +```yaml +material: 'source:ZAPHKIEL:itemId' +material: 'source:ZL:itemId' +``` + +## \* 材质参数 + +### 附加值 + +```yaml +material: '3{data=2}' +``` + +### Model Data + +```yaml +# 1.14+ +material: 'coal{model-data=15}' +``` + +### 皮革染色 + +```yaml +material: 'leather chestplate {dye=255,255,0}' +``` + +### 旗帜模式 + +```yaml +material: 'white banner {banner=RED MOJANG,WHITE}' +``` + diff --git a/menu/icon/display/name.md b/menu/icon/display/name.md new file mode 100644 index 0000000..4544480 --- /dev/null +++ b/menu/icon/display/name.md @@ -0,0 +1,26 @@ +--- +description: 物品显示的名称,支持动态 +--- + +# 名称 + +## 示例 + +```yaml + 'Next': + display: + material: 'Cyan Stained Glass Pane' + name: '&bN&3ext Page' +``` + +```yaml + 'Close': + update: [25, 5, -1, -1] + display: + materials: + - 'Red Stained Glass Pane' + - 'Orange Stained Glass Pane' + # 动态名称,更新周期 5 ticks + name: ['&cC&7lose', '&cCl&7ose', '&cClo&7se', '&cClos&7e', '&cClose'] +``` + diff --git a/menu/icon/display/property.md b/menu/icon/display/property.md new file mode 100644 index 0000000..95fab17 --- /dev/null +++ b/menu/icon/display/property.md @@ -0,0 +1,49 @@ +# 属性 + +## 数量 + +```yaml +# 静态数量 +amount: 10 + +# 动态数量 (需要是返回数值的变量) +amount: '${js: varInt("%server_time_s%") + 1}' +``` + +## 发光 + +```yaml +# 直接设置布尔值,开启或关闭(默认:关闭) +# 实质是为物品添加一个附魔效果 & 隐藏附魔显示的标签,因此不能让所有物品都发光 +shiny: true + +# 动态效果(条件表达式) +shiny: 'all [perm *vip.user money *100]' +``` + +## 标签 + +```yaml +flags: +- 'HIDE_ENCHANTS' +- 'HIDE_ATTRIBUTES' +``` + +> 可用物品标签 +> +> * HIDE\_ENCHANTS +> * HIDE\_ATTRIBUTES +> * HIDE\_UNBREAKABLE +> * HIDE\_POTION\_EFFECTS +> * HIDE\_DESTROYS +> * HIDE\_PLACED\_ON + +## NBT + +```yaml +nbt: + balabala: 12345 +``` + +* 支持使用变量 + diff --git a/menu/icon/handler.md b/menu/icon/handler.md new file mode 100644 index 0000000..b4ff786 --- /dev/null +++ b/menu/icon/handler.md @@ -0,0 +1,95 @@ +--- +description: 图标交互反应 +--- + +# 交互 + +## 示例 + +```yaml +'Close': + update: [-1, 5, -1, -1] + display: + material: Red Stained Glass Pane + name: ['&cC&7lose', '&cCl&7ose', '&cClo&7se', '&cClos&7e', '&cClose'] + # 动作部分 + actions: + # 点击类型 - 反应 + all: close +``` + +```yaml + '*': + update: [-1, 5, 20, -1] + display: # ... + # 动作部分 + actions: + # 点击类型(左键) + left: + - 'set-meta: icon_server_hide true' + - 'sound: BLOCK_NOTE_BLOCK_BIT-1-0' + - 'refresh: *' +``` + +## 类型 + +> ALL +> +> LEFT +> +> RIGHT +> +> SHIFT\_LEFT +> +> SHIFT\_RIGHT +> +> OFFHAND +> +> NUMBER\_KEY +> +> NUMBER\_KEY\_1 +> +> NUMBER\_KEY\_2 +> +> NUMBER\_KEY\_3 +> +> NUMBER\_KEY\_4 +> +> NUMBER\_KEY\_5 +> +> NUMBER\_KEY\_6 +> +> NUMBER\_KEY\_7 +> +> NUMBER\_KEY\_8 +> +> NUMBER\_KEY\_9 +> +> MIDDLE +> +> DROP +> +> CONTROL\_DROP +> +> ABROAD\_LEFT\_EMPTY +> +> ABROAD\_RIGHT\_EMPTY +> +> ABROAD\_LEFT\_ITEM +> +> ABROAD\_RIGHT\_ITEM +> +> LEFT\_MOUSE\_DRAG\_ADD +> +> RIGHT\_MOUSE\_DRAG\_ADD +> +> MIDDLE\_MOUSE\_DRAG\_ADD +> +> DOUBLE\_CLICK + +## 反应 + +{% page-ref page="../action/reactions.md" %} + + + diff --git a/menu/icon/settings.md b/menu/icon/settings.md new file mode 100644 index 0000000..9a293d4 --- /dev/null +++ b/menu/icon/settings.md @@ -0,0 +1,127 @@ +--- +description: 图标设置项均与 显示部分、交互部分 同级节点 +--- + +# 设置 + +## 更新周期 + +* 更新周期包括 4 个(材质,名称,Lore,槽位) +* 你可以通过配置数组的形式独立设置它们 + +```yaml +# 设置一个值,同时应用到 材质,名称,Lore,槽位 中 +update: 20 +``` + +```yaml +# 不完全设置 +# 分别设置材质,名称,Lore的更新周期为 20, -1, 15 +# 槽位的更新周期将选取其中最高的一个,即 20 +update: [20, -1, 15] +``` + +```yaml +# 完全配置, 即按顺序分别对应 +update: [-1, 5, 25, 30] +``` + +* 不配置更新周期项,图标将默认不更新 +* 更新周期应与默认图标同级节点,子图标不支持自定义更新周期 +* 插件会根据属性是否需要更新(包含动画、变量)自动分配周期,因此不必在这麻烦配置 + +## 刷新周期 + +* 按优先级、条件重新计算子图标 +* 只有存在子图标时,此项才有效 +* 使用动作 Refresh,可以主动刷新图标,达到同样效果 + +```yaml +refresh: 20 +``` + +## 图标槽位 + +* 针对静态槽位的图标,推荐通过 **布局** 功能来定义图标槽位 +* 手动配置槽位将覆写布局的配置 + +```yaml +slots: 6 +``` + +```yaml +slots: + - 1 + - 2 + - 3 + # 4-6 = [4, 5, 6] + - '4-6' + - '${js: varInt("%player_health%")}' + - 7 +``` + +```yaml +# 动态槽位,List> 嵌套模式 +slots: +- - 0 + - 1 +- - 1 + - 2 +- - 2 + - 3 +``` + +## \* 继承 + +```yaml +inherit: true +``` + +* 无论配置与否,默认子图标都将继承主图标的**材质** +* 开启子图标继承,将额外继承主图标的**名称,Lore** +* “继承” 是指子图标未配置的前提下,继承状态下子图标仍然可以通过配置来覆写 +* 默认继承与否可通过编辑 `settings.yml` 文件来设置 + +## \* 条件 + +子图标的属性,条件计算结果为真时才可能显示此子图标 + +```yaml +condition: 'perm *trmenu.use' +``` + +## \* 优先级 + +决定该子图标的权重(筛选子图标时的顺序先后) + +```yaml +priority: 20 +``` + +* 该项仅对子图标有效 +* 若不设置,则默认优先级为 **0** +* 选取条件子图标时,按优先级**升序**依次计算子图标到满足条件的为止,即优先级最小的图标最先计算 + +在实际配置 YAML 文件时,此项为非必须项 + +插件会根据图标的 “配置位置” ,按顺序自动的分配一定优先级,例如 + +```yaml +Icons: + 'A': + display: + material: grass_block + icons: + - condition: 'perm *op' + display: + material: beacon + - condition: 'perm *mvp.user' + display: + material: diamond block + - condition: 'perm *vip.user' + display: + material: gold block +``` + +会先计算条件为 `perm *op` 的子图标,再依次 `perm *mvp.user` 、`perm *vip.user` + diff --git a/migrate.md b/migrate.md new file mode 100644 index 0000000..79b04ee --- /dev/null +++ b/migrate.md @@ -0,0 +1,52 @@ +--- +description: Only applicable to 3.0-pre9 or earlier v3 version TrMenu +--- + +# Migration Guide + +## Migrate from the earlier v3 version of TrMenu to 3.0-pre12 and above. +> Although TabooLib-pre29 and later versions have automatic migration, we still recommend you to migrate according to the following instructions to avoid migration failure. + +After TabooLib 6.0.0 released the preview version, it has been adapted to the server above 1.17. +However, its framework is greatly changed, which may cause some features of TrMenu to be different from that of TabooLib 5.x. +Therefore, some content needs to be manually migrated to quickly update to TrMenu-pre12. + +## Migration Instructions +To help you update to the latest version faster, some mandatory and optional \(optional\) migration options are marked below. +### Configuration File +#### [Optional] settings.yml +Because TabooLib 6.x updated the feature of automatically detecting the language of the executor, the node `Options.Language` has been removed, +Just remove the relevant nodes in the configuration file: +```yaml +Options: + Language: en_US +``` +#### [Requirement] lang/*.yml \(I.e. all language files\) +Because TabooLib 6.x has updated the function of automatically updating configuration files, in order to better realize automatic updating of configuration files. +Therefore, the language file is required to be 'flattened' instead of 'multi-level', so the original language file format cannot be read again. + +Suggestion: Delete all language files in the `plugins/TrMenu/lang` directory. +#### [1.8 User] datasource.yml +Because the 1.8 server-side JDBC driver is too old, it is necessary to make some modifications to the configuration file. +However, the configuration file did not appear in the previous TrMenu \(pre9 and previous\) versions, so after completing the migration of the previous two configuration files, you need to start the server first. + +Not surprisingly, you will see the following error message: +``` +[xx:xx:xx] [Server thread/INFO]: [TrMenu] Enabling TrMenu v3.0-PRE-15 +[xx:xx:xx] [Server thread/WARN]: SLF4J: No SLF4J providers were found. +[xx:xx:xx] [Server thread/WARN]: SLF4J: Defaulting to no-operation (NOP) logger implementation +[xx:xx:xx] [Server thread/WARN]: SLF4J: See http://www.slf4j.org/codes.html#noProviders for further details. +[xx:xx:xx] [Server thread/ERROR]: Error occurred while enabling TrMenu v3.0-PRE-15 (Is it up to date?) +java.lang.AbstractMethodError: org.sqlite.Conn.isValid(I)Z + at com.zaxxer.hikari_4_0_3.pool.PoolBase.checkValidationSupport(PoolBase.java:464) ~[?:?] + at com.zaxxer.hikari_4_0_3.pool.PoolBase.checkDriverSupport(PoolBase.java:447) ~[?:?] + at com.zaxxer.hikari_4_0_3.pool.PoolBase.setupConnection(PoolBase.java:416) ~[?:?] + at com.zaxxer.hikari_4_0_3.pool.PoolBase.newConnection(PoolBase.java:369) ~[?:?] + at com.zaxxer.hikari_4_0_3.pool.PoolBase.newPoolEntry(PoolBase.java:206) ~[?:?] + at com.zaxxer.hikari_4_0_3.pool.HikariPool.createPoolEntry(HikariPool.java:476) ~[?:?] + at com.zaxxer.hikari_4_0_3.pool.HikariPool.checkFailFast(HikariPool.java:561) ~[?:?] +... +``` +At this time, the configuration file `plugins/TrMenu/datasource.yml` will appear. + +Then find the configuration file node `DefaultSettings.ConnectionTestQuery`, the default value is `~`, just change it to `SELECT 1` to complete the migration. diff --git a/script/javascript.md b/script/javascript.md new file mode 100644 index 0000000..11eb4bf --- /dev/null +++ b/script/javascript.md @@ -0,0 +1,34 @@ +--- +description: This section requires a certain Java / JavaScript or similar programming foundation to facilitate understanding +--- + +# JavaScript + +## Object + +TrMenu’s JavaScript engine currently provides the following objects + +* `bukkitServer` 即 Bukkit.getServer\(\) +* `utils` 即 me.arasple.mc.trmenu.module.internal.script.js.Assist.INSTANCE +* `player` I.e. the player itself +* `session` 即 me.arasple.mc.trmenu.module.display.MenuSession + +## Function + +The JavaScript engine of TrMenu currently provides the following functions + +* `vars(String input)` Return the replacement function variable +* `varInt(String input)` Replace variable and convert to integer + +## Note + +* TrMenu’s JavaScript will be pre-compiled and cached, and all variables need to be processed through functions +* The symbol for "or" is `||`, and the symbol for "and" is `&&` + +## Instance + +* Check whether there is permission + * You can directly call the method of the player object, `player.hasPermission("perm")` + + + diff --git a/script/kether.md b/script/kether.md new file mode 100644 index 0000000..4cd6f70 --- /dev/null +++ b/script/kether.md @@ -0,0 +1,12 @@ +# Kether + +## Learn + +![](https://kether.tabooproject.org/static/images/ketherx.png) + +{% embed url="https://kether.tabooproject.org" caption="KetherX Chinese document" %} + +## Note + +Kether conditions are used in icons and actions, because **does not include** any delayed or delayed return content + diff --git a/start/README.md b/start/README.md new file mode 100644 index 0000000..855d422 --- /dev/null +++ b/start/README.md @@ -0,0 +1,2 @@ +# Start + diff --git a/start/configurate.md b/start/configurate.md new file mode 100644 index 0000000..033038b --- /dev/null +++ b/start/configurate.md @@ -0,0 +1,137 @@ +--- +description: After installing this plugin for the first time, some files will be generated in the plugin directory +--- + +# Configuration + +## Document + +{% tabs %} +{% tab title="lang/zh\_CN.yml" %} +TLocale language file, you can edit almost all messages of this plugin +{% endtab %} + +{% tab title="data/globalData.yml" %} +Where the global cache data variables are stored + +Do not edit when the server is enabled +{% endtab %} + +{% tab title="data/itemRepository.yml" %} +Where the item repository data is stored + +Do not edit when the server is enabled +{% endtab %} + +{% tab title="menus" %} +The default menu loading directory + +The menu file (YAML) can be placed in this directory or its subdirectories, and it will be automatically loaded by the plug-in +{% endtab %} + +{% tab title="settings.yml" %} +The main configuration file of TrMenu +{% endtab %} +{% endtabs %} + +## 设置 + +{% code title="settings.yml \(v3.0 BETA-2\)" %} +```yaml +# +# Plugin options +# +Options: + # Performance mode: High, Normal, Low + Running-Performance: Normal + # Multithreading + Multi-Thread: true + # Load menu asynchronously + Async-Load-Menus: true +# +# Plugin player data storage method +# +Database: + # storage method: LOCAL, MONGODB + Method: LOCAL + Url: + Client: 'mongodb://localhost:3307' + Database: trixey + Collection: menu + +# +# Menu loader +# +Loader: + # Custom load directory + Menu-Files: + - 'plugins/CustomMenusFolder' + +# +# Menu Settings +# +Menu: + # 选项 + Settings: + # The minimum interval for the bound item to trigger the opening of the menu (to prevent frequent brushing) + Bound-Item-Interval: 3 + # Icon + Icon: + # Whether to enable sub-icons to inherit the main icon by default + Inherit: false + # Display items + Item: + # Default Name color + Default-Name-Color: "&7" + # Default Lore color + Default-Lore-Color: "&7" + # Priority coloring + # If turned on, first replace the color and then process the function variable + Pre-Color: false + +# +# Action related +# +Action: + # Catcher + Inputer: + # Cancellation word (regular) + Cancel-Words: + - 'cancel|quit|end' + - 'q' + +# +# Actions performed by quick binding +# For specific notes, please refer to the [USAGE-Quick Binding] chapter +# +Shortcuts: + Offhand: [] + Sneaking-Offhand: + - condition: 'perm *trmenu.shortcut' + execute: 'open: Example' + deny: 'return' + Right-Click-Player: 'open: Profile' + Sneaking-Right-Click-Player: [ ] + PlayerInventory-Border-Left: [ ] + PlayerInventory-Border-Right: [ ] + PlayerInventory-Border-Middle: [ ] + +# +# Register custom commands +# Please refer to the [USAGE-Command Registration] chapter for specific notes +# +RegisterCommands: + openMenus: + aliases: [ ] + permission: null + execute: + - 'tell: &7Argument `example` Required!' + arguments: + example: 'open: example' +``` +{% endcode %} + +## Language + +TODO + diff --git a/start/install.md b/start/install.md new file mode 100644 index 0000000..d535fc6 --- /dev/null +++ b/start/install.md @@ -0,0 +1,51 @@ +--- +description: After obtaining the JAR file of this plugin from a regular channel, install it on the server +--- + +# Install + +## Requirement + +{% hint style="info" %} +TrMenu needs to be downloaded and installed on the Internet +{% endhint %} + +| Software | Version | +| :--- | :--- | +| Spigot | 1.8-1.17 | +| Paper / Tuinity | 1.8-1.17 | +| CatSever \(Forge\) | √ | +| Akarin | unknown | +| Arclight \(Forge\) | √ | +| Mohist \(Forge\) | √ | +| Other unknown servers so | uncertain | + +* Recommended:[PlaceholderAPI](http://ci.extendedclip.com/job/PlaceholderAPI/) \(Must ensure that the version is in **v2.10.10 +**\) + +## Install + +* Throw **TrMenu.jar** into the plugins directory +* Restart the server + +## Compatible + +Known incompatible plugins & solutions: + +* **LibsDisguises** + * Reason: Incompatible + * Solution: + * Set `Remove-Armor` to `false` in the configuration file of LibsDisguises + +## Note + +{% hint style="warning" %} +TrMenu **does not support all** forms of **reload**, + +Forcible hot reload operation may cause unknown errors, please operate with caution + +including but not limited to: + +* TrMenu reload TrMenu via Plugman \(or similar plugin\) +* Restart the server through the /reload \(confirm\) command +{% endhint %} + diff --git a/start/purchase.md b/start/purchase.md new file mode 100644 index 0000000..31fcec6 --- /dev/null +++ b/start/purchase.md @@ -0,0 +1,48 @@ +--- +description: TrMenu is a paid open source plugin, dedicated to providing the best quality menu system experience +--- + +# Purchase + +> Before paying for this plugin, please confirm whether your server core and version are compatible + +## Price + +* **129** RMB + +## Terms + +* TrMenu is an open source plugin, the complete code is on Github and is kept updated +* Option to pay for the plugin, which can support the author to continue to maintain the project & get after-sales service and help +* Voluntary payment support projects, prohibit all forms and reasons for disputes and initiate refund appeals +* You can obtain plugin JAR files through non-paid means, but secondary distribution in all ways is prohibited without permission + +Non-purchasers can learn the plugin by consulting this document (the document is fully open) + +But it may not receive any kind of help or support from the developer + +## Channel +> The original author is about to take the college entrance examination, so it is now maintained by Score2 free of charge. +> +> You can also selectively support and encourage Score2 to continue to update: +> [https://afdian.net/@Score2](https://afdian.net/@Score2) + +{% embed url="http://afdian.net/@Arasple" %} + +~~TrMenu Love Power Purchasing Channel~~ +* ~~After paying for the purchase, you will receive the key of this plugin in a private message on iPower~~ +* ~~Add bot **193432545**'s friends and chat privately to exchange keys~~ +* ~~Send message in private chat `#密匙` ,for example send `#27EFEF9092434D64AB3190B5DA596ADO`.~~ +* ~~Get the latest stable version download address through the robot private chat command `/trmenu`~~ + +Obtain the plugin JAR file from any third-party channel, + +**Will not receive any help or support from the author**,nor can we guarantee the security of the plugin + +## Compile by yourself + +* This plugin is completely open source and provides complete build files. If you don’t want to pay for it, you can build it yourself +* Will not receive any help or support + +{% embed url="https://github.com/TrMenu/TrMenu" %} + diff --git a/usage/arguments.md b/usage/arguments.md new file mode 100644 index 0000000..8a63deb --- /dev/null +++ b/usage/arguments.md @@ -0,0 +1,30 @@ +--- +description: TrMenu menu supports the use of arguments with the suffix of the trigger command as a variable +--- + +# Menu Arguments + +## Function Enable + +Without any configuration, the menu arguments transfer function is **enabled** by default. + +If you encounter command incompatibility, you can turn it off through options + +{% page-ref page="../menu/configuration/option.md" %} + +## Transfer Method + +* Binding custom open command transfer +* Plugin turn on menu command transfer +* Call action second modification + +## How To Use + +For example, the player’s current arguments are apple, juice, orange, which is an array of size 3 + +Almost anywhere in the menu can be used as a variable, {0} will return apple, {1} will return juice, and so on + +## Note + +* By default, closing the menu does not delete existing arguments + diff --git a/usage/command.md b/usage/command.md new file mode 100644 index 0000000..d2e6452 --- /dev/null +++ b/usage/command.md @@ -0,0 +1,88 @@ +--- +description: '[] is a required argument, <> is an optional argument' +--- + +# Plugin Commands + +## Main Command + +> The main command of the plugin + +* Name:`trmenu` `menu` +* Permission: `trmenu.access` + +## List + +> List loaded menus + +* Permission: `trmenu.command.list` +* Argument + * <Filter> Filter menu name + +## Open + +> Open the specified menu + +* Permission: `trmenu.command.open` +* Argument + * \[ID\]:<Page> Menu name and specified page number + * <Player> Specify the player, if you leave it blank, you will default to yourself + * <Menu Argument> The incoming menu argument, used as a variable +* Example + * `trmenu open Example BlackSKY` Open the Example menu for player BlackSKY + * `trmenu open Shop:3` Open the Shop menu for yourself, page 3 + +## Reload + +> Reload menu + +* Permission: `trmenu.command.reload` + +## Template + +> Template creation function, quick layout menu \(currently only supports box containers\) + +* Permission: `trmenu.command.template` +* Argument + * <Rows> The row size of the menu \(1~6\) + +## Action + +> Test TrMenu action + +* Permission: `trmenu.command.action` +* Argument + * \[ID\] The name of the player object performing the action + * \[Action\] Action line +* Note + * By default, the action return & related costs will be printed to the commander + * To hide this print function, you need to put a `#` mark at the top of the action line + +## Item + +> Management control items + +* Permission: `trmenu.command.item` +* Argument + * \[Method\] Operation type + * toJson Convert items in hand to JSON text format + * fromJson Convert the text content of Argument2 into an item object + * save Save items to the item repository + * get Get items from the item repository + * delete Delete items from the item repository + * <Value> value + +## Sounds + +> Preview sound effects + +* Permission: `trmenu.command.sounds` +* Argument + * <Filter> Filter sound effect name + +## Debug + +> Debug function + +* Permission: `trmenu.command.debug` + diff --git a/usage/functions.md b/usage/functions.md new file mode 100644 index 0000000..62bacda --- /dev/null +++ b/usage/functions.md @@ -0,0 +1,59 @@ +# Icon variable +```text +@iconid@ +``` + +# Function variable + +{% hint style="info" %} +Inside the function variable `{}` If you need to use the `}` closing curly brace, please escape it with `\}` +{% endhint %} + +## Menu arguments function + +```text +{} +``` + +## Kether Function + +```text +{ke: } +``` + +## JavaScript Function + +```text +{js: } +``` + +## Meta variable + +```text +{meta: } +``` + +## Data variable + +```text +{data: } +``` + +## GlobalData variable + +```text +{gdata: } +``` + +## 内置节点变量 + +```text +{node: } +``` + +## 内置脚本函数 + +```text +${__......} +``` + diff --git a/usage/item-matchers.md b/usage/item-matchers.md new file mode 100644 index 0000000..47e734a --- /dev/null +++ b/usage/item-matchers.md @@ -0,0 +1,26 @@ +# Item Matcher + +## Format + +* Match different features of the same item, use, to separate +* Use for multiple item features; separate + +## Example + +```text +material:DIAMOND,amount:64 ; material:EMERALD,amount:32 +``` + +## 特征 + +* Material: Material +* Data: Data +* Amount: Amount +* Name: Name +* Lore: Lore +* Head material: Texture +* ModelData + +[ +](https://trmenu.trixey.cc/v/chinese/functions/arguments) + diff --git a/usage/node-getter.md b/usage/node-getter.md new file mode 100644 index 0000000..c21df09 --- /dev/null +++ b/usage/node-getter.md @@ -0,0 +1,159 @@ +--- +description: TrMenu also supports getting the nodes in the configuration file as variables to avoid writing the same content repeatedly. +--- + +# Node variable + +## Usage +```text +{node: } + +E.g: +{node: Title} + -> Get the Title of the menu + +{node: Icons.A.display.name} + -> Get the display name of icon A + +{node: Icons.@iconid@.display.name} + -> If the variable exists in any area of B that supports function variables, the display name of icon B will be obtained +``` +* **Note**: When getting the node, pay attention to whether the case matches as much as possible +* As a function variable, it can be used in any area that supports variables +* **@iconid@** is a static variable, which will be parsed as the id of the icon under the icon + +## 示例 +```yaml +Title: '子服列表' + +Layout: + - '012345678' + +Icons: + 0: + display: + material: 'wool' + name: '&3前往 => &b{node: Icons.@iconid@.properties.name}' + lore: + - '&7有关信息: {node: Icons.@iconid@.properties.desc}' + - '' + actions: + all: + - 'server: {node: Icons.@iconid@.properties.server}' + properties: + name: '子服-@iconId@' + server: 'server-@iconId@' + desc: '这里有许多友爱的玩家们' + 1: + display: + material: 'wool' + name: '&3前往 => &b{node: Icons.@iconid@.properties.name}' + lore: + - '&7有关信息: {node: Icons.@iconid@.properties.desc}' + - '' + actions: + all: + - 'server: {node: Icons.@iconid@.properties.server}' + properties: + name: '子服-@iconId@' + server: 'server-@iconId@' + desc: '这里风和日丽' + 2: + display: + material: 'wool' + name: '&3前往 => &b{node: Icons.@iconid@.properties.name}' + lore: + - '&7有关信息: {node: Icons.@iconid@.properties.desc}' + - '' + actions: + all: + - 'server: {node: Icons.@iconid@.properties.server}' + properties: + name: '子服-@iconId@' + server: 'server-@iconId@' + desc: '这里危机重重' + 3: + display: + material: 'wool' + name: '&3前往 => &b{node: Icons.@iconid@.properties.name}' + lore: + - '&7有关信息: {node: Icons.@iconid@.properties.desc}' + - '' + actions: + all: + - 'server: {node: Icons.@iconid@.properties.server}' + properties: + name: '子服-@iconId@' + server: 'server-@iconId@' + desc: '这里适合养老' + 4: + display: + material: 'wool' + name: '&3前往 => &b{node: Icons.@iconid@.properties.name}' + lore: + - '&7有关信息: {node: Icons.@iconid@.properties.desc}' + - '' + actions: + all: + - 'server: {node: Icons.@iconid@.properties.server}' + properties: + name: '子服-@iconId@' + server: 'server-@iconId@' + desc: '这里怪物很多' + 5: + display: + material: 'wool' + name: '&3前往 => &b{node: Icons.@iconid@.properties.name}' + lore: + - '&7有关信息: {node: Icons.@iconid@.properties.desc}' + - '' + actions: + all: + - 'server: {node: Icons.@iconid@.properties.server}' + properties: + name: '子服-@iconId@' + server: 'server-@iconId@' + desc: '这里适合生电' + 6: + display: + material: 'wool' + name: '&3前往 => &b{node: Icons.@iconid@.properties.name}' + lore: + - '&7有关信息: {node: Icons.@iconid@.properties.desc}' + - '' + actions: + all: + - 'server: {node: Icons.@iconid@.properties.server}' + properties: + name: '子服-@iconId@' + server: 'server-@iconId@' + desc: '这里经常崩服' + 7: + display: + material: 'wool' + name: '&3前往 => &b{node: Icons.@iconid@.properties.name}' + lore: + - '&7有关信息: {node: Icons.@iconid@.properties.desc}' + - '' + actions: + all: + - 'server: {node: Icons.@iconid@.properties.server}' + properties: + name: '子服-@iconId@' + server: 'server-@iconId@' + desc: '这里有剧情' + 8: + display: + material: 'wool' + name: '&3前往 => &b{node: Icons.@iconid@.properties.name}' + lore: + - '&7有关信息: {node: Icons.@iconid@.properties.desc}' + - '' + actions: + all: + - 'server: {node: Icons.@iconid@.properties.server}' + properties: + name: '子服-@iconId@' + server: 'server-@iconId@' + desc: '这里可以上天' +``` \ No newline at end of file diff --git a/usage/register-commands.md b/usage/register-commands.md new file mode 100644 index 0000000..5bcd432 --- /dev/null +++ b/usage/register-commands.md @@ -0,0 +1,22 @@ +# Command Register + +## Configuration + +{% code title="settings.yml" %} +```yaml +RegisterCommands: + # Main command name + openMenus: + # Sub command name + aliases: [ ] + # Required permissions, no need permissions set it to null + permission: null + # Action + execute: + - 'tell: &7Argument `example` Required!' + # Arguments + arguments: + # /openMenu example + # Action + example: 'open: example' +``` \ No newline at end of file diff --git a/usage/shortcuts.md b/usage/shortcuts.md new file mode 100644 index 0000000..e313980 --- /dev/null +++ b/usage/shortcuts.md @@ -0,0 +1,34 @@ +# Quick binding + +## Configuration + +{% code title="settings.yml" %} +```yaml +Shortcuts: + Offhand: [] + Sneaking-Offhand: + - condition: 'perm *trmenu.shortcut' + execute: 'open: Example' + deny: 'return' + Right-Click-Player: 'open: Profile' + Sneaking-Right-Click-Player: [ ] + PlayerInventory-Border-Left: [ ] + PlayerInventory-Border-Right: [ ] + PlayerInventory-Border-Middle: [ ] +``` +{% endcode %} + +## Type + +* Switch Off-hand OFFHAND +* Sneak + Switch Off-hand SNEAKING\_OFFHAND +* Right click player RIGHT\_CLICK\_PLAYER +* Sneak + Right click player SNEAKING\_RIGHT\_CLICK\_PLAYER +* \(In survival mode\) Click outside the player’s inventory PLAYER\_INVENTORY\_BORDER\_LEFT/RIGHT/MIDDLE + +## Note + +* When the execution result of **action group** is true, this event will be canceled \(for example, switch deputy\) +* To prevent interference with the game experience, all shortcut combinations of \[sneak + operation\] need to be completed within **1.5s** before they can be triggered +* The shortcut of the right-click player will pass in the menu argument as the name of the target player by default +