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;
|
|
|
|
import net.minecraft.entity.EntityType;
|
|
|
|
import net.minecraft.entity.LivingEntity;
|
|
|
|
import net.minecraft.entity.ai.attributes.AttributeModifierMap;
|
|
|
|
|
|
|
|
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-20 09:55:51 -05:00
|
|
|
@ParametersAreNonnullByDefault
|
|
|
|
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-23 09:51:14 -05:00
|
|
|
public static void injectEntityTypeHolder( MobFamily.Species<?> species ) {
|
|
|
|
try {
|
|
|
|
final Field field = getFieldOptional( species.entityClass, SpecialMob.TypeHolder.class );
|
|
|
|
if( field != null ) field.set( null, species.entityType );
|
|
|
|
}
|
|
|
|
catch( IllegalAccessException ex ) {
|
|
|
|
throw new RuntimeException( "Entity class for " + species.name + " has invalid entity type holder", ex );
|
|
|
|
}
|
2022-06-21 22:40:16 -05:00
|
|
|
}
|
|
|
|
|
2022-06-20 09:55:51 -05:00
|
|
|
/** Creates an entity factory 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> EntityType.IFactory<T> getEntityFactory( MobFamily.Species<T> species ) {
|
|
|
|
try {
|
|
|
|
final Constructor<T> constructor = getConstructor( species.entityClass, SpecialMob.Constructor.class );
|
|
|
|
return ( entityType, world ) -> {
|
|
|
|
try {
|
|
|
|
return constructor.newInstance( entityType, world );
|
|
|
|
}
|
|
|
|
catch( InstantiationException | IllegalAccessException | InvocationTargetException ex ) {
|
|
|
|
throw new RuntimeException( "Class for " + constructor.getDeclaringClass().getName() + " has invalid constructor", ex );
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
catch( NoSuchMethodException ex ) {
|
|
|
|
throw new RuntimeException( "Entity class for " + species.name + " has no valid constructors", ex );
|
|
|
|
}
|
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 {
|
|
|
|
return ((AttributeModifierMap.MutableAttribute) getMethod( species.entityClass, SpecialMob.AttributeCreator.class )
|
|
|
|
.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
|
|
|
}
|
|
|
|
|
|
|
|
/** 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
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//--------------- RAW ANNOTATION METHODS ----------------
|
|
|
|
|
2022-06-21 22:40:16 -05:00
|
|
|
/**
|
|
|
|
* @return Pulls a static field with a specific annotation from a class.
|
|
|
|
* @throws NoSuchFieldException if the field does not exist.
|
|
|
|
*/
|
|
|
|
private static Field getField( Class<?> type, Class<? extends Annotation> annotation ) throws NoSuchFieldException {
|
|
|
|
final Field field = getFieldOptional( type, annotation );
|
|
|
|
if( field == null ) {
|
|
|
|
throw new NoSuchFieldException( String.format( "Could not find static @%s annotated field in %s",
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
private static Field getFieldOptional( Class<?> type, Class<? extends Annotation> annotation ) {
|
|
|
|
for( Field field : type.getFields() ) {
|
|
|
|
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.
|
|
|
|
* @throws NoSuchMethodException if the method does not exist.
|
|
|
|
*/
|
|
|
|
private static Method getMethod( Class<?> type, Class<? extends Annotation> annotation ) throws NoSuchMethodException {
|
|
|
|
final Method method = getMethodOptional( type, annotation );
|
|
|
|
if( method == null ) {
|
|
|
|
throw new NoSuchMethodException( String.format( "Could not find static @%s annotated method in %s",
|
|
|
|
annotation.getSimpleName(), type.getName() ) );
|
|
|
|
}
|
|
|
|
return method;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return Pulls a static method with a specific annotation from a class, or null if the method does not exist.
|
|
|
|
*/
|
|
|
|
private static Method getMethodOptional( Class<?> type, Class<? extends Annotation> annotation ) {
|
|
|
|
for( Method method : type.getMethods() ) {
|
|
|
|
if( Modifier.isStatic( method.getModifiers() ) && method.isAnnotationPresent( annotation ) )
|
|
|
|
return method;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @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 {
|
|
|
|
for( Constructor<?> constructor : type.getConstructors() ) {
|
|
|
|
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() ) );
|
|
|
|
}
|
|
|
|
}
|