3 Commits

Author SHA1 Message Date
TheWhiteDog9487
8b03098280 0.5.0
支持通过对角线坐标设置随机坐标选取范围
远离边界这个保护的实现有问题,然后地狱末地什么玩意的也不对,重试机制也不存在,这些之后再鸽
2026-02-11 18:12:06 +08:00
TheWhiteDog9487
a0a6e1c35d ?为啥你是反过来的 2026-02-11 18:08:00 +08:00
TheWhiteDog9487
5f3ca6c68f 更新依赖项 2026-02-11 18:08:00 +08:00
7 changed files with 220 additions and 86 deletions

View File

@@ -1,61 +1,84 @@
# 介绍 # 介绍
这个模组增加了一个命令(/rtp),用于将玩家随机传送到世界的任何一个位置。 这个模组增加了一个命令 `/rtp`,用于将玩家随机传送到世界的任何一个位置。
# 命令格式 # 命令格式
- /rtp - `/rtp`
- /rtp <Radius(半径)> - `/rtp <随机半径>`
- /rtp <被传送玩家名(PlayerID)> - `/rtp <被传送玩家ID>`
- /rtp <Radius(半径)> <被传送玩家名(PlayerID)> - `/rtp <随机半径> <被传送玩家ID>`
- /rtp <被传送玩家名(PlayerID)> <Radius(半径)> - `/rtp <被传送玩家ID> <随机半径>`
- /rtp <Radius(半径)> <OriginPos(随机中心坐标)> - `/rtp <随机半径> <随机中心坐标>`
- /rtp <Radius(半径)> <被传送玩家名(PlayerID)> <OriginEntity(随机中心实体)> - `/rtp <随机半径> <被传送玩家ID> <作为随机中心实体>`
- /rtp <Radius(半径)> <被传送玩家名(PlayerID)> <OriginPos(随机中心坐标)> - `/rtp <随机半径> <被传送玩家ID> <随机中心坐标>`
- /rtp <被传送玩家名(PlayerID)> <Radius(半径)> <OriginEntity(随机中心实体)> - `/rtp <被传送玩家ID> <随机半径> <作为随机中心实体>`
- /rtp <被传送玩家名(PlayerID)> <Radius(半径)> <OriginPos(随机中心坐标)> - `/rtp <被传送玩家ID> <随机半径> <随机中心坐标>`
- `/rtp <随机区域起点坐标> <随机区域终点坐标>`
- `/rtp <被传送玩家ID> <随机区域起点坐标> <随机区域终点坐标>`
- `/rtp <随机区域起点坐标> <随机区域终点坐标> <被传送玩家ID>`
- `/rtp <随机区域起点实体> <随机区域终点实体>`
- `/rtp <随机区域起点实体> <随机区域终点实体> <被传送玩家ID>`
## 命令示例 ## 命令示例
- /rtp - `/rtp`
将执行命令的玩家随机传送到以(0,0)为中心点2.9e7 - 1e4作为随机半径的范围内的随机一点 将执行命令的玩家随机传送到以`(0,0)`为中心点2.9e7 - 1e4作为随机半径的范围内的随机一点
2.9e+7 = 2.9 x 10^7 = 29000000 = 两千九百万 2.9e+7 = 2.9 x 10^7 = 29000000 = 两千九百万
1e4 = 10^4 = 10000 = 一万 1e4 = 10^4 = 10000 = 一万
- /rtp 1000 - `/rtp 1000`
将执行命令的玩家随机传送到以(0,0)为中心点1000作为随机半径的范围内的随机一点 将执行命令的玩家随机传送到以`(0,0)`为中心点1000作为随机半径的范围内的随机一点
- /rtp TheWhiteDog9487 - `/rtp TheWhiteDog9487`
将TheWhiteDog9487随机传送到以(0,0)为中心点2.9e7 - 1e4作为随机半径的范围内的随机一点 将TheWhiteDog9487随机传送到以`(0,0)`为中心点2.9e7 - 1e4作为随机半径的范围内的随机一点
- /rtp TheWhiteDog9487 1000 - `/rtp TheWhiteDog9487 1000`
将TheWhiteDog9487随机传送到以(0,0)为中心点1000作为随机半径的范围内的随机一点 将TheWhiteDog9487随机传送到以`(0,0)`为中心点1000作为随机半径的范围内的随机一点
- /rtp 1000 TheWhiteDog9487 - `/rtp 1000 TheWhiteDog9487`
将TheWhiteDog9487随机传送到以(0,0)为中心点1000作为随机半径的范围内的随机一点 将TheWhiteDog9487随机传送到以`(0,0)`为中心点1000作为随机半径的范围内的随机一点
- /rtp 1000 10000 10000 - `/rtp 1000 10000 10000`
将执行命令的玩家随机传送到以(10000,10000)为中心点1000作为随机半径的范围内的随机一点 将执行命令的玩家随机传送到以`(10000,10000)`为中心点1000作为随机半径的范围内的随机一点
- /rtp 1000 TheWhiteDog9487 TheWhiteDog_CN - `/rtp 1000 TheWhiteDog9487 TheWhiteDog_CN`
将TheWhiteDog9487随机传送到以TheWhiteDog_CN所在位置为中心点1000作为随机半径的范围内的随机一点 将TheWhiteDog9487随机传送到以TheWhiteDog_CN所在位置为中心点1000作为随机半径的范围内的随机一点
- /rtp 1000 TheWhiteDog9487 10000 10000 - `/rtp 1000 TheWhiteDog9487 10000 10000`
将TheWhiteDog9487随机传送到以(10000,10000)为中心点1000作为随机半径的范围内的随机一点 将TheWhiteDog9487随机传送到以`(10000,10000)`为中心点1000作为随机半径的范围内的随机一点
- /rtp TheWhiteDog9487 1000 TheWhiteDog_CN - `/rtp TheWhiteDog9487 1000 TheWhiteDog_CN`
将TheWhiteDog9487随机传送到以TheWhiteDog_CN所在位置为中心点1000作为随机半径的范围内的随机一点 将TheWhiteDog9487随机传送到以TheWhiteDog_CN所在位置为中心点1000作为随机半径的范围内的随机一点
- /rtp TheWhiteDog9487 1000 10000 10000 - `/rtp TheWhiteDog9487 1000 10000 10000`
将TheWhiteDog9487随机传送到以(10000,10000)为中心点1000作为随机半径的范围内的随机一点 将TheWhiteDog9487随机传送到以`(10000,10000)`为中心点1000作为随机半径的范围内的随机一点
- `/rtp 10000 10000 20000 20000`
将执行命令的玩家随机传送到以`(10000,10000)`,`(20000,10000)`,`(20000,20000)`,`(10000,20000)`四个顶点围成的长方形区域内的随机一点
您需要提供的坐标是这个长方形的任意一个顶点和这个顶点对应的斜对角顶点的位置
- `/rtp TheWhiteDog9487 10000 10000 20000 20000`
将TheWhiteDog9487随机传送到以`(10000,10000)`,`(20000,10000)`,`(20000,20000)`,`(10000,20000)`四个顶点围成的长方形区域内的随机一点
- `/rtp 10000 10000 20000 20000 TheWhiteDog9487`
将TheWhiteDog9487随机传送到以`(10000,10000)`,`(20000,10000)`,`(20000,20000)`,`(10000,20000)`四个顶点围成的长方形区域内的随机一点
- `/rtp TheWhiteDog9487 TheWhiteDog_CN`
将命令执行者传送到以TheWhiteDog9487和TheWhiteDog_CN当前所在位置为对角线的长方形区域内的随机一点
- `/rtp TheWhiteDog9487 TheWhiteDog_CN TheWhiteDog4568`
将TheWhiteDog4568随机传送到以TheWhiteDog9487和TheWhiteDog_CN当前所在位置为对角线的长方形区域内的随机一点
### 特别提示 ### 特别提示
/rtp <Radius(半径)> <Origin(随机中心实体)> 这种格式不存在。 `/rtp <随机半径> <作为随机中心实体>` 这种格式不存在。
因为第二个参数可能是被传送玩家名,也可能是做随机中心点的实体。 因为第二个参数可能是被传送玩家名,也可能是做随机中心点的实体。
这两种都是实体类型,无法区分到底是哪一种,存在歧义。 这两种都是实体类型,无法区分到底是哪一种,存在歧义。
同样,`/rtp <被传送玩家ID> <随机区域起点实体> <随机区域终点实体>` 这一组也是不存在的
三个参数都是实体类型,没办法区分`被传送玩家ID`是第一个还是第三个
# 依赖项 # 依赖项
由于我使用了fabric.api.command.v2中的CommandRegistrationCallback.EVENT来向游戏注册命令所以这个模组需要依赖Fabric API Fabric API
# 关于玩家权限 # 关于玩家权限
我参照原版的 /tp 命令,给 /rtp 设置了2级的权限要求。 我参照原版的 `/tp` 命令,给 `/rtp` 设置了2级的权限要求。
如果是原版或者类原版,你只需要让玩家有作弊的权限就可以用。 如果是原版或者类原版,你只需要让玩家有作弊的权限就可以用。
插件服务器方面那些具体的权限分配,因为我自己没玩过所以我也没法给出参考意见。 插件服务器方面那些具体的权限分配,因为我自己没玩过所以我也没法给出参考意见。
@@ -67,7 +90,7 @@
2. 单人游戏 + 开放局域网 2. 单人游戏 + 开放局域网
1. 使用客户端内置的服务器,开放局域网的那位玩家的客户端需要安装 1. 使用客户端内置的服务器,开放局域网的那位玩家的客户端需要安装
2. 其他加入游戏的玩家不需要安装 2. 其他加入游戏的玩家不需要安装
3. 使用独立服务器类似server.jar这种文件 3. 使用独立服务器(类似`server.jar`这种文件)
1. 服务器需要安装 1. 服务器需要安装
2. 客户端不需要 2. 客户端不需要
@@ -75,11 +98,11 @@
如果有查看多语言翻译文本的需求 ,那建议还是都装一下吧。** 如果有查看多语言翻译文本的需求 ,那建议还是都装一下吧。**
# 一些小彩蛋 # 一些小彩蛋
你可以使用 /随机传送 来替代 /rtp 你可以使用 `/随机传送` 来替代 `/rtp`
没错Minecraft的命令是可以存在非ASCII字符的所以我就整了一个 没错Minecraft的命令是可以存在非ASCII字符的所以我就整了一个
例如: 例如:
- /rtp TheWhiteDog9487 1000 - `/rtp TheWhiteDog9487 1000`
- /随机传送 TheWhiteDog9487 1000 - `/随机传送 TheWhiteDog9487 1000`
这两个命令的效果没有任何差别 这两个命令的效果没有任何差别
玩家权限限制和 /rtp 当然也是一样的都是2级 玩家权限限制和 `/rtp` 当然也是一样的都是2级

