private static void Scatter(OCMap map, List <Vector3i> list)          // рассеивание
        {
            OCSunLightMap lightmap = map.GetSunLightmap();

            for (int i = 0; i < list.Count; i++)
            {
                Vector3i pos = list[i];
                if (pos.y < 0)
                {
                    continue;
                }

                OCBlockData block = map.GetBlock(pos);
                int         light = lightmap.GetLight(pos) - OCLightComputerUtils.GetLightStep(block);
                if (light <= MIN_LIGHT)
                {
                    continue;
                }

                foreach (Vector3i dir in Vector3i.directions)
                {
                    Vector3i nextPos = pos + dir;
                    block = map.GetBlock(nextPos);
                    if (block.IsAlpha() && lightmap.SetMaxLight((byte)light, nextPos))
                    {
                        list.Add(nextPos);
                    }
                    if (!block.IsEmpty())
                    {
                        OCLightComputerUtils.SetLightDirty(map, nextPos);
                    }
                }
            }
        }
        public static void Scatter(OCMap map, OCColumnMap columnMap, int cx, int cz)
        {
            int x1 = cx * OCChunk.SIZE_X;
            int z1 = cz * OCChunk.SIZE_Z;

            int x2 = x1 + OCChunk.SIZE_X;
            int z2 = z1 + OCChunk.SIZE_Z;

            OCSunLightMap lightmap = map.GetSunLightmap();

            list.Clear();
            for (int x = x1; x < x2; x++)
            {
                for (int z = z1; z < z2; z++)
                {
                    int maxY = ComputeMaxY(lightmap, x, z) + 1;
                    for (int y = 0; y < maxY; y++)
                    {
                        if (lightmap.GetLight(x, y, z) > MIN_LIGHT)
                        {
                            list.Add(new Vector3i(x, y, z));
                        }
                    }
                }
            }
            Scatter(map, columnMap, list);
        }
        private static int ComputeMaxY(OCSunLightMap lightmap, int x, int z)
        {
            int maxY = lightmap.GetSunHeight(x, z);

            maxY = Mathf.Max(maxY, lightmap.GetSunHeight(x - 1, z));
            maxY = Mathf.Max(maxY, lightmap.GetSunHeight(x + 1, z));
            maxY = Mathf.Max(maxY, lightmap.GetSunHeight(x, z - 1));
            maxY = Mathf.Max(maxY, lightmap.GetSunHeight(x, z + 1));
            return(maxY);
        }
        private static void RemoveLight(OCMap map, List <Vector3i> list)
        {
            OCSunLightMap lightmap = map.GetSunLightmap();

            foreach (Vector3i pos in list)
            {
                lightmap.SetLight(MAX_LIGHT, pos);
            }

            List <Vector3i> lightPoints = new List <Vector3i>();

            for (int i = 0; i < list.Count; i++)
            {
                Vector3i pos = list[i];
                if (pos.y < 0)
                {
                    continue;
                }
                if (lightmap.IsSunLight(pos.x, pos.y, pos.z))
                {
                    lightPoints.Add(pos);
                    continue;
                }
                byte light = (byte)(lightmap.GetLight(pos) - STEP_LIGHT);
                lightmap.SetLight(MIN_LIGHT, pos);
                if (light <= MIN_LIGHT)
                {
                    continue;
                }

                foreach (Vector3i dir in Vector3i.directions)
                {
                    Vector3i    nextPos = pos + dir;
                    OCBlockData block   = map.GetBlock(nextPos);
                    if (block.IsAlpha())
                    {
                        if (lightmap.GetLight(nextPos) <= light)
                        {
                            list.Add(nextPos);
                        }
                        else
                        {
                            lightPoints.Add(nextPos);
                        }
                    }
                    if (!block.IsEmpty())
                    {
                        OCLightComputerUtils.SetLightDirty(map, nextPos);
                    }
                }
            }

            Scatter(map, lightPoints);
        }
        public static void RecomputeLightAtPosition(OCMap map, Vector3i pos)
        {
            OCSunLightMap lightmap     = map.GetSunLightmap();
            int           oldSunHeight = lightmap.GetSunHeight(pos.x, pos.z);

            ComputeRayAtPosition(map, pos.x, pos.z);
            int newSunHeight = lightmap.GetSunHeight(pos.x, pos.z);

            if (newSunHeight < oldSunHeight)              // свет опустился
            // добавляем свет
            {
                list.Clear();
                for (int ty = newSunHeight; ty <= oldSunHeight; ty++)
                {
                    pos.y = ty;
                    lightmap.SetLight(MIN_LIGHT, pos);
                    list.Add(pos);
                }
                Scatter(map, list);
            }
            if (newSunHeight > oldSunHeight)              // свет поднялся
            // удаляем свет
            {
                list.Clear();
                for (int ty = oldSunHeight; ty <= newSunHeight; ty++)
                {
                    pos.y = ty;
                    list.Add(pos);
                }
                RemoveLight(map, list);
            }

            if (newSunHeight == oldSunHeight)
            {
                if (map.GetBlock(pos).IsAlpha())
                {
                    UpdateLight(map, pos);
                }
                else
                {
                    RemoveLight(map, pos);
                }
            }
        }
		private static int ComputeMaxY(OCSunLightMap lightmap, int x, int z) {
			int maxY = lightmap.GetSunHeight(x, z);
			maxY = Mathf.Max(maxY, lightmap.GetSunHeight(x-1, z  ));
			maxY = Mathf.Max(maxY, lightmap.GetSunHeight(x+1, z  ));
			maxY = Mathf.Max(maxY, lightmap.GetSunHeight(x,   z-1));
			maxY = Mathf.Max(maxY, lightmap.GetSunHeight(x,   z+1));
			return maxY;
		}