using System; using System.Collections; using System.Collections.Generic; using Substrate.Utility; namespace Substrate { public class BlockLight { private struct LightRecord { public int x; public int y; public int z; public int str; public LightRecord (int _x, int _y, int _z, int s) { x = _x; y = _y; z = _z; str = s; } } private IBoundedLitBlockCollection _blockset; // Maintains internal state of multi-block relighting algorithms private BitArray _lightbit; private Queue _update; public delegate IBoundedLitBlockCollection NeighborLookupHandler (int relx, int rely, int relz); public event NeighborLookupHandler ResolveNeighbor; public BlockLight (IBoundedLitBlockCollection blockset) { _blockset = blockset; _lightbit = new BitArray(_blockset.XDim * 3 * _blockset.ZDim * 3 * _blockset.YDim); _update = new Queue(); } public BlockLight (BlockLight bl) { _blockset = bl._blockset; _lightbit = new BitArray(_blockset.XDim * 3 * _blockset.ZDim * 3 * _blockset.YDim); _update = new Queue(); } public void UpdateBlockLight (int lx, int ly, int lz) { BlockKey primary = new BlockKey(lx, ly, lz); _update.Enqueue(primary); BlockInfo info = _blockset.GetInfo(lx, ly, lz); if (info.Luminance > BlockInfo.MIN_LUMINANCE || info.TransmitsLight) { QueueRelight(new BlockKey(lx - 1, ly, lz)); QueueRelight(new BlockKey(lx + 1, ly, lz)); QueueRelight(new BlockKey(lx, ly - 1, lz)); QueueRelight(new BlockKey(lx, ly + 1, lz)); QueueRelight(new BlockKey(lx, ly, lz - 1)); QueueRelight(new BlockKey(lx, ly, lz + 1)); } UpdateBlockLight(); } public void UpdateBlockSkyLight (int lx, int ly, int lz) { BlockKey primary = new BlockKey(lx, ly, lz); _update.Enqueue(primary); UpdateBlockSkyLight(); } public void RebuildBlockLight () { for (int x = 0; x < _blockset.XDim; x++) { for (int z = 0; z < _blockset.ZDim; z++) { for (int y = 0; y < _blockset.YDim; y++) { BlockInfo info = _blockset.GetInfo(x, y, z); if (info.Luminance > 0) { SpreadBlockLight(x, y, z); } } } } } public void RebuildBlockSkyLight () { IBoundedLitBlockCollection[,] chunkMap = LocalBlockLightMap(); int[,] heightMap = LocalHeightMap(chunkMap); // Optimization - only need to queue at level of highest neighbor's height for (int x = 0; x < _blockset.XDim; x++) { for (int z = 0; z < _blockset.ZDim; z++) { int xi = x + _blockset.XDim; int zi = z + _blockset.ZDim; int h = heightMap[xi, zi]; h = Math.Max(h, heightMap[xi, zi - 1]); h = Math.Max(h, heightMap[xi - 1, zi]); h = Math.Max(h, heightMap[xi + 1, zi]); h = Math.Max(h, heightMap[xi, zi + 1]); for (int y = h + 1; y < _blockset.YDim; y++) { _blockset.SetSkyLight(x, y, z, BlockInfo.MAX_LUMINANCE); } //QueueRelight(new BlockKey(x, h, z)); SpreadSkyLight(chunkMap, heightMap, x, h, z); } } } public void RebuildHeightMap () { for (int x = 0; x < _blockset.XDim; x++) { for (int z = 0; z < _blockset.ZDim; z++) { for (int y = _blockset.YDim - 1; y >= 0; y--) { BlockInfo info = _blockset.GetInfo(x, y, z); if (info.Opacity > BlockInfo.MIN_OPACITY || !info.TransmitsLight) { _blockset.SetHeight(x, z, y); break; } } } } } public void StitchBlockLight () { IBoundedLitBlockCollection[,] map = LocalBlockLightMap(); if (map[1, 0] != null) { StitchBlockLight(map[1, 0], BlockCollectionEdge.EAST); } if (map[0, 1] != null) { StitchBlockLight(map[0, 1], BlockCollectionEdge.NORTH); } if (map[1, 2] != null) { StitchBlockLight(map[1, 2], BlockCollectionEdge.WEST); } if (map[2, 1] != null) { StitchBlockLight(map[2, 1], BlockCollectionEdge.SOUTH); } } // TODO: Revise to cache the specified chunk into local map public void StitchBlockLight (IBoundedLitBlockCollection chunk, BlockCollectionEdge edge) { if (chunk.XDim != _blockset.XDim || chunk.YDim != _blockset.YDim || chunk.ZDim != _blockset.ZDim) { throw new InvalidOperationException("BlockLight must have same dimensions to be stitched"); } switch (edge) { case BlockCollectionEdge.EAST: for (int x = 0; x < _blockset.XDim; x++) { for (int y = 0; y < _blockset.YDim; y++) { TestBlockLight(chunk, x, y, 0, x, y, _blockset.ZDim - 1); } } break; case BlockCollectionEdge.NORTH: for (int z = 0; z < _blockset.ZDim; z++) { for (int y = 0; y < _blockset.YDim; y++) { TestBlockLight(chunk, 0, y, z, _blockset.XDim - 1, y, z); } } break; case BlockCollectionEdge.WEST: for (int x = 0; x < _blockset.XDim; x++) { for (int y = 0; y < _blockset.YDim; y++) { TestBlockLight(chunk, x, y, _blockset.ZDim - 1, x, y, 0); } } break; case BlockCollectionEdge.SOUTH: for (int z = 0; z < _blockset.ZDim; z++) { for (int y = 0; y < _blockset.YDim; y++) { TestBlockLight(chunk, _blockset.XDim - 1, y, z, 0, y, z); } } break; } UpdateBlockLight(); } public void StitchBlockSkyLight () { IBoundedLitBlockCollection[,] map = LocalBlockLightMap(); if (map[1, 0] != null) { StitchBlockSkyLight(map[1, 0], BlockCollectionEdge.EAST); } if (map[0, 1] != null) { StitchBlockSkyLight(map[0, 1], BlockCollectionEdge.NORTH); } if (map[1, 2] != null) { StitchBlockSkyLight(map[1, 2], BlockCollectionEdge.WEST); } if (map[2, 1] != null) { StitchBlockSkyLight(map[2, 1], BlockCollectionEdge.SOUTH); } } public void StitchBlockSkyLight (IBoundedLitBlockCollection chunk, BlockCollectionEdge edge) { if (chunk.XDim != _blockset.XDim || chunk.YDim != _blockset.YDim || chunk.ZDim != _blockset.ZDim) { throw new InvalidOperationException("BlockLight must have same dimensions to be stitched"); } switch (edge) { case BlockCollectionEdge.EAST: for (int x = 0; x < _blockset.XDim; x++) { for (int y = 0; y < _blockset.YDim; y++) { TestSkyLight(chunk, x, y, 0, x, y, _blockset.ZDim - 1); } } break; case BlockCollectionEdge.NORTH: for (int z = 0; z < _blockset.ZDim; z++) { for (int y = 0; y < _blockset.YDim; y++) { TestSkyLight(chunk, 0, y, z, _blockset.XDim - 1, y, z); } } break; case BlockCollectionEdge.WEST: for (int x = 0; x < _blockset.XDim; x++) { for (int y = 0; y < _blockset.YDim; y++) { TestSkyLight(chunk, x, y, _blockset.ZDim - 1, x, y, 0); } } break; case BlockCollectionEdge.SOUTH: for (int z = 0; z < _blockset.ZDim; z++) { for (int y = 0; y < _blockset.YDim; y++) { TestSkyLight(chunk, _blockset.XDim - 1, y, z, 0, y, z); } } break; } UpdateBlockSkyLight(); } private void UpdateBlockLight () { while (_update.Count > 0) { BlockKey k = _update.Dequeue(); int index = LightBitmapIndex(k); _lightbit[index] = false; IBoundedLitBlockCollection cc = LocalChunk(k.x, k.y, k.z); if (cc == null) { continue; } int lle = NeighborLight(k.x, k.y, k.z - 1); int lln = NeighborLight(k.x - 1, k.y, k.z); int lls = NeighborLight(k.x, k.y, k.z + 1); int llw = NeighborLight(k.x + 1, k.y, k.z); int lld = NeighborLight(k.x, k.y - 1, k.z); int llu = NeighborLight(k.x, k.y + 1, k.z); int x = (k.x + _blockset.XDim * 2) % _blockset.XDim; int y = k.y; int z = (k.z + _blockset.ZDim * 2) % _blockset.ZDim; int lightval = cc.GetBlockLight(x, y, z); BlockInfo info = cc.GetInfo(x, y, z); int light = Math.Max(info.Luminance, 0); light = Math.Max(light, lle - 1); light = Math.Max(light, lln - 1); light = Math.Max(light, lls - 1); light = Math.Max(light, llw - 1); light = Math.Max(light, lld - 1); light = Math.Max(light, llu - 1); light = Math.Max(light - info.Opacity, 0); if (light != lightval) { //Console.WriteLine("Block Light: ({0},{1},{2}) " + lightval + " -> " + light, k.x, k.y, k.z); cc.SetBlockLight(x, y, z, light); if (info.TransmitsLight) { QueueRelight(new BlockKey(k.x - 1, k.y, k.z)); QueueRelight(new BlockKey(k.x + 1, k.y, k.z)); QueueRelight(new BlockKey(k.x, k.y - 1, k.z)); QueueRelight(new BlockKey(k.x, k.y + 1, k.z)); QueueRelight(new BlockKey(k.x, k.y, k.z - 1)); QueueRelight(new BlockKey(k.x, k.y, k.z + 1)); } } } } private void UpdateBlockSkyLight () { while (_update.Count > 0) { BlockKey k = _update.Dequeue(); int index = LightBitmapIndex(k); _lightbit[index] = false; IBoundedLitBlockCollection cc = LocalChunk(k.x, k.y, k.z); if (cc == null) { continue; } int x = (k.x + _blockset.XDim) % _blockset.XDim; int y = k.y; int z = (k.z + _blockset.ZDim) % _blockset.ZDim; int lightval = cc.GetSkyLight(x, y, z); BlockInfo info = cc.GetInfo(x, y, z); int light = BlockInfo.MIN_LUMINANCE; if (cc.GetHeight(x, z) <= y) { light = BlockInfo.MAX_LUMINANCE; } else { int lle = NeighborSkyLight(k.x, k.y, k.z - 1); int lln = NeighborSkyLight(k.x - 1, k.y, k.z); int lls = NeighborSkyLight(k.x, k.y, k.z + 1); int llw = NeighborSkyLight(k.x + 1, k.y, k.z); int lld = NeighborSkyLight(k.x, k.y - 1, k.z); int llu = NeighborSkyLight(k.x, k.y + 1, k.z); light = Math.Max(light, lle - 1); light = Math.Max(light, lln - 1); light = Math.Max(light, lls - 1); light = Math.Max(light, llw - 1); light = Math.Max(light, lld - 1); light = Math.Max(light, llu - 1); } light = Math.Max(light - info.Opacity, 0); if (light != lightval) { //Console.WriteLine("Block SkyLight: ({0},{1},{2}) " + lightval + " -> " + light, k.x, k.y, k.z); cc.SetSkyLight(x, y, z, light); if (info.TransmitsLight) { QueueRelight(new BlockKey(k.x - 1, k.y, k.z)); QueueRelight(new BlockKey(k.x + 1, k.y, k.z)); QueueRelight(new BlockKey(k.x, k.y - 1, k.z)); QueueRelight(new BlockKey(k.x, k.y + 1, k.z)); QueueRelight(new BlockKey(k.x, k.y, k.z - 1)); QueueRelight(new BlockKey(k.x, k.y, k.z + 1)); } } } } private void SpreadBlockLight (int lx, int ly, int lz) { BlockInfo primary = _blockset.GetInfo(lx, ly, lz); int primaryLight = _blockset.GetBlockLight(lx, ly, lz); int priLum = Math.Max(primary.Luminance - primary.Opacity, 0); if (primaryLight < priLum) { _blockset.SetBlockLight(lx, ly, lz, priLum); } if (primaryLight > primary.Luminance - 1 && !primary.TransmitsLight) { return; } Queue spread = new Queue(); spread.Enqueue(new LightRecord(lx - 1, ly, lz, primary.Luminance - 1)); spread.Enqueue(new LightRecord(lx + 1, ly, lz, primary.Luminance - 1)); spread.Enqueue(new LightRecord(lx, ly - 1, lz, primary.Luminance - 1)); spread.Enqueue(new LightRecord(lx, ly + 1, lz, primary.Luminance - 1)); spread.Enqueue(new LightRecord(lx, ly, lz - 1, primary.Luminance - 1)); spread.Enqueue(new LightRecord(lx, ly, lz + 1, primary.Luminance - 1)); while (spread.Count > 0) { LightRecord rec = spread.Dequeue(); IBoundedLitBlockCollection cc = LocalChunk(rec.x, rec.y, rec.z); if (cc == null) { continue; } int x = (rec.x + _blockset.XDim) % _blockset.XDim; int y = rec.y; int z = (rec.z + _blockset.ZDim) % _blockset.ZDim; BlockInfo info = cc.GetInfo(x, y, z); int light = cc.GetBlockLight(x, y, z); int dimStr = Math.Max(rec.str - info.Opacity, 0); if (dimStr > light) { cc.SetBlockLight(x, y, z, dimStr); if (info.TransmitsLight) { spread.Enqueue(new LightRecord(rec.x - 1, rec.y, rec.z, dimStr - 1)); spread.Enqueue(new LightRecord(rec.x + 1, rec.y, rec.z, dimStr - 1)); spread.Enqueue(new LightRecord(rec.x, rec.y - 1, rec.z, dimStr - 1)); spread.Enqueue(new LightRecord(rec.x, rec.y + 1, rec.z, dimStr - 1)); spread.Enqueue(new LightRecord(rec.x, rec.y, rec.z - 1, dimStr - 1)); spread.Enqueue(new LightRecord(rec.x, rec.y, rec.z + 1, dimStr - 1)); } } } } private void SpreadSkyLight (IBoundedLitBlockCollection[,] chunkMap, int[,] heightMap, int lx, int ly, int lz) { BlockInfo primary = _blockset.GetInfo(lx, ly, lz); int primaryLight = _blockset.GetSkyLight(lx, ly, lz); int priLum = Math.Max(BlockInfo.MAX_LUMINANCE - primary.Opacity, 0); if (primaryLight < priLum) { _blockset.SetSkyLight(lx, ly, lz, priLum); } if (primaryLight > BlockInfo.MAX_LUMINANCE - 1 || !primary.TransmitsLight) { return; } Queue spread = new Queue(); int lxi = lx + _blockset.XDim; int lzi = lz + _blockset.ZDim; if (heightMap[lxi, lzi] > ly - 1) { spread.Enqueue(new LightRecord(lx, ly - 1, lz, BlockInfo.MAX_LUMINANCE - 1)); } else { spread.Enqueue(new LightRecord(lx, ly - 1, lz, BlockInfo.MAX_LUMINANCE)); } if (heightMap[lxi - 1, lzi] > ly) { spread.Enqueue(new LightRecord(lx - 1, ly, lz, BlockInfo.MAX_LUMINANCE - 1)); } if (heightMap[lxi + 1, lzi] > ly) { spread.Enqueue(new LightRecord(lx + 1, ly, lz, BlockInfo.MAX_LUMINANCE - 1)); } if (heightMap[lxi, lzi] > ly + 1) { spread.Enqueue(new LightRecord(lx, ly + 1, lz, BlockInfo.MAX_LUMINANCE - 1)); } if (heightMap[lxi, lzi] > ly) { spread.Enqueue(new LightRecord(lx, ly, lz - 1, BlockInfo.MAX_LUMINANCE - 1)); } if (heightMap[lxi, lzi + 1] > ly) { spread.Enqueue(new LightRecord(lx, ly, lz + 1, BlockInfo.MAX_LUMINANCE - 1)); } while (spread.Count > 0) { LightRecord rec = spread.Dequeue(); int xi = rec.x + _blockset.XDim; int zi = rec.z + _blockset.ZDim; IBoundedLitBlockCollection cc = chunkMap[xi / _blockset.XDim, zi / _blockset.ZDim]; if (cc == null) { continue; } int x = xi % _blockset.XDim; int y = rec.y; int z = zi % _blockset.ZDim; BlockInfo info = cc.GetInfo(x, y, z); int light = cc.GetSkyLight(x, y, z); int dimStr = Math.Max(rec.str - info.Opacity, 0); if (dimStr > light) { cc.SetSkyLight(x, y, z, dimStr); if (info.TransmitsLight) { /*spread.Enqueue(new LightRecord(rec.x - 1, rec.y, rec.z, dimStr - 1)); spread.Enqueue(new LightRecord(rec.x + 1, rec.y, rec.z, dimStr - 1)); if (rec.y > 0) spread.Enqueue(new LightRecord(rec.x, rec.y - 1, rec.z, dimStr - 1)); if (rec.y < _blockset.YDim - 1) spread.Enqueue(new LightRecord(rec.x, rec.y + 1, rec.z, dimStr - 1)); spread.Enqueue(new LightRecord(rec.x, rec.y, rec.z - 1, dimStr - 1)); spread.Enqueue(new LightRecord(rec.x, rec.y, rec.z + 1, dimStr - 1));*/ if (heightMap[xi, zi] > rec.y - 1) { spread.Enqueue(new LightRecord(rec.x, rec.y - 1, rec.z, dimStr - 1)); } else { spread.Enqueue(new LightRecord(rec.x, rec.y - 1, rec.z, dimStr)); } if (heightMap[xi - 1, zi] > rec.y) { spread.Enqueue(new LightRecord(rec.x - 1, rec.y, rec.z, dimStr - 1)); } if (heightMap[xi + 1, zi] > rec.y) { spread.Enqueue(new LightRecord(rec.x + 1, rec.y, rec.z, dimStr - 1)); } if (heightMap[xi, zi] > rec.y + 1) { spread.Enqueue(new LightRecord(rec.x, rec.y + 1, rec.z, dimStr - 1)); } if (heightMap[xi, zi - 1] > rec.y) { spread.Enqueue(new LightRecord(rec.x, rec.y, rec.z - 1, dimStr - 1)); } if (heightMap[xi, zi + 1] > rec.y) { spread.Enqueue(new LightRecord(rec.x, rec.y, rec.z + 1, dimStr - 1)); } } } } } private int LightBitmapIndex (BlockKey key) { int x = key.x + _blockset.XDim; int y = key.y; int z = key.z + _blockset.ZDim; int zstride = _blockset.YDim; int xstride = _blockset.ZDim * 3 * zstride; return (x * xstride) + (z * zstride) + y; } private void QueueRelight (BlockKey key) { if (key.x < -15 || key.x >= 31 || key.z < -15 || key.z >= 31) { return; } int index = LightBitmapIndex(key); if (!_lightbit[index]) { _lightbit[index] = true; _update.Enqueue(key); } } private IBoundedLitBlockCollection LocalChunk (int lx, int ly, int lz) { if (ly < 0 || ly >= _blockset.YDim) { return null; } if (lx < 0) { if (lz < 0) { return OnResolveNeighbor(-1, 0, -1); } else if (lz >= _blockset.ZDim) { return OnResolveNeighbor(-1, 0, 1); } return OnResolveNeighbor(-1, 0, 0); } else if (lx >= _blockset.XDim) { if (lz < 0) { return OnResolveNeighbor(1, 0, -1); } else if (lz >= _blockset.ZDim) { return OnResolveNeighbor(1, 0, 1); } return OnResolveNeighbor(1, 0, 0); } else { if (lz < 0) { return OnResolveNeighbor(0, 0, -1); } else if (lz >= _blockset.ZDim) { return OnResolveNeighbor(0, 0, 1); } return _blockset; } } private int NeighborLight (int x, int y, int z) { if (y < 0 || y >= _blockset.YDim) { return 0; } IBoundedLitBlockCollection src = LocalChunk(x, y, z); if (src == null) { return 0; } x = (x + _blockset.XDim * 2) % _blockset.XDim; z = (z + _blockset.ZDim * 2) % _blockset.ZDim; BlockInfo info = src.GetInfo(x, y, z); if (!info.TransmitsLight) { return info.Luminance; } int light = src.GetBlockLight(x, y, z); return Math.Max(light, info.Luminance); } private int NeighborSkyLight (int x, int y, int z) { if (y < 0 || y >= _blockset.YDim) { return 0; } IBoundedLitBlockCollection src = LocalChunk(x, y, z); if (src == null) { return 0; } x = (x + _blockset.XDim * 2) % _blockset.XDim; z = (z + _blockset.ZDim * 2) % _blockset.ZDim; BlockInfo info = src.GetInfo(x, y, z); if (!info.TransmitsLight) { return BlockInfo.MIN_LUMINANCE; } int light = src.GetSkyLight(x, y, z); return light; } private int NeighborHeight (int x, int z) { IBoundedLitBlockCollection src = LocalChunk(x, 0, z); if (src == null) { return _blockset.YDim - 1; } x = (x + _blockset.XDim * 2) % _blockset.XDim; z = (z + _blockset.ZDim * 2) % _blockset.ZDim; return src.GetHeight(x, z); } private void TestBlockLight (IBoundedLitBlockCollection chunk, int x1, int y1, int z1, int x2, int y2, int z2) { int light1 = _blockset.GetBlockLight(x1, y1, z1); int light2 = chunk.GetBlockLight(x2, y2, z2); int lum1 = _blockset.GetInfo(x1, y1, z1).Luminance; int lum2 = chunk.GetInfo(x2, y2, z2).Luminance; int v1 = Math.Max(light1, lum1); int v2 = Math.Max(light2, lum2); if (Math.Abs(v1 - v2) > 1) { QueueRelight(new BlockKey(x1, y1, z1)); } } private void TestSkyLight (IBoundedLitBlockCollection chunk, int x1, int y1, int z1, int x2, int y2, int z2) { int light1 = _blockset.GetSkyLight(x1, y1, z1); int light2 = chunk.GetSkyLight(x2, y2, z2); if (Math.Abs(light1 - light2) > 1) { QueueRelight(new BlockKey(x1, y1, z1)); } } private IBoundedLitBlockCollection[,] LocalBlockLightMap () { IBoundedLitBlockCollection[,] map = new IBoundedLitBlockCollection[3, 3]; map[0, 0] = OnResolveNeighbor(-1, 0, -1); map[0, 1] = OnResolveNeighbor(-1, 0, 0); map[0, 2] = OnResolveNeighbor(-1, 0, 1); map[1, 0] = OnResolveNeighbor(0, 0, -1); map[1, 1] = _blockset; map[1, 2] = OnResolveNeighbor(0, 0, 1); map[2, 0] = OnResolveNeighbor(1, 0, -1); map[2, 1] = OnResolveNeighbor(1, 0, 0); map[2, 2] = OnResolveNeighbor(1, 0, 1); return map; } private int[,] LocalHeightMap (IBoundedLitBlockCollection[,] chunkMap) { int[,] map = new int[3 * _blockset.XDim, 3 * _blockset.ZDim]; for (int xi = 0; xi < 3; xi++) { int xoff = xi * _blockset.XDim; for (int zi = 0; zi < 3; zi++) { int zoff = zi * _blockset.ZDim; if (chunkMap[xi, zi] == null) { continue; } for (int x = 0; x < _blockset.XDim; x++) { int xx = xoff + x; for (int z = 0; z < _blockset.ZDim; z++) { int zz = zoff + z; map[xx, zz] = chunkMap[xi, zi].GetHeight(x, z); } } } } return map; } private IBoundedLitBlockCollection OnResolveNeighbor (int relX, int relY, int relZ) { if (ResolveNeighbor != null) { IBoundedLitBlockCollection n = ResolveNeighbor(relX, relY, relZ); if (n == null) { return null; } if (n.XDim != _blockset.XDim || n.YDim != _blockset.YDim || n.ZDim != _blockset.ZDim) { throw new InvalidOperationException("Subscriber returned incompatible ILitBlockCollection"); } return n; } return null; } } }