first commit

This commit is contained in:
Soph :3 2024-10-13 15:19:27 +03:00
commit 2f5a4bc647
Signed by: sophie
GPG key ID: EDA5D222A0C270F2
74 changed files with 2126 additions and 0 deletions

12
.gitattributes vendored Normal file
View file

@ -0,0 +1,12 @@
#
# https://help.github.com/articles/dealing-with-line-endings/
#
# Linux start script should use lf
/gradlew text eol=lf
# These are Windows script files and should use crlf
*.bat text eol=crlf
# Binary files should be left untouched
*.jar binary

5
.gitignore vendored Normal file
View file

@ -0,0 +1,5 @@
# Ignore Gradle project-specific cache directory
.gradle
# Ignore Gradle build output directory
build

4
.vscode/settings.json vendored Normal file
View file

@ -0,0 +1,4 @@
{
"java.configuration.updateBuildConfiguration": "interactive",
"java.compile.nullAnalysis.mode": "disabled"
}

1
README.md Normal file
View file

@ -0,0 +1 @@
this is unfinished

View file

@ -0,0 +1,14 @@
{
"no_chat_command_enabled": "AnimalRP's chat modifications are now enabled for you.",
"no_chat_command_disabled": "AnimalRP's chat modifications are now disabled for you.",
"animal_not_set": "You do not have a animal set.",
"animal_removed": "You no longer have a animal set.",
"animal_set": "You are now an %s! ",
"your_options": "Your options are: ",
"invalid_animal": "Invalid animal!",
"tf_off": "Use /tf off to disable the changes.",
"only_animals": "Only animals can interact with other animals :(",
"no_console": "I'm sorry console.",
"no_self_argument": "You can't %1 yourself.",
"not_animal": "%1 is not an animal :("
}

View file

@ -0,0 +1,5 @@
apply plugin: 'java-library'
dependencies {
implementation 'com.google.code.gson:gson:2.11.0'
}

View file

@ -0,0 +1,30 @@
package ovh.sad.animalrp.common;
import java.util.HashMap;
import java.util.UUID;
import ovh.sad.animalrp.common.util.Animal;
import ovh.sad.animalrp.common.util.HashmapStore;
public interface AnimalRP {
public static final String MOD_ID = "animal-rp";
public static HashMap<UUID, Animal<?,?>> users = new HashMap<>();
public static HashMap<UUID, Boolean> noChat = new HashMap<>();
public static HashMap<String, Animal<?,?>> animals = new HashMap<>();
public static String welcomeMessage = """
|\\ _,,,---,,_
ZZZzz /,`.-'`' -. ;-;;,_
|,4- ) )-,_. ,\\ ( `'-'
'---''(_/--' `-'\\_)
furry animal mod
by fucksophie
rewritten for paper & fabric on October 5th
""";
}

View file

@ -0,0 +1,24 @@
package ovh.sad.animalrp.common.util;
import java.util.ArrayList;
import java.util.HashMap;
public abstract class Animal<T,A> {
public String name;
public String catchphrase;
public String color;
public HashMap<Mood, T> moodSounds = new HashMap<Mood, T>();
public ArrayList<A> superfoods = new ArrayList<A>();
public String toString() {
return this.name;
}
public Animal(String name, String catchphrase, String color) {
this.name = name;
this.catchphrase = catchphrase;
this.color = color;
}
public abstract String chatTransformations(String message);
}

View file

@ -0,0 +1,35 @@
package ovh.sad.animalrp.common.util;
import com.google.gson.JsonObject;
public class Cooldown {
public long timeCreated;
public Integer length;
public String type;
public long getTime() {
return this.timeCreated - (System.currentTimeMillis() - this.length);
}
public boolean isExpired() {
return this.getTime() <= 0;
}
public JsonObject toJson() {
JsonObject obj = new JsonObject();
obj.addProperty("timeCreated", this.timeCreated);
obj.addProperty("length", this.length);
obj.addProperty("type", this.type);
return obj;
}
public static Cooldown fromJson(JsonObject obj) {
Cooldown cldn = new Cooldown();
cldn.timeCreated = obj.get("timeCreated").getAsLong();
cldn.length = obj.get("length").getAsInt();
cldn.type = obj.get("type").getAsString();
return cldn;
}
}

View file

@ -0,0 +1,66 @@
package ovh.sad.animalrp.common.util;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.HashMap;
import com.google.gson.Gson;
import com.google.gson.JsonIOException;
import com.google.gson.JsonObject;
import com.google.gson.JsonSyntaxException;
public class HashmapStore {
public Gson gson = new Gson();
public Path folder;
public HashmapStore(Path folder) {
this.folder = folder;
}
@SuppressWarnings("unchecked")
public HashMap<String, String> get(String name) {
Path filePath = folder.resolve(name);
if (!Files.exists(filePath)) {
return new HashMap<String, String>();
}
try {
return gson.fromJson(new FileReader(filePath.toString()), HashMap.class);
} catch (JsonSyntaxException | JsonIOException | FileNotFoundException e) {
e.printStackTrace();
return new HashMap<String, String>();
}
}
@SuppressWarnings("unchecked")
public void save(String name, @SuppressWarnings("rawtypes") HashMap hashmap) {
if (!Files.exists(folder)) {
try {
Files.createDirectory(folder);
} catch (IOException e) {
e.printStackTrace();
}
}
Path filePath = folder.resolve(name);
if (!Files.exists(filePath))
try {
Files.createFile(filePath);
} catch (IOException e) {
e.printStackTrace();
}
JsonObject jsobj = new JsonObject();
hashmap.forEach((a, b) -> {
jsobj.addProperty(a.toString(), b.toString());
});
try {
Files.write(filePath, gson.toJson(jsobj).getBytes());
} catch (IOException e) {
e.printStackTrace();
}
}
}

View file

@ -0,0 +1,34 @@
package ovh.sad.animalrp.common.util;
import java.io.IOException;
import com.google.gson.Gson;
import com.google.gson.JsonObject;
import com.google.gson.JsonParser;
public class Messages {
public Gson gson = new Gson();
public static JsonObject json;
static {
String result = "";
try {
result = new String(Messages.class.getResourceAsStream("/messages.json").readAllBytes());
} catch (IOException e) {
e.printStackTrace();
}
json = JsonParser.parseString(result).getAsJsonObject();
}
/**
* All available members:
*
*
* no_chat_command_enabled
* no_chat_command_disabled
*
*/
public static String get(String memberName) {
return Messages.json.get(memberName).getAsString();
}
}

View file

@ -0,0 +1,5 @@
package ovh.sad.animalrp.common.util;
public enum Mood {
HAPPY,SAD,STRESSED,ANGRY,CUTE
}

View file

@ -0,0 +1,44 @@
package ovh.sad.animalrp.common.util;
import java.util.ArrayList;
import java.util.List;
import java.util.Random;
public class TextDestroyer {
String[] expressions;
String[][] replaces;
public TextDestroyer(String[] expressions, String[][] replaces) {
this.expressions = expressions;
this.replaces = replaces;
}
public String destroy(String message) {
final String[] words = message.split(" ");
final List<String> out = new ArrayList<>();
Random y = new Random();
for (String word : words) {
if((word.startsWith("[") && word.endsWith("]")) || word.startsWith("@")) {
out.add(word);
continue;
}
if(y.nextBoolean()){
out.add(word);
continue;
};
for(String[] replacing: this.replaces) {
word = word.replace(replacing[0], replacing[1]);
}
out.add(word);
if(y.nextDouble() < 0.12) out.add(this.expressions[y.nextInt(this.expressions.length)]);
}
return String.join(" ", out);
}
}

View file

@ -0,0 +1,14 @@
{
"no_chat_command_enabled": "AnimalRP's chat modifications are now enabled for you.",
"no_chat_command_disabled": "AnimalRP's chat modifications are now disabled for you.",
"animal_not_set": "You do not have a animal set.",
"animal_removed": "You no longer have a animal set.",
"animal_set": "You are now an %s! ",
"your_options": "Your options are: ",
"invalid_animal": "Invalid animal!",
"tf_off": "Use /tf off to disable the changes.",
"only_animals": "Only animals can interact with other animals :(",
"no_console": "I'm sorry console.",
"no_self_argument": "You can't %1 yourself.",
"not_animal": "%1 is not an animal :("
}

View file

@ -0,0 +1,13 @@
{
"required": true,
"package": "ovh.sad.animalrp.fabric.mixin",
"compatibilityLevel": "JAVA_21",
"server": [
"DecoratedMessage",
"FoodEating",
"Sneaking"
],
"injectors": {
"defaultRequire": 1
}
}

View file

