Allow new entries to be created in the data watcher

Addresses #160
This commit is contained in:
Dan Mulloy 2016-03-05 20:36:54 -05:00
parent ec8fa0d1fe
commit a39a37ec51
3 changed files with 110 additions and 25 deletions

View File

@ -217,7 +217,7 @@ class EntityUtilities {
} }
Object trackerEntry = WrappedIntHashMap.fromHandle(trackedEntities).get(entityID); Object trackerEntry = WrappedIntHashMap.fromHandle(trackedEntities).get(entityID);
Class<?> entryClass = MinecraftReflection.getEntityTrackerClass(); Class<?> entryClass = MinecraftReflection.getMinecraftClass("EntityTrackerEntry");
return entryClass.cast(trackerEntry); return entryClass.cast(trackerEntry);
} }

View File

@ -17,6 +17,7 @@
package com.comphenix.protocol.wrappers; package com.comphenix.protocol.wrappers;
import java.lang.reflect.Field; import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier; import java.lang.reflect.Modifier;
import java.lang.reflect.ParameterizedType; import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type; import java.lang.reflect.Type;
@ -27,6 +28,7 @@ import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Map.Entry; import java.util.Map.Entry;
import org.apache.commons.lang.Validate;
import org.bukkit.entity.Entity; import org.bukkit.entity.Entity;
import org.bukkit.inventory.ItemStack; import org.bukkit.inventory.ItemStack;
@ -39,6 +41,7 @@ import com.comphenix.protocol.reflect.accessors.Accessors;
import com.comphenix.protocol.reflect.accessors.ConstructorAccessor; import com.comphenix.protocol.reflect.accessors.ConstructorAccessor;
import com.comphenix.protocol.reflect.accessors.FieldAccessor; import com.comphenix.protocol.reflect.accessors.FieldAccessor;
import com.comphenix.protocol.reflect.accessors.MethodAccessor; import com.comphenix.protocol.reflect.accessors.MethodAccessor;
import com.comphenix.protocol.reflect.fuzzy.FuzzyMethodContract;
import com.comphenix.protocol.utility.MinecraftReflection; import com.comphenix.protocol.utility.MinecraftReflection;
import com.google.common.base.Optional; import com.google.common.base.Optional;
@ -50,7 +53,8 @@ public class WrappedDataWatcher extends AbstractWrapper implements Iterable<Wrap
private static final Class<?> HANDLE_TYPE = MinecraftReflection.getDataWatcherClass(); private static final Class<?> HANDLE_TYPE = MinecraftReflection.getDataWatcherClass();
private static MethodAccessor GETTER = null; private static MethodAccessor GETTER = null;
private static MethodAccessor SETTER = null; public static MethodAccessor SETTER = null;
public static MethodAccessor REGISTER = null;
private static FieldAccessor ENTITY_FIELD = null; private static FieldAccessor ENTITY_FIELD = null;
private static FieldAccessor MAP_FIELD = null; private static FieldAccessor MAP_FIELD = null;
@ -297,15 +301,37 @@ public class WrappedDataWatcher extends AbstractWrapper implements Iterable<Wrap
/** /**
* Sets the DataWatcher Item associated with a given watcher object to a new value. * Sets the DataWatcher Item associated with a given watcher object to a new value.
* @param watcherObject Associated watcher object * @param object Associated watcher object
* @param value New value * @param value New value
*/ */
public void setObject(WrappedDataWatcherObject watcherObject, Object value) { public void setObject(WrappedDataWatcherObject object, Object value) {
if (SETTER == null) { Validate.notNull(object, "Watcher object cannot be null!");
SETTER = Accessors.getMethodAccessor(handleType, "set", watcherObject.getHandleType(), Object.class);
if (SETTER == null || REGISTER == null) {
FuzzyReflection fuzzy = FuzzyReflection.fromClass(handleType, true);
List<Method> methods = fuzzy.getMethodList(FuzzyMethodContract.newBuilder()
.banModifier(Modifier.STATIC)
.requireModifier(Modifier.PUBLIC)
.parameterExactArray(object.getHandleType(), Object.class)
.build());
for (Method method : methods) {
if (method.getName().equals("set")) {
SETTER = Accessors.getMethodAccessor(method);
} else if (method.getName().equals("register")) {
REGISTER = Accessors.getMethodAccessor(method);
} else {
System.out.println(method);
}
}
} }
SETTER.invoke(handle, watcherObject.getHandle(), WrappedWatchableObject.getUnwrapped(value)); if (hasIndex(object.getIndex())) {
SETTER.invoke(handle, object.getHandle(), WrappedWatchableObject.getUnwrapped(value));
} else {
Serializer serializer = object.getSerializer();
Validate.notNull(serializer, "You must specify a serializer to register an object!");
REGISTER.invoke(handle, object.getHandle(), WrappedWatchableObject.getUnwrapped(value));
}
} }
// TODO Add support for setting the dirty state // TODO Add support for setting the dirty state
@ -318,8 +344,8 @@ public class WrappedDataWatcher extends AbstractWrapper implements Iterable<Wrap
public WrappedDataWatcher deepClone() { public WrappedDataWatcher deepClone() {
WrappedDataWatcher clone = new WrappedDataWatcher(getEntity()); WrappedDataWatcher clone = new WrappedDataWatcher(getEntity());
for (Entry<Integer, WrappedWatchableObject> entry : asMap().entrySet()) { for (WrappedWatchableObject wrapper : this) {
clone.setObject(entry.getKey(), entry.getValue()); clone.setObject(wrapper.getWatcherObject(), wrapper);
} }
return clone; return clone;
@ -452,6 +478,7 @@ public class WrappedDataWatcher extends AbstractWrapper implements Iterable<Wrap
public static class WrappedDataWatcherObject extends AbstractWrapper { public static class WrappedDataWatcherObject extends AbstractWrapper {
private static final Class<?> HANDLE_TYPE = MinecraftReflection.getDataWatcherObjectClass(); private static final Class<?> HANDLE_TYPE = MinecraftReflection.getDataWatcherObjectClass();
private static ConstructorAccessor constructor = null; private static ConstructorAccessor constructor = null;
private static MethodAccessor getSerializer = null;
private final StructureModifier<Object> modifier; private final StructureModifier<Object> modifier;
@ -492,10 +519,21 @@ public class WrappedDataWatcher extends AbstractWrapper implements Iterable<Wrap
return (int) modifier.read(0); return (int) modifier.read(0);
} }
// TODO this public Serializer getSerializer() {
/* public Serializer getSerializer() { if (getSerializer == null) {
return null; getSerializer = Accessors.getMethodAccessor(FuzzyReflection.fromClass(HANDLE_TYPE, true).getMethodByParameters(
} */ "getSerializer", MinecraftReflection.getDataWatcherSerializerClass(), new Class[0]));
}
Object serializer = getSerializer.invoke(handle);
Serializer wrapper = Registry.fromHandle(serializer);
if (wrapper != null) {
return wrapper;
} else {
return new Serializer(null, serializer, false);
}
}
} }
/** /**
@ -538,6 +576,11 @@ public class WrappedDataWatcher extends AbstractWrapper implements Iterable<Wrap
public boolean isOptional() { public boolean isOptional() {
return optional; return optional;
} }
@Override
public String toString() {
return "Serializer[type=" + type + ", handle=" + handle + ", optional=" + optional + "]";
}
} }
/** /**
@ -556,15 +599,29 @@ public class WrappedDataWatcher extends AbstractWrapper implements Iterable<Wrap
* @return The serializer, or null if none exists * @return The serializer, or null if none exists
*/ */
public static Serializer get(Class<?> clazz) { public static Serializer get(Class<?> clazz) {
if (! INITIALIZED) { initialize();
initialize();
INITIALIZED = true;
}
return REGISTRY.get(clazz); return REGISTRY.get(clazz);
} }
public static Serializer fromHandle(Object handle) {
initialize();
for (Serializer serializer : REGISTRY.values()) {
if (serializer.getHandle().equals(handle)) {
return serializer;
}
}
return null;
}
private static void initialize() { private static void initialize() {
if (!INITIALIZED) {
INITIALIZED = true;
} else {
return;
}
List<Field> candidates = FuzzyReflection.fromClass(MinecraftReflection.getMinecraftClass("DataWatcherRegistry"), true) List<Field> candidates = FuzzyReflection.fromClass(MinecraftReflection.getMinecraftClass("DataWatcherRegistry"), true)
.getFieldListByType(MinecraftReflection.getDataWatcherSerializerClass()); .getFieldListByType(MinecraftReflection.getDataWatcherSerializerClass());
for (Field candidate : candidates) { for (Field candidate : candidates) {

View File

@ -3,6 +3,7 @@
*/ */
package com.comphenix.protocol.wrappers; package com.comphenix.protocol.wrappers;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertTrue; import static org.junit.Assert.assertTrue;
import net.minecraft.server.v1_9_R1.DataWatcher; import net.minecraft.server.v1_9_R1.DataWatcher;
import net.minecraft.server.v1_9_R1.Entity; import net.minecraft.server.v1_9_R1.Entity;
@ -28,19 +29,46 @@ public class WrappedDataWatcherTest {
} }
@Test @Test
public void test() { public void testBytes() {
Entity entity = new EntityLightning(null, 0, 0, 0, true); WrappedDataWatcher wrapper = create();
DataWatcher handle = entity.getDataWatcher(); WrappedWatchableObject watchable = wrapper.getWatchableObject(0);
WrappedDataWatcherObject object = watchable.getWatcherObject();
WrappedDataWatcher wrapper = new WrappedDataWatcher(handle); // Make sure the serializers work
assertEquals(object.getSerializer(), Registry.get(Byte.class));
// Make sure we can set existing objects
wrapper.setObject(0, (byte) 1); wrapper.setObject(0, (byte) 1);
assertTrue(wrapper.getByte(0) == 1); assertTrue(wrapper.getByte(0) == 1);
}
@Test
public void testStrings() {
WrappedDataWatcher wrapper = create();
// Make sure we can create watcher objects
Serializer serializer = Registry.get(String.class); Serializer serializer = Registry.get(String.class);
WrappedDataWatcherObject watcherObject = new WrappedDataWatcherObject(3, serializer); WrappedDataWatcherObject object = new WrappedDataWatcherObject(3, serializer);
wrapper.setObject(watcherObject, "Hiya"); wrapper.setObject(object, "Test");
assertTrue(wrapper.getString(3).equals("Hiya")); assertEquals(wrapper.getString(3), "Test");
}
@Test
public void testFloats() {
WrappedDataWatcher wrapper = create();
// Make sure we can add new entries
Serializer serializer = Registry.get(Float.class);
WrappedDataWatcherObject object = new WrappedDataWatcherObject(10, serializer);
wrapper.setObject(object, 1.0F);
assertTrue(wrapper.hasIndex(10));
}
private WrappedDataWatcher create() {
Entity entity = new EntityLightning(null, 0, 0, 0, true);
DataWatcher handle = entity.getDataWatcher();
return new WrappedDataWatcher(handle);
} }
} }