364 lines
10 KiB
Java

package fr.evolving.automata;
import java.io.Serializable;
import java.util.Iterator;
import com.badlogic.gdx.Gdx;
import com.badlogic.gdx.math.Vector2;
import com.badlogic.gdx.utils.Array;
import com.badlogic.gdx.utils.ObjectMap.Entry;
import fr.evolving.automata.Particle.Orientation;
import fr.evolving.automata.Particle.Type;
public class Grid implements Serializable,Cloneable {
protected Cell[][] Cells;
public Integer sizeX, sizeY;
public transient Array<Particle> particles;
public Grid(Integer X, Integer Y) {
Reinit();
this.sizeX = X;
this.sizeY = Y;
this.Cells = new Cell[this.sizeX][this.sizeY];
for (int x = 0; x < this.sizeX; x++) {
for (int y = 0; y < this.sizeY; y++) {
this.Cells[x][y] = new Cell();
}
}
}
public void Reinit() {
if (particles==null)
particles=new Array<Particle>();
}
//Réalise un cycle de simulation dans la grille
public void Cycle() {
for(Particle particle: particles) {
Gdx.app.debug("wirechem-Grid", "Grid Cycle -> Particle "+particle.getType()+"/"+particle.getSize()+ " coords:"+particle.getCoordx()+","+particle.getCoordy()+"/"+particle.getOrientation()+" charge:"+particle.getCharge());
if (particle.getType()==Type.Photon) {
particle.Next();
if (!particle.isAlive()) {
Gdx.app.debug("wirechem-Particle", "coords:"+particle.getCoordx()+","+particle.getCoordy()+" killed & removed");
particles.removeValue(particle, true);
}
}
}
}
//Affiche le cycle en cours à l'écran
public void tiling_particle() {
for (int x = 0; x < this.sizeX; x++)
for (int y = 0; y < this.sizeY; y++)
if (GetXY(x, y).Fiber)
GetXY(x, y).Fiber_state = 0;
for(Particle particle: particles) {
if (particle.getType()==Type.Photon) {
GetXY(particle.getCoordx(), particle.getCoordy()).Fiber_state=Math.floorDiv(29-particle.getLife(),3);
Gdx.app.debug("wirechem-Grid", "Grid Tiling -> Photon state :"+GetXY(particle.getCoordx(), particle.getCoordy()).Fiber_state+":"+particle.getCoordx()+","+particle.getCoordy());
}
}
}
//Initialise la simulation pour permettre ensuite de faire des cycles
public void Initialize() {
particles.clear();
this.tiling_particle();
particles.add(new Particle(this));
particles.get(0).setType(Type.Photon);
particles.get(0).setCoordx(6);
particles.get(0).setCoordy(3);
particles.get(0).setOrientation(Orientation.E);
}
//Genère des tiles qui correspondent aux transmuteurs sur la grille
public int tiling_transmuter() {
int result=0;
for (int x = 0; x < this.sizeX; x++)
for (int y = 0; y < this.sizeY; y++)
{
GetXY(x, y).Transmuter_calc = 0;
if (GetXY(x, y).Transmuter!=null && !GetXY(x, y).Free)
result+=GetXY(x, y).Transmuter.getPrice();
}
for (int x = 0; x < this.sizeX; x++)
for (int y = 0; y < this.sizeY; y++) {
Transmuter transmuter = getTransmuter(x, y);
if (transmuter != null) {
Iterator<Entry<Vector2, Integer>> tiles = transmuter.getTilesidrotated().iterator();
while (tiles.hasNext()) {
Entry<Vector2, Integer> all = tiles.next();
Cell cell=GetXY(x + all.key.x, y + all.key.y);
if (cell!=null) {
cell.Transmuter_calc = (1 << 16) * transmuter.getRotation().ordinal()+ all.value;
cell.Transmuter_movex = (int) -all.key.x;
cell.Transmuter_movey = (int) -all.key.y;
}
else
{
result-=GetXY(x, y).Transmuter.getPrice();
Iterator<Entry<Vector2, Integer>> tileseraser = transmuter.getTilesidrotated().iterator();
while (tileseraser.hasNext()) {
Entry<Vector2, Integer> allereaser = tileseraser.next();
Cell celleraser=GetXY(x + allereaser.key.x, y + allereaser.key.y);
if (celleraser!=null) {
celleraser.Transmuter=null;
celleraser.Transmuter_calc=0;
celleraser.Transmuter_movex=0;
celleraser.Transmuter_movey=0;
}
}
break;
}
}
}
}
for (int x = 0; x < this.sizeX; x++)
for (int y = 0; y < this.sizeY; y++) {
if (GetXY(x, y).Transmuter_calc > 0)
Gdx.app.debug("wirechem-Grid", x + "," + y + ">"+ GetXY(x, y).Transmuter_calc);
}
return result;
}
//
public int tiling_copper() {
int result=0;
for (int x = 0; x < this.sizeX; x++)
for (int y = 0; y < this.sizeY; y++) {
if (getFiber(x,y) && !GetXY(x, y).Free)
result+=5;
if (getCopper(x, y)) {
if (!GetXY(x, y).Free)
{
result++;
if (getFiber(x,y))
result+=45;
}
int value = 0;
if (getCopper(x, y + 1))
value++;
if (getCopper(x - 1, y))
value += 8;
if (getCopper(x, y - 1))
value += 4;
if (getCopper(x + 1, y))
value += 2;
GetXY(x, y).Copper_calc = value;
} else
GetXY(x, y).Copper_calc = -1;
}
for (int x = 0; x < this.sizeX; x++)
for (int y = 0; y < this.sizeY; y++) {
int value = 0;
if (getCoppercalc(x, y) == 15) {
if (getCopper(x - 1, y - 1))
value++;
if (getCopper(x, y - 1))
value++;
if (getCopper(x + 1, y - 1))
value++;
if (getCopper(x - 1, y))
value++;
if (getCopper(x + 1, y))
value++;
if (getCopper(x - 1, y + 1))
value++;
if (getCopper(x, y + 1))
value++;
if (getCopper(x + 1, y + 1))
value++;
if (value >= 5)
GetXY(x, y).Copper_calc = GetXY(x, y).Copper_calc + 20;
} else {
if (getCoppercalc(x, y) != -1) {
int oldvalue = GetXY(x, y).Copper_calc;
if (getCoppercalc(x - 1, y - 1) == 15
|| getCoppercalc(x - 1, y - 1) == 35)
value++;
if (getCoppercalc(x, y - 1) == 15
|| getCoppercalc(x, y - 1) == 35)
value++;
if (getCoppercalc(x + 1, y - 1) == 15
|| getCoppercalc(x + 1, y - 1) == 35)
value++;
if (getCoppercalc(x - 1, y) == 15
|| getCoppercalc(x - 1, y) == 35)
value++;
if (getCoppercalc(x + 1, y) == 15
|| getCoppercalc(x + 1, y) == 35)
value++;
if (getCoppercalc(x - 1, y + 1) == 15
|| getCoppercalc(x - 1, y + 1) == 35)
value++;
if (getCoppercalc(x, y + 1) == 15
|| getCoppercalc(x, y + 1) == 35)
value++;
if (getCoppercalc(x + 1, y + 1) == 15
|| getCoppercalc(x + 1, y + 1) == 35)
value++;
if (value >= 1 && oldvalue != 1 && oldvalue != 2
&& oldvalue != 4 && oldvalue != 8
&& oldvalue != 10 && oldvalue != 5)
GetXY(x, y).Copper_calc = oldvalue + 20;
}
}
}
for (int x = 0; x < this.sizeX; x++)
for (int y = 0; y < this.sizeY; y++) {
if (getCoppercalc(x, y) == 35) {
int value = 0;
if (!getCopper(x + 1, y + 1))
value += 2;
if (!getCopper(x - 1, y - 1))
value += 8;
if (!getCopper(x + 1, y - 1))
value += 4;
if (!getCopper(x - 1, y + 1))
value += 1;
GetXY(x, y).Copper_calc = GetXY(x, y).Copper_calc + value;
}
}
for (int x = 0; x < this.sizeX; x++)
for (int y = 0; y < this.sizeY; y++) {
int oldvalue = GetXY(x, y).Copper_calc;
if (oldvalue == 27 || oldvalue == 31 || oldvalue == 33
|| oldvalue == 34) {
int value = 0;
if (getCopper(x, y + 1) && getCoppercalc(x, y + 1) < 15)
value += 1;
if (getCopper(x - 1, y) && getCoppercalc(x - 1, y) < 15)
value += 6;
if (getCopper(x, y - 1) && getCoppercalc(x, y - 1) < 15)
value += 2;
if (getCopper(x + 1, y) && getCoppercalc(x + 1, y) < 15)
value += 2;
if (value > 0)
GetXY(x, y).Copper_calc = oldvalue + 22 + value;
}
int value = 0;
if (oldvalue == 34
&& (getCoppercalc(x - 1, y) == 31
|| getCoppercalc(x - 1, y) == 55 || getCoppercalc(
x - 1, y) == 58))
value = 62;
if (oldvalue == 34
&& (getCoppercalc(x + 1, y) == 31
|| getCoppercalc(x + 1, y) == 55 || getCoppercalc(
x + 1, y) == 58))
value = 58;
if (oldvalue == 31
&& (getCoppercalc(x - 1, y) == 34
|| getCoppercalc(x - 1, y) == 58 || getCoppercalc(
x - 1, y) == 62))
value = 59;
if (oldvalue == 31
&& (getCoppercalc(x + 1, y) == 34
|| getCoppercalc(x + 1, y) == 58 || getCoppercalc(
x + 1, y) == 62))
value = 55;
if (oldvalue == 33
&& (getCoppercalc(x, y - 1) == 27
|| getCoppercalc(x, y - 1) == 50 || getCoppercalc(
x, y - 1) == 51))
value = 57;
if (oldvalue == 33
&& (getCoppercalc(x, y + 1) == 27
|| getCoppercalc(x, y + 1) == 50 || getCoppercalc(
x, y + 1) == 51))
value = 56;
if (oldvalue == 27
&& (getCoppercalc(x, y - 1) == 33
|| getCoppercalc(x, y - 1) == 56 || getCoppercalc(
x, y - 1) == 57))
value = 51;
if (oldvalue == 27
&& (getCoppercalc(x, y + 1) == 33
|| getCoppercalc(x, y + 1) == 56 || getCoppercalc(
x, y + 1) == 57))
value = 50;
if (value > 0)
GetXY(x, y).Copper_calc = value;
}
return result;
}
public Cell GetXY(float X, float Y) {
if (X < 0 || Y < 0 || X >= this.sizeX || Y >= this.sizeY)
return null;
else
return this.Cells[(int) X][(int) Y];
}
public Transmuter getTransmuter(float X, float Y) {
Cell cell = GetXY(X, Y);
if (cell == null)
return null;
else
return cell.Transmuter;
}
public int getTransmutercalc(float X, float Y) {
Cell cell = GetXY(X, Y);
if (cell == null)
return 0;
else
return cell.Transmuter_calc & 0xFFFF;
}
public int getTransmuterrot(float X, float Y) {
Cell cell = GetXY(X, Y);
if (cell == null)
return 0;
else
return cell.Transmuter_calc >> 16;
}
public boolean getCopper(float X, float Y) {
Cell cell = GetXY(X, Y);
if (cell == null)
return false;
else
return cell.Copper;
}
public boolean getFiber(float X, float Y) {
Cell cell = GetXY(X, Y);
if (cell == null)
return false;
else
return cell.Fiber;
}
public int getCoppercalc(float X, float Y) {
Cell cell = GetXY(X, Y);
if (cell == null)
return 0;
else
return cell.Copper_calc;
}
public Object clone() {
Grid result = new Grid(this.sizeX,this.sizeY);
for (int x = 0; x < this.sizeX; x++)
for (int y = 0; y < this.sizeY; y++)
result.Cells[x][y] = (Cell)this.Cells[x][y].clone();
return result;
}
public Object clone(int newsizex,int newsizey) {
if (newsizex<3) newsizex=3;
if (newsizey<3) newsizey=3;
Grid result = new Grid(newsizex,newsizey);
for (int x = 0; x < newsizex; x++)
for (int y = 0; y < newsizey; y++)
if (x<this.sizeX && y<this.sizeY)
result.Cells[x][y] = (Cell)this.Cells[x][y].clone();
return result;
}
}