-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathstructure.go
65 lines (56 loc) · 2.25 KB
/
structure.go
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
package schematic
import (
"fmt"
"github.com/df-mc/dragonfly/server/block"
"github.com/df-mc/dragonfly/server/world"
"reflect"
)
// schematic implements the structure of a Schematic, providing methods to read from it.
type schematic struct {
Data map[string]interface{}
w, h, l int
materials string
blocks []uint8
metadata []uint8
}
// init initialises the schematic structure, parsing several values from the NBT data.
func (s *schematic) init() error {
s.w, s.h, s.l = int(s.Data["Width"].(int16)), int(s.Data["Height"].(int16)), int(s.Data["Length"].(int16))
s.materials = s.Data["Materials"].(string)
blocks, metadata := reflect.ValueOf(s.Data["Blocks"]), reflect.ValueOf(s.Data["Data"])
blockSlice, metadataSlice := reflect.MakeSlice(reflect.SliceOf(blocks.Type().Elem()), blocks.Len(), blocks.Len()), reflect.MakeSlice(reflect.SliceOf(blocks.Type().Elem()), metadata.Len(), metadata.Len())
reflect.Copy(blockSlice, blocks)
reflect.Copy(metadataSlice, metadata)
s.blocks, s.metadata = blockSlice.Interface().([]byte), metadataSlice.Interface().([]byte)
if len(s.blocks) != s.w*s.h*s.l || len(s.metadata) != s.w*s.h*s.l {
return fmt.Errorf("blocks and metadata were expected to be %v bytes long both (%v*%v*%v), but blocks has length %v and metadata has length %v", s.w*s.h*s.l, s.w, s.h, s.l, len(s.blocks), len(s.metadata))
}
return nil
}
// Dimensions returns the dimensions of the schematic as width, height and length respectively.
func (s *schematic) Dimensions() [3]int {
return [3]int{s.w, s.h, s.l}
}
// At returns the block found at a given position in the schematic. If any of the X, Y or Z coordinates passed
// are out of the bounds of the schematic, At will panic.
func (s *schematic) At(x, y, z int, _ func(int, int, int) world.Block) (world.Block, world.Liquid) {
index := (y*s.l+z)*s.w + x
id, meta := s.blocks[index], s.metadata[index]
if id == 0 {
// Don't write air blocks: We simply return nil so that no block is placed at all.
return nil, nil
}
old := oldBlock{id: id, metadata: meta}
if converted, ok := editionConversion[old]; ok {
old = converted
}
n, ok := conversion[old]
if !ok {
return block.Air{}, nil
}
ret, ok := world.BlockByName(n.name, n.properties)
if !ok {
return block.Air{}, nil
}
return ret, nil
}