blob: 3ea303839011c3358b29f1560fd6529b5ee1a2f2 [file] [log] [blame] [raw]
package protocolsupport.api;
import java.text.MessageFormat;
import java.util.Arrays;
import java.util.Collections;
import java.util.EnumMap;
import java.util.ArrayList;
import java.util.List;
import java.util.Comparator;
import java.util.Iterator;
import gnu.trove.map.hash.TIntObjectHashMap;
public enum ProtocolVersion {
UNKNOWN(-1),
MINECRAFT_LEGACY(-1),
MINECRAFT_1_4_6(false, 51, "1.4.7"),
MINECRAFT_1_5_0(false, 60, "1.5.1"),
MINECRAFT_2_0_BLUE(false, 90, "2.0 blue"),
MINECRAFT_2_0_RED(false, 91, "2.0 red"),
MINECRAFT_2_0_PURPLE(false, 92, "2.0 purple"),
MINECRAFT_1_5_2(false, 61, "1.5.2"),
MINECRAFT_1_6_1(false, 73, "1.6.1"),
MINECRAFT_1_6_2(false, 74, "1.6.2"),
MINECRAFT_1_6_4(false, 78, "1.6.4"),
MINECRAFT_1_7_2(true, 4, "1.7.5"),
MINECRAFT_1_7_6(true, 5, "1.7.10"),
MINECRAFT_1_8(true, 47, "1.8"),
MINECRAFT_1_9_0(true, 107, "1.9"),
MINECRAFT_1_9_1(true, 108, "1.9.1"),
MINECRAFT_1_9_2(true, 109, "1.9.2"),
MINECRAFT_1_9_4(true, 110, "1.9.4"),
MINECRAFT_1_10(true, 210, "1.10"),
MINECRAFT_1_11(true, 315, "1.11"),
MINECRAFT_1_11_1(true, 316, "1.11.2"),
MINECRAFT_FUTURE(-1);
public static final ProtocolVersion MINECRAFT_1_4_7 = MINECRAFT_1_4_6;
public static final ProtocolVersion MINECRAFT_1_5_1 = MINECRAFT_1_5_0;
public static final ProtocolVersion MINECRAFT_1_7_5 = MINECRAFT_1_7_2;
public static final ProtocolVersion MINECRAFT_1_7_10 = MINECRAFT_1_7_6;
public static final ProtocolVersion MINECRAFT_1_9 = MINECRAFT_1_9_0;
private boolean is_modern_protocol;
private final int id;
private final String name;
ProtocolVersion(int id) {
this(false, id, null);
}
ProtocolVersion(boolean is_modern_protocol, int id, String name) {
this.is_modern_protocol = is_modern_protocol;
this.id = id;
this.name = name;
}
/**
* Returns whether this protocol version is after-Netty-rewrite since 13w41a.
* @return true if this version is used by Minecraft 13w41a or later
*/
public boolean isModernProtocol() {
return is_modern_protocol;
}
/**
* Returns the network version id of this protocol version
* @return network id of this protocol version
*/
public int getId() {
return id;
}
/**
* Returns user friendly version name
* Notice: This name can change, so it shouldn't be used as a key anywhere
* @return user friendly version name
*/
public String getName() {
return name;
}
/**
* Returns if this version is supported as game version (i.e.: player can join and play on the server)
* @return true if this protocol version is supported
*/
public boolean isSupported() {
return name != null;
}
/**
* Returns if the game version used by this protocol released after the game version used by another protocol version
* @param another another protocol version
* @return true if game version is released after the game version used by another protocol version
* @throws IllegalArgumentException if protocol versions use different protocol types
*/
public boolean isAfter(ProtocolVersion another) {
return ordinal() > another.ordinal();
}
/**
* Returns if the game version used by this protocol released after (or is the same) the game version used by another protocol version
* @param another another protocol version
* @return true if game version is released after (or is the same) the game version used by another protocol version
* @throws IllegalArgumentException if protocol versions use different protocol types
*/
public boolean isAfterOrEq(ProtocolVersion another) {
return ordinal() >= another.ordinal();
}
/**
* Returns if the game version used by this protocol released before the game version used by another protocol version
* @param another another protocol version
* @return true if game version is released before the game version used by another protocol version
* @throws IllegalArgumentException if protocol versions use different protocol types
*/
public boolean isBefore(ProtocolVersion another) {
return ordinal() < another.ordinal();
}
/**
* Returns if the game version used by this protocol released before (or is the same) the game version used by another protocol version
* @param another another protocol version
* @return true if game version is released before (or is the same) the game version used by another protocol version
* @throws IllegalArgumentException if protocol versions use different protocol types
*/
public boolean isBeforeOrEq(ProtocolVersion another) {
return ordinal() <= another.ordinal();
}
/**
* Returns if the game version used by this protocol released in between (or is the same) of other game versions used by others protocol versions
* @param one one protocol version
* @param another another protocol version
* @return true if game version is released before (or is the same) the game version used by another protocol version
* @throws IllegalArgumentException if protocol versions use different protocol types
*/
public boolean isBetween(ProtocolVersion one, ProtocolVersion another) {
return (isAfterOrEq(one) && isBeforeOrEq(another)) || (isBeforeOrEq(one) && isAfterOrEq(another));
}
private static final TIntObjectHashMap<ProtocolVersion> modern_versions_by_number = new TIntObjectHashMap<>();
private static final TIntObjectHashMap<ProtocolVersion> legacy_versions_by_number = new TIntObjectHashMap<>();
static {
for(ProtocolVersion version : ProtocolVersion.values()) {
if(!version.isSupported()) continue;
if(version.isModernProtocol()) {
modern_versions_by_number.put(version.id, version);
} else {
legacy_versions_by_number.put(version.id, version);
}
}
}
/**
* Get ProtocolVersion by network protocol version number
* @param is_modern_protocol specific whether it is a modern protocol version
* @param number the protocol version number
* @return Returns protocol version or UNKNOWN if not found
*/
public static ProtocolVersion fromProtocolVersionNumber(boolean is_modern_protocol, int number) {
ProtocolVersion version =
(is_modern_protocol ? modern_versions_by_number : legacy_versions_by_number).
get(number);
return version != null ? version : UNKNOWN;
}
/**
* Returns protocol version by network game id
* @param id network version id
* @return Returns protocol version by network game id or UNKNOWN if not found
* @deprecated legacy and modern protocol versions may share same numeric IDs
*/
@Deprecated
public static ProtocolVersion fromId(int id) {
return fromProtocolVersionNumber(true, id);
}
/**
* Returns protocol version that is used by the game version released after game version used by this protocol
* Returns null if next game version doesn't exist
* @return protocol version that is used by the game version released after game version used by this protocol
* @throws IllegalArgumentException if protocol type is UNKNOWN
*/
public ProtocolVersion next() {
ProtocolVersion[] versions = values();
int nextVersionOrderId = ordinal() + 1;
if (nextVersionOrderId < versions.length) {
return versions[nextVersionOrderId];
} else {
return null;
}
}
/**
* Returns protocol version that is used by the game version released before game version used by this protocol
* Returns null if previous game version doesn't exist
* @return protocol version that is used by the game version released before game version used by this protocol
* @throws IllegalArgumentException if protocol type is UNKNOWN
*/
public ProtocolVersion previous() {
ProtocolVersion[] versions = values();
int previousVersionOrderId = ordinal() - 1;
if (previousVersionOrderId >= 0) {
return versions[previousVersionOrderId];
} else {
return null;
}
}
/**
* Returns all protocol versions that are between specified ones (inclusive)
* Throws {@link IllegalArgumentException} if protocol versions types are not the same or one of the types is UNKNOWN
* @param one one protocol version
* @param another one protocol version
* @return all protocol versions that are between specified ones (inclusive)
*/
public static ProtocolVersion[] getAllBetween(ProtocolVersion one, ProtocolVersion another) {
ProtocolVersion[] versions = values();
int startId = Math.min(one.ordinal(), another.ordinal());
int endId = Math.max(one.ordinal(), another.ordinal());
ProtocolVersion[] between = new ProtocolVersion[(endId - startId) + 1];
for (int i = startId; i <= endId; i++) {
between[i - startId] = versions[i];
}
return between;
}
/**
* Returns latest supported protocol version for specified protocol type
* @param type protocol type
* @return latest supported protocol version for specified protocol type
* @throws IllegalArgumentException if protocol type has not supported protocol versions
*/
public static ProtocolVersion getLatest() {
return MINECRAFT_1_11_1;
}
/**
* Returns oldest supported protocol version for specified protocol type
* @param type protocol type
* @return oldest supported protocol version for specified protocol type
* @throws IllegalArgumentException if protocol type has not supported protocol versions
*/
public static ProtocolVersion getOldest() {
return MINECRAFT_1_4_6;
}
/**
* Returns all protocol versions that are after specified one (inclusive)
* @param version protocol version
* @return all protocol versions that are after specified one (inclusive)
* @throws IllegalArgumentException if getAllBetween(version, getLatest(version.getType())) throws one
* @deprecated non intuitive behavior
*/
@Deprecated
public static ProtocolVersion[] getAllAfter(ProtocolVersion version) {
return getAllBetween(version, getLatest());
}
/**
* Returns all protocol versions that are before specified one (inclusive)
* @param version protocol version
* @return all protocol versions that are before specified one
* @throws IllegalArgumentException if getAllBetween(getOldest(version.getType()), version) throws one
* @deprecated non intuitive behavior
*/
@Deprecated
public static ProtocolVersion[] getAllBefore(ProtocolVersion version) {
return getAllBetween(getOldest(), version);
}
}