Initial port of version 2.1.0 from the 1.16 branch.
This commit is contained in:
@ -11,7 +11,7 @@ public class ProtectedPlayer {
|
|||||||
|
|
||||||
public ProtectedPlayer(UUID player_uuid) {
|
public ProtectedPlayer(UUID player_uuid) {
|
||||||
this.player_uuid = player_uuid;
|
this.player_uuid = player_uuid;
|
||||||
this.grace_period = (Config.POST_GRACE_DURATION.get() * 40);
|
this.grace_period = (Config.Server.POST_GRACE_DURATION.get() * 40);
|
||||||
this.is_loading = true;
|
this.is_loading = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -39,8 +39,10 @@ public class ProtectedPlayers {
|
|||||||
protected_players.remove(protected_player);
|
protected_players.remove(protected_player);
|
||||||
|
|
||||||
ServerPlayer player = ServerLifecycleHooks.getCurrentServer().getPlayerList().getPlayer(player_uuid);
|
ServerPlayer player = ServerLifecycleHooks.getCurrentServer().getPlayerList().getPlayer(player_uuid);
|
||||||
if (player == null || !Config.POST_GRACE_ENABLED.get()) { return; }
|
if (player == null) { return; }
|
||||||
player.sendMessage(new TextComponent("Grace Period over!"), player_uuid);
|
|
||||||
|
if (!Config.Server.POST_GRACE_ENABLED.get()) { return; }
|
||||||
|
player.sendMessage(new TextComponent("[LoginProtection] Grace period ended!"), player_uuid);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void updateGracePeriod(UUID player_uuid) {
|
public void updateGracePeriod(UUID player_uuid) {
|
||||||
@ -51,7 +53,6 @@ public class ProtectedPlayers {
|
|||||||
protected_player.setGracePeriod(grace_period);
|
protected_player.setGracePeriod(grace_period);
|
||||||
if (grace_period <= 0) {
|
if (grace_period <= 0) {
|
||||||
removePlayer(player_uuid);
|
removePlayer(player_uuid);
|
||||||
System.out.println(protected_player.getPlayerUUID() + " is no longer being protected!");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -2,19 +2,25 @@ package com.micle.loginprotection.events;
|
|||||||
|
|
||||||
import com.micle.loginprotection.LoginProtection;
|
import com.micle.loginprotection.LoginProtection;
|
||||||
import com.micle.loginprotection.network.C2SKeyPress;
|
import com.micle.loginprotection.network.C2SKeyPress;
|
||||||
|
import com.micle.loginprotection.setup.Config;
|
||||||
import net.minecraft.client.Minecraft;
|
import net.minecraft.client.Minecraft;
|
||||||
import net.minecraft.client.multiplayer.ServerData;
|
import net.minecraft.client.multiplayer.ServerData;
|
||||||
import net.minecraftforge.api.distmarker.Dist;
|
import net.minecraftforge.api.distmarker.Dist;
|
||||||
import net.minecraftforge.api.distmarker.OnlyIn;
|
import net.minecraftforge.api.distmarker.OnlyIn;
|
||||||
import net.minecraftforge.client.event.InputEvent;
|
import net.minecraftforge.client.event.InputEvent;
|
||||||
import net.minecraftforge.eventbus.api.SubscribeEvent;
|
import net.minecraftforge.eventbus.api.SubscribeEvent;
|
||||||
|
import org.lwjgl.glfw.GLFW;
|
||||||
|
|
||||||
public class OnKeyPressEventHandler {
|
public class OnKeyPressEventHandler {
|
||||||
@SubscribeEvent
|
@SubscribeEvent
|
||||||
@OnlyIn(Dist.CLIENT)
|
@OnlyIn(Dist.CLIENT)
|
||||||
public void KeyPressEvent(InputEvent.KeyInputEvent event) {
|
public void KeyPressEvent(InputEvent.KeyInputEvent event) {
|
||||||
ServerData server = Minecraft.getInstance().getCurrentServer();
|
Minecraft instance = Minecraft.getInstance();
|
||||||
|
ServerData server = instance.getCurrentServer();
|
||||||
if (server == null) { return; }
|
if (server == null) { return; }
|
||||||
|
if (Minecraft.getInstance().screen != null) { return; }
|
||||||
|
if (checkKeyAllowed(instance, event.getKey())) { return; }
|
||||||
|
|
||||||
try {
|
try {
|
||||||
LoginProtection.INSTANCE.sendToServer(new C2SKeyPress());
|
LoginProtection.INSTANCE.sendToServer(new C2SKeyPress());
|
||||||
} catch (NullPointerException ignored) { }
|
} catch (NullPointerException ignored) { }
|
||||||
@ -23,10 +29,58 @@ public class OnKeyPressEventHandler {
|
|||||||
@SubscribeEvent
|
@SubscribeEvent
|
||||||
@OnlyIn(Dist.CLIENT)
|
@OnlyIn(Dist.CLIENT)
|
||||||
public void MouseClickEvent(InputEvent.ClickInputEvent event) {
|
public void MouseClickEvent(InputEvent.ClickInputEvent event) {
|
||||||
ServerData server = Minecraft.getInstance().getCurrentServer();
|
Minecraft instance = Minecraft.getInstance();
|
||||||
|
ServerData server = instance.getCurrentServer();
|
||||||
if (server == null) { return; }
|
if (server == null) { return; }
|
||||||
|
if (Minecraft.getInstance().screen != null) { return; }
|
||||||
|
if (checkKeyAllowed(instance, event.getKeyBinding().getKey().getValue())) { return; }
|
||||||
|
|
||||||
try {
|
try {
|
||||||
LoginProtection.INSTANCE.sendToServer(new C2SKeyPress());
|
LoginProtection.INSTANCE.sendToServer(new C2SKeyPress());
|
||||||
} catch (NullPointerException ignored) { }
|
} catch (NullPointerException ignored) { }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@OnlyIn(Dist.CLIENT)
|
||||||
|
private static boolean checkKeyAllowed(Minecraft instance, int key) {
|
||||||
|
boolean isAllowed = false;
|
||||||
|
|
||||||
|
if (key == GLFW.GLFW_KEY_ESCAPE) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.PAUSE.toString()); }
|
||||||
|
if (key == GLFW.GLFW_KEY_F3) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.DEBUG.toString()); }
|
||||||
|
if (key == instance.options.keyFullscreen.getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.FULLSCREEN.toString()); }
|
||||||
|
if (key == instance.options.keyTogglePerspective.getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.PERSPECTIVE.toString()); }
|
||||||
|
if (key == instance.options.keySmoothCamera.getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.SMOOTH_CAMERA.toString()); }
|
||||||
|
if (key == instance.options.keyScreenshot.getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.SCREENSHOT.toString()); }
|
||||||
|
if (key == instance.options.keySpectatorOutlines.getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.SPECTATOR_OUTLINES.toString()); }
|
||||||
|
if (key == instance.options.keyAdvancements.getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.ADVANCEMENTS.toString()); }
|
||||||
|
if (key == instance.options.keyPlayerList.getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.PLAYER_LIST.toString()); }
|
||||||
|
if (key == instance.options.keyChat.getKey().getValue() ||
|
||||||
|
key == instance.options.keyCommand.getKey().getValue() ||
|
||||||
|
key == GLFW.GLFW_KEY_ENTER) {
|
||||||
|
isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.CHAT.toString());
|
||||||
|
}
|
||||||
|
if (key == instance.options.keySocialInteractions.getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.SOCIAL_INTERACTIONS.toString()); }
|
||||||
|
if (key == instance.options.keyLoadHotbarActivator.getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.LOAD_HOTBAR_ACTIVATOR.toString()); }
|
||||||
|
if (key == instance.options.keySaveHotbarActivator.getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.SAVE_HOTBAR_ACTIVATOR.toString()); }
|
||||||
|
if (key == instance.options.keySwapOffhand.getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.SWAP_ITEM.toString()); }
|
||||||
|
if (key == instance.options.keyInventory.getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.INVENTORY.toString()); }
|
||||||
|
for (int i = 0; i < instance.options.keyHotbarSlots.length; i++) {
|
||||||
|
if (key == instance.options.keyHotbarSlots[i].getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.HOTBAR.toString()); }
|
||||||
|
}
|
||||||
|
if (key == instance.options.keyDrop.getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.DROP_ITEM.toString()); }
|
||||||
|
if (key == instance.options.keyUse.getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.USE_ITEM.toString()); }
|
||||||
|
if (key == instance.options.keyPickItem.getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.PICK_BLOCK.toString()); }
|
||||||
|
if (key == instance.options.keyAttack.getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.ATTACK.toString()); }
|
||||||
|
if (key == instance.options.keyUp.getKey().getValue() ||
|
||||||
|
key == instance.options.keyRight.getKey().getValue() ||
|
||||||
|
key == instance.options.keyDown.getKey().getValue() ||
|
||||||
|
key == instance.options.keyLeft.getKey().getValue() ||
|
||||||
|
key == instance.options.keySprint.getKey().getValue()) {
|
||||||
|
isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.MOVE.toString());
|
||||||
|
}
|
||||||
|
if (key == instance.options.keyShift.getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.SNEAK.toString()); }
|
||||||
|
if (key == instance.options.keyJump.getKey().getValue()) { isAllowed = Config.Server.MAIN_KEY_ALLOW_LIST.get().contains(Config.Server.KEYS.JUMP.toString()); }
|
||||||
|
|
||||||
|
return isAllowed;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -15,8 +15,8 @@ public class OnLivingSetAttackTargetEventHandler {
|
|||||||
|
|
||||||
ProtectedPlayer player = LoginProtection.protected_players.getPlayer(target.getUUID());
|
ProtectedPlayer player = LoginProtection.protected_players.getPlayer(target.getUUID());
|
||||||
if (player == null) { return; }
|
if (player == null) { return; }
|
||||||
if (player.isLoading() && !Config.MAIN_IGNORE_PLAYER_ENABLED.get()) { return; }
|
if (player.isLoading() && !Config.Server.MAIN_IGNORE_PLAYER_ENABLED.get()) { return; }
|
||||||
if (!player.isLoading() && !Config.POST_GRACE_IGNORE_PLAYER_ENABLED.get()) { return; }
|
if (!player.isLoading() && !Config.Server.POST_GRACE_IGNORE_PLAYER_ENABLED.get()) { return; }
|
||||||
((Mob) event.getEntityLiving()).setTarget(null);
|
((Mob) event.getEntityLiving()).setTarget(null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -29,25 +29,26 @@ public class C2SKeyPress {
|
|||||||
if (LoginProtection.protected_players.getPlayer(sender.getUUID()) == null) { return; }
|
if (LoginProtection.protected_players.getPlayer(sender.getUUID()) == null) { return; }
|
||||||
if (!LoginProtection.protected_players.getPlayer(sender.getUUID()).isLoading()) { return; }
|
if (!LoginProtection.protected_players.getPlayer(sender.getUUID()).isLoading()) { return; }
|
||||||
|
|
||||||
if (!Config.POST_GRACE_ENABLED.get()) {
|
if (!Config.Server.POST_GRACE_ENABLED.get()) {
|
||||||
LoginProtection.protected_players.removePlayer(sender.getUUID());
|
LoginProtection.protected_players.removePlayer(sender.getUUID());
|
||||||
|
sender.sendMessage(new TextComponent("[LoginProtection] You are now seen as active."), sender.getUUID());
|
||||||
} else {
|
} else {
|
||||||
LoginProtection.protected_players.getPlayer(sender.getUUID()).setLoading(false);
|
LoginProtection.protected_players.getPlayer(sender.getUUID()).setLoading(false);
|
||||||
sender.sendMessage(new TextComponent("Grace Period started!"), sender.getUUID());
|
sender.sendMessage(new TextComponent("[LoginProtection] Grace period started!"), sender.getUUID());
|
||||||
}
|
}
|
||||||
if (sender.isInWater()) {
|
if (sender.isInWater()) {
|
||||||
if (Config.POST_DROWN_ENABLED.get()) {
|
if (Config.Server.POST_DROWN_ENABLED.get()) {
|
||||||
sender.setAirSupply(sender.getMaxAirSupply());
|
sender.setAirSupply(sender.getMaxAirSupply());
|
||||||
}
|
}
|
||||||
if (Config.POST_WATER_ENABLED.get()) {
|
if (Config.Server.POST_WATER_ENABLED.get()) {
|
||||||
sender.addEffect(new MobEffectInstance(MobEffects.WATER_BREATHING, Config.POST_WATER_DURATION.get()*20, 0));
|
sender.addEffect(new MobEffectInstance(MobEffects.WATER_BREATHING, Config.Server.POST_WATER_DURATION.get()*20, 0));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (sender.isInLava() && Config.POST_LAVA_ENABLED.get()) {
|
if (sender.isInLava() && Config.Server.POST_LAVA_ENABLED.get()) {
|
||||||
sender.addEffect(new MobEffectInstance(MobEffects.FIRE_RESISTANCE, Config.POST_LAVA_DURATION.get()*20, 0));
|
sender.addEffect(new MobEffectInstance(MobEffects.FIRE_RESISTANCE, Config.Server.POST_LAVA_DURATION.get()*20, 0));
|
||||||
}
|
}
|
||||||
if (sender.isOnFire() && Config.POST_FIRE_ENABLED.get()) {
|
if (sender.isOnFire() && Config.Server.POST_FIRE_ENABLED.get()) {
|
||||||
sender.addEffect(new MobEffectInstance(MobEffects.FIRE_RESISTANCE, Config.POST_FIRE_DURATION.get()*20, 0));
|
sender.addEffect(new MobEffectInstance(MobEffects.FIRE_RESISTANCE, Config.Server.POST_FIRE_DURATION.get()*20, 0));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
context.setPacketHandled(true);
|
context.setPacketHandled(true);
|
||||||
|
|||||||
@ -1,11 +1,55 @@
|
|||||||
package com.micle.loginprotection.setup;
|
package com.micle.loginprotection.setup;
|
||||||
|
|
||||||
|
import com.google.common.collect.Lists;
|
||||||
|
import com.micle.loginprotection.LoginProtection;
|
||||||
import net.minecraftforge.common.ForgeConfigSpec;
|
import net.minecraftforge.common.ForgeConfigSpec;
|
||||||
import net.minecraftforge.fml.ModLoadingContext;
|
import net.minecraftforge.fml.ModLoadingContext;
|
||||||
|
import net.minecraftforge.fml.common.Mod;
|
||||||
import net.minecraftforge.fml.config.ModConfig;
|
import net.minecraftforge.fml.config.ModConfig;
|
||||||
|
import org.apache.commons.lang3.tuple.Pair;
|
||||||
|
|
||||||
public class Config {
|
import java.util.List;
|
||||||
public static ForgeConfigSpec SERVER_CONFIG;
|
|
||||||
|
@Mod.EventBusSubscriber(modid = LoginProtection.MOD_ID, bus = Mod.EventBusSubscriber.Bus.MOD)
|
||||||
|
public final class Config {
|
||||||
|
public static final Server SERVER;
|
||||||
|
public static final ForgeConfigSpec SERVER_SPEC;
|
||||||
|
static {
|
||||||
|
Pair<Server, ForgeConfigSpec> spec_pair = new ForgeConfigSpec.Builder().configure(Server::new);
|
||||||
|
SERVER = spec_pair.getLeft();
|
||||||
|
SERVER_SPEC = spec_pair.getRight();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void init() {
|
||||||
|
ModLoadingContext.get().registerConfig(ModConfig.Type.SERVER, SERVER_SPEC);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class Server {
|
||||||
|
public enum KEYS {
|
||||||
|
PAUSE,
|
||||||
|
DEBUG,
|
||||||
|
FULLSCREEN,
|
||||||
|
PERSPECTIVE,
|
||||||
|
SMOOTH_CAMERA,
|
||||||
|
SCREENSHOT,
|
||||||
|
SPECTATOR_OUTLINES,
|
||||||
|
ADVANCEMENTS,
|
||||||
|
PLAYER_LIST,
|
||||||
|
CHAT,
|
||||||
|
SOCIAL_INTERACTIONS,
|
||||||
|
LOAD_HOTBAR_ACTIVATOR,
|
||||||
|
SAVE_HOTBAR_ACTIVATOR,
|
||||||
|
SWAP_ITEM,
|
||||||
|
INVENTORY,
|
||||||
|
HOTBAR,
|
||||||
|
DROP_ITEM,
|
||||||
|
USE_ITEM,
|
||||||
|
PICK_BLOCK,
|
||||||
|
ATTACK,
|
||||||
|
MOVE,
|
||||||
|
SNEAK,
|
||||||
|
JUMP
|
||||||
|
}
|
||||||
|
|
||||||
public static ForgeConfigSpec.BooleanValue POST_GRACE_ENABLED;
|
public static ForgeConfigSpec.BooleanValue POST_GRACE_ENABLED;
|
||||||
public static ForgeConfigSpec.BooleanValue POST_GRACE_IGNORE_PLAYER_ENABLED;
|
public static ForgeConfigSpec.BooleanValue POST_GRACE_IGNORE_PLAYER_ENABLED;
|
||||||
@ -19,20 +63,23 @@ public class Config {
|
|||||||
public static ForgeConfigSpec.IntValue POST_FIRE_DURATION;
|
public static ForgeConfigSpec.IntValue POST_FIRE_DURATION;
|
||||||
|
|
||||||
public static ForgeConfigSpec.BooleanValue MAIN_IGNORE_PLAYER_ENABLED;
|
public static ForgeConfigSpec.BooleanValue MAIN_IGNORE_PLAYER_ENABLED;
|
||||||
|
public static ForgeConfigSpec.ConfigValue<List<? extends String>> MAIN_KEY_ALLOW_LIST;
|
||||||
|
|
||||||
public static void init() {
|
Server(ForgeConfigSpec.Builder builder) {
|
||||||
initServer();
|
|
||||||
|
|
||||||
ModLoadingContext.get().registerConfig(ModConfig.Type.SERVER, SERVER_CONFIG);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void initServer() {
|
|
||||||
ForgeConfigSpec.Builder builder = new ForgeConfigSpec.Builder();
|
|
||||||
|
|
||||||
builder.comment("Main protection settings for protecting inactive (loading) players.").push("main");
|
builder.comment("Main protection settings for protecting inactive (loading) players.").push("main");
|
||||||
MAIN_IGNORE_PLAYER_ENABLED = builder
|
MAIN_IGNORE_PLAYER_ENABLED = builder
|
||||||
.comment("Whether mobs will ignore a protected player. (They will not attack/aggro)")
|
.comment("Whether mobs will ignore a protected player. (They will not attack/aggro)")
|
||||||
.define("ignorePlayerEnabled", true);
|
.define("ignorePlayerEnabled", true);
|
||||||
|
builder.push("allow_keys");
|
||||||
|
MAIN_KEY_ALLOW_LIST = builder
|
||||||
|
.comment("Allowed keys players can press without becoming active.\n" +
|
||||||
|
"Available values: PAUSE, DEBUG, FULLSCREEN, PERSPECTIVE, SMOOTH_CAMERA, SCREENSHOT, SPECTATOR_OUTLINES,\n" +
|
||||||
|
"ADVANCEMENTS, PLAYER_LIST, CHAT, SOCIAL_INTERACTIONS, LOAD_HOTBAR_ACTIVATOR, SAVE_HOTBAR_ACTIVATOR,\n" +
|
||||||
|
"SWAP_ITEM, INVENTORY, HOTBAR, DROP_ITEM, USE_ITEM, PICK_BLOCK, ATTACK, MOVE, SNEAK, JUMP")
|
||||||
|
.defineList("allowedKeys", Lists.newArrayList(KEYS.PAUSE.toString(), KEYS.DEBUG.toString(), KEYS.FULLSCREEN.toString(), KEYS.PERSPECTIVE.toString(), KEYS.SMOOTH_CAMERA.toString(),
|
||||||
|
KEYS.SCREENSHOT.toString(), KEYS.SPECTATOR_OUTLINES.toString(), KEYS.ADVANCEMENTS.toString(), KEYS.PLAYER_LIST.toString(), KEYS.CHAT.toString(), KEYS.SOCIAL_INTERACTIONS.toString(),
|
||||||
|
KEYS.LOAD_HOTBAR_ACTIVATOR.toString(), KEYS.SAVE_HOTBAR_ACTIVATOR.toString(), KEYS.SWAP_ITEM.toString(), KEYS.HOTBAR.toString(), KEYS.PICK_BLOCK.toString()), o -> o instanceof String);
|
||||||
|
builder.pop();
|
||||||
builder.pop();
|
builder.pop();
|
||||||
|
|
||||||
builder.comment("Additional protection settings that apply as soon as a player becomes active.").push("post");
|
builder.comment("Additional protection settings that apply as soon as a player becomes active.").push("post");
|
||||||
@ -75,7 +122,6 @@ public class Config {
|
|||||||
.defineInRange("fireDuration", 10, 1, Integer.MAX_VALUE/20);
|
.defineInRange("fireDuration", 10, 1, Integer.MAX_VALUE/20);
|
||||||
builder.pop();
|
builder.pop();
|
||||||
builder.pop();
|
builder.pop();
|
||||||
|
}
|
||||||
SERVER_CONFIG = builder.build();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Reference in New Issue
Block a user