2014-11-14 12:02:52 +01:00
|
|
|
/*
|
|
|
|
* This file is part of Applied Energistics 2.
|
|
|
|
* Copyright (c) 2013 - 2014, AlgorithmX2, All rights reserved.
|
|
|
|
*
|
|
|
|
* Applied Energistics 2 is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* Applied Energistics 2 is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
|
|
* along with Applied Energistics 2. If not, see <http://www.gnu.org/licenses/lgpl>.
|
|
|
|
*/
|
|
|
|
|
2014-03-02 09:35:11 +01:00
|
|
|
package appeng.services;
|
|
|
|
|
2015-01-03 02:53:14 +01:00
|
|
|
|
2014-03-02 09:35:11 +01:00
|
|
|
import java.io.File;
|
|
|
|
import java.util.HashMap;
|
2015-01-03 02:53:14 +01:00
|
|
|
import java.util.Map;
|
2014-03-11 05:33:55 +01:00
|
|
|
import java.util.concurrent.ExecutorService;
|
|
|
|
import java.util.concurrent.Executors;
|
|
|
|
import java.util.concurrent.Future;
|
|
|
|
import java.util.concurrent.ThreadFactory;
|
|
|
|
import java.util.concurrent.TimeUnit;
|
2014-03-02 09:35:11 +01:00
|
|
|
|
|
|
|
import net.minecraft.block.Block;
|
|
|
|
import net.minecraft.world.World;
|
2014-08-09 03:45:31 +02:00
|
|
|
import net.minecraft.world.chunk.Chunk;
|
2014-12-29 21:59:05 +01:00
|
|
|
|
2014-03-02 09:35:11 +01:00
|
|
|
import appeng.api.AEApi;
|
|
|
|
import appeng.api.util.DimensionalCoord;
|
2015-03-09 17:35:19 +01:00
|
|
|
import appeng.services.compass.CompassReader;
|
|
|
|
import appeng.services.compass.ICompassCallback;
|
2014-03-02 09:35:11 +01:00
|
|
|
|
2015-01-03 02:53:14 +01:00
|
|
|
|
2014-03-11 05:33:55 +01:00
|
|
|
public class CompassService implements ThreadFactory
|
2014-03-02 09:35:11 +01:00
|
|
|
{
|
2015-01-03 02:53:14 +01:00
|
|
|
private static final int CHUNK_SIZE = 16;
|
|
|
|
private final Map<World, CompassReader> worldSet = new HashMap<World, CompassReader>();
|
|
|
|
private final ExecutorService executor;
|
|
|
|
private final File rootFolder;
|
|
|
|
private int jobSize = 0;
|
|
|
|
|
|
|
|
public CompassService( File aEFolder )
|
|
|
|
{
|
|
|
|
this.rootFolder = aEFolder;
|
|
|
|
this.executor = Executors.newSingleThreadExecutor( this );
|
|
|
|
this.jobSize = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public Future<?> getCompassDirection( DimensionalCoord coord, int maxRange, ICompassCallback cc )
|
|
|
|
{
|
|
|
|
this.jobSize++;
|
|
|
|
return this.executor.submit( new CMDirectionRequest( coord, maxRange, cc ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
public int jobSize()
|
|
|
|
{
|
|
|
|
return this.jobSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void cleanUp()
|
|
|
|
{
|
|
|
|
for ( CompassReader cr : this.worldSet.values() )
|
|
|
|
cr.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void updateArea( World w, int chunkX, int chunkZ )
|
|
|
|
{
|
|
|
|
int x = chunkX << 4;
|
|
|
|
int z = chunkZ << 4;
|
|
|
|
|
|
|
|
this.updateArea( w, x, CHUNK_SIZE, z );
|
|
|
|
this.updateArea( w, x, CHUNK_SIZE + 32, z );
|
|
|
|
this.updateArea( w, x, CHUNK_SIZE + 64, z );
|
|
|
|
this.updateArea( w, x, CHUNK_SIZE + 96, z );
|
|
|
|
|
|
|
|
this.updateArea( w, x, CHUNK_SIZE + 128, z );
|
|
|
|
this.updateArea( w, x, CHUNK_SIZE + 160, z );
|
|
|
|
this.updateArea( w, x, CHUNK_SIZE + 192, z );
|
|
|
|
this.updateArea( w, x, CHUNK_SIZE + 224, z );
|
|
|
|
}
|
|
|
|
|
|
|
|
public Future<?> updateArea( World w, int x, int y, int z )
|
|
|
|
{
|
|
|
|
this.jobSize++;
|
|
|
|
|
|
|
|
int cx = x >> 4;
|
|
|
|
int cdy = y >> 5;
|
|
|
|
int cz = z >> 4;
|
|
|
|
|
|
|
|
int low_y = cdy << 5;
|
|
|
|
int hi_y = low_y + 32;
|
|
|
|
|
|
|
|
// lower level...
|
|
|
|
Chunk c = w.getChunkFromBlockCoords( x, z );
|
|
|
|
|
|
|
|
for ( Block skyStoneBlock : AEApi.instance().definitions().blocks().skyStone().maybeBlock().asSet() )
|
|
|
|
{
|
|
|
|
for ( int i = 0; i < CHUNK_SIZE; i++ )
|
|
|
|
{
|
|
|
|
for ( int j = 0; j < CHUNK_SIZE; j++ )
|
|
|
|
{
|
|
|
|
for ( int k = low_y; k < hi_y; k++ )
|
|
|
|
{
|
|
|
|
Block blk = c.getBlock( i, k, j );
|
|
|
|
if ( blk == skyStoneBlock && c.getBlockMetadata( i, k, j ) == 0 )
|
|
|
|
{
|
|
|
|
return this.executor.submit( new CMUpdatePost( w, cx, cz, cdy, true ) );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.executor.submit( new CMUpdatePost( w, cx, cz, cdy, false ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
private CompassReader getReader( World w )
|
|
|
|
{
|
|
|
|
CompassReader cr = this.worldSet.get( w );
|
|
|
|
|
|
|
|
if ( cr == null )
|
|
|
|
{
|
|
|
|
cr = new CompassReader( w.provider.dimensionId, this.rootFolder );
|
|
|
|
this.worldSet.put( w, cr );
|
|
|
|
}
|
|
|
|
|
|
|
|
return cr;
|
|
|
|
}
|
|
|
|
|
|
|
|
private int dist( int ax, int az, int bx, int bz )
|
|
|
|
{
|
|
|
|
int up = ( bz - az ) * CHUNK_SIZE;
|
|
|
|
int side = ( bx - ax ) * CHUNK_SIZE;
|
|
|
|
|
|
|
|
return up * up + side * side;
|
|
|
|
}
|
|
|
|
|
|
|
|
private double rad( int ax, int az, int bx, int bz )
|
|
|
|
{
|
|
|
|
int up = bz - az;
|
|
|
|
int side = bx - ax;
|
|
|
|
|
|
|
|
return Math.atan2( -up, side ) - Math.PI / 2.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void kill()
|
|
|
|
{
|
|
|
|
this.executor.shutdown();
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
this.executor.awaitTermination( 6, TimeUnit.MINUTES );
|
|
|
|
this.jobSize = 0;
|
|
|
|
|
|
|
|
for ( CompassReader cr : this.worldSet.values() )
|
|
|
|
{
|
|
|
|
cr.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
this.worldSet.clear();
|
|
|
|
}
|
|
|
|
catch ( InterruptedException e )
|
|
|
|
{
|
|
|
|
// wrap this up..
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public Thread newThread( Runnable job )
|
|
|
|
{
|
|
|
|
return new Thread( job, "AE Compass Service" );
|
|
|
|
}
|
2014-03-02 09:35:11 +01:00
|
|
|
|
2014-08-07 08:47:42 +02:00
|
|
|
|
2014-03-11 05:33:55 +01:00
|
|
|
private class CMUpdatePost implements Runnable
|
2014-03-02 09:35:11 +01:00
|
|
|
{
|
|
|
|
|
|
|
|
public final World world;
|
|
|
|
|
2015-03-26 12:13:34 +01:00
|
|
|
public final int chunkX;
|
|
|
|
public final int chunkZ;
|
2014-03-02 09:35:11 +01:00
|
|
|
public final int doubleChunkY; // 32 blocks instead of 16.
|
|
|
|
public final boolean value;
|
|
|
|
|
2015-01-03 02:53:14 +01:00
|
|
|
public CMUpdatePost( World w, int cx, int cz, int dcy, boolean val )
|
|
|
|
{
|
2014-12-29 15:13:47 +01:00
|
|
|
this.world = w;
|
|
|
|
this.chunkX = cx;
|
|
|
|
this.doubleChunkY = dcy;
|
|
|
|
this.chunkZ = cz;
|
|
|
|
this.value = val;
|
2014-03-02 09:35:11 +01:00
|
|
|
}
|
2014-03-11 05:33:55 +01:00
|
|
|
|
|
|
|
@Override
|
|
|
|
public void run()
|
|
|
|
{
|
2014-12-29 15:13:47 +01:00
|
|
|
CompassService.this.jobSize--;
|
2014-08-07 08:47:42 +02:00
|
|
|
|
2014-12-29 15:13:47 +01:00
|
|
|
CompassReader cr = CompassService.this.getReader( this.world );
|
|
|
|
cr.setHasBeacon( this.chunkX, this.chunkZ, this.doubleChunkY, this.value );
|
2014-08-07 08:47:42 +02:00
|
|
|
|
2014-12-29 15:13:47 +01:00
|
|
|
if ( CompassService.this.jobSize() < 2 )
|
|
|
|
CompassService.this.cleanUp();
|
2014-03-11 05:33:55 +01:00
|
|
|
}
|
2014-09-28 00:50:06 +02:00
|
|
|
}
|
2014-03-02 09:35:11 +01:00
|
|
|
|
2015-01-03 02:53:14 +01:00
|
|
|
|
2014-03-11 05:33:55 +01:00
|
|
|
private class CMDirectionRequest implements Runnable
|
2014-03-02 09:35:11 +01:00
|
|
|
{
|
|
|
|
|
2014-03-03 00:18:15 +01:00
|
|
|
public final int maxRange;
|
2014-03-05 04:12:23 +01:00
|
|
|
public final DimensionalCoord coord;
|
2014-03-02 09:35:11 +01:00
|
|
|
public final ICompassCallback callback;
|
|
|
|
|
2015-01-03 02:53:14 +01:00
|
|
|
public CMDirectionRequest( DimensionalCoord coord, int getMaxRange, ICompassCallback cc )
|
|
|
|
{
|
2014-03-02 09:35:11 +01:00
|
|
|
this.coord = coord;
|
2014-03-03 00:18:15 +01:00
|
|
|
this.maxRange = getMaxRange;
|
2014-12-29 15:13:47 +01:00
|
|
|
this.callback = cc;
|
2014-03-02 09:35:11 +01:00
|
|
|
}
|
|
|
|
|
2014-03-11 05:33:55 +01:00
|
|
|
@Override
|
|
|
|
public void run()
|
|
|
|
{
|
2014-12-29 15:13:47 +01:00
|
|
|
CompassService.this.jobSize--;
|
2014-08-07 08:47:42 +02:00
|
|
|
|
2014-12-29 15:13:47 +01:00
|
|
|
int cx = this.coord.x >> 4;
|
|
|
|
int cz = this.coord.z >> 4;
|
2014-03-11 05:33:55 +01:00
|
|
|
|
2014-12-29 15:13:47 +01:00
|
|
|
CompassReader cr = CompassService.this.getReader( this.coord.getWorld() );
|
2014-03-11 05:33:55 +01:00
|
|
|
|
|
|
|
// Am I standing on it?
|
|
|
|
if ( cr.hasBeacon( cx, cz ) )
|
|
|
|
{
|
2014-12-29 15:13:47 +01:00
|
|
|
this.callback.calculatedDirection( true, true, -999, 0 );
|
2014-08-07 08:47:42 +02:00
|
|
|
|
2014-12-29 15:13:47 +01:00
|
|
|
if ( CompassService.this.jobSize() < 2 )
|
|
|
|
CompassService.this.cleanUp();
|
2014-08-07 08:47:42 +02:00
|
|
|
|
2014-03-11 05:33:55 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// spiral outward...
|
2015-01-03 02:53:14 +01:00
|
|
|
for ( int offset = 1; offset < this.maxRange; offset++ )
|
2014-03-11 05:33:55 +01:00
|
|
|
{
|
2014-09-28 11:47:17 +02:00
|
|
|
int minX = cx - offset;
|
|
|
|
int minZ = cz - offset;
|
|
|
|
int maxX = cx + offset;
|
|
|
|
int maxZ = cz + offset;
|
2014-03-11 05:33:55 +01:00
|
|
|
|
|
|
|
int closest = Integer.MAX_VALUE;
|
|
|
|
int chosen_x = cx;
|
|
|
|
int chosen_z = cz;
|
|
|
|
|
2015-01-03 02:53:14 +01:00
|
|
|
for ( int z = minZ; z <= maxZ; z++ )
|
2014-03-11 05:33:55 +01:00
|
|
|
{
|
2014-09-28 11:47:17 +02:00
|
|
|
if ( cr.hasBeacon( minX, z ) )
|
2014-03-11 05:33:55 +01:00
|
|
|
{
|
2014-12-29 15:13:47 +01:00
|
|
|
int closeness = CompassService.this.dist( cx, cz, minX, z );
|
2014-09-21 01:05:13 +02:00
|
|
|
if ( closeness < closest )
|
2014-03-11 05:33:55 +01:00
|
|
|
{
|
2014-09-21 01:05:13 +02:00
|
|
|
closest = closeness;
|
2014-09-28 11:47:17 +02:00
|
|
|
chosen_x = minX;
|
2014-03-11 05:33:55 +01:00
|
|
|
chosen_z = z;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-28 11:47:17 +02:00
|
|
|
if ( cr.hasBeacon( maxX, z ) )
|
2014-03-11 05:33:55 +01:00
|
|
|
{
|
2014-12-29 15:13:47 +01:00
|
|
|
int closeness = CompassService.this.dist( cx, cz, maxX, z );
|
2014-09-21 01:05:13 +02:00
|
|
|
if ( closeness < closest )
|
2014-03-11 05:33:55 +01:00
|
|
|
{
|
2014-09-21 01:05:13 +02:00
|
|
|
closest = closeness;
|
2014-09-28 11:47:17 +02:00
|
|
|
chosen_x = maxX;
|
2014-03-11 05:33:55 +01:00
|
|
|
chosen_z = z;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-03 02:53:14 +01:00
|
|
|
for ( int x = minX + 1; x < maxX; x++ )
|
2014-03-11 05:33:55 +01:00
|
|
|
{
|
2014-09-28 11:47:17 +02:00
|
|
|
if ( cr.hasBeacon( x, minZ ) )
|
2014-03-11 05:33:55 +01:00
|
|
|
{
|
2014-12-29 15:13:47 +01:00
|
|
|
int closeness = CompassService.this.dist( cx, cz, x, minZ );
|
2014-09-21 01:05:13 +02:00
|
|
|
if ( closeness < closest )
|
2014-03-11 05:33:55 +01:00
|
|
|
{
|
2014-09-21 01:05:13 +02:00
|
|
|
closest = closeness;
|
2014-03-11 05:33:55 +01:00
|
|
|
chosen_x = x;
|
2014-09-28 11:47:17 +02:00
|
|
|
chosen_z = minZ;
|
2014-03-11 05:33:55 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-28 11:47:17 +02:00
|
|
|
if ( cr.hasBeacon( x, maxZ ) )
|
2014-03-11 05:33:55 +01:00
|
|
|
{
|
2014-12-29 15:13:47 +01:00
|
|
|
int closeness = CompassService.this.dist( cx, cz, x, maxZ );
|
2014-09-21 01:05:13 +02:00
|
|
|
if ( closeness < closest )
|
2014-03-11 05:33:55 +01:00
|
|
|
{
|
2014-09-21 01:05:13 +02:00
|
|
|
closest = closeness;
|
2014-03-11 05:33:55 +01:00
|
|
|
chosen_x = x;
|
2014-09-28 11:47:17 +02:00
|
|
|
chosen_z = maxZ;
|
2014-03-11 05:33:55 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( closest < Integer.MAX_VALUE )
|
|
|
|
{
|
2014-12-29 15:13:47 +01:00
|
|
|
this.callback.calculatedDirection( true, false, CompassService.this.rad( cx, cz, chosen_x, chosen_z ), CompassService.this.dist( cx, cz, chosen_x, chosen_z ) );
|
2014-08-07 08:47:42 +02:00
|
|
|
|
2014-12-29 15:13:47 +01:00
|
|
|
if ( CompassService.this.jobSize() < 2 )
|
|
|
|
CompassService.this.cleanUp();
|
2014-08-07 08:47:42 +02:00
|
|
|
|
2014-03-11 05:33:55 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// didn't find shit...
|
2014-12-29 15:13:47 +01:00
|
|
|
this.callback.calculatedDirection( false, true, -999, 999 );
|
2014-08-07 08:47:42 +02:00
|
|
|
|
2014-12-29 15:13:47 +01:00
|
|
|
if ( CompassService.this.jobSize() < 2 )
|
|
|
|
CompassService.this.cleanUp();
|
2014-03-11 05:33:55 +01:00
|
|
|
}
|
2014-09-28 00:50:06 +02:00
|
|
|
}
|
2014-03-02 09:35:11 +01:00
|
|
|
}
|