@ -0,0 +1,32 @@
{
"schemaVersion": 1,
"id": "animalrp",
"version": "${version}",
"name": "AnimalRP",
"description": "AnimalRP, an plugin for furry minecraft servers that mangles your text, adds specific types of animals that have different improvements and more!",
"authors": [
"@fucksophie"
],
"contact": {
"homepage": "https://sad.ovh",
"sources": "https://git.sad.ovh/sophie/animalrp2"
},
"license": "CC0-1.0",
"icon": "assets/animal-rp/icon.png",
"environment": "*",
"entrypoints": {
"server": [
"ovh.sad.animalrp.fabric.AnimalRPFabric"
]
},
"mixins": [
"animal-rp.mixins.json"
],
"depends": {
"fabricloader": ">=0.16.4",
"minecraft": "~1.21.1",
"java": ">=21",
"fabric-api": "*",
"placeholder-api": "*"
}
}

View file

@ -0,0 +1,61 @@
import net.fabricmc.loom.task.RemapJarTask
plugins {
id 'fabric-loom' version '1.7-SNAPSHOT'
id "com.github.johnrengelman.shadow" version "8.1.1"
}
archivesBaseName = 'animalrp'
repositories {
mavenCentral()
maven { url 'https://maven.fabricmc.net/' }
maven {
url "https://maven.nucleoid.xyz/"
name "Nucleoid"
}
}
dependencies {
minecraft "com.mojang:minecraft:${project.minecraft_version}"
mappings "net.fabricmc:yarn:${project.yarn_mappings}:v2"
modImplementation "net.fabricmc:fabric-loader:${project.loader_version}"
modImplementation "net.fabricmc.fabric-api:fabric-api:${project.fabric_version}"
include(modImplementation("eu.pb4:placeholder-api:2.4.1+1.21"))
implementation project(':animalrp-common')
}
processResources {
inputs.property 'version', project.mod_version
filesMatching('**/fabric.mod.json') {
expand 'version': project.mod_version
}
}
shadowJar {
archiveFileName = "animalrpfabric-${project.mod_version}-dev.jar"
dependencies {
exclude('net.fabricmc:.*')
include(dependency('ovh.sad:.*'))
include(dependency('ovh.sad.animalrp:.*'))
// We don't want to include the mappings in the jar do we?
exclude '/mappings/*'
}
}
task remappedShadowJar(type: RemapJarTask) {
dependsOn tasks.shadowJar
input = tasks.shadowJar.archiveFile
addNestedDependencies = true
archiveFileName = "AnimalRP-Fabric-${project.mod_version}.jar"
}
tasks.assemble.dependsOn tasks.remappedShadowJar
artifacts {
archives remappedShadowJar
shadow shadowJar
}

View file

@ -0,0 +1,64 @@
package ovh.sad.animalrp.fabric;
import java.util.UUID;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import org.jetbrains.annotations.Nullable;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import eu.pb4.placeholders.api.PlaceholderContext;
import eu.pb4.placeholders.api.PlaceholderHandler;
import eu.pb4.placeholders.api.PlaceholderResult;
import eu.pb4.placeholders.api.Placeholders;
import net.fabricmc.api.ModInitializer;
import net.fabricmc.loader.api.FabricLoader;
import net.minecraft.util.Identifier;
import ovh.sad.animalrp.common.AnimalRP;
import ovh.sad.animalrp.common.util.Animal;
import ovh.sad.animalrp.common.util.HashmapStore;
import ovh.sad.animalrp.fabric.animals.Bee;
import ovh.sad.animalrp.fabric.animals.Cat;
import ovh.sad.animalrp.fabric.animals.Dog;
import ovh.sad.animalrp.fabric.animals.Fox;
public class AnimalRPFabric implements ModInitializer, AnimalRP {
public static ScheduledExecutorService executor = Executors.newSingleThreadScheduledExecutor();
public static HashmapStore hashmapStore = new HashmapStore(
FabricLoader.getInstance().getConfigDir().resolve(AnimalRP.MOD_ID).toAbsolutePath());
public static final Logger LOGGER = LoggerFactory.getLogger(MOD_ID);
@Override
public void onInitialize() {
LOGGER.info(welcomeMessage);
animals.put("cat", new Cat());
animals.put("dog", new Dog());
animals.put("fox", new Fox());
animals.put("bee", new Bee());
hashmapStore.get("users.json").forEach((k, v) -> {
users.put(UUID.fromString(k), animals.get(v));
});
hashmapStore.get("nochat.json").forEach((k, v) -> {
noChat.put(UUID.fromString(k), Boolean.valueOf(v));
});
Placeholders.register(Identifier.of("animalrp", "animalcolor"), new PlaceholderHandler() {
@Override
public PlaceholderResult onPlaceholderRequest(PlaceholderContext ctx, @Nullable String arg) {
if (!ctx.hasPlayer())
return PlaceholderResult.invalid("No player!");
Animal<?, ?> animal = users.get(ctx.player().getUuid());
if (animal == null)
return PlaceholderResult.value("");
if (noChat.get(ctx.player().getUuid()) != null)
return PlaceholderResult.value("");
return PlaceholderResult.value(animal.color);
}
});
}
}

View file

@ -0,0 +1,194 @@
package ovh.sad.animalrp.fabric.animals;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.UUID;
import java.util.concurrent.TimeUnit;
import net.fabricmc.fabric.api.event.player.UseItemCallback;
import net.minecraft.block.Block;
import net.minecraft.block.Blocks;
import net.minecraft.component.DataComponentTypes;
import net.minecraft.component.type.FoodComponent;
import net.minecraft.entity.effect.StatusEffectInstance;
import net.minecraft.entity.effect.StatusEffects;
import net.minecraft.item.Item;
import net.minecraft.item.ItemStack;
import net.minecraft.item.Items;
import net.minecraft.server.network.ServerPlayerEntity;
import net.minecraft.sound.SoundCategory;
import net.minecraft.sound.SoundEvent;
import net.minecraft.sound.SoundEvents;
import net.minecraft.util.Identifier;
import net.minecraft.util.TypedActionResult;
import ovh.sad.animalrp.common.util.Animal;
import ovh.sad.animalrp.common.util.Mood;
import ovh.sad.animalrp.common.util.TextDestroyer;
import ovh.sad.animalrp.fabric.AnimalRPFabric;
public class Bee extends Animal<SoundEvent, Item> {
class Row {
Item mat;
Integer times;
}
private static Item[] _allFlowers = { Items.ALLIUM, Items.AZURE_BLUET, Items.BLUE_ORCHID,
Items.CORNFLOWER, Items.DANDELION, Items.LILY_OF_THE_VALLEY, Items.OXEYE_DAISY,
Items.POPPY, Items.TORCHFLOWER, Items.ORANGE_TULIP, Items.PINK_TULIP, Items.RED_TULIP,
Items.WHITE_TULIP };
public static List<Item> allFlowers = Arrays.asList(_allFlowers);
public static Identifier beeFoodKey = Identifier.of("animalrp", "bee_food");
public static HashMap<UUID, Row> inARow = new HashMap<UUID, Row>();
TextDestroyer destroyer = new TextDestroyer(new String[] {
">_<", "*buzz*",
";3", ":3", "εწз", " ≧◠◡◠≦ ", "*stings you*", "*humms*",
"*i'm a bee*"
}, new String[][] {
{ "e", "ee" },
{ "b", "bzz" },
{ "h", "hh" },
{ "ie", "ee" },
{ "be", "bee" },
{ "E", "EE" },
{ "B", "BZZ" },
{ "H", "HH" },
{ "IE", "EE" },
{ "BE", "BEE" }
});
ArrayList<UUID> sneakers = new ArrayList<UUID>();
public static Boolean isItemARP(ItemStack is) {
return (allFlowers.contains(is.getItem()) && is.get(DataComponentTypes.FOOD) != null);
}
public Bee() {
super("bee", "Buzz...", "#FFFF00");
this.moodSounds.put(Mood.HAPPY, SoundEvents.ENTITY_BEE_LOOP);
this.moodSounds.put(Mood.CUTE, SoundEvents.ENTITY_BEE_LOOP);
this.moodSounds.put(Mood.SAD, SoundEvents.ENTITY_BEE_HURT);
this.moodSounds.put(Mood.STRESSED, SoundEvents.ENTITY_BEE_STING);
this.moodSounds.put(Mood.ANGRY, SoundEvents.ENTITY_BEE_LOOP_AGGRESSIVE);
UseItemCallback.EVENT.register((player, world, hand) -> {
Animal<?,?> animal = AnimalRPFabric.users.get(player.getUuid());
ItemStack item = player.getStackInHand(hand);
if (item == null) // air interact
return TypedActionResult.pass(item);
if (!allFlowers.contains(item.getItem())) { // not a flower
return TypedActionResult.pass(item);
}
Boolean incorrect = false;
if (animal == null) {
incorrect = true;
} else {
if (animal.name != this.name) {
incorrect = true;
}
}
if (incorrect) {
if (Bee.isItemARP(item)) {
item.remove(DataComponentTypes.FOOD);
return TypedActionResult.pass(item);
}
return TypedActionResult.pass(item);
}
if (Bee.isItemARP(item)) { // correct animal, but foodkey already set
return TypedActionResult.pass(item);
}
FoodComponent food = new FoodComponent.Builder()
.statusEffect(new StatusEffectInstance(StatusEffects.SPEED, 20 * 4, 1, true, true, true), 1)
.alwaysEdible()
.nutrition(4)
.saturationModifier(9.4f)
.build();
item.set(DataComponentTypes.FOOD, food);
return TypedActionResult.pass(item);
});
}
// Called from the FoodEating mixin.
// Called only if the player is a animal and a bee.
public void onEat(ServerPlayerEntity player, ItemStack item) {
if (!allFlowers.contains(item.getItem())) { // not a flower
return;
}
Row row = inARow.get(player.getUuid()); // make a new row
if (row == null) { // none yet
row = new Row();
row.mat = item.getItem();
row.times = 1;
} else {
if (row.mat.equals(item.getItem())) { // mat is same as in row, increase time
row.times += 1;
} else {
row.mat = item.getItem(); // mat not same, change mat, reset time
row.times = 1;
}
}
if (row.times > 20) {
player.addStatusEffect(new StatusEffectInstance(StatusEffects.NAUSEA, 20 * 10, 1, true, true));
}
if (row.times > 30) {
if (row.times > 40) {
player.addStatusEffect(new StatusEffectInstance(StatusEffects.WEAKNESS, 20 * 10, 3, true, true));
} else {
player.addStatusEffect(new StatusEffectInstance(StatusEffects.WEAKNESS, 20 * 5, 2, true, true));
}
}
inARow.put(player.getUuid(), row);
}
// Called from the 'Sneaking' mixin.
public void onSneak(ServerPlayerEntity player, boolean status) {
@SuppressWarnings("unchecked")
Animal<SoundEvent,?> animal = (Animal<SoundEvent, Item>) AnimalRPFabric.users.get(player.getUuid());
if (animal == null)
return;
if (animal.name != this.name)
return;
Block type = player.getWorld().getBlockState(player.getBlockPos().down()).getBlock();
if (status
&& type != Blocks.AIR && type != Blocks.WATER) {
if (!sneakers.contains(player.getUuid())) {
sneakers.add(player.getUuid());
AnimalRPFabric.executor.schedule(new Runnable() {
@Override
public void run() {
if (sneakers.contains(player.getUuid()))
sneakers.remove(player.getUuid());
}
}, 1, TimeUnit.SECONDS);
} else {
sneakers.remove(player.getUuid());
player.addStatusEffect(
new StatusEffectInstance(StatusEffects.LEVITATION, 20, 5, true, true, true));
player.getWorld().playSound(player, player.getBlockPos(),
animal.moodSounds.get(Mood.HAPPY), SoundCategory.PLAYERS, 10F, 1);
}
}
}
@Override
public String chatTransformations(String message) {
return destroyer.destroy(message);
}
}