View File

@@ -1,5 +1,5 @@
plugins { plugins {
id("fabric-loom") version "1.14-SNAPSHOT" id("fabric-loom") version "1.15-SNAPSHOT"
id("maven-publish") id("maven-publish")
} }

View File

@@ -8,7 +8,7 @@ minecraft_version=1.21.11
loader_version=0.18.2 loader_version=0.18.2
# Mod Properties # Mod Properties
mod_version=0.4.0 mod_version=0.5.0
maven_group=xyz.thewhitedog9487 maven_group=xyz.thewhitedog9487
archives_base_name=RandomTeleporter archives_base_name=RandomTeleporter
@@ -17,4 +17,4 @@ fabric_version=0.139.4+1.21.11
# https://modrinth.com/mod/modmenu/versions # https://modrinth.com/mod/modmenu/versions
# https://maven.terraformersmc.com/releases/com/terraformersmc/modmenu # https://maven.terraformersmc.com/releases/com/terraformersmc/modmenu
modmenu_version=17.0.0-alpha.1 modmenu_version=17.0.0-beta.2

View File

@@ -18,7 +18,7 @@ import net.minecraft.core.BlockPos;
import net.minecraft.world.phys.Vec2; import net.minecraft.world.phys.Vec2;
import net.minecraft.world.level.Level; import net.minecraft.world.level.Level;
import net.minecraft.world.level.chunk.status.ChunkStatus; import net.minecraft.world.level.chunk.status.ChunkStatus;
import org.jetbrains.annotations.Nullable; import org.jspecify.annotations.Nullable;
import java.util.HashSet; import java.util.HashSet;
import java.util.Set; import java.util.Set;
@@ -88,7 +88,7 @@ public class CommandRegister {
// /rtp // /rtp
.requires(Commands.hasPermission(PermissionLevel)) .requires(Commands.hasPermission(PermissionLevel))
.executes(context -> ExecuteCommand( .executes(context -> ExecuteCommand(
context.getSource(),null,null, null)) context.getSource(),null,null, null, null, null))
// /rtp <Radius(半径)> // /rtp <Radius(半径)>
.then(argument(CommandArgumentName_Radius, IntegerArgumentType.integer(0)) .then(argument(CommandArgumentName_Radius, IntegerArgumentType.integer(0))
.requires(Commands.hasPermission(PermissionLevel)) .requires(Commands.hasPermission(PermissionLevel))
@@ -96,16 +96,20 @@ public class CommandRegister {
context.getSource(), context.getSource(),
IntegerArgumentType.getInteger(context, CommandArgumentName_Radius), IntegerArgumentType.getInteger(context, CommandArgumentName_Radius),
null, null,
null,
null,
null))) null)))
// /rtp <被传送玩家名(PlayerID)> // /rtp <PlayerID(被传送玩家名)>
.then(argument(CommandArgumentName_Target, EntityArgument.entity()) .then(argument(CommandArgumentName_Target, EntityArgument.entity())
.requires(Commands.hasPermission(PermissionLevel)) .requires(Commands.hasPermission(PermissionLevel))
.executes(context -> ExecuteCommand( .executes(context -> ExecuteCommand(
context.getSource(), context.getSource(),
null, null,
EntityArgument.getEntity(context,CommandArgumentName_Target), EntityArgument.getEntity(context,CommandArgumentName_Target),
null,
null,
null))) null)))
// /rtp <Radius(半径)> <被传送玩家名(PlayerID)> // /rtp <Radius(半径)> <PlayerID(被传送玩家名)>
.then(argument(CommandArgumentName_Radius, IntegerArgumentType.integer(0)) .then(argument(CommandArgumentName_Radius, IntegerArgumentType.integer(0))
.then(argument(CommandArgumentName_Target, EntityArgument.entity()) .then(argument(CommandArgumentName_Target, EntityArgument.entity())
.requires(Commands.hasPermission(PermissionLevel)) .requires(Commands.hasPermission(PermissionLevel))
@@ -113,8 +117,10 @@ public class CommandRegister {
context.getSource(), context.getSource(),
IntegerArgumentType.getInteger(context, CommandArgumentName_Radius), IntegerArgumentType.getInteger(context, CommandArgumentName_Radius),
EntityArgument.getEntity(context,CommandArgumentName_Target), EntityArgument.getEntity(context,CommandArgumentName_Target),
null,
null,
null)))) null))))
// /rtp <被传送玩家名(PlayerID)> <Radius(半径)> // /rtp <PlayerID(被传送玩家名)> <Radius(半径)>
.then(argument(CommandArgumentName_Target, EntityArgument.entity()) .then(argument(CommandArgumentName_Target, EntityArgument.entity())
.then(argument(CommandArgumentName_Radius, IntegerArgumentType.integer(0)) .then(argument(CommandArgumentName_Radius, IntegerArgumentType.integer(0))
.requires(Commands.hasPermission(PermissionLevel)) .requires(Commands.hasPermission(PermissionLevel))
@@ -122,6 +128,8 @@ public class CommandRegister {
context.getSource(), context.getSource(),
IntegerArgumentType.getInteger(context, CommandArgumentName_Radius), IntegerArgumentType.getInteger(context, CommandArgumentName_Radius),
EntityArgument.getEntity(context,CommandArgumentName_Target), EntityArgument.getEntity(context,CommandArgumentName_Target),
null,
null,
null)))) null))))
// // /rtp <Radius(半径)> <Origin(随机中心)> // // /rtp <Radius(半径)> <Origin(随机中心)>
// .then(argument("Radius(半径)", LongArgumentType.longArg()) // .then(argument("Radius(半径)", LongArgumentType.longArg())
@@ -140,8 +148,10 @@ public class CommandRegister {
context.getSource(), context.getSource(),
IntegerArgumentType.getInteger(context, CommandArgumentName_Radius), IntegerArgumentType.getInteger(context, CommandArgumentName_Radius),
null, null,
Vec2Argument.getVec2(context,CommandArgumentName_OriginPosition))))) Vec2Argument.getVec2(context,CommandArgumentName_OriginPosition),
// /rtp <Radius(半径)> <被传送玩家名(PlayerID)> <OriginEntity(随机中心,实体)> null,
null))))
// /rtp <Radius(半径)> <PlayerID(被传送玩家名)> <OriginEntity(随机中心,实体)>
.then(argument(CommandArgumentName_Radius, IntegerArgumentType.integer(0)) .then(argument(CommandArgumentName_Radius, IntegerArgumentType.integer(0))
.then(argument(CommandArgumentName_Target, EntityArgument.entity()) .then(argument(CommandArgumentName_Target, EntityArgument.entity())
.then(argument(CommandArgumentName_OriginEntity, EntityArgument.entity()) .then(argument(CommandArgumentName_OriginEntity, EntityArgument.entity())
@@ -151,8 +161,10 @@ public class CommandRegister {
IntegerArgumentType.getInteger(context, CommandArgumentName_Radius), IntegerArgumentType.getInteger(context, CommandArgumentName_Radius),
EntityArgument.getEntity(context,CommandArgumentName_Target), EntityArgument.getEntity(context,CommandArgumentName_Target),
new Vec2( (float) EntityArgument.getEntity( context,CommandArgumentName_OriginEntity).position().x, new Vec2( (float) EntityArgument.getEntity( context,CommandArgumentName_OriginEntity).position().x,
(float) EntityArgument.getEntity( context,CommandArgumentName_OriginEntity).position().z )))))) (float) EntityArgument.getEntity( context,CommandArgumentName_OriginEntity).position().z ),
// /rtp <Radius(半径)> <被传送玩家名(PlayerID)> <OriginPos(随机中心,坐标)> null,
null)))))
// /rtp <Radius(半径)> <PlayerID(被传送玩家名)> <OriginPos(随机中心,坐标)>
.then(argument(CommandArgumentName_Radius, IntegerArgumentType.integer(0)) .then(argument(CommandArgumentName_Radius, IntegerArgumentType.integer(0))
.then(argument(CommandArgumentName_Target, EntityArgument.entity()) .then(argument(CommandArgumentName_Target, EntityArgument.entity())
.then(argument(CommandArgumentName_OriginPosition, Vec2Argument.vec2()) .then(argument(CommandArgumentName_OriginPosition, Vec2Argument.vec2())
@@ -161,8 +173,10 @@ public class CommandRegister {
context.getSource(), context.getSource(),
IntegerArgumentType.getInteger(context, CommandArgumentName_Radius), IntegerArgumentType.getInteger(context, CommandArgumentName_Radius),
EntityArgument.getEntity(context,CommandArgumentName_Target), EntityArgument.getEntity(context,CommandArgumentName_Target),
Vec2Argument.getVec2(context,CommandArgumentName_OriginPosition)))))) Vec2Argument.getVec2(context,CommandArgumentName_OriginPosition),
// /rtp <被传送玩家名(PlayerID)> <Radius(半径)> <OriginEntity(随机中心,实体)> null,
null)))))
// /rtp <PlayerID(被传送玩家名)> <Radius(半径)> <OriginEntity(随机中心,实体)>
.then(argument(CommandArgumentName_Target, EntityArgument.entity()) .then(argument(CommandArgumentName_Target, EntityArgument.entity())
.then(argument(CommandArgumentName_Radius, IntegerArgumentType.integer(0)) .then(argument(CommandArgumentName_Radius, IntegerArgumentType.integer(0))
.then(argument(CommandArgumentName_OriginEntity, EntityArgument.entity()) .then(argument(CommandArgumentName_OriginEntity, EntityArgument.entity())
@@ -172,8 +186,10 @@ public class CommandRegister {
IntegerArgumentType.getInteger(context, CommandArgumentName_Radius), IntegerArgumentType.getInteger(context, CommandArgumentName_Radius),
EntityArgument.getEntity(context,CommandArgumentName_Target), EntityArgument.getEntity(context,CommandArgumentName_Target),
new Vec2( (float) EntityArgument.getEntity( context,CommandArgumentName_OriginEntity).position().x, new Vec2( (float) EntityArgument.getEntity( context,CommandArgumentName_OriginEntity).position().x,
(float) EntityArgument.getEntity( context,CommandArgumentName_OriginEntity).position().z )))))) (float) EntityArgument.getEntity( context,CommandArgumentName_OriginEntity).position().z ),
// /rtp <被传送玩家名(PlayerID)> <Radius(半径)> <OriginPos(随机中心,坐标)> null,
null)))))
// /rtp <PlayerID(被传送玩家名)> <Radius(半径)> <OriginPos(随机中心,坐标)>
.then(argument(CommandArgumentName_Target, EntityArgument.entity()) .then(argument(CommandArgumentName_Target, EntityArgument.entity())
.then(argument(CommandArgumentName_Radius, IntegerArgumentType.integer(0)) .then(argument(CommandArgumentName_Radius, IntegerArgumentType.integer(0))
.then(argument(CommandArgumentName_OriginPosition, Vec2Argument.vec2()) .then(argument(CommandArgumentName_OriginPosition, Vec2Argument.vec2())
@@ -182,7 +198,71 @@ public class CommandRegister {
context.getSource(), context.getSource(),
IntegerArgumentType.getInteger(context, CommandArgumentName_Radius), IntegerArgumentType.getInteger(context, CommandArgumentName_Radius),
EntityArgument.getEntity(context,CommandArgumentName_Target), EntityArgument.getEntity(context,CommandArgumentName_Target),
Vec2Argument.getVec2(context,CommandArgumentName_OriginPosition)))))) );});} Vec2Argument.getVec2(context,CommandArgumentName_OriginPosition),
null,
null)))))
// /rtp <RegionFrom(随机区域起点,坐标)> <RegionTo(随机区域终点,坐标)>
.then(argument(CommandArgumentName_RegionFromPosition, Vec2Argument.vec2())
.then(argument(CommandArgumentName_RegionToPosition, Vec2Argument.vec2())
.requires(Commands.hasPermission(PermissionLevel))
.executes(context -> ExecuteCommand(
context.getSource(),
null,
null,
null,
Vec2Argument.getVec2(context, CommandArgumentName_RegionFromPosition),
Vec2Argument.getVec2(context, CommandArgumentName_RegionToPosition)))))
// /rtp <RegionFrom(随机区域起点,坐标)> <RegionTo(随机区域终点,坐标)> <PlayerID(被传送玩家名)>
.then(argument(CommandArgumentName_RegionFromPosition, Vec2Argument.vec2())
.then(argument(CommandArgumentName_RegionToPosition, Vec2Argument.vec2())
.then(argument(CommandArgumentName_Target, EntityArgument.entity())
.requires(Commands.hasPermission(PermissionLevel))
.executes(context -> ExecuteCommand(
context.getSource(),
null,
EntityArgument.getEntity(context,CommandArgumentName_Target),
null,
Vec2Argument.getVec2(context, CommandArgumentName_RegionFromPosition),
Vec2Argument.getVec2(context, CommandArgumentName_RegionToPosition))))))
// /rtp <PlayerID(被传送玩家名)> <RegionFrom(随机区域起点,坐标)> <RegionTo(随机区域终点,坐标)>
.then(argument(CommandArgumentName_Target, EntityArgument.entity())
.then(argument(CommandArgumentName_RegionFromPosition, Vec2Argument.vec2())
.then(argument(CommandArgumentName_RegionToPosition, Vec2Argument.vec2())
.requires(Commands.hasPermission(PermissionLevel))
.executes(context -> ExecuteCommand(
context.getSource(),
null,
EntityArgument.getEntity(context,CommandArgumentName_Target),
null,
Vec2Argument.getVec2(context, CommandArgumentName_RegionFromPosition),
Vec2Argument.getVec2(context, CommandArgumentName_RegionToPosition))))))
// /rtp <RegionFrom(随机区域起点,实体)> <RegionTo(随机区域终点,实体)>
.then(argument(CommandArgumentName_RegionFromEntity, EntityArgument.entity())
.then(argument(CommandArgumentName_RegionToEntity, EntityArgument.entity())
.requires(Commands.hasPermission(PermissionLevel))
.executes(context -> ExecuteCommand(
context.getSource(),
null,
null,
null,
new Vec2( (float) EntityArgument.getEntity( context,CommandArgumentName_RegionFromEntity).position().x,
(float) EntityArgument.getEntity( context,CommandArgumentName_RegionFromEntity).position().z ),
new Vec2( (float) EntityArgument.getEntity( context,CommandArgumentName_RegionToEntity).position().x,
(float) EntityArgument.getEntity( context,CommandArgumentName_RegionToEntity).position().z ) ) )))
// /rtp <RegionFrom(随机区域起点,实体)> <RegionTo(随机区域终点,实体)> <PlayerID(被传送玩家名)>
.then(argument(CommandArgumentName_RegionFromEntity, EntityArgument.entity())
.then(argument(CommandArgumentName_RegionToEntity, EntityArgument.entity())
.then(argument(CommandArgumentName_Target, EntityArgument.entity())
.requires(Commands.hasPermission(PermissionLevel))
.executes(context -> ExecuteCommand(
context.getSource(),
null,
EntityArgument.getEntity(context,CommandArgumentName_Target),
null,
new Vec2( (float) EntityArgument.getEntity( context,CommandArgumentName_RegionFromEntity).position().x,
(float) EntityArgument.getEntity( context,CommandArgumentName_RegionFromEntity).position().z ),
new Vec2( (float) EntityArgument.getEntity( context,CommandArgumentName_RegionToEntity).position().x,
(float) EntityArgument.getEntity( context,CommandArgumentName_RegionToEntity).position().z )))))) );});}
/** /**
* 向游戏内注册命令 * 向游戏内注册命令
@@ -201,9 +281,11 @@ public class CommandRegister {
* @param Radius 随机选择的目的坐标距离参数 {@code Origin} 的最大距离 * @param Radius 随机选择的目的坐标距离参数 {@code Origin} 的最大距离
* @param Entity 被传送的实体 * @param Entity 被传送的实体
* @param Origin 随机选择的目的坐标的中心 * @param Origin 随机选择的目的坐标的中心
* @param RegionFrom 选择随机坐标的范围的起始坐标
* @param RegionTo 选择随机坐标的范围的结束坐标
* @return 命令运行是否成功 * @return 命令运行是否成功
*/ */
static int ExecuteCommand(CommandSourceStack Source, @Nullable Integer Radius, @Nullable Entity Entity, @Nullable Vec2 Origin){ static int ExecuteCommand(CommandSourceStack Source, @Nullable Integer Radius, @Nullable Entity Entity, @Nullable Vec2 Origin, @Nullable Vec2 RegionFrom, @Nullable Vec2 RegionTo){
Entity TargetEntity = Entity == null ? Source.getPlayer() : Entity; Entity TargetEntity = Entity == null ? Source.getPlayer() : Entity;
/* /*
@@ -215,34 +297,46 @@ public class CommandRegister {
*/ */
if (TargetEntity == null) { if (TargetEntity == null) {
Source.sendSuccess(()->{ return Component.translatableWithFallback("error.no_target","不存在被传送目标由非玩家物体执行命令时请显式指定被传送玩家ID"); }, true); Source.sendSuccess(()->{ return Component.translatableWithFallback("error.no_target","不存在被传送目标由非玩家物体执行命令时请显式指定被传送玩家ID"); }, true);
return -1;} return -1; }
if (Radius == null){Radius = (int) (WorldBorder - 1e4);}
// ↑ 远离世界边界
int Coordinate_X = 0; int Coordinate_X = 0;
int Coordinate_Z = 0; int Coordinate_Z = 0;
if (RegionFrom == null || RegionTo == null) { // ← 按半径和中心取随机
if (Radius == null) { Radius = (int) (WorldBorder - 1e4); }
// ↑ 远离世界边界
try { try {
if (Origin == null){ if (Origin == null) {
Coordinate_X = SR.nextInt(-Radius, Radius + 1); Coordinate_X = SR.nextInt(-Radius, Radius + 1);
Coordinate_Z = SR.nextInt(-Radius, Radius + 1); } Coordinate_Z = SR.nextInt(-Radius, Radius + 1); }
else{ else {
Coordinate_X = SR.nextInt((int) Origin.x - Radius, (int) Origin.x + Radius + 1); Coordinate_X = SR.nextInt((int) Origin.x - Radius, (int) Origin.x + Radius + 1);
Coordinate_Z = SR.nextInt((int) Origin.y - Radius, (int) Origin.y + Radius + 1); } } Coordinate_Z = SR.nextInt((int) Origin.y - Radius, (int) Origin.y + Radius + 1); } }
catch (IllegalArgumentException e) { catch (IllegalArgumentException e) {
// 半径为零 // 半径为零
if (Origin == null) { if (Origin == null) {
Source.sendSuccess(()->{ return Component.translatableWithFallback("warning.radius_equal_zero_no_target", "由于你设置的随机半径为0并且未设置随机中心点坐标因此什么都不会发生"); },true); Source.sendSuccess(() -> {
return -1;} return Component.translatableWithFallback("warning.radius_equal_zero_no_target", "由于你设置的随机半径为0并且未设置随机中心点坐标因此什么都不会发生"); }, true);
return -1; }
else { else {
Coordinate_X = (int) Origin.x; Coordinate_X = (int) Origin.x;
Coordinate_Z = (int) Origin.y; Coordinate_Z = (int) Origin.y;
int finalCoordinate_X1 = Coordinate_X; int finalCoordinate_X1 = Coordinate_X;
int finalCoordinate_Z1 = Coordinate_Z; int finalCoordinate_Z1 = Coordinate_Z;
// ↑ "lambda 表达式中使用的变量应为 final 或有效 final" // ↑ "lambda 表达式中使用的变量应为 final 或有效 final"
Source.sendSuccess(()->{ return Component.translatableWithFallback("warning.radius_equal_zero", "警告由于你设置的随机半径为0因此在选择出合适高度之后将直接把你传送至%d %d", finalCoordinate_X1, finalCoordinate_Z1); },true);}} Source.sendSuccess(() -> {
return Component.translatableWithFallback("warning.radius_equal_zero", "警告由于你设置的随机半径为0因此在选择出合适高度之后将直接把你传送至%d %d", finalCoordinate_X1, finalCoordinate_Z1); }, true); } } }
else { // ← 按范围取随机
Coordinate_X = SR.nextInt( Math.min( (int)RegionFrom.x, (int)RegionTo.x ), Math.max( (int)RegionFrom.x, (int)RegionTo.x ) + 1 );
Coordinate_Z = SR.nextInt( Math.min( (int)RegionFrom.y, (int)RegionTo.y ), Math.max( (int)RegionFrom.y, (int)RegionTo.y ) + 1 ); }
if (Coordinate_X >= WorldBorder){ Coordinate_X = (int) (WorldBorder - 1e4); }
else if (Coordinate_X <= -WorldBorder){ Coordinate_X = (int) (-WorldBorder + 1e4); }
if (Coordinate_Z >= WorldBorder){ Coordinate_Z = (int) (WorldBorder - 1e4); }
else if (Coordinate_Z <= -WorldBorder){ Coordinate_Z = (int) (-WorldBorder + 1e4); }
// ↑ 远离世界边界
Level EntityWorld = TargetEntity.level(); Level EntityWorld = TargetEntity.level();
EntityWorld.getChunk(Coordinate_X >> 4, Coordinate_Z >> 4, ChunkStatus.FULL, true); EntityWorld.getChunk(Coordinate_X >> 4, Coordinate_Z >> 4, ChunkStatus.FULL, true);
// ↑ 加载目标区块不然下面获取到的Coordinate_Y一定是-64 // ↑ 加载目标区块不然下面获取到的Coordinate_Y一定是-64
// 必须是 >> 4 ,不能是 / 16 // 按位右移4和除以16在被除数小于0时的结果不一致一个是向上取整一个是向下取整
// 因此必须是 >> 4 ,不能是 / 16
int Coordinate_Y = EntityWorld.getHeight(MOTION_BLOCKING_NO_LEAVES, Coordinate_X, Coordinate_Z); int Coordinate_Y = EntityWorld.getHeight(MOTION_BLOCKING_NO_LEAVES, Coordinate_X, Coordinate_Z);
BlockPos.MutableBlockPos BlockPos = new BlockPos.MutableBlockPos(); BlockPos.MutableBlockPos BlockPos = new BlockPos.MutableBlockPos();
for (int x = -1; x <= 1; x++) { for (int x = -1; x <= 1; x++) {
@@ -261,5 +355,5 @@ public class CommandRegister {
// ↑ "lambda 表达式中使用的变量应为 final 或有效 final" // ↑ "lambda 表达式中使用的变量应为 final 或有效 final"
final var FeedbackFallbackString = String.format("已将玩家%s传送到%d %d %d", TargetEntity.getName().getString(), Coordinate_X, Coordinate_Y, Coordinate_Z); final var FeedbackFallbackString = String.format("已将玩家%s传送到%d %d %d", TargetEntity.getName().getString(), Coordinate_X, Coordinate_Y, Coordinate_Z);
Source.sendSuccess(()->{ return Component.translatableWithFallback("info.success", FeedbackFallbackString, TargetEntity.getName(), finalCoordinate_X, Coordinate_Y, finalCoordinate_Z); },true); Source.sendSuccess(()->{ return Component.translatableWithFallback("info.success", FeedbackFallbackString, TargetEntity.getName(), finalCoordinate_X, Coordinate_Y, finalCoordinate_Z); },true);
return Command.SINGLE_SUCCESS;} return Command.SINGLE_SUCCESS; }
} }

View File

@@ -4,6 +4,7 @@ import net.fabricmc.fabric.api.resource.v1.ResourceLoader;
import net.fabricmc.fabric.api.resource.v1.reloader.SimpleResourceReloader; import net.fabricmc.fabric.api.resource.v1.reloader.SimpleResourceReloader;
import net.minecraft.network.chat.Component; import net.minecraft.network.chat.Component;
import net.minecraft.resources.Identifier; import net.minecraft.resources.Identifier;
import org.jspecify.annotations.NonNull;
import static net.minecraft.server.packs.PackType.CLIENT_RESOURCES; import static net.minecraft.server.packs.PackType.CLIENT_RESOURCES;
import static xyz.thewhitedog9487.RandomTeleporter.MOD_ID; import static xyz.thewhitedog9487.RandomTeleporter.MOD_ID;
@@ -13,9 +14,13 @@ import static xyz.thewhitedog9487.RandomTeleporter.MOD_ID;
*/ */
public class ResourceReloaderListener { public class ResourceReloaderListener {
public static String CommandArgumentName_Radius = "Radius(半径)"; public static String CommandArgumentName_Radius = "Radius(半径)";
public static String CommandArgumentName_Target = "被传送玩家名(PlayerID)"; public static String CommandArgumentName_Target = "PlayerID(被传送玩家名)";
public static String CommandArgumentName_OriginPosition = "OriginPos(随机中心,坐标)"; public static String CommandArgumentName_OriginPosition = "OriginPos(随机中心,坐标)";
public static String CommandArgumentName_OriginEntity = "OriginEntity(随机中心,实体)"; public static String CommandArgumentName_OriginEntity = "OriginEntity(随机中心,实体)";
public static String CommandArgumentName_RegionFromPosition = "RegionFrom(随机范围起始位置,坐标)";
public static String CommandArgumentName_RegionToPosition = "RegionTo(随机范围结束位置,坐标)";
public static String CommandArgumentName_RegionFromEntity = "RegionFrom(随机范围起始位置,实体)";
public static String CommandArgumentName_RegionToEntity = "RegionTo(随机范围结束位置,实体)";
public static void Register(){ public static void Register(){
ResourceLoader.get(CLIENT_RESOURCES).registerReloader(Identifier.fromNamespaceAndPath(MOD_ID, "translate_text_apply"), new SimpleResourceReloader<Void>() { ResourceLoader.get(CLIENT_RESOURCES).registerReloader(Identifier.fromNamespaceAndPath(MOD_ID, "translate_text_apply"), new SimpleResourceReloader<Void>() {
@@ -23,12 +28,16 @@ public class ResourceReloaderListener {
* 用不到,不用管 * 用不到,不用管
*/ */
@Override @Override
protected Void prepare(SharedState store) { protected Void prepare(@NonNull SharedState store) {
return null; } return null; }
@Override @Override
protected void apply(Void prepared, SharedState store) { protected void apply(Void prepared, @NonNull SharedState store) {
CommandArgumentName_Radius = Component.translatableWithFallback("command.argument.radius", "Radius(半径)").getString(); CommandArgumentName_Radius = Component.translatableWithFallback("command.argument.radius", "Radius(半径)").getString();
CommandArgumentName_Target = Component.translatableWithFallback("command.argument.target", "被传送玩家名(PlayerID)").getString(); CommandArgumentName_Target = Component.translatableWithFallback("command.argument.target", "PlayerID(被传送玩家名)").getString();
CommandArgumentName_OriginPosition = Component.translatableWithFallback("command.argument.origin_pos", "OriginPos(随机中心,坐标)").getString(); CommandArgumentName_OriginPosition = Component.translatableWithFallback("command.argument.origin_pos", "OriginPos(随机中心,坐标)").getString();
CommandArgumentName_OriginEntity = Component.translatableWithFallback("command.argument.origin_entity", "OriginEntity(随机中心,实体)").getString(); } } ); } } CommandArgumentName_OriginEntity = Component.translatableWithFallback("command.argument.origin_entity", "OriginEntity(随机中心,实体)").getString();
CommandArgumentName_RegionFromPosition = Component.translatableWithFallback("command.argument.region_from_pos", "RegionFrom(随机范围起始位置,坐标)").getString();
CommandArgumentName_RegionToPosition = Component.translatableWithFallback("command.argument.region_to_pos", "RegionTo(随机范围结束位置,坐标)").getString();
CommandArgumentName_RegionFromEntity = Component.translatableWithFallback("command.argument.region_from_entity", "RegionFrom(随机范围起始位置,实体)").getString();
CommandArgumentName_RegionToEntity = Component.translatableWithFallback("command.argument.region_to_entity", "RegionTo(随机范围结束位置,实体)").getString(); } } ); } }

View File

@@ -9,5 +9,9 @@
"command.argument.radius": "radius", "command.argument.radius": "radius",
"command.argument.target": "teleported entity", "command.argument.target": "teleported entity",
"command.argument.origin_pos": "origin(pos)", "command.argument.origin_pos": "origin(pos)",
"command.argument.origin_entity": "origin(entity)" "command.argument.origin_entity": "origin(entity)",
"command.argument.region_from_pos": "region start(pos)",
"command.argument.region_to_pos": "region end(pos)",
"command.argument.region_from_entity": "region start(entity)",
"command.argument.region_to_entity": "region end(entity)"
} }

View File

@@ -9,5 +9,9 @@
"command.argument.radius": "半径", "command.argument.radius": "半径",
"command.argument.target": "被传送实体", "command.argument.target": "被传送实体",
"command.argument.origin_pos": "随机中心(坐标)", "command.argument.origin_pos": "随机中心(坐标)",
"command.argument.origin_entity": "随机中心(实体)" "command.argument.origin_entity": "随机中心(实体)",
"command.argument.region_from_pos": "坐标选取区域起点(坐标)",
"command.argument.region_to_pos": "坐标选取区域终点(坐标)",
"command.argument.region_from_entity": "坐标选取区域起点(实体)",
"command.argument.region_to_entity": "坐标选取区域终点(实体)"
} }