Massive amount of fixes and changes with multiplayer/annotations
This commit is contained in:
@@ -15,22 +15,19 @@ public class Annotations {
|
|||||||
@Target(ElementType.METHOD)
|
@Target(ElementType.METHOD)
|
||||||
@Retention(RetentionPolicy.CLASS)
|
@Retention(RetentionPolicy.CLASS)
|
||||||
public @interface Remote {
|
public @interface Remote {
|
||||||
/**If true, this method can only be invoked on clients from the server.
|
/**Specifies the locations where this method can be invoked.*/
|
||||||
* If false, this method can only be invoked on servers from a client.*/
|
Loc targets() default Loc.server;
|
||||||
boolean server() default true;
|
/**Specifies which methods are generated. Only affects client methods.*/
|
||||||
/**Whether a client-specific method is generated that accepts a connecton ID and sends to only one player. Default is false.
|
Variant variants() default Variant.all;
|
||||||
* Only affects client methods.*/
|
/**The local locations where this method is called.*/
|
||||||
boolean one() default false;
|
Loc called() default Loc.none;
|
||||||
/**Whether a 'global' method is generated that sends the event to all players. Default is true.
|
/**Whether to forward this packet to all other clients.*/
|
||||||
* Only affects client methods.*/
|
boolean forward() default false;
|
||||||
boolean all() default true;
|
|
||||||
/**Whether this method is invoked locally as well as remotely.*/
|
|
||||||
boolean local() default true;
|
|
||||||
/**Whether the packet for this method is sent with UDP instead of TCP.
|
/**Whether the packet for this method is sent with UDP instead of TCP.
|
||||||
* UDP is faster, but is prone to packet loss and duplication.*/
|
* UDP is faster, but is prone to packet loss and duplication.*/
|
||||||
boolean unreliable() default false;
|
boolean unreliable() default false;
|
||||||
/**The simple class name where this method is placed.*/
|
/**The simple class name where this method is placed.*/
|
||||||
String target() default "Call";
|
String in() default "Call";
|
||||||
}
|
}
|
||||||
|
|
||||||
/**Specifies that this method will be used to write classes of the type returned by {@link #value()}.<br>
|
/**Specifies that this method will be used to write classes of the type returned by {@link #value()}.<br>
|
||||||
@@ -50,4 +47,42 @@ public class Annotations {
|
|||||||
public @interface ReadClass {
|
public @interface ReadClass {
|
||||||
Class<?> value();
|
Class<?> value();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**A set of two booleans, one specifying server and one specifying client.*/
|
||||||
|
public enum Loc {
|
||||||
|
/**Method can only be invoked on the client from the server.*/
|
||||||
|
server(true, false),
|
||||||
|
/**Method can only be invoked on the server from the client.*/
|
||||||
|
client(false, true),
|
||||||
|
/**Method can be invoked from anywhere*/
|
||||||
|
both(true, true),
|
||||||
|
/**Neither server no client.*/
|
||||||
|
none(false, false);
|
||||||
|
|
||||||
|
/**If true, this method can be invoked ON clients FROM servers.*/
|
||||||
|
public final boolean isServer;
|
||||||
|
/**If true, this method can be invoked ON servers FROM clients.*/
|
||||||
|
public final boolean isClient;
|
||||||
|
|
||||||
|
Loc(boolean server, boolean client){
|
||||||
|
this.isServer = server;
|
||||||
|
this.isClient = client;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public enum Variant {
|
||||||
|
/**Method can only be invoked targeting one player.*/
|
||||||
|
one(true, false),
|
||||||
|
/**Method can only be invoked targeting all players.*/
|
||||||
|
all(false, true),
|
||||||
|
/**Method targets both one player and all players.*/
|
||||||
|
both(true, true);
|
||||||
|
|
||||||
|
public final boolean isOne, isAll;
|
||||||
|
|
||||||
|
Variant(boolean isOne, boolean isAll){
|
||||||
|
this.isOne = isOne;
|
||||||
|
this.isAll = isAll;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,8 @@
|
|||||||
package io.anuke.annotations;
|
package io.anuke.annotations;
|
||||||
|
|
||||||
|
import io.anuke.annotations.Annotations.Loc;
|
||||||
|
import io.anuke.annotations.Annotations.Variant;
|
||||||
|
|
||||||
import javax.lang.model.element.ExecutableElement;
|
import javax.lang.model.element.ExecutableElement;
|
||||||
|
|
||||||
/**Class that repesents a remote method to be constructed and put into a class.*/
|
/**Class that repesents a remote method to be constructed and put into a class.*/
|
||||||
@@ -9,26 +12,28 @@ public class MethodEntry {
|
|||||||
/**Fully qualified target method to call.*/
|
/**Fully qualified target method to call.*/
|
||||||
public final String targetMethod;
|
public final String targetMethod;
|
||||||
/**Whether this method can be called on a client/server.*/
|
/**Whether this method can be called on a client/server.*/
|
||||||
public final boolean server;
|
public final Loc where;
|
||||||
/**Whether an additional 'one' and 'all' method variant is generated. At least one of these must be true.
|
/**Whether an additional 'one' and 'all' method variant is generated. At least one of these must be true.
|
||||||
* Only applicable to client (server-invoked) methods.*/
|
* Only applicable to client (server-invoked) methods.*/
|
||||||
public final boolean allVariant, oneVariant;
|
public final Variant target;
|
||||||
/**Whether this method is called locally as well as remotely.*/
|
/**Whether this method is called locally as well as remotely.*/
|
||||||
public final boolean local;
|
public final Loc local;
|
||||||
/**Whether this method is unreliable and uses UDP.*/
|
/**Whether this method is unreliable and uses UDP.*/
|
||||||
public final boolean unreliable;
|
public final boolean unreliable;
|
||||||
|
/**Whether to forward this method call to all other clients when a client invokes it. Server only.*/
|
||||||
|
public final boolean forward;
|
||||||
/**Unique method ID.*/
|
/**Unique method ID.*/
|
||||||
public final int id;
|
public final int id;
|
||||||
/**The element method associated with this entry.*/
|
/**The element method associated with this entry.*/
|
||||||
public final ExecutableElement element;
|
public final ExecutableElement element;
|
||||||
|
|
||||||
public MethodEntry(String className, String targetMethod, boolean server,
|
public MethodEntry(String className, String targetMethod, Loc where, Variant target,
|
||||||
boolean allVariant, boolean oneVariant, boolean local, boolean unreliable, int id, ExecutableElement element) {
|
Loc local, boolean unreliable, boolean forward, int id, ExecutableElement element) {
|
||||||
this.className = className;
|
this.className = className;
|
||||||
|
this.forward = forward;
|
||||||
this.targetMethod = targetMethod;
|
this.targetMethod = targetMethod;
|
||||||
this.server = server;
|
this.where = where;
|
||||||
this.allVariant = allVariant;
|
this.target = target;
|
||||||
this.oneVariant = oneVariant;
|
|
||||||
this.local = local;
|
this.local = local;
|
||||||
this.id = id;
|
this.id = id;
|
||||||
this.element = element;
|
this.element = element;
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ package io.anuke.annotations;
|
|||||||
import com.squareup.javapoet.FieldSpec;
|
import com.squareup.javapoet.FieldSpec;
|
||||||
import com.squareup.javapoet.JavaFile;
|
import com.squareup.javapoet.JavaFile;
|
||||||
import com.squareup.javapoet.TypeSpec;
|
import com.squareup.javapoet.TypeSpec;
|
||||||
|
import io.anuke.annotations.Annotations.Loc;
|
||||||
import io.anuke.annotations.Annotations.Remote;
|
import io.anuke.annotations.Annotations.Remote;
|
||||||
import io.anuke.annotations.IOFinder.ClassSerializer;
|
import io.anuke.annotations.IOFinder.ClassSerializer;
|
||||||
|
|
||||||
@@ -38,8 +39,19 @@ public class RemoteMethodAnnotationProcessor extends AbstractProcessor {
|
|||||||
/**Name of class that handles reading and invoking packets on the client.*/
|
/**Name of class that handles reading and invoking packets on the client.*/
|
||||||
private static final String readClientName = "RemoteReadClient";
|
private static final String readClientName = "RemoteReadClient";
|
||||||
|
|
||||||
/**Whether the initial round is done.*/
|
/**Processing round number.*/
|
||||||
private boolean done;
|
private int round;
|
||||||
|
|
||||||
|
//class serializers
|
||||||
|
private HashMap<String, ClassSerializer> serializers;
|
||||||
|
//all elements with the Remote annotation
|
||||||
|
private Set<? extends Element> elements;
|
||||||
|
//map of all classes to generate by name
|
||||||
|
private HashMap<String, ClassEntry> classMap;
|
||||||
|
//list of all method entries
|
||||||
|
private ArrayList<MethodEntry> methods;
|
||||||
|
//list of all method entries
|
||||||
|
private ArrayList<ClassEntry> classes;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized void init(ProcessingEnvironment processingEnv) {
|
public synchronized void init(ProcessingEnvironment processingEnv) {
|
||||||
@@ -53,83 +65,91 @@ public class RemoteMethodAnnotationProcessor extends AbstractProcessor {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
|
public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
|
||||||
if(done) return false; //only process 1 round
|
if(round > 1) return false; //only process 2 rounds
|
||||||
done = true;
|
|
||||||
|
round ++;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
|
||||||
//get serializers
|
//round 1: find all annotations, generate *writers*
|
||||||
HashMap<String, ClassSerializer> serializers = new IOFinder().findSerializers(roundEnv);
|
if(round == 1) {
|
||||||
|
//get serializers
|
||||||
|
serializers = new IOFinder().findSerializers(roundEnv);
|
||||||
|
|
||||||
//last method ID used
|
//last method ID used
|
||||||
int lastMethodID = 0;
|
int lastMethodID = 0;
|
||||||
//find all elements with the Remote annotation
|
//find all elements with the Remote annotation
|
||||||
Set<? extends Element> elements = roundEnv.getElementsAnnotatedWith(Remote.class);
|
elements = roundEnv.getElementsAnnotatedWith(Remote.class);
|
||||||
//map of all classes to generate by name
|
//map of all classes to generate by name
|
||||||
HashMap<String, ClassEntry> classMap = new HashMap<>();
|
classMap = new HashMap<>();
|
||||||
//list of all method entries
|
//list of all method entries
|
||||||
ArrayList<MethodEntry> methods = new ArrayList<>();
|
methods = new ArrayList<>();
|
||||||
//list of all method entries
|
//list of all method entries
|
||||||
ArrayList<ClassEntry> classes = new ArrayList<>();
|
classes = new ArrayList<>();
|
||||||
|
|
||||||
//create methods
|
//create methods
|
||||||
for (Element element : elements) {
|
for (Element element : elements) {
|
||||||
Remote annotation = element.getAnnotation(Remote.class);
|
Remote annotation = element.getAnnotation(Remote.class);
|
||||||
|
|
||||||
//check for static
|
//check for static
|
||||||
if(!element.getModifiers().contains(Modifier.STATIC)) {
|
if (!element.getModifiers().contains(Modifier.STATIC) || !element.getModifiers().contains(Modifier.PUBLIC)) {
|
||||||
Utils.messager.printMessage(Kind.ERROR, "All Remote methods must be static: ", element);
|
Utils.messager.printMessage(Kind.ERROR, "All @Remote methods must be public and static: ", element);
|
||||||
|
}
|
||||||
|
|
||||||
|
//can't generate none methods
|
||||||
|
if (annotation.targets() == Loc.none) {
|
||||||
|
Utils.messager.printMessage(Kind.ERROR, "A @Remote method's where() cannot be equal to 'none':", element);
|
||||||
|
}
|
||||||
|
|
||||||
|
//get and create class entry if needed
|
||||||
|
if (!classMap.containsKey(annotation.in())) {
|
||||||
|
ClassEntry clas = new ClassEntry(annotation.in());
|
||||||
|
classMap.put(annotation.in(), clas);
|
||||||
|
classes.add(clas);
|
||||||
|
}
|
||||||
|
|
||||||
|
ClassEntry entry = classMap.get(annotation.in());
|
||||||
|
|
||||||
|
//create and add entry
|
||||||
|
MethodEntry method = new MethodEntry(entry.name, Utils.getMethodName(element), annotation.targets(), annotation.variants(),
|
||||||
|
annotation.called(), annotation.unreliable(), annotation.forward(), lastMethodID++, (ExecutableElement) element);
|
||||||
|
|
||||||
|
entry.methods.add(method);
|
||||||
|
methods.add(method);
|
||||||
}
|
}
|
||||||
|
|
||||||
//get and create class entry if needed
|
//create read/write generators
|
||||||
if (!classMap.containsKey(annotation.target())) {
|
RemoteWriteGenerator writegen = new RemoteWriteGenerator(serializers);
|
||||||
ClassEntry clas = new ClassEntry(annotation.target());
|
|
||||||
classMap.put(annotation.target(), clas);
|
|
||||||
classes.add(clas);
|
|
||||||
}
|
|
||||||
|
|
||||||
ClassEntry entry = classMap.get(annotation.target());
|
//generate the methods to invoke (write)
|
||||||
|
writegen.generateFor(classes, packageName);
|
||||||
|
|
||||||
//make sure that each server annotation has at least one method to generate, otherwise throw an error
|
return true;
|
||||||
if (annotation.server() && !annotation.all() && !annotation.one()) {
|
}else if(round == 2) { //round 2: generate all *readers*
|
||||||
Utils.messager.printMessage(Kind.ERROR, "A client method must not have all() and one() both be false!", element);
|
RemoteReadGenerator readgen = new RemoteReadGenerator(serializers);
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
//create and add entry
|
//generate server readers
|
||||||
MethodEntry method = new MethodEntry(entry.name, Utils.getMethodName(element), annotation.server(),
|
readgen.generateFor(methods.stream().filter(method -> method.where.isClient).collect(Collectors.toList()), readServerName, packageName, true);
|
||||||
annotation.all(), annotation.one(), annotation.local(), annotation.unreliable(), lastMethodID ++, (ExecutableElement)element);
|
//generate client readers
|
||||||
|
readgen.generateFor(methods.stream().filter(method -> method.where.isServer).collect(Collectors.toList()), readClientName, packageName, false);
|
||||||
|
|
||||||
entry.methods.add(method);
|
//create class for storing unique method hash
|
||||||
methods.add(method);
|
TypeSpec.Builder hashBuilder = TypeSpec.classBuilder("MethodHash").addModifiers(Modifier.PUBLIC);
|
||||||
|
hashBuilder.addField(FieldSpec.builder(int.class, "HASH", Modifier.STATIC, Modifier.PUBLIC, Modifier.FINAL)
|
||||||
|
.initializer("$1L", Objects.hash(methods)).build());
|
||||||
|
|
||||||
|
//build and write resulting hash class
|
||||||
|
TypeSpec spec = hashBuilder.build();
|
||||||
|
JavaFile.builder(packageName, spec).build().writeTo(Utils.filer);
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
//create read/write generators
|
|
||||||
RemoteReadGenerator readgen = new RemoteReadGenerator(serializers);
|
|
||||||
RemoteWriteGenerator writegen = new RemoteWriteGenerator(serializers);
|
|
||||||
|
|
||||||
//generate server readers
|
|
||||||
readgen.generateFor(methods.stream().filter(method -> !method.server).collect(Collectors.toList()), readServerName, packageName, true);
|
|
||||||
//generate client readers
|
|
||||||
readgen.generateFor(methods.stream().filter(method -> method.server).collect(Collectors.toList()), readClientName, packageName, false);
|
|
||||||
|
|
||||||
//generate the methods to invoke (write)
|
|
||||||
writegen.generateFor(classes, packageName);
|
|
||||||
|
|
||||||
//create class for storing unique method hash
|
|
||||||
TypeSpec.Builder hashBuilder = TypeSpec.classBuilder("MethodHash").addModifiers(Modifier.PUBLIC);
|
|
||||||
hashBuilder.addField(FieldSpec.builder(int.class, "HASH", Modifier.STATIC, Modifier.PUBLIC, Modifier.FINAL)
|
|
||||||
.initializer("$1L", Objects.hash(methods)).build());
|
|
||||||
|
|
||||||
//build and write resulting hash class
|
|
||||||
TypeSpec spec = hashBuilder.build();
|
|
||||||
JavaFile.builder(packageName, spec).build().writeTo(Utils.filer);
|
|
||||||
|
|
||||||
return true;
|
|
||||||
|
|
||||||
}catch (Exception e){
|
}catch (Exception e){
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
throw new RuntimeException(e);
|
throw new RuntimeException(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -70,9 +70,8 @@ public class RemoteReadGenerator {
|
|||||||
for(int i = 0; i < entry.element.getParameters().size(); i ++){
|
for(int i = 0; i < entry.element.getParameters().size(); i ++){
|
||||||
VariableElement var = entry.element.getParameters().get(i);
|
VariableElement var = entry.element.getParameters().get(i);
|
||||||
|
|
||||||
if(entry.server || i != 0) { //if client, skip first parameter since it's always of type player and doesn't need to be read
|
if(!needsPlayer || i != 0) { //if client, skip first parameter since it's always of type player and doesn't need to be read
|
||||||
//full type name of parameter
|
//full type name of parameter
|
||||||
//TODO check if the result is correct
|
|
||||||
String typeName = var.asType().toString();
|
String typeName = var.asType().toString();
|
||||||
//name of parameter
|
//name of parameter
|
||||||
String varName = var.getSimpleName().toString();
|
String varName = var.getSimpleName().toString();
|
||||||
@@ -98,16 +97,41 @@ public class RemoteReadGenerator {
|
|||||||
//add statement for reading it
|
//add statement for reading it
|
||||||
readBlock.addStatement(typeName + " " + varName + " = " + ser.readMethod + "(buffer)");
|
readBlock.addStatement(typeName + " " + varName + " = " + ser.readMethod + "(buffer)");
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
//append variable name to string builder
|
//append variable name to string builder
|
||||||
varResult.append(var.getSimpleName());
|
varResult.append(var.getSimpleName());
|
||||||
if(i != entry.element.getParameters().size() - 1) varResult.append(", ");
|
if(i != entry.element.getParameters().size() - 1) varResult.append(", ");
|
||||||
|
}else{
|
||||||
|
varResult.append("player");
|
||||||
|
if(i != entry.element.getParameters().size() - 1) varResult.append(", ");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//now execute it
|
|
||||||
readBlock.addStatement("com.badlogic.gdx.Gdx.app.postRunnable(() -> $N." + entry.element.getSimpleName() + "(" + varResult.toString() + "))",
|
|
||||||
((TypeElement)entry.element.getEnclosingElement()).getQualifiedName().toString());
|
//begin lambda control flow
|
||||||
|
readBlock.beginControlFlow("com.badlogic.gdx.Gdx.app.postRunnable(() -> ");
|
||||||
|
|
||||||
|
//call forwarded method before the method, so if it throws a ValidateException, the method won't be forwarded
|
||||||
|
if(entry.forward && entry.where.isServer){
|
||||||
|
//try block to catch validate exception
|
||||||
|
readBlock.beginControlFlow("try");
|
||||||
|
|
||||||
|
//call forwarded method
|
||||||
|
readBlock.addStatement(packageName + "." + entry.className + "." + entry.element.getSimpleName() +
|
||||||
|
"__forward(player.clientid" + (varResult.length() == 0 ? "" : ", ") + varResult.toString() + ")");
|
||||||
|
|
||||||
|
//when a ValidateException is caught, print the error and return
|
||||||
|
readBlock.nextControlFlow("catch (io.anuke.mindustry.net.ValidateException e)");
|
||||||
|
readBlock.addStatement("e.printStackTrace()");
|
||||||
|
readBlock.addStatement("return");
|
||||||
|
readBlock.endControlFlow();
|
||||||
|
}
|
||||||
|
|
||||||
|
//execute the relevant method
|
||||||
|
readBlock.addStatement("$N." + entry.element.getSimpleName() + "(" + varResult.toString() + ")", ((TypeElement) entry.element.getEnclosingElement()).getQualifiedName().toString());
|
||||||
|
//end lambda
|
||||||
|
readBlock.endControlFlow(")");
|
||||||
}
|
}
|
||||||
|
|
||||||
//end control flow if necessary
|
//end control flow if necessary
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
package io.anuke.annotations;
|
package io.anuke.annotations;
|
||||||
|
|
||||||
import com.squareup.javapoet.*;
|
import com.squareup.javapoet.*;
|
||||||
|
import io.anuke.annotations.Annotations.Loc;
|
||||||
import io.anuke.annotations.IOFinder.ClassSerializer;
|
import io.anuke.annotations.IOFinder.ClassSerializer;
|
||||||
|
|
||||||
import javax.lang.model.element.ExecutableElement;
|
import javax.lang.model.element.ExecutableElement;
|
||||||
@@ -27,6 +28,7 @@ public class RemoteWriteGenerator {
|
|||||||
|
|
||||||
for(ClassEntry entry : entries){
|
for(ClassEntry entry : entries){
|
||||||
//create builder
|
//create builder
|
||||||
|
System.out.println("Generating class! " + entry.name);
|
||||||
TypeSpec.Builder classBuilder = TypeSpec.classBuilder(entry.name).addModifiers(Modifier.PUBLIC);
|
TypeSpec.Builder classBuilder = TypeSpec.classBuilder(entry.name).addModifiers(Modifier.PUBLIC);
|
||||||
|
|
||||||
//add temporary write buffer
|
//add temporary write buffer
|
||||||
@@ -36,13 +38,18 @@ public class RemoteWriteGenerator {
|
|||||||
//go through each method entry in this class
|
//go through each method entry in this class
|
||||||
for(MethodEntry methodEntry : entry.methods){
|
for(MethodEntry methodEntry : entry.methods){
|
||||||
//write the 'send event to all players' variant: always happens for clients, but only happens if 'all' is enabled on the server method
|
//write the 'send event to all players' variant: always happens for clients, but only happens if 'all' is enabled on the server method
|
||||||
if(!methodEntry.server || methodEntry.allVariant){
|
if(methodEntry.where.isClient || methodEntry.target.isAll){
|
||||||
writeMethodVariant(classBuilder, methodEntry, true);
|
writeMethodVariant(classBuilder, methodEntry, true, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
//write the 'send even to one player' variant, which is only applicable on the server
|
//write the 'send event to one player' variant, which is only applicable on the server
|
||||||
if(methodEntry.server && methodEntry.oneVariant){
|
if(methodEntry.where.isServer && methodEntry.target.isOne){
|
||||||
writeMethodVariant(classBuilder, methodEntry, false);
|
writeMethodVariant(classBuilder, methodEntry, false, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
//write the forwarded method version
|
||||||
|
if(methodEntry.where.isServer && methodEntry.forward){
|
||||||
|
writeMethodVariant(classBuilder, methodEntry, true, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -53,16 +60,16 @@ public class RemoteWriteGenerator {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**Creates a specific variant for a method entry.*/
|
/**Creates a specific variant for a method entry.*/
|
||||||
private void writeMethodVariant(TypeSpec.Builder classBuilder, MethodEntry methodEntry, boolean toAll){
|
private void writeMethodVariant(TypeSpec.Builder classBuilder, MethodEntry methodEntry, boolean toAll, boolean forwarded){
|
||||||
ExecutableElement elem = methodEntry.element;
|
ExecutableElement elem = methodEntry.element;
|
||||||
|
|
||||||
//create builder
|
//create builder
|
||||||
MethodSpec.Builder method = MethodSpec.methodBuilder(elem.getSimpleName().toString())
|
MethodSpec.Builder method = MethodSpec.methodBuilder(elem.getSimpleName().toString() + (forwarded ? "__forward" : "")) //add except suffix when forwarding
|
||||||
.addModifiers(Modifier.PUBLIC, Modifier.STATIC)
|
.addModifiers(Modifier.PUBLIC, Modifier.STATIC, Modifier.SYNCHRONIZED)
|
||||||
.returns(void.class);
|
.returns(void.class);
|
||||||
|
|
||||||
//validate client methods to make sure
|
//validate client methods to make sure
|
||||||
if(!methodEntry.server){
|
if(methodEntry.where.isClient){
|
||||||
if(elem.getParameters().isEmpty()){
|
if(elem.getParameters().isEmpty()){
|
||||||
Utils.messager.printMessage(Kind.ERROR, "Client invoke methods must have a first parameter of type Player.", elem);
|
Utils.messager.printMessage(Kind.ERROR, "Client invoke methods must have a first parameter of type Player.", elem);
|
||||||
return;
|
return;
|
||||||
@@ -79,8 +86,18 @@ public class RemoteWriteGenerator {
|
|||||||
method.addParameter(int.class, "playerClientID");
|
method.addParameter(int.class, "playerClientID");
|
||||||
}
|
}
|
||||||
|
|
||||||
//call local method if applicable
|
//add sender to ignore
|
||||||
if(methodEntry.local && methodEntry.server){
|
if(forwarded){
|
||||||
|
method.addParameter(int.class, "exceptSenderID");
|
||||||
|
}
|
||||||
|
|
||||||
|
//call local method if applicable, shouldn't happen when forwarding method as that already happens by default
|
||||||
|
if(!forwarded && methodEntry.local != Loc.none){
|
||||||
|
//add in local checks
|
||||||
|
if(methodEntry.local != Loc.both){
|
||||||
|
method.beginControlFlow("if("+getCheckString(methodEntry.local) + " || !io.anuke.mindustry.net.Net.active())");
|
||||||
|
}
|
||||||
|
|
||||||
//concatenate parameters
|
//concatenate parameters
|
||||||
int index = 0;
|
int index = 0;
|
||||||
StringBuilder results = new StringBuilder();
|
StringBuilder results = new StringBuilder();
|
||||||
@@ -93,21 +110,27 @@ public class RemoteWriteGenerator {
|
|||||||
//add the statement to call it
|
//add the statement to call it
|
||||||
method.addStatement("$N." + elem.getSimpleName() + "(" + results.toString() + ")",
|
method.addStatement("$N." + elem.getSimpleName() + "(" + results.toString() + ")",
|
||||||
((TypeElement)elem.getEnclosingElement()).getQualifiedName().toString());
|
((TypeElement)elem.getEnclosingElement()).getQualifiedName().toString());
|
||||||
|
|
||||||
|
if(methodEntry.local != Loc.both){
|
||||||
|
method.endControlFlow();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//start control flow to check if it's actually client/server so no netcode is called
|
//start control flow to check if it's actually client/server so no netcode is called
|
||||||
method.beginControlFlow("if(io.anuke.mindustry.net.Net." + (!methodEntry.server ? "client" : "server")+"())");
|
method.beginControlFlow("if("+getCheckString(methodEntry.where)+")");
|
||||||
|
|
||||||
//add statement to create packet from pool
|
//add statement to create packet from pool
|
||||||
method.addStatement("$1N packet = $2N.obtain($1N.class)", "io.anuke.mindustry.net.Packets.InvokePacket", "com.badlogic.gdx.utils.Pools");
|
method.addStatement("$1N packet = $2N.obtain($1N.class)", "io.anuke.mindustry.net.Packets.InvokePacket", "com.badlogic.gdx.utils.Pools");
|
||||||
//assign buffer
|
//assign buffer
|
||||||
method.addStatement("packet.writeBuffer = TEMP_BUFFER");
|
method.addStatement("packet.writeBuffer = TEMP_BUFFER");
|
||||||
|
//assign method ID
|
||||||
|
method.addStatement("packet.type = (byte)" + methodEntry.id);
|
||||||
//rewind buffer
|
//rewind buffer
|
||||||
method.addStatement("TEMP_BUFFER.position(0)");
|
method.addStatement("TEMP_BUFFER.position(0)");
|
||||||
|
|
||||||
for(int i = 0; i < elem.getParameters().size(); i ++){
|
for(int i = 0; i < elem.getParameters().size(); i ++){
|
||||||
//first argument is skipped as it is always the player caller
|
//first argument is skipped as it is always the player caller
|
||||||
if(!methodEntry.server && i == 0){
|
if((!methodEntry.where.isServer/* || methodEntry.mode == Loc.both*/) && i == 0){
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -147,21 +170,31 @@ public class RemoteWriteGenerator {
|
|||||||
//assign packet length
|
//assign packet length
|
||||||
method.addStatement("packet.writeLength = TEMP_BUFFER.position()");
|
method.addStatement("packet.writeLength = TEMP_BUFFER.position()");
|
||||||
|
|
||||||
//send the actual packet
|
String sendString;
|
||||||
if(toAll){
|
|
||||||
//send to all players / to server
|
if(forwarded){ //forward packet
|
||||||
method.addStatement("io.anuke.mindustry.net.Net.send(packet, "+
|
sendString = "sendExcept(exceptSenderID, ";
|
||||||
(methodEntry.unreliable ? "io.anuke.mindustry.net.Net.SendMode.udp" : "io.anuke.mindustry.net.Net.SendMode.tcp")+")");
|
}else if(toAll){ //send to all players / to server
|
||||||
}else{
|
sendString = "send(";
|
||||||
//send to specific client from server
|
}else{ //send to specific client from server
|
||||||
method.addStatement("io.anuke.mindustry.net.Net.sendTo(playerClientID, packet, "+
|
sendString = "sendTo(playerClientID, ";
|
||||||
(methodEntry.unreliable ? "io.anuke.mindustry.net.Net.SendMode.udp" : "io.anuke.mindustry.net.Net.SendMode.tcp")+")");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//send the actual packet
|
||||||
|
method.addStatement("io.anuke.mindustry.net.Net." + sendString + "packet, "+
|
||||||
|
(methodEntry.unreliable ? "io.anuke.mindustry.net.Net.SendMode.udp" : "io.anuke.mindustry.net.Net.SendMode.tcp")+")");
|
||||||
|
|
||||||
|
|
||||||
//end check for server/client
|
//end check for server/client
|
||||||
method.endControlFlow();
|
method.endControlFlow();
|
||||||
|
|
||||||
//add method to class, finally
|
//add method to class, finally
|
||||||
classBuilder.addMethod(method.build());
|
classBuilder.addMethod(method.build());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private String getCheckString(Loc loc){
|
||||||
|
return loc.isClient && loc.isServer ? "io.anuke.mindustry.net.Net.server() || io.anuke.mindustry.net.Net.client()" :
|
||||||
|
loc.isClient ? "io.anuke.mindustry.net.Net.client()" :
|
||||||
|
loc.isServer ? "io.anuke.mindustry.net.Net.server()" : "false";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -31,7 +31,6 @@ public class Mindustry extends ModuleCore {
|
|||||||
module(ui = new UI());
|
module(ui = new UI());
|
||||||
module(netServer = new NetServer());
|
module(netServer = new NetServer());
|
||||||
module(netClient = new NetClient());
|
module(netClient = new NetClient());
|
||||||
module(netCommon = new NetCommon());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|||||||
@@ -123,7 +123,6 @@ public class Vars{
|
|||||||
public static Renderer renderer;
|
public static Renderer renderer;
|
||||||
public static UI ui;
|
public static UI ui;
|
||||||
public static World world;
|
public static World world;
|
||||||
public static NetCommon netCommon;
|
|
||||||
public static NetServer netServer;
|
public static NetServer netServer;
|
||||||
public static NetClient netClient;
|
public static NetClient netClient;
|
||||||
|
|
||||||
|
|||||||
@@ -138,7 +138,7 @@ public class Control extends Module{
|
|||||||
|
|
||||||
Events.on(PlayEvent.class, () -> {
|
Events.on(PlayEvent.class, () -> {
|
||||||
for(Player player : players){
|
for(Player player : players){
|
||||||
player.dead = true;
|
player.add();
|
||||||
}
|
}
|
||||||
|
|
||||||
state.set(State.playing);
|
state.set(State.playing);
|
||||||
|
|||||||
@@ -4,10 +4,12 @@ import com.badlogic.gdx.graphics.Color;
|
|||||||
import com.badlogic.gdx.utils.reflect.ClassReflection;
|
import com.badlogic.gdx.utils.reflect.ClassReflection;
|
||||||
import com.badlogic.gdx.utils.reflect.ReflectionException;
|
import com.badlogic.gdx.utils.reflect.ReflectionException;
|
||||||
import io.anuke.annotations.Annotations.Remote;
|
import io.anuke.annotations.Annotations.Remote;
|
||||||
|
import io.anuke.annotations.Annotations.Variant;
|
||||||
import io.anuke.mindustry.core.GameState.State;
|
import io.anuke.mindustry.core.GameState.State;
|
||||||
import io.anuke.mindustry.entities.Player;
|
import io.anuke.mindustry.entities.Player;
|
||||||
import io.anuke.mindustry.entities.traits.SyncTrait;
|
import io.anuke.mindustry.entities.traits.SyncTrait;
|
||||||
import io.anuke.mindustry.gen.Call;
|
import io.anuke.mindustry.gen.Call;
|
||||||
|
import io.anuke.mindustry.gen.RemoteReadClient;
|
||||||
import io.anuke.mindustry.net.Net;
|
import io.anuke.mindustry.net.Net;
|
||||||
import io.anuke.mindustry.net.Net.SendMode;
|
import io.anuke.mindustry.net.Net.SendMode;
|
||||||
import io.anuke.mindustry.net.NetworkIO;
|
import io.anuke.mindustry.net.NetworkIO;
|
||||||
@@ -102,7 +104,10 @@ public class NetClient extends Module {
|
|||||||
finishConnecting();
|
finishConnecting();
|
||||||
});
|
});
|
||||||
|
|
||||||
Net.handleClient(InvokePacket.class, packet -> {});
|
Net.handleClient(InvokePacket.class, packet -> {
|
||||||
|
packet.writeBuffer.position(0);
|
||||||
|
RemoteReadClient.readPacket(packet.writeBuffer, packet.type);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -165,9 +170,16 @@ public class NetClient extends Module {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Remote(one = true, all = false, unreliable = true)
|
@Remote(variants = Variant.one)
|
||||||
public static void onSnapshot(byte[] snapshot, int snapshotID){
|
public static void onKick(KickReason reason){
|
||||||
|
netClient.disconnectQuietly();
|
||||||
|
state.set(State.menu);
|
||||||
|
if(!reason.quiet) ui.showError("$text.server.kicked." + reason.name());
|
||||||
|
ui.loadfrag.hide();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Remote(variants = Variant.one, unreliable = true)
|
||||||
|
public static void onSnapshot(byte[] snapshot, int snapshotID){
|
||||||
//skip snapshot IDs that have already been recieved
|
//skip snapshot IDs that have already been recieved
|
||||||
if(snapshotID == netClient.lastSnapshotID){
|
if(snapshotID == netClient.lastSnapshotID){
|
||||||
return;
|
return;
|
||||||
@@ -177,7 +189,7 @@ public class NetClient extends Module {
|
|||||||
|
|
||||||
byte[] result;
|
byte[] result;
|
||||||
int length;
|
int length;
|
||||||
if (snapshotID == -1) { //-1 = fresh snapshot
|
if (snapshotID == 0) { //fresh snapshot
|
||||||
result = snapshot;
|
result = snapshot;
|
||||||
length = snapshot.length;
|
length = snapshot.length;
|
||||||
netClient.lastSnapshot = snapshot;
|
netClient.lastSnapshot = snapshot;
|
||||||
@@ -189,6 +201,8 @@ public class NetClient extends Module {
|
|||||||
netClient.lastSnapshot = Arrays.copyOf(result, length);
|
netClient.lastSnapshot = Arrays.copyOf(result, length);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
netClient.lastSnapshotID = snapshotID;
|
||||||
|
|
||||||
//set stream bytes to begin write
|
//set stream bytes to begin write
|
||||||
netClient.byteStream.setBytes(result, 0, length);
|
netClient.byteStream.setBytes(result, 0, length);
|
||||||
|
|
||||||
@@ -214,6 +228,7 @@ public class NetClient extends Module {
|
|||||||
//entity must not be added yet, so create it
|
//entity must not be added yet, so create it
|
||||||
if(entity == null){
|
if(entity == null){
|
||||||
entity = (SyncTrait) ClassReflection.newInstance(group.getType()); //TODO solution without reflection?
|
entity = (SyncTrait) ClassReflection.newInstance(group.getType()); //TODO solution without reflection?
|
||||||
|
entity.resetID(id);
|
||||||
entity.add();
|
entity.add();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -222,15 +237,11 @@ public class NetClient extends Module {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//confirm that snapshot 0 has been recieved if this is the initial snapshot
|
//confirm that snapshot has been recieved
|
||||||
if(snapshotID == -1){
|
netClient.lastSnapshotID = snapshotID;
|
||||||
netClient.lastSnapshotID = 0;
|
|
||||||
}else{ //confirm that the snapshot has been recieved
|
|
||||||
netClient.lastSnapshotID = snapshotID;
|
|
||||||
}
|
|
||||||
|
|
||||||
}catch (IOException | ReflectionException e){
|
}catch (IOException | ReflectionException e){
|
||||||
throw new RuntimeException(e);
|
e.printStackTrace();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,19 +0,0 @@
|
|||||||
package io.anuke.mindustry.core;
|
|
||||||
|
|
||||||
import io.anuke.mindustry.entities.Player;
|
|
||||||
import io.anuke.ucore.modules.Module;
|
|
||||||
|
|
||||||
import static io.anuke.mindustry.Vars.playerGroup;
|
|
||||||
|
|
||||||
public class NetCommon extends Module {
|
|
||||||
|
|
||||||
public void sendMessage(String message){
|
|
||||||
//TODO implement
|
|
||||||
}
|
|
||||||
|
|
||||||
public String colorizeName(int id, String name){
|
|
||||||
Player player = playerGroup.getByID(id);
|
|
||||||
if(name == null || player == null) return null;
|
|
||||||
return "[#" + player.color.toString().toUpperCase() + "]" + name;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -3,6 +3,7 @@ package io.anuke.mindustry.core;
|
|||||||
import com.badlogic.gdx.utils.Base64Coder;
|
import com.badlogic.gdx.utils.Base64Coder;
|
||||||
import com.badlogic.gdx.utils.IntMap;
|
import com.badlogic.gdx.utils.IntMap;
|
||||||
import com.badlogic.gdx.utils.TimeUtils;
|
import com.badlogic.gdx.utils.TimeUtils;
|
||||||
|
import io.anuke.annotations.Annotations.Loc;
|
||||||
import io.anuke.annotations.Annotations.Remote;
|
import io.anuke.annotations.Annotations.Remote;
|
||||||
import io.anuke.mindustry.content.Mechs;
|
import io.anuke.mindustry.content.Mechs;
|
||||||
import io.anuke.mindustry.core.GameState.State;
|
import io.anuke.mindustry.core.GameState.State;
|
||||||
@@ -32,10 +33,8 @@ import java.io.IOException;
|
|||||||
import static io.anuke.mindustry.Vars.*;
|
import static io.anuke.mindustry.Vars.*;
|
||||||
|
|
||||||
public class NetServer extends Module{
|
public class NetServer extends Module{
|
||||||
private final static float serverSyncTime = 4, itemSyncTime = 10, kickDuration = 30 * 1000;
|
private final static float serverSyncTime = 4, kickDuration = 30 * 1000;
|
||||||
|
private final static boolean preventDuplicatNames = false;
|
||||||
private final static int timerEntitySync = 0;
|
|
||||||
private final static int timerStateSync = 1;
|
|
||||||
|
|
||||||
public final Administration admins = new Administration();
|
public final Administration admins = new Administration();
|
||||||
|
|
||||||
@@ -58,7 +57,13 @@ public class NetServer extends Module{
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
Net.handleServer(Disconnect.class, (id, packet) -> {});
|
Net.handleServer(Disconnect.class, (id, packet) -> {
|
||||||
|
Player player = connections.get(id);
|
||||||
|
if(player != null){
|
||||||
|
Call.sendMessage("[accent]" + player.name + " has disconnected.");
|
||||||
|
player.remove();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
Net.handleServer(ConnectPacket.class, (id, packet) -> {
|
Net.handleServer(ConnectPacket.class, (id, packet) -> {
|
||||||
String uuid = new String(Base64Coder.encode(packet.uuid));
|
String uuid = new String(Base64Coder.encode(packet.uuid));
|
||||||
@@ -81,10 +86,12 @@ public class NetServer extends Module{
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
for(Player player : playerGroup.all()){
|
if(preventDuplicatNames) {
|
||||||
if(player.name.equalsIgnoreCase(packet.name)){
|
for (Player player : playerGroup.all()) {
|
||||||
kick(id, KickReason.nameInUse);
|
if (player.name.equalsIgnoreCase(packet.name)) {
|
||||||
return;
|
kick(id, KickReason.nameInUse);
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -172,6 +179,7 @@ public class NetServer extends Module{
|
|||||||
}
|
}
|
||||||
|
|
||||||
//TODO kick player, send kick packet
|
//TODO kick player, send kick packet
|
||||||
|
Call.onKick(connection, reason);
|
||||||
|
|
||||||
Timers.runTask(2f, con::close);
|
Timers.runTask(2f, con::close);
|
||||||
|
|
||||||
@@ -184,15 +192,21 @@ public class NetServer extends Module{
|
|||||||
|
|
||||||
void sync(){
|
void sync(){
|
||||||
try {
|
try {
|
||||||
//TODO implement snapshot packets w/ delta compression
|
|
||||||
|
|
||||||
//iterate through each player
|
//iterate through each player
|
||||||
for (Player player : connections.values()) {
|
for (Player player : connections.values()) {
|
||||||
NetConnection connection = Net.getConnection(player.clientid);
|
NetConnection connection = Net.getConnection(player.clientid);
|
||||||
|
|
||||||
|
if(connection == null){
|
||||||
|
Log.err("Player {0} failed to connect.", player.name);
|
||||||
|
connections.remove(player.clientid);
|
||||||
|
player.remove();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if(!player.timer.get(Player.timeSync, serverSyncTime)) continue;
|
if(!player.timer.get(Player.timeSync, serverSyncTime)) continue;
|
||||||
|
|
||||||
//if the player hasn't acknolwedged that it has recieved the packet, send the same thing again
|
//if the player hasn't acknowledged that it has recieved the packet, send the same thing again
|
||||||
if(connection.lastSentSnapshotID > connection.lastSnapshotID){
|
if(connection.lastSentSnapshotID > connection.lastSnapshotID){
|
||||||
Call.onSnapshot(connection.id, connection.lastSentSnapshot, connection.lastSentSnapshotID);
|
Call.onSnapshot(connection.id, connection.lastSentSnapshot, connection.lastSentSnapshotID);
|
||||||
return;
|
return;
|
||||||
@@ -215,7 +229,7 @@ public class NetServer extends Module{
|
|||||||
|
|
||||||
//check for syncable groups
|
//check for syncable groups
|
||||||
for (EntityGroup<?> group : Entities.getAllGroups()) {
|
for (EntityGroup<?> group : Entities.getAllGroups()) {
|
||||||
//TODO range-check sync positions?
|
//TODO range-check sync positions to optimize?
|
||||||
if (group.isEmpty() || !(group.all().get(0) instanceof SyncTrait)) continue;
|
if (group.isEmpty() || !(group.all().get(0) instanceof SyncTrait)) continue;
|
||||||
|
|
||||||
//make sure mapping is enabled for this group
|
//make sure mapping is enabled for this group
|
||||||
@@ -223,13 +237,20 @@ public class NetServer extends Module{
|
|||||||
throw new RuntimeException("Entity group '" + group.getType() + "' contains SyncTrait entities, yet mapping is not enabled. In order for syncing to work, you must enable mapping for this group.");
|
throw new RuntimeException("Entity group '" + group.getType() + "' contains SyncTrait entities, yet mapping is not enabled. In order for syncing to work, you must enable mapping for this group.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int size = group.size();
|
||||||
|
|
||||||
|
if(group.getType() == Player.class){
|
||||||
|
size --;
|
||||||
|
}
|
||||||
|
|
||||||
//write group ID + group size
|
//write group ID + group size
|
||||||
dataStream.writeByte(group.getID());
|
dataStream.writeByte(group.getID());
|
||||||
dataStream.writeShort(group.size());
|
dataStream.writeShort(size);
|
||||||
//write timestamp
|
//write timestamp
|
||||||
dataStream.writeLong(TimeUtils.millis());
|
dataStream.writeLong(TimeUtils.millis());
|
||||||
|
|
||||||
for(Entity entity : group.all()){
|
for(Entity entity : group.all()){
|
||||||
|
if(entity == player) continue;
|
||||||
//write all entities now
|
//write all entities now
|
||||||
dataStream.writeInt(entity.getID());
|
dataStream.writeInt(entity.getID());
|
||||||
((SyncTrait)entity).write(dataStream);
|
((SyncTrait)entity).write(dataStream);
|
||||||
@@ -237,17 +258,17 @@ public class NetServer extends Module{
|
|||||||
}
|
}
|
||||||
|
|
||||||
byte[] bytes = syncStream.toByteArray();
|
byte[] bytes = syncStream.toByteArray();
|
||||||
if(connection.lastSnapshot == null){
|
connection.lastSentSnapshot = bytes;
|
||||||
|
if(connection.lastSnapshotID == -1){
|
||||||
//no snapshot to diff, send it all
|
//no snapshot to diff, send it all
|
||||||
Call.onSnapshot(connection.id, bytes, -1);
|
Call.onSnapshot(connection.id, bytes, 0);
|
||||||
|
connection.lastSnapshotID = 0;
|
||||||
}else{
|
}else{
|
||||||
//increment snapshot ID
|
|
||||||
connection.lastSnapshotID ++;
|
|
||||||
//send diff, otherwise
|
//send diff, otherwise
|
||||||
byte[] diff = ByteDeltaEncoder.toDiff(new ByteMatcherHash(connection.lastSnapshot, bytes), encoder);
|
byte[] diff = ByteDeltaEncoder.toDiff(new ByteMatcherHash(connection.lastSnapshot, bytes), encoder);
|
||||||
Call.onSnapshot(connection.id, diff, connection.lastSnapshotID);
|
Call.onSnapshot(connection.id, diff, connection.lastSnapshotID + 1);
|
||||||
|
//increment snapshot ID
|
||||||
connection.lastSentSnapshot = bytes;
|
connection.lastSentSnapshotID ++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -256,10 +277,10 @@ public class NetServer extends Module{
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Remote(server = false)
|
@Remote(targets = Loc.client)
|
||||||
public static void connectConfirm(Player player){
|
public static void connectConfirm(Player player){
|
||||||
player.add();
|
player.add();
|
||||||
netCommon.sendMessage("[accent]" + player.name + " has connected.");
|
Call.sendMessage("[accent]" + player.name + " has connected.");
|
||||||
Log.info("&y{0} has connected.", player.name);
|
Log.info("&y{0} has connected.", player.name);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -30,7 +30,9 @@ import io.anuke.ucore.graphics.Fill;
|
|||||||
import io.anuke.ucore.graphics.Lines;
|
import io.anuke.ucore.graphics.Lines;
|
||||||
import io.anuke.ucore.util.*;
|
import io.anuke.ucore.util.*;
|
||||||
|
|
||||||
import java.io.*;
|
import java.io.DataInput;
|
||||||
|
import java.io.DataOutput;
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
import static io.anuke.mindustry.Vars.*;
|
import static io.anuke.mindustry.Vars.*;
|
||||||
|
|
||||||
@@ -192,6 +194,7 @@ public class Player extends Unit implements BuilderTrait, CarryTrait {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void removed() {
|
public void removed() {
|
||||||
|
Log.info("\n\nPLAYER REMOVED\n\n");
|
||||||
dropCarry();
|
dropCarry();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -368,7 +371,7 @@ public class Player extends Unit implements BuilderTrait, CarryTrait {
|
|||||||
hitTime = Math.max(0f, hitTime - Timers.delta());
|
hitTime = Math.max(0f, hitTime - Timers.delta());
|
||||||
|
|
||||||
if(!isLocal){
|
if(!isLocal){
|
||||||
interpolate();
|
//interpolate();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -552,9 +555,9 @@ public class Player extends Unit implements BuilderTrait, CarryTrait {
|
|||||||
dead = true;
|
dead = true;
|
||||||
respawning = false;
|
respawning = false;
|
||||||
trail.clear();
|
trail.clear();
|
||||||
|
health = maxHealth();
|
||||||
|
|
||||||
add();
|
add();
|
||||||
heal();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean isShooting(){
|
public boolean isShooting(){
|
||||||
@@ -601,7 +604,6 @@ public class Player extends Unit implements BuilderTrait, CarryTrait {
|
|||||||
|
|
||||||
private void readSaveSuper(DataInput stream) throws IOException {
|
private void readSaveSuper(DataInput stream) throws IOException {
|
||||||
super.readSave(stream);
|
super.readSave(stream);
|
||||||
|
|
||||||
byte uamount = stream.readByte();
|
byte uamount = stream.readByte();
|
||||||
for (int i = 0; i < uamount; i++) {
|
for (int i = 0; i < uamount; i++) {
|
||||||
upgrades.add(Upgrade.getByID(stream.readByte()));
|
upgrades.add(Upgrade.getByID(stream.readByte()));
|
||||||
@@ -611,13 +613,23 @@ public class Player extends Unit implements BuilderTrait, CarryTrait {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void write(DataOutput buffer) {
|
public void write(DataOutput buffer) throws IOException {
|
||||||
//todo
|
super.writeSave(buffer);
|
||||||
|
buffer.writeUTF(name);
|
||||||
|
buffer.writeInt(Color.rgba8888(color));
|
||||||
|
buffer.writeBoolean(dead);
|
||||||
|
buffer.writeByte(weapon.id);
|
||||||
|
buffer.writeByte(mech.id);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void read(DataInput buffer, long time) {
|
public void read(DataInput buffer, long time) throws IOException {
|
||||||
//todo
|
super.readSave(buffer);
|
||||||
|
name = buffer.readUTF();
|
||||||
|
color.set(buffer.readInt());
|
||||||
|
dead = buffer.readBoolean();
|
||||||
|
weapon = Upgrade.getByID(buffer.readByte());
|
||||||
|
mech = Upgrade.getByID(buffer.readByte());
|
||||||
}
|
}
|
||||||
|
|
||||||
//endregion
|
//endregion
|
||||||
|
|||||||
@@ -63,7 +63,9 @@ public abstract class Unit extends DestructibleEntity implements SaveTrait, Targ
|
|||||||
|
|
||||||
x = interpolator.pos.x;
|
x = interpolator.pos.x;
|
||||||
y = interpolator.pos.y;
|
y = interpolator.pos.y;
|
||||||
rotation = interpolator.values[0];
|
if(interpolator.values.length > 0){
|
||||||
|
rotation = interpolator.values[0];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -100,6 +102,7 @@ public abstract class Unit extends DestructibleEntity implements SaveTrait, Targ
|
|||||||
stream.writeByte(team.ordinal());
|
stream.writeByte(team.ordinal());
|
||||||
stream.writeFloat(x);
|
stream.writeFloat(x);
|
||||||
stream.writeFloat(y);
|
stream.writeFloat(y);
|
||||||
|
stream.writeFloat(rotation);
|
||||||
stream.writeShort((short)health);
|
stream.writeShort((short)health);
|
||||||
stream.writeByte(status.current().id);
|
stream.writeByte(status.current().id);
|
||||||
stream.writeFloat(status.getTime());
|
stream.writeFloat(status.getTime());
|
||||||
@@ -111,6 +114,7 @@ public abstract class Unit extends DestructibleEntity implements SaveTrait, Targ
|
|||||||
byte team = stream.readByte();
|
byte team = stream.readByte();
|
||||||
float x = stream.readFloat();
|
float x = stream.readFloat();
|
||||||
float y = stream.readFloat();
|
float y = stream.readFloat();
|
||||||
|
float rotation = stream.readFloat();
|
||||||
int health = stream.readShort();
|
int health = stream.readShort();
|
||||||
byte effect = stream.readByte();
|
byte effect = stream.readByte();
|
||||||
float etime = stream.readFloat();
|
float etime = stream.readFloat();
|
||||||
@@ -120,6 +124,7 @@ public abstract class Unit extends DestructibleEntity implements SaveTrait, Targ
|
|||||||
this.health = health;
|
this.health = health;
|
||||||
this.x = x;
|
this.x = x;
|
||||||
this.y = y;
|
this.y = y;
|
||||||
|
this.rotation = rotation;
|
||||||
this.status.set(StatusEffect.getByID(effect), etime);
|
this.status.set(StatusEffect.getByID(effect), etime);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -51,9 +51,7 @@ public class BasicBulletType extends BulletType {
|
|||||||
for (int i = 0; i < fragBullets; i++) {
|
for (int i = 0; i < fragBullets; i++) {
|
||||||
float len = Mathf.random(1f, 7f);
|
float len = Mathf.random(1f, 7f);
|
||||||
float a = Mathf.random(360f);
|
float a = Mathf.random(360f);
|
||||||
Bullet bullet = Bullet.create(fragBullet, b,
|
Bullet.create(fragBullet, b, x + Angles.trnsx(a, len), y + Angles.trnsy(a, len), a, Mathf.random(fragVelocityMin, fragVelocityMax));
|
||||||
x + Angles.trnsx(a, len), y + Angles.trnsy(a, len), a);
|
|
||||||
bullet.getVelocity().scl(Mathf.random(fragVelocityMin, fragVelocityMax));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,10 +3,8 @@ package io.anuke.mindustry.entities.bullet;
|
|||||||
import com.badlogic.gdx.math.Vector2;
|
import com.badlogic.gdx.math.Vector2;
|
||||||
import com.badlogic.gdx.utils.Pools;
|
import com.badlogic.gdx.utils.Pools;
|
||||||
import io.anuke.mindustry.entities.Unit;
|
import io.anuke.mindustry.entities.Unit;
|
||||||
import io.anuke.mindustry.entities.traits.SyncTrait;
|
|
||||||
import io.anuke.mindustry.entities.traits.TeamTrait;
|
import io.anuke.mindustry.entities.traits.TeamTrait;
|
||||||
import io.anuke.mindustry.game.Team;
|
import io.anuke.mindustry.game.Team;
|
||||||
import io.anuke.mindustry.net.Interpolator;
|
|
||||||
import io.anuke.mindustry.world.Tile;
|
import io.anuke.mindustry.world.Tile;
|
||||||
import io.anuke.ucore.entities.EntityGroup;
|
import io.anuke.ucore.entities.EntityGroup;
|
||||||
import io.anuke.ucore.entities.impl.BulletEntity;
|
import io.anuke.ucore.entities.impl.BulletEntity;
|
||||||
@@ -15,31 +13,31 @@ import io.anuke.ucore.entities.trait.SolidTrait;
|
|||||||
import io.anuke.ucore.entities.trait.VelocityTrait;
|
import io.anuke.ucore.entities.trait.VelocityTrait;
|
||||||
import io.anuke.ucore.util.Timer;
|
import io.anuke.ucore.util.Timer;
|
||||||
|
|
||||||
import java.io.DataInput;
|
|
||||||
import java.io.DataOutput;
|
|
||||||
import java.io.IOException;
|
|
||||||
|
|
||||||
import static io.anuke.mindustry.Vars.bulletGroup;
|
import static io.anuke.mindustry.Vars.bulletGroup;
|
||||||
import static io.anuke.mindustry.Vars.world;
|
import static io.anuke.mindustry.Vars.world;
|
||||||
|
|
||||||
public class Bullet extends BulletEntity<BulletType> implements TeamTrait, SyncTrait{
|
public class Bullet extends BulletEntity<BulletType> implements TeamTrait{
|
||||||
private static Vector2 vector = new Vector2();
|
private static Vector2 vector = new Vector2();
|
||||||
|
|
||||||
private Interpolator interpolator = new Interpolator();
|
//private Interpolator interpolator = new Interpolator();
|
||||||
private Team team;
|
private Team team;
|
||||||
|
|
||||||
public Timer timer = new Timer(3);
|
public Timer timer = new Timer(3);
|
||||||
|
|
||||||
public static Bullet create(BulletType type, TeamTrait owner, float x, float y, float angle){
|
public static void create (BulletType type, TeamTrait owner, float x, float y, float angle){
|
||||||
return create(type, owner, owner.getTeam(), x, y, angle);
|
create(type, owner, owner.getTeam(), x, y, angle);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static Bullet create (BulletType type, Entity owner, Team team, float x, float y, float angle){
|
public static void create (BulletType type, Entity owner, Team team, float x, float y, float angle){
|
||||||
|
create(type, owner, team, x, y, angle, 1f);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void create (BulletType type, Entity owner, Team team, float x, float y, float angle, float velocityScl){
|
||||||
Bullet bullet = Pools.obtain(Bullet.class);
|
Bullet bullet = Pools.obtain(Bullet.class);
|
||||||
bullet.type = type;
|
bullet.type = type;
|
||||||
bullet.owner = owner;
|
bullet.owner = owner;
|
||||||
|
|
||||||
bullet.velocity.set(0, type.speed).setAngle(angle);
|
bullet.velocity.set(0, type.speed).setAngle(angle).scl(velocityScl);
|
||||||
bullet.velocity.add(owner instanceof VelocityTrait ? ((VelocityTrait)owner).getVelocity() : Vector2.Zero);
|
bullet.velocity.add(owner instanceof VelocityTrait ? ((VelocityTrait)owner).getVelocity() : Vector2.Zero);
|
||||||
bullet.hitbox.setSize(type.hitsize);
|
bullet.hitbox.setSize(type.hitsize);
|
||||||
|
|
||||||
@@ -47,11 +45,14 @@ public class Bullet extends BulletEntity<BulletType> implements TeamTrait, SyncT
|
|||||||
bullet.type = type;
|
bullet.type = type;
|
||||||
bullet.set(x, y);
|
bullet.set(x, y);
|
||||||
bullet.add();
|
bullet.add();
|
||||||
return bullet;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static Bullet create(BulletType type, Bullet parent, float x, float y, float angle){
|
public static void create(BulletType type, Bullet parent, float x, float y, float angle){
|
||||||
return create(type, parent.owner, parent.team, x, y, angle);
|
create(type, parent.owner, parent.team, x, y, angle);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void create(BulletType type, Bullet parent, float x, float y, float angle, float velocityScl){
|
||||||
|
create(type, parent.owner, parent.team, x, y, angle, velocityScl);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**Internal use only!*/
|
/**Internal use only!*/
|
||||||
@@ -60,7 +61,7 @@ public class Bullet extends BulletEntity<BulletType> implements TeamTrait, SyncT
|
|||||||
public boolean collidesTiles(){
|
public boolean collidesTiles(){
|
||||||
return true; //TODO make artillery and such not do this
|
return true; //TODO make artillery and such not do this
|
||||||
}
|
}
|
||||||
|
/*
|
||||||
@Override
|
@Override
|
||||||
public boolean doSync(){
|
public boolean doSync(){
|
||||||
return type.syncable;
|
return type.syncable;
|
||||||
@@ -85,7 +86,7 @@ public class Bullet extends BulletEntity<BulletType> implements TeamTrait, SyncT
|
|||||||
y = data.readFloat();
|
y = data.readFloat();
|
||||||
team = Team.values()[data.readByte()];
|
team = Team.values()[data.readByte()];
|
||||||
type = BulletType.getByID(data.readByte());
|
type = BulletType.getByID(data.readByte());
|
||||||
}
|
}*/
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Team getTeam() {
|
public Team getTeam() {
|
||||||
|
|||||||
@@ -1,19 +1,18 @@
|
|||||||
package io.anuke.mindustry.entities.units;
|
package io.anuke.mindustry.entities.units;
|
||||||
|
|
||||||
import io.anuke.mindustry.content.fx.ExplosionFx;
|
import io.anuke.mindustry.content.fx.ExplosionFx;
|
||||||
import io.anuke.mindustry.entities.traits.TargetTrait;
|
|
||||||
import io.anuke.mindustry.entities.TileEntity;
|
import io.anuke.mindustry.entities.TileEntity;
|
||||||
import io.anuke.mindustry.entities.Unit;
|
import io.anuke.mindustry.entities.Unit;
|
||||||
import io.anuke.mindustry.entities.Units;
|
import io.anuke.mindustry.entities.Units;
|
||||||
import io.anuke.mindustry.entities.bullet.Bullet;
|
import io.anuke.mindustry.entities.bullet.Bullet;
|
||||||
|
import io.anuke.mindustry.entities.traits.TargetTrait;
|
||||||
import io.anuke.mindustry.game.Team;
|
import io.anuke.mindustry.game.Team;
|
||||||
import io.anuke.mindustry.net.Net;
|
import io.anuke.mindustry.net.Net;
|
||||||
import io.anuke.mindustry.type.AmmoType;
|
import io.anuke.mindustry.type.AmmoType;
|
||||||
import io.anuke.mindustry.type.Item;
|
import io.anuke.mindustry.type.Item;
|
||||||
import io.anuke.mindustry.world.meta.BlockFlag;
|
|
||||||
import io.anuke.mindustry.world.Tile;
|
import io.anuke.mindustry.world.Tile;
|
||||||
|
import io.anuke.mindustry.world.meta.BlockFlag;
|
||||||
import io.anuke.ucore.core.Effects;
|
import io.anuke.ucore.core.Effects;
|
||||||
import io.anuke.ucore.core.Effects.Effect;
|
|
||||||
import io.anuke.ucore.core.Timers;
|
import io.anuke.ucore.core.Timers;
|
||||||
import io.anuke.ucore.entities.EntityGroup;
|
import io.anuke.ucore.entities.EntityGroup;
|
||||||
import io.anuke.ucore.util.Angles;
|
import io.anuke.ucore.util.Angles;
|
||||||
@@ -21,7 +20,9 @@ import io.anuke.ucore.util.Geometry;
|
|||||||
import io.anuke.ucore.util.Mathf;
|
import io.anuke.ucore.util.Mathf;
|
||||||
import io.anuke.ucore.util.Timer;
|
import io.anuke.ucore.util.Timer;
|
||||||
|
|
||||||
import java.io.*;
|
import java.io.DataInput;
|
||||||
|
import java.io.DataOutput;
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
import static io.anuke.mindustry.Vars.*;
|
import static io.anuke.mindustry.Vars.*;
|
||||||
|
|
||||||
@@ -55,12 +56,6 @@ public abstract class BaseUnit extends Unit{
|
|||||||
rotation = Mathf.slerpDelta(rotation, angle, type.rotatespeed);
|
rotation = Mathf.slerpDelta(rotation, angle, type.rotatespeed);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void effectAt(Effect effect, float rotation, float dx, float dy){
|
|
||||||
Effects.effect(effect,
|
|
||||||
x + Angles.trnsx(rotation, dx, dy),
|
|
||||||
y + Angles.trnsy(rotation, dx, dy), Mathf.atan2(dx, dy) + rotation);
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean targetHasFlag(BlockFlag flag){
|
public boolean targetHasFlag(BlockFlag flag){
|
||||||
return target instanceof TileEntity &&
|
return target instanceof TileEntity &&
|
||||||
((TileEntity)target).tile.block().flags.contains(flag);
|
((TileEntity)target).tile.block().flags.contains(flag);
|
||||||
@@ -248,12 +243,13 @@ public abstract class BaseUnit extends Unit{
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void write(DataOutput data) {
|
public void write(DataOutput data) throws IOException{
|
||||||
//todo
|
writeSave(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void read(DataInput data, long time) {
|
public void read(DataInput data, long time) throws IOException{
|
||||||
//todo
|
super.readSave(data);
|
||||||
|
this.type = UnitType.getByID(data.readByte());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,6 +3,9 @@ package io.anuke.mindustry.io;
|
|||||||
import io.anuke.annotations.Annotations.ReadClass;
|
import io.anuke.annotations.Annotations.ReadClass;
|
||||||
import io.anuke.annotations.Annotations.WriteClass;
|
import io.anuke.annotations.Annotations.WriteClass;
|
||||||
import io.anuke.mindustry.entities.Player;
|
import io.anuke.mindustry.entities.Player;
|
||||||
|
import io.anuke.mindustry.net.Packets.KickReason;
|
||||||
|
import io.anuke.mindustry.type.Upgrade;
|
||||||
|
import io.anuke.mindustry.type.Weapon;
|
||||||
import io.anuke.mindustry.world.Tile;
|
import io.anuke.mindustry.world.Tile;
|
||||||
|
|
||||||
|
|
||||||
@@ -34,19 +37,47 @@ public class TypeIO {
|
|||||||
return world.tile(buffer.getInt());
|
return world.tile(buffer.getInt());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@WriteClass(KickReason.class)
|
||||||
|
public static void writeKick(ByteBuffer buffer, KickReason reason){
|
||||||
|
buffer.put((byte)reason.ordinal());
|
||||||
|
}
|
||||||
|
|
||||||
|
@ReadClass(KickReason.class)
|
||||||
|
public static KickReason readKick(ByteBuffer buffer){
|
||||||
|
return KickReason.values()[buffer.get()];
|
||||||
|
}
|
||||||
|
|
||||||
|
@WriteClass(Weapon.class)
|
||||||
|
public static void writeWeapon(ByteBuffer buffer, Weapon weapon){
|
||||||
|
buffer.put(weapon.id);
|
||||||
|
}
|
||||||
|
|
||||||
|
@ReadClass(Weapon.class)
|
||||||
|
public static Weapon readWeapon(ByteBuffer buffer){
|
||||||
|
return Upgrade.getByID(buffer.get());
|
||||||
|
}
|
||||||
|
|
||||||
@WriteClass(String.class)
|
@WriteClass(String.class)
|
||||||
public static void writeString(ByteBuffer buffer, String string){
|
public static void writeString(ByteBuffer buffer, String string){
|
||||||
byte[] bytes = string.getBytes();
|
if(string != null) {
|
||||||
buffer.putShort((short)bytes.length);
|
byte[] bytes = string.getBytes();
|
||||||
buffer.put(bytes);
|
buffer.putShort((short) bytes.length);
|
||||||
|
buffer.put(bytes);
|
||||||
|
}else{
|
||||||
|
buffer.putShort((short)-1);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ReadClass(String.class)
|
@ReadClass(String.class)
|
||||||
public static String readString(ByteBuffer buffer){
|
public static String readString(ByteBuffer buffer){
|
||||||
short length = buffer.getShort();
|
short length = buffer.getShort();
|
||||||
byte[] bytes = new byte[length];
|
if(length != -1) {
|
||||||
buffer.get(bytes);
|
byte[] bytes = new byte[length];
|
||||||
return new String(bytes);
|
buffer.get(bytes);
|
||||||
|
return new String(bytes);
|
||||||
|
}else{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@WriteClass(byte[].class)
|
@WriteClass(byte[].class)
|
||||||
|
|||||||
8
core/src/io/anuke/mindustry/net/In.java
Normal file
8
core/src/io/anuke/mindustry/net/In.java
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
package io.anuke.mindustry.net;
|
||||||
|
|
||||||
|
/**Stores class nameas for remote method invocation for consistency's sake.*/
|
||||||
|
public class In {
|
||||||
|
public static final String normal = "Call";
|
||||||
|
public static final String entities = "CallEntity";
|
||||||
|
public static final String blocks = "CallBlocks";
|
||||||
|
}
|
||||||
@@ -10,7 +10,7 @@ public class Interpolator {
|
|||||||
//used for movement
|
//used for movement
|
||||||
public Vector2 target = new Vector2();
|
public Vector2 target = new Vector2();
|
||||||
public Vector2 last = new Vector2();
|
public Vector2 last = new Vector2();
|
||||||
public float[] targets;
|
public float[] targets = {};
|
||||||
public float spacing = 1f;
|
public float spacing = 1f;
|
||||||
public float time;
|
public float time;
|
||||||
|
|
||||||
|
|||||||
@@ -56,7 +56,6 @@ public class Net{
|
|||||||
for(int i = 0; i < packetQueue.size; i ++){
|
for(int i = 0; i < packetQueue.size; i ++){
|
||||||
Log.info("Processing {0} packet post-load.", ClassReflection.getSimpleName(packetQueue.get(i).getClass()));
|
Log.info("Processing {0} packet post-load.", ClassReflection.getSimpleName(packetQueue.get(i).getClass()));
|
||||||
handleClientReceived(packetQueue.get(i));
|
handleClientReceived(packetQueue.get(i));
|
||||||
Pools.free(packetQueue.get(i));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
//clear inbound packet queue
|
//clear inbound packet queue
|
||||||
@@ -180,6 +179,7 @@ public class Net{
|
|||||||
}
|
}
|
||||||
}else if(clientListeners.get(object.getClass()) != null ||
|
}else if(clientListeners.get(object.getClass()) != null ||
|
||||||
listeners.get(object.getClass()) != null){
|
listeners.get(object.getClass()) != null){
|
||||||
|
|
||||||
if(clientLoaded || object instanceof ImportantPacket){
|
if(clientLoaded || object instanceof ImportantPacket){
|
||||||
if(clientListeners.get(object.getClass()) != null) clientListeners.get(object.getClass()).accept(object);
|
if(clientListeners.get(object.getClass()) != null) clientListeners.get(object.getClass()).accept(object);
|
||||||
if(listeners.get(object.getClass()) != null) listeners.get(object.getClass()).accept(object);
|
if(listeners.get(object.getClass()) != null) listeners.get(object.getClass()).accept(object);
|
||||||
|
|||||||
@@ -1,5 +1,32 @@
|
|||||||
package io.anuke.mindustry.net;
|
package io.anuke.mindustry.net;
|
||||||
|
|
||||||
|
import io.anuke.annotations.Annotations.Loc;
|
||||||
|
import io.anuke.annotations.Annotations.Remote;
|
||||||
|
import io.anuke.annotations.Annotations.Variant;
|
||||||
|
import io.anuke.mindustry.Vars;
|
||||||
|
import io.anuke.mindustry.entities.Player;
|
||||||
|
|
||||||
|
import static io.anuke.mindustry.Vars.playerGroup;
|
||||||
|
|
||||||
public class NetEvents {
|
public class NetEvents {
|
||||||
|
|
||||||
|
@Remote(called = Loc.both, targets = Loc.both)
|
||||||
|
public static void sendMessage(Player player, String message){
|
||||||
|
if(Vars.ui != null){
|
||||||
|
Vars.ui.chatfrag.addMessage(message, player == null ? null : colorizeName(player.id, player.name));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Remote(called = Loc.both, variants = Variant.both)
|
||||||
|
public static void sendMessage(String message){
|
||||||
|
if(Vars.ui != null){
|
||||||
|
Vars.ui.chatfrag.addMessage(message, null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static String colorizeName(int id, String name){
|
||||||
|
Player player = playerGroup.getByID(id);
|
||||||
|
if(name == null || player == null) return null;
|
||||||
|
return "[#" + player.color.toString().toUpperCase() + "]" + name;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,15 +2,15 @@ package io.anuke.mindustry.net;
|
|||||||
|
|
||||||
import com.badlogic.gdx.utils.ObjectMap;
|
import com.badlogic.gdx.utils.ObjectMap;
|
||||||
import com.badlogic.gdx.utils.TimeUtils;
|
import com.badlogic.gdx.utils.TimeUtils;
|
||||||
import io.anuke.mindustry.content.Weapons;
|
|
||||||
import io.anuke.mindustry.content.blocks.Blocks;
|
import io.anuke.mindustry.content.blocks.Blocks;
|
||||||
import io.anuke.mindustry.entities.Player;
|
import io.anuke.mindustry.entities.Player;
|
||||||
import io.anuke.mindustry.game.GameMode;
|
import io.anuke.mindustry.game.GameMode;
|
||||||
import io.anuke.mindustry.game.Team;
|
import io.anuke.mindustry.game.Team;
|
||||||
|
import io.anuke.mindustry.game.TeamInfo;
|
||||||
|
import io.anuke.mindustry.game.TeamInfo.TeamData;
|
||||||
import io.anuke.mindustry.io.Map;
|
import io.anuke.mindustry.io.Map;
|
||||||
import io.anuke.mindustry.io.MapMeta;
|
import io.anuke.mindustry.io.MapMeta;
|
||||||
import io.anuke.mindustry.io.Version;
|
import io.anuke.mindustry.io.Version;
|
||||||
import io.anuke.mindustry.type.Upgrade;
|
|
||||||
import io.anuke.mindustry.world.Tile;
|
import io.anuke.mindustry.world.Tile;
|
||||||
import io.anuke.mindustry.world.blocks.BlockPart;
|
import io.anuke.mindustry.world.blocks.BlockPart;
|
||||||
import io.anuke.ucore.core.Timers;
|
import io.anuke.ucore.core.Timers;
|
||||||
@@ -37,16 +37,11 @@ public class NetworkIO {
|
|||||||
|
|
||||||
stream.writeInt(state.wave); //wave
|
stream.writeInt(state.wave); //wave
|
||||||
stream.writeFloat(state.wavetime); //wave countdown
|
stream.writeFloat(state.wavetime); //wave countdown
|
||||||
stream.writeInt(state.enemies); //enemy amount
|
|
||||||
|
|
||||||
stream.writeBoolean(state.friendlyFire); //friendly fire state
|
stream.writeBoolean(state.friendlyFire); //friendly fire state
|
||||||
stream.writeInt(player.id); //player remap ID
|
|
||||||
stream.writeBoolean(player.isAdmin);
|
|
||||||
|
|
||||||
stream.writeByte(player.upgrades.size);
|
stream.writeInt(player.id);
|
||||||
for(Upgrade u : player.upgrades){
|
player.write(stream);
|
||||||
stream.writeByte(u.id);
|
|
||||||
}
|
|
||||||
|
|
||||||
//--MAP DATA--
|
//--MAP DATA--
|
||||||
|
|
||||||
@@ -80,6 +75,17 @@ public class NetworkIO {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//write team data
|
||||||
|
stream.writeByte(state.teams.getTeams().size);
|
||||||
|
for(TeamData data : state.teams.getTeams()){
|
||||||
|
stream.writeByte(data.team.ordinal());
|
||||||
|
stream.writeBoolean(data.ally);
|
||||||
|
stream.writeShort(data.cores.size);
|
||||||
|
for(Tile tile : data.cores){
|
||||||
|
stream.writeInt(tile.packedPosition());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}catch (IOException e){
|
}catch (IOException e){
|
||||||
throw new RuntimeException(e);
|
throw new RuntimeException(e);
|
||||||
}
|
}
|
||||||
@@ -105,29 +111,18 @@ public class NetworkIO {
|
|||||||
|
|
||||||
int wave = stream.readInt();
|
int wave = stream.readInt();
|
||||||
float wavetime = stream.readFloat();
|
float wavetime = stream.readFloat();
|
||||||
int enemies = stream.readInt();
|
|
||||||
boolean friendlyfire = stream.readBoolean();
|
boolean friendlyfire = stream.readBoolean();
|
||||||
|
|
||||||
state.enemies = enemies;
|
|
||||||
state.wave = wave;
|
state.wave = wave;
|
||||||
state.wavetime = wavetime;
|
state.wavetime = wavetime;
|
||||||
state.mode = GameMode.values()[mode];
|
state.mode = GameMode.values()[mode];
|
||||||
state.friendlyFire = friendlyfire;
|
state.friendlyFire = friendlyfire;
|
||||||
|
|
||||||
int pid = stream.readInt();
|
|
||||||
boolean admin = stream.readBoolean();
|
|
||||||
|
|
||||||
byte weapons = stream.readByte();
|
|
||||||
|
|
||||||
for(int i = 0; i < weapons; i ++){
|
|
||||||
player.upgrades.add(Upgrade.getByID(stream.readByte()));
|
|
||||||
}
|
|
||||||
|
|
||||||
player.weapon = Weapons.blaster;
|
|
||||||
|
|
||||||
Entities.clear();
|
Entities.clear();
|
||||||
player.id = pid;
|
int id = stream.readInt();
|
||||||
player.isAdmin = admin;
|
player.read(stream, TimeUtils.millis());
|
||||||
|
player.resetID(id);
|
||||||
player.add();
|
player.add();
|
||||||
|
|
||||||
world.beginMapLoad();
|
world.beginMapLoad();
|
||||||
@@ -174,7 +169,20 @@ public class NetworkIO {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
player.dead = true;
|
player.reset();
|
||||||
|
state.teams = new TeamInfo();
|
||||||
|
|
||||||
|
byte teams = stream.readByte();
|
||||||
|
for (int i = 0; i < teams; i++) {
|
||||||
|
Team team = Team.values()[stream.readByte()];
|
||||||
|
boolean ally = stream.readBoolean();
|
||||||
|
short cores = stream.readShort();
|
||||||
|
state.teams.add(team, ally);
|
||||||
|
|
||||||
|
for (int j = 0; j < cores; j++) {
|
||||||
|
state.teams.get(team).cores.add(world.tile(stream.readInt()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
world.endMapLoad();
|
world.endMapLoad();
|
||||||
|
|
||||||
|
|||||||
@@ -3,14 +3,14 @@ package io.anuke.mindustry.net;
|
|||||||
import com.badlogic.gdx.utils.TimeUtils;
|
import com.badlogic.gdx.utils.TimeUtils;
|
||||||
import io.anuke.mindustry.Vars;
|
import io.anuke.mindustry.Vars;
|
||||||
import io.anuke.mindustry.entities.Player;
|
import io.anuke.mindustry.entities.Player;
|
||||||
import io.anuke.mindustry.gen.RemoteReadClient;
|
|
||||||
import io.anuke.mindustry.io.Version;
|
import io.anuke.mindustry.io.Version;
|
||||||
import io.anuke.mindustry.net.Packet.ImportantPacket;
|
import io.anuke.mindustry.net.Packet.ImportantPacket;
|
||||||
import io.anuke.mindustry.net.Packet.UnimportantPacket;
|
import io.anuke.mindustry.net.Packet.UnimportantPacket;
|
||||||
|
import io.anuke.ucore.io.ByteBufferInput;
|
||||||
import io.anuke.ucore.io.ByteBufferOutput;
|
import io.anuke.ucore.io.ByteBufferOutput;
|
||||||
import io.anuke.ucore.io.IOUtils;
|
import io.anuke.ucore.io.IOUtils;
|
||||||
import io.anuke.ucore.io.ByteBufferInput;
|
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
|
|
||||||
/**Class for storing all packets.*/
|
/**Class for storing all packets.*/
|
||||||
@@ -66,19 +66,17 @@ public class Packets {
|
|||||||
@Override
|
@Override
|
||||||
public void read(ByteBuffer buffer) {
|
public void read(ByteBuffer buffer) {
|
||||||
type = buffer.get();
|
type = buffer.get();
|
||||||
|
writeLength = buffer.getShort();
|
||||||
if(Net.client()){
|
byte[] bytes = new byte[writeLength];
|
||||||
RemoteReadClient.readPacket(buffer, type);
|
buffer.get(bytes);
|
||||||
}else{
|
writeBuffer = ByteBuffer.wrap(bytes);
|
||||||
byte[] bytes = new byte[writeLength];
|
|
||||||
buffer.get(bytes);
|
|
||||||
writeBuffer = ByteBuffer.wrap(bytes);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void write(ByteBuffer buffer) {
|
public void write(ByteBuffer buffer) {
|
||||||
buffer.put(type);
|
buffer.put(type);
|
||||||
|
buffer.putShort((short)writeLength);
|
||||||
|
|
||||||
writeBuffer.position(0);
|
writeBuffer.position(0);
|
||||||
for(int i = 0; i < writeLength; i ++){
|
for(int i = 0; i < writeLength; i ++){
|
||||||
buffer.put(writeBuffer.get());
|
buffer.put(writeBuffer.get());
|
||||||
@@ -109,7 +107,11 @@ public class Packets {
|
|||||||
buffer.putInt(lastSnapshot);
|
buffer.putInt(lastSnapshot);
|
||||||
buffer.putInt(player.id);
|
buffer.putInt(player.id);
|
||||||
buffer.putLong(TimeUtils.millis());
|
buffer.putLong(TimeUtils.millis());
|
||||||
player.write(out);
|
try {
|
||||||
|
player.write(out);
|
||||||
|
}catch (IOException e){
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -120,7 +122,11 @@ public class Packets {
|
|||||||
int id = buffer.getInt();
|
int id = buffer.getInt();
|
||||||
long time = buffer.getLong();
|
long time = buffer.getLong();
|
||||||
player = Vars.playerGroup.getByID(id);
|
player = Vars.playerGroup.getByID(id);
|
||||||
player.read(in, time);
|
try {
|
||||||
|
player.read(in, time);
|
||||||
|
}catch (IOException e){
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
13
core/src/io/anuke/mindustry/net/ValidateException.java
Normal file
13
core/src/io/anuke/mindustry/net/ValidateException.java
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
package io.anuke.mindustry.net;
|
||||||
|
|
||||||
|
import io.anuke.mindustry.entities.Player;
|
||||||
|
|
||||||
|
/**Thrown when a client sends invalid information.*/
|
||||||
|
public class ValidateException extends RuntimeException{
|
||||||
|
public final Player player;
|
||||||
|
|
||||||
|
public ValidateException(Player player, String s) {
|
||||||
|
super(s);
|
||||||
|
this.player = player;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -8,7 +8,7 @@ import io.anuke.ucore.core.Effects.Effect;
|
|||||||
|
|
||||||
public class AmmoType implements Content{
|
public class AmmoType implements Content{
|
||||||
private static int lastID = 0;
|
private static int lastID = 0;
|
||||||
private static Array<AmmoType> allTypes = new Array<>();
|
private static Array<AmmoType> allTypes = new Array<>(32);
|
||||||
|
|
||||||
public final byte id;
|
public final byte id;
|
||||||
/**The item used. Always null if liquid isn't.*/
|
/**The item used. Always null if liquid isn't.*/
|
||||||
|
|||||||
@@ -2,10 +2,14 @@ package io.anuke.mindustry.type;
|
|||||||
|
|
||||||
import com.badlogic.gdx.graphics.g2d.TextureRegion;
|
import com.badlogic.gdx.graphics.g2d.TextureRegion;
|
||||||
import com.badlogic.gdx.utils.ObjectMap;
|
import com.badlogic.gdx.utils.ObjectMap;
|
||||||
|
import io.anuke.annotations.Annotations.Remote;
|
||||||
|
import io.anuke.annotations.Annotations.Loc;
|
||||||
import io.anuke.mindustry.content.fx.Fx;
|
import io.anuke.mindustry.content.fx.Fx;
|
||||||
import io.anuke.mindustry.entities.Player;
|
import io.anuke.mindustry.entities.Player;
|
||||||
import io.anuke.mindustry.entities.Unit;
|
import io.anuke.mindustry.entities.Unit;
|
||||||
import io.anuke.mindustry.entities.bullet.Bullet;
|
import io.anuke.mindustry.entities.bullet.Bullet;
|
||||||
|
import io.anuke.mindustry.gen.CallEntity;
|
||||||
|
import io.anuke.mindustry.net.In;
|
||||||
import io.anuke.ucore.core.Effects;
|
import io.anuke.ucore.core.Effects;
|
||||||
import io.anuke.ucore.core.Effects.Effect;
|
import io.anuke.ucore.core.Effects.Effect;
|
||||||
import io.anuke.ucore.graphics.Draw;
|
import io.anuke.ucore.graphics.Draw;
|
||||||
@@ -51,7 +55,7 @@ public class Weapon extends Upgrade {
|
|||||||
|
|
||||||
public void update(Player p, boolean left, float pointerX, float pointerY){
|
public void update(Player p, boolean left, float pointerX, float pointerY){
|
||||||
int t = left ? Player.timerShootLeft : Player.timerShootRight;
|
int t = left ? Player.timerShootLeft : Player.timerShootRight;
|
||||||
int t2 = !left ? Player.timerShootRight : Player.timerShootLeft;
|
int t2 = !left ? Player.timerShootLeft : Player.timerShootRight;
|
||||||
if(p.inventory.hasAmmo() && p.timer.get(t, reload)){
|
if(p.inventory.hasAmmo() && p.timer.get(t, reload)){
|
||||||
if(roundrobin){
|
if(roundrobin){
|
||||||
p.timer.reset(t2, reload/2f);
|
p.timer.reset(t2, reload/2f);
|
||||||
@@ -78,8 +82,7 @@ public class Weapon extends Upgrade {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public void shoot(Player p, float x, float y, float angle, boolean left){
|
public void shoot(Player p, float x, float y, float angle, boolean left){
|
||||||
shootInternal(p, x, y, angle, left);
|
CallEntity.onShootWeapon(p, this, x, y, angle, left);
|
||||||
|
|
||||||
p.inventory.useAmmo();
|
p.inventory.useAmmo();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -92,26 +95,30 @@ public class Weapon extends Upgrade {
|
|||||||
ammoMap.put(type.item, type);
|
ammoMap.put(type.item, type);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void shootInternal(Player p, float x, float y, float rotation, boolean left){
|
|
||||||
Angles.shotgun(shots, spacing, rotation, f -> bullet(p, x, y, f + Mathf.range(inaccuracy)));
|
|
||||||
|
|
||||||
AmmoType type = p.inventory.getAmmo();
|
|
||||||
|
|
||||||
tr.trns(rotation + 180f, type.recoil);
|
|
||||||
|
|
||||||
p.getVelocity().add(tr);
|
|
||||||
|
|
||||||
tr.trns(rotation, 3f);
|
|
||||||
|
|
||||||
Effects.shake(shake, shake, x, y);
|
|
||||||
Effects.effect(ejectEffect, x, y, rotation * -Mathf.sign(left));
|
|
||||||
Effects.effect(type.shootEffect, x + tr.x, y + tr.y, rotation, p);
|
|
||||||
Effects.effect(type.smokeEffect, x + tr.x, y + tr.y, rotation, p);
|
|
||||||
}
|
|
||||||
|
|
||||||
void bullet(Unit owner, float x, float y, float angle){
|
void bullet(Unit owner, float x, float y, float angle){
|
||||||
tr.trns(angle, 3f);
|
tr.trns(angle, 3f);
|
||||||
Bullet.create(owner.inventory.getAmmo().bullet, owner, x + tr.x, y + tr.y, angle);
|
Bullet.create(owner.inventory.getAmmo().bullet, owner, x + tr.x, y + tr.y, angle);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Remote(targets = Loc.both, called = Loc.both, in = In.entities, forward = true)
|
||||||
|
public static void onShootWeapon(Player player, Weapon weapon, float x, float y, float rotation, boolean left){
|
||||||
|
Angles.shotgun(weapon.shots, weapon.spacing, rotation, f -> weapon.bullet(player, x, y, f + Mathf.range(weapon.inaccuracy)));
|
||||||
|
|
||||||
|
AmmoType type = player.inventory.getAmmo();
|
||||||
|
|
||||||
|
weapon.tr.trns(rotation + 180f, type.recoil);
|
||||||
|
|
||||||
|
player.getVelocity().add(weapon.tr);
|
||||||
|
|
||||||
|
weapon.tr.trns(rotation, 3f);
|
||||||
|
|
||||||
|
Effects.shake(weapon.shake, weapon.shake, x, y);
|
||||||
|
Effects.effect(weapon.ejectEffect, x, y, rotation * -Mathf.sign(left));
|
||||||
|
Effects.effect(type.shootEffect, x + weapon.tr.x, y + weapon.tr.y, rotation, player);
|
||||||
|
Effects.effect(type.smokeEffect, x + weapon.tr.x, y + weapon.tr.y, rotation, player);
|
||||||
|
|
||||||
|
//reset timer for remote players
|
||||||
|
player.timer.getTime(left ? Player.timerShootLeft : Player.timerShootRight);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ import com.badlogic.gdx.utils.Array;
|
|||||||
import io.anuke.mindustry.Vars;
|
import io.anuke.mindustry.Vars;
|
||||||
import io.anuke.mindustry.core.GameState.State;
|
import io.anuke.mindustry.core.GameState.State;
|
||||||
import io.anuke.mindustry.core.Platform;
|
import io.anuke.mindustry.core.Platform;
|
||||||
|
import io.anuke.mindustry.gen.Call;
|
||||||
import io.anuke.mindustry.net.Net;
|
import io.anuke.mindustry.net.Net;
|
||||||
import io.anuke.ucore.core.Core;
|
import io.anuke.ucore.core.Core;
|
||||||
import io.anuke.ucore.core.Inputs;
|
import io.anuke.ucore.core.Inputs;
|
||||||
@@ -21,6 +22,7 @@ import io.anuke.ucore.scene.ui.layout.Table;
|
|||||||
import io.anuke.ucore.scene.ui.layout.Unit;
|
import io.anuke.ucore.scene.ui.layout.Unit;
|
||||||
import io.anuke.ucore.util.Mathf;
|
import io.anuke.ucore.util.Mathf;
|
||||||
|
|
||||||
|
import static io.anuke.mindustry.Vars.players;
|
||||||
import static io.anuke.mindustry.Vars.state;
|
import static io.anuke.mindustry.Vars.state;
|
||||||
import static io.anuke.ucore.core.Core.scene;
|
import static io.anuke.ucore.core.Core.scene;
|
||||||
import static io.anuke.ucore.core.Core.skin;
|
import static io.anuke.ucore.core.Core.skin;
|
||||||
@@ -166,7 +168,8 @@ public class ChatFragment extends Table implements Fragment{
|
|||||||
if(message.replaceAll(" ", "").isEmpty()) return;
|
if(message.replaceAll(" ", "").isEmpty()) return;
|
||||||
|
|
||||||
history.insert(1, message);
|
history.insert(1, message);
|
||||||
//TODO send the message
|
|
||||||
|
Call.sendMessage(players[0], message);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void toggle(){
|
public void toggle(){
|
||||||
|
|||||||
@@ -39,7 +39,7 @@ import static io.anuke.mindustry.Vars.*;
|
|||||||
|
|
||||||
public class Block extends BaseBlock implements UnlockableContent{
|
public class Block extends BaseBlock implements UnlockableContent{
|
||||||
private static int lastid;
|
private static int lastid;
|
||||||
private static Array<Block> blocks = new Array<>();
|
private static Array<Block> blocks = new Array<>(140);
|
||||||
private static ObjectMap<String, Block> map = new ObjectMap<>();
|
private static ObjectMap<String, Block> map = new ObjectMap<>();
|
||||||
|
|
||||||
protected Array<Tile> tempTiles = new Array<>();
|
protected Array<Tile> tempTiles = new Array<>();
|
||||||
|
|||||||
@@ -189,6 +189,7 @@ public class Tile implements PosTrait, TargetTrait {
|
|||||||
return isLinked() || !((floor.solid && (block == Blocks.air || block.solidifes)) || (block.solid && (!block.destructible && !block.update)));
|
return isLinked() || !((floor.solid && (block == Blocks.air || block.solidifes)) || (block.solid && (!block.destructible && !block.update)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**Whether this block was placed by a player/unit.*/
|
||||||
public boolean synthetic(){
|
public boolean synthetic(){
|
||||||
Block block = block();
|
Block block = block();
|
||||||
return block.update || block.destructible;
|
return block.update || block.destructible;
|
||||||
@@ -197,7 +198,8 @@ public class Tile implements PosTrait, TargetTrait {
|
|||||||
public boolean solid(){
|
public boolean solid(){
|
||||||
Block block = block();
|
Block block = block();
|
||||||
Block floor = floor();
|
Block floor = floor();
|
||||||
return block.solid || (floor.solid && (block == Blocks.air || block.solidifes)) || block.isSolidFor(this);
|
return block.solid || (floor.solid && (block == Blocks.air || block.solidifes)) || block.isSolidFor(this)
|
||||||
|
|| (isLinked() && getLinked().block().isSolidFor(getLinked()));
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean breakable(){
|
public boolean breakable(){
|
||||||
|
|||||||
@@ -2,17 +2,24 @@ package io.anuke.mindustry.world.blocks.storage;
|
|||||||
|
|
||||||
import com.badlogic.gdx.graphics.g2d.TextureRegion;
|
import com.badlogic.gdx.graphics.g2d.TextureRegion;
|
||||||
import com.badlogic.gdx.math.Rectangle;
|
import com.badlogic.gdx.math.Rectangle;
|
||||||
|
import io.anuke.annotations.Annotations.Loc;
|
||||||
|
import io.anuke.annotations.Annotations.Remote;
|
||||||
import io.anuke.mindustry.Vars;
|
import io.anuke.mindustry.Vars;
|
||||||
import io.anuke.mindustry.content.fx.Fx;
|
import io.anuke.mindustry.content.fx.Fx;
|
||||||
import io.anuke.mindustry.entities.*;
|
import io.anuke.mindustry.entities.Player;
|
||||||
|
import io.anuke.mindustry.entities.TileEntity;
|
||||||
|
import io.anuke.mindustry.entities.Unit;
|
||||||
|
import io.anuke.mindustry.entities.Units;
|
||||||
import io.anuke.mindustry.entities.effect.ItemTransfer;
|
import io.anuke.mindustry.entities.effect.ItemTransfer;
|
||||||
|
import io.anuke.mindustry.gen.CallBlocks;
|
||||||
import io.anuke.mindustry.graphics.Palette;
|
import io.anuke.mindustry.graphics.Palette;
|
||||||
import io.anuke.mindustry.graphics.Shaders;
|
import io.anuke.mindustry.graphics.Shaders;
|
||||||
|
import io.anuke.mindustry.net.In;
|
||||||
import io.anuke.mindustry.net.Net;
|
import io.anuke.mindustry.net.Net;
|
||||||
import io.anuke.mindustry.type.Item;
|
import io.anuke.mindustry.type.Item;
|
||||||
import io.anuke.mindustry.type.ItemType;
|
import io.anuke.mindustry.type.ItemType;
|
||||||
import io.anuke.mindustry.world.meta.BlockFlag;
|
|
||||||
import io.anuke.mindustry.world.Tile;
|
import io.anuke.mindustry.world.Tile;
|
||||||
|
import io.anuke.mindustry.world.meta.BlockFlag;
|
||||||
import io.anuke.ucore.core.Effects;
|
import io.anuke.ucore.core.Effects;
|
||||||
import io.anuke.ucore.core.Graphics;
|
import io.anuke.ucore.core.Graphics;
|
||||||
import io.anuke.ucore.core.Timers;
|
import io.anuke.ucore.core.Timers;
|
||||||
@@ -21,6 +28,10 @@ import io.anuke.ucore.graphics.Lines;
|
|||||||
import io.anuke.ucore.util.EnumSet;
|
import io.anuke.ucore.util.EnumSet;
|
||||||
import io.anuke.ucore.util.Mathf;
|
import io.anuke.ucore.util.Mathf;
|
||||||
|
|
||||||
|
import java.io.DataInputStream;
|
||||||
|
import java.io.DataOutputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
import static io.anuke.mindustry.Vars.debug;
|
import static io.anuke.mindustry.Vars.debug;
|
||||||
import static io.anuke.mindustry.Vars.state;
|
import static io.anuke.mindustry.Vars.state;
|
||||||
|
|
||||||
@@ -130,7 +141,7 @@ public class CoreBlock extends StorageBlock {
|
|||||||
CoreEntity entity = tile.entity();
|
CoreEntity entity = tile.entity();
|
||||||
|
|
||||||
if(!entity.solid && !Units.anyEntities(tile)){
|
if(!entity.solid && !Units.anyEntities(tile)){
|
||||||
entity.solid = true;
|
CallBlocks.setCoreSolid(tile, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(entity.currentPlayer != null){
|
if(entity.currentPlayer != null){
|
||||||
@@ -145,8 +156,8 @@ public class CoreBlock extends StorageBlock {
|
|||||||
|
|
||||||
if(entity.progress >= 1f){
|
if(entity.progress >= 1f){
|
||||||
Effects.effect(Fx.spawn, entity);
|
Effects.effect(Fx.spawn, entity);
|
||||||
|
CallBlocks.setCoreSolid(tile, false);
|
||||||
entity.progress = 0;
|
entity.progress = 0;
|
||||||
entity.solid = false;
|
|
||||||
entity.currentPlayer.heal();
|
entity.currentPlayer.heal();
|
||||||
entity.currentPlayer.rotation = 90f;
|
entity.currentPlayer.rotation = 90f;
|
||||||
entity.currentPlayer.baseRotation = 90f;
|
entity.currentPlayer.baseRotation = 90f;
|
||||||
@@ -182,6 +193,12 @@ public class CoreBlock extends StorageBlock {
|
|||||||
return new CoreEntity();
|
return new CoreEntity();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Remote(called = Loc.server, in = In.blocks)
|
||||||
|
public static void setCoreSolid(Tile tile, boolean solid){
|
||||||
|
CoreEntity entity = tile.entity();
|
||||||
|
entity.solid = solid;
|
||||||
|
}
|
||||||
|
|
||||||
public class CoreEntity extends TileEntity{
|
public class CoreEntity extends TileEntity{
|
||||||
Player currentPlayer;
|
Player currentPlayer;
|
||||||
boolean solid = true;
|
boolean solid = true;
|
||||||
@@ -196,5 +213,15 @@ public class CoreBlock extends StorageBlock {
|
|||||||
progress = 0f;
|
progress = 0f;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void write(DataOutputStream stream) throws IOException {
|
||||||
|
stream.writeBoolean(solid);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void read(DataInputStream stream) throws IOException {
|
||||||
|
solid = stream.readBoolean();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,7 +4,6 @@ import com.badlogic.gdx.Gdx;
|
|||||||
import com.badlogic.gdx.utils.Array;
|
import com.badlogic.gdx.utils.Array;
|
||||||
import com.badlogic.gdx.utils.ObjectMap;
|
import com.badlogic.gdx.utils.ObjectMap;
|
||||||
import com.badlogic.gdx.utils.ObjectSet;
|
import com.badlogic.gdx.utils.ObjectSet;
|
||||||
import com.badlogic.gdx.utils.Pools;
|
|
||||||
import com.esotericsoftware.kryonet.*;
|
import com.esotericsoftware.kryonet.*;
|
||||||
import com.esotericsoftware.kryonet.Listener.LagListener;
|
import com.esotericsoftware.kryonet.Listener.LagListener;
|
||||||
import com.esotericsoftware.minlog.Log;
|
import com.esotericsoftware.minlog.Log;
|
||||||
@@ -133,7 +132,6 @@ public class KryoClient implements ClientProvider{
|
|||||||
}else{
|
}else{
|
||||||
client.sendUDP(object);
|
client.sendUDP(object);
|
||||||
}
|
}
|
||||||
Pools.free(object);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ package io.anuke.kryonet;
|
|||||||
import com.badlogic.gdx.Gdx;
|
import com.badlogic.gdx.Gdx;
|
||||||
import com.badlogic.gdx.utils.Array;
|
import com.badlogic.gdx.utils.Array;
|
||||||
import com.badlogic.gdx.utils.Base64Coder;
|
import com.badlogic.gdx.utils.Base64Coder;
|
||||||
import com.badlogic.gdx.utils.Pools;
|
|
||||||
import com.esotericsoftware.kryonet.Connection;
|
import com.esotericsoftware.kryonet.Connection;
|
||||||
import com.esotericsoftware.kryonet.FrameworkMessage;
|
import com.esotericsoftware.kryonet.FrameworkMessage;
|
||||||
import com.esotericsoftware.kryonet.Listener;
|
import com.esotericsoftware.kryonet.Listener;
|
||||||
@@ -367,7 +366,6 @@ public class KryoServer implements ServerProvider {
|
|||||||
Log.info("Connection removed {0}", k);
|
Log.info("Connection removed {0}", k);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Pools.free(object);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|||||||
@@ -28,7 +28,6 @@ public class MindustryServer extends ModuleCore {
|
|||||||
module(logic = new Logic());
|
module(logic = new Logic());
|
||||||
module(world = new World());
|
module(world = new World());
|
||||||
module(netServer = new NetServer());
|
module(netServer = new NetServer());
|
||||||
module(netCommon = new NetCommon());
|
|
||||||
module(new ServerControl(args));
|
module(new ServerControl(args));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -229,7 +229,7 @@ public class ServerControl extends Module {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
netCommon.sendMessage("[GRAY][[Server]:[] " + arg[0]);
|
//netCommon.sendMessage("[GRAY][[Server]:[] " + arg[0]);
|
||||||
info("&lyServer: &lb{0}", arg[0]);
|
info("&lyServer: &lb{0}", arg[0]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user