View file

@ -0,0 +1,64 @@
package ovh.sad.animalrp.fabric.animals;
import net.fabricmc.fabric.api.entity.event.v1.ServerLivingEntityEvents;
import net.minecraft.entity.damage.DamageTypes;
import net.minecraft.item.Item;
import net.minecraft.item.Items;
import net.minecraft.server.network.ServerPlayerEntity;
import net.minecraft.sound.SoundEvent;
import net.minecraft.sound.SoundEvents;
import ovh.sad.animalrp.common.util.Animal;
import ovh.sad.animalrp.common.util.Mood;
import ovh.sad.animalrp.common.util.TextDestroyer;
import ovh.sad.animalrp.fabric.AnimalRPFabric;
public class Cat extends Animal<SoundEvent, Item> {
TextDestroyer destroyer = new TextDestroyer(new String[]{
">_<", ":3", "ʕʘ‿ʘʔ", ":D", "._.",
";3", "xD", "ㅇㅅㅇ",
">_>", "ÙωÙ", "UwU", "OwO", ":P",
"(◠‿◠✿)", "^_^", ";_;",
"x3", "(• o •)", "<_<"
}, new String[][]{
{"l", "w"},
{"r", "w"},
{"th", "d"},
{"L", "W"},
{"R", "W"},
{"TH", "D"}
});
public Cat() {
super("cat", "Nya~", "#F2BDCD");
this.moodSounds.put(Mood.HAPPY, SoundEvents.ENTITY_CAT_PURR);
this.moodSounds.put(Mood.CUTE, SoundEvents.ENTITY_CAT_PURREOW);
this.moodSounds.put(Mood.SAD, SoundEvents.ENTITY_CAT_AMBIENT);
this.moodSounds.put(Mood.STRESSED, SoundEvents.ENTITY_CAT_STRAY_AMBIENT);
this.moodSounds.put(Mood.ANGRY, SoundEvents.ENTITY_CAT_HISS);
this.superfoods.add(Items.COOKED_COD);
this.superfoods.add(Items.COD);
this.superfoods.add(Items.COOKED_SALMON);
this.superfoods.add(Items.SALMON);
ServerLivingEntityEvents.AFTER_DAMAGE.register((entity, source, baseDmg, dmg, blocked) -> {
if (entity instanceof ServerPlayerEntity player) {
Animal<?,?> animal = AnimalRPFabric.users.get(player.getUuid());
if (animal == null || !animal.name.equals(this.name)) {
return;
}
if (source.isOf(DamageTypes.FALL)) {
player.setHealth(Math.max(player.getMaxHealth(), player.getHealth() + 5));
}
}
});
}
@Override
public String chatTransformations(String message) {
return destroyer.destroy(message);
}
}

View file

@ -0,0 +1,53 @@
package ovh.sad.animalrp.fabric.animals;
import net.fabricmc.fabric.api.event.player.AttackEntityCallback;
import net.minecraft.entity.effect.StatusEffectInstance;
import net.minecraft.entity.effect.StatusEffects;
import net.minecraft.item.Item;
import net.minecraft.item.Items;
import net.minecraft.sound.SoundEvent;
import net.minecraft.sound.SoundEvents;
import net.minecraft.util.ActionResult;
import ovh.sad.animalrp.common.util.Animal;
import ovh.sad.animalrp.common.util.Mood;
import ovh.sad.animalrp.common.util.TextDestroyer;
import ovh.sad.animalrp.fabric.AnimalRPFabric;
public class Dog extends Animal<SoundEvent, Item> {
TextDestroyer destroyer = new TextDestroyer(new String[] {
"Woof!", "Bark :3",
"Arf", "bark bark bark", "arf~"
}, new String[][] {
});
public Dog() {
super("dog", "Arf!", "#ff8c00");
this.moodSounds.put(Mood.HAPPY, SoundEvents.ENTITY_WOLF_AMBIENT);
this.moodSounds.put(Mood.CUTE, SoundEvents.ENTITY_WOLF_STEP);
this.moodSounds.put(Mood.SAD, SoundEvents.ENTITY_WOLF_WHINE);
this.moodSounds.put(Mood.STRESSED, SoundEvents.ENTITY_WOLF_SHAKE);
this.moodSounds.put(Mood.ANGRY, SoundEvents.ENTITY_WOLF_GROWL);
this.superfoods.add(Items.CHICKEN);
this.superfoods.add(Items.BEEF);
this.superfoods.add(Items.PORKCHOP);
AttackEntityCallback.EVENT.register((player, world, hand, entity, hitResult) -> {
Animal<?,?> animal = AnimalRPFabric.users.get(player.getUuid());
if (animal == null)
return ActionResult.PASS;
if (animal.name != this.name)
return ActionResult.PASS;
player.removeStatusEffect(StatusEffects.SPEED);
player.addStatusEffect(
new StatusEffectInstance(StatusEffects.SPEED, 20, 2, true, true, true));
return ActionResult.PASS;
});
}
@Override
public String chatTransformations(String message) {
return destroyer.destroy(message);
}
}

View file

