/
ReflectionUtil.java
121 lines (108 loc) · 5.55 KB
/
ReflectionUtil.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
package knightminer.inspirations.library.util;
import knightminer.inspirations.library.InspirationsRegistry;
import net.minecraft.item.crafting.Ingredient;
import net.minecraftforge.fml.common.ObfuscationReflectionHelper;
import net.minecraftforge.fml.common.ObfuscationReflectionHelper.UnableToFindFieldException;
import net.minecraftforge.fml.common.ObfuscationReflectionHelper.UnableToFindMethodException;
import net.minecraftforge.registries.ForgeRegistryEntry;
import net.minecraftforge.registries.IRegistryDelegate;
import javax.annotation.Nullable;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Map;
public final class ReflectionUtil {
private ReflectionUtil() {}
private static final Map<String,Field> FIELDS = new HashMap<>();
private static final Map<String,Method> METHODS = new HashMap<>();
private static final Map<String,Class<?>> CLASS = new HashMap<>();
/* PotionBrewing.MixPredicate */
@Nullable
public static <T extends ForgeRegistryEntry<T>> T getMixPredicateInput(Object mixPredicate) {
IRegistryDelegate<T> effect = getPrivateValue(getClass("net.minecraft.potion.PotionBrewing$MixPredicate"), mixPredicate, "field_185198_a");
if (effect != null) {
return effect.get();
}
return null;
}
@Nullable
public static Ingredient getMixPredicateReagent(Object mixPredicate) {
return getPrivateValue(getClass("net.minecraft.potion.PotionBrewing$MixPredicate"), mixPredicate, "field_185199_b");
}
@Nullable
public static <T extends ForgeRegistryEntry<T>> T getMixPredicateOutput(Object mixPredicate) {
IRegistryDelegate<T> effect = getPrivateValue(getClass("net.minecraft.potion.PotionBrewing$MixPredicate"), mixPredicate, "field_185200_c");
if (effect != null) {
return effect.get();
}
return null;
}
/* Base Methods */
/**
* Searches the instance for the occurrence of a method either named by its SRG name (obfuscated) or MCP name (development),
* caches the reference for further use, invokes the method and returns the value returned by the invocation (or <tt>null</tt>, if the method is <tt>void</tt>).<br>
* If it can't find the method or something went wrong during invocation, it will be logged and <tt>null</tt> is returned.
* @param instance The instance to be accessed. Cannot be <tt>null</tt>!
* @param name The name used in a development environment
* @param <T> The type of the return value. Must be the same as or a superclass of the return type of the method! {@link Object}, if the method is <tt>void</tt>
* @return The return value of the method or <tt>null</tt>, if it fails or the method is <tt>void</tt>
*/
@Nullable
@SuppressWarnings("unchecked")
private static <T> T invokeMethod(final Class<?> classToSearch, final Object instance, final String name, final Class<?>[] paramTypes, final Object... params) {
try {
Method m = METHODS.computeIfAbsent(name, key -> ObfuscationReflectionHelper.findMethod(classToSearch, name, paramTypes));
return (T)m.invoke(instance, params);
} catch (IllegalAccessException | IllegalArgumentException | InvocationTargetException | ClassCastException | UnableToFindMethodException e) {
InspirationsRegistry.log.error(e);
METHODS.putIfAbsent(name, null); // set cache of method to null if it errors trying to find the method in the first place
return null;
}
}
/**
* Looks up a class by its name, caches it for further use and returns it.<br>
* If it can't find the class, it will be logged and <tt>null</tt> is returned.
* @param className The class name to be searched for
* @return The class found or <tt>null</tt>, if the class is unavailable
*/
@Nullable
@SuppressWarnings("unchecked")
private static Class<? super Object> getClass(String className) {
return (Class<? super Object>)CLASS.computeIfAbsent(className, key -> {
try {
return Class.forName(key, false, InspirationsRegistry.class.getClassLoader());
} catch (ClassNotFoundException e) {
InspirationsRegistry.log.error(e);
return null;
}
});
}
/**
* Searches the class for the occurrence of a field either named by its SRG name (obfuscated) or MCP name (development),
* caches the reference for further use and returns the value of the field from the instance (or statically, if the instance is <tt>null</tt>.<br>
* If it can't find the field or something went wrong with getting the value, it will be logged and <tt>null</tt> is returned.
* @param instance The instance to be accessed, <tt>null</tt> if the field is static
* @param name SRG name of the field to find
* @param <T> The type of the return value. Must be the same as or a superclass of the type of the field!
* @return The value of the field or <tt>null</tt>, if it fails
*/
@Nullable
@SuppressWarnings("unchecked")
private static <C, T> T getPrivateValue(@Nullable Class<? super C> clazz, final Object instance, final String name) {
if (clazz == null) {
return null;
}
if (!clazz.isInstance(instance)) {
return null;
}
try {
Field f = FIELDS.computeIfAbsent(name, key -> ObfuscationReflectionHelper.findField(clazz, name));
return (T)f.get(instance);
} catch (IllegalAccessException | UnableToFindFieldException | ClassCastException e) {
InspirationsRegistry.log.error(e);
FIELDS.putIfAbsent(name, null); // set cache of field to null if it errors trying to find the field in the first place
return null;
}
}
}