summaryrefslogtreecommitdiff
path: root/main.lua
blob: 93dadd9c7b91188159d9ffc017cc079c2ee91a66 (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
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
-- advtrains track map generator
-- Usage: lua main.lua path/to/world

-- Viewport maximum coordinate in all directions
local maxc = 5000

-- embed an image called "world.png"
local wimg = false
-- image file resolution (not world resolution!)
local wimresx = 3000
local wimresy = 3000
-- one pixel is ... nodes
local wimscale = 4

-- y ranges and line colors
-- Minimum y level drawn
local drminy = -30
-- Color of min y level
local colminy = {r=0, g=0, b=255}
-- Intermediate color
local colmedy = {r=255, g=0, b=0}
-- Maximum y level drawn
local drmaxy = 80
-- Color of max y level
local colmaxy = {r=255, g=255, b=0}


datapath = (arg[1] or "").."/"


--Constant for maximum connection value/division of the circle
AT_CMAX = 16

advtrains = {}
minetest = {}
core = minetest

--table for track nodes/connections
trackconns = {}

-- math library seems to be missing this function
math.hypot = function(a,b) return math.sqrt(a*a + b*b) end

-- need to declare this for trackdefs
function attrans(str) return str end

-- pos to string
local function pts(pos)
	return pos.x .. "," .. pos.y .. "," .. pos.z
end

--Advtrains dump (special treatment of pos and sigd)
function atdump(t, intend)
	local str
	if type(t)=="table" then
		if t.x and t.y and t.z then
			str=minetest.pos_to_string(t)
		elseif t.p and t.s then -- interlocking sigd
			str="S["..minetest.pos_to_string(t.p).."/"..t.s.."]"
		else
			str="{"
			local intd = (intend or "") .. "  "
			for k,v in pairs(t) do
				if type(k)~="string" or not string.match(k, "^path[_]?") then
					-- do not print anything path-related
					str = str .. "\n" .. intd .. atdump(k, intd) .. " = " ..atdump(v, intd)
				end
			end
			str = str .. "\n" .. (intend or "") .. "}"
		end
	elseif type(t)=="boolean" then
		if t then
			str="true"
		else
			str="false"
		end
	elseif type(t)=="function" then
		str="<function>"
	elseif type(t)=="userdata" then
		str="<userdata>"
	else
		str=""..t
	end
	return str
end

dofile("vector.lua")
local serialize = dofile("serialize.lua")
dofile("helpers.lua")
dofile("tracks.lua")
dofile("track_defs.lua")

dofile("nodedb.lua")


function parse_args(argv) 
	local i = 1
	local no_trains = false
	local datapath, mappath, worldimage
	while i <= #argv do
		local a = argv[i]
		if (a == "-m") or (a == "--map-file") then
			-- only draw trains – requires specifying an already drawn file
			i = i+1
			if not argv[i] then
				error(("missing filename after `%s'"):format(a))
			end
			mappath = argv[i]
		elseif (a == "-t") or (a == "--no-trains") then
			-- do not draw trains
			no_trains = true
		elseif (a == "-w") or (a == "--world-image") then
			-- overlay over world image
			i = i+1
			if not argv[i] then
				error(("missing filename after `%s'"):format(a))
			end
			worldimage = argv[i]
		else
			datapath = a
		end
		
		i = i + 1
	end
	return datapath, mappath, no_trains, worldimage
end

datapath, mappath, no_trains, worldimage = parse_args(arg)

-- Load saves
local tbl = serialize.read_from_file(datapath.."advtrains_core.ls")
--local file, err = io.open(datapath.."advtrains_trains", "r")
--local tbl = minetest.deserialize(file:read("*a"))
if type(tbl) ~= "table" then
	error("Trains file: not a table")
end
advtrains.trains = tbl.trains
--file:close()

--ndb contains the defs, while ndb2 is the actual contents
dofile("nodedb.lua")
local file, err = io.open(datapath.."advtrains_ndb4.ls", "r")
--tbl = minetest.deserialize(file:read("*a"))
--if type(tbl) ~= "table" then
	--error("Node database file: not a table")
--end
--advtrains.ndb.load_data(tbl)
advtrains.ndb.load_callback(file)
--file:close()

-- open svg file

local svgfile = io.open(datapath.."out.svg", "w")


if mappath then
	mapfile = io.open(mappath, "r")
	cont = mapfile:read("*a"):sub(1, -7) -- remove </svg> end tag
	svgfile:write(cont)
else
	svgfile:write([[
<?xml version="1.0" standalone="no" ?>
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 20010904//EN"
  "http://www.w3.org/TR/2001/REC-SVG-20010904/DTD/svg10.dtd">
<svg width="1024" height="800" xmlns="http://www.w3.org/2000/svg"
  xmlns:xlink="http://www.w3.org/1999/xlink" ]])


	svgfile:write('viewBox="'..(-maxc)..' '..(-maxc)..' '..(2*maxc)..' '..(2*maxc)..'" >')

	svgfile:write([[
<circle cx="0" cy="0" r="5" stroke="red" stroke-width="1" />
]])
end
if worldimage then
	local wimx = -(wimresx*wimscale/2)
	local wimy = -(wimresy*wimscale/2)
	local wimw = wimresx*wimscale
	local wimh = wimresy*wimscale
	
	svgfile:write('<image xlink:href="'..worldimage..'" x="'..wimx..'" y="'..wimy..'" height="'..wimh..'px" width="'..wimw..'px"/>')
end

local function writec(text)
	--print("\n"..text)
	svgfile:write("<!-- " .. text .. " -->\n")
end


-- everything set up. Start generating an SVG
-- All nodes that have been fit into a polyline are removed from the NDB, in order to not draw them again.

-- "Restart points" for the breadth-first traverser (set when more than 2 conns present)
-- {pos = <position>, connid = <int>, conn = <conndef>}
-- Note that the node at "pos" is already deleted from the NDB at the time of recall, therefore "conn" is specified
local bfs_rsp = {}

local ndb_nodes_handled = 0

-- Points of the current polyline. Inserted as xyz vectors, maybe we'll use the y value one day
local current_polyline = {}

-- Traverser function from interlocking, highly modified
local function gen_rsp_polyline(rsp)

	-- trick a bit
	local pos, connid, conns = rsp.pos, 1, {rsp.conn}
	current_polyline[#current_polyline+1] = pos
	
	while true do
		local adj_pos, adj_connid, conn_idx, nextrail_y, next_conns = advtrains.get_adjacent_rail(pos, conns, connid)
		if not adj_pos then
			-- proceed one more node, for seamless turnout transitions
			current_polyline[#current_polyline+1] = advtrains.pos_add_dir(pos, conns[connid].c)
			return
		end
		-- continue traversing
		local conn_mainbranch
		for nconnid, nconn in ipairs(next_conns) do
			if adj_connid ~= nconnid then
				if not conn_mainbranch then
					--use the first one found to continue
					conn_mainbranch = nconnid
					--writec(nconnid.." nconn mainbranch")
				else
					-- insert bfs reminders for other conns
					table.insert(bfs_rsp, {pos = adj_pos, connid = nconnid, conn = nconn})
					--writec(nconnid.." nconn bfs")
				end
			end
		end
		
		-- save in polyline and delete from ndb
		--writec("Saved pos: "..pts(adj_pos).." mainbranch cont "..conn_mainbranch.." nextconns "..atdump(next_conns))
		current_polyline[#current_polyline+1] = adj_pos
		advtrains.ndb.clear(adj_pos)
		ndb_nodes_handled = ndb_nodes_handled + 1
		
		pos, connid, conns = adj_pos, conn_mainbranch, next_conns
		
	end
end

plcnt = 0


local function hexcolor(clr)
	return "#"..advtrains.hex(clr.r)..advtrains.hex(clr.g)..advtrains.hex(clr.b)
end

local function cfactor(ry)
	local y = ry - (ry%4)

	local fac = (y-drminy)/(drmaxy-drminy)
	return fac
end

local function pl_header(fac)
	
	local color
	if fac<0.5 then
		color = {
			r = colminy.r + (colmedy.r-colminy.r)*2*fac,
			g = colminy.g + (colmedy.g-colminy.g)*2*fac,
			b = colminy.b + (colmedy.b-colminy.b)*2*fac,
		}
	else
		color = {
			r = colmedy.r + (colmaxy.r-colmedy.r)*(2*fac-1),
			g = colmedy.g + (colmaxy.g-colmedy.g)*(2*fac-1),
			b = colmedy.b + (colmaxy.b-colmedy.b)*(2*fac-1),
		}
	end
	
	local c = hexcolor(color)
	return '<polyline style="fill:none;stroke:'..c..';stroke-width:1" points="'
end

local function polyline_write(pl)
	local p1y = cfactor(pl[1].y)
	local str = {}
	
	if p1y <= 1 and p1y >= 0 then
		table.insert(str, pl_header(p1y))
	end
	
	local i
	local e
	local lastcf = p1y
	local lastldir = {x=0, y=0}
	for i=1,#pl do
		e = pl[i]
		local cf = cfactor(e.y)
		if cf ~= lastcf then
			if lastcf <= 1 and lastcf >= 0 then
				-- insert final point
				-- Note that we mirror y, so positive z is up
				table.insert(str, e.x .. "," .. -(e.z) .. " ")
				table.insert(str, '" />\n')
				plcnt = plcnt + 1
			end
			if cf <= 1 and cf >= 0 then
				table.insert(str, pl_header(cf))
			end
		end
		if cf <= 1 and cf >= 0 then
			-- Note that we mirror y, so positive z is up
			table.insert(str, e.x .. "," .. -(e.z) .. " ")
		end
		lastcf = cf
	end
	if lastcf <= 1 and lastcf >= 0 then
		table.insert(str, '" />\n')
	end
	svgfile:write(table.concat(str))
	plcnt = plcnt + 1
end


	

-- while there are entries in the nodedb
-- 1. find a starting point
if not mappath then
	local stpos, conns = advtrains.ndb.mapper_find_starting_point()
	while stpos do
		
		writec("Restart at position "..pts(stpos))
		for connid, conn in ipairs(conns) do
			table.insert(bfs_rsp, {pos = stpos, connid = connid, conn = conn})
		end
		advtrains.ndb.clear(stpos)
		
		-- 2. while there are BFS entries
		while #bfs_rsp > 0 do
			-- make polylines
			local current_rsp = bfs_rsp[#bfs_rsp]
			bfs_rsp[#bfs_rsp] = nil
			--print("Starting polyline at "..pts(current_rsp.pos).."/"..current_rsp.connid)
			
			
			current_polyline = {}
			
			gen_rsp_polyline(current_rsp)
			
			polyline_write(current_polyline)
			
			io.write("Progress ", ndb_nodes_handled, "+", ndb_nodes_notrack, "/", ndb_nodes_total, "=", math.floor(((ndb_nodes_handled+ndb_nodes_notrack)/ndb_nodes_total)*100), "%\r")
		end
		stpos, conns = advtrains.ndb.mapper_find_starting_point()
	end
end
-- draw trains
trains = 0
stopped = 0
lines = {}
running = {}
if not no_trains then
	for i,v in pairs(advtrains.trains) do
		pos = v.last_pos
		color = "green"
		if v.velocity == 0 then
			color = "orange"
			stopped = stopped + 1
		end
		svgfile:write("<circle cx=\""..pos.x.."\" cy=\""..-pos.z.."\" r=\"3\" stroke=\""..color.."\" stroke-width=\"1\" fill=\"none\" />")
		if v.line then
			lines[v.line] = (lines[v.line] or 0) + 1
			if v.velocity ~= 0 then
				running[v.line] = (running[v.line] or 0) + 1
			end
			svgfile:write(" <text x=\""..(pos.x+5).."\" y=\""..-pos.z.."\" class=\"trainline\">"..v.line.."</text>")
		end
		trains = trains+1
	end
end

svgfile:write("</svg>")
svgfile:close()

print("\nWrote",plcnt,"polylines. Processed", ndb_nodes_handled, "track,",ndb_nodes_notrack, "non-track nodes out of", ndb_nodes_total)
print("Drew "..trains.." trains. "..stopped.." stopped trains.")
print("\n Number of trains moving/total:")
for i,v in pairs(lines) do
	print(i..":            "..(running[i] or 0).."/"..v)
end