* Assorted terrain fixes
parent
aa704172d1
commit
cd88a4914d
|
@ -277,7 +277,7 @@ namespace OpenSim.Region.Terrain
|
||||||
resultText += "terrain save grdmap <filename> <gradient map> - creates a PNG snapshot of the region using a named gradient map\n";
|
resultText += "terrain save grdmap <filename> <gradient map> - creates a PNG snapshot of the region using a named gradient map\n";
|
||||||
resultText += "terrain rescale <min> <max> - rescales a terrain to be between <min> and <max> meters high\n";
|
resultText += "terrain rescale <min> <max> - rescales a terrain to be between <min> and <max> meters high\n";
|
||||||
resultText += "terrain fill <val> - fills a terrain at the specified height\n";
|
resultText += "terrain fill <val> - fills a terrain at the specified height\n";
|
||||||
resultText += "terrain erode aerobic <windspeed> <pickupmin> <dropmin> <carry> <rounds> <lowest>\n";
|
resultText += "terrain erode aerobic <windspeed> <pickupmin> <dropmin> <carry> <rounds> <lowest t/f> <fluid dynamics t/f>\n";
|
||||||
resultText += "terrain erode thermal <talus> <rounds> <carry>\n";
|
resultText += "terrain erode thermal <talus> <rounds> <carry>\n";
|
||||||
resultText += "terrain erode hydraulic <rain> <evaporation> <solubility> <frequency> <rounds>\n";
|
resultText += "terrain erode hydraulic <rain> <evaporation> <solubility> <frequency> <rounds>\n";
|
||||||
resultText += "terrain multiply <val> - multiplies a terrain by <val>\n";
|
resultText += "terrain multiply <val> - multiplies a terrain by <val>\n";
|
||||||
|
@ -426,11 +426,14 @@ namespace OpenSim.Region.Terrain
|
||||||
|
|
||||||
private bool ConsoleErosion(string[] args, ref string resultText)
|
private bool ConsoleErosion(string[] args, ref string resultText)
|
||||||
{
|
{
|
||||||
|
double min = heightmap.FindMin();
|
||||||
|
double max = heightmap.FindMax();
|
||||||
|
|
||||||
switch (args[1].ToLower())
|
switch (args[1].ToLower())
|
||||||
{
|
{
|
||||||
case "aerobic":
|
case "aerobic":
|
||||||
// WindSpeed, PickupMinimum,DropMinimum,Carry,Rounds,Lowest
|
// WindSpeed, PickupMinimum,DropMinimum,Carry,Rounds,Lowest
|
||||||
heightmap.AerobicErosion(Convert.ToDouble(args[2]), Convert.ToDouble(args[3]), Convert.ToDouble(args[4]), Convert.ToDouble(args[5]), Convert.ToInt32(args[6]), Convert.ToBoolean(args[7]), true);
|
heightmap.AerobicErosion(Convert.ToDouble(args[2]), Convert.ToDouble(args[3]), Convert.ToDouble(args[4]), Convert.ToDouble(args[5]), Convert.ToInt32(args[6]), Convert.ToBoolean(args[7]), Convert.ToBoolean(args[8]));
|
||||||
break;
|
break;
|
||||||
case "thermal":
|
case "thermal":
|
||||||
heightmap.ThermalWeathering(Convert.ToDouble(args[2]), Convert.ToInt32(args[3]), Convert.ToDouble(args[4]));
|
heightmap.ThermalWeathering(Convert.ToDouble(args[2]), Convert.ToInt32(args[3]), Convert.ToDouble(args[4]));
|
||||||
|
@ -444,6 +447,9 @@ namespace OpenSim.Region.Terrain
|
||||||
resultText = "Unknown erosion type";
|
resultText = "Unknown erosion type";
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
heightmap.Normalise(min, max);
|
||||||
|
|
||||||
tainted++;
|
tainted++;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
|
@ -257,5 +257,21 @@ namespace libTerrain
|
||||||
{
|
{
|
||||||
return Sum() / (w * h);
|
return Sum() / (w * h);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public bool ContainsNaN()
|
||||||
|
{
|
||||||
|
int x, y;
|
||||||
|
for (x = 0; x < w; x++)
|
||||||
|
{
|
||||||
|
for (y = 0; y < h; y++)
|
||||||
|
{
|
||||||
|
double elm = map[x, y];
|
||||||
|
|
||||||
|
if (Double.IsNaN(elm))
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -57,13 +57,16 @@ namespace libTerrain
|
||||||
|
|
||||||
public void SaveImage(string filename)
|
public void SaveImage(string filename)
|
||||||
{
|
{
|
||||||
|
Channel outmap = this.Copy();
|
||||||
|
outmap.Normalise();
|
||||||
|
|
||||||
Bitmap bit = new Bitmap(w, h, System.Drawing.Imaging.PixelFormat.Format24bppRgb);
|
Bitmap bit = new Bitmap(w, h, System.Drawing.Imaging.PixelFormat.Format24bppRgb);
|
||||||
int x, y;
|
int x, y;
|
||||||
for (x = 0; x < w; x++)
|
for (x = 0; x < w; x++)
|
||||||
{
|
{
|
||||||
for (y = 0; y < h; y++)
|
for (y = 0; y < h; y++)
|
||||||
{
|
{
|
||||||
int val = Math.Min(255, (int)(map[x,y] * 255));
|
int val = Math.Min(255, (int)(outmap.map[x,y] * 255));
|
||||||
Color col = Color.FromArgb(val,val,val);
|
Color col = Color.FromArgb(val,val,val);
|
||||||
bit.SetPixel(x, y, col);
|
bit.SetPixel(x, y, col);
|
||||||
}
|
}
|
||||||
|
|
|
@ -43,6 +43,8 @@ namespace libTerrain
|
||||||
|
|
||||||
int x, y;
|
int x, y;
|
||||||
|
|
||||||
|
if (max != min)
|
||||||
|
{
|
||||||
for (x = 0; x < w; x++)
|
for (x = 0; x < w; x++)
|
||||||
{
|
{
|
||||||
for (y = 0; y < h; y++)
|
for (y = 0; y < h; y++)
|
||||||
|
@ -50,6 +52,11 @@ namespace libTerrain
|
||||||
map[x, y] = (map[x, y] - min) * (1.0 / (max - min));
|
map[x, y] = (map[x, y] - min) * (1.0 / (max - min));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
this.Fill(0.5);
|
||||||
|
}
|
||||||
|
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
@ -244,6 +251,76 @@ namespace libTerrain
|
||||||
map = manipulated;
|
map = manipulated;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void Distort(Channel mask, double str)
|
||||||
|
{
|
||||||
|
// Simple pertubation filter
|
||||||
|
double[,] manipulated = new double[w, h];
|
||||||
|
|
||||||
|
double amount;
|
||||||
|
|
||||||
|
int x, y;
|
||||||
|
for (x = 0; x < w; x++)
|
||||||
|
{
|
||||||
|
for (y = 0; y < h; y++)
|
||||||
|
{
|
||||||
|
amount = mask.map[x, y];
|
||||||
|
double offset_x = (double)x + (amount * str) - (0.5 * str);
|
||||||
|
double offset_y = (double)y + (amount * str) - (0.5 * str);
|
||||||
|
|
||||||
|
if (offset_x > w)
|
||||||
|
offset_x = w - 1;
|
||||||
|
if (offset_y > h)
|
||||||
|
offset_y = h - 1;
|
||||||
|
if (offset_y < 0)
|
||||||
|
offset_y = 0;
|
||||||
|
if (offset_x < 0)
|
||||||
|
offset_x = 0;
|
||||||
|
|
||||||
|
double p = GetBilinearInterpolate(offset_x, offset_y);
|
||||||
|
manipulated[x, y] = p;
|
||||||
|
SetDiff(x, y);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
map = manipulated;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Distort(Channel mask, Channel mask2, double str)
|
||||||
|
{
|
||||||
|
// Simple pertubation filter
|
||||||
|
double[,] manipulated = new double[w, h];
|
||||||
|
|
||||||
|
double amountX;
|
||||||
|
double amountY;
|
||||||
|
|
||||||
|
int x, y;
|
||||||
|
for (x = 0; x < w; x++)
|
||||||
|
{
|
||||||
|
for (y = 0; y < h; y++)
|
||||||
|
{
|
||||||
|
amountX = mask.map[x, y];
|
||||||
|
amountY = mask2.map[x, y];
|
||||||
|
double offset_x = (double)x + (amountX * str) - (0.5 * str);
|
||||||
|
double offset_y = (double)y + (amountY * str) - (0.5 * str);
|
||||||
|
|
||||||
|
if (offset_x > w)
|
||||||
|
offset_x = w - 1;
|
||||||
|
if (offset_y > h)
|
||||||
|
offset_y = h - 1;
|
||||||
|
if (offset_y < 0)
|
||||||
|
offset_y = 0;
|
||||||
|
if (offset_x < 0)
|
||||||
|
offset_x = 0;
|
||||||
|
|
||||||
|
double p = GetBilinearInterpolate(offset_x, offset_y);
|
||||||
|
manipulated[x, y] = p;
|
||||||
|
SetDiff(x, y);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
map = manipulated;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
public Channel Blend(Channel other, double amount)
|
public Channel Blend(Channel other, double amount)
|
||||||
{
|
{
|
||||||
int x, y;
|
int x, y;
|
||||||
|
|
|
@ -76,13 +76,19 @@ namespace libTerrain
|
||||||
/// <param name="lowest">Drop sediment at the lowest point?</param>
|
/// <param name="lowest">Drop sediment at the lowest point?</param>
|
||||||
public void AerobicErosion(double windspeed, double pickupTalusMinimum, double dropTalusMinimum, double carry, int rounds, bool lowest, bool usingFluidDynamics)
|
public void AerobicErosion(double windspeed, double pickupTalusMinimum, double dropTalusMinimum, double carry, int rounds, bool lowest, bool usingFluidDynamics)
|
||||||
{
|
{
|
||||||
|
bool debugImages = true;
|
||||||
|
|
||||||
Channel wind = new Channel(w, h) ;
|
Channel wind = new Channel(w, h) ;
|
||||||
Channel sediment = new Channel(w, h);
|
Channel sediment = new Channel(w, h);
|
||||||
int x, y, i, j;
|
int x, y, i, j;
|
||||||
|
|
||||||
|
this.Normalise();
|
||||||
|
|
||||||
wind = this.Copy();
|
wind = this.Copy();
|
||||||
wind.Normalise(); // Cheap wind calculations
|
wind.Noise();
|
||||||
wind *= windspeed;
|
|
||||||
|
if (debugImages)
|
||||||
|
wind.SaveImage("testimg/wind_start.png");
|
||||||
|
|
||||||
if (usingFluidDynamics)
|
if (usingFluidDynamics)
|
||||||
{
|
{
|
||||||
|
@ -90,9 +96,12 @@ namespace libTerrain
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
wind.Pertubation(30); // Can do better later
|
wind.Pertubation(30);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (debugImages)
|
||||||
|
wind.SaveImage("testimg/wind_begin.png");
|
||||||
|
|
||||||
for (i = 0; i < rounds; i++)
|
for (i = 0; i < rounds; i++)
|
||||||
{
|
{
|
||||||
// Convert some rocks to sand
|
// Convert some rocks to sand
|
||||||
|
@ -127,7 +136,14 @@ namespace libTerrain
|
||||||
if (usingFluidDynamics)
|
if (usingFluidDynamics)
|
||||||
{
|
{
|
||||||
sediment.navierStokes(7, 0.1, 0.0, 0.1);
|
sediment.navierStokes(7, 0.1, 0.0, 0.1);
|
||||||
wind.navierStokes(10, 0.1, 0.0, 0.0);
|
|
||||||
|
Channel noiseChan = new Channel(w, h);
|
||||||
|
noiseChan.Noise();
|
||||||
|
wind.Blend(noiseChan, 0.01);
|
||||||
|
|
||||||
|
wind.navierStokes(10, 0.1, 0.01, 0.01);
|
||||||
|
|
||||||
|
sediment.Distort(wind, windspeed);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -137,6 +153,9 @@ namespace libTerrain
|
||||||
sediment.seed++;
|
sediment.seed++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (debugImages)
|
||||||
|
wind.SaveImage("testimg/wind_" + i.ToString() + ".png");
|
||||||
|
|
||||||
// Convert some sand to rock
|
// Convert some sand to rock
|
||||||
for (x = 1; x < w - 1; x++)
|
for (x = 1; x < w - 1; x++)
|
||||||
{
|
{
|
||||||
|
@ -174,11 +193,21 @@ namespace libTerrain
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (debugImages)
|
||||||
|
sediment.SaveImage("testimg/sediment_" + i.ToString() + ".png");
|
||||||
|
|
||||||
|
wind.Normalise();
|
||||||
|
wind *= windspeed;
|
||||||
|
|
||||||
|
this.Normalise();
|
||||||
}
|
}
|
||||||
|
|
||||||
Channel myself = this;
|
Channel myself = this;
|
||||||
myself += sediment;
|
myself += sediment;
|
||||||
myself.Normalise();
|
myself.Normalise();
|
||||||
|
|
||||||
|
if (debugImages)
|
||||||
|
this.SaveImage("testimg/output.png");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -211,11 +211,11 @@ namespace libTerrain
|
||||||
int i;
|
int i;
|
||||||
int j;
|
int j;
|
||||||
|
|
||||||
for (i = 0; i <= N; i++)
|
for (i = 1; i <= N; i++)
|
||||||
{
|
{
|
||||||
for (j = 0; j <= N; j++)
|
for (j = 1; j <= N; j++)
|
||||||
{
|
{
|
||||||
doubles[i, j] = dens[nsIX(i, j, N)];
|
doubles[i - 1, j - 1] = dens[nsIX(i, j, N)];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -229,7 +229,7 @@ namespace libTerrain
|
||||||
{
|
{
|
||||||
for (j = 1; j <= N; j++)
|
for (j = 1; j <= N; j++)
|
||||||
{
|
{
|
||||||
dens[nsIX(i, j, N)] = doubles[i, j];
|
dens[nsIX(i, j, N)] = doubles[i - 1, j - 1];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -272,5 +272,36 @@ namespace libTerrain
|
||||||
{
|
{
|
||||||
nsSimulate(this.h, rounds, dt, diff, visc);
|
nsSimulate(this.h, rounds, dt, diff, visc);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void navierStokes(int rounds, double dt, double diff, double visc, ref double[,] uret, ref double[,] vret)
|
||||||
|
{
|
||||||
|
int N = this.h;
|
||||||
|
|
||||||
|
int size = (N * 2) * (N * 2);
|
||||||
|
|
||||||
|
double[] u = new double[size]; // Force, X axis
|
||||||
|
double[] v = new double[size]; // Force, Y axis
|
||||||
|
double[] u_prev = new double[size];
|
||||||
|
double[] v_prev = new double[size];
|
||||||
|
double[] dens = new double[size];
|
||||||
|
double[] dens_prev = new double[size];
|
||||||
|
|
||||||
|
nsDoublesToBuffer(this.map, N, ref dens);
|
||||||
|
nsDoublesToBuffer(this.map, N, ref dens_prev);
|
||||||
|
|
||||||
|
for (int i = 0; i < rounds; i++)
|
||||||
|
{
|
||||||
|
u_prev = u;
|
||||||
|
v_prev = v;
|
||||||
|
dens_prev = dens;
|
||||||
|
|
||||||
|
nsVelStep(N, ref u, ref v, ref u_prev, ref v_prev, visc, dt);
|
||||||
|
nsDensStep(N, ref dens, ref dens_prev, ref u, ref v, diff, dt);
|
||||||
|
}
|
||||||
|
|
||||||
|
nsBufferToDoubles(ref u, N, ref uret);
|
||||||
|
nsBufferToDoubles(ref v, N, ref vret);
|
||||||
|
nsBufferToDoubles(ref dens, N, ref this.map);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
Loading…
Reference in New Issue