@ -0,0 +1,64 @@
package ovh.sad.animalrp.fabric.animals;
import net.fabricmc.fabric.api.entity.event.v1.ServerLivingEntityEvents;
import net.minecraft.entity.Entity;
import net.minecraft.entity.player.PlayerEntity;
import net.minecraft.item.Item;
import net.minecraft.item.Items;
import net.minecraft.server.network.ServerPlayerEntity;
import net.minecraft.sound.SoundEvent;
import net.minecraft.sound.SoundEvents;
import ovh.sad.animalrp.common.util.Animal;
import ovh.sad.animalrp.common.util.Mood;
import ovh.sad.animalrp.common.util.TextDestroyer;
import ovh.sad.animalrp.fabric.AnimalRPFabric;
public class Fox extends Animal<SoundEvent, Item> {
TextDestroyer destroyer = new TextDestroyer(new String[] {
"yap",
"*yap yap*",
"*beeps*",
"*barks*",
"*screeches*",
":3"
}, new String[][] {
{ "you", "u" },
{ "o", "yo" },
{ "i", "yi" },
{ "!", " !" },
{ "?", " ?" }
});
public Fox() {
super("fox", "Yap!", "#FF8000");
this.moodSounds.put(Mood.HAPPY, SoundEvents.ENTITY_FOX_SNIFF);
this.moodSounds.put(Mood.CUTE, SoundEvents.ENTITY_FOX_SLEEP);
this.moodSounds.put(Mood.SAD, SoundEvents.ENTITY_FOX_SNIFF);
this.moodSounds.put(Mood.STRESSED, SoundEvents.ENTITY_FOX_AGGRO);
this.moodSounds.put(Mood.ANGRY, SoundEvents.ENTITY_FOX_BITE);
this.superfoods.add(Items.APPLE);
this.superfoods.add(Items.GLOW_BERRIES);
ServerLivingEntityEvents.AFTER_DAMAGE.register((entity, source, baseDmg, dmg, blocked) -> {
if(!(source.getSource() instanceof ServerPlayerEntity)) return;
if(entity instanceof PlayerEntity) return;
Entity victim = entity;
ServerPlayerEntity damager = (ServerPlayerEntity) source.getSource();
Animal<?,?> animal = AnimalRPFabric.users.get(damager.getUuid());
if (animal == null || !animal.name.equals(this.name)) {
return;
}
victim.damage(victim.getDamageSources().playerAttack(damager), dmg * 0.25F);
});
}
@Override
public String chatTransformations(String message) {
return this.destroyer.destroy(message);
}
}

View file

@ -0,0 +1,95 @@
package ovh.sad.animalrp.fabric.commands;
import com.mojang.brigadier.CommandDispatcher;
import eu.pb4.placeholders.api.TextParserUtils;
import net.minecraft.command.CommandRegistryAccess;
import net.minecraft.command.argument.EntityArgumentType;
import net.minecraft.entity.Entity;
import net.minecraft.server.command.CommandManager;
import net.minecraft.server.command.ServerCommandSource;
import net.minecraft.server.network.ServerPlayerEntity;
import net.minecraft.sound.SoundCategory;
import net.minecraft.sound.SoundEvent;
import net.minecraft.text.Text;
import ovh.sad.animalrp.common.util.Animal;
import ovh.sad.animalrp.common.util.Messages;
import ovh.sad.animalrp.common.util.Mood;
import ovh.sad.animalrp.fabric.AnimalRPFabric;
public class InteractionCommand {
String command;
String toThem;
String toYou;
Mood mood;
public InteractionCommand(String command, Mood mood, String toThem, String toYou) {
this.command = command;
this.toThem = toThem;
this.toYou = toYou;
this.mood = mood;
}
@SuppressWarnings("deprecation")
public void Command(CommandDispatcher<ServerCommandSource> dispatcher, CommandRegistryAccess registryAccess,
CommandManager.RegistrationEnvironment environment) {
dispatcher.register(CommandManager.literal(this.command)
.then(CommandManager.argument("player", EntityArgumentType.player())
.executes(context -> {
Entity sender = context.getSource().getEntity();
if (!(sender instanceof ServerPlayerEntity)) {
context.getSource().sendFeedback(
() -> Text.literal(Messages.get("no_console")).withColor(8421504), false);
return 0;
}
ServerPlayerEntity player = context.getSource().getPlayer();
Animal<?,?> aplayer = AnimalRPFabric.users.get(player.getUuid());
if (aplayer == null) {
context.getSource().sendFeedback(
() -> Text.literal(Messages.get("only_animals"))
.withColor(8421504),
false);
return 0;
}
ServerPlayerEntity splayer = EntityArgumentType.getPlayer(context, "player");
if (splayer.getName() == player.getName()) {
context.getSource().sendFeedback(
() -> Text.literal(String.format(Messages.get("no_self_argument"), this.command))
.withColor(8421504),
false);
return 0;
}
@SuppressWarnings("unchecked")
Animal<SoundEvent,Mood> asplayer = (Animal<SoundEvent, Mood>) AnimalRPFabric.users.get(splayer.getUuid());
if (asplayer == null) {
context.getSource().sendFeedback(
() -> Text.literal(String.format(Messages.get("not_animal"), splayer.getName()))
.withColor(8421504),
false);
return 0;
}
splayer.sendMessage(TextParserUtils.formatText(
String.format(this.toThem,
"<light_purple>" + player.getName().getString() + "</light_purple>",
"<italic><gray>" + aplayer.catchphrase)));
player.sendMessage(TextParserUtils.formatText(
String.format(this.toYou,
"<light_purple>" + splayer.getName().getString() + "</light_purple>",
"<italic><gray>" + asplayer.catchphrase)));
player.getWorld().playSound(splayer, splayer.getBlockPos(),
asplayer.moodSounds.get(this.mood), SoundCategory.PLAYERS, 1F,
1);
return 1;
})));
}
}

View file

@ -0,0 +1,40 @@
package ovh.sad.animalrp.fabric.commands;
import java.util.UUID;
import com.mojang.brigadier.CommandDispatcher;
import net.minecraft.command.CommandRegistryAccess;
import net.minecraft.server.command.CommandManager;
import net.minecraft.server.command.ServerCommandSource;
import net.minecraft.text.Text;
import ovh.sad.animalrp.common.AnimalRP;
import ovh.sad.animalrp.common.util.Messages;
import ovh.sad.animalrp.fabric.AnimalRPFabric;
public class NoChatCommand {
public void Command(CommandDispatcher<ServerCommandSource> dispatcher, CommandRegistryAccess registryAccess,
CommandManager.RegistrationEnvironment environment) {
dispatcher.register(CommandManager.literal("disableanimalchat").executes(context -> {
UUID userUuid = context.getSource().getEntity().getUuid();
Boolean isDisabled = AnimalRP.noChat.get(context.getSource().getEntity().getUuid());
if (isDisabled == null)
isDisabled = false;
if (isDisabled) { //
context.getSource().sendFeedback(
() -> Text.literal(Messages.get("no_chat_command_enabled")).withColor(65280),
false);
AnimalRPFabric.noChat.remove(userUuid);
} else {
context.getSource().sendFeedback(() -> Text
.literal(Messages.get("no_chat_command_disabled")).withColor(16711680), false);
AnimalRPFabric.noChat.put(userUuid, true);
}
AnimalRPFabric.hashmapStore.save("nochat.json", AnimalRPFabric.noChat);
return 0;
}));
}
}

View file

@ -0,0 +1,72 @@
package ovh.sad.animalrp.fabric.commands;
import java.util.Map.Entry;
import com.mojang.brigadier.CommandDispatcher;
import com.mojang.brigadier.arguments.StringArgumentType;
import net.minecraft.command.CommandRegistryAccess;
import net.minecraft.entity.Entity;
import net.minecraft.server.command.CommandManager;
import net.minecraft.server.command.ServerCommandSource;
import net.minecraft.text.MutableText;
import net.minecraft.text.Text;
import net.minecraft.util.Formatting;
import ovh.sad.animalrp.common.util.Animal;
import ovh.sad.animalrp.common.util.Messages;
import ovh.sad.animalrp.fabric.AnimalRPFabric;
public class TfCommand {
public void Command(CommandDispatcher<ServerCommandSource> dispatcher, CommandRegistryAccess registryAccess,
CommandManager.RegistrationEnvironment environment) {
dispatcher.register(CommandManager.literal("tf")
.then(CommandManager.argument("animal", StringArgumentType.string())
.executes(context -> {
final Entity entity = context.getSource().getEntity();
final String animalString = StringArgumentType.getString(context, "animal");
Animal<?, ?> animal = AnimalRPFabric.animals.get(animalString);
if (animalString.equals("off")) {
if (AnimalRPFabric.users.get(entity.getUuid()) == null) {
context.getSource().sendFeedback(
() -> Text.literal(Messages.get("animal_not_set")), false);
return 0;
}
AnimalRPFabric.users.remove(entity.getUuid());
context.getSource().sendFeedback(
() -> Text.literal(Messages.get("animal_removed")), false);
return 0;
}
if (animal == null) {
classicError(context.getSource());
return 0;
}
AnimalRPFabric.users.put(entity.getUuid(), animal);
context.getSource()
.sendFeedback(
() -> Text.literal(String.format(Messages.get("animal_set"), animalString))
.append(Text.literal(animal.catchphrase)
.formatted(Formatting.ITALIC).withColor(
Integer.parseInt(animal.color.substring(1),
16))),
false);
AnimalRPFabric.hashmapStore.save("users.json", AnimalRPFabric.users);
return 1;
})));
}
void classicError(ServerCommandSource source) {
MutableText options = Text.literal(Messages.get("your_options"));
for (Entry<String, Animal<?, ?>> entry : AnimalRPFabric.animals.entrySet()) {
options.append(Text.literal(entry.getKey() + " ")
.withColor(Integer.parseInt(entry.getValue().color.substring(1), 16)));
}
source.sendFeedback(() -> Text.literal(Messages.get("invalid_animal")).withColor(16711680), false);
source.sendFeedback(() -> options, false);
source.sendFeedback(() -> Text.literal(Messages.get("tf_off")), false);
}
}

