forked from mirrors/NBTExplorer
259 lines
9.4 KiB
C#
259 lines
9.4 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Text;
|
|
using System.Globalization;
|
|
using NDesk.Options;
|
|
using Substrate;
|
|
|
|
namespace NBToolkit
|
|
{
|
|
public class ReplaceOptions : TKOptions, IChunkFilterable
|
|
{
|
|
private OptionSet _filterOpt = null;
|
|
private ChunkFilter _chunkFilter = null;
|
|
|
|
public int? OPT_BEFORE = null;
|
|
public int? OPT_AFTER = null;
|
|
|
|
public int? OPT_DATA = null;
|
|
public double? OPT_PROB = null;
|
|
|
|
// Block coordinate conditions
|
|
public int? BL_X_GE = null;
|
|
public int? BL_X_LE = null;
|
|
public int? BL_Y_GE = null;
|
|
public int? BL_Y_LE = null;
|
|
public int? BL_Z_GE = null;
|
|
public int? BL_Z_LE = null;
|
|
|
|
// Neighbor conditions
|
|
public int? OPT_NEIGHBOR = null;
|
|
public int? OPT_NEIGHBOR_SIDE = null;
|
|
public int? OPT_NEIGHBOR_E = null;
|
|
public int? OPT_NEIGHBOR_W = null;
|
|
public int? OPT_NEIGHBOR_N = null;
|
|
public int? OPT_NEIGHBOR_S = null;
|
|
public int? OPT_NEIGHBOR_T = null;
|
|
public int? OPT_NEIGHBOR_B = null;
|
|
|
|
public ReplaceOptions ()
|
|
: base()
|
|
{
|
|
_filterOpt = new OptionSet()
|
|
{
|
|
{ "b|before=", "Replace instances of block type {ID} with another block type",
|
|
v => OPT_BEFORE = Convert.ToInt32(v) % 256 },
|
|
{ "a|after=", "Replace the selected blocks with block type {ID}",
|
|
v => OPT_AFTER = Convert.ToInt32(v) % 256 },
|
|
{ "d|data=", "Set the new block's data value to {VAL} (0-15)",
|
|
v => OPT_DATA = Convert.ToInt32(v) % 16 },
|
|
{ "p|prob=", "Replace any matching block with probability {VAL} (0.0-1.0)",
|
|
v => { OPT_PROB = Convert.ToDouble(v, new CultureInfo("en-US"));
|
|
OPT_PROB = Math.Max((double)OPT_PROB, 0.0);
|
|
OPT_PROB = Math.Min((double)OPT_PROB, 1.0); } },
|
|
{ "bxr|BlockXRange=", "Update blocks with X-coord between {0:V1} and {1:V2}, inclusive. V1 or V2 may be left blank.",
|
|
(v1, v2) => {
|
|
try { BL_X_GE = Convert.ToInt32(v1); } catch (FormatException) { }
|
|
try { BL_X_LE = Convert.ToInt32(v2); } catch (FormatException) { }
|
|
} },
|
|
{ "byr|BlockYRange=", "Update blocks with Y-coord between {0:V1} and {1:V2}, inclusive. V1 or V2 may be left blank",
|
|
(v1, v2) => {
|
|
try { BL_Y_GE = Convert.ToInt32(v1); } catch (FormatException) { }
|
|
try { BL_Y_LE = Convert.ToInt32(v2); } catch (FormatException) { }
|
|
} },
|
|
{ "bzr|BlockZRange=", "Update blocks with Z-coord between {0:V1} and {1:V2}, inclusive. V1 or V2 may be left blank",
|
|
(v1, v2) => {
|
|
try { BL_Z_GE = Convert.ToInt32(v1); } catch (FormatException) { }
|
|
try { BL_Z_LE = Convert.ToInt32(v2); } catch (FormatException) { }
|
|
} },
|
|
/*{ "nb=", "Update blocks that have block type {ID} as any neighbor",
|
|
v => OPT_NEIGHBOR = Convert.ToInt32(v) % 256 },
|
|
{ "nbs=", "Update blocks that have block type {ID} as any x/z neighbor",
|
|
v => OPT_NEIGHBOR_SIDE = Convert.ToInt32(v) % 256 },
|
|
{ "nbxa=", "Update blocks that have block type {ID} as their south neighbor",
|
|
v => OPT_NEIGHBOR_S = Convert.ToInt32(v) % 256 },
|
|
{ "nbxb=", "Update blocks that have block type {ID} as their north neighbor",
|
|
v => OPT_NEIGHBOR_N = Convert.ToInt32(v) % 256 },
|
|
{ "nbya=", "Update blocks that have block type {ID} as their top neighbor",
|
|
v => OPT_NEIGHBOR_T = Convert.ToInt32(v) % 256 },
|
|
{ "nbyb=", "Update blocks that have block type {ID} as their bottom neighbor",
|
|
v => OPT_NEIGHBOR_B = Convert.ToInt32(v) % 256 },
|
|
{ "nbza=", "Update blocks that have block type {ID} as their west neighbor",
|
|
v => OPT_NEIGHBOR_W = Convert.ToInt32(v) % 256 },
|
|
{ "nbzb=", "Update blocks that have block type {ID} as their east neighbor",
|
|
v => OPT_NEIGHBOR_E = Convert.ToInt32(v) % 256 },*/
|
|
};
|
|
|
|
_chunkFilter = new ChunkFilter();
|
|
}
|
|
|
|
public ReplaceOptions (string[] args)
|
|
: this()
|
|
{
|
|
Parse(args);
|
|
}
|
|
|
|
public override void Parse (string[] args)
|
|
{
|
|
base.Parse(args);
|
|
|
|
_filterOpt.Parse(args);
|
|
_chunkFilter.Parse(args);
|
|
}
|
|
|
|
public override void PrintUsage ()
|
|
{
|
|
Console.WriteLine("Usage: nbtoolkit replace -b <id> -a <id> [options]");
|
|
Console.WriteLine();
|
|
Console.WriteLine("Options for command 'replace':");
|
|
|
|
_filterOpt.WriteOptionDescriptions(Console.Out);
|
|
|
|
Console.WriteLine();
|
|
_chunkFilter.PrintUsage();
|
|
|
|
Console.WriteLine();
|
|
base.PrintUsage();
|
|
}
|
|
|
|
public override void SetDefaults ()
|
|
{
|
|
base.SetDefaults();
|
|
|
|
// Check for required parameters
|
|
if (OPT_BEFORE == null || OPT_AFTER == null) {
|
|
Console.WriteLine("Error: You must specify a before and after Block ID");
|
|
Console.WriteLine();
|
|
PrintUsage();
|
|
|
|
throw new TKOptionException();
|
|
}
|
|
}
|
|
|
|
public IChunkFilter GetChunkFilter ()
|
|
{
|
|
return _chunkFilter;
|
|
}
|
|
}
|
|
|
|
public class Replace : TKFilter
|
|
{
|
|
private ReplaceOptions opt;
|
|
|
|
private static Random rand = new Random();
|
|
|
|
public Replace (ReplaceOptions o)
|
|
{
|
|
opt = o;
|
|
}
|
|
|
|
public override void Run ()
|
|
{
|
|
World world = new World(opt.OPT_WORLD);
|
|
ChunkManager cm = world.GetChunkManager() as ChunkManager;
|
|
|
|
FilteredChunkManager fcm = new FilteredChunkManager(cm, opt.GetChunkFilter());
|
|
|
|
int affectedChunks = 0;
|
|
foreach (ChunkRef chunk in fcm) {
|
|
affectedChunks++;
|
|
|
|
ApplyChunk(world, chunk);
|
|
|
|
world.GetChunkManager().Save();
|
|
}
|
|
|
|
Console.WriteLine("Affected Chunks: " + affectedChunks);
|
|
}
|
|
|
|
public void ApplyChunk (World world, ChunkRef chunk)
|
|
{
|
|
int xBase = chunk.X * chunk.XDim;
|
|
int zBase = chunk.Z * chunk.ZDim;
|
|
|
|
// Determine X range
|
|
int xmin = 0;
|
|
int xmax = 15;
|
|
|
|
if (opt.BL_X_GE != null) {
|
|
xmin = (int)opt.BL_X_GE - xBase;
|
|
}
|
|
if (opt.BL_X_LE != null) {
|
|
xmax = (int)opt.BL_X_LE - xBase;
|
|
}
|
|
|
|
xmin = (xmin < 0) ? 0 : xmin;
|
|
xmax = (xmin > 15) ? 15 : xmax;
|
|
|
|
if (xmin > 15 || xmax < 0 || xmin > xmax) {
|
|
return;
|
|
}
|
|
|
|
// Determine Y range
|
|
int ymin = 0;
|
|
int ymax = 127;
|
|
|
|
if (opt.BL_Y_GE != null) {
|
|
ymin = (int)opt.BL_Y_GE;
|
|
}
|
|
if (opt.BL_Y_LE != null) {
|
|
ymax = (int)opt.BL_Y_LE;
|
|
}
|
|
|
|
if (ymin > ymax) {
|
|
return;
|
|
}
|
|
|
|
// Determine X range
|
|
int zmin = 0;
|
|
int zmax = 15;
|
|
|
|
if (opt.BL_Z_GE != null) {
|
|
zmin = (int)opt.BL_Z_GE - zBase;
|
|
}
|
|
if (opt.BL_Z_LE != null) {
|
|
zmax = (int)opt.BL_Z_LE - zBase;
|
|
}
|
|
|
|
zmin = (zmin < 0) ? 0 : zmin;
|
|
zmax = (zmin > 15) ? 15 : zmax;
|
|
|
|
if (zmin > 15 || zmax < 0 || zmin > zmax) {
|
|
return;
|
|
}
|
|
|
|
// Process Chunk
|
|
for (int y = ymin; y <= ymax; y++) {
|
|
for (int x = xmin; x <= xmax; x++) {
|
|
for (int z = zmin; z <= zmax; z++) {
|
|
// Probability test
|
|
if (opt.OPT_PROB != null) {
|
|
double c = rand.NextDouble();
|
|
if (c > opt.OPT_PROB) {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
int lx = x & (chunk.XDim - 1);
|
|
int ly = y & (chunk.YDim - 1);
|
|
int lz = z & (chunk.ZDim - 1);
|
|
|
|
// Attempt to replace block
|
|
int oldBlock = chunk.GetBlockID(lx , ly, lz);
|
|
if (oldBlock == opt.OPT_BEFORE) {
|
|
chunk.SetBlockID(lx, ly, lz, (int)opt.OPT_AFTER);
|
|
|
|
if (opt.OPT_VV) {
|
|
Console.WriteLine("Replaced block at {0},{1},{2}", lx, ly, lz);
|
|
}
|
|
|
|
if (opt.OPT_DATA != null) {
|
|
chunk.SetBlockData(lx, ly, lz, (int)opt.OPT_DATA);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|