mirror of
https://gitea.wildfiregames.com/0ad/0ad
synced 2026-06-19 06:43:58 -07:00
An alternative to 6a43f91630 which doesn't restrict
`playerPlacementByPattern` to return exactly two values.
425 lines
13 KiB
JavaScript
425 lines
13 KiB
JavaScript
Engine.LoadLibrary("rmgen");
|
|
Engine.LoadLibrary("rmgen-common");
|
|
Engine.LoadLibrary("rmbiome");
|
|
|
|
export function* generateMap(mapSettings)
|
|
{
|
|
setBiome(mapSettings.Biome);
|
|
|
|
const tMainTerrain = g_Terrains.mainTerrain;
|
|
const tForestFloor1 = g_Terrains.forestFloor1;
|
|
const tForestFloor2 = g_Terrains.forestFloor2;
|
|
const tCliff = g_Terrains.cliff;
|
|
const tTier1Terrain = g_Terrains.tier1Terrain;
|
|
const tTier2Terrain = g_Terrains.tier2Terrain;
|
|
const tTier3Terrain = g_Terrains.tier3Terrain;
|
|
const tHill = g_Terrains.hill;
|
|
const tRoad = g_Terrains.road;
|
|
const tRoadWild = g_Terrains.roadWild;
|
|
const tTier4Terrain = g_Terrains.tier4Terrain;
|
|
const tShore = g_Terrains.shore;
|
|
const tWater = g_Terrains.water;
|
|
|
|
const oTree1 = g_Gaia.tree1;
|
|
const oTree2 = g_Gaia.tree2;
|
|
const oTree3 = g_Gaia.tree3;
|
|
const oTree4 = g_Gaia.tree4;
|
|
const oTree5 = g_Gaia.tree5;
|
|
const oFruitBush = g_Gaia.fruitBush;
|
|
const oMainHuntableAnimal = g_Gaia.mainHuntableAnimal;
|
|
const oFish = g_Gaia.fish;
|
|
const oSecondaryHuntableAnimal = g_Gaia.secondaryHuntableAnimal;
|
|
const oStoneLarge = g_Gaia.stoneLarge;
|
|
const oStoneSmall = g_Gaia.stoneSmall;
|
|
const oMetalLarge = g_Gaia.metalLarge;
|
|
const oWoodTreasure = "gaia/treasure/wood";
|
|
const oDock = "skirmish/structures/default_dock";
|
|
|
|
const aGrass = g_Decoratives.grass;
|
|
const aGrassShort = g_Decoratives.grassShort;
|
|
const aRockLarge = g_Decoratives.rockLarge;
|
|
const aRockMedium = g_Decoratives.rockMedium;
|
|
const aBushMedium = g_Decoratives.bushMedium;
|
|
const aBushSmall = g_Decoratives.bushSmall;
|
|
|
|
const pForest1 = [tForestFloor2 + TERRAIN_SEPARATOR + oTree1, tForestFloor2 + TERRAIN_SEPARATOR + oTree2, tForestFloor2];
|
|
const pForest2 = [tForestFloor1 + TERRAIN_SEPARATOR + oTree4, tForestFloor1 + TERRAIN_SEPARATOR + oTree5, tForestFloor1];
|
|
|
|
const heightSeaGround = -5;
|
|
const heightLand = 3;
|
|
const heightHill = 18;
|
|
const heightOffsetBump = 2;
|
|
|
|
globalThis.g_Map = new RandomMap(heightSeaGround, tWater);
|
|
|
|
const numPlayers = getNumPlayers();
|
|
const mapSize = g_Map.getSize();
|
|
const mapCenter = g_Map.getCenter();
|
|
|
|
const clPlayer = g_Map.createTileClass();
|
|
const clHill = g_Map.createTileClass();
|
|
const clForest = g_Map.createTileClass();
|
|
const clForestIsland = g_Map.createTileClass();
|
|
const clDirt = g_Map.createTileClass();
|
|
const clRock = g_Map.createTileClass();
|
|
const clMetal = g_Map.createTileClass();
|
|
const clFood = g_Map.createTileClass();
|
|
const clIslandHunt = g_Map.createTileClass();
|
|
const clBaseResource = g_Map.createTileClass();
|
|
const clLand = g_Map.createTileClass();
|
|
const clIsland = g_Map.createTileClass();
|
|
|
|
const startAngle = randomAngle();
|
|
const pattern = g_MapSettings.PlayerPlacement;
|
|
|
|
const teamDist = {
|
|
"circle": 0.42,
|
|
"river": 0.65,
|
|
"stronghold": 0.44
|
|
}[pattern];
|
|
|
|
const playerDist = {
|
|
"circle": 0.1,
|
|
"river": 0.1,
|
|
"stronghold": 0.06
|
|
}[pattern];
|
|
|
|
const { playerIDs, playerPosition } =
|
|
playerPlacementByPattern(
|
|
pattern,
|
|
fractionToTiles(teamDist),
|
|
fractionToTiles(playerDist),
|
|
startAngle,
|
|
undefined);
|
|
|
|
g_Map.log("Creating player islands");
|
|
for (let i = 0; i < numPlayers; ++i)
|
|
{
|
|
createArea(
|
|
new ClumpPlacer(diskArea(defaultPlayerBaseRadius() * 1.75), 0.8, 0.1, Infinity, playerPosition[i]),
|
|
[
|
|
new LayeredPainter([tWater, tShore, tMainTerrain], [1, 4]),
|
|
new SmoothElevationPainter(ELEVATION_SET, heightLand, 4),
|
|
new TileClassPainter(clIsland),
|
|
new TileClassPainter(mapSettings.Nomad ? clLand : clPlayer)
|
|
]);
|
|
}
|
|
yield 10;
|
|
|
|
placePlayerBases({
|
|
"PlayerPlacement": [playerIDs, playerPosition],
|
|
"PlayerTileClass": clPlayer,
|
|
"BaseResourceClass": clBaseResource,
|
|
"Walls": false,
|
|
// No city patch
|
|
"StartingAnimal": {
|
|
},
|
|
"Berries": {
|
|
"template": oFruitBush
|
|
},
|
|
"Mines": {
|
|
"types": [
|
|
{ "template": oMetalLarge },
|
|
{ "template": oStoneLarge }
|
|
]
|
|
},
|
|
"Trees": {
|
|
"template": oTree1,
|
|
"count": scaleByMapSize(12, 30)
|
|
},
|
|
"Decoratives": {
|
|
"template": aGrassShort
|
|
}
|
|
});
|
|
yield 15;
|
|
|
|
g_Map.log("Create the continent body");
|
|
const continentPosition = Vector2D.add(mapCenter, new Vector2D(0, fractionToTiles(0))).round();
|
|
createArea(
|
|
new ClumpPlacer(diskArea(fractionToTiles(0.30)), 0.8, 0.08, Infinity, continentPosition),
|
|
[
|
|
new LayeredPainter([tWater, tShore, tMainTerrain], [4, 2]),
|
|
new SmoothElevationPainter(ELEVATION_SET, heightLand, 4),
|
|
new TileClassPainter(clLand)
|
|
],
|
|
avoidClasses(clIsland, 22));
|
|
yield 20;
|
|
|
|
g_Map.log("Creating shore jaggedness");
|
|
createAreas(
|
|
new ClumpPlacer(scaleByMapSize(10, 65), 0.2, 0.1, Infinity),
|
|
[
|
|
new LayeredPainter([tMainTerrain, tMainTerrain], [2]),
|
|
new SmoothElevationPainter(ELEVATION_SET, heightLand, 4),
|
|
new TileClassPainter(clLand)
|
|
],
|
|
[
|
|
borderClasses(clLand, 6, 3),
|
|
avoidClasses(clIsland, 16)
|
|
],
|
|
scaleByMapSize(2, 15) * 20,
|
|
150);
|
|
|
|
paintTerrainBasedOnHeight(1, 3, 0, tShore);
|
|
paintTerrainBasedOnHeight(-8, 1, 2, tWater);
|
|
yield 25;
|
|
|
|
g_Map.log("Creating bumps");
|
|
createAreas(
|
|
new ClumpPlacer(scaleByMapSize(20, 50), 0.3, 0.06, Infinity),
|
|
new SmoothElevationPainter(ELEVATION_MODIFY, heightOffsetBump, 2),
|
|
[avoidClasses(clIsland, 10), stayClasses(clLand, 3)],
|
|
scaleByMapSize(100, 200)
|
|
);
|
|
yield 30;
|
|
|
|
g_Map.log("Creating hills");
|
|
createAreas(
|
|
new ClumpPlacer(scaleByMapSize(20, 150), 0.2, 0.1, Infinity),
|
|
[
|
|
new LayeredPainter([tCliff, tHill], [2]),
|
|
new SmoothElevationPainter(ELEVATION_SET, heightHill, 2),
|
|
new TileClassPainter(clHill)
|
|
],
|
|
[avoidClasses(clIsland, 10, clHill, 50), stayClasses(clLand, 12)],
|
|
scaleByMapSize(1, 4) * numPlayers
|
|
);
|
|
yield 34;
|
|
|
|
g_Map.log("Creating forests");
|
|
{
|
|
const [forestTrees, stragglerTrees] = getTreeCounts(...rBiomeTreeCount(1));
|
|
const types = [
|
|
[[tForestFloor2, tMainTerrain, pForest1], [tForestFloor2, pForest1]],
|
|
[[tForestFloor1, tMainTerrain, pForest2], [tForestFloor1, pForest2]]
|
|
];
|
|
|
|
const size = forestTrees / (scaleByMapSize(2, 8) * numPlayers) *
|
|
(currentBiome() == "generic/savanna" ? 2 : 1);
|
|
|
|
const num = Math.floor(size / types.length);
|
|
for (const type of types)
|
|
createAreas(
|
|
new ClumpPlacer(forestTrees / num, 0.1, 0.1, Infinity),
|
|
[
|
|
new LayeredPainter(type, [2]),
|
|
new TileClassPainter(clForest)
|
|
],
|
|
[avoidClasses(clForest, 10, clHill, 2), stayClasses(clLand, 7)],
|
|
num);
|
|
}
|
|
yield 38;
|
|
|
|
g_Map.log("Creating dirt patches");
|
|
for (const dirtClumpSize of [scaleByMapSize(3, 48), scaleByMapSize(5, 84), scaleByMapSize(8, 128)])
|
|
createAreas(
|
|
new ClumpPlacer(dirtClumpSize, 0.3, 0.06, 0.5),
|
|
[
|
|
new LayeredPainter(
|
|
[[tMainTerrain, tTier1Terrain], [tTier1Terrain, tTier2Terrain], [tTier2Terrain, tTier3Terrain]],
|
|
[1, 1]),
|
|
new TileClassPainter(clDirt)
|
|
],
|
|
[
|
|
avoidClasses(
|
|
clForest, 0,
|
|
clHill, 0,
|
|
clDirt, 5,
|
|
clIsland, 0),
|
|
stayClasses(clLand, 7)
|
|
],
|
|
scaleByMapSize(15, 45));
|
|
|
|
yield 42;
|
|
|
|
g_Map.log("Creating grass patches");
|
|
for (const grassClumpSize of [scaleByMapSize(2, 32), scaleByMapSize(3, 48), scaleByMapSize(5, 80)])
|
|
createAreas(
|
|
new ClumpPlacer(grassClumpSize, 0.3, 0.06, 0.5),
|
|
new TerrainPainter(tTier4Terrain),
|
|
[avoidClasses(clForest, 0, clHill, 0, clDirt, 5, clIsland, 0), stayClasses(clLand, 7)],
|
|
scaleByMapSize(15, 45));
|
|
yield 46;
|
|
|
|
g_Map.log("Creating stone mines");
|
|
let group = new SimpleGroup([new SimpleObject(oStoneSmall, 0, 2, 0, 4, 0, 2 * Math.PI, 1), new SimpleObject(oStoneLarge, 1, 1, 0, 4, 0, 2 * Math.PI, 4)], true, clRock);
|
|
createObjectGroupsDeprecated(group, 0,
|
|
[avoidClasses(clForest, 0, clRock, 18, clHill, 1), stayClasses(clLand, 4)],
|
|
scaleByMapSize(26, 30), 800
|
|
);
|
|
yield 50;
|
|
|
|
g_Map.log("Creating metal mines");
|
|
group = new SimpleGroup([new SimpleObject(oMetalLarge, 1, 1, 0, 4)], true, clMetal);
|
|
createObjectGroupsDeprecated(group, 0,
|
|
[avoidClasses(clForest, 0, clMetal, 18, clRock, 8, clHill, 1), stayClasses(clLand, 4)],
|
|
scaleByMapSize(28, 32), 800
|
|
);
|
|
|
|
g_Map.log("Creating small stone quarries");
|
|
group = new SimpleGroup([new SimpleObject(oStoneSmall, 2, 5, 1, 3)], true, clRock);
|
|
createObjectGroupsDeprecated(group, 0,
|
|
[avoidClasses(clForest, 0, clMetal, 6, clRock, 12, clHill, 1), stayClasses(clLand, 4)],
|
|
scaleByMapSize(25, 30), 800
|
|
);
|
|
yield 54;
|
|
|
|
g_Map.log("Creating island forests");
|
|
var [forestTrees, stragglerTrees] = getTreeCounts(...rBiomeTreeCount(1));
|
|
var types = [
|
|
[[tForestFloor2, tMainTerrain, pForest1], [tForestFloor2, pForest1]],
|
|
[[tForestFloor1, tMainTerrain, pForest2], [tForestFloor1, pForest2]]
|
|
];
|
|
|
|
var size = forestTrees / (scaleByMapSize(2, 8) * numPlayers) *
|
|
(currentBiome() == "generic/savanna" ? 2 : 3);
|
|
|
|
var num = Math.floor((size / types.length));
|
|
for (const type of types)
|
|
createAreas(
|
|
new ClumpPlacer(forestTrees / num, 0.1, 0.1, Infinity),
|
|
[
|
|
new LayeredPainter(type, [2]),
|
|
new TileClassPainter(clForestIsland)
|
|
],
|
|
[avoidClasses(clBaseResource, 11, clForestIsland, 6, clHill, 0), stayClasses(clIsland, 5)],
|
|
num);
|
|
|
|
yield 58;
|
|
|
|
g_Map.log("Creating small decorative rocks");
|
|
group = new SimpleGroup(
|
|
[new SimpleObject(aRockMedium, 1, 3, 0, 1)],
|
|
true
|
|
);
|
|
createObjectGroupsDeprecated(
|
|
group, 0,
|
|
[avoidClasses(clForest, 0, clPlayer, 0, clHill, 0), stayClasses(clLand, 6)],
|
|
scaleByMapSize(16, 262), 50
|
|
);
|
|
yield 62;
|
|
|
|
g_Map.log("Creating large decorative rocks");
|
|
group = new SimpleGroup(
|
|
[new SimpleObject(aRockLarge, 1, 2, 0, 1), new SimpleObject(aRockMedium, 1, 3, 0, 2)],
|
|
true
|
|
);
|
|
createObjectGroupsDeprecated(
|
|
group, 0,
|
|
[avoidClasses(clForest, 0, clPlayer, 0, clHill, 0), stayClasses(clLand, 6)],
|
|
scaleByMapSize(8, 131), 50
|
|
);
|
|
yield 66;
|
|
|
|
g_Map.log("Creating deer");
|
|
group = new SimpleGroup(
|
|
[new SimpleObject(oMainHuntableAnimal, 5, 7, 0, 4)],
|
|
true, clFood
|
|
);
|
|
createObjectGroupsDeprecated(group, 0,
|
|
[avoidClasses(clForest, 0, clPlayer, 10, clHill, 1, clFood, 20), stayClasses(clLand, 7)],
|
|
3 * numPlayers, 50
|
|
);
|
|
|
|
if (currentBiome() == "generic/savanna")
|
|
{
|
|
g_Map.log("Creating Island Fauna");
|
|
group = new SimpleGroup(
|
|
[new SimpleObject(oMainHuntableAnimal, 5, 7, 0, 4)],
|
|
true, clIslandHunt
|
|
);
|
|
createObjectGroupsDeprecated(group, 0,
|
|
[avoidClasses(clBaseResource, 6, clIslandHunt, 58), stayClasses(clIsland, 12)],
|
|
numPlayers * 20, 400
|
|
);
|
|
}
|
|
|
|
yield 70;
|
|
|
|
g_Map.log("Creating sheep");
|
|
group = new SimpleGroup(
|
|
[new SimpleObject(oSecondaryHuntableAnimal, 2, 3, 0, 2)],
|
|
true, clFood
|
|
);
|
|
createObjectGroupsDeprecated(group, 0,
|
|
[avoidClasses(clForest, 0, clPlayer, 10, clHill, 1, clFood, 20), stayClasses(clLand, 7)],
|
|
3 * numPlayers, 50
|
|
);
|
|
yield 74;
|
|
|
|
g_Map.log("Creating fruit bush");
|
|
group = new SimpleGroup(
|
|
[new SimpleObject(oFruitBush, 5, 7, 0, 4)],
|
|
true, clFood
|
|
);
|
|
createObjectGroupsDeprecated(group, 0,
|
|
[avoidClasses(clForest, 0, clPlayer, 8, clHill, 1, clFood, 20), stayClasses(clLand, 7)],
|
|
randIntInclusive(1, 4) * numPlayers + 2, 50
|
|
);
|
|
yield 78;
|
|
|
|
g_Map.log("Creating fish");
|
|
createObjectGroupsDeprecated(
|
|
new SimpleGroup([new SimpleObject(oFish, 2, 3, 0, 2)], true, clFood),
|
|
0,
|
|
avoidClasses(clLand, 2, clPlayer, 2, clHill, 0, clFood, 10),
|
|
scaleByMapSize(500, 700), 500
|
|
);
|
|
yield 82;
|
|
|
|
createStragglerTrees(
|
|
[oTree1, oTree2, oTree4, oTree3],
|
|
[avoidClasses(clForest, 1, clHill, 1, clPlayer, 9, clMetal, 6, clRock, 6), stayClasses(clLand, 9)],
|
|
clForest,
|
|
stragglerTrees);
|
|
|
|
createStragglerTrees(
|
|
[oTree1, oTree2, oTree4, oTree3],
|
|
[avoidClasses(clForestIsland, 1, clBaseResource, 8), stayClasses(clIsland, 5)],
|
|
clForest,
|
|
stragglerTrees);
|
|
|
|
|
|
yield 85;
|
|
|
|
const planetm = currentBiome() == "generic/india" ? 8 : 1;
|
|
|
|
g_Map.log("Creating small grass tufts");
|
|
group = new SimpleGroup(
|
|
[new SimpleObject(aGrassShort, 1, 2, 0, 1, -Math.PI / 8, Math.PI / 8)]
|
|
);
|
|
createObjectGroupsDeprecated(group, 0,
|
|
[avoidClasses(clHill, 2, clPlayer, 2, clDirt, 0), stayClasses(clLand, 6)],
|
|
planetm * scaleByMapSize(13, 200)
|
|
);
|
|
yield 90;
|
|
|
|
g_Map.log("Creating large grass tufts");
|
|
group = new SimpleGroup(
|
|
[new SimpleObject(aGrass, 2, 4, 0, 1.8, -Math.PI / 8, Math.PI / 8), new SimpleObject(aGrassShort, 3, 6, 1.2, 2.5, -Math.PI / 8, Math.PI / 8)]
|
|
);
|
|
createObjectGroupsDeprecated(group, 0,
|
|
[avoidClasses(clHill, 2, clPlayer, 2, clDirt, 1, clForest, 0), stayClasses(clLand, 6)],
|
|
planetm * scaleByMapSize(13, 200)
|
|
);
|
|
yield 94;
|
|
|
|
g_Map.log("Creating bushes");
|
|
group = new SimpleGroup(
|
|
[new SimpleObject(aBushMedium, 1, 2, 0, 2), new SimpleObject(aBushSmall, 2, 4, 0, 2)]
|
|
);
|
|
createObjectGroupsDeprecated(group, 0,
|
|
[avoidClasses(clHill, 1, clPlayer, 1, clDirt, 1), stayClasses(clLand, 6)],
|
|
planetm * scaleByMapSize(13, 200), 50
|
|
);
|
|
yield 98;
|
|
|
|
setSkySet(pickRandom(["cirrus", "cumulus", "sunny"]));
|
|
setSunRotation(randomAngle());
|
|
setSunElevation(randFloat(1 / 5, 1 / 3) * Math.PI);
|
|
setWaterWaviness(2);
|
|
|
|
placePlayersNomad(clPlayer, [stayClasses(clIsland, 4), avoidClasses(clForest, 1, clMetal, 4, clRock, 4, clHill, 4, clFood, 2)]);
|
|
|
|
return g_Map;
|
|
}
|