View file

@ -0,0 +1,46 @@
package ovh.sad.animalrp.fabric.mixin;
import java.util.Random;
import org.jetbrains.annotations.NotNull;
import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.Shadow;
import org.spongepowered.asm.mixin.Unique;
import org.spongepowered.asm.mixin.injection.At;
import org.spongepowered.asm.mixin.injection.ModifyVariable;
import net.minecraft.network.message.SignedMessage;
import net.minecraft.server.network.ServerPlayNetworkHandler;
import net.minecraft.server.network.ServerPlayerEntity;
import net.minecraft.sound.SoundCategory;
import net.minecraft.sound.SoundEvent;
import net.minecraft.text.Text;
import ovh.sad.animalrp.common.util.Animal;
import ovh.sad.animalrp.common.util.Mood;
import ovh.sad.animalrp.fabric.AnimalRPFabric;
@Mixin(value = ServerPlayNetworkHandler.class, priority = 1) // make sure we run FIRST
public abstract class DecoratedMessage {
@Shadow
public ServerPlayerEntity player;
@Unique
Random random = new Random();
@ModifyVariable(method = "handleDecoratedMessage", at = @At(value = "HEAD"), argsOnly = true)
public @NotNull SignedMessage modifyChatMessageSentByPlayers(@NotNull SignedMessage original) {
if (AnimalRPFabric.noChat.get(player.getUuid()) != null)
return original;
@SuppressWarnings("unchecked")
Animal<SoundEvent, Mood> animal = (Animal<SoundEvent, Mood>) AnimalRPFabric.users.get(player.getUuid());
if (animal == null)
return original;
if (random.nextDouble() < 0.08) {
player.getWorld().playSound(player, player.getBlockPos(),
animal.moodSounds.get(Mood.HAPPY), SoundCategory.PLAYERS, 10F, 1);
}
return original
.withUnsignedContent(Text.literal(animal.chatTransformations(original.getContent().getString())));
}
}

View file

@ -0,0 +1,49 @@
package ovh.sad.animalrp.fabric.mixin;
import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.injection.At;
import org.spongepowered.asm.mixin.injection.Inject;
import org.spongepowered.asm.mixin.injection.callback.CallbackInfoReturnable;
import net.minecraft.component.type.FoodComponent;
import net.minecraft.entity.EntityType;
import net.minecraft.entity.LivingEntity;
import net.minecraft.entity.effect.StatusEffectInstance;
import net.minecraft.entity.effect.StatusEffects;
import net.minecraft.item.ItemStack;
import net.minecraft.server.network.ServerPlayerEntity;
import net.minecraft.world.World;
import ovh.sad.animalrp.common.util.Animal;
import ovh.sad.animalrp.fabric.AnimalRPFabric;
import ovh.sad.animalrp.fabric.animals.Bee;
@Mixin(value = LivingEntity.class)
public class FoodEating {
@Inject(method = "eatFood", at = @At("HEAD"))
public void eatFood(World world, ItemStack stack, FoodComponent foodComponent, CallbackInfoReturnable<?> cfr) {
LivingEntity entity = (LivingEntity) (Object) this;
if (entity.getType().equals(EntityType.PLAYER)) {
ServerPlayerEntity player = (ServerPlayerEntity) entity;
Animal<?, ?> animal = AnimalRPFabric.users.get(player.getUuid());
if (animal != null) {
if (animal.name == "bee") {
((Bee) animal).onEat(player, stack);
}
if (animal.superfoods.contains(stack.getItem())) {
player.getHungerManager().add(4, 9.4f);
StatusEffectInstance effect = player.getStatusEffect(StatusEffects.SPEED);
int duration = 20 * 4;
if (effect != null) {
duration += effect.getDuration();
}
player.addStatusEffect(
new StatusEffectInstance(StatusEffects.SPEED, duration, 1, true, true, true));
}
}
}
}
}

View file

@ -0,0 +1,24 @@
package ovh.sad.animalrp.fabric.mixin;
import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.injection.At;
import org.spongepowered.asm.mixin.injection.Inject;
import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
import net.minecraft.entity.Entity;
import net.minecraft.entity.EntityType;
import net.minecraft.server.network.ServerPlayerEntity;
import ovh.sad.animalrp.fabric.AnimalRPFabric;
import ovh.sad.animalrp.fabric.animals.Bee;
@Mixin(Entity.class)
public class Sneaking {
@Inject(method = "setSneaking", at = @At("HEAD"))
public void setSneaking(boolean sneaking, CallbackInfo info) {
Entity entity = (Entity) (Object) this;
if (entity.getType() == EntityType.PLAYER) {
Bee bee = (Bee) AnimalRPFabric.animals.get("bee");
bee.onSneak((ServerPlayerEntity) entity, sneaking);
}
}
}

View file

@ -0,0 +1,13 @@
{
"required": true,
"package": "ovh.sad.animalrp.fabric.mixin",
"compatibilityLevel": "JAVA_21",
"server": [
"DecoratedMessage",
"FoodEating",
"Sneaking"
],
"injectors": {
"defaultRequire": 1
}
}

View file

@ -0,0 +1,32 @@
{
"schemaVersion": 1,
"id": "animalrp",
"version": "${version}",
"name": "AnimalRP",
"description": "AnimalRP, an plugin for furry minecraft servers that mangles your text, adds specific types of animals that have different improvements and more!",
"authors": [
"@fucksophie"
],
"contact": {
"homepage": "https://sad.ovh",
"sources": "https://git.sad.ovh/sophie/animalrp2"
},
"license": "CC0-1.0",
"icon": "assets/animal-rp/icon.png",
"environment": "*",
"entrypoints": {
"server": [
"ovh.sad.animalrp.fabric.AnimalRPFabric"
]
},
"mixins": [
"animal-rp.mixins.json"
],
"depends": {
"fabricloader": ">=0.16.4",
"minecraft": "~1.21.1",
"java": ">=21",
"fabric-api": "*",
"placeholder-api": "*"
}
}

View file

@ -0,0 +1,18 @@
apply plugin: 'java'
repositories {
maven {
name = "papermc"
url = uri("https://repo.papermc.io/repository/maven-public/")
}
}
dependencies {
compileOnly("io.papermc.paper:paper-api:1.21.1-R0.1-SNAPSHOT")
implementation project(':animalrp-common')
}
java {
toolchain.languageVersion.set(JavaLanguageVersion.of(21))
}

View file

@ -0,0 +1,34 @@
package ovh.sad.animalrp.paper;
import java.util.UUID;
import java.util.logging.Level;
import org.bukkit.plugin.java.JavaPlugin;
import ovh.sad.animalrp.common.AnimalRP;
import ovh.sad.animalrp.common.util.HashmapStore;
public class AnimalRPPaper extends JavaPlugin implements AnimalRP {
public static HashmapStore hashmapStore;
@Override
public void onEnable() {
AnimalRPPaper.hashmapStore = new HashmapStore(this.getDataFolder().toPath());
this.getLogger().log(Level.INFO, welcomeMessage);
hashmapStore.get("users.json").forEach((k, v) -> {
users.put(UUID.fromString(k), animals.get(v));
});
hashmapStore.get("nochat.json").forEach((k, v) -> {
noChat.put(UUID.fromString(k), Boolean.valueOf(v));
});
}
@Override
public void onDisable() {
hashmapStore.save("users.json", users);
hashmapStore.save("nochat.json", noChat);
}
}

View file

