merge
This commit is contained in:
@@ -124,7 +124,7 @@ save.new = New Save
|
|||||||
save.overwrite = Are you sure you want to overwrite\nthis save slot?
|
save.overwrite = Are you sure you want to overwrite\nthis save slot?
|
||||||
overwrite = Overwrite
|
overwrite = Overwrite
|
||||||
save.none = No saves found!
|
save.none = No saves found!
|
||||||
saveload = [accent]Saving...
|
saveload = Saving...
|
||||||
savefail = Failed to save game!
|
savefail = Failed to save game!
|
||||||
save.delete.confirm = Are you sure you want to delete this save?
|
save.delete.confirm = Are you sure you want to delete this save?
|
||||||
save.delete = Delete
|
save.delete = Delete
|
||||||
|
|||||||
@@ -119,7 +119,7 @@ public class BlockIndexer{
|
|||||||
|
|
||||||
ObjectSet<Tile> set = damagedTiles[team.ordinal()];
|
ObjectSet<Tile> set = damagedTiles[team.ordinal()];
|
||||||
for(Tile tile : set){
|
for(Tile tile : set){
|
||||||
if((tile.entity == null || tile.entity.getTeam() != team || !tile.entity.damaged()) && !(tile.block() instanceof BuildBlock)){
|
if((tile.entity == null || tile.entity.getTeam() != team || !tile.entity.damaged()) || tile.block() instanceof BuildBlock){
|
||||||
returnArray.add(tile);
|
returnArray.add(tile);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -84,7 +84,7 @@ public class Control implements ApplicationListener{
|
|||||||
});
|
});
|
||||||
|
|
||||||
Events.on(PlayEvent.class, event -> {
|
Events.on(PlayEvent.class, event -> {
|
||||||
player.setTeam(defaultTeam);
|
player.setTeam(state.rules.pvp ? netServer.assignTeam(playerGroup.all()) : defaultTeam);
|
||||||
player.setDead(true);
|
player.setDead(true);
|
||||||
player.add();
|
player.add();
|
||||||
|
|
||||||
@@ -237,7 +237,7 @@ public class Control implements ApplicationListener{
|
|||||||
public void playMap(Map map, Rules rules){
|
public void playMap(Map map, Rules rules){
|
||||||
ui.loadAnd(() -> {
|
ui.loadAnd(() -> {
|
||||||
logic.reset();
|
logic.reset();
|
||||||
world.loadMap(map);
|
world.loadMap(map, rules);
|
||||||
state.rules = rules;
|
state.rules = rules;
|
||||||
logic.play();
|
logic.play();
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -40,7 +40,7 @@ import static io.anuke.mindustry.Vars.*;
|
|||||||
|
|
||||||
public class NetServer implements ApplicationListener{
|
public class NetServer implements ApplicationListener{
|
||||||
public final static int maxSnapshotSize = 430;
|
public final static int maxSnapshotSize = 430;
|
||||||
private final static float serverSyncTime = 15, kickDuration = 30 * 1000;
|
private final static float serverSyncTime = 12, kickDuration = 30 * 1000;
|
||||||
private final static Vector2 vector = new Vector2();
|
private final static Vector2 vector = new Vector2();
|
||||||
private final static Rectangle viewport = new Rectangle();
|
private final static Rectangle viewport = new Rectangle();
|
||||||
/** If a player goes away of their server-side coordinates by this distance, they get teleported back. */
|
/** If a player goes away of their server-side coordinates by this distance, they get teleported back. */
|
||||||
|
|||||||
@@ -220,6 +220,10 @@ public class World implements ApplicationListener{
|
|||||||
}
|
}
|
||||||
|
|
||||||
public void loadMap(Map map){
|
public void loadMap(Map map){
|
||||||
|
loadMap(map, new Rules());
|
||||||
|
}
|
||||||
|
|
||||||
|
public void loadMap(Map map, Rules checkRules){
|
||||||
try{
|
try{
|
||||||
SaveIO.load(map.file, new FilterContext(map));
|
SaveIO.load(map.file, new FilterContext(map));
|
||||||
}catch(Exception e){
|
}catch(Exception e){
|
||||||
@@ -238,20 +242,21 @@ public class World implements ApplicationListener{
|
|||||||
invalidMap = false;
|
invalidMap = false;
|
||||||
|
|
||||||
if(!headless){
|
if(!headless){
|
||||||
if(state.teams.get(defaultTeam).cores.size == 0){
|
if(state.teams.get(defaultTeam).cores.size == 0 && !checkRules.pvp){
|
||||||
ui.showError("$map.nospawn");
|
ui.showError("$map.nospawn");
|
||||||
invalidMap = true;
|
invalidMap = true;
|
||||||
}else if(state.rules.pvp){ //pvp maps need two cores to be valid
|
}else if(checkRules.pvp){ //pvp maps need two cores to be valid
|
||||||
invalidMap = true;
|
int teams = 0;
|
||||||
for(Team team : Team.all){
|
for(Team team : Team.all){
|
||||||
if(state.teams.get(team).cores.size != 0 && team != defaultTeam){
|
if(state.teams.get(team).cores.size != 0){
|
||||||
invalidMap = false;
|
teams ++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if(invalidMap){
|
if(teams < 2){
|
||||||
|
invalidMap = true;
|
||||||
ui.showError("$map.nospawn.pvp");
|
ui.showError("$map.nospawn.pvp");
|
||||||
}
|
}
|
||||||
}else if(state.rules.attackMode){ //pvp maps need two cores to be valid
|
}else if(checkRules.attackMode){ //attack maps need two cores to be valid
|
||||||
invalidMap = state.teams.get(waveTeam).cores.isEmpty();
|
invalidMap = state.teams.get(waveTeam).cores.isEmpty();
|
||||||
if(invalidMap){
|
if(invalidMap){
|
||||||
ui.showError("$map.nospawn.attack");
|
ui.showError("$map.nospawn.attack");
|
||||||
|
|||||||
@@ -1,33 +1,28 @@
|
|||||||
package io.anuke.mindustry.entities.traits;
|
package io.anuke.mindustry.entities.traits;
|
||||||
|
|
||||||
import io.anuke.annotations.Annotations.*;
|
import io.anuke.annotations.Annotations.*;
|
||||||
import io.anuke.arc.Core;
|
import io.anuke.arc.*;
|
||||||
import io.anuke.arc.Events;
|
|
||||||
import io.anuke.arc.collection.Array;
|
|
||||||
import io.anuke.arc.collection.Queue;
|
import io.anuke.arc.collection.Queue;
|
||||||
|
import io.anuke.arc.collection.*;
|
||||||
import io.anuke.arc.graphics.g2d.*;
|
import io.anuke.arc.graphics.g2d.*;
|
||||||
import io.anuke.arc.math.Angles;
|
import io.anuke.arc.math.*;
|
||||||
import io.anuke.arc.math.Mathf;
|
import io.anuke.arc.math.geom.*;
|
||||||
import io.anuke.arc.math.geom.Vector2;
|
import io.anuke.arc.util.*;
|
||||||
import io.anuke.arc.util.Time;
|
import io.anuke.mindustry.*;
|
||||||
import io.anuke.mindustry.Vars;
|
import io.anuke.mindustry.content.*;
|
||||||
import io.anuke.mindustry.content.Blocks;
|
import io.anuke.mindustry.entities.type.*;
|
||||||
import io.anuke.mindustry.entities.type.TileEntity;
|
import io.anuke.mindustry.game.EventType.*;
|
||||||
import io.anuke.mindustry.entities.type.Unit;
|
import io.anuke.mindustry.gen.*;
|
||||||
import io.anuke.mindustry.game.EventType.BuildSelectEvent;
|
import io.anuke.mindustry.graphics.*;
|
||||||
import io.anuke.mindustry.gen.Call;
|
|
||||||
import io.anuke.mindustry.graphics.Pal;
|
|
||||||
import io.anuke.mindustry.net.Net;
|
|
||||||
import io.anuke.mindustry.world.*;
|
import io.anuke.mindustry.world.*;
|
||||||
import io.anuke.mindustry.world.blocks.BuildBlock;
|
import io.anuke.mindustry.world.blocks.*;
|
||||||
import io.anuke.mindustry.world.blocks.BuildBlock.BuildEntity;
|
import io.anuke.mindustry.world.blocks.BuildBlock.*;
|
||||||
|
|
||||||
import java.io.*;
|
import java.io.*;
|
||||||
import java.util.Arrays;
|
import java.util.*;
|
||||||
|
|
||||||
import static io.anuke.mindustry.Vars.*;
|
import static io.anuke.mindustry.Vars.*;
|
||||||
import static io.anuke.mindustry.entities.traits.BuilderTrait.BuildDataStatic.removal;
|
import static io.anuke.mindustry.entities.traits.BuilderTrait.BuildDataStatic.*;
|
||||||
import static io.anuke.mindustry.entities.traits.BuilderTrait.BuildDataStatic.tmptr;
|
|
||||||
|
|
||||||
/** Interface for units that build things.*/
|
/** Interface for units that build things.*/
|
||||||
public interface BuilderTrait extends Entity, TeamTrait{
|
public interface BuilderTrait extends Entity, TeamTrait{
|
||||||
@@ -106,8 +101,6 @@ public interface BuilderTrait extends Entity, TeamTrait{
|
|||||||
unit.rotation = Mathf.slerpDelta(unit.rotation, unit.angleTo(entity), 0.4f);
|
unit.rotation = Mathf.slerpDelta(unit.rotation, unit.angleTo(entity), 0.4f);
|
||||||
}
|
}
|
||||||
|
|
||||||
//progress is synced, thus not updated clientside
|
|
||||||
if(!Net.client()){
|
|
||||||
//deconstructing is 2x as fast
|
//deconstructing is 2x as fast
|
||||||
if(current.breaking){
|
if(current.breaking){
|
||||||
entity.deconstruct(unit, core, 2f / entity.buildCost * Time.delta() * getBuildPower(tile) * state.rules.buildSpeedMultiplier);
|
entity.deconstruct(unit, core, 2f / entity.buildCost * Time.delta() * getBuildPower(tile) * state.rules.buildSpeedMultiplier);
|
||||||
@@ -115,10 +108,7 @@ public interface BuilderTrait extends Entity, TeamTrait{
|
|||||||
entity.construct(unit, core, 1f / entity.buildCost * Time.delta() * getBuildPower(tile) * state.rules.buildSpeedMultiplier);
|
entity.construct(unit, core, 1f / entity.buildCost * Time.delta() * getBuildPower(tile) * state.rules.buildSpeedMultiplier);
|
||||||
}
|
}
|
||||||
|
|
||||||
current.progress = entity.progress();
|
current.progress = entity.progress;
|
||||||
}else{
|
|
||||||
entity.progress = current.progress;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Returns the queue for storing build requests. */
|
/** Returns the queue for storing build requests. */
|
||||||
@@ -135,7 +125,7 @@ public interface BuilderTrait extends Entity, TeamTrait{
|
|||||||
default void writeBuilding(DataOutput output) throws IOException{
|
default void writeBuilding(DataOutput output) throws IOException{
|
||||||
BuildRequest request = buildRequest();
|
BuildRequest request = buildRequest();
|
||||||
|
|
||||||
if(request != null){
|
if(request != null && (request.block != null || request.breaking)){
|
||||||
output.writeByte(request.breaking ? 1 : 0);
|
output.writeByte(request.breaking ? 1 : 0);
|
||||||
output.writeInt(Pos.get(request.x, request.y));
|
output.writeInt(Pos.get(request.x, request.y));
|
||||||
output.writeFloat(request.progress);
|
output.writeFloat(request.progress);
|
||||||
@@ -174,7 +164,11 @@ public interface BuilderTrait extends Entity, TeamTrait{
|
|||||||
if(applyChanges){
|
if(applyChanges){
|
||||||
buildQueue().addLast(request);
|
buildQueue().addLast(request);
|
||||||
}else if(isBuilding()){
|
}else if(isBuilding()){
|
||||||
buildRequest().progress = progress;
|
BuildRequest last = buildRequest();
|
||||||
|
last.progress = progress;
|
||||||
|
if(last.tile() != null && last.tile().entity instanceof BuildEntity){
|
||||||
|
((BuildEntity)last.tile().entity).progress = progress;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import io.anuke.mindustry.entities.type.TileEntity;
|
|||||||
import io.anuke.mindustry.entities.units.UnitState;
|
import io.anuke.mindustry.entities.units.UnitState;
|
||||||
import io.anuke.mindustry.world.Pos;
|
import io.anuke.mindustry.world.Pos;
|
||||||
import io.anuke.mindustry.world.Tile;
|
import io.anuke.mindustry.world.Tile;
|
||||||
|
import io.anuke.mindustry.world.blocks.*;
|
||||||
|
|
||||||
import java.io.*;
|
import java.io.*;
|
||||||
|
|
||||||
@@ -23,6 +24,10 @@ public class RepairDrone extends BaseDrone{
|
|||||||
target = Units.findDamagedTile(team, x, y);
|
target = Units.findDamagedTile(team, x, y);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(target instanceof TileEntity && ((TileEntity)target).block instanceof BuildBlock){
|
||||||
|
target = null;
|
||||||
|
}
|
||||||
|
|
||||||
if(target != null){
|
if(target != null){
|
||||||
if(target.dst(RepairDrone.this) > type.range){
|
if(target.dst(RepairDrone.this) > type.range){
|
||||||
circle(type.range * 0.9f);
|
circle(type.range * 0.9f);
|
||||||
|
|||||||
@@ -1,14 +1,15 @@
|
|||||||
package io.anuke.mindustry.net;
|
package io.anuke.mindustry.net;
|
||||||
|
|
||||||
import io.anuke.arc.math.Mathf;
|
import io.anuke.arc.math.*;
|
||||||
import io.anuke.arc.math.geom.Vector2;
|
import io.anuke.arc.math.geom.*;
|
||||||
import io.anuke.arc.util.Time;
|
import io.anuke.arc.util.*;
|
||||||
|
|
||||||
public class Interpolator{
|
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[] lasts = {};
|
||||||
public long lastUpdated, updateSpacing;
|
public long lastUpdated, updateSpacing;
|
||||||
|
|
||||||
//current state
|
//current state
|
||||||
@@ -21,6 +22,12 @@ public class Interpolator{
|
|||||||
lastUpdated = Time.millis();
|
lastUpdated = Time.millis();
|
||||||
|
|
||||||
targets = target1ds;
|
targets = target1ds;
|
||||||
|
if(lasts.length != values.length){
|
||||||
|
lasts = new float[values.length];
|
||||||
|
}
|
||||||
|
for(int i = 0; i < values.length; i++){
|
||||||
|
lasts[i] = values[i];
|
||||||
|
}
|
||||||
last.set(cx, cy);
|
last.set(cx, cy);
|
||||||
target.set(x, y);
|
target.set(x, y);
|
||||||
}
|
}
|
||||||
@@ -46,8 +53,12 @@ public class Interpolator{
|
|||||||
values = new float[targets.length];
|
values = new float[targets.length];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(lasts.length != targets.length){
|
||||||
|
lasts = new float[targets.length];
|
||||||
|
}
|
||||||
|
|
||||||
for(int i = 0; i < values.length; i++){
|
for(int i = 0; i < values.length; i++){
|
||||||
values[i] = Mathf.slerp(values[i], targets[i], alpha);
|
values[i] = Mathf.slerp(lasts[i], targets[i], alpha);
|
||||||
}
|
}
|
||||||
}else{
|
}else{
|
||||||
pos.set(target);
|
pos.set(target);
|
||||||
|
|||||||
@@ -162,8 +162,8 @@ public class Net{
|
|||||||
/**
|
/**
|
||||||
* Returns a list of all connections IDs.
|
* Returns a list of all connections IDs.
|
||||||
*/
|
*/
|
||||||
public static Array<NetConnection> getConnections(){
|
public static Iterable<NetConnection> getConnections(){
|
||||||
return (Array<NetConnection>)serverProvider.getConnections();
|
return (Iterable<NetConnection>)serverProvider.getConnections();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -377,16 +377,53 @@ public class Net{
|
|||||||
void host(int port) throws IOException;
|
void host(int port) throws IOException;
|
||||||
|
|
||||||
/** Sends a large stream of data to a specific client. */
|
/** Sends a large stream of data to a specific client. */
|
||||||
void sendStream(int id, Streamable stream);
|
default void sendStream(int id, Streamable stream){
|
||||||
|
NetConnection connection = getByID(id);
|
||||||
|
if(connection == null) return;
|
||||||
|
try{
|
||||||
|
int cid;
|
||||||
|
StreamBegin begin = new StreamBegin();
|
||||||
|
begin.total = stream.stream.available();
|
||||||
|
begin.type = Registrator.getID(stream.getClass());
|
||||||
|
connection.send(begin, SendMode.tcp);
|
||||||
|
cid = begin.id;
|
||||||
|
|
||||||
/** Send an object to everyone connected. */
|
while(stream.stream.available() > 0){
|
||||||
void send(Object object, SendMode mode);
|
byte[] bytes = new byte[Math.min(512, stream.stream.available())];
|
||||||
|
stream.stream.read(bytes);
|
||||||
|
|
||||||
/** Send an object to a specific client ID. */
|
StreamChunk chunk = new StreamChunk();
|
||||||
void sendTo(int id, Object object, SendMode mode);
|
chunk.id = cid;
|
||||||
|
chunk.data = bytes;
|
||||||
|
connection.send(chunk, SendMode.tcp);
|
||||||
|
}
|
||||||
|
}catch(IOException e){
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/** Send an object to everyone <i>except</i> a client ID. */
|
default void send(Object object, SendMode mode){
|
||||||
void sendExcept(int id, Object object, SendMode mode);
|
for(NetConnection con : getConnections()){
|
||||||
|
con.send(object, mode);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
default void sendTo(int id, Object object, SendMode mode){
|
||||||
|
NetConnection conn = getByID(id);
|
||||||
|
if(conn == null){
|
||||||
|
Log.err("Failed to find connection with ID {0}.", id);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
conn.send(object, mode);
|
||||||
|
}
|
||||||
|
|
||||||
|
default void sendExcept(int id, Object object, SendMode mode){
|
||||||
|
for(NetConnection con : getConnections()){
|
||||||
|
if(con.id != id){
|
||||||
|
con.send(object, mode);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/** Close the server connection. */
|
/** Close the server connection. */
|
||||||
void close();
|
void close();
|
||||||
@@ -395,7 +432,7 @@ public class Net{
|
|||||||
byte[] compressSnapshot(byte[] input);
|
byte[] compressSnapshot(byte[] input);
|
||||||
|
|
||||||
/** Return all connected users. */
|
/** Return all connected users. */
|
||||||
Array<? extends NetConnection> getConnections();
|
Iterable<? extends NetConnection> getConnections();
|
||||||
|
|
||||||
/** Returns a connection by ID. */
|
/** Returns a connection by ID. */
|
||||||
NetConnection getByID(int id);
|
NetConnection getByID(int id);
|
||||||
|
|||||||
@@ -3,6 +3,8 @@ package io.anuke.mindustry.net;
|
|||||||
import io.anuke.mindustry.net.Net.SendMode;
|
import io.anuke.mindustry.net.Net.SendMode;
|
||||||
|
|
||||||
public abstract class NetConnection{
|
public abstract class NetConnection{
|
||||||
|
private static int lastID;
|
||||||
|
|
||||||
public final int id;
|
public final int id;
|
||||||
public final String address;
|
public final String address;
|
||||||
|
|
||||||
@@ -18,8 +20,9 @@ public abstract class NetConnection{
|
|||||||
public boolean hasBegunConnecting = false;
|
public boolean hasBegunConnecting = false;
|
||||||
public float viewWidth, viewHeight, viewX, viewY;
|
public float viewWidth, viewHeight, viewX, viewY;
|
||||||
|
|
||||||
public NetConnection(int id, String address){
|
/** Assigns this connection a unique ID. No two connections will ever have the same ID.*/
|
||||||
this.id = id;
|
public NetConnection(String address){
|
||||||
|
this.id = lastID++;
|
||||||
this.address = address;
|
this.address = address;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -106,7 +106,7 @@ public class PausedDialog extends FloatingDialog{
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(control.saves.getCurrent() == null || !control.saves.getCurrent().isAutosave() || state.rules.tutorial){
|
if(control.saves.getCurrent() == null || !control.saves.getCurrent().isAutosave() || state.rules.tutorial || Net.client()){
|
||||||
state.set(State.menu);
|
state.set(State.menu);
|
||||||
logic.reset();
|
logic.reset();
|
||||||
return;
|
return;
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import io.anuke.arc.*;
|
|||||||
import io.anuke.arc.collection.*;
|
import io.anuke.arc.collection.*;
|
||||||
import io.anuke.arc.function.*;
|
import io.anuke.arc.function.*;
|
||||||
import io.anuke.arc.net.*;
|
import io.anuke.arc.net.*;
|
||||||
|
import io.anuke.arc.util.async.*;
|
||||||
import io.anuke.arc.util.pooling.*;
|
import io.anuke.arc.util.pooling.*;
|
||||||
import io.anuke.mindustry.net.Net.*;
|
import io.anuke.mindustry.net.Net.*;
|
||||||
import io.anuke.mindustry.net.Packets.*;
|
import io.anuke.mindustry.net.Packets.*;
|
||||||
@@ -84,7 +85,7 @@ public class ArcNetClient implements ClientProvider{
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void connect(String ip, int port, Runnable success){
|
public void connect(String ip, int port, Runnable success){
|
||||||
runAsync(() -> {
|
Threads.daemon(() -> {
|
||||||
try{
|
try{
|
||||||
//just in case
|
//just in case
|
||||||
client.stop();
|
client.stop();
|
||||||
@@ -99,7 +100,7 @@ public class ArcNetClient implements ClientProvider{
|
|||||||
updateThread.setDaemon(true);
|
updateThread.setDaemon(true);
|
||||||
updateThread.start();
|
updateThread.start();
|
||||||
|
|
||||||
client.connect(5000, ip, port, port);
|
client.connect(5000, ip, port);
|
||||||
success.run();
|
success.run();
|
||||||
}catch(Exception e){
|
}catch(Exception e){
|
||||||
handleException(e);
|
handleException(e);
|
||||||
@@ -115,11 +116,7 @@ public class ArcNetClient implements ClientProvider{
|
|||||||
@Override
|
@Override
|
||||||
public void send(Object object, SendMode mode){
|
public void send(Object object, SendMode mode){
|
||||||
try{
|
try{
|
||||||
if(mode == SendMode.tcp){
|
|
||||||
client.sendTCP(object);
|
client.sendTCP(object);
|
||||||
}else{
|
|
||||||
client.sendUDP(object);
|
|
||||||
}
|
|
||||||
//sending things can cause an under/overflow, catch it and disconnect instead of crashing
|
//sending things can cause an under/overflow, catch it and disconnect instead of crashing
|
||||||
}catch(BufferOverflowException | BufferUnderflowException e){
|
}catch(BufferOverflowException | BufferUnderflowException e){
|
||||||
Net.showError(e);
|
Net.showError(e);
|
||||||
@@ -140,7 +137,7 @@ public class ArcNetClient implements ClientProvider{
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void pingHost(String address, int port, Consumer<Host> valid, Consumer<Exception> invalid){
|
public void pingHost(String address, int port, Consumer<Host> valid, Consumer<Exception> invalid){
|
||||||
runAsync(() -> {
|
Threads.daemon(() -> {
|
||||||
try{
|
try{
|
||||||
DatagramSocket socket = new DatagramSocket();
|
DatagramSocket socket = new DatagramSocket();
|
||||||
socket.send(new DatagramPacket(new byte[]{-2, 1}, 2, InetAddress.getByName(address), port));
|
socket.send(new DatagramPacket(new byte[]{-2, 1}, 2, InetAddress.getByName(address), port));
|
||||||
@@ -189,12 +186,6 @@ public class ArcNetClient implements ClientProvider{
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void runAsync(Runnable run){
|
|
||||||
Thread thread = new Thread(run, "Client Async Run");
|
|
||||||
thread.setDaemon(true);
|
|
||||||
thread.start();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void handleException(Exception e){
|
private void handleException(Exception e){
|
||||||
if(e instanceof ArcNetException){
|
if(e instanceof ArcNetException){
|
||||||
Core.app.post(() -> Net.showError(new IOException("mismatch")));
|
Core.app.post(() -> Net.showError(new IOException("mismatch")));
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
package io.anuke.mindustry.net;
|
package io.anuke.mindustry.net;
|
||||||
|
|
||||||
import io.anuke.arc.*;
|
import io.anuke.arc.*;
|
||||||
import io.anuke.arc.collection.*;
|
|
||||||
import io.anuke.arc.net.*;
|
import io.anuke.arc.net.*;
|
||||||
import io.anuke.arc.util.*;
|
import io.anuke.arc.util.*;
|
||||||
|
import io.anuke.arc.util.async.*;
|
||||||
import io.anuke.mindustry.net.Net.*;
|
import io.anuke.mindustry.net.Net.*;
|
||||||
import io.anuke.mindustry.net.Packets.*;
|
import io.anuke.mindustry.net.Packets.*;
|
||||||
import net.jpountz.lz4.*;
|
import net.jpountz.lz4.*;
|
||||||
@@ -17,14 +17,10 @@ import static io.anuke.mindustry.Vars.*;
|
|||||||
|
|
||||||
public class ArcNetServer implements ServerProvider{
|
public class ArcNetServer implements ServerProvider{
|
||||||
final Server server;
|
final Server server;
|
||||||
final CopyOnWriteArrayList<KryoConnection> connections = new CopyOnWriteArrayList<>();
|
final CopyOnWriteArrayList<ArcConnection> connections = new CopyOnWriteArrayList<>();
|
||||||
final CopyOnWriteArraySet<Integer> missing = new CopyOnWriteArraySet<>();
|
|
||||||
final Array<KryoConnection> array = new Array<>();
|
|
||||||
final LZ4Compressor compressor = LZ4Factory.fastestInstance().fastCompressor();
|
final LZ4Compressor compressor = LZ4Factory.fastestInstance().fastCompressor();
|
||||||
Thread serverThread;
|
Thread serverThread;
|
||||||
|
|
||||||
int lastconnection = 0;
|
|
||||||
|
|
||||||
public ArcNetServer(){
|
public ArcNetServer(){
|
||||||
server = new Server(4096 * 2, 4096, new PacketSerializer());
|
server = new Server(4096 * 2, 4096, new PacketSerializer());
|
||||||
server.setMulticast(multicastGroup, multicastPort);
|
server.setMulticast(multicastGroup, multicastPort);
|
||||||
@@ -40,7 +36,7 @@ public class ArcNetServer implements ServerProvider{
|
|||||||
public void connected(Connection connection){
|
public void connected(Connection connection){
|
||||||
String ip = connection.getRemoteAddressTCP().getAddress().getHostAddress();
|
String ip = connection.getRemoteAddressTCP().getAddress().getHostAddress();
|
||||||
|
|
||||||
KryoConnection kn = new KryoConnection(lastconnection++, ip, connection);
|
ArcConnection kn = new ArcConnection(ip, connection);
|
||||||
|
|
||||||
Connect c = new Connect();
|
Connect c = new Connect();
|
||||||
c.id = kn.id;
|
c.id = kn.id;
|
||||||
@@ -54,7 +50,7 @@ public class ArcNetServer implements ServerProvider{
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void disconnected(Connection connection){
|
public void disconnected(Connection connection){
|
||||||
KryoConnection k = getByKryoID(connection.getID());
|
ArcConnection k = getByKryoID(connection.getID());
|
||||||
if(k == null) return;
|
if(k == null) return;
|
||||||
|
|
||||||
Disconnect c = new Disconnect();
|
Disconnect c = new Disconnect();
|
||||||
@@ -68,7 +64,7 @@ public class ArcNetServer implements ServerProvider{
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void received(Connection connection, Object object){
|
public void received(Connection connection, Object object){
|
||||||
KryoConnection k = getByKryoID(connection.getID());
|
ArcConnection k = getByKryoID(connection.getID());
|
||||||
if(object instanceof FrameworkMessage || k == null) return;
|
if(object instanceof FrameworkMessage || k == null) return;
|
||||||
|
|
||||||
Core.app.post(() -> {
|
Core.app.post(() -> {
|
||||||
@@ -97,18 +93,14 @@ public class ArcNetServer implements ServerProvider{
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Array<KryoConnection> getConnections(){
|
public Iterable<ArcConnection> getConnections(){
|
||||||
array.clear();
|
return connections;
|
||||||
for(KryoConnection c : connections){
|
|
||||||
array.add(c);
|
|
||||||
}
|
|
||||||
return array;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public KryoConnection getByID(int id){
|
public ArcConnection getByID(int id){
|
||||||
for(int i = 0; i < connections.size(); i++){
|
for(int i = 0; i < connections.size(); i++){
|
||||||
KryoConnection con = connections.get(i);
|
ArcConnection con = connections.get(i);
|
||||||
if(con.id == id){
|
if(con.id == id){
|
||||||
return con;
|
return con;
|
||||||
}
|
}
|
||||||
@@ -119,16 +111,14 @@ public class ArcNetServer implements ServerProvider{
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void host(int port) throws IOException{
|
public void host(int port) throws IOException{
|
||||||
lastconnection = 0;
|
|
||||||
connections.clear();
|
connections.clear();
|
||||||
missing.clear();
|
server.bind(port);
|
||||||
server.bind(port, port);
|
|
||||||
|
|
||||||
serverThread = new Thread(() -> {
|
serverThread = new Thread(() -> {
|
||||||
try{
|
try{
|
||||||
server.run();
|
server.run();
|
||||||
}catch(Throwable e){
|
}catch(Throwable e){
|
||||||
if(!(e instanceof ClosedSelectorException)) handleException(e);
|
if(!(e instanceof ClosedSelectorException)) Threads.throwAppException(e);
|
||||||
}
|
}
|
||||||
}, "Net Server");
|
}, "Net Server");
|
||||||
serverThread.setDaemon(true);
|
serverThread.setDaemon(true);
|
||||||
@@ -138,97 +128,12 @@ public class ArcNetServer implements ServerProvider{
|
|||||||
@Override
|
@Override
|
||||||
public void close(){
|
public void close(){
|
||||||
connections.clear();
|
connections.clear();
|
||||||
lastconnection = 0;
|
Threads.daemon(server::stop);
|
||||||
|
|
||||||
async(server::stop);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
ArcConnection getByKryoID(int id){
|
||||||
public void sendStream(int id, Streamable stream){
|
|
||||||
KryoConnection connection = getByID(id);
|
|
||||||
if(connection == null) return;
|
|
||||||
try{
|
|
||||||
|
|
||||||
if(connection.connection != null){
|
|
||||||
|
|
||||||
connection.connection.addListener(new InputStreamSender(stream.stream, 512){
|
|
||||||
int id;
|
|
||||||
|
|
||||||
protected void start(){
|
|
||||||
//send an object so the receiving side knows how to handle the following chunks
|
|
||||||
StreamBegin begin = new StreamBegin();
|
|
||||||
begin.total = stream.stream.available();
|
|
||||||
begin.type = Registrator.getID(stream.getClass());
|
|
||||||
connection.connection.sendTCP(begin);
|
|
||||||
id = begin.id;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected Object next(byte[] bytes){
|
|
||||||
StreamChunk chunk = new StreamChunk();
|
|
||||||
chunk.id = id;
|
|
||||||
chunk.data = bytes;
|
|
||||||
return chunk; //wrap the byte[] with an object so the receiving side knows how to handle it.
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}else{
|
|
||||||
int cid;
|
|
||||||
StreamBegin begin = new StreamBegin();
|
|
||||||
begin.total = stream.stream.available();
|
|
||||||
begin.type = Registrator.getID(stream.getClass());
|
|
||||||
connection.send(begin, SendMode.tcp);
|
|
||||||
cid = begin.id;
|
|
||||||
|
|
||||||
while(stream.stream.available() > 0){
|
|
||||||
byte[] bytes = new byte[Math.min(512, stream.stream.available())];
|
|
||||||
stream.stream.read(bytes);
|
|
||||||
|
|
||||||
StreamChunk chunk = new StreamChunk();
|
|
||||||
chunk.id = cid;
|
|
||||||
chunk.data = bytes;
|
|
||||||
connection.send(chunk, SendMode.tcp);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}catch(IOException e){
|
|
||||||
throw new RuntimeException(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void send(Object object, SendMode mode){
|
|
||||||
for(int i = 0; i < connections.size(); i++){
|
for(int i = 0; i < connections.size(); i++){
|
||||||
connections.get(i).send(object, mode);
|
ArcConnection con = connections.get(i);
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void sendTo(int id, Object object, SendMode mode){
|
|
||||||
NetConnection conn = getByID(id);
|
|
||||||
if(conn == null){
|
|
||||||
if(!missing.contains(id))
|
|
||||||
Log.err("Failed to find connection with ID {0}.", id);
|
|
||||||
missing.add(id);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
conn.send(object, mode);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void sendExcept(int id, Object object, SendMode mode){
|
|
||||||
for(int i = 0; i < connections.size(); i++){
|
|
||||||
KryoConnection conn = connections.get(i);
|
|
||||||
if(conn.id != id) conn.send(object, mode);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void handleException(Throwable e){
|
|
||||||
Time.run(0f, () -> {
|
|
||||||
throw new RuntimeException(e);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
KryoConnection getByKryoID(int id){
|
|
||||||
for(int i = 0; i < connections.size(); i++){
|
|
||||||
KryoConnection con = connections.get(i);
|
|
||||||
if(con.connection != null && con.connection.getID() == id){
|
if(con.connection != null && con.connection.getID() == id){
|
||||||
return con;
|
return con;
|
||||||
}
|
}
|
||||||
@@ -237,17 +142,11 @@ public class ArcNetServer implements ServerProvider{
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
void async(Runnable run){
|
class ArcConnection extends NetConnection{
|
||||||
Thread thread = new Thread(run);
|
|
||||||
thread.setDaemon(true);
|
|
||||||
thread.start();
|
|
||||||
}
|
|
||||||
|
|
||||||
class KryoConnection extends NetConnection{
|
|
||||||
public final Connection connection;
|
public final Connection connection;
|
||||||
|
|
||||||
public KryoConnection(int id, String address, Connection connection){
|
public ArcConnection(String address, Connection connection){
|
||||||
super(id, address);
|
super(address);
|
||||||
this.connection = connection;
|
this.connection = connection;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -259,17 +158,13 @@ public class ArcNetServer implements ServerProvider{
|
|||||||
@Override
|
@Override
|
||||||
public void send(Object object, SendMode mode){
|
public void send(Object object, SendMode mode){
|
||||||
try{
|
try{
|
||||||
if(mode == SendMode.tcp){
|
|
||||||
connection.sendTCP(object);
|
connection.sendTCP(object);
|
||||||
}else{
|
|
||||||
connection.sendUDP(object);
|
|
||||||
}
|
|
||||||
}catch(Exception e){
|
}catch(Exception e){
|
||||||
Log.err(e);
|
Log.err(e);
|
||||||
Log.info("Error sending packet. Disconnecting invalid client!");
|
Log.info("Error sending packet. Disconnecting invalid client!");
|
||||||
connection.close();
|
connection.close();
|
||||||
|
|
||||||
KryoConnection k = getByKryoID(connection.getID());
|
ArcConnection k = getByKryoID(connection.getID());
|
||||||
if(k != null) connections.remove(k);
|
if(k != null) connections.remove(k);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -157,7 +157,7 @@ public class ServerControl implements ApplicationListener{
|
|||||||
|
|
||||||
info("Selected next map to be {0}.", map.name());
|
info("Selected next map to be {0}.", map.name());
|
||||||
|
|
||||||
play(true, () -> world.loadMap(map));
|
play(true, () -> world.loadMap(map, map.applyRules(lastMode)));
|
||||||
}
|
}
|
||||||
}else{
|
}else{
|
||||||
netServer.kickAll(KickReason.gameover);
|
netServer.kickAll(KickReason.gameover);
|
||||||
@@ -231,7 +231,7 @@ public class ServerControl implements ApplicationListener{
|
|||||||
logic.reset();
|
logic.reset();
|
||||||
lastMode = preset;
|
lastMode = preset;
|
||||||
try{
|
try{
|
||||||
world.loadMap(result);
|
world.loadMap(result, result.applyRules(lastMode));
|
||||||
state.rules = result.applyRules(preset);
|
state.rules = result.applyRules(preset);
|
||||||
logic.play();
|
logic.play();
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user