summaryrefslogtreecommitdiff
path: root/src/main/kotlin/io/dico/parcels2/blockvisitor/Schematic.kt
blob: 8d2084b0ddb8d49f0be13352990ac32a518c2d72 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
package io.dico.parcels2.blockvisitor

import io.dico.parcels2.util.Region
import io.dico.parcels2.util.Vec3i
import io.dico.parcels2.util.get
import org.bukkit.Bukkit
import org.bukkit.Material
import org.bukkit.World
import org.bukkit.block.data.BlockData

private val air = Bukkit.createBlockData(Material.AIR)

// TODO order paste such that attachables are placed after the block they depend on
class Schematic {
    val size: Vec3i get() = _size!!
    private var _size: Vec3i? = null
        set(value) {
            field?.let { throw IllegalStateException() }
            field = value
        }

    private var blockDatas: Array<BlockData?>? = null
    //private var extra: Map<Vec3i, (Block) -> Unit>? = null
    private var isLoaded = false; private set
    private val traverser: RegionTraverser = RegionTraverser.upward

    fun getLoadTask(world: World, region: Region): TimeLimitedTask = {
        _size = region.size

        val data = arrayOfNulls<BlockData>(region.blockCount).also { blockDatas = it }
        //val extra = mutableMapOf<Vec3i, (Block) -> Unit>().also { extra = it }
        val blocks = traverser.traverseRegion(region)

        for ((index, vec) in blocks.withIndex()) {
            markSuspensionPoint()
            val block = world[vec]
            if (block.y > 255) continue
            val blockData = block.blockData
            data[index] = blockData
        }

        isLoaded = true
    }

    fun getPasteTask(world: World, position: Vec3i): TimeLimitedTask = {
        if (!isLoaded) throw IllegalStateException()
        val region = Region(position, _size!!)
        val blocks = traverser.traverseRegion(region, worldHeight = world.maxHeight)
        val blockDatas = blockDatas!!

        val postponed = mutableListOf<Pair<Vec3i, BlockData>>()

        for ((index, vec) in blocks.withIndex()) {
            markSuspensionPoint()
            val block = world[vec]
            val type = blockDatas[index] ?: air
            if (type !== air && isAttachable(type.material)) {


                postponed += vec to type
            } else {
                block.blockData = type
            }
        }

        for ((vec, data) in postponed) {

        }
    }

}