@ -0,0 +1,201 @@
package ovh.sad.animalrp.paper.animals;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.UUID;
import org.bukkit.Material;
import org.bukkit.Sound;
import org.bukkit.block.BlockFace;
import org.bukkit.entity.Player;
import org.bukkit.event.EventHandler;
import org.bukkit.event.Listener;
import org.bukkit.event.player.PlayerInteractEvent;
import org.bukkit.event.player.PlayerItemConsumeEvent;
import org.bukkit.event.player.PlayerToggleSneakEvent;
import org.bukkit.inventory.ItemStack;
import org.bukkit.inventory.meta.ItemMeta;
import org.bukkit.inventory.meta.components.FoodComponent;
import org.bukkit.potion.PotionEffect;
import org.bukkit.potion.PotionEffectType;
import org.bukkit.scheduler.BukkitRunnable;
import ovh.sad.animalrp.common.AnimalRP;
import ovh.sad.animalrp.common.util.Animal;
import ovh.sad.animalrp.common.util.Mood;
import ovh.sad.animalrp.common.util.TextDestroyer;
import ovh.sad.animalrp.paper.AnimalRPPaper;
public class Bee extends Animal<Sound, Material> implements Listener {
class Row {
Material mat;
Integer times;
}
private static Material[] _allFlowers = { Material.ALLIUM, Material.AZURE_BLUET, Material.BLUE_ORCHID,
Material.CORNFLOWER, Material.DANDELION, Material.LILY_OF_THE_VALLEY, Material.OXEYE_DAISY,
Material.POPPY, Material.TORCHFLOWER, Material.ORANGE_TULIP, Material.PINK_TULIP, Material.RED_TULIP,
Material.WHITE_TULIP };
public static List<Material> allFlowers = Arrays.asList(_allFlowers);
public static HashMap<UUID, Row> inARow = new HashMap<UUID, Row>();
TextDestroyer destroyer = new TextDestroyer(new String[] {
">_<", "*buzz*",
";3", ":3", "εწз", " ≧◠◡◠≦ ", "*stings you*", "*humms*",
"*i'm a bee*"
}, new String[][] {
{ "e", "ee" },
{ "b", "bzz" },
{ "h", "hh" },
{ "ie", "ee" },
{ "be", "bee" },
{ "E", "EE" },
{ "B", "BZZ" },
{ "H", "HH" },
{ "IE", "EE" },
{ "BE", "BEE" }
});
ArrayList<UUID> sneakers = new ArrayList<UUID>();
public static Boolean isItemARP(ItemStack is) {
return (allFlowers.contains(is.getType()) && is.getItemMeta().hasFood());
}
public Bee() {
super("bee", "Buzz...", "#FFFF00");
this.moodSounds.put(Mood.HAPPY, Sound.ENTITY_BEE_LOOP);
this.moodSounds.put(Mood.CUTE, Sound.ENTITY_BEE_LOOP);
this.moodSounds.put(Mood.SAD, Sound.ENTITY_BEE_HURT);
this.moodSounds.put(Mood.STRESSED, Sound.ENTITY_BEE_STING);
this.moodSounds.put(Mood.ANGRY, Sound.ENTITY_BEE_LOOP_AGGRESSIVE);
}
@EventHandler
public void onInteract(PlayerInteractEvent event) {
Player player = event.getPlayer();
Animal<?,?> animal = AnimalRP.users.get(player.getUniqueId());
ItemStack item = event.getItem();
if (item == null) // air interact
return;
if (!allFlowers.contains(item.getType())) { // not a flower
return;
}
ItemMeta meta = item.getItemMeta();
Boolean incorrect = false;
if (animal == null) {
incorrect = true;
} else {
if (animal.name != this.name) {
incorrect = true;
}
}
if (incorrect) {
if (Bee.isItemARP(item)) {
meta.setFood(null);
item.setItemMeta(meta);
event.setCancelled(true);
}
return;
}
if (Bee.isItemARP(item)) { // correct animal, but foodkey already set
return;
}
FoodComponent food = meta.getFood();
food.addEffect(new PotionEffect(PotionEffectType.SPEED, 20 * 4, 1, true), 1);
food.setCanAlwaysEat(true);
food.setNutrition(4); // these values match the 'superfood' of animalrps
food.setSaturation(9.4f);
meta.setFood(food);
item.setItemMeta(meta);
}
@EventHandler
public void onConsume(PlayerItemConsumeEvent event) {
Player player = event.getPlayer();
Animal<?,?> animal = AnimalRP.users.get(player.getUniqueId());
ItemStack item = event.getItem();
if (!allFlowers.contains(item.getType())) { // not a flower
return;
}
if (animal == null)
return;
if (animal.name != this.name)
return;
Row row = inARow.get(player.getUniqueId()); // make a new row
if (row == null) { // none yet
row = new Row();
row.mat = item.getType();
row.times = 1;
} else {
if (row.mat.equals(item.getType())) { // mat is same as in row, increase time
row.times += 1;
} else {
row.mat = item.getType(); // mat not same, change mat, reset time
row.times = 1;
}
}
if (row.times > 20) {
player.addPotionEffect(new PotionEffect(PotionEffectType.NAUSEA, 20 * 10, 1, true));
}
if (row.times > 30) {
if (row.times > 40) {
player.addPotionEffect(new PotionEffect(PotionEffectType.WEAKNESS, 20 * 10, 3, true));
} else {
player.addPotionEffect(new PotionEffect(PotionEffectType.WEAKNESS, 20 * 5, 2, true));
}
}
inARow.put(player.getUniqueId(), row);
}
@EventHandler
public void onSneak(PlayerToggleSneakEvent event) {
Animal<?,?> animal = AnimalRP.users.get(event.getPlayer().getUniqueId());
if (animal == null)
return;
if (animal.name != this.name)
return;
Material type = event.getPlayer().getLocation().getBlock().getRelative(BlockFace.DOWN).getType();
if (event.isSneaking()
&& type != Material.AIR && type != Material.WATER) {
Player player = event.getPlayer();
if (!sneakers.contains(player.getUniqueId())) {
sneakers.add(player.getUniqueId());
new BukkitRunnable() {
@Override
public void run() {
if (sneakers.contains(player.getUniqueId()))
sneakers.remove(player.getUniqueId());
}
}.runTaskLater(AnimalRPPaper.getProvidingPlugin(Animal.class), 20);
} else {
sneakers.remove(player.getUniqueId());
player.addPotionEffect(new PotionEffect(PotionEffectType.LEVITATION, 20 * 1, 5, true));
player.getWorld().playSound(player.getLocation(), this.moodSounds.get(Mood.HAPPY), 1F, 1);
}
}
}
@Override
public String chatTransformations(String message) {
return destroyer.destroy(message);
}
}

View file

@ -0,0 +1,62 @@
package ovh.sad.animalrp.paper.animals;
import org.bukkit.Material;
import org.bukkit.Sound;
import org.bukkit.entity.Player;
import org.bukkit.event.EventHandler;
import org.bukkit.event.entity.EntityDamageEvent;
import org.bukkit.event.entity.EntityDamageEvent.DamageCause;
import ovh.sad.animalrp.common.AnimalRP;
import ovh.sad.animalrp.common.util.Animal;
import ovh.sad.animalrp.common.util.Mood;
import ovh.sad.animalrp.common.util.TextDestroyer;
public class Cat extends Animal<Sound, Material> {
TextDestroyer destroyer = new TextDestroyer(new String[]{
">_<", ":3", "ʕʘ‿ʘʔ", ":D", "._.",
";3", "xD", "ㅇㅅㅇ",
">_>", "ÙωÙ", "UwU", "OwO", ":P",
"(◠‿◠✿)", "^_^", ";_;",
"x3", "(• o •)", "<_<"
}, new String[][]{
{"l", "w"},
{"r", "w"},
{"th", "d"},
{"L", "W"},
{"R", "W"},
{"TH", "D"}
});
public Cat() {
super("cat", "Nya~", "#F2BDCD");
this.moodSounds.put(Mood.HAPPY, Sound.ENTITY_CAT_PURR);
this.moodSounds.put(Mood.CUTE, Sound.ENTITY_CAT_PURREOW);
this.moodSounds.put(Mood.SAD, Sound.ENTITY_CAT_AMBIENT);
this.moodSounds.put(Mood.STRESSED, Sound.ENTITY_CAT_STRAY_AMBIENT);
this.moodSounds.put(Mood.ANGRY, Sound.ENTITY_CAT_HISS);
this.superfoods.add(Material.COOKED_COD);
this.superfoods.add(Material.COD);
this.superfoods.add(Material.COOKED_SALMON);
this.superfoods.add(Material.SALMON);
}
@EventHandler
public void onDamage(EntityDamageEvent event) {
if(!(event.getEntity() instanceof Player)) return;
Player player = (Player)event.getEntity();
Animal<?,?> animal = AnimalRP.users.get(player.getUniqueId());
if(animal == null) return;
if(animal.name != this.name) return;
if(event.getCause() == DamageCause.FALL) {
event.setDamage(event.getDamage() - 5);
}
}
@Override
public String chatTransformations(String message) {
return destroyer.destroy(message);
}
}

View file

