Add a quick and dirty wrapper for statistics maps.

This commit is contained in:
Kristian S. Stangeland 2014-01-02 07:23:27 +01:00
parent 5c92a46cfa
commit c06aeb2836
4 changed files with 116 additions and 1 deletions

View File

@ -30,6 +30,7 @@ import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Collection;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentMap;
@ -76,6 +77,7 @@ import com.comphenix.protocol.wrappers.WrappedChatComponent;
import com.comphenix.protocol.wrappers.WrappedDataWatcher;
import com.comphenix.protocol.wrappers.WrappedGameProfile;
import com.comphenix.protocol.wrappers.WrappedServerPing;
import com.comphenix.protocol.wrappers.WrappedStatistic;
import com.comphenix.protocol.wrappers.WrappedWatchableObject;
import com.comphenix.protocol.wrappers.EnumWrappers.ChatVisibility;
import com.comphenix.protocol.wrappers.EnumWrappers.ClientCommand;
@ -357,6 +359,21 @@ public class PacketContainer implements Serializable {
BukkitConverters.getIgnoreNull(new ItemStackArrayConverter()));
}
/**
* Retrieve a read/write structure for maps of statistics.
* <p>
* Note that you must write back the changed map to persist it.
* @return A modifier for maps of statistics.
*/
public StructureModifier<Map<WrappedStatistic, Integer>> getStatisticMaps() {
return structureModifier.withType(Map.class,
BukkitConverters.<WrappedStatistic, Integer>getMapConverter(
MinecraftReflection.getStatisticClass(),
BukkitConverters.getWrappedStatisticConverter()
)
);
}
/**
* Retrieves a read/write structure for the world type enum.
* <p>

View File

@ -832,6 +832,24 @@ public class MinecraftReflection {
}
}
/**
* Retrieve the NMS statistics class.
* @return The statistics class.
*/
public static Class<?> getStatisticClass() {
// TODO: Implement fallback
return getMinecraftClass("Statistic");
}
/**
* Retrieve the NMS statistic list class.
* @return The statistic list class.
*/
public static Class<?> getStatisticListClass() {
// TODO: Implement fallback
return getMinecraftClass("StatisticList");
}
/**
* Fallback method that can determine the MinecraftServer and the ServerConfigurationManager.
*/

View File

@ -28,7 +28,7 @@ public abstract class AbstractWrapper {
if (handle == null)
throw new IllegalArgumentException("handle cannot be NULL.");
if (!handleType.isAssignableFrom(handle.getClass()))
throw new IllegalArgumentException("handle (" + handle + ") is not a " + handleType);
throw new IllegalArgumentException("handle (" + handle + ") is not a " + handleType + ", but " + handle.getClass());
this.handle = handle;
}

View File

@ -57,6 +57,7 @@ import com.google.common.base.Objects;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.Lists;
import com.google.common.collect.Maps;
/**
* Contains several useful equivalent converters for normal Bukkit types.
@ -191,6 +192,60 @@ public class BukkitConverters {
}
}
/**
* Retrieve an equivalent converter for a map of generic keys and primitive values.
* @param genericItemType - the generic item type.
* @param itemConverter - an equivalent converter for the generic type.
* @return An equivalent converter.
*/
public static <T, U> EquivalentConverter<Map<T, U>> getMapConverter(
final Class<?> genericKeyType, final EquivalentConverter<T> keyConverter) {
// Convert to and from the wrapper
return new IgnoreNullConverter<Map<T, U>>() {
@SuppressWarnings("unchecked")
@Override
protected Map<T, U> getSpecificValue(Object generic) {
if (generic instanceof Map) {
Map<T, U> result = Maps.newHashMap();
// Copy everything to a new list
for (Entry<Object, Object> entry : ((Map<Object, Object>) generic).entrySet()) {
result.put(
keyConverter.getSpecific(entry.getKey()),
(U) entry.getValue()
);
}
return result;
}
// Not valid
return null;
}
@SuppressWarnings("unchecked")
@Override
protected Object getGenericValue(Class<?> genericType, Map<T, U> specific) {
Map<Object, Object> newContainer = (Map<Object, Object>) DefaultInstances.DEFAULT.getDefault(genericType);
// Convert each object
for (Entry<T, U> entry : specific.entrySet()) {
newContainer.put(
keyConverter.getGeneric(genericKeyType, entry.getKey()),
entry.getValue()
);
}
return newContainer;
}
@SuppressWarnings("unchecked")
@Override
public Class<Map<T, U>> getSpecificType() {
Class<?> dummy = Map.class;
return (Class<Map<T, U>>) dummy;
}
};
}
/**
* Retrieve an equivalent converter for a list of generic items.
* @param genericItemType - the generic item type.
@ -594,6 +649,29 @@ public class BukkitConverters {
};
}
/**
* Retrieve the converter for a statistic.
* @return Statistic converter.
*/
public static EquivalentConverter<WrappedStatistic> getWrappedStatisticConverter() {
return new IgnoreNullConverter<WrappedStatistic>() {
@Override
protected Object getGenericValue(Class<?> genericType, WrappedStatistic specific) {
return specific.getHandle();
}
@Override
protected WrappedStatistic getSpecificValue(Object generic) {
return WrappedStatistic.fromHandle(generic);
}
@Override
public Class<WrappedStatistic> getSpecificType() {
return WrappedStatistic.class;
}
};
}
/**
* Retrieve a converter for block instances.
* @return A converter for block instances.
@ -756,6 +834,7 @@ public class BukkitConverters {
builder.put(WrappedGameProfile.class, (EquivalentConverter) getWrappedGameProfileConverter());
builder.put(WrappedChatComponent.class, (EquivalentConverter) getWrappedChatComponentConverter());
builder.put(WrappedServerPing.class, (EquivalentConverter) getWrappedServerPingConverter());
builder.put(WrappedStatistic.class, (EquivalentConverter) getWrappedStatisticConverter());
for (Entry<Class<?>, EquivalentConverter<?>> entry : EnumWrappers.getFromWrapperMap().entrySet()) {
builder.put((Class) entry.getKey(), (EquivalentConverter) entry.getValue());
@ -798,6 +877,7 @@ public class BukkitConverters {
builder.put(MinecraftReflection.getGameProfileClass(), (EquivalentConverter) getWrappedGameProfileConverter());
builder.put(MinecraftReflection.getIChatBaseComponentClass(), (EquivalentConverter) getWrappedChatComponentConverter());
builder.put(MinecraftReflection.getServerPingClass(), (EquivalentConverter) getWrappedServerPingConverter());
builder.put(MinecraftReflection.getStatisticClass(), (EquivalentConverter) getWrappedStatisticConverter());
for (Entry<Class<?>, EquivalentConverter<?>> entry : EnumWrappers.getFromNativeMap().entrySet()) {
builder.put((Class) entry.getKey(), (EquivalentConverter) entry.getValue());