2022-06-20 09:55:51 -05:00
|
|
|
package fathertoast.specialmobs.common.util;
|
|
|
|
|
|
|
|
import fathertoast.specialmobs.common.bestiary.BestiaryInfo;
|
|
|
|
import fathertoast.specialmobs.common.bestiary.MobFamily;
|
|
|
|
import fathertoast.specialmobs.common.bestiary.SpecialMob;
|
|
|
|
import fathertoast.specialmobs.datagen.loot.LootTableBuilder;
|
2022-06-27 10:38:10 -05:00
|
|
|
import mcp.MethodsReturnNonnullByDefault;
|
2022-06-20 09:55:51 -05:00
|
|
|
import net.minecraft.entity.EntityType;
|
|
|
|
import net.minecraft.entity.LivingEntity;
|
|
|
|
import net.minecraft.entity.ai.attributes.AttributeModifierMap;
|
2022-06-24 05:19:49 +02:00
|
|
|
import net.minecraft.item.Item;
|
2022-06-20 09:55:51 -05:00
|
|
|
|
2022-06-27 10:38:10 -05:00
|
|
|
import javax.annotation.Nullable;
|
2022-06-20 09:55:51 -05:00
|
|
|
import javax.annotation.ParametersAreNonnullByDefault;
|
|
|
|
import java.lang.annotation.Annotation;
|
2022-06-21 22:40:16 -05:00
|
|
|
import java.lang.reflect.*;
|
2022-06-20 09:55:51 -05:00
|
|
|
|
2022-06-23 09:51:14 -05:00
|
|
|
/**
|
|
|
|
* Provides helper methods to handle annotation processing through reflection.
|
|
|
|
*/
|
2022-06-27 10:38:10 -05:00
|
|
|
@SuppressWarnings( "SameParameterValue" )
|
2022-06-20 09:55:51 -05:00
|
|
|
@ParametersAreNonnullByDefault
|
2022-06-27 10:38:10 -05:00
|
|
|
@MethodsReturnNonnullByDefault
|
2022-06-20 09:55:51 -05:00
|
|
|
public final class AnnotationHelper {
|
|
|
|
|
|
|
|
//--------------- PRETTY HELPER METHODS ----------------
|
|
|
|
|
2022-06-21 22:40:16 -05:00
|
|
|
/** Creates an entity factory from a special mob species. Throws an exception if anything goes wrong. */
|
2022-06-27 10:38:10 -05:00
|
|
|
public static void injectSpeciesReference( MobFamily.Species<?> species ) {
|
2022-06-23 09:51:14 -05:00
|
|
|
try {
|
2022-06-27 10:38:10 -05:00
|
|
|
final Field field = getField( species.entityClass, SpecialMob.SpeciesReference.class );
|
|
|
|
field.set( null, species );
|
2022-06-23 09:51:14 -05:00
|
|
|
}
|
2022-06-27 10:38:10 -05:00
|
|
|
catch( IllegalAccessException | NoSuchFieldException ex ) {
|
|
|
|
throw new RuntimeException( "Entity class for " + species.name + " has invalid species reference holder", ex );
|
2022-06-23 09:51:14 -05:00
|
|
|
}
|
2022-06-21 22:40:16 -05:00
|
|
|
}
|
|
|
|
|
2022-06-20 09:55:51 -05:00
|
|
|
/** Gets bestiary info from a special mob species. Throws an exception if anything goes wrong. */
|
2022-06-23 09:51:14 -05:00
|
|
|
public static <T extends LivingEntity> BestiaryInfo getBestiaryInfo( MobFamily.Species<T> species, EntityType.Builder<T> entityType ) {
|
|
|
|
try {
|
|
|
|
return (BestiaryInfo) getMethod( species.entityClass, SpecialMob.BestiaryInfoSupplier.class ).invoke( null, entityType );
|
|
|
|
}
|
|
|
|
catch( IllegalAccessException | NoSuchMethodException | InvocationTargetException ex ) {
|
|
|
|
throw new RuntimeException( "Entity class for " + species.name + " has invalid bestiary info method", ex );
|
|
|
|
}
|
2022-06-20 09:55:51 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/** Creates an attribute modifier map from a special mob species. Throws an exception if anything goes wrong. */
|
2022-06-23 09:51:14 -05:00
|
|
|
public static AttributeModifierMap createAttributes( MobFamily.Species<?> species ) {
|
|
|
|
try {
|
2022-06-27 10:38:10 -05:00
|
|
|
return ((AttributeModifierMap.MutableAttribute) getMethodOrSuper( species.entityClass, SpecialMob.AttributeCreator.class )
|
2022-06-23 09:51:14 -05:00
|
|
|
.invoke( null )).build();
|
|
|
|
}
|
|
|
|
catch( NoSuchMethodException | InvocationTargetException | IllegalAccessException ex ) {
|
|
|
|
throw new RuntimeException( "Entity class for " + species.name + " has invalid attribute creation method", ex );
|
|
|
|
}
|
2022-06-20 12:37:58 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/** Gets the translations from a special mob species. Throws an exception if anything goes wrong. */
|
2022-06-23 09:51:14 -05:00
|
|
|
public static String[] getTranslations( MobFamily.Species<?> species ) {
|
|
|
|
try {
|
|
|
|
return (String[]) getMethod( species.entityClass, SpecialMob.LanguageProvider.class )
|
|
|
|
.invoke( null, species.entityType.get().getDescriptionId() );
|
|
|
|
}
|
|
|
|
catch( NoSuchMethodException | InvocationTargetException | IllegalAccessException ex ) {
|
|
|
|
throw new RuntimeException( "Entity class for " + species.name + " has invalid language provider method", ex );
|
|
|
|
}
|
2022-06-20 09:55:51 -05:00
|
|
|
}
|
2022-06-24 13:54:20 -05:00
|
|
|
|
2022-06-24 05:19:49 +02:00
|
|
|
/** Gets the translations from a mod item. Throws an exception if anything goes wrong. */
|
2022-06-24 13:54:20 -05:00
|
|
|
public static String[] getTranslations( Item item ) {
|
2022-06-24 05:19:49 +02:00
|
|
|
try {
|
|
|
|
return (String[]) getMethod( item.getClass(), SpecialMob.LanguageProvider.class )
|
|
|
|
.invoke( null, item.getDescriptionId() );
|
|
|
|
}
|
|
|
|
catch( NoSuchMethodException | InvocationTargetException | IllegalAccessException ex ) {
|
|
|
|
throw new RuntimeException( "Item class for " + item.getRegistryName() + " has invalid language provider method", ex );
|
|
|
|
}
|
|
|
|
}
|
2022-06-24 13:54:20 -05:00
|
|
|
|
2022-06-20 09:55:51 -05:00
|
|
|
/** Builds a loot table from a special mob species. Throws an exception if anything goes wrong. */
|
2022-06-23 09:51:14 -05:00
|
|
|
public static LootTableBuilder buildLootTable( MobFamily.Species<?> species ) {
|
|
|
|
try {
|
|
|
|
final LootTableBuilder builder = new LootTableBuilder();
|
|
|
|
getMethod( species.entityClass, SpecialMob.LootTableProvider.class ).invoke( null, builder );
|
|
|
|
return builder;
|
|
|
|
}
|
|
|
|
catch( NoSuchMethodException | InvocationTargetException | IllegalAccessException ex ) {
|
|
|
|
throw new RuntimeException( "Entity class for " + species.name + " has invalid loot table builder method", ex );
|
|
|
|
}
|
2022-06-20 09:55:51 -05:00
|
|
|
}
|
|
|
|
|
2022-06-29 09:14:28 -05:00
|
|
|
/** Creates an entity factory from a special mob species. Throws an exception if anything goes wrong. */
|
|
|
|
public static <T extends LivingEntity> EntityType.IFactory<T> getEntityFactory( MobFamily.Species<T> species ) {
|
|
|
|
try {
|
|
|
|
//noinspection unchecked
|
|
|
|
return (EntityType.IFactory<T>) getMethod( species.entityClass, SpecialMob.Factory.class ).invoke( null );
|
|
|
|
}
|
|
|
|
catch( NoSuchMethodException | InvocationTargetException | IllegalAccessException ex ) {
|
|
|
|
throw new RuntimeException( "Entity class for " + species.name + " has invalid factory provider method", ex );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-20 09:55:51 -05:00
|
|
|
|
|
|
|
//--------------- RAW ANNOTATION METHODS ----------------
|
|
|
|
|
2022-06-21 22:40:16 -05:00
|
|
|
/**
|
|
|
|
* @return Pulls a static field with a specific annotation from a class.
|
2022-06-27 10:38:10 -05:00
|
|
|
* @throws NoSuchFieldException if the field does not exist in the class.
|
2022-06-21 22:40:16 -05:00
|
|
|
*/
|
|
|
|
private static Field getField( Class<?> type, Class<? extends Annotation> annotation ) throws NoSuchFieldException {
|
|
|
|
final Field field = getFieldOptional( type, annotation );
|
|
|
|
if( field == null ) {
|
2022-06-27 10:38:10 -05:00
|
|
|
throw new NoSuchFieldException( String.format( "Could not find required static @%s annotated field in %s",
|
2022-06-21 22:40:16 -05:00
|
|
|
annotation.getSimpleName(), type.getName() ) );
|
|
|
|
}
|
|
|
|
return field;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return Pulls a static field with a specific annotation from a class, or null if the field does not exist.
|
|
|
|
*/
|
2022-06-27 10:38:10 -05:00
|
|
|
@Nullable
|
2022-06-21 22:40:16 -05:00
|
|
|
private static Field getFieldOptional( Class<?> type, Class<? extends Annotation> annotation ) {
|
2022-06-27 10:38:10 -05:00
|
|
|
for( Field field : type.getDeclaredFields() ) {
|
2022-06-21 22:40:16 -05:00
|
|
|
if( Modifier.isStatic( field.getModifiers() ) && field.isAnnotationPresent( annotation ) )
|
|
|
|
return field;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2022-06-20 09:55:51 -05:00
|
|
|
/**
|
|
|
|
* @return Pulls a static method with a specific annotation from a class.
|
2022-06-27 10:38:10 -05:00
|
|
|
* @throws NoSuchMethodException if the method does not exist in the class.
|
2022-06-20 09:55:51 -05:00
|
|
|
*/
|
|
|
|
private static Method getMethod( Class<?> type, Class<? extends Annotation> annotation ) throws NoSuchMethodException {
|
2022-06-27 10:38:10 -05:00
|
|
|
for( Method method : type.getDeclaredMethods() ) {
|
|
|
|
if( Modifier.isStatic( method.getModifiers() ) && method.isAnnotationPresent( annotation ) )
|
|
|
|
return method;
|
2022-06-20 09:55:51 -05:00
|
|
|
}
|
2022-06-27 10:38:10 -05:00
|
|
|
throw new NoSuchMethodException( String.format( "Could not find required static @%s annotated method in %s",
|
|
|
|
annotation.getSimpleName(), type.getName() ) );
|
2022-06-20 09:55:51 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2022-06-27 10:38:10 -05:00
|
|
|
* @return Pulls a static method with a specific annotation from a class, or its super class(es) if none is defined in the class.
|
|
|
|
* @throws NoSuchMethodException if the method does not exist in the class or any of its parents.
|
2022-06-20 09:55:51 -05:00
|
|
|
*/
|
2022-06-27 10:38:10 -05:00
|
|
|
private static Method getMethodOrSuper( Class<? extends LivingEntity> type, Class<? extends Annotation> annotation ) throws NoSuchMethodException {
|
|
|
|
Class<?> currentType = type;
|
|
|
|
while( currentType != LivingEntity.class ) {
|
|
|
|
for( Method method : currentType.getDeclaredMethods() ) {
|
|
|
|
if( Modifier.isStatic( method.getModifiers() ) && method.isAnnotationPresent( annotation ) )
|
|
|
|
return method;
|
|
|
|
}
|
|
|
|
currentType = currentType.getSuperclass();
|
|
|
|
}
|
|
|
|
throw new NoSuchMethodException( String.format( "Could not find 'overridable' static @%s annotated method in %s or its parents",
|
|
|
|
annotation.getSimpleName(), type.getName() ) );
|
2022-06-20 09:55:51 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return Pulls a constructor with a specific annotation from a class.
|
|
|
|
* @throws NoSuchMethodException if the constructor does not exist.
|
|
|
|
*/
|
|
|
|
private static <T> Constructor<T> getConstructor( Class<T> type, Class<? extends Annotation> annotation ) throws NoSuchMethodException {
|
2022-06-28 13:04:46 -05:00
|
|
|
for( Constructor<?> constructor : type.getDeclaredConstructors() ) {
|
2022-06-20 09:55:51 -05:00
|
|
|
if( constructor.isAnnotationPresent( annotation ) )
|
|
|
|
//noinspection unchecked
|
|
|
|
return (Constructor<T>) constructor;
|
|
|
|
}
|
|
|
|
throw new NoSuchMethodException( String.format( "Could not find @%s annotated constructor in %s",
|
|
|
|
annotation.getSimpleName(), type.getName() ) );
|
|
|
|
}
|
|
|
|
}
|