Skip to content

Commit c26eb87

Browse files
committedDec 15, 2015
Mgfractal: Add 3D and 4D fractals
3D Mandelbrot/Mandelbar 3D Christmas Tree 3D Mandelbulb 3D Cosine Mandelbulb 4D Mandelbulb Plus corresponding julia set for each Add credits for formulas Rename parameter 'formula' to 'fractal' Speed optimisations
1 parent aed1076 commit c26eb87

File tree

4 files changed

+144
-58
lines changed

4 files changed

+144
-58
lines changed
 

‎builtin/settingtypes.txt

+31-20
Original file line numberDiff line numberDiff line change
@@ -970,49 +970,60 @@ mgflat_np_cave2 (Mapgen flat cave2 noise parameters) noise_params 0, 12, (128, 1
970970

971971
[***Mapgen fractal]
972972

973-
# Choice of 8 4-dimensional fractals.
974-
# 1 = "Roundy" mandelbrot set.
975-
# 2 = "Roundy" julia set.
976-
# 3 = "Squarry" mandelbrot set.
977-
# 4 = "Squarry" julia set.
978-
# 5 = "Mandy Cousin" mandelbrot set.
979-
# 6 = "Mandy Cousin" julia set.
980-
# 7 = "Variation" mandelbrot set.
981-
# 8 = "Variation" julia set.
982-
mgfractal_formula (Mapgen fractal formula) int 1 1 8
973+
# Choice of 18 fractals from 9 formulas.
974+
# 1 = 4D "Roundy" mandelbrot set.
975+
# 2 = 4D "Roundy" julia set.
976+
# 3 = 4D "Squarry" mandelbrot set.
977+
# 4 = 4D "Squarry" julia set.
978+
# 5 = 4D "Mandy Cousin" mandelbrot set.
979+
# 6 = 4D "Mandy Cousin" julia set.
980+
# 7 = 4D "Variation" mandelbrot set.
981+
# 8 = 4D "Variation" julia set.
982+
# 9 = 3D "Mandelbrot/Mandelbar" mandelbrot set.
983+
# 10 = 3D "Mandelbrot/Mandelbar" julia set.
984+
# 11 = 3D "Christmas Tree" mandelbrot set.
985+
# 12 = 3D "Christmas Tree" julia set.
986+
# 13 = 3D "Mandelbulb" mandelbrot set.
987+
# 14 = 3D "Mandelbulb" julia set.
988+
# 15 = 3D "Cosine Mandelbulb" mandelbrot set.
989+
# 16 = 3D "Cosine Mandelbulb" julia set.
990+
# 17 = 4D "Mandelbulb" mandelbrot set.
991+
# 18 = 4D "Mandelbulb" julia set.
992+
mgfractal_fractal (Mapgen fractal fractal) int 1 1 18
983993

984994
# Iterations of the recursive function.
985-
# Controls scale of finest detail.
995+
# Controls the amount of fine detail.
986996
mgfractal_iterations (Mapgen fractal iterations) int 11
987997

988998
# Approximate (X,Y,Z) scale of fractal in nodes.
989999
mgfractal_scale (Mapgen fractal scale) v3f (4096.0, 1024.0, 4096.0)
9901000

991-
# (X,Y,Z) offset of fractal from world centre.
1001+
# (X,Y,Z) offset of fractal from world centre in units of 'scale'.
9921002
# Used to move a suitable spawn area of low land close to (0, 0).
993-
# The default is suitable for mandelbrot sets, it needs to be edited for julia sets,
994-
# do this by greatly reducing 'scale' and setting 'offset' initially to (0, 0, 0).
1003+
# The default is suitable for mandelbrot sets, it needs to be edited for julia sets.
9951004
# Range roughly -2 to 2. Multiply by 'scale' for offset in nodes.
9961005
mgfractal_offset (Mapgen fractal offset) v3f (1.79, 0.0, 0.0)
9971006

998-
# W co-ordinate of the generated 3D slice of the 4D shape.
999-
# Alters the generated 3D shape.
1007+
# W co-ordinate of the generated 3D slice of a 4D fractal.
1008+
# Determines which 3D slice of the 4D shape is generated.
1009+
# Has no effect on 3D fractals.
10001010
# Range roughly -2 to 2.
10011011
mgfractal_slice_w (Mapgen fractal slice w) float 0.0
10021012

1003-
# Julia set only: X value determining the 4D shape.
1013+
# Julia set only: X component of hypercomplex constant determining julia shape.
10041014
# Range roughly -2 to 2.
10051015
mgfractal_julia_x (Mapgen fractal julia x) float 0.33
10061016

1007-
# Julia set only: Y value determining the 4D shape.
1017+
# Julia set only: Y component of hypercomplex constant determining julia shape.
10081018
# Range roughly -2 to 2.
10091019
mgfractal_julia_y (Mapgen fractal julia y) float 0.33
10101020

1011-
# Julia set only: Z value determining the 4D shape.
1021+
# Julia set only: Z component of hypercomplex constant determining julia shape.
10121022
# Range roughly -2 to 2.
10131023
mgfractal_julia_z (Mapgen fractal julia z) float 0.33
10141024

1015-
# Julia set only: W value determining the 4D shape.
1025+
# Julia set only: W component of hypercomplex constant determining julia shape.
1026+
# Has no effect on 3D fractals.
10161027
# Range roughly -2 to 2.
10171028
mgfractal_julia_w (Mapgen fractal julia w) float 0.33
10181029

‎minetest.conf.example

+32-21
Original file line numberDiff line numberDiff line change
@@ -1234,57 +1234,68 @@
12341234

12351235
#### Mapgen fractal
12361236

1237-
# Choice of 8 4-dimensional fractals.
1238-
# 1 = "Roundy" mandelbrot set.
1239-
# 2 = "Roundy" julia set.
1240-
# 3 = "Squarry" mandelbrot set.
1241-
# 4 = "Squarry" julia set.
1242-
# 5 = "Mandy Cousin" mandelbrot set.
1243-
# 6 = "Mandy Cousin" julia set.
1244-
# 7 = "Variation" mandelbrot set.
1245-
# 8 = "Variation" julia set.
1246-
# type: int min: 1 max: 8
1247-
# mgfractal_formula = 1
1237+
# Choice of 18 fractals from 9 formulas.
1238+
# 1 = 4D "Roundy" mandelbrot set.
1239+
# 2 = 4D "Roundy" julia set.
1240+
# 3 = 4D "Squarry" mandelbrot set.
1241+
# 4 = 4D "Squarry" julia set.
1242+
# 5 = 4D "Mandy Cousin" mandelbrot set.
1243+
# 6 = 4D "Mandy Cousin" julia set.
1244+
# 7 = 4D "Variation" mandelbrot set.
1245+
# 8 = 4D "Variation" julia set.
1246+
# 9 = 3D "Mandelbrot/Mandelbar" mandelbrot set.
1247+
# 10 = 3D "Mandelbrot/Mandelbar" julia set.
1248+
# 11 = 3D "Christmas Tree" mandelbrot set.
1249+
# 12 = 3D "Christmas Tree" julia set.
1250+
# 13 = 3D "Mandelbulb" mandelbrot set.
1251+
# 14 = 3D "Mandelbulb" julia set.
1252+
# 15 = 3D "Cosine Mandelbulb" mandelbrot set.
1253+
# 16 = 3D "Cosine Mandelbulb" julia set.
1254+
# 17 = 4D "Mandelbulb" mandelbrot set.
1255+
# 18 = 4D "Mandelbulb" julia set.
1256+
# type: int min: 1 max: 18
1257+
# mgfractal_fractal = 1
12481258

12491259
# Iterations of the recursive function.
1250-
# Controls scale of finest detail.
1260+
# Controls the amount of fine detail.
12511261
# type: int
12521262
# mgfractal_iterations = 11
12531263

12541264
# Approximate (X,Y,Z) scale of fractal in nodes.
12551265
# type: v3f
12561266
# mgfractal_scale = (4096.0, 1024.0, 4096.0)
12571267

1258-
# (X,Y,Z) offset of fractal from world centre.
1268+
# (X,Y,Z) offset of fractal from world centre in units of 'scale'.
12591269
# Used to move a suitable spawn area of low land close to (0, 0).
1260-
# The default is suitable for mandelbrot sets, it needs to be edited for julia sets,
1261-
# do this by greatly reducing 'scale' and setting 'offset' initially to (0, 0, 0).
1270+
# The default is suitable for mandelbrot sets, it needs to be edited for julia sets.
12621271
# Range roughly -2 to 2. Multiply by 'scale' for offset in nodes.
12631272
# type: v3f
12641273
# mgfractal_offset = (1.79, 0.0, 0.0)
12651274

1266-
# W co-ordinate of the generated 3D slice of the 4D shape.
1267-
# Alters the generated 3D shape.
1275+
# W co-ordinate of the generated 3D slice of a 4D fractal.
1276+
# Determines which 3D slice of the 4D shape is generated.
1277+
# Has no effect on 3D fractals.
12681278
# Range roughly -2 to 2.
12691279
# type: float
12701280
# mgfractal_slice_w = 0.0
12711281

1272-
# Julia set only: X value determining the 4D shape.
1282+
# Julia set only: X component of hypercomplex constant determining julia shape.
12731283
# Range roughly -2 to 2.
12741284
# type: float
12751285
# mgfractal_julia_x = 0.33
12761286

1277-
# Julia set only: Y value determining the 4D shape.
1287+
# Julia set only: Y component of hypercomplex constant determining julia shape.
12781288
# Range roughly -2 to 2.
12791289
# type: float
12801290
# mgfractal_julia_y = 0.33
12811291

1282-
# Julia set only: Z value determining the 4D shape.
1292+
# Julia set only: Z component of hypercomplex constant determining julia shape.
12831293
# Range roughly -2 to 2.
12841294
# type: float
12851295
# mgfractal_julia_z = 0.33
12861296

1287-
# Julia set only: W value determining the 4D shape.
1297+
# Julia set only: W component of hypercomplex constant determining julia shape.
1298+
# Has no effect on 3D fractals.
12881299
# Range roughly -2 to 2.
12891300
# type: float
12901301
# mgfractal_julia_w = 0.33

‎src/mapgen_fractal.cpp

+72-14
Original file line numberDiff line numberDiff line change
@@ -66,7 +66,7 @@ MapgenFractal::MapgenFractal(int mapgenid, MapgenParams *params, EmergeManager *
6666
MapgenFractalParams *sp = (MapgenFractalParams *)params->sparams;
6767
this->spflags = sp->spflags;
6868

69-
this->formula = sp->formula;
69+
this->fractal = sp->fractal;
7070
this->iterations = sp->iterations;
7171
this->scale = sp->scale;
7272
this->offset = sp->offset;
@@ -77,6 +77,9 @@ MapgenFractal::MapgenFractal(int mapgenid, MapgenParams *params, EmergeManager *
7777
this->julia_z = sp->julia_z;
7878
this->julia_w = sp->julia_w;
7979

80+
this->formula = fractal / 2 + fractal % 2;
81+
this->julia = fractal % 2 == 0;
82+
8083
//// 2D terrain noise
8184
noise_seabed = new Noise(&sp->np_seabed, seed, csize.X, csize.Z);
8285
noise_filler_depth = new Noise(&sp->np_filler_depth, seed, csize.X, csize.Z);
@@ -141,7 +144,7 @@ MapgenFractalParams::MapgenFractalParams()
141144
{
142145
spflags = 0;
143146

144-
formula = 1;
147+
fractal = 1;
145148
iterations = 11;
146149
scale = v3f(4096.0, 1024.0, 4096.0);
147150
offset = v3f(1.79, 0.0, 0.0);
@@ -163,7 +166,7 @@ void MapgenFractalParams::readParams(const Settings *settings)
163166
{
164167
settings->getFlagStrNoEx("mgfractal_spflags", spflags, flagdesc_mapgen_fractal);
165168

166-
settings->getU16NoEx("mgfractal_formula", formula);
169+
settings->getU16NoEx("mgfractal_fractal", fractal);
167170
settings->getU16NoEx("mgfractal_iterations", iterations);
168171
settings->getV3FNoEx("mgfractal_scale", scale);
169172
settings->getV3FNoEx("mgfractal_offset", offset);
@@ -185,7 +188,7 @@ void MapgenFractalParams::writeParams(Settings *settings) const
185188
{
186189
settings->setFlagStr("mgfractal_spflags", spflags, flagdesc_mapgen_fractal, U32_MAX);
187190

188-
settings->setU16("mgfractal_formula", formula);
191+
settings->setU16("mgfractal_fractal", fractal);
189192
settings->setU16("mgfractal_iterations", iterations);
190193
settings->setV3F("mgfractal_scale", scale);
191194
settings->setV3F("mgfractal_offset", offset);
@@ -368,7 +371,7 @@ bool MapgenFractal::getFractalAtPoint(s16 x, s16 y, s16 z)
368371
{
369372
float cx, cy, cz, cw, ox, oy, oz, ow;
370373

371-
if (formula % 2 == 0) { // Julia sets, formula = 2, 4, 6, 8
374+
if (julia) { // Julia set
372375
cx = julia_x;
373376
cy = julia_y;
374377
cz = julia_z;
@@ -377,7 +380,7 @@ bool MapgenFractal::getFractalAtPoint(s16 x, s16 y, s16 z)
377380
oy = (float)y / scale.Y - offset.Y;
378381
oz = (float)z / scale.Z - offset.Z;
379382
ow = slice_w;
380-
} else { // Mandelbrot sets, formula = 1, 3, 5, 7
383+
} else { // Mandelbrot set
381384
cx = (float)x / scale.X - offset.X;
382385
cy = (float)y / scale.Y - offset.Y;
383386
cz = (float)z / scale.Z - offset.Z;
@@ -388,32 +391,87 @@ bool MapgenFractal::getFractalAtPoint(s16 x, s16 y, s16 z)
388391
ow = 0.0f;
389392
}
390393

394+
float nx = 0.0f;
395+
float ny = 0.0f;
396+
float nz = 0.0f;
397+
float nw = 0.0f;
398+
391399
for (u16 iter = 0; iter < iterations; iter++) {
392-
float nx = 0.0f;
393-
float ny = 0.0f;
394-
float nz = 0.0f;
395-
float nw = 0.0f;
396400

397-
if (formula == 1 || formula == 2) { // 4D "Roundy" Mandelbrot/Julia Set
401+
if (formula == 1) { // 4D "Roundy"
398402
nx = ox * ox - oy * oy - oz * oz - ow * ow + cx;
399403
ny = 2.0f * (ox * oy + oz * ow) + cy;
400404
nz = 2.0f * (ox * oz + oy * ow) + cz;
401405
nw = 2.0f * (ox * ow + oy * oz) + cw;
402-
} else if (formula == 3 || formula == 4) { // 4D "Squarry" Mandelbrot/Julia Set
406+
} else if (formula == 2) { // 4D "Squarry"
403407
nx = ox * ox - oy * oy - oz * oz - ow * ow + cx;
404408
ny = 2.0f * (ox * oy + oz * ow) + cy;
405409
nz = 2.0f * (ox * oz + oy * ow) + cz;
406410
nw = 2.0f * (ox * ow - oy * oz) + cw;
407-
} else if (formula == 5 || formula == 6) { // 4D "Mandy Cousin" Mandelbrot/Julia Set
411+
} else if (formula == 3) { // 4D "Mandy Cousin"
408412
nx = ox * ox - oy * oy - oz * oz + ow * ow + cx;
409413
ny = 2.0f * (ox * oy + oz * ow) + cy;
410414
nz = 2.0f * (ox * oz + oy * ow) + cz;
411415
nw = 2.0f * (ox * ow + oy * oz) + cw;
412-
} else if (formula == 7 || formula == 8) { // 4D "Variation" Mandelbrot/Julia Set
416+
} else if (formula == 4) { // 4D "Variation"
413417
nx = ox * ox - oy * oy - oz * oz - ow * ow + cx;
414418
ny = 2.0f * (ox * oy + oz * ow) + cy;
415419
nz = 2.0f * (ox * oz - oy * ow) + cz;
416420
nw = 2.0f * (ox * ow + oy * oz) + cw;
421+
} else if (formula == 5) { // 3D "Mandelbrot/Mandelbar"
422+
nx = ox * ox - oy * oy - oz * oz + cx;
423+
ny = 2.0f * ox * oy + cy;
424+
nz = -2.0f * ox * oz + cz;
425+
} else if (formula == 6) { // 3D "Christmas Tree"
426+
// Altering the formula here is necessary to avoid division by zero
427+
if (fabs(oz) < 0.000000001f) {
428+
nx = ox * ox - oy * oy - oz * oz + cx;
429+
ny = 2.0f * oy * ox + cy;
430+
nz = 4.0f * oz * ox + cz;
431+
} else {
432+
float a = (2.0f * ox) / (sqrt(oy * oy + oz * oz));
433+
nx = ox * ox - oy * oy - oz * oz + cx;
434+
ny = a * (oy * oy - oz * oz) + cy;
435+
nz = a * 2.0f * oy * oz + cz;
436+
}
437+
} else if (formula == 7) { // 3D "Mandelbulb"
438+
if (fabs(oy) < 0.000000001f) {
439+
nx = ox * ox - oz * oz + cx;
440+
ny = cy;
441+
nz = -2.0f * oz * sqrt(ox * ox) + cz;
442+
} else {
443+
float a = 1.0f - (oz * oz) / (ox * ox + oy * oy);
444+
nx = (ox * ox - oy * oy) * a + cx;
445+
ny = 2.0f * ox * oy * a + cy;
446+
nz = -2.0f * oz * sqrt(ox * ox + oy * oy) + cz;
447+
}
448+
} else if (formula == 8) { // 3D "Cosine Mandelbulb"
449+
if (fabs(oy) < 0.000000001f) {
450+
nx = 2.0f * ox * oz + cx;
451+
ny = 4.0f * oy * oz + cy;
452+
nz = oz * oz - ox * ox - oy * oy + cz;
453+
} else {
454+
float a = (2.0f * oz) / sqrt(ox * ox + oy * oy);
455+
nx = (ox * ox - oy * oy) * a + cx;
456+
ny = 2.0f * ox * oy * a + cy;
457+
nz = oz * oz - ox * ox - oy * oy + cz;
458+
}
459+
} else if (formula == 9) { // 4D "Mandelbulb"
460+
float rxy = sqrt(ox * ox + oy * oy);
461+
float rxyz = sqrt(ox * ox + oy * oy + oz * oz);
462+
if (fabs(ow) < 0.000000001f && fabs(oz) < 0.000000001f) {
463+
nx = (ox * ox - oy * oy) + cx;
464+
ny = 2.0f * ox * oy + cy;
465+
nz = -2.0f * rxy * oz + cz;
466+
nw = 2.0f * rxyz * ow + cw;
467+
} else {
468+
float a = 1.0f - (ow * ow) / (rxyz * rxyz);
469+
float b = a * (1.0f - (oz * oz) / (rxy * rxy));
470+
nx = (ox * ox - oy * oy) * b + cx;
471+
ny = 2.0f * ox * oy * b + cy;
472+
nz = -2.0f * rxy * oz * a + cz;
473+
nw = 2.0f * rxyz * ow + cw;
474+
}
417475
}
418476

419477
if (nx * nx + ny * ny + nz * nz + nw * nw > 4.0f)

‎src/mapgen_fractal.h

+9-3
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,9 @@ Minetest
33
Copyright (C) 2010-2015 kwolekr, Ryan Kwolek <kwolekr@minetest.net>
44
Copyright (C) 2010-2015 paramat, Matt Gregory
55
6+
Fractal formulas from http://www.bugman123.com/Hypercomplex/index.html
7+
by Paul Nylander, and from http://www.fractalforums.com, thank you.
8+
69
This program is free software; you can redistribute it and/or modify
710
it under the terms of the GNU Lesser General Public License as published by
811
the Free Software Foundation; either version 2.1 of the License, or
@@ -33,7 +36,7 @@ extern FlagDesc flagdesc_mapgen_fractal[];
3336
struct MapgenFractalParams : public MapgenSpecificParams {
3437
u32 spflags;
3538

36-
u16 formula;
39+
u16 fractal;
3740
u16 iterations;
3841
v3f scale;
3942
v3f offset;
@@ -63,14 +66,17 @@ class MapgenFractal : public Mapgen {
6366

6467
int ystride;
6568
int zstride;
66-
u32 spflags;
69+
u16 formula;
70+
bool julia;
6771

6872
v3s16 node_min;
6973
v3s16 node_max;
7074
v3s16 full_node_min;
7175
v3s16 full_node_max;
7276

73-
u16 formula;
77+
u32 spflags;
78+
79+
u16 fractal;
7480
u16 iterations;
7581
v3f scale;
7682
v3f offset;

0 commit comments

Comments
 (0)
Please sign in to comment.