@ -0,0 +1,54 @@
package ovh.sad.animalrp.paper.animals;
import org.bukkit.Material;
import org.bukkit.Sound;
import org.bukkit.entity.Player;
import org.bukkit.event.EventHandler;
import org.bukkit.event.entity.EntityDamageByEntityEvent;
import org.bukkit.potion.PotionEffect;
import org.bukkit.potion.PotionEffectType;
import ovh.sad.animalrp.common.util.Animal;
import ovh.sad.animalrp.common.util.Mood;
import ovh.sad.animalrp.common.util.TextDestroyer;
import ovh.sad.animalrp.paper.AnimalRPPaper;
public class Dog extends Animal<Sound, Material> {
TextDestroyer destroyer = new TextDestroyer(new String[]{
"Woof!", "Bark :3",
"Arf", "bark bark bark", "arf~"
}, new String[][]{
});
public Dog() {
super("dog", "Arf!", "#ff8c00");
this.moodSounds.put(Mood.HAPPY, Sound.ENTITY_WOLF_AMBIENT);
this.moodSounds.put(Mood.CUTE, Sound.ENTITY_WOLF_STEP);
this.moodSounds.put(Mood.SAD, Sound.ENTITY_WOLF_WHINE);
this.moodSounds.put(Mood.STRESSED, Sound.ENTITY_WOLF_SHAKE);
this.moodSounds.put(Mood.ANGRY, Sound.ENTITY_WOLF_GROWL);
this.superfoods.add(Material.CHICKEN);
this.superfoods.add(Material.BEEF);
this.superfoods.add(Material.PORKCHOP);
}
@EventHandler
public void onDamage(EntityDamageByEntityEvent event) {
if(event.getEntity() instanceof Player) {
Player player = (Player)event.getEntity();
Animal<?, ?> animal = AnimalRPPaper.users.get(player.getUniqueId());
if(animal == null) return;
if(animal.name != this.name) return;
player.removePotionEffect(PotionEffectType.SPEED);
player.addPotionEffect(new PotionEffect(PotionEffectType.SPEED, 20, 2));
}
}
@Override
public String chatTransformations(String message) {
return destroyer.destroy(message);
}
}

View file

@ -0,0 +1,67 @@
package ovh.sad.animalrp.paper.animals;
import java.util.Random;
import org.bukkit.Material;
import org.bukkit.Sound;
import org.bukkit.entity.EntityType;
import org.bukkit.entity.Player;
import org.bukkit.event.EventHandler;
import org.bukkit.event.Listener;
import org.bukkit.event.entity.EntityDamageByEntityEvent;
import org.bukkit.event.entity.EntityDamageEvent.DamageCause;
import ovh.sad.animalrp.common.AnimalRP;
import ovh.sad.animalrp.common.util.Animal;
import ovh.sad.animalrp.common.util.Mood;
import ovh.sad.animalrp.common.util.TextDestroyer;
public class Fox extends Animal<Sound, Material> implements Listener {
TextDestroyer destroyer = new TextDestroyer(new String[]{
"yap",
"*yap yap*",
"*beeps*",
"*barks*",
"*screeches*",
":3"
}, new String[][]{
{"you", "u"},
{"o", "yo"},
{"i", "yi"},
{"!", " !"},
{"?", " ?"}
});
Random rand = new Random();
public Fox() {
super("fox", "Yap!", "#FF8000");
this.moodSounds.put(Mood.HAPPY, Sound.ENTITY_FOX_SNIFF);
this.moodSounds.put(Mood.CUTE, Sound.ENTITY_FOX_SLEEP);
this.moodSounds.put(Mood.SAD, Sound.ENTITY_FOX_SNIFF);
this.moodSounds.put(Mood.STRESSED, Sound.ENTITY_FOX_AGGRO);
this.moodSounds.put(Mood.ANGRY, Sound.ENTITY_FOX_BITE);
this.superfoods.add(Material.APPLE);
this.superfoods.add(Material.GLOW_BERRIES);
}
@EventHandler
public void onDamage(EntityDamageByEntityEvent event) {
if(event.getDamager() instanceof Player) {
Player player = (Player)event.getDamager();
Animal<?, ?> animal = AnimalRP.users.get(player.getUniqueId());
if(animal == null) return;
if(animal.name != this.name) return;
if(event.getCause() == DamageCause.ENTITY_ATTACK && event.getEntity().getType() != EntityType.PLAYER) {
event.setDamage(event.getDamage()*1.25);
}
}
}
@Override
public String chatTransformations(String message) {
return this.destroyer.destroy(message);
}
}

View file

@ -0,0 +1,40 @@
package ovh.sad.animalrp.paper.listeners;
import java.util.Random;
import org.bukkit.Material;
import org.bukkit.Sound;
import org.bukkit.SoundCategory;
import org.bukkit.entity.Player;
import org.bukkit.event.EventHandler;
import org.bukkit.event.EventPriority;
import org.bukkit.event.Listener;
import io.papermc.paper.event.player.AsyncChatEvent;
import ovh.sad.animalrp.common.util.Animal;
import ovh.sad.animalrp.common.util.Mood;
import ovh.sad.animalrp.paper.AnimalRPPaper;
public abstract class PlayerChat implements Listener {
Random random = new Random();
@EventHandler(priority = EventPriority.LOWEST)
public void onPlayerChat(final AsyncChatEvent event) {
Player player = event.getPlayer();
if (AnimalRPPaper.noChat.get(player.getUniqueId()) != null)
return;
@SuppressWarnings("unchecked")
Animal<Sound, Material> animal = (Animal<Sound, Material>) AnimalRPPaper.users.get(player.getUuid());
if (animal == null)
return;
if (random.nextDouble() < 0.08) {
player.getWorld().playSound(player.getLocation(),
animal.moodSounds.get(Mood.HAPPY), SoundCategory.PLAYERS, 10F, 1);
}
event.message()
return original
.withUnsignedContent(Text.literal(animal.chatTransformations(original.getContent().getString())));
}
}

42
build.gradle Normal file
View file

@ -0,0 +1,42 @@
import org.gradle.api.tasks.testing.logging.TestExceptionFormat
import org.gradle.api.tasks.testing.logging.TestLogEvent
defaultTasks 'build'
subprojects {
apply plugin: 'java'
apply plugin: 'maven-publish'
group = 'ovh.sad.animalrp'
version = "${project.mod_version}"
sourceCompatibility = 21
targetCompatibility = 21
tasks.withType(JavaCompile).configureEach {
options.encoding = 'UTF-8'
options.release = 21;
}
jar {
from '../LICENSE.txt'
}
repositories {
// Fix issue with lwjgl-freetype not being found on macOS / ForgeGradle issue
//
// Could not resolve all files for configuration ':_compileJava_1'.
// Could not find lwjgl-freetype-3.3.3-natives-macos-patch.jar (org.lwjgl:lwjgl-freetype:3.3.3).
maven {
url "https://libraries.minecraft.net"
content {
includeModule("org.lwjgl", "lwjgl-freetype")
}
}
mavenCentral()
maven { url 'https://s01.oss.sonatype.org/content/repositories/snapshots/' }
maven { url 'https://oss.sonatype.org/content/repositories/snapshots/' }
maven { url 'https://libraries.minecraft.net/' }
maven { url "https://repo.william278.net/releases" }
}
}

19
gradle.properties Normal file
View file

@ -0,0 +1,19 @@
# Done to increase the memory available to gradle.
org.gradle.jvmargs=-Xmx2G
org.gradle.parallel=true
# Fabric Properties
# check these on https://fabricmc.net/develop
minecraft_version=1.21.1
yarn_mappings=1.21.1+build.3
loader_version=0.16.4
# Mod Properties
mod_version=0.1
group=ovh.sad.animalrp
maven_group=ovh.sad.animalrp
archives_base_name=animal-rp
# Dependencies
fabric_version=0.103.0+1.21.1

View file

@ -0,0 +1,7 @@
[versions]
placeholderApi = "2.4.1+1.21"
tomlj = "1.1.1"
[libraries]
placeholderApi = { group = "eu.pb4", name = "placeholder-api", version.ref = "placeholderApi"}
tomlj = { group = "org.tomlj", name = "tomlj", version.ref = "tomlj" }

BIN
gradle/wrapper/gradle-wrapper.jar vendored Normal file

Binary file not shown.

View file

@ -0,0 +1,7 @@
distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-8.9-bin.zip
networkTimeout=10000
validateDistributionUrl=true
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists

252
gradlew vendored Executable file
View file

