From 17e8f78857553484dca599d7458df64f9c30e0d2 Mon Sep 17 00:00:00 2001 From: Vadim Lopatin Date: Mon, 25 Apr 2016 08:39:24 +0300 Subject: [PATCH] dminer example, initial versin --- dlangui-msvc.sln | 15 + dub.json | 1 + examples/dminer/dminer.visualdproj | 220 ++++++ examples/dminer/dub.json | 21 + examples/dminer/src/dminer/core/blocks.d | 287 ++++++++ examples/dminer/src/dminer/core/minetypes.d | 731 ++++++++++++++++++++ examples/dminer/src/dminer/core/terrain.d | 153 ++++ examples/dminer/src/dminer/core/world.d | 501 ++++++++++++++ examples/dminer/src/minermain.d | 380 ++++++++++ examples/dminer/views/res/i18n/en.ini | 40 ++ examples/dminer/views/res/i18n/ru.ini | 34 + examples/dminer/views/res/mdpi/blocks.png | Bin 0 -> 31980 bytes examples/dminer/views/res/mdpi/cr3_logo.png | Bin 0 -> 8433 bytes examples/dminer/views/resources.list | 4 + 14 files changed, 2387 insertions(+) create mode 100644 examples/dminer/dminer.visualdproj create mode 100644 examples/dminer/dub.json create mode 100644 examples/dminer/src/dminer/core/blocks.d create mode 100644 examples/dminer/src/dminer/core/minetypes.d create mode 100644 examples/dminer/src/dminer/core/terrain.d create mode 100644 examples/dminer/src/dminer/core/world.d create mode 100644 examples/dminer/src/minermain.d create mode 100644 examples/dminer/views/res/i18n/en.ini create mode 100644 examples/dminer/views/res/i18n/ru.ini create mode 100644 examples/dminer/views/res/mdpi/blocks.png create mode 100644 examples/dminer/views/res/mdpi/cr3_logo.png create mode 100644 examples/dminer/views/resources.list diff --git a/dlangui-msvc.sln b/dlangui-msvc.sln index 0ed91bba..c5a5adc3 100644 --- a/dlangui-msvc.sln +++ b/dlangui-msvc.sln @@ -50,6 +50,11 @@ Project("{002A2DE9-8BB6-484D-9802-7E4AD4084715}") = "ircclient", "examples\irccl {52A2ABB9-2CF7-4D5F-AE8C-75B21F8585A5} = {52A2ABB9-2CF7-4D5F-AE8C-75B21F8585A5} EndProjectSection EndProject +Project("{002A2DE9-8BB6-484D-9802-7E4AD4084715}") = "dminer", "examples\dminer\dminer.visualdproj", "{5F443F6A-6612-4404-B89E-D0D0205DC8E5}" + ProjectSection(ProjectDependencies) = postProject + {52A2ABB9-2CF7-4D5F-AE8C-75B21F8585A5} = {52A2ABB9-2CF7-4D5F-AE8C-75B21F8585A5} + EndProjectSection +EndProject Global GlobalSection(SolutionConfigurationPlatforms) = preSolution Debug|Win32 = Debug|Win32 @@ -173,6 +178,16 @@ Global {20722E6B-CA27-467F-8BB8-07F80106B478}.Unittest|Win32.Build.0 = Release|Win32 {20722E6B-CA27-467F-8BB8-07F80106B478}.Unittest|x64.ActiveCfg = Release|Win32 {20722E6B-CA27-467F-8BB8-07F80106B478}.Unittest|x64.Build.0 = Release|Win32 + {5F443F6A-6612-4404-B89E-D0D0205DC8E5}.Debug|Win32.ActiveCfg = Debug|Win32 + {5F443F6A-6612-4404-B89E-D0D0205DC8E5}.Debug|Win32.Build.0 = Debug|Win32 + {5F443F6A-6612-4404-B89E-D0D0205DC8E5}.Debug|x64.ActiveCfg = Debug|Win32 + {5F443F6A-6612-4404-B89E-D0D0205DC8E5}.Release|Win32.ActiveCfg = Release|Win32 + {5F443F6A-6612-4404-B89E-D0D0205DC8E5}.Release|Win32.Build.0 = Release|Win32 + {5F443F6A-6612-4404-B89E-D0D0205DC8E5}.Release|x64.ActiveCfg = Release|Win32 + {5F443F6A-6612-4404-B89E-D0D0205DC8E5}.Unittest|Win32.ActiveCfg = Release|Win32 + {5F443F6A-6612-4404-B89E-D0D0205DC8E5}.Unittest|Win32.Build.0 = Release|Win32 + {5F443F6A-6612-4404-B89E-D0D0205DC8E5}.Unittest|x64.ActiveCfg = Release|Win32 + {5F443F6A-6612-4404-B89E-D0D0205DC8E5}.Unittest|x64.Build.0 = Release|Win32 EndGlobalSection GlobalSection(SolutionProperties) = preSolution HideSolutionNode = FALSE diff --git a/dub.json b/dub.json index 8365a6ed..e4fd8c63 100644 --- a/dub.json +++ b/dub.json @@ -36,6 +36,7 @@ "./examples/example1/", "./examples/dmledit/", "./examples/d3d/", + "./examples/dminer/", "./examples/tetris/", "./examples/opengl/", "./examples/ircclient/", diff --git a/examples/dminer/dminer.visualdproj b/examples/dminer/dminer.visualdproj new file mode 100644 index 00000000..e2fe6214 --- /dev/null +++ b/examples/dminer/dminer.visualdproj @@ -0,0 +1,220 @@ + + {5F443F6A-6612-4404-B89E-D0D0205DC8E5} + + 0 + 0 + 0 + 2 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 3 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 2.043 + 0 + 0 + 0 + 0 + 0 + $(CC) -c + 1 + $(DMDInstallDir)windows\bin\dmd.exe + $(SolutionDir)/src $(SolutionDir)/3rdparty $(SolutionDir)/deps/DerelictGL3/source $(SolutionDir)/deps/DerelictUtil/source $(SolutionDir)/deps/DerelictFT/source $(SolutionDir)/deps/DerelictSDL2/source + views views/res views/res/i18n views/res/mdpi views/res/hdpi + $(ConfigurationName) + $(OutDir) + + + 0 + + + + + 0 + + + 1 + $(IntDir)\$(TargetName).json + 0 + + 0 + + 0 + 0 + 0 + + + + 0 + + 1 + $(VisualDInstallDir)cv2pdb\cv2pdb.exe + 0 + 0 + 0 + + + + ole32.lib kernel32.lib user32.lib comctl32.lib comdlg32.lib + + + + $(OutDir)\$(ProjectName).exe + 1 + 2 + 0 + + + + *.obj;*.cmd;*.build;*.json;*.dep + + + 0 + 0 + 0 + 2 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 1 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 1 + 1 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 0 + 2.043 + 0 + 0 + 0 + 0 + 0 + $(CC) -c + 1 + $(DMDInstallDir)windows\bin\dmd.exe + $(SolutionDir)/src $(SolutionDir)/3rdparty $(SolutionDir)/deps/DerelictGL3/source $(SolutionDir)/deps/DerelictUtil/source $(SolutionDir)/deps/DerelictFT/source $(SolutionDir)/deps/DerelictSDL2/source + views views/res views/res/i18n views/res/mdpi views/res/hdpi + $(ConfigurationName) + $(OutDir) + + + 0 + + + + + 0 + + + 1 + $(IntDir)\$(TargetName).json + 0 + + 0 + + 0 + 0 + 0 + + + + 0 + + 0 + $(VisualDInstallDir)cv2pdb\cv2pdb.exe + 0 + 0 + 0 + + + + ole32.lib kernel32.lib user32.lib comctl32.lib comdlg32.lib + + + + $(OutDir)\$(ProjectName).exe + 1 + 1 + 0 + + + + *.obj;*.cmd;*.build;*.json;*.dep + + + + + + + + + + + + + + + diff --git a/examples/dminer/dub.json b/examples/dminer/dub.json new file mode 100644 index 00000000..730836db --- /dev/null +++ b/examples/dminer/dub.json @@ -0,0 +1,21 @@ +{ + "name": "dminer", + "description": "dlangui library example DMiner", + "homepage": "https://github.com/buggins/dlangui", + "license": "Boost", + "authors": ["Vadim Lopatin"], + + "stringImportPaths": ["views", "views/res", "views/res/i18n", "views/res/mdpi"], + + "targetPath": "bin", + "targetName": "dminer", + "targetType": "executable", + + "sourceFiles-windows": ["$PACKAGE_DIR/src/win_app.def"], + + "versions": ["EmbedStandardResources"], + + "dependencies": { + "dlangui": {"path": "../../"} + } +} diff --git a/examples/dminer/src/dminer/core/blocks.d b/examples/dminer/src/dminer/core/blocks.d new file mode 100644 index 00000000..3ce10bf4 --- /dev/null +++ b/examples/dminer/src/dminer/core/blocks.d @@ -0,0 +1,287 @@ +module dminer.core.blocks; + +import dminer.core.minetypes; +import dminer.core.world; +import dlangui.graphics.scene.mesh; + +immutable string BLOCK_TEXTURE_FILENAME = "blocks"; +immutable int BLOCK_TEXTURE_DX = 1024; +immutable int BLOCK_TEXTURE_DY = 1024; +immutable int BLOCK_SPRITE_SIZE = 16; +immutable int BLOCK_SPRITE_STEP = 16; +immutable int BLOCK_SPRITE_OFFSET = 0; +immutable int BLOCK_TEXTURE_SPRITES_PER_LINE = 1024/16; +immutable int VERTEX_COMPONENTS = 12; + +enum BlockVisibility { + INVISIBLE, + OPAQUE, // completely opaque (cells covered by this block are invisible) + OPAQUE_SEPARATE_TX, + HALF_OPAQUE, // partially paque, cells covered by this block can be visible, render as normal block + HALF_OPAQUE_SEPARATE_TX, + HALF_TRANSPARENT, // should be rendered last (semi transparent texture) +} + +class BlockDef { +public: + cell_t id; + string name; + BlockVisibility visibility = BlockVisibility.INVISIBLE; + int txIndex; + this() { + } + this(cell_t blockId, string blockName, BlockVisibility v, int tx) { + id = blockId; + name = blockName; + visibility = v; + txIndex = tx; + } + ~this() { + } + // blocks behind this block can be visible + @property bool canPass() { + return visibility == BlockVisibility.INVISIBLE + || visibility == BlockVisibility.HALF_OPAQUE + || visibility == BlockVisibility.HALF_OPAQUE_SEPARATE_TX + || visibility == BlockVisibility.HALF_TRANSPARENT; + } + // block is fully opaque (all blocks behind are invisible) + @property bool isOpaque() { + return visibility == BlockVisibility.OPAQUE + || visibility == BlockVisibility.OPAQUE_SEPARATE_TX; + } + // block is visible + @property bool isVisible() { + return visibility != BlockVisibility.INVISIBLE; + } + + @property bool terrainSmoothing() { + return false; + } + + /// create cube face + void createFace(World world, ref Position camPosition, Vector3d pos, Dir face, Mesh mesh) { + // default implementation + ushort startVertexIndex = cast(ushort)mesh.vertexCount; + float[VERTEX_COMPONENTS * 4] vptr; + ushort[6] iptr; + createFaceMesh(vptr.ptr, face, pos.x, pos.y, pos.z, txIndex); + for (int i = 0; i < 6; i++) + iptr[i] = cast(ushort)(startVertexIndex + face_indexes[i]); + //if (HIGHLIGHT_GRID && ((pos.x & 7) == 0 || (pos.z & 7) == 0)) { + // for (int i = 0; i < 4; i++) { + // vptr[11 * i + 6 + 0] = 1.4f; + // vptr[11 * i + 6 + 1] = 1.4f; + // vptr[11 * i + 6 + 2] = 1.4f; + // } + //} + mesh.addVertexes(vptr); + mesh.addPart(PrimitiveType.triangles, iptr); + } + /// create faces + void createFaces(World world, ref Position camPosition, Vector3d pos, int visibleFaces, Mesh mesh) { + for (int i = 0; i < 6; i++) + if (visibleFaces & (1 << i)) + createFace(world, camPosition, pos, cast(Dir)i, mesh); + } +} + +// pos, normal, color, tx +static const float[VERTEX_COMPONENTS * 4] face_vertices_north = +[ + -0.5, 0.5, -0.5, 0.0, 0.0, -1.0, 1.0, 1.0, 1.0, 1.0, 0.0, 0.0, + 0.5, 0.5, -0.5, 0.0, 0.0, -1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 0.0, + -0.5, -0.5, -0.5, 0.0, 0.0, -1.0, 1.0, 1.0, 1.0, 1.0, 0.0, 1.0, + 0.5, -0.5, -0.5, 0.0, 0.0, -1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, +]; + +static const float[VERTEX_COMPONENTS * 4] face_vertices_south = +[ + -0.5, -0.5, 0.5, 0.0, 0.0, 1.0, 1.0, 1.0, 1.0, 1.0, 0.0, 0.0, + 0.5, -0.5, 0.5, 0.0, 0.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 0.0, + -0.5, 0.5, 0.5, 0.0, 0.0, 1.0, 1.0, 1.0, 1.0, 1.0, 0.0, 1.0, + 0.5, 0.5, 0.5, 0.0, 0.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, +]; + +static const float[VERTEX_COMPONENTS * 4] face_vertices_west = +[ + -0.5, -0.5, -0.5, -1.0, 0.0, 0.0, 1.0, 1.0, 1.0, 1.0, 0.0, 0.0, + -0.5, -0.5, 0.5, -1.0, 0.0, 0.0, 1.0, 1.0, 1.0, 1.0, 1.0, 0.0, + -0.5, 0.5, -0.5, -1.0, 0.0, 0.0, 1.0, 1.0, 1.0, 1.0, 0.0, 1.0, + -0.5, 0.5, 0.5, -1.0, 0.0, 0.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0 +]; + +static const float[VERTEX_COMPONENTS * 4] face_vertices_east = +[ + 0.5, -0.5, 0.5, 1.0, 0.0, 0.0, 1.0, 1.0, 1.0, 1.0, 0.0, 0.0, + 0.5, -0.5, -0.5, 1.0, 0.0, 0.0, 1.0, 1.0, 1.0, 1.0, 1.0, 0.0, + 0.5, 0.5, 0.5, 1.0, 0.0, 0.0, 1.0, 1.0, 1.0, 1.0, 0.0, 1.0, + 0.5, 0.5, -0.5, 1.0, 0.0, 0.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, +]; + +static const float[VERTEX_COMPONENTS * 4] face_vertices_up = +[ + -0.5, 0.5, 0.5, 0.0, 1.0, 0.0, 1.0, 1.0, 1.0, 1.0, 0.0, 0.0, + 0.5, 0.5, 0.5, 0.0, 1.0, 0.0, 1.0, 1.0, 1.0, 1.0, 1.0, 0.0, + -0.5, 0.5, -0.5, 0.0, 1.0, 0.0, 1.0, 1.0, 1.0, 1.0, 0.0, 1.0, + 0.5, 0.5, -0.5, 0.0, 1.0, 0.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, +]; + +static const float[VERTEX_COMPONENTS * 4] face_vertices_down = +[ + -0.5, -0.5, -0.5, 0.0, -1.0, 0.0, 1.0, 1.0, 1.0, 1.0, 0.0, 0.0, + 0.5, -0.5, -0.5, 0.0, -1.0, 0.0, 1.0, 1.0, 1.0, 1.0, 1.0, 0.0, + -0.5, -0.5, 0.5, 0.0, -1.0, 0.0, 1.0, 1.0, 1.0, 1.0, 0.0, 1.0, + 0.5, -0.5, 0.5, 0.0, -1.0, 0.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, +]; + +static const ushort[6] face_indexes = +[ + 0, 1, 2, 2, 1, 3 +]; + +static const ushort[6] face_indexes_back = +[ + 0, 2, 1, 2, 3, 1 +]; + +static void fillFaceMesh(float * data, const float * src, float x0, float y0, float z0, int tileX, int tileY) { + for (int i = 0; i < 4; i++) { + const float * srcvertex = src + i * VERTEX_COMPONENTS; + float * dstvertex = data + i * VERTEX_COMPONENTS; + for (int j = 0; j < VERTEX_COMPONENTS; j++) { + float v = srcvertex[j]; + switch (j) { + case 0: // x + v += x0; + break; + case 1: // y + v += y0; + break; + case 2: // z + v += z0; + break; + case 10: // tx.u + v = ((tileX + v * BLOCK_SPRITE_SIZE)) / cast(float)BLOCK_TEXTURE_DX; + break; + case 11: // tx.v + //v = (BLOCK_TEXTURE_DY - (tileY + v * BLOCK_SPRITE_SIZE)) / cast(float)BLOCK_TEXTURE_DY; + v = ((tileY + v * BLOCK_SPRITE_SIZE)) / cast(float)BLOCK_TEXTURE_DY; + break; + default: + break; + } + dstvertex[j] = v; + } + } +} + +static void createFaceMesh(float * data, Dir face, float x0, float y0, float z0, int tileIndex) { + + int tileX = (tileIndex % BLOCK_TEXTURE_SPRITES_PER_LINE) * BLOCK_SPRITE_STEP + BLOCK_SPRITE_OFFSET; + int tileY = (tileIndex / BLOCK_TEXTURE_SPRITES_PER_LINE) * BLOCK_SPRITE_STEP + BLOCK_SPRITE_OFFSET; + // data is 11 comp * 4 vert floats + switch (face) with(Dir) { + default: + case NORTH: + fillFaceMesh(data, face_vertices_north.ptr, x0, y0, z0, tileX, tileY); + break; + case SOUTH: + fillFaceMesh(data, face_vertices_south.ptr, x0, y0, z0, tileX, tileY); + break; + case WEST: + fillFaceMesh(data, face_vertices_west.ptr, x0, y0, z0, tileX, tileY); + break; + case EAST: + fillFaceMesh(data, face_vertices_east.ptr, x0, y0, z0, tileX, tileY); + break; + case UP: + fillFaceMesh(data, face_vertices_up.ptr, x0, y0, z0, tileX, tileY); + break; + case DOWN: + fillFaceMesh(data, face_vertices_down.ptr, x0, y0, z0, tileX, tileY); + break; + } +} + + + +// block type definitions +__gshared BlockDef[256] BLOCK_DEFS; +// faster check for block->canPass() +__gshared bool[256] BLOCK_TYPE_CAN_PASS; +// faster check for block->isOpaque() +__gshared bool[256] BLOCK_TYPE_OPAQUE; +// faster check for block->isVisible() +__gshared bool[256] BLOCK_TYPE_VISIBLE; +// faster check for block->isVisible() +__gshared bool[256] BLOCK_TERRAIN_SMOOTHING; + +/// registers new block type +void registerBlockType(BlockDef def) { + if (BLOCK_DEFS[def.id]) { + if (BLOCK_DEFS[def.id] is def) + return; + destroy(BLOCK_DEFS[def.id]); + } + BLOCK_DEFS[def.id] = def; + // init property shortcuts + BLOCK_TYPE_CAN_PASS[def.id] = def.canPass; + BLOCK_TYPE_OPAQUE[def.id] = def.isOpaque; + BLOCK_TYPE_VISIBLE[def.id] = def.isVisible; + BLOCK_TERRAIN_SMOOTHING[def.id] = def.terrainSmoothing; +} + +enum BlockImage : int { + stone, + grass_top, + grass_side, + grass_top_footsteps, + dirt, + bedrock, + sand, + gravel, + sandstone, + clay, + cobblestone, + cobblestone_mossy, + brick, + stonebrick, + red_sand, +} + +/// init block types array +__gshared static this() { + import std.string; + for (int i = 0; i < 256; i++) { + if (!BLOCK_DEFS[i]) { + registerBlockType(new BlockDef(cast(cell_t)i, "undef%d".format(i), BlockVisibility.INVISIBLE, 0)); + } + } + BLOCK_TYPE_CAN_PASS[BOUND_SKY] = false; + BLOCK_TYPE_VISIBLE[BOUND_SKY] = false; + BLOCK_TYPE_CAN_PASS[BOUND_BOTTOM] = false; + BLOCK_TYPE_VISIBLE[BOUND_BOTTOM] = true; + + // empty cell + registerBlockType(new BlockDef(0, "empty", BlockVisibility.INVISIBLE, 0)); + // standard block types + registerBlockType(new BlockDef(1, "gray_brick", BlockVisibility.OPAQUE, BlockImage.stonebrick)); + registerBlockType(new BlockDef(2, "brick", BlockVisibility.OPAQUE, BlockImage.brick)); + registerBlockType(new BlockDef(3, "bedrock", BlockVisibility.OPAQUE, BlockImage.bedrock)); + registerBlockType(new BlockDef(4, "clay", BlockVisibility.OPAQUE, BlockImage.clay)); + registerBlockType(new BlockDef(5, "cobblestone", BlockVisibility.OPAQUE, BlockImage.cobblestone)); + registerBlockType(new BlockDef(6, "gravel", BlockVisibility.OPAQUE, BlockImage.gravel)); + registerBlockType(new BlockDef(7, "red_sand", BlockVisibility.OPAQUE, BlockImage.red_sand)); + registerBlockType(new BlockDef(8, "sand", BlockVisibility.OPAQUE, BlockImage.sand)); + + registerBlockType(new BlockDef(50, "box", BlockVisibility.HALF_OPAQUE, 50)); + + //registerBlockType(new TerrainBlock(100, "terrain_bedrock", 2)); + //registerBlockType(new TerrainBlock(101, "terrain_clay", 3)); + //registerBlockType(new TerrainBlock(102, "terrain_cobblestone", 4)); + //registerBlockType(new TerrainBlock(103, "terrain_gravel", 5)); + //registerBlockType(new TerrainBlock(104, "terrain_red_sand", 6)); + //registerBlockType(new TerrainBlock(105, "terrain_sand", 7)); + +} diff --git a/examples/dminer/src/dminer/core/minetypes.d b/examples/dminer/src/dminer/core/minetypes.d new file mode 100644 index 00000000..6f326f40 --- /dev/null +++ b/examples/dminer/src/dminer/core/minetypes.d @@ -0,0 +1,731 @@ +module dminer.core.minetypes; + +alias cell_t = ubyte; + +immutable cell_t NO_CELL = 0; +immutable cell_t END_OF_WORLD = 253; +immutable cell_t VISITED_CELL = 255; +immutable cell_t VISITED_OCCUPIED = 254; + +immutable cell_t BOUND_BOTTOM = 253; +immutable cell_t BOUND_SKY = 252; + +enum Dir : ubyte { + NORTH = 0, + SOUTH, + EAST, + WEST, + UP, + DOWN, +} + +// 26 direction masks based on Dir +enum DirMask : ubyte { + MASK_NORTH = (1 << Dir.NORTH), + MASK_SOUTH = (1 << Dir.SOUTH), + MASK_EAST = (1 << Dir.EAST), + MASK_WEST = (1 << Dir.WEST), + MASK_UP = (1 << Dir.UP), + MASK_DOWN = (1 << Dir.DOWN), + MASK_ALL = 0x3F, + //MASK_WEST_UP = (1 << Dir.WEST) | MASK_UP, + //MASK_EAST_UP = (1 << Dir.EAST) | MASK_UP, + //MASK_WEST_DOWN = (1 << Dir.WEST) | MASK_DOWN, + //MASK_EAST_DOWN = (1 << Dir.EAST) | MASK_DOWN, + //MASK_NORTH_WEST = MASK_NORTH | MASK_WEST, + //MASK_NORTH_EAST = MASK_NORTH | MASK_EAST, + //MASK_NORTH_UP = MASK_NORTH | MASK_UP, + //MASK_NORTH_DOWN = MASK_NORTH | MASK_DOWN, + //MASK_NORTH_WEST_UP = MASK_NORTH | MASK_WEST | MASK_UP, + //MASK_NORTH_EAST_UP = MASK_NORTH | MASK_EAST | MASK_UP, + //MASK_NORTH_WEST_DOWN = MASK_NORTH | MASK_WEST | MASK_DOWN, + //MASK_NORTH_EAST_DOWN = MASK_NORTH | MASK_EAST | MASK_DOWN, + //MASK_SOUTH_WEST = MASK_SOUTH | MASK_WEST, + //MASK_SOUTH_EAST = MASK_SOUTH | MASK_EAST, + //MASK_SOUTH_UP = MASK_SOUTH | MASK_UP, + //MASK_SOUTH_DOWN = MASK_SOUTH | MASK_DOWN, + //MASK_SOUTH_WEST_UP = MASK_SOUTH | MASK_WEST | MASK_UP, + //MASK_SOUTH_EAST_UP = MASK_SOUTH | MASK_EAST | MASK_UP, + //MASK_SOUTH_WEST_DOWN = MASK_SOUTH | MASK_WEST | MASK_DOWN, + //MASK_SOUTH_EAST_DOWN = MASK_SOUTH | MASK_EAST | MASK_DOWN, +} + +struct Vector2d { + int x; + int y; + this(int xx, int yy) { + x = xx; + y = yy; + } + //bool opEqual(Vector2d v) const { + // return x == v.x && y == v.y; + //} +} + +immutable Vector2d ZERO2 = Vector2d(0, 0); + +struct Vector3d { + int x; + int y; + int z; + this(int xx, int yy, int zz) { + x = xx; + y = yy; + z = zz; + } + //bool opEqual(const Vector3d v) const { + // return x == v.x && y == v.y && z == v.z; + //} + + /// returns vector with all components which are negative of components for this vector + Vector3d opUnary(string op : "-")() const { + return Vector3d(-x, -y, -z); + } + /// subtract vectors + Vector3d opBinary(string op : "-")(const Vector3d v) const { + return Vector3d(x - v.x, y - v.y, z - v.z); + } + /// add vectors + Vector3d opBinary(string op : "+")(const Vector3d v) const { + return Vector3d(x + v.x, y + v.y, z + v.z); + } + /// + int opBinary(string op : "*")(const Vector3d v) const { + return x*v.x + y*v.y + z*v.z; + } + /// multiply vector elements by constant + Vector3d opBinary(string op : "*")(int n) const { + return Vector3d(x * n, y * n, z * n); + } + + /// + ref Vector3d opOpAssign(string op : "+")(const Vector3d v) { + x += v.x; + y += v.y; + z += v.z; + return this; + } + /// + ref Vector3d opOpAssign(string op : "-")(const Vector3d v) { + x -= v.x; + y -= v.y; + z -= v.z; + return this; + } + /// + ref Vector3d opOpAssign(string op : "*")(int n) { + x *= n; + y *= n; + z *= n; + return this; + } + Vector3d turnLeft() { + return Vector3d(z, y, -x); + } + Vector3d turnRight() { + return Vector3d(-z, y, x); + } + Vector3d turnUp() { + return Vector3d(x, -z, y); + } + Vector3d turnDown() { + return Vector3d(x, z, -y); + } + Vector3d move(Dir dir) { + Vector3d res = this; + switch (dir) with(Dir) { + case NORTH: + res.z--; + break; + case SOUTH: + res.z++; + break; + case WEST: + res.x--; + break; + case EAST: + res.x++; + break; + case UP: + res.y++; + break; + case DOWN: + res.y--; + break; + default: + break; + } + return res; + } +} + +const Vector3d ZERO3 = Vector3d(0, 0, 0); + +struct Array(T) { +private: + int _length; + T[] _data; +public: + T * ptr(int index = 0) { + return _data.ptr + index; + } + void swap(ref Array v) { + int tmp; + tmp = _length; _length = v._length; v._length = tmp; + T[] ptmp; + ptmp = _data; _data = v._data; v._data = ptmp; + } + /// ensure capacity is enough to fit sz items + void reserve(int sz) { + sz += _length; + if (_data.length < sz) { + int oldsize = cast(int)_data.length; + int newsize = 1024; + while (newsize < sz) + newsize <<= 1; + _data.length = newsize; + for (int i = oldsize; i < newsize; i++) + _data.ptr[i] = T.init; + _data.assumeSafeAppend(); + } + } + @property int length() { + return _length; + } + /// append single item by ref + void append(ref const T value) { + if (_length >= _data.length) + reserve(cast(int)(_data.length == 0 ? 64 : _data.length * 2 - _length)); + _data.ptr[_length++] = value; + } + /// append single item by value + void append(T value) { + if (_length >= _data.length) + reserve(cast(int)(_data.length == 0 ? 64 : _data.length * 2 - _length)); + _data.ptr[_length++] = value; + } + /// append single item w/o check + void appendNoCheck(ref const T value) { + _data.ptr[_length++] = value; + } + /// append single item w/o check + void appendNoCheck(T value) { + _data.ptr[_length++] = value; + } + /// appends same value several times, return pointer to appended items + T* append(ref const T value, int count) { + reserve(count); + int startLen = _length; + for (int i = 0; i < count; i++) + _data.ptr[_length++] = value; + return _data.ptr + startLen; + } + /// appends same value several times, return pointer to appended items + T* append(T value, int count) { + reserve(count); + int startLen = _length; + for (int i = 0; i < count; i++) + _data.ptr[_length++] = value; + return _data.ptr + startLen; + } + void clear() { + _length = 0; + } + T get(int index) { + return _data.ptr[index]; + } + void set(int index, T value) { + _data.ptr[index] = value; + } + ref T opIndex(int index) { + return _data.ptr[index]; + } +} + +alias FloatArray = Array!(float); +alias IntArray = Array!(int); +alias CellArray = Array!(cell_t); +alias Vector2dArray = Array!(Vector2d); +alias Vector3dArray = Array!(Vector3d); + +/// array with support of both positive and negative indexes +struct InfiniteArray(T) { +private: + T[] dataPlus; + T[] dataMinus; + int minIdx; + int maxIdx; +public: + @property int minIndex() { return minIdx; } + @property int maxIndex() { return maxIdx; } + void disposeFunction(T p) { + destroy(p); + } + ~this() { + foreach(p; dataPlus) + if (p !is T.init) + disposeFunction(p); + foreach(p; dataMinus) + if (p !is T.init) + disposeFunction(p); + } + T get(int index) { + if (index >= 0) { + if (index >= maxIdx) + return T.init; + return dataPlus[index]; + } else { + if (index <= minIdx) + return T.init; + return dataMinus[-index]; + } + } + void set(int index, T value) { + if (index >= 0) { + if (index >= maxIdx) { + // extend array + if (index <= dataPlus.length) { + int oldsize = dataPlus.length; + int newsize = 1024; + while (newsize <= index) + newsize <<= 1; + dataPlus.length = newsize; + dataPlus.assumeSafeAppend; + for(int i = oldsize; i < newsize; i++) + dataPlus[i] = T.init; + } + maxIdx = index + 1; + } + if (dataPlus[index] !is T.init && dataPlus[index] !is value) + disposeFunction(dataPlus[index]); + dataPlus[index] = value; + } else { + if (index <= minIdx) { + // extend array + if (-index <= dataMinus.length) { + int oldsize = dataMinus.length; + int newsize = 1024; + while (newsize <= -index) + newsize <<= 1; + dataMinus.length = newsize; + dataMinus.assumeSafeAppend; + for(int i = oldsize; i < newsize; i++) + dataMinus[i] = T.init; + } + maxIdx = index - 1; + } + if (dataMinus[-index] !is T.init && dataMinus[-index] !is value) + disposeFunction(dataMinus[-index]); + dataMinus[-index] = value; + } + } +} + +struct InfiniteMatrix(T) { +private: + int _size = 0; + int _sizeShift = 0; + int _sizeShiftMul2 = 0; + int _sizeMask = 0; + int _invSizeMask = 0; + T[] _data; + void resize(int newSizeShift) { + int newSize = (1<= newSize || y < -newSize || y >= newSize) { + // destory: // outside new size + destroy(v); + } else { + // move + newdata[((y + newSize) << (newSizeShift + 1)) | (x + newSize)] = v; + } + } + } + _data = newdata; + _size = newSize; + _sizeShift = newSizeShift; + _sizeShiftMul2 = _sizeShift + 1; + _sizeMask = (1 << _sizeShiftMul2) - 1; + _invSizeMask = ~_sizeMask; + } + int calcIndex(int x, int y) { + return (y << _sizeShiftMul2) + x; + } +public: + @property int size() { return _size; } + T get(int x, int y) { + if (!_data) + return null; + x += _size; + y += _size; + if (!((x | y) & ~_sizeMask)) { + return _data.ptr[(y << _sizeShiftMul2) + x]; //calcIndex(x, y) + } + return null; + } + void set(int x, int y, T v) { + if (x < -_size || x >= _size || y < -_size || y >= _size) { + int newSizeShift = _sizeShift < 6 ? 6 : _sizeShift + 1; + for (; ;newSizeShift++) { + int sz = 1 << newSizeShift; + if (x < -sz || x >= sz || y < -sz || y >= sz) + continue; + break; + } + resize(newSizeShift); + } + x += _size; + y += _size; + int index = calcIndex(x, y); + if (_data.ptr[index]) + destroy(_data.ptr[index]); + _data.ptr[index] = v; + } + ~this() { + foreach(ref v; _data) + if (v) + destroy(v); + } +} + +struct Position { + Vector3d pos; + Direction direction; + this(ref Position p) { + pos = p.pos; + direction = p.direction; + } + this(Vector3d position, Vector3d dir) { + pos = position; + direction = dir; + } + Vector2d calcPlaneCoords(Vector3d v) { + v = v - pos; + switch (direction.dir) with(Dir) { + default: + case NORTH: + return Vector2d(v.x, v.y); + case SOUTH: + return Vector2d(-v.x, v.y); + case EAST: + return Vector2d(v.z, v.y); + case WEST: + return Vector2d(-v.z, v.y); + case UP: + return Vector2d(-v.z, v.x); + case DOWN: + return Vector2d(v.z, v.x); + } + } + void turnLeft() { + direction.turnLeft(); + } + void turnRight() { + direction.turnRight(); + } + void shiftLeft(int step = 1) { + pos += direction.left * step; + } + void shiftRight(int step = 1) { + pos += direction.right * step; + } + void turnUp() { + direction.turnUp(); + } + void turnDown() { + direction.turnDown(); + } + void forward(int step = 1) { + pos += direction.forward * step; + } + void backward(int step = 1) { + pos -= direction.forward * step; + } + void moveUp(int step = 1) { + pos += direction.up * step; + } + void moveDown(int step = 1) { + pos += direction.down * step; + } + void moveLeft(int step = 1) { + pos += direction.left * step; + } + void moveRight(int step = 1) { + pos += direction.right * step; + } +} + + +/// returns opposite direction to specified direction +Dir opposite(Dir d) { + return cast(Dir)(d ^ 1); +} + +Dir turnLeft(Dir d) { + switch (d) with (Dir) { + case WEST: + return SOUTH; + case EAST: + return NORTH; + default: + case NORTH: + return WEST; + case SOUTH: + return EAST; + case UP: + return SOUTH; + case DOWN: + return NORTH; + } +} + +Dir turnRight(Dir d) { + switch (d) with (Dir) { + case WEST: + return NORTH; + case EAST: + return SOUTH; + default: + case NORTH: + return EAST; + case SOUTH: + return WEST; + case UP: + return NORTH; + case DOWN: + return SOUTH; + } +} + +Dir turnUp(Dir d) { + switch (d) with (Dir) { + case WEST: + return UP; + case EAST: + return UP; + default: + case NORTH: + return UP; + case SOUTH: + return UP; + case UP: + return SOUTH; + case DOWN: + return NORTH; + } +} + +Dir turnDown(Dir d) { + switch (d) with (Dir) { + case WEST: + return DOWN; + case EAST: + return DOWN; + default: + case NORTH: + return DOWN; + case SOUTH: + return DOWN; + case UP: + return NORTH; + case DOWN: + return SOUTH; + } +} + + +struct Direction { + this(int x, int y, int z) { + set(x, y, z); + } + this(Vector3d v) { + set(v); + } + this(Dir d) { + set(d); + } + /// returns Y axis rotation angle in degrees (0, 90, 180, 270) + @property float angle() { + switch (dir) with (Dir) { + default: + case NORTH: + return 0; + case SOUTH: + return 180; + case WEST: + return 90; + case EAST: + return 270; + case UP: + case DOWN: + return 0; + } + } + /// set by direction code + void set(Dir d) { + switch (d) with (Dir) { + default: + case NORTH: + set(0, 0, -1); + break; + case SOUTH: + set(0, 0, 1); + break; + case WEST: + set(-1, 0, 0); + break; + case EAST: + set(1, 0, 0); + break; + case UP: + set(0, 1, 0); + break; + case DOWN: + set(0, -1, 0); + break; + } + } + /// set by vector + void set(Vector3d v) { set(v.x, v.y, v.z); } + /// set by vector + void set(int x, int y, int z) { + forward = Vector3d(x, y, z); + if (x) { + dir = (x > 0) ? Dir.EAST : Dir.WEST; + } + else if (y) { + dir = (y > 0) ? Dir.UP : Dir.DOWN; + } + else { + dir = (z > 0) ? Dir.SOUTH : Dir.NORTH; + } + switch (dir) with (Dir) { + case UP: + up = Vector3d(1, 0, 0); + left = Vector3d(0, 0, 1); + break; + case DOWN: + up = Vector3d(1, 0, 0); + left = Vector3d(0, 0, -1); + break; + default: + case NORTH: + up = Vector3d(0, 1, 0); + left = Vector3d(-1, 0, 0); + break; + case SOUTH: + up = Vector3d(0, 1, 0); + left = Vector3d(1, 0, 0); + break; + case EAST: + up = Vector3d(0, 1, 0); + left = Vector3d(0, 0, -1); + break; + case WEST: + up = Vector3d(0, 1, 0); + left = Vector3d(0, 0, 1); + break; + } + down = -up; + right = -left; + forwardUp = forward + up; + forwardDown = forward + down; + forwardLeft = forward + left; + forwardLeftUp = forward + left + up; + forwardLeftDown = forward + left + down; + forwardRight = forward + right; + forwardRightUp = forward + right + up; + forwardRightDown = forward + right + down; + } + + void turnLeft() { + set(.turnLeft(dir)); + } + void turnRight() { + set(.turnRight(dir)); + } + void turnUp() { + set(.turnUp(dir)); + } + void turnDown() { + set(.turnDown(dir)); + } + + Dir dir; + Vector3d forward; + Vector3d up; + Vector3d right; + Vector3d left; + Vector3d down; + Vector3d forwardUp; + Vector3d forwardDown; + Vector3d forwardLeft; + Vector3d forwardLeftUp; + Vector3d forwardLeftDown; + Vector3d forwardRight; + Vector3d forwardRightUp; + Vector3d forwardRightDown; +} + +/// returns number of bits to store integer +int bitsFor(int n) { + int res; + for (res = 0; n > 0; res++) + n >>= 1; + return res; +} + +/// returns 0 for 0, 1 for negatives, 2 for positives +int mySign(int n) { + if (n > 0) + return 1; + else if (n < 0) + return -1; + else + return 0; +} + +immutable ulong RANDOM_MULTIPLIER = 0x5DEECE66D; +immutable ulong RANDOM_MASK = ((cast(ulong)1 << 48) - 1); +immutable ulong RANDOM_ADDEND = cast(ulong)0xB; + +struct Random { + ulong seed; + //Random(); + void setSeed(ulong value) { + seed = (value ^ RANDOM_MULTIPLIER) & RANDOM_MASK; + } + + int next(int bits) { + seed = (seed * RANDOM_MULTIPLIER + RANDOM_ADDEND) & RANDOM_MASK; + return cast(int)(seed >> (48 - bits)); + } + + int nextInt() { + return next(31); + } + int nextInt(int n) { + if ((n & -n) == n) // i.e., n is a power of 2 + return cast(int)((n * cast(long)next(31)) >> 31); + int bits, val; + do { + bits = next(31); + val = bits % n; + } while (bits - val + (n - 1) < 0); + return val; + } +} + +const Vector3d[6] DIRECTION_VECTORS = [ + Vector3d(0, 0, -1), + Vector3d(0, 0, 1), + Vector3d(-1, 0, 0), + Vector3d(1, 0, 0), + Vector3d(0, 1, 0), + Vector3d(0, -1, 0) +]; diff --git a/examples/dminer/src/dminer/core/terrain.d b/examples/dminer/src/dminer/core/terrain.d new file mode 100644 index 00000000..2310200d --- /dev/null +++ b/examples/dminer/src/dminer/core/terrain.d @@ -0,0 +1,153 @@ +module dminer.core.terrain; + +import dminer.core.minetypes; + + +struct TerrainGen { + private int dx; + private int dy; + private int xpow; + private int ypow; + private short[] data; + private Random rnd; + private void diamond(int x, int y, int size, int offset) { + int avg = (get(x, y - size) + get(x + size, y) + get(x, y + size) + get(x - size, y)) >> 2; + set(x, y, avg + offset); + } + private void square(int x, int y, int size, int offset) { + int avg = (get(x - size, y - size) + get(x + size, y - size) + get(x - size, y + size) + get(x - size, y - size)) >> 2; + set(x, y, avg + offset); + } + + this(int xbits, int zbits) { + xpow = xbits; + ypow = zbits; + dx = (1 << xpow) + 1; + dy = (1 << ypow) + 1; + data = new short[dx * dy]; + } + ~this() { + } + void filter(int range) { + short[] tmp = new short[dx * dy]; + int div = (range * 2 + 1) * (range * 2 + 1); + for (int y = 0; y < dy; y++) { + for (int x = 0; x < dx; x++) { + int s = 0; + for (int yy = -range; yy <= range; yy++) { + for (int xx = -range; xx <= range; xx++) { + s += get(x + xx, y + yy); + } + } + s /= div; + tmp[(y << ypow) + y + x] = cast(short)s; + } + } + int sz = dx * dy; + data[0 .. sz] = tmp[0 .. sz]; + } + + void generate(int seed, short[] initData, int stepBits) { + rnd.setSeed(seed); + int step = 1 << stepBits; + int index = 0; + for (int y = 0; y <= dy; y += step) { + for (int x = 0; x <= dx; x += step) { + set(x, y, initData[index++]); + } + } + int half = step >> 1; + while (half > 0) { + int scale = step; + for (int y = half; y < dy; y += step) { + for (int x = half; x < dx; x++) { + square(x, y, half, rnd.nextInt(scale * 2) - scale); + } + } + for (int y = 0; y <= dy; y += half) { + for (int x = (y + half) % step; x <= dx; x += step) { + diamond(x, y, half, rnd.nextInt(scale * 2) - scale); + } + } + step >>= 1; + half >>= 1; + } + } + void generateWithScale(int seed, short[] initData, int stepBits, TerrainGen scaleMap) { + rnd.setSeed(seed); + int step = 1 << stepBits; + int index = 0; + for (int y = 0; y <= dy; y += step) { + for (int x = 0; x <= dx; x += step) { + set(x, y, initData[index++]); + } + } + int half = step >> 1; + while (half > 0) { + for (int y = half; y < dy; y += step) { + for (int x = half; x < dx; x++) { + int scale = (scaleMap.get(x, y) * step) >> 8; + scale = rnd.nextInt(scale * 2) - scale; + if (step < 4) + scale = 0; + square(x, y, half, scale); + } + } + for (int y = 0; y <= dy; y += half) { + for (int x = (y + half) % step; x <= dx; x += step) { + int scale = (scaleMap.get(x, y) * step) >> 8; + scale = rnd.nextInt(scale * 2) - scale; + if (step < 4) + scale = 0; + diamond(x, y, half, scale); + } + } + step >>= 1; + half >>= 1; + } + } + @property int width() { + return dx - 1; + } + @property int height() { + return dy - 1; + } + int get(int x, int y) { + if (x < 0 || y < 0 || x >= dx || y >= dy) + return 0; + return data[(y << ypow) + y + x]; + } + void set(int x, int y, int value) { + if (x < 0 || y < 0 || x >= dx || y >= dy) + return; + if (value < -32767) + value = -32767; + if (value > 32767) + value = 32767; + data[(y << ypow) + y + x] = cast(short)value; + } + /// ensure that data is in range [minvalue, maxvalue] + void limit(int minvalue, int maxvalue) { + // find actual min/max + int minv, maxv; + minv = maxv = get(0, 0); + for (int y = 0; y <= dy; y++) { + for (int x = 0; x <= dx; x++) { + int v = get(x, y); + if (minv > v) + minv = v; + if (maxv < v) + maxv = v; + } + } + int mul = (maxvalue - minvalue); + int div = (maxv - minv); + if (div > 0) { + for (int y = 0; y <= dy; y++) { + for (int x = 0; x <= dx; x++) { + set(x, y, minvalue + (get(x, y) - minv) * mul / div); + } + } + } + } +} diff --git a/examples/dminer/src/dminer/core/world.d b/examples/dminer/src/dminer/core/world.d new file mode 100644 index 00000000..1bd4312e --- /dev/null +++ b/examples/dminer/src/dminer/core/world.d @@ -0,0 +1,501 @@ +module dminer.core.world; + +import dminer.core.minetypes; +import dminer.core.blocks; + +const int MAX_VIEW_DISTANCE_BITS = 6; +const int MAX_VIEW_DISTANCE = (1 << MAX_VIEW_DISTANCE_BITS); + +// Layer is 16x16 (CHUNK_DX_SHIFT x CHUNK_DX_SHIFT) cells +immutable int CHUNK_DX_SHIFT = 4; +immutable int CHUNK_DX = (1< bottomLayer) + minLayer = bottomLayer; + if (maxLayer == -1 || maxLayer < topLayer) + maxLayer = topLayer; + } + cell_t get(int x, int y, int z) { + //if (!this) + // return NO_CELL; + ChunkLayer * layer = layers[y & CHUNK_DY_MASK]; + if (!layer) + return NO_CELL; + return layer.get(x & CHUNK_DX_MASK, z & CHUNK_DY_MASK); + } + + /// get, x, y, z are already checked for bounds + cell_t getNoCheck(int x, int y, int z) { + ChunkLayer * layer = layers.ptr[y]; + if (!layer) // likely + return NO_CELL; + return layer.cells.ptr[(z << CHUNK_DX_SHIFT) + x]; // inlined return layer.get(x, z); + } + + void set(int x, int y, int z, cell_t cell) { + int layerIndex = y & CHUNK_DY_MASK; + ChunkLayer * layer = layers.ptr[layerIndex]; + if (!layer) { + layer = new ChunkLayer(); + layers.ptr[layerIndex] = layer; + if (topLayer == -1 || topLayer < layerIndex) + topLayer = layerIndex; + if (bottomLayer == -1 || bottomLayer > layerIndex) + bottomLayer = layerIndex; + } + layer.set(x & CHUNK_DX_MASK, z & CHUNK_DY_MASK, cell); + } + + /// srcpos coords x, z are in chunk bounds + //void getCells(Vector3d srcpos, Vector3d dstpos, Vector3d size, VolumeData & buf); +} + +alias ChunkMatrix = InfiniteMatrix!(Chunk *); + +/// Voxel World +class World { +private: + Position _camPosition; + int maxVisibleRange = MAX_VIEW_DISTANCE; + ChunkMatrix chunks; + DiamondVisitor visitorHelper; +public: + this() { + _camPosition = Position(Vector3d(0, 13, 0), Vector3d(0, 0, 1)); + } + ~this() { + } + @property final ref Position camPosition() { return _camPosition; } + + final cell_t getCell(int x, int y, int z) { + if (!(y & CHUNK_DY_INV_MASK)) { + if (Chunk * p = chunks.get(x >> CHUNK_DX_SHIFT, z >> CHUNK_DX_SHIFT)) + return p.getNoCheck(x & CHUNK_DX_MASK, y, z & CHUNK_DX_MASK); + return NO_CELL; + } + // y out of bounds + if (y < 0) + return BOUND_BOTTOM; + //if (y >= CHUNK_DY) + else + return BOUND_SKY; + } + + final bool isOpaque(int x, int y, int z) { + cell_t cell = getCell(x, y, z); + return BLOCK_TYPE_OPAQUE.ptr[cell] && cell != BOUND_SKY; + } + + final void setCell(int x, int y, int z, cell_t value) { + int chunkx = x >> CHUNK_DX_SHIFT; + int chunkz = z >> CHUNK_DX_SHIFT; + Chunk * p = chunks.get(chunkx, chunkz); + if (!p) { + p = new Chunk(); + chunks.set(chunkx, chunkz, p); + } + p.set(x & CHUNK_DX_MASK, y, z & CHUNK_DX_MASK, value); + } + + void setCellRange(Vector3d pos, Vector3d sz, cell_t value) { + for (int x = 0; x < sz.x; x++) + for (int y = 0; y < sz.y; y++) + for (int z = 0; z < sz.z; z++) + setCell(pos.x + x, pos.y + y, pos.z + z, value); + } + + bool canPass(Vector3d pos) { + return canPass(Vector3d(pos.x - 2, pos.y - 3, pos.z - 2), Vector3d(4, 5, 4)); + } + + bool canPass(Vector3d pos, Vector3d size) { + for (int x = 0; x <= size.x; x++) + for (int z = 0; z <= size.z; z++) + for (int y = 0; y < size.y; y++) { + if (isOpaque(pos.x + x, pos.y + y, pos.z + z)) + return false; + } + return true; + } + final void visitVisibleCells(ref Position position, CellVisitor visitor) { + visitorHelper.init(this, + &position, + visitor); + visitorHelper.visitAll(maxVisibleRange); + } +} + +interface CellVisitor { + //void newDirection(ref Position camPosition); + //void visitFace(World world, ref Position camPosition, Vector3d pos, cell_t cell, Dir face); + void visit(World world, ref Position camPosition, Vector3d pos, cell_t cell, int visibleFaces); +} + +struct DiamondVisitor { + int maxDist; + int maxDistBits; + int dist; + World world; + Position * position; + Vector3d pos0; + CellVisitor visitor; + CellArray visited; + cell_t * visited_ptr; + Vector3dArray oldcells; + Vector3dArray newcells; + ubyte visitedId; + //ubyte visitedEmpty; + int m0; + int m0mask; + void init(World w, Position * pos, CellVisitor v) { + world = w; + position = pos; + visitor = v; + pos0 = position.pos; + } + void visitCell(int vx, int vy, int vz) { + //CRLog::trace("visitCell(%d %d %d) dist=%d", v.x, v.y, v.z, myAbs(v.x) + myAbs(v.y) + myAbs(v.z)); + + //int occupied = visitedOccupied; + int index = (vx + m0) + ((vz + m0) << (maxDistBits + 1)); + if (vy < 0) { + // inverse index for lower half + index ^= m0mask; + } + //int index = diamondIndex(v, maxDistBits); + if (visited_ptr[index] == visitedId)// || cell == visitedEmpty) + return; + visitCellNoCheck(vx, vy, vz); + visited_ptr[index] = visitedId; // cell; + } + + void visitCellNoCheck(int vx, int vy, int vz) { + //if (v * position.direction.forward < dist / 3) // limit by visible from cam + // return; + //Vector3d pos = pos0 + v; + int posx = pos0.x + vx; + int posy = pos0.y + vy; + int posz = pos0.z + vz; + cell_t cell = world.getCell(posx, posy, posz); + + // read cell from world + if (BLOCK_TYPE_VISIBLE.ptr[cell]) { + int visibleFaces = 0; + if (vy <= 0 && !world.isOpaque(posx, posy + 1, posz)) + visibleFaces |= DirMask.MASK_UP; + if (vy >= 0 && !world.isOpaque(posx, posy - 1, posz)) + visibleFaces |= DirMask.MASK_DOWN; + if (vx <= 0 && !world.isOpaque(posx + 1, posy, posz)) + visibleFaces |= DirMask.MASK_EAST; + if (vx >= 0 && !world.isOpaque(posx - 1, posy, posz)) + visibleFaces |= DirMask.MASK_WEST; + if (vz <= 0 && !world.isOpaque(posx, posy, posz + 1)) + visibleFaces |= DirMask.MASK_SOUTH; + if (vz >= 0 && !world.isOpaque(posx, posy, posz - 1)) + visibleFaces |= DirMask.MASK_NORTH; + visitor.visit(world, *position, Vector3d(posx, posy, posz), cell, visibleFaces); + } + // mark as visited + if (BLOCK_TYPE_CAN_PASS.ptr[cell]) + newcells.append(Vector3d(vx, vy, vz)); + //cell = BLOCK_TYPE_CAN_PASS[cell] ? visitedEmpty : visitedOccupied; + } + + bool needVisit(int index) { + if (visited_ptr[index] != visitedId) { + visited_ptr[index] = visitedId; + return true; + } + return false; + } + + static int myAbs(int n) { + return n < 0 ? -n : n; + } + + void visitAll(int maxDistance) { + maxDist = maxDistance; + maxDistance *= 2; + maxDistBits = bitsFor(maxDist); + int maxDistMask = ~((1 << maxDistBits) - 1); + maxDistBits++; + + m0 = 1 << maxDistBits; + m0mask = (m0 - 1) + ((m0 - 1) << (maxDistBits + 1)); + + oldcells.clear(); + newcells.clear(); + oldcells.reserve(maxDist * 4 * 4); + newcells.reserve(maxDist * 4 * 4); + + dist = 1; + + int vsize = ((1 << maxDistBits) * (1 << maxDistBits)) << 2; + visited.clear(); + visited.append(cast(ubyte)0, vsize); + visited_ptr = visited.ptr(); + visitedId = 2; + oldcells.clear(); + oldcells.append(Vector3d(0, 0, 0)); + Dir dir = position.direction.dir; + + int zstep = 1 << (maxDistBits + 1); + for (; dist < maxDistance; dist++) { + // for each distance + if (oldcells.length() == 0) { // no cells to pass through + import dlangui.core.logger; + Log.d("No more cells at distance ", dist); + break; + } + newcells.clear(); + visitedId++; + int maxUp = (((dist + 1) * 7) / 8) + 1; + int maxDown = - (dist < 3 ? 3 : (((dist + 1) * 7) / 8)) - 1; + //CRLog::trace("dist: %d cells: %d", dist, oldcells.length()); + for (int i = 0; i < oldcells.length(); i++) { + Vector3d pt = oldcells[i]; + assert(myAbs(pt.x) + myAbs(pt.y) + myAbs(pt.z) == dist - 1); + if (((pt.x + maxDist) | (pt.y + maxDist) | (pt.z + maxDist)) & maxDistMask) + continue; + if (dist > 2) { + // skip some directions + if (pt.y > maxUp || pt.y < maxDown) + continue; + if (dir == Dir.SOUTH) { + if (pt.z < -1) + continue; + } else if (dir == Dir.NORTH) { + if (pt.z > 1) + continue; + } else if (dir == Dir.EAST) { + if (pt.x < -1) + continue; + } else { // WEST + if (pt.x > 1) + continue; + } + } + int mx = pt.x; + int my = pt.y; + int mz = pt.z; + int sx = mx > 0 ? 1 : 0; + int sy = my > 0 ? 1 : 0; + int sz = mz > 0 ? 1 : 0; + if (mx < 0) { + mx = -mx; + sx = -1; + } + if (my < 0) { + my = -my; + sy = -1; + } + if (mz < 0) { + mz = -mz; + sz = -1; + } + int ymask = sy < 0 ? m0mask : 0; + int index = ((pt.x + m0) + ((pt.z + m0) << (maxDistBits + 1))) ^ ymask; + if (sx && sy && sz) { + //bool noStepZ = (mx > mz) || (my > mz); + // 1, 1, 1 + int xindex = index + (sy < 0 ? -sx : sx); + if (visited_ptr[xindex] != visitedId) { + visitCellNoCheck(pt.x + sx, pt.y, pt.z); + visited_ptr[xindex] = visitedId; + } + int zindex = index + (sz * sy > 0 ? zstep : -zstep); + if (visited_ptr[zindex] != visitedId) { + visitCellNoCheck(pt.x, pt.y, pt.z + sz); + visited_ptr[zindex] = visitedId; + } + if (!ymask && sy < 0) + index ^= m0mask; + if (visited_ptr[index] != visitedId) { + visitCellNoCheck(pt.x, pt.y + sy, pt.z); + visited_ptr[index] = visitedId; + } + } else { + // has 0 in one of coords + if (!sx) { + if (!sy) { + if (!sz) { + // 0, 0, 0 + visitCell(pt.x + 1, pt.y, pt.z); + visitCell(pt.x - 1, pt.y, pt.z); + visitCell(pt.x, pt.y + 1, pt.z); + visitCell(pt.x, pt.y - 1, pt.z); + visitCell(pt.x, pt.y, pt.z + 1); + visitCell(pt.x, pt.y, pt.z - 1); + } else { + // 0, 0, 1 + visitCell(pt.x, pt.y, pt.z + sz); + visitCell(pt.x + 1, pt.y, pt.z); + visitCell(pt.x - 1, pt.y, pt.z); + visitCell(pt.x, pt.y + 1, pt.z); + visitCell(pt.x, pt.y - 1, pt.z); + } + } else { + if (!sz) { + // 0, 1, 0 + visitCell(pt.x, pt.y + sy, pt.z); + visitCell(pt.x + 1, pt.y, pt.z); + visitCell(pt.x - 1, pt.y, pt.z); + visitCell(pt.x, pt.y, pt.z + 1); + visitCell(pt.x, pt.y, pt.z - 1); + } else { + // 0, 1, 1 + visitCell(pt.x, pt.y + sy, pt.z); + visitCell(pt.x, pt.y, pt.z + sz); + visitCell(pt.x + 1, pt.y, pt.z); + visitCell(pt.x - 1, pt.y, pt.z); + } + } + } else { + if (!sy) { + if (!sz) { + // 1, 0, 0 + visitCell(pt.x + sx, pt.y, pt.z); + visitCell(pt.x, pt.y + 1, pt.z); + visitCell(pt.x, pt.y - 1, pt.z); + visitCell(pt.x, pt.y, pt.z + 1); + visitCell(pt.x, pt.y, pt.z - 1); + } else { + // 1, 0, 1 + visitCell(pt.x + sx, pt.y, pt.z); + visitCell(pt.x, pt.y, pt.z + sz); + visitCell(pt.x, pt.y + 1, pt.z); + visitCell(pt.x, pt.y - 1, pt.z); + } + } else { + // 1, 1, 0 + visitCell(pt.x + sx, pt.y, pt.z); + visitCell(pt.x, pt.y + sy, pt.z); + visitCell(pt.x, pt.y, pt.z + 1); + visitCell(pt.x, pt.y, pt.z - 1); + } + } + } + } + newcells.swap(oldcells); + } + } +} + +static short[] TERRAIN_INIT_DATA = [ + // V + 10, 10, 10, 10, 30, 30, 30, 30, 30, 30, 30, 30, 10, 10, 10, 10, 10, + 10, 10, 20, 50, 50, 50, 50, 50, 50, 50, 50, 50, 20, 20, 20, 20, 10, + 10, 20, 20, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 20, 20, 10, + 10, 20, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 20, 10, + 10, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 20, 30, + 30, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 30, + 30, 50, 50, 50, 50, 50, 50, 50, 120, 50, 50, 50, 50, 50, 50, 50, 30, + 30, 50, 50, 50, 50, 50, 50, 110, 140, 130, 50, 50, 50, 50, 50, 50, 30, + 30, 50, 50, 50, 50, 50, 50, 140, 150, 140, 50, 50, 50, 50, 50, 50, 30, // <== + 30, 50, 50, 50, 50, 50, 50, 110, 140, 120, 50, 50, 50, 50, 50, 50, 30, + 30, 50, 50, 50, 50, 50, 50, 50, 110, 50, 50, 50, 50, 50, 50, 50, 30, + 30, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 10, + 30, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 10, + 30, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 40, 50, 10, + 30, 20, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 50, 40, 20, 20, 10, + 30, 20, 20, 50, 50, 50, 50, 50, 50, 50, 40, 20, 20, 20, 20, 20, 10, + 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 10, 10, 10, 10, 10, + // ^ +]; + +static short[] TERRAIN_SCALE_DATA = [ + // V + 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, + 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, + 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, + 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 30, 20, 20, 20, + 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 45, 20, 20, 20, + 20, 20, 20, 20, 20, 20, 20, 20, 80, 20, 20, 20, 40, 50, 40, 20, 20, + 20, 20, 20, 20, 20, 20, 90, 20, 80, 20, 30, 20, 20, 30, 20, 20, 20, + 20, 20, 20, 20, 20, 90, 20, 80, 30, 20, 40, 20, 20, 20, 20, 20, 20, + 20, 20, 20, 20, 20, 20, 90, 30, 40, 30, 50, 20, 20, 20, 20, 20, 20, // <== + 20, 20, 20, 20, 20, 20, 50, 20, 30, 20, 20, 20, 20, 20, 20, 20, 20, + 20, 20, 20, 20, 20, 20, 40, 70, 40, 90, 20, 40, 20, 20, 20, 20, 20, + 20, 20, 20, 20, 20, 20, 20, 20, 80, 20, 50, 70, 50, 20, 20, 20, 20, + 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 60, 20, 20, 20, 20, 20, + 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, + 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, + 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, + 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, + // ^ +]; + +void initWorldTerrain(World world, int terrSizeBits = 10, int x0 = 0, int z0 = 0) { + import dminer.core.terrain; + int terrSize = 1 << terrSizeBits; + TerrainGen scaleterr = TerrainGen(terrSizeBits, terrSizeBits); // 512x512 + scaleterr.generate(4321, TERRAIN_SCALE_DATA, terrSizeBits - 4); // init grid is 16x16 (1 << (9-7)) + scaleterr.filter(1); + //scaleterr.filter(2); + scaleterr.limit(0, 90); + TerrainGen terr = TerrainGen(terrSizeBits, terrSizeBits); // 512x512 + terr.generateWithScale(123456, TERRAIN_INIT_DATA, terrSizeBits - 4, scaleterr); // init grid is 16x16 (1 << (9-7)) + terr.filter(1); + terr.limit(5, CHUNK_DY * 3 / 4); + terr.filter(1); + for (int x = 0; x < terrSize; x++) { + for (int z = 0; z < terrSize; z++) { + int h = terr.get(x, z); + cell_t cell = 1; + //if (h < CHUNK_DY / 10) + // cell = 100; + //else if (h < CHUNK_DY / 5) + // cell = 101; + //else if (h < CHUNK_DY / 4) + // cell = 102; + //else if (h < CHUNK_DY / 3) + // cell = 103; + //else if (h < CHUNK_DY / 2) + // cell = 104; + //else + // cell = 105; + for (int y = 0; y < h; y++) { + world.setCell(x0 + x - terrSize / 2, y, z0 + z - terrSize / 2, cell); + } + } + } +} diff --git a/examples/dminer/src/minermain.d b/examples/dminer/src/minermain.d new file mode 100644 index 00000000..425f9ea9 --- /dev/null +++ b/examples/dminer/src/minermain.d @@ -0,0 +1,380 @@ +module minermain; + +import dlangui; +import dlangui.graphics.scene.scene3d; +import dlangui.graphics.scene.camera; +import dlangui.graphics.scene.mesh; +import dlangui.graphics.scene.material; +import dlangui.graphics.scene.effect; +import dlangui.graphics.scene.model; +import dlangui.graphics.scene.node; +import dlangui.graphics.scene.light; +import dlangui.graphics.glsupport; +import dlangui.graphics.gldrawbuf; +import dlangui.graphics.scene.effect; +import derelict.opengl3.gl3; +import derelict.opengl3.gl; + +import dminer.core.world; +import dminer.core.minetypes; +import dminer.core.blocks; + +mixin APP_ENTRY_POINT; + +/// entry point for dlangui based application +extern (C) int UIAppMain(string[] args) { + // embed resources listed in views/resources.list into executable + embeddedResourceList.addResources(embedResourcesFromList!("resources.list")()); + //embeddedResourceList.dumpEmbeddedResources(); + + // create window + Window window = Platform.instance.createWindow("DlangUI Voxel RPG", null, WindowFlag.Resizable, 600, 500); + window.mainWidget = new UiWidget(); + + //MeshPart part = new MeshPart(); + + // show window + window.show(); + + // run message loop + return Platform.instance.enterMessageLoop(); +} + +class UiWidget : VerticalLayout, CellVisitor { + this() { + super("OpenGLView"); + layoutWidth = FILL_PARENT; + layoutHeight = FILL_PARENT; + alignment = Align.Center; + try { + parseML(q{ + { + margins: 0 + padding: 0 + //backgroundImageId: "tx_fabric.tiled" + backgroundColor: 0x000000; + layoutWidth: fill + layoutHeight: fill + + VerticalLayout { + id: glView + margins: 0 + padding: 0 + layoutWidth: fill + layoutHeight: fill + TextWidget { text: "MinerD example"; textColor: "red"; fontSize: 150%; fontWeight: 800; fontFace: "Arial" } + VSpacer { layoutWeight: 30 } + TextWidget { id: lblPosition; text: ""; backgroundColor: 0x80202020; textColor: 0xFFE0E0 } + } + } + }, "", this); + } catch (Exception e) { + Log.e("Failed to parse dml", e); + } + // assign OpenGL drawable to child widget background + childById("glView").backgroundDrawable = DrawableRef(new OpenGLDrawable(&doDraw)); + + _scene = new Scene3d(); + + _cam = new Camera(); + _cam.translate(vec3(0, 14, -7)); + + _scene.activeCamera = _cam; + + dirLightNode = new Node3d(); + dirLightNode.rotateY(-15); + dirLightNode.translateX(2); + dirLightNode.translateY(3); + dirLightNode.translateZ(0); + dirLightNode.light = Light.createPoint(vec3(2, 2, 2), 15); //Light.createDirectional(vec3(1, 0.5, 0.5)); + dirLightNode.light.enabled = true; + _scene.addChild(dirLightNode); + + + int x0 = 0; + int y0 = 0; + int z0 = 0; + + + _minerMesh = new Mesh(VertexFormat(VertexElementType.POSITION, VertexElementType.NORMAL, VertexElementType.COLOR, VertexElementType.TEXCOORD0)); + _world = new World(); + + initWorldTerrain(_world); + + int cy0 = 3; + for (int y = CHUNK_DY - 1; y > 0; y--) + if (!_world.canPass(Vector3d(0, y, 0))) { + cy0 = y; + break; + } + _world.camPosition = Position(Vector3d(0, cy0, 0), Vector3d(0, 0, 1)); + + _world.setCellRange(Vector3d(3, 11, 5), Vector3d(1, 100, 1), 1); + _world.setCellRange(Vector3d(13, 11, -5), Vector3d(1, 100, 1), 3); + _world.setCellRange(Vector3d(-6, 11, 10), Vector3d(1, 100, 1), 4); + _world.setCellRange(Vector3d(-8, 11, 15), Vector3d(1, 100, 1), 5); + _world.setCellRange(Vector3d(12, 11, -7), Vector3d(1, 100, 1), 6); + _world.setCellRange(Vector3d(5, 11, 9), Vector3d(1, 100, 1), 7); + _world.setCellRange(Vector3d(9, 11, 5), Vector3d(1, 100, 1), 7); + _world.setCellRange(Vector3d(-5, 11, 9), Vector3d(1, 100, 1), 7); + _world.setCellRange(Vector3d(9, 11, -5), Vector3d(1, 100, 1), 7); + _world.setCellRange(Vector3d(5, 11, -9), Vector3d(1, 100, 1), 7); + _world.setCellRange(Vector3d(-9, 11, 5), Vector3d(1, 100, 1), 7); + _world.setCellRange(Vector3d(7, 11, 3), Vector3d(1, 100, 1), 8); + _world.setCellRange(Vector3d(-7, 11, 3), Vector3d(1, 100, 1), 8); + _world.setCellRange(Vector3d(7, 11, -3), Vector3d(1, 100, 1), 8); + _world.setCellRange(Vector3d(-7, 11, 3), Vector3d(1, 100, 1), 8); + + updateCamPosition(false); + updateMinerMesh(); + + Material minerMaterial = new Material(EffectId("textured.vert", "textured.frag", null), "blocks"); + minerMaterial.ambientColor = vec3(0.2,0.2,0.2); + minerMaterial.textureLinear = false; + Model minerDrawable = new Model(minerMaterial, _minerMesh); + Node3d minerNode = new Node3d("miner", minerDrawable); + _scene.addChild(minerNode); + + + focusable = true; + } + + /// process key event, return true if event is processed. + override bool onKeyEvent(KeyEvent event) { + if (event.action == KeyAction.KeyDown) { + switch(event.keyCode) with(KeyCode) { + case KEY_W: + case UP: + _world.camPosition.forward(1); + updateCamPosition(); + return true; + case DOWN: + case KEY_S: + _world.camPosition.backward(1); + updateCamPosition(); + return true; + case KEY_A: + case LEFT: + _world.camPosition.turnLeft(); + updateCamPosition(); + return true; + case KEY_D: + case RIGHT: + _world.camPosition.turnRight(); + updateCamPosition(); + return true; + case HOME: + case KEY_E: + _world.camPosition.moveUp(); + updateCamPosition(); + return true; + case END: + case KEY_Q: + _world.camPosition.moveDown(); + updateCamPosition(); + return true; + case KEY_Z: + _world.camPosition.moveLeft(); + updateCamPosition(); + return true; + case KEY_C: + _world.camPosition.moveRight(); + updateCamPosition(); + return true; + case KEY_F: + flying = !flying; + if (!flying) + _world.camPosition.pos.y = CHUNK_DY - 3; + updateCamPosition(); + return true; + case KEY_U: + enableMeshUpdate = !enableMeshUpdate; + updateCamPosition(); + return true; + default: + return false; + } + } + return false; + } + + Node3d dirLightNode; + + void visit(World world, ref Position camPosition, Vector3d pos, cell_t cell, int visibleFaces) { + BlockDef def = BLOCK_DEFS[cell]; + def.createFaces(world, world.camPosition, pos, visibleFaces, _minerMesh); + } + + bool flying = false; + bool enableMeshUpdate = true; + + void updateCamPosition(bool animateIt = true) { + import std.string; + import std.conv : to; + import std.utf : toUTF32; + import std.format; + + if (!flying) { + while(_world.canPass(_world.camPosition.pos + Vector3d(0, -1, 0))) + _world.camPosition.pos += Vector3d(0, -1, 0); + if(!_world.canPass(_world.camPosition.pos + Vector3d(0, -1, 0))) { + if (_world.canPass(_world.camPosition.pos + Vector3d(0, 1, 0))) + _world.camPosition.pos += Vector3d(0, 1, 0); + else if (_world.canPass(_world.camPosition.pos + Vector3d(1, 0, 0))) + _world.camPosition.pos += Vector3d(1, 0, 0); + else if (_world.canPass(_world.camPosition.pos + Vector3d(-1, 0, 0))) + _world.camPosition.pos += Vector3d(-1, 0, 0); + else if (_world.canPass(_world.camPosition.pos + Vector3d(0, 0, 1))) + _world.camPosition.pos += Vector3d(0, 0, 1); + else if (_world.canPass(_world.camPosition.pos + Vector3d(0, 0, -1))) + _world.camPosition.pos += Vector3d(0, 0, -1); + while(_world.canPass(_world.camPosition.pos + Vector3d(0, -1, 0))) + _world.camPosition.pos += Vector3d(0, -1, 0); + } + } + + setPos(vec3(_world.camPosition.pos.x + 0.5f, _world.camPosition.pos.y + 0.5f, _world.camPosition.pos.z + 0.5f), animateIt); + setAngle(_world.camPosition.direction.angle, animateIt); + Widget w = childById("lblPosition"); + string dir = _world.camPosition.direction.dir.to!string; + dstring s = format("pos(%d,%d) h=%d %s [F]lying: %s [U]pdateMesh: %s", _world.camPosition.pos.x, _world.camPosition.pos.z, _world.camPosition.pos.y, dir, + flying, enableMeshUpdate).toUTF32; + w.text = s; + if (enableMeshUpdate) + updateMinerMesh(); + } + + void updateMinerMesh() { + _minerMesh.reset(); + long ts = currentTimeMillis; + _world.visitVisibleCells(_world.camPosition, this); + long duration = currentTimeMillis - ts; + Log.d("DiamondVisitor finished in ", duration, " ms ", "Vertex count: ", _minerMesh.vertexCount); + + invalidate(); + //for (int i = 0; i < 20; i++) + // Log.d("vertex: ", _minerMesh.vertex(i)); + } + + World _world; + vec3 _position; + float _angle; + vec3 _animatingPosition; + float _animatingAngle; + + void setPos(vec3 newPos, bool animateIt = false) { + if (animateIt) { + _position = newPos; + } else { + _animatingPosition = newPos; + _position = newPos; + } + } + + void setAngle(float newAngle, bool animateIt = false) { + if (animateIt) { + _angle = newAngle; + } else { + _animatingAngle = newAngle; + _angle = newAngle; + } + } + + /// returns true is widget is being animated - need to call animate() and redraw + @property override bool animating() { return true; } + /// animates window; interval is time left from previous draw, in hnsecs (1/10000000 of second) + override void animate(long interval) { + //Log.d("animating"); + if (_animatingAngle != _angle) { + float delta = _angle - _animatingAngle; + if (delta > 180) + delta -= 360; + else if (delta < -180) + delta += 360; + float dist = delta < 0 ? -delta : delta; + if (dist < 5) { + _animatingAngle = _angle; + } else { + float speed = 360; + float step = speed * interval / 10000000.0f; + //Log.d("Rotate animation delta=", delta, " dist=", dist, " elapsed=", interval, " step=", step); + if (step > dist) + step = dist; + delta = delta * (step /dist); + _animatingAngle += delta; + } + } + if (_animatingPosition != _position) { + vec3 delta = _position - _animatingPosition; + float dist = delta.length; + if (dist < 0.01) { + _animatingPosition = _position; + // done + } else { + float speed = 8; + if (dist > 2) + speed = (dist - 2) * 3 + speed; + float step = speed * interval / 10000000.0f; + //Log.d("Move animation delta=", delta, " dist=", dist, " elapsed=", interval, " step=", step); + if (step > dist) + step = dist; + delta = delta * (step / dist); + _animatingPosition += delta; + } + } + invalidate(); + } + float angle = 0; + + Scene3d _scene; + Camera _cam; + Mesh _minerMesh; + + + /// this is OpenGLDrawableDelegate implementation + private void doDraw(Rect windowRect, Rect rc) { + _cam.setPerspective(rc.width, rc.height, 45.0f, 0.1, 100); + //_cam.translate(vec3( + // childById!ScrollBar("sbTranslationX").position / 10.0f, + // childById!ScrollBar("sbTranslationY").position / 10.0f, + // childById!ScrollBar("sbTranslationZ").position / 10.0f)); + //_world.camPosition.pos.x; + _cam.setIdentity(); + _cam.translate(_animatingPosition); + _cam.rotateY(_animatingAngle); + //_cam.rotateX(-15); + //_cam.lookAt(vec3(dir.x, dir.y, dir.z), vec3(pos.x, pos.y, pos.z), vec3(0,1,0)); + //_cam.translateX(_world.camPosition.pos.x); + //_cam.translateY(_world.camPosition.pos.y); + //_cam.translateZ(_world.camPosition.pos.z); + dirLightNode.setIdentity(); + dirLightNode.translate(_animatingPosition); + dirLightNode.rotateY(_animatingAngle); + //dirLightNode.setIdentity(); + //dirLightNode.translateX(_world.camPosition.pos.x); + //dirLightNode.translateY(_world.camPosition.pos.y); + //dirLightNode.translateZ(_world.camPosition.pos.z); + //_cam.rotateX(rotationX); + //_cam.rotateY(rotationY); + //_cam.rotateZ(rotationZ); + //_cam.translate(vec3(-1, -1.5, -1)); // - angle/1000 + //_cam.translate(vec3(0, 0, -1.1)); // - angle/1000 + //_cam.translate(vec3(0, 3, - angle/1000)); // + //_cam.rotateZ(30.0f + angle * 0.3456778); + + checkgl!glEnable(GL_CULL_FACE); + //checkgl!glDisable(GL_CULL_FACE); + checkgl!glEnable(GL_DEPTH_TEST); + checkgl!glCullFace(GL_BACK); + + _scene.drawScene(false); + + checkgl!glDisable(GL_DEPTH_TEST); + checkgl!glDisable(GL_CULL_FACE); + } + + ~this() { + destroy(_scene); + destroy(_world); + } +} diff --git a/examples/dminer/views/res/i18n/en.ini b/examples/dminer/views/res/i18n/en.ini new file mode 100644 index 00000000..576671ef --- /dev/null +++ b/examples/dminer/views/res/i18n/en.ini @@ -0,0 +1,40 @@ +EXIT=Exit +MENU_FILE=&File +MENU_FILE_NEW=&New +MENU_FILE_OPEN=&Open +MENU_FILE_OPEN_RECENT=Open recent +MENU_FILE_SAVE=&Save +MENU_FILE_EXIT=E&xit +MENU_EDIT=&Edit +MENU_EDIT_COPY=&Copy +MENU_EDIT_PASTE=&Paste +MENU_EDIT_CUT=Cu&t +MENU_EDIT_UNDO=&Undo +MENU_EDIT_REDO=&Redo +MENU_EDIT_INDENT=Indent block +MENU_EDIT_UNINDENT=Unindent block +MENU_EDIT_TOGGLE_LINE_COMMENT=Toggle line comment +MENU_EDIT_TOGGLE_BLOCK_COMMENT=Toggle block comment +MENU_EDIT_PREFERENCES=&Preferences +MENU_VIEW=&View +MENU_VIEW_LANGUAGE=Interface &Language +MENU_VIEW_LANGUAGE_EN=English +MENU_VIEW_LANGUAGE_RU=Русский +MENU_VIEW_THEME=&Theme +MENU_VIEW_THEME_DEFAULT=Default +MENU_VIEW_THEME_DARK=Dark +MENU_VIEW_THEME_CUSTOM1=Custom 1 +MENU_WINDOW=&Window +MENU_WINDOW_PREFERENCES=&Preferences +MENU_HELP=&Help +MENU_HELP_VIEW_HELP=&View help +MENU_HELP_ABOUT=&About +MENU_DEBUG_UPDATE_PREVIEW=Update Preview + +TAB_LONG_LIST=Long list +TAB_BUTTONS=Buttons +TAB_ANIMATION=Animation +TAB_TABLE_LAYOUT=Table layout +TAB_EDITORS=Editors +TAB_CANVAS=Canvas + diff --git a/examples/dminer/views/res/i18n/ru.ini b/examples/dminer/views/res/i18n/ru.ini new file mode 100644 index 00000000..ab52d3d0 --- /dev/null +++ b/examples/dminer/views/res/i18n/ru.ini @@ -0,0 +1,34 @@ +EXIT=Выход +MENU_FILE=&Файл +MENU_FILE_OPEN=&Открыть +MENU_FILE_OPEN_RECENT=Открыть из последних +MENU_FILE_SAVE=&Сохранить +MENU_FILE_EXIT=Вы&ход +MENU_EDIT=&Правка +MENU_EDIT_COPY=&Копировать +MENU_EDIT_PASTE=&Вставить +MENU_EDIT_CUT=Вырезать +MENU_EDIT_UNDO=&Отмена +MENU_EDIT_REDO=&Повторить +MENU_EDIT_PREFERENCES=&Настройки +MENU_VIEW=&Вид +MENU_VIEW_LANGUAGE=&Язык интерфейса +MENU_VIEW_LANGUAGE_EN=English +MENU_VIEW_LANGUAGE_RU=Русский +MENU_VIEW_THEME=&Тема +MENU_VIEW_THEME_DEFAULT=Стандартная +MENU_VIEW_THEME_DARK=Тёмная +MENU_VIEW_THEME_CUSTOM1=Пример 1 +MENU_WINDOW=&Окно +MENU_WINDOW_PREFERENCES=&Настройки +MENU_HELP=&Справка +MENU_HELP_VIEW_HELP=&Просмотр справки +MENU_HELP_ABOUT=&О программе + +TAB_LONG_LIST=Длинный список +TAB_BUTTONS=Кнопки +TAB_ANIMATION=Анимация +TAB_TABLE_LAYOUT=Табличный layout +TAB_EDITORS=Редакторы + + diff --git a/examples/dminer/views/res/mdpi/blocks.png b/examples/dminer/views/res/mdpi/blocks.png new file mode 100644 index 0000000000000000000000000000000000000000..0440ab4bfce11772ed34fd0f01bb48077f628b10 GIT binary patch literal 31980 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 literal 0 HcmV?d00001 diff --git a/examples/dminer/views/res/mdpi/cr3_logo.png b/examples/dminer/views/res/mdpi/cr3_logo.png new file mode 100644 index 0000000000000000000000000000000000000000..6e0d94ea246c86dcc40b8b8495f39b55c65918c6 GIT binary patch literal 8433 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 T;=!W_00000NkvXXu0mjfkMScx literal 0 HcmV?d00001 diff --git a/examples/dminer/views/resources.list b/examples/dminer/views/resources.list new file mode 100644 index 00000000..46354f80 --- /dev/null +++ b/examples/dminer/views/resources.list @@ -0,0 +1,4 @@ +res/i18n/en.ini +res/i18n/ru.ini +res/mdpi/cr3_logo.png +res/mdpi/blocks.png