mirror of
https://github.com/LeavesMC/Leaves.git
synced 2025-12-19 14:59:32 +00:00
543 lines
36 KiB
Diff
543 lines
36 KiB
Diff
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
From: violetc <58360096+s-yh-china@users.noreply.github.com>
|
|
Date: Tue, 4 Feb 2025 19:45:19 +0800
|
|
Subject: [PATCH] Replay Mod API
|
|
|
|
This patch is Powered by ReplayMod(https://github.com/ReplayMod)
|
|
|
|
diff --git a/net/minecraft/commands/CommandSourceStack.java b/net/minecraft/commands/CommandSourceStack.java
|
|
index 3acfb2a78845dd8081dc3c01d653034232c76e60..efe3f1849e68e5bbe2cdb3793dafc8b58c3d8415 100644
|
|
--- a/net/minecraft/commands/CommandSourceStack.java
|
|
+++ b/net/minecraft/commands/CommandSourceStack.java
|
|
@@ -580,7 +580,7 @@ public class CommandSourceStack implements ExecutionCommandSource<CommandSourceS
|
|
|
|
@Override
|
|
public Collection<String> getOnlinePlayerNames() {
|
|
- return this.entity instanceof ServerPlayer sourcePlayer && !sourcePlayer.getBukkitEntity().hasPermission("paper.bypass-visibility.tab-completion") ? this.getServer().getPlayerList().getPlayers().stream().filter(serverPlayer -> sourcePlayer.getBukkitEntity().canSee(serverPlayer.getBukkitEntity())).map(serverPlayer -> serverPlayer.getGameProfile().getName()).toList() : Lists.newArrayList(this.server.getPlayerNames()); // Paper - Make CommandSourceStack respect hidden players
|
|
+ return this.entity instanceof ServerPlayer sourcePlayer && !(sourcePlayer instanceof org.leavesmc.leaves.replay.ServerPhotographer) && !sourcePlayer.getBukkitEntity().hasPermission("paper.bypass-visibility.tab-completion") ? this.getServer().getPlayerList().getPlayers().stream().filter(serverPlayer -> sourcePlayer.getBukkitEntity().canSee(serverPlayer.getBukkitEntity())).map(serverPlayer -> serverPlayer.getGameProfile().getName()).toList() : Lists.newArrayList(this.server.getPlayerNames()); // Paper - Make CommandSourceStack respect hidden players // Leaves - only real player
|
|
}
|
|
|
|
@Override
|
|
diff --git a/net/minecraft/commands/arguments/selector/EntitySelector.java b/net/minecraft/commands/arguments/selector/EntitySelector.java
|
|
index 514f8fbdeb776087608665c35de95294aadf5cf0..b75772897cabc3e7c59301d451685378fa55b6c3 100644
|
|
--- a/net/minecraft/commands/arguments/selector/EntitySelector.java
|
|
+++ b/net/minecraft/commands/arguments/selector/EntitySelector.java
|
|
@@ -128,11 +128,12 @@ public class EntitySelector {
|
|
return this.findPlayers(source);
|
|
} else if (this.playerName != null) {
|
|
ServerPlayer playerByName = source.getServer().getPlayerList().getPlayerByName(this.playerName);
|
|
+ playerByName = playerByName instanceof org.leavesmc.leaves.replay.ServerPhotographer ? null : playerByName; // Leaves - skip photographer
|
|
return playerByName == null ? List.of() : List.of(playerByName);
|
|
} else if (this.entityUUID != null) {
|
|
for (ServerLevel serverLevel : source.getServer().getAllLevels()) {
|
|
Entity entity = serverLevel.getEntity(this.entityUUID);
|
|
- if (entity != null) {
|
|
+ if (entity != null && !(entity instanceof org.leavesmc.leaves.replay.ServerPhotographer)) {
|
|
if (entity.getType().isEnabled(source.enabledFeatures())) {
|
|
return List.of(entity);
|
|
}
|
|
@@ -146,7 +147,7 @@ public class EntitySelector {
|
|
AABB absoluteAabb = this.getAbsoluteAabb(vec3);
|
|
if (this.currentEntity) {
|
|
Predicate<Entity> predicate = this.getPredicate(vec3, absoluteAabb, null);
|
|
- return source.getEntity() != null && predicate.test(source.getEntity()) ? List.of(source.getEntity()) : List.of();
|
|
+ return source.getEntity() != null && !(source.getEntity() instanceof org.leavesmc.leaves.replay.ServerPhotographer) && predicate.test(source.getEntity()) ? List.of(source.getEntity()) : List.of(); // Leaves - skip photographer
|
|
} else {
|
|
Predicate<Entity> predicate = this.getPredicate(vec3, absoluteAabb, source.enabledFeatures());
|
|
List<Entity> list = new ObjectArrayList<>();
|
|
@@ -157,6 +158,7 @@ public class EntitySelector {
|
|
this.addEntities(list, serverLevel1, absoluteAabb, predicate);
|
|
}
|
|
}
|
|
+ list.removeIf(entity -> entity instanceof org.leavesmc.leaves.replay.ServerPhotographer); // Leaves - skip photographer
|
|
|
|
return this.sortAndLimit(vec3, list);
|
|
}
|
|
@@ -192,9 +194,11 @@ public class EntitySelector {
|
|
this.checkPermissions(source);
|
|
if (this.playerName != null) {
|
|
ServerPlayer playerByName = source.getServer().getPlayerList().getPlayerByName(this.playerName);
|
|
+ playerByName = playerByName instanceof org.leavesmc.leaves.replay.ServerPhotographer ? null : playerByName; // Leaves - skip photographer
|
|
return playerByName == null ? List.of() : List.of(playerByName);
|
|
} else if (this.entityUUID != null) {
|
|
ServerPlayer playerByName = source.getServer().getPlayerList().getPlayer(this.entityUUID);
|
|
+ playerByName = playerByName instanceof org.leavesmc.leaves.replay.ServerPhotographer ? null : playerByName; // Leaves - skip photographer
|
|
return playerByName == null ? List.of() : List.of(playerByName);
|
|
} else {
|
|
Vec3 vec3 = this.position.apply(source.getPosition());
|
|
@@ -206,11 +210,11 @@ public class EntitySelector {
|
|
int resultLimit = this.getResultLimit();
|
|
List<ServerPlayer> players;
|
|
if (this.isWorldLimited()) {
|
|
- players = source.getLevel().getPlayers(predicate, resultLimit);
|
|
+ players = source.getLevel().getPlayers((player -> !(player instanceof org.leavesmc.leaves.replay.ServerPhotographer) && predicate.test(player)), resultLimit); // Leaves - skip photographer
|
|
} else {
|
|
players = new ObjectArrayList<>();
|
|
|
|
- for (ServerPlayer serverPlayer1 : source.getServer().getPlayerList().getPlayers()) {
|
|
+ for (ServerPlayer serverPlayer1 : source.getServer().getPlayerList().realPlayers) { // Leaves - only real players
|
|
if (predicate.test(serverPlayer1)) {
|
|
players.add(serverPlayer1);
|
|
if (players.size() >= resultLimit) {
|
|
diff --git a/net/minecraft/server/MinecraftServer.java b/net/minecraft/server/MinecraftServer.java
|
|
index 39898e5e2bd75eced05f0b90b3881270d144fd93..2b29a8cdf8956e13143a59c16fcee2f5c9b49086 100644
|
|
--- a/net/minecraft/server/MinecraftServer.java
|
|
+++ b/net/minecraft/server/MinecraftServer.java
|
|
@@ -1638,7 +1638,7 @@ public abstract class MinecraftServer extends ReentrantBlockableEventLoop<TickTa
|
|
}
|
|
|
|
private ServerStatus.Players buildPlayerStatus() {
|
|
- List<ServerPlayer> players = this.playerList.getPlayers();
|
|
+ List<ServerPlayer> players = this.playerList.realPlayers; // Leaves - only real player
|
|
int maxPlayers = this.getMaxPlayers();
|
|
if (this.hidesOnlinePlayers()) {
|
|
return new ServerStatus.Players(maxPlayers, players.size(), List.of());
|
|
@@ -1852,7 +1852,7 @@ public abstract class MinecraftServer extends ReentrantBlockableEventLoop<TickTa
|
|
|
|
@Override
|
|
public int getPlayerCount() {
|
|
- return this.playerList.getPlayerCount();
|
|
+ return this.playerList.realPlayers.size(); // Leaves - only real player
|
|
}
|
|
|
|
@Override
|
|
@@ -2246,7 +2246,7 @@ public abstract class MinecraftServer extends ReentrantBlockableEventLoop<TickTa
|
|
if (Thread.currentThread() != this.serverThread) return; // Paper
|
|
// Paper start - we don't need to save everything, just advancements
|
|
// this.getPlayerList().saveAll();
|
|
- for (final ServerPlayer player : this.getPlayerList().getPlayers()) {
|
|
+ for (final ServerPlayer player : this.getPlayerList().realPlayers) { // Leaves - only real players
|
|
player.getAdvancements().save();
|
|
}
|
|
// Paper end - we don't need to save everything, just advancements
|
|
@@ -2385,7 +2385,7 @@ public abstract class MinecraftServer extends ReentrantBlockableEventLoop<TickTa
|
|
if (!playerList.isUsingWhitelist()) return; // Paper - whitelist not enabled
|
|
UserWhiteList whiteList = playerList.getWhiteList();
|
|
|
|
- for (ServerPlayer serverPlayer : Lists.newArrayList(playerList.getPlayers())) {
|
|
+ for (ServerPlayer serverPlayer : Lists.newArrayList(playerList.realPlayers)) { // Leaves - only real player
|
|
if (!whiteList.isWhiteListed(serverPlayer.getGameProfile()) && !this.getPlayerList().isOp(serverPlayer.getGameProfile())) { // Paper - Fix kicking ops when whitelist is reloaded (MC-171420)
|
|
serverPlayer.connection.disconnect(net.kyori.adventure.text.Component.text(org.spigotmc.SpigotConfig.whitelistMessage), org.bukkit.event.player.PlayerKickEvent.Cause.WHITELIST); // Paper - use configurable message & kick event cause
|
|
}
|
|
diff --git a/net/minecraft/server/PlayerAdvancements.java b/net/minecraft/server/PlayerAdvancements.java
|
|
index a14401edec04964e6c596c16ba58643b835ef9c1..924829d46f789752aab52c0a51f4d0925710eadf 100644
|
|
--- a/net/minecraft/server/PlayerAdvancements.java
|
|
+++ b/net/minecraft/server/PlayerAdvancements.java
|
|
@@ -173,7 +173,7 @@ public class PlayerAdvancements {
|
|
}
|
|
// Leaves end - spectator don't get advancement
|
|
// Leaves start - bot can't get advancement
|
|
- if (player instanceof org.leavesmc.leaves.bot.ServerBot) {
|
|
+ if (player instanceof org.leavesmc.leaves.bot.ServerBot || player instanceof org.leavesmc.leaves.replay.ServerPhotographer) { // Leaves - and photographer
|
|
return false;
|
|
}
|
|
// Leaves end - bot can't get advancement
|
|
diff --git a/net/minecraft/server/ServerScoreboard.java b/net/minecraft/server/ServerScoreboard.java
|
|
index 5c9f2a8476ff260ec7f0843f7c956c71e9b0d34d..22d67e72e0bf2dbe8c7a936260bb3577c2cfa10b 100644
|
|
--- a/net/minecraft/server/ServerScoreboard.java
|
|
+++ b/net/minecraft/server/ServerScoreboard.java
|
|
@@ -244,7 +244,7 @@ public class ServerScoreboard extends Scoreboard {
|
|
public void startTrackingObjective(Objective objective) {
|
|
List<Packet<?>> startTrackingPackets = this.getStartTrackingPackets(objective);
|
|
|
|
- for (ServerPlayer serverPlayer : this.server.getPlayerList().getPlayers()) {
|
|
+ for (ServerPlayer serverPlayer : this.server.getPlayerList().realPlayers) { // Leaves - only real players
|
|
if (serverPlayer.getBukkitEntity().getScoreboard().getHandle() != this) continue; // CraftBukkit - Only players on this board
|
|
for (Packet<?> packet : startTrackingPackets) {
|
|
serverPlayer.connection.send(packet);
|
|
@@ -270,7 +270,7 @@ public class ServerScoreboard extends Scoreboard {
|
|
public void stopTrackingObjective(Objective objective) {
|
|
List<Packet<?>> stopTrackingPackets = this.getStopTrackingPackets(objective);
|
|
|
|
- for (ServerPlayer serverPlayer : this.server.getPlayerList().getPlayers()) {
|
|
+ for (ServerPlayer serverPlayer : this.server.getPlayerList().realPlayers) { // Leaves - only real players
|
|
if (serverPlayer.getBukkitEntity().getScoreboard().getHandle() != this) continue; // CraftBukkit - Only players on this board
|
|
for (Packet<?> packet : stopTrackingPackets) {
|
|
serverPlayer.connection.send(packet);
|
|
diff --git a/net/minecraft/server/commands/DefaultGameModeCommands.java b/net/minecraft/server/commands/DefaultGameModeCommands.java
|
|
index 84baf29ad8af6a8af0cd00ce3921b3c5b8b89451..17f5543b802d88811c3e373e579a6e8b4c9f263a 100644
|
|
--- a/net/minecraft/server/commands/DefaultGameModeCommands.java
|
|
+++ b/net/minecraft/server/commands/DefaultGameModeCommands.java
|
|
@@ -27,7 +27,7 @@ public class DefaultGameModeCommands {
|
|
server.setDefaultGameType(gamemode);
|
|
GameType forcedGameType = server.getForcedGameType();
|
|
if (forcedGameType != null) {
|
|
- for (ServerPlayer serverPlayer : server.getPlayerList().getPlayers()) {
|
|
+ for (ServerPlayer serverPlayer : server.getPlayerList().realPlayers) { // Leaves - only real players
|
|
// Paper start - Expand PlayerGameModeChangeEvent
|
|
org.bukkit.event.player.PlayerGameModeChangeEvent event = serverPlayer.setGameMode(gamemode, org.bukkit.event.player.PlayerGameModeChangeEvent.Cause.DEFAULT_GAMEMODE, net.kyori.adventure.text.Component.empty());
|
|
if (event != null && event.isCancelled()) {
|
|
diff --git a/net/minecraft/server/commands/ListPlayersCommand.java b/net/minecraft/server/commands/ListPlayersCommand.java
|
|
index c6ae34f91b3629990294fc5e69237a1e600ef038..2109b0a2d4099e64c34cd1c45b83f72654d3b615 100644
|
|
--- a/net/minecraft/server/commands/ListPlayersCommand.java
|
|
+++ b/net/minecraft/server/commands/ListPlayersCommand.java
|
|
@@ -33,7 +33,7 @@ public class ListPlayersCommand {
|
|
private static int format(CommandSourceStack source, Function<ServerPlayer, Component> nameExtractor) {
|
|
PlayerList playerList = source.getServer().getPlayerList();
|
|
// CraftBukkit start
|
|
- List<ServerPlayer> playersTemp = playerList.getPlayers();
|
|
+ List<ServerPlayer> playersTemp = playerList.realPlayers;
|
|
if (source.getBukkitSender() instanceof org.bukkit.entity.Player) {
|
|
org.bukkit.entity.Player sender = (org.bukkit.entity.Player) source.getBukkitSender();
|
|
playersTemp = playersTemp.stream().filter((ep) -> sender.canSee(ep.getBukkitEntity())).collect(java.util.stream.Collectors.toList());
|
|
diff --git a/net/minecraft/server/commands/OpCommand.java b/net/minecraft/server/commands/OpCommand.java
|
|
index f2286b96b8f40b4588f817913c42ae7b4a92340f..dbe6c37642d35ac6ee8b428cf1e45878a5dfa9da 100644
|
|
--- a/net/minecraft/server/commands/OpCommand.java
|
|
+++ b/net/minecraft/server/commands/OpCommand.java
|
|
@@ -25,7 +25,7 @@ public class OpCommand {
|
|
(commandContext, suggestionsBuilder) -> {
|
|
PlayerList playerList = commandContext.getSource().getServer().getPlayerList();
|
|
return SharedSuggestionProvider.suggest(
|
|
- playerList.getPlayers()
|
|
+ playerList.realPlayers // Leaves - only real player
|
|
.stream()
|
|
.filter(serverPlayer -> !playerList.isOp(serverPlayer.getGameProfile()))
|
|
.map(serverPlayer -> serverPlayer.getGameProfile().getName()),
|
|
diff --git a/net/minecraft/server/commands/ParticleCommand.java b/net/minecraft/server/commands/ParticleCommand.java
|
|
index 33d96239f4b72a5587dc70f9602847a870d6d6a5..a83ce2cd112fca02cf3545f8c38e5cafae2c7c0e 100644
|
|
--- a/net/minecraft/server/commands/ParticleCommand.java
|
|
+++ b/net/minecraft/server/commands/ParticleCommand.java
|
|
@@ -36,7 +36,7 @@ public class ParticleCommand {
|
|
0.0F,
|
|
0,
|
|
false,
|
|
- commandContext.getSource().getServer().getPlayerList().getPlayers()
|
|
+ commandContext.getSource().getServer().getPlayerList().realPlayers // Leaves - only real player
|
|
)
|
|
)
|
|
.then(
|
|
@@ -50,7 +50,7 @@ public class ParticleCommand {
|
|
0.0F,
|
|
0,
|
|
false,
|
|
- context1.getSource().getServer().getPlayerList().getPlayers()
|
|
+ context1.getSource().getServer().getPlayerList().realPlayers // Leaves - only real player
|
|
)
|
|
)
|
|
.then(
|
|
@@ -68,7 +68,7 @@ public class ParticleCommand {
|
|
FloatArgumentType.getFloat(context1, "speed"),
|
|
IntegerArgumentType.getInteger(context1, "count"),
|
|
false,
|
|
- context1.getSource().getServer().getPlayerList().getPlayers()
|
|
+ context1.getSource().getServer().getPlayerList().realPlayers // Leaves - only real player
|
|
)
|
|
)
|
|
.then(
|
|
@@ -82,7 +82,7 @@ public class ParticleCommand {
|
|
FloatArgumentType.getFloat(context1, "speed"),
|
|
IntegerArgumentType.getInteger(context1, "count"),
|
|
true,
|
|
- context1.getSource().getServer().getPlayerList().getPlayers()
|
|
+ context1.getSource().getServer().getPlayerList().realPlayers // Leaves - only real player
|
|
)
|
|
)
|
|
.then(
|
|
@@ -112,7 +112,7 @@ public class ParticleCommand {
|
|
FloatArgumentType.getFloat(context1, "speed"),
|
|
IntegerArgumentType.getInteger(context1, "count"),
|
|
false,
|
|
- context1.getSource().getServer().getPlayerList().getPlayers()
|
|
+ context1.getSource().getServer().getPlayerList().realPlayers // Leaves - only real player
|
|
)
|
|
)
|
|
.then(
|
|
diff --git a/net/minecraft/server/commands/TeamMsgCommand.java b/net/minecraft/server/commands/TeamMsgCommand.java
|
|
index 134d7b1a9d5a5a47ebf4aabff110dde914cd6fe1..894dd1d048904b8a775416ea6cb3112215c567bc 100644
|
|
--- a/net/minecraft/server/commands/TeamMsgCommand.java
|
|
+++ b/net/minecraft/server/commands/TeamMsgCommand.java
|
|
@@ -40,7 +40,7 @@ public class TeamMsgCommand {
|
|
} else {
|
|
List<ServerPlayer> list = commandSourceStack.getServer()
|
|
.getPlayerList()
|
|
- .getPlayers()
|
|
+ .realPlayers // Leaves - only real players
|
|
.stream()
|
|
.filter(player -> player == entityOrException || player.getTeam() == team)
|
|
.toList();
|
|
diff --git a/net/minecraft/server/commands/WhitelistCommand.java b/net/minecraft/server/commands/WhitelistCommand.java
|
|
index 763a51d9a0859296eb4ea52b6879d6c7703db673..25b4fcf79dbf3e227451b3321d8fd38bab55ffe9 100644
|
|
--- a/net/minecraft/server/commands/WhitelistCommand.java
|
|
+++ b/net/minecraft/server/commands/WhitelistCommand.java
|
|
@@ -43,7 +43,7 @@ public class WhitelistCommand {
|
|
(commandContext, suggestionsBuilder) -> {
|
|
PlayerList playerList = commandContext.getSource().getServer().getPlayerList();
|
|
return SharedSuggestionProvider.suggest(
|
|
- playerList.getPlayers()
|
|
+ playerList.realPlayers // Leaves - only real player
|
|
.stream()
|
|
.filter(serverPlayer -> !playerList.getWhiteList().isWhiteListed(serverPlayer.getGameProfile()))
|
|
.map(serverPlayer -> serverPlayer.getGameProfile().getName()),
|
|
diff --git a/net/minecraft/server/gui/PlayerListComponent.java b/net/minecraft/server/gui/PlayerListComponent.java
|
|
index f5ba0c9a4c3f9eaa38eeb689de915c25c7165433..24bbc32bc17802edbd9cc14310fe8141c0ad85b0 100644
|
|
--- a/net/minecraft/server/gui/PlayerListComponent.java
|
|
+++ b/net/minecraft/server/gui/PlayerListComponent.java
|
|
@@ -17,8 +17,8 @@ public class PlayerListComponent extends JList<String> {
|
|
if (this.tickCount++ % 20 == 0) {
|
|
Vector<String> list = new Vector<>();
|
|
|
|
- for (int i = 0; i < this.server.getPlayerList().getPlayers().size(); i++) {
|
|
- list.add(this.server.getPlayerList().getPlayers().get(i).getGameProfile().getName());
|
|
+ for (int i = 0; i < this.server.getPlayerList().realPlayers.size(); i++) { // Leaves - only real players
|
|
+ list.add(this.server.getPlayerList().realPlayers.get(i).getGameProfile().getName()); // Leaves - only real players
|
|
}
|
|
|
|
this.setListData(list);
|
|
diff --git a/net/minecraft/server/level/ServerLevel.java b/net/minecraft/server/level/ServerLevel.java
|
|
index e6de7ef46d197c14495d4b55d094af34816fc063..5072dc7ac71e1640b2aad35c3c3560e0860ece94 100644
|
|
--- a/net/minecraft/server/level/ServerLevel.java
|
|
+++ b/net/minecraft/server/level/ServerLevel.java
|
|
@@ -2693,7 +2693,7 @@ public class ServerLevel extends Level implements ServerEntityGetter, WorldGenLe
|
|
if (entity instanceof ServerPlayer serverPlayer) {
|
|
ServerLevel.this.players.add(serverPlayer);
|
|
// Leaves start - skip
|
|
- if (!(serverPlayer instanceof org.leavesmc.leaves.bot.ServerBot)) {
|
|
+ if (!(serverPlayer instanceof org.leavesmc.leaves.bot.ServerBot) && !(serverPlayer instanceof org.leavesmc.leaves.replay.ServerPhotographer)) { // and photographer
|
|
ServerLevel.this.realPlayers.add(serverPlayer);
|
|
}
|
|
// Leaves end - skip
|
|
@@ -2776,7 +2776,7 @@ public class ServerLevel extends Level implements ServerEntityGetter, WorldGenLe
|
|
if (entity instanceof ServerPlayer serverPlayer) {
|
|
ServerLevel.this.players.remove(serverPlayer);
|
|
// Leaves start - skip
|
|
- if (!(serverPlayer instanceof org.leavesmc.leaves.bot.ServerBot)) {
|
|
+ if (!(serverPlayer instanceof org.leavesmc.leaves.bot.ServerBot) && !(serverPlayer instanceof org.leavesmc.leaves.replay.ServerPhotographer)) { // and photographer
|
|
ServerLevel.this.realPlayers.remove(serverPlayer);
|
|
}
|
|
// Leaves end - skip
|
|
diff --git a/net/minecraft/server/players/PlayerList.java b/net/minecraft/server/players/PlayerList.java
|
|
index 7a0d36b9c6a633919a116fc4d47e500b2c6cbc28..95cb24f44412e2aad5f15f8256e3506366c1412d 100644
|
|
--- a/net/minecraft/server/players/PlayerList.java
|
|
+++ b/net/minecraft/server/players/PlayerList.java
|
|
@@ -130,6 +130,7 @@ public abstract class PlayerList {
|
|
private boolean allowCommandsForAllPlayers;
|
|
private static final boolean ALLOW_LOGOUTIVATOR = false;
|
|
private int sendAllPlayerInfoIn;
|
|
+ public final List<ServerPlayer> realPlayers = new java.util.concurrent.CopyOnWriteArrayList(); // Leaves - replay api
|
|
|
|
// CraftBukkit start
|
|
private org.bukkit.craftbukkit.CraftServer cserver;
|
|
@@ -148,6 +149,125 @@ public abstract class PlayerList {
|
|
|
|
abstract public void loadAndSaveFiles(); // Paper - fix converting txt to json file; moved from DedicatedPlayerList constructor
|
|
|
|
+ // Leaves start - replay mod api
|
|
+ public void placeNewPhotographer(Connection connection, org.leavesmc.leaves.replay.ServerPhotographer player, ServerLevel worldserver) {
|
|
+ player.isRealPlayer = true; // Paper
|
|
+ player.loginTime = System.currentTimeMillis(); // Paper
|
|
+
|
|
+ ServerLevel worldserver1 = worldserver;
|
|
+
|
|
+ player.setServerLevel(worldserver1);
|
|
+ player.spawnIn(worldserver1);
|
|
+ player.gameMode.setLevel((ServerLevel) player.level());
|
|
+
|
|
+ LevelData worlddata = worldserver1.getLevelData();
|
|
+
|
|
+ player.loadGameTypes(null);
|
|
+ ServerGamePacketListenerImpl playerconnection = new ServerGamePacketListenerImpl(this.server, connection, player, CommonListenerCookie.createInitial(player.gameProfile, false));
|
|
+ GameRules gamerules = worldserver1.getGameRules();
|
|
+ boolean flag = gamerules.getBoolean(GameRules.RULE_DO_IMMEDIATE_RESPAWN);
|
|
+ boolean flag1 = gamerules.getBoolean(GameRules.RULE_REDUCEDDEBUGINFO);
|
|
+ boolean flag2 = gamerules.getBoolean(GameRules.RULE_LIMITED_CRAFTING);
|
|
+
|
|
+ playerconnection.send(new ClientboundLoginPacket(player.getId(), worlddata.isHardcore(), this.server.levelKeys(), this.getMaxPlayers(), worldserver1.getWorld().getSendViewDistance(), worldserver1.getWorld().getSimulationDistance(), flag1, !flag, flag2, player.createCommonSpawnInfo(worldserver1), this.server.enforceSecureProfile())); // Paper - replace old player chunk management
|
|
+ player.getBukkitEntity().sendSupportedChannels(); // CraftBukkit
|
|
+ playerconnection.send(new ClientboundChangeDifficultyPacket(worlddata.getDifficulty(), worlddata.isDifficultyLocked()));
|
|
+ playerconnection.send(new ClientboundPlayerAbilitiesPacket(player.getAbilities()));
|
|
+ playerconnection.send(new ClientboundSetHeldSlotPacket(player.getInventory().getSelectedSlot()));
|
|
+ RecipeManager craftingmanager = this.server.getRecipeManager();
|
|
+ playerconnection.send(new ClientboundUpdateRecipesPacket(craftingmanager.getSynchronizedItemProperties(), craftingmanager.getSynchronizedStonecutterRecipes()));
|
|
+
|
|
+ this.sendPlayerPermissionLevel(player);
|
|
+ player.getStats().markAllDirty();
|
|
+ player.getRecipeBook().sendInitialRecipeBook(player);
|
|
+ this.updateEntireScoreboard(worldserver1.getScoreboard(), player);
|
|
+ this.server.invalidateStatus();
|
|
+
|
|
+ playerconnection.teleport(player.getX(), player.getY(), player.getZ(), player.getYRot(), player.getXRot());
|
|
+ ServerStatus serverping = this.server.getStatus();
|
|
+
|
|
+ if (serverping != null) {
|
|
+ player.sendServerStatus(serverping);
|
|
+ }
|
|
+
|
|
+ this.players.add(player);
|
|
+ this.playersByName.put(player.getScoreboardName().toLowerCase(java.util.Locale.ROOT), player); // Spigot
|
|
+ this.playersByUUID.put(player.getUUID(), player);
|
|
+
|
|
+ player.supressTrackerForLogin = true;
|
|
+ worldserver1.addNewPlayer(player);
|
|
+ this.server.getCustomBossEvents().onPlayerConnect(player);
|
|
+ org.bukkit.craftbukkit.entity.CraftPlayer bukkitPlayer = player.getBukkitEntity();
|
|
+
|
|
+ player.containerMenu.transferTo(player.containerMenu, bukkitPlayer);
|
|
+ if (!player.connection.isAcceptingMessages()) {
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ // org.leavesmc.leaves.protocol.core.LeavesProtocolManager.handlePlayerJoin(player); // Leaves - protocol
|
|
+
|
|
+ // Leaves start - bot support
|
|
+ if (org.leavesmc.leaves.LeavesConfig.modify.fakeplayer.enable) {
|
|
+ org.leavesmc.leaves.bot.ServerBot bot = this.server.getBotList().getBotByName(player.getScoreboardName());
|
|
+ if (bot != null) {
|
|
+ this.server.getBotList().removeBot(bot, org.leavesmc.leaves.event.bot.BotRemoveEvent.RemoveReason.INTERNAL, player.getBukkitEntity(), false);
|
|
+ }
|
|
+ this.server.getBotList().bots.forEach(bot1 -> {
|
|
+ bot1.sendPlayerInfo(player);
|
|
+ bot1.sendFakeDataIfNeed(player, true);
|
|
+ }); // Leaves - render bot
|
|
+ }
|
|
+ // Leaves end - bot support
|
|
+
|
|
+ final List<ServerPlayer> onlinePlayers = Lists.newArrayListWithExpectedSize(this.players.size() - 1);
|
|
+ for (int i = 0; i < this.players.size(); ++i) {
|
|
+ ServerPlayer entityplayer1 = this.players.get(i);
|
|
+
|
|
+ if (entityplayer1 == player || !bukkitPlayer.canSee(entityplayer1.getBukkitEntity())) {
|
|
+ continue;
|
|
+ }
|
|
+
|
|
+ // Leaves start - skip photographer
|
|
+ if (entityplayer1 instanceof org.leavesmc.leaves.replay.ServerPhotographer) {
|
|
+ continue;
|
|
+ }
|
|
+ // Leaves end - skip photographer
|
|
+
|
|
+ onlinePlayers.add(entityplayer1);
|
|
+ }
|
|
+ if (!onlinePlayers.isEmpty()) {
|
|
+ player.connection.send(ClientboundPlayerInfoUpdatePacket.createPlayerInitializing(onlinePlayers, player));
|
|
+ }
|
|
+
|
|
+ player.sentListPacket = true;
|
|
+ player.supressTrackerForLogin = false;
|
|
+ ((ServerLevel)player.level()).getChunkSource().chunkMap.addEntity(player);
|
|
+
|
|
+ this.sendLevelInfo(player, worldserver1);
|
|
+
|
|
+ if (player.level() == worldserver1 && !worldserver1.players().contains(player)) {
|
|
+ worldserver1.addNewPlayer(player);
|
|
+ this.server.getCustomBossEvents().onPlayerConnect(player);
|
|
+ }
|
|
+
|
|
+ worldserver1 = player.level();
|
|
+ java.util.Iterator<net.minecraft.world.effect.MobEffectInstance> iterator = player.getActiveEffects().iterator();
|
|
+ while (iterator.hasNext()) {
|
|
+ MobEffectInstance mobeffect = iterator.next();
|
|
+ playerconnection.send(new ClientboundUpdateMobEffectPacket(player.getId(), mobeffect, false));
|
|
+ }
|
|
+
|
|
+ if (player.isDeadOrDying()) {
|
|
+ net.minecraft.core.Holder<net.minecraft.world.level.biome.Biome> plains = worldserver1.registryAccess().lookupOrThrow(net.minecraft.core.registries.Registries.BIOME)
|
|
+ .getOrThrow(net.minecraft.world.level.biome.Biomes.PLAINS);
|
|
+ player.connection.send(new net.minecraft.network.protocol.game.ClientboundLevelChunkWithLightPacket(
|
|
+ new net.minecraft.world.level.chunk.EmptyLevelChunk(worldserver1, player.chunkPosition(), plains),
|
|
+ worldserver1.getLightEngine(), null, null, false)
|
|
+ );
|
|
+ }
|
|
+ }
|
|
+ // Leaves end - replay mod api
|
|
+
|
|
public void placeNewPlayer(Connection connection, ServerPlayer player, CommonListenerCookie cookie) {
|
|
player.isRealPlayer = true; // Paper
|
|
player.loginTime = System.currentTimeMillis(); // Paper - Replace OfflinePlayer#getLastPlayed
|
|
@@ -310,6 +430,7 @@ public abstract class PlayerList {
|
|
|
|
// player.connection.send(ClientboundPlayerInfoUpdatePacket.createPlayerInitializing(this.players)); // CraftBukkit - replaced with loop below
|
|
this.players.add(player);
|
|
+ this.realPlayers.add(player); // Leaves - replay api
|
|
this.playersByName.put(player.getScoreboardName().toLowerCase(java.util.Locale.ROOT), player); // Spigot
|
|
this.playersByUUID.put(player.getUUID(), player);
|
|
// this.broadcastAll(ClientboundPlayerInfoUpdatePacket.createPlayerInitializing(List.of(player))); // CraftBukkit - replaced with loop below
|
|
@@ -503,6 +624,7 @@ public abstract class PlayerList {
|
|
}
|
|
|
|
protected void save(ServerPlayer player) {
|
|
+ if (player instanceof org.leavesmc.leaves.replay.ServerPhotographer) return; // Leaves - skip photographer
|
|
if (!player.getBukkitEntity().isPersistent()) return; // CraftBukkit
|
|
player.lastSave = MinecraftServer.currentTick; // Paper - Incremental chunk and player saving
|
|
this.playerIo.save(player);
|
|
@@ -517,6 +639,43 @@ public abstract class PlayerList {
|
|
}
|
|
}
|
|
|
|
+ // Leaves start - replay mod api
|
|
+ public void removePhotographer(org.leavesmc.leaves.replay.ServerPhotographer entityplayer) {
|
|
+ ServerLevel worldserver = entityplayer.level();
|
|
+
|
|
+ entityplayer.awardStat(Stats.LEAVE_GAME);
|
|
+
|
|
+ if (entityplayer.containerMenu != entityplayer.inventoryMenu) {
|
|
+ entityplayer.closeContainer(org.bukkit.event.inventory.InventoryCloseEvent.Reason.DISCONNECT);
|
|
+ }
|
|
+
|
|
+ if (server.isSameThread()) entityplayer.doTick();
|
|
+
|
|
+ if (this.collideRuleTeamName != null) {
|
|
+ final net.minecraft.world.scores.Scoreboard scoreBoard = this.server.getLevel(Level.OVERWORLD).getScoreboard();
|
|
+ final PlayerTeam team = scoreBoard.getPlayersTeam(this.collideRuleTeamName);
|
|
+ if (entityplayer.getTeam() == team && team != null) {
|
|
+ scoreBoard.removePlayerFromTeam(entityplayer.getScoreboardName(), team);
|
|
+ }
|
|
+ }
|
|
+
|
|
+ worldserver.removePlayerImmediately(entityplayer, Entity.RemovalReason.UNLOADED_WITH_PLAYER);
|
|
+ entityplayer.retireScheduler();
|
|
+ entityplayer.getAdvancements().stopListening();
|
|
+ this.players.remove(entityplayer);
|
|
+ this.playersByName.remove(entityplayer.getScoreboardName().toLowerCase(java.util.Locale.ROOT));
|
|
+ this.server.getCustomBossEvents().onPlayerDisconnect(entityplayer);
|
|
+ UUID uuid = entityplayer.getUUID();
|
|
+ ServerPlayer entityplayer1 = this.playersByUUID.get(uuid);
|
|
+
|
|
+ if (entityplayer1 == entityplayer) {
|
|
+ this.playersByUUID.remove(uuid);
|
|
+ }
|
|
+
|
|
+ this.cserver.getScoreboardManager().removePlayer(entityplayer.getBukkitEntity());
|
|
+ }
|
|
+ // Leaves stop - replay mod api
|
|
+
|
|
public @Nullable net.kyori.adventure.text.Component remove(ServerPlayer player) { // CraftBukkit - return string // Paper - return Component
|
|
// Paper start - Fix kick event leave message not being sent
|
|
return this.remove(player, net.kyori.adventure.text.Component.translatable("multiplayer.player.left", net.kyori.adventure.text.format.NamedTextColor.YELLOW, io.papermc.paper.configuration.GlobalConfiguration.get().messages.useDisplayNameInQuitMessage ? player.getBukkitEntity().displayName() : io.papermc.paper.adventure.PaperAdventure.asAdventure(player.getDisplayName())));
|
|
@@ -590,6 +749,7 @@ public abstract class PlayerList {
|
|
player.retireScheduler(); // Paper - Folia schedulers
|
|
player.getAdvancements().stopListening();
|
|
this.players.remove(player);
|
|
+ this.realPlayers.remove(player); // Leaves - replay api
|
|
this.playersByName.remove(player.getScoreboardName().toLowerCase(java.util.Locale.ROOT)); // Spigot
|
|
this.server.getCustomBossEvents().onPlayerDisconnect(player);
|
|
UUID uuid = player.getUUID();
|
|
@@ -964,14 +1124,14 @@ public abstract class PlayerList {
|
|
}
|
|
|
|
public String[] getPlayerNamesArray() {
|
|
- String[] strings = new String[this.players.size() + this.server.getBotList().bots.size()]; // Leaves - fakeplayer support
|
|
+ String[] strings = new String[this.realPlayers.size() + this.server.getBotList().bots.size()]; // Leaves - fakeplayer support, and skip photographer
|
|
|
|
- for (int i = 0; i < this.players.size(); i++) {
|
|
- strings[i] = this.players.get(i).getGameProfile().getName();
|
|
+ for (int i = 0; i < this.realPlayers.size(); i++) { // Leaves - only real players
|
|
+ strings[i] = this.realPlayers.get(i).getGameProfile().getName(); // Leaves - only real players
|
|
}
|
|
// Leaves start - fakeplayer support
|
|
- for (int i = this.players.size(); i < strings.length; ++i) {
|
|
- strings[i] = this.server.getBotList().bots.get(i - this.players.size()).getGameProfile().getName();
|
|
+ for (int i = this.realPlayers.size(); i < strings.length; ++i) { // Leaves - only real players
|
|
+ strings[i] = this.server.getBotList().bots.get(i - this.realPlayers.size()).getGameProfile().getName(); // Leaves - only real players
|
|
}
|
|
// Leaves end - fakeplayer support
|
|
|
|
@@ -1030,7 +1190,7 @@ public abstract class PlayerList {
|
|
|
|
// Paper start - whitelist verify event / login event
|
|
public LoginResult canBypassFullServerLogin(final GameProfile profile, final LoginResult currentResult) {
|
|
- final boolean shouldKick = this.players.size() >= this.maxPlayers && !this.canBypassPlayerLimit(profile);
|
|
+ final boolean shouldKick = this.realPlayers.size() >= this.maxPlayers && !this.canBypassPlayerLimit(profile); // Leaves - only real player
|
|
final io.papermc.paper.event.player.PlayerServerFullCheckEvent fullCheckEvent = new io.papermc.paper.event.player.PlayerServerFullCheckEvent(
|
|
com.destroystokyo.paper.profile.CraftPlayerProfile.asBukkitMirror(profile),
|
|
io.papermc.paper.adventure.PaperAdventure.asAdventure(currentResult.message),
|