mirror of
https://github.com/Uberi/Minetest-WorldEdit.git
synced 2025-01-10 01:50:28 +01:00
wip
This commit is contained in:
parent
2f2f5a7def
commit
f43bc5278e
@ -1,9 +1,7 @@
|
|||||||
--- Schematic serialization and deserialiation.
|
--- Schematic serialization and deserialiation.
|
||||||
-- @module worldedit.serialization
|
-- @module worldedit.serialization
|
||||||
|
|
||||||
worldedit.LATEST_SERIALIZATION_VERSION = 5
|
worldedit.LATEST_SERIALIZATION_VERSION = 6
|
||||||
local LATEST_SERIALIZATION_HEADER = worldedit.LATEST_SERIALIZATION_VERSION .. ":"
|
|
||||||
|
|
||||||
|
|
||||||
--[[
|
--[[
|
||||||
Serialization version history:
|
Serialization version history:
|
||||||
@ -15,6 +13,7 @@ Serialization version history:
|
|||||||
`name`, `param1`, `param2`, and `meta` fields.
|
`name`, `param1`, `param2`, and `meta` fields.
|
||||||
5: Added header and made `param1`, `param2`, and `meta` fields optional.
|
5: Added header and made `param1`, `param2`, and `meta` fields optional.
|
||||||
Header format: <Version>,<ExtraHeaderField1>,...:<Content>
|
Header format: <Version>,<ExtraHeaderField1>,...:<Content>
|
||||||
|
6: Much more complicated but also better format
|
||||||
--]]
|
--]]
|
||||||
|
|
||||||
|
|
||||||
@ -66,52 +65,101 @@ function worldedit.serialize(pos1, pos2)
|
|||||||
has_meta[hash_node_position(meta_positions[i])] = true
|
has_meta[hash_node_position(meta_positions[i])] = true
|
||||||
end
|
end
|
||||||
|
|
||||||
local pos = {x=pos1.x, y=0, z=0}
|
-- Decide axis of saved rows
|
||||||
local count = 0
|
local dim = vector.add(vector.subtract(pos2, pos1), 1)
|
||||||
local result = {}
|
local axis
|
||||||
while pos.x <= pos2.x do
|
if dim.x * dim.y < math.min(dim.y * dim.z, dim.x * dim.z) then
|
||||||
pos.y = pos1.y
|
axis = "z"
|
||||||
while pos.y <= pos2.y do
|
elseif dim.x * dim.z < math.min(dim.x * dim.y, dim.y * dim.z) then
|
||||||
pos.z = pos1.z
|
axis = "y"
|
||||||
while pos.z <= pos2.z do
|
elseif dim.y * dim.z < math.min(dim.x * dim.y, dim.x * dim.z) then
|
||||||
local node = get_node(pos)
|
axis = "x"
|
||||||
if node.name ~= "air" and node.name ~= "ignore" then
|
else
|
||||||
count = count + 1
|
axis = "x" -- X or Z are usually most efficient
|
||||||
|
end
|
||||||
|
local other1, other2 = worldedit.get_axis_others(axis)
|
||||||
|
|
||||||
local meta
|
-- Helper functions
|
||||||
if has_meta[hash_node_position(pos)] then
|
local function cur_new(pos, pos1)
|
||||||
meta = get_meta(pos):to_table()
|
return {
|
||||||
|
a = axis,
|
||||||
-- Convert metadata item stacks to item strings
|
p = {pos.x - pos1.x, pos.y - pos1.y, pos.z - pos1.z},
|
||||||
for _, invlist in pairs(meta.inventory) do
|
c = 1,
|
||||||
for index = 1, #invlist do
|
data = {},
|
||||||
local itemstack = invlist[index]
|
param1 = {},
|
||||||
if itemstack.to_string then
|
param2 = {},
|
||||||
invlist[index] = itemstack:to_string()
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
result[count] = {
|
|
||||||
x = pos.x - pos1.x,
|
|
||||||
y = pos.y - pos1.y,
|
|
||||||
z = pos.z - pos1.z,
|
|
||||||
name = node.name,
|
|
||||||
param1 = node.param1 ~= 0 and node.param1 or nil,
|
|
||||||
param2 = node.param2 ~= 0 and node.param2 or nil,
|
|
||||||
meta = meta,
|
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
pos.z = pos.z + 1
|
local function try_match(array, value)
|
||||||
|
local match_dist = 8
|
||||||
|
local first = math.max(1, #array - match_dist + 1)
|
||||||
|
local i = #array
|
||||||
|
while i >= first do
|
||||||
|
if array[i] == value then
|
||||||
|
return -(#array - i + 1)
|
||||||
end
|
end
|
||||||
pos.y = pos.y + 1
|
i = i - 1
|
||||||
end
|
end
|
||||||
pos.x = pos.x + 1
|
return value
|
||||||
|
end
|
||||||
|
local function cur_finish(result, cur)
|
||||||
|
if #cur.param1 == 1 and cur.param1[1] == 0 then
|
||||||
|
cur.param1 = nil
|
||||||
|
end
|
||||||
|
if #cur.param2 == 1 and cur.param2[1] == 0 then
|
||||||
|
cur.param2 = nil
|
||||||
|
end
|
||||||
|
result[#result + 1] = cur
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Serialize stuff
|
||||||
|
local pos = {}
|
||||||
|
local count = 0
|
||||||
|
local result = {}
|
||||||
|
local cur
|
||||||
|
pos[other1] = pos1[other1]
|
||||||
|
while pos[other1] <= pos2[other1] do
|
||||||
|
pos[other2] = pos1[other2]
|
||||||
|
while pos[other2] <= pos2[other2] do
|
||||||
|
pos[axis] = pos1[axis]
|
||||||
|
while pos[axis] <= pos2[axis] do
|
||||||
|
|
||||||
|
local node = get_node(pos)
|
||||||
|
if node.name ~= "air" and node.name ~= "ignore" then
|
||||||
|
if cur == nil then -- Start a new row
|
||||||
|
cur = cur_new(pos, pos1, axis, other1, other2)
|
||||||
|
cur.data[1] = node.name
|
||||||
|
cur.param1[1] = node.param1
|
||||||
|
cur.param2[1] = node.param2
|
||||||
|
else -- Push to existing row
|
||||||
|
local next_c = cur.c + 1
|
||||||
|
cur.c = next_c
|
||||||
|
cur.data[next_c] = try_match(cur.data, node.name)
|
||||||
|
cur.param1[next_c] = try_match(cur.param1, node.param1)
|
||||||
|
cur.param2[next_c] = try_match(cur.param2, node.param2)
|
||||||
|
end
|
||||||
|
count = count + 1
|
||||||
|
else
|
||||||
|
if cur ~= nil then -- Finish row
|
||||||
|
cur_finish(result, cur)
|
||||||
|
cur = nil
|
||||||
|
end
|
||||||
|
end
|
||||||
|
pos[axis] = pos[axis] + 1
|
||||||
|
|
||||||
|
end
|
||||||
|
if cur ~= nil then -- Finish leftover row
|
||||||
|
cur_finish(result, cur)
|
||||||
|
cur = nil
|
||||||
|
end
|
||||||
|
pos[other2] = pos[other2] + 1
|
||||||
|
end
|
||||||
|
pos[other1] = pos[other1] + 1
|
||||||
end
|
end
|
||||||
-- Serialize entries
|
-- Serialize entries
|
||||||
result = minetest.serialize(result)
|
result = minetest.serialize(result)
|
||||||
return LATEST_SERIALIZATION_HEADER .. result, count
|
return tonumber(worldedit.LATEST_SERIALIZATION_VERSION) .. "," ..
|
||||||
|
string.format("%d,%d,%d:", dim.x, dim.y, dim.z) .. result, count
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Contains code based on [table.save/table.load](http://lua-users.org/wiki/SaveTableToFile)
|
-- Contains code based on [table.save/table.load](http://lua-users.org/wiki/SaveTableToFile)
|
||||||
|
Loading…
Reference in New Issue
Block a user