@ -0,0 +1,252 @@
#!/bin/sh
#
# Copyright © 2015-2021 the original authors.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# https://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
# SPDX-License-Identifier: Apache-2.0
#
##############################################################################
#
# Gradle start up script for POSIX generated by Gradle.
#
# Important for running:
#
# (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is
# noncompliant, but you have some other compliant shell such as ksh or
# bash, then to run this script, type that shell name before the whole
# command line, like:
#
# ksh Gradle
#
# Busybox and similar reduced shells will NOT work, because this script
# requires all of these POSIX shell features:
# * functions;
# * expansions «$var», «${var}», «${var:-default}», «${var+SET}»,
# «${var#prefix}», «${var%suffix}», and «$( cmd )»;
# * compound commands having a testable exit status, especially «case»;
# * various built-in commands including «command», «set», and «ulimit».
#
# Important for patching:
#
# (2) This script targets any POSIX shell, so it avoids extensions provided
# by Bash, Ksh, etc; in particular arrays are avoided.
#
# The "traditional" practice of packing multiple parameters into a
# space-separated string is a well documented source of bugs and security
# problems, so this is (mostly) avoided, by progressively accumulating
# options in "$@", and eventually passing that to Java.
#
# Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS,
# and GRADLE_OPTS) rely on word-splitting, this is performed explicitly;
# see the in-line comments for details.
#
# There are tweaks for specific operating systems such as AIX, CygWin,
# Darwin, MinGW, and NonStop.
#
# (3) This script is generated from the Groovy template
# https://github.com/gradle/gradle/blob/HEAD/platforms/jvm/plugins-application/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt
# within the Gradle project.
#
# You can find Gradle at https://github.com/gradle/gradle/.
#
##############################################################################
# Attempt to set APP_HOME
# Resolve links: $0 may be a link
app_path=$0
# Need this for daisy-chained symlinks.
while
APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path
[ -h "$app_path" ]
do
ls=$( ls -ld "$app_path" )
link=${ls#*' -> '}
case $link in #(
/*) app_path=$link ;; #(
*) app_path=$APP_HOME$link ;;
esac
done
# This is normally unused
# shellcheck disable=SC2034
APP_BASE_NAME=${0##*/}
# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036)
APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s
' "$PWD" ) || exit
# Use the maximum available, or set MAX_FD != -1 to use that value.
MAX_FD=maximum
warn () {
echo "$*"
} >&2
die () {
echo
echo "$*"
echo
exit 1
} >&2
# OS specific support (must be 'true' or 'false').
cygwin=false
msys=false
darwin=false
nonstop=false
case "$( uname )" in #(
CYGWIN* ) cygwin=true ;; #(
Darwin* ) darwin=true ;; #(
MSYS* | MINGW* ) msys=true ;; #(
NONSTOP* ) nonstop=true ;;
esac
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
# Determine the Java command to use to start the JVM.
if [ -n "$JAVA_HOME" ] ; then
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
# IBM's JDK on AIX uses strange locations for the executables
JAVACMD=$JAVA_HOME/jre/sh/java
else
JAVACMD=$JAVA_HOME/bin/java
fi
if [ ! -x "$JAVACMD" ] ; then
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
else
JAVACMD=java
if ! command -v java >/dev/null 2>&1
then
die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
fi
# Increase the maximum file descriptors if we can.
if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then
case $MAX_FD in #(
max*)
# In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked.
# shellcheck disable=SC2039,SC3045
MAX_FD=$( ulimit -H -n ) ||
warn "Could not query maximum file descriptor limit"
esac
case $MAX_FD in #(
'' | soft) :;; #(
*)
# In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked.
# shellcheck disable=SC2039,SC3045
ulimit -n "$MAX_FD" ||
warn "Could not set maximum file descriptor limit to $MAX_FD"
esac
fi
# Collect all arguments for the java command, stacking in reverse order:
# * args from the command line
# * the main class name
# * -classpath
# * -D...appname settings
# * --module-path (only if needed)
# * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables.
# For Cygwin or MSYS, switch paths to Windows format before running java
if "$cygwin" || "$msys" ; then
APP_HOME=$( cygpath --path --mixed "$APP_HOME" )
CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" )
JAVACMD=$( cygpath --unix "$JAVACMD" )
# Now convert the arguments - kludge to limit ourselves to /bin/sh
for arg do
if
case $arg in #(
-*) false ;; # don't mess with options #(
/?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath
[ -e "$t" ] ;; #(
*) false ;;
esac
then
arg=$( cygpath --path --ignore --mixed "$arg" )
fi
# Roll the args list around exactly as many times as the number of
# args, so each arg winds up back in the position where it started, but
# possibly modified.
#
# NB: a `for` loop captures its iteration list before it begins, so
# changing the positional parameters here affects neither the number of
# iterations, nor the values presented in `arg`.
shift # remove old arg
set -- "$@" "$arg" # push replacement arg
done
fi
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
# Collect all arguments for the java command:
# * DEFAULT_JVM_OPTS, JAVA_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments,
# and any embedded shellness will be escaped.
# * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be
# treated as '${Hostname}' itself on the command line.
set -- \
"-Dorg.gradle.appname=$APP_BASE_NAME" \
-classpath "$CLASSPATH" \
org.gradle.wrapper.GradleWrapperMain \
"$@"
# Stop when "xargs" is not available.
if ! command -v xargs >/dev/null 2>&1
then
die "xargs is not available"
fi
# Use "xargs" to parse quoted args.
#
# With -n1 it outputs one arg per line, with the quotes and backslashes removed.
#
# In Bash we could simply go:
#
# readarray ARGS < <( xargs -n1 <<<"$var" ) &&
# set -- "${ARGS[@]}" "$@"
#
# but POSIX shell has neither arrays nor command substitution, so instead we
# post-process each arg (as a line of input to sed) to backslash-escape any
# character that might be a shell metacharacter, then use eval to reverse
# that process (while maintaining the separation between arguments), and wrap
# the whole thing up as a single "set" statement.
#
# This will of course break if any of these variables contains a newline or
# an unmatched quote.
#
eval "set -- $(
printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" |
xargs -n1 |
sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' |
tr '\n' ' '
)" '"$@"'
exec "$JAVACMD" "$@"

94
gradlew.bat vendored Normal file
View file

@ -0,0 +1,94 @@
@rem
@rem Copyright 2015 the original author or authors.
@rem
@rem Licensed under the Apache License, Version 2.0 (the "License");
@rem you may not use this file except in compliance with the License.
@rem You may obtain a copy of the License at
@rem
@rem https://www.apache.org/licenses/LICENSE-2.0
@rem
@rem Unless required by applicable law or agreed to in writing, software
@rem distributed under the License is distributed on an "AS IS" BASIS,
@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
@rem See the License for the specific language governing permissions and
@rem limitations under the License.
@rem
@rem SPDX-License-Identifier: Apache-2.0
@rem
@if "%DEBUG%"=="" @echo off
@rem ##########################################################################
@rem
@rem Gradle startup script for Windows
@rem
@rem ##########################################################################
@rem Set local scope for the variables with windows NT shell
if "%OS%"=="Windows_NT" setlocal
set DIRNAME=%~dp0
if "%DIRNAME%"=="" set DIRNAME=.
@rem This is normally unused
set APP_BASE_NAME=%~n0
set APP_HOME=%DIRNAME%
@rem Resolve any "." and ".." in APP_HOME to make it shorter.
for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m"
@rem Find java.exe
if defined JAVA_HOME goto findJavaFromJavaHome
set JAVA_EXE=java.exe
%JAVA_EXE% -version >NUL 2>&1
if %ERRORLEVEL% equ 0 goto execute
echo. 1>&2
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 1>&2
echo. 1>&2
echo Please set the JAVA_HOME variable in your environment to match the 1>&2
echo location of your Java installation. 1>&2
goto fail
:findJavaFromJavaHome
set JAVA_HOME=%JAVA_HOME:"=%
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
if exist "%JAVA_EXE%" goto execute
echo. 1>&2
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 1>&2
echo. 1>&2
echo Please set the JAVA_HOME variable in your environment to match the 1>&2
echo location of your Java installation. 1>&2
goto fail
:execute
@rem Setup the command line
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
@rem Execute Gradle
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %*
:end
@rem End local scope for the variables with windows NT shell
if %ERRORLEVEL% equ 0 goto mainEnd
:fail
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
rem the _cmd.exe /c_ return code!
set EXIT_CODE=%ERRORLEVEL%
if %EXIT_CODE% equ 0 set EXIT_CODE=1
if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE%
exit /b %EXIT_CODE%
:mainEnd
if "%OS%"=="Windows_NT" endlocal
:omega

20
settings.gradle Normal file
View file

@ -0,0 +1,20 @@
pluginManagement {
repositories {
gradlePluginPortal()
maven {
name = 'Fabric'
url = 'https://maven.fabricmc.net/'
}
}
}
plugins {
id("org.gradle.toolchains.foojay-resolver-convention") version("0.8.0")
}
rootProject.name = 'animalrp'
include (
'animalrp-common',
'animalrp-fabric',
'animalrp-paper'
)