2022-02-23 04:09:15 +01:00
|
|
|
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
|
|
From: Jake Potrebic <jake.m.potrebic@gmail.com>
|
|
|
|
Date: Thu, 7 Oct 2021 14:34:55 -0700
|
|
|
|
Subject: [PATCH] Custom Potion Mixes
|
|
|
|
|
|
|
|
|
|
|
|
diff --git a/src/main/java/io/papermc/paper/potion/PaperPotionMix.java b/src/main/java/io/papermc/paper/potion/PaperPotionMix.java
|
|
|
|
new file mode 100644
|
|
|
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000
|
|
|
|
--- /dev/null
|
|
|
|
+++ b/src/main/java/io/papermc/paper/potion/PaperPotionMix.java
|
|
|
|
@@ -0,0 +0,0 @@
|
|
|
|
+package io.papermc.paper.potion;
|
|
|
|
+
|
2023-10-11 01:07:21 +02:00
|
|
|
+import java.util.function.Predicate;
|
2022-02-23 04:09:15 +01:00
|
|
|
+import net.minecraft.world.item.ItemStack;
|
|
|
|
+import org.bukkit.craftbukkit.inventory.CraftItemStack;
|
|
|
|
+import org.bukkit.craftbukkit.inventory.CraftRecipe;
|
2023-10-11 01:07:21 +02:00
|
|
|
+import org.bukkit.inventory.RecipeChoice;
|
2022-02-23 04:09:15 +01:00
|
|
|
+
|
2023-10-11 01:07:21 +02:00
|
|
|
+public record PaperPotionMix(ItemStack result, Predicate<ItemStack> input, Predicate<ItemStack> ingredient) {
|
2022-02-23 04:09:15 +01:00
|
|
|
+
|
|
|
|
+ public PaperPotionMix(PotionMix potionMix) {
|
2023-10-11 01:07:21 +02:00
|
|
|
+ this(CraftItemStack.asNMSCopy(potionMix.getResult()), convert(potionMix.getInput()), convert(potionMix.getIngredient()));
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ static Predicate<ItemStack> convert(final RecipeChoice choice) {
|
|
|
|
+ if (choice instanceof PredicateRecipeChoice predicateRecipeChoice) {
|
|
|
|
+ return stack -> predicateRecipeChoice.test(CraftItemStack.asBukkitCopy(stack));
|
|
|
|
+ }
|
|
|
|
+ return CraftRecipe.toIngredient(choice, true);
|
2022-02-23 04:09:15 +01:00
|
|
|
+ }
|
|
|
|
+}
|
2023-12-06 17:39:37 +01:00
|
|
|
diff --git a/src/main/java/net/minecraft/server/MinecraftServer.java b/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
|
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
|
|
|
--- a/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
|
|
+++ b/src/main/java/net/minecraft/server/MinecraftServer.java
|
2024-04-26 17:33:00 +02:00
|
|
|
@@ -0,0 +0,0 @@ public abstract class MinecraftServer extends ReentrantBlockableEventLoop<TickTa
|
|
|
|
private final StructureTemplateManager structureTemplateManager;
|
|
|
|
private final ServerTickRateManager tickRateManager;
|
|
|
|
protected WorldData worldData;
|
|
|
|
- private final PotionBrewing potionBrewing;
|
|
|
|
+ public PotionBrewing potionBrewing; // Paper - private -> public (remove final)
|
|
|
|
private volatile boolean isSaving;
|
|
|
|
|
|
|
|
// CraftBukkit start
|
2023-12-06 17:39:37 +01:00
|
|
|
@@ -0,0 +0,0 @@ public abstract class MinecraftServer extends ReentrantBlockableEventLoop<TickTa
|
|
|
|
|
|
|
|
this.worldData.setDataConfiguration(worlddataconfiguration);
|
2024-04-24 22:05:42 +02:00
|
|
|
this.resources.managers.updateRegistryTags();
|
2024-04-26 17:33:00 +02:00
|
|
|
+ this.potionBrewing = this.potionBrewing.reload(this.worldData.enabledFeatures()); // Paper - Custom Potion Mixes
|
2024-01-23 12:06:27 +01:00
|
|
|
this.getPlayerList().saveAll();
|
|
|
|
this.getPlayerList().reloadResources();
|
|
|
|
this.functionManager.replaceLibrary(this.resources.managers.getFunctionLibrary());
|
2022-02-23 04:09:15 +01:00
|
|
|
diff --git a/src/main/java/net/minecraft/world/inventory/BrewingStandMenu.java b/src/main/java/net/minecraft/world/inventory/BrewingStandMenu.java
|
|
|
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
|
|
|
--- a/src/main/java/net/minecraft/world/inventory/BrewingStandMenu.java
|
|
|
|
+++ b/src/main/java/net/minecraft/world/inventory/BrewingStandMenu.java
|
|
|
|
@@ -0,0 +0,0 @@ public class BrewingStandMenu extends AbstractContainerMenu {
|
|
|
|
}
|
|
|
|
|
|
|
|
public static boolean mayPlaceItem(ItemStack stack) {
|
|
|
|
- return stack.is(Items.POTION) || stack.is(Items.SPLASH_POTION) || stack.is(Items.LINGERING_POTION) || stack.is(Items.GLASS_BOTTLE);
|
2024-01-18 18:52:00 +01:00
|
|
|
+ return stack.is(Items.POTION) || stack.is(Items.SPLASH_POTION) || stack.is(Items.LINGERING_POTION) || stack.is(Items.GLASS_BOTTLE) || PotionBrewing.isCustomInput(stack); // Paper - Custom Potion Mixes
|
2022-02-23 04:09:15 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
diff --git a/src/main/java/net/minecraft/world/item/alchemy/PotionBrewing.java b/src/main/java/net/minecraft/world/item/alchemy/PotionBrewing.java
|
|
|
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
|
|
|
--- a/src/main/java/net/minecraft/world/item/alchemy/PotionBrewing.java
|
|
|
|
+++ b/src/main/java/net/minecraft/world/item/alchemy/PotionBrewing.java
|
|
|
|
@@ -0,0 +0,0 @@ public class PotionBrewing {
|
2024-04-24 22:05:42 +02:00
|
|
|
private final List<Ingredient> containers;
|
|
|
|
private final List<PotionBrewing.Mix<Potion>> potionMixes;
|
|
|
|
private final List<PotionBrewing.Mix<Item>> containerMixes;
|
2024-01-18 18:52:00 +01:00
|
|
|
+ private static final it.unimi.dsi.fastutil.objects.Object2ObjectLinkedOpenHashMap<org.bukkit.NamespacedKey, io.papermc.paper.potion.PaperPotionMix> CUSTOM_MIXES = new it.unimi.dsi.fastutil.objects.Object2ObjectLinkedOpenHashMap<>(); // Paper - Custom Potion Mixes
|
2024-04-24 22:05:42 +02:00
|
|
|
|
|
|
|
PotionBrewing(List<Ingredient> potionTypes, List<PotionBrewing.Mix<Potion>> potionRecipes, List<PotionBrewing.Mix<Item>> itemRecipes) {
|
|
|
|
this.containers = potionTypes;
|
2022-02-23 04:09:15 +01:00
|
|
|
@@ -0,0 +0,0 @@ public class PotionBrewing {
|
2024-04-24 22:05:42 +02:00
|
|
|
}
|
2022-02-23 04:09:15 +01:00
|
|
|
|
2024-04-24 22:05:42 +02:00
|
|
|
public boolean isIngredient(ItemStack stack) {
|
|
|
|
- return this.isContainerIngredient(stack) || this.isPotionIngredient(stack);
|
|
|
|
+ return this.isContainerIngredient(stack) || this.isPotionIngredient(stack) || isCustomIngredient(stack); // Paper - Custom Potion Mixes
|
2022-02-23 04:09:15 +01:00
|
|
|
}
|
|
|
|
|
2024-04-24 22:05:42 +02:00
|
|
|
private boolean isContainer(ItemStack stack) {
|
2022-02-23 04:09:15 +01:00
|
|
|
@@ -0,0 +0,0 @@ public class PotionBrewing {
|
|
|
|
}
|
|
|
|
|
2024-04-24 22:05:42 +02:00
|
|
|
public boolean hasMix(ItemStack input, ItemStack ingredient) {
|
2024-01-18 18:52:00 +01:00
|
|
|
+ // Paper start - Custom Potion Mixes
|
2022-02-23 04:09:15 +01:00
|
|
|
+ if (hasCustomMix(input, ingredient)) {
|
|
|
|
+ return true;
|
|
|
|
+ }
|
2024-01-18 18:52:00 +01:00
|
|
|
+ // Paper end - Custom Potion Mixes
|
2024-04-24 22:05:42 +02:00
|
|
|
return this.isContainer(input) && (this.hasContainerMix(input, ingredient) || this.hasPotionMix(input, ingredient));
|
2024-04-12 21:14:06 +02:00
|
|
|
}
|
|
|
|
|
2022-02-23 04:09:15 +01:00
|
|
|
@@ -0,0 +0,0 @@ public class PotionBrewing {
|
2024-04-24 22:05:42 +02:00
|
|
|
if (optional.isEmpty()) {
|
|
|
|
return input;
|
|
|
|
} else {
|
|
|
|
+ // Paper start - Custom Potion Mixes
|
|
|
|
+ for (var mix : CUSTOM_MIXES.values()) {
|
|
|
|
+ if (mix.input().test(input) && mix.ingredient().test(ingredient)) {
|
|
|
|
+ return mix.result().copy();
|
|
|
|
+ }
|
2022-02-23 04:09:15 +01:00
|
|
|
+ }
|
2024-04-24 22:05:42 +02:00
|
|
|
+ // Paper end - Custom Potion Mixes
|
|
|
|
for (PotionBrewing.Mix<Item> mix : this.containerMixes) {
|
|
|
|
if (input.is(mix.from) && mix.ingredient.test(ingredient)) {
|
|
|
|
return PotionContents.createItemStack(mix.to.value(), optional.get());
|
2022-02-23 04:09:15 +01:00
|
|
|
@@ -0,0 +0,0 @@ public class PotionBrewing {
|
2024-04-24 22:05:42 +02:00
|
|
|
builder.addMix(Potions.SLOW_FALLING, Items.REDSTONE, Potions.LONG_SLOW_FALLING);
|
2022-02-23 04:09:15 +01:00
|
|
|
}
|
|
|
|
|
2024-01-18 18:52:00 +01:00
|
|
|
+ // Paper start - Custom Potion Mixes
|
2022-02-23 04:09:15 +01:00
|
|
|
+ public static boolean isCustomIngredient(ItemStack stack) {
|
|
|
|
+ for (var mix : CUSTOM_MIXES.values()) {
|
|
|
|
+ if (mix.ingredient().test(stack)) {
|
|
|
|
+ return true;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ public static boolean isCustomInput(ItemStack stack) {
|
|
|
|
+ for (var mix : CUSTOM_MIXES.values()) {
|
|
|
|
+ if (mix.input().test(stack)) {
|
|
|
|
+ return true;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ private static boolean hasCustomMix(ItemStack input, ItemStack ingredient) {
|
|
|
|
+ for (var mix : CUSTOM_MIXES.values()) {
|
|
|
|
+ if (mix.input().test(input) && mix.ingredient().test(ingredient)) {
|
|
|
|
+ return true;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ public static void addPotionMix(io.papermc.paper.potion.PotionMix mix) {
|
|
|
|
+ if (CUSTOM_MIXES.containsKey(mix.getKey())) {
|
|
|
|
+ throw new IllegalArgumentException("Duplicate recipe ignored with ID " + mix.getKey());
|
|
|
|
+ }
|
|
|
|
+ CUSTOM_MIXES.putAndMoveToFirst(mix.getKey(), new io.papermc.paper.potion.PaperPotionMix(mix));
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ public static boolean removePotionMix(org.bukkit.NamespacedKey key) {
|
|
|
|
+ return CUSTOM_MIXES.remove(key) != null;
|
|
|
|
+ }
|
|
|
|
+
|
2024-04-26 17:33:00 +02:00
|
|
|
+ public PotionBrewing reload(FeatureFlagSet flags) {
|
|
|
|
+ return bootstrap(flags);
|
2022-02-23 04:09:15 +01:00
|
|
|
+ }
|
2024-01-18 18:52:00 +01:00
|
|
|
+ // Paper end - Custom Potion Mixes
|
2022-02-23 04:09:15 +01:00
|
|
|
+
|
2024-04-24 22:05:42 +02:00
|
|
|
public static class Builder {
|
|
|
|
private final List<Ingredient> containers = new ArrayList<>();
|
|
|
|
private final List<PotionBrewing.Mix<Potion>> potionMixes = new ArrayList<>();
|
2022-02-23 04:09:15 +01:00
|
|
|
diff --git a/src/main/java/net/minecraft/world/level/block/entity/BrewingStandBlockEntity.java b/src/main/java/net/minecraft/world/level/block/entity/BrewingStandBlockEntity.java
|
|
|
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
|
|
|
--- a/src/main/java/net/minecraft/world/level/block/entity/BrewingStandBlockEntity.java
|
|
|
|
+++ b/src/main/java/net/minecraft/world/level/block/entity/BrewingStandBlockEntity.java
|
|
|
|
@@ -0,0 +0,0 @@ public class BrewingStandBlockEntity extends BaseContainerBlockEntity implements
|
|
|
|
|
2024-04-24 22:05:42 +02:00
|
|
|
return potionbrewer.isIngredient(stack);
|
|
|
|
} else {
|
|
|
|
- return slot == 4 ? stack.is(Items.BLAZE_POWDER) : (stack.is(Items.POTION) || stack.is(Items.SPLASH_POTION) || stack.is(Items.LINGERING_POTION) || stack.is(Items.GLASS_BOTTLE)) && this.getItem(slot).isEmpty();
|
|
|
|
+ return slot == 4 ? stack.is(Items.BLAZE_POWDER) : (stack.is(Items.POTION) || stack.is(Items.SPLASH_POTION) || stack.is(Items.LINGERING_POTION) || stack.is(Items.GLASS_BOTTLE) || PotionBrewing.isCustomInput(stack)) && this.getItem(slot).isEmpty(); // Paper - Custom Potion Mixes
|
|
|
|
}
|
2022-02-23 04:09:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
diff --git a/src/main/java/org/bukkit/craftbukkit/CraftServer.java b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
|
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
|
|
|
--- a/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
|
|
+++ b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
|
|
@@ -0,0 +0,0 @@ public final class CraftServer implements Server {
|
|
|
|
private final io.papermc.paper.datapack.PaperDatapackManager datapackManager; // Paper
|
|
|
|
public static Exception excessiveVelEx; // Paper - Velocity warnings
|
|
|
|
private final io.papermc.paper.logging.SysoutCatcher sysoutCatcher = new io.papermc.paper.logging.SysoutCatcher(); // Paper
|
2024-04-25 19:42:24 +02:00
|
|
|
+ private final org.bukkit.craftbukkit.potion.CraftPotionBrewer potionBrewer; // Paper - Custom Potion Mixes
|
2022-02-23 04:09:15 +01:00
|
|
|
|
|
|
|
static {
|
|
|
|
ConfigurationSerialization.registerClass(CraftOfflinePlayer.class);
|
2024-04-25 19:42:24 +02:00
|
|
|
@@ -0,0 +0,0 @@ public final class CraftServer implements Server {
|
|
|
|
if (this.configuration.getBoolean("settings.use-map-color-cache")) {
|
|
|
|
MapPalette.setMapColorCache(new CraftMapColorCache(this.logger));
|
|
|
|
}
|
|
|
|
+ this.potionBrewer = new org.bukkit.craftbukkit.potion.CraftPotionBrewer(playerList.getServer()); // Paper - custom potion mixes
|
|
|
|
datapackManager = new io.papermc.paper.datapack.PaperDatapackManager(console.getPackRepository()); // Paper
|
|
|
|
}
|
|
|
|
|
2022-02-23 04:09:15 +01:00
|
|
|
@@ -0,0 +0,0 @@ public final class CraftServer implements Server {
|
|
|
|
return datapackManager;
|
|
|
|
}
|
|
|
|
|
|
|
|
+ @Override
|
2024-04-25 19:45:17 +02:00
|
|
|
+ public org.bukkit.craftbukkit.potion.CraftPotionBrewer getPotionBrewer() {
|
2022-02-23 04:09:15 +01:00
|
|
|
+ return this.potionBrewer;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
// Paper end
|
|
|
|
}
|
|
|
|
diff --git a/src/main/java/org/bukkit/craftbukkit/inventory/CraftRecipe.java b/src/main/java/org/bukkit/craftbukkit/inventory/CraftRecipe.java
|
|
|
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
|
|
|
--- a/src/main/java/org/bukkit/craftbukkit/inventory/CraftRecipe.java
|
|
|
|
+++ b/src/main/java/org/bukkit/craftbukkit/inventory/CraftRecipe.java
|
|
|
|
@@ -0,0 +0,0 @@ public interface CraftRecipe extends Recipe {
|
|
|
|
void addToCraftingManager();
|
|
|
|
|
|
|
|
default Ingredient toNMS(RecipeChoice bukkit, boolean requireNotEmpty) {
|
|
|
|
+ // Paper start
|
|
|
|
+ return toIngredient(bukkit, requireNotEmpty);
|
|
|
|
+ }
|
|
|
|
+ static Ingredient toIngredient(RecipeChoice bukkit, boolean requireNotEmpty) {
|
|
|
|
+ // Paper end
|
|
|
|
Ingredient stack;
|
|
|
|
|
|
|
|
if (bukkit == null) {
|
2023-12-06 17:39:37 +01:00
|
|
|
diff --git a/src/main/java/org/bukkit/craftbukkit/potion/CraftPotionBrewer.java b/src/main/java/org/bukkit/craftbukkit/potion/CraftPotionBrewer.java
|
2024-04-24 22:05:42 +02:00
|
|
|
new file mode 100644
|
|
|
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000
|
|
|
|
--- /dev/null
|
2023-12-06 17:39:37 +01:00
|
|
|
+++ b/src/main/java/org/bukkit/craftbukkit/potion/CraftPotionBrewer.java
|
2024-04-24 22:05:42 +02:00
|
|
|
@@ -0,0 +0,0 @@
|
|
|
|
+package org.bukkit.craftbukkit.potion;
|
|
|
|
+
|
|
|
|
+import com.google.common.base.Preconditions;
|
|
|
|
+import java.util.ArrayList;
|
|
|
|
+import java.util.Collection;
|
|
|
|
+import org.bukkit.potion.PotionBrewer;
|
|
|
|
+import org.bukkit.potion.PotionEffect;
|
|
|
|
+import org.bukkit.potion.PotionEffectType;
|
|
|
|
+import org.bukkit.potion.PotionType;
|
|
|
|
+
|
|
|
|
+public class CraftPotionBrewer implements PotionBrewer {
|
|
|
|
+
|
2024-04-25 19:42:24 +02:00
|
|
|
+ private final net.minecraft.server.MinecraftServer minecraftServer;
|
|
|
|
+
|
|
|
|
+ public CraftPotionBrewer(net.minecraft.server.MinecraftServer minecraftServer) {
|
|
|
|
+ this.minecraftServer = minecraftServer;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ // Paper start - keep old spigot methods, removal in 1.20.6
|
2024-04-24 22:05:42 +02:00
|
|
|
+ @Override
|
|
|
|
+ public Collection<PotionEffect> getEffects(PotionType type, boolean upgraded, boolean extended) {
|
2024-04-25 19:42:24 +02:00
|
|
|
+ final org.bukkit.NamespacedKey key = type.getKey();
|
|
|
|
+
|
|
|
|
+ Preconditions.checkArgument(!key.getKey().startsWith("strong_"), "Strong potion type cannot be used directly, got %s", key);
|
|
|
|
+ Preconditions.checkArgument(!key.getKey().startsWith("long_"), "Extended potion type cannot be used directly, got %s", key);
|
|
|
|
+
|
|
|
|
+ org.bukkit.NamespacedKey effectiveKey = key;
|
|
|
|
+ if (upgraded) {
|
|
|
|
+ effectiveKey = new org.bukkit.NamespacedKey(key.namespace(), "strong_" + key.key());
|
|
|
|
+ } else if (extended) {
|
|
|
|
+ effectiveKey = new org.bukkit.NamespacedKey(key.namespace(), "long_" + key.key());
|
|
|
|
+ }
|
2024-04-24 22:05:42 +02:00
|
|
|
+
|
2024-04-25 19:42:24 +02:00
|
|
|
+ final org.bukkit.potion.PotionType effectivePotionType = org.bukkit.Registry.POTION.get(effectiveKey);
|
|
|
|
+ Preconditions.checkNotNull(type, "Unknown potion type from data " + effectiveKey.asMinimalString()); // Legacy error message in 1.20.4
|
|
|
|
+ return effectivePotionType.getPotionEffects();
|
2024-04-24 22:05:42 +02:00
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ @Override
|
|
|
|
+ public Collection<PotionEffect> getEffectsFromDamage(int damage) {
|
2024-04-25 19:42:24 +02:00
|
|
|
+ return new ArrayList<>();
|
2024-04-24 22:05:42 +02:00
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ @Override
|
|
|
|
+ public PotionEffect createEffect(PotionEffectType potion, int duration, int amplifier) {
|
2024-04-25 19:42:24 +02:00
|
|
|
+ return new PotionEffect(potion, potion.isInstant() ? 1 : duration, amplifier);
|
2024-04-24 22:05:42 +02:00
|
|
|
+ }
|
2024-04-25 19:42:24 +02:00
|
|
|
+ // Paper end - keep old spigot methods, removal in 1.20.6
|
2023-12-06 17:39:37 +01:00
|
|
|
+
|
|
|
|
+ // Paper start
|
|
|
|
+ @Override
|
|
|
|
+ public void addPotionMix(io.papermc.paper.potion.PotionMix potionMix) {
|
|
|
|
+ net.minecraft.world.item.alchemy.PotionBrewing.addPotionMix(potionMix);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ @Override
|
|
|
|
+ public void removePotionMix(org.bukkit.NamespacedKey key) {
|
|
|
|
+ net.minecraft.world.item.alchemy.PotionBrewing.removePotionMix(key);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ @Override
|
|
|
|
+ public void resetPotionMixes() {
|
2024-04-26 17:33:00 +02:00
|
|
|
+ this.minecraftServer.potionBrewing = this.minecraftServer.potionBrewing().reload(this.minecraftServer.getWorldData().enabledFeatures());
|
2023-12-06 17:39:37 +01:00
|
|
|
+ }
|
|
|
|
+ // Paper end
|
2024-04-24 22:05:42 +02:00
|
|
|
+}
|