aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorSingularis <singularis@volny.cz>2025-04-21 07:21:53 +0200
committerorwell <orwell@bleipb.de>2025-05-27 20:22:01 +0200
commit95a5586e0fe196e2291acaf128b8d1b4f8a32510 (patch)
tree6cc5ff9a64e533c8d9262f965e2246dfa2b3da24
parent1d6913cec797b3848c93e8c6b16c3b6b2711d38c (diff)
downloadadvtrains-95a5586e0fe196e2291acaf128b8d1b4f8a32510.tar.gz
advtrains-95a5586e0fe196e2291acaf128b8d1b4f8a32510.tar.bz2
advtrains-95a5586e0fe196e2291acaf128b8d1b4f8a32510.zip
[advtrains] přechod na Advtrains 2.5.0 (první pokus)
- [ch_core/lib] drobná oprava - [advtrains_attachment_offset_patch] použití initial_properties
-rw-r--r--.build.yml42
-rw-r--r--.gitignore3
-rw-r--r--advtrains/api_doc.txt2
-rw-r--r--advtrains/atc.lua235
-rw-r--r--advtrains/copytool.lua20
-rw-r--r--advtrains/couple.lua41
-rw-r--r--advtrains/craft_items.lua2
-rw-r--r--advtrains/crafting.lua17
-rw-r--r--advtrains/debugitems.lua86
-rw-r--r--advtrains/formspec.lua111
-rw-r--r--advtrains/helpers.lua187
-rw-r--r--advtrains/init.lua57
l---------advtrains/locale/README.md1
-rw-r--r--advtrains/lzb.lua10
-rw-r--r--advtrains/nodedb.lua14
-rw-r--r--advtrains/occupation.lua128
-rw-r--r--advtrains/p_mesecon_iface.lua26
-rw-r--r--advtrains/passive.lua94
-rw-r--r--advtrains/path.lua75
-rw-r--r--advtrains/po/README.md70
-rw-r--r--advtrains/po/advtrains.pot632
-rw-r--r--advtrains/po/de.po724
-rw-r--r--advtrains/po/fr.po728
-rwxr-xr-xadvtrains/po/update-translations.sh28
-rw-r--r--advtrains/po/zh_CN.po696
-rw-r--r--advtrains/po/zh_TW.po696
-rw-r--r--advtrains/poconvert.lua185
-rw-r--r--advtrains/protection.lua8
-rw-r--r--advtrains/settingtypes.txt7
-rw-r--r--advtrains/signals.lua163
-rw-r--r--advtrains/sounds/advtrains_crossing_bell.oggbin30681 -> 24656 bytes
-rw-r--r--advtrains/spec/poconvert_spec.lua70
-rw-r--r--advtrains/spec/texture_spec.lua19
-rw-r--r--advtrains/spec/wagons_spec.lua40
-rw-r--r--advtrains/texture.lua228
-rw-r--r--advtrains/textures/advtrains_wagon_prop_tool.pngbin0 -> 779 bytes
-rw-r--r--advtrains/track_reg_helper.lua779
-rw-r--r--advtrains/trackdb_legacy.lua27
-rw-r--r--advtrains/trackplacer.lua668
-rw-r--r--advtrains/tracks.lua935
-rw-r--r--advtrains/trainhud.lua197
-rw-r--r--advtrains/trainlogic.lua159
-rw-r--r--advtrains/wagonprop_tool.lua43
-rw-r--r--advtrains/wagons.lua266
-rw-r--r--advtrains_interlocking/README.md85
-rw-r--r--advtrains_interlocking/approach.lua5
-rw-r--r--advtrains_interlocking/ars.lua68
-rw-r--r--advtrains_interlocking/autonaming.lua69
-rw-r--r--advtrains_interlocking/database.lua1028
-rw-r--r--advtrains_interlocking/demosignals.lua104
-rw-r--r--advtrains_interlocking/init.lua6
-rw-r--r--advtrains_interlocking/mod.conf2
-rw-r--r--advtrains_interlocking/route_prog.lua71
-rw-r--r--advtrains_interlocking/route_ui.lua222
-rw-r--r--advtrains_interlocking/routesetting.lua165
-rw-r--r--advtrains_interlocking/signal_api.lua872
-rw-r--r--advtrains_interlocking/signal_aspect_ui.lua281
-rw-r--r--advtrains_interlocking/smartroute.lua257
-rw-r--r--advtrains_interlocking/spec/basic_signalling_spec.lua106
l---------advtrains_interlocking/spec/fixtures/advtrains_helpers.lua1
-rw-r--r--advtrains_interlocking/spec/mineunit.conf0
-rw-r--r--advtrains_interlocking/spec/signal_group_spec.lua95
-rw-r--r--advtrains_interlocking/tcb_ts_ui.lua717
-rw-r--r--advtrains_interlocking/textures/at_il_ts_highlight_particle.pngbin0 -> 7164 bytes
-rw-r--r--advtrains_interlocking/tool.lua115
-rw-r--r--advtrains_interlocking/train_sections.lua93
-rw-r--r--advtrains_interlocking/tsr_rail.lua12
-rw-r--r--advtrains_line_automation/init.lua1
-rw-r--r--advtrains_signals_japan/.gitignore1
-rw-r--r--advtrains_signals_japan/init.lua461
-rw-r--r--advtrains_signals_japan/mod.conf6
-rw-r--r--advtrains_signals_japan/textures/advtrains_signals_japan_mast.pngbin0 -> 180 bytes
-rw-r--r--advtrains_signals_ks/init.lua388
-rw-r--r--advtrains_signals_ks/models/advtrains_signals_ks_distant_smr0.obj3293
-rw-r--r--advtrains_signals_ks/models/advtrains_signals_ks_distant_smr30.obj3291
-rw-r--r--advtrains_signals_ks/models/advtrains_signals_ks_distant_smr45.obj3296
-rw-r--r--advtrains_signals_ks/models/advtrains_signals_ks_distant_smr60.obj3293
-rw-r--r--advtrains_signals_ks/models/advtrains_signals_ks_sign_ne3_smr0.obj212
-rw-r--r--advtrains_signals_ks/models/advtrains_signals_ks_sign_ne3_smr30.obj213
-rw-r--r--advtrains_signals_ks/models/advtrains_signals_ks_sign_ne3_smr45.obj213
-rw-r--r--advtrains_signals_ks/models/advtrains_signals_ks_sign_ne3_smr60.obj213
-rw-r--r--advtrains_signals_ks/models/advtrains_signals_ks_sign_ne4_smr0.obj148
-rw-r--r--advtrains_signals_ks/models/advtrains_signals_ks_sign_ne4_smr30.obj151
-rw-r--r--advtrains_signals_ks/models/advtrains_signals_ks_sign_ne4_smr45.obj151
-rw-r--r--advtrains_signals_ks/models/advtrains_signals_ks_sign_ne4_smr60.obj151
-rw-r--r--advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x1.pngbin0 -> 77 bytes
-rw-r--r--advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x2.pngbin0 -> 83 bytes
-rw-r--r--advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x3.pngbin0 -> 80 bytes
-rw-r--r--advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x4.pngbin0 -> 78 bytes
-rw-r--r--advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x5.pngbin0 -> 75 bytes
-rw-r--r--advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne4.pngbin0 -> 99 bytes
-rw-r--r--advtrains_signals_ks/textures/advtrains_signals_ks_vs_inv.pngbin0 -> 4234 bytes
-rwxr-xr-xadvtrains_signals_muc_ubahn/init.lua88
-rw-r--r--advtrains_signals_muc_ubahn/mod.conf5
-rw-r--r--advtrains_signals_muc_ubahn/models/advtrains_signals_muc_ubahn_wsig_l.obj2137
-rw-r--r--advtrains_signals_muc_ubahn/models/advtrains_signals_muc_ubahn_wsig_r.obj2137
-rw-r--r--advtrains_signals_muc_ubahn/models/advtrains_signals_muc_ubahn_wsig_t.mtl12
-rw-r--r--advtrains_signals_muc_ubahn/models/advtrains_signals_muc_ubahn_wsig_t.obj2138
-rw-r--r--advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_hp0.pngbin0 -> 4991 bytes
-rw-r--r--advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_hp1.pngbin0 -> 4985 bytes
-rw-r--r--advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_hp2.pngbin0 -> 5048 bytes
-rw-r--r--advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_hp3.pngbin0 -> 4984 bytes
-rwxr-xr-xadvtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_template.pngbin0 -> 5395 bytes
-rw-r--r--advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_vr0.pngbin0 -> 5055 bytes
-rw-r--r--advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_vr1.pngbin0 -> 5047 bytes
-rw-r--r--advtrains_trackmap/fsrender.lua41
-rw-r--r--advtrains_trackmap/grid.lua107
-rw-r--r--advtrains_trackmap/init.lua14
-rw-r--r--advtrains_trackmap/mod.conf6
-rw-r--r--advtrains_trackmap/viewer.lua43
-rw-r--r--advtrains_train_track/init.lua587
-rw-r--r--advtrains_train_track/mod.conf2
-rw-r--r--advtrains_train_track/settingtypes.txt4
-rw-r--r--assets/mm_trains_bvbg.pngbin0 -> 31081 bytes
-rw-r--r--assets/muc_ubahn_uvs.pngbin0 -> 93694 bytes
-rw-r--r--assets/signal_wall.blend1bin0 -> 602788 bytes
-rw-r--r--assets/signal_wall_ceiling_muc_ubahn.blendbin0 -> 603780 bytes
-rw-r--r--assets/signal_wall_muc_ubahn.blendbin0 -> 603780 bytes
-rwxr-xr-xmake_release.sh53
-rw-r--r--modpack.conf1
120 files changed, 32471 insertions, 3279 deletions
diff --git a/.build.yml b/.build.yml
index 303348f..5234336 100644
--- a/.build.yml
+++ b/.build.yml
@@ -1,22 +1,38 @@
image: debian/stable
packages:
- git
+- lua5.1
+- luarocks
- curl
- minetest-server
- unzip
- wget
- lua-busted
+- luajit
sources :
- https://git.sr.ht/~gpcf/advtrains
+artifacts:
+- advtrains.luacov.report.out
+- advtrains_interlocking.luacov.report.out
+
tasks:
-- download_mt_server: |
- mkdir bin
- wget https://lifomaps.de/advtrains-test/builtin.tar.gz
- tar xf builtin.tar.gz
- curl https://lifomaps.de/advtrains-test/minetestserver -o ~/bin/minetestserver
- chmod +x ~/bin/minetestserver
+- install_mineunit : |
+ for i in {busted,luacov}; do
+ luarocks install --local --lua-version 5.1 $i >/dev/null
+ done
+ luarocks install --local --lua-version 5.1 --server=https://luarocks.org/dev mineunit
+- run_unit_tests : |
+ cd advtrains/serialize_lib
+ ~/.luarocks/bin/busted
+ for i in {advtrains,advtrains_interlocking}; do
+ cd ../$i
+ ~/.luarocks/bin/mineunit -c
+ ~/.luarocks/bin/mineunit -r
+ sed -n '/^File/,$p' luacov.report.out
+ mv luacov.report.out ~/$i.luacov.report.out
+ done
- install_mt_game : |
curl -L https://github.com/minetest/minetest_game/archive/master.zip -o master.zip
mkdir -p .minetest/games/
@@ -28,13 +44,6 @@ tasks:
curl https://lifomaps.de/advtrains-test/testworld.tar.gz -o ~/testworld.tar.gz
cd .minetest/worlds/
tar xf ../../testworld.tar.gz
-- run_unit_tests : |
- cd advtrains/advtrains
- busted
- cd ../advtrains_interlocking
- busted
- cd ../serialize_lib
- busted
- activate_test_env: |
cd advtrains
git merge --no-commit origin/luaatcdebug
@@ -45,4 +54,9 @@ tasks:
git clone https://git.bananach.space/basic_trains.git/
- run_test_world: |
echo "bind_address = 127.0.0.1" > minetest.conf
- ~/bin/minetestserver --port 31111 --gameid minetest_game --config ~/minetest.conf --world ~/.minetest/worlds/advtrains_testworld
+ minetestserver --port 31111 --gameid minetest_game --config ~/minetest.conf --world ~/.minetest/worlds/advtrains_testworld --logfile ~/minetest.log
+- test_po_files : |
+ cd advtrains/advtrains
+ for f in po/*.po; do
+ luajit -e 'require("poconvert").from_string("advtrains", io.input():read("*a"))' < $f
+ done
diff --git a/.gitignore b/.gitignore
index b3180de..42eaf06 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,3 +1,6 @@
## Eclipse project files & directories
.project
.settings
+luacov.*
+advtrains/locale/*.tr
+advtrains/po/*~
diff --git a/advtrains/api_doc.txt b/advtrains/api_doc.txt
index 5668ba3..6b338a7 100644
--- a/advtrains/api_doc.txt
+++ b/advtrains/api_doc.txt
@@ -18,8 +18,6 @@ advtrains.register_wagon(name, prototype, description, inventory_image)
# Wagon prototype properties
{
... all regular luaentity properties (mesh, textures, collisionbox a.s.o)...
- drives_on = {default=true},
- ^- used to define the tracktypes (see below) that wagon can drive on. The tracktype identifiers are given as keys, similar to privileges)
max_speed = 10,
^- optional, default 10: defines the maximum speed this wagon can drive. The maximum speed of a train is determined by the wagon with the lowest max_speed value.
seats = {
diff --git a/advtrains/atc.lua b/advtrains/atc.lua
index d314b3c..8e54b08 100644
--- a/advtrains/atc.lua
+++ b/advtrains/atc.lua
@@ -94,6 +94,7 @@ function atc.train_reset_command(train, keep_tarvel)
train.atc_brake_target=nil
train.atc_wait_finish=nil
train.atc_wait_autocouple=nil
+ train.atc_wait_signal=nil
train.atc_arrow=nil
if not keep_tarvel then
train.tarvelocity=nil
@@ -106,142 +107,8 @@ local apn_func=function(pos)
-- FIX for long-persisting ndb bug: there's no node in parameter 2 of this function!
local meta=minetest.get_meta(pos)
if meta then
- meta:set_string("infotext", attrans("ATC controller, unconfigured."))
- -- meta:set_string("formspec", atc.get_atc_controller_formspec(pos, meta))
- end
-end
-
--- formspec and callback:
-
-local function get_custom_state(pos)
- local node = minetest.get_node(pos)
- local meta = minetest.get_meta(pos)
- return {
- pos = pos,
- node_name_f = minetest.formspec_escape(node.name),
- mode = tonumber(meta:get_string("mode")) or 1,
- command = meta:get_string("command"),
- command_on = meta:get_string("command_on"),
- channel = meta:get_string("channel"),
- line = meta:get_string("line"),
- routingcode = meta:get_string("routingcode"),
- text_inside = meta:get_string("text_inside"),
- text_outside = meta:get_string("text_outside"),
- ars = meta:get_string("ars"),
- }
-end
-
-local function get_formspec(custom_state)
- local F = minetest.formspec_escape
- local formspec = {
- ch_core.formspec_header({
- formspec_version = 6,
- size = {12,12.75},
- auto_background = true,
- }),
- "style[command,command_on;font=mono]"..
- "item_image[0.25,0.25;1,1;"..custom_state.node_name_f.."]"..
- "label[1.4,0.85;řídicí obvod ATC]"..
- "button_exit[11,0.25;0.75,0.75;close;X]"..
- "textarea[0.25,4.25;11.5,0.75;text_inside;nový text uvnitř (5 ř.):;"..F(custom_state.text_inside).."]"..
- "textarea[0.25,5.5;11.5,0.75;text_outside;nový text venku (3 ř.):;"..F(custom_state.text_outside).."]"..
- "field[0.25,6.75;3,0.75;line;nová linka:;"..F(custom_state.line).."]"..
- "field[0.25,8;3,0.75;routingcode;nový směr. kód:;"..F(custom_state.routingcode).."]"..
- "textarea[0.25,9.25;3,2.25;ars;vlaky\\, pro které platí:;"..F(custom_state.ars).."]"..
- "label[3.5,6.55;nápověda:]"..
- "tablecolumns[text;text]"..
- "table[3.5,6.75;8.25,4.75;atc_help;S{0-20|M},Zpomalí/zrychlí na zadanou rychlost,SM,Zrychlí na maximální rychlost,"..
- "B{0-20},Zpomalí na zadanou rychlost,W,Počká na dosažení cílové rychlosti,D{0-...},Počká zadaný počet sekund,"..
- "R,Pokud vlak stojí\\, obrátí směr jízdy,OL,Otevře levé dveře,OR,Otevře pravé dveře,OC,Zavře všechny dveře,"..
- "K,Vyhodí cestující (ne strojvedoucí/ho),Cpl,Počkat na náraz do vagonu a připojit ho,"..
- "A0,Vypne ARS,A1,Zapne ARS,I+ {...}\\;,Vykonat\\, pokud jede vlak po směru šipky,"..
- "I- {...}\\;,Nevykonat (ř.obvod je jednosměrný),I<{0-20} {...}\\;,Vykonat\\, je-li rychlost menší než uvedená"..
- ",,(analogicky pro op. <=\\, >= a >);]",
- "button_exit[0.25,11.75;11.5,0.75;save;Uložit]"..
- "tooltip[line;Nová linka na odjezdu. Prázdné pole = zachovat stávající linku. Pro smazání linky zadejte znak -]"..
- "tooltip[routingcode;Nový směrový kód na odjezdu. Prázdné pole = zachovat stávající směrový kód. Pro smazání kódu vlaku zadejte znak -]"..
- "tooltip[text_inside;Nastavit text uvnitř vlaku. Prázdné pole = zachovat stávající. - = smazat]"..
- "tooltip[text_outside;Nastavit text vně vlaku. Prázdné pole = zachovat stávající. - = smazat]"..
- "tooltip[ars;Seznam podmínek\\, z nichž musí vlak splnit alespoň jednu\\, aby zde zastavil:\nLN {linka}\nRC {směrovací kód}\n"..
- "* = jakýkoliv vlak\n\\# komentář]",
- }
- if custom_state.mode < 3 then
- table.insert(formspec, "field[0.25,1.75;11.5,0.75;command;program:;"..F(custom_state.command).."]")
- if custom_state.mode == 2 then
- table.insert(formspec, "field[0.25,3;11.5,0.75;command_on;program (při signálu):;"..F(custom_state.command_on).."]")
- end
- else
- table.insert(formspec, "field[0.25,1.75;11.5,0.75;channel;"..attrans("Digiline channel")..";"..F(custom_state.channel).."]")
- end
- return table.concat(formspec)
-end
-
-local function limit_text(t, limit)
- local tbl = t:split("\n")
- if #tbl <= limit then
- return t
- else
- for i = #t, limit + 1, -1 do
- tbl[i] = nil
- end
- return table.concat(tbl, "\n")
- end
-end
-
-local function formspec_callback(custom_state, player, formname, fields)
- local pos = custom_state.pos
- if advtrains.is_protected(pos, player:get_player_name()) then
- minetest.record_protection_violation(pos, player:get_player_name())
- return
- end
-
- if fields.command ~= nil then custom_state.command = fields.command end
- if fields.command_on ~= nil then custom_state.command_on = fields.command_on end
- if fields.line ~= nil then custom_state.line = fields.line end
- if fields.routingcode ~= nil then custom_state.routingcode = fields.routingcode end
- if fields.text_inside ~= nil then custom_state.text_inside = fields.text_inside end
- if fields.text_outside ~= nil then custom_state.text_outside = fields.text_outside end
- if fields.ars ~= nil then custom_state.ars = fields.ars end
- if fields.channel ~= nil then custom_state.channel = fields.channel end
-
- local meta=minetest.get_meta(pos)
- if meta then
- if not fields.save then
- --[[--maybe only the dropdown changed
- if fields.mode then
- meta:set_string("mode", idxtrans[fields.mode])
- if fields.mode=="digiline" then
- meta:set_string("infotext", attrans("ATC controller, mode @1\nChannel: @2", fields.mode, meta:get_string("command")) )
- else
- meta:set_string("infotext", attrans("ATC controller, mode @1\nCommand: @2", fields.mode, meta:get_string("command")) )
- end
- meta:set_string("formspec", atc.get_atc_controller_formspec(pos, meta))
- end]]--
- return
- end
- if custom_state.ars == "" then custom_state.ars = "*" end
- --meta:set_string("mode", idxtrans[fields.mode])
- meta:set_string("command", custom_state.command)
- --meta:set_string("command_on", fields.command_on)
- meta:set_string("channel", custom_state.channel)
- meta:set_string("line", custom_state.line)
- meta:set_string("routingcode", custom_state.routingcode)
- meta:set_string("text_inside", limit_text(custom_state.text_inside, 5))
- meta:set_string("text_outside", limit_text(custom_state.text_outside, 3))
- meta:set_string("ars", custom_state.ars)
- --if fields.mode=="digiline" then
- -- meta:set_string("infotext", attrans("ATC controller, mode @1\nChannel: @2", fields.mode, meta:get_string("command")) )
- --else
- meta:set_string("infotext", attrans("ATC controller, mode @1\nCommand: @2", "-", meta:get_string("command")) )
- --end
- -- meta:set_string("formspec", atc.get_atc_controller_formspec(pos, meta))
-
- local pts=minetest.pos_to_string(pos)
- local _, conns=advtrains.get_rail_info_at(pos, advtrains.all_tracktypes)
- atc.controllers[pts]={command=fields.command}
- if #advtrains.occ.get_trains_at(pos) > 0 then
- atc.send_command(pos)
- end
+ meta:set_string("infotext", attrans("Unconfigured ATC controller"))
+ meta:set_string("formspec", atc.get_atc_controller_formspec(pos, meta))
end
end
@@ -254,40 +121,75 @@ advtrains.atc_function = function(def, preset, suffix, rotation)
local pts=minetest.pos_to_string(pos)
atc.controllers[pts]=nil
end,
- on_rightclick = function(pos, node, clicker, itemstack, pointed_thing)
- if advtrains.is_protected(pos, clicker:get_player_name()) then
- minetest.record_protection_violation(pos, clicker:get_player_name())
+ on_receive_fields = function(pos, formname, fields, player)
+ if advtrains.is_protected(pos, player:get_player_name()) then
+ minetest.record_protection_violation(pos, player:get_player_name())
return
end
- local custom_state = get_custom_state(pos)
- ch_core.show_formspec(clicker, "advtrains:atc", get_formspec(custom_state), formspec_callback, custom_state, {})
- end,
- advtrains = {
- on_train_enter = function(pos, train_id, train, index)
- if train.path_cn[index] ~= 1 then
- return -- opposite direction
- end
- local meta = minetest.get_meta(pos)
- if advtrains.interlocking ~= nil then
- local ars = meta:get_string("ars")
- ars = advtrains.interlocking.text_to_ars(ars) or {default = true}
- if not (ars.default or advtrains.interlocking.ars_check_rule_match(ars, train)) then
- return -- ARS does not match
- end
+ local meta=minetest.get_meta(pos)
+ if meta then
+ if not fields.save then
+ --[[--maybe only the dropdown changed
+ if fields.mode then
+ meta:set_string("mode", idxtrans[fields.mode])
+ if fields.mode=="digiline" then
+ meta:set_string("infotext", attrans("ATC controller, mode @1\nChannel: @2", fields.mode, meta:get_string("command")) )
+ else
+ meta:set_string("infotext", attrans("ATC controller, mode @1\nCommand: @2", fields.mode, meta:get_string("command")) )
+ end
+ meta:set_string("formspec", atc.get_atc_controller_formspec(pos, meta))
+ end]]--
+ return
end
- local function apply_change(old_value, change)
- if change == "-" then return nil elseif change == "" then return old_value else return change end
+ --meta:set_string("mode", idxtrans[fields.mode])
+ meta:set_string("command", fields.command)
+ --meta:set_string("command_on", fields.command_on)
+ meta:set_string("channel", fields.channel)
+ --if fields.mode=="digiline" then
+ -- meta:set_string("infotext", attrans("ATC controller, mode @1\nChannel: @2", fields.mode, meta:get_string("command")) )
+ --else
+ meta:set_string("infotext", attrans("ATC controller, mode @1\nCommand: @2", "-", meta:get_string("command")) )
+ --end
+ meta:set_string("formspec", atc.get_atc_controller_formspec(pos, meta))
+
+ local pts=minetest.pos_to_string(pos)
+ local _, conns=advtrains.get_rail_info_at(pos, advtrains.all_tracktypes)
+ atc.controllers[pts]={command=fields.command}
+ if #advtrains.occ.get_trains_at(pos) > 0 then
+ atc.send_command(pos)
end
- train.line = apply_change(train.line, meta:get_string("line"))
- train.routingcode = apply_change(train.routingcode, meta:get_string("routingcode"))
- train.text_inside = apply_change(train.text_inside, meta:get_string("text_inside"))
- train.text_outside = apply_change(train.text_outside, meta:get_string("text_outside"))
+ end
+ end,
+ advtrains = {
+ on_train_enter = function(pos, train_id)
atc.send_command(pos, train_id)
end,
},
}
end
+function atc.get_atc_controller_formspec(pos, meta)
+ local mode=tonumber(meta:get_string("mode")) or 1
+ local command=meta:get_string("command")
+ local command_on=meta:get_string("command_on")
+ local channel=meta:get_string("channel")
+ local formspec="size[8,6]"
+ -- "dropdown[0,0;3;mode;static,mesecon,digiline;"..mode.."]"
+ if mode<3 then
+ formspec=formspec
+ .."style[command;font=mono]"
+ .."field[0.8,1.5;7,1;command;"..attrans("Command")..";"..minetest.formspec_escape(command).."]"
+ if tonumber(mode)==2 then
+ formspec=formspec
+ .."style[command_on;font=mono]"
+ .."field[0.8,3;7,1;command_on;"..attrans("Command (on)")..";"..minetest.formspec_escape(command_on).."]"
+ end
+ else
+ formspec=formspec.."field[0.8,1.5;7,1;channel;"..attrans("Digiline channel")..";"..minetest.formspec_escape(channel).."]"
+ end
+ return formspec.."button_exit[0.5,4.5;7,1;save;"..attrans("Save").."]"
+end
+
--from trainlogic.lua train step
local matchptn={
["SM"]=function(id, train)
@@ -332,7 +234,7 @@ local matchptn={
advtrains.train_ensure_init(id, train)
-- no one minds if this failed... this shouldn't even be called without train being initialized...
else
- atwarn(sid(id), attrans("ATC Reverse command warning: didn't reverse train, train moving!"))
+ atwarn(sid(id), attrans("ATC Reverse command warning: didn't reverse train, train moving."))
end
return 1
end,
@@ -344,11 +246,11 @@ local matchptn={
end,
["K"] = function(id, train)
if train.door_open == 0 then
- atwarn(sid(id), attrans("ATC Kick command warning: Doors closed"))
+ atwarn(sid(id), attrans("ATC Kick command warning: doors are closed."))
return 1
end
if train.velocity > 0 then
- atwarn(sid(id), attrans("ATC Kick command warning: Train moving"))
+ atwarn(sid(id), attrans("ATC Kick command warning: train moving."))
return 1
end
local tp = train.trainparts
@@ -377,6 +279,10 @@ local matchptn={
train.atc_wait_autocouple=true
return 3
end,
+ ["G"]=function(id, train)
+ train.atc_wait_signal=true
+ return 1
+ end,
}
eval_conditional = function(command, arrow, speed)
@@ -474,7 +380,8 @@ function atc.execute_atc_command(id, train)
train.atc_command=string.sub(command, patlen+1)
if train.atc_delay<=0
and not train.atc_wait_finish
- and not train.atc_wait_autocouple then
+ and not train.atc_wait_autocouple
+ and not train.atc_wait_signal then
--continue (recursive, cmds shouldn't get too long, and it's a end-recursion.)
atc.execute_atc_command(id, train)
end
diff --git a/advtrains/copytool.lua b/advtrains/copytool.lua
index 0c1cdfe..c63551e 100644
--- a/advtrains/copytool.lua
+++ b/advtrains/copytool.lua
@@ -21,12 +21,12 @@ minetest.register_tool("advtrains:copytool", {
local node=minetest.get_node_or_nil(pointed_thing.under)
if not node then atprint("[advtrains]Ignore at placer position") return itemstack end
local nodename=node.name
- if(not advtrains.is_track_and_drives_on(nodename, {default=true})) then
+ if(not advtrains.is_track(nodename)) then
atprint("no track here, not placing.")
return itemstack
end
if not minetest.check_player_privs(placer, {train_operator = true }) then
- minetest.chat_send_player(pname, "You don't have the train_operator privilege.")
+ minetest.chat_send_player(pname, S("You do not have the @1 privilege.", "train_operator"))
return itemstack
end
if not minetest.check_player_privs(placer, {train_admin = true }) and minetest.is_protected(pointed_thing.under, placer:get_player_name()) then
@@ -38,7 +38,7 @@ minetest.register_tool("advtrains:copytool", {
local prevpos = advtrains.get_adjacent_rail(pointed_thing.under, tconns, plconnid, {default=true})
if not prevpos then
- minetest.chat_send_player(pname, "The track you are trying to place the wagon on is not long enough!")
+ minetest.chat_send_player(pname, attrans("The track you are trying to place the wagon on is not long enough."))
return
end
@@ -49,12 +49,12 @@ minetest.register_tool("advtrains:copytool", {
end
local clipboard = meta:get_string("clipboard")
if (clipboard == "") then
- minetest.chat_send_player(pname, "The clipboard is empty.");
+ minetest.chat_send_player(pname, attrans("The clipboard is empty."));
return
end
clipboard = minetest.deserialize(clipboard)
if (clipboard.wagons == nil) then
- minetest.chat_send_player(pname, "The clipboard is empty.");
+ minetest.chat_send_player(pname, attrans("The clipboard is empty."));
return
end
@@ -71,7 +71,7 @@ minetest.register_tool("advtrains:copytool", {
local train = advtrains.trains[id]
train.off_track = train.end_index<train.path_trk_b
if (train.off_track) then
- minetest.chat_send_player(pname, "Back of train would end up off track, cancelling.")
+ minetest.chat_send_player(pname, attrans("Back of train would end up off track, cancelling."))
advtrains.remove_train(id)
return
end
@@ -89,19 +89,19 @@ minetest.register_tool("advtrains:copytool", {
local le = pointed_thing.ref:get_luaentity()
if (le == nil) then
- minetest.chat_send_player(user:get_player_name(), "No such lua entity!")
+ minetest.chat_send_player(user:get_player_name(), attrans("No such lua entity."))
return
end
local wagon = advtrains.wagons[le.id]
if (not (le.id and advtrains.wagons[le.id])) then
- minetest.chat_send_player(user:get_player_name(), string.format("No such wagon: %s", le.id))
+ minetest.chat_send_player(user:get_player_name(), attrans("No such wagon: @1.", le.id))
return
end
local train = advtrains.trains[wagon.train_id]
if (not train) then
- minetest.chat_send_player(user:get_player_name(), string.format("No such train: %s", wagon.train_id))
+ minetest.chat_send_player(user:get_player_name(), attrans("No such train: @1.", wagon.train_id))
return
end
@@ -177,7 +177,7 @@ minetest.register_tool("advtrains:copytool", {
return
end
meta:set_string("clipboard", minetest.serialize(clipboard))
- minetest.chat_send_player(user:get_player_name(), attrans("Train copied!"))
+ minetest.chat_send_player(user:get_player_name(), attrans("Train copied."))
return itemstack
end
})
diff --git a/advtrains/couple.lua b/advtrains/couple.lua
index a4355ca..a561b06 100644
--- a/advtrains/couple.lua
+++ b/advtrains/couple.lua
@@ -28,6 +28,20 @@ end
advtrains.register_coupler_type("chain", attrans("Buffer and Chain Coupler"))
advtrains.register_coupler_type("scharfenberg", attrans("Scharfenberg Coupler"))
+for _, name in pairs {"couple", "decouple"} do
+ local t = {}
+ local function reg(f)
+ table.insert(t, f)
+ end
+ local function cb(...)
+ for _, f in ipairs(t) do
+ f(...)
+ end
+ end
+ advtrains["te_registered_on_" .. name] = t
+ advtrains["te_register_on_" .. name] = reg
+ advtrains["te_run_callbacks_on_" .. name] = cb
+end
local function create_couple_entity(pos, train1, t1_is_front, train2, t2_is_front)
local id1 = train1.id
@@ -79,8 +93,9 @@ function advtrains.train_check_couples(train)
end
if not train.cpl_front then
-- recheck front couple
- local front_trains, pos = advtrains.occ.get_occupations(train, atround(train.index) + CPL_CHK_DST)
+ local pos = advtrains.path_get(train, atround(train.index) + CPL_CHK_DST)
if advtrains.is_node_loaded(pos) then -- if the position is loaded...
+ local front_trains = advtrains.occ.reverse_lookup_sel(pos, "in_train")
for tid, idx in pairs(front_trains) do
local other_train = advtrains.trains[tid]
if not advtrains.train_ensure_init(tid, other_train) then
@@ -109,8 +124,9 @@ function advtrains.train_check_couples(train)
end
if not train.cpl_back then
-- recheck back couple
- local back_trains, pos = advtrains.occ.get_occupations(train, atround(train.end_index) - CPL_CHK_DST)
+ local pos = advtrains.path_get(train, atround(train.end_index) - CPL_CHK_DST)
if advtrains.is_node_loaded(pos) then -- if the position is loaded...
+ local back_trains = advtrains.occ.reverse_lookup_sel(pos, "in_train")
for tid, idx in pairs(back_trains) do
local other_train = advtrains.trains[tid]
if not advtrains.train_ensure_init(tid, other_train) then
@@ -182,8 +198,8 @@ end
function advtrains.safe_couple_trains(train1, t1_is_front, train2, t2_is_front, pname)
if pname and not minetest.check_player_privs(pname, "train_operator") then
- minetest.chat_send_player(pname, "Missing train_operator privilege")
- return false
+ minetest.chat_send_player(pname, S("You are not allowed to couple trains without the train_operator privilege."))
+ return false
end
local wck_t1, wck_t2
@@ -225,6 +241,15 @@ function advtrains.couple_trains(init_train, invert_init_train, stat_train, stat
local stp = stat_train.trainparts
local stat_wagoncnt = #stp
local stat_trainlen = stat_train.trainlen -- save the train length of stat train, to be added to index
+
+ -- sanity check, prevent coupling if train would be longer than 20 after coupling
+ local tot_len = init_wagoncnt + stat_wagoncnt
+ if tot_len > advtrains.TRAIN_MAX_WAGONS then
+ atwarn("Cannot couple",stat_train.id,"and",init_train.id,"- train would have length",tot_len,"which is above the limit of",advtrains.TRAIN_MAX_WAGONS)
+ return
+ end
+
+ advtrains.te_run_callbacks_on_couple(init_train, stat_train)
if stat_train_opposite then
-- insert wagons in inverse order and set their wagon_flipped state
@@ -310,7 +335,7 @@ function advtrains.check_matching_coupler_types(t1, t1_front, t2, t2_front)
--atdebug("CMCT: t1",t1_cplt,"t2",t2_cplt,"")
-- if at least one of the trains has no couplers table, it always couples (fallback behavior and mode for universal shunters)
- if not t1_cplt or not t2_cplt then
+ if minetest.settings:get_bool("advtrains_universal_couplers", false) or not t1_cplt or not t2_cplt then
return true
end
@@ -326,11 +351,11 @@ function advtrains.check_matching_coupler_types(t1, t1_front, t2, t2_front)
for typ,_ in pairs(t1_cplt) do
table.insert(t1_cplhr, advtrains.coupler_types[typ] or typ)
end
- if #t1_cplhr==0 then t1_cplhr[1]=attrans("<none>") end
+ if #t1_cplhr==0 then t1_cplhr[1]=attrans("<No coupler>") end
for typ,_ in pairs(t2_cplt) do
table.insert(t2_cplhr, advtrains.coupler_types[typ] or typ)
end
- if #t2_cplhr==0 then t2_cplhr[1]=attrans("<none>") end
+ if #t2_cplhr==0 then t2_cplhr[1]=attrans("<No coupler>") end
return false, attrans("Can not couple: The couplers of the trains do not match (@1 and @2).", table.concat(t1_cplhr, ","), table.concat(t2_cplhr, ","))
end
@@ -430,7 +455,7 @@ minetest.register_entity("advtrains:discouple", {
self.object:remove()
return
end
- --getyaw seems to be a reliable method to check if an object is loaded...if it returns nil, it is not.
+ --get_yaw seems to be a reliable method to check if an object is loaded...if it returns nil, it is not.
if not self.wagon.object:get_yaw() then
self.object:remove()
return
diff --git a/advtrains/craft_items.lua b/advtrains/craft_items.lua
index 0e693eb..1188b64 100644
--- a/advtrains/craft_items.lua
+++ b/advtrains/craft_items.lua
@@ -6,7 +6,7 @@ core.register_craftitem("advtrains:boiler", {
core.register_craftitem("advtrains:driver_cab", {
- description = attrans("driver's cab"),
+ description = attrans("Driver's cab"),
inventory_image = "advtrains_driver_cab.png",
})
diff --git a/advtrains/crafting.lua b/advtrains/crafting.lua
index f2bfaff..b6f01a6 100644
--- a/advtrains/crafting.lua
+++ b/advtrains/crafting.lua
@@ -22,6 +22,14 @@ minetest.register_craft({
})
--Wallmounted Signal
minetest.register_craft({
+ output = 'advtrains:signal_wall_l_off 2',
+ recipe = {
+ {'default:steel_ingot', 'default:steel_ingot', 'dye:red'},
+ {'', 'default:steel_ingot', ''},
+ {'default:steel_ingot', 'default:steel_ingot', 'dye:dark_green'},
+ },
+})
+minetest.register_craft({
output = 'advtrains:signal_wall_r_off 2',
recipe = {
{'dye:red', 'default:steel_ingot', 'default:steel_ingot'},
@@ -29,6 +37,15 @@ minetest.register_craft({
{'dye:dark_green', 'default:steel_ingot', 'default:steel_ingot'},
},
})
+minetest.register_craft({
+ output = 'advtrains:signal_wall_t_off 2',
+ recipe = {
+ {'default:steel_ingot', '', 'default:steel_ingot'},
+ {'default:steel_ingot', 'default:steel_ingot', 'default:steel_ingot'},
+ {'dye:dark_green', '', 'dye:red'},
+ },
+})
+
--Wallmounted Signals can be converted into every orientation by shapeless crafting
minetest.register_craft({
diff --git a/advtrains/debugitems.lua b/advtrains/debugitems.lua
index e672308..2236cba 100644
--- a/advtrains/debugitems.lua
+++ b/advtrains/debugitems.lua
@@ -51,3 +51,89 @@ minetest.register_chatcommand("atyaw",
end
end,
})
+
+minetest.register_tool("advtrains:wagonpos_tester",
+{
+ description = "Wagon position tester",
+ groups = {cracky=1}, -- key=name, value=rating; rating=1..3.
+ inventory_image = "drwho_screwdriver.png",
+ wield_image = "drwho_screwdriver.png",
+ stack_max = 1,
+ range = 7.0,
+
+ on_place = function(itemstack, placer, pointed_thing)
+
+ end,
+ --[[
+ ^ Shall place item and return the leftover itemstack
+ ^ default: minetest.item_place ]]
+ on_use = function(itemstack, user, pointed_thing)
+ if pointed_thing.type=="node" then
+ local pos = pointed_thing.under
+ local trains = advtrains.occ.get_trains_at(pos)
+ for train_id, index in pairs(trains) do
+ local wagon_num, wagon_id, wagon_data, offset_from_center = advtrains.get_wagon_at_index(train_id, index)
+ if wagon_num then
+ atdebug(wagon_num, wagon_id, offset_from_center)
+ end
+ end
+ end
+ end,
+}
+)
+
+
+local function trackitest(initial_pos, initial_connid)
+ local ti, pos, connid, ok
+ ti = advtrains.get_track_iterator(initial_pos, initial_connid, 500, true)
+ atdebug("Starting at pos:",initial_pos,initial_connid)
+ while ti:has_next_branch() do
+ pos, connid = ti:next_branch() -- in first iteration, this will be the node at initial_pos. In subsequent iterations this will be the switch node from which we are branching off
+ atdebug("Next Branch:",pos, connid)
+ ok = true
+ while ok do
+ ok, pos, connid = ti:next_track()
+ atdebug("Next Track:", ok, pos, connid)
+ end
+ end
+ atdebug("End of traverse. Visited: ",table.concat(ti.visited, ","))
+end
+
+minetest.register_tool("advtrains:trackitest",
+{
+ description = "Track Iterator Tester (leftclick conn 1, rightclick conn 2)",
+ groups = {cracky=1}, -- key=name, value=rating; rating=1..3.
+ inventory_image = "advtrains_track_swlcr_45.png",
+ wield_image = "advtrains_track_swlcr_45.png",
+ stack_max = 1,
+ range = 7.0,
+
+ on_place = function(itemstack, placer, pointed_thing)
+ if pointed_thing.type ~= "node" then return end
+ trackitest(pointed_thing.under, 2)
+ end,
+ on_use = function(itemstack, user, pointed_thing)
+ if pointed_thing.type ~= "node" then return end
+ trackitest(pointed_thing.under, 1)
+ end,
+}
+)
+
+minetest.register_chatcommand("at_trackdef_audit",
+ {
+ params = "",
+ description = "Performs an audit of all track definitions currently loaded and checks for potential problems",
+ func = function(name, param)
+ for name, ndef in pairs(minetest.registered_nodes) do
+ --TODO finish this!
+ if ndef.at_conns then
+ -- check if conn_map is there and if it has enough entries
+ if #ndef.at_conns > 2 then
+ if #ndef.at_conn_map < #ndef.at_conns then
+ atwarn("AUDIT: Node",name,"- Not enough connmap entries! Check ndef:",ndef)
+ end
+ end
+ end
+ end
+ end,
+})
diff --git a/advtrains/formspec.lua b/advtrains/formspec.lua
new file mode 100644
index 0000000..8894354
--- /dev/null
+++ b/advtrains/formspec.lua
@@ -0,0 +1,111 @@
+local sformat = string.format
+local fsescape = minetest.formspec_escape
+
+local function make_list(entries)
+ local t = {}
+ for k, v in ipairs(entries) do
+ t[k] = fsescape(v)
+ end
+ return table.concat(t, ",")
+end
+
+local function S_wrapper(f, i0)
+ return function(...)
+ local args = {...}
+ args[i0] = attrans(unpack(args,i0))
+ return f(unpack(args,1,i0))
+ end
+end
+
+local function f_button(x, y, w, id, text)
+ return sformat("button[%f,%f;%f,0.75;%s;%s]", x, y, w, id, text)
+end
+
+local function f_checkbox(x, y, name, selected, label)
+ return sformat("checkbox[%f,%f;%s;%s;%s]", x, y+0.25, name, label, selected and "true" or "false")
+end
+
+local function f_button_exit(x, y, w, id, text)
+ return sformat("button_exit[%f,%f;%f,0.75;%s;%s]", x, y, w, id, text)
+end
+
+local function f_dropdown(x, y, w, id, entries, sel, indexed)
+ return sformat("dropdown[%f,%f;%f,0.75;%s;%s;%d%s]",
+ x, y, w, id, make_list(entries),
+ sel or 1,
+ indexed and ";true" or "")
+end
+
+local function f_image_button(x, y, w, h, texture, id, label, noclip, drawborder, pressed)
+ local st = {string.format("%f,%f;%f,%f;%s;%s;%s", x, y, w, h, fsescape(texture), fsescape(id), fsescape(label))}
+ if pressed then
+ st[#st+1] = tostring(noclip or false)
+ st[#st+1] = tostring(drawborder or false)
+ st[#st+1] = fsescape(pressed)
+ end
+ return sformat("image_button[%s]", table.concat(st, ";"))
+end
+
+local function f_image_button_exit(x, y, w, h, texture, id, label)
+ local st = {string.format("%f,%f;%f,%f;%s;%s;%s", x, y, w, h, fsescape(texture), fsescape(id), fsescape(label))}
+ return sformat("image_button_exit[%s]", table.concat(st, ";"))
+end
+
+local function f_label(x, y, text)
+ return sformat("label[%f,%f;%s]", x, y+0.25, fsescape(text))
+end
+
+local function f_field_aux(x, y, w, id, default)
+ return sformat("field[%f,%f;%f,0.75;%s;;%s]", x, y, w, id, default)
+end
+
+local function f_field(x, y, w, id, label, default)
+ return f_label(x, y-0.5, label) .. f_field_aux(x, y, w, id, default)
+end
+
+local function f_tabheader(x, y, w, h, id, entries, sel, transparent, border)
+ local st = {string.format("%f,%f",x, y)}
+ if h then
+ if w then
+ st[#st+1] = string.format("%f,%f", w, h)
+ else
+ st[#st+1] = tostring(h)
+ end
+ end
+ st[#st+1] = tostring(id)
+ st[#st+1] = make_list(entries)
+ st[#st+1] = tostring(sel)
+ if transparent ~= nil then
+ st[#st+1] = tostring(transparent)
+ if border ~= nil then
+ st[#st+1] = tostring(border)
+ end
+ end
+ return string.format("tabheader[%s]", table.concat(st, ";"))
+end
+
+local function f_textlist(x, y, w, h, id, entries, sel, transparent)
+ local st = {string.format("%f,%f;%f,%f;%s;%s", x, y, w, h, id, make_list(entries))}
+ if sel then
+ st[#st+1] = tostring(sel)
+ st[#st+1] = tostring(transparent or false)
+ end
+ return string.format("textlist[%s]", table.concat(st, ";"))
+end
+
+return {
+ button = f_button,
+ S_button = S_wrapper(f_button, 5),
+ checkbox = f_checkbox,
+ S_checkbox = S_wrapper(f_checkbox, 5),
+ button_exit = f_button_exit,
+ S_button_exit = S_wrapper(f_button_exit, 5),
+ dropdown = f_dropdown,
+ field = f_field,
+ image_button = f_image_button,
+ image_button_exit = f_image_button_exit,
+ label = f_label,
+ S_label = S_wrapper(f_label, 3),
+ tabheader = f_tabheader,
+ textlist = f_textlist,
+}
diff --git a/advtrains/helpers.lua b/advtrains/helpers.lua
index 33950ec..1ed5a7e 100644
--- a/advtrains/helpers.lua
+++ b/advtrains/helpers.lua
@@ -293,18 +293,19 @@ function advtrains.conn_matches_to(conn, other_conns)
end
-- Going from the rail at pos (does not need to be rounded) along connection with id conn_idx, if there is a matching rail, return it and the matching connid
--- returns: <adjacent pos>, <conn index of adjacent>, <my conn index>, <railheight of adjacent>
+-- returns: <adjacent pos>, <conn index of adjacent>, <my conn index>, <railheight of adjacent>, (adjacent conns table), (adjacent connmap table)
-- parameter this_conns_p is connection table of this rail and is optional, is determined by get_rail_info_at if not provided.
-function advtrains.get_adjacent_rail(this_posnr, this_conns_p, conn_idx, drives_on)
+function advtrains.get_adjacent_rail(this_posnr, this_conns_p, conn_idx)
local this_pos = advtrains.round_vector_floor_y(this_posnr)
local this_conns = this_conns_p
+ local _
if not this_conns then
_, this_conns = advtrains.get_rail_info_at(this_pos)
end
if not conn_idx then
for coni, _ in ipairs(this_conns) do
- local adj_pos, adj_conn_idx, _, nry, nco = advtrains.get_adjacent_rail(this_pos, this_conns, coni)
- if adj_pos then return adj_pos,adj_conn_idx,coni,nry, nco end
+ local adj_pos, adj_conn_idx, _, nry, nco, ncm = advtrains.get_adjacent_rail(this_pos, this_conns, coni)
+ if adj_pos then return adj_pos,adj_conn_idx,coni,nry, nco, ncm end
end
return nil
end
@@ -318,34 +319,46 @@ function advtrains.get_adjacent_rail(this_posnr, this_conns_p, conn_idx, drives_
adj_pos.y = adj_pos.y + 1
end
- local nextnode_ok, nextconns, nextrail_y=advtrains.get_rail_info_at(adj_pos, drives_on)
+ local nextnode_ok, nextconns, nextrail_y, nextconnmap=advtrains.get_rail_info_at(adj_pos)
if not nextnode_ok then
adj_pos.y = adj_pos.y - 1
conn_y = conn_y + 1
- nextnode_ok, nextconns, nextrail_y=advtrains.get_rail_info_at(adj_pos, drives_on)
+ nextnode_ok, nextconns, nextrail_y, nextconnmap=advtrains.get_rail_info_at(adj_pos)
if not nextnode_ok then
return nil
end
end
local adj_connid = advtrains.conn_matches_to({c=conn.c, y=conn_y}, nextconns)
if adj_connid then
- return adj_pos, adj_connid, conn_idx, nextrail_y, nextconns
+ return adj_pos, adj_connid, conn_idx, nextrail_y, nextconns, nextconnmap
end
return nil
end
-- when a train enters a rail on connid 'conn', which connid will it go out?
--- nconns: number of connections in connection table:
--- 2 = straight rail; 3 = turnout, 4 = crossing, 5 = three-way turnout (5th entry is a stub)
+-- Since 2.5: This mapping is contained in the conn_map table in the node definition!
-- returns: connid_out
-local connlku={[2]={2,1}, [3]={2,1,1}, [4]={2,1,4,3}, [5]={2,1,1,1}}
-function advtrains.get_matching_conn(conn, nconns)
- return connlku[nconns][conn]
+function advtrains.get_matching_conn(conn, conn_map)
+ if tonumber(conn_map) then
+ error("Legacy call to get_matching_conn! Instead of nconns, conn_map needs to be provided!")
+ end
+ if not conn_map then
+ --OK for two-conn rails, just return the other
+ if conn==1 then return 2 end
+ if conn==2 then return 1 end
+ error("get_matching_conn: For connid >=3, conn_map must not be nil!")
+ end
+ local cout = conn_map[conn]
+ if not cout then
+ error("get_matching_conn: Connid "..conn.." not found in conn_map which is "..atdump(conn_map))
+ end
+ return cout
end
-function advtrains.random_id()
+function advtrains.random_id(lenp)
local idst=""
- for i=0,5 do
+ local len = lenp or 6
+ for i=1,len do
idst=idst..(math.random(0,9))
end
return idst
@@ -524,3 +537,149 @@ function advtrains.yaw_equals(yaw1, yaw2)
return yaw1 == yaw2
end
end
+
+
+-- TrackIterator interface --
+
+-- Metatable:
+local trackiter_mt = {
+ -- Internal State:
+ -- branches: A list of {pos, connid, limit} for where to restart
+ -- pos: The *next* position that the track iterator will return
+ -- bconnid: The connid of the connection of the rail at pos that points backward
+ -- tconns: The connections of the rail at pos
+ -- limit: the current limit
+ -- visited: a key-boolean table of already visited rails
+
+ -- get whether there are still unprocessed branches
+ has_next_branch = function(self)
+ return #self.branches > 0
+ end,
+ -- go to the next unprocessed branch
+ -- returns track_pos, track_connid of the switch/crossing node where the track branches off
+ next_branch = function(self)
+ local br = table.remove(self.branches, 1)
+ -- Advance internal state
+ local adj_pos, adj_connid, _, _, adj_conns, adj_connmap = advtrains.get_adjacent_rail(br.pos, nil, br.connid)
+ self.pos = adj_pos
+ self.bconnid = adj_connid
+ self.tconns = adj_conns
+ self.tconnmap = adj_connmap
+ self.limit = br.limit - 1
+ self.visited[advtrains.encode_pos(br.pos)] = true
+ self.last_track_already_visited = false
+ return br.pos, br.connid
+ end,
+ -- get the next track along the current branch,
+ -- potentially adding branching tracks to the unprocessed branches list
+ -- returns track_pos, track_connid, track_backwards_connid
+ -- On error, returns nil, reason; reason is one of "track_end", "limit_hit", "already_visited"
+ next_track = function(self)
+ if self.last_track_already_visited then
+ -- see comment below
+ return nil, "already_visited"
+ end
+ local pos = self.pos
+ if not pos then
+ -- last run found track end. Return false
+ return false, "track_end"
+ end
+ -- if limit hit, return nil to signal this
+ if self.limit <= 0 then
+ return nil, "limit_hit"
+ end
+ -- select next conn (main conn to follow is the associated connection)
+ local old_bconnid = self.bconnid
+ local mconnid = advtrains.get_matching_conn(self.bconnid, self.tconnmap)
+ if self.visited[advtrains.encode_pos(pos)] then
+ -- node was already seen
+ -- Due to special requirements for the track section updater, return this first already visited track once
+ -- but do not process any further rails on this branch
+ -- The next call will then throw already_visited error
+ self.last_track_already_visited = true
+ return pos, mconnid, old_bconnid
+ end
+ -- If there are more connections, add these to branches
+ for nconnid,_ in ipairs(self.tconns) do
+ if nconnid~=mconnid and nconnid~=self.bconnid then
+ table.insert(self.branches, {pos = self.pos, connid = nconnid, limit=self.limit})
+ end
+ end
+ -- Advance internal state
+ local adj_pos, adj_connid, _, _, adj_conns, adj_connmap = advtrains.get_adjacent_rail(pos, self.tconns, mconnid)
+ self.pos = adj_pos
+ self.bconnid = adj_connid
+ self.tconns = adj_conns
+ self.tconnmap = adj_connmap
+ self.limit = self.limit - 1
+ self.visited[advtrains.encode_pos(pos)] = true
+ self.last_track_already_visited = false
+ return pos, mconnid, old_bconnid
+ end,
+
+ add_branch = function(self, pos, connid)
+ table.insert(self.branches, {pos = pos, connid = connid, limit=self.limit})
+ end,
+
+ is_visited = function(self, pos)
+ return self.visited[advtrains.encode_pos(pos)]
+ end,
+}
+
+-- Returns a new TrackIterator object
+
+-- Parameters:
+-- initial_pos: the initial track position of the track iterator
+-- initial_connid: the connection index in which to traverse. If nil, adds a "branch" for every connection of the track (traverse in all directions)
+-- limit: maximum distance from the start point after which the traverser stops
+-- follow_all: NOT IMPLEMENTED (supposed: if true, follows all branches at multi-connection tracks, even the ones pointing backwards or the crossing track on crossings. If false, follows only switches in driving direction.)
+
+-- Functions of the returned TrackIterator can be called via the Lua : notation, such as ti:next_track()
+-- If only the main track needs to be followed, use only the ti:next_track() function and do not call ti:next_branch().
+function advtrains.get_track_iterator(initial_pos, initial_connid, limit, follow_all)
+ local ti = {
+ visited = {}
+ }
+ if initial_connid then
+ ti.branches = { {pos = initial_pos, connid = initial_connid, limit=limit} }
+ else
+ -- get track info here
+ local node_ok, conns, rail_y=advtrains.get_rail_info_at(initial_pos)
+ assert(node_ok, "get_track_iterator called with non-track node!")
+ ti.branches = {}
+ for coni, _ in pairs(conns) do
+ table.insert(ti.branches, {pos = initial_pos, connid = coni, limit=limit})
+ end
+ end
+ ti.limit = limit -- safeguard if someone adds a branch before calling anything
+ setmetatable(ti, {__index=trackiter_mt})
+ return ti
+end
+
+--[[
+Example TrackIterator usage structure:
+
+local ti, pos, connid, ok
+ti = advtrains.get_track_iterator(initial_pos, initial_connid, 500, true)
+while ti:has_next_branch() do
+ pos, connid = ti:next_branch() -- in first iteration, this will be the node at initial_pos. In subsequent iterations this will be the switch node from which we are branching off
+ repeat
+ <do something with the track>
+ if <track satisfies an abort condition> then break end --for example, when traversing should stop at TCBs this can check if there is a tcb here
+ pos, connid = ti:next_track()
+ until not pos -- this stops the loop when either the track end is reached or the limit is hit
+ -- while loop continues with the next branch ( diverging branch of one of the switches/crossings) until no more are left
+end
+
+Example for walking only a single track (without branching):
+
+local ti, pos, connid, ok
+ti = advtrains.get_track_iterator(initial_pos, initial_connid, 500, true)
+
+pos, connid = ti:next_branch() -- this always needs to be done at least one time, and gets the track at initial_pos
+repeat
+ <do something with the track>
+ if <track satisfies an abort condition> then break end --for example, when traversing should stop at TCBs this can check if there is a tcb here
+ ok, pos, connid = ti:next_track()
+until not ok -- this stops the loop when either the track end is reached or the limit is hit
+]]
diff --git a/advtrains/init.lua b/advtrains/init.lua
index 688f2a5..f602c44 100644
--- a/advtrains/init.lua
+++ b/advtrains/init.lua
@@ -27,6 +27,9 @@ local has_luaautomation = minetest.get_modpath("advtrains_luaautomation")
-- There is no need to support 0.4.x anymore given that the compatitability with it is already broken by 1bb1d825f46af3562554c12fba35a31b9f7973ff
attrans = minetest.get_translator ("advtrains")
+function attrans_formspec(...)
+ return minetest.formspec_escape(attrans(...))
+end
--advtrains
advtrains = {trains={}, player_to_train_mapping={}}
@@ -51,6 +54,9 @@ advtrains.IGNORE_WORLD = false
local NO_SAVE = false
-- Do not save any data to advtrains save files
+advtrains.TRAIN_MAX_WAGONS = 20
+-- Limit on the maximum number of wagons that may be in a train
+
-- ==========================================================================
-- Use a global slowdown factor to slow down train movements. Now a setting
@@ -201,20 +207,27 @@ advtrains.meseconrules =
advtrains.fpath=minetest.get_worldpath().."/advtrains"
+advtrains.poconvert = dofile(advtrains.modpath.."/poconvert.lua")
+advtrains.poconvert.from_flat("advtrains")
+attrans = minetest.get_translator("advtrains")
+
advtrains.speed = dofile(advtrains.modpath.."/speed.lua")
+advtrains.formspec = dofile(advtrains.modpath.."/formspec.lua")
+advtrains.texture = dofile(advtrains.modpath.."/texture.lua")
dofile(advtrains.modpath.."/path.lua")
dofile(advtrains.modpath.."/trainlogic.lua")
dofile(advtrains.modpath.."/trainhud.lua")
dofile(advtrains.modpath.."/trackplacer.lua")
dofile(advtrains.modpath.."/copytool.lua")
+dofile(advtrains.modpath.."/wagonprop_tool.lua")
dofile(advtrains.modpath.."/tracks.lua")
+dofile(advtrains.modpath.."/track_reg_helper.lua")
dofile(advtrains.modpath.."/occupation.lua")
dofile(advtrains.modpath.."/atc.lua")
dofile(advtrains.modpath.."/wagons.lua")
dofile(advtrains.modpath.."/protection.lua")
-dofile(advtrains.modpath.."/trackdb_legacy.lua")
dofile(advtrains.modpath.."/nodedb.lua")
dofile(advtrains.modpath.."/couple.lua")
@@ -232,6 +245,9 @@ end
dofile(advtrains.modpath.."/lzb.lua")
+if minetest.settings:get_bool("advtrains_register_debugitems") then
+ dofile(advtrains.modpath.."/debugitems.lua")
+end
--load/save
@@ -414,6 +430,16 @@ function advtrains.load_version_4()
if il_save then
advtrains.interlocking.db.load(il_save)
end
+
+ -- TODO 2.5.0 backwards compatibility fallback: Store the pre-v2.5.0 save file so that it can be reverted to if needed
+ local fallback_file = advtrains.fpath.."_interlocking.ls.pre250"
+ local file = io.open(fallback_file, "rb")
+ if file then
+ io.close(file)
+ else
+ atwarn("Backing up pre-2.5.0 version of Interlocking save file to",fallback_file," for potential downgrade to older versions")
+ os.rename(advtrains.fpath.."_interlocking.ls", fallback_file)
+ end
end
--== load lines ==
@@ -473,8 +499,8 @@ advtrains.avt_save = function(remove_players_from_wagons)
"atc_brake_target", "atc_wait_finish", "atc_command", "atc_delay", "door_open",
"text_outside", "text_inside", "line", "routingcode",
"il_sections", "speed_restriction", "speed_restrictions_t", "is_shunt",
- "points_split", "autocouple", "atc_wait_autocouple", "ars_disable",
- "line_status",
+ "path_ori_cp", "autocouple", "atc_wait_autocouple", "ars_disable",
+ "staticdata", "line_status",
})
--then save it
tmp_trains[id]=v
@@ -748,6 +774,21 @@ minetest.register_chatcommand("at_whereis",
end
end,
})
+minetest.register_chatcommand("at_tp",
+ {
+ params = "<train id>",
+ description = attrans("Teleports you to the position of the train with the given id"),
+ privs = {train_operator = true, teleport = true},
+ func = function(name,param)
+ local train = advtrains.trains[param]
+ if not train or not train.last_pos then
+ return false, "Train "..param.." does not exist or is invalid"
+ else
+ minetest.get_player_by_name(name):set_pos(train.last_pos)
+ return true, "Teleporting to train "..param
+ end
+ end,
+})
minetest.register_chatcommand("at_disable_step",
{
params = "<yes/no>",
@@ -769,6 +810,16 @@ minetest.register_chatcommand("at_disable_step",
end,
})
+minetest.register_chatcommand("at_status",
+ {
+ params = "",
+ description = attrans("Print advtrains status info"),
+ privs = {train_operator = true},
+ func = function(name, param)
+ return true, advtrains.print_concat_table({"Advtrains Status: no_action",no_action,"slowdown",advtrains.global_slowdown,"(log",math.log(advtrains.global_slowdown),")"})
+ end,
+})
+
advtrains.is_no_action = function()
return no_action
end
diff --git a/advtrains/locale/README.md b/advtrains/locale/README.md
new file mode 120000
index 0000000..61e473c
--- /dev/null
+++ b/advtrains/locale/README.md
@@ -0,0 +1 @@
+../po/README.md \ No newline at end of file
diff --git a/advtrains/lzb.lua b/advtrains/lzb.lua
index 64e4553..116777c 100644
--- a/advtrains/lzb.lua
+++ b/advtrains/lzb.lua
@@ -29,8 +29,9 @@ The LZB subsystem keeps track of "checkpoints" the train will pass in the future
To perform 2, it populates the train.path_speed table which is handled along with the path subsystem.
This table is used in trainlogic.lua/train_step_b() and applied to the velocity calculations.
-Note: in contrast to node enter callbacks, which are called when the train passes the .5 index mark, LZB callbacks are executed on passing the .0 index mark!
-If an LZB checkpoint has speed 0, the train will still enter the node (the enter callback will be called), but will stop at the 0.9 index mark (for details, see SLOW_APPROACH in trainlogic.lua)
+Note: As of 2024-11-25, node enter callbacks as well as LZB callbacks are executed on passing the .0 index mark!
+If an LZB checkpoint has speed 0, it will stop at the 0.9 index mark (for details, see SLOW_APPROACH in trainlogic.lua) and will not enter the node.
+(previous behavior: node enter callback was called at .5 index mark)
The start point for the LZB traverser (and thus the first node that will receive an approach callback) is floor(train.index) + 1. This means, once the LZB checkpoint callback has fired,
this path node will not receive any further approach callbacks for the same approach situation
@@ -48,7 +49,7 @@ local params = {
ZONE_HOLD = 5, -- added on top of ZONE_ROLL
ZONE_VSLOW = 3, -- When speed is <2, still allow accelerating
- DST_FACTOR = 1.5,
+ DST_FACTOR = 3,--1.5,
SHUNT_SPEED_MAX = advtrains.SHUNT_SPEED_MAX,
}
@@ -264,7 +265,8 @@ end
advtrains.te_register_on_new_path(function(id, train)
advtrains.lzb_invalidate(train)
-- Taken care of in pre-move hook (see train_step_b)
- --look_ahead(id, train)
+ -- 2025-01-28 - do anyway, there seems to be an issue
+ look_ahead(id, train)
end)
advtrains.te_register_on_invalidate_ahead(function(id, train, start_idx)
diff --git a/advtrains/nodedb.lua b/advtrains/nodedb.lua
index b6521d5..408ff5d 100644
--- a/advtrains/nodedb.lua
+++ b/advtrains/nodedb.lua
@@ -212,6 +212,10 @@ function ndb.get_node(pos)
end
return n
end
+function ndb.get_ndef(pos)
+ local n=ndb.get_node_or_nil(pos)
+ return n and minetest.registered_nodes[n.name]
+end
function ndb.get_node_raw(pos)
local cid=ndbget(pos.x, pos.y, pos.z)
if cid then
@@ -264,23 +268,23 @@ end
--get_node with pseudoload. now we only need track data, so we can use the trackdb as second fallback
--nothing new will be saved inside the trackdb.
--returns:
---true, conn1, conn2, rely1, rely2, railheight in case everything's right.
+--true, conns, railheight, connmap in case everything's right.
--false if it's not a rail or the train does not drive on this rail, but it is loaded or
--nil if the node is neither loaded nor in trackdb
--the distraction between false and nil will be needed only in special cases.(train initpos)
-function advtrains.get_rail_info_at(pos, drives_on)
+function advtrains.get_rail_info_at(pos)
local rdp=advtrains.round_vector_floor_y(pos)
local node=ndb.get_node_or_nil(rdp)
if not node then return end
local nodename=node.name
- if(not advtrains.is_track_and_drives_on(nodename, drives_on)) then
+ if(not advtrains.is_track(nodename)) then
return false
end
- local conns, railheight, tracktype=advtrains.get_track_connections(node.name, node.param2)
+ local conns, railheight, connmap = advtrains.get_track_connections(node.name, node.param2)
- return true, conns, railheight
+ return true, conns, railheight, connmap
end
local IGNORE_WORLD = advtrains.IGNORE_WORLD
diff --git a/advtrains/occupation.lua b/advtrains/occupation.lua
index db39991..20a986e 100644
--- a/advtrains/occupation.lua
+++ b/advtrains/occupation.lua
@@ -86,10 +86,11 @@ end
function o.set_item(train_id, pos, idx)
local t = occgetcreate(pos)
+ assert(idx)
local i = 1
while t[i] do
- if t[i]==train_id then
- break
+ if t[i]==train_id and t[i+1]==idx then
+ return
end
i = i + 2
end
@@ -98,25 +99,30 @@ function o.set_item(train_id, pos, idx)
end
-function o.clear_item(train_id, pos)
+function o.clear_all_items(train_id, pos)
local t = occget(pos)
if not t then return end
local i = 1
- local moving = false
while t[i] do
if t[i]==train_id then
- if moving then
- -- if, for some occasion, there should be a duplicate entry, erase this one too
- atwarn("Duplicate occupation entry at",pos,"for train",train_id,":",t)
- i = i - 2
- end
- moving = true
+ table.remove(t, i)
+ table.remove(t, i)
+ else
+ i = i + 2
end
- if moving then
- t[i] = t[i+2]
- t[i+1] = t[i+3]
+ end
+end
+function o.clear_specific_item(train_id, pos, index)
+ local t = occget(pos)
+ if not t then return end
+ local i = 1
+ while t[i] do
+ if t[i]==train_id and t[i+1]==index then
+ table.remove(t, i)
+ table.remove(t, i)
+ else
+ i = i + 2
end
- i = i + 2
end
end
@@ -143,64 +149,88 @@ function o.check_collision(pos, train_id)
return false
end
--- Gets a mapping of train id's to indexes of trains that share this path item with this train
--- The train itself will not be included.
--- If the requested index position is off-track, returns {}.
--- returns (table with train_id->index), position
-function o.get_occupations(train, index)
- local ppos, ontrack = advtrains.path_get(train, index)
- if not ontrack then
- atlog("Train",train.id,"get_occupations requested off-track",index)
- return {}, ppos
- end
+-- Gets a mapping of train id's to indexes of trains that have a path item at this position
+-- Note that the case where 2 or more indices are at a position only occurs if there is a track loop.
+-- returns (table with train_id->{index1, index2...})
+function o.reverse_lookup(ppos)
local pos = advtrains.round_vector_floor_y(ppos)
local t = occget(pos)
if not t then return {} end
local r = {}
local i = 1
- local train_id = train.id
while t[i] do
- if t[i]~=train_id then
- r[t[i]] = t[i+1]
- end
+ if not r[t[i]] then r[t[i]] = {} end
+ table.insert(r[t[i]], t[i+1])
i = i + 2
end
- return r, pos
+ return r
end
--- Gets a mapping of train id's to indexes of trains that stand or drive over
+
+-- Gets a mapping of train id's to indexes of trains that have a path item at this position.
+-- Quick variant: will only return one index per train (the latest one added)
-- returns (table with train_id->index)
-function o.get_trains_at(ppos)
+function o.reverse_lookup_quick(ppos)
local pos = advtrains.round_vector_floor_y(ppos)
local t = occget(pos)
if not t then return {} end
local r = {}
local i = 1
while t[i] do
- local train = advtrains.trains[t[i]]
- local idx = t[i+1]
- if train.end_index - 0.5 <= idx and idx <= train.index + 0.5 then
- r[t[i]] = idx
- end
+ r[t[i]] = t[i+1]
i = i + 2
end
return r
end
--- Gets a mapping of train id's to indexes of trains that have a path
--- generated over this node
--- returns (table with train_id->index)
-function o.get_trains_over(ppos)
- local pos = advtrains.round_vector_floor_y(ppos)
- local t = occget(pos)
- if not t then return {} end
+local OCC_CLOSE_PROXIMITY = 3
+-- Gets a mapping of train id's to index of trains that have a path item at this position. Selects at most one index based on a given heuristic, or even none if it does not match the heuristic criterion
+-- returns (table with train_id->index), position
+-- "in_train": first index that lies between train index and end index
+-- "train_at_node": first index where the train is standing on that node (like in_train but with +-0.5 added to index)
+-- "first_ahead": smallest index that is > current index
+-- "before_end"(default): smallest index that is > end index
+-- "close_proximity": within 3 indices close to the train index and end_index
+-- "any": just output the first index found and do not check further (also occurs if both "in_train" and "first_ahead" heuristics have failed
+function o.reverse_lookup_sel(pos, heuristic)
+ if not heuristic then heuristic = "before_end" end
+ local om = o.reverse_lookup(pos)
local r = {}
- local i = 1
- while t[i] do
- local idx = t[i+1]
- r[t[i]] = idx
- i = i + 2
+ for tid, idxs in pairs(om) do
+ r[tid] = idxs[1]
+ if heuristic~="any" then
+ --must run a heuristic
+ --atdebug("reverse_lookup_sel is running heuristic for", pos,heuristic,"idxs",table.concat(idxs,","))
+ local otrn = advtrains.trains[tid]
+ advtrains.train_ensure_init(tid, otrn)
+ local h_value
+ for _,idx in ipairs(idxs) do
+ if heuristic == "first_ahead" and idx > otrn.index and (not h_value or h_value>idx) then
+ h_value = idx
+ end
+ if heuristic == "before_end" and idx > otrn.end_index and (not h_value or h_value>idx) then
+ h_value = idx
+ end
+ if heuristic == "in_train" and idx < otrn.index and idx > otrn.end_index then
+ h_value = idx
+ end
+ if heuristic == "train_at_node" and idx < (otrn.index+0.5) and idx > (otrn.end_index-0.5) then
+ h_value = idx
+ end
+ if heuristic == "close_proximity" and idx < (otrn.index + OCC_CLOSE_PROXIMITY) and idx > (otrn.end_index - OCC_CLOSE_PROXIMITY) then
+ h_value = idx
+ end
+ end
+ r[tid] = h_value
+ --atdebug(h_value,"chosen")
+ end
end
- return r
+ return r, pos
+end
+-- Gets a mapping of train id's to indexes of trains that stand or drive over
+-- returns (table with train_id->index)
+function o.get_trains_at(ppos)
+ local pos = advtrains.round_vector_floor_y(ppos)
+ return o.reverse_lookup_sel(pos, "train_at_node")
end
advtrains.occ = o
diff --git a/advtrains/p_mesecon_iface.lua b/advtrains/p_mesecon_iface.lua
index 33fcecd..0426e2b 100644
--- a/advtrains/p_mesecon_iface.lua
+++ b/advtrains/p_mesecon_iface.lua
@@ -14,13 +14,11 @@ minetest.override_item("mesecons_switch:mesecon_switch_off", {
minetest.sound_play("mesecons_switch", {pos=pos})
end,
advtrains = {
- getstate = "off",
- setstate = function(pos, node, newstate)
- if newstate=="on" then
- advtrains.ndb.swap_node(pos, {name="mesecons_switch:mesecon_switch_on", param2=node.param2})
- if advtrains.is_node_loaded(pos) then
- mesecon.receptor_on(pos)
- end
+ node_state = "off",
+ node_state_map = { on = "mesecons_switch:mesecon_switch_on", off = "mesecons_switch:mesecon_switch_off" },
+ node_on_switch_state = function(pos, new_node, old_state, new_state)
+ if advtrains.is_node_loaded(pos) then
+ mesecon.receptor_on(pos)
end
end,
on_updated_from_nodedb = function(pos, node)
@@ -41,16 +39,14 @@ minetest.override_item("mesecons_switch:mesecon_switch_on", {
minetest.sound_play("mesecons_switch", {pos=pos})
end,
advtrains = {
- getstate = "on",
- setstate = function(pos, node, newstate)
- if newstate=="off" then
- advtrains.ndb.swap_node(pos, {name="mesecons_switch:mesecon_switch_off", param2=node.param2})
- if advtrains.is_node_loaded(pos) then
- mesecon.receptor_off(pos)
- end
+ node_state = "on",
+ node_state_map = { on = "mesecons_switch:mesecon_switch_on", off = "mesecons_switch:mesecon_switch_off" },
+ node_on_switch_state = function(pos, new_node, old_state, new_state)
+ if advtrains.is_node_loaded(pos) then
+ mesecon.receptor_off(pos)
end
end,
- fallback_state = "off",
+ node_fallback_state = "off",
on_updated_from_nodedb = function(pos, node)
mesecon.receptor_on(pos)
end,
diff --git a/advtrains/passive.lua b/advtrains/passive.lua
index 156e846..59d2ea8 100644
--- a/advtrains/passive.lua
+++ b/advtrains/passive.lua
@@ -1,9 +1,5 @@
-- passive.lua
--- API to passive components, as described in passive_api.txt of advtrains_luaautomation
--- This has been moved to the advtrains core in turn with the interlocking system,
--- to prevent a dependency on luaautomation.
-
-local deprecation_warned = {}
+-- Rework for advtrains 2.5: The passive API now uses the reworked node_state system. Please see the comment in tracks.lua
function advtrains.getstate(parpos, pnode)
local pos
@@ -19,20 +15,8 @@ function advtrains.getstate(parpos, pnode)
local node=pnode or advtrains.ndb.get_node(pos)
local ndef=minetest.registered_nodes[node.name]
local st
- if ndef and ndef.advtrains and ndef.advtrains.getstate then
- st=ndef.advtrains.getstate
- elseif ndef and ndef.luaautomation and ndef.luaautomation.getstate then
- if not deprecation_warned[node.name] then
- minetest.log("warning", node.name.." uses deprecated definition of ATLATC functions in the 'luaautomation' field. Please move them to the 'advtrains' field!")
- end
- st=ndef.luaautomation.getstate
- else
- return nil
- end
- if type(st)=="function" then
- return st(pos, node)
- else
- return st
+ if ndef and ndef.advtrains then
+ return ndef.advtrains.node_state
end
end
@@ -45,31 +29,48 @@ function advtrains.setstate(parpos, newstate, pnode)
end
if type(pos)~="table" or (not pos.x or not pos.y or not pos.z) then
debug.sethook()
- error("Invalid position supplied to getstate")
+ error("Invalid position supplied to setstate")
end
local node=pnode or advtrains.ndb.get_node(pos)
local ndef=minetest.registered_nodes[node.name]
- local st
- if ndef and ndef.advtrains and ndef.advtrains.setstate then
- st=ndef.advtrains.setstate
- elseif ndef and ndef.luaautomation and ndef.luaautomation.setstate then
- if not deprecation_warned[node.name] then
- minetest.log("warning", node.name.." uses deprecated definition of ATLATC functions in the 'luaautomation' field. Please move them to the 'advtrains' field!")
- end
- st=ndef.luaautomation.setstate
- else
- return nil
+
+ if not ndef or not ndef.advtrains then
+ return false, "missing_node_def"
+ end
+ local old_state = ndef.advtrains.node_state
+
+ if old_state == newstate then
+ -- nothing needs to be done
+ return true
end
+ if not ndef.advtrains.node_state_map then
+ return false, "missing_node_state_map"
+ end
+ local new_node_name = ndef.advtrains.node_state_map[newstate]
+ if not new_node_name then
+ return false, "no_such_state"
+ end
+
+ -- prevent state switching when route lock or train is present
if advtrains.get_train_at_pos(pos) then
- return false
+ return false, "train_here"
end
- if advtrains.interlocking and advtrains.interlocking.route.has_route_lock(minetest.pos_to_string(pos)) then
- return false
+ if advtrains.interlocking and advtrains.interlocking.route.has_route_lock(advtrains.encode_pos(pos)) then
+ return false, "route_lock_here"
end
- st(pos, node, newstate)
+ -- perform the switch
+ local new_node = {name = new_node_name, param2 = node.param2}
+ advtrains.ndb.swap_node(pos, new_node)
+ -- if callback is present, call it
+ if ndef.advtrains.node_on_switch_state then
+ ndef.advtrains.node_on_switch_state(pos, new_node, old_state, newstate)
+ end
+ -- invalidate paths (only relevant if this is a track)
+ advtrains.invalidate_all_paths(pos)
+
return true
end
@@ -86,12 +87,7 @@ function advtrains.is_passive(parpos, pnode)
end
local node=pnode or advtrains.ndb.get_node(pos)
local ndef=minetest.registered_nodes[node.name]
- if ndef and ndef.advtrains and ndef.advtrains.getstate then
- return true
- elseif ndef and ndef.luaautomation and ndef.luaautomation.getstate then
- if not deprecation_warned[node.name] then
- minetest.log("warning", node.name.." uses deprecated definition of ATLATC functions in the 'luaautomation' field. Please move them to the 'advtrains' field!")
- end
+ if ndef and ndef.advtrains and ndef.advtrains.node_state_map then
return true
else
return false
@@ -102,20 +98,10 @@ end
function advtrains.set_fallback_state(pos, pnode)
local node=pnode or advtrains.ndb.get_node(pos)
local ndef=minetest.registered_nodes[node.name]
- local st
- if ndef and ndef.advtrains and ndef.advtrains.setstate
- and ndef.advtrains.fallback_state then
- if advtrains.get_train_at_pos(pos) then
- return false
- end
-
- if advtrains.interlocking and advtrains.interlocking.route.has_route_lock(minetest.pos_to_string(pos)) then
- return false
- end
-
- ndef.advtrains.setstate(pos, node, ndef.advtrains.fallback_state)
- return true
- end
+ if not ndef or not ndef.advtrains or not ndef.advtrains.node_fallback_state then
+ return false, "no_fallback_state"
+ end
+ return advtrains.setstate(pos, ndef.advtrains.node_fallback_state, node)
end
diff --git a/advtrains/path.lua b/advtrains/path.lua
index f2b8a13..57829ad 100644
--- a/advtrains/path.lua
+++ b/advtrains/path.lua
@@ -33,17 +33,16 @@
-- If you need to proceed along the path by a specific actual distance, it does NOT work to simply add it to the index. You should use the path_get_index_by_offset() function.
-- creates the path data structure, reconstructing the train from a position and a connid
--- Important! train.drives_on must exist while calling this method
-- returns: true - successful
-- nil - node not yet available/unloaded, please wait
-- false - node definitely gone, remove train
function advtrains.path_create(train, pos, connid, rel_index)
local posr = advtrains.round_vector_floor_y(pos)
- local node_ok, conns, rhe = advtrains.get_rail_info_at(pos, train.drives_on)
+ local node_ok, conns, rhe, connmap = advtrains.get_rail_info_at(pos)
if not node_ok then
return node_ok
end
- local mconnid = advtrains.get_matching_conn(connid, #conns)
+ local mconnid = advtrains.get_matching_conn(connid, connmap)
train.index = rel_index
train.path = { [0] = { x=posr.x, y=posr.y+rhe, z=posr.z } }
train.path_cn = { [0] = connid }
@@ -113,14 +112,21 @@ end
-- before returning from the calling function.
function advtrains.path_invalidate(train, ignore_lock)
if advtrains.lock_path_inval and not ignore_lock then
- atwarn("Train ",train.train_id,": Illegal path invalidation has occured during train step:")
+ atwarn("Train ",train.id,": Illegal path invalidation has occured during train step:")
atwarn(debug.traceback())
end
-
if train.path then
+ --atdebug("path_invalidate for",train.id)
+ local _cnt = 0
for i,p in pairs(train.path) do
- advtrains.occ.clear_item(train.id, advtrains.round_vector_floor_y(p))
+ _cnt = _cnt + 1
+ if _cnt > 10000 then
+ atdebug(train)
+ error("Loop trap in advtrains.path_invalidate was triggered!")
+ end
+ advtrains.occ.clear_all_items(train.id, advtrains.round_vector_floor_y(p))
end
+ --atdebug("occ cleared")
end
train.path = nil
train.path_dist = nil
@@ -162,7 +168,7 @@ function advtrains.path_invalidate_ahead(train, start_idx, ignore_when_passed)
-- leave current node in path, it won't change. What might change is the path onward from here (e.g. switch)
local i = idx + 1
while train.path[i] do
- advtrains.occ.clear_item(train.id, advtrains.round_vector_floor_y(train.path[i]))
+ advtrains.occ.clear_specific_item(train.id, advtrains.round_vector_floor_y(train.path[i]), i)
i = i+1
end
train.path_ext_f=idx
@@ -207,23 +213,19 @@ function advtrains.path_get(train, index)
while index > pef do
local pos = train.path[pef]
local connid = train.path_cn[pef]
- local node_ok, this_conns, adj_pos, adj_connid, conn_idx, nextrail_y, next_conns
+ local node_ok, this_conns, adj_pos, adj_connid, conn_idx, nextrail_y, next_conns, next_connmap
if pef == train.path_trk_f then
node_ok, this_conns = advtrains.get_rail_info_at(pos)
if not node_ok then error("For train "..train.id..": Path item "..pef.." on-track but not a valid node!") end
- adj_pos, adj_connid, conn_idx, nextrail_y, next_conns = advtrains.get_adjacent_rail(pos, this_conns, connid, train.drives_on)
+ adj_pos, adj_connid, conn_idx, nextrail_y, next_conns, next_connmap = advtrains.get_adjacent_rail(pos, this_conns, connid)
end
pef = pef + 1
if adj_pos then
advtrains.occ.set_item(train.id, adj_pos, pef)
-
- -- If we have split points, notify accordingly
- local mconnid = advtrains.get_matching_conn(adj_connid, #next_conns)
- if #next_conns==3 and adj_connid==1 and train.points_split and train.points_split[advtrains.encode_pos(adj_pos)] then
- --atdebug(id,"has split points restored at",adj_pos)
- mconnid = 3
- end
-
+
+ local mconnid = advtrains.get_matching_conn(adj_connid, next_connmap)
+ -- NO split points handling here. It is only required for backwards path calculation
+
adj_pos.y = adj_pos.y + nextrail_y
train.path_cp[pef] = adj_connid
train.path_cn[pef] = mconnid
@@ -246,23 +248,26 @@ function advtrains.path_get(train, index)
while index < peb do
local pos = train.path[peb]
local connid = train.path_cp[peb]
- local node_ok, this_conns, adj_pos, adj_connid, conn_idx, nextrail_y, next_conns
+ local node_ok, this_conns, adj_pos, adj_connid, conn_idx, nextrail_y, next_conns, next_connmap
if peb == train.path_trk_b then
node_ok, this_conns = advtrains.get_rail_info_at(pos)
if not node_ok then error("For train "..train.id..": Path item "..peb.." on-track but not a valid node!") end
- adj_pos, adj_connid, conn_idx, nextrail_y, next_conns = advtrains.get_adjacent_rail(pos, this_conns, connid, train.drives_on)
+ adj_pos, adj_connid, conn_idx, nextrail_y, next_conns, next_connmap = advtrains.get_adjacent_rail(pos, this_conns, connid)
end
peb = peb - 1
if adj_pos then
advtrains.occ.set_item(train.id, adj_pos, peb)
- -- If we have split points, notify accordingly
- local mconnid = advtrains.get_matching_conn(adj_connid, #next_conns)
- if #next_conns==3 and adj_connid==1 and train.points_split and train.points_split[advtrains.encode_pos(adj_pos)] then
- -- atdebug(id,"has split points restored at",adj_pos)
- mconnid = 3
+ local mconnid = advtrains.get_matching_conn(adj_connid, next_connmap)
+ -- If, for this position, we have remembered the origin conn, apply it here
+ if next_connmap then -- only needs to be done when this track is a turnout (>2 conns)
+ local origin_conn = train.path_ori_cp[advtrains.encode_pos(adj_pos)]
+ if origin_conn then
+ --atdebug("Train",train.id,"at",adj_pos,"restoring turnout origin CP",origin_conn,"for path item",index)
+ mconnid = origin_conn
+ end
end
-
+
adj_pos.y = adj_pos.y + nextrail_y
train.path_cn[peb] = adj_connid
train.path_cp[peb] = mconnid
@@ -375,12 +380,25 @@ function advtrains.path_get_index_by_offset(train, index, offset)
return c_idx + frac
end
+
+-- The path_dist[] table contains absolute distance values for every whole index.
+-- Use this function to retrieve the correct absolute distance for a fractional index value (interpolate between floor and ceil index)
+-- returns: absolute distance from path item 0
+function advtrains.path_get_path_dist_fractional(train, index)
+ local start_index_f = atfloor(index)
+ local frac = index - start_index_f
+ -- ensure path exists
+ advtrains.path_get_adjacent(train, index)
+ local dist1, dist2 = train.path_dist[start_index_f], train.path_dist[start_index_f+1]
+ return dist1 + (dist2-dist1)*frac
+end
+
local PATH_CLEAR_KEEP = 4
function advtrains.path_clear_unused(train)
local i
for i = train.path_ext_b, train.path_req_b - PATH_CLEAR_KEEP do
- advtrains.occ.clear_item(train.id, advtrains.round_vector_floor_y(train.path[i]))
+ advtrains.occ.clear_specific_item(train.id, advtrains.round_vector_floor_y(train.path[i]), i)
train.path[i] = nil
train.path_dist[i-1] = nil
train.path_cp[i] = nil
@@ -421,18 +439,19 @@ end
-- Projects the path of "train" onto the path of "onto_train_id", and returns the index on onto_train's path
-- that corresponds to "index" on "train"'s path, as well as whether both trains face each other
-- index may be fractional
+-- heuristic: see advtrains.occ.reverse_lookup_sel()
-- returns: res_index, trains_facing
-- returns nil when path can not be projected, either because trains are on different tracks or
-- node at "index" happens to be on a turnout and it's the wrong direction
-- Note - duplicate with similar functionality is in train_step_b() - that code combines train detection with projecting
-function advtrains.path_project(train, index, onto_train_id)
+function advtrains.path_project(train, index, onto_train_id, heuristic)
local base_idx = atfloor(index)
local frac_part = index - base_idx
local base_pos = advtrains.path_get(train, base_idx)
local base_cn = train.path_cn[base_idx]
local otrn = advtrains.trains[onto_train_id]
-- query occupation
- local occ = advtrains.occ.get_trains_over(base_pos)
+ local occ = advtrains.occ.reverse_lookup_sel(base_pos, heuristic)
-- is wanted train id contained?
local ob_idx = occ[onto_train_id]
if not ob_idx then
diff --git a/advtrains/po/README.md b/advtrains/po/README.md
new file mode 100644
index 0000000..3e94682
--- /dev/null
+++ b/advtrains/po/README.md
@@ -0,0 +1,70 @@
+# Translations
+Please read this document before working on any translations.
+
+Unlike many other mods, Advtrains uses `.po` files for localization,
+which are then automatically converted to `.tr` files when the mod is
+loaded. Therefore, please submit patches that edit the `.po` files.
+
+## Getting Started
+The translation files can be edited like any other `.po` file.
+
+If the translation file for your language does not exist, create it by
+copying `template.txt` to `advtrains.XX.tr`, where `XX` is replaced by
+the language code.
+
+Feel free to use the [discussion mailing list][srht-discuss] if you
+have any questions regarding localization.
+
+You can share your `.po` file directly or [as a patch][gsm] to the [dev
+mailing list][srht-devel]. The latter is encouraged, but, unlike code
+changes, translation files sent directly are also accepted.
+
+[tr-format]: https://minetest.gitlab.io/minetest/translations/#translation-file-format
+[srht-discuss]: https://lists.sr.ht/~gpcf/advtrains-discuss
+[srht-devel]: https://lists.sr.ht/~gpcf/advtrains-devel
+[gsm]: https://git-send-email.io
+
+## Translation Notes
+* Translations should be consistent. You can use other entries or the
+translations in Minetest as a reference.
+* Translations do not have to fully correspond to the original text -
+they only need to provide the same information. In particular,
+translations do not need to have the same linguistical structure as the
+original text.
+* Replacement sequences (`@1`, `@2`, etc) should not be translated.
+* Certain abbreviations or names, such as "Ks" or "Zs 3", should
+generally not be translated.
+
+### (de) German
+* Verwenden Sie die neue Rechtschreibung und die Sie-Form.
+* Mit der deutschen Tastaturbelegung unter Linux können die
+Anführungszeichen „“ mit AltGr-V bzw. AltGr-B eingegeben werden.
+
+### (zh) Chinese
+(This section is written in English to avoid writing the note twice or
+using only one of the variants, as most of this section applies to both
+the traditional and simplified variants.)
+
+* Please use the 「」 quotation marks for Traditional Chinese and “”
+for Simplified Chinese.
+* Please use the fullwidth variants of: , 、 。 ? ! : ;
+* Please use the halfwidth variants of: ( ) [ ] / \ |
+* Please do not leave any space between Han characters (including
+fullwidth punctuation marks).
+* Please leave a space between Han characters (excluding fullwidth
+punctuation marks) and characters from other scripts (including
+halfwidth punctuation marks). However, do not leave any space between
+Han characters and Arabic numerals.
+
+## Notes for developers
+* The `update-translations.sh` script can be used to update the
+translation files. However, please make sure to install the
+`basic_trains` mod before running the script.
+* Please make sure that the first argument to `S` (or `attrans`) _only_
+includes string literals without formatting or concatenation. This is
+unfortunately a limitation of the `xgettext` utility.
+* Avoid word-by-word translations.
+* Avoid manipulating translated strings (except for concatenation). Use
+server-side translations if you have to modify the text sent to users.
+* Avoid truncating strings unless multibyte characters are handled
+properly.
diff --git a/advtrains/po/advtrains.pot b/advtrains/po/advtrains.pot
new file mode 100644
index 0000000..6fda1d7
--- /dev/null
+++ b/advtrains/po/advtrains.pot
@@ -0,0 +1,632 @@
+# SOME DESCRIPTIVE TITLE.
+# Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER
+# This file is distributed under the same license as the advtrains package.
+# FIRST AUTHOR <EMAIL@ADDRESS>, YEAR.
+#
+#, fuzzy
+msgid ""
+msgstr ""
+"Project-Id-Version: advtrains\n"
+"Report-Msgid-Bugs-To: advtrains-discuss@lists.sr.ht\n"
+"POT-Creation-Date: 2023-10-09 11:02+0200\n"
+"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
+"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
+"Language-Team: LANGUAGE <LL@li.org>\n"
+"Language: \n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=CHARSET\n"
+"Content-Transfer-Encoding: 8bit\n"
+
+#: advtrains/atc.lua:109
+msgid "Unconfigured ATC controller"
+msgstr ""
+
+#: advtrains/atc.lua:150
+msgid ""
+"ATC controller, mode @1\n"
+"Command: @2"
+msgstr ""
+
+#: advtrains/atc.lua:180
+msgid "Command"
+msgstr ""
+
+#: advtrains/atc.lua:184
+msgid "Command (on)"
+msgstr ""
+
+#: advtrains/atc.lua:187
+msgid "Digiline channel"
+msgstr ""
+
+#: advtrains/atc.lua:189 advtrains_line_automation/stoprail.lua:65
+#: advtrains_luaautomation/active_common.lua:48
+msgid "Save"
+msgstr ""
+
+#: advtrains/atc.lua:236
+msgid "ATC Reverse command warning: didn't reverse train, train moving."
+msgstr ""
+
+#: advtrains/atc.lua:248
+msgid "ATC Kick command warning: doors are closed."
+msgstr ""
+
+#: advtrains/atc.lua:252
+msgid "ATC Kick command warning: train moving."
+msgstr ""
+
+#: advtrains/atc.lua:322
+msgid "ATC command syntax error: I statement not closed: @1"
+msgstr ""
+
+#: advtrains/atc.lua:385
+msgid "ATC command parse error: Unknown command: @1"
+msgstr ""
+
+#: advtrains/copytool.lua:8
+msgid ""
+"Train copy/paste tool\n"
+"\n"
+"Left-click: copy train\n"
+"Right-click: paste train"
+msgstr ""
+
+#: advtrains/copytool.lua:29
+msgid "You do not have the @1 privilege."
+msgstr ""
+
+#: advtrains/copytool.lua:41
+msgid "The track you are trying to place the wagon on is not long enough."
+msgstr ""
+
+#: advtrains/copytool.lua:47
+msgid "The clipboard couldn't access the metadata. Paste failed."
+msgstr ""
+
+#: advtrains/copytool.lua:52 advtrains/copytool.lua:57
+msgid "The clipboard is empty."
+msgstr ""
+
+#: advtrains/copytool.lua:74
+msgid "Back of train would end up off track, cancelling."
+msgstr ""
+
+#: advtrains/copytool.lua:92
+msgid "No such lua entity."
+msgstr ""
+
+#: advtrains/copytool.lua:98
+msgid "No such wagon: @1."
+msgstr ""
+
+#: advtrains/copytool.lua:104
+msgid "No such train: @1."
+msgstr ""
+
+#: advtrains/copytool.lua:176
+msgid "The clipboard couldn't access the metadata. Copy failed."
+msgstr ""
+
+#: advtrains/copytool.lua:180
+msgid "Train copied."
+msgstr ""
+
+#: advtrains/couple.lua:28
+msgid "Buffer and Chain Coupler"
+msgstr ""
+
+#: advtrains/couple.lua:29
+msgid "Scharfenberg Coupler"
+msgstr ""
+
+#: advtrains/couple.lua:185
+msgid ""
+"You are not allowed to couple trains without the train_operator privilege."
+msgstr ""
+
+#: advtrains/couple.lua:329 advtrains/couple.lua:333
+msgid "<No coupler>"
+msgstr ""
+
+#: advtrains/couple.lua:334
+msgid "Can not couple: The couplers of the trains do not match (@1 and @2)."
+msgstr ""
+
+#: advtrains/craft_items.lua:3
+msgid "Boiler"
+msgstr ""
+
+#: advtrains/craft_items.lua:9
+msgid "Driver's cab"
+msgstr ""
+
+#: advtrains/craft_items.lua:15
+msgid "Wheel"
+msgstr ""
+
+#: advtrains/craft_items.lua:21
+msgid "Chimney"
+msgstr ""
+
+#: advtrains/misc_nodes.lua:16
+msgid "@1 Platform (low)"
+msgstr ""
+
+#: advtrains/misc_nodes.lua:33
+msgid "@1 Platform (high)"
+msgstr ""
+
+#: advtrains/misc_nodes.lua:59
+msgid "@1 Platform (45 degree)"
+msgstr ""
+
+#: advtrains/misc_nodes.lua:81
+msgid "@1 Platform (low, 45 degree)"
+msgstr ""
+
+#: advtrains/protection.lua:7
+msgid "Can place, remove and operate trains"
+msgstr ""
+
+#: advtrains/protection.lua:12
+msgid ""
+"Can place, remove and operate any train, regardless of owner, whitelist, or "
+"protection"
+msgstr ""
+
+#: advtrains/protection.lua:18
+msgid "Can place and dig tracks in unprotected areas"
+msgstr ""
+
+#: advtrains/protection.lua:24
+msgid "Can operate turnouts and signals in unprotected areas"
+msgstr ""
+
+#: advtrains/protection.lua:148
+msgid ""
+"You are not allowed to build near tracks without the track_builder privilege."
+msgstr ""
+
+#: advtrains/protection.lua:148
+msgid ""
+"You are not allowed to build tracks without the track_builder privilege."
+msgstr ""
+
+#: advtrains/protection.lua:153
+msgid "You are not allowed to build near tracks at this protected position."
+msgstr ""
+
+#: advtrains/protection.lua:153
+msgid "You are not allowed to build tracks at this protected position."
+msgstr ""
+
+#: advtrains/protection.lua:184
+msgid ""
+"You are not allowed to operate turnouts and signals without the "
+"railway_operator privilege."
+msgstr ""
+
+#: advtrains/signals.lua:63
+msgid "Lampless Signal"
+msgstr ""
+
+#: advtrains/signals.lua:127
+msgid "Signal"
+msgstr ""
+
+#: advtrains/signals.lua:191
+msgid "Wallmounted Signal (left)"
+msgstr ""
+
+#: advtrains/signals.lua:192
+msgid "Wallmounted Signal (right)"
+msgstr ""
+
+#: advtrains/signals.lua:193
+msgid "Wallmounted Signal (top)"
+msgstr ""
+
+#: advtrains/signals.lua:281 advtrains/signals.lua:322
+msgid "Andrew's Cross"
+msgstr ""
+
+#: advtrains/trackplacer.lua:313
+msgid ""
+"Track Worker Tool\n"
+"\n"
+"Left-click: change rail type (straight/curve/switch)\n"
+"Right-click: rotate object"
+msgstr ""
+
+#: advtrains/trackplacer.lua:340 advtrains/trackplacer.lua:377
+msgid "This node can't be rotated using the trackworker."
+msgstr ""
+
+#: advtrains/trackplacer.lua:350
+msgid "This track can not be rotated."
+msgstr ""
+
+#: advtrains/trackplacer.lua:404
+msgid "This node can't be changed using the trackworker."
+msgstr ""
+
+#: advtrains/trackplacer.lua:414
+msgid "This track can not be changed."
+msgstr ""
+
+#: advtrains/tracks.lua:449
+msgid "This track can not be removed."
+msgstr ""
+
+#: advtrains/tracks.lua:616
+msgid "Position is occupied by a train."
+msgstr ""
+
+#: advtrains/tracks.lua:622
+msgid "There's a Track Circuit Break here."
+msgstr ""
+
+#: advtrains/tracks.lua:626
+msgid "There's a Signal Influence Point here."
+msgstr ""
+
+#: advtrains/tracks.lua:637
+msgid "@1 Slope"
+msgstr ""
+
+#: advtrains/tracks.lua:648 advtrains/tracks.lua:653
+msgid "Can't place slope: not pointing at node."
+msgstr ""
+
+#: advtrains/tracks.lua:658
+msgid "Can't place slope: space occupied."
+msgstr ""
+
+#: advtrains/tracks.lua:711
+msgid "Can't place slope: Not enough slope items left (@1 required)."
+msgstr ""
+
+#: advtrains/tracks.lua:714
+msgid "Can't place slope: There's no slope of length @1."
+msgstr ""
+
+#: advtrains/tracks.lua:721
+msgid "Can't place slope: no supporting node at upper end."
+msgstr ""
+
+#: advtrains/trainhud.lua:305
+msgid "OVERRUN RED SIGNAL! Examine situation and reverse train to move again."
+msgstr ""
+
+#: advtrains/wagons.lua:179
+msgid "This wagon is owned by @1, you can't destroy it."
+msgstr ""
+
+#: advtrains/wagons.lua:203
+msgid "The wagon's inventory is not empty."
+msgstr ""
+
+#: advtrains/wagons.lua:210
+msgid "Wagon needs to be decoupled from other wagons in order to destroy it."
+msgstr ""
+
+#: advtrains/wagons.lua:216
+msgid ""
+"Warning: If you destroy this wagon, you only get some steel back! If you are "
+"sure, hold Sneak and left-click the wagon."
+msgstr ""
+
+#: advtrains/wagons.lua:649 advtrains/wagons.lua:850
+msgid "Show Inventory"
+msgstr ""
+
+#: advtrains/wagons.lua:652
+msgid "Onboard Computer"
+msgstr ""
+
+#: advtrains/wagons.lua:655 advtrains/wagons.lua:1328
+msgid "Wagon properties"
+msgstr ""
+
+#: advtrains/wagons.lua:658
+msgid "Get off"
+msgstr ""
+
+#: advtrains/wagons.lua:661
+msgid "Get off (forced)"
+msgstr ""
+
+#: advtrains/wagons.lua:663
+msgid "(Doors closed)"
+msgstr ""
+
+#: advtrains/wagons.lua:692
+msgid "This wagon has no seats."
+msgstr ""
+
+#: advtrains/wagons.lua:703
+msgid "This wagon is full."
+msgstr ""
+
+#: advtrains/wagons.lua:706
+msgid "Doors are closed! (Try holding sneak key!)"
+msgstr ""
+
+#: advtrains/wagons.lua:712
+msgid "You can't get on this wagon."
+msgstr ""
+
+#: advtrains/wagons.lua:838
+msgid "Select seat:"
+msgstr ""
+
+#: advtrains/wagons.lua:880
+msgid "Save wagon properties"
+msgstr ""
+
+#: advtrains/wagons.lua:965
+msgid "Text displayed outside on train"
+msgstr ""
+
+#: advtrains/wagons.lua:966
+msgid "Text displayed inside train"
+msgstr ""
+
+#: advtrains/wagons.lua:967
+msgid "Line"
+msgstr ""
+
+#: advtrains/wagons.lua:968
+msgid "Routingcode"
+msgstr ""
+
+#: advtrains/wagons.lua:1241
+msgid ""
+"Doors are closed. Use Sneak+rightclick to ignore the closed doors and get "
+"off."
+msgstr ""
+
+#: advtrains/wagons.lua:1250
+msgid "You are not allowed to access the driver stand."
+msgstr ""
+
+#: advtrains_interlocking/tsr_rail.lua:13
+msgid "Point speed restriction: @1"
+msgstr ""
+
+#: advtrains_interlocking/tsr_rail.lua:14
+msgid "Set point speed restriction:"
+msgstr ""
+
+#: advtrains_interlocking/tsr_rail.lua:30
+msgid "You are not allowed to configure this track without the @1 privilege."
+msgstr ""
+
+#: advtrains_interlocking/tsr_rail.lua:34
+#: advtrains_line_automation/stoprail.lua:31
+#: advtrains_line_automation/stoprail.lua:76
+msgid "You are not allowed to configure this track."
+msgstr ""
+
+#: advtrains_interlocking/tsr_rail.lua:64
+msgid "Point Speed Restriction Track"
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:54
+msgid "Station Code"
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:55
+msgid "Station Name"
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:56
+msgid "Door Delay"
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:57
+msgid "Dep. Speed"
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:58 advtrains_train_track/init.lua:11
+#: advtrains_train_track/init.lua:156
+msgid "Track"
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:59
+msgid "Stop Time"
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:60
+msgid "Door Side"
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:62
+msgid "Reverse train"
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:63
+msgid "Kick out passengers"
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:97
+msgid "Station code \"@1\" already exists and is owned by @2."
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:111
+msgid "This station is owned by @1. You are not allowed to edit its name."
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:221
+msgid "Station/Stop Track"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:17
+msgid "Unconfigured LuaATC component"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:46
+msgid "LuaATC Environment"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:49
+msgid "Clear Local Environment"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:50
+msgid "Code"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:64
+msgid ""
+"You are not allowed to configure this LuaATC component without the @1 "
+"privilege."
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:94
+msgid "LuaATC component assigned to environment '@1'"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:96
+msgid "LuaATC component assigned to an invalid environment"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:171
+msgid "LuaATC component with error: @1"
+msgstr ""
+
+#: advtrains_luaautomation/init.lua:13
+msgid ""
+"Can place and configure LuaATC components, including execute potentially "
+"harmful Lua code"
+msgstr ""
+
+#: advtrains_luaautomation/mesecon_controller.lua:211
+msgid "LuaATC Mesecon Controller"
+msgstr ""
+
+#: advtrains_luaautomation/operation_panel.lua:11
+msgid "LuaATC Operation Panel"
+msgstr ""
+
+#: advtrains_luaautomation/pcnaming.lua:28
+msgid ""
+"Passive Component Naming Tool\n"
+"\n"
+"Right-click to name a passive component."
+msgstr ""
+
+#: advtrains_luaautomation/pcnaming.lua:39
+msgid ""
+"You are not allowed to name LuaATC passive components without the @1 "
+"privilege."
+msgstr ""
+
+#: advtrains_luaautomation/pcnaming.lua:62
+msgid "Set name of component (empty to clear)"
+msgstr ""
+
+#: advtrains_train_industrial/init.lua:10
+#: advtrains_train_industrial/init.lua:49 advtrains_train_steam/init.lua:20
+#: advtrains_train_steam/init.lua:91
+msgid "Driver Stand (right)"
+msgstr ""
+
+#: advtrains_train_industrial/init.lua:17
+#: advtrains_train_industrial/init.lua:56 advtrains_train_steam/init.lua:14
+#: advtrains_train_steam/init.lua:85
+msgid "Driver Stand (left)"
+msgstr ""
+
+#: advtrains_train_industrial/init.lua:40
+msgid "Industrial Train Engine"
+msgstr ""
+
+#: advtrains_train_industrial/init.lua:79
+msgid "Big Industrial Train Engine"
+msgstr ""
+
+#: advtrains_train_industrial/init.lua:98
+msgid "Industrial tank wagon"
+msgstr ""
+
+#: advtrains_train_industrial/init.lua:116
+msgid "Industrial wood wagon"
+msgstr ""
+
+#: advtrains_train_japan/init.lua:4
+msgid "Japanese Train Inter-Wagon Connection"
+msgstr ""
+
+#: advtrains_train_japan/init.lua:37
+msgid "Driver stand"
+msgstr ""
+
+#: advtrains_train_japan/init.lua:101
+msgid "Japanese Train Engine"
+msgstr ""
+
+#: advtrains_train_japan/init.lua:176
+msgid "Japanese Train Wagon"
+msgstr ""
+
+#: advtrains_train_steam/init.lua:75
+msgid "Steam Engine"
+msgstr ""
+
+#: advtrains_train_steam/init.lua:159
+msgid "Detailed Steam Engine"
+msgstr ""
+
+#: advtrains_train_steam/init.lua:206
+msgid "Passenger Wagon"
+msgstr ""
+
+#: advtrains_train_steam/init.lua:226
+msgid "Box Wagon"
+msgstr ""
+
+#: advtrains_train_subway/init.lua:144
+msgid "Subway Passenger Wagon"
+msgstr ""
+
+#: advtrains_train_track/init.lua:31
+msgid "Y-turnout"
+msgstr ""
+
+#: advtrains_train_track/init.lua:49
+msgid "3-way turnout"
+msgstr ""
+
+#: advtrains_train_track/init.lua:69
+msgid "Perpendicular Diamond Crossing Track"
+msgstr ""
+
+#: advtrains_train_track/init.lua:91
+msgid "90+Angle Diamond Crossing Track"
+msgstr ""
+
+#: advtrains_train_track/init.lua:132
+msgid "Diagonal Diamond Crossing Track"
+msgstr ""
+
+#: advtrains_train_track/init.lua:179
+msgid "Bumper"
+msgstr ""
+
+#: advtrains_train_track/init.lua:201
+msgid "ATC controller"
+msgstr ""
+
+#: advtrains_train_track/init.lua:317
+msgid "Unloading Track"
+msgstr ""
+
+#: advtrains_train_track/init.lua:342
+msgid "Loading Track"
+msgstr ""
+
+#: advtrains_train_track/init.lua:406
+msgid "Detector Rail"
+msgstr ""
diff --git a/advtrains/po/de.po b/advtrains/po/de.po
new file mode 100644
index 0000000..8821fe3
--- /dev/null
+++ b/advtrains/po/de.po
@@ -0,0 +1,724 @@
+msgid ""
+msgstr ""
+"Project-Id-Version: advtrains\n"
+"Report-Msgid-Bugs-To: advtrains-discuss@lists.sr.ht\n"
+"POT-Creation-Date: 2023-10-09 11:02+0200\n"
+"PO-Revision-Date: 2023-10-09 11:18+0200\n"
+"Last-Translator: Y. Wang <yw05@forksworld.de>\n"
+"Language-Team: German\n"
+"Language: de\n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"X-Generator: Poedit 3.3.2\n"
+
+#: advtrains/atc.lua:109
+msgid "Unconfigured ATC controller"
+msgstr "Nicht konfiguiertes Zugbeeinflussungsgleis"
+
+#: advtrains/atc.lua:150
+msgid ""
+"ATC controller, mode @1\n"
+"Command: @2"
+msgstr ""
+"Zugbeeinflussungsgleis in Betriebsart „@1“\n"
+"Befehl: @2"
+
+#: advtrains/atc.lua:180
+msgid "Command"
+msgstr "Befehl"
+
+#: advtrains/atc.lua:184
+msgid "Command (on)"
+msgstr "Befehl (wenn aktiviert)"
+
+#: advtrains/atc.lua:187
+msgid "Digiline channel"
+msgstr "Digiline-Kanal"
+
+#: advtrains/atc.lua:189 advtrains_line_automation/stoprail.lua:65
+#: advtrains_luaautomation/active_common.lua:48
+msgid "Save"
+msgstr "Speichern"
+
+#: advtrains/atc.lua:236
+msgid "ATC Reverse command warning: didn't reverse train, train moving."
+msgstr ""
+"Zugbeeinflussung: Der Zug befindet sich in Bewegung und kann nicht umgekehrt "
+"werden."
+
+#: advtrains/atc.lua:248
+msgid "ATC Kick command warning: doors are closed."
+msgstr ""
+"Zugbeeinflussung: Wegen geschlossener Türen werden Fahrgäste nicht zum "
+"Ausstieg gezwungen."
+
+#: advtrains/atc.lua:252
+msgid "ATC Kick command warning: train moving."
+msgstr ""
+"Zugbeeinflussung: Der Zug befindet sich in Bewegung, Fahrgäste werden nicht "
+"zum Ausstieg gezwungen."
+
+#: advtrains/atc.lua:322
+msgid "ATC command syntax error: I statement not closed: @1"
+msgstr "Zugbeeinflussung: Unvollständiger I-Befehl: @1"
+
+#: advtrains/atc.lua:385
+msgid "ATC command parse error: Unknown command: @1"
+msgstr "Zugbeeinflussung: Unbekannter Befehl: @1"
+
+#: advtrains/copytool.lua:8
+msgid ""
+"Train copy/paste tool\n"
+"\n"
+"Left-click: copy train\n"
+"Right-click: paste train"
+msgstr ""
+"Werkzeug zur Erstellung von Zugkopien\n"
+"\n"
+"Linksklick: Zug ins Clipboard kopieren\n"
+"Right-click: Kopierten Zug einfügen"
+
+#: advtrains/copytool.lua:29
+msgid "You do not have the @1 privilege."
+msgstr "Ihnen fehlt das „@1“-Privileg."
+
+#: advtrains/copytool.lua:41
+msgid "The track you are trying to place the wagon on is not long enough."
+msgstr "Das Gleis, auf dem der Waggon platziert werden woll, ist zu kurz."
+
+#: advtrains/copytool.lua:47
+msgid "The clipboard couldn't access the metadata. Paste failed."
+msgstr ""
+"Wegen des fehlgeschlagenen Zugriffs auf die Metadaten konnte eine Kopie des "
+"Zuges nicht eingefügt werden."
+
+#: advtrains/copytool.lua:52 advtrains/copytool.lua:57
+msgid "The clipboard is empty."
+msgstr "Das Clipboard ist leer."
+
+#: advtrains/copytool.lua:74
+msgid "Back of train would end up off track, cancelling."
+msgstr "Der hinterer Teil dez Zuges wäre nicht auf dem Gleis."
+
+#: advtrains/copytool.lua:92
+msgid "No such lua entity."
+msgstr ""
+"Sie zeigen nicht auf einem Objekt, das mit diesem Werkzeug kopiert werden "
+"kann."
+
+#: advtrains/copytool.lua:98
+msgid "No such wagon: @1."
+msgstr "Es gibt keinen mit „@1“ identifizierbaren Waggon."
+
+#: advtrains/copytool.lua:104
+msgid "No such train: @1."
+msgstr "Es gibt keinen mit „@1“ identifizierbaren Zug."
+
+#: advtrains/copytool.lua:176
+msgid "The clipboard couldn't access the metadata. Copy failed."
+msgstr ""
+"Wegen des fehlgeschlagenen Zugriffs auf die Metadaten konnte der Zug nicht "
+"kopiert werden."
+
+#: advtrains/copytool.lua:180
+msgid "Train copied."
+msgstr "Der Zug wurde Kopiert."
+
+#: advtrains/couple.lua:28
+msgid "Buffer and Chain Coupler"
+msgstr "Schraubenkupplung"
+
+#: advtrains/couple.lua:29
+msgid "Scharfenberg Coupler"
+msgstr "Scharfenbergkupplung"
+
+#: advtrains/couple.lua:185
+msgid ""
+"You are not allowed to couple trains without the train_operator privilege."
+msgstr "Sie dürfen ohne das „train_operator“-Privileg keine Züge ankuppeln."
+
+#: advtrains/couple.lua:329 advtrains/couple.lua:333
+msgid "<No coupler>"
+msgstr "<Keine Kupplung vorhanden>"
+
+#: advtrains/couple.lua:334
+msgid "Can not couple: The couplers of the trains do not match (@1 and @2)."
+msgstr "Die Kupplungen der Züge passen nicht zueinander (@1 und @2)."
+
+#: advtrains/craft_items.lua:3
+msgid "Boiler"
+msgstr ""
+
+#: advtrains/craft_items.lua:9
+msgid "Driver's cab"
+msgstr "Führerstand"
+
+#: advtrains/craft_items.lua:15
+msgid "Wheel"
+msgstr ""
+
+#: advtrains/craft_items.lua:21
+msgid "Chimney"
+msgstr ""
+
+#: advtrains/misc_nodes.lua:16
+msgid "@1 Platform (low)"
+msgstr "Niedriger @1-Bahnsteig"
+
+#: advtrains/misc_nodes.lua:33
+msgid "@1 Platform (high)"
+msgstr "Hoher @1-Bahnsteig"
+
+#: advtrains/misc_nodes.lua:59
+msgid "@1 Platform (45 degree)"
+msgstr "Hoher @1-Bahnsteig (45°)"
+
+#: advtrains/misc_nodes.lua:81
+msgid "@1 Platform (low, 45 degree)"
+msgstr "Niedriger @1-Bahnsteig (45°)"
+
+#: advtrains/protection.lua:7
+msgid "Can place, remove and operate trains"
+msgstr ""
+
+#: advtrains/protection.lua:12
+msgid ""
+"Can place, remove and operate any train, regardless of owner, whitelist, or "
+"protection"
+msgstr ""
+
+#: advtrains/protection.lua:18
+msgid "Can place and dig tracks in unprotected areas"
+msgstr ""
+
+#: advtrains/protection.lua:24
+msgid "Can operate turnouts and signals in unprotected areas"
+msgstr ""
+
+#: advtrains/protection.lua:148
+msgid ""
+"You are not allowed to build near tracks without the track_builder privilege."
+msgstr ""
+"Sie dürfen ohne das „track_builder“-Privileg nicht in der Nähe von Gleisen "
+"bauen."
+
+#: advtrains/protection.lua:148
+msgid ""
+"You are not allowed to build tracks without the track_builder privilege."
+msgstr "Sie dürfen ohne das „track_builder“-Privileg kein Gleis bauen."
+
+#: advtrains/protection.lua:153
+msgid "You are not allowed to build near tracks at this protected position."
+msgstr "Sie dürfen an geschützten Stellen nicht in der Nähe von Gleisen bauen."
+
+#: advtrains/protection.lua:153
+msgid "You are not allowed to build tracks at this protected position."
+msgstr "Sie dürfen an geschützten Stellen kein Gleis bauen."
+
+#: advtrains/protection.lua:184
+msgid ""
+"You are not allowed to operate turnouts and signals without the "
+"railway_operator privilege."
+msgstr ""
+"Sie dürfen ohne das „railway_operator“-Privileg keine Bahnanlage operieren."
+
+#: advtrains/signals.lua:63
+msgid "Lampless Signal"
+msgstr "Mechanisches Signal"
+
+#: advtrains/signals.lua:127
+msgid "Signal"
+msgstr "Lichtsignal"
+
+#: advtrains/signals.lua:191
+msgid "Wallmounted Signal (left)"
+msgstr "An der linken Seite montiertes Signal"
+
+#: advtrains/signals.lua:192
+msgid "Wallmounted Signal (right)"
+msgstr "An der rechten Seite montiertes Signal"
+
+#: advtrains/signals.lua:193
+msgid "Wallmounted Signal (top)"
+msgstr "An der Decke montiertes Signal"
+
+#: advtrains/signals.lua:281 advtrains/signals.lua:322
+msgid "Andrew's Cross"
+msgstr "Andreaskreuz"
+
+#: advtrains/trackplacer.lua:313
+msgid ""
+"Track Worker Tool\n"
+"\n"
+"Left-click: change rail type (straight/curve/switch)\n"
+"Right-click: rotate object"
+msgstr ""
+"Gleiswerkzeug\n"
+"\n"
+"Linksklick: Gleistyp ändern\n"
+"Rechtsklick: Objekt drehen"
+
+#: advtrains/trackplacer.lua:340 advtrains/trackplacer.lua:377
+msgid "This node can't be rotated using the trackworker."
+msgstr "Dieser Block kann nicht mit dem Gleiswerkzeug gedreht werden."
+
+#: advtrains/trackplacer.lua:350
+msgid "This track can not be rotated."
+msgstr "Dieses Gleis kann nicht gedreht werden."
+
+#: advtrains/trackplacer.lua:404
+msgid "This node can't be changed using the trackworker."
+msgstr "Dieser Block kann nicht mit dem Gleiswerkzeug bearbeitet werden."
+
+#: advtrains/trackplacer.lua:414
+msgid "This track can not be changed."
+msgstr "Dieses Gleis kann nicht geändert werden."
+
+#: advtrains/tracks.lua:449
+msgid "This track can not be removed."
+msgstr "Dieses Gleis kann nicht entfernt werden."
+
+#: advtrains/tracks.lua:616
+msgid "Position is occupied by a train."
+msgstr "Ein Zug steht an dieser Position."
+
+#: advtrains/tracks.lua:622
+msgid "There's a Track Circuit Break here."
+msgstr "Hier ist eine Gleisabschnittsgrenze (TCB)."
+
+#: advtrains/tracks.lua:626
+msgid "There's a Signal Influence Point here."
+msgstr "Hier ist ein Signal-Beeinflussungspunkt."
+
+#: advtrains/tracks.lua:637
+msgid "@1 Slope"
+msgstr "@1 Steigung"
+
+#: advtrains/tracks.lua:648 advtrains/tracks.lua:653
+msgid "Can't place slope: not pointing at node."
+msgstr "Es kann nicht platziert werden: Sie zeigen nicht auf einem Block."
+
+#: advtrains/tracks.lua:658
+msgid "Can't place slope: space occupied."
+msgstr "Es kann nicht platziert werden: Diese Position ist besetzt."
+
+#: advtrains/tracks.lua:711
+msgid "Can't place slope: Not enough slope items left (@1 required)."
+msgstr ""
+"Es kann nicht platziert werden: Sie haben nicht genug Steigungsblöcke, es "
+"werden insgesamt @1 benötigt."
+
+#: advtrains/tracks.lua:714
+msgid "Can't place slope: There's no slope of length @1."
+msgstr ""
+"Es kann nicht platziert werden: die Steigung der Länge @1 ist nicht "
+"definiert."
+
+#: advtrains/tracks.lua:721
+msgid "Can't place slope: no supporting node at upper end."
+msgstr ""
+"Es kann nicht platziert werden: es gibt keinen unterstützenden Block am Ende "
+"der Steigung."
+
+#: advtrains/trainhud.lua:305
+msgid "OVERRUN RED SIGNAL! Examine situation and reverse train to move again."
+msgstr ""
+
+#: advtrains/wagons.lua:179
+msgid "This wagon is owned by @1, you can't destroy it."
+msgstr "Dieser Waggon gehört @1, Sie dürfen ihn nicht abbauen."
+
+#: advtrains/wagons.lua:203
+msgid "The wagon's inventory is not empty."
+msgstr "Das Inventar dieses Waggons ist nicht leer."
+
+#: advtrains/wagons.lua:210
+msgid "Wagon needs to be decoupled from other wagons in order to destroy it."
+msgstr "Der Waggon muss abgekoppelt sein, damit Sie ihn abbauen können."
+
+#: advtrains/wagons.lua:216
+msgid ""
+"Warning: If you destroy this wagon, you only get some steel back! If you are "
+"sure, hold Sneak and left-click the wagon."
+msgstr ""
+"Warnung: Durch den Abbau des Waggons erhalten Sie nur etwas Stahl zurück. "
+"Nutzen Sie Schleichen+Linksklick, um dem Waggon abzubauen."
+
+#: advtrains/wagons.lua:649 advtrains/wagons.lua:850
+msgid "Show Inventory"
+msgstr "Inventar Zeigen"
+
+#: advtrains/wagons.lua:652
+msgid "Onboard Computer"
+msgstr ""
+
+#: advtrains/wagons.lua:655 advtrains/wagons.lua:1328
+msgid "Wagon properties"
+msgstr "Waggon-Einstellungen"
+
+#: advtrains/wagons.lua:658
+msgid "Get off"
+msgstr "Aussteigen"
+
+#: advtrains/wagons.lua:661
+msgid "Get off (forced)"
+msgstr "Ausstieg zwingen"
+
+#: advtrains/wagons.lua:663
+msgid "(Doors closed)"
+msgstr "(Türen geschlossen)"
+
+#: advtrains/wagons.lua:692
+msgid "This wagon has no seats."
+msgstr "In diesem Waggon ist kein Sitzplatz vorhanden."
+
+#: advtrains/wagons.lua:703
+msgid "This wagon is full."
+msgstr "Der Waggon ist voll."
+
+#: advtrains/wagons.lua:706
+msgid "Doors are closed! (Try holding sneak key!)"
+msgstr "Die Türen sind geschlossen."
+
+#: advtrains/wagons.lua:712
+msgid "You can't get on this wagon."
+msgstr "Sie können nicht in diesen Waggon einsteigen."
+
+#: advtrains/wagons.lua:838
+msgid "Select seat:"
+msgstr "Wählen Sie einen Sitzplatz aus:"
+
+#: advtrains/wagons.lua:880
+msgid "Save wagon properties"
+msgstr "Waggon-Einstellungen speichern"
+
+#: advtrains/wagons.lua:965
+msgid "Text displayed outside on train"
+msgstr "Äußere Anzeige"
+
+#: advtrains/wagons.lua:966
+msgid "Text displayed inside train"
+msgstr "Innere Anzeige"
+
+#: advtrains/wagons.lua:967
+msgid "Line"
+msgstr "Linie"
+
+#: advtrains/wagons.lua:968
+msgid "Routingcode"
+msgstr ""
+
+#: advtrains/wagons.lua:1241
+msgid ""
+"Doors are closed. Use Sneak+rightclick to ignore the closed doors and get "
+"off."
+msgstr ""
+"Die Türen sind geschlossen. Nutzen Sie Schleichen+Rechtsklick, um trotz "
+"geschlossener Türen auszusteigen."
+
+#: advtrains/wagons.lua:1250
+msgid "You are not allowed to access the driver stand."
+msgstr "Sie haben keinen Zugang zum Führerstand."
+
+#: advtrains_interlocking/tsr_rail.lua:13
+msgid "Point speed restriction: @1"
+msgstr "Geschwindigkeitskontrolle: @1"
+
+#: advtrains_interlocking/tsr_rail.lua:14
+msgid "Set point speed restriction:"
+msgstr ""
+
+#: advtrains_interlocking/tsr_rail.lua:30
+msgid "You are not allowed to configure this track without the @1 privilege."
+msgstr "Sie dürfen ohne das „@1“-Privileg dieses Gleis nicht konfigurieren."
+
+#: advtrains_interlocking/tsr_rail.lua:34
+#: advtrains_line_automation/stoprail.lua:31
+#: advtrains_line_automation/stoprail.lua:76
+msgid "You are not allowed to configure this track."
+msgstr "Sie dürfen dieses Gleis nicht konfigurieren."
+
+#: advtrains_interlocking/tsr_rail.lua:64
+msgid "Point Speed Restriction Track"
+msgstr "Geschwindigkeitskontrollgleis"
+
+#: advtrains_line_automation/stoprail.lua:54
+msgid "Station Code"
+msgstr "Kennzeichen der Haltestelle"
+
+#: advtrains_line_automation/stoprail.lua:55
+msgid "Station Name"
+msgstr "Name der Haltestelle"
+
+#: advtrains_line_automation/stoprail.lua:56
+msgid "Door Delay"
+msgstr "Zeit für die Türschließung"
+
+#: advtrains_line_automation/stoprail.lua:57
+msgid "Dep. Speed"
+msgstr "Zielgeschwindigkeit bei Abfahrt"
+
+#: advtrains_line_automation/stoprail.lua:58 advtrains_train_track/init.lua:11
+#: advtrains_train_track/init.lua:156
+msgid "Track"
+msgstr "Gleis"
+
+#: advtrains_line_automation/stoprail.lua:59
+msgid "Stop Time"
+msgstr "Wartezeit"
+
+#: advtrains_line_automation/stoprail.lua:60
+msgid "Door Side"
+msgstr "Türseite"
+
+#: advtrains_line_automation/stoprail.lua:62
+msgid "Reverse train"
+msgstr "Zug Umkehren"
+
+#: advtrains_line_automation/stoprail.lua:63
+msgid "Kick out passengers"
+msgstr "Fahrgäste zum Ausstieg zwingen"
+
+#: advtrains_line_automation/stoprail.lua:97
+msgid "Station code \"@1\" already exists and is owned by @2."
+msgstr ""
+"Die Haltestelle mit dem Kennzeichen „@1“ ist bereits vorhanden und wird von "
+"@2 verwaltet."
+
+#: advtrains_line_automation/stoprail.lua:111
+msgid "This station is owned by @1. You are not allowed to edit its name."
+msgstr ""
+"Diese Haltestelle wird von @1 verwaltet. Sie dürfen sie nicht umbenennen."
+
+#: advtrains_line_automation/stoprail.lua:221
+msgid "Station/Stop Track"
+msgstr "Gleis zur Kennzeichnung einer Haltestelle"
+
+#: advtrains_luaautomation/active_common.lua:17
+msgid "Unconfigured LuaATC component"
+msgstr "Nicht konfiguierter LuaATC-Bauteil"
+
+#: advtrains_luaautomation/active_common.lua:46
+msgid "LuaATC Environment"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:49
+msgid "Clear Local Environment"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:50
+msgid "Code"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:64
+msgid ""
+"You are not allowed to configure this LuaATC component without the @1 "
+"privilege."
+msgstr ""
+"Sie dürfen ohne das „@1“-Privileg diesen LuaATC-Bauteil nicht konfigurieren."
+
+#: advtrains_luaautomation/active_common.lua:94
+msgid "LuaATC component assigned to environment '@1'"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:96
+msgid "LuaATC component assigned to an invalid environment"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:171
+msgid "LuaATC component with error: @1"
+msgstr "LuaATC-Bauteil mit Fehlermeldung: @1"
+
+#: advtrains_luaautomation/init.lua:13
+msgid ""
+"Can place and configure LuaATC components, including execute potentially "
+"harmful Lua code"
+msgstr ""
+"Kann LuaATC-Bauteile platzieren und konfigurieren (auch evtl. schädliche "
+"Programme ausführen)"
+
+#: advtrains_luaautomation/mesecon_controller.lua:211
+msgid "LuaATC Mesecon Controller"
+msgstr ""
+
+#: advtrains_luaautomation/operation_panel.lua:11
+msgid "LuaATC Operation Panel"
+msgstr ""
+
+#: advtrains_luaautomation/pcnaming.lua:28
+msgid ""
+"Passive Component Naming Tool\n"
+"\n"
+"Right-click to name a passive component."
+msgstr ""
+"PC-Benennungswerkzeug\n"
+"\n"
+"Rechtsklick zur Benennung der passiven Komponente."
+
+#: advtrains_luaautomation/pcnaming.lua:39
+msgid ""
+"You are not allowed to name LuaATC passive components without the @1 "
+"privilege."
+msgstr "Sie dürfen ohne das „@1“ keinen passiven LuaATC-Bauteil benennen."
+
+#: advtrains_luaautomation/pcnaming.lua:62
+msgid "Set name of component (empty to clear)"
+msgstr ""
+
+#: advtrains_train_industrial/init.lua:10
+#: advtrains_train_industrial/init.lua:49 advtrains_train_steam/init.lua:20
+#: advtrains_train_steam/init.lua:91
+msgid "Driver Stand (right)"
+msgstr "Führerstand Rechts"
+
+#: advtrains_train_industrial/init.lua:17
+#: advtrains_train_industrial/init.lua:56 advtrains_train_steam/init.lua:14
+#: advtrains_train_steam/init.lua:85
+msgid "Driver Stand (left)"
+msgstr "Führerstand Links"
+
+#: advtrains_train_industrial/init.lua:40
+msgid "Industrial Train Engine"
+msgstr "Industrielle Lokomotive"
+
+#: advtrains_train_industrial/init.lua:79
+msgid "Big Industrial Train Engine"
+msgstr "Große Industrielle Lokomotive"
+
+#: advtrains_train_industrial/init.lua:98
+msgid "Industrial tank wagon"
+msgstr "Tankwaggon"
+
+#: advtrains_train_industrial/init.lua:116
+msgid "Industrial wood wagon"
+msgstr "Holztransportwaggon"
+
+#: advtrains_train_japan/init.lua:4
+msgid "Japanese Train Inter-Wagon Connection"
+msgstr "Waggonzwischenverbindung Japanischer Personenzüge"
+
+#: advtrains_train_japan/init.lua:37
+msgid "Driver stand"
+msgstr "Führerstand"
+
+#: advtrains_train_japan/init.lua:101
+msgid "Japanese Train Engine"
+msgstr "Japanische Personenzug-Lokomotive"
+
+#: advtrains_train_japan/init.lua:176
+msgid "Japanese Train Wagon"
+msgstr "Japanischer Personenzug-Passagierwaggon"
+
+#: advtrains_train_steam/init.lua:75
+msgid "Steam Engine"
+msgstr "Dampflokomotive"
+
+#: advtrains_train_steam/init.lua:159
+msgid "Detailed Steam Engine"
+msgstr "Detaillierte Dampflokomotive"
+
+#: advtrains_train_steam/init.lua:206
+msgid "Passenger Wagon"
+msgstr "Passagierwaggon"
+
+#: advtrains_train_steam/init.lua:226
+msgid "Box Wagon"
+msgstr "Güterwaggon"
+
+#: advtrains_train_subway/init.lua:144
+msgid "Subway Passenger Wagon"
+msgstr "U-Bahn-Waggon"
+
+#: advtrains_train_track/init.lua:31
+msgid "Y-turnout"
+msgstr "Y-Weiche"
+
+#: advtrains_train_track/init.lua:49
+msgid "3-way turnout"
+msgstr "Dreiwegweiche"
+
+#: advtrains_train_track/init.lua:69
+msgid "Perpendicular Diamond Crossing Track"
+msgstr "Kreuzung mit zueinander orthogonalen Gleisen"
+
+#: advtrains_train_track/init.lua:91
+msgid "90+Angle Diamond Crossing Track"
+msgstr "Kreuzung mit einem achsenparallelen Gleis"
+
+#: advtrains_train_track/init.lua:132
+msgid "Diagonal Diamond Crossing Track"
+msgstr "Diagonale Gleiskreuzung"
+
+#: advtrains_train_track/init.lua:179
+msgid "Bumper"
+msgstr "Prellbock"
+
+#: advtrains_train_track/init.lua:201
+msgid "ATC controller"
+msgstr "Zugbeeinflussungsgleis"
+
+#: advtrains_train_track/init.lua:317
+msgid "Unloading Track"
+msgstr "Abladungsgleis"
+
+#: advtrains_train_track/init.lua:342
+msgid "Loading Track"
+msgstr "Beladungsgleis"
+
+#: advtrains_train_track/init.lua:406
+msgid "Detector Rail"
+msgstr "Detektorgleis"
+
+#~ msgid ""
+#~ "ATC controller, mode @1\n"
+#~ "Channel: @2"
+#~ msgstr ""
+#~ "Zugbeeinflussungsgleis in Betriebsart „@1“\n"
+#~ "Kanal: @2"
+
+#~ msgid "Access to @1"
+#~ msgstr "Zugang zu @1"
+
+#~ msgid "Can't get on: wagon full or doors closed!"
+#~ msgstr ""
+#~ "Sie können nicht einsteigen: der Waggon ist voll oder die Türen sind "
+#~ "geschlossen."
+
+#~ msgid "Can't place: protected position!"
+#~ msgstr "Es kann nicht platziert werden: diese Position ist geschützt."
+
+#~ msgid "Default Seat"
+#~ msgstr "Standardsitzplatz"
+
+#~ msgid "Default Seat (driver stand)"
+#~ msgstr "Standardsitzplatz (Führerstand)"
+
+#~ msgid "Deprecated Track"
+#~ msgstr "ausrangiertes Gleis, nicht verwenden."
+
+#~ msgid "Lock couples"
+#~ msgstr "Kupplungen sperren"
+
+#~ msgid "Speed:"
+#~ msgstr "Geschw.:"
+
+#~ msgid "Target:"
+#~ msgstr "Zielges.:"
+
+#~ msgid "This position is protected!"
+#~ msgstr "Diese Position ist geschützt!"
+
+#~ msgid "Use Sneak+rightclick to bypass closed doors!"
+#~ msgstr ""
+#~ "Nutzen Sie Schleichen+Rechtsklick, um trotz geschlossener Türen "
+#~ "einzusteigen."
+
+#, fuzzy
+#~ msgid "You are not allowed to modify this protected track."
+#~ msgstr "Sie dürfen an geschützten Stellen kein Gleis bauen."
+
+#~ msgid ""
+#~ "You need to own at least one neighboring wagon to destroy this couple."
+#~ msgstr ""
+#~ "Sie müssen Besitzer eines angrenzenden Waggons sein, um hier abzukuppeln."
diff --git a/advtrains/po/fr.po b/advtrains/po/fr.po
new file mode 100644
index 0000000..c744d2c
--- /dev/null
+++ b/advtrains/po/fr.po
@@ -0,0 +1,728 @@
+msgid ""
+msgstr ""
+"Project-Id-Version: advtrains\n"
+"Report-Msgid-Bugs-To: advtrains-discuss@lists.sr.ht\n"
+"POT-Creation-Date: 2023-10-09 11:02+0200\n"
+"PO-Revision-Date: 2024-11-02 21:31+0100\n"
+"Last-Translator: Tanavit <tanavit@posto.ovh>\n"
+"Language-Team: French\n"
+"Language: fr\n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"X-Generator: Poedit 2.4.2\n"
+
+#: advtrains/atc.lua:109
+msgid "Unconfigured ATC controller"
+msgstr "Controlleur ATC, non-configuré"
+
+#: advtrains/atc.lua:150
+msgid ""
+"ATC controller, mode @1\n"
+"Command: @2"
+msgstr ""
+"Controlleur ATC, mode @1\n"
+"Commande : @2"
+
+#: advtrains/atc.lua:180
+msgid "Command"
+msgstr "Commande"
+
+#: advtrains/atc.lua:184
+msgid "Command (on)"
+msgstr "Commande (marche)"
+
+#: advtrains/atc.lua:187
+msgid "Digiline channel"
+msgstr "Canal Digiline"
+
+#: advtrains/atc.lua:189 advtrains_line_automation/stoprail.lua:65
+#: advtrains_luaautomation/active_common.lua:48
+msgid "Save"
+msgstr "Sauvegarder"
+
+#: advtrains/atc.lua:236
+msgid "ATC Reverse command warning: didn't reverse train, train moving."
+msgstr ""
+"Attention : Commande ATC de renversement impossible car le train se déplace."
+
+#: advtrains/atc.lua:248
+msgid "ATC Kick command warning: doors are closed."
+msgstr "Avertissement commande ATC Éjecter : portes closes."
+
+#: advtrains/atc.lua:252
+msgid "ATC Kick command warning: train moving."
+msgstr "Avertissement commande ATC Éjecter : train en mouvement."
+
+#: advtrains/atc.lua:322
+msgid "ATC command syntax error: I statement not closed: @1"
+msgstr "Erreur de syntaxe de commande ATC : instruction \"I\" incomplète : @1"
+
+#: advtrains/atc.lua:385
+msgid "ATC command parse error: Unknown command: @1"
+msgstr "Erreur d'analyse de commande ATC : Commande inconnue : @1"
+
+#: advtrains/copytool.lua:8
+msgid ""
+"Train copy/paste tool\n"
+"\n"
+"Left-click: copy train\n"
+"Right-click: paste train"
+msgstr ""
+"Outil de copie/collage de train\n"
+"\n"
+"Clic-Gauche : copie\n"
+"\n"
+"Clic-Droit : collage"
+
+#: advtrains/copytool.lua:29
+msgid "You do not have the @1 privilege."
+msgstr "Vous ne possédez pas le privilège \"@1\"."
+
+#: advtrains/copytool.lua:41
+msgid "The track you are trying to place the wagon on is not long enough."
+msgstr "La voie sur laquelle vous tentez de placer le wagon est trop courte."
+
+#: advtrains/copytool.lua:47
+msgid "The clipboard couldn't access the metadata. Paste failed."
+msgstr "Le presse-papier ne peut accéder aux métadonnées. Échec du collage."
+
+#: advtrains/copytool.lua:52 advtrains/copytool.lua:57
+msgid "The clipboard is empty."
+msgstr "Le presse-papier est vide."
+
+#: advtrains/copytool.lua:74
+msgid "Back of train would end up off track, cancelling."
+msgstr "La fin du train serait hors voie : annulation."
+
+#: advtrains/copytool.lua:92
+msgid "No such lua entity."
+msgstr "Pas de telle entité lua."
+
+#: advtrains/copytool.lua:98
+msgid "No such wagon: @1."
+msgstr "Pas de tel wagon : @1."
+
+#: advtrains/copytool.lua:104
+msgid "No such train: @1."
+msgstr "Pas de tel train : @1."
+
+#: advtrains/copytool.lua:176
+msgid "The clipboard couldn't access the metadata. Copy failed."
+msgstr "Le presse-papier ne peut accéder aux métadonnées. Échec de la copie."
+
+#: advtrains/copytool.lua:180
+msgid "Train copied."
+msgstr "Train copié."
+
+#: advtrains/couple.lua:28
+msgid "Buffer and Chain Coupler"
+msgstr "Attelage à tampon et vis"
+
+#: advtrains/couple.lua:29
+msgid "Scharfenberg Coupler"
+msgstr "Attelage Scharfenberg"
+
+#: advtrains/couple.lua:185
+msgid ""
+"You are not allowed to couple trains without the train_operator privilege."
+msgstr ""
+"Vous n'êtes pas autorisé à coupler des trains sans le privilège "
+"\"train_operator\"."
+
+#: advtrains/couple.lua:329 advtrains/couple.lua:333
+msgid "<No coupler>"
+msgstr "<Pas de coupleur>"
+
+#: advtrains/couple.lua:334
+msgid "Can not couple: The couplers of the trains do not match (@1 and @2)."
+msgstr ""
+"Accouplement impossible: les attelages des trains ne concordent pas (@1 et "
+"@2)."
+
+#: advtrains/craft_items.lua:3
+msgid "Boiler"
+msgstr "Chaudière à vapeur"
+
+#: advtrains/craft_items.lua:9
+msgid "Driver's cab"
+msgstr "Cabine de pilotage"
+
+#: advtrains/craft_items.lua:15
+msgid "Wheel"
+msgstr "Roue"
+
+#: advtrains/craft_items.lua:21
+msgid "Chimney"
+msgstr "Cheminée"
+
+#: advtrains/misc_nodes.lua:16
+msgid "@1 Platform (low)"
+msgstr "Quai @1 (bas)"
+
+#: advtrains/misc_nodes.lua:33
+msgid "@1 Platform (high)"
+msgstr "Quai @1 (haut)"
+
+#: advtrains/misc_nodes.lua:59
+msgid "@1 Platform (45 degree)"
+msgstr "Quai @1 (haut, 45°)"
+
+#: advtrains/misc_nodes.lua:81
+msgid "@1 Platform (low, 45 degree)"
+msgstr "Quai @1 (bas, 45°)"
+
+#: advtrains/protection.lua:7
+msgid "Can place, remove and operate trains"
+msgstr "Possibilité de poser, retirer ou opérer les trains"
+
+#: advtrains/protection.lua:12
+msgid ""
+"Can place, remove and operate any train, regardless of owner, whitelist, or "
+"protection"
+msgstr ""
+"Possibilité de poser, retirer ou opérer un quelconque train, indépendamment "
+"du propriétaire, de la liste blanche ou de protection"
+
+#: advtrains/protection.lua:18
+msgid "Can place and dig tracks in unprotected areas"
+msgstr "Possibilité de poser ou retirer des voies dans les zones non protégées"
+
+#: advtrains/protection.lua:24
+msgid "Can operate turnouts and signals in unprotected areas"
+msgstr ""
+"Possibilité d'opérer des embranchements et signaux dans les zones non "
+"protégées"
+
+#: advtrains/protection.lua:148
+msgid ""
+"You are not allowed to build near tracks without the track_builder privilege."
+msgstr ""
+"Vous ne pouvez pas construire à proximité d'une voie sans le privilège "
+"\"track_builder\" (?)"
+
+#: advtrains/protection.lua:148
+msgid ""
+"You are not allowed to build tracks without the track_builder privilege."
+msgstr ""
+"Vous ne pouvez pas construire une voie sans le privilège \"track_builder\"."
+
+#: advtrains/protection.lua:153
+msgid "You are not allowed to build near tracks at this protected position."
+msgstr ""
+"Vous ne pouvez pas construire à proximité d'une voie à cet emplacement "
+"protégé."
+
+#: advtrains/protection.lua:153
+msgid "You are not allowed to build tracks at this protected position."
+msgstr "Vous ne pouvez pas construire une voie à cet emplacement protégé."
+
+#: advtrains/protection.lua:184
+msgid ""
+"You are not allowed to operate turnouts and signals without the "
+"railway_operator privilege."
+msgstr ""
+"Vous ne pouvez pas actionner les aiguillages ou les signaux (privilège "
+"\"railway_operator\" manquant)"
+
+#: advtrains/signals.lua:63
+msgid "Lampless Signal"
+msgstr "Sémaphore"
+
+#: advtrains/signals.lua:127
+msgid "Signal"
+msgstr "Signal"
+
+#: advtrains/signals.lua:191
+msgid "Wallmounted Signal (left)"
+msgstr "Signal mural (gauche)"
+
+#: advtrains/signals.lua:192
+msgid "Wallmounted Signal (right)"
+msgstr "Signal mural (droit)"
+
+#: advtrains/signals.lua:193
+msgid "Wallmounted Signal (top)"
+msgstr "Signal mural (plafond)"
+
+#: advtrains/signals.lua:281 advtrains/signals.lua:322
+msgid "Andrew's Cross"
+msgstr "Croix de Saint André"
+
+#: advtrains/trackplacer.lua:313
+msgid ""
+"Track Worker Tool\n"
+"\n"
+"Left-click: change rail type (straight/curve/switch)\n"
+"Right-click: rotate object"
+msgstr ""
+"Outil \"Trackworker\"\n"
+"\n"
+"Clic-Gauche : change le type de rail (droit/courbé/aiguillage)\n"
+"\n"
+"Clic-Droit : tourne l'objet"
+
+#: advtrains/trackplacer.lua:340 advtrains/trackplacer.lua:377
+msgid "This node can't be rotated using the trackworker."
+msgstr "Ce nœud ne peut être tourné avec l'outil \"Trackworker\"."
+
+#: advtrains/trackplacer.lua:350
+msgid "This track can not be rotated."
+msgstr "Cette voie ne peut pas être tournée."
+
+#: advtrains/trackplacer.lua:404
+msgid "This node can't be changed using the trackworker."
+msgstr "Ce nœud ne peut être modifié avec l'outil \"Trackworker\"."
+
+#: advtrains/trackplacer.lua:414
+msgid "This track can not be changed."
+msgstr "Cette voie ne peut pas être modifiée."
+
+#: advtrains/tracks.lua:449
+msgid "This track can not be removed."
+msgstr "Cette voie ne peut pas être enlevée."
+
+#: advtrains/tracks.lua:616
+msgid "Position is occupied by a train."
+msgstr "Cet emplacement est occupé par un train."
+
+#: advtrains/tracks.lua:622
+msgid "There's a Track Circuit Break here."
+msgstr "Il y a un \"Track Circuit Break\" ici."
+
+#: advtrains/tracks.lua:626
+msgid "There's a Signal Influence Point here."
+msgstr "Il y a un \"Signal Influence Point\" ici."
+
+#: advtrains/tracks.lua:637
+msgid "@1 Slope"
+msgstr "Pente @1"
+
+#: advtrains/tracks.lua:648 advtrains/tracks.lua:653
+msgid "Can't place slope: not pointing at node."
+msgstr "Placement impossible : ne pointe pas un nœud."
+
+#: advtrains/tracks.lua:658
+msgid "Can't place slope: space occupied."
+msgstr "Placement impossible : espace occupé."
+
+#: advtrains/tracks.lua:711
+msgid "Can't place slope: Not enough slope items left (@1 required)."
+msgstr ""
+"Placement impossible : quantité insuffisante de voie pentue (@1 manquant)."
+
+#: advtrains/tracks.lua:714
+msgid "Can't place slope: There's no slope of length @1."
+msgstr "Placement impossible : il n'y a pas de voie pentue de longueur @1."
+
+#: advtrains/tracks.lua:721
+msgid "Can't place slope: no supporting node at upper end."
+msgstr "Placement impossible : pas de nœud d'appui à l'extrémité supérieure."
+
+#: advtrains/trainhud.lua:305
+msgid "OVERRUN RED SIGNAL! Examine situation and reverse train to move again."
+msgstr ""
+"Franchissement de signal rouge : examinez la situation et inversez le sens "
+"de marche du train."
+
+#: advtrains/wagons.lua:179
+msgid "This wagon is owned by @1, you can't destroy it."
+msgstr "Ce wagon est la propriété de @1, vous ne pouvez pas le détruire."
+
+#: advtrains/wagons.lua:203
+msgid "The wagon's inventory is not empty."
+msgstr "Le stock de ce wagon n'est pas vide."
+
+#: advtrains/wagons.lua:210
+msgid "Wagon needs to be decoupled from other wagons in order to destroy it."
+msgstr ""
+"Les wagons doivent être désaccouplés des autres pour pouvoir être détruits."
+
+#: advtrains/wagons.lua:216
+msgid ""
+"Warning: If you destroy this wagon, you only get some steel back! If you are "
+"sure, hold Sneak and left-click the wagon."
+msgstr ""
+"Attention: Si vous détruisez ce wagon, vous ne récupérerez que de la "
+"ferraille ! Si vous êtes sûr de vous, appuyez la touche \"Marcher lentement "
+"(Sneak)\" et Clic-Gauche."
+
+#: advtrains/wagons.lua:649 advtrains/wagons.lua:850
+msgid "Show Inventory"
+msgstr "Montrer le stock"
+
+#: advtrains/wagons.lua:652
+msgid "Onboard Computer"
+msgstr "Ordinateur embarqué"
+
+#: advtrains/wagons.lua:655 advtrains/wagons.lua:1328
+msgid "Wagon properties"
+msgstr "Propriétés du wagon"
+
+#: advtrains/wagons.lua:658
+msgid "Get off"
+msgstr "Débarquer"
+
+#: advtrains/wagons.lua:661
+msgid "Get off (forced)"
+msgstr "Débarquer (de force)"
+
+#: advtrains/wagons.lua:663
+msgid "(Doors closed)"
+msgstr "(Portes closes)"
+
+#: advtrains/wagons.lua:692
+msgid "This wagon has no seats."
+msgstr "Ce wagon n'a pas de siège."
+
+#: advtrains/wagons.lua:703
+msgid "This wagon is full."
+msgstr "Ce wagon est plein."
+
+#: advtrains/wagons.lua:706
+msgid "Doors are closed! (Try holding sneak key!)"
+msgstr "Portes closes : (Essayez la \"sneak key\"!\")"
+
+#: advtrains/wagons.lua:712
+msgid "You can't get on this wagon."
+msgstr "Montée impossible dans ce wagon."
+
+#: advtrains/wagons.lua:838
+msgid "Select seat:"
+msgstr "Choisir le siège :"
+
+#: advtrains/wagons.lua:880
+msgid "Save wagon properties"
+msgstr "Sauvegarder les propriétés du wagon"
+
+#: advtrains/wagons.lua:965
+msgid "Text displayed outside on train"
+msgstr "Texte affiché à l'extérieur du train"
+
+#: advtrains/wagons.lua:966
+msgid "Text displayed inside train"
+msgstr "Texte affiché à l'intérieur du train"
+
+#: advtrains/wagons.lua:967
+msgid "Line"
+msgstr "Ligne"
+
+#: advtrains/wagons.lua:968
+msgid "Routingcode"
+msgstr "Code de routage"
+
+#: advtrains/wagons.lua:1241
+msgid ""
+"Doors are closed. Use Sneak+rightclick to ignore the closed doors and get "
+"off."
+msgstr ""
+"Portes closes ! Utilisez \"Marcher lentement (Sneak)\" et Clic-Droit pour "
+"franchir les portes et débarquer."
+
+#: advtrains/wagons.lua:1250
+msgid "You are not allowed to access the driver stand."
+msgstr "Accès interdit au poste de pilotage."
+
+#: advtrains_interlocking/tsr_rail.lua:13
+msgid "Point speed restriction: @1"
+msgstr "Point de limitation de vitesse : @1"
+
+#: advtrains_interlocking/tsr_rail.lua:14
+msgid "Set point speed restriction:"
+msgstr "Placez un point de limitation de vitesse :"
+
+#: advtrains_interlocking/tsr_rail.lua:30
+msgid "You are not allowed to configure this track without the @1 privilege."
+msgstr "Vous n'êtes pas autorisé à configurer cette voie sans le privilège @1."
+
+#: advtrains_interlocking/tsr_rail.lua:34
+#: advtrains_line_automation/stoprail.lua:31
+#: advtrains_line_automation/stoprail.lua:76
+msgid "You are not allowed to configure this track."
+msgstr "Vous n'êtes pas autorisé à configurer cette voie."
+
+#: advtrains_interlocking/tsr_rail.lua:64
+msgid "Point Speed Restriction Track"
+msgstr "Voie de point de limitation de vitesse"
+
+#: advtrains_line_automation/stoprail.lua:54
+msgid "Station Code"
+msgstr "Code de Station"
+
+#: advtrains_line_automation/stoprail.lua:55
+msgid "Station Name"
+msgstr "Nom de Station"
+
+#: advtrains_line_automation/stoprail.lua:56
+msgid "Door Delay"
+msgstr "Durée d'ouverture des portes"
+
+#: advtrains_line_automation/stoprail.lua:57
+msgid "Dep. Speed"
+msgstr "Vitesse de départ"
+
+#: advtrains_line_automation/stoprail.lua:58 advtrains_train_track/init.lua:11
+#: advtrains_train_track/init.lua:156
+msgid "Track"
+msgstr "Voie"
+
+#: advtrains_line_automation/stoprail.lua:59
+msgid "Stop Time"
+msgstr "Durée d'arrêt"
+
+#: advtrains_line_automation/stoprail.lua:60
+msgid "Door Side"
+msgstr "Coté d'ouvertures des portes"
+
+#: advtrains_line_automation/stoprail.lua:62
+msgid "Reverse train"
+msgstr "Inversion du sens de marche"
+
+#: advtrains_line_automation/stoprail.lua:63
+msgid "Kick out passengers"
+msgstr "Éjecter les passagers"
+
+#: advtrains_line_automation/stoprail.lua:97
+msgid "Station code \"@1\" already exists and is owned by @2."
+msgstr "Le code de station \"@1\" existe et est possédé par @2."
+
+#: advtrains_line_automation/stoprail.lua:111
+msgid "This station is owned by @1. You are not allowed to edit its name."
+msgstr ""
+"Cette station est la propriété de @1. Vous n'êtes pas autorisé à modifier "
+"son nom."
+
+#: advtrains_line_automation/stoprail.lua:221
+msgid "Station/Stop Track"
+msgstr "Voie d'arrêt en station"
+
+#: advtrains_luaautomation/active_common.lua:17
+msgid "Unconfigured LuaATC component"
+msgstr "Composant LuaATC non configuré"
+
+#: advtrains_luaautomation/active_common.lua:46
+msgid "LuaATC Environment"
+msgstr "Environnement LuaATC"
+
+#: advtrains_luaautomation/active_common.lua:49
+msgid "Clear Local Environment"
+msgstr "Effacer l'environnement LuaATC"
+
+#: advtrains_luaautomation/active_common.lua:50
+msgid "Code"
+msgstr "Code"
+
+#: advtrains_luaautomation/active_common.lua:64
+msgid ""
+"You are not allowed to configure this LuaATC component without the @1 "
+"privilege."
+msgstr "Vous ne pouvez configurer ce composant LuaATC sans le privilege @1."
+
+#: advtrains_luaautomation/active_common.lua:94
+msgid "LuaATC component assigned to environment '@1'"
+msgstr "Composant LuaATC assigné à l'environnement '@1'"
+
+#: advtrains_luaautomation/active_common.lua:96
+msgid "LuaATC component assigned to an invalid environment"
+msgstr "Composant LuaATC assigné à un environnement invalide"
+
+#: advtrains_luaautomation/active_common.lua:171
+msgid "LuaATC component with error: @1"
+msgstr "Erreur @1 du composant LuaATC"
+
+#: advtrains_luaautomation/init.lua:13
+msgid ""
+"Can place and configure LuaATC components, including execute potentially "
+"harmful Lua code"
+msgstr ""
+"Permet le placement et la configuration de composants LuaATC avec risque "
+"d'exécution de code Lua dangereux"
+
+#: advtrains_luaautomation/mesecon_controller.lua:211
+msgid "LuaATC Mesecon Controller"
+msgstr "Commande Mesecon de LuaATC"
+
+#: advtrains_luaautomation/operation_panel.lua:11
+msgid "LuaATC Operation Panel"
+msgstr "Panneau de commande de LuaATC"
+
+#: advtrains_luaautomation/pcnaming.lua:28
+msgid ""
+"Passive Component Naming Tool\n"
+"\n"
+"Right-click to name a passive component."
+msgstr ""
+"Outil de nommage de composant passif\n"
+"\n"
+"Clic-Droit pour nommer un composant passif."
+
+#: advtrains_luaautomation/pcnaming.lua:39
+msgid ""
+"You are not allowed to name LuaATC passive components without the @1 "
+"privilege."
+msgstr "Vous ne pouvez nommer un composant LuaATC passif sans le privilege @1."
+
+#: advtrains_luaautomation/pcnaming.lua:62
+msgid "Set name of component (empty to clear)"
+msgstr "Nommer le composant (chaîne vide pour effacer)"
+
+#: advtrains_train_industrial/init.lua:10
+#: advtrains_train_industrial/init.lua:49 advtrains_train_steam/init.lua:20
+#: advtrains_train_steam/init.lua:91
+msgid "Driver Stand (right)"
+msgstr "Poste de pilotage (droit)"
+
+#: advtrains_train_industrial/init.lua:17
+#: advtrains_train_industrial/init.lua:56 advtrains_train_steam/init.lua:14
+#: advtrains_train_steam/init.lua:85
+msgid "Driver Stand (left)"
+msgstr "Poste de pilotage (gauche)"
+
+#: advtrains_train_industrial/init.lua:40
+msgid "Industrial Train Engine"
+msgstr "Locomotive industrielle"
+
+#: advtrains_train_industrial/init.lua:79
+msgid "Big Industrial Train Engine"
+msgstr "Grosse locomotive industrielle"
+
+#: advtrains_train_industrial/init.lua:98
+msgid "Industrial tank wagon"
+msgstr "Wagon-citerne industriel"
+
+#: advtrains_train_industrial/init.lua:116
+msgid "Industrial wood wagon"
+msgstr "Wagon grumier industriel"
+
+#: advtrains_train_japan/init.lua:4
+msgid "Japanese Train Inter-Wagon Connection"
+msgstr "Passage inter-voiture de train Japonais"
+
+#: advtrains_train_japan/init.lua:37
+msgid "Driver stand"
+msgstr "Poste de pilotage"
+
+#: advtrains_train_japan/init.lua:101
+msgid "Japanese Train Engine"
+msgstr "Motrice Japonaise"
+
+#: advtrains_train_japan/init.lua:176
+msgid "Japanese Train Wagon"
+msgstr "Voiture Japonaise"
+
+#: advtrains_train_steam/init.lua:75
+msgid "Steam Engine"
+msgstr "Locomotive à vapeur"
+
+#: advtrains_train_steam/init.lua:159
+msgid "Detailed Steam Engine"
+msgstr "Locomotive à vapeur complexe"
+
+#: advtrains_train_steam/init.lua:206
+msgid "Passenger Wagon"
+msgstr "Voiture passager"
+
+#: advtrains_train_steam/init.lua:226
+msgid "Box Wagon"
+msgstr "Wagon de frêt"
+
+#: advtrains_train_subway/init.lua:144
+msgid "Subway Passenger Wagon"
+msgstr "Voiture de Métropolitain"
+
+#: advtrains_train_track/init.lua:31
+msgid "Y-turnout"
+msgstr "Embranchement en Y"
+
+#: advtrains_train_track/init.lua:49
+msgid "3-way turnout"
+msgstr "Embranchement triple"
+
+#: advtrains_train_track/init.lua:69
+msgid "Perpendicular Diamond Crossing Track"
+msgstr "Croisement perpendiculaire"
+
+#: advtrains_train_track/init.lua:91
+msgid "90+Angle Diamond Crossing Track"
+msgstr "Croisement perpendiculo-diagonal"
+
+#: advtrains_train_track/init.lua:132
+msgid "Diagonal Diamond Crossing Track"
+msgstr "Croisement diagonal"
+
+#: advtrains_train_track/init.lua:179
+msgid "Bumper"
+msgstr "Heurtoir"
+
+#: advtrains_train_track/init.lua:201
+msgid "ATC controller"
+msgstr "Controlleur ATC"
+
+#: advtrains_train_track/init.lua:317
+msgid "Unloading Track"
+msgstr "Voie de Déchargement"
+
+#: advtrains_train_track/init.lua:342
+msgid "Loading Track"
+msgstr "Voie de Chargement"
+
+#: advtrains_train_track/init.lua:406
+msgid "Detector Rail"
+msgstr "Voie détectrice"
+
+#~ msgid ""
+#~ "ATC controller, mode @1\n"
+#~ "Channel: @2"
+#~ msgstr ""
+#~ "Controlleur ATC, mode @1\n"
+#~ "Canal : @2"
+
+#~ msgid "Access to @1"
+#~ msgstr "Accès à @1"
+
+#~ msgid "Can't get on: wagon full or doors closed!"
+#~ msgstr ""
+#~ "Embarquement impossible : le wagon est plein ou ses portes sont closes !"
+
+#~ msgid "Can't place: protected position!"
+#~ msgstr "Placement impossible : emplacement protégé"
+
+#~ msgid "Default Seat"
+#~ msgstr "Siège par défaut"
+
+#~ msgid "Default Seat (driver stand)"
+#~ msgstr "Siège par défaut (poste de pilotage)"
+
+#~ msgid "Deprecated Track"
+#~ msgstr "Voie déconseillée"
+
+#~ msgid "Lock couples"
+#~ msgstr "Verrouiller l'accouplement"
+
+#~ msgid "Speed:"
+#~ msgstr "Vitesse : "
+
+#~ msgid "Target:"
+#~ msgstr "Destination : "
+
+#, fuzzy
+#~ msgid "This node can't be rotated using the trackworker,"
+#~ msgstr "Ce nœud ne peut être tourné avec l'outil \"Trackworker\" !"
+
+#~ msgid "This position is protected!"
+#~ msgstr "Cet emplacement est protégé !"
+
+#~ msgid "Use Sneak+rightclick to bypass closed doors!"
+#~ msgstr ""
+#~ "Utilisez \"Marcher lentement (Sneak)\" et Clic-Droit pour franchir les "
+#~ "portes closes !"
+
+#, fuzzy
+#~ msgid "You are not allowed to modify this protected track."
+#~ msgstr "Vous ne pouvez pas construire une voie à cet emplacement protégé"
+
+#~ msgid ""
+#~ "You need to own at least one neighboring wagon to destroy this couple."
+#~ msgstr ""
+#~ "Vous devez être propriétaire d'au moins un wagon voisin pour supprimer "
+#~ "cet attelage."
diff --git a/advtrains/po/update-translations.sh b/advtrains/po/update-translations.sh
new file mode 100755
index 0000000..3a56c7c
--- /dev/null
+++ b/advtrains/po/update-translations.sh
@@ -0,0 +1,28 @@
+#!/bin/sh
+# NOTE: Please make sure you also have basic_trains installed, as it uses attrans for historical reasons
+
+PODIR=`dirname "$0"`
+ATDIR="$PODIR/../.."
+BTDIR="$ATDIR/../basic_trains"
+POTFILE="$PODIR/advtrains.pot"
+
+xgettext \
+ -D "$ATDIR" \
+ -D "$BTDIR" \
+ -d advtrains \
+ -o "$POTFILE" \
+ -p . \
+ -L lua \
+ --from-code=UTF-8 \
+ --sort-by-file \
+ --keyword='attrans' \
+ --keyword='S' \
+ --package-name='advtrains' \
+ --msgid-bugs-address='advtrains-discuss@lists.sr.ht' \
+ `find $ATDIR $BTDIR -name '*.lua' -printf '%P\n'` \
+ &&
+for i in "$PODIR"/*.po; do
+ msgmerge -U \
+ --sort-by-file \
+ $i "$POTFILE"
+done
diff --git a/advtrains/po/zh_CN.po b/advtrains/po/zh_CN.po
new file mode 100644
index 0000000..5bcc316
--- /dev/null
+++ b/advtrains/po/zh_CN.po
@@ -0,0 +1,696 @@
+msgid ""
+msgstr ""
+"Project-Id-Version: advtrains\n"
+"Report-Msgid-Bugs-To: advtrains-discuss@lists.sr.ht\n"
+"POT-Creation-Date: 2023-10-09 11:02+0200\n"
+"PO-Revision-Date: 2023-10-09 11:24+0200\n"
+"Last-Translator: Y. Wang <yw05@forksworld.de>\n"
+"Language-Team: Chinese (Simplified)\n"
+"Language: zh_CN\n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"X-Generator: Poedit 3.3.2\n"
+
+#: advtrains/atc.lua:109
+msgid "Unconfigured ATC controller"
+msgstr "ATC 控制器 (未配置)"
+
+#: advtrains/atc.lua:150
+msgid ""
+"ATC controller, mode @1\n"
+"Command: @2"
+msgstr ""
+"ATC 控制器\n"
+"模式:@1\n"
+"命令:@2"
+
+#: advtrains/atc.lua:180
+msgid "Command"
+msgstr "命令"
+
+#: advtrains/atc.lua:184
+msgid "Command (on)"
+msgstr "命令 (激活时)"
+
+#: advtrains/atc.lua:187
+msgid "Digiline channel"
+msgstr "Digiline 频道"
+
+#: advtrains/atc.lua:189 advtrains_line_automation/stoprail.lua:65
+#: advtrains_luaautomation/active_common.lua:48
+msgid "Save"
+msgstr "保存"
+
+#: advtrains/atc.lua:236
+msgid "ATC Reverse command warning: didn't reverse train, train moving."
+msgstr "ATC 警告:火车正在移动,无法改变行车方向。"
+
+#: advtrains/atc.lua:248
+msgid "ATC Kick command warning: doors are closed."
+msgstr "ATC 警告:车门已关闭,无法踢出乘客。"
+
+#: advtrains/atc.lua:252
+msgid "ATC Kick command warning: train moving."
+msgstr "ATC 警告:火车正在移动,无法踢出乘客。"
+
+#: advtrains/atc.lua:322
+msgid "ATC command syntax error: I statement not closed: @1"
+msgstr "ATC 语法错误:“I”命令不完整:@1"
+
+#: advtrains/atc.lua:385
+msgid "ATC command parse error: Unknown command: @1"
+msgstr "ATC 语法错误:未知命令:@1"
+
+#: advtrains/copytool.lua:8
+msgid ""
+"Train copy/paste tool\n"
+"\n"
+"Left-click: copy train\n"
+"Right-click: paste train"
+msgstr ""
+"火车复制工具\n"
+"\n"
+"左键单击:复制\n"
+"右键单击:粘帖"
+
+#: advtrains/copytool.lua:29
+msgid "You do not have the @1 privilege."
+msgstr "您没有“@1”权限。"
+
+#: advtrains/copytool.lua:41
+msgid "The track you are trying to place the wagon on is not long enough."
+msgstr "轨道太短。"
+
+#: advtrains/copytool.lua:47
+msgid "The clipboard couldn't access the metadata. Paste failed."
+msgstr "无法粘贴:剪贴板无法访问元数据。"
+
+#: advtrains/copytool.lua:52 advtrains/copytool.lua:57
+msgid "The clipboard is empty."
+msgstr "剪贴板是空的。"
+
+#: advtrains/copytool.lua:74
+msgid "Back of train would end up off track, cancelling."
+msgstr "火车后部不在轨道上。"
+
+#: advtrains/copytool.lua:92
+msgid "No such lua entity."
+msgstr "您没有指向一个可以用火车复制工具复制的物体。"
+
+#: advtrains/copytool.lua:98
+msgid "No such wagon: @1."
+msgstr "ID 为“@1”的车厢不存在。"
+
+#: advtrains/copytool.lua:104
+msgid "No such train: @1."
+msgstr "ID 为“@1”的列车不存在。"
+
+#: advtrains/copytool.lua:176
+msgid "The clipboard couldn't access the metadata. Copy failed."
+msgstr "无法复制:剪贴板无法访问元数据。"
+
+#: advtrains/copytool.lua:180
+msgid "Train copied."
+msgstr "已复制列车。"
+
+#: advtrains/couple.lua:28
+msgid "Buffer and Chain Coupler"
+msgstr "链式车钩"
+
+#: advtrains/couple.lua:29
+msgid "Scharfenberg Coupler"
+msgstr "Scharfenberg 式车钩"
+
+#: advtrains/couple.lua:185
+msgid ""
+"You are not allowed to couple trains without the train_operator privilege."
+msgstr "您没有“train_operator”权限,不能连接这两节车厢。"
+
+#: advtrains/couple.lua:329 advtrains/couple.lua:333
+msgid "<No coupler>"
+msgstr "<没有车钩>"
+
+#: advtrains/couple.lua:334
+msgid "Can not couple: The couplers of the trains do not match (@1 and @2)."
+msgstr "您无法连接这两节车厢:这两节车厢使用不同的车钩 (@1和@2)。"
+
+#: advtrains/craft_items.lua:3
+msgid "Boiler"
+msgstr "锅炉"
+
+#: advtrains/craft_items.lua:9
+msgid "Driver's cab"
+msgstr "驾驶室"
+
+#: advtrains/craft_items.lua:15
+msgid "Wheel"
+msgstr "车轮"
+
+#: advtrains/craft_items.lua:21
+msgid "Chimney"
+msgstr "烟囱"
+
+#: advtrains/misc_nodes.lua:16
+msgid "@1 Platform (low)"
+msgstr "较低的@1站台"
+
+#: advtrains/misc_nodes.lua:33
+msgid "@1 Platform (high)"
+msgstr "较高的@1站台"
+
+#: advtrains/misc_nodes.lua:59
+msgid "@1 Platform (45 degree)"
+msgstr "较高的@1站台 (45°)"
+
+#: advtrains/misc_nodes.lua:81
+msgid "@1 Platform (low, 45 degree)"
+msgstr "较低的@1站台 (45°)"
+
+#: advtrains/protection.lua:7
+msgid "Can place, remove and operate trains"
+msgstr ""
+
+#: advtrains/protection.lua:12
+msgid ""
+"Can place, remove and operate any train, regardless of owner, whitelist, or "
+"protection"
+msgstr ""
+
+#: advtrains/protection.lua:18
+msgid "Can place and dig tracks in unprotected areas"
+msgstr ""
+
+#: advtrains/protection.lua:24
+msgid "Can operate turnouts and signals in unprotected areas"
+msgstr ""
+
+#: advtrains/protection.lua:148
+msgid ""
+"You are not allowed to build near tracks without the track_builder privilege."
+msgstr "您没有“train_operator”权限,不能在铁路附近建任何东西。"
+
+#: advtrains/protection.lua:148
+msgid ""
+"You are not allowed to build tracks without the track_builder privilege."
+msgstr "您没有“train_operator”权限,不能在这里建造铁路。"
+
+#: advtrains/protection.lua:153
+msgid "You are not allowed to build near tracks at this protected position."
+msgstr "这里已被保护,您不能在这里的铁路附近建任何东西。"
+
+#: advtrains/protection.lua:153
+msgid "You are not allowed to build tracks at this protected position."
+msgstr "这里已被保护,您不能在这里建造铁路。"
+
+#: advtrains/protection.lua:184
+msgid ""
+"You are not allowed to operate turnouts and signals without the "
+"railway_operator privilege."
+msgstr "您没有“railway_operator”权限,不能控制铁路设施。"
+
+#: advtrains/signals.lua:63
+msgid "Lampless Signal"
+msgstr "臂板信号机"
+
+#: advtrains/signals.lua:127
+msgid "Signal"
+msgstr "信号灯"
+
+#: advtrains/signals.lua:191
+msgid "Wallmounted Signal (left)"
+msgstr "壁挂式信号灯 (左侧)"
+
+#: advtrains/signals.lua:192
+msgid "Wallmounted Signal (right)"
+msgstr "壁挂式信号灯 (右侧)"
+
+#: advtrains/signals.lua:193
+msgid "Wallmounted Signal (top)"
+msgstr "悬挂式信号灯"
+
+#: advtrains/signals.lua:281 advtrains/signals.lua:322
+msgid "Andrew's Cross"
+msgstr "铁路道口信号灯"
+
+#: advtrains/trackplacer.lua:313
+msgid ""
+"Track Worker Tool\n"
+"\n"
+"Left-click: change rail type (straight/curve/switch)\n"
+"Right-click: rotate object"
+msgstr ""
+"铁路调整工具\n"
+"\n"
+"左键单击:切换轨道类型\n"
+"右键单击:旋转方块"
+
+#: advtrains/trackplacer.lua:340 advtrains/trackplacer.lua:377
+msgid "This node can't be rotated using the trackworker."
+msgstr "您不能使用铁路调整工具旋转这个方块。"
+
+#: advtrains/trackplacer.lua:350
+msgid "This track can not be rotated."
+msgstr "您不能旋转这段轨道。"
+
+#: advtrains/trackplacer.lua:404
+msgid "This node can't be changed using the trackworker."
+msgstr "您不能使用铁路调整工具调整这个方块。"
+
+#: advtrains/trackplacer.lua:414
+msgid "This track can not be changed."
+msgstr "您不能调整这段轨道。"
+
+#: advtrains/tracks.lua:449
+msgid "This track can not be removed."
+msgstr "您不能移除这段轨道。"
+
+#: advtrains/tracks.lua:616
+msgid "Position is occupied by a train."
+msgstr ""
+
+#: advtrains/tracks.lua:622
+msgid "There's a Track Circuit Break here."
+msgstr ""
+
+#: advtrains/tracks.lua:626
+msgid "There's a Signal Influence Point here."
+msgstr ""
+
+#: advtrains/tracks.lua:637
+msgid "@1 Slope"
+msgstr "@1斜坡"
+
+#: advtrains/tracks.lua:648 advtrains/tracks.lua:653
+msgid "Can't place slope: not pointing at node."
+msgstr "无法放置斜坡:您没有选择任何方块。"
+
+#: advtrains/tracks.lua:658
+msgid "Can't place slope: space occupied."
+msgstr "无法放置斜坡:此区域已被占用。"
+
+#: advtrains/tracks.lua:711
+msgid "Can't place slope: Not enough slope items left (@1 required)."
+msgstr "无法放置斜坡:您没有足够的铁路斜坡放置工具 (您总共需要@1个)"
+
+#: advtrains/tracks.lua:714
+msgid "Can't place slope: There's no slope of length @1."
+msgstr "无法放置斜坡:advtrains 不支持长度为@1米的斜坡。"
+
+#: advtrains/tracks.lua:721
+msgid "Can't place slope: no supporting node at upper end."
+msgstr "无法放置斜坡:较高端没有支撑方块。"
+
+#: advtrains/trainhud.lua:305
+msgid "OVERRUN RED SIGNAL! Examine situation and reverse train to move again."
+msgstr ""
+
+#: advtrains/wagons.lua:179
+msgid "This wagon is owned by @1, you can't destroy it."
+msgstr "这是 @1 的车厢,您不能摧毁它。"
+
+#: advtrains/wagons.lua:203
+msgid "The wagon's inventory is not empty."
+msgstr ""
+
+#: advtrains/wagons.lua:210
+msgid "Wagon needs to be decoupled from other wagons in order to destroy it."
+msgstr ""
+
+#: advtrains/wagons.lua:216
+msgid ""
+"Warning: If you destroy this wagon, you only get some steel back! If you are "
+"sure, hold Sneak and left-click the wagon."
+msgstr ""
+"警告:如果您摧毁此车厢,您只能拿到一些钢方块。如果您确定要摧毁这节车厢,请按"
+"潜行键并左键单击此车厢。"
+
+#: advtrains/wagons.lua:649 advtrains/wagons.lua:850
+msgid "Show Inventory"
+msgstr "显示物品栏"
+
+#: advtrains/wagons.lua:652
+msgid "Onboard Computer"
+msgstr ""
+
+#: advtrains/wagons.lua:655 advtrains/wagons.lua:1328
+msgid "Wagon properties"
+msgstr "车厢属性"
+
+#: advtrains/wagons.lua:658
+msgid "Get off"
+msgstr "下车"
+
+#: advtrains/wagons.lua:661
+msgid "Get off (forced)"
+msgstr "强制下车"
+
+#: advtrains/wagons.lua:663
+msgid "(Doors closed)"
+msgstr "(车门已关闭)"
+
+#: advtrains/wagons.lua:692
+msgid "This wagon has no seats."
+msgstr "这节车厢没有座位。"
+
+#: advtrains/wagons.lua:703
+msgid "This wagon is full."
+msgstr "车厢已满。"
+
+#: advtrains/wagons.lua:706
+msgid "Doors are closed! (Try holding sneak key!)"
+msgstr ""
+
+#: advtrains/wagons.lua:712
+msgid "You can't get on this wagon."
+msgstr ""
+
+#: advtrains/wagons.lua:838
+msgid "Select seat:"
+msgstr "请选择座位:"
+
+#: advtrains/wagons.lua:880
+msgid "Save wagon properties"
+msgstr "保存车厢属性"
+
+#: advtrains/wagons.lua:965
+msgid "Text displayed outside on train"
+msgstr "车厢外部显示"
+
+#: advtrains/wagons.lua:966
+msgid "Text displayed inside train"
+msgstr "车厢内部显示"
+
+#: advtrains/wagons.lua:967
+msgid "Line"
+msgstr "火车线路"
+
+#: advtrains/wagons.lua:968
+msgid "Routingcode"
+msgstr "路由码"
+
+#: advtrains/wagons.lua:1241
+msgid ""
+"Doors are closed. Use Sneak+rightclick to ignore the closed doors and get "
+"off."
+msgstr "车门已关闭,请使用潜行+右键单击下车。"
+
+#: advtrains/wagons.lua:1250
+msgid "You are not allowed to access the driver stand."
+msgstr ""
+
+#: advtrains_interlocking/tsr_rail.lua:13
+msgid "Point speed restriction: @1"
+msgstr ""
+
+#: advtrains_interlocking/tsr_rail.lua:14
+msgid "Set point speed restriction:"
+msgstr ""
+
+#: advtrains_interlocking/tsr_rail.lua:30
+msgid "You are not allowed to configure this track without the @1 privilege."
+msgstr "您没有“@1”权限,不能调整这段轨道。"
+
+#: advtrains_interlocking/tsr_rail.lua:34
+#: advtrains_line_automation/stoprail.lua:31
+#: advtrains_line_automation/stoprail.lua:76
+msgid "You are not allowed to configure this track."
+msgstr "您不能调整这段轨道。"
+
+#: advtrains_interlocking/tsr_rail.lua:64
+msgid "Point Speed Restriction Track"
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:54
+msgid "Station Code"
+msgstr "车站代码"
+
+#: advtrains_line_automation/stoprail.lua:55
+msgid "Station Name"
+msgstr "车站名称"
+
+#: advtrains_line_automation/stoprail.lua:56
+msgid "Door Delay"
+msgstr "车门关闭时间"
+
+#: advtrains_line_automation/stoprail.lua:57
+msgid "Dep. Speed"
+msgstr "出发速度"
+
+#: advtrains_line_automation/stoprail.lua:58 advtrains_train_track/init.lua:11
+#: advtrains_train_track/init.lua:156
+msgid "Track"
+msgstr "轨道"
+
+#: advtrains_line_automation/stoprail.lua:59
+msgid "Stop Time"
+msgstr "停站时间"
+
+#: advtrains_line_automation/stoprail.lua:60
+msgid "Door Side"
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:62
+msgid "Reverse train"
+msgstr "改变行车方向"
+
+#: advtrains_line_automation/stoprail.lua:63
+msgid "Kick out passengers"
+msgstr "踢出乘客"
+
+#: advtrains_line_automation/stoprail.lua:97
+msgid "Station code \"@1\" already exists and is owned by @2."
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:111
+msgid "This station is owned by @1. You are not allowed to edit its name."
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:221
+msgid "Station/Stop Track"
+msgstr "车站轨道"
+
+#: advtrains_luaautomation/active_common.lua:17
+msgid "Unconfigured LuaATC component"
+msgstr "LuaATC 部件 (未配置)"
+
+#: advtrains_luaautomation/active_common.lua:46
+msgid "LuaATC Environment"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:49
+msgid "Clear Local Environment"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:50
+msgid "Code"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:64
+msgid ""
+"You are not allowed to configure this LuaATC component without the @1 "
+"privilege."
+msgstr "您没有“@1”权限,不能配置这个 LuaATC 部件。"
+
+#: advtrains_luaautomation/active_common.lua:94
+msgid "LuaATC component assigned to environment '@1'"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:96
+msgid "LuaATC component assigned to an invalid environment"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:171
+msgid "LuaATC component with error: @1"
+msgstr ""
+
+#: advtrains_luaautomation/init.lua:13
+msgid ""
+"Can place and configure LuaATC components, including execute potentially "
+"harmful Lua code"
+msgstr ""
+
+#: advtrains_luaautomation/mesecon_controller.lua:211
+msgid "LuaATC Mesecon Controller"
+msgstr ""
+
+#: advtrains_luaautomation/operation_panel.lua:11
+msgid "LuaATC Operation Panel"
+msgstr ""
+
+#: advtrains_luaautomation/pcnaming.lua:28
+msgid ""
+"Passive Component Naming Tool\n"
+"\n"
+"Right-click to name a passive component."
+msgstr ""
+"被动元件命名工具\n"
+"\n"
+"右键单击命名所选元件。"
+
+#: advtrains_luaautomation/pcnaming.lua:39
+msgid ""
+"You are not allowed to name LuaATC passive components without the @1 "
+"privilege."
+msgstr "您没有“@1”权限,不能命名被动元件。"
+
+#: advtrains_luaautomation/pcnaming.lua:62
+msgid "Set name of component (empty to clear)"
+msgstr ""
+
+#: advtrains_train_industrial/init.lua:10
+#: advtrains_train_industrial/init.lua:49 advtrains_train_steam/init.lua:20
+#: advtrains_train_steam/init.lua:91
+msgid "Driver Stand (right)"
+msgstr "右侧司机座位"
+
+#: advtrains_train_industrial/init.lua:17
+#: advtrains_train_industrial/init.lua:56 advtrains_train_steam/init.lua:14
+#: advtrains_train_steam/init.lua:85
+msgid "Driver Stand (left)"
+msgstr "左侧司机座位"
+
+#: advtrains_train_industrial/init.lua:40
+msgid "Industrial Train Engine"
+msgstr "工业用火车头"
+
+#: advtrains_train_industrial/init.lua:79
+msgid "Big Industrial Train Engine"
+msgstr "大型工业用火车头"
+
+#: advtrains_train_industrial/init.lua:98
+msgid "Industrial tank wagon"
+msgstr "液体运输车厢"
+
+#: advtrains_train_industrial/init.lua:116
+msgid "Industrial wood wagon"
+msgstr "木材运输车厢"
+
+#: advtrains_train_japan/init.lua:4
+msgid "Japanese Train Inter-Wagon Connection"
+msgstr "日本火车车钩"
+
+#: advtrains_train_japan/init.lua:37
+msgid "Driver stand"
+msgstr "司机座位"
+
+#: advtrains_train_japan/init.lua:101
+msgid "Japanese Train Engine"
+msgstr "高速列车车头"
+
+#: advtrains_train_japan/init.lua:176
+msgid "Japanese Train Wagon"
+msgstr "高速列车车厢"
+
+#: advtrains_train_steam/init.lua:75
+msgid "Steam Engine"
+msgstr "蒸汽机车"
+
+#: advtrains_train_steam/init.lua:159
+msgid "Detailed Steam Engine"
+msgstr "精细的蒸汽机车"
+
+#: advtrains_train_steam/init.lua:206
+msgid "Passenger Wagon"
+msgstr "客车"
+
+#: advtrains_train_steam/init.lua:226
+msgid "Box Wagon"
+msgstr "货运车厢"
+
+#: advtrains_train_subway/init.lua:144
+msgid "Subway Passenger Wagon"
+msgstr "地铁车厢"
+
+#: advtrains_train_track/init.lua:31
+msgid "Y-turnout"
+msgstr "对称道岔"
+
+#: advtrains_train_track/init.lua:49
+msgid "3-way turnout"
+msgstr "三开道岔"
+
+#: advtrains_train_track/init.lua:69
+msgid "Perpendicular Diamond Crossing Track"
+msgstr "垂直交叉轨道"
+
+#: advtrains_train_track/init.lua:91
+msgid "90+Angle Diamond Crossing Track"
+msgstr "交叉轨道 (其中一条轨道与坐标轴平行)"
+
+#: advtrains_train_track/init.lua:132
+msgid "Diagonal Diamond Crossing Track"
+msgstr "交叉轨道"
+
+#: advtrains_train_track/init.lua:179
+msgid "Bumper"
+msgstr "保险杠"
+
+#: advtrains_train_track/init.lua:201
+msgid "ATC controller"
+msgstr "ATC 控制器"
+
+#: advtrains_train_track/init.lua:317
+msgid "Unloading Track"
+msgstr "卸货轨道"
+
+#: advtrains_train_track/init.lua:342
+msgid "Loading Track"
+msgstr "装货轨道"
+
+#: advtrains_train_track/init.lua:406
+msgid "Detector Rail"
+msgstr "探测轨道"
+
+#~ msgid ""
+#~ "ATC controller, mode @1\n"
+#~ "Channel: @2"
+#~ msgstr ""
+#~ "ATC 控制器\n"
+#~ "模式:@1\n"
+#~ "频道:@2"
+
+#~ msgid "Access to @1"
+#~ msgstr "可前往@1"
+
+#~ msgid "Can't get on: wagon full or doors closed!"
+#~ msgstr "无法上车:车门已关闭或车厢已满。"
+
+#~ msgid "Can't place: protected position!"
+#~ msgstr "无法放置:此区域已被保护。"
+
+#~ msgid "Default Seat"
+#~ msgstr "默认座位"
+
+#~ msgid "Default Seat (driver stand)"
+#~ msgstr "默认座位 (司机座位)"
+
+#~ msgid "Deprecated Track"
+#~ msgstr "请不要使用"
+
+#~ msgid "Lock couples"
+#~ msgstr "锁定连接处"
+
+#~ msgid "Speed:"
+#~ msgstr "速度"
+
+#~ msgid "Target:"
+#~ msgstr "目标速度"
+
+#, fuzzy
+#~ msgid "This node can't be rotated using the trackworker,"
+#~ msgstr "您不能使用铁路调整工具旋转这个方块。"
+
+#~ msgid "This position is protected!"
+#~ msgstr "这里已被保护。"
+
+#~ msgid "Use Sneak+rightclick to bypass closed doors!"
+#~ msgstr "请使用潜行+右键上车。"
+
+#, fuzzy
+#~ msgid "You are not allowed to modify this protected track."
+#~ msgstr "这里已被保护,您不能在这里建造铁路。"
+
+#~ msgid ""
+#~ "You need to own at least one neighboring wagon to destroy this couple."
+#~ msgstr "您必须至少拥有其中一节车厢才能分开这两节车厢。"
diff --git a/advtrains/po/zh_TW.po b/advtrains/po/zh_TW.po
new file mode 100644
index 0000000..ece82c3
--- /dev/null
+++ b/advtrains/po/zh_TW.po
@@ -0,0 +1,696 @@
+msgid ""
+msgstr ""
+"Project-Id-Version: advtrains\n"
+"Report-Msgid-Bugs-To: advtrains-discuss@lists.sr.ht\n"
+"POT-Creation-Date: 2023-10-09 11:02+0200\n"
+"PO-Revision-Date: 2023-10-09 11:31+0200\n"
+"Last-Translator: Y. Wang <yw05@forksworld.de>\n"
+"Language-Team: Chinese (Traditional)\n"
+"Language: zh_TW\n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"X-Generator: Poedit 3.3.2\n"
+
+#: advtrains/atc.lua:109
+msgid "Unconfigured ATC controller"
+msgstr "ATC 控制器 (未配置)"
+
+#: advtrains/atc.lua:150
+msgid ""
+"ATC controller, mode @1\n"
+"Command: @2"
+msgstr ""
+"ATC 控制器\n"
+"模式:@1\n"
+"命令:@2"
+
+#: advtrains/atc.lua:180
+msgid "Command"
+msgstr "命令"
+
+#: advtrains/atc.lua:184
+msgid "Command (on)"
+msgstr "命令 (啟用時)"
+
+#: advtrains/atc.lua:187
+msgid "Digiline channel"
+msgstr "Digiline 頻道"
+
+#: advtrains/atc.lua:189 advtrains_line_automation/stoprail.lua:65
+#: advtrains_luaautomation/active_common.lua:48
+msgid "Save"
+msgstr "儲存"
+
+#: advtrains/atc.lua:236
+msgid "ATC Reverse command warning: didn't reverse train, train moving."
+msgstr "ATC 警告:火車正在移動,無法改變行車方向。"
+
+#: advtrains/atc.lua:248
+msgid "ATC Kick command warning: doors are closed."
+msgstr "ATC 警告:車門已關閉,無法踢出乘客。"
+
+#: advtrains/atc.lua:252
+msgid "ATC Kick command warning: train moving."
+msgstr "ATC 警告:火車正在移動,無法踢出乘客。"
+
+#: advtrains/atc.lua:322
+msgid "ATC command syntax error: I statement not closed: @1"
+msgstr "ATC 語法錯誤:「I」命令不完整:@1"
+
+#: advtrains/atc.lua:385
+msgid "ATC command parse error: Unknown command: @1"
+msgstr "ATC 語法錯誤:未知命令:@1"
+
+#: advtrains/copytool.lua:8
+msgid ""
+"Train copy/paste tool\n"
+"\n"
+"Left-click: copy train\n"
+"Right-click: paste train"
+msgstr ""
+"火車複製工具\n"
+"\n"
+"左鍵單擊:複製\n"
+"右鍵單擊:粘帖"
+
+#: advtrains/copytool.lua:29
+msgid "You do not have the @1 privilege."
+msgstr "您沒有「@1」許可權。"
+
+#: advtrains/copytool.lua:41
+msgid "The track you are trying to place the wagon on is not long enough."
+msgstr "軌道太短。"
+
+#: advtrains/copytool.lua:47
+msgid "The clipboard couldn't access the metadata. Paste failed."
+msgstr "無法貼上:剪貼簿無法訪問元資料。"
+
+#: advtrains/copytool.lua:52 advtrains/copytool.lua:57
+msgid "The clipboard is empty."
+msgstr "剪貼簿是空的。"
+
+#: advtrains/copytool.lua:74
+msgid "Back of train would end up off track, cancelling."
+msgstr "火車後部不在軌道上。"
+
+#: advtrains/copytool.lua:92
+msgid "No such lua entity."
+msgstr "您沒有指向一個可以用火車複製工具複製的物體。"
+
+#: advtrains/copytool.lua:98
+msgid "No such wagon: @1."
+msgstr "ID 為「@1」的車廂不存在。"
+
+#: advtrains/copytool.lua:104
+msgid "No such train: @1."
+msgstr "ID 為「@1」的列車不存在。"
+
+#: advtrains/copytool.lua:176
+msgid "The clipboard couldn't access the metadata. Copy failed."
+msgstr "無法複製:剪貼簿無法訪問元資料。"
+
+#: advtrains/copytool.lua:180
+msgid "Train copied."
+msgstr "已複製火車。"
+
+#: advtrains/couple.lua:28
+msgid "Buffer and Chain Coupler"
+msgstr "鏈式連結器"
+
+#: advtrains/couple.lua:29
+msgid "Scharfenberg Coupler"
+msgstr "Scharfenberg 式連結器"
+
+#: advtrains/couple.lua:185
+msgid ""
+"You are not allowed to couple trains without the train_operator privilege."
+msgstr "您沒有「train_operator」許可權,不能連結這兩節車廂。"
+
+#: advtrains/couple.lua:329 advtrains/couple.lua:333
+msgid "<No coupler>"
+msgstr "<無連結器>"
+
+#: advtrains/couple.lua:334
+msgid "Can not couple: The couplers of the trains do not match (@1 and @2)."
+msgstr "您無法連結這兩節車廂:這兩節車廂使用不同的連結器 (@1和@2)。"
+
+#: advtrains/craft_items.lua:3
+msgid "Boiler"
+msgstr "鍋爐"
+
+#: advtrains/craft_items.lua:9
+msgid "Driver's cab"
+msgstr "駕駛室"
+
+#: advtrains/craft_items.lua:15
+msgid "Wheel"
+msgstr "車輪"
+
+#: advtrains/craft_items.lua:21
+msgid "Chimney"
+msgstr "煙囪"
+
+#: advtrains/misc_nodes.lua:16
+msgid "@1 Platform (low)"
+msgstr "較低的@1月臺"
+
+#: advtrains/misc_nodes.lua:33
+msgid "@1 Platform (high)"
+msgstr "較高的@1月臺"
+
+#: advtrains/misc_nodes.lua:59
+msgid "@1 Platform (45 degree)"
+msgstr "較高的@1月臺 (45°)"
+
+#: advtrains/misc_nodes.lua:81
+msgid "@1 Platform (low, 45 degree)"
+msgstr "較低的@1月臺 (45°)"
+
+#: advtrains/protection.lua:7
+msgid "Can place, remove and operate trains"
+msgstr ""
+
+#: advtrains/protection.lua:12
+msgid ""
+"Can place, remove and operate any train, regardless of owner, whitelist, or "
+"protection"
+msgstr ""
+
+#: advtrains/protection.lua:18
+msgid "Can place and dig tracks in unprotected areas"
+msgstr ""
+
+#: advtrains/protection.lua:24
+msgid "Can operate turnouts and signals in unprotected areas"
+msgstr ""
+
+#: advtrains/protection.lua:148
+msgid ""
+"You are not allowed to build near tracks without the track_builder privilege."
+msgstr "您沒有「train_operator」許可權,不能在鐵路附近建任何東西。"
+
+#: advtrains/protection.lua:148
+msgid ""
+"You are not allowed to build tracks without the track_builder privilege."
+msgstr "您沒有「train_operator」許可權,不能在這裡建造鐵路。"
+
+#: advtrains/protection.lua:153
+msgid "You are not allowed to build near tracks at this protected position."
+msgstr "這裡已被保護,您不能在這裡的鐵路附近建任何東西。"
+
+#: advtrains/protection.lua:153
+msgid "You are not allowed to build tracks at this protected position."
+msgstr "這裡已被保護,您不能在這裡建造鐵路。"
+
+#: advtrains/protection.lua:184
+msgid ""
+"You are not allowed to operate turnouts and signals without the "
+"railway_operator privilege."
+msgstr "您沒有「railway_operator」許可權,不能控制鐵路設施。"
+
+#: advtrains/signals.lua:63
+msgid "Lampless Signal"
+msgstr "臂木式號誌機"
+
+#: advtrains/signals.lua:127
+msgid "Signal"
+msgstr "色燈號誌機"
+
+#: advtrains/signals.lua:191
+msgid "Wallmounted Signal (left)"
+msgstr "壁掛式色燈號誌機 (左側)"
+
+#: advtrains/signals.lua:192
+msgid "Wallmounted Signal (right)"
+msgstr "壁掛式色燈號誌機 (右側)"
+
+#: advtrains/signals.lua:193
+msgid "Wallmounted Signal (top)"
+msgstr "懸掛式色燈號誌機"
+
+#: advtrains/signals.lua:281 advtrains/signals.lua:322
+msgid "Andrew's Cross"
+msgstr "平交道號誌燈"
+
+#: advtrains/trackplacer.lua:313
+msgid ""
+"Track Worker Tool\n"
+"\n"
+"Left-click: change rail type (straight/curve/switch)\n"
+"Right-click: rotate object"
+msgstr ""
+"鐵路調整工具\n"
+"\n"
+"左鍵單擊:切換軌道型別\n"
+"右鍵單擊:旋轉方塊"
+
+#: advtrains/trackplacer.lua:340 advtrains/trackplacer.lua:377
+msgid "This node can't be rotated using the trackworker."
+msgstr "您不能使用鐵路調整工具旋轉這個方塊。"
+
+#: advtrains/trackplacer.lua:350
+msgid "This track can not be rotated."
+msgstr "您不能旋轉這段軌道。"
+
+#: advtrains/trackplacer.lua:404
+msgid "This node can't be changed using the trackworker."
+msgstr "您不能使用鐵路調整工具調整這個方塊。"
+
+#: advtrains/trackplacer.lua:414
+msgid "This track can not be changed."
+msgstr "您不能調整這段軌道。"
+
+#: advtrains/tracks.lua:449
+msgid "This track can not be removed."
+msgstr "您不能移除這段軌道。"
+
+#: advtrains/tracks.lua:616
+msgid "Position is occupied by a train."
+msgstr ""
+
+#: advtrains/tracks.lua:622
+msgid "There's a Track Circuit Break here."
+msgstr ""
+
+#: advtrains/tracks.lua:626
+msgid "There's a Signal Influence Point here."
+msgstr ""
+
+#: advtrains/tracks.lua:637
+msgid "@1 Slope"
+msgstr "@1斜坡"
+
+#: advtrains/tracks.lua:648 advtrains/tracks.lua:653
+msgid "Can't place slope: not pointing at node."
+msgstr "無法放置斜坡:您沒有選擇任何方塊。"
+
+#: advtrains/tracks.lua:658
+msgid "Can't place slope: space occupied."
+msgstr "無法放置斜坡:此區域已被佔用。"
+
+#: advtrains/tracks.lua:711
+msgid "Can't place slope: Not enough slope items left (@1 required)."
+msgstr "無法放置斜坡:您沒有足夠的鐵路斜坡放置工具 (您總共需要@1個)"
+
+#: advtrains/tracks.lua:714
+msgid "Can't place slope: There's no slope of length @1."
+msgstr "無法放置斜坡:advtrains 不支援長度為@1米的斜坡。"
+
+#: advtrains/tracks.lua:721
+msgid "Can't place slope: no supporting node at upper end."
+msgstr "無法放置斜坡:較高階沒有支撐方塊。"
+
+#: advtrains/trainhud.lua:305
+msgid "OVERRUN RED SIGNAL! Examine situation and reverse train to move again."
+msgstr ""
+
+#: advtrains/wagons.lua:179
+msgid "This wagon is owned by @1, you can't destroy it."
+msgstr "這是 @1 的車廂,您不能摧毀它。"
+
+#: advtrains/wagons.lua:203
+msgid "The wagon's inventory is not empty."
+msgstr ""
+
+#: advtrains/wagons.lua:210
+msgid "Wagon needs to be decoupled from other wagons in order to destroy it."
+msgstr ""
+
+#: advtrains/wagons.lua:216
+msgid ""
+"Warning: If you destroy this wagon, you only get some steel back! If you are "
+"sure, hold Sneak and left-click the wagon."
+msgstr ""
+"警告:如果您摧毀此車廂,您只能拿到一些鋼方塊。如果您確定要摧毀這節車廂,請按"
+"潛行鍵並左鍵單擊此車廂。"
+
+#: advtrains/wagons.lua:649 advtrains/wagons.lua:850
+msgid "Show Inventory"
+msgstr "顯示物品欄"
+
+#: advtrains/wagons.lua:652
+msgid "Onboard Computer"
+msgstr ""
+
+#: advtrains/wagons.lua:655 advtrains/wagons.lua:1328
+msgid "Wagon properties"
+msgstr "車廂屬性"
+
+#: advtrains/wagons.lua:658
+msgid "Get off"
+msgstr "下車"
+
+#: advtrains/wagons.lua:661
+msgid "Get off (forced)"
+msgstr "強制下車"
+
+#: advtrains/wagons.lua:663
+msgid "(Doors closed)"
+msgstr "(車門已關閉)"
+
+#: advtrains/wagons.lua:692
+msgid "This wagon has no seats."
+msgstr "這節車廂沒有座位。"
+
+#: advtrains/wagons.lua:703
+msgid "This wagon is full."
+msgstr "車廂已滿。"
+
+#: advtrains/wagons.lua:706
+msgid "Doors are closed! (Try holding sneak key!)"
+msgstr ""
+
+#: advtrains/wagons.lua:712
+msgid "You can't get on this wagon."
+msgstr ""
+
+#: advtrains/wagons.lua:838
+msgid "Select seat:"
+msgstr "請選擇座位:"
+
+#: advtrains/wagons.lua:880
+msgid "Save wagon properties"
+msgstr "儲存車廂屬性"
+
+#: advtrains/wagons.lua:965
+msgid "Text displayed outside on train"
+msgstr "車廂外部顯示"
+
+#: advtrains/wagons.lua:966
+msgid "Text displayed inside train"
+msgstr "車廂內部顯示"
+
+#: advtrains/wagons.lua:967
+msgid "Line"
+msgstr "火車線路"
+
+#: advtrains/wagons.lua:968
+msgid "Routingcode"
+msgstr "路由碼"
+
+#: advtrains/wagons.lua:1241
+msgid ""
+"Doors are closed. Use Sneak+rightclick to ignore the closed doors and get "
+"off."
+msgstr "車門已關閉,請使用潛行+右鍵單擊下車。"
+
+#: advtrains/wagons.lua:1250
+msgid "You are not allowed to access the driver stand."
+msgstr ""
+
+#: advtrains_interlocking/tsr_rail.lua:13
+msgid "Point speed restriction: @1"
+msgstr ""
+
+#: advtrains_interlocking/tsr_rail.lua:14
+msgid "Set point speed restriction:"
+msgstr ""
+
+#: advtrains_interlocking/tsr_rail.lua:30
+msgid "You are not allowed to configure this track without the @1 privilege."
+msgstr "您沒有「@1」許可權,不能調整這段軌道。"
+
+#: advtrains_interlocking/tsr_rail.lua:34
+#: advtrains_line_automation/stoprail.lua:31
+#: advtrains_line_automation/stoprail.lua:76
+msgid "You are not allowed to configure this track."
+msgstr "您不能調整這段軌道。"
+
+#: advtrains_interlocking/tsr_rail.lua:64
+msgid "Point Speed Restriction Track"
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:54
+msgid "Station Code"
+msgstr "車站碼"
+
+#: advtrains_line_automation/stoprail.lua:55
+msgid "Station Name"
+msgstr "車站名稱"
+
+#: advtrains_line_automation/stoprail.lua:56
+msgid "Door Delay"
+msgstr "車門關閉時間"
+
+#: advtrains_line_automation/stoprail.lua:57
+msgid "Dep. Speed"
+msgstr "出發速度"
+
+#: advtrains_line_automation/stoprail.lua:58 advtrains_train_track/init.lua:11
+#: advtrains_train_track/init.lua:156
+msgid "Track"
+msgstr "軌道"
+
+#: advtrains_line_automation/stoprail.lua:59
+msgid "Stop Time"
+msgstr "停站時間"
+
+#: advtrains_line_automation/stoprail.lua:60
+msgid "Door Side"
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:62
+msgid "Reverse train"
+msgstr "改變行車方向"
+
+#: advtrains_line_automation/stoprail.lua:63
+msgid "Kick out passengers"
+msgstr "踢出乘客"
+
+#: advtrains_line_automation/stoprail.lua:97
+msgid "Station code \"@1\" already exists and is owned by @2."
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:111
+msgid "This station is owned by @1. You are not allowed to edit its name."
+msgstr ""
+
+#: advtrains_line_automation/stoprail.lua:221
+msgid "Station/Stop Track"
+msgstr "車站軌道"
+
+#: advtrains_luaautomation/active_common.lua:17
+msgid "Unconfigured LuaATC component"
+msgstr "LuaATC 元件 (未配置)"
+
+#: advtrains_luaautomation/active_common.lua:46
+msgid "LuaATC Environment"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:49
+msgid "Clear Local Environment"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:50
+msgid "Code"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:64
+msgid ""
+"You are not allowed to configure this LuaATC component without the @1 "
+"privilege."
+msgstr "您沒有「@1」許可權,不能配置這個 LuaATC 元件。"
+
+#: advtrains_luaautomation/active_common.lua:94
+msgid "LuaATC component assigned to environment '@1'"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:96
+msgid "LuaATC component assigned to an invalid environment"
+msgstr ""
+
+#: advtrains_luaautomation/active_common.lua:171
+msgid "LuaATC component with error: @1"
+msgstr ""
+
+#: advtrains_luaautomation/init.lua:13
+msgid ""
+"Can place and configure LuaATC components, including execute potentially "
+"harmful Lua code"
+msgstr ""
+
+#: advtrains_luaautomation/mesecon_controller.lua:211
+msgid "LuaATC Mesecon Controller"
+msgstr ""
+
+#: advtrains_luaautomation/operation_panel.lua:11
+msgid "LuaATC Operation Panel"
+msgstr ""
+
+#: advtrains_luaautomation/pcnaming.lua:28
+msgid ""
+"Passive Component Naming Tool\n"
+"\n"
+"Right-click to name a passive component."
+msgstr ""
+"被動元件命名工具\n"
+"\n"
+"右鍵單擊命名所選元件。"
+
+#: advtrains_luaautomation/pcnaming.lua:39
+msgid ""
+"You are not allowed to name LuaATC passive components without the @1 "
+"privilege."
+msgstr "您沒有「@1」許可權,不能命名這個元件。"
+
+#: advtrains_luaautomation/pcnaming.lua:62
+msgid "Set name of component (empty to clear)"
+msgstr ""
+
+#: advtrains_train_industrial/init.lua:10
+#: advtrains_train_industrial/init.lua:49 advtrains_train_steam/init.lua:20
+#: advtrains_train_steam/init.lua:91
+msgid "Driver Stand (right)"
+msgstr "右側司機座位"
+
+#: advtrains_train_industrial/init.lua:17
+#: advtrains_train_industrial/init.lua:56 advtrains_train_steam/init.lua:14
+#: advtrains_train_steam/init.lua:85
+msgid "Driver Stand (left)"
+msgstr "左側司機座位"
+
+#: advtrains_train_industrial/init.lua:40
+msgid "Industrial Train Engine"
+msgstr "工業用火車頭"
+
+#: advtrains_train_industrial/init.lua:79
+msgid "Big Industrial Train Engine"
+msgstr "大型工業用火車頭"
+
+#: advtrains_train_industrial/init.lua:98
+msgid "Industrial tank wagon"
+msgstr "液體運輸車廂"
+
+#: advtrains_train_industrial/init.lua:116
+msgid "Industrial wood wagon"
+msgstr "木材運輸車廂"
+
+#: advtrains_train_japan/init.lua:4
+msgid "Japanese Train Inter-Wagon Connection"
+msgstr "日本火車連結器"
+
+#: advtrains_train_japan/init.lua:37
+msgid "Driver stand"
+msgstr "司機座位"
+
+#: advtrains_train_japan/init.lua:101
+msgid "Japanese Train Engine"
+msgstr "高速列車車頭"
+
+#: advtrains_train_japan/init.lua:176
+msgid "Japanese Train Wagon"
+msgstr "高速列車車廂"
+
+#: advtrains_train_steam/init.lua:75
+msgid "Steam Engine"
+msgstr "蒸汽機車"
+
+#: advtrains_train_steam/init.lua:159
+msgid "Detailed Steam Engine"
+msgstr "精細的蒸汽機車"
+
+#: advtrains_train_steam/init.lua:206
+msgid "Passenger Wagon"
+msgstr "客車"
+
+#: advtrains_train_steam/init.lua:226
+msgid "Box Wagon"
+msgstr "貨運車廂"
+
+#: advtrains_train_subway/init.lua:144
+msgid "Subway Passenger Wagon"
+msgstr "地鐵車廂"
+
+#: advtrains_train_track/init.lua:31
+msgid "Y-turnout"
+msgstr "對稱道岔"
+
+#: advtrains_train_track/init.lua:49
+msgid "3-way turnout"
+msgstr "三開道岔"
+
+#: advtrains_train_track/init.lua:69
+msgid "Perpendicular Diamond Crossing Track"
+msgstr "垂直交叉軌道"
+
+#: advtrains_train_track/init.lua:91
+msgid "90+Angle Diamond Crossing Track"
+msgstr "交叉軌道 (其中一條軌道與座標軸平行)"
+
+#: advtrains_train_track/init.lua:132
+msgid "Diagonal Diamond Crossing Track"
+msgstr "交叉軌道"
+
+#: advtrains_train_track/init.lua:179
+msgid "Bumper"
+msgstr "保險槓"
+
+#: advtrains_train_track/init.lua:201
+msgid "ATC controller"
+msgstr "ATC 控制器"
+
+#: advtrains_train_track/init.lua:317
+msgid "Unloading Track"
+msgstr "卸貨軌道"
+
+#: advtrains_train_track/init.lua:342
+msgid "Loading Track"
+msgstr "裝貨軌道"
+
+#: advtrains_train_track/init.lua:406
+msgid "Detector Rail"
+msgstr "探測軌道"
+
+#~ msgid ""
+#~ "ATC controller, mode @1\n"
+#~ "Channel: @2"
+#~ msgstr ""
+#~ "ATC 控制器\n"
+#~ "模式:@1\n"
+#~ "頻道:@2"
+
+#~ msgid "Access to @1"
+#~ msgstr "可前往@1"
+
+#~ msgid "Can't get on: wagon full or doors closed!"
+#~ msgstr "無法上車:車門已關閉或車廂已滿。"
+
+#~ msgid "Can't place: protected position!"
+#~ msgstr "無法放置:此區域已被保護。"
+
+#~ msgid "Default Seat"
+#~ msgstr "預設座位"
+
+#~ msgid "Default Seat (driver stand)"
+#~ msgstr "預設座位 (司機座位)"
+
+#~ msgid "Deprecated Track"
+#~ msgstr "請不要使用"
+
+#~ msgid "Lock couples"
+#~ msgstr "鎖定連結處"
+
+#~ msgid "Speed:"
+#~ msgstr "速度"
+
+#~ msgid "Target:"
+#~ msgstr "目標速度"
+
+#, fuzzy
+#~ msgid "This node can't be rotated using the trackworker,"
+#~ msgstr "您不能使用鐵路調整工具旋轉這個方塊。"
+
+#~ msgid "This position is protected!"
+#~ msgstr "這裡已被保護。"
+
+#~ msgid "Use Sneak+rightclick to bypass closed doors!"
+#~ msgstr "請使用潛行+右鍵上車。"
+
+#, fuzzy
+#~ msgid "You are not allowed to modify this protected track."
+#~ msgstr "這裡已被保護,您不能在這裡建造鐵路。"
+
+#~ msgid ""
+#~ "You need to own at least one neighboring wagon to destroy this couple."
+#~ msgstr "您必須至少擁有其中一節車廂才能分開這兩節車廂。"
diff --git a/advtrains/poconvert.lua b/advtrains/poconvert.lua
new file mode 100644
index 0000000..74f962e
--- /dev/null
+++ b/advtrains/poconvert.lua
@@ -0,0 +1,185 @@
+local unescape_string
+do
+ local schartbl = { -- https://en.wikipedia.org/wiki/Escape_sequences_in_C
+ a = "\a",
+ b = "\b",
+ e = string.char(0x1b),
+ f = "\f",
+ n = "\n",
+ r = "\r",
+ t = "\t",
+ v = "\v",
+ }
+ local function replace_single(pfx, c)
+ local pl = #pfx
+ if pl % 2 == 0 then
+ return string.sub(pfx, 1, pl/2) .. c
+ end
+ return string.sub(pfx, 1, math.floor(pl/2)) .. (schartbl[c] or c)
+ end
+ unescape_string = function(str)
+ return string.gsub(str, [[(\+)([abefnrtv'"?])]], replace_single)
+ end
+end
+
+local function readstring_aux(str, pos)
+ local _, spos = string.find(str, [[^%s*"]], pos)
+ if not spos then
+ return nil
+ end
+ local ipos = spos
+ while true do
+ local _, epos, m = string.find(str, [[(\*)"]], ipos+1)
+ if not epos then
+ return error("String extends beyond the end of input")
+ end
+ ipos = epos
+ if #m % 2 == 0 then
+ return unescape_string(string.sub(str, spos+1, epos-1)), epos+1
+ end
+ end
+end
+
+local function readstring(str, pos)
+ local st = {}
+ local nxt = pos
+ while true do
+ local s, npos = readstring_aux(str, nxt)
+ if not s then
+ if not st[1] then
+ return nil, nxt
+ else
+ return table.concat(st), nxt
+ end
+ end
+ nxt = npos
+ table.insert(st, s)
+ end
+end
+
+local function readtoken(str, pos)
+ local _, epos, tok = string.find(str, [[^%s*(%S+)]], pos)
+ if epos == nil then
+ return nil, pos
+ end
+ return tok, epos+1
+end
+
+local function readcomment_add_flags(flags, s)
+ for flag in string.gmatch(s, ",%s*([^,]+)") do
+ flags[flag] = true
+ end
+end
+
+local function readcomment_aux(str, pos)
+ local _, epos, sval = string.find(str, "^\n*#([^\n]*)", pos)
+ if not epos then
+ return nil
+ end
+ return sval, epos+1
+end
+
+local function readcomment(str, pos)
+ local st = {}
+ local nxt = pos
+ local flags = {}
+ while true do
+ local s, npos = readcomment_aux(str, nxt)
+ if not npos then
+ local t = {
+ comment = table.concat(st, "\n"),
+ flags = flags,
+ }
+ return t, nxt
+ end
+ if string.sub(s, 1, 1) == "," then
+ readcomment_add_flags(flags, s)
+ end
+ table.insert(st, s)
+ nxt = npos
+ end
+end
+
+local function readpo(str)
+ local st = {}
+ local pos = 1
+ while true do
+ local entry, nxt = readcomment(str, pos)
+ local msglines = 0
+ while true do
+ local tok, npos = readtoken(str, nxt)
+ if tok == nil or string.sub(tok, 1, 1) == "#" then
+ break
+ elseif string.sub(tok, 1, 3) ~= "msg" then
+ return error("Invalid token: " .. tok)
+ elseif entry[tok] ~= nil then
+ break
+ else
+ local value, npos = readstring(str, npos)
+ assert(value ~= nil, "No string provided for " .. tok)
+ entry[tok] = value
+ nxt = npos
+ msglines = msglines+1
+ end
+ end
+ if msglines == 0 then
+ return st
+ elseif entry.msgid ~= "" then
+ assert(entry.msgid ~= nil, "Missing untranslated string")
+ assert(entry.msgstr ~= nil, "Missing translated string")
+ table.insert(st, entry)
+ end
+ pos = nxt
+ end
+end
+
+local escape_lookup = {
+ ["="] = "@=",
+ ["\n"] = "@n"
+}
+local function escape_string(st)
+ return (string.gsub(st, "[=\n]", escape_lookup))
+end
+
+local function convert_po_string(textdomain, str)
+ local st = {string.format("# textdomain: %s", textdomain)}
+ for _, entry in ipairs(readpo(str)) do
+ local line = ("%s=%s"):format(escape_string(entry.msgid), escape_string(entry.msgstr))
+ if entry.flags.fuzzy then
+ line = "#" .. line
+ end
+ table.insert(st, line)
+ end
+ return table.concat(st, "\n")
+end
+
+local function convert_po_file(textdomain, inpath, outpath)
+ local f, err = io.open(inpath, "rb")
+ assert(f, err)
+ local str = convert_po_string(textdomain, f:read("*a"))
+ f:close()
+ minetest.safe_file_write(outpath, str)
+ return str
+end
+
+local function convert_flat_po_directory(textdomain, modpath)
+ assert(textdomain, "No textdomain specified for po file conversion")
+ local mp = modpath or minetest.get_modpath(textdomain)
+ assert(mp ~= nil, "No path to write for " .. textdomain)
+ local popath = mp .. "/po"
+ local trpath = mp .. "/locale"
+ for _, infile in pairs(minetest.get_dir_list(popath, false)) do
+ local lang = string.match(infile, [[^([^%.]+)%.po$]])
+ if lang then
+ local inpath = popath .. "/" .. infile
+ local outpath = ("%s/%s.%s.tr"):format(trpath, textdomain, lang)
+ convert_po_file(textdomain, inpath, outpath)
+ end
+ end
+end
+
+return {
+ from_string = convert_po_string,
+ from_file = convert_po_file,
+ from_flat = convert_flat_po_directory,
+}
diff --git a/advtrains/protection.lua b/advtrains/protection.lua
index 7c5cf0b..36f4192 100644
--- a/advtrains/protection.lua
+++ b/advtrains/protection.lua
@@ -4,24 +4,24 @@
-- Privileges to control TRAIN DRIVING/COUPLING
minetest.register_privilege("train_operator", {
- description = "Without this privilege, a player can't do anything about trains, neither place or remove them nor drive or couple them (but he can build tracks if he has track_builder)",
+ description = attrans("Can place, remove and operate trains"),
give_to_singleplayer= true,
});
minetest.register_privilege("train_admin", {
- description = "Player may drive, place or remove any trains from/to anywhere, regardless of owner, whitelist or protection",
+ description = attrans("Can place, remove and operate any train, regardless of owner, whitelist, or protection"),
give_to_singleplayer= true,
});
-- Privileges to control TRACK BUILDING
minetest.register_privilege("track_builder", {
- description = "Player can place and/or dig rails not protected from him. If he also has protection_bypass, he can place/dig any rails",
+ description = attrans("Can place and dig tracks in unprotected areas"),
give_to_singleplayer= true,
});
-- Privileges to control OPERATING TURNOUTS/SIGNALS
minetest.register_privilege("railway_operator", {
- description = "Player can operate turnouts and signals not protected from him. If he also has protection_bypass, he can operate any turnouts/signals",
+ description = attrans("Can operate turnouts and signals in unprotected areas"),
give_to_singleplayer= true,
});
diff --git a/advtrains/settingtypes.txt b/advtrains/settingtypes.txt
index 2b627cb..a09da71 100644
--- a/advtrains/settingtypes.txt
+++ b/advtrains/settingtypes.txt
@@ -7,6 +7,10 @@ advtrains_show_ids (Show ID's in infotext) bool false
# You probably want to leave this setting set to false.
advtrains_enable_debugging (Enable debugging) bool false
+# Register certain debug items, for example the tunnelborer
+# Do not use on productive servers!
+advtrains_register_debugitems (Register Debug Items) bool false
+
# Enable the logging of certain events related to advtrains
# Logs are saved in the world directory as advtrains.log
# This setting is useful for multiplayer servers
@@ -61,3 +65,6 @@ advtrains_save_interval (Save Interval) int 60 20 3600
# If enabled, trains only collide with nodes with "normal" drawtype.
advtrains_forgiving_collision (Forgiving Collision mode) bool false
+# Enable universal couplers for wagons
+# If enabled, wagons will bypass the checks that compare the coupler types when coupling.
+advtrains_universal_couplers (Universal Couplers) bool false \ No newline at end of file
diff --git a/advtrains/signals.lua b/advtrains/signals.lua
index d533227..93fd99e 100644
--- a/advtrains/signals.lua
+++ b/advtrains/signals.lua
@@ -3,15 +3,15 @@
local mrules_wallsignal = advtrains.meseconrules
-local function can_dig_func(pos)
+local function can_dig_func(pos, player)
if advtrains.interlocking then
- return advtrains.interlocking.signal_can_dig(pos)
+ return advtrains.interlocking.signal.can_dig(pos, player)
end
return true
end
-local function after_dig_func(pos)
+local function after_dig_func(pos, oldnode, oldmetadata, digger)
if advtrains.interlocking then
- return advtrains.interlocking.signal_after_dig(pos)
+ return advtrains.interlocking.signal.after_dig(pos, oldnode, oldmetadata, digger)
end
return true
end
@@ -26,22 +26,24 @@ return {
}
end
-local suppasp = {
- main = {0, -1},
- dst = {false},
- shunt = nil,
- proceed_as_main = true,
- info = {
- call_on = false,
- dead_end = false,
- w_speed = nil,
- }
+local main_aspects = {
+ { name = "free", description = "Free" }
}
+local function simple_apply_aspect(offname, onname)
+ return function(pos, node, main_aspect, rem_aspect, rem_aspinfo)
+ if main_aspect.halt then
+ advtrains.ndb.swap_node(pos, {name = offname, param2 = node.param2})
+ else
+ advtrains.ndb.swap_node(pos, {name = onname, param2 = node.param2})
+ end
+ end
+end
+
for r,f in pairs({on={as="off", ls="green", als="red"}, off={as="on", ls="red", als="green"}}) do
- advtrains.trackplacer.register_tracktype("advtrains:retrosignal", "")
- advtrains.trackplacer.register_tracktype("advtrains:signal", "")
+ -- advtrains.trackplacer.register_tracktype("advtrains:retrosignal", "")
+ -- advtrains.trackplacer.register_tracktype("advtrains:signal", "")
for rotid, rotation in ipairs({"", "_30", "_45", "_60"}) do
local crea=1
@@ -75,7 +77,10 @@ for r,f in pairs({on={as="off", ls="green", als="red"}, off={as="on", ls="red",
["action_"..f.as] = function (pos, node)
advtrains.ndb.swap_node(pos, {name = "advtrains:retrosignal_"..f.as..rotation, param2 = node.param2}, true)
if advtrains.interlocking then
- advtrains.interlocking.signal_on_aspect_changed(pos)
+ -- forcefully clears any set aspect, so that aspect system doesnt override it again
+ advtrains.interlocking.signal.unregister_aspect(pos)
+ -- notify trains
+ advtrains.interlocking.signal.notify_trains(pos)
end
end
}},
@@ -89,29 +94,24 @@ for r,f in pairs({on={as="off", ls="green", als="red"}, off={as="on", ls="red",
elseif advtrains.check_turnout_signal_protection(pos, player:get_player_name()) then
advtrains.ndb.swap_node(pos, {name = "advtrains:retrosignal_"..f.as..rotation, param2 = node.param2}, true)
if advtrains.interlocking then
- advtrains.interlocking.signal_on_aspect_changed(pos)
+ -- forcefully clears any set aspect, so that aspect system doesnt override it again
+ advtrains.interlocking.signal.unregister_aspect(pos)
+ -- notify trains
+ advtrains.interlocking.signal.notify_trains(pos)
end
end
end,
- -- new signal API
+ -- very new signal API
advtrains = {
- set_aspect = function(pos, node, asp)
- if asp.main ~= 0 then
- advtrains.ndb.swap_node(pos, {name = "advtrains:retrosignal_on"..rotation, param2 = node.param2}, true)
- else
- advtrains.ndb.swap_node(pos, {name = "advtrains:retrosignal_off"..rotation, param2 = node.param2}, true)
- end
- end,
- get_aspect = function(pos, node)
- return aspect(r=="on")
- end,
- supported_aspects = suppasp,
+ main_aspects = main_aspects,
+ apply_aspect = simple_apply_aspect("advtrains:retrosignal_off"..rotation, "advtrains:retrosignal_on"..rotation),
+ get_aspect_info = function() return aspect(r=="on") end,
},
can_dig = can_dig_func,
after_dig_node = after_dig_func,
- check_for_pole = true,
+ --TODO add rotation using trackworker
})
- advtrains.trackplacer.add_worked("advtrains:retrosignal", r, rotation, nil)
+ -- advtrains.trackplacer.add_worked("advtrains:retrosignal", r, rotation, nil)
minetest.register_node("advtrains:signal_"..r..rotation, {
drawtype = "mesh",
@@ -140,9 +140,6 @@ for r,f in pairs({on={as="off", ls="green", als="red"}, off={as="on", ls="red",
rules=advtrains.meseconrules,
["action_"..f.as] = function (pos, node)
advtrains.setstate(pos, f.als, node)
- if advtrains.interlocking then
- advtrains.interlocking.signal_on_aspect_changed(pos)
- end
end
}},
on_rightclick=function(pos, node, player)
@@ -154,36 +151,29 @@ for r,f in pairs({on={as="off", ls="green", als="red"}, off={as="on", ls="red",
advtrains.interlocking.show_ip_form(pos, pname)
elseif advtrains.check_turnout_signal_protection(pos, player:get_player_name()) then
advtrains.setstate(pos, f.als, node)
- if advtrains.interlocking then
- advtrains.interlocking.signal_on_aspect_changed(pos)
- end
end
end,
- -- new signal API
+ -- very new signal API
advtrains = {
- set_aspect = function(pos, node, asp)
- if asp.main ~= 0 then
- advtrains.ndb.swap_node(pos, {name = "advtrains:signal_on"..rotation, param2 = node.param2}, true)
- else
- advtrains.ndb.swap_node(pos, {name = "advtrains:signal_off"..rotation, param2 = node.param2}, true)
- end
- end,
- get_aspect = function(pos, node)
- return aspect(r=="on")
- end,
- supported_aspects = suppasp,
- getstate = f.ls,
- setstate = function(pos, node, newstate)
- if newstate == f.als then
- advtrains.ndb.swap_node(pos, {name = "advtrains:signal_"..f.as..rotation, param2 = node.param2}, true)
+ main_aspects = main_aspects,
+ apply_aspect = simple_apply_aspect("advtrains:signal_off"..rotation, "advtrains:signal_on"..rotation),
+ get_aspect_info = function() return aspect(r=="on") end,
+ node_state = f.ls,
+ node_state_map = { red = "advtrains:signal_off"..rotation, green = "advtrains:signal_on"..rotation},
+ node_on_switch_state = function(pos, new_node, old_state, new_state)
+ if advtrains.interlocking then
+ -- forcefully clears any set aspect, so that aspect system doesnt override it again
+ advtrains.interlocking.signal.unregister_aspect(pos)
+ -- notify trains
+ advtrains.interlocking.signal.notify_trains(pos)
end
end,
},
can_dig = can_dig_func,
after_dig_node = after_dig_func,
- check_for_pole = true,
+ --TODO add rotation using trackworker
})
- advtrains.trackplacer.add_worked("advtrains:signal", r, rotation, nil)
+ -- advtrains.trackplacer.add_worked("advtrains:signal", r, rotation, nil)
end
local crea=1
@@ -197,7 +187,7 @@ for r,f in pairs({on={as="off", ls="green", als="red"}, off={as="on", ls="red",
b={-1/4, -1/2, 1/2 - 1/8, 1/4, 1/2, 1/2},
p={-1/4, -1/2, 5/8, 1/4, 1/2, 7/8},
}) do
- local def ={
+ local def = {
drawtype = "mesh",
paramtype="light",
paramtype2="4dir",
@@ -223,9 +213,6 @@ for r,f in pairs({on={as="off", ls="green", als="red"}, off={as="on", ls="red",
rules = mrules_wallsignal,
["action_"..f.as] = function (pos, node)
advtrains.setstate(pos, f.als, node)
- if advtrains.interlocking then
- advtrains.interlocking.signal_on_aspect_changed(pos)
- end
end
}},
on_rightclick=function(pos, node, player)
@@ -237,28 +224,21 @@ for r,f in pairs({on={as="off", ls="green", als="red"}, off={as="on", ls="red",
advtrains.interlocking.show_ip_form(pos, pname)
elseif advtrains.check_turnout_signal_protection(pos, player:get_player_name()) then
advtrains.setstate(pos, f.als, node)
- if advtrains.interlocking then
- advtrains.interlocking.signal_on_aspect_changed(pos)
- end
end
end,
- -- new signal API
+ -- very new signal API
advtrains = {
- set_aspect = function(pos, node, asp)
- if asp.main ~= 0 then
- advtrains.ndb.swap_node(pos, {name = "advtrains:signal_wall_"..loc.."_on", param2 = node.param2}, true)
- else
- advtrains.ndb.swap_node(pos, {name = "advtrains:signal_wall_"..loc.."_off", param2 = node.param2}, true)
- end
- end,
- get_aspect = function(pos, node)
- return aspect(r=="on")
- end,
- supported_aspects = suppasp,
- getstate = f.ls,
- setstate = function(pos, node, newstate)
- if newstate == f.als then
- advtrains.ndb.swap_node(pos, {name = "advtrains:signal_wall_"..loc.."_"..f.as, param2 = node.param2}, true)
+ main_aspects = main_aspects,
+ apply_aspect = simple_apply_aspect("advtrains:signal_wall_"..loc.."_off", "advtrains:signal_wall_"..loc.."_on"),
+ get_aspect_info = function() return aspect(r=="on") end,
+ node_state = f.ls,
+ node_state_map = { red = "advtrains:signal_wall_"..loc.."_off", green = "advtrains:signal_wall_"..loc.."_on" },
+ node_on_switch_state = function(pos, new_node, old_state, new_state)
+ if advtrains.interlocking then
+ -- forcefully clears any set aspect, so that aspect system doesnt override it again
+ advtrains.interlocking.signal.unregister_aspect(pos)
+ -- notify trains
+ advtrains.interlocking.signal.notify_trains(pos)
end
end,
},
@@ -298,22 +278,15 @@ minetest.register_node("advtrains:across_off", {
mesecons = {effector = {
rules = advtrains.meseconrules,
action_on = function (pos, node)
- minetest.get_meta(pos):set_int("crossing_state", 0)
advtrains.ndb.swap_node(pos, {name = "advtrains:across_on", param2 = node.param2}, true)
end
}},
advtrains = {
- getstate = "off",
- setstate = function(pos, node, newstate)
- if newstate == "on" then
- minetest.get_meta(pos):set_int("crossing_state", 0)
- advtrains.ndb.swap_node(pos, {name = "advtrains:across_on", param2 = node.param2}, true)
- end
- end,
+ node_state = "off",
+ node_state_map = { on = "advtrains:across_on", off = "advtrains:across_off" },
},
on_rightclick=function(pos, node, player)
if advtrains.check_turnout_signal_protection(pos, player:get_player_name()) then
- minetest.get_meta(pos):set_int("crossing_state", 0)
advtrains.ndb.swap_node(pos, {name = "advtrains:across_on", param2 = node.param2}, true)
end
end,
@@ -330,7 +303,7 @@ minetest.register_node("advtrains:across_on", {
mesh = "advtrains_across.obj",
tiles = {{name="advtrains_across_anim.png", animation={type="vertical_frames", aspect_w=64, aspect_h=64, length=1.0}}},
drop="advtrains:across_off",
- description=attrans("Andrew's Cross (on) (you hacker you)"),
+ description=attrans("Andrew's Cross"),
groups = {
cracky=3,
not_blocking_trains=1,
@@ -346,13 +319,9 @@ minetest.register_node("advtrains:across_on", {
end
}},
advtrains = {
- getstate = "on",
- setstate = function(pos, node, newstate)
- if newstate == "off" then
- advtrains.ndb.swap_node(pos, {name = "advtrains:across_off", param2 = node.param2}, true)
- end
- end,
- fallback_state = "off",
+ node_state = "on",
+ node_state_map = { on = "advtrains:across_on", off = "advtrains:across_off" },
+ node_fallback_state = "off",
},
on_rightclick=function(pos, node, player)
if advtrains.check_turnout_signal_protection(pos, player:get_player_name()) then
@@ -374,7 +343,7 @@ minetest.register_abm(
{
label = "Sound for Level Crossing",
nodenames = {"advtrains:across_on"},
- interval = 1,
+ interval = 3,
chance = 1,
action = function(pos, node, active_object_count, active_object_count_wider)
local meta = minetest.get_meta(pos)
diff --git a/advtrains/sounds/advtrains_crossing_bell.ogg b/advtrains/sounds/advtrains_crossing_bell.ogg
index 3235b52..2b441ae 100644
--- a/advtrains/sounds/advtrains_crossing_bell.ogg
+++ b/advtrains/sounds/advtrains_crossing_bell.ogg
Binary files differ
diff --git a/advtrains/spec/poconvert_spec.lua b/advtrains/spec/poconvert_spec.lua
new file mode 100644
index 0000000..51f33e7
--- /dev/null
+++ b/advtrains/spec/poconvert_spec.lua
@@ -0,0 +1,70 @@
+package.path = "../?.lua;" .. package.path
+advtrains = {}
+_G.advtrains = advtrains
+local poconvert = require("poconvert")
+
+describe("PO file converter", function()
+ it("should convert PO files", function()
+ assert.equals([[
+# textdomain: foo
+foo=bar
+baz=
+#@=wh\at\\@n=@=w\as\\@n
+multiline@nstrings=multiline@nresult
+with context?=oder doch nicht]], poconvert.from_string("foo", [[
+msgid ""
+msgstr "whatever metadata"
+
+msgid "foo"
+msgstr "bar"
+
+msgid "baz"
+msgstr ""
+
+#, fuzzy
+msgid "=wh\\at\\\\\n"
+msgstr "=w\\as\\\\\n"
+
+msgid "multi"
+"line\n"
+"strings"
+msgstr "multi"
+"line\n"
+"result"
+
+msgctxt "i18n context"
+msgid "with context?"
+msgstr "oder doch nicht"]]))
+ end)
+ it("should reject invalid tokens", function()
+ assert.has.errors(function()
+ poconvert.from_string("", [[
+foo ""
+bar ""]])
+ end, "Invalid token: foo")
+ end)
+ it("should reject entries without a msgstr", function()
+ assert.has.errors(function()
+ poconvert.from_string("", [[msgid "foo"]])
+ end, "Missing translated string")
+ end)
+ it("should reject entries without a msgid", function()
+ assert.has.errors(function()
+ poconvert.from_string("", [[msgstr "foo"]])
+ end, "Missing untranslated string")
+ end)
+ it("should reject entries with improperly enclosed strings", function()
+ assert.has.errors(function()
+ poconvert.from_string("", [[
+msgid "foo"
+msgstr "bar \]])
+ end, "String extends beyond the end of input")
+ end)
+ it("should reject incomplete input", function()
+ assert.has.errors(function()
+ poconvert.from_string("", [[
+msgid "foo"
+msgstr]])
+ end, "No string provided for msgstr")
+ end)
+end)
diff --git a/advtrains/spec/texture_spec.lua b/advtrains/spec/texture_spec.lua
new file mode 100644
index 0000000..2e3bd5d
--- /dev/null
+++ b/advtrains/spec/texture_spec.lua
@@ -0,0 +1,19 @@
+package.path = "../?.lua;" .. package.path
+local T = require "texture"
+
+describe("Texture creation", function()
+ it("works", function()
+ assert.same("^.png", tostring(T.raw"^.png"))
+ assert.same("foo\\:bar.png", tostring(T"foo:bar.png"))
+ end)
+end)
+
+describe("Texture modifiers", function()
+ it("work", function()
+ assert.same("x^[colorize:c", tostring(T"x":colorize"c"))
+ assert.same("x^[colorize:c:alpha", tostring(T"x":colorize("c", "alpha")))
+ assert.same("x^[multiply:c", tostring(T"x":multiply"c"))
+ assert.same("x^[resize:2x3", tostring(T"x":resize(2, 3)))
+ assert.same("x^[transformI", tostring(T"x":transform"I"))
+ end)
+end)
diff --git a/advtrains/spec/wagons_spec.lua b/advtrains/spec/wagons_spec.lua
new file mode 100644
index 0000000..df0687b
--- /dev/null
+++ b/advtrains/spec/wagons_spec.lua
@@ -0,0 +1,40 @@
+require "mineunit"
+mineunit "core"
+
+_G.advtrains = {
+ wagon_load_range = 32
+}
+sourcefile "wagons"
+
+local myproto = {_test = true}
+advtrains.register_wagon(":mywagon", myproto, "My wagon", "", false)
+advtrains.register_wagon_alias(":myalias", ":mywagon")
+advtrains.register_wagon_alias(":myotheralias", ":myalias")
+
+local myotherproto = {_other = true}
+advtrains.register_wagon(":noalias", myotherproto, "Not aliased wagon", "", false)
+advtrains.register_wagon_alias(":noalias", ":mywagon")
+
+advtrains.register_wagon_alias(":nilalias", ":nil")
+
+advtrains.register_wagon_alias(":R1", ":R2")
+advtrains.register_wagon_alias(":R2", ":R3")
+advtrains.register_wagon_alias(":R3", ":R1")
+
+describe("wagon alias system", function()
+ it("should work", function()
+ assert.same({":mywagon", myproto}, {advtrains.resolve_wagon_alias(":myalias")})
+ assert.equal(myproto, advtrains.wagon_prototypes[":myalias"])
+ assert.same({":mywagon", myproto}, {advtrains.resolve_wagon_alias(":myotheralias")})
+ end)
+ it("should respect wagon registration", function()
+ assert.same({":noalias", myotherproto}, {advtrains.resolve_wagon_alias(":noalias")})
+ end)
+ it("should handle recursive loops", function()
+ assert.same({}, {advtrains.resolve_wagon_alias(":R1")})
+ end)
+ it("should return nil for missing entries", function()
+ assert.same({}, {advtrains.resolve_wagon_alias(":what")})
+ assert.same({}, {advtrains.resolve_wagon_alias(":nilalias")})
+ end)
+end)
diff --git a/advtrains/texture.lua b/advtrains/texture.lua
new file mode 100644
index 0000000..e6d83b0
--- /dev/null
+++ b/advtrains/texture.lua
@@ -0,0 +1,228 @@
+local tx = {}
+setmetatable(tx, {__call = function(_, ...) return tx.base(...) end})
+
+function tx.escape(str)
+ return (string.gsub(tostring(str), [[([%^:\])]], [[\%1]]))
+end
+
+local function getargs(...)
+ return select("#", ...), {...}
+end
+
+local function curry(f, x)
+ return function(...)
+ return f(x, ...)
+ end
+end
+
+local function xmkmodifier(func)
+ return function(self, ...)
+ table.insert(self, (func(...)))
+ return self
+ end
+end
+
+local function mkmodifier(fmt, spec)
+ return xmkmodifier(function(...)
+ local count = select("#", ...)
+ local args = {...}
+ for k, f in pairs(spec) do
+ args[k] = f(args[k])
+ end
+ return string.format(fmt, unpack(args, 1, count))
+ end)
+end
+
+-- Texture object
+local tx_lib = {}
+local tx_mt = {
+ __index = tx_lib,
+ __tostring = function(self)
+ return table.concat(self, "^")
+ end,
+ __concat = function(a, b)
+ return tx.raw(("%s^%s"):format(tostring(a), tostring(b)))
+ end,
+}
+
+function tx.raw(str)
+ return setmetatable({str}, tx_mt)
+end
+function tx.base(str)
+ return tx.raw(tx.escape(str))
+end
+-- TODO: use [fill when 5.8.0 becomes widely used client-side
+function tx.fill(w, h, color)
+ return tx"advtrains_hud_bg.png":resize(w, h):colorize(color)
+end
+
+-- Most texture modifiers
+tx_lib.colorize = xmkmodifier(function(c, a)
+ local str = ("[colorize:%s"):format(tx.escape(c))
+ if a then
+ str = str .. ":" .. a
+ end
+ return str
+end)
+tx_lib.multiply = mkmodifier("[multiply:%s", {tx.escape})
+tx_lib.resize = mkmodifier("[resize:%dx%d", {})
+tx_lib.transform = mkmodifier("[transform%s", {tx.escape})
+
+-- [combine
+
+local combine = {}
+
+function combine:add(x, y, ent)
+ table.insert(self.st, ([[%d,%d=%s]]):format(x, y, tx.escape(tostring(ent))))
+ return self
+end
+
+local combine_mt = {
+ __index = combine,
+ __tostring = function(self)
+ return table.concat(self.st, ":")
+ end,
+}
+
+function tx.combine(w, h, bg)
+ local base = ("[combine:%dx%d"):format(w, h)
+ local obj = setmetatable({width = w, height = h, st = {base}}, combine_mt)
+ if bg then
+ obj:add_fill(0, 0, w, h, bg)
+ end
+ return obj
+end
+
+function combine:add_fill(x, y, ...)
+ return self:add(x, y, tx.fill(...))
+end
+
+local function add_multicolor_fill(n, self, x, y, w, h, ...)
+ local argc, argv = getargs(...)
+ local t = 0
+ for k = 1, argc, 2 do
+ t = t + argv[k]
+ end
+ local newargs = {x, y, w, h}
+ local sk, wk = n, n+2
+ local s = newargs[wk]/t
+ for k = 1, argc, 2 do
+ local v = argv[k] * s
+ newargs[wk] = v
+ newargs[5] = argv[k+1]
+ self:add_fill(unpack(newargs))
+ newargs[sk] = newargs[sk] + v
+ end
+ return self
+end
+combine.add_multicolor_fill_topdown = curry(add_multicolor_fill, 2)
+combine.add_multicolor_fill_leftright = curry(add_multicolor_fill, 1)
+
+local function add_segmentbar(n, self, x, y, w, h, m, c, ...)
+ local argc, argv = getargs(...)
+ local baseargs = {x, y, w, h}
+ local ss = (baseargs[n+2]+m)/c
+ local bs = ss - m
+ for k = 1, argc, 3 do
+ local lower, upper, fill = argv[k], argv[k+1], argv[k+2]
+ lower = math.max(0, math.floor(lower))+1
+ upper = math.min(c, math.floor(upper))
+ if lower <= upper then
+ local args = {x, y, w, h, fill}
+ args[n+2] = bs
+ args[n] = args[n] + ss*(lower-1)
+ for i = lower, upper do
+ self:add_fill(unpack(args))
+ args[n] = args[n] + ss
+ end
+ end
+ end
+ return self
+end
+combine.add_segmentbar_topdown = curry(add_segmentbar, 2)
+combine.add_segmentbar_leftright = curry(add_segmentbar, 1)
+
+local function add_lever(n, self, x, y, w, h, hs, ss, val, hf, sf)
+ local baseargs = {x, y, w, h}
+ local sargs = {x, y, w, h, sf}
+ sargs[5-n] = ss
+ sargs[n+2] = baseargs[n+2] + ss - hs
+ for k = 1, 2 do
+ sargs[k] = baseargs[k] + (baseargs[k+2] - sargs[k+2])/2
+ end
+ self:add_fill(unpack(sargs))
+ local hargs = {x, y, w, h, hf}
+ hargs[n+2] = hs
+ hargs[n] = baseargs[n] + (baseargs[n+2]-hs)*val
+ self:add_fill(unpack(hargs))
+ return self
+end
+combine.add_lever_topdown = curry(add_lever, 2)
+combine.add_lever_leftright = curry(add_lever, 1)
+
+--[[ Seven-segment display
+ -1-
+6 2
+ -7-
+5 3
+ -4-
+--]]
+local sevenseg_digits = {
+ ["0"] = {1, 2, 3, 4, 5, 6},
+ ["1"] = {2, 3},
+ ["2"] = {1, 2, 4, 5, 7},
+ ["3"] = {1, 2, 3, 4, 7},
+ ["4"] = {2, 3, 6, 7},
+ ["5"] = {1, 3, 4, 6, 7},
+ ["6"] = {1, 3, 4, 5, 6, 7},
+ ["7"] = {1, 2, 3},
+ ["8"] = {1, 2, 3, 4, 5, 6, 7},
+ ["9"] = {1, 2, 3, 4, 6, 7},
+}
+
+function combine:add_str7seg(x0, y0, tw, th, str, fill)
+ --[[ w and h (as width/height of individual (horizontal) segments) have the following properties:
+ tw = n(w+3h)-h
+ th = 2w+3h
+ --]]
+ local len = #str
+ local h = (2*tw-len*th)/(3*len-2)
+ local w = (th-3*h)/2
+ local ws = w+3*h
+ local segs = {
+ {h, 0, w, h},
+ {w+h, h, h, w},
+ {w+h, w+2*h, h, w},
+ {h, 2*(w+h), w, h},
+ {0, w+2*h, h, w},
+ {0, h, h, w},
+ {h, w+h, w, h},
+ }
+ for i = 1, len do
+ for _, k in pairs(sevenseg_digits[string.sub(str, i, i)] or {}) do
+ local s = segs[k]
+ self:add_fill(s[1]+x0, s[2]+y0, s[3], s[4], fill)
+ end
+ x0 = x0 + ws
+ end
+ return self
+end
+
+function combine:add_n7seg(x, y, w, h, n, prec, ...)
+ if not (type(n) == "number" and type(prec) == "number") then
+ error("passed non-numeric value or precision to numeric display")
+ elseif prec < 0 then
+ error("negative length")
+ end
+ local pfx = ""
+ if n >= 0 then
+ n = math.min(10^prec-1, n)
+ else
+ n = math.min(10^(prec-1)-1, -n)
+ pfx = "-"
+ end
+ local str = ("%d"):format(n)
+ return self:add_str7seg(x, y, w, h, pfx .. ("0"):rep(prec-#str-#pfx) .. str, ...)
+end
+
+return tx
diff --git a/advtrains/textures/advtrains_wagon_prop_tool.png b/advtrains/textures/advtrains_wagon_prop_tool.png
new file mode 100644
index 0000000..6352c55
--- /dev/null
+++ b/advtrains/textures/advtrains_wagon_prop_tool.png
Binary files differ
diff --git a/advtrains/track_reg_helper.lua b/advtrains/track_reg_helper.lua
new file mode 100644
index 0000000..97ab069
--- /dev/null
+++ b/advtrains/track_reg_helper.lua
@@ -0,0 +1,779 @@
+-- New track registration helper
+-- Retains the old table-template-based definition format, but adapts it to the new (advtrains 2.5)
+-- track definition system
+-- Note to future: This is actually just a work-saver, avoiding me to port over all the crossing nodes as well as the linetrack tracks.
+-- Future track mods should please directly use the appropriate advtrains.register_node_4rot() API and not rely on this!
+
+--definition preparation
+local function conns(c1, c2, r1, r2) return {{c=c1, y=r1}, {c=c2, y=r2}} end
+local function conns3(c1, c2, c3, r1, r2, r3) return {{c=c1, y=r1}, {c=c2, y=r2}, {c=c3, y=r3}} end
+
+advtrains.ap={}
+advtrains.ap.t_30deg_flat={
+ v25_format = true,
+ regstep=1,
+ variant={
+ st={
+ conns = conns(0,8),
+ desc = "straight",
+ tpdouble = true,
+ tpsingle = true,
+ trackworker = "cr",
+ },
+ cr={
+ conns = conns(0,7),
+ desc = "curve",
+ tpdouble = true,
+ trackworker = "swlst",
+ },
+ swlst={
+ conns = conns3(0,8,7),
+ desc = "left switch (straight)",
+ trackworker = "swrst",
+ switchalt = "cr",
+ switchmc = "on",
+ switchst = "st",
+ switchprefix = "swl",
+ conn_map = {2,1,1},
+ stmref = "swl",
+ },
+ swlcr={
+ conns = conns3(0,8,7),
+ desc = "left switch (curve)",
+ trackworker = "swrcr",
+ switchalt = "st",
+ switchmc = "off",
+ switchst = "cr",
+ switchprefix = "swl",
+ conn_map = {3,1,1},
+ stmref = "swl",
+ },
+ swrst={
+ conns = conns3(0,8,9),
+ desc = "right switch (straight)",
+ trackworker = "st",
+ switchalt = "cr",
+ switchmc = "on",
+ switchst = "st",
+ switchprefix = "swr",
+ conn_map = {2,1,1},
+ stmref = "swr",
+ },
+ swrcr={
+ conns = conns3(0,8,9),
+ desc = "right switch (curve)",
+ trackworker = "st",
+ switchalt = "st",
+ switchmc = "off",
+ switchst = "cr",
+ switchprefix = "swr",
+ conn_map = {3,1,1},
+ stmref = "swr",
+ },
+ },
+ regtp=true,
+ tpdefault="st",
+ trackworker={
+ ["swrcr"]="st",
+ ["swrst"]="st",
+ ["cr"]="swlst",
+ ["swlcr"]="swrcr",
+ ["swlst"]="swrst",
+ },
+ rotation={"", "_30", "_45", "_60"},
+ statemaps = {
+ swl = { st = "swlst", cr = "swlcr"},
+ swr = { st = "swrst", cr = "swrcr"}
+ }
+}
+advtrains.ap.t_yturnout={
+ v25_format = true,
+ regstep=1,
+ variant={
+ l={
+ conns = conns3(0,7,9),
+ desc = "Y-turnout (left)",
+ switchalt = "r",
+ switchmc = "off",
+ switchst = "l",
+ switchprefix = "",
+ conn_map = {2,1,1},
+ stmref = "sw",
+ tpsingle = true,
+ },
+ r={
+ conns = conns3(0,7,9),
+ desc = "Y-turnout (right)",
+ switchalt = "l",
+ switchmc = "on",
+ switchst = "r",
+ switchprefix = "",
+ conn_map = {3,1,1},
+ stmref = "sw",
+ }
+ },
+ regtp=true,
+ tpdefault="l",
+ rotation={"", "_30", "_45", "_60"},
+ statemaps = {
+ sw = { l = "l", r = "r"}
+ }
+}
+advtrains.ap.t_s3way={
+ v25_format = true,
+ regstep=1,
+ variant={
+ l={
+ conns = { {c=0}, {c=7}, {c=8}, {c=9}},
+ desc = "3-way turnout (left)",
+ switchalt = "s",
+ switchst="l",
+ switchprefix = "",
+ conn_map = {2,1,1,1},
+ stmref = "sw",
+ },
+ s={
+ conns = { {c=0}, {c=7}, {c=8}, {c=9}},
+ desc = "3-way turnout (straight)",
+ switchalt ="r",
+ switchst = "s",
+ switchprefix = "",
+ conn_map = {3,1,1,1},
+ stmref = "sw",
+ tpsingle = true,
+ },
+ r={
+ conns = { {c=0}, {c=7}, {c=8}, {c=9}},
+ desc = "3-way turnout (right)",
+ switchalt = "l",
+ switchst="r",
+ switchprefix = "",
+ conn_map = {4,1,1,1},
+ stmref = "sw",
+ }
+ },
+ regtp=true,
+ tpdefault="l",
+ rotation={"", "_30", "_45", "_60"},
+ statemaps = {
+ sw = { l = "l", s = "s", r = "r"}
+ }
+}
+advtrains.ap.t_30deg_slope={
+ v25_format = true,
+ regstep=1,
+ variant={
+ vst1={conns = conns(8,0,0,0.5), rail_y = 0.25, desc = "steep uphill 1/2", slope=true},
+ vst2={conns = conns(8,0,0.5,1), rail_y = 0.75, desc = "steep uphill 2/2", slope=true},
+ vst31={conns = conns(8,0,0,0.33), rail_y = 0.16, desc = "uphill 1/3", slope=true},
+ vst32={conns = conns(8,0,0.33,0.66), rail_y = 0.5, desc = "uphill 2/3", slope=true},
+ vst33={conns = conns(8,0,0.66,1), rail_y = 0.83, desc = "uphill 3/3", slope=true},
+ vst41={conns = conns(8,0,0,1/4), rail_y = 1/4 - 1/8, desc = "uphill 1/4", slope=true},
+ vst42={conns = conns(8,0,1/4,2/4), rail_y = 2/4 - 1/8, desc = "uphill 2/4", slope=true},
+ vst43={conns = conns(8,0,2/4,3/4), rail_y = 3/4 - 1/8, desc = "uphill 3/4", slope=true},
+ vst44={conns = conns(8,0,3/4,1), rail_y = 1 - 1/8, desc = "uphill 4/4", slope=true},
+ vst51={conns = conns(8,0,0,1/5), rail_y = 1/5 - 1/10, desc = "uphill 1/5", slope=true},
+ vst52={conns = conns(8,0,1/5,2/5), rail_y = 2/5 - 1/10, desc = "uphill 2/5", slope=true},
+ vst53={conns = conns(8,0,2/5,3/5), rail_y = 3/5 - 1/10, desc = "uphill 3/5", slope=true},
+ vst54={conns = conns(8,0,3/5,4/5), rail_y = 4/5 - 1/10, desc = "uphill 4/5", slope=true},
+ vst55={conns = conns(8,0,4/5,1), rail_y = 5/5 - 1/10, desc = "uphill 5/5", slope=true},
+ vst61={conns = conns(8,0,0,1/6), rail_y = 1/6 - 1/12, desc = "uphill 1/6", slope=true},
+ vst62={conns = conns(8,0,1/6,2/6), rail_y = 2/6 - 1/12, desc = "uphill 2/6", slope=true},
+ vst63={conns = conns(8,0,2/6,3/6), rail_y = 3/6 - 1/12, desc = "uphill 3/6", slope=true},
+ vst64={conns = conns(8,0,3/6,4/6), rail_y = 4/6 - 1/12, desc = "uphill 4/6", slope=true},
+ vst65={conns = conns(8,0,4/6,5/6), rail_y = 5/6 - 1/12, desc = "uphill 5/6", slope=true},
+ vst66={conns = conns(8,0,5/6,1), rail_y = 6/6 - 1/12, desc = "uphill 6/6", slope=true},
+ vst71={conns = conns(8,0,0,1/7), rail_y = 1/7 - 1/14, desc = "uphill 1/7", slope=true},
+ vst72={conns = conns(8,0,1/7,2/7), rail_y = 2/7 - 1/14, desc = "uphill 2/7", slope=true},
+ vst73={conns = conns(8,0,2/7,3/7), rail_y = 3/7 - 1/14, desc = "uphill 3/7", slope=true},
+ vst74={conns = conns(8,0,3/7,4/7), rail_y = 4/7 - 1/14, desc = "uphill 4/7", slope=true},
+ vst75={conns = conns(8,0,4/7,5/7), rail_y = 5/7 - 1/14, desc = "uphill 5/7", slope=true},
+ vst76={conns = conns(8,0,5/7,6/7), rail_y = 6/7 - 1/14, desc = "uphill 6/7", slope=true},
+ vst77={conns = conns(8,0,6/7,1), rail_y = 7/7 - 1/14, desc = "uphill 7/7", slope=true},
+ vst81={conns = conns(8,0,0,1/8), rail_y = 1/8 - 1/16, desc = "uphill 1/8", slope=true},
+ vst82={conns = conns(8,0,1/8,2/8), rail_y = 2/8 - 1/16, desc = "uphill 2/8", slope=true},
+ vst83={conns = conns(8,0,2/8,3/8), rail_y = 3/8 - 1/16, desc = "uphill 3/8", slope=true},
+ vst84={conns = conns(8,0,3/8,4/8), rail_y = 4/8 - 1/16, desc = "uphill 4/8", slope=true},
+ vst85={conns = conns(8,0,4/8,5/8), rail_y = 5/8 - 1/16, desc = "uphill 5/8", slope=true},
+ vst86={conns = conns(8,0,5/8,6/8), rail_y = 6/8 - 1/16, desc = "uphill 6/8", slope=true},
+ vst87={conns = conns(8,0,6/8,7/8), rail_y = 7/8 - 1/16, desc = "uphill 7/8", slope=true},
+ vst88={conns = conns(8,0,7/8,1), rail_y = 8/8 - 1/16, desc = "uphill 8/8", slope=true},
+ },
+ regsp=true,
+ slopeplacer={
+ [2]={"vst1", "vst2"},
+ [3]={"vst31", "vst32", "vst33"},
+ [4]={"vst41", "vst42", "vst43", "vst44"},
+ [5]={"vst51", "vst52", "vst53", "vst54", "vst55"},
+ [6]={"vst61", "vst62", "vst63", "vst64", "vst65", "vst66"},
+ [7]={"vst71", "vst72", "vst73", "vst74", "vst75", "vst76", "vst77"},
+ [8]={"vst81", "vst82", "vst83", "vst84", "vst85", "vst86", "vst87", "vst88"},
+ max=8,--highest entry
+ },
+ slopeplacer_45={
+ [2]={"vst1_45", "vst2_45"},
+ max=2,
+ },
+ rotation={"", "_30", "_45", "_60"},
+ trackworker={},
+ increativeinv={},
+}
+advtrains.ap.t_30deg_straightonly={
+ v25_format = true,
+ regstep=1,
+ variant={
+ st={
+ conns = conns(0,8),
+ desc = "straight",
+ tpdouble = true,
+ tpsingle = true,
+ trackworker = "st",
+ },
+ },
+ regtp=true,
+ tpdefault="st",
+ rotation={"", "_30", "_45", "_60"},
+}
+advtrains.ap.t_30deg_straightonly_noplacer={
+ v25_format = true,
+ regstep=1,
+ variant={
+ st={
+ conns = conns(0,8),
+ desc = "straight",
+ tpdouble = true,
+ tpsingle = true,
+ trackworker = "st",
+ },
+ },
+ tpdefault="st",
+ rotation={"", "_30", "_45", "_60"},
+}
+advtrains.ap.t_45deg={
+ v25_format = true,
+ regstep=2,
+ variant={
+ st={
+ conns = conns(0,8),
+ desc = "straight",
+ tpdouble = true,
+ tpsingle = true,
+ trackworker = "cr",
+ },
+ cr={
+ conns = conns(0,6),
+ desc = "curve",
+ tpdouble = true,
+ trackworker = "swlst",
+ },
+ swlst={
+ conns = conns3(0,8,6),
+ desc = "left switch (straight)",
+ trackworker = "swrst",
+ switchalt = "cr",
+ switchmc = "on",
+ switchst = "st",
+ },
+ swlcr={
+ conns = conns3(0,6,8),
+ desc = "left switch (curve)",
+ trackworker = "swrcr",
+ switchalt = "st",
+ switchmc = "off",
+ switchst = "cr",
+ },
+ swrst={
+ conns = conns3(0,8,10),
+ desc = "right switch (straight)",
+ trackworker = "st",
+ switchalt = "cr",
+ switchmc = "on",
+ switchst = "st",
+ },
+ swrcr={
+ conns = conns3(0,10,8),
+ desc = "right switch (curve)",
+ trackworker = "st",
+ switchalt = "st",
+ switchmc = "off",
+ switchst = "cr",
+ },
+ },
+ regtp=true,
+ tpdefault="st",
+ trackworker={
+ ["swrcr"]="st",
+ ["swrst"]="st",
+ ["cr"]="swlst",
+ ["swlcr"]="swrcr",
+ ["swlst"]="swrst",
+ },
+ rotation={"", "_30", "_45", "_60"},
+}
+advtrains.ap.t_perpcrossing={
+ v25_format = true,
+ regstep = 1,
+ variant={
+ st={
+ conns = { {c=0}, {c=8}, {c=4}, {c=12} },
+ desc = "perpendicular crossing",
+ tpdouble = true,
+ tpsingle = true,
+ trackworker = "st",
+ conn_map = {2,1,4,3},
+ },
+ },
+ regtp=true,
+ tpdefault="st",
+ rotation={"", "_30", "_45", "_60"},
+}
+advtrains.ap.t_90plusx_crossing={
+ v25_format = true,
+ regstep = 1,
+ variant={
+ ["30l"]={
+ conns = { {c=0}, {c=8}, {c=1}, {c=9} },
+ desc = "30/90 degree crossing (left)",
+ tpdouble = true,
+ tpsingle = true,
+ trackworker = "45l",
+ conn_map = {2,1,4,3},
+ },
+ ["45l"]={
+ conns = { {c=0}, {c=8}, {c=2}, {c=10} },
+ desc = "45/90 degree crossing (left)",
+ tpdouble = true,
+ tpsingle = true,
+ trackworker = "60l",
+ conn_map = {2,1,4,3},
+ },
+ ["60l"]={
+ conns = { {c=0}, {c=8}, {c=3}, {c=11}},
+ desc = "60/90 degree crossing (left)",
+ tpdouble = true,
+ tpsingle = true,
+ trackworker = "60r",
+ conn_map = {2,1,4,3},
+ },
+ ["60r"]={
+ conns = { {c=0}, {c=8}, {c=5}, {c=13} },
+ desc = "60/90 degree crossing (right)",
+ tpdouble = true,
+ tpsingle = true,
+ trackworker = "45r",
+ conn_map = {2,1,4,3},
+ },
+ ["45r"]={
+ conns = { {c=0}, {c=8}, {c=6}, {c=14} },
+ desc = "45/90 degree crossing (right)",
+ tpdouble = true,
+ tpsingle = true,
+ trackworker = "30r",
+ conn_map = {2,1,4,3},
+ },
+ ["30r"]={
+ conns = { {c=0}, {c=8}, {c=7}, {c=15}},
+ desc = "30/90 degree crossing (right)",
+ tpdouble = true,
+ tpsingle = true,
+ trackworker = "30l",
+ conn_map = {2,1,4,3},
+ },
+ },
+ regtp=true,
+ tpdefault="30l",
+ rotation={""},
+ trackworker = {
+ ["30l"] = "45l",
+ ["45l"] = "60l",
+ ["60l"] = "60r",
+ ["60r"] = "45r",
+ ["45r"] = "30r",
+ ["30r"] = "30l",
+ }
+}
+
+advtrains.ap.t_diagonalcrossing = {
+ v25_format = true,
+ regstep=1,
+ variant={
+ ["30l45r"]={
+ conns = {{c=1}, {c=9}, {c=6}, {c=14}},
+ desc = "30left-45right diagonal crossing",
+ tpdouble=true,
+ tpsingle=true,
+ trackworker="60l30l",
+ conn_map = {2,1,4,3},
+ },
+ ["60l30l"]={
+ conns = {{c=3}, {c=11}, {c=1}, {c=9}},
+ desc = "30left-60right diagonal crossing",
+ tpdouble=true,
+ tpsingle=true,
+ trackworker="60l45r",
+ conn_map = {2,1,4,3},
+ },
+ ["60l45r"]={
+ conns = {{c=3}, {c=11}, {c=6}, {c=14}},
+ desc = "60left-45right diagonal crossing",
+ tpdouble=true,
+ tpsingle=true,
+ trackworker="60l60r",
+ conn_map = {2,1,4,3},
+ },
+ ["60l60r"]={
+ conns = {{c=3}, {c=11}, {c=5}, {c=13}},
+ desc = "60left-60right diagonal crossing",
+ tpdouble=true,
+ tpsingle=true,
+ trackworker="60r45l",
+ conn_map = {2,1,4,3},
+ },
+ --If 60l60r had a mirror image, it would be here, but it's symmetric.
+ -- 60l60r is also equivalent to 30l30r but rotated 90 degrees.
+ ["60r45l"]={
+ conns = {{c=5}, {c=13}, {c=2}, {c=10}},
+ desc = "60right-45left diagonal crossing",
+ tpdouble=true,
+ tpsingle=true,
+ trackworker="60r30r",
+ conn_map = {2,1,4,3},
+ },
+ ["60r30r"]={
+ conns = {{c=5}, {c=13}, {c=7}, {c=15}},
+ desc = "60right-30right diagonal crossing",
+ tpdouble=true,
+ tpsingle=true,
+ trackworker="30r45l",
+ conn_map = {2,1,4,3},
+ },
+ ["30r45l"]={
+ conns = {{c=7}, {c=15}, {c=2}, {c=10}},
+ desc = "30right-45left diagonal crossing",
+ tpdouble=true,
+ tpsingle=true,
+ trackworker="30l45r",
+ conn_map = {2,1,4,3},
+ },
+
+ },
+ regtp=true,
+ tpdefault="30l45r",
+ rotation={""},
+ trackworker = {
+ ["30l45r"] = "60l30l",
+ ["60l30l"] = "60l45r",
+ ["60l45r"] = "60l60r",
+ ["60l60r"] = "60r45l",
+ ["60r45l"] = "60r30r",
+ ["60r30r"] = "30r45l",
+ ["30r45l"] = "30l45r",
+ }
+}
+
+advtrains.trackpresets = advtrains.ap
+
+--definition format: ([] optional)
+--[[{
+ nodename_prefix
+ texture_prefix
+ [shared_texture]
+ models_prefix
+ models_suffix (with dot)
+ [shared_model]
+ formats={
+ st,cr,swlst,swlcr,swrst,swrcr,vst1,vst2
+ (each a table with indices 0-3, for if to register a rail with this 'rotation' table entry. nil is assumed as 'all', set {} to not register at all)
+ }
+ common={} change something on common rail appearance
+}
+[18.12.17] Note on new connection system:
+In order to support real rail crossing nodes and finally make the trackplacer respect switches, I changed the connection system.
+There can be a variable number of connections available. These are specified as tuples {c=<connection>, y=<rely>}
+The table "at_conns" consists of {<conn1>, <conn2>...}
+the "at_rail_y" property holds the value that was previously called "railheight"
+Depending on the number of connections:
+2 conns: regular rail
+3 conns: switch:
+ - when train passes in at conn1, will move out of conn2
+ - when train passes in at conn2 or conn3, will move out of conn1
+4 conns: cross (or cross switch, depending on arrangement of conns):
+ - conn1 <> conn2
+ - conn3 <> conn4
+]]
+
+-- Notify the user if digging the rail is not allowed
+local function can_dig_callback(pos, player)
+ local ok, reason = advtrains.can_dig_or_modify_track(pos)
+ if not ok and player then
+ minetest.chat_send_player(player:get_player_name(), attrans("This track can not be removed!") .. " " .. reason)
+ end
+ return ok
+end
+
+local function append_statemap_suffix(state_map, nnpref, rot)
+ local t = {}
+ for state, nn in pairs(state_map) do
+ t[state] = nnpref .. "_" .. nn .. rot
+ end
+ return t
+end
+
+function advtrains.register_tracks(tracktype, def, preset)
+ if not preset.v25_format then
+ error("advtrains.register_tracks(): A track preset for pre-v2.5 is used with advtrains 2.5+. Mod probably defines own track preset instead of using it from the advtrains.ap table! Please check track mod compatibility!")
+ end
+
+ if preset.regtp then
+ local nnprefix = def.nodename_prefix
+ minetest.register_craftitem(":"..nnprefix.."_placer", {
+ description = def.description,
+ inventory_image = def.texture_prefix.."_placer.png",
+ wield_image = def.texture_prefix.."_placer.png",
+ groups={advtrains_trackplacer=1, digtron_on_place=1},
+ liquids_pointable = false,
+ on_place = function(itemstack, placer, pointed_thing)
+ local name = placer:get_player_name()
+ if not name then
+ return itemstack, false
+ end
+ if pointed_thing.type=="node" then
+ local pos=pointed_thing.above
+ local upos=vector.subtract(pointed_thing.above, {x=0, y=1, z=0})
+ if not advtrains.check_track_protection(pos, name) then
+ return itemstack, false
+ end
+ if minetest.registered_nodes[minetest.get_node(pos).name] and minetest.registered_nodes[minetest.get_node(pos).name].buildable_to then
+ local s = minetest.registered_nodes[minetest.get_node(upos).name] and minetest.registered_nodes[minetest.get_node(upos).name].walkable
+ if s then
+ -- minetest.chat_send_all(nnprefix)
+ local yaw = placer:get_look_horizontal()
+ advtrains.trackplacer.place_track(pos, nnprefix, name, yaw)
+ if not advtrains.is_creative(name) then
+ itemstack:take_item()
+ end
+ end
+ end
+ end
+ return itemstack, true
+ end,
+ })
+
+ advtrains.trackplacer.set_default_place_candidate(def.nodename_prefix, def.nodename_prefix.."_"..preset.tpdefault)
+ end
+ if preset.regsp then
+ advtrains.slope.register_placer(def, preset)
+ end
+
+ for suffix, var in pairs(preset.variant) do
+ for rotid, rotation in ipairs(preset.rotation) do
+ if not def.formats[suffix] or def.formats[suffix][rotid] then
+ local img_suffix = suffix..rotation
+ local ndef = advtrains.merge_tables({
+ description=def.description.."("..(var.desc or "any")..rotation..")",
+ drawtype = "mesh",
+ paramtype="light",
+ paramtype2="facedir",
+ use_texture_alpha = "clip",
+ walkable = false,
+ selection_box = {
+ type = "fixed",
+ fixed = {-1/2-1/16, -1/2, -1/2, 1/2+1/16, -1/2+2/16, 1/2},
+ },
+
+ mesh = def.shared_model or (def.models_prefix.."_"..img_suffix..def.models_suffix),
+ tiles = {def.shared_texture or (def.texture_prefix.."_"..img_suffix..".png"), def.second_texture},
+
+ groups = {
+ attached_node = advtrains.IGNORE_WORLD and 0 or 1,
+ advtrains_track=1,
+ ["advtrains_track_"..tracktype]=1,
+ save_in_at_nodedb=1,
+ dig_immediate=2,
+ not_in_creative_inventory=1,
+ not_blocking_trains=1,
+ },
+
+ can_dig = advtrains.track_can_dig_callback,
+ after_dig_node = advtrains.track_update_callback,
+ after_place_node = advtrains.track_update_callback,
+
+ at_rail_y = var.rail_y
+ }, def.common or {})
+
+ if preset.regtp then
+ ndef.drop = def.nodename_prefix.."_placer"
+ end
+ if preset.regsp and var.slope then
+ ndef.drop = def.nodename_prefix.."_slopeplacer"
+ end
+
+ --connections
+ ndef.at_conns = advtrains.rotate_conn_by(var.conns, (rotid-1)*preset.regstep)
+ -- NEW since 2.5
+ ndef.at_conn_map = var.conn_map
+
+ local ndef_avt_table = {}
+
+ if var.switchalt and var.switchst then
+ -- NEW since 2.5
+ ndef.on_rightclick = advtrains.state_node_on_rightclick_callback
+ ndef_avt_table.node_state = var.switchst
+ ndef_avt_table.node_next_state = var.switchalt
+ -- obtain and build statemap
+ local state_map = preset.statemaps[var.stmref]
+ if not state_map then error("On registering "..def.nodename_prefix.."_"..suffix..rotation..", stmref of variant doesn't reference entry in preset.statemaps") end
+ ndef_avt_table.node_state_map = append_statemap_suffix(state_map, def.nodename_prefix, rotation)
+
+ if var.switchmc then
+ local vswitchalt = var.switchalt
+ ndef.mesecons = {effector = {
+ ["action_"..var.switchmc] = function(pos, node)
+ advtrains.setstate(pos, vswitchalt, node)
+ end,
+ rules=advtrains.meseconrules
+ }}
+ end
+ end
+
+ local adef={}
+ if def.get_additional_definiton then
+ adef=def.get_additional_definiton(def, preset, suffix, rotation)
+ end
+ ndef = advtrains.merge_tables(ndef, adef)
+
+ -- insert getstate/setstate functions after merging the additional definitions
+ if ndef_avt_table then
+ ndef.advtrains = advtrains.merge_tables(ndef.advtrains or {}, ndef_avt_table)
+ end
+
+ -- NEW since 2.5: add appropriate fields for trackworker rotation
+ -- get the next rotation step
+ local num_rots = #preset.rotation
+ if rotid >= num_rots then
+ ndef.advtrains.trackworker_next_rot = def.nodename_prefix.."_"..suffix..preset.rotation[1]
+ ndef.advtrains.trackworker_rot_incr_param2 = true
+ else
+ ndef.advtrains.trackworker_next_rot = def.nodename_prefix.."_"..suffix..preset.rotation[rotid+1]
+ end
+ if var.trackworker then
+ ndef.advtrains.trackworker_next_var = def.nodename_prefix.."_"..var.trackworker..rotation
+ end
+
+ local the_node_name = def.nodename_prefix.."_"..suffix..rotation
+
+ --trackplacer
+ if preset.regtp and (var.tpsingle or var.tpdouble) then
+ advtrains.trackplacer.register_candidate(
+ def.nodename_prefix,
+ the_node_name,
+ ndef,
+ var.tpsingle,
+ var.tpdouble,
+ true)
+ end
+
+ -- All set, go ahead and register node!
+ --atdebug("track_reg_helper: Registering ",the_node_name," as",ndef)
+ minetest.register_node(":"..the_node_name, ndef)
+
+ end
+ end
+ end
+end
+
+-- slope placer. Defined in register_tracks.
+--crafted with rail and gravel
+local sl={}
+function sl.register_placer(def, preset)
+ minetest.register_craftitem(":"..def.nodename_prefix.."_slopeplacer",{
+ description = attrans("@1 Slope", def.description),
+ inventory_image = def.texture_prefix.."_slopeplacer.png",
+ wield_image = def.texture_prefix.."_slopeplacer.png",
+ groups={},
+ on_place = sl.create_slopeplacer_on_place(def, preset)
+ })
+end
+--(itemstack, placer, pointed_thing)
+function sl.create_slopeplacer_on_place(def, preset)
+ return function(istack, player, pt)
+ if not pt.type=="node" then
+ minetest.chat_send_player(player:get_player_name(), attrans("Can't place: not pointing at node"))
+ return istack
+ end
+ local pos=pt.above
+ if not pos then
+ minetest.chat_send_player(player:get_player_name(), attrans("Can't place: not pointing at node"))
+ return istack
+ end
+ local node=minetest.get_node(pos)
+ if not minetest.registered_nodes[node.name] or not minetest.registered_nodes[node.name].buildable_to then
+ minetest.chat_send_player(player:get_player_name(), attrans("Can't place: space occupied!"))
+ return istack
+ end
+ if not advtrains.check_track_protection(pos, player:get_player_name()) then
+ minetest.record_protection_violation(pos, player:get_player_name())
+ return istack
+ end
+ --determine player orientation (only horizontal component)
+ --get_look_horizontal may not be available
+ local yaw=player.get_look_horizontal and player:get_look_horizontal() or (player:get_look_yaw() - math.pi/2)
+
+ --rounding unit vectors is a nice way for selecting 1 of 8 directions since sin(30°) is 0.5.
+ local dirvec={x=math.floor(math.sin(-yaw)+0.5), y=0, z=math.floor(math.cos(-yaw)+0.5)}
+ --translate to direction to look up inside the preset table
+ local param2, rot45=({
+ [-1]={
+ [-1]=2,
+ [0]=3,
+ [1]=3,
+ },
+ [0]={
+ [-1]=2,
+ [1]=0,
+ },
+ [1]={
+ [-1]=1,
+ [0]=1,
+ [1]=0,
+ },
+ })[dirvec.x][dirvec.z], dirvec.x~=0 and dirvec.z~=0
+ local lookup=preset.slopeplacer
+ if rot45 then lookup=preset.slopeplacer_45 end
+
+ --go unitvector forward and look how far the next node is
+ local step=1
+ while step<=lookup.max do
+ local node=minetest.get_node(vector.add(pos, dirvec))
+ --next node solid?
+ if not minetest.registered_nodes[node.name] or not minetest.registered_nodes[node.name].buildable_to or advtrains.is_protected(pos, player:get_player_name()) then
+ --do slopes of this distance exist?
+ if lookup[step] then
+ if minetest.settings:get_bool("creative_mode") or istack:get_count()>=step then
+ --start placing
+ local placenodes=lookup[step]
+ while step>0 do
+ minetest.set_node(pos, {name=def.nodename_prefix.."_"..placenodes[step], param2=param2})
+ if not minetest.settings:get_bool("creative_mode") then
+ istack:take_item()
+ end
+ step=step-1
+ pos=vector.subtract(pos, dirvec)
+ end
+ else
+ minetest.chat_send_player(player:get_player_name(), attrans("Can't place: Not enough slope items left (@1 required)", step))
+ end
+ else
+ minetest.chat_send_player(player:get_player_name(), attrans("Can't place: There's no slope of length @1",step))
+ end
+ return istack
+ end
+ step=step+1
+ pos=vector.add(pos, dirvec)
+ end
+ minetest.chat_send_player(player:get_player_name(), attrans("Can't place: no supporting node at upper end."))
+ return itemstack
+ end
+end
+
+advtrains.slope=sl
diff --git a/advtrains/trackdb_legacy.lua b/advtrains/trackdb_legacy.lua
deleted file mode 100644
index 99349e8..0000000
--- a/advtrains/trackdb_legacy.lua
+++ /dev/null
@@ -1,27 +0,0 @@
---trackdb_legacy.lua
---loads the (old) track database. the only use for this is to provide data for rails that haven't been written into the ndb database.
---nothing will be saved.
---if the user thinks that he has loaded every track in his world at least once, he can delete the track database.
-
---trackdb[[y][x][z]={conn1, conn2, rely1, rely2, railheight}
-
-
---trackdb keeps its own save file.
-advtrains.fpath_tdb=minetest.get_worldpath().."/advtrains_trackdb2"
-local file, err = io.open(advtrains.fpath_tdb, "r")
-if not file then
- atprint("Not loading a trackdb file.")
-else
- local tbl = minetest.deserialize(file:read("*a"))
- if type(tbl) == "table" then
- advtrains.trackdb=tbl
- atprint("Loaded trackdb file.")
- end
- file:close()
-end
-
-
-
-
-
-
diff --git a/advtrains/trackplacer.lua b/advtrains/trackplacer.lua
index 5291923..3278b8c 100644
--- a/advtrains/trackplacer.lua
+++ b/advtrains/trackplacer.lua
@@ -3,475 +3,333 @@
--all new trackplacer code
local tp={
- tracks={}
+ groups={}
}
-function tp.register_tracktype(nnprefix, n_suffix)
- if tp.tracks[nnprefix] then return end--due to the separate registration of slopes and flats for the same nnpref, definition would be overridden here. just don't.
- tp.tracks[nnprefix]={
- default=n_suffix,
- single_conn={},
- single_conn_1={},
- single_conn_2={},
- double_conn={},
- double_conn_1={},
- double_conn_2={},
- --keys:conn1_conn2 (example:1_4)
- --values:{name=x, param2=x}
- twcycle={},
- twrotate={},--indexed by suffix, list, tells order of rotations
- modify={},
- }
-end
-function tp.add_double_conn(nnprefix, suffix, rotation, conns)
- local nodename=nnprefix.."_"..suffix..rotation
- for i=0,3 do
- tp.tracks[nnprefix].double_conn[((conns.conn1+4*i)%16).."_"..((conns.conn2+4*i)%16)]={name=nodename, param2=i}
- tp.tracks[nnprefix].double_conn[((conns.conn2+4*i)%16).."_"..((conns.conn1+4*i)%16)]={name=nodename, param2=i}
- tp.tracks[nnprefix].double_conn_1[((conns.conn1+4*i)%16).."_"..((conns.conn2+4*i)%16)]={name=nodename, param2=i}
- tp.tracks[nnprefix].double_conn_2[((conns.conn2+4*i)%16).."_"..((conns.conn1+4*i)%16)]={name=nodename, param2=i}
- end
- tp.tracks[nnprefix].modify[nodename]=true
-end
-function tp.add_single_conn(nnprefix, suffix, rotation, conns)
- local nodename=nnprefix.."_"..suffix..rotation
- for i=0,3 do
- tp.tracks[nnprefix].single_conn[((conns.conn1+4*i)%16)]={name=nodename, param2=i}
- tp.tracks[nnprefix].single_conn[((conns.conn2+4*i)%16)]={name=nodename, param2=i}
- tp.tracks[nnprefix].single_conn_1[((conns.conn1+4*i)%16)]={name=nodename, param2=i}
- tp.tracks[nnprefix].single_conn_2[((conns.conn2+4*i)%16)]={name=nodename, param2=i}
- end
- tp.tracks[nnprefix].modify[nodename]=true
-end
+--[[ New in version 2.5:
+The track placer no longer uses hacky nodename pattern matching.
+The base criterion for rotating or matching tracks is the common "ndef.advtrains.track_place_group" property.
+Only rails where this field is set are considered for replacement. Other rails can still be considered for connection.
+Replacement ("bending") of rails can only happen within their respective track place group. Only two-conn rails are allowed in the trackplacer.
-function tp.add_worked(nnprefix, suffix, rotation, cycle_follows)
- tp.tracks[nnprefix].twcycle[suffix]=cycle_follows
- if not tp.tracks[nnprefix].twrotate[suffix] then tp.tracks[nnprefix].twrotate[suffix]={} end
- table.insert(tp.tracks[nnprefix].twrotate[suffix], rotation)
-end
+The track registration functions register the candidates for any given track_place_group in two separate collections:
+- double: tracks that can be used to connect both ends of the rail
+- single: tracks that will be used to connect conn1 when only a single end is to be connected
+When track placing is requested, the calling code just supplies the track_place_group to be placed.
---[[
- rewrite algorithm.
- selection criteria: these will never be changed or even selected:
- - tracks being already connected on both sides
- - tracks that are already connected on one side but are not bendable to the desired position
- the following situations can occur:
- 1. there are two more than two rails around
- 1.1 there is one or more subset(s) that can be directly connected
- -> choose the first possibility
- 2.2 not
- -> choose the first one and orient straight
- 2. there's exactly 1 rail around
- -> choose and orient straight
- 3. there's no rail around
- -> set straight
-]]
+]]--
-local function istrackandbc(pos_p, conn)
- local tpos = pos_p
- local cnode=minetest.get_node(advtrains.dirCoordSet(tpos, conn.c))
- if advtrains.is_track_and_drives_on(cnode.name, advtrains.all_tracktypes) then
- local cconns=advtrains.get_track_connections(cnode.name, cnode.param2)
- return advtrains.conn_matches_to(conn, cconns)
- end
- --try the same 1 node below
- tpos = {x=tpos.x, y=tpos.y-1, z=tpos.z}
- cnode=minetest.get_node(advtrains.dirCoordSet(tpos, conn.c))
- if advtrains.is_track_and_drives_on(cnode.name, advtrains.all_tracktypes) then
- local cconns=advtrains.get_track_connections(cnode.name, cnode.param2)
- return advtrains.conn_matches_to(conn, cconns)
- end
- return false
+local function rotate(conn, rot)
+ return (conn + rot) % 16
end
-function tp.find_already_connected(pos)
- local dnode=minetest.get_node(pos)
- local dconns=advtrains.get_track_connections(dnode.name, dnode.param2)
- local found_conn
- for connid, conn in ipairs(dconns) do
- if istrackandbc(pos, conn) then
- if found_conn then --we found one in previous iteration
- return true, true --signal that it's connected
- else
- found_conn = conn.c
- end
- end
+-- Register a track node as candidate
+-- tpg: the track place group to register the candidates for
+-- NOTE: This value is automatically added to the node definition (ndef) in field ndef.advtrains.track_place_group!
+-- name, ndef: the node name and node definition table to register
+-- as_single: whether the rail should be considered as candidate for one-endpoint connection
+-- Typically only set for the straight rail variants
+-- as_double: whether the rail should be considered as candidate for two-endpoint connection
+-- Typically set for straights and curves
+-- ignore_2conn_for_legacy_xing: skips the 2-connection assertion - ONLY for compatibility with the legacy crossing nodes, DO NOT USE!
+function tp.register_candidate(tpg, name, ndef, as_single, as_double, ignore_2conn_for_legacy_xing)
+ --atdebug("TP Register candidate:",tpg, name, as_single, as_double)
+ --get or create TP group
+ if not tp.groups[tpg] then
+ tp.groups[tpg] = {double = {}, single1 = {}, single2 = {}, default = {name = name, param2 = 0} }
+ -- note: this causes the first candidate to ever be registered to be the default (which is typically what you want)
+ -- But it can be overwritten using tp.set_default_place_candidate
end
- return found_conn
-end
-function tp.rail_and_can_be_bent(originpos, conn)
- local pos=advtrains.dirCoordSet(originpos, conn)
- local newdir=(conn+8)%16
- local node=minetest.get_node(pos)
- if not advtrains.is_track_and_drives_on(node.name, advtrains.all_tracktypes) then
- return false
+ local g = tp.groups[tpg]
+
+ -- get conns
+ if not ignore_2conn_for_legacy_xing then
+ assert(#ndef.at_conns == 2)
end
- local ndef=minetest.registered_nodes[node.name]
- local nnpref = ndef and ndef.at_nnpref
- if not nnpref then return false end
- local tr=tp.tracks[nnpref]
- if not tr then return false end
- if not tr.modify[node.name] then
- --we actually can use this rail, but only if it already points to the desired direction.
- if advtrains.is_track_and_drives_on(node.name, advtrains.all_tracktypes) then
- local cconns=advtrains.get_track_connections(node.name, node.param2)
- return advtrains.conn_matches_to(conn, cconns)
+ local c1, c2 = ndef.at_conns[1].c, ndef.at_conns[2].c
+ local is_symmetrical = (rotate(c1, 8) == c2)
+
+ -- store all possible rotations (param2 values)
+ for i=0,3 do
+ if as_double then
+ g.double[rotate(c1,i*4).."_"..rotate(c2,i*4)] = {name=name, param2=i}
+ if not is_symmetrical then
+ g.double[rotate(c2,i*4).."_"..rotate(c1,i*4)] = {name=name, param2=i}
+ -- if the track is unsymmetric (e.g. a curve), we may require the "wrong" orientation to fill a gap.
+ end
+ end
+ if as_single then
+ g.single1[rotate(c1,i*4)] = {name=name, param2=i}
+ g.single2[rotate(c2,i*4)] = {name=name, param2=i}
end
end
- -- If the rail is not allowed to be modified, also only use if already in desired direction
- if not advtrains.can_dig_or_modify_track(pos) then
- local cconns=advtrains.get_track_connections(node.name, node.param2)
- return advtrains.conn_matches_to(conn, cconns)
+
+ -- Set track place group on the node
+ if not ndef.advtrains then
+ ndef.advtrains = {}
end
- --rail at other end?
- local adj1, adj2=tp.find_already_connected(pos)
- if adj1 and adj2 then
- return false--dont destroy existing track
- elseif adj1 and not adj2 then
- if tr.double_conn[adj1.."_"..newdir] then
- return true--if exists, connect new rail and old end
- end
- return false
+ ndef.advtrains.track_place_group = tpg
+end
+
+-- Sets the node that is placed by the track placer when there is no track nearby. param2 defaults to 0
+function tp.set_default_place_candidate(tpg, name, param2)
+ if not tp.groups[tpg] then
+ tp.groups[tpg] = {double = {}, single1 = {}, single2 = {}, default = {name = name, param2 = param2 or 0} }
else
- if tr.single_conn[newdir] then--just rotate old rail to right orientation
- return true
- end
- return false
+ tp.groups[tpg].default = {name = name, param2 = param2 or 0}
end
end
-function tp.bend_rail(originpos, conn)
- local pos=advtrains.dirCoordSet(originpos, conn)
- local newdir=advtrains.oppd(conn)
- local node=minetest.get_node(pos)
- local ndef=minetest.registered_nodes[node.name]
- local nnpref = ndef and ndef.at_nnpref
- if not nnpref then return false end
- local tr=tp.tracks[nnpref]
- if not tr then return false end
- --is rail already connected? no need to bend.
- local conns=advtrains.get_track_connections(node.name, node.param2)
- if advtrains.conn_matches_to(conn, conns) then
- return
+
+local function check_or_bend_rail(origin, dir, pname, commit)
+ local pos = advtrains.pos_add_dir(origin, dir)
+ local back_dir = advtrains.oppd(dir);
+
+ local node_ok, conns = advtrains.get_rail_info_at(pos)
+ if not node_ok then
+ -- try the node one level below
+ pos.y = pos.y - 1
+ node_ok, conns = advtrains.get_rail_info_at(pos)
end
- --rail at other end?
- local adj1, adj2=tp.find_already_connected(pos)
- if adj1 and adj2 then
- return false--dont destroy existing track
- elseif adj1 and not adj2 then
- if tr.double_conn[adj1.."_"..newdir] then
- advtrains.ndb.swap_node(pos, tr.double_conn[adj1.."_"..newdir])
- return true--if exists, connect new rail and old end
- end
+ if not node_ok then
return false
- else
- if tr.single_conn[newdir] then--just rotate old rail to right orientation
- advtrains.ndb.swap_node(pos, tr.single_conn[newdir])
+ end
+ -- if one conn of the node here already points towards us, nothing to do
+ for connid, conn in ipairs(conns) do
+ if back_dir == conn.c then
return true
end
+ end
+ -- can we bend the node here?
+ local node = advtrains.ndb.get_node(pos)
+ local ndef = minetest.registered_nodes[node.name]
+ if not ndef or not ndef.advtrains or not ndef.advtrains.track_place_group then
return false
end
-end
-function tp.placetrack(pos, nnpref, placer, itemstack, pointed_thing, yaw)
- --1. find all rails that are likely to be connected
- local tr=tp.tracks[nnpref]
- local p_rails={}
- local p_railpos={}
- for i=0,15 do
- if tp.rail_and_can_be_bent(pos, i, nnpref) then
- p_rails[#p_rails+1]=i
- p_railpos[i] = pos
- else
- local upos = {x=pos.x, y=pos.y-1, z=pos.z}
- if tp.rail_and_can_be_bent(upos, i, nnpref) then
- p_rails[#p_rails+1]=i
- p_railpos[i] = upos
- end
- end
+ -- now the track must be two-conn, else it wouldn't be allowed to have track_place_group set.
+ --assert(#conns == 2) -- cannot check here, because of legacy crossing hack
+ -- Is player and game allowed to do this?
+ if not advtrains.can_dig_or_modify_track(pos) then
+ return false
end
-
- -- try double_conn
- if #p_rails > 1 then
- --iterate subsets
- for k1, conn1 in ipairs(p_rails) do
- for k2, conn2 in ipairs(p_rails) do
- if k1~=k2 then
- local dconn1 = tr.double_conn_1
- local dconn2 = tr.double_conn_2
- if not (advtrains.yawToDirection(yaw, conn1, conn2) == conn1) then
- dconn1 = tr.double_conn_2
- dconn2 = tr.double_conn_1
- end
- -- Checks are made this way round so that dconn1 has priority (this will make arrows of atc rails
- -- point in the right direction)
- local using
- if (dconn2[conn1.."_"..conn2]) then
- using = dconn2[conn1.."_"..conn2]
- end
- if (dconn1[conn1.."_"..conn2]) then
- using = dconn1[conn1.."_"..conn2]
- end
- if using then
- -- has found a fitting rail in either direction
- -- if not, continue loop
- tp.bend_rail(p_railpos[conn1], conn1, nnpref)
- tp.bend_rail(p_railpos[conn2], conn2, nnpref)
- advtrains.ndb.swap_node(pos, using)
- local nname=using.name
- if minetest.registered_nodes[nname] and minetest.registered_nodes[nname].after_place_node then
- minetest.registered_nodes[nname].after_place_node(pos, placer, itemstack, pointed_thing)
- end
- return
- end
- end
- end
+ if not advtrains.check_track_protection(pos, pname) then
+ return false
+ end
+ -- we confirmed that track can be modified. Does there exist a suitable connection candidate?
+ -- check if there are any unbound ends
+ local bound_connids = {}
+ for connid, conn in ipairs(conns) do
+ local adj_pos, adj_connid = advtrains.get_adjacent_rail(pos, conns, connid)
+ if adj_pos then
+ bound_connids[#bound_connids+1] = connid
end
end
- -- try single_conn
- if #p_rails > 0 then
- for ix, p_rail in ipairs(p_rails) do
- local sconn1 = tr.single_conn_1
- local sconn2 = tr.single_conn_2
- if not (advtrains.yawToDirection(yaw, p_rail, (p_rail+8)%16) == p_rail) then
- sconn1 = tr.single_conn_2
- sconn2 = tr.single_conn_1
- end
- if sconn1[p_rail] then
- local using = sconn1[p_rail]
- tp.bend_rail(p_railpos[p_rail], p_rail, nnpref)
- advtrains.ndb.swap_node(pos, using)
- local nname=using.name
- if minetest.registered_nodes[nname] and minetest.registered_nodes[nname].after_place_node then
- minetest.registered_nodes[nname].after_place_node(pos, placer, itemstack, pointed_thing)
- end
- return
+ -- depending on the nummber of ends, decide
+ if #bound_connids == 2 then
+ -- rail is within a fixed track, do not break up
+ return false
+ end
+ -- obtain the group table
+ local g = tp.groups[ndef.advtrains.track_place_group]
+ if #bound_connids == 1 then
+ -- we can attempt double
+ local bound_dir = conns[bound_connids[1]].c
+ if g.double[back_dir.."_"..bound_dir] then
+ if commit then
+ advtrains.ndb.swap_node(pos, g.double[back_dir.."_"..bound_dir])
end
- if sconn2[p_rail] then
- local using = sconn2[p_rail]
- tp.bend_rail(p_railpos[p_rail], p_rail, nnpref)
- advtrains.ndb.swap_node(pos, using)
- local nname=using.name
- if minetest.registered_nodes[nname] and minetest.registered_nodes[nname].after_place_node then
- minetest.registered_nodes[nname].after_place_node(pos, placer, itemstack, pointed_thing)
- end
- return
+ return true
+ end
+ else
+ -- rail is entirely unbound, we can attempt single1
+ if g.single1[back_dir] then
+ if commit then
+ advtrains.ndb.swap_node(pos, g.single1[back_dir])
end
+ return true
end
end
- --use default
- minetest.set_node(pos, {name=nnpref.."_"..tr.default})
- if minetest.registered_nodes[nnpref.."_"..tr.default] and minetest.registered_nodes[nnpref.."_"..tr.default].after_place_node then
- minetest.registered_nodes[nnpref.."_"..tr.default].after_place_node(pos, placer, itemstack, pointed_thing)
- end
end
-local function on_place(itemstack, placer, pointed_thing, nnprefix, def)
- local s, pos, name
-
- name = placer:get_player_name()
- if not name then
- return itemstack, false
- end
-
- if pointed_thing.type ~= "node" then
- return itemstack, true
- end
-
- if pointed_thing.above.y == pointed_thing.under.y then
- pos = pointed_thing.under
- else
- pos = pointed_thing.above
+local function track_place_node(pos, node, ndef_p, pname)
+ --atdebug("track_place_node: ",pos, node)
+ advtrains.ndb.swap_node(pos, node)
+ local ndef = ndef_p or minetest.registered_nodes[node.name]
+ if ndef and ndef.after_place_node then
+ -- resolve player again
+ local player = pname and core.get_player_by_name(pname) or nil
+ ndef.after_place_node(pos, player) -- note: itemstack and pointed_thing are NOT available here anymore (crap!)
end
+end
- local upos = vector.offset(pos, 0, -1, 0)
- local ndef = minetest.registered_nodes[minetest.get_node(pos).name]
- local undef = minetest.registered_nodes[minetest.get_node(upos).name] -- definition of the node under
- if not advtrains.check_track_protection(pos, name) then
- return itemstack, false
- end
- if not ndef or not ndef.buildable_to then
- return itemstack, true -- not place for a track
- end
- if def.suitable_substrate then
- s = def.suitable_substrate(upos)
- else
- s = undef and undef.walkable
- end
- if s then
- -- minetest.chat_send_all(nnprefix)
- local yaw = placer:get_look_horizontal()
- tp.placetrack(pos, nnprefix, placer, itemstack, pointed_thing, yaw)
- if not minetest.is_creative_enabled(name) then
- itemstack:take_item()
+-- Main API function to place a track. Replaces the older "placetrack"
+-- This function will attempt to place a track of the specified track placing group at the specified position, connecting it
+-- with neighboring rails. Neighboring rails can themselves be replaced ("bent") within their own track place group,
+-- if the player is permitted to do this.
+-- Order of preference is:
+-- Connect two track ends if possible
+-- Connect one track end if any rail is near
+-- Place the default track if no tracks are near
+-- The function returns true on success.
+function tp.place_track(pos, tpg, pname, yaw)
+ -- 1. collect neighboring tracks and whether they can be connected
+ --atdebug("tp.place_track(",pos, tpg, pname, yaw,")")
+ local cand = {}
+ for i=0,15 do
+ if check_or_bend_rail(pos, i, pname) then
+ cand[#cand+1] = i
end
- return itemstack, true
end
-
- -- try the position below
- pos.y = pos.y - 1
- upos.y = upos.y - 1
- ndef = undef
- undef = minetest.registered_nodes[minetest.get_node(upos).name]
- pointed_thing = table.copy(pointed_thing)
- pointed_thing.below = upos
- pointed_thing.above = pos
-
- if not advtrains.check_track_protection(pos, name) then
- return itemstack, false
- end
- if not ndef or not ndef.buildable_to then
- return itemstack, true -- not place for a track
+ --atdebug("Candidates: ",cand)
+ -- obtain the group table
+ local g = tp.groups[tpg]
+ if not g then
+ error("tp.place_track: for tpg="..tpg.." couldn't find the group table")
end
- if def.suitable_substrate then
- s = def.suitable_substrate(upos)
- else
- s = undef and undef.walkable
+ --atdebug("Group table:",g)
+ -- 2. try all possible two-endpoint connections
+ for k1, conn1 in ipairs(cand) do
+ for k2, conn2 in ipairs(cand) do
+ if k1~=k2 then
+ -- order of conn1/conn2: prefer conn1 being in the direction of the player facing.
+ -- the combination the other way round will be run through in a later loop iteration
+ if advtrains.yawToDirection(yaw, conn1, conn2) == conn1 then
+ -- does there exist a suitable double-connection rail?
+ --atdebug("Try double conn: ",conn1, conn2)
+ local node = g.double[conn1.."_"..conn2]
+ if node then
+ check_or_bend_rail(pos, conn1, pname, true)
+ check_or_bend_rail(pos, conn2, pname, true)
+ track_place_node(pos, node, nil, pname) -- calls after_place_node implicitly
+ return true
+ end
+ end
+ end
+ end
end
- if s then
- -- minetest.chat_send_all(nnprefix)
- local yaw = placer:get_look_horizontal()
- tp.placetrack(pos, nnprefix, placer, itemstack, pointed_thing, yaw)
- if not minetest.is_creative_enabled(name) then
- itemstack:take_item()
+ -- 3. try all possible one_endpoint connections
+ for k1, conn1 in ipairs(cand) do
+ -- select single1 or single2? depending on yaw
+ local single
+ if advtrains.yawToDirection(yaw, conn1, advtrains.oppd(conn1)) == conn1 then
+ single = g.single1
+ else
+ single = g.single2
+ end
+ --atdebug("Try single conn: ",conn1)
+ local node = single[conn1]
+ if node then
+ check_or_bend_rail(pos, conn1, pname, true)
+ track_place_node(pos, node, nil, pname) -- calls after_place_node implicitly
+ return true
end
end
- return itemstack, true
+ -- 4. if nothing worked, set the default
+ local node = g.default
+ track_place_node(pos, node, nil, pname) -- calls after_place_node implicitly
+ return true
end
-function tp.register_track_placer(nnprefix, imgprefix, dispname, def)
- minetest.register_craftitem(":"..nnprefix.."_placer",{
- description = dispname,
- inventory_image = imgprefix.."_placer.png",
- wield_image = imgprefix.."_placer.png",
- groups={advtrains_trackplacer=1, digtron_on_place=1},
- liquids_pointable = def.liquids_pointable,
- on_place = function(itemstack, placer, pointed_thing)
- return on_place(itemstack, placer, pointed_thing, nnprefix, def)
- end,
- })
-end
+-- TRACK WORKER --
minetest.register_craftitem("advtrains:trackworker",{
- description = attrans("Track Worker Tool\n\nLeft-click: change rail type (straight/curve/switch)\nRight-click: rotate rail/bumper/signal/etc."),
+ description = attrans("Track Worker Tool\n\nLeft-click: change rail type (straight/curve/switch)\nRight-click: rotate object"),
groups = {cracky=1}, -- key=name, value=rating; rating=1..3.
inventory_image = "advtrains_trackworker.png",
wield_image = "advtrains_trackworker.png",
stack_max = 1,
on_place = function(itemstack, placer, pointed_thing)
- local name = placer:get_player_name()
- if not name then
+ local name = placer:get_player_name()
+ if not name then
+ return
+ end
+ local has_aux1_down = placer:get_player_control().aux1
+ if pointed_thing.type=="node" then
+ local pos=pointed_thing.under
+ if not advtrains.check_track_protection(pos, name) then
return
end
- local has_aux1_down = placer:get_player_control().aux1
- if pointed_thing.type=="node" then
- local pos=pointed_thing.under
- if not advtrains.check_track_protection(pos, name) then
- return
- end
- local node=minetest.get_node(pos)
+ local node=minetest.get_node(pos)
- --if not advtrains.is_track_and_drives_on(minetest.get_node(pos).name, advtrains.all_tracktypes) then return end
-
- local nnprefix, suffix, rotation=string.match(node.name, "^(.+)_([^_]+)(_[^_]+)$")
- --atdebug(node.name.."\npattern recognizes:"..nnprefix.." / "..suffix.." / "..rotation)
- --atdebug("nntab: ",tp.tracks[nnprefix])
- if not tp.tracks[nnprefix] or not tp.tracks[nnprefix].twrotate[suffix] then
- nnprefix, suffix=string.match(node.name, "^(.+)_([^_]+)$")
- rotation = ""
- if not tp.tracks[nnprefix] or not tp.tracks[nnprefix].twrotate[suffix] then
- minetest.chat_send_player(placer:get_player_name(), attrans("This node can't be rotated using the trackworker!"))
- return
- end
- end
-
- -- check if the node is modify-protected
- if advtrains.is_track_and_drives_on(minetest.get_node(pos).name, advtrains.all_tracktypes) then
- -- is a track, we can query
- local can_modify, reason = advtrains.can_dig_or_modify_track(pos)
- if not can_modify then
- local str = attrans("This track can not be rotated!")
- if reason then
- str = str .. " " .. reason
- end
- minetest.chat_send_player(placer:get_player_name(), str)
- return
+ -- New since 2.5: only the fields in the node definition are considered, no more hacky pattern matching on the nodename
+
+ local ndef = minetest.registered_nodes[node.name]
+
+ if not ndef.advtrains or not ndef.advtrains.trackworker_next_rot then
+ minetest.chat_send_player(placer:get_player_name(), attrans("This node can't be rotated using the trackworker!"))
+ return
+ end
+
+ -- check if the node is modify-protected
+ if advtrains.is_track(node.name) then
+ -- is a track, we can query
+ local can_modify, reason = advtrains.can_dig_or_modify_track(pos)
+ if not can_modify then
+ local str = attrans("This track can not be rotated!")
+ if reason then
+ str = str .. " " .. reason
end
- end
-
- if has_aux1_down then
- --feature: flip the node by 180°
- --i've always wanted this!
- advtrains.ndb.swap_node(pos, {name=node.name, param2=(node.param2+2)%4})
+ minetest.chat_send_player(placer:get_player_name(), str)
return
end
-
- local modext=tp.tracks[nnprefix].twrotate[suffix]
-
- if rotation==modext[#modext] then --increase param2
- advtrains.ndb.swap_node(pos, {name=nnprefix.."_"..suffix..modext[1], param2=(node.param2+1)%4})
- return
- else
- local modpos
- for k,v in pairs(modext) do
- if v==rotation then modpos=k end
- end
- if not modpos then
- minetest.chat_send_player(placer:get_player_name(), attrans("This node can't be rotated using the trackworker!"))
- return
- end
- advtrains.ndb.swap_node(pos, {name=nnprefix.."_"..suffix..modext[modpos+1], param2=node.param2})
- end
end
+
+ if has_aux1_down then
+ --feature: flip the node by 180°
+ --i've always wanted this!
+ advtrains.ndb.swap_node(pos, {name=node.name, param2=(node.param2+2)%4})
+ return
+ end
+
+ local new_node = {name = ndef.advtrains.trackworker_next_rot, param2 = node.param2}
+ if ndef.advtrains.trackworker_rot_incr_param2 then
+ new_node.param2 = ((node.param2 + 1) % 4)
+ end
+ advtrains.ndb.swap_node(pos, new_node)
+ end
end,
- on_use=function(itemstack, user, pointed_thing)
- local name = user:get_player_name()
- if not name then
- return
+ on_use=function(itemstack, player, pointed_thing)
+ local name = player:get_player_name()
+ if not name then
+ return
+ end
+ if pointed_thing.type=="node" then
+ local pos=pointed_thing.under
+ local node=minetest.get_node(pos)
+ if not advtrains.check_track_protection(pos, name) then
+ return
end
- if pointed_thing.type=="node" then
- local pos=pointed_thing.under
- local node=minetest.get_node(pos)
- if not advtrains.check_track_protection(pos, name) then
- return
- end
-
- --if not advtrains.is_track_and_drives_on(minetest.get_node(pos).name, advtrains.all_tracktypes) then return end
- if advtrains.get_train_at_pos(pos) then return end
- local nnprefix, suffix, rotation=string.match(node.name, "^(.+)_([^_]+)(_[^_]+)$")
- --atdebug(node.name.."\npattern recognizes:"..nodeprefix.." / "..railtype.." / "..rotation)
- if not tp.tracks[nnprefix] or not tp.tracks[nnprefix].twcycle[suffix] then
- nnprefix, suffix=string.match(node.name, "^(.+)_([^_]+)$")
- rotation = ""
- if not tp.tracks[nnprefix] or not tp.tracks[nnprefix].twcycle[suffix] then
- minetest.chat_send_player(user:get_player_name(), attrans("This node can't be changed using the trackworker!"))
- return
- end
- end
-
- -- check if the node is modify-protected
- if advtrains.is_track_and_drives_on(minetest.get_node(pos).name, advtrains.all_tracktypes) then
- -- is a track, we can query
- local can_modify, reason = advtrains.can_dig_or_modify_track(pos)
- if not can_modify then
- local str = attrans("This track can not be changed!")
- if reason then
- str = str .. " " .. reason
- end
- minetest.chat_send_player(user:get_player_name(), str)
- return
+
+ -- New since 2.5: only the fields in the node definition are considered, no more hacky pattern matching on the nodename
+
+ local ndef = minetest.registered_nodes[node.name]
+
+ if not ndef.advtrains or not ndef.advtrains.trackworker_next_var then
+ minetest.chat_send_player(name, attrans("This node can't be changed using the trackworker!"))
+ return
+ end
+
+ -- check if the node is modify-protected
+ if advtrains.is_track(node.name) then
+ -- is a track, we can query
+ local can_modify, reason = advtrains.can_dig_or_modify_track(pos)
+ if not can_modify then
+ local str = attrans("This track can not be rotated!")
+ if reason then
+ str = str .. " " .. reason
end
+ minetest.chat_send_player(name, str)
+ return
end
-
- local nextsuffix=tp.tracks[nnprefix].twcycle[suffix]
- advtrains.ndb.swap_node(pos, {name=nnprefix.."_"..nextsuffix..rotation, param2=node.param2})
-
- else
- atprint(name, dump(tp.tracks))
end
+
+ local new_node = {name = ndef.advtrains.trackworker_next_var, param2 = node.param2}
+ advtrains.ndb.swap_node(pos, new_node)
+ end
end,
})
diff --git a/advtrains/tracks.lua b/advtrains/tracks.lua
index 3acde74..dba0d7a 100644
--- a/advtrains/tracks.lua
+++ b/advtrains/tracks.lua
@@ -1,484 +1,128 @@
---advtrains by orwell96, see readme.txt
+-- tracks.lua
+-- rewritten with advtrains 2.5 according to new track registration system
---dev-time settings:
---EDIT HERE
---If the old non-model rails on straight tracks should be replaced by the new...
---false: no
---true: yes
-advtrains.register_replacement_lbms=false
---[[TracksDefinition
-nodename_prefix
-texture_prefix
-description
-common={}
-straight={}
-straight45={}
-curve={}
-curve45={}
-lswitchst={}
-lswitchst45={}
-rswitchst={}
-rswitchst45={}
-lswitchcr={}
-lswitchcr45={}
-rswitchcr={}
-rswitchcr45={}
-vert1={
- --you'll probably want to override mesh here
-}
-vert2={
- --you'll probably want to override mesh here
-}
-]]--
-advtrains.all_tracktypes={}
+--[[
---definition preparation
-local function conns(c1, c2, r1, r2) return {{c=c1, y=r1}, {c=c2, y=r2}} end
-local function conns3(c1, c2, c3, r1, r2, r3) return {{c=c1, y=r1}, {c=c2, y=r2}, {c=c3, y=r3}} end
+Tracks in advtrains are defined by the node definition. They must have at least 2 connections, but can have any number.
+Switchable nodes (turnouts, single/double-slip switches) are implemented by having a separate node (node name) for each of the possible states.
-advtrains.ap={}
-advtrains.ap.t_30deg_flat={
- regstep=1,
- variant={
- st={
- conns = conns(0,8),
- desc = "straight",
- tpdouble = true,
- tpsingle = true,
- trackworker = "cr",
- },
- cr={
- conns = conns(0,7),
- desc = "curve",
- tpdouble = true,
- trackworker = "swlst",
- },
- swlst={
- conns = conns3(0,8,7),
- desc = "left switch (straight)",
- trackworker = "swrst",
- switchalt = "cr",
- switchmc = "on",
- switchst = "st",
- switchprefix = "swl",
- },
- swlcr={
- conns = conns3(0,7,8),
- desc = "left switch (curve)",
- trackworker = "swrcr",
- switchalt = "st",
- switchmc = "off",
- switchst = "cr",
- switchprefix = "swl",
- },
- swrst={
- conns = conns3(0,8,9),
- desc = "right switch (straight)",
- trackworker = "st",
- switchalt = "cr",
- switchmc = "on",
- switchst = "st",
- switchprefix = "swr",
- },
- swrcr={
- conns = conns3(0,9,8),
- desc = "right switch (curve)",
- trackworker = "st",
- switchalt = "st",
- switchmc = "off",
- switchst = "cr",
- switchprefix = "swr",
- },
- },
- regtp=true,
- tpdefault="st",
- trackworker={
- ["swrcr"]="st",
- ["swrst"]="st",
- ["cr"]="swlst",
- ["swlcr"]="swrcr",
- ["swlst"]="swrst",
- },
- rotation={"", "_30", "_45", "_60"},
-}
-advtrains.ap.t_yturnout={
- regstep=1,
- variant={
- l={
- conns = conns3(0,7,9),
- desc = "Y-turnout (left)",
- switchalt = "r",
- switchmc = "off",
- switchst = "l",
- switchprefix = "",
- },
- r={
- conns = conns3(0,9,7),
- desc = "Y-turnout (right)",
- switchalt = "l",
- switchmc = "on",
- switchst = "r",
- switchprefix = "",
- }
+ minetest.register_node(nodename, {
+ ... usual node definition ...
+ groups = {
+ advtrains_track = 1,
+ advtrains_track_<tracktype>=1
+ ^- these groups tell that the node is a track
+ not_blocking_trains=1,
+ ^- this group tells that the node should not block trains although it's walkable.
},
- regtp=true,
- tpdefault="l",
- rotation={"", "_30", "_45", "_60"},
-}
-advtrains.ap.t_s3way={
- regstep=1,
- variant={
- l={
- conns = { {c=0}, {c=7}, {c=8}, {c=9}, {c=0} },
- desc = "3-way turnout (left)",
- switchalt = "s",
- switchst="l",
- switchprefix = "",
- },
- s={
- conns = { {c=0}, {c=8}, {c=7}, {c=9}, {c=0} },
- desc = "3-way turnout (straight)",
- switchalt ="r",
- switchst = "s",
- switchprefix = "",
- },
- r={
- conns = { {c=0}, {c=9}, {c=8}, {c=7}, {c=0} },
- desc = "3-way turnout (right)",
- switchalt = "l",
- switchst="r",
- switchprefix = "",
+
+ at_rail_y = 0,
+ ^- Height of this rail node (the y position of a wagon that stands centered on this rail)
+ at_conns = {
+ [1] = { c=0..15, y=0..1 },
+ [2] = { c=0..15, y=0..1 },
+ ( [3] = { c=0..15, y=0..1 }, )
+ ( [4] = { c=0..15, y=0..1 }, )
+ ( ... )
+ }
+ ^- Connections of this rail. There are two general cases:
+ a) SIMPLE TRACK - the track has exactly 2 connections, and does not feature a turnout, crossing or other contraption
+ For simple tracks, except for the at_conns table no further setup needs to be specified. A train entering on conn 1 will go out at conn 2 and vice versa.
+ A track with only one connection defined is not permitted.
+ b) COMPOUND TRACK - the track has more than 2 connections
+ This will be used for turnouts and crossings. Tracks with more than 2 conns MUST define 'at_conn_map'.
+ Switchable nodes, whose state can be changed (e.g. turnouts) MUST define a 'state_map' within the advtrains table as well.
+ This differs from the behavior up until 2.4.2, where the conn mapping was fixed.
+ ^- Connection definition:
+ - c is the direction of the connection (0-16). For the mapping to world directions see helpers.lua.
+ - Connections will be auto-rotated with param2 of the node (horizontal, param2 values 0-3 only)
+ - y is the height of the connection (rail will only connect when this matches)
+ ^- The index of a connection inside the conns table (1, 2, 3, ...) is referred throughout advtrains code as 'connid'
+ ^- IMPORTANT: For switchable nodes (any kind of turnout), it is crucial that for all of the node's variants the at_conns table stays the same. See below.
+
+ at_conn_map = {
+ [1] = 2,
+ [2] = 1,
+ [3] = 1,
+ }
+ ^- Connection map of this rail. It specifies when a train enters the track on connid X, on which connid it will leave
+ This field MUST be specified when the number of connections in at_conns is greater than 2
+ This field may, and obviously will, vary between nodes for switchable nodes.
+
+ can_dig = advtrains.track_can_dig_callback
+ after_dig_node = advtrains.track_update_callback
+ after_place_node = advtrains.track_update_callback
+ ^- the code in these 3 default minetest API functions is required for advtrains to work, however you can add your own code
+
+ on_rightclick = advtrains.state_node_on_rightclick_callback
+ ^- Must be added if the node is a turnout and if it should be switched by right-click. It will cause the turnout to be switched to next_state.
+
+ advtrains = {
+ on_train_enter=function(pos, train_id, train, index) end
+ ^- called when a train enters the rail
+ on_train_leave=function(pos, train_id, train, index) end
+ ^- called when a train leaves the rail
+
+ -- The following function is only in effect when interlocking is enabled:
+ on_train_approach = function(pos, train_id, train, index, has_entered, lzbdata)
+ ^- called when a train is approaching this position, called exactly once for every path recalculation (which can happen at any time)
+ ^- This is called so that if the train would start braking now, it would come to halt about(wide approx) 5 nodes before the rail.
+ ^- has_entered: when true, the train is already standing on this node with its front tip, and the enter callback has already been called.
+ Possibly, some actions need not to be taken in this case. Only set if it's the very first node the train is standing on.
+ ^- lzbdata should be ignored and nothing should be assigned to it
+
+ -- The following information is required if the node is a turnout (e.g. can be switched into different positions)
+ node_state = "st"
+ ^- The name of the state this node represents
+ ^- Conventions for this field are as follows:
+ - Two-way straight/turn switches: 'st'=straight branch, 'cr'=diverting/turn branch
+ - 3-way turnouts, Y-turnouts: 'l'=left branch, 's'=straight branch, 'r'=right branch
+
+ node_next_state = "cr"
+ ^- The name of the state that the turnout should be switched to when it is right-clicked
+
+ node_fallback_state = "st"
+ ^- The name of the state that the turnout should "fall back" to when it is released
+ Only used by the interlocking system, when a route on the node is released it is switched back to this state.
+
+ node_state_map = {
+ ["st"] = "<node name of the st variant>",
+ ["cr"] = "<node name of the cr variant>",
+ ... etc ...
}
- },
- regtp=true,
- tpdefault="l",
- rotation={"", "_30", "_45", "_60"},
-}
-advtrains.ap.t_30deg_slope={
- regstep=1,
- variant={
- vst1={conns = conns(8,0,0,0.5), rail_y = 0.25, desc = "steep uphill 1/2", slope=true},
- vst2={conns = conns(8,0,0.5,1), rail_y = 0.75, desc = "steep uphill 2/2", slope=true},
- vst31={conns = conns(8,0,0,0.33), rail_y = 0.16, desc = "uphill 1/3", slope=true},
- vst32={conns = conns(8,0,0.33,0.66), rail_y = 0.5, desc = "uphill 2/3", slope=true},
- vst33={conns = conns(8,0,0.66,1), rail_y = 0.83, desc = "uphill 3/3", slope=true},
- vst41={conns = conns(8,0,0,1/4), rail_y = 1/4 - 1/8, desc = "uphill 1/4", slope=true},
- vst42={conns = conns(8,0,1/4,2/4), rail_y = 2/4 - 1/8, desc = "uphill 2/4", slope=true},
- vst43={conns = conns(8,0,2/4,3/4), rail_y = 3/4 - 1/8, desc = "uphill 3/4", slope=true},
- vst44={conns = conns(8,0,3/4,1), rail_y = 1 - 1/8, desc = "uphill 4/4", slope=true},
- vst51={conns = conns(8,0,0,1/5), rail_y = 1/5 - 1/10, desc = "uphill 1/5", slope=true},
- vst52={conns = conns(8,0,1/5,2/5), rail_y = 2/5 - 1/10, desc = "uphill 2/5", slope=true},
- vst53={conns = conns(8,0,2/5,3/5), rail_y = 3/5 - 1/10, desc = "uphill 3/5", slope=true},
- vst54={conns = conns(8,0,3/5,4/5), rail_y = 4/5 - 1/10, desc = "uphill 4/5", slope=true},
- vst55={conns = conns(8,0,4/5,1), rail_y = 5/5 - 1/10, desc = "uphill 5/5", slope=true},
- vst61={conns = conns(8,0,0,1/6), rail_y = 1/6 - 1/12, desc = "uphill 1/6", slope=true},
- vst62={conns = conns(8,0,1/6,2/6), rail_y = 2/6 - 1/12, desc = "uphill 2/6", slope=true},
- vst63={conns = conns(8,0,2/6,3/6), rail_y = 3/6 - 1/12, desc = "uphill 3/6", slope=true},
- vst64={conns = conns(8,0,3/6,4/6), rail_y = 4/6 - 1/12, desc = "uphill 4/6", slope=true},
- vst65={conns = conns(8,0,4/6,5/6), rail_y = 5/6 - 1/12, desc = "uphill 5/6", slope=true},
- vst66={conns = conns(8,0,5/6,1), rail_y = 6/6 - 1/12, desc = "uphill 6/6", slope=true},
- vst71={conns = conns(8,0,0,1/7), rail_y = 1/7 - 1/14, desc = "uphill 1/7", slope=true},
- vst72={conns = conns(8,0,1/7,2/7), rail_y = 2/7 - 1/14, desc = "uphill 2/7", slope=true},
- vst73={conns = conns(8,0,2/7,3/7), rail_y = 3/7 - 1/14, desc = "uphill 3/7", slope=true},
- vst74={conns = conns(8,0,3/7,4/7), rail_y = 4/7 - 1/14, desc = "uphill 4/7", slope=true},
- vst75={conns = conns(8,0,4/7,5/7), rail_y = 5/7 - 1/14, desc = "uphill 5/7", slope=true},
- vst76={conns = conns(8,0,5/7,6/7), rail_y = 6/7 - 1/14, desc = "uphill 6/7", slope=true},
- vst77={conns = conns(8,0,6/7,1), rail_y = 7/7 - 1/14, desc = "uphill 7/7", slope=true},
- vst81={conns = conns(8,0,0,1/8), rail_y = 1/8 - 1/16, desc = "uphill 1/8", slope=true},
- vst82={conns = conns(8,0,1/8,2/8), rail_y = 2/8 - 1/16, desc = "uphill 2/8", slope=true},
- vst83={conns = conns(8,0,2/8,3/8), rail_y = 3/8 - 1/16, desc = "uphill 3/8", slope=true},
- vst84={conns = conns(8,0,3/8,4/8), rail_y = 4/8 - 1/16, desc = "uphill 4/8", slope=true},
- vst85={conns = conns(8,0,4/8,5/8), rail_y = 5/8 - 1/16, desc = "uphill 5/8", slope=true},
- vst86={conns = conns(8,0,5/8,6/8), rail_y = 6/8 - 1/16, desc = "uphill 6/8", slope=true},
- vst87={conns = conns(8,0,6/8,7/8), rail_y = 7/8 - 1/16, desc = "uphill 7/8", slope=true},
- vst88={conns = conns(8,0,7/8,1), rail_y = 8/8 - 1/16, desc = "uphill 8/8", slope=true},
- },
- regsp=true,
- slopeplacer={
- [2]={"vst1", "vst2"},
- [3]={"vst31", "vst32", "vst33"},
- [4]={"vst41", "vst42", "vst43", "vst44"},
- [5]={"vst51", "vst52", "vst53", "vst54", "vst55"},
- [6]={"vst61", "vst62", "vst63", "vst64", "vst65", "vst66"},
- [7]={"vst71", "vst72", "vst73", "vst74", "vst75", "vst76", "vst77"},
- [8]={"vst81", "vst82", "vst83", "vst84", "vst85", "vst86", "vst87", "vst88"},
- max=8,--highest entry
- },
- slopeplacer_45={
- [2]={"vst1_45", "vst2_45"},
- max=2,
- },
- rotation={"", "_30", "_45", "_60"},
- trackworker={},
- increativeinv={},
-}
-advtrains.ap.t_30deg_straightonly={
- regstep=1,
- variant={
- st={
- conns = conns(0,8),
- desc = "straight",
- tpdouble = true,
- tpsingle = true,
- trackworker = "st",
- },
- },
- regtp=true,
- tpdefault="st",
- rotation={"", "_30", "_45", "_60"},
-}
-advtrains.ap.t_30deg_straightonly_noplacer={
- regstep=1,
- variant={
- st={
- conns = conns(0,8),
- desc = "straight",
- tpdouble = true,
- tpsingle = true,
- trackworker = "st",
- },
- },
- tpdefault="st",
- rotation={"", "_30", "_45", "_60"},
-}
-advtrains.ap.t_45deg={
- regstep=2,
- variant={
- st={
- conns = conns(0,8),
- desc = "straight",
- tpdouble = true,
- tpsingle = true,
- trackworker = "cr",
- },
- cr={
- conns = conns(0,6),
- desc = "curve",
- tpdouble = true,
- trackworker = "swlst",
- },
- swlst={
- conns = conns3(0,8,6),
- desc = "left switch (straight)",
- trackworker = "swrst",
- switchalt = "cr",
- switchmc = "on",
- switchst = "st",
- },
- swlcr={
- conns = conns3(0,6,8),
- desc = "left switch (curve)",
- trackworker = "swrcr",
- switchalt = "st",
- switchmc = "off",
- switchst = "cr",
- },
- swrst={
- conns = conns3(0,8,10),
- desc = "right switch (straight)",
- trackworker = "st",
- switchalt = "cr",
- switchmc = "on",
- switchst = "st",
- },
- swrcr={
- conns = conns3(0,10,8),
- desc = "right switch (curve)",
- trackworker = "st",
- switchalt = "st",
- switchmc = "off",
- switchst = "cr",
- },
- },
- regtp=true,
- tpdefault="st",
- trackworker={
- ["swrcr"]="st",
- ["swrst"]="st",
- ["cr"]="swlst",
- ["swlcr"]="swrcr",
- ["swlst"]="swrst",
- },
- rotation={"", "_30", "_45", "_60"},
-}
-advtrains.ap.t_perpcrossing={
- regstep = 1,
- variant={
- st={
- conns = { {c=0}, {c=8}, {c=4}, {c=12} },
- desc = "perpendicular crossing",
- tpdouble = true,
- tpsingle = true,
- trackworker = "st",
- },
- },
- regtp=true,
- tpdefault="st",
- rotation={"", "_30", "_45", "_60"},
-}
-advtrains.ap.t_90plusx_crossing={
- regstep = 1,
- variant={
- ["30l"]={
- conns = { {c=0}, {c=8}, {c=1}, {c=9} },
- desc = "30/90 degree crossing (left)",
- tpdouble = true,
- tpsingle = true,
- trackworker = "45l"
- },
- ["45l"]={
- conns = { {c=0}, {c=8}, {c=2}, {c=10} },
- desc = "45/90 degree crossing (left)",
- tpdouble = true,
- tpsingle = true,
- trackworker = "60l",
- },
- ["60l"]={
- conns = { {c=0}, {c=8}, {c=3}, {c=11}},
- desc = "60/90 degree crossing (left)",
- tpdouble = true,
- tpsingle = true,
- trackworker = "60r",
- },
- ["60r"]={
- conns = { {c=0}, {c=8}, {c=5}, {c=13} },
- desc = "60/90 degree crossing (right)",
- tpdouble = true,
- tpsingle = true,
- trackworker = "45r"
- },
- ["45r"]={
- conns = { {c=0}, {c=8}, {c=6}, {c=14} },
- desc = "45/90 degree crossing (right)",
- tpdouble = true,
- tpsingle = true,
- trackworker = "30r",
- },
- ["30r"]={
- conns = { {c=0}, {c=8}, {c=7}, {c=15}},
- desc = "30/90 degree crossing (right)",
- tpdouble = true,
- tpsingle = true,
- trackworker = "30l",
- },
- },
- regtp=true,
- tpdefault="30l",
- rotation={""},
- trackworker = {
- ["30l"] = "45l",
- ["45l"] = "60l",
- ["60l"] = "60r",
- ["60r"] = "45r",
- ["45r"] = "30r",
- ["30r"] = "30l",
+ ^- Map of state name to the appropriate node name that should be set by advtrains when a switch is requested
+ Note that for all of those nodes, the at_conns table must be identical (however the conn_map will vary)
+
+ node_on_switch_state = function(pos, node, oldstate, newstate)
+ ^- Called when the node state is switched by advtrains, after the node replacement has commenced.
+
+ Turnout switching can happen programmatically via advtrains.setstate(pos, state), via user right_click or via the interlocking system.
+ In no other situation is it permissible to exchange track nodes in-place, unless both at_conns and at_conn_map stay identical.
+
+ Note that the fields node_state, node_next_state and node_state_map completely replace the getstate/setstate functions.
+ There must be a one-to-one mapping between states and node names and no function can be defined for state switching.
+ This principle enables the seamless working of the interlocking autorouter and reduces failure points.
+ The node_state_* system can also be used as drop-in replacement for the passive-API-enabled nodes (andrews-cross, mesecon_switch etc.)
+ The advtrains API functions advtrains.getstate() and advtrains.setstate() remain the programmatic access points, but will now utilize the new state system.
+
+
+ trackworker_next_rot = <nodename of next rotation step>,
+ ^- if set, right-click with trackworker will set this node
+ trackworker_rot_incr_param2 = true
+ ^- if set, trackworker will increase node param2 on rightclick
+
+ trackworker_next_var = <nodename of next variant>
+ ^- if set, left-click with trackworker will set this node
}
-}
-
-advtrains.ap.t_diagonalcrossing = {
- regstep=1,
- variant={
- ["30l45r"]={
- conns = {{c=1}, {c=9}, {c=6}, {c=14}},
- desc = "30left-45right diagonal crossing",
- tpdouble=true,
- tpsingle=true,
- trackworker="60l30l",
- },
- ["60l30l"]={
- conns = {{c=3}, {c=11}, {c=1}, {c=9}},
- desc = "30left-60right diagonal crossing",
- tpdouble=true,
- tpsingle=true,
- trackworker="60l45r"
- },
- ["60l45r"]={
- conns = {{c=3}, {c=11}, {c=6}, {c=14}},
- desc = "60left-45right diagonal crossing",
- tpdouble=true,
- tpsingle=true,
- trackworker="60l60r"
- },
- ["60l60r"]={
- conns = {{c=3}, {c=11}, {c=5}, {c=13}},
- desc = "60left-60right diagonal crossing",
- tpdouble=true,
- tpsingle=true,
- trackworker="60r45l",
- },
- --If 60l60r had a mirror image, it would be here, but it's symmetric.
- -- 60l60r is also equivalent to 30l30r but rotated 90 degrees.
- ["60r45l"]={
- conns = {{c=5}, {c=13}, {c=2}, {c=10}},
- desc = "60right-45left diagonal crossing",
- tpdouble=true,
- tpsingle=true,
- trackworker="60r30r",
- },
- ["60r30r"]={
- conns = {{c=5}, {c=13}, {c=7}, {c=15}},
- desc = "60right-30right diagonal crossing",
- tpdouble=true,
- tpsingle=true,
- trackworker="30r45l",
- },
- ["30r45l"]={
- conns = {{c=7}, {c=15}, {c=2}, {c=10}},
- desc = "30right-45left diagonal crossing",
- tpdouble=true,
- tpsingle=true,
- trackworker="30l45r",
- },
+ })
- },
- regtp=true,
- tpdefault="30l45r",
- rotation={""},
- trackworker = {
- ["30l45r"] = "60l30l",
- ["60l30l"] = "60l45r",
- ["60l45r"] = "60l60r",
- ["60l60r"] = "60r45l",
- ["60r45l"] = "60r30r",
- ["60r30r"] = "30r45l",
- ["30r45l"] = "30l45r",
- }
-}
+]]--
-advtrains.trackpresets = advtrains.ap
+-- This file provides some utilities to register tracks, but tries to not get into the way too much
---definition format: ([] optional)
---[[{
- nodename_prefix
- texture_prefix
- [shared_texture]
- models_prefix
- models_suffix (with dot)
- [shared_model]
- formats={
- st,cr,swlst,swlcr,swrst,swrcr,vst1,vst2
- (each a table with indices 0-3, for if to register a rail with this 'rotation' table entry. nil is assumed as 'all', set {} to not register at all)
- }
- common={} change something on common rail appearance
-}
-[18.12.17] Note on new connection system:
-In order to support real rail crossing nodes and finally make the trackplacer respect switches, I changed the connection system.
-There can be a variable number of connections available. These are specified as tuples {c=<connection>, y=<rely>}
-The table "at_conns" consists of {<conn1>, <conn2>...}
-the "at_rail_y" property holds the value that was previously called "railheight"
-Depending on the number of connections:
-2 conns: regular rail
-3 conns: switch:
- - when train passes in at conn1, will move out of conn2
- - when train passes in at conn2 or conn3, will move out of conn1
-4 conns: cross (or cross switch, depending on arrangement of conns):
- - conn1 <> conn2
- - conn3 <> conn4
-]]
--- Notify the user if digging the rail is not allowed
-local function can_dig_callback(pos, player)
+function advtrains.track_can_dig_callback(pos, player)
local ok, reason = advtrains.can_dig_or_modify_track(pos)
if not ok and player then
minetest.chat_send_player(player:get_player_name(), attrans("This track can not be removed!") .. " " .. reason)
@@ -486,148 +130,113 @@ local function can_dig_callback(pos, player)
return ok
end
-function advtrains.register_tracks(tracktype, def, preset)
- advtrains.trackplacer.register_tracktype(def.nodename_prefix, preset.tpdefault)
- if preset.regtp then
- advtrains.trackplacer.register_track_placer(def.nodename_prefix, def.texture_prefix, def.description, def)
- end
- if preset.regsp then
- advtrains.slope.register_placer(def, preset)
- end
- for suffix, var in pairs(preset.variant) do
- for rotid, rotation in ipairs(preset.rotation) do
- if not def.formats[suffix] or def.formats[suffix][rotid] then
- local img_suffix = suffix..rotation
- local ndef = advtrains.merge_tables({
- description=def.description.."("..(var.desc or "any")..rotation..")",
- drawtype = "mesh",
- paramtype="light",
- paramtype2="facedir",
- walkable = false,
- selection_box = {
- type = "fixed",
- fixed = {-1/2, -1/2, -1/2, 1/2, -1/2+1/16, 1/2},
- },
-
- mesh = def.shared_model or (def.models_prefix.."_"..img_suffix..def.models_suffix),
- tiles = {def.shared_texture or (def.texture_prefix.."_"..img_suffix..".png"), def.second_texture},
- use_texture_alpha = "clip",
+function advtrains.track_update_callback(pos)
+ advtrains.ndb.update(pos)
+end
- groups = {
- attached_node = advtrains.IGNORE_WORLD and 0 or 1,
- advtrains_track=1,
- ["advtrains_track_"..tracktype]=1,
- save_in_at_nodedb=1,
- dig_immediate=2,
- not_in_creative_inventory=1,
- not_blocking_trains=1,
- },
-
- can_dig = can_dig_callback,
- after_dig_node=function(pos)
- advtrains.ndb.update(pos)
- end,
- after_place_node=function(pos)
- advtrains.ndb.update(pos)
- end,
- at_nnpref = def.nodename_prefix,
- at_suffix = suffix,
- at_rotation = rotation,
- at_rail_y = var.rail_y
- }, def.common or {})
-
- if preset.regtp then
- ndef.drop = def.nodename_prefix.."_placer"
- end
- if preset.regsp and var.slope then
- ndef.drop = def.nodename_prefix.."_slopeplacer"
- end
-
- --connections
- ndef.at_conns = advtrains.rotate_conn_by(var.conns, (rotid-1)*preset.regstep)
-
- local ndef_avt_table
-
- if var.switchalt and var.switchst then
- local switchfunc=function(pos, node, newstate)
- newstate = newstate or var.switchalt -- support for 3 (or more) state switches
- -- this code is only called from the internal setstate function, which
- -- ensures that it is safe to switch the turnout
- if newstate~=var.switchst then
- advtrains.ndb.swap_node(pos, {name=def.nodename_prefix.."_"..(var.switchprefix or "")..newstate..rotation, param2=node.param2})
- advtrains.invalidate_all_paths(pos)
- end
- end
- ndef.on_rightclick = function(pos, node, player)
- if advtrains.check_turnout_signal_protection(pos, player:get_player_name()) then
- advtrains.setstate(pos, nil, node)
- advtrains.log("Switch", player:get_player_name(), pos)
- end
- end
- if var.switchmc then
- ndef.mesecons = {effector = {
- ["action_"..var.switchmc] = function(pos, node)
- advtrains.setstate(pos, nil, node)
- end,
- rules=advtrains.meseconrules
- }}
- end
- ndef_avt_table = {
- getstate = var.switchst,
- setstate = switchfunc,
- }
- end
-
- local adef={}
- if def.get_additional_definiton then
- adef=def.get_additional_definiton(def, preset, suffix, rotation)
- end
- ndef = advtrains.merge_tables(ndef, adef)
-
- -- insert getstate/setstate functions after merging the additional definitions
- if ndef_avt_table then
- ndef.advtrains = advtrains.merge_tables(ndef.advtrains or {}, ndef_avt_table)
- end
+function advtrains.state_node_on_rightclick_callback(pos, node, player)
+ if advtrains.check_turnout_signal_protection(pos, player:get_player_name()) then
+ local ndef = minetest.registered_nodes[node.name]
+ if ndef and ndef.advtrains and ndef.advtrains.node_next_state then
+ advtrains.setstate(pos, ndef.advtrains.node_next_state, node)
+ advtrains.log("Switch", player:get_player_name(), pos)
+ end
+ end
+end
- minetest.register_node(":"..def.nodename_prefix.."_"..suffix..rotation, ndef)
- --trackplacer
- if preset.regtp then
- local tpconns = {conn1=ndef.at_conns[1].c, conn2=ndef.at_conns[2].c}
- if var.tpdouble then
- advtrains.trackplacer.add_double_conn(def.nodename_prefix, suffix, rotation, tpconns)
- end
- if var.tpsingle then
- advtrains.trackplacer.add_single_conn(def.nodename_prefix, suffix, rotation, tpconns)
- end
+-- advtrains.register_node_4rot(name, nodedef)
+-- Registers four rotations for the node defined by nodedef (0°, 30°, 45° and 60°; the 4 90°-steps are already handled by the param2, resulting in 16 directions total).
+-- You must provide the definition for the base node, and certain fields are altered automatically for the 3 additional rotations:
+-- name: appends the suffix "_30", "_45" or "_60"
+-- description: appends the rotation (human-readable) in parenthesis
+-- tiles_prefix: if defined, "tiles" field will be set as prefix..rotationExtension..".png"
+-- mesh_prefix, mesh_suffix: if defined, "mesh" field will be set as prefix..rotationExtension..suffix
+-- at_conns: are rotated according to the node rotation
+-- node_state_map, trackworker_next_var: appends the suffix appropriately.
+-- groups: applies save_in_at_nodedb and not_blocking_trains groups if not already present
+-- The nodes are registered in the trackworker to be rotated with right-click.
+-- definition_mangling_function is an optional parameter. For each of the 4 rotations, it gets passed the modified node definition and may perform final modifications to it.
+-- signature: function definition_mangling_function(name, nodedef, rotationIndex, rotationSuffix)
+-- Example usage: define the setstate function of turnouts (if that is not done via the "automatic" way of state_node_map)
+local rotations = {
+ {i = 0, s = "", h = " (0)", n = "_30"},
+ {i = 1, s = "_30", h = " (30)", n = "_45"},
+ {i = 2, s = "_45", h = " (45)", n = "_60"},
+ {i = 3, s = "_60", h = " (60)", n = ""},
+}
+function advtrains.register_node_4rot(ori_name, ori_ndef, definition_mangling_function)
+ for _, rot in ipairs(rotations) do
+ local ndef = table.copy(ori_ndef)
+ if ori_ndef.advtrains then
+ -- make sure advtrains table is deep-copied because we may need to replace node_state_map
+ ndef.advtrains = table.copy(ori_ndef.advtrains)
+ else
+ ndef.advtrains = {} -- we need the table later for trackworker
+ end
+ -- Perform the name mangling
+ local suffix = rot.s
+ local name = ori_name..suffix
+ ndef.description = ori_ndef.description .. rot.h
+ if ori_ndef.tiles_prefix then
+ ndef.tiles = { ori_ndef.tiles_prefix .. suffix .. ".png" }
+ end
+ if ori_ndef.mesh_prefix then
+ ndef.mesh = ori_ndef.mesh_prefix .. suffix .. ori_ndef.mesh_suffix
+ end
+ -- rotate connections
+ if ori_ndef.at_conns then
+ ndef.at_conns = advtrains.rotate_conn_by(ori_ndef.at_conns, rot.i)
+ end
+ -- update node state map if present
+ if ori_ndef.advtrains then
+ if ori_ndef.advtrains.node_state_map then
+ local new_nsm = {}
+ for state, nname in pairs(ori_ndef.advtrains.node_state_map) do
+ new_nsm[state] = nname .. suffix
end
- advtrains.trackplacer.add_worked(def.nodename_prefix, suffix, rotation, var.trackworker)
+ ndef.advtrains.node_state_map = new_nsm
end
+ if ori_ndef.advtrains.trackworker_next_var then
+ ndef.advtrains.trackworker_next_var = ori_ndef.advtrains.trackworker_next_var .. suffix
+ end
+ -- apply trackworker rot field
+ ndef.advtrains.trackworker_next_rot = ori_name .. rot.n
+ ndef.advtrains.trackworker_rot_incr_param2 = (rot.n=="")
+ end
+ -- apply groups
+ ndef.groups.save_in_at_nodedb = 1
+ ndef.groups.not_blocking_trains = 1
+
+ -- give the definition mangling function an option to do some adjustments
+ if definition_mangling_function then
+ definition_mangling_function(name, ndef, rot.i, suffix)
+ end
+
+ -- register node
+ minetest.register_node(":"..name, ndef)
+
+ -- if this has the track_place_group set, register as a candidate for the track_place_group
+ if ndef.advtrains.track_place_group then
+ advtrains.trackplacer.register_candidate(ndef.advtrains.track_place_group, name, ndef, ndef.advtrains.track_place_single, true)
end
end
- advtrains.all_tracktypes[tracktype]=true
end
-function advtrains.is_track_and_drives_on(nodename, drives_on_p)
- local drives_on = drives_on_p
- if not drives_on then drives_on = advtrains.all_tracktypes end
- local hasentry = false
- for _,_ in pairs(drives_on) do
- hasentry=true
- end
- if not hasentry then drives_on = advtrains.all_tracktypes end
-
+-- track-related helper functions
+
+function advtrains.is_track(nodename)
if not minetest.registered_nodes[nodename] then
return false
end
local nodedef=minetest.registered_nodes[nodename]
- for k,v in pairs(drives_on) do
- if nodedef.groups["advtrains_track_"..k] then
- return true
- end
+ if nodedef and nodedef.groups.advtrains_track then
+ return true
end
return false
end
+-- returns the connection tables of the track with given node details
+-- returns: conns table, railheight, conn_map table
function advtrains.get_track_connections(name, param2)
local nodedef=minetest.registered_nodes[name]
if not nodedef then atprint(" get_track_connections couldn't find nodedef for nodename "..(name or "nil")) return nil end
@@ -635,18 +244,16 @@ function advtrains.get_track_connections(name, param2)
if not param2 then noderot=0 end
if noderot > 3 then atprint(" get_track_connections: rail has invaild param2 of "..noderot) noderot=0 end
- local tracktype
- for k,_ in pairs(nodedef.groups) do
- local tt=string.match(k, "^advtrains_track_(.+)$")
- if tt then
- tracktype=tt
- end
+ if not nodedef.at_conns then
+ return nil
end
- return advtrains.rotate_conn_by(nodedef.at_conns, noderot*AT_CMAX/4), (nodedef.at_rail_y or 0), tracktype
+ --atdebug("Track connections of ",name,param2,":",nodedef.at_conns)
+ return advtrains.rotate_conn_by(nodedef.at_conns, noderot*AT_CMAX/4), (nodedef.at_rail_y or 0), nodedef.at_conn_map
end
-- Function called when a track is about to be dug or modified by the trackworker
-- Returns either true (ok) or false,"translated string describing reason why it isn't allowed"
+-- Impl Note: possibly duplicate code in "self contained TCB" - see interlocking/tcb_ts_ui.lua!
function advtrains.can_dig_or_modify_track(pos)
if advtrains.get_train_at_pos(pos) then
return false, attrans("Position is occupied by a train.")
@@ -664,125 +271,3 @@ function advtrains.can_dig_or_modify_track(pos)
end
return true
end
-
--- slope placer. Defined in register_tracks.
---crafted with rail and gravel
-local sl={}
-function sl.register_placer(def, preset)
- minetest.register_craftitem(":"..def.nodename_prefix.."_slopeplacer",{
- description = attrans("@1 Slope", def.description),
- inventory_image = def.texture_prefix.."_slopeplacer.png",
- wield_image = def.texture_prefix.."_slopeplacer.png",
- groups={},
- on_place = sl.create_slopeplacer_on_place(def, preset),
- liquids_pointable = def.liquids_pointable,
- })
-end
---(itemstack, placer, pointed_thing)
-function sl.create_slopeplacer_on_place(def, preset)
- return function(istack, player, pt)
- if not pt.type=="node" then
- minetest.chat_send_player(player:get_player_name(), attrans("Can't place: not pointing at node"))
- return istack
- end
- local pos=pt.above
- if not pos then
- minetest.chat_send_player(player:get_player_name(), attrans("Can't place: not pointing at node"))
- return istack
- end
- local node=minetest.get_node(pos)
- if not minetest.registered_nodes[node.name] or not minetest.registered_nodes[node.name].buildable_to then
- minetest.chat_send_player(player:get_player_name(), attrans("Can't place: space occupied!"))
- return istack
- end
- if not advtrains.check_track_protection(pos, player:get_player_name()) then
- minetest.record_protection_violation(pos, player:get_player_name())
- return istack
- end
- --determine player orientation (only horizontal component)
- --get_look_horizontal may not be available
- local yaw=player.get_look_horizontal and player:get_look_horizontal() or (player:get_look_yaw() - math.pi/2)
-
- --rounding unit vectors is a nice way for selecting 1 of 8 directions since sin(30°) is 0.5.
- local dirvec={x=math.floor(math.sin(-yaw)+0.5), y=0, z=math.floor(math.cos(-yaw)+0.5)}
- --translate to direction to look up inside the preset table
- local param2, rot45=({
- [-1]={
- [-1]=2,
- [0]=3,
- [1]=3,
- },
- [0]={
- [-1]=2,
- [1]=0,
- },
- [1]={
- [-1]=1,
- [0]=1,
- [1]=0,
- },
- })[dirvec.x][dirvec.z], dirvec.x~=0 and dirvec.z~=0
- local lookup=preset.slopeplacer
- if rot45 then lookup=preset.slopeplacer_45 end
-
- --go unitvector forward and look how far the next node is
- local step=1
- while step<=lookup.max do
- local node=minetest.get_node(vector.add(pos, dirvec))
- --next node solid?
- if not minetest.registered_nodes[node.name] or not minetest.registered_nodes[node.name].buildable_to or advtrains.is_protected(pos, player:get_player_name()) then
- --do slopes of this distance exist?
- if lookup[step] then
- if minetest.is_creative_enabled(player:get_player_name()) or istack:get_count()>=step then
- --start placing
- local placenodes=lookup[step]
- while step>0 do
- minetest.set_node(pos, {name=def.nodename_prefix.."_"..placenodes[step], param2=param2})
- if not minetest.is_creative_enabled(player:get_player_name()) then
- istack:take_item()
- end
- step=step-1
- pos=vector.subtract(pos, dirvec)
- end
- else
- minetest.chat_send_player(player:get_player_name(), attrans("Can't place: Not enough slope items left (@1 required)", step))
- end
- else
- minetest.chat_send_player(player:get_player_name(), attrans("Can't place: There's no slope of length @1",step))
- end
- return istack
- end
- step=step+1
- pos=vector.add(pos, dirvec)
- end
- minetest.chat_send_player(player:get_player_name(), attrans("Can't place: no supporting node at upper end."))
- return istack
- end
-end
-
-advtrains.slope=sl
-
---END code, BEGIN definition
---definition format: ([] optional)
---[[{
- nodename_prefix
- texture_prefix
- [shared_texture]
- models_prefix
- models_suffix (with dot)
- [shared_model]
- formats={
- st,cr,swlst,swlcr,swrst,swrcr,vst1,vst2
- (each a table with indices 0-3, for if to register a rail with this 'rotation' table entry. nil is assumed as 'all', set {} to not register at all)
- }
- common={} change something on common rail appearance
-}]]
-
-
-
-
-
-
-
-
-
diff --git a/advtrains/trainhud.lua b/advtrains/trainhud.lua
index e0fc259..9d9d475 100644
--- a/advtrains/trainhud.lua
+++ b/advtrains/trainhud.lua
@@ -1,5 +1,7 @@
--trainhud.lua: holds all the code for train controlling
+local T = advtrains.texture
+
advtrains.hud = {}
advtrains.hhud = {}
@@ -8,6 +10,8 @@ advtrains.hud[player:get_player_name()] = nil
advtrains.hhud[player:get_player_name()] = nil
end)
+local hud_type_key = minetest.features.hud_def_type_field and "type" or "hud_elem_type"
+
local mletter={[1]="F", [-1]="R", [0]="N"}
function advtrains.on_control_change(pc, train, flip)
@@ -46,6 +50,10 @@ function advtrains.on_control_change(pc, train, flip)
train.ctrl_user = 1
act=true
end
+ if train.ars_disable and pc.up then
+ -- up clears ars disable flag in any situation
+ train.ars_disable = nil
+ end
-- If atc command set, only "Jump" key can clear command. To prevent accidental control.
if train.tarvelocity or train.atc_command then
return
@@ -101,19 +109,21 @@ function advtrains.set_trainhud(name, text, driver)
if not player then
return
end
+ local drivertext = driver or ""
local driverhud = {
- type = "image",
+ [hud_type_key] = "image",
name = "ADVTRAINS_DRIVER",
position = {x=0.5, y=1},
offset = {x=0,y=-170},
- text = driver or "",
+ text = drivertext,
alignment = {x=0,y=-1},
- scale = {x=1,y=1},}
+ scale = {x=1,y=1},
+ }
if not hud then
- hud = {["driver"]={}}
+ hud = {}
advtrains.hud[name] = hud
hud.id = player:hud_add({
- type = "text",
+ [hud_type_key] = "text",
name = "ADVTRAINS",
number = 0xFFFFFF,
position = {x=0.5, y=1},
@@ -122,17 +132,22 @@ function advtrains.set_trainhud(name, text, driver)
scale = {x=200, y=60},
alignment = {x=0, y=-1},
})
- hud.oldText=text
hud.driver = player:hud_add(driverhud)
+ hud.oldText = text
+ hud.oldDriver = drivertext
else
if hud.oldText ~= text then
player:hud_change(hud.id, "text", text)
hud.oldText=text
end
if hud.driver then
- player:hud_change(hud.driver, "text", driver or "")
+ if hud.oldDriver ~= drivertext then
+ player:hud_change(hud.driver, "text", drivertext)
+ hud.oldDriver = drivertext
+ end
elseif driver then
hud.driver = player:hud_add(driverhud)
+ hud.oldDriver = drivertext
end
end
end
@@ -147,7 +162,7 @@ function advtrains.set_help_hud(name, text)
hud = {}
advtrains.hhud[name] = hud
hud.id = player:hud_add({
- type = "text",
+ [hud_type_key] = "text",
name = "ADVTRAINS_HELP",
number = 0xFFFFFF,
position = {x=1, y=0.3},
@@ -184,138 +199,108 @@ function advtrains.hud_train_format(train, flip)
local vel = advtrains.abs_ceil(train.velocity)
local vel_kmh=advtrains.abs_ceil(advtrains.ms_to_kmh(train.velocity))
- local tlev=train.lever or 1
+ local tlev=train.lever or 3
if train.velocity==0 and not train.active_control then tlev=1 end
if train.hud_lzb_effect_tmr then
tlev=1
end
- local ht = {"[combine:440x110:0,0=(advtrains_hud_bg.png^[resize\\:440x110)"}
+ local hud = T.combine(440, 110, "black")
local st = {}
if train.debug then st = {train.debug} end
- -- seven-segment display
- local function sevenseg(digit, x, y, w, h, m)
- --[[
- -1-
- 2 3
- -4-
- 5 6
- -7-
- ]]
- local segs = {
- {h, 0, w, h},
- {0, h, h, w},
- {w+h, h, h, w},
- {h, w+h, w, h},
- {0, w+2*h, h, w},
- {w+h, w+2*h, h, w},
- {h, 2*(w+h), w, h}}
- local trans = {
- [0] = {true, true, true, false, true, true, true},
- [1] = {false, false, true, false, false, true, false},
- [2] = {true, false, true, true, true, false, true},
- [3] = {true, false, true, true, false, true, true},
- [4] = {false, true, true, true, false, true, false},
- [5] = {true, true, false, true, false, true, true},
- [6] = {true, true, false, true, true, true, true},
- [7] = {true, false, true, false, false, true, false},
- [8] = {true, true, true, true, true, true, true},
- [9] = {true, true, true, true, false, true, true}}
- local ent = trans[digit or 10]
- if not ent then return end
- for i = 1, 7, 1 do
- if ent[i] then
- local s = segs[i]
- ht[#ht+1] = sformat("%d,%d=(advtrains_hud_bg.png^[resize\\:%dx%d^%s)",x+s[1], y+s[2], s[3], s[4], m)
- end
- end
- end
-
-- lever
- ht[#ht+1] = "275,10=(advtrains_hud_bg.png^[colorize\\:cyan^[resize\\:5x18)"
- ht[#ht+1] = "275,28=(advtrains_hud_bg.png^[colorize\\:white^[resize\\:5x18)"
- ht[#ht+1] = "275,46=(advtrains_hud_bg.png^[colorize\\:orange^[resize\\:5x36)"
- ht[#ht+1] = "275,82=(advtrains_hud_bg.png^[colorize\\:red^[resize\\:5x18)"
- ht[#ht+1] = "292,16=(advtrains_hud_bg.png^[colorize\\:darkslategray^[resize\\:6x78)"
- ht[#ht+1] = sformat("280,%s=(advtrains_hud_bg.png^[colorize\\:gray^[resize\\:30x18)",18*(4-tlev)+10)
+ hud:add_multicolor_fill_topdown(275, 10, 5, 90, 1, "cyan", 1, "white", 2, "orange", 1, "red")
+ hud:add_lever_topdown(280, 10, 30, 90, 18, 6, (4-tlev)/4, "gray", "darkslategray")
-- reverser
- ht[#ht+1] = sformat("245,10=(advtrains_hud_arrow.png^[transformFY%s)", flip and "" or "^[multiply\\:cyan")
- ht[#ht+1] = sformat("245,85=(advtrains_hud_arrow.png%s)", flip and "^[multiply\\:orange" or "")
- ht[#ht+1] = "250,35=(advtrains_hud_bg.png^[colorize\\:darkslategray^[resize\\:5x40)"
- ht[#ht+1] = sformat("240,%s=(advtrains_hud_bg.png^[resize\\:25x15^[colorize\\:gray)", flip and 65 or 30)
+ hud:add(245, 10, T"advtrains_hud_arrow.png":transform"FY":multiply(flip and "gray" or "cyan"))
+ hud:add(245, 85, T"advtrains_hud_arrow.png":multiply(flip and "orange" or "gray"))
+ hud:add_lever_topdown(240, 30, 25, 50, 15, 5, flip and 1 or 0, "gray", "darkslategray")
-- train control/safety indication
- if train.tarvelocity or train.atc_command then
- ht[#ht+1] = "10,10=(advtrains_hud_atc.png^[resize\\:30x30^[multiply\\:cyan)"
- end
- if train.hud_lzb_effect_tmr then
- ht[#ht+1] = "50,10=(advtrains_hud_lzb.png^[resize\\:30x30^[multiply\\:red)"
- end
- if train.is_shunt then
- ht[#ht+1] = "90,10=(advtrains_hud_shunt.png^[resize\\:30x30^[multiply\\:orange)"
- end
+ hud:add(10, 10, T"advtrains_hud_atc.png":resize(30, 30):multiply((train.tarvelocity or train.atc_command) and "cyan" or "darkslategray"))
+ hud:add(50, 10, T"advtrains_hud_lzb.png":resize(30, 30):multiply(train.hud_lzb_effect_tmr and "red" or "darkslategray"))
+ hud:add(90, 10, T"advtrains_hud_shunt.png":resize(30, 30):multiply(train.is_shunt and "orange" or "darkslategray"))
-- door
- ht[#ht+1] = "187,10=(advtrains_hud_bg.png^[resize\\:26x30^[colorize\\:white)"
- ht[#ht+1] = "189,12=(advtrains_hud_bg.png^[resize\\:22x11)"
- ht[#ht+1] = sformat("170,10=(advtrains_hud_bg.png^[resize\\:15x30^[colorize\\:%s)", train.door_open==-1 and "white" or "darkslategray")
- ht[#ht+1] = "172,12=(advtrains_hud_bg.png^[resize\\:11x11)"
- ht[#ht+1] = sformat("215,10=(advtrains_hud_bg.png^[resize\\:15x30^[colorize\\:%s)", train.door_open==1 and "white" or "darkslategray")
- ht[#ht+1] = "217,12=(advtrains_hud_bg.png^[resize\\:11x11)"
+ hud:add_fill(187, 10, 26, 30, "white"):add_fill(189, 12, 22, 11, "black")
+ hud:add_fill(170, 10, 15, 30, train.door_open==-1 and "white" or "darkslategray"):add_fill(172, 12, 11, 11, "black")
+ hud:add_fill(215, 10, 15, 30, train.door_open==1 and "white" or "darkslategray"):add_fill(217, 12, 11, 11, "black")
-- speed indication(s)
- sevenseg(math.floor(vel/10), 320, 10, 30, 10, "[colorize\\:red\\:255")
- sevenseg(vel%10, 380, 10, 30, 10, "[colorize\\:red\\:255")
- for i = 1, vel, 1 do
- ht[#ht+1] = sformat("%d,65=(advtrains_hud_bg.png^[resize\\:8x20^[colorize\\:white)", i*11-1)
- end
- for i = max+1, 20, 1 do
- ht[#ht+1] = sformat("%d,65=(advtrains_hud_bg.png^[resize\\:8x20^[colorize\\:darkslategray)", i*11-1)
- end
+ hud:add_n7seg(320, 10, 110, 90, vel, 2, "red")
+ hud:add_segmentbar_leftright(10, 65, 217, 20, 3, 20, max, 20, "darkslategray", 0, vel, "white")
if res and res > 0 then
- ht[#ht+1] = sformat("%d,60=(advtrains_hud_bg.png^[resize\\:3x30^[colorize\\:red\\:255)", 7+res*11)
+ hud:add_fill(7+res*11, 60, 3, 30, "red")
end
if train.tarvelocity then
- ht[#ht+1] = sformat("%d,85=(advtrains_hud_arrow.png^[multiply\\:cyan^[transformFY^[makealpha\\:#000000)", 1+train.tarvelocity*11)
+ hud:add(1+train.tarvelocity*11, 85, T"advtrains_hud_arrow.png":transform"FY":multiply"cyan")
end
local lzb = train.lzb
- if lzb and lzb.checkpoints then
- local oc = lzb.checkpoints
- for i = 1, #oc do
- local spd = oc[i].speed
- spd = advtrains.speed.min(spd, train.speed_restriction)
- if spd == -1 then spd = nil end
- local c = not spd and "lime" or (type(spd) == "number" and (spd == 0) and "red" or "orange") or nil
- if c then
- ht[#ht+1] = sformat("130,10=(advtrains_hud_bg.png^[resize\\:30x5^[colorize\\:%s)",c)
- ht[#ht+1] = sformat("130,35=(advtrains_hud_bg.png^[resize\\:30x5^[colorize\\:%s)",c)
- if spd and spd~=0 then
- ht[#ht+1] = sformat("%d,50=(advtrains_hud_arrow.png^[multiply\\:red^[makealpha\\:#000000)", 1+spd*11)
- end
- local floor = math.floor
- local dist = floor(((oc[i].index or train.index)-train.index))
- dist = math.max(0, math.min(999, dist))
- for j = 1, 3, 1 do
- sevenseg(floor((dist/10^(3-j))%10), 119+j*11, 18, 4, 2, "[colorize\\:"..c)
+ local lzbdisp = {c = "darkslategray", d = 888}
+ if lzb and lzb.checkpoints and lzb.checkpoints[1] then
+ local cp = lzb.checkpoints[1]
+ if advtrains.interlocking and cp.udata and cp.udata.signal_pos then
+ local sigd = advtrains.interlocking.db.get_sigd_for_signal(cp.udata.signal_pos)
+ if sigd then
+ local tcbs = advtrains.interlocking.db.get_tcbs(sigd) or {}
+ if tcbs.route_rsn then
+ table.insert(st, ("%s: %s"):format(minetest.pos_to_string(sigd.p), tcbs.route_rsn))
end
- break
end
end
+ local spd = cp.speed
+ local c, arrow
+ if not spd or spd == -1 then
+ c = "lime"
+ elseif spd == 0 then
+ c = "red"
+ elseif not res or spd <= res then
+ c = "orange"
+ arrow = true
+ elseif spd <= max then
+ c = "yellow"
+ arrow = true
+ else
+ c = "cyan"
+ arrow = true
+ end
+ if arrow then
+ hud:add(1+spd*11, 50, T"advtrains_hud_arrow.png":multiply"red")
+ end
+ local dist = math.floor(((cp.index or train.index)-train.index))
+ dist = math.max(0, math.min(999, dist))
+ lzbdisp = {c = c, d = dist}
end
+ hud:add_fill(130, 10, 30, 5, lzbdisp.c)
+ hud:add_fill(130, 35, 30, 5, lzbdisp.c)
+ hud:add_n7seg(131, 18, 28, 14, lzbdisp.d, 3, lzbdisp.c)
if res and res == 0 then
- st[#st+1] = attrans("OVERRUN RED SIGNAL! Examine situation and reverse train to move again.")
+ table.insert(st, attrans("OVERRUN RED SIGNAL! Examine situation and reverse train to move again."))
end
if train.atc_command then
- st[#st+1] = sformat("ATC: %s%s", train.atc_delay and advtrains.abs_ceil(train.atc_delay).."s " or "", train.atc_command or "")
+ local delay_str = ""
+ if train.atc_delay and train.atc_delay >= 0 then
+ delay_str = advtrains.abs_ceil(train.atc_delay).."s "
+ end
+ if train.atc_wait_finish then
+ delay_str = delay_str.."[W] "
+ end
+ if train.atc_wait_autocouple then
+ delay_str = delay_str.."[Cpl] "
+ end
+ if train.atc_wait_signal then
+ delay_str = delay_str.."[G] "
+ end
+ table.insert(st, ("ATC: %s%s"):format(delay_str, train.atc_command or ""))
end
- return table.concat(st,"\n"), table.concat(ht,":")
+ return table.concat(st,"\n"), tostring(hud)
end
local _, texture = advtrains.hud_train_format { -- dummy train object to demonstrate the train hud
max_speed = 15, speed_restriction = 15, velocity = 15, tarvelocity = 12,
active_control = true, lever = 3, ctrl = {lzb = true}, is_shunt = true,
- door_open = 1, lzb = {oncoming = {{spd=6, idx=125.7}}}, index = 0,
+ door_open = 1, lzb = {checkpoints = {{speed=6, index=125.7}}}, index = 100,
}
minetest.register_node("advtrains:hud_demo",{
diff --git a/advtrains/trainlogic.lua b/advtrains/trainlogic.lua
index 4177bc2..14f3f8b 100644
--- a/advtrains/trainlogic.lua
+++ b/advtrains/trainlogic.lua
@@ -149,10 +149,8 @@ minetest.register_on_joinplayer(function(player)
local pname = player:get_player_name()
local id=advtrains.player_to_train_mapping[pname]
if id then
- for _,wagon in pairs(minetest.luaentities) do
- if wagon.is_wagon and wagon.initialized and wagon.train_id==id then
- wagon:reattach_all()
- end
+ for _, wagon in advtrains.wagon_entity_pairs_in_train(id) do
+ wagon:reattach_all()
end
end
end)
@@ -164,12 +162,10 @@ minetest.register_on_dieplayer(function(player)
if id then
local train=advtrains.trains[id]
if not train then advtrains.player_to_train_mapping[pname]=nil return end
- for _,wagon in pairs(minetest.luaentities) do
- if wagon.is_wagon and wagon.initialized and wagon.train_id==id then
- --when player dies, detach him from the train
- --call get_off_plr on every wagon since we don't know which one he's on.
- wagon:get_off_plr(pname)
- end
+ for _, wagon in advtrains.wagon_entity_pairs_in_train(id) do
+ --when player dies, detach him from the train
+ --call get_off_plr on every wagon since we don't know which one he's on.
+ wagon:get_off_plr(pname)
end
-- just in case no wagon felt responsible for this player: clear train mapping
advtrains.player_to_train_mapping[pname] = nil
@@ -274,6 +270,10 @@ function advtrains.train_ensure_init(id, train)
atwarn(debug.traceback())
return nil
end
+
+ if not train.staticdata then
+ train.staticdata = {}
+ end
train.dirty = true
if train.no_step then
@@ -284,10 +284,12 @@ function advtrains.train_ensure_init(id, train)
assertdef(train, "velocity", 0)
--assertdef(train, "tarvelocity", 0)
assertdef(train, "acceleration", 0)
- assertdef(train, "id", id)
-
+ if train.id ~= id then
+ train.id = id
+ end
+ assertdef(train, "path_ori_cp", {})
- if not train.drives_on or not train.max_speed then
+ if not train.max_speed then
--atprint("in ensure_init: missing properties, updating!")
advtrains.update_trainpart_properties(id)
end
@@ -432,7 +434,8 @@ function advtrains.train_step_b(id, train, dtime)
if train.atc_command then
if (not train.atc_delay or train.atc_delay<=0)
and not train.atc_wait_finish
- and not train.atc_wait_autocouple then
+ and not train.atc_wait_autocouple
+ and not train.atc_wait_signal then
advtrains.atc.execute_atc_command(id, train)
elseif train.atc_delay and train.atc_delay > 0 then
train.atc_delay=train.atc_delay-dtime
@@ -461,6 +464,15 @@ function advtrains.train_step_b(id, train, dtime)
train.atc_wait_finish=nil
end
end
+ -- clear atc_wait_signal immediately when the next LZB checkpoint is not a "stop"
+ -- (but make sure lzb is initialized, otherwise wait for it)
+ if train.atc_wait_signal and train.lzb then
+ local first_ckp = train.lzb.checkpoints and train.lzb.checkpoints[1]
+ -- no checkpoint exists, or it has a speed of either nil or >0
+ if not first_ckp or first_ckp.speed ~= 0 then
+ train.atc_wait_signal = nil
+ end
+ end
if train.tarvelocity and train.tarvelocity>v0 then
--atprint("in train_step_b: applying ATC ACCEL", train.tarvelocity)
@@ -623,7 +635,7 @@ function advtrains.train_step_b(id, train, dtime)
local base_cn = train.path_cn[base_idx]
--atdebug(id,"Begin Checking for on-track collisions new_idx=",new_index_curr_tv,"base_idx=",base_idx,"base_pos=",base_pos,"base_cn=",base_cn)
-- query occupation
- local occ = advtrains.occ.get_trains_over(base_pos)
+ local occ = advtrains.occ.reverse_lookup_sel(base_pos, "close_proximity")
-- iterate other trains
for otid, ob_idx in pairs(occ) do
if otid ~= id then
@@ -653,7 +665,7 @@ function advtrains.train_step_b(id, train, dtime)
-- Phase 2 - project ref_index back onto our path and check again (necessary because there might be a turnout on the way and we are driving into the flank
if target_is_inside then
- local our_index = advtrains.path_project(otrn, ref_index, id)
+ local our_index = advtrains.path_project(otrn, ref_index, id, "before_end")
--atdebug("Backprojected our_index",our_index)
if our_index and our_index <= new_index_curr_tv
and our_index >= train.index then --FIX: If train was already past the collision point in the previous step, there is no collision! Fixes bug with split_at_index
@@ -820,9 +832,12 @@ function advtrains.train_step_c(id, train, dtime)
if is_loaded_area then
local objs = minetest.get_objects_inside_radius(rcollpos, 2)
for _,obj in ipairs(objs) do
- if not obj:is_player() and obj:get_armor_groups().fleshy and obj:get_armor_groups().fleshy > 0
- and obj:get_luaentity() and obj:get_luaentity().name~="signs_lib:text" then
- obj:punch(obj, 1, { full_punch_interval = 1.0, damage_groups = {fleshy = 1000}, }, nil)
+ if not obj:is_player() then
+ local armor = obj:get_armor_groups()
+ local luaentity = obj:get_luaentity()
+ if armor.fleshy and armor.fleshy > 0 and luaentity and luaentity.name ~= "signs_lib:text" then
+ obj:punch(obj, 1, { full_punch_interval = 1.0, damage_groups = {fleshy = 1000}, }, nil)
+ end
end
end
end
@@ -860,7 +875,7 @@ local callbacks_leave_node, run_callbacks_leave_node = mknodecallback("leave")
-- Node callback for approaching
-- Might be called multiple times, whenever path is recalculated. Also called for the first node the train is standing on, then has_entered is true.
-- signature is function(pos, id, train, index, has_entered, lzbdata)
--- has_entered: true if the "enter" callback has already been executed for this train in this location
+-- has_entered: Provided for legacy reasons. Always false. (used to signify that the enter callback has already been called, at a time where enter was still called at .5 index)
-- lzbdata: arbitrary data (shared between all callbacks), deleted when LZB is restarted.
-- These callbacks are called in order of distance as train progresses along tracks, so lzbdata can be used to
-- keep track of a train's state once it passes this point
@@ -879,13 +894,10 @@ local function tnc_call_enter_callback(pos, train_id, train, index)
run_callbacks_enter_node(pos, train_id, train, index)
-- check for split points
- if mregnode and mregnode.at_conns and #mregnode.at_conns == 3 and train.path_cp[index] == 3 then
- -- train came from connection 3 of a switch, so it split points.
- if not train.points_split then
- train.points_split = {}
- end
- train.points_split[advtrains.encode_pos(pos)] = true
- --atdebug(train_id,"split points at",pos)
+ if mregnode and mregnode.at_conn_map then
+ -- If this node has >2 conns (and a connmap), remember the connection where we came from to handle split points
+ --atdebug("Train",train_id,"at",pos,"saving turnout origin CP",train.path_cp[index],"for path item",index)
+ train.path_ori_cp[advtrains.encode_pos(pos)] = train.path_cp[index]
end
end
local function tnc_call_leave_callback(pos, train_id, train, index)
@@ -900,23 +912,16 @@ local function tnc_call_leave_callback(pos, train_id, train, index)
run_callbacks_leave_node(pos, train_id, train, index)
-- split points do not matter anymore. clear them
- if train.points_split then
- if train.points_split[advtrains.encode_pos(pos)] then
- train.points_split[advtrains.encode_pos(pos)] = nil
- --atdebug(train_id,"has passed split points at",pos)
- end
- -- any entries left?
- for _,_ in pairs(train.points_split) do
- return
- end
- train.points_split = nil
+ if mregnode and mregnode.at_conn_map then
+ -- If this node has >2 conns (and a connmap), remember the connection where we came from to handle split points
+ --atdebug("Train",train_id,"at",pos,"removing turnout origin CP for path item",index," because train has left it")
+ train.path_ori_cp[advtrains.encode_pos(pos)] = nil
end
- -- WARNING possibly unreachable place!
end
function advtrains.tnc_call_approach_callback(pos, train_id, train, index, lzbdata)
--atdebug("tnc approach",pos,train_id, lzbdata)
- local has_entered = atround(train.index) == index
+ local has_entered = false -- 2024-11-25: has_entered is now always false, because enter point = LZB hit point!
local node = advtrains.ndb.get_node(pos) --this spares the check if node is nil, it has a name in any case
local mregnode=minetest.registered_nodes[node.name]
@@ -929,18 +934,21 @@ function advtrains.tnc_call_approach_callback(pos, train_id, train, index, lzbda
end
-- === te callback definition for tnc node callbacks ===
+-- Change 2024-11-25: Enter node happens when index surpasses whole number (i.e. at center of rail)
+-- Instead of atround (prev behavior) nouw use floor and ceil (note this also fixes issues where index was exactly on .0)
+local atceil = math.ceil
advtrains.te_register_on_new_path(function(id, train)
train.tnc = {
- old_index = atround(train.index),
- old_end_index = atround(train.end_index),
+ old_index = atfloor(train.index),
+ old_end_index = atceil(train.end_index),
}
--atdebug(id,"tnc init",train.index,train.end_index)
end)
advtrains.te_register_on_update(function(id, train)
- local new_index = atround(train.index)
- local new_end_index = atround(train.end_index)
+ local new_index = atfloor(train.index)
+ local new_end_index = atceil(train.end_index)
local old_index = train.tnc.old_index
local old_end_index = train.tnc.old_end_index
while old_index < new_index do
@@ -958,8 +966,8 @@ advtrains.te_register_on_update(function(id, train)
end)
advtrains.te_register_on_create(function(id, train)
- local index = atround(train.index)
- local end_index = atround(train.end_index)
+ local index = atfloor(train.index)
+ local end_index = atceil(train.end_index)
while end_index <= index do
local pos = advtrains.round_vector_floor_y(advtrains.path_get(train,end_index))
tnc_call_enter_callback(pos, id, train, end_index)
@@ -969,8 +977,8 @@ advtrains.te_register_on_create(function(id, train)
end)
advtrains.te_register_on_remove(function(id, train)
- local index = atround(train.index)
- local end_index = atround(train.end_index)
+ local index = atfloor(train.index)
+ local end_index = atceil(train.end_index)
while end_index <= index do
local pos = advtrains.round_vector_floor_y(advtrains.path_get(train,end_index))
tnc_call_leave_callback(pos, id, train, end_index)
@@ -1052,10 +1060,9 @@ end
-- Note: safe_decouple_wagon() has been moved to wagons.lua
--- this function sets wagon's pos_in_train(parts) properties and train's max_speed and drives_on (and more)
+-- this function sets wagon's pos_in_train(parts) properties and train's max_speed (and more)
function advtrains.update_trainpart_properties(train_id, invert_flipstate)
local train=advtrains.trains[train_id]
- train.drives_on=advtrains.merge_tables(advtrains.all_tracktypes)
--FIX: deep-copy the table!!!
train.max_speed=20
train.extent_h = 0;
@@ -1071,7 +1078,16 @@ function advtrains.update_trainpart_properties(train_id, invert_flipstate)
if data then
local wagon = advtrains.wagon_prototypes[data.type or data.entity_name]
if not wagon then
- atwarn("Wagon '",data.type,"' couldn't be found. Please check that all required modules are loaded!")
+ local ent = advtrains.wagon_objects[w_id]
+ local pdesc
+ if ent and ent:get_pos() then
+ pdesc = "at " .. minetest.pos_to_string(ent:get_pos())
+ elseif train.last_pos then
+ pdesc = "near " .. minetest.pos_to_string(train.last_pos)
+ else
+ pdesc = "at an unknown location"
+ end
+ atwarn(string.format("Wagon %q %s could not be found. Please check that all required modules are loaded!", data.type, pdesc))
wagon = advtrains.wagon_prototypes["advtrains:wagon_placeholder"]
end
@@ -1088,13 +1104,6 @@ function advtrains.update_trainpart_properties(train_id, invert_flipstate)
end
rel_pos=rel_pos+wagon.wagon_span
- if wagon.drives_on then
- for k,_ in pairs(train.drives_on) do
- if not wagon.drives_on[k] then
- train.drives_on[k]=nil
- end
- end
- end
train.max_speed=math.min(train.max_speed, wagon.max_speed)
train.extent_h = math.max(train.extent_h, wagon.extent_h or 1);
end
@@ -1126,8 +1135,17 @@ function advtrains.spawn_wagons(train_id)
if advtrains.position_in_range(pos, ablkrng) then
--atdebug("wagon",w_id,"spawning")
local wt = advtrains.get_wagon_prototype(data)
- local wagon = minetest.add_entity(pos, wt):get_luaentity()
- wagon:set_id(w_id)
+ local wobj = minetest.add_entity(pos, wt)
+ if not wobj then
+ atwarn("Failed to spawn wagon", w_id, "of type", wt)
+ else
+ local wagon = wobj:get_luaentity()
+ if not wagon then
+ atwarn("Wagon", w_id, "of type", wt, "spawned with nil luaentity")
+ else
+ wagon:set_id(w_id)
+ end
+ end
end
end
else
@@ -1194,6 +1212,8 @@ function advtrains.split_train_at_index(train, index)
newtrain.points_split = advtrains.merge_tables(train.points_split)
newtrain.autocouple = train.autocouple
+ advtrains.te_run_callbacks_on_decouple(train, newtrain, index)
+
return newtrain_id -- return new train ID, so new train can be manipulated
end
@@ -1206,8 +1226,24 @@ function advtrains.invert_train(train_id)
return
end
+ -- Before flipping the train, we must check if there are any points on the path
+ -- which will become split on rotating, and store their cn (which will become the cp)
+ local ori_cp_after_flip = {}
+ for index = atround(train.end_index),atround(train.index) do
+ local pos = advtrains.path_get(train, index)
+ local ok, conns, railheight, connmap = advtrains.get_rail_info_at(pos)
+ if ok and connmap then
+ --atdebug("Reversing Train",train.id," ori_cp Checks: at",pos,"saving turnout origin CP",train.path_cn[index],"for path item",index)
+ ori_cp_after_flip[advtrains.encode_pos(pos)] = train.path_cn[index]
+ end
+ end
+
+ -- Actual rotation happens here! This sets the path restore position to the end of the train, inverting the connid.
advtrains.path_setrestore(train, true)
+ -- clear the origin cp list because it is now invalid, and replace it by what we built prior.
+ train.path_ori_cp = ori_cp_after_flip
+
-- rotate some other stuff
if train.door_open then
train.door_open = - train.door_open
@@ -1227,7 +1263,6 @@ function advtrains.invert_train(train_id)
advtrains.update_trainpart_properties(train_id, true)
-- recalculate path
- advtrains.train_ensure_init(train_id, train)
-- If interlocking present, check whether this train is in a section and then set as shunt move after reversion
if advtrains.interlocking and train.il_sections and #train.il_sections > 0 then
@@ -1253,7 +1288,7 @@ function advtrains.invalidate_all_paths(pos)
local tab
if pos then
-- if position given, check occupation system
- tab = advtrains.occ.get_trains_over(pos)
+ tab = advtrains.occ.reverse_lookup_quick(pos)
else
tab = advtrains.trains
end
@@ -1266,7 +1301,7 @@ end
-- Calls invalidate_path_ahead on all trains occupying (having paths over) this node
-- Can be called during train step.
function advtrains.invalidate_all_paths_ahead(pos)
- local tab = advtrains.occ.get_trains_over(pos)
+ local tab = advtrains.occ.reverse_lookup_sel(pos, "first_ahead")
for id,index in pairs(tab) do
local train = advtrains.trains[id]
diff --git a/advtrains/wagonprop_tool.lua b/advtrains/wagonprop_tool.lua
new file mode 100644
index 0000000..2a4a9e2
--- /dev/null
+++ b/advtrains/wagonprop_tool.lua
@@ -0,0 +1,43 @@
+minetest.register_craftitem("advtrains:wagon_prop_tool",{ --craftitem because it does nothing on its own
+ description = attrans("Wagon Properties Tool\nPunch a wagon to view and edit the Wagon Properties"),
+ short_description = attrans("Wagon Properties Tool"),
+ groups = {},
+ inventory_image = "advtrains_wagon_prop_tool.png",
+ wield_image = "advtrains_wagon_prop_tool.png",
+ stack_max = 1,
+ on_use = function(itemstack, user, pointed_thing)
+ local pname = user:get_player_name()
+ if not pname or pname == "" then
+ return
+ end
+
+ --sanity checks in case of clicking the wrong entity/node/nothing
+ if pointed_thing.type ~= "object" then return end --not an entity
+ local object = pointed_thing.ref:get_luaentity()
+ if not object.id then return end --entity doesn't have an id field
+
+ local wagon = advtrains.wagons[object.id] --check if wagon exists in advtrains
+ if not wagon then --not a wagon
+ return
+ end --end sanity checks
+
+ --whitelist protection check
+ if not advtrains.check_driving_couple_protection(pname,wagon.owner,wagon.whitelist) then
+ minetest.chat_send_player(pname, attrans("Insufficient privileges to use this!"))
+ return
+ end
+ object:show_wagon_properties(pname)
+ return itemstack
+ end,
+})
+
+if minetest.get_modpath("default") then --register recipe
+ minetest.register_craft({
+ output = "advtrains:wagon_prop_tool",
+ recipe = {
+ {"advtrains:dtrack_placer","dye:black","default:paper"},
+ {"screwdriver:screwdriver","default:paper","default:paper"},
+ {"","","group:wood"},
+ }
+ })
+end \ No newline at end of file
diff --git a/advtrains/wagons.lua b/advtrains/wagons.lua
index 7c952f5..38210ef 100644
--- a/advtrains/wagons.lua
+++ b/advtrains/wagons.lua
@@ -14,7 +14,13 @@ local IGNORE_WORLD = advtrains.IGNORE_WORLD
local has_wielded_light = core.get_modpath("wielded_light")
advtrains.wagons = {}
-advtrains.wagon_prototypes = {}
+advtrains.wagon_alias = {}
+advtrains.wagon_prototypes = setmetatable({}, {
+ __index = function(t, k)
+ local _, proto = advtrains.resolve_wagon_alias(k)
+ return proto
+ end
+})
advtrains.wagon_objects = {}
local unload_wgn_range = advtrains.wagon_load_range + 32
@@ -61,6 +67,7 @@ local wagon = {
has_inventory=false,
}
+
function wagon:train()
local data = advtrains.wagons[self.id]
return advtrains.trains[data.train_id]
@@ -203,7 +210,7 @@ function wagon:on_punch(puncher, time_from_last_punch, tool_capabilities, direct
end
for listname, _ in pairs(inv:get_lists()) do
if not inv:is_empty(listname) then
- minetest.chat_send_player(puncher:get_player_name(), attrans("The wagon's inventory is not empty!"));
+ minetest.chat_send_player(puncher:get_player_name(), attrans("The wagon's inventory is not empty."));
return
end
end
@@ -283,7 +290,7 @@ function wagon:on_step(dtime)
local data = advtrains.wagons[self.id]
if not pos then
- --atdebug("["..self.id.."][fatal] missing position (object:getpos() returned nil)")
+ --atdebug("["..self.id.."][fatal] missing position (object:get_pos() returned nil)")
return
end
@@ -366,7 +373,16 @@ function wagon:on_step(dtime)
--show off-track information in outside text instead of notifying the whole server about this
if train.off_track then
- outside = outside .."\n" .. attrans("!!! Train off track !!!")
+ outside = outside .."\n!!! Vlak mimo koleje !!!"
+ end
+
+ -- liquid container: display liquid contents in infotext
+ if self.techage_liquid_capacity then
+ if data.techage_liquid and data.techage_liquid.name then
+ outside = outside .."\nKapalina: "..data.techage_liquid.name..", "..data.techage_liquid.amount.." jednotek"
+ else
+ outside = outside .."\nKapalina: empty"
+ end
end
if self.infotext_cache~=outside then
@@ -696,7 +712,7 @@ function wagon:on_rightclick(clicker)
end
local doors_open = self:train().door_open~=0 or clicker:get_player_control().sneak
- local allow, rsn=false, "Wagon has no seats!"
+ local allow, rsn=false, attrans("This wagon has no seats.")
for _,sgr in ipairs(self.assign_to_seat_group) do
allow, rsn = self:check_seat_group_access(pname, sgr)
if allow then
@@ -707,16 +723,16 @@ function wagon:on_rightclick(clicker)
self:get_on(clicker, seatid)
return
else
- rsn=attrans("Wagon is full.")
+ rsn=attrans("This wagon is full.")
end
else
- rsn=attrans("Doors are closed! (try holding sneak key!)")
+ rsn=attrans("Doors are closed! (Try holding sneak key!)")
end
end
end
end
end
- minetest.chat_send_player(pname, attrans("Can't get on: @1", rsn))
+ minetest.chat_send_player(pname, rsn or attrans("You can't get on this wagon."))
else
self:show_get_on_form(pname)
end
@@ -866,8 +882,9 @@ function wagon:show_wagon_properties(pname)
]]
local data = advtrains.wagons[self.id]
local form="size[5,5]"
- form = form .. "field[0.5,1;4.5,1;whitelist;" .. attrans("Allow these players to access your wagon:") .. ";"..minetest.formspec_escape(data.whitelist or "").."]"
- form = form .. "field[0.5,2;4.5,1;roadnumber;" .. attrans("Wagon road number:") .. ";"..minetest.formspec_escape(data.roadnumber or "").."]"
+ form=form.."label[0.2,0;"..attrans("This Wagon ID")..": "..self.id.."]"
+ form = form .. "field[0.5,1;4.5,1;whitelist;Allow these players to access your wagon:;"..minetest.formspec_escape(data.whitelist or "").."]"
+ form = form .. "field[0.5,2;4.5,1;roadnumber;Wagon road number:;"..minetest.formspec_escape(data.roadnumber or "").."]"
local fc = ""
if data.fc then
fc = table.concat(data.fc, "!")
@@ -979,6 +996,7 @@ function wagon:show_bordcom(pname)
local linhei
local form = "size[11,9]label[0.5,0;AdvTrains Boardcom v0.1]"
+ form=form.."textarea[7.5,0.05;10,1;;"..attrans("Train ID")..": "..(minetest.formspec_escape(train.id or ""))..";]"
form=form.."textarea[0.5,1.5;7,1;text_outside;"..attrans("Text displayed outside on train")..";"..(minetest.formspec_escape(train.text_outside or "")).."]"
form=form.."textarea[0.5,3;7,1;text_inside;"..attrans("Text displayed inside train")..";"..(minetest.formspec_escape(train.text_inside or "")).."]"
form=form.."field[7.5,1.75;3,1;line;"..attrans("Line")..";"..(minetest.formspec_escape(train.line or "")).."]"
@@ -1096,12 +1114,11 @@ function wagon:handle_bordcom_fields(pname, formname, fields)
for i, tpid in ipairs(train.trainparts) do
if fields["dcpl_"..i] then
advtrains.safe_decouple_wagon(tpid, pname)
- elseif fields["wgprp"..i] then
- for _,wagon in pairs(minetest.luaentities) do
- if wagon.is_wagon and wagon.initialized and wagon.id==tpid and data.owner==pname then
- wagon:show_wagon_properties(pname)
- return
- end
+ elseif fields["wgprp"..i] and data.owner==pname then
+ local wagon = advtrains.get_wagon_entity(tpid)
+ if wagon then
+ wagon:show_wagon_properties(pname)
+ return
end
end
end
@@ -1147,44 +1164,48 @@ end
minetest.register_on_player_receive_fields(function(player, formname, fields)
local uid=string.match(formname, "^advtrains_geton_(.+)$")
if uid then
- for _,wagon in pairs(minetest.luaentities) do
- if wagon.is_wagon and wagon.initialized and wagon.id==uid then
- local data = advtrains.wagons[wagon.id]
- if fields.inv then
- if wagon.has_inventory and wagon.get_inventory_formspec then
- minetest.show_formspec(player:get_player_name(), "advtrains_inv_"..uid, wagon:get_inventory_formspec(player:get_player_name(), make_inv_name(uid)))
- end
- elseif fields.seat then
- local val=minetest.explode_textlist_event(fields.seat)
- if val and val.type~="INV" and not data.seatp[player:get_player_name()] then
- --get on
- wagon:get_on(player, val.index)
- --will work with the new close_formspec functionality. close exactly this formspec.
- minetest.show_formspec(player:get_player_name(), formname, "")
- end
+ local wagon = advtrains.get_wagon_entity(uid)
+ if wagon then
+ local data = advtrains.wagons[wagon.id]
+ if fields.inv then
+ if wagon.has_inventory and wagon.get_inventory_formspec then
+ minetest.show_formspec(player:get_player_name(), "advtrains_inv_"..uid, wagon:get_inventory_formspec(player:get_player_name(), make_inv_name(uid)))
+ end
+ elseif fields.seat then
+ local val=minetest.explode_textlist_event(fields.seat)
+ if val and val.type~="INV" and not data.seatp[player:get_player_name()] then
+ --get on
+ wagon:get_on(player, val.index)
+ --will work with the new close_formspec functionality. close exactly this formspec.
+ minetest.show_formspec(player:get_player_name(), formname, "")
end
end
end
+ return true
end
+
uid=string.match(formname, "^advtrains_seating_(.+)$")
if uid then
- for _,wagon in pairs(minetest.luaentities) do
- if wagon.is_wagon and wagon.initialized and wagon.id==uid then
- local pname=player:get_player_name()
- local no=wagon:get_seatno(pname)
- if no then
- if wagon.seat_groups then
- wagon:seating_from_key_helper(pname, fields, no)
- end
+ local wagon = advtrains.get_wagon_entity(uid)
+ if wagon then
+ local pname=player:get_player_name()
+ local no=wagon:get_seatno(pname)
+ if no then
+ if wagon.seat_groups then
+ wagon:seating_from_key_helper(pname, fields, no)
end
end
end
+ return true
end
+
uid=string.match(formname, "^advtrains_prop_(.+)$")
if uid then
local pname=player:get_player_name()
local data = advtrains.wagons[uid]
- if pname~=data.owner and not minetest.check_player_privs(pname, {train_admin = true}) then
+ if not data then
+ return true
+ elseif pname~=data.owner and not minetest.check_player_privs(pname, {train_admin = true}) then
return true
end
if fields.save or not fields.quit then
@@ -1206,29 +1227,32 @@ minetest.register_on_player_receive_fields(function(player, formname, fields)
wagon.show_wagon_properties({id=uid}, pname)
end
end
+ return true
end
uid=string.match(formname, "^advtrains_bordcom_(.+)$")
if uid then
- for _,wagon in pairs(minetest.luaentities) do
- if wagon.is_wagon and wagon.initialized and wagon.id==uid then
- wagon:handle_bordcom_fields(player:get_player_name(), formname, fields)
- end
+ local wagon = advtrains.get_wagon_entity(uid)
+ if wagon then
+ wagon:handle_bordcom_fields(player:get_player_name(), formname, fields)
end
+ return true
end
+
uid=string.match(formname, "^advtrains_inv_(.+)$")
if uid then
local pname=player:get_player_name()
local data = advtrains.wagons[uid]
if fields.prop and data.owner==pname then
- for _,wagon in pairs(minetest.luaentities) do
- if wagon.is_wagon and wagon.initialized and wagon.id==uid and data.owner==pname then
- wagon:show_wagon_properties(pname)
- --wagon:handle_bordcom_fields(player:get_player_name(), formname, fields)
- end
+ local wagon = advtrains.get_wagon_entity(uid)
+ if wagon then
+ wagon:show_wagon_properties(pname)
+ --wagon:handle_bordcom_fields(player:get_player_name(), formname, fields)
end
end
+ return true
end
end)
+
function wagon:seating_from_key_helper(pname, fields, no)
local data = advtrains.wagons[self.id]
local sgr=self.seats[no].group
@@ -1255,7 +1279,7 @@ function wagon:seating_from_key_helper(pname, fields, no)
minetest.after(0.1, function(pn) self:show_bordcom(pn) end, pname)
end
if fields.dcwarn then
- minetest.chat_send_player(pname, attrans("Doors are closed! Use Sneak+rightclick to ignore the closed doors and get off!"))
+ minetest.chat_send_player(pname, attrans("Doors are closed. Use Sneak+rightclick to ignore the closed doors and get off."))
end
if fields.off then
self:get_off(no)
@@ -1264,7 +1288,7 @@ end
function wagon:check_seat_group_access(pname, sgr)
local data = advtrains.wagons[self.id]
if self.seat_groups[sgr].driving_ctrl_access and not (advtrains.check_driving_couple_protection(pname, data.owner, data.whitelist)) then
- return false, "Not allowed to access a driver stand!"
+ return false, attrans("You are not allowed to access the driver stand.")
end
if self.seat_groups[sgr].driving_ctrl_access then
advtrains.log("Drive", pname, self.object:get_pos(), self:train().text_outside)
@@ -1325,11 +1349,33 @@ function advtrains.get_wagon_prototype(data)
data.type = data.entity_name
data.entity_name = nil
end
- if not wt or not advtrains.wagon_prototypes[wt] then
- atwarn("Unable to load wagon type",wt,", using placeholder")
- wt="advtrains:wagon_placeholder"
+ local rt, proto = advtrains.resolve_wagon_alias(wt)
+ if not rt then
+ --atwarn("Unable to load wagon type",wt,", using placeholder")
+ rt = "advtrains:wagon_placeholder"
+ proto = advtrains.wagon_prototypes[rt]
+ end
+ return rt, proto
+end
+
+function advtrains.register_wagon_alias(src, dst)
+ advtrains.wagon_alias[src] = dst
+end
+
+local function recursive_resolve_alias(name, seen)
+ local prototype = rawget(advtrains.wagon_prototypes, name)
+ if prototype then
+ return name, prototype
+ end
+ local resolved = advtrains.wagon_alias[name]
+ if resolved and not seen[resolved] then
+ seen[name] = true
+ return recursive_resolve_alias(resolved, seen)
end
- return wt, advtrains.wagon_prototypes[wt]
+end
+
+function advtrains.resolve_wagon_alias(name)
+ return recursive_resolve_alias(name, {})
end
function advtrains.standard_inventory_formspec(self, pname, invname)
@@ -1394,15 +1440,25 @@ function advtrains.register_wagon(sysname_p, prototype, desc, inv_img, nincreati
groups = wagon_groups,
on_place = function(itemstack, placer, pointed_thing)
- if not pointed_thing.type == "node" then
+ if pointed_thing.type ~= "node" then
return
end
+
+ local pos = pointed_thing.under
+ local node = minetest.get_node(pos)
+ local pointed_def = minetest.registered_nodes[node.name]
+ if pointed_def and pointed_def.on_rightclick then
+ local controls = placer:get_player_control()
+ if not controls.sneak then
+ return pointed_def.on_rightclick(pos, node, placer, itemstack, pointed_thing)
+ end
+ end
+
local pname = placer:get_player_name()
- local node=minetest.get_node_or_nil(pointed_thing.under)
- if not node then atprint("[advtrains]Ignore at placer position") return itemstack end
+ if node.name == "ignore" then atprint("[advtrains]Ignore at placer position") return itemstack end
local nodename=node.name
- if(not advtrains.is_track_and_drives_on(nodename, prototype.drives_on)) then
+ if(not advtrains.is_track(nodename)) then
atprint("no track here, not placing.")
return itemstack
end
@@ -1410,7 +1466,7 @@ function advtrains.register_wagon(sysname_p, prototype, desc, inv_img, nincreati
minetest.chat_send_player(pname, attrans("You don't have the train_operator privilege."))
return itemstack
end
- if not minetest.check_player_privs(placer, {train_admin = true }) and minetest.is_protected(pointed_thing.under, placer:get_player_name()) then
+ if not minetest.check_player_privs(placer, {train_admin = true }) and minetest.is_protected(pos, placer:get_player_name()) then
return itemstack
end
local tconns=advtrains.get_track_connections(node.name, node.param2)
@@ -1425,7 +1481,7 @@ function advtrains.register_wagon(sysname_p, prototype, desc, inv_img, nincreati
local wid = advtrains.create_wagon(sysname, pname)
- local id=advtrains.create_new_train_at(pointed_thing.under, plconnid, 0, {wid})
+ local id=advtrains.create_new_train_at(pos, plconnid, 0, {wid})
if not advtrains.is_creative(pname) then
itemstack:take_item()
@@ -1442,7 +1498,6 @@ advtrains.register_wagon("advtrains:wagon_placeholder", {
collisionbox = {-0.3,-0.3,-0.3, 0.3,0.3,0.3},
visual_size = {x=0.7, y=0.7},
initial_sprite_basepos = {x=0, y=0},
- drives_on = advtrains.all_tracktypes,
max_speed = 5,
seats = {
},
@@ -1451,5 +1506,90 @@ advtrains.register_wagon("advtrains:wagon_placeholder", {
assign_to_seat_group = {},
wagon_span=1,
drops={},
-}, attrans("Wagon placeholder"), "advtrains_wagon_placeholder.png", true)
+}, "Wagon placeholder", "advtrains_wagon_placeholder.png", true)
+
+
+-- Helper function to retrieve the wagon at a certain position in a train, given its train ID and the desired index within that train's path
+--
+-- Returns: wagon_num, wagon_id, wagon_data, offset_from_center
+-- wagon_num: The n'th wagon in the train (index into "trainparts" table)
+-- wagon_id: The wagon ID. Obtain wagon data from advtrains.wagons[wagon_id], and subsequently the wagon prototype via advtrains.get_wagon_prototype(data)
+-- offset_from_center: The offset (an absolute distance value) from the center point of the wagon. Positive is towards the end of the train, negative towards the start. (note that this is inverse to the counting direction of the index!)
+--
+--[[ To get the wagon standing at a certain world position, you first need to retrieve the index via the occupation table, as follows:
+ local trains = advtrains.occ.get_trains_at(pos)
+ for train_id, index in pairs(trains) do
+ local wagon_num, wagon_id, wagon_data, offset_from_center = advtrains.get_wagon_at_index(train_id, index)
+ if wagon_num then
+ ...
+ end
+ end
+]]--
+function advtrains.get_wagon_at_index(train_id, w_index)
+ local train = advtrains.trains[train_id]
+ if not train then error("Passed train id "..train_id.." doesnt exist") end
+ -- ensure init - always required
+ advtrains.train_ensure_init(train_id, train)
+ -- Use path dist to determine the offset from the start of the train
+ local dstart = advtrains.path_get_path_dist_fractional(train, train.index)
+ local dtarget = advtrains.path_get_path_dist_fractional(train, w_index)
+ local dist_from_start = dstart - dtarget -- NOTE: dist_from_start is supposed to be positive, but dtarget will be smaller than dstart
+ -- if dist_from_start is <0, we are outside of train
+ if dist_from_start < 0 then
+ return nil
+ end
+ -- scan over wagons to see if dist_from_start falls into its window
+ local start_pos = 0
+ local center_pos
+ local end_pos
+ local i = 1
+ while train.trainparts[i] do
+ local w_id = train.trainparts[i]
+ -- get wagon prototype to retrieve wagon span
+ local wdata = advtrains.wagons[w_id]
+ if wdata then
+ local wtype, wproto = advtrains.get_wagon_prototype(wdata)
+ local wagon_span = wproto.wagon_span
+ -- determine center and end pos
+ center_pos = start_pos + wagon_span
+ end_pos = center_pos + wagon_span
+ if start_pos <= dist_from_start and dist_from_start < end_pos then
+ -- Found the correct wagon in the train!
+ local offset_from_center = dist_from_start - center_pos
+ return i, w_id, wdata, offset_from_center
+ end
+ -- go on
+ start_pos = end_pos
+ else
+ error("Wagon "..w_id.." from train "..train_id.." doesnt exist!")
+ end
+ i = i + 1
+ end
+ -- nothing found, dist must be further back
+ return nil
+end
+
+function advtrains.get_wagon_entity(wagon_id)
+ if not advtrains.wagons[wagon_id] then return end
+ local object = advtrains.wagon_objects[wagon_id]
+ if object then
+ return object:get_luaentity()
+ end
+end
+
+function advtrains.next_wagon_entity_in_train(train, i)
+ local wagon_id = train.trainparts[i + 1]
+ if wagon_id then
+ local wagon = advtrains.get_wagon_entity(wagon_id)
+ if wagon then
+ return i + 1, wagon
+ end
+ end
+end
+
+function advtrains.wagon_entity_pairs_in_train(train_id)
+ local train = advtrains.trains[train_id]
+ if not train then return function() end end
+ return advtrains.next_wagon_entity_in_train, train, 0
+end
diff --git a/advtrains_interlocking/README.md b/advtrains_interlocking/README.md
new file mode 100644
index 0000000..d4a2699
--- /dev/null
+++ b/advtrains_interlocking/README.md
@@ -0,0 +1,85 @@
+# Interlocking for Advtrains
+
+The `advtrains_interlocking` mod provides various interlocking and signaling features for Advtrains.
+
+## Signal aspect tables
+
+Signal aspects are represented using tables with the following (optional) fields:
+
+* `main`: The main signal aspect. It provides information on the permitted speed after passing the signal.
+* `dst`: The distant signal aspect. It provides information on the permitted speed after passing the next signal.
+* `shunt`: Whether the train may proceed in shunt mode and, if the main aspect is danger, proceed in shunt mode.
+* `proceed_as_main`: Whether the train should exit shunt mode when proceeding.
+* `group`: The name of the signal group.
+* `name`: The name of the signal aspect.
+
+The `main` and `dst` fields may be:
+
+* An positive number indicating the permitted speed,
+* The number 0, indicating that the train should expect to stop at the current signal (or, for the `dst` field, the next signal),
+* The number -1, indicating that the train can proceed (or, for the `dst` field, expect to proceed) at maximum speed,
+* The constant `false`, indicating no change to the speed restriction, or
+* The constant `nil`, indicating that the default value for the name aspect (if present) is used. If no valid signal aspect is named, or the signal aspect does not provide a default value, the value is assumed to be `false`.
+
+### Node definitions
+
+Signals should belong the following node groups:
+
+* `advtrains_signal`: `1` for static signals, `2` for signals with variable aspects.
+* `save_in_at_nodedb`: This should be set to `1` to make sure that Advtrains always has access to the signal.
+* `not_blocking_trains`: Setting this to `1` prevents trains from colliding with the signal. Setting this is not necessary, but recommended.
+
+The node definition should contain an `advtrains` field.
+
+The `advtrains` field of the node definition should contain a `supported_aspects` table for signals with variable aspects.
+
+The `supported_aspects` table should contain the following fields:
+
+* `main`: A list of values supported for the main aspect.
+* `dst`: A list of values supported for the distant aspect.
+* `shunt`: The value for the `shunt` field of the signal aspect or `nil` if the value is variable.
+* `proceed_as_main`: The value for the `proceed_as_main` field of the signal aspect.
+* `group`: The name of the signal group.
+* `name`: A list of supported (named) aspects.
+* `dst_shift`: The phase shift for distant/repeater signals. This field should not be set for main signals.
+
+The `advtrains` field of the node definition should contain a `get_aspect` function. This function is given the position of the signal and the node at the position. It should return the signal aspect table or, in the case of type 2 signals, the name of the signal aspect.
+
+For signals with variable aspects, a corresponding `set_aspect` function should also be defined. This function is given the position of the signal, the node at the position, and the new aspect. The new aspect is not guaranteed to be supported by the signal itself.
+
+Signals should also have the following callbacks set:
+
+* `on_rightclick` should be set to `advtrains.interlocking.signal_rc_handler`
+* `can_dig` should be set to `advtrains.interlocking.signal_can_dig`
+* `after_dig_node` should be set to `advtrains.interlocking.signal_after_dig`
+
+Alternatively, custom callbacks should call the respective functions.
+
+## Signal groups
+
+Signals may belong to signal groups are registered using `advtrains.interlocking.aspect.register_group`.
+
+Signal group definitions include the following fields:
+
+* `name`: The internal name of the signal group. It is recommended to use the mod name as a prefix to avoid name collisions.
+* `label`: The description of the signal group.
+* `aspects`: A table of signal aspects. Entries with string indices define the signal aspect with the name. Entries with numeric indices (starting from 1, counting upward) contain a list of corresponding aspect names (the first entry is preferred) and are mainly used for routing, where larger indices indicate that the signal with the aspect is closer to the signal with the "danger" (or similar) aspect.
+
+Each aspect in the signal group definition table should contain the following fields:
+
+* `label`: The description of the signal aspect.
+* `main`, `shunt`, `proceed_as_main`: The default values for the aspect. Note that the `dst` field has no default value as it is automatically adjusted.
+
+## Notes
+
+It is allowed to provide other methods of setting the signal aspect. However:
+
+* These changes are ignored by the routesetting system.
+* Please call `advtrains.interlocking.signal_readjust_aspect` after the signal aspect has changed.
+
+## Examples
+An example of speed signals can be found in `advtrains_signals_ks`, which provides a subset of German signals.
+
+An example of route signals can be found in `advtrains_signals_japan`, which provides a subset of Japanese signals.
+
+The mods mentioned above are also used for demonstation purposes and can also be used for testing.
diff --git a/advtrains_interlocking/approach.lua b/advtrains_interlocking/approach.lua
index eecf09a..eaf0248 100644
--- a/advtrains_interlocking/approach.lua
+++ b/advtrains_interlocking/approach.lua
@@ -64,10 +64,7 @@ advtrains.tnc_register_on_approach(function(pos, id, train, index, has_entered,
-- resetting the path does not matter to the set route and ARS doesn't need to be called again.
if spos and ars_enabled then
--atdebug(id,"IL Spos (ARS)",spos,asp)
- local sigd = il.db.get_sigd_for_signal(spos)
- if sigd then
- il.ars_check(sigd, train)
- end
+ il.ars_check(spos, train)
end
--atdebug("trav: ",pos, cn, asp, spos, "travsht=", lzb.travsht)
local lspd
diff --git a/advtrains_interlocking/ars.lua b/advtrains_interlocking/ars.lua
index 4f50df9..eb10497 100644
--- a/advtrains_interlocking/ars.lua
+++ b/advtrains_interlocking/ars.lua
@@ -52,7 +52,7 @@ function il.ars_to_text(arstab)
end
function il.text_to_ars(t)
- if t=="" then
+ if not string.match(t, "%S+") then
return nil
elseif t=="*" then
return {default=true}
@@ -129,29 +129,49 @@ function il.ars_check_rule_match(ars, train)
return nil
end
-function advtrains.interlocking.ars_check(sigd, train)
- local tcbs = il.db.get_tcbs(sigd)
- if not tcbs or not tcbs.routes then return end
-
- if tcbs.ars_disabled or tcbs.ars_ignore_next then
- -- No-ARS mode of signal.
- -- ignore...
- -- Note: ars_ignore_next is set by signalling formspec when route is cancelled
- tcbs.ars_ignore_next = nil
- return
- end
-
- if tcbs.routeset then
- -- ARS is not in effect when a route is already set
- -- just "punch" routesetting, just in case callback got lost.
- minetest.after(0, il.route.update_route, sigd, tcbs, nil, nil)
- return
+function advtrains.interlocking.ars_check(signalpos, train, trig_from_dst)
+ -- check for distant signal
+ -- this whole check must be delayed until after the route setting has taken place,
+ -- because before that the distant signal is yet unknown
+ if not trig_from_dst then
+ minetest.after(0.5, function()
+ -- does signal have dst?
+ local _, remote = il.signal.get_aspect(signalpos)
+ if remote then
+ advtrains.interlocking.ars_check(remote, train, true)
+ end
+ end)
end
-
- local rteid = find_rtematch(tcbs.routes, train)
- if rteid then
- --delay routesetting, it should not occur inside train step
- -- using after here is OK because that gets called on every path recalculation
- minetest.after(0, il.route.update_route, sigd, tcbs, rteid, nil)
+
+ local sigd = il.db.get_sigd_for_signal(signalpos)
+ local tcbs = sigd and il.db.get_tcbs(sigd)
+ -- trigger ARS on this signal
+ if tcbs and tcbs.routes then
+
+ if tcbs.ars_disabled or tcbs.ars_ignore_next then
+ -- No-ARS mode of signal.
+ -- ignore...
+ -- Note: ars_ignore_next is set by signalling formspec when route is cancelled
+ tcbs.ars_ignore_next = nil
+ return
+ end
+ if trig_from_dst and tcbs.no_dst_ars_trig then
+ -- signal not to be triggered from distant
+ return
+ end
+
+ if tcbs.routeset then
+ -- ARS is not in effect when a route is already set
+ -- just "punch" routesetting, just in case callback got lost.
+ minetest.after(0, il.route.update_route, sigd, tcbs, nil, nil)
+ return
+ end
+
+ local rteid = find_rtematch(tcbs.routes, train)
+ if rteid then
+ --delay routesetting, it should not occur inside train step
+ -- using after here is OK because that gets called on every path recalculation
+ minetest.after(0, il.route.update_route, sigd, tcbs, rteid, nil)
+ end
end
end
diff --git a/advtrains_interlocking/autonaming.lua b/advtrains_interlocking/autonaming.lua
new file mode 100644
index 0000000..f88ccbd
--- /dev/null
+++ b/advtrains_interlocking/autonaming.lua
@@ -0,0 +1,69 @@
+-- autonaming.lua
+-- Automatically set names of signals (and maybe track sections) based on numbering
+
+
+local function find_highest_count(prefix)
+ local cnt = 0
+ local pattern = "^"..prefix.."(%d+)$"
+ local alltcb = advtrains.interlocking.db.get_all_tcb()
+ for _,tcb in pairs(alltcb) do
+ for _,tcbs in pairs(tcb) do
+ if tcbs.signal_name then
+ local mnum = string.match(tcbs.signal_name, pattern)
+ if mnum then
+ local n = tonumber(mnum)
+ if n and n > cnt then
+ cnt = n
+ end
+ end
+ end
+ end
+ end
+ return cnt
+end
+
+-- { pname = { prefix = "FOO", count = 7 } }
+local player_prefix_info = {}
+
+function advtrains.interlocking.set_autoname_prefix(pname, prefix)
+ if prefix and #prefix>0 then
+ -- check that it is valid
+ if not string.match(prefix,"[A-Za-z_]+") then
+ return false, "Illegal prefix, only letters and _ allowed"
+ end
+ -- scan database for this prefix to find out the highest count
+ local count = find_highest_count(prefix)
+ player_prefix_info[pname] = { prefix = prefix, count = count}
+ return true, "Prefix set, next signal name will be: ".. advtrains.interlocking.get_next_autoname(pname, true)
+ else
+ player_prefix_info[pname] = nil
+ return true, "Prefix unset, signals are not auto-named for you!"
+ end
+end
+
+function advtrains.interlocking.get_next_autoname(pname, no_increment)
+ local pi = player_prefix_info[pname]
+ if pi then
+ local name = pi.prefix..(pi.count+1)
+ if not no_increment then pi.count = pi.count+1 end
+ return name
+ else
+ return nil
+ end
+end
+
+function advtrains.interlocking.add_autoname_to_tcbs(tcbs, pname)
+ if not tcbs or not pname then return end
+ if tcbs.signal_name then return end -- name already set
+
+ tcbs.signal_name = advtrains.interlocking.get_next_autoname(pname)
+end
+
+minetest.register_chatcommand("at_nameprefix",
+ {
+ params = "<prefix>",
+ description = "Sets the current prefix for automatically naming interlocking components. Example: '/at_nameprefix TEST' - signals will be named TEST1, TEST2 and so on",
+ privs = {interlocking = true},
+ func = advtrains.interlocking.set_autoname_prefix,
+})
+
diff --git a/advtrains_interlocking/database.lua b/advtrains_interlocking/database.lua
index a362296..844d350 100644
--- a/advtrains_interlocking/database.lua
+++ b/advtrains_interlocking/database.lua
@@ -2,71 +2,19 @@
-- saving the location of TCB's, their neighbors and their state
--[[
-== THIS COMMENT IS PARTIALLY INCORRECT AND OUTDATED! ==
-
-The interlocking system is based on track circuits.
-Track circuit breaks must be manually set by the user. Signals must be assigned to track circuit breaks and to a direction(connid).
-To simplify the whole system, there is no overlap.
-== Trains ==
-Trains always occupy certain track circuits. These are shown red in the signalbox view (TRAIN occupation entry).
-== Database storage ==
-The things that are actually saved are the Track Circuit Breaks. Each TCB holds a list of the TCBs that are adjacent in each direction.
-TC occupation/state is then saved inside each (TCB,Direction) and held in sync across all TCBs adjacent to this one. If something should not be in sync,
-all entries are merged to perform the most restrictive setup.
-== Traverser function ==
-To determine and update the list of neighboring TCBs, we need a traverser function.
-It will start at one TCB in a specified direction (connid) and use get_adjacent_rail to crawl along the track. When encountering a turnout or a crossing,
-it needs to branch(call itself recursively) to find all required TCBs. Those found TCBs are then saved in a list as tuples (TCB,Dir)
-In the last step, they exchange their neighbors.
-== TC states ==
-A track circuit does not have a state as such, but has more or less a list of "reservations"
-type can be one of these:
-TRAIN See Trains obove
-ROUTE Route set from a signal, but no train has yet passed that signal.
-Not implemented (see note by reversible): OWNED - former ROUTE segments that a train has begun passing (train_id assigned)
- - Space behind a train up to the next signal, when a TC is set as REVERSIBLE
-Certain TCs can be marked as "allow call-on".
-== Route setting: ==
-Routes are set from a signal (the entry signal) to another signal facing the same direction (the exit signal)
-Remember that signals are assigned to a TCB and a connid.
-Whenever this is done, the following track circuits are set "reserved" by the train by saving the entry signal's ID:
-- all TCs on the direct way of the route - set as ROUTE
-Route setting fails whenever any TC that we want to set ROUTE to is already set ROUTE or TRAIN from another signal (except call-on, see below)
-Apart from this, we need to set turnouts
-- Turnouts on the track are set held as ROUTE
-- Turnouts that purpose as flank protection are set held as FLANK (NOTE: left as an idea for later, because it's not clear how to do this properly without an engineer)
-Note: In SimSig, it is possible to set a route into an still occupied section on the victoria line sim. (at the depot exit at seven sisters), although
- there are still segments set ahead of the first train passing, remaining from another route.
- Because our system will be able to remember "requested routes" and set them automatically once ready, this is not necessary here.
-== Call-On/Multiple Trains ==
-It will be necessary to join and split trains using call-on routes. A call-on route may be set when:
-- there are no ROUTE reservations
-- there are TRAIN reservations only inside TCs that have "allow call-on" set
-== TC Properties ==
-Note: Reversible property will not be implemented, assuming everything as non-rev.
-This is sufficient to cover all use cases, and is done this way in reality.
- REVERSIBLE - Whether trains are allowed to reverse while on track circuit
- This property is supposed to be set for station tracks, where there is a signal at each end, and for sidings.
- It should in no case be set for TCs covering turnouts, or for main running lines.
- When a TC is not set as reversible, the OWNED status is cleared from the TC right after the train left it,
- to allow other trains to pass it.
- If it is set reversible, interlocking will keep the OWNED state behind the train up to the next signal, clearing it
- as soon as the train passes another signal or enters a non-reversible section.
-CALL_ON_ALLOWED - Whether this TC being blocked (TRAIN or ROUTE) does not prevent shunt routes being set through this TC
-== More notes ==
-- It may not be possible to switch turnouts when their TC has any state entry
-
== Route releasing (TORR) ==
A train passing through a route happens as follows:
Route set from entry to exit signal
-Train passes entry signal and enters first TC past the signal
--> Route from signal cleared (TCs remain locked)
--> ROUTE status of first TC past signal cleared
+Train passes entry signal and enters first TS past the signal
+-> Route from signal cleared (TSs remain locked)
+-> 'route' status of first TS past signal cleared
+-> 'route_post' (holding the turnout locks) remains set
Train continues along the route.
-Whenever train leaves a TC
+Whenever train leaves a TS
-> Clearing any routes set from this TC outward recursively - see "Reversing problem"
-Whenever train enters a TC
--> Clear route status from the just entered TC
+-> Free turnout locks and clear 'route_post'
+Whenever train enters a TS
+-> Clear route status from the just entered TC (but not route_post)
Note that this prohibits by design that the train clears the route ahead of it.
== Reversing Problem ==
Encountered at the Royston simulation in SimSig. It is solved there by imposing a time limit on the set route. Call-on routes can somehow be set anyway.
@@ -89,6 +37,22 @@ Another case of this:
The / here is a non-interlocked turnout (to a non-frequently used siding). For some reason, there is no exit node there,
so the route is set to the signal at the right end. The train is taking the exit to the siding and frees the TC, without ever
having touched the right TC.
+
+
+== Terminology / Variable Names ==
+
+"tcb" : A TCB table (as in track_circuit_breaks)
+"tcbs" : One side of a tcb (that is tcb == {[1] = tcbs, [2] = tcbs})
+"sigd" : A table of format {p=<position>, s=<side aka connid>} by which a "tcbs" is uniqely identified.
+
+== Section Autorepair & Turnout Cache ==
+
+As fundamental part of reworked route programming mechanism, Track Section objects become weak now. They are created and destroyed on demand.
+ildb.repair_tcb automatically checks all nearby sections for issues and repairs them automatically.
+
+Also the database now holds a cache of the turnouts in the section and their position for all possible driving paths.
+Every time a repair operation takes place, and on every track edit operation, the affected sections need to have their cache updated.
+
]]--
local TRAVERSER_LIMIT = 1000
@@ -111,7 +75,45 @@ advtrains.interlocking.npr_rails = {}
function ildb.load(data)
if not data then return end
if data.tcbs then
- track_circuit_breaks = data.tcbs
+ if data.tcbpts_conversion_applied then
+ track_circuit_breaks = data.tcbs
+ else
+ -- Convert legacy pos_to_string tcbs to new advtrains.encode_pos position strings
+ for pts, tcb in pairs(data.tcbs) do
+ local pos = minetest.string_to_pos(pts)
+ if pos then
+ -- that was a pos_to_string
+ local epos = advtrains.encode_pos(pos)
+ atdebug("ILDB converting TCB position format",pts,"->",epos)
+ track_circuit_breaks[epos] = tcb
+ else
+ -- keep entry, it is already new
+ track_circuit_breaks[pts] = tcb
+ end
+ -- convert the routes.[].locks table keys
+ for t_side,tcbs in pairs(tcb) do
+ if tcbs.routes then
+ for t_rnum,route in pairs(tcbs.routes) do
+ for t_rsnm,rseg in ipairs(route) do
+ local locks_n = {}
+ for lpts,state in pairs(rseg.locks) do
+ local lpos = minetest.string_to_pos(lpts)
+ if lpos then
+ local epos = advtrains.encode_pos(lpos)
+ atdebug("ILDB converting tcb",pts,"side",t_side,"route",t_rnum,"lock position format",lpts,"->",epos)
+ locks_n[epos] = state
+ else
+ -- already correct format
+ locks_n[lpts] = state
+ end
+ end
+ rseg.locks = locks_n
+ end
+ end
+ end
+ end
+ end
+ end
end
if data.ts then
track_sections = data.ts
@@ -120,7 +122,23 @@ function ildb.load(data)
signal_assignments = data.signalass
end
if data.rs_locks then
- advtrains.interlocking.route.rte_locks = data.rs_locks
+ if data.tcbpts_conversion_applied then
+ advtrains.interlocking.route.rte_locks = data.rs_locks
+ else
+ advtrains.interlocking.route.rte_locks = {}
+ for pts, lta in pairs(data.rs_locks) do
+ local pos = minetest.string_to_pos(pts)
+ if pos then
+ -- that was a pos_to_string
+ local epos = advtrains.encode_pos(pos)
+ atdebug("ILDB converting Route Lock position format",pts,"->",epos)
+ advtrains.interlocking.route.rte_locks[epos] = lta
+ else
+ -- keep entry, it is already new
+ advtrains.interlocking.route.rte_locks[pts] = lta
+ end
+ end
+ end
end
if data.rs_callbacks then
advtrains.interlocking.route.rte_callbacks = data.rs_callbacks
@@ -132,6 +150,9 @@ function ildb.load(data)
advtrains.interlocking.npr_rails = data.npr_rails
end
+ -- let signal_api load data
+ advtrains.interlocking.signal.load(data)
+
--COMPATIBILITY to Signal aspect format
-- TODO remove in time...
for pts,tcb in pairs(track_circuit_breaks) do
@@ -165,7 +186,7 @@ function ildb.load(data)
end
function ildb.save()
- return {
+ local data = {
tcbs = track_circuit_breaks,
ts=track_sections,
signalass = signal_assignments,
@@ -173,7 +194,10 @@ function ildb.save()
rs_callbacks = advtrains.interlocking.route.rte_callbacks,
influence_points = influence_points,
npr_rails = advtrains.interlocking.npr_rails,
+ tcbpts_conversion_applied = true, -- remark that legacy pos conversion has taken place
}
+ advtrains.interlocking.signal.save(data)
+ return data
end
--
@@ -183,6 +207,9 @@ TCB data structure
-- This is the "A" side of the TCB
[1] = { -- Variant: with adjacent TCs.
ts_id = <id> -- ID of the assigned track section
+ xlink = <other sigd> -- If two sections of track are not physically joined but must function as one TS (e.g. knights move crossing), a bidirectional link can be added with exactly one other TCB.
+ -- TS search will behave as if these two TCBs were physically connected.
+
signal = <pos> -- optional: when set, routes can be set from this tcb/direction and signal
-- aspect will be set accordingly.
routeset = <index in routes> -- Route set from this signal. This is the entry that is cleared once
@@ -195,20 +222,63 @@ TCB data structure
signal_name = <string> -- The human-readable name of the signal, only for documenting purposes
routes = { <route definition> } -- a collection of routes from this signal
route_auto = <boolean> -- When set, we will automatically re-set the route (designated by routeset)
+
+ auto_block_signal_mode = <boolean> -- Simplified mode for simple block signals:
+ -- Signal has only one route which is constantly re-set (route_auto is implied)
+ -- Supposed to be used when only a single track section is ahead and it directly ends at the next signal
+ -- UI only offers to enable or disable the signal
+ -- ARS is implicitly disabled on the signal
},
-- This is the "B" side of the TCB
[2] = { -- Variant: end of track-circuited area (initial state of TC)
ts_id = nil, -- this is the indication for end_of_interlocking
- section_free = <boolean>, --this can be set by an exit node via mesecons or atlatc,
- -- or from the tc formspec.
}
}
+Route definition
+routes = {
+ [i] = {
+ -- one table for each track section on the route
+ -- Note that the section ID is implicitly inferred from the TCB
+ 1 = {
+ locks = { -- component locks for this section of the route.
+ 800080008000 = st
+ }
+ next = S[(-23,9,0)/2] -- the start TCB of the next route segment (pointing forward)
+ -- Signal info: relates to the signal at the start of this section:
+ main_aspect = "_free" -- The main aspect that the route start signal is to show
+ assign_dst = false -- Whether to assign distant signal (affects only the signal at the start of the route)
+ -- false: start signal does not set distant signal (the default), for long blocks
+ -- it is assumed that the next main signal will have its own distant sig
+ -- true: start signal sets distant signal to the next signal on the route with route_role "main" (typically the end signal)
+ -- for short blocks where end signal doesn't have its own distant sig
+ call_on = false -- if true, when this route is set, section is allowed to be occupied by a train (but it must not have a route set in)
+ }
+ 2 = {
+ locks = {}
+ -- if next is omitted, then there is no final TCB (e.g. a buffer)
+ }
+ name = "<the route name>"
+ ars = { <ARS rule definition table> }
+ use_rscache = false -- if true, the track section's rs_cache will be used to set locks in addition to the locks table
+ -- this is disabled for legacy routes, but enabled for all new routes by default
+ default_autoworking = false -- if true, when route is set autoworking will be by default on. Used for Blocksignal mode
+ }
+}
+
Track section
[id] = {
name = "Some human-readable name"
tc_breaks = { <signal specifier>,... } -- Bounding TC's (signal specifiers)
- -- Can be direct ends (auto-detected), conflicting routes or TCBs that are too far away from each other
+ rs_cache = { [startTcbPosEnc] = { [endTcbPosEnc] = { [componentPosEnc] = "state" } } }
+ -- Saves the turnout states that need to be locked when a route is set from tcb#x to tcb#y
+ -- e.g. "800080008005" = { "800080007EEA" = { "800080008000" = "st" } }
+ -- start TCB end TCB switch pos
+ -- Recalculated on every change via update_rs_cache
+ -- Note that the tcb side number is not saved because it is unnecessary
+ fixed_locks = { "800080008000" = "st" }
+ -- table of fixed locks to be set for all routes thru this section (e.g. level crossings
+
route = {
origin = <signal>, -- route origin
entry = <sigd>, -- supposed train entry point
@@ -224,7 +294,9 @@ Track section
-- first says whether to clear the routesetting status from the origin signal.
-- locks contains the positions where locks are held by this ts.
-- 'route' is cleared when train enters the section, while 'route_post' cleared when train leaves section.
+
trains = {<id>, ...} -- Set whenever a train (or more) reside in this TC
+ -- Note: The same train ID may be contained in this mapping multiple times, when it has entered the section in two different places.
}
@@ -237,24 +309,13 @@ signal_assignments = {
}
]]
+-- Maximum scan length for track iterator
+local TS_MAX_SCAN = 1000
---
-function ildb.create_tcb(pos)
- local new_tcb = {
- [1] = {},
- [2] = {},
- }
- local pts = advtrains.roundfloorpts(pos)
- if not track_circuit_breaks[pts] then
- track_circuit_breaks[pts] = new_tcb
- return true
- else
- return false
- end
-end
+-- basic functions
function ildb.get_tcb(pos)
- local pts = advtrains.roundfloorpts(pos)
+ local pts = advtrains.encode_pos(pos)
return track_circuit_breaks[pts]
end
@@ -264,227 +325,635 @@ function ildb.get_tcbs(sigd)
return tcb[sigd.s]
end
-
-function ildb.create_ts(sigd)
- local tcbs = ildb.get_tcbs(sigd)
- local id = advtrains.random_id()
-
- while track_sections[id] do
- id = advtrains.random_id()
- end
-
- track_sections[id] = {
- name = tostring(id),
- tc_breaks = { sigd }
- }
- tcbs.ts_id = id
-end
-
function ildb.get_ts(id)
return track_sections[id]
end
+-- retrieve full tables. Please use only read-only!
+function ildb.get_all_tcb()
+ return track_circuit_breaks
+end
+function ildb.get_all_ts()
+ return track_sections
+end
-
--- various helper functions handling sigd's
-local sigd_equal = advtrains.interlocking.sigd_equal
-local function insert_sigd_nodouble(list, sigd)
- for idx, cmp in pairs(list) do
- if sigd_equal(sigd, cmp) then
- return
- end
+function tsrepair_notify(notify_pname, ...)
+ if notify_pname then
+ minetest.chat_send_player(notify_pname, advtrains.print_concat_table({"TS Check:",...}))
end
- table.insert(list, sigd)
end
-
--- This function will actually handle the node that is in connid direction from the node at pos
--- so, this needs the conns of the node at pos, since these are already calculated
-local function traverser(found_tcbs, pos, conns, connid, count, brk_when_found_n)
- local adj_pos, adj_connid, conn_idx, nextrail_y, next_conns = advtrains.get_adjacent_rail(pos, conns, connid, advtrains.all_tracktypes)
- if not adj_pos then
- --atdebug("Traverser found end-of-track at",pos, connid)
- return
+-- Checks the consistency of the track section at the given position, attempts to autorepair track sections if they are inconsistent
+-- There are 2 operation modes:
+-- 1: pos is NOT a TCB, tcb_connid MUST be nil
+-- 2: pos is a TCB, tcb_connid MUST be given
+-- @param pos: the position to start from
+-- @param tcb_connid: If provided node is a TCB, the direction in which to search
+-- @param notify_pname: the player to notify about reparations
+-- Returns:
+-- ts_id - the track section that was found
+-- nil - No track section exists
+function ildb.check_and_repair_ts_at_pos(pos, tcb_connid, notify_pname, force_create)
+ --atdebug("check_and_repair_ts_at_pos", pos, tcb_connid)
+ -- check prereqs
+ if ildb.get_tcb(pos) then
+ if not tcb_connid then error("check_and_repair_ts_at_pos: Startpoint is TCB, must provide tcb_connid!") end
+ else
+ -- FIX 2025-01-07: If the checked pos is not a TCB, we always need to search in both directions, otherwise we errorneously split a ts
+ tcb_connid = nil
end
- -- look whether there is a TCB here
- if #next_conns == 2 then --if not, don't even try!
- local tcb = ildb.get_tcb(adj_pos)
- if tcb then
- -- done with this branch
- --atdebug("Traverser found tcb at",adj_pos, adj_connid)
- insert_sigd_nodouble(found_tcbs, {p=adj_pos, s=adj_connid})
- return
+ -- STEP 1: Ensure that only one section is at this place
+ -- get all TCBs adjacent to this
+ local all_tcbs = ildb.get_all_tcbs_adjacent(pos, tcb_connid)
+ local first_ts = true
+ local ts_id
+ for _,sigd in ipairs(all_tcbs) do
+ ildb.tcbs_ensure_ts_ref_exists(sigd)
+ local tcbs_ts_id = sigd.tcbs.ts_id
+ if first_ts then
+ -- this one determines
+ ts_id = tcbs_ts_id
+ first_ts = false
+ else
+ -- these must be the same as the first
+ if ts_id ~= tcbs_ts_id then
+ -- inconsistency is found, repair it
+ --atdebug("check_and_repair_ts_at_pos: Inconsistency is found!")
+ tsrepair_notify(notify_pname, "Track section inconsistent here, repairing...")
+ return ildb.repair_ts_merge_all(all_tcbs, force_create, notify_pname)
+ -- Step2 check is no longer necessary since we just created that new section
+ end
end
end
- -- recursion abort condition
- if count > TRAVERSER_LIMIT then
- --atdebug("Traverser hit counter at",adj_pos, adj_connid)
- return true
- end
- -- continue traversing
- local counter_hit = false
- for nconnid, nconn in ipairs(next_conns) do
- if adj_connid ~= nconnid then
- counter_hit = counter_hit or traverser(found_tcbs, adj_pos, next_conns, nconnid, count + 1, brk_when_found_n)
- if brk_when_found_n and #found_tcbs>=brk_when_found_n then
- break
- end
+ -- only one found (it is either nil or a ts id)
+ --atdebug("check_and_repair_ts_at_pos: TS consistent id=",ts_id,"")
+ if not ts_id then
+ if force_create and next(all_tcbs) then --ensure at least one tcb is in list
+ return ildb.repair_ts_merge_all(all_tcbs, force_create, notify_pname)
+ else
+ --tsrepair_notify(notify_pname, "No track section found here.")
+ return nil
+ -- All TCBs agreed that there is no section here
end
end
- return counter_hit
+
+ local ts = ildb.get_ts(ts_id)
+ if not ts then
+ -- This branch may never be reached, because ildb.tcbs_ensure_ts_ref_exists(sigd) is already supposed to clear out missing sections
+ error("check_and_repair_ts_at_pos: Resolved to nonexisting section although ildb.tcbs_ensure_ts_ref_exists(sigd) was supposed to prevent this. Panic!")
+ end
+ ildb.purge_ts_tcb_refs(ts_id)
+ -- STEP 2: Ensure that all_tcbs is equal to the track section's TCB list. If there are extra TCBs then the section should be split
+ -- ildb.tcbs_ensure_ts_ref_exists(sigd) has already make sure that all tcbs are found in the ts's tc_breaks list
+ -- That means it is sufficient to compare the LENGTHS of both lists, if one is longer then it is inconsistent
+ if #ts.tc_breaks ~= #all_tcbs then
+ --atdebug("check_and_repair_ts_at_pos: Partition is found!")
+ tsrepair_notify(notify_pname, "Track section partition found, repairing...")
+ return ildb.repair_ts_merge_all(all_tcbs, force_create, notify_pname)
+ end
+ --tsrepair_notify(notify_pname, "Found section", ts.name or ts_id, "here.")
+ ildb.update_rs_cache(ts_id)
+ return ts_id
end
+-- Helper function to prevent duplicates
+local function insert_sigd_if_not_present(tab, sigd)
+ local found = false
+ for _, ssigd in ipairs(tab) do
+ if vector.equals(sigd.p, ssigd.p) and sigd.s==ssigd.s then
+ found = true
+ end
+ end
+ if not found then
+ table.insert(tab, sigd)
+ end
+ return not found
+end
-
--- Merges the TS with merge_id into root_id and then deletes merge_id
-local function merge_ts(root_id, merge_id)
- local rts = ildb.get_ts(root_id)
- local mts = ildb.get_ts(merge_id)
- if not mts then return end -- This may be the case when sync_tcb_neighbors
- -- inserts the same id twice. do nothing.
-
- if not ildb.may_modify_ts(rts) then return false end
- if not ildb.may_modify_ts(mts) then return false end
-
- -- cobble together the list of TCBs
- for _, msigd in ipairs(mts.tc_breaks) do
- local tcbs = ildb.get_tcbs(msigd)
- if tcbs then
- insert_sigd_nodouble(rts.tc_breaks, msigd)
- tcbs.ts_id = root_id
+-- Starting from a position, search all TCBs that can be reached from this position.
+-- In a non-faulty setup, all of these should have the same track section assigned.
+-- This function does not trigger a repair.
+-- @param inipos: the initial position
+-- @param inidir: the initial direction, or nil to search in all directions
+-- @param per_track_callback: A callback function called with signature (pos, connid, bconnid) for every track encountered
+-- Returns: a list of sigd's describing the TCBs found (sigd's point inward):
+-- {p=<pos>, s=<side>, tcbs=<ref to tcbs table>}
+function ildb.get_all_tcbs_adjacent(inipos, inidir, per_track_callback)
+ --atdebug("get_all_tcbs_adjacent: inipos",inipos,"inidir",inidir,"")
+ local found_sigd = {}
+ local ti = advtrains.get_track_iterator(inipos, inidir, TS_MAX_SCAN, true)
+ -- if initial start is TCBS and has xlink, need to add that to the TI
+ local inisi = {p=inipos, s=inidir};
+ local initcbs = ildb.get_tcbs(inisi)
+ if initcbs then
+ ildb.validate_tcb_xlink(inisi, true)
+ if initcbs.xlink then
+ -- adding the tcb will happen when this branch is retrieved again using ti:next_branch()
+ --atdebug("get_all_tcbs_adjacent: Putting xlink Branch for initial node",initcbs.xlink)
+ ti:add_branch(initcbs.xlink.p, initcbs.xlink.s)
end
- advtrains.interlocking.show_tcb_marker(msigd.p)
end
- -- done
- track_sections[merge_id] = nil
+ local pos, connid, bconnid, tcb
+ while ti:has_next_branch() do
+ pos, connid = ti:next_branch()
+ --atdebug("get_all_tcbs_adjacent: BRANCH: ",pos, connid)
+ bconnid = nil
+ local is_branch_start = true
+ repeat
+ -- callback
+ if per_track_callback then
+ per_track_callback(pos, connid, bconnid)
+ end
+ tcb = ildb.get_tcb(pos)
+ if tcb then
+ local using_connid = bconnid
+ -- found a tcb
+ if is_branch_start then
+ -- A branch cannot be a TCB, as that would imply that it was a turnout/crossing (illegal)
+ -- UNLESS: (a) it is the start point or (b) it was added via xlink
+ -- Then the correct conn to use is connid (pointing forward)
+ --atdebug("get_all_tcbs_adjacent: Inserting TCB at branch start",pos, connid)
+ using_connid = connid
+ end
+ -- add the sigd of this tcb and a reference to the tcb table in it
+ --atdebug("get_all_tcbs_adjacent: Found TCB: ",pos, using_connid, "ts=", tcb[using_connid].ts_id)
+ local si = {p=pos, s=using_connid, tcbs=tcb[using_connid]}
+ -- if xlink exists, add it now (only if we are not in branch start)
+ ildb.validate_tcb_xlink(si, true)
+ if not is_branch_start and si.tcbs.xlink then
+ -- adding the tcb will happen when this branch is retrieved again using ti:next_branch()
+ --atdebug("get_all_tcbs_adjacent: Putting xlink Branch",si.tcbs.xlink)
+ ti:add_branch(si.tcbs.xlink.p, si.tcbs.xlink.s)
+ end
+ insert_sigd_if_not_present(found_sigd, si)
+ if not is_branch_start then
+ break
+ end
+ end
+ pos, connid, bconnid = ti:next_track()
+ is_branch_start = false
+ --atdebug("get_all_tcbs_adjacent: TRACK: ",pos, connid, bconnid)
+ until not pos
+ end
+ return found_sigd
end
-local lntrans = { "A", "B" }
-local function sigd_to_string(sigd)
- return minetest.pos_to_string(sigd.p).." / "..lntrans[sigd.s]
+-- Called by frontend functions when multiple tcbs's that logically belong to one section have been determined to have different sections
+-- Parameter is the output of ildb.get_all_tcbs_adjacent(pos)
+-- Returns the ID of the track section that results after the merge
+function ildb.repair_ts_merge_all(all_tcbs, force_create, notify_pname)
+ --atdebug("repair_ts_merge_all: Instructed to merge sections of following TCBs:")
+ -- The first loop does the following for each TCBS:
+ -- a) Store the TS ID in the set of TS to update
+ -- b) Set the TS ID to nil, so that the TCBS gets removed from the section
+ local ts_to_update = {}
+ local ts_name_repo = {}
+ local any_ts = false
+ for _,sigd in ipairs(all_tcbs) do
+ local ts_id = sigd.tcbs.ts_id
+ --atdebug(sigd, "ts=", ts_id)
+ if ts_id then
+ local ts = track_sections[ts_id]
+ if ts then
+ any_ts = true
+ ts_to_update[ts_id] = true
+ -- if nonstandard name, store this
+ if ts.name and not string.match(ts.name, "^Section") then
+ ts_name_repo[#ts_name_repo+1] = ts.name
+ end
+ end
+ end
+ sigd.tcbs.ts_id = nil
+ end
+ if not any_ts and not force_create then
+ -- nothing to do at all, just no interlocking. Why were we even called
+ --atdebug("repair_ts_merge_all: No track section present, will not create a new one")
+ return nil
+ end
+ -- Purge every TS in turn. TS's that are now empty will be deleted. TS's that still have TCBs will be kept
+ for ts_id, _ in pairs(ts_to_update) do
+ local remain_ts = ildb.purge_ts_tcb_refs(ts_id)
+ end
+ -- Create a new fresh track section with all the TCBs we have in our collection
+ local new_ts_id, new_ts = ildb.create_ts_from_tcb_list(all_tcbs)
+ tsrepair_notify(notify_pname, "Created track section",new_ts_id,"from",#all_tcbs,"TCBs")
+ return new_ts_id
end
--- Check for near TCBs and connect to their TS if they have one, and syncs their data.
-function ildb.sync_tcb_neighbors(pos, connid)
- local found_tcbs = { {p = pos, s = connid} }
- local node_ok, conns, rhe = advtrains.get_rail_info_at(pos, advtrains.all_tracktypes)
- if not node_ok then
- atwarn("update_tcb_neighbors but node is NOK: "..minetest.pos_to_string(pos))
- return
+-- For the specified TS, go through the list of TCBs and purge all TCBs that have no corresponding backreference in their TCBS table.
+-- If the track section ends up empty, it is deleted in this process.
+-- Should the track section still exist after the purge operation, it is returned.
+function ildb.purge_ts_tcb_refs(ts_id)
+ local ts = track_sections[ts_id]
+ if not ts then
+ return nil
end
-
- --atdebug("Traversing from ",pos, connid)
- local counter_hit = traverser(found_tcbs, pos, conns, connid, 0)
-
- local ts_id
- local list_eoi = {}
- local list_ok = {}
- local list_mismatch = {}
- local ts_to_merge = {}
-
- for idx, sigd in pairs(found_tcbs) do
+ local has_changed = false
+ local i = 1
+ while ts.tc_breaks[i] do
+ -- get TCBS
+ local sigd = ts.tc_breaks[i]
local tcbs = ildb.get_tcbs(sigd)
- if not tcbs.ts_id then
- --atdebug("Sync: put",sigd_to_string(sigd),"into list_eoi")
- table.insert(list_eoi, sigd)
- elseif not ts_id and tcbs.ts_id then
- if not ildb.get_ts(tcbs.ts_id) then
- atwarn("Track section database is inconsistent, there's no TS with ID=",tcbs.ts_id)
- tcbs.ts_id = nil
- table.insert(list_eoi, sigd)
+ if tcbs then
+ if tcbs.ts_id == ts_id then
+ -- this one is legit
+ i = i+1
else
- --atdebug("Sync: put",sigd_to_string(sigd),"into list_ok")
- ts_id = tcbs.ts_id
- table.insert(list_ok, sigd)
+ -- this one is to be purged
+ --atdebug("purge_ts_tcb_refs(",ts_id,"): purging",sigd,"(backreference = ",tcbs.ts_id,")")
+ table.remove(ts.tc_breaks, i)
+ has_changed = true
end
- elseif ts_id and tcbs.ts_id and tcbs.ts_id ~= ts_id then
- atwarn("Track section database is inconsistent, sections share track!")
- atwarn("Merging",tcbs.ts_id,"into",ts_id,".")
- table.insert(list_mismatch, sigd)
- table.insert(ts_to_merge, tcbs.ts_id)
+ else
+ -- if not tcbs: was anyway an orphan, remove it
+ --atdebug("purge_ts_tcb_refs(",ts_id,"): purging",sigd,"(referred nonexisting TCB)")
+ table.remove(ts.tc_breaks, i)
+ has_changed = true
end
end
- if ts_id then
- local ts = ildb.get_ts(ts_id)
- for _, sigd in ipairs(list_eoi) do
- local tcbs = ildb.get_tcbs(sigd)
- tcbs.ts_id = ts_id
- table.insert(ts.tc_breaks, sigd)
- advtrains.interlocking.show_tcb_marker(sigd.p)
- end
- for _, mts in ipairs(ts_to_merge) do
- merge_ts(ts_id, mts)
+ if #ts.tc_breaks == 0 then
+ -- remove the section completely
+ --atdebug("purge_ts_tcb_refs(",ts_id,"): after purging, the section is empty, is being deleted")
+ track_sections[ts_id] = nil
+ return nil
+ else
+ if has_changed then
+ -- needs to update route cache
+ ildb.update_rs_cache(ts_id)
end
+ return ts
end
end
-function ildb.link_track_sections(merge_id, root_id)
- if merge_id == root_id then
+-- For the specified TCBS, make sure that the track section referenced by it
+-- (a) exists and
+-- (b) has a backreference to this TCBS stored in its tc_breaks list
+function ildb.tcbs_ensure_ts_ref_exists(sigd)
+ local tcbs = sigd.tcbs or ildb.get_tcbs(sigd)
+ if not tcbs or not tcbs.ts_id then return end
+ local ts = ildb.get_ts(tcbs.ts_id)
+ if not ts then
+ --atdebug("tcbs_ensure_ts_ref_exists(",sigd,"): TS does not exist, setting to nil")
+ -- TS is deleted, clear own ts id
+ tcbs.ts_id = nil
return
end
- merge_ts(root_id, merge_id)
+ local did_insert = insert_sigd_if_not_present(ts.tc_breaks, {p=sigd.p, s=sigd.s})
+ if did_insert then
+ --atdebug("tcbs_ensure_ts_ref_exists(",sigd,"): TCBS was missing reference in TS",tcbs.ts_id)
+ ildb.update_rs_cache(tcbs.ts_id)
+ end
end
-function ildb.remove_from_interlocking(sigd)
- local tcbs = ildb.get_tcbs(sigd)
- if not ildb.may_modify_tcbs(tcbs) then return false end
+function ildb.create_ts_from_tcb_list(sigd_list)
+ local id = advtrains.random_id(8)
- if tcbs.ts_id then
- local tsid = tcbs.ts_id
- local ts = ildb.get_ts(tsid)
- if not ts then
- tcbs.ts_id = nil
- return true
+ while track_sections[id] do
+ id = advtrains.random_id(8)
+ end
+ --atdebug("create_ts_from_tcb_list: sigd_list=",sigd_list, "new ID will be ",id)
+
+ local tcbr = {}
+ -- makes a copy of the sigd list, for use in repair mechanisms where sigd may contain a tcbs field which we dont want
+ for _, sigd in ipairs(sigd_list) do
+ table.insert(tcbr, {p=sigd.p, s=sigd.s})
+ local tcbs = sigd.tcbs or ildb.get_tcbs(sigd)
+ if tcbs.ts_id then
+ error("Trying to create TS with TCBS that is already assigned to other section")
end
-
- -- remove entry from the list
- local idx = 1
- while idx <= #ts.tc_breaks do
- local cmp = ts.tc_breaks[idx]
- if sigd_equal(sigd, cmp) then
- table.remove(ts.tc_breaks, idx)
+ tcbs.ts_id = id
+ end
+
+ local new_ts = {
+ tc_breaks = tcbr
+ }
+ track_sections[id] = new_ts
+ -- update the TCB markers
+ for _, sigd in ipairs(sigd_list) do
+ advtrains.interlocking.show_tcb_marker(sigd.p)
+ end
+
+
+ ildb.update_rs_cache(id)
+ return id, new_ts
+end
+
+-- RS CACHE --
+
+--[[
+node_from_to_list - cache of from-to connid mappings and their associated state.
+Acts like a cache, built on demand by ildb.get_possible_out_connids(nodename)
+node_name = {
+ from_connid = {
+ to_connid = state
+ }
+}
+]]
+local node_from_to_state_cache = {}
+
+function ildb.get_possible_out_connids(node_name, in_connid)
+ if not node_from_to_state_cache[node_name] then
+ node_from_to_state_cache[node_name] = {}
+ end
+ local nt = node_from_to_state_cache[node_name]
+ if not nt[in_connid] then
+ local ta = {}
+ --atdebug("Node From/To State Cache: Caching for ",node_name,"connid",in_connid)
+ local ndef = minetest.registered_nodes[node_name]
+ if ndef.advtrains.node_state_map then
+ for state, tnode in pairs(ndef.advtrains.node_state_map) do
+ local tndef = minetest.registered_nodes[tnode]
+ -- verify that the conns table is identical - this is purely to catch setup errors!
+ if not tndef.at_conns or not tndef.at_conn_map then
+ --atdebug("ndef:",ndef,", tndef:",tndef)
+ error("In AT setup for node "..tnode..": Node set as state "..state.." of "..node_name.." in state_map, but is missing at_conns/at_conn_map!")
+ end
+ if #ndef.at_conns ~= #tndef.at_conns then
+ --atdebug("ndef:",ndef,", tndef:",tndef)
+ error("In AT setup for node "..tnode..": Conns table does not match that of "..node_name.." (of which this is state "..state..")")
+ end
+ for connid=1,#ndef.at_conns do
+ if ndef.at_conns[connid].c ~= tndef.at_conns[connid].c then
+ --atdebug("ndef:",ndef,", tndef:",tndef)
+ error("In AT setup for node "..tnode..": Conns table does not match that of "..node_name.." (of which this is state "..state..")")
+ end
+ end
+ -- do the actual caching by looking at the conn_map
+ local target_connid = tndef.at_conn_map[in_connid]
+ if ta[target_connid] then
+ -- Select the variant for which the other way would back-connect. This way, turnouts will switch to the appropriate branch if the train joins
+ local have_back_conn = (tndef.at_conn_map[target_connid])==in_connid
+ --atdebug("Found second state mapping",in_connid,"-",target_connid,"have_back_conn=",have_back_conn)
+ if have_back_conn then
+ --atdebug("Overwriting",in_connid,"-",target_connid,"=",state)
+ ta[target_connid] = state
+ end
+ else
+ --atdebug("Setting",in_connid,"-",target_connid,"=",state)
+ ta[target_connid] = state
+ end
+ end
+ else
+ error("Node From/To State Cache: "..node_name.." doesn't have a state map, is not a switchable track! Panic!")
+ end
+ nt[in_connid] = ta
+ end
+ return nt[in_connid]
+end
+
+local function recursively_find_routes(s_pos, s_connid, locks_found, result_table, scan_limit)
+ --atdebug("Recursively restarting at ",s_pos, s_connid, "limit left", scan_limit)
+ local ti = advtrains.get_track_iterator(s_pos, s_connid, scan_limit, false)
+ local pos, connid, bconnid = ti:next_branch()
+ pos, connid, bconnid = ti:next_track()-- step once to get ahead of previous turnout
+ local last_pos
+ while pos do -- this stops the loop when either the track end is reached or the limit is hit
+ local node = advtrains.ndb.get_node_or_nil(pos)
+ --atdebug("Walk ",pos, "nodename", node.name, "entering at conn",bconnid)
+ local ndef = minetest.registered_nodes[node.name]
+ if ndef.advtrains and ndef.advtrains.node_state_map then
+ -- Stop, this is a switchable node. Find out which conns we can go at
+ --atdebug("Found turnout ",pos, "nodename", node.name, "entering at conn",bconnid)
+ local pts = advtrains.encode_pos(pos)
+ if locks_found[pts] then
+ -- we've been here before. Stop
+ --atdebug("Was already seen! returning")
+ return
+ end
+ local out_conns = ildb.get_possible_out_connids(node.name, bconnid)
+ -- note 2025-01-06: get_possible_out_connids contains some logic so that the correct switch state is selected even if the turnout is entered from the branch side (see "have_back_conn")
+ for oconnid, state in pairs(out_conns) do
+ --atdebug("Going in direction",oconnid,"state",state)
+ locks_found[pts] = state
+ recursively_find_routes(pos, oconnid, locks_found, result_table, ti.limit)
+ locks_found[pts] = nil
+ end
+ return
+ end
+ --otherwise, this might be a tcb
+ local tcb = ildb.get_tcb(pos)
+ if tcb then
+ -- we found a tcb, store the current locks in the result_table
+ local end_pkey = advtrains.encode_pos(pos)
+ --atdebug("Found end TCB", pos, end_pkey,", returning")
+ if result_table[end_pkey] then
+ atwarn("While caching track section routing, found multiple route paths within same track section. Only first one found will be used")
else
- idx = idx + 1
+ result_table[end_pkey] = table.copy(locks_found)
end
+ return
end
- tcbs.ts_id = nil
-
- --ildb.sync_tcb_neighbors(sigd.p, sigd.s)
+ -- Go forward
+ last_pos = pos
+ pos, connid, bconnid = ti:next_track()
+ end
+ --atdebug("recursively_find_routes: Reached track end or limit at", last_pos, ". This path is not saved, returning")
+end
+
+-- Updates the turnout cache of the given track section
+function ildb.update_rs_cache(ts_id)
+ local ts = ildb.get_ts(ts_id)
+ if not ts then
+ error("Update TS Cache called with nonexisting ts_id "..(ts_id or "nil"))
+ end
+ local rscache = {}
+ --atdebug("== Running update_rs_cache for ",ts_id)
+ -- start on every of the TS's TCBs, walk the track forward and store locks along the way
+ for start_tcbi, start_tcbs in ipairs(ts.tc_breaks) do
+ start_pkey = advtrains.encode_pos(start_tcbs.p)
+ rscache[start_pkey] = {}
+ --atdebug("Starting for ",start_tcbi, start_tcbs)
+ local locks_found = {}
+ local result_table = {}
- if #ts.tc_breaks == 0 then
- track_sections[tsid] = nil
+ recursively_find_routes(start_tcbs.p, start_tcbs.s, locks_found, result_table, TS_MAX_SCAN)
+ -- now result_table contains found route locks. Match them with the other TCBs we have in this section
+ for end_tcbi, end_tcbs in ipairs(ts.tc_breaks) do
+ if end_tcbi ~= start_tcbi then
+ end_pkey = advtrains.encode_pos(end_tcbs.p)
+ if result_table[end_pkey] then
+ --atdebug("Set RSCache entry",end_pkey.."-"..end_pkey,"=",result_table[end_pkey])
+ local lockstab = result_table[end_pkey]
+ if ts.fixed_locks then -- include the sections fixedlocks if present
+ for pts, st in pairs(ts.fixed_locks) do lockstab[pts] = st end
+ end
+ rscache[start_pkey][end_pkey] = lockstab
+ result_table[end_pkey] = nil
+ end
+ end
+ end
+ -- warn about superfluous entry
+ for sup_end_pkey, sup_entry in pairs(result_table) do
+ atwarn("In update_rs_cache for section",ts_id,"found superfluous endpoint",sup_end_pkey,"->",sup_entry)
end
end
- advtrains.interlocking.show_tcb_marker(sigd.p)
- if tcbs.signal then
- return false
- end
- return true
+ ts.rs_cache = rscache
+ --atdebug("== Done update_rs_cache for ",ts_id, "result:",rscache)
end
-function ildb.remove_tcb(pos)
- local pts = advtrains.roundfloorpts(pos)
- if not track_circuit_breaks[pts] then
- return true --FIX: not an error, because tcb is already removed
- end
+
+--- DB API functions
+
+local lntrans = { "A", "B" }
+function ildb.sigd_to_string(sigd)
+ return minetest.pos_to_string(sigd.p).." / "..lntrans[sigd.s]
+end
+
+-- Create a new TCB at the position and update/repair the adjoining sections
+function ildb.create_tcb_at(pos)
+ --atdebug("create_tcb_at",pos)
+ local pts = advtrains.encode_pos(pos)
+ track_circuit_breaks[pts] = {[1] = {}, [2] = {}}
+ local all_tcbs_1 = ildb.get_all_tcbs_adjacent(pos, 1)
+ --atdebug("TCBs on A side",all_tcbs_1)
+ local all_tcbs_2 = ildb.get_all_tcbs_adjacent(pos, 2)
+ --atdebug("TCBs on B side",all_tcbs_2)
+ -- perform TS repair
+ ildb.repair_ts_merge_all(all_tcbs_1, false)
+ ildb.repair_ts_merge_all(all_tcbs_2, false)
+end
+
+-- Remove TCB at the position and update/repair the now joined section
+-- skip_tsrepair: should be set to true when the rail node at the TCB position is already gone.
+-- Assumes the track sections are now separated and does not attempt the repair process.
+function ildb.remove_tcb_at(pos, pname, skip_tsrepair)
+ --atdebug("remove_tcb_at",pos)
+ local pts = advtrains.encode_pos(pos)
+ local old_tcb = track_circuit_breaks[pts]
+ -- unassign signals if defined
for connid=1,2 do
- if not ildb.remove_from_interlocking({p=pos, s=connid}) then
- return false
+ if old_tcb[connid].signal then
+ ildb.set_sigd_for_signal(old_tcb[connid].signal, nil)
end
end
track_circuit_breaks[pts] = nil
+ -- purge the track sections adjacent
+ if old_tcb[1].ts_id then
+ ildb.purge_ts_tcb_refs(old_tcb[1].ts_id)
+ end
+ if old_tcb[2].ts_id then
+ ildb.purge_ts_tcb_refs(old_tcb[2].ts_id)
+ end
+ -- update xlink partners
+ if old_tcb[1].xlink then
+ ildb.validate_tcb_xlink(old_tcb[1].xlink)
+ end
+ if old_tcb[2].xlink then
+ ildb.validate_tcb_xlink(old_tcb[2].xlink)
+ end
+ advtrains.interlocking.remove_tcb_marker(pos)
+ -- If needed, merge the track sections here
+ if not skip_tsrepair then
+ ildb.check_and_repair_ts_at_pos(pos, nil, pname)
+ end
return true
end
-function ildb.dissolve_ts(ts_id)
- local ts = ildb.get_ts(ts_id)
- if not ildb.may_modify_ts(ts) then return false end
- local tcbr = advtrains.merge_tables(ts.tc_breaks)
- for _,sigd in ipairs(tcbr) do
- ildb.remove_from_interlocking(sigd)
+-- Xlink: Connecting not-physically-connected sections handling
+
+-- Ensures that the xlink of this tcbs is bidirectional
+function ildb.validate_tcb_xlink(sigd, suppress_repairs)
+ local tcbs = sigd.tcbs or ildb.get_tcbs(sigd)
+ local osigd = tcbs.xlink
+ if not osigd then return end
+ local otcbs = ildb.get_tcbs(tcbs.xlink)
+ if not otcbs then
+ --atdebug("validate_tcb_xlink",sigd,": Link partner ",osigd,"orphaned")
+ tcbs.xlink = nil
+ if not suppress_repairs then
+ ildb.check_and_repair_ts_at_pos(sigd.p, sigd.s)
+ end
+ return
end
- -- Note: ts gets removed in the moment of the removal of the last TCB.
- return true
+ if otcbs.xlink then
+ if not vector.equals(otcbs.xlink.p, sigd.p) or otcbs.xlink.s~=sigd.s then
+ --atdebug("validate_tcb_xlink",sigd,": Link partner ",osigd,"backreferencing to someone else (namely",otcbs.xlink,") clearing it")
+ tcbs.xlink = nil
+ if not suppress_repairs then
+ ildb.check_and_repair_ts_at_pos(sigd.p, sigd.s)
+ --atdebug("validate_tcb_xlink",sigd,": Link partner ",osigd," was backreferencing to someone else, now updating that")
+ ildb.validate_tcb_xlink(osigd)
+ end
+ end
+ else
+ --atdebug("validate_tcb_xlink",sigd,": Link partner ",osigd,"wasn't backreferencing, clearing it")
+ tcbs.xlink = nil
+ if not suppress_repairs then
+ ildb.check_and_repair_ts_at_pos(sigd.p, sigd.s)
+ end
+ end
+end
+
+function ildb.add_tcb_xlink(sigd1, sigd2)
+ --("add_tcb_xlink",sigd1, sigd2)
+ local tcbs1 = sigd1.tcbs or ildb.get_tcbs(sigd1)
+ local tcbs2 = sigd2.tcbs or ildb.get_tcbs(sigd2)
+ if vector.equals(sigd1.p, sigd2.p) then
+ --atdebug("add_tcb_xlink Cannot xlink with same TCB")
+ return
+ end
+ if not tcbs1 or not tcbs2 then
+ --atdebug("add_tcb_xlink TCBS doesnt exist")
+ return
+ end
+ if tcbs1.xlink or tcbs2.xlink then
+ --atdebug("add_tcb_xlink One already linked")
+ return
+ end
+ -- apply link
+ tcbs1.xlink = {p=sigd2.p, s=sigd2.s}
+ tcbs2.xlink = {p=sigd1.p, s=sigd1.s}
+ -- update section. It should be sufficient to call update only once because the TCBs are linked anyway now
+ ildb.check_and_repair_ts_at_pos(sigd1.p, sigd1.s)
+end
+
+function ildb.remove_tcb_xlink(sigd)
+ --atdebug("remove_tcb_xlink",sigd)
+ -- Validate first. If Xlink is gone already then, nothing to do
+ ildb.validate_tcb_xlink(sigd)
+ -- Checking all of these already done by validate
+ local tcbs = sigd.tcbs or ildb.get_tcbs(sigd)
+ local osigd = tcbs.xlink
+ if not osigd then
+ -- validate already cleared us
+ --atdebug("remove_tcb_xlink: Already gone by validate")
+ return
+ end
+ local otcbs = ildb.get_tcbs(tcbs.xlink)
+ -- clear it
+ otcbs.xlink = nil
+ tcbs.xlink = nil
+ -- Update section for ourself and the other one
+ ildb.check_and_repair_ts_at_pos(sigd.p, sigd.s)
+ ildb.check_and_repair_ts_at_pos(osigd.p, osigd.s)
+end
+
+function ildb.create_ts_from_tcbs(sigd)
+ --atdebug("create_ts_from_tcbs",sigd)
+ local all_tcbs = ildb.get_all_tcbs_adjacent(sigd.p, sigd.s)
+ ildb.repair_ts_merge_all(all_tcbs, true)
+end
+
+-- Remove the given track section, leaving its TCBs with no section assigned
+function ildb.remove_ts(ts_id)
+ --atdebug("remove_ts",ts_id)
+ local ts = track_sections[ts_id]
+ if not ts then
+ error("remove_ts: "..ts_id.." doesn't exist")
+ end
+ while ts.tc_breaks[i] do
+ -- get TCBS
+ local sigd = ts.tc_breaks[i]
+ local tcbs = ildb.get_tcbs(sigd)
+ if tcbs then
+ --atdebug("cleared TCB",sigd)
+ tcbs.ts_id = nil
+ else
+ --atdebug("orphan TCB",sigd)
+ end
+ i = i+1
+ end
+ track_sections[ts_id] = nil
end
-- Returns true if it is allowed to modify any property of a track section, such as
@@ -509,38 +978,8 @@ function ildb.may_modify_tcbs(tcbs)
return true
end
--- Utilize the traverser to find the track section at the specified position
--- Returns:
--- ts_id, origin - the first found ts and the sigd of the found tcb
--- nil - there were no TCBs in TRAVERSER_MAX range of the position
--- false - the first found TCB stated End-Of-Interlocking, or track ends were reached
-function ildb.get_ts_at_pos(pos)
- local node_ok, conns, rhe = advtrains.get_rail_info_at(pos, advtrains.all_tracktypes)
- if not node_ok then
- error("get_ts_at_pos but node is NOK: "..minetest.pos_to_string(pos))
- end
- local limit_hit = false
- local found_tcbs = {}
- for connid, conn in ipairs(conns) do -- Note: a breadth-first-search would be better for performance
- limit_hit = limit_hit or traverser(found_tcbs, pos, conns, connid, 0, 1)
- if #found_tcbs >= 1 then
- local tcbs = ildb.get_tcbs(found_tcbs[1])
- local ts
- if tcbs.ts_id then
- return tcbs.ts_id, found_tcbs[1]
- else
- return false
- end
- end
- end
- if limit_hit then
- -- there was at least one limit hit
- return nil
- else
- -- all traverser ends were track ends
- return false
- end
-end
+
+-- Signals/IP --
-- returns the sigd the signal at pos belongs to, if this is known
@@ -556,11 +995,34 @@ function ildb.get_sigd_for_signal(pos)
end
return nil
end
-function ildb.set_sigd_for_signal(pos, sigd)
+function ildb.set_sigd_for_signal(pos, sigd) -- do not use!
local pts = advtrains.roundfloorpts(pos)
signal_assignments[pts] = sigd
end
+-- Assign the signal at pos to the given TCB side.
+function ildb.assign_signal_to_tcbs(pos, sigd)
+ local tcbs = ildb.get_tcbs(sigd)
+ assert(tcbs, "assign_signal_to_tcbs invalid sigd!")
+ tcbs.signal = pos
+ if not tcbs.routes then
+ tcbs.routes = {}
+ end
+ ildb.set_sigd_for_signal(pos, sigd)
+end
+
+-- unassign the signal from the given sigd (looks in tcbs.signal for the signalpos)
+function ildb.unassign_signal_for_tcbs(sigd)
+ local tcbs = advtrains.interlocking.db.get_tcbs(sigd)
+ if not tcbs then return end
+ local pos = tcbs.signal
+ if not pos then return end
+ ildb.set_sigd_for_signal(pos, nil)
+ tcbs.signal = nil
+ tcbs.route_aspect = nil
+ tcbs.route_remote = nil
+end
+
-- checks if there's any influence point set to this position
-- if purge is true, checks whether the associated signal still exists
-- and deletes the ip if not.
@@ -570,7 +1032,7 @@ function ildb.is_ip_at(pos, purge)
if purge then
-- is there still a signal assigned to it?
for connid, sigpos in pairs(influence_points[pts]) do
- local asp = advtrains.interlocking.signal_get_aspect(sigpos)
+ local asp = advtrains.interlocking.signal.get_aspect_info(sigpos)
if not asp then
atlog("Clearing orphaned signal influence point", pts, "/", connid)
ildb.clear_ip_signal(pts, connid)
@@ -597,7 +1059,7 @@ end
function ildb.get_ip_signal_asp(pts, connid)
local p = ildb.get_ip_signal(pts, connid)
if p then
- local asp = advtrains.interlocking.signal_get_aspect(p)
+ local asp = advtrains.interlocking.signal.get_aspect_info(p)
if not asp then
atlog("Clearing orphaned signal influence point", pts, "/", connid)
ildb.clear_ip_signal(pts, connid)
diff --git a/advtrains_interlocking/demosignals.lua b/advtrains_interlocking/demosignals.lua
deleted file mode 100644
index 98fa421..0000000
--- a/advtrains_interlocking/demosignals.lua
+++ /dev/null
@@ -1,104 +0,0 @@
--- Demonstration signals
--- Those can display the 3 main aspects of Ks signals
-
--- Note that the group value of advtrains_signal is 2, which means "step 2 of signal capabilities"
--- advtrains_signal=1 is meant for signals that do not implement set_aspect.
-
-
-local setaspect = function(pos, node, asp)
- if asp.main == 0 then
- advtrains.ndb.swap_node(pos, {name="advtrains_interlocking:ds_danger"})
- else
- if asp.dst ~= 0 and asp.main == -1 then
- advtrains.ndb.swap_node(pos, {name="advtrains_interlocking:ds_free"})
- else
- advtrains.ndb.swap_node(pos, {name="advtrains_interlocking:ds_slow"})
- end
- end
- local meta = minetest.get_meta(pos)
- if meta then
- meta:set_string("infotext", minetest.serialize(asp))
- end
-end
-
-local suppasp = {
- main = {0, 6, -1},
- dst = {0, false},
- shunt = false,
- proceed_as_main = true,
- info = {
- call_on = false,
- dead_end = false,
- w_speed = nil,
- }
-}
-
-minetest.register_node("advtrains_interlocking:ds_danger", {
- description = attrans("Demo signal at Danger"),
- tiles = {"at_il_signal_asp_danger.png"},
- groups = {
- cracky = 3,
- advtrains_signal = 2,
- save_in_at_nodedb = 1,
- },
- advtrains = {
- set_aspect = setaspect,
- supported_aspects = suppasp,
- get_aspect = function(pos, node)
- return advtrains.interlocking.DANGER
- end,
- },
- on_rightclick = advtrains.interlocking.signal_rc_handler,
- can_dig = advtrains.interlocking.signal_can_dig,
- after_dig_node = advtrains.interlocking.signal_after_dig,
-})
-minetest.register_node("advtrains_interlocking:ds_free", {
- description = attrans("Demo signal at Free"),
- tiles = {"at_il_signal_asp_free.png"},
- groups = {
- cracky = 3,
- advtrains_signal = 2,
- save_in_at_nodedb = 1,
- },
- advtrains = {
- set_aspect = setaspect,
- supported_aspects = suppasp,
- get_aspect = function(pos, node)
- return {
- main = -1,
- }
- end,
- },
- on_rightclick = advtrains.interlocking.signal_rc_handler,
- can_dig = advtrains.interlocking.signal_can_dig,
- after_dig_node = advtrains.interlocking.signal_after_dig,
-})
-minetest.register_node("advtrains_interlocking:ds_slow", {
- description = attrans("Demo signal at Slow"),
- tiles = {"at_il_signal_asp_slow.png"},
- groups = {
- cracky = 3,
- advtrains_signal = 2,
- save_in_at_nodedb = 1,
- },
- advtrains = {
- set_aspect = setaspect,
- supported_aspects = suppasp,
- get_aspect = function(pos, node)
- return {
- main = 6,
- }
- end,
- },
- on_rightclick = advtrains.interlocking.signal_rc_handler,
- can_dig = advtrains.interlocking.signal_can_dig,
- after_dig_node = advtrains.interlocking.signal_after_dig,
-})
-
-minetest.register_craft({
- output = "advtrains_interlocking:ds_danger",
- recipe = {
- {"bakedclay:black", "dye:red"},
- {"default:glass", "default:glass"},
- },
-})
diff --git a/advtrains_interlocking/init.lua b/advtrains_interlocking/init.lua
index ae73a37..596641e 100644
--- a/advtrains_interlocking/init.lua
+++ b/advtrains_interlocking/init.lua
@@ -2,7 +2,6 @@ ch_base.open_mod(minetest.get_current_modname())
-- Advtrains interlocking system
-- See database.lua for a detailed explanation
-
advtrains.interlocking = {}
advtrains.SHUNT_SPEED_MAX = 6
@@ -16,16 +15,19 @@ local modpath = minetest.get_modpath(minetest.get_current_modname()) .. DIR_DELI
dofile(modpath.."database.lua")
dofile(modpath.."signal_api.lua")
-dofile(modpath.."demosignals.lua")
+dofile(modpath.."signal_aspect_ui.lua")
dofile(modpath.."train_sections.lua")
dofile(modpath.."route_prog.lua")
dofile(modpath.."routesetting.lua")
dofile(modpath.."tcb_ts_ui.lua")
dofile(modpath.."route_ui.lua")
+dofile(modpath.."smartroute.lua")
+dofile(modpath.."autonaming.lua")
dofile(modpath.."tool.lua")
dofile(modpath.."approach.lua")
dofile(modpath.."ars.lua")
+
dofile(modpath.."tsr_rail.lua")
diff --git a/advtrains_interlocking/mod.conf b/advtrains_interlocking/mod.conf
index 3b2d029..9191dd9 100644
--- a/advtrains_interlocking/mod.conf
+++ b/advtrains_interlocking/mod.conf
@@ -4,4 +4,4 @@ description=Interlocking system for Advanced Trains
author=orwell96
depends=advtrains
-optional_depends=advtrains_train_track
+#optional_depends=advtrains_train_track
diff --git a/advtrains_interlocking/route_prog.lua b/advtrains_interlocking/route_prog.lua
index cdc98c0..e0634d4 100644
--- a/advtrains_interlocking/route_prog.lua
+++ b/advtrains_interlocking/route_prog.lua
@@ -19,6 +19,11 @@ C. punch a turnout (or some other passive component) to fix its state (toggle)
The route visualization will also be used to visualize routes after they have been programmed.
]]--
+-- TODO duplicate
+local lntrans = { "A", "B" }
+local function sigd_to_string(sigd)
+ return minetest.pos_to_string(sigd.p).." / "..lntrans[sigd.s]
+end
-- table with objectRefs
local markerent = {}
@@ -108,15 +113,7 @@ end
--[[
Route definition:
-route = {
- name = <string>
- [n] = {
- next = <sigd>, -- of the next (note: next) TCB on the route
- locks = {<pts> = "state"} -- route locks of this route segment
- }
- terminal = <sigd>,
- aspect = <signal aspect>,--note, might change in future
-}
+=== See database.lua L238
The first item in the TCB path (namely i=0) is always the start signal of this route,
so this is left out.
All subsequent entries, starting from 1, contain:
@@ -206,7 +203,7 @@ function advtrains.interlocking.visualize_route(origin, route, context, tmp_lcks
end
-- display locks set by player
for pts, state in pairs(tmp_lcks) do
- local pos = minetest.string_to_pos(pts)
+ local pos = advtrains.decode_pos(pts)
routesprite(context, pos, "fixp"..pts, "at_il_route_lock_edit.png", "Zajištěna ve stavu '"..state.."' cestou "..route.name.." (levý klik pro uvolnění)",
function() clear_lock(tmp_lcks, pname, pts) end)
end
@@ -216,20 +213,33 @@ end
local player_rte_prog = {}
-function advtrains.interlocking.init_route_prog(pname, sigd)
+function advtrains.interlocking.init_route_prog(pname, sigd, default_route)
if not minetest.check_player_privs(pname, "interlocking") then
- minetest.chat_send_player(pname, attrans("Insufficient privileges to use this!"))
+ minetest.chat_send_player(pname, "Insufficient privileges to use this!")
return
end
- player_rte_prog[pname] = {
+ local rp = {
origin = sigd,
- route = {
- name = "PROG["..pname.."]",
- },
- tmp_lcks = {},
}
- advtrains.interlocking.visualize_route(sigd, player_rte_prog[pname].route, "prog_"..pname, player_rte_prog[pname].tmp_lcks, pname)
- minetest.chat_send_player(pname, "Režim programování cesty je aktivní. Klikejte levým tlačítkem na TCB pro přidání úseků a na výhybky pro jejich uzamčení.")
+ if default_route then
+ rp.route = table.copy(default_route)
+
+ -- "Step back one section", but keeping turnouts
+ local last_route = rp.route[#rp.route]
+ if last_route then
+ rp.tmp_lcks = last_route.locks
+ rp.route[#rp.route] = nil
+ end
+ rp.route.name = "PROG["..pname.."]"
+ else
+ rp.route = {
+ name = "PROG["..pname.."]"
+ }
+ rp.tmp_lcks = {}
+ end
+ player_rte_prog[pname] = rp
+ advtrains.interlocking.visualize_route(sigd, rp.route, "prog_"..pname, rp.tmp_lcks, pname)
+ minetest.chat_send_player(pname, "Route programming mode active. Punch TCBs to add route segments, punch turnouts to lock them.")
end
local function get_last_route_item(origin, route)
@@ -239,10 +249,10 @@ local function get_last_route_item(origin, route)
return route[#route].next
end
-local function do_advance_route(pname, rp, sigd, tsname)
+local function do_advance_route(pname, rp, sigd, tsref)
table.insert(rp.route, {next = sigd, locks = rp.tmp_lcks})
rp.tmp_lcks = {}
- chat(pname, "Úsek '"..tsname.."' přidán na cestu.")
+ chat(pname, "Úsek '"..(tsref and (tsref.name or "") or "--EOI--").."' přidán na cestu.")
end
local function finishrpform(pname)
@@ -425,20 +435,20 @@ minetest.register_on_player_receive_fields(function(player, formname, fields)
if fields.advance then
-- advance route
if not is_endpoint then
- do_advance_route(pname, rp, this_sigd, this_ts.name)
+ do_advance_route(pname, rp, this_sigd, this_ts)
end
end
if fields.endhere then
if not is_endpoint then
- do_advance_route(pname, rp, this_sigd, this_ts.name)
+ do_advance_route(pname, rp, this_sigd, this_ts)
end
finishrpform(pname)
end
if can_over and fields.endover then
if not is_endpoint then
- do_advance_route(pname, rp, this_sigd, this_ts.name)
+ do_advance_route(pname, rp, this_sigd, this_ts)
end
- do_advance_route(pname, rp, over_sigd, over_ts and over_ts.name or "--EOI--")
+ do_advance_route(pname, rp, over_sigd, over_ts)
finishrpform(pname)
end
end
@@ -488,6 +498,8 @@ minetest.register_on_player_receive_fields(function(player, formname, fields)
local terminal = get_last_route_item(rp.origin, rp.route)
rp.route.terminal = terminal
rp.route.name = fields.name
+ -- new routes now always use the rscache
+ rp.route.use_rscache = true
table.insert(tcbs.routes, rp.route)
@@ -524,13 +536,16 @@ minetest.register_on_punchnode(function(pos, node, player, pointed_thing)
-- show formspec
show_routing_form(pname, tcbpos)
-
advtrains.interlocking.visualize_route(rp.origin, rp.route, "prog_"..pname, rp.tmp_lcks, pname)
-
+ return
+ elseif advtrains.interlocking.db.get_tcb(pos) then
+ -- the punched node itself is a TCB
+ show_routing_form(pname, pos)
+ advtrains.interlocking.visualize_route(rp.origin, rp.route, "prog_"..pname, rp.tmp_lcks, pname)
return
end
if advtrains.is_passive(pos) then
- local pts = advtrains.roundfloorpts(pos)
+ local pts = advtrains.encode_pos(pos)
if rp.tmp_lcks[pts] then
clear_lock(rp.tmp_lcks, pname, pts)
else
diff --git a/advtrains_interlocking/route_ui.lua b/advtrains_interlocking/route_ui.lua
index 0903b27..3fb5bc6 100644
--- a/advtrains_interlocking/route_ui.lua
+++ b/advtrains_interlocking/route_ui.lua
@@ -3,6 +3,7 @@
local atil = advtrains.interlocking
local ildb = atil.db
+local F = advtrains.formspec
-- TODO duplicate
local lntrans = { "A", "B" }
@@ -10,9 +11,10 @@ local function sigd_to_string(sigd)
return minetest.pos_to_string(sigd.p).." / "..lntrans[sigd.s]
end
+-- indexed by pname
+local sel_rpartcache = {}
-
-function atil.show_route_edit_form(pname, sigd, routeid)
+function atil.show_route_edit_form(pname, sigd, routeid, sel_rpartidx)
if not minetest.check_player_privs(pname, {train_operator=true, interlocking=true}) then
minetest.chat_send_player(pname, attrans("Insufficient privileges to use this!"))
@@ -30,16 +32,18 @@ function atil.show_route_edit_form(pname, sigd, routeid)
-- construct textlist for route information
local tab = {}
- local function itab(t)
+ local tabref = {}
+ local function itab(rseg, t, rty, rpara)
tab[#tab+1] = minetest.formspec_escape(string.gsub(t, ",", " "))
+ tabref[#tab] = { [rty] = true, param = rpara, seg = rseg, idx = #tab }
end
- itab("TCB "..sigd_to_string(sigd).." ("..tcbs.signal_name..") Route #"..routeid)
+ itab(1, "("..(tcbs.signal_name or "+")..") Route #"..routeid, "signal", sigd)
-- this code is partially copy-pasted from routesetting.lua
-- we start at the tc designated by signal
local c_sigd = sigd
local i = 1
- local c_tcbs, c_ts_id, c_ts, c_rseg, c_lckp
+ local c_tcbs, c_ts_id, c_ts, c_rseg
while c_sigd and i<=#route do
c_tcbs = ildb.get_tcbs(c_sigd)
if not c_tcbs then
@@ -54,50 +58,127 @@ function atil.show_route_edit_form(pname, sigd, routeid)
c_ts = ildb.get_ts(c_ts_id)
c_rseg = route[i]
- c_lckp = {}
- itab(""..i.." Vjezd "..sigd_to_string(c_sigd).." -> Ús. "..(c_ts and c_ts.name or "-").." -> Výjezd "..(c_rseg.next and sigd_to_string(c_rseg.next) or "KONEC"))
+ local signame = "-"
+ if c_tcbs and c_tcbs.signal then signame = c_tcbs.signal_name or "o" end
+ itab(i, ""..i.." "..sigd_to_string(c_sigd).." ("..signame..")", "signal", c_sigd)
+ itab(i, "= "..(c_ts and c_ts.name or c_ts_id).." ="..(c_rseg.call_on and " [CO]" or ""), "section", c_ts_id)
if c_rseg.locks then
for pts, state in pairs(c_rseg.locks) do
- local pos = minetest.string_to_pos(pts)
- itab(" Zámek: "..pts.." -> "..state)
+ local pos = advtrains.decode_pos(pts)
+ itab(i, "Zámek "..pts.." -> "..state, "lock", pos)
if not advtrains.is_passive(pos) then
itab("-!- Na "..pts.." není žádná pasivní komponenta. Prosím, přenastavte cestu!")
break
end
end
end
+ -- sanity check, is section at next the same as the current?
+ local nvar = c_rseg.next
+ if nvar then
+ local re_tcbs = ildb.get_tcbs({p = nvar.p, s = (nvar.s==1) and 2 or 1})
+ if not re_tcbs or not re_tcbs.ts_id or re_tcbs.ts_id~=c_ts_id then
+ itab(i, "-!- At "..sigd_to_string(c_sigd)..".Section Start and End do not match!", "err", nil)
+ break
+ end
+ end
-- advance
- c_sigd = c_rseg.next
+ c_sigd = nvar
i = i + 1
end
if c_sigd then
local e_tcbs = ildb.get_tcbs(c_sigd)
- itab("Konec cesty: "..sigd_to_string(c_sigd).." ("..(e_tcbs and e_tcbs.signal_name or "-")..")")
+ local signame = "-"
+ if e_tcbs and e_tcbs.signal then signame = e_tcbs.signal_name or "o" end
+ itab(i, "E "..sigd_to_string(c_sigd).." ("..signame..")", "end", c_sigd)
else
- itab("Cesta je slepá")
+ itab(i, "E (none)", "end", nil)
end
- form = form.."textlist[0.5,2;7.75,3.9;rtelog;"..table.concat(tab, ",").."]"
+ if not sel_rpartidx then sel_rpartidx = 1 end
+ form = form.."textlist[0.5,2;3.5,3.9;routelog;"..table.concat(tab, ",")..";"..(sel_rpartidx or 1)..";false]"
- form = form.."button[0.5,6;3,1;back;<<< Zpět na signalizaci]"
- form = form.."button[4.5,6;2,1;aspect;Signál...]"
- form = form.."button[6.5,6;2,1;delete;Smazat cestu]"
+ -- to the right of rtelog, controls are displayed for the thing in focus
+ -- What is in focus is determined by the parameter sel_rpartidx
+
+ local sel_rpart = tabref[sel_rpartidx]
+ --atdebug("sel rpart",sel_rpart)
+
+ if sel_rpart and sel_rpart.signal then
+ -- get TCBS here and rseg selected
+ local s_tcbs = ildb.get_tcbs(sel_rpart.param)
+ local rseg = route[sel_rpart.seg]
+ -- main aspect list
+ local signalpos = s_tcbs and s_tcbs.signal
+ if signalpos and rseg then
+ form = form..F.label(4.5, 2, "Signal Aspect:")
+ local ndef = signalpos and advtrains.ndb.get_ndef(signalpos)
+ if ndef and ndef.advtrains and ndef.advtrains.main_aspects then
+ local entries = { "<Default Aspect>" }
+ local sel = 1
+ for i, mae in ipairs(ndef.advtrains.main_aspects) do
+ entries[i+1] = mae.description
+ if mae.name == rseg.main_aspect then
+ sel = i+1
+ end
+ end
+ form = form..F.dropdown(4.5, 3.0, 4, "sa_main_aspect", entries, sel, true)
+ end
+ -- checkbox for assign distant signal
+ local assign_dst = rseg.assign_dst
+ if assign_dst == nil then
+ assign_dst = (sel_rpart.seg~=1) -- special behavior when assign_dst is nil (and not false):
+ -- defaults to false for the very first signal and true for all others (= minimal user configuration overhead)
+ -- Note: on save, the value will be fixed at either false or true
+ end
+ form = form..string.format("checkbox[4.5,4.0;sa_distant;Announce distant signal;%s]", assign_dst)
+ else
+ form = form..F.label(4.5, 2, "No Signal at this TCB")
+ end
+ elseif sel_rpart and sel_rpart.section then
+ local rseg = route[sel_rpart.seg]
+ if rseg then
+ form = form..F.label(4.5, 2, "Section Options:")
+ -- checkbox for call-on
+ form = form..string.format("checkbox[4.5,4.0;se_callon;Call-on (section may be occupied);%s]", rseg.call_on)
+ end
+ elseif sel_rpart and sel_rpart.err then
+ form = form.."textarea[4.5,2.5;4,4;errorta;Error:;"..tab[sel_rpartidx].."]"
+ else
+ form = form..F.label(4.5, 2, "<< Select a route part to edit options")
+ end
+
+ form = form.."button[0.5,6;1,1;prev;<<<]"
+ form = form.."button[1.5,6;1,1;back;"..routeid.."/"..#tcbs.routes.."]"
+ form = form.."button[2.5,6;1,1;next;>>>]"
+
+
+ --if route.smartroute_generated or route.default_autoworking then
+ -- form = form.."button[3.5,6;2,1;noautogen;Clr Autogen]"
+ --end
+ form = form.."button[5.5,6;3,1;delete;Delete Route]"
+ form = form.."button[0.5,7;3,1;back;Back to signal]"
+ form = form.."button[3.5,7;2,1;clone;Clone Route]"
+ form = form.."button[5.5,7;3,1;newfrom;New From Route]"
--atdebug(route.ars)
form = form.."style[ars;font=mono]"
- form = form.."textarea[0.8,7.3;5,3;ars;Pravidla ARS;"..atil.ars_to_text(route.ars).."]"
- form = form.."button[5.5,7.23;3,1;savears;Uložit pravidla ARS]"
+ form = form.."textarea[0.8,8.3;5,3;ars;ARS Rule List;"..atil.ars_to_text(route.ars).."]"
+ form = form.."button[5.5,8.23;3,1;savears;Save ARS List]"
- minetest.show_formspec(pname, "at_il_routeedit_"..minetest.pos_to_string(sigd.p).."_"..sigd.s.."_"..routeid, form)
-
+ local formname = "at_il_routeedit_"..minetest.pos_to_string(sigd.p).."_"..sigd.s.."_"..routeid
+ minetest.show_formspec(pname, formname, form)
+ -- record selected entry from routelog for receive fields callback
+ sel_rpartcache[pname] = sel_rpart
end
minetest.register_on_player_receive_fields(function(player, formname, fields)
local pname = player:get_player_name()
+ -- retreive sel_rpart from the cache in any case and clear it out
+ local sel_rpart = sel_rpartcache[pname]
if not minetest.check_player_privs(pname, {train_operator=true, interlocking=true}) then
return
end
@@ -118,26 +199,90 @@ minetest.register_on_player_receive_fields(function(player, formname, fields)
local route = tcbs.routes[routeid]
if not route then return end
+ if fields.prev then
+ atil.show_route_edit_form(pname, sigd, routeid - 1)
+ return
+ end
+ if fields.next then
+ atil.show_route_edit_form(pname, sigd, routeid + 1)
+ return
+ end
+
if fields.setname and fields.name then
route.name = fields.name
end
- if fields.aspect then
- local suppasp = advtrains.interlocking.signal_get_supported_aspects(tcbs.signal)
-
- local callback = function(pname, asp)
- route.aspect = asp
- advtrains.interlocking.show_route_edit_form(pname, sigd, routeid)
+ if fields.sa_main_aspect and sel_rpart and sel_rpart.signal then
+ local idx = tonumber(fields.sa_main_aspect)
+ -- get TCBS here and rseg selected
+ local s_tcbs = ildb.get_tcbs(sel_rpart.param)
+ local rseg = route[sel_rpart.seg]
+ -- main aspect list
+ local signalpos = s_tcbs and s_tcbs.signal
+ if rseg then
+ rseg.main_aspect = nil
+ if idx > 1 then
+ local ndef = signalpos and advtrains.ndb.get_ndef(signalpos)
+ if ndef and ndef.advtrains and ndef.advtrains.main_aspects then
+ rseg.main_aspect = ndef.advtrains.main_aspects[idx - 1].name
+ end
+ end
end
-
- advtrains.interlocking.show_signal_aspect_selector(pname, suppasp, route.name, callback, route.aspect or advtrains.interlocking.GENERIC_FREE)
- return
end
+ if fields.sa_distant and sel_rpart and sel_rpart.signal then
+ local rseg = route[sel_rpart.seg]
+ if rseg then
+ rseg.assign_dst = minetest.is_yes(fields.sa_distant)
+ end
+ end
+ if fields.se_callon and sel_rpart and sel_rpart.section then
+ local rseg = route[sel_rpart.seg]
+ if rseg then
+ rseg.call_on = minetest.is_yes(fields.se_callon)
+ -- reshow form to update CO marker
+ atil.show_route_edit_form(pname, sigd, routeid, sel_rpart.idx)
+ return
+ end
+ end
+
+ --if fields.noautogen then
+ -- route.smartroute_generated = nil
+ -- route.default_autoworking = nil
+ -- -- reshow form for the button to disappear
+ -- atil.show_route_edit_form(pname, sigd, routeid, sel_rpart and sel_rpart.idx)
+ -- return
+ --end
+
if fields.delete then
-- if something set the route in the meantime, make sure this doesn't break.
atil.route.update_route(sigd, tcbs, nil, true)
table.remove(tcbs.routes, routeid)
advtrains.interlocking.show_signalling_form(sigd, pname)
+ -- cleanup
+ sel_rpartcache[pname] = nil
+ return
+ end
+
+ if fields.clone then
+ -- if something set the route in the meantime, make sure this doesn't break.
+ atil.route.update_route(sigd, tcbs, nil, true)
+ local rcopy = table.copy(route)
+ rcopy.name = route.name.."_copy"
+ rcopy.smartroute_generated = nil
+ table.insert(tcbs.routes, routeid+1, rcopy)
+ advtrains.interlocking.show_signalling_form(sigd, pname)
+ -- cleanup
+ sel_rpartcache[pname] = nil
+ return
+ end
+
+ if fields.newfrom then
+ advtrains.interlocking.init_route_prog(pname, sigd, route)
+ minetest.close_formspec(pname, formname)
+ tcbs.ars_ignore_next = nil
+ -- cleanup
+ sel_rpartcache[pname] = nil
+ return
end
if fields.ars and fields.savears then
@@ -147,6 +292,25 @@ minetest.register_on_player_receive_fields(function(player, formname, fields)
if fields.back then
advtrains.interlocking.show_signalling_form(sigd, pname)
+ -- cleanup
+ sel_rpartcache[pname] = nil
+ return
+ end
+
+ -- if an entry was selected in the textlist (and its not the current one) update the form
+ if fields.routelog then
+ local prev_idx = sel_rpart and sel_rpart.idx or 1
+ local tev = minetest.explode_textlist_event(fields.routelog)
+ local new_idx = tev and tev.index
+ if new_idx and new_idx ~= prev_idx then
+ atil.show_route_edit_form(pname, sigd, routeid, new_idx)
+ return
+ end
+ end
+
+ if fields.quit then
+ -- cleanup
+ sel_rpartcache[pname] = nil
end
end
diff --git a/advtrains_interlocking/routesetting.lua b/advtrains_interlocking/routesetting.lua
index 67efaea..6544a92 100644
--- a/advtrains_interlocking/routesetting.lua
+++ b/advtrains_interlocking/routesetting.lua
@@ -43,13 +43,19 @@ function ilrs.set_route(signal, route, try)
local first = true
local i = 1
local rtename = route.name
- local signalname = ildb.get_tcbs(signal).signal_name
+ local signalname = (ildb.get_tcbs(signal).signal_name or "") .. sigd_to_string(signal)
local c_tcbs, c_ts_id, c_ts, c_rseg, c_lckp
+ -- signals = { { pos = ., tcbs_ref = <tcbs>, role = "main_distant", main_aspect = nil, dst_type = "next_main" or "none" }
+ local signals = {}
+ local nodst
while c_sigd and i<=#route do
c_tcbs = ildb.get_tcbs(c_sigd)
if not c_tcbs then
if not try then atwarn("Did not find TCBS",c_sigd,"while setting route",rtename,"of",signal) end
- return false, "No TCB found at "..sigd_to_string(c_sigd)..". Please reconfigure route!"
+ return false, "No TCB found at "..sigd_to_string(c_sigd)..". Please update or reconfigure route!"
+ end
+ if i == 1 then
+ nodst = c_tcbs.nodst
end
c_ts_id = c_tcbs.ts_id
if not c_ts_id then
@@ -60,42 +66,79 @@ function ilrs.set_route(signal, route, try)
c_rseg = route[i]
c_lckp = {}
- if c_ts.route then
+ if not c_ts then
+ if not try then atwarn("Encountered ts missing during a real run of routesetting routine, at ts=",c_ts_id,"while setting route",rtename,"of",signal) end
+ return false, "Section '"..(c_ts_id).."' not found!", c_ts_id, nil
+ elseif c_ts.route then
if not try then atwarn("Encountered ts lock during a real run of routesetting routine, at ts=",c_ts_id,"while setting route",rtename,"of",signal) end
- return false, "Section '"..c_ts.name.."' already has route set from "..sigd_to_string(c_ts.route.origin)..":\n"..c_ts.route.rsn, c_ts_id, nil
+ return false, "Section '"..(c_ts.name or c_ts_id).."' already has route set from "..sigd_to_string(c_ts.route.origin)..":\n"..c_ts.route.rsn, c_ts_id, nil
end
if c_ts.trains and #c_ts.trains>0 then
- if not try then atwarn("Encountered ts occupied during a real run of routesetting routine, at ts=",c_ts_id,"while setting route",rtename,"of",signal) end
- return false, "Section '"..c_ts.name.."' is occupied!", c_ts_id, nil
+ if c_rseg.call_on then
+ --atdebug("Routesetting: Call-on situation in", c_ts_id)
+ else
+ if not try then atwarn("Encountered ts occupied during a real run of routesetting routine, at ts=",c_ts_id,"while setting route",rtename,"of",signal) end
+ return false, "Section '"..(c_ts.name or c_ts_id).."' is occupied!", c_ts_id, nil
+ end
end
- for pts, state in pairs(c_rseg.locks) do
- local confl = ilrs.has_route_lock(pts, state)
+ -- collect locks from rs cache and from route def
+ local c_locks = {}
+ if route.use_rscache and c_ts.rs_cache and c_rseg.next then
+ -- rscache needs to be enabled, present and next must be defined
+ start_pkey = advtrains.encode_pos(c_sigd.p)
+ end_pkey = advtrains.encode_pos(c_rseg.next.p)
+ if c_ts.rs_cache[start_pkey] and c_ts.rs_cache[start_pkey][end_pkey] then
+ for lp,lst in pairs(c_ts.rs_cache[start_pkey][end_pkey]) do
+ --atdebug("Add lock from RSCache:",lp,"->",lst)
+ c_locks[lp] = lst
+ end
+ elseif not try then
+ atwarn("While setting route",rtename,"of",signal,"segment "..i..",required path from",c_tcbs,"to",c_rseg.next,"was not found in the track section's RS cache. Please check!")
+ end
+ end
+ -- add all from locks, these override the rscache
+ for lpts,lst in pairs(c_rseg.locks) do
+ --atdebug("Add lock from Routedef:",lpts,"->",lst,"overrides",c_locks[lpts] or "none")
+ c_locks[lpts] = lst
+ end
+
+ for lp, state in pairs(c_locks) do
+ local confl = ilrs.has_route_lock(lp, state)
- local pos = minetest.string_to_pos(pts)
+ local pos = advtrains.decode_pos(lp)
if advtrains.is_passive(pos) then
local cstate = advtrains.getstate(pos)
if cstate ~= state then
- local confl = ilrs.has_route_lock(pts)
+ local confl = ilrs.has_route_lock(lp)
if confl then
- if not try then atwarn("Encountered route lock while a real run of routesetting routine, at position",pts,"while setting route",rtename,"of",signal) end
- return false, "Lock conflict at "..pts..", Held locked by:\n"..confl, nil, pts
+ if not try then atwarn("Encountered route lock while a real run of routesetting routine, at position",pos,"while setting route",rtename,"of",signal) end
+ return false, "Lock conflict at "..minetest.pos_to_string(pos)..", Held locked by:\n"..confl, nil, lp
elseif not try then
advtrains.setstate(pos, state)
end
end
if not try then
- ilrs.add_route_lock(pts, c_ts_id, "Route '"..rtename.."' from signal '"..signalname.."'", signal)
- c_lckp[#c_lckp+1] = pts
+ ilrs.add_route_lock(lp, c_ts_id, "Route '"..rtename.."' from signal '"..signalname.."'", signal)
+ c_lckp[#c_lckp+1] = lp
end
else
if not try then atwarn("Encountered route lock misconfiguration (no passive component) while a real run of routesetting routine, at position",pts,"while setting route",rtename,"of",signal) end
- return false, "No passive component at "..pts..". Please reconfigure route!"
+ return false, "No passive component at "..minetest.pos_to_string(pos)..". Please update track section or reconfigure route!"
+ end
+ end
+ -- sanity check, is section at next the same as the current?
+ local nvar = c_rseg.next
+ if nvar then
+ local re_tcbs = ildb.get_tcbs({p = nvar.p, s = (nvar.s==1) and 2 or 1})
+ if (not re_tcbs or not re_tcbs.ts_id or re_tcbs.ts_id~=c_ts_id)
+ and route[i+1] then --FIX 2025-01-08: in old worlds the final TCB may be wrong (it didn't matter back then), don't error out here (route still shown invalid in UI)
+ if not try then atwarn("Encountered inconsistent ts (front~=back) while a real run of routesetting routine, at position",pts,"while setting route",rtename,"of",signal) end
+ return false, "TCB at "..minetest.pos_to_string(nvar.p).." has different section than previous TCB. Please update track section or reconfigure route!"
end
end
-- reserve ts and write locks
if not try then
- local nvar = c_rseg.next
if not route[i+1] then
-- We shouldn't use the "next" value of the final route segment, because this can lead to accidental route-cancelling of already set routes from another signal.
nvar = nil
@@ -112,9 +155,22 @@ function ilrs.set_route(signal, route, try)
}
if c_tcbs.signal then
c_tcbs.route_committed = true
- c_tcbs.aspect = route.aspect or advtrains.interlocking.GENERIC_FREE
c_tcbs.route_origin = signal
- advtrains.interlocking.update_signal_aspect(c_tcbs)
+ -- determine route role
+ local ndef = advtrains.ndb.get_ndef(c_tcbs.signal)
+ local assign_dst = c_rseg.assign_dst
+ if assign_dst == nil then
+ assign_dst = (i~=1) -- special behavior when assign_dst is nil (and not false):
+ -- defaults to false for the very first signal and true for all others (= minimal user configuration overhead)
+ end
+ local sig_table = {
+ pos = c_tcbs.signal,
+ tcbs_ref = c_tcbs,
+ role = ndef and ndef.advtrains and ndef.advtrains.route_role,
+ main_aspect = c_rseg.main_aspect,
+ assign_dst = assign_dst
+ }
+ signals[#signals+1] = sig_table
end
end
-- advance
@@ -122,10 +178,47 @@ function ilrs.set_route(signal, route, try)
c_sigd = c_rseg.next
i = i + 1
end
+
+ -- Get reference to signal at end of route
+ local last_mainsig = nil
+ if c_sigd then
+ local e_tcbs = ildb.get_tcbs(c_sigd)
+ local pos = e_tcbs and e_tcbs.signal
+ if pos then
+ last_mainsig = pos
+ end
+ end
+ for i = #signals, 1, -1 do
+ -- note the signals are iterated backwards. Switch depending on the role
+ local sig = signals[i]
+ -- apply mainaspect
+ sig.tcbs_ref.route_aspect = sig.main_aspect or "_default" -- or route.main_aspect : TODO this does not work if a distant signal is on the path! Implement per-sig aspects!
+ if sig.role == "distant" or sig.role == "distant_repeater" or sig.role == "main_distant" then
+ if last_mainsig then
+ -- assign the remote as the last mainsig if desired
+ if sig.assign_dst then
+ sig.tcbs_ref.route_remote = last_mainsig
+ end
+ -- if it wasn't a distant_repeater clear the mainsig
+ if sig.role ~= "distant_repeater" then
+ last_mainsig = false
+ end
+ end
+ end
+ if sig.role == "main" or sig.role == "main_distant" or sig.role == "end" then
+ -- record this as the new last mainsig
+ last_mainsig = sig.pos
+ end
+ -- for shunt signals nothing happens
+ -- update the signal aspect on map
+ advtrains.interlocking.signal.update_route_aspect(sig.tcbs_ref, i ~= 1)
+ end
return true
end
+-- Change 2024-01-27: pts is not an encoded pos, not a pos-to-string!
+
-- Checks whether there is a route lock that prohibits setting the component
-- to the wanted state. returns string with reasons on conflict
function ilrs.has_route_lock(pts)
@@ -172,6 +265,11 @@ function ilrs.free_route_locks(ts, lcks, nocallbacks)
end
function ilrs.free_route_locks_indiv(pts, ts, nocallbacks)
+ -- legacy: if starts with bracket then pts is still in old pos_to_string format (may happen because ts.route_post is not migrated)
+ if string.match(pts, "^%(") then
+ atdebug("free_route_locks_indiv: converting position",pts)
+ pts = advtrains.encode_pos(minetest.string_to_pos(pts))
+ end
local e = ilrs.rte_locks[pts]
if not e then return nil
elseif #e==0 then
@@ -191,7 +289,7 @@ function ilrs.free_route_locks_indiv(pts, ts, nocallbacks)
-- TODO use luaautomation timers?
if not nocallbacks then
minetest.after(0, ilrs.update_waiting, "lck", pts)
- minetest.after(0.5, advtrains.set_fallback_state, minetest.string_to_pos(pts))
+ minetest.after(0.5, advtrains.set_fallback_state, advtrains.decode_pos(pts))
end
end
-- frees all route locks, even manual ones set with the tool, at a specific position
@@ -224,12 +322,13 @@ function ilrs.cancel_route_from(sigd)
--atdebug("cancelling",c_ts.route.rsn)
-- clear signal aspect and routesetting state
c_tcbs.route_committed = nil
- c_tcbs.aspect = nil
+ c_tcbs.route_aspect = nil
+ c_tcbs.route_remote = nil
c_tcbs.routeset = nil
c_tcbs.route_auto = nil
c_tcbs.route_origin = nil
- advtrains.interlocking.update_signal_aspect(c_tcbs)
+ advtrains.interlocking.signal.update_route_aspect(c_tcbs)
c_ts_id = c_tcbs.ts_id
if not c_tcbs then
@@ -273,17 +372,19 @@ function ilrs.update_route(sigd, tcbs, newrte, cancel)
--atdebug("Signal not in control, held by",tcbs.signal_name)
return
end
+ -- clear route_rsn, it will be set again if needed
+ tcbs.route_rsn = nil
if (newrte and tcbs.routeset and tcbs.routeset ~= newrte) or cancel then
if tcbs.route_committed then
--atdebug("Cancelling:",tcbs.routeset)
advtrains.interlocking.route.cancel_route_from(sigd)
end
tcbs.route_committed = nil
- tcbs.aspect = nil
+ tcbs.route_aspect = nil
+ tcbs.route_remote = nil
has_changed_aspect = true
tcbs.routeset = nil
tcbs.route_auto = nil
- tcbs.route_rsn = nil
end
if newrte or tcbs.routeset then
if tcbs.route_committed then
@@ -291,7 +392,14 @@ function ilrs.update_route(sigd, tcbs, newrte, cancel)
end
if newrte then tcbs.routeset = newrte end
--atdebug("Setting:",tcbs.routeset)
- local succ, rsn, cbts, cblk = ilrs.set_route(sigd, tcbs.routes[tcbs.routeset])
+ local succ, rsn, cbts, cblk
+ local route = tcbs.routes[tcbs.routeset]
+ if route then
+ succ, rsn, cbts, cblk = ilrs.set_route(sigd, route)
+ else
+ succ = false
+ rsn = attrans("Route state changed.")
+ end
if not succ then
tcbs.route_rsn = rsn
--atdebug("Routesetting failed:",rsn)
@@ -308,12 +416,17 @@ function ilrs.update_route(sigd, tcbs, newrte, cancel)
end
else
--atdebug("Committed Route:",tcbs.routeset)
- has_changed_aspect = true
+ -- set_route now sets the signal aspects
+ --has_changed_aspect = true
+ -- route success. apply default_autoworking flag if requested
+ if route.default_autoworking then
+ tcbs.route_auto = true --FIX 2025-01-08: never set it to false if it was true!
+ end
end
end
if has_changed_aspect then
-- FIX: prevent an minetest.after() loop caused by update_signal_aspect dispatching path invalidation, which in turn calls ARS again
- advtrains.interlocking.update_signal_aspect(tcbs)
+ advtrains.interlocking.signal.update_route_aspect(tcbs)
end
advtrains.interlocking.update_player_forms(sigd)
end
diff --git a/advtrains_interlocking/signal_api.lua b/advtrains_interlocking/signal_api.lua
index 15f26ca..7d718a4 100644
--- a/advtrains_interlocking/signal_api.lua
+++ b/advtrains_interlocking/signal_api.lua
@@ -1,539 +1,479 @@
-- Signal API implementation
+local F = advtrains.formspec
---[[
-Signal aspect table:
-Note: All speeds are measured in m/s, aka the number of + signs in the HUD.
-asp = {
- main = <int speed>,
- -- Main signal aspect, tells state and permitted speed of next section
- -- 0 = section is blocked
- -- >0 = section is free, speed limit is this value
- -- -1 = section is free, maximum speed permitted
- -- false/nil = Signal doesn't provide main signal information, retain current speed limit.
- shunt = <boolean>,
- -- Whether train may proceed as shunt move, on sight
- -- main aspect takes precedence over this
- -- When main==0, train switches to shunt move and is restricted to speed 6
- proceed_as_main = <boolean>,
- -- If an approaching train is a shunt move and 'shunt' is false,
- -- the train may proceed as a train move under the "main" aspect
- -- if the main aspect permits it (i.e. main!=0)
- -- If this is not set, shunt moves are NOT allowed to switch to
- -- a train move, and must stop even if "main" would permit passing.
- -- This is intended to be used for "Halt for shunt moves" signs.
-
- dst = <int speed>,
- -- Distant signal aspect, tells state and permitted speed of the section after next section
- -- The character of these information is purely informational
- -- At this time, this field is not actively used
- -- 0 = section is blocked
- -- >0 = section is free, speed limit is this value
- -- -1 = section is free, maximum speed permitted
- -- false/nil = Signal doesn't provide distant signal information.
-
- -- the character of call_on and dead_end is purely informative
- call_on = <boolean>, -- Call-on route, expect train in track ahead (not implemented yet)
- dead_end = <boolean>, -- Route ends on a dead end (e.g. bumper) (not implemented yet)
-
- w_speed = <integer>,
- -- "Warning speed restriction". Supposed for short-term speed
- -- restrictions which always override any other restrictions
- -- imposed by "speed" fields, until lifted by a value of -1
- -- (Example: german Langsamfahrstellen-Signale)
- }
-}
-
-== How signals actually work in here ==
-Each signal (in the advtrains universe) is some node that has at least the
-following things:
-- An "influence point" that is set somewhere on a rail
-- An aspect which trains that pass the "influence point" have to obey
+local signal = {}
-There can be static and dynamic signals. Static signals are, roughly
-spoken, signs, while dynamic signals are "real" signals which can display
-different things.
-
-The node definition of a signal node should contain those fields:
-groups = {
- advtrains_signal = 2,
- save_in_at_nodedb = 1,
+signal.MASP_HALT = {
+ name = "_halt",
+ halt = true,
}
-advtrains = {
- set_aspect = function(pos, node, asp)
- -- This function gets called whenever the signal should display
- -- a new or changed signal aspect. It is not required that
- -- the signal actually displays the exact same aspect, since
- -- some signals can not do this by design. However, it must
- -- display an aspect that is at least as restrictive as the passed
- -- aspect as far as it is capable of doing so.
- -- Examples:
- -- - pure shunt signals can not display a "main" aspect
- -- and have no effect on train moves, so they will only ever
- -- honor the shunt.free field for their aspect.
- -- - the german Hl system can only signal speeds of 40, 60
- -- and 100 km/h, a speed of 80km/h should then be signalled
- -- as 60 km/h instead.
- -- In turn, it is not guaranteed that the aspect will fulfill the
- -- criteria put down in supported_aspects.
- -- If set_aspect is present, supported_aspects should also be declared.
-
- -- The aspect passed in here can always be queried using the
- -- advtrains.interlocking.signal_get_supposed_aspect(pos) function.
- -- It is always DANGER when the signal is not used as route signal.
-
- -- For static signals, this function should be completely omitted
- -- If this function is omitted, it won't be possible to use
- -- route setting on this signal.
- end,
- supported_aspects = {
- -- A table which tells which different types of aspects this signal
- -- is able to display. It is used to construct the "aspect editing"
- -- formspec for route programming (and others) It should always be
- -- present alongside with set_aspect. If this is not specified but
- -- set_aspect is, the user will be allowed to select any aspect.
- -- Any of the fields marked with <boolean/nil> support 3 types of values:
- nil: if this signal can switch between free/blocked
- false: always shows "blocked", unchangable
- true: always shows "free", unchangable
- -- Any of the "speed" fields should contain a list of possible values
- -- to be set as restriction. If omitted, the value of the described
- -- field is always assumed to be false (no information)
- -- A speed of 0 means that the signal can show a "blocked" aspect
- -- (which is probably the case for most signals)
- -- If the signal can signal "no information" on one of the fields
- -- (thus false is an acceptable value), include false in the list
- -- If your signal can only display a single speed (may it be -1),
- -- always enclose that single value into a list. (such as {-1})
- main = {<speed1>, ..., <speedn>} or nil,
- dst = {<speed1>, ..., <speedn>} or nil,
- shunt = <boolean/nil>,
-
- call_on = <boolean/nil>,
- dead_end = <boolean/nil>,
- w_speed = {<speed1>, ..., <speedn>} or nil,
-
- },
- Example for supported_aspects:
- supported_aspects = {
- main = {0, 6, -1}, -- can show either "Section blocked", "Proceed at speed 6" or "Proceed at maximum speed"
- dst = {0, false}, -- can show only if next signal shows "blocked", no other information.
- shunt = false, -- shunting by this signal is never allowed.
-
- call_on = false,
- dead_end = false,
- w_speed = nil,
- -- none of the information can be shown by the signal
-
- },
-
- get_aspect = function(pos, node)
- -- This function gets called by the train safety system. It
- should return the aspect that this signal actually displays,
- not preferably the input of set_aspect.
- -- For regular, full-featured light signals, they will probably
- honor all entries in the original aspect, however, e.g.
- simple shunt signals always return main=false regardless of
- the set_aspect input because they can not signal "Halt" to
- train moves.
- -- advtrains.interlocking.DANGER contains a default "all-danger" aspect.
- -- If your signal does not cover certain sub-tables of the aspect,
- the following reasonable defaults are automatically assumed:
- main = false (unchanged)
- dst = false (unchanged)
- shunt = false (shunting not allowed)
- info = {} (no further information)
- end,
+
+signal.MASP_DEFAULT = {
+ name = "_default",
+ default = true,
}
-on_rightclick = advtrains.interlocking.signal_rc_handler
-can_dig = advtrains.interlocking.signal_can_dig
-after_dig_node = advtrains.interlocking.signal_after_dig
-
-(If you need to specify custom can_dig or after_dig_node callbacks,
-please call those functions anyway!)
-
-Important note: If your signal should support external ways to set its
-aspect (e.g. via mesecons), there are some things that need to be considered:
-- advtrains.interlocking.signal_get_supposed_aspect(pos) won't respect this
-- Whenever you change the signal aspect, and that aspect change
-did not happen through a call to
-advtrains.interlocking.signal_set_aspect(pos, asp), you are
-*required* to call this function:
-advtrains.interlocking.signal_on_aspect_changed(pos)
-in order to notify trains about the aspect change.
-This function will query get_aspect to retrieve the new aspect.
-
-]]--
-
-local DANGER = {
+
+signal.ASPI_HALT = {
main = 0,
- dst = false,
shunt = false,
}
-advtrains.interlocking.DANGER = DANGER
-advtrains.interlocking.GENERIC_FREE = {
+signal.ASPI_FREE = {
main = -1,
shunt = false,
- dst = false,
+ proceed_as_main = true,
}
-local function convert_aspect_if_necessary(asp)
- if type(asp.main) == "table" then
- local newasp = {}
- if asp.main.free then
- newasp.main = asp.main.speed
- else
- newasp.main = 0
- end
- if asp.dst and asp.dst.free then
- newasp.dst = asp.dst.speed
- else
- newasp.dst = 0
+--[[
+Implementation plan orwell 2024-01-28:
+Most parts of ywang's implementation are fine, especially I like the formspecs. But I would like to change a few aspects (no pun intended) of this.
+- Signal gets distant assigned via field in signal aspect table (instead of explicitly)
+- Signal speed/shunt are no longer free-text but rather they need to be predefined in the node definition
+To do this: Differentiation between:
+== Main Aspect ==
+This is what a signal is assigned by either the route system or the user.
+It is a string key which has an appropriate entry in the node definition (where it has a description assigned)
+The signal mod defines a function to set a signal to the most appropriate aspect. This function gets
+a) the main aspect table (straight from node def)
+b) the distant signal's aspect group name & aspect table
+
+== Aspect ==
+One concrete combination of lights/shapes that a signal signal shows. Handling these is at the discretion of
+the signal mod defining the signal, and they are typically combinations of main aspect and distant aspect
+Example:
+- A Ks signal has the main_aspect="proceed_12" set for a route
+- The signal at the end of the route shows main_aspect="proceed_8", advtrains also passes on that this means {main=8, shunt=false}
+- The ndef.afunction(pos, node, main_aspect, rem_aspect, rem_aspinfo) determines that the signal should now show
+ blinking green with main indicator 12 and dst indicator 8, and sets the nodes accordingly.
+ This function can now return the Aspect Info table, which will be cached by advtrains until the aspect changes again
+ and will be used when a train approaches the signal. If nil is returned, then the aspect will be queried next time
+ by calling ndef.advtrains.get_aspect_info(pos)
+
+Note that once apply_aspect returns, there is no need for advtrains anymore to query the aspect info.
+When the signal, for any reason, wants to change its aspect by itself *without* going through the signal API then
+it should update the aspect info cache by calling advtrains.interlocking.signal.update_aspect_info(pos)
+
+Apply_aspect may also receive the special main aspect { name = "_halt", halt = true }. It usually means that the signal is not assigned to anything particular,
+and it should cause the signal to show its most restrictive aspect. Typically it is a halt aspect, but e.g. for distant-only
+signals this would be "expect stop".
+
+A special case occurs for pure distant signals: Such signals must set apply_aspect, but must not set main_aspects. Behavior is as follows:
+- Signal is uninitialized, distant signal is not assigned to a main signal, or no route is set: main_aspect == { name = "_halt", halt = true } and rem_aspect == nil
+- A remote main signal is assigned (either by user or by route): main_aspect is always { name = "_default" } and rem_aspect / rem_aspinfo give the correct information
+
+Main aspect names starting with underscore (e.g. "_default") are reserved and must not be used!
+
+== Aspect Info ==
+The actual signal aspect in the already-known format. This is what the trains use to determine halt/proceed and speed.
+asp = {
+ main = 0 (halt) / -1 (max speed) / false (no info) / <number> (speed limit)
+ shunt = true (shunt free) / false (shunt not free)
+ proceed_as_main = true (shunt move can proceed and become train move when main!=0) / false (no)
+ dst = speed of the remote signal (like main, informative character, not actually used)
+}
+
+Node definition of signals:
+- The signal needs some logic to figure out, for each combination of its own aspect group and the distant signal's aspect, what aspect info it can/will show.
+ndef.advtrains = {
+ main_aspects = {
+ { name = "proceed" description = "Proceed at full speed", <more data at discretion of signal>}
+ { name = "reduced" description = "Proceed at reduced speed", <more data at discretion of signal>}
+ }
+ -- This list is mainly for the selection dialog. Order of entries determines list order in the dropdown.
+ -- Some fields have special meaning:
+ -- name: A unique key to identify the main aspect. Might be required by some code.
+ -- description: Text shown in UI dropdown
+ -- Node can set any other fields at its discretion. They are not touched.
+ -- Note: Pure distant signals should set one main aspect, and set the "pure_distant = true" field
+ apply_aspect = function(pos, node, main_aspect, rem_aspect, rem_aspinfo)
+ -- set the node to show the desired aspect
+ -- called by advtrains when this signal's aspect group or the remote signal's aspect changes
+ -- main_aspect is never nil, but can be the special aspect { name = "_halt", halt = true }
+ -- MAY return the aspect_info. If it returns nil then get_aspect_info will be queried at a later point.
+ get_aspect_info(pos, main_aspect)
+ -- Returns the aspect info table (main, shunt, dst etc.)
+ distant_support = true or false
+ -- If true, signal is considered in distant signalling. If false or nil, rem_aspect and rem_aspinfo are never set.
+ route_role = one of "main", "main_distant", "shunt", "distant", "distant_repeater", "end"
+ -- Determines how the signal behaves when routes are set. Only in effect when signal is assigned to a TCB.
+ -- main: The signal is a possible endpoint for a train move route. Distant signals before it refer to it.
+ -- shunt: The signal is a possible endpoint for a shunt move route. Ignored for distant signals.
+ -- distant, distant_repeater: The next signal with role="main" is set as the remote signal. main_aspects may be undefined, the main aspect passed to apply_aspect is a dummy one in this case.
+ -- distant: if more than one distant signal is before a main signal, only the last one is assigned (but any number of distant_repeater signals are allowed)
+ -- main_distant: Combination of main and distant - like "main", but additionally gets assigned to the next main like a "distant"
+ -- end: like main, but signifies that it marks an end of track and trains cannot continue further. (currently no practical implications above main)
+ pure_distant = true / false
+ -- If true, this signal is assumed to be a pure distant signal (its halt aspect is rather "expect halt" and it cannot show a true "stop here")
+ -- The following special behavior applies when this signal is assigned to a TCB: When a train passes the signal, the aspect is not reset to the
+ -- halt aspect and it continues to announce the remote signal's aspect (like in real life)
+ -- Typically such signals have one main aspect, their appearance depends almost exclusively on their remote signal and the halt aspect is the same as
+ -- the aspect shown when the main aspect is set but the remote signal is at halt.
+}
+
+== Nomenclature ==
+The distant/main relation is named as follows:
+ V M
+=====>====>
+Main signal (main) always refers to the signal that is in focus right now (even if that is a distant-only signal)
+From the standpoint of M, V is the distant (dst) signal. M does not need to concern itself with V's aspect but needs to notify V when it changes
+From the standpoint of V, M is the remote (rem) signal. V needs to show an aspect that matches its remote signal M
+
+== Criteria for which signals are eligible for routes ==
+
+All signals must define:
+- get_aspect_info()
+
+Signals that can be assigned to a TCB must satisfy:
+- apply_aspect() defined
+
+Signals that are possible start and end points for a route must satisfy:
+- main_aspects defined (note, pure distant signals should therefore not define main_aspects)
+
+]]
+
+-- Database
+-- Signal Aspect store
+-- Stores for each signal the main aspect and other info, like the assigned remote signal
+-- [signal encodePos] = { main = <table or string>, [remote = encodedPos] }
+-- main is a string: "named aspect" is looked up in the main_aspects table of the ndef
+-- main is a table: this table directly is the main aspect (used for advanced signals with additional lights/indicators)
+signal.aspects = {}
+
+-- Distant signal notification. Records for each signal the distant signals that refer to it
+-- Note: this mapping is weak. Needs always backreference check.
+-- [signal encodePos] = { [distant signal encodePos] = true }
+signal.distant_refs = {}
+
+function signal.load(data)
+ signal.aspects = data.aspects or {}
+ -- rebuild distant_refs after load
+ signal.distant_refs = {}
+ for main, aspt in pairs(signal.aspects) do
+ if aspt.remote then
+ if not signal.distant_refs[aspt.remote] then
+ signal.distant_refs[aspt.remote] = {}
+ end
+ signal.distant_refs[aspt.remote][main] = true
end
- newasp.proceed_as_main = asp.shunt.proceed_as_main
- newasp.shunt = asp.shunt.free
- -- Note: info table not transferred, it's not used right now
- return newasp
end
- return asp
end
-function advtrains.interlocking.update_signal_aspect(tcbs)
- if tcbs.signal then
- local asp = tcbs.aspect or DANGER
- advtrains.interlocking.signal_set_aspect(tcbs.signal, asp)
- end
+function signal.save(data)
+ data.aspects = signal.aspects
end
-function advtrains.interlocking.signal_can_dig(pos)
- return not advtrains.interlocking.db.get_sigd_for_signal(pos)
-end
-function advtrains.interlocking.signal_after_dig(pos)
- -- clear influence point
- advtrains.interlocking.db.clear_ip_by_signalpos(pos)
+-- Set a signal's aspect.
+-- Signal aspects should only be set through this function. It takes care of:
+-- - Storing the main aspect and dst pos for this signal permanently (until next change)
+-- - Assigning the distant signal for this signal
+-- - Calling apply_aspect() in the signal's node definition to make the signal show the aspect
+-- - Calling apply_aspect() again whenever the remote signal changes its aspect
+-- - Notifying this signal's distant signals about changes to this signal (unless skip_dst_notify is specified)
+-- main_asp: either a string (==name in ndef.advtrains.main_aspects) or the main aspect table directly (for advanced signals)
+function signal.set_aspect(pos, main_asp, rem_pos, skip_dst_notify)
+ -- safeguard for the two integrated aspects (these two must be passed as string key)
+ if type(main_asp)=="table" and (main_asp.name=="_default" or main_asp.name=="_halt") then
+ error("MASP_HALT and MASP_DEFAULT must be passed via string keys _halt or _default, not as tables!")
+ end
+ local main_pts = advtrains.encode_pos(pos)
+ local old_tbl = signal.aspects[main_pts]
+ local old_remote = old_tbl and old_tbl.remote
+ local new_remote = rem_pos and advtrains.encode_pos(rem_pos)
+
+ -- if remote has changed, unregister from old remote
+ if old_remote and old_remote~=new_remote and signal.distant_refs[old_remote] then
+ --atdebug("unregister old remote: ",old_remote,"from",main_pts)
+ signal.distant_refs[old_remote][main_pts] = nil
+ end
+
+ signal.aspects[main_pts] = { main = main_asp, remote = new_remote }
+ -- apply aspect on main signal, this also checks new_remote
+ signal.reapply_aspect(main_pts)
+
+ -- notify my distants about this change (with limit 2)
+ if not skip_dst_notify then
+ signal.notify_distants_of(main_pts, 2)
+ end
end
-function advtrains.interlocking.signal_set_aspect(pos, asp)
- asp = convert_aspect_if_necessary(asp)
- local node=advtrains.ndb.get_node(pos)
- local ndef=minetest.registered_nodes[node.name]
- if ndef and ndef.advtrains and ndef.advtrains.set_aspect then
- ndef.advtrains.set_aspect(pos, node, asp)
- advtrains.interlocking.signal_on_aspect_changed(pos)
+function signal.clear_aspect(pos, skip_dst_notify)
+ local main_pts = advtrains.encode_pos(pos)
+ local old_tbl = signal.aspects[main_pts]
+ local old_remote = old_tbl and old_tbl.remote
+
+ -- unregister from old remote
+ if old_remote then
+ signal.distant_refs[old_remote][main_pts] = nil
+ end
+
+ signal.aspects[main_pts] = nil
+ -- apply aspect on main signal, this also checks new_remote
+ signal.reapply_aspect(main_pts)
+
+ -- notify my distants about this change (with limit 2)
+ if not skip_dst_notify then
+ signal.notify_distants_of(main_pts, 2)
end
end
--- should be called when aspect has changed on this signal.
-function advtrains.interlocking.signal_on_aspect_changed(pos)
- local ipts, iconn = advtrains.interlocking.db.get_ip_by_signalpos(pos)
- if not ipts then return end
- local ipos = minetest.string_to_pos(ipts)
+-- Clear any info about aspects from this signal, without resetting/reapplying the aspect.
+-- Supposed to be used for legacy on-off signals when the on-off toggle is used
+function signal.unregister_aspect(pos)
+ local main_pts = advtrains.encode_pos(pos)
+ local old_tbl = signal.aspects[main_pts]
+ local old_remote = old_tbl and old_tbl.remote
- advtrains.invalidate_all_paths_ahead(ipos)
+ -- unregister from old remote
+ if old_remote then
+ signal.distant_refs[old_remote][main_pts] = nil
+ end
+
+ signal.aspects[main_pts] = nil
end
-function advtrains.interlocking.signal_rc_handler(pos, node, player, itemstack, pointed_thing)
- local pname = player:get_player_name()
- local control = player:get_player_control()
- if control.aux1 then
- advtrains.interlocking.show_ip_form(pos, pname)
+-- Notify distant signals of main_pts of a change in the aspect of this signal
+--
+function signal.notify_distants_of(main_pts, limit)
+ --atdebug("notify_distants_of",advtrains.decode_pos(main_pts),"limit",limit)
+ if limit <= 0 then
return
end
-
- local sigd = advtrains.interlocking.db.get_sigd_for_signal(pos)
- if sigd then
- advtrains.interlocking.show_signalling_form(sigd, pname)
- else
- local ndef = minetest.registered_nodes[node.name]
- if ndef.advtrains and ndef.advtrains.set_aspect then
- -- permit to set aspect manually
- local function callback(pname, aspect)
- advtrains.interlocking.signal_set_aspect(pos, aspect)
+ local dstrefs = signal.distant_refs[main_pts]
+ --atdebug("dstrefs",dstrefs,"")
+ if dstrefs then
+ for dst,_ in pairs(dstrefs) do
+ -- ensure that the backref is still valid
+ local dst_asp = signal.aspects[dst]
+ if dst_asp and dst_asp.remote == main_pts then
+ signal.reapply_aspect(dst)
+ signal.notify_distants_of(dst, limit - 1)
+ else
+ atwarn("Distant signal backref is not purged: main =",main_pts,", distant =",dst,", remote =",dst_asp.remote,"")
end
- local isasp = ndef.advtrains.get_aspect(pos, node)
-
- advtrains.interlocking.show_signal_aspect_selector(
- pname,
- ndef.advtrains.supported_aspects,
- attrans("Set aspect manually"), callback,
- isasp)
- else
- --static signal - only IP
- advtrains.interlocking.show_ip_form(pos, pname)
end
end
end
--- Returns the aspect the signal at pos is supposed to show
-function advtrains.interlocking.signal_get_supposed_aspect(pos)
- local sigd = advtrains.interlocking.db.get_sigd_for_signal(pos)
- if sigd then
- local tcbs = advtrains.interlocking.db.get_tcbs(sigd)
- if tcbs.aspect then
- return convert_aspect_if_necessary(tcbs.aspect)
- end
- end
- return DANGER;
+function signal.notify_trains(pos)
+ local ipts, iconn = advtrains.interlocking.db.get_ip_by_signalpos(pos)
+ if not ipts then return end
+ local ipos = minetest.string_to_pos(ipts)
+
+ -- FIXME: invalidate_all_paths_ahead does not appear to always work as expected
+ --advtrains.invalidate_all_paths_ahead(ipos)
+ minetest.after(0, advtrains.invalidate_all_paths, ipos)
end
--- Returns the actual aspect of the signal at position, as returned by the nodedef.
--- returns nil when there's no signal at the position
-function advtrains.interlocking.signal_get_aspect(pos)
- local node=advtrains.ndb.get_node(pos)
- local ndef=minetest.registered_nodes[node.name]
- if ndef and ndef.advtrains and ndef.advtrains.get_aspect then
- local asp = ndef.advtrains.get_aspect(pos, node)
- if not asp then asp = DANGER end
- return convert_aspect_if_necessary(asp)
+-- Update waiting trains and distant signals about a changed signal aspect
+-- Must be called when a signal's aspect changes through some other means
+-- and not via the signal mechanism
+function signal.notify_on_aspect_changed(pos, skip_dst_notify)
+ signal.notify_trains(pos)
+ if not skip_dst_notify then
+ signal.notify_distants_of(advtrains.encode_pos(pos), 2)
end
- return nil
end
--- Returns the "supported_aspects" of the signal at position, as returned by the nodedef.
--- returns nil when there's no signal at the position
-function advtrains.interlocking.signal_get_supported_aspects(pos)
- local node=advtrains.ndb.get_node(pos)
- local ndef=minetest.registered_nodes[node.name]
- if ndef and ndef.advtrains and ndef.advtrains.supported_aspects then
- local asp = ndef.advtrains.supported_aspects
- return asp
- end
- return nil
+-- Gets the stored main aspect and distant signal position for this signal
+-- This information equals the information last passed to set_aspect
+-- It does not take into consideration the actual speed signalling, please use
+-- get_aspect_info() for this
+-- pos: the position of the signal
+-- returns: main_aspect, dst_pos
+function signal.get_aspect(pos)
+ local aspt = signal.aspects[advtrains.encode_pos(pos)]
+ local ma,dp = signal.get_aspect_internal(pos, aspt)
+ return ma, dp and advtrains.decode_pos(dp)
+end
+
+local function cache_mainaspects(ndefat)
+ ndefat.main_aspects_lookup = {}
+ for _,ma in ipairs(ndefat.main_aspects) do
+ ndefat.main_aspects_lookup[ma.name] = ma
+ end
+ ndefat.main_aspects_lookup[signal.MASP_HALT.name] = signal.MASP_HALT.name -- halt is always defined
+ ndefat.main_aspects_lookup[signal.MASP_DEFAULT.name] = ndefat.main_aspects[1] -- default is the first one
end
-local players_assign_ip = {}
-local function ipmarker(ipos, connid)
- local node_ok, conns, rhe = advtrains.get_rail_info_at(ipos, advtrains.all_tracktypes)
- if not node_ok then return end
- local yaw = advtrains.dir_to_angle(conns[connid].c)
+-- gets the main aspect. resolves named aspects to aspect table on demand
+function signal.get_aspect_internal(pos, aspt)
+ -- look up node and nodedef
+ local node = advtrains.ndb.get_node_or_nil(pos)
+ local ndef = node and minetest.registered_nodes[node.name]
+ if not aspt or not ndef then
+ -- oh, no main aspect, nevermind
+ return signal.MASP_HALT, nil, node, ndef
+ end
+ local ndefat = ndef.advtrains or {}
+ local masp = aspt.main or signal.MASP_HALT
- -- using tcbmarker here
- local obj = minetest.add_entity(vector.add(ipos, {x=0, y=0.2, z=0}), "advtrains_interlocking:tcbmarker")
- if not obj then return end
- obj:set_yaw(yaw)
- obj:set_properties({
- textures = { "at_il_signal_ip.png" },
- })
+ if type(masp) == "string" then
+ if masp=="_halt" then
+ masp = signal.MASP_HALT
+ elseif masp=="_default" and not ndefat.main_aspects then
+ -- case is fine, distant only signal
+ masp = signal.MASP_DEFAULT
+ else
+ assert(ndefat.main_aspects, "With named aspects, node "..node.name.." needs advtrains.main_aspects table!")
+ -- resolve the main aspect from the mainaspects table
+ if not ndefat.main_aspects_lookup then
+ cache_mainaspects(ndefat)
+ end
+ masp = ndefat.main_aspects_lookup[aspt.main] or signal.MASP_DEFAULT
+ end
+ end
+ -- return whatever the main aspect is
+ return masp, aspt.remote, node, ndef
end
--- shows small info form for signal IP state/assignment
--- only_notset: show only if it is not set yet (used by signal tcb assignment)
-function advtrains.interlocking.show_ip_form(pos, pname, only_notset)
- if not minetest.check_player_privs(pname, "interlocking") then
- return
- end
- local form = "size[7,5]label[0.5,0.5;" .. attrans("Signal at @1", minetest.pos_to_string(pos)).."]"
- advtrains.interlocking.db.check_for_duplicate_ip(pos)
- local pts, connid = advtrains.interlocking.db.get_ip_by_signalpos(pos)
- if pts then
- form = form.."label[0.5,1.5;" .. attrans("Influence point is set at @1/@2", pts, connid).."]"
- form = form.."button_exit[0.5,2.5; 5,1;set;"..attrans("Move") .. "]"
- form = form.."button_exit[0.5,3.5; 5,1;clear;" .. attrans("Clear") .. "]"
- local ipos = minetest.string_to_pos(pts)
- ipmarker(ipos, connid)
- else
- form = form.."label[0.5,1.5;" .. attrans("Influence point is not set.") .. "]"
- form = form.."label[0.5,2.0;" .. attrans("It is recommended to set an influence point.") .. "]"
- form = form.."label[0.5,2.5;" .. attrans("This is the point where trains will obey the signal.") .. "]"
+-- For the signal at pos, get the "aspect info" table. This contains the speed signalling information at this location
+function signal.get_aspect_info(pos)
+ -- get aspect internal
+ local aspt = signal.aspects[advtrains.encode_pos(pos)]
+ local masp, remote, node, ndef = signal.get_aspect_internal(pos, aspt)
+ -- call into ndef
+ if ndef and ndef.advtrains and ndef.advtrains.get_aspect_info then
+ local ai = ndef.advtrains.get_aspect_info
+ if type(ai)=="function" then
+ ai = ai(pos, masp)
+ end
+ if type(ai)=="table" then
+ --atdebug(pos,"aspectinfo",ai)
+ return ai
+ else
+ error("For node "..node.name..": ndef.advtrains.get_aspect_info must be function or table")
+ end
- form = form.."button_exit[0.5,3.5; 5,1;set;" .. attrans("Set") .. "]"
- end
- if not only_notset or not pts then
- minetest.show_formspec(pname, "at_il_ipassign_"..minetest.pos_to_string(pos), form)
end
end
-minetest.register_on_player_receive_fields(function(player, formname, fields)
- local pname = player:get_player_name()
- if not minetest.check_player_privs(pname, {train_operator=true, interlocking=true}) then
- return
- end
- local pts = string.match(formname, "^at_il_ipassign_([^_]+)$")
- local pos
- if pts then
- pos = minetest.string_to_pos(pts)
- end
- if pos then
- if fields.set then
- advtrains.interlocking.signal_init_ip_assign(pos, pname)
- elseif fields.clear then
- advtrains.interlocking.db.clear_ip_by_signalpos(pos)
+
+-- Called when either this signal has changed its main aspect
+-- or when this distant signal's currently assigned main signal has changed its aspect
+-- It retrieves the signal's main aspect and aspect info and calls apply_aspect of the node definition
+-- to update the signal's appearance and aspect info
+-- pts: The signal position to update as encoded_pos
+-- returns: the return value of the nodedef call which may be aspect_info
+function signal.reapply_aspect(pts)
+ -- get aspt
+ local aspt = signal.aspects[pts]
+ --atdebug("reapply_aspect",advtrains.decode_pos(pts),"aspt",aspt)
+ local pos = advtrains.decode_pos(pts)
+ -- resolve mainaspect table by name
+ local masp, remote, node, ndef = signal.get_aspect_internal(pos, aspt)
+ -- if we have remote, resolve remote
+ local rem_masp, rem_aspi
+ if remote then
+ -- register in remote signal as distant
+ if not signal.distant_refs[remote] then
+ signal.distant_refs[remote] = {}
+ end
+ signal.distant_refs[remote][pts] = true
+ local rem_aspt = signal.aspects[remote]
+ --atdebug("resolving remote",advtrains.decode_pos(remote),"aspt",rem_aspt)
+ local rem_pos = advtrains.decode_pos(remote)
+ local _,rem_ndef
+ rem_masp, _, _, rem_ndef = signal.get_aspect_internal(rem_pos, rem_aspt)
+ if rem_masp then
+ if rem_ndef.advtrains and rem_ndef.advtrains.get_aspect_info then
+ rem_aspi = rem_ndef.advtrains.get_aspect_info(rem_pos, rem_masp)
+ end
end
end
-end)
-
--- inits the signal IP assignment process
-function advtrains.interlocking.signal_init_ip_assign(pos, pname)
- if not minetest.check_player_privs(pname, "interlocking") then
- minetest.chat_send_player(pname, attrans("Insufficient privileges to use this!"))
- return
+ -- call into ndef
+ --atdebug("applying to",pos,": main_asp",masp,"rem_masp",rem_masp,"rem_aspi",rem_aspi)
+ if ndef.advtrains and ndef.advtrains.apply_aspect then
+ ndef.advtrains.apply_aspect(pos, node, masp, rem_masp, rem_aspi)
end
- --remove old IP
- --advtrains.interlocking.db.clear_ip_by_signalpos(pos)
- minetest.chat_send_player(pname, attrans("Configuring Signal: Please look in train's driving direction and punch rail to set influence point."))
-
- players_assign_ip[pname] = pos
+ -- notify trains
+ signal.notify_trains(pos)
end
-minetest.register_on_punchnode(function(pos, node, player, pointed_thing)
- local pname = player:get_player_name()
- if not minetest.check_player_privs(pname, "interlocking") then
- return
- end
- -- IP assignment
- local signalpos = players_assign_ip[pname]
- if signalpos then
- if vector.distance(pos, signalpos)<=50 then
- local node_ok, conns, rhe = advtrains.get_rail_info_at(pos, advtrains.all_tracktypes)
- if node_ok and #conns == 2 then
-
- local yaw = player:get_look_horizontal()
- local plconnid = advtrains.yawToClosestConn(yaw, conns)
-
- -- add assignment if not already present.
- local pts = advtrains.roundfloorpts(pos)
- if not advtrains.interlocking.db.get_ip_signal_asp(pts, plconnid) then
- advtrains.interlocking.db.set_ip_signal(pts, plconnid, signalpos)
- ipmarker(pos, plconnid)
- minetest.chat_send_player(pname, attrans("Configuring Signal: Successfully set influence point"))
- else
- minetest.chat_send_player(pname, attrans("Configuring Signal: Influence point of another signal is already present!"))
- end
- else
- minetest.chat_send_player(pname, attrans("Configuring Signal: This is not a normal two-connection rail! Aborted."))
- end
- else
- minetest.chat_send_player(pname, attrans("Configuring Signal: Node is too far away. Aborted."))
+-- Update this signal's aspect based on the set route
+--
+function signal.update_route_aspect(tcbs, skip_dst_notify)
+ if tcbs.signal then
+ if not tcbs.route_aspect and signal.get_signal_cap_level(tcbs.signal) == 2 then
+ return
+ -- Special behavior for pure-distant signals assigned to TCBs: retain their last assigned main signal
+ -- and do not fall back to halt. This mirrors real-life, where the distant signal goes back to
+ -- expect halt only when the main signal falls into halt
end
- players_assign_ip[pname] = nil
+ local asp = tcbs.route_aspect or "_halt"
+ local rem = tcbs.route_remote
+ signal.set_aspect(tcbs.signal, asp, rem, skip_dst_notify)
end
-end)
-
-
---== aspect selector ==--
-
-local players_aspsel = {}
+end
---[[
-suppasp: "supported_aspects" table
-purpose: form title string
-callback: func(pname, aspect) called on form submit
-isasp: aspect currently set
-]]
-function advtrains.interlocking.show_signal_aspect_selector(pname, p_suppasp, p_purpose, callback, isasp)
- local suppasp = p_suppasp or {
- main = {0, -1}, dst = {false}, shunt = false, info = {},
- }
- local purpose = p_purpose or ""
-
- local form = "size[7,7]label[0.5,0.5;" .. attrans("Select Signal Aspect:") .. "]"
- form = form.."label[0.5,1;"..purpose.."]"
-
- form = form.."label[0.5,1.5;" .. attrans("== Main Signal ==") .. "]"
- local selid = 1
- local entries = {}
- for idx, spv in ipairs(suppasp.main) do
- local entry
- if spv == 0 then
- entry = attrans("Halt")
- elseif spv == -1 then
- entry = attrans("Continue at maximum speed")
- elseif not spv then
- entry = attrans("Continue\\, speed limit unchanged (no info)")
- else
- entry = attrans("Continue at speed of @1", spv)
- end
- -- hack: the crappy formspec system returns the label, not the index. save the index in it.
- entries[idx] = idx.."| "..entry
- if isasp and spv == (isasp.main or false) then
- selid = idx
+-- Returns how capable the signal is with regards to aspect setting
+-- 0: not a signal at all
+-- 1: signal has get_aspect_info() but the aspect is not variable (e.g. a sign)
+-- 2: signal has apply_aspect() and main aspects but has "pure_distant" flag set (cannot be start/endpoint of a route, special behavior that its route aspect is not cleared on train pass)
+-- 3: signal has main signal role but can only ever display a halt aspect, such as a bumper (can be endpoint, but not startpoint, of a route)
+-- 4: Full capabilities, signal has main aspects and can be used as main/shunt signal (can be start/endpoint of a route)
+function signal.get_signal_cap_level(pos)
+ local node = advtrains.ndb.get_node_or_nil(pos)
+ local ndef = node and minetest.registered_nodes[node.name]
+ local ndefat = ndef and ndef.advtrains
+ if ndefat and ndefat.get_aspect_info then
+ if ndefat.apply_aspect and ndefat.main_aspects then
+ if not ndefat.pure_distant then
+ -- if the table contains anything, 4, otherwise 3
+ for _,_ in pairs(ndefat.main_aspects) do
+ return 4
+ end
+ return 3
+ end
+ return 2
end
+ return 1
end
- form = form.."dropdown[0.5,2;6;main;"..table.concat(entries, ",")..";"..selid.."]"
+ return 0
+end
-
- form = form.."label[0.5,3;" .. attrans("== Shunting ==") .. "]"
- if suppasp.shunt == nil then
- local st = 1
- if isasp and isasp.shunt then st=2 end
- form = form.."dropdown[0.5,3.5;6;shunt_free;---," .. attrans("allowed") .. ";"..st.."]"
- end
+----------------
- form = form.."label[0.5,4.5;" .. attrans("== Distant Signal ==").."]"
- local selid = 1
- local entries = {}
- for idx, spv in ipairs(suppasp.dst) do
- local entry
- if spv == 0 then
- entry = attrans("Expect to stop at the next signal")
- elseif spv == -1 then
- entry = attrans("Expect to pass the next signal at maximum speed")
- elseif not spv then
- entry = attrans("No info")
- else
- entry = attrans("Expect to pass the next signal at speed of @1", string.format("%d", spv))
+function signal.can_dig(pos, player)
+ local sigd = advtrains.interlocking.db.get_sigd_for_signal(pos)
+ if sigd then
+ -- check privileges
+ if not player or not minetest.check_player_privs(player:get_player_name(), "interlocking") then
+ if not player then -- intermediate debug to uncover hard-to-find bugz
+ atwarn("advtrains.interlocking.signal.can_dig(",pos,") called with player==nil!")
+ end
+ return false
end
- entries[idx] = idx.."| "..entry
- if isasp and spv == (isasp.dst or false) then
- selid = idx
+ -- check if route is set
+ local tcbs = advtrains.interlocking.db.get_tcbs(sigd)
+ if tcbs.routeset then
+ return false
end
end
- form = form.."dropdown[0.5,5;6;dst;"..table.concat(entries, ",")..";"..selid.."]"
-
- form = form.."button_exit[0.5,6;5,1;save;" .. attrans("Save signal aspect") .. "]"
-
- local token = advtrains.random_id()
-
- minetest.show_formspec(pname, "at_il_sigaspdia_"..token, form)
-
- minetest.after(1, function()
- players_aspsel[pname] = {
- suppasp = suppasp,
- callback = callback,
- token = token,
- }
- end)
+ return true
end
-local function usebool(sup, val, free)
- if sup == nil then
- return val==free
- else
- return sup
+function signal.after_dig(pos, oldnode, oldmetadata, player)
+ -- unassign signal if necessary
+ local sigd = advtrains.interlocking.db.get_sigd_for_signal(pos)
+ if sigd then
+ advtrains.interlocking.db.unassign_signal_for_tcbs(sigd)
+ minetest.chat_send_player(player:get_player_name(), "Signal has been unassigned. Name and routes are kept for reuse.")
+ end
+ -- clear influence point
+ local ipts,iconnid = advtrains.interlocking.db.get_ip_by_signalpos(pos)
+ if ipts then
+ advtrains.interlocking.db.clear_ip_signal(ipts, iconnid)
end
+ advtrains.interlocking.signal.unregister_aspect(pos)
end
--- other side of hack: extract the index
-local function ddindex(val)
- return tonumber(string.match(val, "^(%d+)|"))
+function signal.on_rightclick(pos, node, player, itemstack, pointed_thing)
+ local pname = player:get_player_name()
+ local control = player:get_player_control()
+ advtrains.interlocking.show_signal_form(pos, node, pname, control.aux1)
end
--- TODO use non-hacky way to parse outputs
-
-minetest.register_on_player_receive_fields(function(player, formname, fields)
- local pname = player:get_player_name()
- local psl = players_aspsel[pname]
- if psl then
- if formname == "at_il_sigaspdia_"..psl.token then
- if fields.save then
- local maini = ddindex(fields.main)
- if not maini then return end
- local dsti = ddindex(fields.dst)
- if not dsti then return end
- local asp = {
- main = psl.suppasp.main[maini],
- dst = psl.suppasp.dst[dsti],
- shunt = usebool(psl.suppasp.shunt, fields.shunt_free, attrans("allowed")),
- info = {}
- }
- psl.callback(pname, asp)
- end
- else
- players_aspsel[pname] = nil
- end
- end
-
-end)
+advtrains.interlocking.signal = signal
diff --git a/advtrains_interlocking/signal_aspect_ui.lua b/advtrains_interlocking/signal_aspect_ui.lua
new file mode 100644
index 0000000..98a332a
--- /dev/null
+++ b/advtrains_interlocking/signal_aspect_ui.lua
@@ -0,0 +1,281 @@
+local F = advtrains.formspec
+
+function advtrains.interlocking.show_signal_form(pos, node, pname, aux_key)
+ local sigd = advtrains.interlocking.db.get_sigd_for_signal(pos)
+ if sigd and not aux_key then
+ advtrains.interlocking.show_signalling_form(sigd, pname)
+ else
+ if advtrains.interlocking.signal.get_signal_cap_level(pos) >= 2 then
+ advtrains.interlocking.show_ip_sa_form(pos, pname)
+ else
+ advtrains.interlocking.show_ip_form(pos, pname)
+ end
+ end
+end
+
+local players_assign_ip = {}
+local players_assign_distant = {}
+
+local function ipmarker(ipos, connid)
+ local node_ok, conns, rhe = advtrains.get_rail_info_at(ipos, advtrains.all_tracktypes)
+ if not node_ok then return end
+ local yaw = advtrains.dir_to_angle(conns[connid].c)
+
+ -- using tcbmarker here
+ local obj = minetest.add_entity(vector.add(ipos, {x=0, y=0.2, z=0}), "advtrains_interlocking:tcbmarker")
+ if not obj then return end
+ obj:set_yaw(yaw)
+ obj:set_properties({
+ textures = { "at_il_signal_ip.png" },
+ })
+end
+
+function advtrains.interlocking.make_ip_formspec_component(pos, x, y, w)
+ advtrains.interlocking.db.check_for_duplicate_ip(pos)
+ local pts, connid = advtrains.interlocking.db.get_ip_by_signalpos(pos)
+ if pts then
+ -- display marker
+ local ipos = minetest.string_to_pos(pts)
+ ipmarker(ipos, connid)
+ return table.concat {
+ F.S_label(x, y, "Influence point is set at @1.", string.format("%s/%s", pts, connid)),
+ F.S_button_exit(x, y+0.5, w/2-0.125, "ip_set", "Modify"),
+ F.S_button_exit(x+w/2+0.125, y+0.5, w/2-0.125, "ip_clear", "Clear"),
+ }
+ else
+ return table.concat {
+ F.S_label(x, y, "Influence point is not set."),
+ F.S_button_exit(x, y+0.5, w, "ip_set", "Set influence point"),
+ }
+ end
+end
+
+-- shows small formspec to set the signal influence point
+-- only_notset: show only if it is not set yet (used by signal tcb assignment)
+function advtrains.interlocking.show_ip_form(pos, pname, only_notset)
+ if not minetest.check_player_privs(pname, "interlocking") then
+ return
+ end
+ local ipform = advtrains.interlocking.make_ip_formspec_component(pos, 0.5, 0.5, 7)
+ local form = {
+ "formspec_version[4]",
+ "size[8,2.25]",
+ ipform,
+ }
+ if not only_notset or not pts then
+ minetest.show_formspec(pname, "at_il_ipsaform_"..minetest.pos_to_string(pos), table.concat(form))
+ end
+end
+
+-- shows larger formspec to set the signal influence point, main aspect and distant signal pos
+-- only_notset: show only if it is not set yet (used by signal tcb assignment)
+function advtrains.interlocking.show_ip_sa_form(pos, pname)
+ if not minetest.check_player_privs(pname, "interlocking") then
+ return
+ end
+ local ipform = advtrains.interlocking.make_ip_formspec_component(pos, 0.5, 0.5, 7)
+ local ma, rpos = advtrains.interlocking.signal.get_aspect(pos)
+ local form = {
+ "formspec_version[4]",
+ "size[8,4.5]",
+ ipform,
+ }
+ -- Create Signal aspect formspec elements
+ local ndef = advtrains.ndb.get_ndef(pos)
+ if ndef and ndef.advtrains then
+ form[#form+1] = F.label(0.5, 2, "Signal Aspect:")
+ -- main aspect list
+ if ndef.advtrains.main_aspects then
+ local entries = { "<none>" }
+ local sel = 1
+ for i, mae in ipairs(ndef.advtrains.main_aspects) do
+ entries[i+1] = mae.description
+ if ma and ma.name == mae.name then
+ sel = i+1
+ end
+ end
+ form[#form+1] = F.dropdown(0.5, 2.5, 4, "sa_mainaspect", entries, sel, true)
+ end
+ -- distant signal assign (is shown either when main_aspect is not none, or when pure distant signal)
+ if rpos then
+ form[#form+1] = F.button_exit(0.5, 3.5, 4, "sa_undistant", "Dst: " .. minetest.pos_to_string(rpos))
+ elseif (ma and not ma.halt) or not ndef.advtrains.main_aspects then
+ form[#form+1] = F.button_exit(0.5, 3.5, 4, "sa_distant", "<assign distant>")
+ end
+ end
+
+ minetest.show_formspec(pname, "at_il_ipsaform_"..minetest.pos_to_string(pos), table.concat(form))
+end
+
+function advtrains.interlocking.handle_ip_sa_formspec_fields(pname, pos, fields)
+ if not (pos and minetest.check_player_privs(pname, {train_operator=true, interlocking=true})) then
+ return
+ end
+ local ma, rpos = advtrains.interlocking.signal.get_aspect(pos)
+ -- mainaspect dropdown
+ if fields.sa_mainaspect then
+ local idx = tonumber(fields.sa_mainaspect)
+ local new_ma = nil
+ if idx > 1 then
+ local ndef = advtrains.ndb.get_ndef(pos)
+ if ndef and ndef.advtrains and ndef.advtrains.main_aspects then
+ new_ma = ndef.advtrains.main_aspects[idx - 1]
+ end
+ end
+ if new_ma then
+ advtrains.interlocking.signal.set_aspect(pos, new_ma.name, rpos)
+ else
+ -- reset everything
+ advtrains.interlocking.signal.clear_aspect(pos)
+ end
+
+ end
+ -- buttons
+ if fields.ip_set then
+ advtrains.interlocking.init_ip_assign(pos, pname)
+ return
+ elseif fields.ip_clear then
+ advtrains.interlocking.db.clear_ip_by_signalpos(pos)
+ return
+ elseif fields.sa_distant then
+ advtrains.interlocking.init_distant_assign(pos, pname)
+ return
+ elseif fields.sa_undistant then
+ advtrains.interlocking.signal.set_aspect(pos, ma.name, nil)
+ return
+ end
+ -- show the form again unless one of the buttons was clicked
+ if not fields.quit then
+ advtrains.interlocking.show_ip_sa_form(pos, pname)
+ end
+end
+
+minetest.register_on_player_receive_fields(function(player, formname, fields)
+ local pname = player:get_player_name()
+ local pts = string.match(formname, "^at_il_ipsaform_([^_]+)$")
+ local pos
+ if pts then
+ pos = minetest.string_to_pos(pts)
+ end
+ if pos then
+ advtrains.interlocking.handle_ip_sa_formspec_fields(pname, pos, fields)
+ end
+end)
+
+-- inits the signal IP assignment process
+function advtrains.interlocking.init_ip_assign(pos, pname)
+ if not minetest.check_player_privs(pname, "interlocking") then
+ minetest.chat_send_player(pname, "Insufficient privileges to use this!")
+ return
+ end
+ --remove old IP
+ --advtrains.interlocking.db.clear_ip_by_signalpos(pos)
+ minetest.chat_send_player(pname, "Configuring Signal: Please look in train's driving direction and punch rail to set influence point.")
+
+ players_assign_ip[pname] = pos
+end
+
+-- inits the distant signal assignment process
+function advtrains.interlocking.init_distant_assign(pos, pname)
+ if not minetest.check_player_privs(pname, "interlocking") then
+ minetest.chat_send_player(pname, "Insufficient privileges to use this!")
+ return
+ end
+ minetest.chat_send_player(pname, "Set distant signal: Punch the main signal to assign!")
+
+ players_assign_distant[pname] = pos
+end
+
+-- Tries to automatically find a TCB to assign to the signal, or a main signal if this is a pure distant signal and another signal is found before the TCB
+local function try_auto_assign_to_tcb(signalpos, pos, connid, pname)
+ local pure_distant = advtrains.interlocking.signal.get_signal_cap_level(signalpos) == 2 -- exactly 2: pure distant sig
+ local is_past_first = false
+ local ti = advtrains.get_track_iterator(pos, connid, pure_distant and 150 or 16, false) -- maximum 16 track nodes ahead
+ local apos, aconnid = ti:next_branch()
+ while apos do
+ -- check for presence of a tcb
+ local tcb = advtrains.interlocking.db.get_tcb(apos)
+ if tcb then
+ -- check on the pointing connid whether it has a signal already
+ if not tcb[aconnid].signal then
+ -- go ahead and assign
+ local sigd = { p=apos, s=aconnid }
+ advtrains.interlocking.db.assign_signal_to_tcbs(signalpos, sigd)
+ -- use auto-naming
+ advtrains.interlocking.add_autoname_to_tcbs(tcb[aconnid], pname)
+ minetest.chat_send_player(pname, "Assigned signal to the TCB at "..core.pos_to_string(apos))
+ advtrains.interlocking.show_tcb_marker(apos)
+ advtrains.interlocking.show_signalling_form(sigd, pname)
+ end
+ -- in all cases return
+ return
+ elseif pure_distant and is_past_first then
+ -- try to find another signal's influence point here which could be the remote of a distant signal
+ local pts = advtrains.roundfloorpts(apos)
+ local mainsig = advtrains.interlocking.db.get_ip_signal(pts, aconnid)
+ if mainsig and advtrains.interlocking.signal.get_signal_cap_level(mainsig) >= 3 then
+ advtrains.interlocking.signal.set_aspect(signalpos, "_default", mainsig)
+ minetest.chat_send_player(pname, "Assigned distant signal to the main signal at "..core.pos_to_string(mainsig))
+ return
+ end
+ end
+ apos, aconnid = ti:next_track()
+ is_past_first = true
+ end
+ -- if we end up here limit is up
+end
+
+minetest.register_on_punchnode(function(pos, node, player, pointed_thing)
+ local pname = player:get_player_name()
+ if not minetest.check_player_privs(pname, "interlocking") then
+ return
+ end
+ -- IP assignment
+ local signalpos = players_assign_ip[pname]
+ if signalpos then
+ if vector.distance(pos, signalpos)<=50 then
+ local node_ok, conns, rhe = advtrains.get_rail_info_at(pos, advtrains.all_tracktypes)
+ if node_ok and #conns == 2 then
+
+ local yaw = player:get_look_horizontal()
+ local plconnid = advtrains.yawToClosestConn(yaw, conns)
+
+ -- add assignment if not already present.
+ local pts = advtrains.roundfloorpts(pos)
+ if not advtrains.interlocking.db.get_ip_signal_asp(pts, plconnid) then
+ advtrains.interlocking.db.set_ip_signal(pts, plconnid, signalpos)
+ ipmarker(pos, plconnid)
+ minetest.chat_send_player(pname, "Configuring Signal: Successfully set influence point")
+ -- Try to find a TCB ahead and auto assign this signal there
+ if advtrains.interlocking.signal.get_signal_cap_level(signalpos) >= 2 then
+ try_auto_assign_to_tcb(signalpos, pos, plconnid, pname)
+ end
+ else
+ minetest.chat_send_player(pname, "Configuring Signal: Influence point of another signal is already present!")
+ end
+ else
+ minetest.chat_send_player(pname, "Configuring Signal: This is not a normal two-connection rail! Aborted.")
+ end
+ else
+ minetest.chat_send_player(pname, "Configuring Signal: Node is too far away. Aborted.")
+ end
+ players_assign_ip[pname] = nil
+ end
+ -- DST assignment
+ signalpos = players_assign_distant[pname]
+ if signalpos then
+ -- get current mainaspect
+ local ma, rpos = advtrains.interlocking.signal.get_aspect(signalpos)
+ -- if punched pos is valid signal then set it as the new remote, otherwise nil
+ local nrpos
+ if advtrains.interlocking.signal.get_signal_cap_level(pos) > 1 then
+ nrpos = pos
+ if not ma or ma.halt then -- make sure that dst is never set without a main aspect (esp. for pure distant signal case)
+ ma = "_default"
+ end
+ advtrains.interlocking.signal.set_aspect(signalpos, ma, nrpos)
+ end
+ players_assign_distant[pname] = nil
+ end
+end)
+
diff --git a/advtrains_interlocking/smartroute.lua b/advtrains_interlocking/smartroute.lua
new file mode 100644
index 0000000..f03ece0
--- /dev/null
+++ b/advtrains_interlocking/smartroute.lua
@@ -0,0 +1,257 @@
+-- smartroute.lua
+-- Implementation of the advtrains auto-route search
+
+local atil = advtrains.interlocking
+local ildb = atil.db
+local sr = {}
+
+
+-- Start the SmartRoute process. This searches for routes and tries to match them with existing routes, showing them in a form
+function sr.start(pname, sigd)
+ -- is start signal a shunt signal? This becomes default setting for searching_shunt
+ local is_startsignal_shunt = false
+ local tcbs = ildb.get_tcbs(sigd)
+ if tcbs.signal then
+ local ndef = advtrains.ndb.get_ndef(tcbs.signal)
+ if ndef and ndef.advtrains then
+ if ndef.advtrains.route_role == "shunt" then
+ is_startsignal_shunt = true
+ end
+ end
+ end
+ sr.propose_next(pname, sigd, 0, is_startsignal_shunt)
+end
+
+
+local function otherside(s)
+ if s==1 then return 2 else return 1 end
+end
+
+--route search implementation
+-- new 2025-01-06: rely on the already present info from rscache to traverse sections
+-- this allows to implement a breadth first search
+-- format of foundroute:
+-- { name = "the name", tcbseq = { list of sigds in sequence, not containing the start sigd }}
+
+local function build_route_from_foundroute(froute, name)
+ local route = {
+ name = froute.name,
+ use_rscache = true,
+ smartroute_generated = true,
+ }
+ for _, sigd in ipairs(froute.tcbseq) do
+ route[#route+1] = { next = sigd, locks = {} }
+ end
+ return route
+end
+
+-- Maximum num of sections for routes to be found
+local RTE_MAX_SECS = 16
+
+-- scan for possible routes from the start tcb in a bread-first-search manner
+-- find_more_than: search is aborted only if more than the specified number of routes are found
+function sr.rescan(pname, sigd, tcbs, find_more_than, searching_shunt, pname)
+ local found_routes = {}
+ local restart_tcbs = { {sigd = sigd, tcbseq = {} } }
+ local last_len = 0
+ while true do
+ -- take first entry out of restart_tcbs (due to the way it is inserted the first entry will always be the one with the lowest length
+ local cur_restart
+ for idx, rst in ipairs(restart_tcbs) do
+ cur_restart = rst
+ table.remove(restart_tcbs, idx)
+ break
+ end
+ if not cur_restart then
+ -- we have no candidates left. Give up and return what we have
+ --atdebug("(SR) No Candidates left, end rescan")
+ return found_routes
+ end
+ -- check if we need to stop due to having found enough routes
+ local cur_len = #cur_restart.tcbseq
+ if cur_len > last_len then
+ -- one level is finished, check if enoufh routes are found
+ if #found_routes > find_more_than then
+ --atdebug("(SR) Layer finished and enough routes found, end rescan")
+ return found_routes
+ end
+ last_len = cur_len
+ end
+ -- our current restart point is nouw in cur_restart
+ local c_sigd = cur_restart.sigd
+ --atdebug("(SR) Search continues at",c_sigd,"seqlen",#cur_restart.tcbseq)
+ -- do a TS repair, this also updates the RS cache should it be out of date
+ local c_ts_id = ildb.check_and_repair_ts_at_pos(c_sigd.p, c_sigd.s, pname, false)
+ if c_ts_id then
+ local c_ts = ildb.get_ts(c_ts_id)
+ local bgn_pts = advtrains.encode_pos(c_sigd.p)
+ local rsout = c_ts.rs_cache[bgn_pts]
+ if rsout then
+ for _, end_sigd in ipairs(c_ts.tc_breaks) do
+ end_pkey = advtrains.encode_pos(end_sigd.p)
+ if rsout[end_pkey] then
+ --atdebug("(SR) Section",c_ts_id,c_ts.name,"has way",c_sigd,"->",end_sigd)
+ local nsigd = {p=end_sigd.p, s = end_sigd.s==1 and 2 or 1} -- invert to other side
+ -- record nsigd in the tcbseq
+ local ntcbseq = table.copy(cur_restart.tcbseq)
+ ntcbseq[#ntcbseq+1] = nsigd
+ local shall_continue = true
+ -- check if that sigd is a route target
+ local tcbs = ildb.get_tcbs(nsigd)
+ if tcbs.signal then
+ local ndef = advtrains.ndb.get_ndef(tcbs.signal)
+ if ndef and ndef.advtrains then
+ if ndef.advtrains.route_role == "main" or ndef.advtrains.route_role == "main_distant"
+ or ndef.advtrains.route_role == "end" or ndef.advtrains.route_role == "shunt" then
+ -- signal is suitable target
+ local is_mainsignal = ndef.advtrains.route_role ~= "shunt"
+ --atdebug("(SR) Suitable end signal at",nsigd,", recording route!")
+ -- record the found route in the results
+ found_routes[#found_routes+1] = {
+ tcbseq = ntcbseq,
+ shunt_route = not is_mainsignal,
+ name = tcbs.signal_name or atil.sigd_to_string(nsigd)
+ }
+ -- if this is a main signal and/or we are only searching shunt routes, stop the search here
+ if is_mainsignal or searching_shunt then
+ --atdebug("(SR) Not continuing this branch!")
+ shall_continue = false
+ end
+ end
+ end
+ end
+ -- unless overridden, insert the next restart point
+ if shall_continue then
+ restart_tcbs[#restart_tcbs+1] = {sigd = nsigd, tcbseq = ntcbseq }
+ end
+ end
+ end
+ else
+ --atdebug("(SR) Section",c_ts_id,c_ts.name,"found no rscache entry for start ",bgn_pts)
+ end
+ else
+ --atdebug("(SR) Stop at",c_sigd,"because no sec ahead")
+ end
+ end
+end
+
+local players_smartroute_actions = {}
+-- Propose to pname the smartroute actions in a form, with the current settings as passed to this function
+function sr.propose_next(pname, sigd, find_more_than, searching_shunt)
+ local tcbs = ildb.get_tcbs(sigd)
+ if not tcbs or not tcbs.routes then
+ minetest.chat_send_player(pname, "Smartroute: TCBS or routes don't exist here!")
+ return
+ elseif not tcbs.ts_id then
+ minetest.chat_send_player(pname, "Smartroute: No track section directly ahead!")
+ return
+ end
+ -- Step 1: search for routes using the current settings
+ local found_routes = sr.rescan(pname, sigd, tcbs, find_more_than, searching_shunt, pname)
+ -- Step 2: store in actions table
+ players_smartroute_actions[pname] = {
+ sigd = sigd,
+ searching_shunt = searching_shunt,
+ found_routes = found_routes
+ }
+ -- step 3: build form
+ local form = "size[5,5]label[0,0;Route search: "..#found_routes.." found]"
+ local tab = {}
+ for idx, froute in ipairs(found_routes) do
+ tab[idx] = minetest.formspec_escape(froute.name.." (Len="..#froute.tcbseq..")")
+ end
+ form=form.."textlist[0.5,1;4,3;rtelist;"..table.concat(tab, ",").."]"
+ form=form.."button[0.5,4;2,1;continue;Search further]"
+ form=form.."button[2.5,4;2,1;apply;Apply]"
+
+ minetest.show_formspec(pname, "at_il_smartroute_propose", form)
+end
+
+minetest.register_on_player_receive_fields(function(player, formname, fields)
+ local pname = player:get_player_name()
+ if not minetest.check_player_privs(pname, "interlocking") then
+ return
+ end
+ if formname ~= "at_il_smartroute_propose" then
+ return
+ end
+ -- retrieve from the storage the current search result
+ local srtab = players_smartroute_actions[pname]
+ if not srtab then
+ return
+ end
+ local sigd = srtab.sigd
+ local found_routes = srtab.found_routes
+
+ if fields.continue then
+ -- search on, but find at least one route more
+ sr.propose_next(pname, sigd, #found_routes, srtab.searching_shunt)
+ return
+ end
+
+ if fields.apply then
+ -- user is happy with the found routes. Proceed to save them in the signal
+ local tcbs = ildb.get_tcbs(sigd)
+ if not tcbs then return end
+ -- remove routes for endpoints for which routes already exist
+ local ex_endpts = {} -- key = sigd_to_string
+ for rtid, route in ipairs(tcbs.routes) do
+ local valid = advtrains.interlocking.check_route_valid(route, sigd)
+ local endpoint = route[#route].next -- 'next' field of the last route segment (the segment with index==len)
+ if valid and endpoint then
+ local endstr = advtrains.interlocking.sigd_to_string(endpoint)
+ --atdebug("(Smartroute) Find existing endpoint:",route.name,"ends at",endstr)
+ ex_endpts[endstr] = route.name
+ else
+ --atdebug("(Smartroute) Find existing endpoint:",route.name," not considered, endpoint",endpoint,"valid",valid)
+ end
+ end
+ local new_frte = {}
+ for _,froute in ipairs(found_routes) do
+ local endpoint = froute.tcbseq[#froute.tcbseq]
+ local endstr = advtrains.interlocking.sigd_to_string(endpoint)
+ if not ex_endpts[endstr] then
+ new_frte[#new_frte+1] = froute
+ else
+ --atdebug("(Smartroute) Throwing away",froute.name,"because endpoint",endstr,"already reached by route",ex_endpts[endstr])
+ end
+ end
+
+ -- All remaining routes will be applied to the signal
+ local sel_rte = #tcbs.routes+1
+ for idx, froute in ipairs(new_frte) do
+ tcbs.routes[#tcbs.routes+1] = build_route_from_foundroute(froute)
+ end
+ -- if only one route present and it is newly created (there was no route before, thus sel_rte==1), make default
+ if sel_rte == 1 and #tcbs.routes == 1 then
+ local route1 = tcbs.routes[1]
+ route1.ars = {default=true}
+ -- if that only route furthermore is a suitable block signal route (1 section with no locks), set it into block signal mode
+ if #route1 == 1 then
+ local ts = tcbs.ts_id and advtrains.interlocking.db.get_ts(tcbs.ts_id)
+ if ts and #ts.tc_breaks == 2 then
+ -- check for presence of any locks
+ local epos1 = advtrains.encode_pos(ts.tc_breaks[1].p)
+ local epos2 = advtrains.encode_pos(ts.tc_breaks[2].p)
+ local haslocks =
+ (route1[1].locks and next(route1[1].locks)) -- the route itself has no locks
+ or (ts.fixed_locks and next(ts.fixed_locks)) -- the section has no fixedlocks
+ or (ts.rs_cache and ts.rs_cache[epos1] and ts.rs_cache[epos1][epos2] and next(ts.rs_cache[epos1][epos2])) -- the section has no locks in rscache
+ if not haslocks then
+ -- yeah, blocksignal!
+ route1.default_autoworking = true
+ end
+ end
+ end
+ end
+ --atdebug("Smartroute done!")
+ advtrains.interlocking.show_signalling_form(sigd, pname, sel_rte)
+ players_smartroute_actions[pname] = nil
+ end
+ if fields.quit then
+ players_smartroute_actions[pname] = nil
+ end
+end)
+
+
+advtrains.interlocking.smartroute = sr
diff --git a/advtrains_interlocking/spec/basic_signalling_spec.lua b/advtrains_interlocking/spec/basic_signalling_spec.lua
new file mode 100644
index 0000000..a4e1e3a
--- /dev/null
+++ b/advtrains_interlocking/spec/basic_signalling_spec.lua
@@ -0,0 +1,106 @@
+--[[
+This file tests a large part of the signaling system, as a lot of tests for the
+signaling system tend to overlap for various parts of the system.
+]]
+
+require("mineunit")
+mineunit("core")
+
+_G.advtrains = {
+ interlocking = {
+ aspect = fixture("../../aspect"),
+ },
+ ndb = {
+ get_node = minetest.get_node,
+ swap_node = minetest.swap_node,
+ }
+}
+
+fixture("advtrains_helpers")
+fixture("../../database")
+sourcefile("distant")
+sourcefile("signal_api")
+sourcefile("signal_aspect_accessors")
+fixture("../../demosignals")
+
+minetest.register_node("advtrains_interlocking:signal_sign", {
+ advtrains = {
+ get_aspcet = function() return {main = 19} end
+ }
+})
+
+local D = advtrains.distant
+local I = advtrains.interlocking
+local A = I.aspect
+
+local stub_aspect_t1 = {
+ free = {main = -1},
+ slow = {main = 6},
+ danger = {main = 0, shunt = false},
+}
+for k, v in pairs(stub_aspect_t1) do
+ stub_aspect_t1[k] = A(v)
+end
+local stub_pos_t1 = {}
+for i = 1, 4 do
+ stub_pos_t1[i] = {x = 1, y = 0, z = i}
+end
+
+world.layout {
+ {stub_pos_t1[1], "advtrains_interlocking:ds_danger"},
+ {stub_pos_t1[2], "advtrains_interlocking:ds_slow"},
+ {stub_pos_t1[3], "advtrains_interlocking:ds_free"},
+ {stub_pos_t1[4], "advtrains_interlocking:signal_sign"},
+}
+
+describe("API for supposed signal aspects", function()
+ it("should load and save data properly", function()
+ local tbl = {_foo = {}}
+ I.load_supposed_aspects(tbl)
+ assert.same(tbl, I.save_supposed_aspects())
+ end)
+ it("should set and get signals properly", function ()
+ local pos = stub_pos_t1[2]
+ local asp = stub_aspect_t1.slow
+ local newasp = A{ main = math.random(1,5) }
+ assert.equal(asp, I.signal_get_aspect(pos))
+ I.signal_set_aspect(pos, newasp)
+ assert.equal(newasp, I.signal_get_aspect(pos))
+ assert.equal(asp, I.signal_get_real_aspect(pos))
+ I.signal_set_aspect(pos, asp)
+ end)
+end)
+
+describe("Distant signaling", function()
+ it("should assign distant signals and set the distant aspect correspondingly", function()
+ for i = 1, 2 do
+ D.assign(stub_pos_t1[i], stub_pos_t1[i+1])
+ end
+ assert.equal(stub_aspect_t1.danger, I.signal_get_aspect(stub_pos_t1[1]))
+ assert.equal(A{main = 6, dst = 0}, I.signal_get_aspect(stub_pos_t1[2]))
+ assert.equal(A{main = -1, dst = 6}, I.signal_get_aspect(stub_pos_t1[3]))
+ end)
+ it("should report assignments properly", function()
+ assert.same({stub_pos_t1[1], "manual"}, {D.get_main(stub_pos_t1[2])})
+ assert.same({[advtrains.encode_pos(stub_pos_t1[3])] = "manual"}, D.get_dst(stub_pos_t1[2]))
+ end)
+ it("should update distant aspects automatically", function()
+ I.signal_set_aspect(stub_pos_t1[2], {main = 2, dst = -1})
+ assert.equal(A{main = 2, dst = 0}, I.signal_get_aspect(stub_pos_t1[2]))
+ assert.equal(A{main = -1, dst = 2}, I.signal_get_aspect(stub_pos_t1[3]))
+ end)
+ it("should unassign signals when one is removed", function()
+ world.set_node(stub_pos_t1[2], "air")
+ assert.same({}, D.get_dst(stub_pos_t1[1]))
+ assert.same({}, {D.get_main(stub_pos_t1[3])})
+ assert.same(stub_aspect_t1.free, I.signal_get_aspect(stub_pos_t1[3]))
+ end)
+ it("should reject signal signs", function()
+ D.assign(stub_pos_t1[1], stub_pos_t1[4])
+ assert.same({}, D.get_dst(stub_pos_t1[1]))
+ assert.same({}, {D.get_main(stub_pos_t1[4])})
+ D.assign(stub_pos_t1[4], stub_pos_t1[1])
+ assert.same({}, D.get_dst(stub_pos_t1[4]))
+ assert.same({}, {D.get_main(stub_pos_t1[1])})
+ end)
+end)
diff --git a/advtrains_interlocking/spec/fixtures/advtrains_helpers.lua b/advtrains_interlocking/spec/fixtures/advtrains_helpers.lua
new file mode 120000
index 0000000..9b0ab67
--- /dev/null
+++ b/advtrains_interlocking/spec/fixtures/advtrains_helpers.lua
@@ -0,0 +1 @@
+../../../advtrains/helpers.lua \ No newline at end of file
diff --git a/advtrains_interlocking/spec/mineunit.conf b/advtrains_interlocking/spec/mineunit.conf
new file mode 100644
index 0000000..e69de29
--- /dev/null
+++ b/advtrains_interlocking/spec/mineunit.conf
diff --git a/advtrains_interlocking/spec/signal_group_spec.lua b/advtrains_interlocking/spec/signal_group_spec.lua
new file mode 100644
index 0000000..bc9d007
--- /dev/null
+++ b/advtrains_interlocking/spec/signal_group_spec.lua
@@ -0,0 +1,95 @@
+require "mineunit"
+mineunit("core")
+
+_G.advtrains = {
+ interlocking = {
+ aspect = sourcefile("aspect"),
+ },
+ ndb = {
+ get_node = minetest.get_node,
+ swap_node = minetest.swap_node,
+ }
+}
+
+fixture("advtrains_helpers")
+sourcefile("database")
+sourcefile("signal_api")
+sourcefile("distant")
+sourcefile("signal_aspect_accessors")
+
+local A = advtrains.interlocking.aspect
+local D = advtrains.distant
+local I = advtrains.interlocking
+local N = advtrains.ndb
+
+local groupdef = {
+ name = "foo",
+ aspects = {
+ proceed = {main = -1},
+ caution = {},
+ danger = {main = 0},
+ "proceed",
+ {"caution"},
+ "danger",
+ },
+}
+
+for k, v in pairs(groupdef.aspects) do
+ minetest.register_node("advtrains_interlocking:" .. k, {
+ advtrains = {
+ supported_aspects = {
+ group = "foo",
+ },
+ get_aspect = function() return A{group = "foo", name = k} end,
+ set_aspect = function(pos, _, name)
+ N.swap_node(pos, {name = "advtrains_interlocking:" .. name})
+ end,
+ }
+ })
+end
+
+local origin = vector.new(0, 0, 0)
+local dstpos = vector.new(0, 0, 1)
+
+world.layout {
+ {origin, "advtrains_interlocking:danger"},
+ {dstpos, "advtrains_interlocking:proceed"},
+}
+
+describe("signal group registration", function()
+ it("should work", function()
+ A.register_group(groupdef)
+ assert(A.get_group_definition("foo"))
+ end)
+ it("should only be allowed once for the same group", function()
+ assert.has.errors(function() A.register_group(type2def) end)
+ end)
+ it("should handle nonexistant groups", function()
+ assert.is_nil(A.get_group_definition("something_else"))
+ end)
+ it("should reject invalid definitions", function()
+ assert.has.errors(function() A.register_group({}) end)
+ assert.has.errors(function() A.register_group({name="",label={}}) end)
+ assert.has.errors(function() A.register_group({name="",aspects={}}) end)
+ end)
+end)
+
+describe("signal aspect", function()
+ it("should handle empty fields properly", function()
+ assert.equal(A{main = 0}, A{group="foo", name="danger"}:to_group())
+ end)
+ it("should be converted properly", function()
+ assert.equal(A{main = 0}, A{group="foo", name="danger"})
+ assert.equal(A{}, A{group="foo", name="caution"})
+ assert.equal(A{main = -1}, A{group="foo", name="proceed"})
+ end)
+end)
+
+describe("signals in groups", function()
+ it("should support distant signaling", function()
+ assert.equal("caution", A():adjust_distant(A{group="foo",name="danger"}).name)
+ assert.equal("proceed", A():adjust_distant(A{group="foo",name="caution"}).name)
+ assert.equal("proceed", A():adjust_distant(A{group="foo",name="proceed"}).name)
+ assert.equal("danger", A{group="foo",name="danger"}:adjust_distant{}.name)
+ end)
+end)
diff --git a/advtrains_interlocking/tcb_ts_ui.lua b/advtrains_interlocking/tcb_ts_ui.lua
index 263a1f0..9a44b4d 100644
--- a/advtrains_interlocking/tcb_ts_ui.lua
+++ b/advtrains_interlocking/tcb_ts_ui.lua
@@ -2,8 +2,11 @@
local players_assign_tcb = {}
local players_assign_signal = {}
+local players_assign_xlink = {}
local players_link_ts = {}
+local players_assign_fixedlocks = {}
+local atil = advtrains.interlocking
local ildb = advtrains.interlocking.db
local ilrs = advtrains.interlocking.route
@@ -14,6 +17,7 @@ local lntrans = { "A", "B" }
local function sigd_to_string(sigd)
return minetest.pos_to_string(sigd.p).." / "..lntrans[sigd.s]
end
+advtrains.interlocking.sigd_to_string = sigd_to_string
minetest.register_node("advtrains_interlocking:tcb_node", {
drawtype = "mesh",
@@ -88,12 +92,8 @@ minetest.register_node("advtrains_interlocking:tcb_node", {
local tcb = ildb.get_tcb(tcbpos)
if not tcb then return true end
for connid=1,2 do
- if tcb[connid].ts_id or tcb[connid].signal then
- minetest.chat_send_player(pname, attrans("Can't remove TCB: Both sides must have no track section and no signal assigned!"))
- return false
- end
- if not ildb.may_modify_tcbs(tcb[connid]) then
- minetest.chat_send_player(pname, attrans("Can't remove TCB: Side @1 forbids modification (shouldn't happen).", connid))
+ if tcb[connid].signal then
+ minetest.chat_send_player(pname, attrans("Can't remove TCB: Both sides must have no signal assigned!"))
return false
end
end
@@ -102,18 +102,11 @@ minetest.register_node("advtrains_interlocking:tcb_node", {
end,
after_dig_node = function(pos, oldnode, oldmetadata, player)
if not oldmetadata or not oldmetadata.fields then return end
+ local pname = player:get_player_name()
local tcbpts = oldmetadata.fields.tcb_pos
if tcbpts and tcbpts ~= "" then
local tcbpos = minetest.string_to_pos(tcbpts)
- local success = ildb.remove_tcb(tcbpos)
- if success and player then
- minetest.chat_send_player(player:get_player_name(), attrans("TCB has been removed."))
- else
- minetest.chat_send_player(player:get_player_name(), attrans("Failed to remove TCB!"))
- minetest.set_node(pos, oldnode)
- local meta = minetest.get_meta(pos)
- meta:set_string("tcb_pos", minetest.pos_to_string(tcbpos))
- end
+ ildb.remove_tcb_at(tcbpos, pname)
end
end,
})
@@ -165,23 +158,22 @@ minetest.register_on_punchnode(function(pos, node, player, pointed_thing)
local tcbnpos = players_assign_tcb[pname]
if tcbnpos then
if vector.distance(pos, tcbnpos)<=20 then
- local node_ok, conns, rhe = advtrains.get_rail_info_at(pos, advtrains.all_tracktypes)
+ local node_ok, conns, rhe = advtrains.get_rail_info_at(pos)
if node_ok and #conns == 2 then
- local ok = ildb.create_tcb(pos)
-
- if not ok then
- minetest.chat_send_player(pname, attrans("Configuring TCB: TCB already exists at this position! It has now been re-assigned."))
+ -- if there is already a tcb here, reassign it
+ if ildb.get_tcb(pos) then
+ minetest.chat_send_player(pname, "Configuring TCB: Already existed at this position, it is now linked to this TCB marker")
+ else
+ ildb.create_tcb_at(pos, pname)
end
-
- ildb.sync_tcb_neighbors(pos, 1)
- ildb.sync_tcb_neighbors(pos, 2)
-
+
local meta = minetest.get_meta(tcbnpos)
meta:set_string("tcb_pos", minetest.pos_to_string(pos))
meta:set_string("infotext", attrans("TCB assigned to @1", minetest.pos_to_string(pos)))
minetest.chat_send_player(pname, attrans("Configuring TCB: Successfully configured TCB"))
+ advtrains.interlocking.show_tcb_marker(pos)
else
- minetest.chat_send_player(pname, attrans("Configuring TCB: This is not a normal two-connection rail! Aborted."))
+ minetest.chat_send_player(pname, "Configuring TCB: This is not a normal two-connection rail! Aborted.")
end
else
minetest.chat_send_player(pname, attrans("Configuring TCB: Node is too far away. Aborted."))
@@ -196,18 +188,13 @@ minetest.register_on_punchnode(function(pos, node, player, pointed_thing)
local is_signal = minetest.get_item_group(node.name, "advtrains_signal") >= 2
if is_signal then
local ndef = minetest.registered_nodes[node.name]
- if ndef and ndef.advtrains and ndef.advtrains.set_aspect then
+ if ndef and ndef.advtrains and ndef.advtrains.apply_aspect then
local tcbs = ildb.get_tcbs(sigd)
if tcbs then
- tcbs.signal = pos
- if not tcbs.signal_name then
- tcbs.signal_name = minetest.pos_to_string(sigd.p)
- end
- if not tcbs.routes then
- tcbs.routes = {}
- end
- ildb.set_sigd_for_signal(pos, sigd)
- minetest.chat_send_player(pname, attrans("Configuring TCB: Successfully assigned signal."))
+ ildb.assign_signal_to_tcbs(pos, sigd)
+ -- use auto-naming
+ advtrains.interlocking.add_autoname_to_tcbs(tcbs, pname)
+ minetest.chat_send_player(pname, "Configuring TCB: Successfully assigned signal.")
advtrains.interlocking.show_ip_form(pos, pname, true)
else
minetest.chat_send_player(pname, attrans("Configuring TCB: Internal error, TCBS doesn't exist. Aborted."))
@@ -223,40 +210,201 @@ minetest.register_on_punchnode(function(pos, node, player, pointed_thing)
end
players_assign_signal[pname] = nil
end
+
+ -- FixedLocks assignment
+ local ts_id = players_assign_fixedlocks[pname]
+ if ts_id then
+ if advtrains.is_passive(pos) then
+ local pts = advtrains.encode_pos(pos)
+ local state = advtrains.getstate(pos)
+ local ts = ildb.get_ts(ts_id)
+ if ts and ts.fixed_locks then
+ minetest.chat_send_player(pname, minetest.pos_to_string(pos).." locks in state "..state)
+ ts.fixed_locks[pts] = state
+ else
+ minetest.chat_send_player(pname, "Error: TS modified, abort!")
+ players_assign_fixedlocks[pname] = nil
+ end
+ else
+ minetest.chat_send_player(pname, "Setting fixed locks finished!")
+ players_assign_fixedlocks[pname] = nil
+ ildb.update_rs_cache(ts_id)
+ advtrains.interlocking.show_ts_form(ts_id, pname)
+ end
+ end
end)
+-- "Self-contained TCB"
+-- 2024-11-25: Buffers should become their own TCB (and signal) automatically to permit setting routes to them
+-- These are support functions for this kind of node.
+
+-- Create an after_place_node callback for a self-contained TCB node. The parameters control additional behavior:
+-- fail_silently_on_noprivs: (boolean) Does not give an error in case the placer does not have the interlocking privilege
+-- auto_create_self_signal: (boolean) Automatically assign this same node as signal to the A side of the newly-created TCB
+-- (this is useful for buffers as they serve both as TCB and as an always-halt signal)
+function advtrains.interlocking.self_tcb_make_after_place_callback(fail_silently_on_noprivs, auto_create_self_signal)
+ return function(pos, player, itemstack, pointed_thing)
+ local pname = player:get_player_name()
+ if not minetest.check_player_privs(pname, "interlocking") then
+ if not fail_silently_on_noprivs then
+ minetest.chat_send_player(pname, "Insufficient privileges to use this!")
+ end
+ return
+ end
+ if ildb.get_tcb(pos) then
+ minetest.chat_send_player(pname, "TCB already existed at this position, now linked to this node")
+ else
+ ildb.create_tcb_at(pos, pname)
+ end
+ if auto_create_self_signal then
+ local sigd = { p = pos, s = 1 }
+ local tcbs = ildb.get_tcbs(sigd)
+ -- make sure signal doesn't already exist
+ if tcbs.signal then
+ minetest.chat_send_player(pname, "Signal on B side already assigned!")
+ return
+ end
+ ildb.assign_signal_to_tcbs(pos, sigd)
+ -- assign influence point to itself
+ ildb.set_ip_signal(advtrains.roundfloorpts(pos), 1, pos)
+ -- use auto-naming
+ advtrains.interlocking.add_autoname_to_tcbs(tcbs, pname)
+ end
+ end
+end
+
+-- Create an can_dig callback for a self-contained TCB node. The parameters control additional behavior:
+-- is_signal: (boolean) Whether this node is also a signal (in addition to being a TCB), e.g. when auto_create_self_signal was set.
+-- Causes also the signal API's can_dig to be called
+function advtrains.interlocking.self_tcb_make_can_dig_callback(is_signal)
+ return function(pos, player)
+ local pname = player and player:get_player_name() or ""
+ -- need to duplicate logic of the regular "can_dig_or_modify_track()" function in core/tracks.lua
+ if advtrains.get_train_at_pos(pos) then
+ minetest.chat_send_player(pname, "Can't remove track, a train is here!")
+ return false
+ end
+ -- end of standard checks
+ local tcb = ildb.get_tcb(pos)
+ if not tcb then
+ -- digging always allowed because the TCB hasn't been created (unless signal callback interjects)
+ if is_signal then
+ return advtrains.interlocking.signal.can_dig(pos, player)
+ else
+ return true
+ end
+ end
+ -- TCB exists
+ if not minetest.check_player_privs(pname, "interlocking") then
+ return false
+ end
+ -- fine to remove (unless signal callback interjects)
+ if is_signal then
+ return advtrains.interlocking.signal.can_dig(pos, player)
+ else
+ return true
+ end
+ end
+end
+
+-- Create an after_dig_node callback for a self-contained TCB node. The parameters control additional behavior:
+-- is_signal: (boolean) Whether this node is also a signal (in addition to being a TCB), e.g. when auto_create_self_signal was set.
+-- Causes also the signal API's after_dig_node to be called
+function advtrains.interlocking.self_tcb_make_after_dig_callback(is_signal)
+ return function(pos, oldnode, oldmetadata, player)
+ local pname = player:get_player_name()
+ if is_signal then
+ -- "dig" the signal first
+ advtrains.interlocking.signal.after_dig(pos, oldnode, oldmetadata, player)
+ end
+ if ildb.get_tcb(pos) then
+ -- remove the TCB
+ ildb.remove_tcb_at(pos, pname, true)
+ end
+ end
+end
+
+-- Create an on_rightclick callback for a self-contained TCB node. The rightclick callback tries to repeat the TCB assignment
+-- if necessary and otherwise shows the TCB formspec. The parameters control additional behavior:
+-- fail_silently_on_noprivs: (boolean) Does not give an error in case the placer does not have the interlocking privilege
+-- auto_create_self_signal: (boolean) Automatically assign this same node as signal to the B side of the
+-- newly-created TCB if that has not already happened during place.
+-- Otherwise, opens the signal dialog instead of the TCB dialog on rightclick
+function advtrains.interlocking.self_tcb_make_on_rightclick_callback(fail_silently_on_noprivs, auto_create_self_signal)
+ return function(pos, node, player, itemstack, pointed_thing)
+ local pname = player:get_player_name()
+ if not minetest.check_player_privs(pname, "interlocking") then
+ if not fail_silently_on_noprivs then
+ minetest.chat_send_player(pname, "Insufficient privileges to use this!")
+ end
+ return
+ end
+ if ildb.get_tcb(pos) then
+ -- TCB already here. go on
+ else
+ -- otherwise create tcb
+ ildb.create_tcb_at(pos, pname)
+ end
+ if auto_create_self_signal then
+ local sigd = { p = pos, s = 1 }
+ local tcbs = ildb.get_tcbs(sigd)
+ -- make sure signal doesn't already exist
+ if not tcbs.signal then
+ -- go ahead and assign signal
+ ildb.assign_signal_to_tcbs(pos, sigd)
+ -- assign influence point to itself
+ ildb.set_ip_signal(advtrains.roundfloorpts(pos), 1, pos)
+ -- use auto-naming
+ advtrains.interlocking.add_autoname_to_tcbs(tcbs, pname)
+ end
+ -- in any case open the signalling form nouw
+ local control = player:get_player_control()
+ advtrains.interlocking.show_signal_form(pos, node, pname, control.aux1)
+ return
+ else
+ -- not an autosignal. Then show the TCB form
+ advtrains.interlocking.show_tcb_form(pos, pname)
+ return
+ end
+ end
+end
+
-- TCB Form
local sidecolorA = minetest.get_color_escape_sequence("#f40000")
local sidecolorB = minetest.get_color_escape_sequence("#068b00")
-local function mktcbformspec(tcbs, btnpref, offset, pname)
+local function mktcbformspec(pos, side, tcbs, offset, pname)
local form = ""
+ local btnpref = side==1 and "A" or "B"
local ts
- local sidecolor = sidecolorB
- if btnpref == "A" then
- sidecolor = sidecolorA
- end
+ -- ensure that mapping and xlink are up to date
+ ildb.tcbs_ensure_ts_ref_exists({p=pos, s=side, tcbs=tcbs})
+ ildb.validate_tcb_xlink({p=pos, s=side, tcbs=tcbs})
+ -- Note: repair operations may have been triggered by this
if tcbs.ts_id then
ts = ildb.get_ts(tcbs.ts_id)
end
if ts then
- form = form.."label[0.5,"..offset..";"..sidecolor..attrans("Side @1", btnpref)..": "..minetest.formspec_escape(ts.name).."]"
- form = form.."button[0.5,"..(offset+0.5)..";5,1;"..btnpref.."_gotots;"..attrans("Show track section").."]"
- if ildb.may_modify_tcbs(tcbs) then
- -- Note: the security check to prohibit those actions is located in database.lua in the corresponding functions.
- form = form.."button[0.5,"..(offset+1.5)..";2.5,1;"..btnpref.."_update;"..attrans("Update near TCBs").."]"
- form = form.."button[3 ,"..(offset+1.5)..";2.5,1;"..btnpref.."_remove;"..attrans("Remove from section").."]"
- end
+ form = form.."label[0.5,"..offset..";Side "..btnpref..": "..minetest.formspec_escape(ts.name or tcbs.ts_id).."]"
+ form = form.."button[0.5,"..(offset+0.5)..";5,1;"..btnpref.."_gotots;Show track section]"
else
tcbs.ts_id = nil
- form = form.."label[0.5,"..offset..";"..sidecolor.."Strana "..btnpref..": "..attrans("End of interlocking").."]"
- form = form.."button[0.5,"..(offset+0.5)..";5,1;"..btnpref.."_makeil;"..attrans("Create Interlocked Track Section").."]"
- --if tcbs.section_free then
- --form = form.."button[0.5,"..(offset+1.5)..";5,1;"..btnpref.."_setlocked;Section is free]"
- --else
- --form = form.."button[0.5,"..(offset+1.5)..";5,1;"..btnpref.."_setfree;Section is blocked]"
- --end
+ form = form.."label[0.5,"..offset..";Side "..btnpref..": ".."End of interlocking]"
+ form = form.."button[0.5,"..(offset+0.5)..";5,1;"..btnpref.."_makeil;Create Interlocked Track Section]"
+ end
+ -- xlink
+ if tcbs.xlink then
+ form = form.."label[0.5,"..(offset+1.5)..";Link:"..ildb.sigd_to_string(tcbs.xlink).."]"
+ form = form.."button[4.5,"..(offset+1.5)..";1,1;"..btnpref.."_xlinkdel;X]"
+ else
+ if players_assign_xlink[pname] then
+ form = form.."button[0.5,"..(offset+1.5)..";4,1;"..btnpref.."_xlinklink;Link "..ildb.sigd_to_string(players_assign_xlink[pname]).."]"
+ form = form.."button[4.5,"..(offset+1.5)..";1,1;"..btnpref.."_xlinkabrt;X]"
+ else
+ form = form.."label[0.5,"..(offset+1.5)..";No Link]"
+ form = form.."button[4.5,"..(offset+1.5)..";1,1;"..btnpref.."_xlinkadd;+]"
+ end
end
if tcbs.signal then
form = form.."button[0.5,"..(offset+2.5)..";5,1;"..btnpref.."_sigdia;"..attrans("Signalling").."]"
@@ -276,8 +424,8 @@ function advtrains.interlocking.show_tcb_form(pos, pname)
if not tcb then return end
local form = "size[6,9] label[0.5,0.5;"..attrans("Track Circuit Break Configuration").."]"
- form = form .. mktcbformspec(tcb[1], "A", 1, pname)
- form = form .. mktcbformspec(tcb[2], "B", 5, pname)
+ form = form .. mktcbformspec(pos, 1, tcb[1], 1, pname)
+ form = form .. mktcbformspec(pos, 2, tcb[2], 5, pname)
minetest.show_formspec(pname, "at_il_tcbconfig_"..minetest.pos_to_string(pos), form)
advtrains.interlocking.show_tcb_marker(pos)
@@ -304,13 +452,13 @@ minetest.register_on_player_receive_fields(function(player, formname, fields)
local tcb = ildb.get_tcb(pos)
if not tcb then return end
local f_gotots = {fields.A_gotots, fields.B_gotots}
- local f_update = {fields.A_update, fields.B_update}
- local f_remove = {fields.A_remove, fields.B_remove}
local f_makeil = {fields.A_makeil, fields.B_makeil}
- local f_setlocked = {fields.A_setlocked, fields.B_setlocked}
- local f_setfree = {fields.A_setfree, fields.B_setfree}
local f_asnsig = {fields.A_asnsig, fields.B_asnsig}
local f_sigdia = {fields.A_sigdia, fields.B_sigdia}
+ local f_xlinkadd = {fields.A_xlinkadd, fields.B_xlinkadd}
+ local f_xlinkdel = {fields.A_xlinkdel, fields.B_xlinkdel}
+ local f_xlinklink = {fields.A_xlinklink, fields.B_xlinklink}
+ local f_xlinkabrt = {fields.A_xlinkabrt, fields.B_xlinkabrt}
for connid=1,2 do
local tcbs = tcb[connid]
@@ -319,28 +467,33 @@ minetest.register_on_player_receive_fields(function(player, formname, fields)
advtrains.interlocking.show_ts_form(tcbs.ts_id, pname)
return
end
- if f_update[connid] then
- ildb.sync_tcb_neighbors(pos, connid)
- end
- if f_remove[connid] then
- ildb.remove_from_interlocking({p=pos, s=connid})
- end
else
if f_makeil[connid] then
- -- try sinc_tcb_neighbors first
- ildb.sync_tcb_neighbors(pos, connid)
- -- if that didn't work, create new section
if not tcbs.ts_id then
- ildb.create_ts({p=pos, s=connid})
- ildb.sync_tcb_neighbors(pos, connid)
+ ildb.create_ts_from_tcbs({p=pos, s=connid})
end
end
- -- non-interlocked
- if f_setfree[connid] then
- tcbs.section_free = true
+ end
+ if tcbs.xlink then
+ if f_xlinkdel[connid] then
+ ildb.remove_tcb_xlink({p=pos, s=connid})
end
- if f_setlocked[connid] then
- tcbs.section_free = nil
+ else
+ local osigd = players_assign_xlink[pname]
+ if osigd then
+ if f_xlinklink[connid] then
+ ildb.add_tcb_xlink({p=pos, s=connid}, osigd)
+ players_assign_xlink[pname] = nil
+ elseif f_xlinkabrt[connid] then
+ players_assign_xlink[pname] = nil
+ end
+ else
+ if f_xlinkadd[connid] then
+ players_assign_xlink[pname] = {p=pos, s=connid}
+ minetest.chat_send_player(pname, "TCB Link: Select linked TCB now!")
+ minetest.close_formspec(pname, formname)
+ return -- to not reopen form
+ end
end
end
if f_asnsig[connid] and not tcbs.signal then
@@ -364,10 +517,7 @@ end)
-- TS Formspec
--- textlist selection temporary storage
-local ts_pselidx = {}
-
-function advtrains.interlocking.show_ts_form(ts_id, pname, sel_tcb)
+function advtrains.interlocking.show_ts_form(ts_id, pname)
if not minetest.check_player_privs(pname, "interlocking") then
minetest.chat_send_player(pname, attrans("Insufficient privileges to use this!"))
return
@@ -386,29 +536,27 @@ function advtrains.interlocking.show_ts_form(ts_id, pname, sel_tcb)
advtrains.interlocking.show_tcb_marker(sigd.p)
end
- form = form.."textlist[0.5,3;5,3;tcblist;"..table.concat(strtab, ",").."]"
+ form = form.."label[0.5,2.5;Boundary TCBs:]"
+ form = form.."textlist[0.5,3;4,3;tcblist;"..table.concat(strtab, ",").."]"
+
+ -- additional route locks (e.g. for level crossings)
+
+ strtab = {}
+ if ts.fixed_locks then
+ for pts, state in pairs(ts.fixed_locks) do
+ strtab[#strtab+1] = minetest.formspec_escape(
+ minetest.pos_to_string(advtrains.decode_pos(pts)).." = "..state)
+ end
+ end
+ form = form.."label[5.5,2.5;Fixed route locks (e.g. level crossings):]"
+ form = form.."textlist[5.5,3;4,3;fixedlocks;"..table.concat(strtab, ",").."]"
if ildb.may_modify_ts(ts) then
+ form = form.."button[5.5,6;2,1;flk_add;Add locks]"
+ form = form.."button[7.5,6;2,1;flk_clear;Clear locks]"
- if players_link_ts[pname] then
- local other_id = players_link_ts[pname]
- local other_ts = ildb.get_ts(other_id)
- if other_ts then
- if ildb.may_modify_ts(other_ts) then
- form = form.."button[5.5,3;3.5,1;mklink;"..attrans("Join with @1", minetest.formspec_escape(other_ts.name)).."]"
- form = form.."button[9 ,3;0.5,1;cancellink;X]"
- end
- end
- else
- form = form.."button[5.5,3;4,1;link;"..attrans("Join into other section").."]"
- hint = 1
- end
- form = form.."button[5.5,4;4,1;dissolve;"..attrans("Dissolve Section").."]"
- form = form.."tooltip[dissolve;"..attrans("This will remove the track section and set all its end points to End Of Interlocking").."]"
- if sel_tcb then
- form = form.."button[5.5,5;4,1;del_tcb;"..attrans("Unlink selected TCB").."]"
- hint = 2
- end
+ form = form.."button[5.5,8;4,1;remove;Remove Section]"
+ form = form.."tooltip[remove;This will remove the track section and set all its end points to End Of Interlocking]"
else
hint=3
end
@@ -426,18 +574,13 @@ function advtrains.interlocking.show_ts_form(ts_id, pname, sel_tcb)
form = form.."label[0.5,7.1;"..attrans("No trains on this section.").."]"
end
- form = form.."button[5.5,7;4,1;reset;"..attrans("Reset section state").."]"
-
- if hint == 1 then
- form = form.."label[0.5,0.75;"..attrans("Use the 'Join' button to designate rail crosses and link not listed far-away TCBs").."]"
- elseif hint == 2 then
- form = form.."label[0.5,0.75;"..attrans("Unlinking a TCB will set it to non-interlocked mode.").."]"
- elseif hint == 3 then
- form = form.."label[0.5,0.75;"..attrans("You cannot modify track sections when a route is set or a train is on the section.").."]"
+ form = form.."button[5.5,7;4,1;reset;Reset section state]"
+
+ if hint == 3 then
+ form = form.."label[0.5,0.75;You cannot modify track sections when a route is set or a train is on the section.]"
--form = form.."label[0.5,1;Trying to unlink a TCB directly connected to this track will not work.]"
end
- ts_pselidx[pname]=sel_tcb
minetest.show_formspec(pname, "at_il_tsconfig_"..ts_id, form)
end
@@ -449,45 +592,15 @@ minetest.register_on_player_receive_fields(function(player, formname, fields)
return
end
-- independent of the formspec, clear this whenever some formspec event happens
- local tpsi = ts_pselidx[pname]
- ts_pselidx[pname] = nil
local ts_id = string.match(formname, "^at_il_tsconfig_(.+)$")
if ts_id and not fields.quit then
local ts = ildb.get_ts(ts_id)
if not ts then return end
- local sel_tcb
- if fields.tcblist then
- local tev = minetest.explode_textlist_event(fields.tcblist)
- sel_tcb = tev.index
- ts_pselidx[pname] = sel_tcb
- elseif tpsi then
- sel_tcb = tpsi
- end
-
if ildb.may_modify_ts(ts) then
- if players_link_ts[pname] then
- if fields.cancellink then
- players_link_ts[pname] = nil
- elseif fields.mklink then
- ildb.link_track_sections(players_link_ts[pname], ts_id)
- players_link_ts[pname] = nil
- end
- end
-
- if fields.del_tcb and sel_tcb and sel_tcb > 0 and sel_tcb <= #ts.tc_breaks then
- if not ildb.remove_from_interlocking(ts.tc_breaks[sel_tcb]) then
- minetest.chat_send_player(pname, attrans("Please unassign signal first!"))
- end
- sel_tcb = nil
- end
-
- if fields.link then
- players_link_ts[pname] = ts_id
- end
- if fields.dissolve then
- ildb.dissolve_ts(ts_id)
+ if fields.remove then
+ ildb.remove_ts(ts_id)
minetest.close_formspec(pname, formname)
return
end
@@ -496,10 +609,22 @@ minetest.register_on_player_receive_fields(function(player, formname, fields)
if fields.setname then
ts.name = fields.name
if ts.name == "" then
- ts.name = ts_id
+ ts.name = nil
end
end
+ if fields.flk_add then
+ if not ts.fixed_locks then
+ ts.fixed_locks = {}
+ end
+ players_assign_fixedlocks[pname] = ts_id
+ minetest.chat_send_player(pname, "Punch components to add fixed locks. (punch anything else = end)")
+ minetest.close_formspec(pname, formname)
+ return
+ elseif fields.flk_clear then
+ ts.fixed_locks = nil
+ end
+
if fields.reset then
-- User requested resetting the section
-- Show him what this means...
@@ -527,7 +652,7 @@ minetest.register_on_player_receive_fields(function(player, formname, fields)
ts.route = nil
for _, sigd in ipairs(ts.tc_breaks) do
local tcbs = ildb.get_tcbs(sigd)
- advtrains.interlocking.update_signal_aspect(tcbs)
+ advtrains.interlocking.signal.update_route_aspect(tcbs)
end
minetest.chat_send_player(pname, attrans("Reset track section @1!", ts_id))
end
@@ -559,6 +684,13 @@ minetest.register_entity("advtrains_interlocking:tcbmarker", {
on_activate = function(self, sdata) if sdata=="STATIC" then self.object:remove() end end,
})
+function advtrains.interlocking.remove_tcb_marker_pts(pts)
+ if markerent[pts] then
+ markerent[pts]:remove()
+ markerent[pts] = nil
+ end
+end
+
function advtrains.interlocking.show_tcb_marker(pos)
--atdebug("showing tcb marker",pos)
local tcb = ildb.get_tcb(pos)
@@ -575,16 +707,14 @@ function advtrains.interlocking.show_tcb_marker(pos)
ts = ildb.get_ts(tcbs.ts_id)
end
if ts then
- itex[connid] = ts.name
+ itex[connid] = ts.name or tcbs.ts_id or "???"
else
itex[connid] = "--EOI--"
end
end
local pts = advtrains.roundfloorpts(pos)
- if markerent[pts] then
- markerent[pts]:remove()
- end
+ advtrains.interlocking.remove_tcb_marker_pts(pts)
local obj = minetest.add_entity(pos, "advtrains_interlocking:tcbmarker")
if not obj then return end
@@ -598,6 +728,88 @@ function advtrains.interlocking.show_tcb_marker(pos)
markerent[pts] = obj
end
+function advtrains.interlocking.remove_tcb_marker(pos)
+ local pts = advtrains.roundfloorpts(pos)
+ if markerent[pts] then
+ markerent[pts]:remove()
+ end
+ markerent[pts] = nil
+end
+
+local ts_showparticles_callback = function(pos, connid, bconnid)
+ minetest.add_particle({
+ pos = pos,
+ velocity = {x=0, y=0, z=0},
+ acceleration = {x=0, y=0, z=0},
+ expirationtime = 10,
+ size = 7,
+ vertical = true,
+ texture = "at_il_ts_highlight_particle.png",
+ glow = 6,
+ })
+end
+
+-- Spawns particles to highlight the clicked track section
+-- TODO: Adapt behavior to not dumb-walk anymore
+function advtrains.interlocking.highlight_track_section(pos)
+ local all_tcbs = ildb.get_all_tcbs_adjacent(pos, nil, ts_showparticles_callback)
+ for _,sigd in ipairs(all_tcbs) do
+ advtrains.interlocking.show_tcb_marker(sigd.p)
+ end
+end
+
+-- checks that the given route is still valid (i.e. all its TCBs, sections and locks exist)
+-- returns true (ok) or false, reason (on issue)
+function advtrains.interlocking.check_route_valid(route, sigd)
+ -- this code is partially copy-pasted from routesetting.lua
+ -- we start at the tc designated by signal
+ local c_sigd = sigd
+ local i = 1
+ local c_tcbs, c_ts_id, c_ts, c_rseg
+ while c_sigd and i<=#route do
+ c_tcbs = ildb.get_tcbs(c_sigd)
+ if not c_tcbs then
+ return false, "No TCBS at "..sigd_to_string(c_sigd)
+ end
+ c_ts_id = c_tcbs.ts_id
+ if not c_ts_id then
+ return false, "No track section adjacent to "..sigd_to_string(c_sigd)
+ end
+ c_ts = ildb.get_ts(c_ts_id)
+
+ c_rseg = route[i]
+
+ if c_rseg.locks then
+ for pts, state in pairs(c_rseg.locks) do
+ local pos = advtrains.decode_pos(pts)
+ if not advtrains.is_passive(pos) then
+ return false, "No passive component for lock at "..pts
+ end
+ end
+ end
+ -- sanity check, is section at next the same as the current?
+ local nvar = c_rseg.next
+ if nvar then
+ local re_tcbs = ildb.get_tcbs({p = nvar.p, s = (nvar.s==1) and 2 or 1})
+ if not re_tcbs or not re_tcbs.ts_id or re_tcbs.ts_id~=c_ts_id then
+ return false, "TCB at "..minetest.pos_to_string(nvar.p).." has different section than previous TCB."
+ end
+ end
+ -- advance
+ c_sigd = nvar
+ i = i + 1
+ end
+ -- check end TCB
+ if not c_sigd then
+ return false, "Final TCBS unset (legacy-style buffer route)"
+ end
+ c_tcbs = ildb.get_tcbs(c_sigd)
+ if not c_tcbs then
+ return false, "Final TCBS missing at "..sigd_to_string(c_sigd)
+ end
+ return true, nil, c_sigd
+end
+
-- Signalling formspec - set routes a.s.o
-- textlist selection temporary storage
@@ -614,7 +826,6 @@ function advtrains.interlocking.show_signalling_form(sigd, pname, sel_rte, calle
local tcbs = ildb.get_tcbs(sigd)
if not tcbs.signal then return end
- if not tcbs.signal_name then tcbs.signal_name = "Signal at "..minetest.pos_to_string(sigd.p) end
if not tcbs.routes then tcbs.routes = {} end
local form = "size[7,10]label[0.5,0.5;"..attrans("Signal at @1", minetest.pos_to_string(sigd.p)).."]"
@@ -649,42 +860,76 @@ function advtrains.interlocking.show_signalling_form(sigd, pname, sel_rte, calle
form = form.."button[0.5,6; 5,1;cancelroute;"..attrans("Cancel Route").."]"
else
if not tcbs.route_origin then
- local strtab = {}
- for idx, route in ipairs(tcbs.routes) do
- local clr = ""
- if route.ars then
- clr = "#FF5555"
- if route.ars.default then
- clr = "#55FF55"
+ if #tcbs.routes > 0 then
+ -- at least one route is defined, show normal dialog
+ local strtab = {}
+ for idx, route in ipairs(tcbs.routes) do
+ local rname = route.name
+ local valid = atil.check_route_valid(route, sigd)
+ local clr = ""
+ if not valid then
+ clr = "#FF5555"
+ rname = rname.." (invalid)"
+ elseif route.ars then
+ clr = "#FFFF55"
+ if route.ars.default then
+ clr = "#55FF55"
+ end
+ end
+ strtab[#strtab+1] = clr .. minetest.formspec_escape(rname)
+ end
+ form = form.."label[0.5,2.5;Routes:]"
+ form = form.."textlist[0.5,3;5,3;rtelist;"..table.concat(strtab, ",")
+ if sel_rte then
+ form = form .. ";" .. sel_rte .."]"
+ form = form.."button[0.5,6; 5,1;setroute;Set Route]"
+ form = form.."button[0.5,7;2,1;dsproute;Show]"
+ if hasprivs then
+ form = form.."button[5.5,3.3;1,0.3;setarsdefault;D]tooltip[setarsdefault;Set ARS default route]"
+ form = form.."button[3.5,7;2,1;editroute;Edit]"
+ if sel_rte > 1 then
+ form = form .. "button[5.5,4;1,0.3;moveup;↑]"
+ end
+ if sel_rte < #strtab then
+ form = form .. "button[5.5,4.7;1,0.3;movedown;↓]"
+ end
+ form = form.."button[5.5,5.4;1,0.3;delroute;X]tooltip[delroute;Delete this route]"
+ end
+ else
+ form = form .. "]"
+ if tcbs.ars_disabled then
+ form = form.."label[0.5,6 ;NOTE: ARS is disabled.]"
+ form = form.."label[0.5,6.5;Routes are not automatically set.]"
end
end
- strtab[#strtab+1] = clr .. minetest.formspec_escape(route.name)
- end
- form = form.."label[0.5,2.5;"..attrans("Routes:").."]"
- form = form.."textlist[0.5,3;5,3;rtelist;"..table.concat(strtab, ",").."]"
- if sel_rte then
- form = form.."button[0.5,6;5,1;setroute;"..attrans("Set Route").."]"
- form = form.."button[0.5,7;2,1;dsproute;"..attrans("Show").."]"
if hasprivs then
- form = form.."button[3.5,7;2,1;editroute;"..attrans("Edit").."]"
- -- form = form.."button[5.75,3;1,1;routeup;^]"
+ form = form.."button[0.5,8;2.5,1;smartroute;Smart Route]"
+ form = form.."button[ 3,8;2.5,1;newroute;New (Manual)]"
+ form = form..string.format("checkbox[0.5,8.75;ars;Automatic routesetting;%s]", not tcbs.ars_disabled)
+ form = form..string.format("checkbox[0.5,9.25;dstarstrig;Distant signal triggers ARS;%s]", not tcbs.no_dst_ars_trig)
end
else
- if tcbs.ars_disabled then
- form = form.."label[0.5,6 ;"..attrans("NOTE: ARS is disabled.").."]"
- form = form.."label[0.5,6.5;"..attrans("Routes are not automatically set.").."]"
+ -- no route is active, and no route is so far defined
+ if not tcbs.signal then atwarn("signalling form missing signal?!", pos) return end -- safeguard, nothing else in this function checks tcbs.signal
+ local caps = advtrains.interlocking.signal.get_signal_cap_level(tcbs.signal)
+ if caps >= 4 then
+ -- offer user the "block signal mode"
+ form = form.."label[0.5,2.5;No routes are yet defined.]"
+ if hasprivs then
+ form = form.."button[0.5,4;2.5,1;smartroute;Smart Route]"
+ form = form.."button[ 3,4;2.5,1;newroute;New (Manual)]"
+ end
+ elseif caps >= 3 then
+ -- it's a buffer!
+ form = form.."label[0.5,2.5;This is an always-halt signal (e.g. a buffer)\n"
+ .."No routes can be set from here.]"
+ else
+ -- signal caps say it cannot be route start/end
+ form = form.."label[0.5,2.5;This is a pure distant signal\n"
+ .."No route is currently set through.]"
end
end
- if hasprivs then
- form = form.."button[0.5,8;2.5,1;newroute;"..attrans("New Route").."]"
- form = form.."button[ 3,8;2.5,1;unassign;"..attrans("Unassign Signal").."]"
- form = form.."button[ 3,9;2.5,1;influp;"..attrans("Influence Point").."]"
- end
- if tcbs.ars_disabled then
- form = form.."button[0.5,9;2.5,1;arsenable;"..attrans("Enable ARS").."]"
- else
- form = form.."button[0.5,9;2.5,1;arsdisable;"..attrans("Disable ARS").."]"
- end
+
elseif sigd_equal(tcbs.route_origin, sigd) then
-- something has gone wrong: tcbs.routeset should have been set...
form = form.."label[0.5,2.5;"..attrans("Inconsistent state: route_origin is same TCBS but no route set. Try again.").."]"
@@ -701,14 +946,14 @@ function advtrains.interlocking.show_signalling_form(sigd, pname, sel_rte, calle
-- always a good idea to update the signal aspect
if not called_from_form_update then
-- FIX prevent a callback loop
- advtrains.interlocking.update_signal_aspect(tcbs)
+ advtrains.interlocking.signal.update_route_aspect(tcbs)
end
end
function advtrains.interlocking.update_player_forms(sigd)
for pname, tsigd in pairs(p_open_sig_form) do
if advtrains.interlocking.sigd_equal(sigd, tsigd) then
- advtrains.interlocking.show_signalling_form(sigd, pname, nil)
+ advtrains.interlocking.show_signalling_form(sigd, pname, nil, true)
end
end
end
@@ -721,10 +966,7 @@ minetest.register_on_player_receive_fields(function(player, formname, fields)
end
local hasprivs = minetest.check_player_privs(pname, "interlocking")
- -- independent of the formspec, clear this whenever some formspec event happens
local tpsi = sig_pselidx[pname]
- sig_pselidx[pname] = nil
- p_open_sig_form[pname] = nil
local pts, connids = string.match(formname, "^at_il_signalling_([^_]+)_(%d)$")
local pos, connid
@@ -739,6 +981,8 @@ minetest.register_on_player_receive_fields(function(player, formname, fields)
if not tcbs then return end
if fields.quit then
+ sig_pselidx[pname] = nil
+ p_open_sig_form[pname] = nil
-- form quit: disable temporary ARS ignore
tcbs.ars_ignore_next = nil
return
@@ -747,12 +991,18 @@ minetest.register_on_player_receive_fields(function(player, formname, fields)
local sel_rte
if fields.rtelist then
local tev = minetest.explode_textlist_event(fields.rtelist)
- sel_rte = tev.index
+ if tev.type ~= "INV" then
+ sel_rte = tev.index
+ end
elseif tpsi then
sel_rte = tpsi
end
if fields.setname and fields.name and hasprivs then
- tcbs.signal_name = fields.name
+ if fields.name == "" then
+ tcbs.signal_name = nil -- do not save a signal name if it isnt used (equivalent to track sections)
+ else
+ tcbs.signal_name = fields.name
+ end
end
if tcbs.routeset and fields.cancelroute then
if tcbs.routes[tcbs.routeset] and tcbs.routes[tcbs.routeset].ars then
@@ -768,6 +1018,11 @@ minetest.register_on_player_receive_fields(function(player, formname, fields)
tcbs.ars_ignore_next = nil
return
end
+ if fields.smartroute and hasprivs then
+ advtrains.interlocking.smartroute.start(pname, sigd)
+ tcbs.ars_ignore_next = nil
+ return
+ end
if sel_rte and tcbs.routes[sel_rte] then
if fields.setroute then
ilrs.update_route(sigd, tcbs, sel_rte)
@@ -779,41 +1034,45 @@ minetest.register_on_player_receive_fields(function(player, formname, fields)
end
if fields.editroute and hasprivs then
advtrains.interlocking.show_route_edit_form(pname, sigd, sel_rte)
- --local rte = tcbs.routes[sel_rte]
- --minetest.show_formspec(pname, formname.."_renroute_"..sel_rte, "field[name;Enter new route name;"..rte.name.."]")
return
end
+ if fields.setarsdefault and hasprivs then
+ for rid, route in ipairs(tcbs.routes) do
+ local isdefault = rid == sel_rte
+ if route.ars then
+ if route.ars.default and isdefault then
+ -- D button pressed but route was already default - remove ars default field!
+ route.ars.default = nil
+ elseif isdefault then
+ route.ars.default = true
+ else
+ route.ars.default = nil
+ end
+ -- if the table is nouw empty delete it
+ if not next(route.ars) then
+ route.ars = nil
+ end
+ elseif isdefault then
+ route.ars = {default = true}
+ end
+ end
+ end
+ if fields.delroute and hasprivs then
+ if tcbs.routes[sel_rte] and tcbs.routes[sel_rte].ars then
+ minetest.chat_send_player(pname, "Cannot delete route which has ARS rules, please review and then delete through edit dialog!")
+ else
+ table.remove(tcbs.routes,sel_rte)
+ end
+ end
end
end
- if fields.unassign and hasprivs then
- -- unassigning the signal from the tcbs
- -- only when no route is set.
- -- Routes and name remain saved, in case the player wants to reassign a new signal
- if not tcbs.routeset then
- local signal_pos = tcbs.signal
- ildb.set_sigd_for_signal(signal_pos, nil)
- tcbs.signal = nil
- tcbs.aspect = nil
- minetest.close_formspec(pname, formname)
- minetest.chat_send_player(pname, attrans("Signal has been unassigned. Name and routes are kept for reuse."))
- return
- else
- minetest.chat_send_player(pname, attrans("Please cancel route first!"))
- end
- end
- if fields.influp and hasprivs then
- advtrains.interlocking.show_ip_form(tcbs.signal, pname)
- return
+ if fields.ars then
+ tcbs.ars_disabled = not minetest.is_yes(fields.ars)
end
- if tcbs.ars_disabled and fields.arsenable then
- core.log("action", pname.." turn ARS on at "..pts)
- tcbs.ars_disabled = nil
- end
- if not tcbs.ars_disabled and fields.arsdisable then
- core.log("action", pname.." turn ARS off at "..pts)
- tcbs.ars_disabled = true
+ if fields.dstarstrig then
+ tcbs.no_dst_ars_trig = not minetest.is_yes(fields.dstarstrig)
end
if fields.auto then
@@ -822,28 +1081,22 @@ minetest.register_on_player_receive_fields(function(player, formname, fields)
if fields.noauto then
tcbs.route_auto = false
end
+
+ if sel_rte and tcbs.routes[sel_rte]and not tcbs.routeset then
+ if fields.moveup then
+ if tcbs.routes[sel_rte - 1] then
+ tcbs.routes[sel_rte - 1], tcbs.routes[sel_rte] = tcbs.routes[sel_rte], tcbs.routes[sel_rte - 1]
+ sel_rte = sel_rte - 1
+ end
+ elseif fields.movedown then
+ if tcbs.routes[sel_rte + 1] then
+ tcbs.routes[sel_rte + 1], tcbs.routes[sel_rte] = tcbs.routes[sel_rte], tcbs.routes[sel_rte + 1]
+ sel_rte = sel_rte + 1
+ end
+ end
+ end
advtrains.interlocking.show_signalling_form(sigd, pname, sel_rte, true)
return
end
-
-
- if not hasprivs then return end
- -- rename route
- local rind, rte_id
- pts, connids, rind = string.match(formname, "^at_il_signalling_([^_]+)_(%d)_renroute_(%d+)$")
- if pts then
- pos = minetest.string_to_pos(pts)
- connid = tonumber(connids)
- rte_id = tonumber(rind)
- if not connid or connid<1 or connid>2 then return end
- end
- if pos and connid and rind and fields.name then
- local sigd = {p=pos, s=connid}
- local tcbs = ildb.get_tcbs(sigd)
- if tcbs.routes[rte_id] then
- tcbs.routes[rte_id].name = fields.name
- advtrains.interlocking.show_signalling_form(sigd, pname)
- end
- end
end)
diff --git a/advtrains_interlocking/textures/at_il_ts_highlight_particle.png b/advtrains_interlocking/textures/at_il_ts_highlight_particle.png
new file mode 100644
index 0000000..4ba3622
--- /dev/null
+++ b/advtrains_interlocking/textures/at_il_ts_highlight_particle.png
Binary files differ
diff --git a/advtrains_interlocking/tool.lua b/advtrains_interlocking/tool.lua
index 82b295e..7ccab46 100644
--- a/advtrains_interlocking/tool.lua
+++ b/advtrains_interlocking/tool.lua
@@ -3,14 +3,78 @@
local ilrs = advtrains.interlocking.route
+local function node_right_click(pos, pname, player)
+ if advtrains.is_passive(pos) then
+ local form = "size[7,5]label[0.5,0.5;Route lock inspector]"
+ local pts = advtrains.encode_pos(pos)
+
+ local rtl = ilrs.has_route_lock(pts)
+
+ if rtl then
+ form = form.."label[0.5,1;Route locks currently put:\n"..rtl.."]"
+ form = form.."button_exit[0.5,3.5; 5,1;clear;Clear]"
+ else
+ form = form.."label[0.5,1;No route locks set]"
+ form = form.."button_exit[0.5,3.5; 5,1;emplace;Emplace manual lock]"
+ end
+
+ minetest.show_formspec(pname, "at_il_rtool_"..pts, form)
+ return
+ end
+
+ -- If not a turnout, check the track section and show a form
+ local node_ok, conns, rail_y=advtrains.get_rail_info_at(pos)
+ if not node_ok then
+ minetest.chat_send_player(pname, "Node is not a track!")
+ return
+ end
+ if advtrains.interlocking.db.get_tcb(pos) then
+ advtrains.interlocking.show_tcb_form(pos, pname)
+ return
+ end
+
+ local ts_id = advtrains.interlocking.db.check_and_repair_ts_at_pos(pos, nil, pname)
+ if ts_id then
+ advtrains.interlocking.show_ts_form(ts_id, pname)
+ else
+ minetest.chat_send_player(pname, "No track section at this location!")
+ end
+end
+
+local function node_left_click(pos, pname, player)
+ local node_ok, conns, rail_y=advtrains.get_rail_info_at(pos)
+ if not node_ok then
+ minetest.chat_send_player(pname, "Node is not a track!")
+ return
+ end
+
+ if advtrains.interlocking.db.get_tcb(pos) then
+ advtrains.interlocking.show_tcb_marker(pos)
+ return
+ end
+
+ -- create track section if aux1 button down
+ local pc = player:get_player_control()
+ local force_create = pc.aux1
+
+ local ts_id = advtrains.interlocking.db.check_and_repair_ts_at_pos(pos, nil, pname, force_create)
+ if ts_id then
+ advtrains.interlocking.db.update_rs_cache(ts_id)
+ advtrains.interlocking.highlight_track_section(pos)
+ else
+ minetest.chat_send_player(pname, "No track section at this location!")
+ end
+end
+
+
minetest.register_craftitem("advtrains_interlocking:tool",{
- description = attrans("Interlocking tool\nright-click turnouts to inspect route locks"),
+ description = "Interlocking tool\nPunch: Highlight track section\nPlace: check route locks/show track section info",
groups = {cracky=1}, -- key=name, value=rating; rating=1..3.
inventory_image = "at_il_tool.png",
wield_image = "at_il_tool.png",
stack_max = 1,
- on_place = function(itemstack, placer, pointed_thing)
- local pname = placer:get_player_name()
+ on_place = function(itemstack, player, pointed_thing)
+ local pname = player:get_player_name()
if not pname then
return
end
@@ -20,36 +84,23 @@ minetest.register_craftitem("advtrains_interlocking:tool",{
end
if pointed_thing.type=="node" then
local pos=pointed_thing.under
- if advtrains.is_passive(pos) then
- local form = "size[7,5]label[0.5,0.5;" .. attrans("Route lock inspector") .. "]"
- local pts = minetest.pos_to_string(pos)
-
- local rtl = ilrs.has_route_lock(pts)
-
- if rtl then
- form = form.."label[0.5,1;" .. attrans("Route locks currently put:") .. "\n"..rtl.."]"
- form = form.."button_exit[0.5,3.5; 5,1;clear;" .. attrans("Clear") .. "]"
- else
- form = form.."label[0.5,1;" .. attrans("No route locks set") .. "]"
- form = form.."button_exit[0.5,3.5; 5,1;emplace;" .. attrans("Emplace manual lock") .. "]"
- end
-
- minetest.show_formspec(pname, "at_il_rtool_"..pts, form)
- else
- minetest.chat_send_player(pname, attrans("Cannot use this here."))
- return
- end
+ node_right_click(pos, pname, player)
end
end,
-})
-
-minetest.register_craft({
- output = "advtrains_interlocking:tool",
- recipe = {
- {"default:diamond", "", ""},
- {"ch_extras:lupa", "", ""},
- {"default:steel_ingot", "", ""},
- },
+ on_use = function(itemstack, player, pointed_thing)
+ local pname = player:get_player_name()
+ if not pname then
+ return
+ end
+ if not minetest.check_player_privs(pname, {interlocking=true}) then
+ minetest.chat_send_player(pname, "Insufficient privileges to use this!")
+ return
+ end
+ if pointed_thing.type=="node" then
+ local pos=pointed_thing.under
+ node_left_click(pos, pname, player)
+ end
+ end
})
minetest.register_on_player_receive_fields(function(player, formname, fields)
@@ -60,7 +111,7 @@ minetest.register_on_player_receive_fields(function(player, formname, fields)
local pos
local pts = string.match(formname, "^at_il_rtool_(.+)$")
if pts then
- pos = minetest.string_to_pos(pts)
+ pos = advtrains.decode_pos(pts)
end
if pos then
if advtrains.is_passive(pos) then
diff --git a/advtrains_interlocking/train_sections.lua b/advtrains_interlocking/train_sections.lua
index 757f36a..a1ee87f 100644
--- a/advtrains_interlocking/train_sections.lua
+++ b/advtrains_interlocking/train_sections.lua
@@ -40,83 +40,99 @@ local function itkexist(tbl, ikey, com)
return false
end
-local function itremove(tbl, com)
+local function itremove(tbl, com, once)
local i=1
while i <= #tbl do
if tbl[i] == com then
table.remove(tbl, i)
+ if once then return end
else
i = i + 1
end
end
end
-local function itkremove(tbl, ikey, com)
+local function itkremove(tbl, ikey, com, once)
local i=1
while i <= #tbl do
if tbl[i][ikey] == com then
table.remove(tbl, i)
+ if once then return end
else
i = i + 1
end
end
end
-local function setsection(tid, train, ts_id, ts, sigd)
+local function setsection(tid, train, ts_id, ts, sigd, only_if_not_exist)
-- train
if not train.il_sections then train.il_sections = {} end
- if not itkexist(train.il_sections, "ts_id", ts_id) then
+ if only_if_not_exist then
+ -- called for the back connid on enter, only to ensure that section is blocked if train was so far not registered
+ if not itkexist(train.il_sections, "ts_id", ts_id) then
+ table.insert(train.il_sections, {ts_id = ts_id, origin = sigd})
+ end
+ else
+ -- insert always, this leads to duplicate entries if the train enters the same section a second time
table.insert(train.il_sections, {ts_id = ts_id, origin = sigd})
end
-- ts
if not ts.trains then ts.trains = {} end
- if not itexist(ts.trains, tid) then
+ if only_if_not_exist then
+ -- called for the back connid on enter, only to ensure that section is blocked if train was so far not registered
+ if not itexist(ts.trains, tid) then
+ table.insert(ts.trains, tid)
+ end
+ else
table.insert(ts.trains, tid)
end
-- routes
- local tcbs = advtrains.interlocking.db.get_tcbs(sigd)
+ local tcbs
+ if sigd then
+ tcbs = advtrains.interlocking.db.get_tcbs(sigd)
+ end
-- route setting - clear route state
if ts.route then
--atdebug(tid,"enters",ts_id,"examining Routestate",ts.route)
- if not sigd_equal(ts.route.entry, sigd) then
+ if sigd and not sigd_equal(ts.route.entry, sigd) then
-- Train entered not from the route. Locate origin and cancel route!
atwarn("Train",tid,"hit route",ts.route.rsn,"!")
advtrains.interlocking.route.cancel_route_from(ts.route.origin)
atwarn("Route was cancelled.")
else
- -- train entered route regularily. Reset route and signal
- tcbs.route_committed = nil
- tcbs.route_comitted = nil -- TODO compatibility cleanup
- tcbs.aspect = nil
- tcbs.route_origin = nil
- advtrains.interlocking.update_signal_aspect(tcbs)
- if tcbs.signal and sigd_equal(ts.route.entry, ts.route.origin) then
- if tcbs.route_auto and tcbs.routeset then
- --atdebug("Resetting route (",ts.route.origin,")")
- advtrains.interlocking.route.update_route(ts.route.origin, tcbs)
- else
- tcbs.routeset = nil
- end
- end
+ -- train entered route regularily.
end
ts.route = nil
end
- if tcbs.signal then
+ if tcbs and tcbs.signal then
+ -- Reset route and signal
+ -- Note that the hit-route case is already handled by cancel_route_from
+ -- this code only handles signal at entering tcb and also triggers for non-route ts
+ tcbs.route_committed = nil
+ tcbs.route_aspect = nil
+ tcbs.route_remote = nil
+ tcbs.route_origin = nil
+ tcbs.route_rsn = nil
+ if not tcbs.route_auto then
+ tcbs.routeset = nil
+ end
+ advtrains.interlocking.signal.update_route_aspect(tcbs)
advtrains.interlocking.route.update_route(sigd, tcbs)
end
end
-local function freesection(tid, train, ts_id, ts)
+local function freesection(tid, train, ts_id, ts, clear_all)
-- train
if not train.il_sections then train.il_sections = {} end
- itkremove(train.il_sections, "ts_id", ts_id)
+ itkremove(train.il_sections, "ts_id", ts_id, not clear_all)
-- ts
if not ts.trains then ts.trains = {} end
- itremove(ts.trains, tid)
+ itremove(ts.trains, tid, not clear_all)
+ -- route locks
if ts.route_post then
advtrains.interlocking.route.free_route_locks(ts_id, ts.route_post.locks)
if ts.route_post.next then
@@ -138,12 +154,18 @@ end
-- This sets the section for both directions, to be failsafe
advtrains.tnc_register_on_enter(function(pos, id, train, index)
local tcb = ildb.get_tcb(pos)
- if tcb then
- for connid=1,2 do
- local ts = tcb[connid].ts_id and ildb.get_ts(tcb[connid].ts_id)
- if ts then
- setsection(id, train, tcb[connid].ts_id, ts, {p=pos, s=connid})
- end
+ if tcb and train.path_cp[index] and train.path_cn[index] then
+ -- forward conn
+ local connid = train.path_cn[index]
+ local ts = tcb[connid] and tcb[connid].ts_id and ildb.get_ts(tcb[connid].ts_id)
+ if ts then
+ setsection(id, train, tcb[connid].ts_id, ts, {p=pos, s=connid})
+ end
+ -- backward conn (safety only)
+ connid = train.path_cp[index]
+ ts = tcb[connid] and tcb[connid].ts_id and ildb.get_ts(tcb[connid].ts_id)
+ if ts then
+ setsection(id, train, tcb[connid].ts_id, ts, {p=pos, s=connid}, true)
end
end
end)
@@ -153,8 +175,9 @@ end)
advtrains.tnc_register_on_leave(function(pos, id, train, index)
local tcb = ildb.get_tcb(pos)
if tcb and train.path_cp[index] then
+ -- backward conn
local connid = train.path_cp[index]
- local ts = tcb[connid].ts_id and ildb.get_ts(tcb[connid].ts_id)
+ local ts = tcb[connid] and tcb[connid].ts_id and ildb.get_ts(tcb[connid].ts_id)
if ts then
freesection(id, train, tcb[connid].ts_id, ts)
end
@@ -167,13 +190,13 @@ advtrains.te_register_on_create(function(id, train)
-- let's see what track sections we find here
local index = atround(train.index)
local pos = advtrains.path_get(train, index)
- local ts_id, origin = ildb.get_ts_at_pos(pos)
+ local ts_id = ildb.check_and_repair_ts_at_pos(pos, 1) -- passing connid 1 - that always exists
if ts_id then
local ts = ildb.get_ts(ts_id)
if ts then
- setsection(id, train, ts_id, ts, origin)
+ setsection(id, train, ts_id, ts, nil, true)
else
- atwarn("ILDB corruption: TCB",origin," has invalid TS reference")
+ atwarn("While placing train, TS didnt exist ",ts_id)
end
-- Make train a shunt move
train.is_shunt = true
diff --git a/advtrains_interlocking/tsr_rail.lua b/advtrains_interlocking/tsr_rail.lua
index 7ad6a7a..9fdc3d9 100644
--- a/advtrains_interlocking/tsr_rail.lua
+++ b/advtrains_interlocking/tsr_rail.lua
@@ -3,13 +3,15 @@
-- Simple rail whose only purpose is to place a TSR on the position, as a temporary solution until the timetable system covers everything.
-- This code resembles the code in lines/stoprail.lua
+local S = attrans
+
local function updateform(pos)
local meta = minetest.get_meta(pos)
local pe = advtrains.encode_pos(pos)
local npr = advtrains.interlocking.npr_rails[pe] or 2
- meta:set_string("infotext", attrans("Point speed restriction") .. ": "..npr)
- meta:set_string("formspec", "field[npr;"..attrans("Set point speed restriction:")..";"..npr.."]")
+ meta:set_string("infotext", S("Point speed restriction: @1",npr))
+ meta:set_string("formspec", "field[npr;"..S("Set point speed restriction:")..";"..npr.."]")
end
@@ -25,11 +27,11 @@ local adefunc = function(def, preset, suffix, rotation)
on_receive_fields = function(pos, formname, fields, player)
local pname = player:get_player_name()
if not minetest.check_player_privs(pname, {interlocking=true}) then
- minetest.chat_send_player(pname, attrans("Interlocking privilege required!"))
+ minetest.chat_send_player(pname, S("You are not allowed to configure this track without the @1 privilege.", "interlocking"))
return
end
if minetest.is_protected(pos, pname) then
- minetest.chat_send_player(pname, attrans("This rail is protected!"))
+ minetest.chat_send_player(pname, S("You are not allowed to configure this track."))
minetest.record_protection_violation(pos, pname)
return
end
@@ -59,7 +61,7 @@ if minetest.get_modpath("advtrains_train_track") ~= nil then
models_prefix="advtrains_dtrack",
models_suffix=".b3d",
shared_texture="advtrains_dtrack_shared_npr.png",
- description=attrans("Point Speed Restriction Rail"),
+ description=S("Point Speed Restriction Track"),
formats={},
get_additional_definiton = adefunc,
}, advtrains.trackpresets.t_30deg_straightonly)
diff --git a/advtrains_line_automation/init.lua b/advtrains_line_automation/init.lua
index a54fdfe..65631a9 100644
--- a/advtrains_line_automation/init.lua
+++ b/advtrains_line_automation/init.lua
@@ -1,5 +1,6 @@
ch_base.open_mod(minetest.get_current_modname())
-- Advtrains line automation system
+
advtrains.lines = {
-- [station code] = {name=..., owner=...}
stations = {},
diff --git a/advtrains_signals_japan/.gitignore b/advtrains_signals_japan/.gitignore
new file mode 100644
index 0000000..0969b6b
--- /dev/null
+++ b/advtrains_signals_japan/.gitignore
@@ -0,0 +1 @@
+models/*
diff --git a/advtrains_signals_japan/init.lua b/advtrains_signals_japan/init.lua
new file mode 100644
index 0000000..1140b6b
--- /dev/null
+++ b/advtrains_signals_japan/init.lua
@@ -0,0 +1,461 @@
+local pole_texture = "advtrains_signals_japan_mast.png"
+local signal_face_texture = "advtrains_hud_bg.png^[colorize:#000000:255"
+local pole_radius = 1/16
+local pole_box = {-pole_radius,-1/2,-pole_radius,pole_radius,1/2,pole_radius}
+local light_radius = 1/20
+local signal_width = 6*light_radius
+local signal_thickness = pole_radius*3
+local signal_height = {}
+local signal_box = {}
+local light_red = "advtrains_hud_bg.png^[colorize:red:255"
+local light_yellow = "advtrains_hud_bg.png^[colorize:orange:255"
+local light_green = "advtrains_hud_bg.png^[colorize:lime:255"
+local light_purple = "advtrains_hud_bg.png^[colorize:purple:255"
+local light_distant = light_purple
+local light_off = signal_face_texture
+
+do
+ local model_path_prefix = table.concat({minetest.get_modpath("advtrains_signals_japan"), "models", "advtrains_signals_japan_"}, DIR_DELIM)
+
+ local function vertex(x, y, z)
+ return string.format("v %f %f %f", x, y, z)
+ end
+ local function texture(u, v)
+ return string.format("vt %f %f", u, v)
+ end
+ local function face_element(v, vt)
+ if vt then
+ return string.format("%d/%d", v, vt)
+ end
+ return tonumber(v)
+ end
+ local function face_elements(...)
+ local st = {"f"}
+ local args = {...}
+ local len = #args
+ for i = 1, len, 2 do
+ st[(i+3)/2] = face_element(args[i], args[i+1])
+ end
+ return table.concat(st, " ")
+ end
+ local function sequential_elements(v0, vt0, count)
+ local st = {}
+ for i = 1, count do
+ st[i] = face_element(v0+i, vt0+i)
+ end
+ return table.concat(st, " ")
+ end
+ local function mod_lower(min, a, b)
+ return min + (a-min)%b
+ end
+ local function connect_circular(v0, vt0, count)
+ return "f " .. sequential_elements(v0, vt0, count)
+ end
+ local function connect_cylindrical(v0, vt0, count)
+ local st = {}
+ for i = 0, count-1 do
+ local j = (i+1)%count
+ local v1 = v0+i+1
+ local v2 = v1+count
+ local v3 = v0+j+1
+ local v4 = v3+count
+ local vt1 = vt0+i+1
+ local vt2 = vt1+count+1
+ st[i+1] = face_elements(v1, vt1, v3, vt1+1, v4, vt2+1, v2, vt2)
+ end
+ return table.concat(st, "\n")
+ end
+ local function circular_textures(u0, v0, r, count, total, angular_offset, direction)
+ local st = {}
+ if not angular_offset then
+ angular_offset = 0
+ end
+ if not total then
+ total = count
+ end
+ if not direction then
+ direction = 1
+ end
+ for i = 0, count-1 do
+ local theta = angular_offset + direction*i/total*2*math.pi
+ local u, v = r*math.cos(theta), r*math.sin(theta)
+ st[i+1] = texture(u0+u, v0+v)
+ end
+ return table.concat(st, "\n")
+ end
+ local function rectangular_textures(u0, v0, u1, v1, count)
+ local st = {}
+ local width = u1-u0
+ for i = 0, count do
+ local u = u0+i/count*width
+ st[i+1] = texture(u, v0)
+ st[i+count+2] = texture(u, v1)
+ end
+ return table.concat(st, "\n")
+ end
+
+ -- generate pole model
+ local pole_npolygon = 32
+ local pole_vertex_count = pole_npolygon*2
+ local pole_uv_count = pole_npolygon*3+2
+ local pole_vertices = {}
+ local pole_objdef = {
+ "g pole",
+ "usemtl pole",
+ connect_circular(0, 0, pole_npolygon),
+ connect_circular(pole_npolygon, 0, pole_npolygon),
+ connect_cylindrical(0, pole_npolygon, pole_npolygon),
+ }
+ local pole_uv = {
+ circular_textures(0.5, 0.5, 0.5, pole_npolygon),
+ rectangular_textures(0, 0, 1, 1, pole_npolygon),
+ }
+ for i = 0, pole_npolygon-1 do
+ local theta = i*2/pole_npolygon*math.pi
+ local r = pole_radius
+ local x, z = r*math.sin(theta), r*math.cos(theta)
+ local lower_index = i+1
+ local upper_index = lower_index+pole_npolygon
+ pole_vertices[lower_index] = vertex(x, -0.5, z)
+ pole_vertices[upper_index] = vertex(x, 0.5, z)
+ end
+ pole_vertices = table.concat(pole_vertices, "\n")
+ pole_objdef = table.concat(pole_objdef, "\n")
+ pole_uv = table.concat(pole_uv, "\n")
+ minetest.safe_file_write(model_path_prefix .. "pole.obj", table.concat({pole_vertices, pole_uv, pole_objdef}, "\n"))
+
+ -- generate signals
+ for lightcount = 5, 6 do
+ for rotname, rot in pairs {["0"] = 0, ["30"] = 26.5, ["45"] = 45, ["60"] = 63.5} do
+ local rot = math.rad(rot)
+ local lightradius = 0.05
+ local lightspacing = 0.04
+ local halfwidth = signal_width/2
+ local halfheight = (2+lightcount)*lightradius+(lightcount-1)*lightspacing/2
+ local halfthickness = signal_thickness/2
+ local half_npolygon = pole_npolygon/2
+ local quarter_npolygon = pole_npolygon/4
+ local boxside = math.max(halfwidth, halfthickness*2)
+ signal_height[lightcount] = halfheight*2
+ signal_box[lightcount] = {-boxside, -halfheight, -boxside, boxside, halfheight, boxside}
+
+ local _vertex = vertex
+ local rv = vector.new(0, rot, 0)
+ local function vertex(x, y, z)
+ local v = vector.rotate(vector.new(x, y, z), rv)
+ return _vertex(v.x, v.y, v.z)
+ end
+
+ -- generate signal face
+ local face_vertices = {}
+ local face_uv = {
+ circular_textures(0.5, 0.5+halfheight-3*lightradius, halfwidth, half_npolygon+1, pole_npolygon),
+ circular_textures(0.5, 0.5-halfheight+3*lightradius, halfwidth, half_npolygon+1, pole_npolygon, math.pi),
+ rectangular_textures(0, 0, 1, 1, 2+pole_npolygon),
+ }
+ local face_objdef = {
+ "g face",
+ "usemtl face",
+ connect_circular(pole_vertex_count+2+pole_npolygon, pole_uv_count, 2+pole_npolygon),
+ connect_circular(pole_vertex_count, pole_uv_count, 2+pole_npolygon),
+ connect_cylindrical(pole_vertex_count, pole_uv_count+2+pole_npolygon, 2+pole_npolygon),
+ }
+ local face_vertex_count = 4*half_npolygon+4
+ local face_uv_count = 2*(half_npolygon+1) + 2*(pole_npolygon+3)
+ for i = 0, half_npolygon do
+ local theta = i/half_npolygon*math.pi
+ local r = halfwidth
+ local x, y = r*math.cos(theta), halfheight-3*lightradius+r*math.sin(theta)
+ face_vertices[i+1] = vertex(x, y, -halfthickness)
+ face_vertices[i+2+half_npolygon] = vertex(-x, -y, -halfthickness)
+ face_vertices[i+3+2*half_npolygon] = vertex(x, y, halfthickness)
+ face_vertices[i+4+3*half_npolygon] = vertex(-x, -y, halfthickness)
+ end
+
+ -- generate lights
+ local light_vertices = {}
+ local light_vertex_count = 8*(half_npolygon+1)+pole_npolygon
+ local light_uv = {rectangular_textures(0, 0, 1, 1, half_npolygon)}
+ local light_uv_count = 2*(half_npolygon+1)+pole_npolygon*lightcount
+ local light_objdef_face = {}
+ local light_objdef_main = {
+ "g light",
+ "usemtl light",
+ }
+ for i = 1, lightcount do
+ local x0, y0 = 0, -halfheight + (2*i+1)*lightradius + (i-1)*lightspacing
+ local v0 = light_vertex_count*(i-1)
+ for j = 0, half_npolygon do
+ local theta = j/half_npolygon*math.pi
+ local xs, ys = math.cos(theta), math.sin(theta)
+ for k, v in pairs {
+ {xm = -1, ym = 1, rm = 1, z = 1},
+ {xm = 1, ym = 1, rm = 0.8, z = 1},
+ {xm = -1, ym = 1, rm = 1, z = 2},
+ {xm = 1, ym = 1, rm = 0.8, z = 2},
+ {xm = 1, ym = -1, rm = 1, z = 1},
+ {xm = -1, ym = -1, rm = 0.8, z = 1},
+ {xm = 1, ym = -1, rm = 1, z = 1.5},
+ {xm = -1, ym = -1, rm = 0.8, z = 1.5},
+ } do
+ local x = x0+xs*lightradius*v.xm*v.rm
+ local y = y0+ys*lightradius*v.ym*v.rm
+ light_vertices[v0+(k-1)*(half_npolygon+1)+j+1] = vertex(x, y, -halfthickness*v.z)
+ end
+ end
+ for j = 0, pole_npolygon-1 do
+ local theta = j/pole_npolygon*2*math.pi
+ local x, y = math.cos(theta), math.sin(theta)
+ light_vertices[v0+8*(half_npolygon+1)+1+j] = vertex(x0+lightradius*x, y0+lightradius*y, -halfthickness*1.05)
+ end
+ local v0 = pole_vertex_count+face_vertex_count+v0
+ local vt0 = pole_uv_count + face_uv_count
+ local ostep = 2*half_npolygon+2
+ for j = 1, half_npolygon do
+ local dv = 2*(half_npolygon+1)
+ local v0 = v0 + dv
+ local vn = v0 + dv
+ light_objdef_face[i*ostep-j+1] = face_elements(v0+j, vt0+j, v0+j+1, vt0+j+1, vn-j, vt0+half_npolygon+2+j, vn-j+1, vt0+half_npolygon+1+j)
+ local v0 = vn + dv
+ local vn = v0 + dv
+ light_objdef_face[i*ostep-half_npolygon-j+1] = face_elements(v0+j, vt0+j, v0+j+1, vt0+j+1, vn-j, vt0+half_npolygon+2+j, vn-j+1, vt0+half_npolygon+1+j)
+ end
+ local vt0 = vt0 + 2*(half_npolygon+1) + (i-1)*pole_npolygon
+ light_uv[i+1] = circular_textures(0.5, (i-1/2)/lightcount, 0.4/lightcount, pole_npolygon)
+ light_objdef_face[(i-1)*ostep+1] = connect_cylindrical(v0, pole_uv_count+2+pole_npolygon, 2+pole_npolygon)
+ light_objdef_face[(i-1)*ostep+2] = connect_cylindrical(v0+4*(half_npolygon+1), pole_uv_count+2+pole_npolygon, 2+pole_npolygon)
+ light_objdef_main[2+i] = connect_circular(v0+8*(half_npolygon+1), vt0, pole_npolygon)
+ end
+
+ -- write file
+ face_vertices = table.concat(face_vertices, "\n")
+ face_uv = table.concat(face_uv, "\n")
+ face_objdef = table.concat(face_objdef, "\n")
+ minetest.safe_file_write(model_path_prefix .. lightcount .. "_" .. rotname .. ".obj", table.concat({
+ pole_vertices,
+ face_vertices,
+ table.concat(light_vertices, "\n"),
+ pole_uv,
+ face_uv,
+ table.concat(light_uv, "\n"),
+ pole_objdef,
+ face_objdef,
+ table.concat(light_objdef_face, "\n"),
+ table.concat(light_objdef_main, "\n"),
+ }, "\n"))
+ end
+ end
+end
+
+local S = attrans
+
+minetest.register_node("advtrains_signals_japan:pole_0", {
+ description = S("Japanese signal pole"),
+ drawtype = "mesh",
+ mesh = "advtrains_signals_japan_pole.obj",
+ tiles = {pole_texture},
+
+ paramtype = "light",
+ sunlight_propagates = true,
+
+ paramtype2 = "none",
+ selection_box = {
+ type = "fixed",
+ fixed = {pole_box},
+ },
+ collision_box = {
+ type = "fixed",
+ fixed = {pole_box},
+ },
+ groups = {
+ cracky = 2,
+ not_blocking_trains = 1,
+ not_in_creative_inventory = 0,
+ },
+ drop = "advtrains_signals_japan:pole_0",
+})
+
+--[[
+advtrains.interlocking.aspect.register_group {
+ name = "advtrains_signals_japan:5a",
+ label = S("Japanese signal"),
+ aspects = {
+ danger = {
+ label = S"Danger (halt)",
+ main = 0,
+ },
+ restrictedspeed = {
+ label = S"Restricted speed",
+ },
+ caution = {
+ label = S"Caution",
+ },
+ reducedspeed = {
+ label = S"Reduced speed",
+ },
+ clear = {
+ label = S"Clear (proceed)",
+ },
+ "clear",
+ "reducedspeed",
+ "caution",
+ "restrictedspeed",
+ "danger",
+ }
+}]]
+
+local sigdefs = {}
+local lightcolors = {
+ red = "red",
+ green = "lime",
+ yellow = "orange",
+ distant = "purple",
+}
+local function process_signal(name, sigdata, isrpt)
+ local def = {}
+ local tx = {}
+ def.textures = tx
+ def.desc = sigdata.desc
+ def.isdst = isrpt
+ def.aspects = sigdata.aspects
+ local lights = sigdata.lights
+ local lightcount = #lights
+ if isrpt then
+ lightcount = lightcount+1
+ end
+ def.lightcount = lightcount
+ def.suppasp_names = {}
+ for idx, asp in ipairs(sigdata.aspects) do
+ local aspname = asp.name
+ local tt = {
+ string.format("[combine:1x%d", lightcount),
+ string.format("0,0=(advtrains_hud_bg.png\\^[resize\\:1x%d\\^[colorize\\:#000)", lightcount),
+ }
+ for _, i in pairs(asp.lights) do
+ local color = lightcolors[lights[i]]
+ tt[#tt+1] = string.format("0,%d=(advtrains_hud_bg.png\\^[colorize\\:%s)", i-1, color)
+ end
+ if isrpt then
+ local color = lightcolors.distant
+ tt[#tt+1] = string.format("0,%d=(advtrains_hud_bg.png\\^[colorize\\:%s)", lightcount-1, color)
+ end
+ tx[aspname] = table.concat(tt, ":")
+ def.suppasp_names[idx] = aspname
+ end
+ local invimg = {
+ string.format("[combine:%dx%d", lightcount*4+1, lightcount*4+1),
+ string.format("%d,0=(advtrains_hud_bg.png\\^[resize\\:5x%d\\^[colorize\\:#000)", lightcount*2-2, lightcount*4+1),
+ }
+ for i, c in pairs(lights) do
+ local color = lightcolors[c]
+ invimg[i+2] = string.format("%d,%d=(advtrains_hud_bg.png\\^[resize\\:3x3\\^[colorize\\:%s)", 2*lightcount-1, 4*i-3, color)
+ end
+ if isrpt then
+ invimg[lightcount+2] = string.format("%d,%d=(advtrains_hud_bg.png\\^[resize\\:3x3\\^[colorize\\:%s)", 2*lightcount-1, 4*lightcount-3, lightcolors.distant)
+ end
+ def.inventory_image = table.concat(invimg, ":")
+ return def
+end
+for sigtype, sigdata in pairs {
+ ["5a"] = {
+ desc = "5A",
+ lights = {"yellow", "yellow", "red", "yellow", "green"},
+ aspects = {
+ {name = "clear", description = S"Clear (proceed)", lights = {5}, main = -1},
+ {name = "reducedspeed", description = S"Reduced speed", lights = {2, 5}, main = 12},
+ {name = "caution", description = S"Caution", lights = {4}},
+ {name = "restrictedspeed", description = S"Restricted speed", lights = {1, 4}, main = 6},
+ {name = "danger", description = S"Danger (halt)", lights = {3}, main = 0},
+ }
+ }
+} do
+ sigdefs["main_"..sigtype] = process_signal(sigtype, sigdata)
+ -- TODO re-enable this once ready
+ --sigdefs["rpt_"..sigtype] = process_signal(sigtype, sigdata, true)
+end
+
+for _, rtab in ipairs {
+ {rot = "0", ici = true},
+ {rot = "30"},
+ {rot = "45"},
+ {rot = "60"},
+} do
+ local rot = rtab.rot
+ for sigtype, siginfo in pairs(sigdefs) do
+ local lightcount = siginfo.lightcount
+ for asp, texture in pairs(siginfo.textures) do
+ minetest.register_node("advtrains_signals_japan:"..sigtype.."_"..asp.."_"..rot, {
+ description = attrans(string.format("Japanese%s signal (type %s)", siginfo.isdst and " repeating" or "", siginfo.desc)),
+ drawtype = "mesh",
+ mesh = string.format("advtrains_signals_japan_%d_%s.obj", lightcount, rot),
+ tiles = {pole_texture, signal_face_texture, texture},
+ paramtype = "light",
+ sunlight_propagates = true,
+ light_source = 4,
+ paramtype2 = "facedir",
+ selection_box = {
+ type = "fixed",
+ fixed = {pole_box, signal_box[lightcount]},
+ },
+ collision_box = {
+ type = "fixed",
+ fixed = {pole_box, signal_box[lightcount]},
+ },
+ groups = {
+ cracky = 2,
+ advtrains_signal = 2,
+ not_blocking_trains = 1,
+ save_in_at_nodedb = 1,
+ not_in_creative_inventory = rtab.ici and asp == "danger" and 0 or 1,
+ },
+ inventory_image = siginfo.inventory_image,
+ drop = "advtrains_signals_japan:"..sigtype.."_danger_0",
+ advtrains = {
+ main_aspects = siginfo.aspects,
+ apply_aspect = function(pos, node, main_aspect, rem_aspect, rem_aspinfo)
+ local asp_name = main_aspect and main_aspect.name or "danger"
+ -- if this signal is clear and remote signal is restrictive (<= 10) then degrade to caution aspect
+ if not main_aspect or main_aspect.halt then
+ asp_name = "danger"
+ elseif main_aspect.name == "clear" and rem_aspinfo and rem_aspinfo.main and rem_aspinfo.main >= 0 and rem_aspinfo.main <= 10 then
+ asp_name = "caution"
+ end
+ advtrains.ndb.swap_node(pos, {name="advtrains_signals_japan:"..sigtype.."_"..asp_name.."_"..rot, param2 = node.param2})
+ end,
+ get_aspect_info = function(pos, main_aspect)
+ if main_aspect.halt then
+ return { main = 0 } -- generic halt
+ end
+ return {
+ main = main_aspect.main,
+ proceed_as_main = true,
+ }
+ end,
+ route_role = "main_distant",
+ --[[
+ supported_aspects = {
+ group = "advtrains_signals_japan:5a",
+ name = siginfo.suppasp_names,
+ dst_shift = siginfo.isdst and 0,
+ main = (not siginfo.isdst) and {} or false
+ },
+ get_aspect = function()
+ local main
+ if siginfo.isdst then
+ main = false
+ end
+ return {group = "advtrains_signals_japan:5a", name = asp, main = main}
+ end,
+ set_aspect = function(pos, node, asp)
+ advtrains.ndb.swap_node(pos, {name = "advtrains_signals_japan:"..sigtype.."_"..(asp.name).."_"..rot, param2 = node.param2})
+ end,
+ ]]
+ },
+ on_rightclick = advtrains.interlocking.signal.on_rightclick,
+ can_dig = advtrains.interlocking.signal.can_dig,
+ after_dig_node = advtrains.interlocking.signal.after_dig,
+ })
+ --advtrains.trackplacer.add_worked("advtrains_signals_japan:"..sigtype, asp, "_"..rot)
+ end
+ end
+end
diff --git a/advtrains_signals_japan/mod.conf b/advtrains_signals_japan/mod.conf
new file mode 100644
index 0000000..8cb2cb3
--- /dev/null
+++ b/advtrains_signals_japan/mod.conf
@@ -0,0 +1,6 @@
+name=advtrains_signals_japan
+title=Advtrains Interlocking Signal Set - Japanese Signals
+description=Japanese signal set for the Advanced Trains Interlocking system
+author=yw05
+
+depends=advtrains_interlocking
diff --git a/advtrains_signals_japan/textures/advtrains_signals_japan_mast.png b/advtrains_signals_japan/textures/advtrains_signals_japan_mast.png
new file mode 100644
index 0000000..8d288a9
--- /dev/null
+++ b/advtrains_signals_japan/textures/advtrains_signals_japan_mast.png
Binary files differ
diff --git a/advtrains_signals_ks/init.lua b/advtrains_signals_ks/init.lua
index d406bdc..9c59401 100644
--- a/advtrains_signals_ks/init.lua
+++ b/advtrains_signals_ks/init.lua
@@ -14,10 +14,9 @@ local function asp_to_zs3type(asp)
return math.min(16,4*math.floor(n/4))
end
-local function setzs3(msp, lim, rot)
+local function setzs3(msp, asp, rot)
local pos = {x = msp.x, y = msp.y+1, z = msp.z}
local node = advtrains.ndb.get_node(pos)
- local asp = asp_to_zs3type(lim)
if node.name:find("^advtrains_signals_ks:zs3_") then
advtrains.ndb.swap_node(pos, {name="advtrains_signals_ks:zs3_"..asp.."_"..rot, param2 = node.param2})
end
@@ -52,88 +51,147 @@ local function getzs3v(msp)
return speed
end
-local setaspectf = function(rot)
- return function(pos, node, asp)
- setzs3(pos, asp.main, rot)
- if asp.main == 0 then
- if asp.shunt then
- advtrains.ndb.swap_node(pos, {name="advtrains_signals_ks:hs_shunt_"..rot, param2 = node.param2})
- else
- advtrains.ndb.swap_node(pos, {name="advtrains_signals_ks:hs_danger_"..rot, param2 = node.param2})
- end
+local applyaspectf_main = function(rot)
+ return function(pos, node, main_aspect, dst_aspect, dst_aspect_info)
+ if main_aspect.halt then
+ -- halt aspect, set red and don't do anything further
+ advtrains.ndb.swap_node(pos, {name="advtrains_signals_ks:hs_danger_"..rot, param2 = node.param2})
+ setzs3(pos, "off", rot)
+ setzs3v(pos, nil, rot)
+ return
+ end
+ -- set zs3 signal to show speed according to main_aspect
+ setzs3(pos, main_aspect.zs3, rot)
+ -- select appropriate lamps based on mainaspect and dst
+ if main_aspect.shunt then
+ advtrains.ndb.swap_node(pos, {name="advtrains_signals_ks:hs_shunt_"..rot, param2 = node.param2})
setzs3v(pos, nil, rot)
else
- if not asp.dst or asp.dst == -1 then
+ if not dst_aspect_info
+ or not dst_aspect_info.main
+ or dst_aspect_info.main == -1 then
advtrains.ndb.swap_node(pos, {name="advtrains_signals_ks:hs_free_"..rot, param2 = node.param2})
- elseif asp.dst == 0 then
+ setzs3v(pos, nil, rot)
+ elseif dst_aspect_info.main == 0 then
advtrains.ndb.swap_node(pos, {name="advtrains_signals_ks:hs_slow_"..rot, param2 = node.param2})
+ setzs3v(pos, nil, rot)
else
advtrains.ndb.swap_node(pos, {name="advtrains_signals_ks:hs_nextslow_"..rot, param2 = node.param2})
+ setzs3v(pos, dst_aspect_info.main, rot)
end
- setzs3v(pos, asp.dst, rot)
end
end
end
+-- Main aspects main signal
+-- These aspects tell only the speed signalization at this signal.
+-- Actual signal aspect is chosen based on this and the Dst signal.
+local mainaspects_main = {
+ {
+ name = "proceed",
+ description = "Proceed",
+ zs3 = "off"
+ },
+ {
+ name = "shunt",
+ description = "Shunt",
+ zs3 = "off",
+ shunt = true,
+ },
+ {
+ name = "proceed_16",
+ description = "Proceed (speed 16)",
+ zs3 = "16",
+ },
+ {
+ name = "proceed_12",
+ description = "Proceed (speed 12)",
+ zs3 = "12",
+ },
+ {
+ name = "proceed_8",
+ description = "Proceed (speed 8)",
+ zs3 = "8",
+ },
+ {
+ name = "proceed_6",
+ description = "Proceed (speed 6)",
+ zs3 = "6",
+ },
+ {
+ name = "proceed_4",
+ description = "Proceed (speed 4)",
+ zs3 = "4",
+ },
+}
-local suppasp = {
- main = {0, 4, 6, 8, 12, 16, -1},
- dst = {0, 4, 6, 8, 12, 16, -1, false},
- shunt = nil,
- proceed_as_main = true,
- info = {
- call_on = false,
- dead_end = false,
- w_speed = nil,
- }
+local applyaspectf_distant = function(rot)
+ return function(pos, node, main_aspect, dst_aspect, dst_aspect_info)
+ if main_aspect.halt then
+ -- halt aspect, set red and don't do anything further
+ advtrains.ndb.swap_node(pos, {name="advtrains_signals_ks:vs_slow_"..rot, param2 = node.param2})
+ setzs3v(pos, nil, rot)
+ return
+ end
+ -- select appropriate lamps based on mainaspect and dst
+ if not dst_aspect_info
+ or not dst_aspect_info.main
+ or dst_aspect_info.main == -1 then
+ advtrains.ndb.swap_node(pos, {name="advtrains_signals_ks:vs_free_"..rot, param2 = node.param2})
+ setzs3v(pos, nil, rot)
+ elseif dst_aspect_info.main == 0 then
+ advtrains.ndb.swap_node(pos, {name="advtrains_signals_ks:vs_slow_"..rot, param2 = node.param2})
+ setzs3v(pos, nil, rot)
+ else
+ advtrains.ndb.swap_node(pos, {name="advtrains_signals_ks:vs_nextslow_"..rot, param2 = node.param2})
+ setzs3v(pos, dst_aspect_info.main, rot)
+ end
+ end
+end
+
+-- Main aspects distant signal
+-- Only one aspect for "expect free". Whether green or yellow lamp is shown and which speed indicator is determined by remote signal
+local mainaspects_dst = {
+ {
+ name = "expectclear",
+ description = "Expect Clear",
+ },
}
--Rangiersignal
-local setaspectf_ra = function(rot)
- return function(pos, node, asp)
- if asp.shunt then
+local applyaspectf_ra = function(rot)
+ -- we get here the full main_aspect table
+ return function(pos, node, main_aspect, dst_aspect, dst_aspect_info)
+ if not main_aspect.halt then
+ -- any non-halt main aspect is fine, there's only one anyway
advtrains.ndb.swap_node(pos, {name="advtrains_signals_ks:ra_shuntd_"..rot, param2 = node.param2})
else
advtrains.ndb.swap_node(pos, {name="advtrains_signals_ks:ra_danger_"..rot, param2 = node.param2})
end
- local meta = minetest.get_meta(pos)
- if meta then
- meta:set_string("infotext", minetest.serialize(asp))
- end
end
end
-local suppasp_ra = {
- main = { false },
- dst = { false },
- shunt = nil,
- proceed_as_main = false,
-
- info = {
- call_on = false,
- dead_end = false,
- w_speed = nil,
- }
+-- Main aspects shunt signal
+-- Shunt signals have only two states, distant doesn't matter
+local mainaspects_ra = {
+ {
+ name = "shunt",
+ description = "Shunt",
+ shunt = true,
+ },
}
-advtrains.trackplacer.register_tracktype("advtrains_signals_ks:hs")
-advtrains.trackplacer.register_tracktype("advtrains_signals_ks:ra")
-advtrains.trackplacer.register_tracktype("advtrains_signals_ks:sign")
-advtrains.trackplacer.register_tracktype("advtrains_signals_ks:sign_lf")
-advtrains.trackplacer.register_tracktype("advtrains_signals_ks:sign_lf7")
-advtrains.trackplacer.register_tracktype("advtrains_signals_ks:zs3")
-advtrains.trackplacer.register_tracktype("advtrains_signals_ks:zs3v")
-advtrains.trackplacer.register_tracktype("advtrains_signals_ks:mast")
-
for _, rtab in ipairs({
- {rot = "0", sbox = {-1/8, -1/2, -1/2, 1/8, 1/2, -1/4}, ici=true},
- {rot = "30", sbox = {-3/8, -1/2, -1/2, -1/8, 1/2, -1/4},},
- {rot = "45", sbox = {-1/2, -1/2, -1/2, -1/4, 1/2, -1/4},},
- {rot = "60", sbox = {-1/2, -1/2, -3/8, -1/4, 1/2, -1/8},},
+ {rot = "0", sbox = {-1/8, -1/2, -1/2, 1/8, 1/2, -1/4}, ici=true, nextrot = "30"},
+ {rot = "30", sbox = {-3/8, -1/2, -1/2, -1/8, 1/2, -1/4}, nextrot = "45"},
+ {rot = "45", sbox = {-1/2, -1/2, -1/2, -1/4, 1/2, -1/4}, nextrot = "60"},
+ {rot = "60", sbox = {-1/2, -1/2, -3/8, -1/4, 1/2, -1/8}, nextrot = "0"},
}) do
local rot = rtab.rot
+
+ -- Hauptsignal
for typ, prts in pairs({
- danger = {asp = advtrains.interlocking.DANGER, n = "slow", ici=true},
+ danger = {asp = advtrains.interlocking.signal.ASPI_HALT, n = "slow", ici=true},
slow = {
asp = function(pos)
return { main = getzs3(pos) or -1, proceed_as_main = true, dst = 0 }
@@ -196,17 +254,91 @@ for _, rtab in ipairs({
drop = "advtrains_signals_ks:hs_danger_0",
inventory_image = "advtrains_signals_ks_hs_inv.png",
advtrains = {
- set_aspect = setaspectf(rot),
- supported_aspects = suppasp,
- get_aspect = afunc,
+ main_aspects = mainaspects_main,
+ apply_aspect = applyaspectf_main(rot),
+ get_aspect_info = afunc,
+ route_role = "main_distant",
+ trackworker_next_rot = "advtrains_signals_ks:hs_"..typ.."_"..rtab.nextrot,
+ trackworker_rot_incr_param2 = (rot=="60")
+ },
+ on_rightclick = advtrains.interlocking.signal.on_rightclick,
+ can_dig = advtrains.interlocking.signal.can_dig,
+ after_dig_node = advtrains.interlocking.signal.after_dig,
+ })
+ -- rotatable by trackworker
+ end
+
+ -- Vorsignal (NEU!)
+ for typ, prts in pairs({
+ -- note: the names are taken from the main signal equivalent so that the same names for the lamp images can be used
+ slow = {asp = function(pos) return { dst = 0, shunt = true } end, n = "nextslow", ici=true},
+ nextslow = {
+ asp = function(pos)
+ return { dst = getzs3v(pos) or 6, shunt = true }
+ end,
+ n = "free"
+ },
+ free = {
+ asp = function(pos)
+ return { dst = -1, shunt = true }
+ end,
+ n = "slow"
+ },
+ }) do
+ local tile = "advtrains_signals_ks_ltm_"..typ..".png"
+ local afunc = prts.asp
+ if type(afunc) == "table" then
+ afunc = function() return prts.asp end
+ end
+ if typ == "nextslow" then
+ tile = {
+ name = tile,
+ animation = {
+ type = "vertical_frames",
+ aspect_w = 32,
+ aspect_h = 32,
+ length = 1,
+ }
+ }
+ end
+ minetest.register_node("advtrains_signals_ks:vs_"..typ.."_"..rot, {
+ description = "Ks Distant Signal",
+ drawtype = "mesh",
+ mesh = "advtrains_signals_ks_distant_smr"..rot..".obj",
+ tiles = {"advtrains_signals_ks_mast.png", "advtrains_signals_ks_head.png", "advtrains_signals_ks_head.png", tile},
+
+ paramtype="light",
+ sunlight_propagates=true,
+ light_source = 4,
+
+ paramtype2 = "facedir",
+ selection_box = {
+ type = "fixed",
+ fixed = {rtab.sbox, {-1/4, -1/2, -1/4, 1/4, -7/16, 1/4}}
+ },
+ groups = {
+ cracky = 2,
+ advtrains_signal = 2,
+ not_blocking_trains = 1,
+ save_in_at_nodedb = 1,
+ not_in_creative_inventory = (rtab.ici and prts.ici) and 0 or 1,
+ },
+ drop = "advtrains_signals_ks:vs_slow_0",
+ inventory_image = "advtrains_signals_ks_vs_inv.png",
+ advtrains = {
+ main_aspects = mainaspects_dst,
+ apply_aspect = applyaspectf_distant(rot),
+ get_aspect_info = afunc,
+ route_role = "distant",
+ pure_distant = true,
+ trackworker_next_rot = "advtrains_signals_ks:vs_"..typ.."_"..rtab.nextrot,
+ trackworker_rot_incr_param2 = (rot=="60")
},
- on_rightclick = advtrains.interlocking.signal_rc_handler,
- can_dig = advtrains.interlocking.signal_can_dig,
- after_dig_node = advtrains.interlocking.signal_after_dig,
- after_place_node = advtrains.after_place_signal,
+ on_rightclick = advtrains.interlocking.signal.on_rightclick,
+ can_dig = advtrains.interlocking.signal.can_dig,
+ after_dig_node = advtrains.interlocking.signal.after_dig,
})
-- rotatable by trackworker
- advtrains.trackplacer.add_worked("advtrains_signals_ks:hs", typ, "_"..rot)
end
@@ -215,14 +347,18 @@ for _, rtab in ipairs({
danger = {asp = { main = false, shunt = false }, n = "shuntd", ici=true},
shuntd = {asp = { main = false, shunt = true } , n = "danger"},
}) do
+ local sbox = table.copy(rtab.sbox)
+ sbox[5] = 0
+ local afunc = prts.asp
+ if type(afunc) == "table" then
+ afunc = function() return prts.asp end
+ end
minetest.register_node("advtrains_signals_ks:ra_"..typ.."_"..rot, {
- description = attrans("Ks Shunting Signal"),
+ description = "Ks Shunting Signal",
drawtype = "mesh",
mesh = "advtrains_signals_ks_sht_smr"..rot..".obj",
- tiles = {advtrains_signals_ks_mast_tile, "advtrains_signals_ks_head.png",
- "advtrains_signals_ks_head.png", "advtrains_signals_ks_ltm_"..typ..".png"},
- use_texture_alpha = "opaque",
-
+ tiles = {"advtrains_signals_ks_mast.png", "advtrains_signals_ks_head.png", "advtrains_signals_ks_head.png", "advtrains_signals_ks_ltm_"..typ..".png"},
+
paramtype="light",
sunlight_propagates=true,
light_source = 4,
@@ -230,7 +366,11 @@ for _, rtab in ipairs({
paramtype2 = "facedir",
selection_box = {
type = "fixed",
- fixed = {-1/4, -1/2, -1/4, 1/4, 0, 1/4}
+ fixed = {sbox, rotation_sbox}
+ },
+ collision_box = {
+ type = "fixed",
+ fixed = sbox,
},
groups = {
cracky = 2,
@@ -242,19 +382,18 @@ for _, rtab in ipairs({
drop = "advtrains_signals_ks:ra_danger_0",
inventory_image = "advtrains_signals_ks_ra_inv.png",
advtrains = {
- set_aspect = setaspectf_ra(rot),
- supported_aspects = suppasp_ra,
- get_aspect = function(pos, node)
- return prts.asp
- end,
+ main_aspects = mainaspects_ra,
+ apply_aspect = applyaspectf_ra(rot),
+ get_aspect_info = afunc,
+ route_role = "shunt",
+ trackworker_next_rot = "advtrains_signals_ks:ra_"..typ.."_"..rtab.nextrot,
+ trackworker_rot_incr_param2 = (rot=="60")
},
- on_rightclick = advtrains.interlocking.signal_rc_handler,
- can_dig = advtrains.interlocking.signal_can_dig,
- after_dig_node = advtrains.interlocking.signal_after_dig,
- after_place_node = advtrains.after_place_signal,
+ on_rightclick = advtrains.interlocking.signal.on_rightclick,
+ can_dig = advtrains.interlocking.signal.can_dig,
+ after_dig_node = advtrains.interlocking.signal.after_dig,
})
-- rotatable by trackworker
- advtrains.trackplacer.add_worked("advtrains_signals_ks:ra", typ, "_"..rot)
end
-- Schilder:
@@ -285,34 +424,35 @@ for _, rtab in ipairs({
drop = "advtrains_signals_ks:"..prefix.."_"..typ.."_0",
inventory_image = inv,
advtrains = {
- get_aspect = function() return asp end
+ get_aspect_info = asp,
+ trackworker_next_rot = "advtrains_signals_ks:"..prefix.."_"..typ.."_"..rtab.nextrot,
+ trackworker_rot_incr_param2 = (rot=="60"),
+ trackworker_next_var = "advtrains_signals_ks:"..prefix.."_"..nxt.."_"..rot,
},
- on_rightclick = advtrains.interlocking.signal_rc_handler,
- can_dig = advtrains.interlocking.signal_can_dig,
- after_dig_node = advtrains.interlocking.signal_after_dig,
- after_place_node = advtrains.after_place_signal,
+ on_rightclick = advtrains.interlocking.signal.on_rightclick,
+ can_dig = advtrains.interlocking.signal.can_dig,
+ after_dig_node = advtrains.interlocking.signal.after_dig,
})
-- rotatable by trackworker
- advtrains.trackplacer.add_worked("advtrains_signals_ks:"..prefix, typ, "_"..rot, nxt)
+ --TODO add rotation using trackworker
end
for typ, prts in pairs {
- ["hfs"] = {
- asp = {main = false, shunt = false},
- n = "pam",
- --[[mesh = "_hfs",]]
- name = "železniční návěst: posun zakázán",
- tile2 = "advtrains_signals_ks_sign_lf7.png^(advtrains_signals_ks_sign_hfs.png^[makealpha:255,255,255)",
- },
- ["pam"] = {
- asp = {main = -1, shunt = false, proceed_as_main = true},
- n = "hfs",
- name = "železniční návěst: posun pokračuje jako normální vlak",
- tile2 = "advtrains_signals_ks_sign_lf7.png^(advtrains_signals_ks_sign_pam.png^[makealpha:255,255,255)"
- }
+ ["hfs"] = {asp = {main = false, shunt = false}, n = "pam", mesh = "_hfs", owntile = true},
+ ["pam"] = {asp = {main = -1, shunt = false, proceed_as_main = true}, n = "ne4"},
+ ["ne4"] = {asp = {}, n = "ne3x1", mesh="_ne4", owntile = true},
+ ["ne3x1"] = {asp = {}, n = "ne3x2", mesh="_ne3", owntile = true},
+ ["ne3x2"] = {asp = {}, n = "ne3x3", mesh="_ne3", owntile = true},
+ ["ne3x3"] = {asp = {}, n = "ne3x4", mesh="_ne3", owntile = true},
+ ["ne3x4"] = {asp = {}, n = "ne3x5", mesh="_ne3", owntile = true},
+ ["ne3x5"] = {asp = {}, n = "hfs", mesh="_ne3", owntile = true},
} do
local mesh = prts.mesh or ""
- register_sign("sign", typ, prts.n, prts.name, "sign"..mesh, prts.tile2, typ, prts.tile2, prts.asp)
+ local tile2 = "advtrains_signals_ks_sign_lf7.png^(advtrains_signals_ks_sign_"..typ..".png^[makealpha:255,255,255)"
+ if prts.owntile then
+ tile2 = "advtrains_signals_ks_sign_"..typ..".png"
+ end
+ register_sign("sign", typ, prts.n, "Signal Sign", "sign"..mesh, tile2, "hfs", "advtrains_signals_ks_sign_lf7.png", prts.asp)
end
for typ, prts in pairs {
@@ -330,8 +470,7 @@ for _, rtab in ipairs({
if typ == "e" then
tile2 = "advtrains_signals_ks_sign_zs10.png"
end
- register_sign("sign", typ, prts.n, attrans("Permanent local speed restriction sign").." ("..(typ == "e" and attrans("end") or typ) ..")",
- "sign"..mesh, tile2, "8", "advtrains_signals_ks_sign_8.png^[invert:rgb", prts.asp)
+ register_sign("sign", typ, prts.n, attrans("Permanent local speed restriction sign").." ("..(typ == "e" and attrans("end") or typ) ..")", "sign"..mesh, tile2, "8", "advtrains_signals_ks_sign_8.png^[invert:rgb", prts.asp)
end
for typ, prts in pairs {
@@ -344,8 +483,7 @@ for _, rtab in ipairs({
} do
local tile2 = "advtrains_signals_ks_sign_lf7.png^(advtrains_signals_ks_sign_"..typ..".png^[makealpha:255,255,255)"..(typ == "e" and "" or "^[multiply:orange")
local inv = "advtrains_signals_ks_sign_lf7.png^(advtrains_signals_ks_sign_8.png^[makealpha:255,255,255)^[multiply:orange"
- register_sign("sign_lf", typ, prts.n, attrans("Temporary local speed restriction sign").." ("..(typ == "e" and attrans("end") or typ) ..")",
- "sign", tile2, "8", inv, {main = prts.main, shunt = true, type = "temp"})
+ register_sign("sign_lf", typ, prts.n, "Temporary local speed restriction sign", "sign", tile2, "8", inv, {main = prts.main, shunt = true, type = "temp"})
end
for typ, prts in pairs {
@@ -353,8 +491,8 @@ for _, rtab in ipairs({
["6"] = {main = 6, n = "8"},
["8"] = {main = 8, n = "12"},
["12"] = {main = 12, n = "16"},
- ["16"] = {main = 16, n = "20"},
- ["20"] = {main = 20, n = "4"},
+ ["16"] = {main = 16, n = "e"},
+ ["e"] = {main = -1, n = "4"},
} do
local tile2 = "advtrains_signals_ks_sign_lf7.png^(advtrains_signals_ks_sign_"..typ..".png^[makealpha:255,255,255)"
local inv = "advtrains_signals_ks_sign_lf7.png^(advtrains_signals_ks_sign_8.png^[makealpha:255,255,255)"
@@ -390,8 +528,7 @@ for _, rtab in ipairs({
save_in_at_nodedb = 1,
not_in_creative_inventory = (rtab.ici and prts.ici) and 0 or 1,
},
- after_dig_node = function(pos) advtrains.ndb.update(pos) end,
- after_place_node = advtrains.after_place_signal,
+ after_dig_node = function(pos) advtrains.ndb.update(pos) end
}
-- Zs 3
@@ -400,8 +537,12 @@ for _, rtab in ipairs({
t.mesh = "advtrains_signals_ks_zs_top_smr"..rot..".obj"
t.drop = "advtrains_signals_ks:zs3_off_0"
t.selection_box.fixed[1][5] = 0
+ t.advtrains = {
+ trackworker_next_rot = "advtrains_signals_ks:zs3_"..typ.."_"..rtab.nextrot,
+ trackworker_rot_incr_param2 = (rot=="60")
+ }
minetest.register_node("advtrains_signals_ks:zs3_"..typ.."_"..rot, t)
- advtrains.trackplacer.add_worked("advtrains_signals_ks:zs3", typ, "_"..rot)
+ --TODO add rotation using trackworker
-- Zs 3v
local t = table.copy(def)
@@ -409,8 +550,11 @@ for _, rtab in ipairs({
t.mesh = "advtrains_signals_ks_zs_bottom_smr"..rot..".obj"
t.drop = "advtrains_signals_ks:zs3v_off_0"
t.tiles[3] = t.tiles[3] .. "^[multiply:yellow"
+ t.advtrains = {
+ trackworker_next_rot = "advtrains_signals_ks:zs3v_"..typ.."_"..rtab.nextrot,
+ trackworker_rot_incr_param2 = (rot=="60")
+ }
minetest.register_node("advtrains_signals_ks:zs3v_"..typ.."_"..rot, t)
- advtrains.trackplacer.add_worked("advtrains_signals_ks:zs3v", typ, "_"..rot)
end
minetest.register_node("advtrains_signals_ks:mast_mast_"..rot, {
@@ -435,10 +579,13 @@ for _, rtab in ipairs({
not_blocking_trains = 1,
not_in_creative_inventory = (rtab.ici) and 0 or 1,
},
+ advtrains = {
+ trackworker_next_rot = "advtrains_signals_ks:mast_mast_"..rtab.nextrot,
+ trackworker_rot_incr_param2 = (rot=="60")
+ },
drop = "advtrains_signals_ks:mast_mast_0",
- after_place_node = advtrains.after_place_signal,
})
- advtrains.trackplacer.add_worked("advtrains_signals_ks:mast","mast", "_"..rot)
+ --TODO add rotation using trackworker
end
-- Crafting
@@ -470,6 +617,23 @@ minetest.register_craft({
},
})
+minetest.register_craft({
+ output = "advtrains_signals_ks:zs3_off_0 2",
+ recipe = {
+ {"","default:steel_ingot",""},
+ {"default:steel_ingot","dye:white","default:steel_ingot"},
+ {"","advtrains_signals_ks:mast_mast_0",""}
+ },
+})
+minetest.register_craft({
+ output = "advtrains_signals_ks:zs3v_off_0 2",
+ recipe = {
+ {"","default:steel_ingot",""},
+ {"default:steel_ingot","dye:yellow","default:steel_ingot"},
+ {"","advtrains_signals_ks:mast_mast_0",""}
+ },
+})
+
local sign_material = "default:sign_wall_steel" --fallback
if minetest.get_modpath("basic_materials") then
sign_material = "basic_materials:plastic_sheet"
diff --git a/advtrains_signals_ks/models/advtrains_signals_ks_distant_smr0.obj b/advtrains_signals_ks/models/advtrains_signals_ks_distant_smr0.obj
new file mode 100644
index 0000000..9655dc4
--- /dev/null
+++ b/advtrains_signals_ks/models/advtrains_signals_ks_distant_smr0.obj
@@ -0,0 +1,3293 @@
+# Blender 3.4.1
+# www.blender.org
+mtllib advtrains_signals_ks_distant_smr0.mtl
+o KsHauptsignal_Plane.006
+v 0.028888 -0.503378 -0.439591
+v -0.028274 -0.503378 -0.439631
+v -0.028235 -0.503378 -0.496794
+v 0.028928 -0.503378 -0.496754
+v 0.028809 -0.503378 -0.325555
+v 0.028769 -0.503378 -0.268392
+v -0.028394 -0.503378 -0.268432
+v -0.028354 -0.503378 -0.325595
+v 0.028888 -0.299114 -0.439591
+v 0.028888 -0.172218 -0.439591
+v -0.028274 -0.172218 -0.439631
+v -0.028274 -0.299114 -0.439631
+v -0.028235 -0.299114 -0.496794
+v -0.028235 -0.172218 -0.496794
+v 0.028928 -0.172218 -0.496754
+v 0.028928 -0.299114 -0.496754
+v 0.028809 -0.299114 -0.325555
+v 0.028809 -0.172218 -0.325555
+v 0.028769 -0.172218 -0.268392
+v 0.028769 -0.299114 -0.268392
+v -0.028394 -0.299114 -0.268432
+v -0.028394 -0.172218 -0.268432
+v -0.028354 -0.172218 -0.325595
+v -0.028354 -0.299114 -0.325595
+v -0.135201 0.243339 -0.586523
+v -0.135219 0.242789 -0.559563
+v -0.132013 0.253858 -0.639022
+v -0.132068 0.253308 -0.559514
+v -0.126870 0.263561 -0.638975
+v -0.126926 0.263011 -0.559467
+v -0.119933 0.272074 -0.638933
+v -0.119989 0.271524 -0.559424
+v -0.111469 0.279070 -0.638896
+v -0.111524 0.278520 -0.559387
+v -0.101802 0.284280 -0.638865
+v -0.101858 0.283730 -0.559357
+v -0.091305 0.287504 -0.638844
+v -0.091361 0.286954 -0.559335
+v -0.080380 0.288619 -0.638831
+v -0.080436 0.288069 -0.559323
+v -0.069448 0.287581 -0.638828
+v -0.069504 0.287030 -0.559320
+v -0.058929 0.284430 -0.638835
+v -0.058984 0.283880 -0.559327
+v -0.049226 0.279287 -0.638851
+v -0.049281 0.278737 -0.559343
+v -0.040713 0.272350 -0.638876
+v -0.040768 0.271800 -0.559368
+v -0.033717 0.263886 -0.638909
+v -0.033772 0.263336 -0.559401
+v -0.028506 0.254219 -0.638948
+v -0.028562 0.253669 -0.559440
+v -0.025318 0.243722 -0.586445
+v -0.025337 0.243172 -0.559485
+v -0.024183 0.232316 -0.586441
+v -0.024223 0.232252 -0.559532
+v -0.025274 0.221688 -0.586283
+v -0.025260 0.221322 -0.559578
+v -0.028394 0.211344 -0.586324
+v -0.028411 0.210797 -0.559621
+v -0.033537 0.201642 -0.586366
+v -0.033554 0.201094 -0.559663
+v -0.040474 0.193129 -0.586405
+v -0.040491 0.192581 -0.559701
+v -0.048938 0.186134 -0.586439
+v -0.048955 0.185585 -0.559735
+v -0.058604 0.180924 -0.586466
+v -0.058622 0.180374 -0.559762
+v -0.069101 0.177700 -0.586486
+v -0.069119 0.177150 -0.559782
+v -0.080026 0.176585 -0.586498
+v -0.080044 0.176035 -0.559794
+v -0.090958 0.177623 -0.586502
+v -0.090977 0.177073 -0.559798
+v -0.101477 0.180774 -0.586496
+v -0.101497 0.180224 -0.559793
+v -0.111180 0.185916 -0.586483
+v -0.111200 0.185367 -0.559779
+v -0.119693 0.192853 -0.586461
+v -0.119713 0.192304 -0.559758
+v -0.126689 0.201317 -0.586433
+v -0.126709 0.200769 -0.559729
+v -0.131899 0.210983 -0.586398
+v -0.131919 0.210436 -0.559694
+v -0.135091 0.221305 -0.586361
+v -0.135144 0.220938 -0.559656
+v -0.136256 0.231924 -0.586521
+v -0.136258 0.231861 -0.559612
+v -0.137672 0.243790 -0.586520
+v -0.137717 0.243271 -0.523031
+v -0.134343 0.254782 -0.639014
+v -0.134424 0.254264 -0.522977
+v -0.128969 0.264921 -0.638963
+v -0.129050 0.264402 -0.522926
+v -0.121720 0.273816 -0.638916
+v -0.121801 0.273298 -0.522879
+v -0.112875 0.281127 -0.638876
+v -0.112956 0.280609 -0.522839
+v -0.102774 0.286572 -0.638843
+v -0.102855 0.286053 -0.522806
+v -0.091805 0.289941 -0.638820
+v -0.091886 0.289423 -0.522782
+v -0.080389 0.291105 -0.638806
+v -0.080470 0.290587 -0.522769
+v -0.068965 0.290021 -0.638803
+v -0.069047 0.289502 -0.522766
+v -0.057973 0.286728 -0.638811
+v -0.058054 0.286210 -0.522774
+v -0.047834 0.281354 -0.638829
+v -0.047915 0.280836 -0.522792
+v -0.038938 0.274105 -0.638857
+v -0.039019 0.273587 -0.522820
+v -0.031628 0.265261 -0.638893
+v -0.031709 0.264742 -0.522856
+v -0.026183 0.255160 -0.638937
+v -0.026264 0.254641 -0.522900
+v -0.022850 0.244190 -0.586438
+v -0.022895 0.243672 -0.522949
+v -0.021686 0.232775 -0.586437
+v -0.021730 0.232256 -0.523002
+v -0.022771 0.221351 -0.586289
+v -0.022815 0.220833 -0.523056
+v -0.026063 0.210359 -0.586343
+v -0.026107 0.209840 -0.523110
+v -0.031437 0.200220 -0.586394
+v -0.031481 0.199702 -0.523162
+v -0.038686 0.191324 -0.586441
+v -0.038730 0.190806 -0.523209
+v -0.047531 0.184013 -0.586481
+v -0.047575 0.183495 -0.523249
+v -0.057632 0.178569 -0.586514
+v -0.057676 0.178051 -0.523282
+v -0.068601 0.175200 -0.586538
+v -0.068645 0.174682 -0.523305
+v -0.080017 0.174035 -0.586551
+v -0.080061 0.173517 -0.523319
+v -0.091441 0.175120 -0.586554
+v -0.091485 0.174602 -0.523322
+v -0.102433 0.178413 -0.586546
+v -0.102477 0.177894 -0.523314
+v -0.112572 0.183787 -0.586528
+v -0.112616 0.183268 -0.523296
+v -0.121468 0.191035 -0.586500
+v -0.121512 0.190517 -0.523268
+v -0.128778 0.199880 -0.586464
+v -0.128823 0.199362 -0.523231
+v -0.134223 0.209981 -0.586420
+v -0.134267 0.209463 -0.523188
+v -0.137592 0.220950 -0.586371
+v -0.137637 0.220432 -0.523138
+v -0.138757 0.232366 -0.586521
+v -0.138801 0.231848 -0.523086
+v 0.097038 0.383581 -0.586361
+v 0.097019 0.383353 -0.559401
+v 0.098378 0.387932 -0.638861
+v 0.098323 0.387704 -0.559353
+v 0.100505 0.391945 -0.638817
+v 0.100450 0.391717 -0.559308
+v 0.103374 0.395466 -0.638777
+v 0.103319 0.395239 -0.559268
+v 0.106875 0.398360 -0.638743
+v 0.106820 0.398132 -0.559235
+v 0.110874 0.400515 -0.638717
+v 0.110818 0.400287 -0.559209
+v 0.115216 0.401848 -0.638700
+v 0.115160 0.401621 -0.559191
+v 0.119734 0.402309 -0.638692
+v 0.119679 0.402082 -0.559183
+v 0.124256 0.401880 -0.638693
+v 0.124201 0.401652 -0.559185
+v 0.128607 0.400577 -0.638704
+v 0.128552 0.400349 -0.559196
+v 0.132620 0.398450 -0.638724
+v 0.132565 0.398222 -0.559216
+v 0.136141 0.395581 -0.638753
+v 0.136086 0.395353 -0.559244
+v 0.139035 0.392080 -0.638788
+v 0.138980 0.391852 -0.559280
+v 0.141190 0.388081 -0.638830
+v 0.141135 0.387854 -0.559322
+v 0.142487 0.383739 -0.586328
+v 0.142469 0.383512 -0.559367
+v 0.142957 0.379021 -0.586324
+v 0.142930 0.378995 -0.559416
+v 0.142506 0.374626 -0.586166
+v 0.142501 0.374474 -0.559461
+v 0.141215 0.370347 -0.586206
+v 0.141197 0.370121 -0.559502
+v 0.139088 0.366334 -0.586246
+v 0.139070 0.366108 -0.559542
+v 0.136219 0.362813 -0.586282
+v 0.136201 0.362586 -0.559578
+v 0.132718 0.359920 -0.586312
+v 0.132700 0.359693 -0.559608
+v 0.128720 0.357765 -0.586335
+v 0.128702 0.357538 -0.559632
+v 0.124378 0.356431 -0.586351
+v 0.124360 0.356204 -0.559647
+v 0.119860 0.355970 -0.586358
+v 0.119841 0.355743 -0.559655
+v 0.115338 0.356399 -0.586358
+v 0.115319 0.356172 -0.559654
+v 0.110987 0.357703 -0.586348
+v 0.110968 0.357476 -0.559645
+v 0.106974 0.359830 -0.586331
+v 0.106954 0.359603 -0.559627
+v 0.103453 0.362699 -0.586305
+v 0.103433 0.362472 -0.559602
+v 0.100559 0.366200 -0.586274
+v 0.100539 0.365973 -0.559570
+v 0.098404 0.370198 -0.586237
+v 0.098384 0.369972 -0.559533
+v 0.097083 0.374467 -0.586199
+v 0.097050 0.374316 -0.559494
+v 0.096602 0.378860 -0.586358
+v 0.096590 0.378833 -0.559449
+v 0.096016 0.383767 -0.586357
+v 0.095972 0.383553 -0.522868
+v 0.097414 0.388314 -0.638853
+v 0.097333 0.388099 -0.522815
+v 0.099637 0.392507 -0.638803
+v 0.099556 0.392293 -0.522766
+v 0.102635 0.396187 -0.638760
+v 0.102554 0.395973 -0.522722
+v 0.106294 0.399211 -0.638723
+v 0.106213 0.398996 -0.522686
+v 0.110472 0.401463 -0.638694
+v 0.110391 0.401248 -0.522657
+v 0.115009 0.402856 -0.638675
+v 0.114928 0.402642 -0.522638
+v 0.119731 0.403338 -0.638666
+v 0.119650 0.403124 -0.522629
+v 0.124456 0.402889 -0.638668
+v 0.124375 0.402675 -0.522631
+v 0.129002 0.401527 -0.638681
+v 0.128921 0.401313 -0.522643
+v 0.133196 0.399305 -0.638703
+v 0.133115 0.399090 -0.522666
+v 0.136876 0.396306 -0.638734
+v 0.136795 0.396092 -0.522697
+v 0.139899 0.392648 -0.638774
+v 0.139818 0.392434 -0.522737
+v 0.142151 0.388470 -0.638820
+v 0.142070 0.388256 -0.522783
+v 0.143508 0.383933 -0.586322
+v 0.143464 0.383719 -0.522833
+v 0.143990 0.379211 -0.586322
+v 0.143946 0.378997 -0.522886
+v 0.143541 0.374486 -0.586173
+v 0.143497 0.374272 -0.522940
+v 0.142179 0.369939 -0.586225
+v 0.142135 0.369725 -0.522993
+v 0.139957 0.365746 -0.586274
+v 0.139912 0.365532 -0.523042
+v 0.136958 0.362067 -0.586318
+v 0.136914 0.361852 -0.523086
+v 0.133300 0.359043 -0.586355
+v 0.133256 0.358828 -0.523123
+v 0.129122 0.356791 -0.586384
+v 0.129078 0.356576 -0.523151
+v 0.124585 0.355397 -0.586403
+v 0.124541 0.355183 -0.523170
+v 0.119863 0.354916 -0.586412
+v 0.119819 0.354701 -0.523179
+v 0.115138 0.355364 -0.586410
+v 0.115094 0.355150 -0.523177
+v 0.110591 0.356726 -0.586397
+v 0.110547 0.356512 -0.523165
+v 0.106398 0.358949 -0.586375
+v 0.106354 0.358735 -0.523143
+v 0.102718 0.361947 -0.586344
+v 0.102674 0.361733 -0.523111
+v 0.099694 0.365605 -0.586304
+v 0.099650 0.365391 -0.523072
+v 0.097442 0.369783 -0.586258
+v 0.097398 0.369569 -0.523026
+v 0.096049 0.374320 -0.586208
+v 0.096005 0.374106 -0.522975
+v 0.095567 0.379042 -0.586357
+v 0.095523 0.378828 -0.522922
+v 0.064799 0.243339 -0.586384
+v 0.064781 0.242789 -0.559423
+v 0.067987 0.253858 -0.638883
+v 0.067932 0.253308 -0.559374
+v 0.073130 0.263561 -0.638836
+v 0.073074 0.263011 -0.559327
+v 0.080067 0.272074 -0.638793
+v 0.080011 0.271524 -0.559285
+v 0.088531 0.279070 -0.638756
+v 0.088476 0.278520 -0.559248
+v 0.098198 0.284280 -0.638726
+v 0.098142 0.283730 -0.559217
+v 0.108695 0.287504 -0.638704
+v 0.108639 0.286954 -0.559196
+v 0.119620 0.288619 -0.638692
+v 0.119564 0.288069 -0.559183
+v 0.130552 0.287581 -0.638689
+v 0.130496 0.287030 -0.559180
+v 0.141071 0.284430 -0.638695
+v 0.141016 0.283880 -0.559187
+v 0.150774 0.279287 -0.638712
+v 0.150719 0.278737 -0.559203
+v 0.159287 0.272350 -0.638736
+v 0.159232 0.271800 -0.559228
+v 0.166283 0.263886 -0.638769
+v 0.166228 0.263336 -0.559261
+v 0.171494 0.254219 -0.638809
+v 0.171438 0.253669 -0.559300
+v 0.174681 0.243722 -0.586305
+v 0.174663 0.243172 -0.559345
+v 0.175817 0.232316 -0.586301
+v 0.175777 0.232252 -0.559393
+v 0.174726 0.221688 -0.586144
+v 0.174740 0.221322 -0.559438
+v 0.171606 0.211344 -0.586185
+v 0.171589 0.210797 -0.559481
+v 0.166463 0.201642 -0.586227
+v 0.166446 0.201094 -0.559523
+v 0.159526 0.193129 -0.586265
+v 0.159509 0.192581 -0.559562
+v 0.151062 0.186134 -0.586299
+v 0.151045 0.185585 -0.559595
+v 0.141396 0.180924 -0.586326
+v 0.141378 0.180374 -0.559623
+v 0.130899 0.177700 -0.586346
+v 0.130881 0.177150 -0.559643
+v 0.119974 0.176585 -0.586358
+v 0.119956 0.176035 -0.559655
+v 0.109042 0.177623 -0.586362
+v 0.109023 0.177073 -0.559659
+v 0.098523 0.180774 -0.586357
+v 0.098503 0.180224 -0.559653
+v 0.088820 0.185916 -0.586343
+v 0.088800 0.185368 -0.559640
+v 0.080307 0.192853 -0.586322
+v 0.080287 0.192304 -0.559618
+v 0.073311 0.201317 -0.586293
+v 0.073291 0.200769 -0.559589
+v 0.068101 0.210983 -0.586258
+v 0.068081 0.210436 -0.559555
+v 0.064909 0.221305 -0.586222
+v 0.064856 0.220938 -0.559516
+v 0.063744 0.231924 -0.586381
+v 0.063742 0.231861 -0.559472
+v 0.062328 0.243790 -0.586381
+v 0.062283 0.243271 -0.522892
+v 0.065657 0.254782 -0.638875
+v 0.065576 0.254264 -0.522838
+v 0.071031 0.264921 -0.638823
+v 0.070950 0.264402 -0.522786
+v 0.078280 0.273816 -0.638777
+v 0.078199 0.273298 -0.522739
+v 0.087125 0.281127 -0.638736
+v 0.087044 0.280609 -0.522699
+v 0.097226 0.286572 -0.638704
+v 0.097145 0.286053 -0.522666
+v 0.108195 0.289941 -0.638680
+v 0.108114 0.289423 -0.522643
+v 0.119611 0.291105 -0.638667
+v 0.119530 0.290587 -0.522629
+v 0.131035 0.290021 -0.638664
+v 0.130953 0.289502 -0.522627
+v 0.142027 0.286728 -0.638672
+v 0.141946 0.286210 -0.522634
+v 0.152166 0.281354 -0.638690
+v 0.152085 0.280836 -0.522652
+v 0.161062 0.274105 -0.638717
+v 0.160981 0.273587 -0.522680
+v 0.168372 0.265261 -0.638754
+v 0.168291 0.264742 -0.522717
+v 0.173817 0.255160 -0.638798
+v 0.173736 0.254641 -0.522760
+v 0.177150 0.244190 -0.586299
+v 0.177105 0.243672 -0.522810
+v 0.178314 0.232775 -0.586298
+v 0.178270 0.232256 -0.522862
+v 0.177229 0.221351 -0.586149
+v 0.177185 0.220833 -0.522917
+v 0.173937 0.210359 -0.586203
+v 0.173893 0.209840 -0.522971
+v 0.168563 0.200220 -0.586254
+v 0.168519 0.199702 -0.523022
+v 0.161314 0.191324 -0.586301
+v 0.161270 0.190806 -0.523069
+v 0.152469 0.184014 -0.586342
+v 0.152425 0.183495 -0.523109
+v 0.142368 0.178569 -0.586374
+v 0.142324 0.178051 -0.523142
+v 0.131399 0.175200 -0.586398
+v 0.131355 0.174682 -0.523166
+v 0.119983 0.174035 -0.586411
+v 0.119939 0.173517 -0.523179
+v 0.108559 0.175120 -0.586414
+v 0.108515 0.174602 -0.523182
+v 0.097567 0.178413 -0.586406
+v 0.097523 0.177894 -0.523174
+v 0.087428 0.183787 -0.586388
+v 0.087384 0.183268 -0.523156
+v 0.078532 0.191035 -0.586360
+v 0.078488 0.190517 -0.523128
+v 0.071222 0.199880 -0.586324
+v 0.071177 0.199362 -0.523092
+v 0.065777 0.209981 -0.586280
+v 0.065733 0.209463 -0.523048
+v 0.062408 0.220950 -0.586231
+v 0.062363 0.220432 -0.522999
+v 0.061243 0.232366 -0.586381
+v 0.061199 0.231848 -0.522946
+v 0.095523 0.050272 -0.522922
+v 0.095567 0.050487 -0.586357
+v 0.096005 0.045551 -0.522975
+v 0.096049 0.045765 -0.586208
+v 0.097398 0.041014 -0.523026
+v 0.097442 0.041228 -0.586258
+v 0.099650 0.036836 -0.523072
+v 0.099694 0.037050 -0.586304
+v 0.102674 0.033177 -0.523111
+v 0.102718 0.033392 -0.586344
+v 0.106354 0.030179 -0.523143
+v 0.106398 0.030393 -0.586375
+v 0.110547 0.027956 -0.523165
+v 0.110591 0.028171 -0.586397
+v 0.115094 0.026594 -0.523177
+v 0.115138 0.026809 -0.586410
+v 0.119819 0.026146 -0.523179
+v 0.119863 0.026360 -0.586412
+v 0.124541 0.026627 -0.523170
+v 0.124585 0.026842 -0.586403
+v 0.129078 0.028021 -0.523151
+v 0.129122 0.028235 -0.586384
+v 0.133256 0.030273 -0.523123
+v 0.133300 0.030487 -0.586355
+v 0.136914 0.033297 -0.523086
+v 0.136958 0.033511 -0.586318
+v 0.139912 0.036976 -0.523042
+v 0.139957 0.037191 -0.586274
+v 0.142135 0.041170 -0.522993
+v 0.142179 0.041384 -0.586225
+v 0.143497 0.045716 -0.522940
+v 0.143541 0.045931 -0.586173
+v 0.143946 0.050441 -0.522886
+v 0.143990 0.050656 -0.586322
+v 0.143464 0.055163 -0.522833
+v 0.143508 0.055377 -0.586322
+v 0.142070 0.059700 -0.522783
+v 0.142151 0.059915 -0.638820
+v 0.139818 0.063878 -0.522737
+v 0.139899 0.064093 -0.638774
+v 0.136795 0.067537 -0.522697
+v 0.136876 0.067751 -0.638734
+v 0.133115 0.070535 -0.522666
+v 0.133196 0.070749 -0.638703
+v 0.128921 0.072757 -0.522643
+v 0.129002 0.072972 -0.638681
+v 0.124375 0.074119 -0.522631
+v 0.124456 0.074334 -0.638668
+v 0.119650 0.074568 -0.522629
+v 0.119731 0.074782 -0.638666
+v 0.114928 0.074086 -0.522638
+v 0.115009 0.074301 -0.638675
+v 0.110391 0.072693 -0.522657
+v 0.110472 0.072907 -0.638694
+v 0.106213 0.070441 -0.522686
+v 0.106294 0.070655 -0.638723
+v 0.102554 0.067417 -0.522722
+v 0.102635 0.067631 -0.638760
+v 0.099556 0.063738 -0.522766
+v 0.099637 0.063952 -0.638803
+v 0.097333 0.059544 -0.522815
+v 0.097414 0.059758 -0.638853
+v 0.095972 0.054997 -0.522868
+v 0.096016 0.055212 -0.586357
+v 0.096590 0.050278 -0.559449
+v 0.096602 0.050304 -0.586358
+v 0.097050 0.045760 -0.559494
+v 0.097083 0.045912 -0.586199
+v 0.098384 0.041416 -0.559533
+v 0.098404 0.041642 -0.586237
+v 0.100539 0.037418 -0.559570
+v 0.100559 0.037644 -0.586274
+v 0.103433 0.033917 -0.559602
+v 0.103453 0.034143 -0.586305
+v 0.106954 0.031047 -0.559627
+v 0.106974 0.031274 -0.586331
+v 0.110968 0.028920 -0.559645
+v 0.110987 0.029147 -0.586348
+v 0.115319 0.027617 -0.559654
+v 0.115338 0.027844 -0.586358
+v 0.119841 0.027187 -0.559655
+v 0.119860 0.027415 -0.586358
+v 0.124360 0.027648 -0.559647
+v 0.124378 0.027876 -0.586351
+v 0.128702 0.028982 -0.559632
+v 0.128720 0.029209 -0.586335
+v 0.132700 0.031137 -0.559608
+v 0.132718 0.031364 -0.586312
+v 0.136201 0.034031 -0.559578
+v 0.136219 0.034258 -0.586282
+v 0.139070 0.037552 -0.559542
+v 0.139088 0.037779 -0.586246
+v 0.141197 0.041566 -0.559502
+v 0.141215 0.041792 -0.586206
+v 0.142501 0.045919 -0.559461
+v 0.142506 0.046070 -0.586166
+v 0.142930 0.050440 -0.559416
+v 0.142957 0.050466 -0.586324
+v 0.142469 0.054956 -0.559367
+v 0.142487 0.055184 -0.586328
+v 0.141135 0.059298 -0.559322
+v 0.141190 0.059526 -0.638830
+v 0.138980 0.063296 -0.559280
+v 0.139035 0.063524 -0.638788
+v 0.136086 0.066797 -0.559244
+v 0.136141 0.067025 -0.638753
+v 0.132565 0.069667 -0.559216
+v 0.132620 0.069894 -0.638724
+v 0.128552 0.071794 -0.559196
+v 0.128607 0.072021 -0.638704
+v 0.124201 0.073097 -0.559185
+v 0.124256 0.073325 -0.638693
+v 0.119679 0.073526 -0.559183
+v 0.119734 0.073754 -0.638692
+v 0.115160 0.073065 -0.559191
+v 0.115216 0.073293 -0.638700
+v 0.110818 0.071732 -0.559209
+v 0.110874 0.071959 -0.638717
+v 0.106820 0.069577 -0.559235
+v 0.106875 0.069804 -0.638743
+v 0.103319 0.066683 -0.559268
+v 0.103374 0.066911 -0.638777
+v 0.100450 0.063162 -0.559308
+v 0.100505 0.063390 -0.638817
+v 0.098323 0.059149 -0.559353
+v 0.098378 0.059376 -0.638861
+v 0.097019 0.054798 -0.559401
+v 0.097038 0.055025 -0.586361
+v 0.028888 -0.168051 -0.439591
+v -0.028274 -0.168051 -0.439631
+v -0.028235 -0.168051 -0.496794
+v 0.028928 -0.168051 -0.496754
+v 0.028809 -0.168051 -0.325555
+v 0.028769 -0.168051 -0.268392
+v -0.028394 -0.168051 -0.268432
+v -0.028354 -0.168051 -0.325595
+v -0.028394 -0.168051 -0.268432
+v 0.028769 -0.168051 -0.268392
+v 0.028928 -0.168051 -0.496754
+v -0.028235 -0.168051 -0.496794
+v -0.028354 -0.168051 -0.325595
+v -0.028274 -0.168051 -0.439631
+v 0.028809 -0.168051 -0.325555
+v 0.028888 -0.168051 -0.439591
+v 0.028888 0.037812 -0.439591
+v 0.028888 0.165702 -0.439591
+v -0.028274 0.165702 -0.439631
+v -0.028274 0.037812 -0.439631
+v -0.028235 0.037812 -0.496794
+v -0.028235 0.165702 -0.496794
+v 0.028928 0.165702 -0.496754
+v 0.028928 0.037812 -0.496754
+v 0.028809 0.037812 -0.325555
+v 0.028809 0.165702 -0.325555
+v 0.028769 0.165702 -0.268392
+v 0.028769 0.037812 -0.268392
+v -0.028394 0.037812 -0.268432
+v -0.028394 0.165702 -0.268432
+v -0.028354 0.165702 -0.325595
+v -0.028354 0.037812 -0.325595
+v 0.028888 0.166622 -0.439591
+v -0.028274 0.166622 -0.439631
+v -0.028235 0.166622 -0.496794
+v 0.028928 0.166622 -0.496754
+v 0.028809 0.166622 -0.325555
+v 0.028769 0.166622 -0.268392
+v -0.028394 0.166622 -0.268432
+v -0.028354 0.166622 -0.325595
+v -0.028394 0.166622 -0.268432
+v 0.028769 0.166622 -0.268392
+v 0.028928 0.166622 -0.496754
+v -0.028235 0.166622 -0.496794
+v -0.028354 0.166622 -0.325595
+v -0.028274 0.166622 -0.439631
+v 0.028809 0.166622 -0.325555
+v 0.028888 0.166622 -0.439591
+v 0.028888 0.370885 -0.439591
+v 0.028888 0.497782 -0.439591
+v -0.028274 0.497782 -0.439631
+v -0.028274 0.370885 -0.439631
+v -0.028235 0.370885 -0.496794
+v -0.028235 0.497782 -0.496794
+v 0.028928 0.497782 -0.496754
+v 0.028928 0.370885 -0.496754
+v 0.028809 0.370885 -0.325555
+v 0.028809 0.497782 -0.325555
+v 0.028769 0.497782 -0.268392
+v 0.028769 0.370885 -0.268392
+v -0.028394 0.370885 -0.268432
+v -0.028394 0.497782 -0.268432
+v -0.028354 0.497782 -0.325595
+v -0.028354 0.370885 -0.325595
+v 0.269752 -0.080213 -0.548298
+v -0.230248 -0.080213 -0.548647
+v 0.269752 0.496942 -0.548297
+v -0.230248 0.496942 -0.548646
+v 0.269719 0.496942 -0.500879
+v 0.269719 -0.080213 -0.500879
+v -0.230281 -0.080213 -0.501228
+v -0.230281 0.496942 -0.501228
+vn -1.0000 -0.0000 -0.0007
+vn 1.0000 -0.0000 0.0007
+vn 0.0007 -0.0000 -1.0000
+vn -0.0007 -0.0000 1.0000
+vn -0.0000 -1.0000 -0.0000
+vn -0.0000 1.0000 -0.0000
+vn -0.0945 -0.9955 -0.0070
+vn 0.9955 -0.0946 -0.0013
+vn -0.9948 -0.1015 -0.0028
+vn -0.9955 0.0944 0.0018
+vn 0.9948 0.1014 0.0034
+vn -0.9955 0.0945 0.0001
+vn 0.9948 0.1015 0.0015
+vn 0.9955 -0.0946 -0.0001
+vn -0.9948 -0.1014 -0.0015
+vn 0.7707 0.6370 0.0136
+vn 0.0087 0.0090 -0.9999
+vn 0.0032 0.0067 -1.0000
+vn 0.0012 0.0100 -0.9999
+vn -0.0016 0.0089 -1.0000
+vn -0.0019 0.0091 -1.0000
+vn 0.0021 0.0096 -1.0000
+vn 0.0029 0.0054 -1.0000
+vn -0.0023 0.0081 -1.0000
+vn 0.0002 0.0101 -0.9999
+vn 0.0031 0.0077 -1.0000
+vn -0.0008 0.0093 -1.0000
+vn 0.0029 0.0087 -1.0000
+vn -0.0026 0.0071 -1.0000
+vn -0.0020 0.0059 -1.0000
+vn 0.0025 0.0057 -1.0000
+vn -0.0011 0.0002 -1.0000
+vn -0.0024 -0.0134 -0.9999
+vn -0.0044 -0.0144 -0.9999
+vn 0.1826 -0.9600 -0.2122
+vn 0.0096 0.0120 -0.9999
+vn -0.0081 0.0119 -0.9999
+vn 0.0092 0.0151 -0.9998
+vn 0.0078 0.0180 -0.9998
+vn 0.0052 0.0198 -0.9998
+vn -0.0010 0.0208 -0.9998
+vn -0.0040 0.0200 -0.9998
+vn -0.0063 0.0178 -0.9998
+vn -0.0076 0.0147 -0.9999
+vn -0.3615 -0.9097 -0.2042
+vn -0.0073 0.0089 -0.9999
+vn -0.0058 0.0057 -1.0000
+vn 0.9958 -0.0920 -0.0001
+vn -0.6371 0.7708 0.0030
+vn 0.6317 0.7752 0.0039
+vn -0.7708 -0.6371 -0.0049
+vn 0.4683 0.8836 0.0043
+vn -0.2870 -0.9579 -0.0081
+vn -0.2870 -0.9579 -0.0068
+vn 0.0945 0.9955 0.0045
+vn 0.1015 -0.9948 -0.0068
+vn 0.2870 0.9579 0.0045
+vn 0.4744 -0.8803 -0.0058
+vn 0.6371 -0.7708 -0.0049
+vn -0.8803 -0.4745 -0.0039
+vn 0.8836 -0.4683 -0.0026
+vn 0.9551 -0.2964 -0.0014
+vn -0.9951 -0.0990 -0.0017
+vn -0.9948 0.1021 -0.0012
+vn 0.8802 0.4744 0.0104
+vn 0.0022 0.0208 -0.9998
+vn 0.9554 0.2954 0.0060
+vn 0.0016 -0.0002 -1.0000
+vn 0.6316 0.7751 0.0164
+vn 0.4682 0.8834 0.0185
+vn 0.7708 0.6371 0.0034
+vn -0.4683 -0.8835 -0.0076
+vn -0.2936 0.9557 0.0195
+vn -0.4744 0.8801 0.0178
+vn -0.6370 0.7707 0.0154
+vn -0.7752 0.6316 0.0125
+vn -0.8835 0.4683 0.0090
+vn -0.9574 0.2888 0.0045
+vn 0.9940 0.1091 0.0003
+vn -0.9955 0.0946 0.0001
+vn -0.7708 -0.6370 -0.0058
+vn -0.6316 -0.7752 -0.0068
+vn 0.8803 0.4745 0.0027
+vn 0.0945 0.9953 0.0206
+vn 0.7752 -0.6317 -0.0038
+vn -0.4683 -0.8835 -0.0064
+vn -0.0945 -0.9955 -0.0082
+vn -0.1014 0.9948 0.0044
+vn -0.2936 0.9559 0.0041
+vn -0.4745 0.8802 0.0036
+vn -0.9529 -0.3033 -0.0028
+vn -0.7752 0.6317 0.0023
+vn -0.8836 0.4683 0.0015
+vn -0.9562 0.2926 0.0006
+vn 0.9948 0.1014 0.0015
+vn 0.2869 0.9578 0.0199
+vn -0.9559 -0.2936 -0.0031
+vn -0.8803 -0.4745 -0.0045
+vn 0.9559 0.2936 0.0031
+vn -0.1015 0.9946 0.0204
+vn -0.6317 -0.7752 -0.0058
+vn 0.2936 -0.9559 -0.0064
+vn 0.1015 -0.9948 -0.0081
+vn 0.2936 -0.9559 -0.0076
+vn 0.4744 -0.8803 -0.0069
+vn 0.6371 -0.7708 -0.0059
+vn 0.7752 -0.6317 -0.0046
+vn 0.8836 -0.4683 -0.0032
+vn 0.9580 -0.2869 -0.0017
+vn -0.9948 -0.1015 -0.0015
+vn 0.0197 0.0217 -0.9996
+vn -0.8803 -0.4745 -0.0022
+vn 0.9553 0.2956 0.0029
+vn 0.9955 -0.0946 0.0004
+vn 0.8803 0.4745 0.0015
+vn -0.6316 -0.7753 -0.0031
+vn -0.9530 -0.3031 -0.0015
+vn -0.2935 0.9560 0.0016
+vn -0.6371 0.7708 0.0010
+vn -0.0195 0.0359 -0.9992
+vn 0.0946 0.9955 0.0019
+vn 0.7752 -0.6317 -0.0013
+vn 0.1015 -0.9948 -0.0028
+vn 0.9949 0.1013 0.0018
+vn -0.8835 0.4684 0.0003
+vn 0.9958 -0.0920 0.0004
+vn -0.0051 0.0218 -0.9997
+vn -0.0054 0.0220 -0.9997
+vn 0.0057 0.0207 -0.9998
+vn 0.0115 0.0479 -0.9988
+vn -0.0033 0.0503 -0.9987
+vn -0.9948 0.1020 -0.0009
+vn 0.6371 -0.7708 -0.0018
+vn -0.0944 -0.9955 -0.0029
+vn 0.0064 0.0131 -0.9999
+vn -0.0944 -0.9955 -0.0034
+vn -0.0110 -0.0348 -0.9993
+vn 0.0019 0.0244 -0.9997
+vn -0.7752 0.6316 0.0048
+vn -0.9955 0.0946 -0.0004
+vn 0.7751 -0.6318 -0.0016
+vn -0.9955 0.0945 0.0004
+vn 0.4683 0.8835 0.0079
+vn 0.0062 0.0160 -0.9999
+vn -0.0062 0.0192 -0.9998
+vn -0.4683 -0.8836 -0.0029
+vn -0.6317 -0.7752 -0.0027
+vn 0.0044 0.0503 -0.9987
+vn 0.0044 0.0504 -0.9987
+vn -0.0205 0.0286 -0.9994
+vn 0.2935 -0.9560 -0.0025
+vn 0.1018 -0.9948 -0.0033
+vn 0.0053 0.0138 -0.9999
+vn 0.2871 0.9579 0.0083
+vn -0.1015 0.9948 0.0084
+vn 0.8803 0.4744 0.0047
+vn -0.8836 0.4683 0.0034
+vn 0.0041 0.0229 -0.9997
+vn -0.2937 0.9559 0.0079
+vn -0.0163 0.0428 -0.9989
+vn -0.0164 0.0429 -0.9989
+vn -0.9948 -0.1015 -0.0010
+vn 0.2870 0.9579 0.0020
+vn 0.8836 -0.4682 -0.0010
+vn 0.0212 0.0364 -0.9991
+vn 0.9941 0.1089 0.0006
+vn 0.9955 -0.0945 -0.0001
+vn -0.0040 0.0004 -1.0000
+vn -0.7708 -0.6371 -0.0024
+vn -0.9951 -0.0991 -0.0011
+vn 0.4744 -0.8803 -0.0022
+vn -0.9948 -0.1016 -0.0010
+vn 0.6372 -0.7707 -0.0022
+vn -0.7707 -0.6371 -0.0027
+vn 0.8836 -0.4683 -0.0007
+vn -0.1015 0.9948 0.0018
+vn 0.6317 0.7752 0.0071
+vn -0.9562 0.2926 -0.0001
+vn 0.0175 0.0430 -0.9989
+vn -0.6370 0.7708 0.0061
+vn 0.9559 0.2937 0.0017
+vn -0.2869 -0.9580 -0.0029
+vn -0.0106 0.0478 -0.9988
+vn -0.0106 0.0481 -0.9988
+vn -0.3676 -0.9260 -0.0861
+vn 0.9579 -0.2869 -0.0003
+vn 0.0222 0.0291 -0.9993
+vn -0.0068 -0.0325 -0.9995
+vn 0.6317 0.7752 0.0019
+vn 0.2937 -0.9559 -0.0030
+vn -0.7753 0.6316 0.0006
+vn -0.9560 -0.2935 -0.0017
+vn 0.9948 0.1015 0.0010
+vn 0.0067 0.0187 -0.9998
+vn 0.0068 0.0187 -0.9998
+vn -0.0060 0.0143 -0.9999
+vn 0.7708 0.6370 0.0017
+vn -0.4745 0.8803 0.0013
+vn 0.4745 -0.8803 -0.0027
+vn -0.0138 0.0135 -0.9998
+vn -0.0143 0.0136 -0.9998
+vn -0.0141 0.0136 -0.9998
+vn -0.8803 -0.4744 -0.0020
+vn 0.0944 0.9955 0.0085
+vn 0.1861 -0.9785 -0.0894
+vn -0.0067 0.0168 -0.9998
+vn -0.0028 0.0230 -0.9997
+vn -0.4745 0.8802 0.0072
+vn 0.7708 0.6371 0.0060
+vn -0.0181 0.0214 -0.9996
+vn -0.0178 0.0212 -0.9996
+vn -0.4684 -0.8835 -0.0033
+vn -0.9574 0.2888 0.0015
+vn 0.0024 -0.0005 -1.0000
+vn 0.0023 -0.0006 -1.0000
+vn 0.9551 -0.2964 -0.0002
+vn -0.2869 -0.9579 -0.0034
+vn 0.4682 0.8836 0.0020
+vn -0.0007 0.0244 -0.9997
+vn -0.0946 -0.9955 -0.0070
+vn 0.7708 0.6370 0.0137
+vn 0.0064 0.0071 -1.0000
+vn 0.0038 0.0070 -1.0000
+vn 0.0012 0.0101 -0.9999
+vn -0.0014 0.0087 -1.0000
+vn 0.0023 0.0098 -0.9999
+vn 0.0070 0.0057 -1.0000
+vn -0.0020 0.0078 -1.0000
+vn 0.0002 0.0099 -1.0000
+vn 0.0037 0.0081 -1.0000
+vn -0.0010 0.0097 -1.0000
+vn 0.0032 0.0091 -1.0000
+vn -0.0015 0.0065 -1.0000
+vn -0.0012 0.0057 -1.0000
+vn 0.0037 0.0060 -1.0000
+vn -0.0001 -0.0002 -1.0000
+vn 0.0060 -0.0144 -0.9999
+vn 0.0057 -0.0144 -0.9999
+vn 0.0041 -0.0133 -0.9999
+vn 0.0038 -0.0134 -0.9999
+vn 0.3684 -0.9071 -0.2037
+vn 0.0080 0.0099 -0.9999
+vn -0.0066 0.0100 -0.9999
+vn 0.0082 0.0133 -0.9999
+vn 0.0073 0.0163 -0.9998
+vn 0.0052 0.0191 -0.9998
+vn -0.0010 0.0206 -0.9998
+vn -0.0039 0.0188 -0.9998
+vn -0.0058 0.0159 -0.9999
+vn -0.0069 0.0132 -0.9999
+vn -0.1757 -0.9613 -0.2124
+vn -0.0052 0.0071 -1.0000
+vn -0.0016 0.0054 -1.0000
+vn 0.4683 0.8835 0.0043
+vn -0.2869 -0.9579 -0.0080
+vn -0.2868 -0.9580 -0.0068
+vn 0.0946 0.9955 0.0045
+vn 0.2869 0.9580 0.0045
+vn 0.4745 -0.8802 -0.0058
+vn -0.8803 -0.4744 -0.0039
+vn 0.8836 -0.4682 -0.0026
+vn 0.9579 -0.2869 -0.0052
+vn 0.8803 0.4744 0.0104
+vn 0.0023 0.0203 -0.9998
+vn 0.9559 0.2935 0.0067
+vn 0.0028 0.0002 -1.0000
+vn 0.4683 0.8834 0.0185
+vn 0.7708 0.6370 0.0034
+vn -0.4745 0.8801 0.0178
+vn -0.6371 0.7707 0.0154
+vn -0.9580 0.2868 0.0053
+vn -0.7708 -0.6371 -0.0058
+vn -0.6317 -0.7752 -0.0068
+vn -0.2937 0.9559 0.0041
+vn -0.4744 0.8803 0.0036
+vn -0.9559 -0.2937 -0.0067
+vn -0.8835 0.4683 0.0015
+vn -0.9579 0.2869 0.0017
+vn -0.9559 -0.2937 -0.0031
+vn -0.8803 -0.4744 -0.0045
+vn 0.9541 0.2995 0.0020
+vn 0.8835 -0.4683 -0.0032
+vn -0.9948 -0.1015 -0.0027
+vn 0.0028 0.0065 -1.0000
+vn -0.0017 0.0089 -1.0000
+vn 0.0020 0.0092 -1.0000
+vn 0.0028 0.0053 -1.0000
+vn -0.0024 0.0081 -1.0000
+vn 0.0001 0.0101 -0.9999
+vn 0.0032 0.0076 -1.0000
+vn -0.0007 0.0093 -1.0000
+vn -0.0022 0.0069 -1.0000
+vn -0.0023 0.0061 -1.0000
+vn -0.0012 0.0001 -1.0000
+vn -0.0039 -0.0144 -0.9999
+vn 0.1825 -0.9600 -0.2122
+vn 0.0093 0.0118 -0.9999
+vn 0.0076 0.0177 -0.9998
+vn -0.0039 0.0200 -0.9998
+vn -0.0064 0.0177 -0.9998
+vn -0.0078 0.0151 -0.9999
+vn -0.0069 0.0087 -0.9999
+vn -0.0054 0.0056 -1.0000
+vn -0.9951 -0.0991 -0.0017
+vn 0.0013 -0.0002 -1.0000
+vn -0.9529 -0.3032 -0.0028
+vn -0.9562 0.2927 0.0006
+vn 0.4745 -0.8802 -0.0069
+vn -0.0015 0.0087 -1.0000
+vn 0.0024 0.0098 -0.9999
+vn -0.0016 0.0075 -1.0000
+vn 0.0001 0.0099 -1.0000
+vn 0.0038 0.0081 -1.0000
+vn -0.0012 0.0097 -1.0000
+vn 0.0029 0.0088 -1.0000
+vn -0.0019 0.0067 -1.0000
+vn -0.0011 0.0057 -1.0000
+vn 0.0033 0.0059 -1.0000
+vn -0.0001 -0.0003 -1.0000
+vn 0.0059 -0.0144 -0.9999
+vn 0.0037 -0.0134 -0.9999
+vn 0.3680 -0.9073 -0.2037
+vn 0.0080 0.0100 -0.9999
+vn -0.0063 0.0096 -0.9999
+vn 0.0083 0.0133 -0.9999
+vn 0.0083 0.0134 -0.9999
+vn 0.0081 0.0130 -0.9999
+vn 0.0073 0.0164 -0.9998
+vn 0.0051 0.0187 -0.9998
+vn -0.0038 0.0188 -0.9998
+vn -0.0060 0.0163 -0.9998
+vn -0.0048 0.0069 -1.0000
+vn -0.0016 0.0053 -1.0000
+vn -0.0013 0.0053 -1.0000
+vn 0.0024 0.0002 -1.0000
+vn -0.9559 -0.2937 -0.0066
+vn -0.0947 -0.9955 -0.0029
+vn 0.0145 0.0170 -0.9997
+vn 0.0142 0.0169 -0.9998
+vn 0.0088 0.0173 -0.9998
+vn 0.0020 0.0245 -0.9997
+vn 0.0020 0.0247 -0.9997
+vn -0.0041 0.0206 -0.9998
+vn 0.0046 0.0238 -0.9997
+vn 0.0158 0.0138 -0.9998
+vn -0.0051 0.0184 -0.9998
+vn -0.0006 0.0244 -0.9997
+vn -0.0006 0.0241 -0.9997
+vn 0.0078 0.0195 -0.9998
+vn -0.0034 0.0238 -0.9997
+vn -0.0035 0.0241 -0.9997
+vn 0.0064 0.0215 -0.9997
+vn -0.0048 0.0159 -0.9999
+vn -0.0036 0.0137 -0.9999
+vn -0.0037 0.0137 -0.9999
+vn 0.0074 0.0144 -0.9999
+vn 0.0073 0.0144 -0.9999
+vn -0.0010 -0.0005 -1.0000
+vn 0.0128 -0.0347 -0.9993
+vn 0.0080 -0.0323 -0.9994
+vn 0.3746 -0.9232 -0.0856
+vn 0.0183 0.0239 -0.9995
+vn -0.0168 0.0238 -0.9996
+vn -0.0167 0.0237 -0.9996
+vn 0.0190 0.0320 -0.9993
+vn 0.0189 0.0320 -0.9993
+vn 0.0166 0.0394 -0.9991
+vn 0.0166 0.0391 -0.9991
+vn 0.0166 0.0396 -0.9991
+vn 0.0113 0.0455 -0.9989
+vn -0.0035 0.0496 -0.9988
+vn -0.0104 0.0459 -0.9989
+vn -0.0153 0.0394 -0.9991
+vn -0.0153 0.0393 -0.9991
+vn -0.0175 0.0316 -0.9993
+vn -0.0174 0.0314 -0.9994
+vn -0.1791 -0.9797 -0.0896
+vn -0.0127 0.0166 -0.9998
+vn -0.0050 0.0129 -0.9999
+vn -0.6369 0.7709 0.0010
+vn 0.6316 0.7753 0.0019
+vn 0.4683 0.8836 0.0020
+vn 0.4745 -0.8803 -0.0022
+vn 0.6370 -0.7709 -0.0018
+vn -0.8803 -0.4745 -0.0020
+vn 0.9579 -0.2871 -0.0018
+vn 0.8803 0.4743 0.0047
+vn 0.0047 0.0493 -0.9988
+vn 0.9559 0.2935 0.0032
+vn 0.0051 0.0005 -1.0000
+vn 0.4682 0.8836 0.0078
+vn -0.7753 0.6315 0.0048
+vn -0.9579 0.2869 0.0018
+vn -0.7708 -0.6371 -0.0027
+vn 0.7753 -0.6316 -0.0013
+vn -0.0946 -0.9955 -0.0034
+vn -0.1016 0.9948 0.0018
+vn -0.2937 0.9559 0.0016
+vn -0.8836 0.4683 0.0002
+vn -0.9580 0.2868 0.0003
+vn 0.9541 0.2995 0.0012
+vn 0.2937 -0.9559 -0.0025
+vn 0.1016 -0.9948 -0.0033
+vn 0.2935 -0.9559 -0.0030
+vn 0.7751 -0.6319 -0.0016
+vn 0.8835 -0.4684 -0.0010
+vn -0.9580 0.2867 0.0003
+vn -0.2871 -0.9579 -0.0029
+vn 0.9941 0.1089 0.0005
+vn 0.0025 -0.0005 -1.0000
+vn 0.0027 -0.0005 -1.0000
+vn 0.0026 -0.0005 -1.0000
+vn 0.0133 -0.0347 -0.9993
+vn 0.0130 -0.0347 -0.9993
+vn 0.0131 -0.0347 -0.9993
+vn 0.0044 0.0502 -0.9987
+vn 0.0050 0.0137 -0.9999
+vn 0.0066 0.0218 -0.9997
+vn 0.6318 0.7751 0.0071
+vn -0.1013 0.9948 0.0018
+vn 0.8836 -0.4683 -0.0010
+vn -0.7708 -0.6370 -0.0027
+vn -0.0051 0.0129 -0.9999
+vn -0.0069 0.0168 -0.9998
+vn -0.8836 0.4682 0.0034
+vn 0.0224 0.0294 -0.9993
+vn 0.0226 0.0297 -0.9993
+vn 0.0223 0.0293 -0.9993
+vn -0.9574 0.2889 0.0015
+vn 0.0039 0.0226 -0.9997
+vn 0.0947 0.9955 0.0086
+vn -0.0042 0.0206 -0.9998
+vn 0.6317 0.7752 0.0070
+vn 0.0182 0.0239 -0.9995
+vn -0.4683 -0.8836 -0.0028
+vn -0.0128 0.0166 -0.9998
+vn -0.0067 -0.0325 -0.9994
+vn -0.2937 0.9559 0.0080
+vn 0.0944 0.9955 0.0019
+vn -0.0009 -0.0005 -1.0000
+vn -0.6371 0.7707 0.0061
+vn -0.2869 -0.9579 -0.0035
+vn 0.7752 -0.6317 -0.0016
+vn -0.0049 0.0217 -0.9998
+vn 0.0213 0.0366 -0.9991
+vn -0.0039 0.0004 -1.0000
+vn -0.2935 0.9559 0.0079
+vn 0.9579 -0.2871 -0.0017
+vn 0.0068 0.0142 -0.9999
+vn 0.4745 -0.8803 -0.0026
+vn -0.0053 0.0161 -0.9999
+vn -0.0055 0.0163 -0.9999
+vn 0.1855 -0.9786 -0.0894
+vn 0.0020 0.0244 -0.9997
+vn -0.0165 0.0434 -0.9989
+vn -0.0028 0.0226 -0.9997
+vn -0.0028 0.0225 -0.9997
+vn -0.0028 0.0227 -0.9997
+vn 0.2869 0.9579 0.0083
+vn -0.6318 -0.7751 -0.0027
+vn 0.0018 0.0240 -0.9997
+vn -0.0033 0.0505 -0.9987
+vn -0.0152 0.0389 -0.9991
+vn -0.0946 -0.9955 -0.0035
+vn -0.0166 0.0237 -0.9996
+vn 0.0076 0.0195 -0.9998
+vn 0.0144 0.0171 -0.9998
+vn 0.1013 -0.9948 -0.0033
+vn -0.9951 -0.0986 -0.0011
+vn -0.9951 -0.0987 -0.0011
+vn -0.0175 0.0315 -0.9994
+vn -0.0172 0.0312 -0.9994
+vn -0.9955 0.0943 -0.0005
+vn 0.7709 0.6369 0.0017
+vn -0.7752 0.6317 0.0006
+vn 0.0193 0.0326 -0.9993
+vn 0.9579 -0.2870 -0.0003
+vn 0.0083 -0.0323 -0.9994
+vn -0.4683 -0.8836 -0.0033
+vn 0.0085 0.0171 -0.9998
+vn 0.0166 0.0397 -0.9991
+vn -0.9529 -0.3031 -0.0015
+vn 0.0063 0.0161 -0.9999
+vn 0.0061 0.0159 -0.9999
+vn 0.0058 0.0006 -1.0000
+vn -0.9955 0.0945 0.0003
+vn 0.9541 0.2994 0.0012
+vn -0.0033 0.0237 -0.9997
+vn 0.8836 -0.4682 -0.0007
+vn -0.0114 -0.0347 -0.9993
+vn -0.6317 -0.7752 -0.0031
+vn 0.9948 0.1016 0.0010
+vn -0.0061 0.0143 -0.9999
+vn -0.0062 0.0144 -0.9999
+vn -0.0180 0.0213 -0.9996
+vn 0.9553 0.2955 0.0029
+vn 0.0061 0.0131 -0.9999
+vn -0.0065 0.0195 -0.9998
+vn -0.0061 0.0192 -0.9998
+vn -0.0200 0.0283 -0.9994
+vn -0.0034 0.0136 -0.9999
+vn -0.0196 0.0359 -0.9992
+vn 0.0059 0.0210 -0.9998
+vn 0.0007 0.0044 -1.0000
+vn 0.0007 0.0040 -1.0000
+vn 0.0007 0.0045 -1.0000
+vn 0.0002 0.0042 -1.0000
+vn 0.0006 0.0040 -1.0000
+vn 0.0008 0.0044 -1.0000
+vn 0.0007 0.0046 -1.0000
+vn 0.0008 0.0045 -1.0000
+vn 0.0005 0.0043 -1.0000
+vn 0.0004 0.0044 -1.0000
+vn 0.0001 0.0043 -1.0000
+vn 0.0011 0.0038 -1.0000
+vn 0.0008 0.0038 -1.0000
+vn 0.0006 0.0041 -1.0000
+vn 0.0007 0.0039 -1.0000
+vn 0.0008 0.0041 -1.0000
+vn 0.0006 0.0039 -1.0000
+vn 0.0020 0.0042 -1.0000
+vn 0.0008 0.0043 -1.0000
+vn 0.0005 0.0109 -0.9999
+vn 0.0008 0.0097 -1.0000
+vn 0.0013 0.0092 -1.0000
+vn 0.0004 0.0107 -0.9999
+vn 0.0036 0.0102 -0.9999
+vn 0.0037 0.0102 -0.9999
+vn 0.0009 0.0095 -1.0000
+vn 0.0004 0.0102 -0.9999
+vn 0.0005 0.0099 -1.0000
+vn 0.0008 0.0109 -0.9999
+vn 0.0006 0.0095 -1.0000
+vn 0.0005 0.0092 -1.0000
+vn 0.0008 0.0098 -1.0000
+vn 0.0008 0.0099 -1.0000
+vn 0.0007 0.0096 -1.0000
+vn 0.0008 0.0106 -0.9999
+vn 0.0007 0.0107 -0.9999
+vn 0.0007 0.0109 -0.9999
+vn 0.0009 0.0104 -0.9999
+vn 0.0007 0.0110 -0.9999
+vn 0.0006 0.0107 -0.9999
+vn 0.0007 0.0108 -0.9999
+vn 0.0008 0.0108 -0.9999
+vn 0.0009 0.0108 -0.9999
+vn 0.0009 0.0093 -1.0000
+vn -0.0003 0.0100 -1.0000
+vn 0.0008 0.0093 -1.0000
+vn 0.0008 0.0107 -0.9999
+vn 0.0011 0.0107 -0.9999
+vn 0.0005 0.0104 -0.9999
+vn 0.0010 0.0104 -0.9999
+vn 0.0004 0.0041 -1.0000
+vn 0.0006 0.0047 -1.0000
+vn 0.0005 0.0041 -1.0000
+vn 0.0008 0.0040 -1.0000
+vn 0.0010 0.0044 -1.0000
+vn 0.0003 0.0047 -1.0000
+vn 0.0008 0.0039 -1.0000
+vn 0.0006 0.0044 -1.0000
+vn 0.0009 0.0038 -1.0000
+vn 0.0010 0.0038 -1.0000
+vn 0.0008 0.0046 -1.0000
+vn 0.0009 0.0044 -1.0000
+vn 0.0003 0.0037 -1.0000
+vn 0.0015 0.0042 -1.0000
+vn -0.0001 0.0099 -1.0000
+vn 0.0006 0.0110 -0.9999
+vn 0.0005 0.0103 -0.9999
+vn 0.0008 0.0096 -1.0000
+vn 0.0008 0.0095 -1.0000
+vn 0.0039 0.0103 -0.9999
+vn 0.0038 0.0102 -0.9999
+vn 0.0019 0.0089 -1.0000
+vn 0.0009 0.0109 -0.9999
+vn 0.0017 0.0104 -0.9999
+vn 0.0014 0.0106 -0.9999
+vn 0.0011 0.0096 -1.0000
+vn 0.0009 0.0096 -1.0000
+vn 0.0004 0.0097 -1.0000
+vn 0.0005 0.0096 -1.0000
+vn 0.0006 0.0097 -1.0000
+vn -0.0002 0.0099 -1.0000
+vn 0.0005 0.0095 -1.0000
+vn -0.0002 0.0100 -1.0000
+vt 0.357039 0.250000
+vt 0.785559 0.750000
+vt 0.214079 0.250000
+vt 0.714079 0.750000
+vt 0.285559 0.250000
+vt 0.571118 0.250000
+vt 0.285559 0.250000
+vt 0.642599 0.250000
+vt 0.928520 0.154203
+vt 0.499638 0.250000
+vt 0.714079 0.750000
+vt 0.571118 0.250000
+vt 0.642599 0.750000
+vt 0.000000 0.250000
+vt 0.857040 0.154203
+vt 0.071480 0.250000
+vt 0.857040 0.480583
+vt 0.357039 0.095797
+vt 0.785559 0.904203
+vt 0.785559 1.000000
+vt 0.357039 0.000000
+vt 0.785559 0.043154
+vt 0.714079 1.000000
+vt 0.214079 0.000000
+vt 0.857040 0.043154
+vt 0.928520 0.480583
+vt 0.214079 0.095797
+vt 0.714079 0.904203
+vt 0.285559 0.095797
+vt 0.571118 0.095797
+vt 0.285559 0.000000
+vt 0.571118 0.000000
+vt 0.857040 0.000000
+vt 0.285559 0.000000
+vt 0.642598 0.000000
+vt 0.785559 0.000000
+vt 0.285559 0.095797
+vt 0.642599 0.095797
+vt 0.857040 0.394495
+vt 0.928520 0.000000
+vt 0.499638 0.095797
+vt 0.499638 0.000000
+vt 0.785559 0.129242
+vt 0.714079 1.000000
+vt 0.571118 0.000000
+vt 0.785559 0.172396
+vt 0.714079 0.904203
+vt 0.571118 0.095797
+vt 0.642599 0.904203
+vt 0.000000 0.095797
+vt 0.642599 1.000000
+vt 0.000000 0.000000
+vt 0.857040 0.172396
+vt 0.071480 0.000000
+vt 0.857040 0.129242
+vt 0.928520 0.394495
+vt 0.857040 0.000000
+vt 0.071480 0.095797
+vt 0.578089 0.326242
+vt 0.370472 0.723575
+vt 0.442971 0.724510
+vt 0.846857 0.620495
+vt 0.579868 0.178984
+vt 0.229113 0.694984
+vt 0.442901 0.694984
+vt 0.817898 0.626255
+vt 0.589057 0.151745
+vt 0.229049 0.665457
+vt 0.442837 0.665456
+vt 0.788371 0.626255
+vt 0.605204 0.126737
+vt 0.228992 0.635930
+vt 0.442780 0.635930
+vt 0.759412 0.620495
+vt 0.626230 0.105756
+vt 0.228946 0.606403
+vt 0.442734 0.606402
+vt 0.732132 0.609195
+vt 0.650942 0.089526
+vt 0.228912 0.576877
+vt 0.442700 0.576876
+vt 0.707581 0.592792
+vt 0.678244 0.078632
+vt 0.228890 0.547350
+vt 0.442678 0.547349
+vt 0.686702 0.571913
+vt 0.228883 0.517823
+vt 0.706979 0.073452
+vt 0.442670 0.517822
+vt 0.670298 0.547363
+vt 0.228889 0.488295
+vt 0.735929 0.074111
+vt 0.442678 0.488295
+vt 0.658999 0.520082
+vt 0.763884 0.080481
+vt 0.228910 0.458769
+vt 0.442698 0.458768
+vt 0.653238 0.491123
+vt 0.789700 0.092175
+vt 0.228944 0.429242
+vt 0.442732 0.429241
+vt 0.653238 0.461596
+vt 0.812355 0.108585
+vt 0.228990 0.399716
+vt 0.442778 0.399714
+vt 0.658998 0.432637
+vt 0.831035 0.128981
+vt 0.229046 0.370189
+vt 0.442833 0.370188
+vt 0.670298 0.405358
+vt 0.844628 0.151667
+vt 0.229109 0.340661
+vt 0.442897 0.340661
+vt 0.686702 0.380807
+vt 0.879631 0.287703
+vt 0.370467 0.312068
+vt 0.442967 0.311134
+vt 0.707581 0.359928
+vt 0.370705 0.281246
+vt 0.885704 0.319257
+vt 0.443056 0.281620
+vt 0.732120 0.343530
+vt 0.371348 0.252525
+vt 0.885423 0.348954
+vt 0.443160 0.252098
+vt 0.759394 0.332230
+vt 0.879533 0.378395
+vt 0.371442 0.223476
+vt 0.443252 0.222558
+vt 0.788366 0.326464
+vt 0.867982 0.406513
+vt 0.371495 0.193949
+vt 0.443305 0.193031
+vt 0.817893 0.326463
+vt 0.851292 0.431762
+vt 0.371495 0.164423
+vt 0.443305 0.163503
+vt 0.846854 0.332224
+vt 0.830124 0.453224
+vt 0.371427 0.134897
+vt 0.443237 0.133976
+vt 0.874134 0.343522
+vt 0.805300 0.470133
+vt 0.371280 0.105372
+vt 0.443090 0.104449
+vt 0.898684 0.359926
+vt 0.777761 0.481894
+vt 0.371043 0.075847
+vt 0.442853 0.074922
+vt 0.919564 0.380805
+vt 0.748545 0.488099
+vt 0.370694 0.989321
+vt 0.370712 0.046322
+vt 0.442504 0.990249
+vt 0.442522 0.045396
+vt 0.935969 0.405357
+vt 0.718747 0.488548
+vt 0.371029 0.959797
+vt 0.442838 0.960722
+vt 0.947270 0.432638
+vt 0.689480 0.483245
+vt 0.371269 0.930272
+vt 0.443079 0.931196
+vt 0.953029 0.461597
+vt 0.661830 0.472412
+vt 0.371421 0.900745
+vt 0.443231 0.901668
+vt 0.953029 0.491125
+vt 0.371492 0.871219
+vt 0.636826 0.456466
+vt 0.443302 0.872141
+vt 0.947269 0.520085
+vt 0.615392 0.436018
+vt 0.371495 0.841694
+vt 0.443305 0.842613
+vt 0.935968 0.547365
+vt 0.598317 0.411848
+vt 0.371443 0.812167
+vt 0.443254 0.813086
+vt 0.919563 0.571916
+vt 0.371352 0.783117
+vt 0.586362 0.385333
+vt 0.443163 0.783546
+vt 0.898673 0.592803
+vt 0.370709 0.754396
+vt 0.579568 0.357437
+vt 0.443060 0.754024
+vt 0.874125 0.609201
+vt 0.571207 0.326153
+vt 0.177644 0.253134
+vt 0.006929 0.253167
+vt 0.319029 0.223358
+vt 0.573209 0.178195
+vt 0.007025 0.222313
+vt 0.582798 0.148853
+vt 0.319126 0.192504
+vt 0.007122 0.191458
+vt 0.599888 0.122511
+vt 0.319224 0.161649
+vt 0.007220 0.160604
+vt 0.621946 0.100560
+vt 0.319322 0.130795
+vt 0.007319 0.129750
+vt 0.647824 0.083609
+vt 0.319423 0.099941
+vt 0.007419 0.098896
+vt 0.676401 0.072253
+vt 0.319524 0.069087
+vt 0.007520 0.068042
+vt 0.706468 0.066881
+vt 0.319626 0.038233
+vt 0.007623 0.037188
+vt 0.319730 0.007378
+vt 0.736753 0.067631
+vt 0.319730 0.992485
+vt 0.007727 0.006334
+vt 0.007728 0.993666
+vt 0.765981 0.074365
+vt 0.319616 0.961631
+vt 0.007613 0.962812
+vt 0.792953 0.086670
+vt 0.319502 0.930776
+vt 0.007499 0.931958
+vt 0.816609 0.103895
+vt 0.319390 0.899923
+vt 0.007387 0.901104
+vt 0.836176 0.125250
+vt 0.319278 0.869069
+vt 0.007276 0.870250
+vt 0.850517 0.149568
+vt 0.319168 0.838215
+vt 0.007165 0.839396
+vt 0.886223 0.285996
+vt 0.177771 0.808498
+vt 0.007055 0.808541
+vt 0.177512 0.777645
+vt 0.892437 0.317355
+vt 0.006941 0.777687
+vt 0.176852 0.746796
+vt 0.892437 0.349226
+vt 0.006827 0.746834
+vt 0.886228 0.380492
+vt 0.176742 0.715941
+vt 0.006716 0.715979
+vt 0.874138 0.409852
+vt 0.176640 0.685087
+vt 0.006614 0.685125
+vt 0.856681 0.436209
+vt 0.176549 0.654233
+vt 0.006524 0.654271
+vt 0.834553 0.458608
+vt 0.176474 0.623379
+vt 0.006448 0.623417
+vt 0.808610 0.476251
+vt 0.176415 0.592525
+vt 0.006390 0.592562
+vt 0.779839 0.488518
+vt 0.176377 0.561671
+vt 0.006351 0.561708
+vt 0.749318 0.494983
+vt 0.176359 0.530816
+vt 0.006334 0.530854
+vt 0.176362 0.499962
+vt 0.718195 0.495436
+vt 0.006337 0.500000
+vt 0.687628 0.489885
+vt 0.176387 0.469108
+vt 0.006361 0.469146
+vt 0.658752 0.478560
+vt 0.176430 0.438254
+vt 0.006405 0.438291
+vt 0.632640 0.461897
+vt 0.176492 0.407399
+vt 0.006467 0.407437
+vt 0.610260 0.440534
+vt 0.176568 0.376545
+vt 0.006543 0.376583
+vt 0.592437 0.415287
+vt 0.176656 0.345691
+vt 0.006631 0.345729
+vt 0.176753 0.314837
+vt 0.579817 0.387126
+vt 0.006727 0.314875
+vt 0.177399 0.283987
+vt 0.572791 0.357181
+vt 0.006828 0.284020
+vt 0.578089 0.326242
+vt 0.370472 0.723575
+vt 0.442971 0.724510
+vt 0.231212 0.903091
+vt 0.579868 0.178984
+vt 0.229113 0.694984
+vt 0.442901 0.694984
+vt 0.214996 0.906316
+vt 0.589057 0.151745
+vt 0.229049 0.665457
+vt 0.442837 0.665456
+vt 0.198460 0.906317
+vt 0.605204 0.126737
+vt 0.228992 0.635930
+vt 0.442780 0.635930
+vt 0.182243 0.903091
+vt 0.626230 0.105756
+vt 0.228946 0.606403
+vt 0.442734 0.606402
+vt 0.166967 0.896763
+vt 0.650942 0.089526
+vt 0.228912 0.576877
+vt 0.442700 0.576876
+vt 0.153218 0.887577
+vt 0.678244 0.078632
+vt 0.228890 0.547350
+vt 0.442678 0.547349
+vt 0.141526 0.875885
+vt 0.228883 0.517823
+vt 0.706979 0.073452
+vt 0.442670 0.517822
+vt 0.132340 0.862137
+vt 0.228889 0.488295
+vt 0.735929 0.074111
+vt 0.442678 0.488295
+vt 0.126012 0.846860
+vt 0.763884 0.080481
+vt 0.228910 0.458769
+vt 0.442698 0.458768
+vt 0.122786 0.830643
+vt 0.789700 0.092175
+vt 0.228944 0.429242
+vt 0.442732 0.429241
+vt 0.122786 0.814107
+vt 0.812355 0.108585
+vt 0.228990 0.399716
+vt 0.442778 0.399714
+vt 0.126012 0.797890
+vt 0.831035 0.128981
+vt 0.229046 0.370189
+vt 0.442833 0.370188
+vt 0.132339 0.782614
+vt 0.844628 0.151667
+vt 0.229109 0.340661
+vt 0.442897 0.340661
+vt 0.141526 0.768865
+vt 0.879631 0.287703
+vt 0.370467 0.312068
+vt 0.442967 0.311134
+vt 0.153218 0.757173
+vt 0.370705 0.281246
+vt 0.885704 0.319257
+vt 0.443056 0.281620
+vt 0.166960 0.747990
+vt 0.371348 0.252525
+vt 0.885423 0.348954
+vt 0.443160 0.252098
+vt 0.182233 0.741662
+vt 0.879533 0.378395
+vt 0.371442 0.223476
+vt 0.443252 0.222558
+vt 0.198458 0.738433
+vt 0.867982 0.406513
+vt 0.371495 0.193949
+vt 0.443305 0.193031
+vt 0.214993 0.738433
+vt 0.851292 0.431762
+vt 0.371495 0.164423
+vt 0.443305 0.163503
+vt 0.231211 0.741659
+vt 0.830124 0.453224
+vt 0.371427 0.134897
+vt 0.443237 0.133976
+vt 0.246487 0.747986
+vt 0.805300 0.470133
+vt 0.371280 0.105372
+vt 0.443090 0.104449
+vt 0.260236 0.757172
+vt 0.777761 0.481894
+vt 0.371043 0.075847
+vt 0.442853 0.074922
+vt 0.271928 0.768865
+vt 0.748545 0.488099
+vt 0.370694 0.989321
+vt 0.370712 0.046322
+vt 0.442504 0.990249
+vt 0.442522 0.045396
+vt 0.281116 0.782613
+vt 0.718747 0.488548
+vt 0.371029 0.959797
+vt 0.442838 0.960722
+vt 0.287444 0.797891
+vt 0.689480 0.483245
+vt 0.371269 0.930272
+vt 0.443079 0.931196
+vt 0.290669 0.814108
+vt 0.661830 0.472412
+vt 0.371421 0.900745
+vt 0.443231 0.901668
+vt 0.290669 0.830644
+vt 0.371492 0.871219
+vt 0.636826 0.456466
+vt 0.443302 0.872141
+vt 0.287443 0.846861
+vt 0.615392 0.436018
+vt 0.371495 0.841694
+vt 0.443305 0.842613
+vt 0.281115 0.862138
+vt 0.598317 0.411848
+vt 0.371443 0.812167
+vt 0.443254 0.813086
+vt 0.271928 0.875886
+vt 0.371352 0.783117
+vt 0.586362 0.385333
+vt 0.443163 0.783546
+vt 0.260230 0.887583
+vt 0.370709 0.754396
+vt 0.579568 0.357437
+vt 0.443060 0.754024
+vt 0.246483 0.896766
+vt 0.571207 0.326153
+vt 0.177644 0.253134
+vt 0.006929 0.253167
+vt 0.319029 0.223358
+vt 0.573209 0.178195
+vt 0.007025 0.222313
+vt 0.582798 0.148853
+vt 0.319126 0.192504
+vt 0.007122 0.191458
+vt 0.599888 0.122511
+vt 0.319224 0.161649
+vt 0.007220 0.160604
+vt 0.621946 0.100560
+vt 0.319322 0.130795
+vt 0.007319 0.129750
+vt 0.647824 0.083609
+vt 0.319423 0.099941
+vt 0.007419 0.098896
+vt 0.676401 0.072253
+vt 0.319524 0.069087
+vt 0.007520 0.068042
+vt 0.706468 0.066881
+vt 0.319626 0.038233
+vt 0.007623 0.037188
+vt 0.319730 0.007378
+vt 0.736753 0.067631
+vt 0.319730 0.992485
+vt 0.007727 0.006334
+vt 0.007728 0.993666
+vt 0.765981 0.074365
+vt 0.319616 0.961631
+vt 0.007613 0.962812
+vt 0.792953 0.086670
+vt 0.319502 0.930776
+vt 0.007499 0.931958
+vt 0.816609 0.103895
+vt 0.319390 0.899923
+vt 0.007387 0.901104
+vt 0.836176 0.125250
+vt 0.319278 0.869069
+vt 0.007276 0.870250
+vt 0.850517 0.149568
+vt 0.319168 0.838215
+vt 0.007165 0.839396
+vt 0.886223 0.285996
+vt 0.177771 0.808498
+vt 0.007055 0.808541
+vt 0.177512 0.777645
+vt 0.892437 0.317355
+vt 0.006941 0.777687
+vt 0.176852 0.746796
+vt 0.892437 0.349226
+vt 0.006827 0.746834
+vt 0.886228 0.380492
+vt 0.176742 0.715941
+vt 0.006716 0.715979
+vt 0.874138 0.409852
+vt 0.176640 0.685087
+vt 0.006614 0.685125
+vt 0.856681 0.436209
+vt 0.176549 0.654233
+vt 0.006524 0.654271
+vt 0.834553 0.458608
+vt 0.176474 0.623379
+vt 0.006448 0.623417
+vt 0.808610 0.476251
+vt 0.176415 0.592525
+vt 0.006390 0.592562
+vt 0.779839 0.488518
+vt 0.176377 0.561671
+vt 0.006351 0.561708
+vt 0.749318 0.494983
+vt 0.176359 0.530816
+vt 0.006334 0.530854
+vt 0.176362 0.499962
+vt 0.718195 0.495436
+vt 0.006337 0.500000
+vt 0.687628 0.489885
+vt 0.176387 0.469108
+vt 0.006361 0.469146
+vt 0.658752 0.478560
+vt 0.176430 0.438254
+vt 0.006405 0.438291
+vt 0.632640 0.461897
+vt 0.176492 0.407399
+vt 0.006467 0.407437
+vt 0.610260 0.440534
+vt 0.176568 0.376545
+vt 0.006543 0.376583
+vt 0.592437 0.415287
+vt 0.176656 0.345691
+vt 0.006631 0.345729
+vt 0.176753 0.314837
+vt 0.579817 0.387126
+vt 0.006727 0.314875
+vt 0.177399 0.283987
+vt 0.572791 0.357181
+vt 0.006828 0.284020
+vt 0.578089 0.326242
+vt 0.370472 0.723575
+vt 0.442971 0.724510
+vt 0.278498 0.614635
+vt 0.579868 0.178984
+vt 0.229113 0.694984
+vt 0.442901 0.694984
+vt 0.249539 0.620395
+vt 0.589057 0.151745
+vt 0.229049 0.665457
+vt 0.442837 0.665456
+vt 0.220011 0.620396
+vt 0.605204 0.126737
+vt 0.228992 0.635930
+vt 0.442780 0.635930
+vt 0.191052 0.614636
+vt 0.626230 0.105756
+vt 0.228946 0.606403
+vt 0.442734 0.606402
+vt 0.163773 0.603336
+vt 0.650942 0.089526
+vt 0.228912 0.576877
+vt 0.442700 0.576876
+vt 0.139222 0.586933
+vt 0.678244 0.078632
+vt 0.228890 0.547350
+vt 0.442678 0.547349
+vt 0.118343 0.566054
+vt 0.228883 0.517823
+vt 0.706979 0.073452
+vt 0.442670 0.517822
+vt 0.101939 0.541503
+vt 0.228889 0.488295
+vt 0.735929 0.074111
+vt 0.442678 0.488295
+vt 0.090640 0.514223
+vt 0.763884 0.080481
+vt 0.228910 0.458769
+vt 0.442698 0.458768
+vt 0.084879 0.485264
+vt 0.789700 0.092175
+vt 0.228944 0.429242
+vt 0.442732 0.429241
+vt 0.084879 0.455737
+vt 0.812355 0.108585
+vt 0.228990 0.399716
+vt 0.442778 0.399714
+vt 0.090639 0.426777
+vt 0.831035 0.128981
+vt 0.229046 0.370189
+vt 0.442833 0.370188
+vt 0.101938 0.399499
+vt 0.844628 0.151667
+vt 0.229109 0.340661
+vt 0.442897 0.340661
+vt 0.118343 0.374948
+vt 0.879631 0.287703
+vt 0.370467 0.312068
+vt 0.442967 0.311134
+vt 0.139222 0.354069
+vt 0.370705 0.281246
+vt 0.885704 0.319257
+vt 0.443056 0.281620
+vt 0.163760 0.337671
+vt 0.371348 0.252525
+vt 0.885423 0.348954
+vt 0.443160 0.252098
+vt 0.191035 0.326370
+vt 0.879533 0.378395
+vt 0.371442 0.223476
+vt 0.443252 0.222558
+vt 0.220007 0.320605
+vt 0.867982 0.406513
+vt 0.371495 0.193949
+vt 0.443305 0.193031
+vt 0.249534 0.320604
+vt 0.851292 0.431762
+vt 0.371495 0.164423
+vt 0.443305 0.163503
+vt 0.278494 0.326364
+vt 0.830124 0.453224
+vt 0.371427 0.134897
+vt 0.443237 0.133976
+vt 0.305774 0.337663
+vt 0.805300 0.470133
+vt 0.371280 0.105372
+vt 0.443090 0.104449
+vt 0.330325 0.354067
+vt 0.777761 0.481894
+vt 0.371043 0.075847
+vt 0.442853 0.074922
+vt 0.351205 0.374946
+vt 0.748545 0.488099
+vt 0.370694 0.989321
+vt 0.370712 0.046322
+vt 0.442504 0.990249
+vt 0.442522 0.045396
+vt 0.367610 0.399497
+vt 0.718747 0.488548
+vt 0.371029 0.959797
+vt 0.442838 0.960722
+vt 0.378910 0.426778
+vt 0.689480 0.483245
+vt 0.371269 0.930272
+vt 0.443079 0.931196
+vt 0.384670 0.455738
+vt 0.661830 0.472412
+vt 0.371421 0.900745
+vt 0.443231 0.901668
+vt 0.384670 0.485266
+vt 0.371492 0.871219
+vt 0.636826 0.456466
+vt 0.443302 0.872141
+vt 0.378909 0.514226
+vt 0.615392 0.436018
+vt 0.371495 0.841694
+vt 0.443305 0.842613
+vt 0.367609 0.541505
+vt 0.598317 0.411848
+vt 0.371443 0.812167
+vt 0.443254 0.813086
+vt 0.351204 0.566056
+vt 0.371352 0.783117
+vt 0.586362 0.385333
+vt 0.443163 0.783546
+vt 0.330314 0.586943
+vt 0.370709 0.754396
+vt 0.579568 0.357437
+vt 0.443060 0.754024
+vt 0.305766 0.603342
+vt 0.571207 0.326153
+vt 0.177644 0.253134
+vt 0.006929 0.253167
+vt 0.319029 0.223358
+vt 0.573209 0.178195
+vt 0.007025 0.222313
+vt 0.582798 0.148853
+vt 0.319126 0.192504
+vt 0.007122 0.191458
+vt 0.599888 0.122511
+vt 0.319224 0.161649
+vt 0.007220 0.160604
+vt 0.621946 0.100560
+vt 0.319322 0.130795
+vt 0.007319 0.129750
+vt 0.647824 0.083609
+vt 0.319423 0.099941
+vt 0.007419 0.098896
+vt 0.676401 0.072253
+vt 0.319524 0.069087
+vt 0.007520 0.068042
+vt 0.706468 0.066881
+vt 0.319626 0.038233
+vt 0.007623 0.037188
+vt 0.319730 0.007378
+vt 0.736753 0.067631
+vt 0.319730 0.992485
+vt 0.007727 0.006334
+vt 0.007728 0.993666
+vt 0.765981 0.074365
+vt 0.319616 0.961631
+vt 0.007613 0.962812
+vt 0.792953 0.086670
+vt 0.319502 0.930776
+vt 0.007499 0.931958
+vt 0.816609 0.103895
+vt 0.319390 0.899923
+vt 0.007387 0.901104
+vt 0.836176 0.125250
+vt 0.319278 0.869069
+vt 0.007276 0.870250
+vt 0.850517 0.149568
+vt 0.319168 0.838215
+vt 0.007165 0.839396
+vt 0.886223 0.285996
+vt 0.177771 0.808498
+vt 0.007055 0.808541
+vt 0.177512 0.777645
+vt 0.892437 0.317355
+vt 0.006941 0.777687
+vt 0.176852 0.746796
+vt 0.892437 0.349226
+vt 0.006827 0.746834
+vt 0.886228 0.380492
+vt 0.176742 0.715941
+vt 0.006716 0.715979
+vt 0.874138 0.409852
+vt 0.176640 0.685087
+vt 0.006614 0.685125
+vt 0.856681 0.436209
+vt 0.176549 0.654233
+vt 0.006524 0.654271
+vt 0.834553 0.458608
+vt 0.176474 0.623379
+vt 0.006448 0.623417
+vt 0.808610 0.476251
+vt 0.176415 0.592525
+vt 0.006390 0.592562
+vt 0.779839 0.488518
+vt 0.176377 0.561671
+vt 0.006351 0.561708
+vt 0.749318 0.494983
+vt 0.176359 0.530816
+vt 0.006334 0.530854
+vt 0.176362 0.499962
+vt 0.718195 0.495436
+vt 0.006337 0.500000
+vt 0.687628 0.489885
+vt 0.176387 0.469108
+vt 0.006361 0.469146
+vt 0.658752 0.478560
+vt 0.176430 0.438254
+vt 0.006405 0.438291
+vt 0.632640 0.461897
+vt 0.176492 0.407399
+vt 0.006467 0.407437
+vt 0.610260 0.440534
+vt 0.176568 0.376545
+vt 0.006543 0.376583
+vt 0.592437 0.415287
+vt 0.176656 0.345691
+vt 0.006631 0.345729
+vt 0.176753 0.314837
+vt 0.579817 0.387126
+vt 0.006727 0.314875
+vt 0.177399 0.283987
+vt 0.572791 0.357181
+vt 0.006828 0.284020
+vt 0.006828 0.284020
+vt 0.572791 0.357181
+vt 0.177399 0.283987
+vt 0.006727 0.314875
+vt 0.176753 0.314837
+vt 0.579817 0.387126
+vt 0.006631 0.345729
+vt 0.176656 0.345691
+vt 0.592437 0.415287
+vt 0.006543 0.376583
+vt 0.610260 0.440534
+vt 0.176568 0.376545
+vt 0.006467 0.407437
+vt 0.632640 0.461897
+vt 0.176492 0.407399
+vt 0.006405 0.438291
+vt 0.176430 0.438254
+vt 0.658752 0.478560
+vt 0.006361 0.469146
+vt 0.176387 0.469108
+vt 0.687628 0.489885
+vt 0.006337 0.500000
+vt 0.718195 0.495436
+vt 0.176362 0.499962
+vt 0.006334 0.530854
+vt 0.749318 0.494983
+vt 0.176359 0.530816
+vt 0.006351 0.561708
+vt 0.779839 0.488518
+vt 0.176377 0.561671
+vt 0.006390 0.592562
+vt 0.808610 0.476251
+vt 0.176415 0.592525
+vt 0.006448 0.623417
+vt 0.176474 0.623379
+vt 0.834553 0.458608
+vt 0.006524 0.654271
+vt 0.176549 0.654233
+vt 0.856681 0.436209
+vt 0.006614 0.685125
+vt 0.176640 0.685087
+vt 0.874138 0.409852
+vt 0.006716 0.715979
+vt 0.886228 0.380492
+vt 0.176742 0.715941
+vt 0.006827 0.746834
+vt 0.892437 0.349226
+vt 0.176852 0.746796
+vt 0.006941 0.777687
+vt 0.892437 0.317355
+vt 0.177512 0.777645
+vt 0.007055 0.808541
+vt 0.886223 0.285996
+vt 0.177771 0.808498
+vt 0.007165 0.839396
+vt 0.850517 0.149568
+vt 0.319168 0.838215
+vt 0.007276 0.870250
+vt 0.836176 0.125250
+vt 0.319278 0.869069
+vt 0.007387 0.901104
+vt 0.816609 0.103895
+vt 0.319390 0.899923
+vt 0.007499 0.931958
+vt 0.319502 0.930776
+vt 0.792953 0.086670
+vt 0.007613 0.962812
+vt 0.319616 0.961631
+vt 0.765981 0.074365
+vt 0.007728 0.993666
+vt 0.007727 0.006334
+vt 0.319730 0.007378
+vt 0.736753 0.067631
+vt 0.319730 0.992485
+vt 0.007623 0.037188
+vt 0.319626 0.038233
+vt 0.706468 0.066881
+vt 0.007520 0.068042
+vt 0.676401 0.072253
+vt 0.319524 0.069087
+vt 0.007419 0.098896
+vt 0.319423 0.099941
+vt 0.647824 0.083609
+vt 0.007319 0.129750
+vt 0.319322 0.130795
+vt 0.621946 0.100560
+vt 0.007220 0.160604
+vt 0.599888 0.122511
+vt 0.319224 0.161649
+vt 0.007122 0.191458
+vt 0.319126 0.192504
+vt 0.582798 0.148853
+vt 0.007025 0.222313
+vt 0.319029 0.223358
+vt 0.573209 0.178195
+vt 0.006929 0.253167
+vt 0.177644 0.253134
+vt 0.571207 0.326153
+vt 0.443060 0.754024
+vt 0.208826 0.204735
+vt 0.579568 0.357437
+vt 0.370709 0.754396
+vt 0.443163 0.783546
+vt 0.222573 0.195552
+vt 0.586362 0.385333
+vt 0.371352 0.783117
+vt 0.443254 0.813086
+vt 0.234272 0.183855
+vt 0.598317 0.411848
+vt 0.371443 0.812167
+vt 0.443305 0.842613
+vt 0.243459 0.170107
+vt 0.615392 0.436018
+vt 0.371495 0.841694
+vt 0.443302 0.872141
+vt 0.249787 0.154830
+vt 0.636826 0.456466
+vt 0.371492 0.871219
+vt 0.443231 0.901668
+vt 0.253013 0.138613
+vt 0.371421 0.900745
+vt 0.661830 0.472412
+vt 0.443079 0.931196
+vt 0.253013 0.122077
+vt 0.371269 0.930272
+vt 0.689480 0.483245
+vt 0.442838 0.960722
+vt 0.249787 0.105860
+vt 0.718747 0.488548
+vt 0.371029 0.959797
+vt 0.442504 0.990249
+vt 0.442522 0.045396
+vt 0.243459 0.090582
+vt 0.370694 0.989321
+vt 0.748545 0.488099
+vt 0.370712 0.046322
+vt 0.442853 0.074922
+vt 0.234272 0.076833
+vt 0.777761 0.481894
+vt 0.371043 0.075847
+vt 0.443090 0.104449
+vt 0.222579 0.065141
+vt 0.805300 0.470133
+vt 0.371280 0.105372
+vt 0.443237 0.133976
+vt 0.208831 0.055955
+vt 0.371427 0.134897
+vt 0.830124 0.453224
+vt 0.443305 0.163503
+vt 0.193554 0.049628
+vt 0.371495 0.164423
+vt 0.851292 0.431762
+vt 0.443305 0.193031
+vt 0.177337 0.046402
+vt 0.867982 0.406513
+vt 0.371495 0.193949
+vt 0.443252 0.222558
+vt 0.160801 0.046402
+vt 0.879533 0.378395
+vt 0.371442 0.223476
+vt 0.443160 0.252098
+vt 0.144577 0.049631
+vt 0.371348 0.252525
+vt 0.885423 0.348954
+vt 0.443056 0.281620
+vt 0.129303 0.055959
+vt 0.370705 0.281246
+vt 0.885704 0.319257
+vt 0.442967 0.311134
+vt 0.115562 0.065142
+vt 0.370467 0.312068
+vt 0.879631 0.287703
+vt 0.442897 0.340661
+vt 0.103870 0.076834
+vt 0.229109 0.340661
+vt 0.844628 0.151667
+vt 0.442833 0.370188
+vt 0.094683 0.090583
+vt 0.229046 0.370189
+vt 0.831035 0.128981
+vt 0.442778 0.399714
+vt 0.088355 0.105859
+vt 0.812355 0.108585
+vt 0.228990 0.399716
+vt 0.442732 0.429241
+vt 0.085130 0.122076
+vt 0.228944 0.429242
+vt 0.789700 0.092175
+vt 0.442698 0.458768
+vt 0.085130 0.138611
+vt 0.763884 0.080481
+vt 0.228910 0.458769
+vt 0.442678 0.488295
+vt 0.088356 0.154828
+vt 0.228889 0.488295
+vt 0.735929 0.074111
+vt 0.442670 0.517822
+vt 0.094683 0.170106
+vt 0.706979 0.073452
+vt 0.228883 0.517823
+vt 0.442678 0.547349
+vt 0.103870 0.183854
+vt 0.678244 0.078632
+vt 0.228890 0.547350
+vt 0.442700 0.576876
+vt 0.115562 0.195546
+vt 0.228912 0.576877
+vt 0.650942 0.089526
+vt 0.442734 0.606402
+vt 0.129310 0.204732
+vt 0.228946 0.606403
+vt 0.626230 0.105756
+vt 0.442780 0.635930
+vt 0.144587 0.211060
+vt 0.605204 0.126737
+vt 0.228992 0.635930
+vt 0.442837 0.665456
+vt 0.160804 0.214285
+vt 0.589057 0.151745
+vt 0.229049 0.665457
+vt 0.442901 0.694984
+vt 0.177339 0.214285
+vt 0.229113 0.694984
+vt 0.579868 0.178984
+vt 0.442971 0.724510
+vt 0.193556 0.211059
+vt 0.578089 0.326242
+vt 0.370472 0.723575
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.846034 0.501950
+vt 0.307543 0.501950
+vt 0.769051 0.501950
+vt 0.307542 0.501950
+vt 0.000000 0.501950
+vt 0.692068 0.501950
+vt 0.615085 0.501950
+vt 0.846034 0.996502
+vt 0.384526 0.501950
+vt 0.538102 0.501950
+vt 0.769051 0.501950
+vt 0.923017 0.996502
+vt 0.230559 0.501950
+vt 0.076983 0.501950
+vt 0.692068 0.501950
+vt 0.692068 0.192341
+vt 0.923017 0.650215
+vt 0.076983 0.192341
+vt 0.692068 0.000000
+vt 0.076983 0.000000
+vt 0.846034 0.429417
+vt 0.769051 0.000000
+vt 0.538102 0.000000
+vt 0.923017 0.429417
+vt 0.769051 0.192341
+vt 0.538102 0.192341
+vt 1.000000 0.650215
+vt 0.615085 0.192341
+vt 0.615085 0.000000
+vt 0.923017 0.343447
+vt 0.000000 0.000000
+vt 0.692068 0.000000
+vt 0.846034 0.343447
+vt 0.000000 0.192341
+vt 0.692068 0.192341
+vt 0.923017 0.686893
+vt 0.923017 0.478709
+vt 0.230559 0.192341
+vt 0.230559 0.000000
+vt 0.846034 0.600923
+vt 0.769051 0.000000
+vt 0.307542 0.000000
+vt 0.846034 0.686893
+vt 0.769051 0.192341
+vt 0.307542 0.192341
+vt 0.846034 0.192341
+vt 0.307542 0.192341
+vt 0.846034 0.000000
+vt 0.307542 0.000000
+vt 0.923017 0.686893
+vt 0.384525 0.000000
+vt 0.923017 0.600923
+vt 0.846034 0.686893
+vt 0.384525 0.192341
+vt 1.000000 0.478709
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.846034 1.000000
+vt 0.307543 1.000000
+vt 0.769051 1.000000
+vt 0.307542 1.000000
+vt 0.000000 1.000000
+vt 0.769051 1.000000
+vt 0.615085 1.000000
+vt 0.692068 1.000000
+vt 0.923017 0.307203
+vt 0.384526 1.000000
+vt 0.538102 1.000000
+vt 0.615085 0.501950
+vt 1.000000 0.307203
+vt 0.230559 1.000000
+vt 0.076983 1.000000
+vt 0.692068 0.501950
+vt 0.692068 0.809153
+vt 0.923017 0.478709
+vt 0.076983 0.692797
+vt 0.692068 1.000000
+vt 0.076983 0.501950
+vt 0.846034 0.085970
+vt 0.615085 1.000000
+vt 0.538102 0.501950
+vt 0.923017 0.085970
+vt 0.615085 0.809153
+vt 0.538102 0.692796
+vt 1.000000 0.478709
+vt 0.615085 0.692796
+vt 0.692068 0.692796
+vt 0.615085 0.501950
+vt 0.692068 0.501950
+vt 0.923017 0.000000
+vt 0.000000 0.501950
+vt 0.769051 0.501950
+vt 0.846034 0.000000
+vt 0.000000 0.692797
+vt 0.769051 0.692796
+vt 1.000000 0.000000
+vt 0.923017 0.307203
+vt 0.230559 0.692797
+vt 0.230559 0.501950
+vt 0.846034 0.257476
+vt 0.769051 0.501950
+vt 0.307542 0.501950
+vt 0.846034 0.343447
+vt 0.769051 0.692796
+vt 0.307542 0.692796
+vt 0.846034 0.692796
+vt 0.307542 0.692797
+vt 0.846034 0.501950
+vt 0.307542 0.501950
+vt 0.923017 0.343447
+vt 0.384525 0.501950
+vt 0.923017 0.257476
+vt 0.923017 0.000000
+vt 0.384525 0.692797
+vt 1.000000 0.307203
+vt 0.875462 0.748915
+vt 0.958487 0.000000
+vt 1.000000 1.000000
+vt 0.437731 0.748915
+vt 1.000000 0.500000
+vt 0.875462 0.000000
+vt 0.875462 0.000000
+vt 0.958487 0.748915
+vt 0.958487 0.500000
+vt 0.437731 0.000000
+vt 0.958487 0.000000
+vt 0.875462 0.748915
+vt 0.437731 0.748915
+vt 0.916975 0.748915
+vt 1.000000 0.500000
+vt 0.437731 0.000000
+vt 0.916975 0.000000
+vt 0.958487 1.000000
+vt 0.000000 0.000000
+vt 0.958487 0.500000
+vt 0.916974 0.000000
+vt 0.000000 0.748915
+vt 1.000000 0.000000
+vt 0.916975 0.748915
+s 0
+g KsHauptsignal_Plane.006_Mast
+usemtl Mast
+f 13/29/1 12/27/1 11/24/1 14/31/1
+f 9/18/2 16/37/2 15/34/2 10/21/2
+f 16/38/3 13/30/3 14/32/3 15/35/3
+f 12/28/4 9/19/4 10/20/4 11/23/4
+f 21/49/4 20/47/4 19/44/4 22/51/4
+f 24/58/1 21/50/1 22/52/1 23/54/1
+f 20/48/2 17/41/2 18/42/2 19/45/2
+f 24/58/1 23/54/1 11/24/1 12/27/1
+f 17/39/5 24/56/5 12/26/5 9/17/5
+f 18/42/2 17/41/2 9/18/2 10/21/2
+f 23/55/6 18/43/6 10/22/6 11/25/6
+f 19/46/6 18/43/6 23/55/6 22/53/6
+f 11/25/6 10/22/6 15/36/6 14/33/6
+f 3/5/1 2/3/1 12/27/1 13/29/1
+f 1/1/2 4/7/2 16/37/2 9/18/2
+f 4/8/3 3/6/3 13/30/3 16/38/3
+f 2/4/4 1/2/4 9/19/4 12/28/4
+f 7/13/4 6/11/4 20/47/4 21/49/4
+f 5/9/3 8/15/3 24/57/3 17/40/3
+f 8/16/1 7/14/1 21/50/1 24/58/1
+f 6/12/2 5/10/2 17/41/2 20/48/2
+f 544/974/6 541/973/6 537/971/6 538/972/6
+f 548/981/1 550/984/1 556/1000/1 557/1002/1
+f 557/1002/1 556/1000/1 555/997/1 558/1003/1
+f 552/988/2 547/979/2 560/1008/2 553/992/2
+f 553/992/2 560/1008/2 559/1005/2 554/994/2
+f 547/980/3 548/981/3 557/1002/3 560/1009/3
+f 560/1009/3 557/1002/3 558/1003/3 559/1006/3
+f 550/985/4 552/989/4 553/990/4 556/999/4
+f 556/999/4 553/990/4 554/993/4 555/996/4
+f 545/975/4 546/977/4 564/1018/4 565/1020/4
+f 565/1020/4 564/1018/4 563/1015/4 566/1022/4
+f 551/986/3 549/982/3 568/1027/3 561/1010/3
+f 549/983/1 545/976/1 565/1021/1 568/1028/1
+f 568/1028/1 565/1021/1 566/1023/1 567/1025/1
+f 546/978/2 551/987/2 561/1012/2 564/1019/2
+f 564/1019/2 561/1012/2 562/1013/2 563/1016/2
+f 568/1028/1 567/1025/1 555/997/1 556/1000/1
+f 561/1011/5 568/1029/5 556/1001/5 553/991/5
+f 562/1013/2 561/1012/2 553/992/2 554/994/2
+f 567/1026/6 562/1014/6 554/995/6 555/998/6
+f 563/1017/6 562/1014/6 567/1026/6 566/1024/6
+f 555/998/6 554/995/6 559/1007/6 558/1004/6
+f 576/1033/6 573/1032/6 569/1030/6 570/1031/6
+f 580/1040/1 582/1044/1 588/1060/1 589/1062/1
+f 589/1062/1 588/1060/1 587/1057/1 590/1064/1
+f 584/1048/2 579/1038/2 592/1070/2 585/1052/2
+f 585/1052/2 592/1070/2 591/1067/2 586/1054/2
+f 579/1039/3 580/1041/3 589/1063/3 592/1071/3
+f 592/1071/3 589/1063/3 590/1065/3 591/1068/3
+f 582/1045/4 584/1049/4 585/1050/4 588/1059/4
+f 588/1059/4 585/1050/4 586/1053/4 587/1056/4
+f 577/1034/4 578/1036/4 596/1080/4 597/1082/4
+f 597/1082/4 596/1080/4 595/1077/4 598/1084/4
+f 583/1046/3 581/1042/3 600/1089/3 593/1072/3
+f 581/1043/1 577/1035/1 597/1083/1 600/1090/1
+f 600/1090/1 597/1083/1 598/1085/1 599/1087/1
+f 578/1037/2 583/1047/2 593/1074/2 596/1081/2
+f 596/1081/2 593/1074/2 594/1075/2 595/1078/2
+f 600/1090/1 599/1087/1 587/1057/1 588/1060/1
+f 593/1073/5 600/1091/5 588/1061/5 585/1051/5
+f 594/1075/2 593/1074/2 585/1052/2 586/1054/2
+f 599/1088/6 594/1076/6 586/1055/6 587/1058/6
+f 595/1079/6 594/1076/6 599/1088/6 598/1086/6
+f 587/1058/6 586/1055/6 591/1069/6 590/1066/6
+g KsHauptsignal_Plane.006_Schirm
+usemtl Schirm
+f 601/1092/3 602/1095/3 604/1101/3 603/1098/3
+f 606/1107/4 605/1104/4 608/1113/4 607/1110/4
+f 603/1099/2 605/1105/2 606/1108/2 601/1093/2
+f 604/1102/6 608/1114/6 605/1106/6 603/1100/6
+f 601/1094/5 606/1109/5 607/1111/5 602/1096/5
+f 602/1097/1 607/1112/1 608/1115/1 604/1103/1
+g KsHauptsignal_Plane.006_Lampenrahmen
+usemtl Lampenrahmen
+f 40/89/7 39/87/7 41/91/7
+f 26/61/8 88/187/8 25/60/8
+f 54/117/9 53/116/9 56/121/9
+f 58/125/10 56/121/10 57/123/10
+f 86/183/11 85/181/11 88/187/11
+f 90/191/12 89/190/12 152/286/12
+f 118/235/13 120/238/13 117/234/13
+f 122/241/14 121/239/14 120/238/14
+f 150/283/15 152/286/15 149/281/15
+f 79/169/16 81/174/16 82/175/16
+f 81/173/17 145/275/17 147/278/17
+f 111/224/18 113/227/18 49/107/18
+f 103/210/19 105/214/19 41/92/19
+f 99/204/20 35/79/20 33/75/21
+f 105/214/22 107/218/22 43/95/22
+f 147/278/23 149/282/23 85/182/23
+f 97/201/24 33/75/24 31/71/24
+f 103/210/25 39/88/25 37/83/25
+f 109/221/26 111/224/26 47/103/26
+f 35/79/27 99/204/27 101/207/27
+f 107/218/28 109/221/28 45/99/28
+f 95/198/29 31/71/29 29/67/29
+f 93/195/30 29/67/30 27/63/30
+f 113/227/31 115/230/31 51/111/31
+f 151/285/32 89/189/32 25/59/32
+f 149/282/33 151/285/33 87/186/33
+f 121/240/34 57/124/34 55/120/34
+f 115/230/35 117/233/35 53/115/35
+f 79/170/36 143/272/36 145/275/36
+f 125/245/37 127/248/37 63/135/37
+f 77/165/38 141/269/38 143/272/38
+f 75/161/39 139/266/39 141/269/39
+f 73/157/40 137/264/40 139/266/40
+f 133/257/41 135/260/41 71/151/41
+f 131/254/42 133/257/42 69/147/42
+f 129/251/43 131/254/43 67/143/43
+f 127/248/44 129/251/44 65/139/44
+f 91/193/45 27/63/45 25/59/45
+f 123/242/46 125/245/46 61/131/46
+f 123/242/47 59/127/47 57/124/47
+f 88/187/48 87/185/48 25/60/48
+f 97/202/49 95/199/49 96/200/49
+f 112/226/50 111/225/50 109/222/50
+f 48/105/51 47/104/51 49/108/51
+f 110/223/52 109/222/52 107/219/52
+f 140/268/53 139/267/53 137/263/53
+f 42/93/54 41/91/54 43/96/54
+f 106/216/55 105/213/55 103/211/55
+f 37/84/56 39/87/56 40/89/56
+f 108/220/57 107/219/57 105/215/57
+f 33/76/58 35/80/58 36/81/58
+f 31/72/59 33/76/59 34/77/59
+f 50/109/60 49/108/60 51/112/60
+f 27/64/61 29/68/61 30/69/61
+f 25/60/62 27/64/62 28/65/62
+f 53/116/63 55/119/63 56/121/63
+f 56/121/64 55/119/64 57/123/64
+f 81/174/65 83/178/65 84/179/65
+f 71/151/66 135/260/66 137/264/66
+f 83/178/67 85/181/67 86/183/67
+f 119/237/68 55/120/68 53/115/68
+f 77/166/69 79/169/69 80/171/69
+f 75/162/70 77/166/70 78/167/70
+f 114/229/71 113/228/71 111/225/71
+f 142/271/72 141/270/72 139/267/72
+f 68/145/73 67/144/73 69/148/73
+f 66/141/74 65/140/74 67/144/74
+f 64/137/75 63/136/75 65/140/75
+f 62/133/76 61/132/76 63/136/76
+f 60/129/77 59/128/77 61/132/77
+f 58/125/78 57/123/78 59/128/78
+f 85/181/79 87/185/79 88/187/79
+f 89/190/80 151/284/80 152/286/80
+f 146/277/81 145/276/81 143/273/81
+f 144/274/82 143/273/82 141/270/82
+f 116/232/83 115/231/83 113/228/83
+f 71/152/84 73/158/84 74/159/84
+f 29/68/85 31/72/85 32/73/85
+f 44/97/86 43/96/86 45/100/86
+f 138/265/87 137/263/87 135/261/87
+f 103/211/88 101/208/88 102/209/88
+f 101/208/89 99/205/89 100/206/89
+f 99/205/90 97/202/90 98/203/90
+f 52/113/91 51/112/91 53/116/91
+f 95/199/92 93/196/92 94/197/92
+f 93/196/93 91/192/93 92/194/93
+f 92/194/94 91/192/94 89/190/94
+f 120/238/95 119/236/95 117/234/95
+f 121/239/14 119/236/14 120/238/14
+f 73/158/96 75/162/96 76/163/96
+f 150/283/97 149/281/97 147/279/97
+f 148/280/98 147/279/98 145/276/98
+f 116/232/99 118/235/99 117/234/99
+f 70/149/100 69/148/100 71/153/100
+f 46/101/101 45/100/101 47/104/101
+f 35/80/102 37/84/102 38/85/102
+f 135/261/103 133/258/103 134/259/103
+f 133/258/104 131/255/104 132/256/104
+f 131/255/105 129/252/105 130/253/105
+f 129/252/106 127/249/106 128/250/106
+f 127/249/107 125/246/107 126/247/107
+f 125/246/108 123/243/108 124/244/108
+f 123/243/109 121/239/109 122/241/109
+f 152/286/110 151/284/110 149/281/110
+f 209/401/111 273/503/111 275/506/111
+f 276/508/112 275/507/112 273/504/112
+f 211/406/113 213/409/113 214/411/113
+f 249/467/114 247/464/114 248/466/114
+f 244/460/115 243/459/115 241/456/115
+f 272/502/116 271/501/116 269/498/116
+f 180/341/117 179/340/117 181/344/117
+f 229/436/118 227/433/118 228/434/118
+f 225/430/119 223/427/119 224/428/119
+f 255/476/120 257/479/120 193/367/120
+f 234/444/121 233/441/121 231/439/121
+f 157/296/122 159/300/122 160/301/122
+f 182/345/110 181/344/110 184/349/110
+f 165/312/123 167/315/123 168/317/123
+f 214/411/124 213/409/124 216/415/124
+f 221/424/125 219/420/125 220/422/125
+f 216/415/126 215/413/126 153/288/126
+f 227/432/127 163/307/128 161/303/128
+f 235/446/129 237/449/129 173/327/129
+f 201/385/130 265/492/130 267/494/130
+f 261/485/131 263/488/131 199/379/131
+f 184/349/132 183/347/132 185/351/132
+f 159/300/133 161/304/133 162/305/133
+f 168/317/134 167/315/134 169/319/134
+f 275/506/135 277/510/135 213/410/135
+f 266/493/136 265/491/136 263/489/136
+f 249/468/137 185/352/137 183/348/137
+f 231/438/138 233/442/138 169/320/138
+f 190/361/139 189/360/139 191/364/139
+f 217/418/140 279/512/140 280/514/140
+f 255/477/141 253/474/141 254/475/141
+f 186/353/142 184/349/142 185/351/142
+f 203/390/143 205/394/143 206/395/143
+f 239/452/144 241/455/144 177/335/144
+f 225/429/145 161/303/145 159/299/145
+f 172/325/146 171/324/146 173/328/146
+f 174/329/147 173/328/147 175/332/147
+f 199/379/148 263/488/148 265/492/149
+f 253/473/150 255/476/150 191/363/150
+f 163/308/151 165/312/151 166/313/151
+f 263/489/152 261/486/152 262/487/152
+f 241/455/153 243/458/153 179/339/153
+f 201/386/154 203/390/154 204/391/154
+f 198/377/155 197/376/155 199/381/155
+f 209/402/156 211/406/156 212/407/156
+f 188/357/157 187/356/157 189/360/157
+f 233/442/158 235/446/158 171/323/158
+f 196/373/159 195/372/159 197/376/159
+f 257/479/160 259/482/161 195/371/161
+f 278/511/162 280/514/162 277/509/162
+f 236/448/163 235/447/163 233/443/163
+f 250/469/114 249/467/114 248/466/114
+f 253/474/164 251/471/164 252/472/164
+f 205/393/165 269/497/165 271/500/165
+f 213/409/166 215/413/166 216/415/166
+f 154/289/167 216/415/167 153/288/167
+f 218/419/140 217/418/140 280/514/140
+f 279/513/168 217/417/168 153/287/168
+f 176/333/169 175/332/169 177/336/169
+f 181/344/170 183/347/170 184/349/170
+f 161/304/171 163/308/171 164/309/171
+f 280/514/172 279/512/172 277/509/172
+f 257/480/173 255/477/173 256/478/173
+f 274/505/174 273/504/174 271/501/174
+f 155/292/175 157/296/175 158/297/175
+f 231/439/176 229/436/176 230/437/176
+f 205/394/177 207/397/177 208/399/177
+f 220/422/178 219/420/178 217/418/178
+f 203/389/179 267/494/179 269/497/179
+f 192/365/180 191/364/180 193/368/180
+f 244/460/181 246/463/181 245/462/181
+f 170/321/182 169/319/182 171/324/182
+f 259/482/183 261/485/184 197/375/183
+f 219/421/185 155/291/185 153/287/185
+f 251/471/186 249/467/186 250/469/186
+f 207/398/187 271/500/187 273/503/187
+f 277/510/188 279/513/188 215/414/188
+f 240/454/189 239/453/189 237/450/189
+f 261/486/190 259/483/190 260/484/190
+f 223/427/191 221/424/191 222/425/191
+f 278/511/192 277/509/192 275/507/192
+f 248/466/193 247/464/193 245/462/193
+f 237/449/194 239/452/194 175/331/195
+f 246/463/193 248/466/193 245/462/193
+f 221/423/196 157/295/196 155/291/196
+f 242/457/197 241/456/197 239/453/197
+f 227/433/198 225/430/198 226/431/198
+f 259/483/199 257/480/199 258/481/199
+f 251/470/200 187/355/201 185/352/202
+f 178/337/203 177/336/203 179/340/203
+f 199/380/204 201/386/204 202/387/204
+f 243/458/205 245/461/205 181/343/205
+f 223/426/206 159/299/206 157/295/206
+f 163/307/207 227/432/207 229/435/207
+f 194/369/208 193/368/208 195/372/208
+f 207/397/209 209/402/209 210/403/209
+f 251/470/210 253/473/211 189/359/210
+f 270/499/212 269/498/212 267/495/212
+f 186/353/213 185/351/213 187/356/213
+f 247/465/214 183/348/214 181/343/215
+f 153/288/216 155/292/216 156/293/216
+f 268/496/217 267/495/217 265/491/217
+f 238/451/218 237/450/218 235/447/218
+f 231/438/219 167/316/219 165/311/219
+f 42/93/220 40/89/220 41/91/220
+f 80/171/221 79/169/221 82/175/221
+f 83/177/222 81/173/222 147/278/222
+f 47/103/223 111/224/223 49/107/223
+f 39/88/224 103/210/224 41/92/224
+f 97/201/225 99/204/225 33/75/225
+f 41/92/226 105/214/226 43/95/226
+f 83/177/227 147/278/227 85/182/227
+f 95/198/228 97/201/228 31/71/228
+f 101/207/229 103/210/229 37/83/229
+f 45/99/230 109/221/230 47/103/230
+f 37/83/231 35/79/231 101/207/231
+f 43/95/232 107/218/232 45/99/232
+f 93/195/233 95/198/233 29/67/233
+f 91/193/234 93/195/234 27/63/234
+f 49/107/235 113/227/235 51/111/235
+f 87/186/236 151/285/236 25/59/236
+f 85/182/237 149/282/237 87/186/238
+f 119/237/239 121/240/239 55/120/240
+f 51/111/241 115/230/241 53/115/241
+f 81/173/242 79/170/242 145/275/242
+f 61/131/243 125/245/243 63/135/243
+f 79/170/244 77/165/244 143/272/244
+f 77/165/245 75/161/245 141/269/245
+f 75/161/246 73/157/246 139/266/246
+f 69/147/247 133/257/247 71/151/247
+f 67/143/248 131/254/248 69/147/248
+f 65/139/249 129/251/249 67/143/249
+f 63/135/250 127/248/250 65/139/250
+f 89/189/251 91/193/251 25/59/251
+f 59/127/252 123/242/252 61/131/252
+f 121/240/253 123/242/253 57/124/253
+f 98/203/49 97/202/49 96/200/49
+f 110/223/50 112/226/50 109/222/50
+f 50/109/51 48/105/51 49/108/51
+f 108/220/254 110/223/254 107/219/254
+f 138/265/255 140/268/255 137/263/255
+f 44/97/256 42/93/256 43/96/256
+f 104/212/257 106/216/257 103/211/257
+f 38/85/56 37/84/56 40/89/56
+f 106/217/258 108/220/258 105/215/258
+f 34/77/259 33/76/259 36/81/259
+f 32/73/59 31/72/59 34/77/59
+f 52/113/260 50/109/260 51/112/260
+f 28/65/261 27/64/261 30/69/261
+f 26/61/262 25/60/262 28/65/262
+f 82/175/263 81/174/263 84/179/263
+f 73/157/264 71/151/264 137/264/264
+f 84/179/265 83/178/265 86/183/265
+f 117/233/266 119/237/266 53/115/266
+f 78/167/69 77/166/69 80/171/69
+f 76/163/267 75/162/267 78/167/267
+f 112/226/268 114/229/268 111/225/268
+f 140/268/72 142/271/72 139/267/72
+f 70/149/73 68/145/73 69/148/73
+f 68/145/269 66/141/269 67/144/269
+f 66/141/270 64/137/270 65/140/270
+f 64/137/76 62/133/76 63/136/76
+f 62/133/77 60/129/77 61/132/77
+f 60/129/271 58/125/271 59/128/271
+f 144/274/272 146/277/272 143/273/272
+f 142/271/273 144/274/273 141/270/273
+f 114/229/83 116/232/83 113/228/83
+f 72/154/84 71/152/84 74/159/84
+f 30/69/85 29/68/85 32/73/85
+f 46/101/86 44/97/86 45/100/86
+f 136/262/87 138/265/87 135/261/87
+f 104/212/88 103/211/88 102/209/88
+f 102/209/274 101/208/274 100/206/274
+f 100/206/275 99/205/275 98/203/275
+f 54/117/276 52/113/276 53/116/276
+f 96/200/92 95/199/92 94/197/92
+f 94/197/277 93/196/277 92/194/277
+f 90/191/278 92/194/278 89/190/278
+f 74/159/96 73/158/96 76/163/96
+f 148/280/279 150/283/279 147/279/279
+f 146/277/280 148/280/280 145/276/280
+f 115/231/281 116/232/281 117/234/281
+f 72/155/100 70/149/100 71/153/100
+f 48/105/101 46/101/101 47/104/101
+f 36/81/102 35/80/102 38/85/102
+f 136/262/103 135/261/103 134/259/103
+f 134/259/104 133/258/104 132/256/104
+f 132/256/105 131/255/105 130/253/105
+f 130/253/106 129/252/106 128/250/106
+f 128/250/107 127/249/107 126/247/107
+f 126/247/282 125/246/282 124/244/282
+f 124/244/109 123/243/109 122/241/109
+f 296/545/7 295/543/7 297/547/7
+f 282/517/8 344/643/8 281/516/8
+f 310/573/283 309/572/283 312/577/283
+f 314/581/10 312/577/10 313/579/10
+f 342/639/11 341/637/11 344/643/11
+f 346/647/12 345/646/12 408/742/12
+f 374/691/13 376/694/13 373/690/13
+f 378/697/14 377/695/14 376/694/14
+f 406/739/15 408/742/15 405/737/15
+f 335/625/16 337/630/16 338/631/16
+f 337/629/17 401/731/17 403/734/17
+f 367/680/284 369/683/284 305/563/284
+f 359/666/19 361/670/19 297/548/19
+f 355/660/285 291/535/285 289/531/285
+f 361/670/286 363/674/286 299/551/286
+f 403/734/287 405/738/287 341/638/287
+f 353/657/288 289/531/288 287/527/288
+f 359/666/289 295/544/289 293/539/289
+f 365/677/290 367/680/290 303/559/290
+f 291/535/291 355/660/291 357/663/291
+f 363/674/28 365/677/28 301/555/28
+f 351/654/292 287/527/292 285/523/292
+f 349/651/293 285/523/293 283/519/293
+f 369/683/31 371/686/31 307/567/31
+f 407/741/294 345/645/294 281/515/294
+f 405/738/33 407/741/33 343/642/33
+f 377/696/295 313/580/295 311/576/295
+f 371/686/296 373/689/296 309/571/296
+f 335/626/297 399/728/297 401/731/297
+f 381/701/37 383/704/37 319/591/37
+f 333/621/38 397/725/38 399/728/38
+f 331/617/298 395/722/298 397/725/298
+f 329/613/40 393/720/40 395/722/40
+f 389/713/41 391/716/41 327/607/41
+f 387/710/299 389/713/299 325/603/299
+f 385/707/300 387/710/300 323/599/300
+f 383/704/301 385/707/301 321/595/301
+f 347/649/45 283/519/45 281/515/45
+f 379/698/302 381/701/302 317/587/302
+f 379/698/303 315/583/303 313/580/303
+f 344/643/48 343/641/48 281/516/48
+f 353/658/49 351/655/49 352/656/49
+f 368/682/50 367/681/50 365/678/50
+f 304/561/51 303/560/51 305/564/51
+f 366/679/52 365/678/52 363/675/52
+f 396/724/53 395/723/53 393/719/53
+f 298/549/54 297/547/54 299/552/54
+f 362/672/55 361/669/55 359/667/55
+f 293/540/56 295/543/56 296/545/56
+f 364/676/57 363/675/57 361/671/57
+f 289/532/58 291/536/58 292/537/58
+f 287/528/59 289/532/59 290/533/59
+f 306/565/60 305/564/60 307/568/60
+f 283/520/61 285/524/61 286/525/61
+f 281/516/62 283/520/62 284/521/62
+f 309/572/304 311/575/304 312/577/304
+f 312/577/64 311/575/64 313/579/64
+f 337/630/65 339/634/65 340/635/65
+f 327/607/66 391/716/66 393/720/66
+f 339/634/67 341/637/67 342/639/67
+f 375/693/305 311/576/305 309/571/305
+f 333/622/69 335/625/69 336/627/69
+f 331/618/70 333/622/70 334/623/70
+f 370/685/71 369/684/71 367/681/71
+f 398/727/72 397/726/72 395/723/72
+f 324/601/73 323/600/73 325/604/73
+f 322/597/74 321/596/74 323/600/74
+f 320/593/75 319/592/75 321/596/75
+f 318/589/76 317/588/76 319/592/76
+f 316/585/77 315/584/77 317/588/77
+f 314/581/78 313/579/78 315/584/78
+f 341/637/79 343/641/79 344/643/79
+f 345/646/80 407/740/80 408/742/80
+f 402/733/81 401/732/81 399/729/81
+f 400/730/82 399/729/82 397/726/82
+f 372/688/83 371/687/83 369/684/83
+f 327/608/84 329/614/84 330/615/84
+f 285/524/85 287/528/85 288/529/85
+f 300/553/86 299/552/86 301/556/86
+f 394/721/87 393/719/87 391/717/87
+f 359/667/88 357/664/88 358/665/88
+f 357/664/89 355/661/89 356/662/89
+f 355/661/90 353/658/90 354/659/90
+f 308/569/306 307/568/306 309/572/306
+f 351/655/92 349/652/92 350/653/92
+f 349/652/93 347/648/93 348/650/93
+f 348/650/307 347/648/307 345/646/307
+f 376/694/95 375/692/95 373/690/95
+f 377/695/14 375/692/14 376/694/14
+f 329/614/96 331/618/96 332/619/96
+f 406/739/97 405/737/97 403/735/97
+f 404/736/98 403/735/98 401/732/98
+f 372/688/99 374/691/99 373/690/99
+f 326/605/100 325/604/100 327/609/100
+f 302/557/101 301/556/101 303/560/101
+f 291/536/102 293/540/102 294/541/102
+f 391/717/103 389/714/103 390/715/103
+f 389/714/104 387/711/104 388/712/104
+f 387/711/308 385/708/308 386/709/308
+f 385/708/106 383/705/106 384/706/106
+f 383/705/107 381/702/107 382/703/107
+f 381/702/108 379/699/108 380/700/108
+f 379/699/109 377/695/109 378/697/109
+f 408/742/110 407/740/110 405/737/110
+f 298/549/220 296/545/220 297/547/220
+f 336/627/221 335/625/221 338/631/221
+f 339/633/222 337/629/222 403/734/222
+f 303/559/223 367/680/223 305/563/223
+f 295/544/224 359/666/224 297/548/224
+f 353/657/309 355/660/309 289/531/309
+f 297/548/310 361/670/310 299/551/310
+f 339/633/227 403/734/227 341/638/227
+f 351/654/311 353/657/311 287/527/311
+f 357/663/312 359/666/312 293/539/312
+f 301/555/313 365/677/313 303/559/313
+f 293/539/314 291/535/314 357/663/314
+f 299/551/315 363/674/315 301/555/315
+f 349/651/316 351/654/316 285/523/316
+f 347/649/317 349/651/317 283/519/317
+f 305/563/318 369/683/318 307/567/318
+f 343/642/319 407/741/319 281/515/319
+f 341/638/320 405/738/320 343/642/320
+f 375/693/321 377/696/321 311/576/321
+f 307/567/322 371/686/322 309/571/322
+f 337/629/323 335/626/323 401/731/323
+f 317/587/324 381/701/324 319/591/324
+f 335/626/325 333/621/326 399/728/327
+f 333/621/328 331/617/328 397/725/328
+f 331/617/329 329/613/329 395/722/329
+f 325/603/247 389/713/247 327/607/247
+f 323/599/330 387/710/330 325/603/330
+f 321/595/331 385/707/331 323/599/331
+f 319/591/250 383/704/250 321/595/250
+f 345/645/251 347/649/251 281/515/251
+f 315/583/332 379/698/332 317/587/332
+f 377/696/333 379/698/334 313/580/334
+f 354/659/49 353/658/49 352/656/49
+f 366/679/50 368/682/50 365/678/50
+f 306/565/51 304/561/51 305/564/51
+f 364/676/254 366/679/254 363/675/254
+f 394/721/255 396/724/255 393/719/255
+f 300/553/256 298/549/256 299/552/256
+f 360/668/257 362/672/257 359/667/257
+f 294/541/56 293/540/56 296/545/56
+f 362/673/258 364/676/258 361/671/258
+f 290/533/259 289/532/259 292/537/259
+f 288/529/59 287/528/59 290/533/59
+f 308/569/260 306/565/260 307/568/260
+f 284/521/261 283/520/261 286/525/261
+f 282/517/262 281/516/262 284/521/262
+f 338/631/263 337/630/263 340/635/263
+f 329/613/264 327/607/264 393/720/264
+f 340/635/265 339/634/265 342/639/265
+f 373/689/335 375/693/335 309/571/335
+f 334/623/69 333/622/69 336/627/69
+f 332/619/267 331/618/267 334/623/267
+f 368/682/268 370/685/268 367/681/268
+f 396/724/72 398/727/72 395/723/72
+f 326/605/73 324/601/73 325/604/73
+f 324/601/269 322/597/269 323/600/269
+f 322/597/270 320/593/270 321/596/270
+f 320/593/76 318/589/76 319/592/76
+f 318/589/77 316/585/77 317/588/77
+f 316/585/271 314/581/271 315/584/271
+f 400/730/272 402/733/272 399/729/272
+f 398/727/273 400/730/273 397/726/273
+f 370/685/83 372/688/83 369/684/83
+f 328/610/84 327/608/84 330/615/84
+f 286/525/85 285/524/85 288/529/85
+f 302/557/86 300/553/86 301/556/86
+f 392/718/87 394/721/87 391/717/87
+f 360/668/88 359/667/88 358/665/88
+f 358/665/274 357/664/274 356/662/274
+f 356/662/275 355/661/275 354/659/275
+f 310/573/336 308/569/336 309/572/336
+f 352/656/92 351/655/92 350/653/92
+f 350/653/277 349/652/277 348/650/277
+f 346/647/278 348/650/278 345/646/278
+f 330/615/96 329/614/96 332/619/96
+f 404/736/279 406/739/279 403/735/279
+f 402/733/280 404/736/280 401/732/280
+f 371/687/281 372/688/281 373/690/281
+f 328/611/100 326/605/100 327/609/100
+f 304/561/101 302/557/101 303/560/101
+f 292/537/102 291/536/102 294/541/102
+f 392/718/103 391/717/103 390/715/103
+f 390/715/104 389/714/104 388/712/104
+f 388/712/105 387/711/105 386/709/105
+f 386/709/106 385/708/106 384/706/106
+f 384/706/107 383/705/107 382/703/107
+f 382/703/282 381/702/282 380/700/282
+f 380/700/109 379/699/109 378/697/109
+f 170/321/337 168/317/337 169/319/337
+f 208/399/209 207/397/209 210/403/209
+f 211/405/338 209/401/339 275/506/338
+f 175/331/340 239/452/340 177/335/340
+f 167/316/341 231/438/342 169/320/342
+f 225/429/343 227/432/343 161/303/343
+f 169/320/344 233/442/344 171/323/344
+f 211/405/345 275/506/345 213/410/345
+f 223/426/346 225/429/346 159/299/346
+f 229/435/347 231/438/347 165/311/348
+f 173/327/349 237/449/349 175/331/349
+f 165/311/350 163/307/351 229/435/351
+f 171/323/352 235/446/352 173/327/352
+f 221/423/353 223/426/353 157/295/353
+f 219/421/354 221/423/355 155/291/354
+f 177/335/356 241/455/356 179/339/357
+f 215/414/358 279/513/358 153/287/358
+f 213/410/359 277/510/359 215/414/359
+f 247/465/360 249/468/360 183/348/360
+f 179/339/361 243/458/361 181/343/361
+f 209/401/362 207/398/362 273/503/362
+f 189/359/363 253/473/364 191/363/363
+f 207/398/365 205/393/366 271/500/366
+f 205/393/367 203/389/368 269/497/369
+f 203/389/370 201/385/370 267/494/370
+f 197/375/371 261/485/371 199/379/371
+f 195/371/372 259/482/372 197/375/372
+f 193/367/373 257/479/374 195/371/374
+f 191/363/375 255/476/375 193/367/376
+f 217/417/377 219/421/377 153/287/377
+f 187/355/378 251/470/378 189/359/378
+f 249/468/379 251/470/379 185/352/379
+f 226/431/380 225/430/380 224/428/380
+f 238/451/381 240/454/381 237/450/381
+f 178/337/169 176/333/169 177/336/169
+f 236/448/382 238/451/382 235/447/382
+f 266/493/217 268/496/217 265/491/217
+f 172/325/182 170/321/182 171/324/182
+f 232/440/121 234/444/121 231/439/121
+f 166/313/123 165/312/123 168/317/123
+f 234/445/163 236/448/163 233/443/163
+f 162/305/383 161/304/383 164/309/383
+f 160/301/384 159/300/384 162/305/384
+f 180/341/385 178/337/385 179/340/385
+f 156/293/175 155/292/175 158/297/175
+f 154/289/386 153/288/386 156/293/386
+f 210/403/387 209/402/387 212/407/387
+f 201/385/388 199/379/388 265/492/388
+f 212/407/389 211/406/389 214/411/389
+f 245/461/390 247/465/390 181/343/390
+f 206/395/177 205/394/177 208/399/177
+f 204/391/391 203/390/391 206/395/391
+f 240/454/197 242/457/197 239/453/197
+f 268/496/212 270/499/212 267/495/212
+f 198/377/159 196/373/159 197/376/159
+f 196/373/208 194/369/208 195/372/208
+f 194/369/180 192/365/180 193/368/180
+f 192/365/392 190/361/392 191/364/392
+f 190/361/157 188/357/157 189/360/157
+f 188/357/393 186/353/393 187/356/393
+f 272/502/394 274/505/394 271/501/394
+f 270/499/116 272/502/116 269/498/116
+f 242/457/115 244/460/115 241/456/115
+f 200/382/204 199/380/204 202/387/204
+f 158/297/395 157/296/395 160/301/395
+f 174/329/146 172/325/146 173/328/146
+f 264/490/396 266/493/396 263/489/396
+f 232/440/397 231/439/397 230/437/397
+f 230/437/398 229/436/398 228/434/398
+f 228/434/198 227/433/198 226/431/198
+f 182/345/97 180/341/97 181/344/97
+f 224/428/191 223/427/191 222/425/191
+f 222/425/399 221/424/399 220/422/399
+f 218/419/400 220/422/400 217/418/400
+f 202/387/154 201/386/154 204/391/154
+f 276/508/192 278/511/192 275/507/192
+f 274/505/112 276/508/112 273/504/112
+f 243/459/401 244/460/401 245/462/401
+f 200/383/155 198/377/155 199/381/155
+f 176/333/147 174/329/147 175/332/147
+f 164/309/402 163/308/402 166/313/402
+f 264/490/403 263/489/403 262/487/403
+f 262/487/404 261/486/404 260/484/404
+f 260/484/199 259/483/199 258/481/199
+f 258/481/173 257/480/173 256/478/173
+f 256/478/405 255/477/405 254/475/405
+f 254/475/406 253/474/406 252/472/406
+f 252/472/186 251/471/186 250/469/186
+f 471/838/407 469/835/407 472/839/407
+f 519/935/408 520/937/408 518/934/408
+f 519/935/134 521/939/134 520/937/134
+f 413/749/112 414/750/112 416/754/112
+f 486/868/370 488/871/370 422/763/370
+f 476/848/409 474/844/409 473/841/409
+f 535/967/167 473/841/167 536/970/167
+f 521/939/134 522/942/134 520/937/134
+f 440/790/114 442/793/114 441/791/114
+f 442/792/410 506/910/411 508/914/412
+f 476/847/413 412/748/414 474/843/415
+f 490/876/204 488/872/204 487/869/204
+f 490/877/416 426/768/416 424/765/416
+f 448/801/417 446/798/153 510/918/417
+f 462/824/198 464/827/198 463/826/198
+f 486/867/143 484/863/143 483/861/143
+f 461/823/198 462/824/198 463/826/198
+f 518/933/418 454/811/418 516/930/418
+f 483/861/419 484/863/419 481/857/419
+f 511/919/169 513/923/169 512/921/169
+f 458/818/420 460/822/420 459/820/420
+f 430/774/184 428/771/184 492/881/184
+f 436/783/164 438/787/164 437/785/164
+f 480/856/156 478/852/156 477/849/156
+f 435/782/421 436/783/421 437/785/421
+f 417/755/422 415/752/422 418/757/422
+f 440/789/423 438/786/423 504/906/423
+f 466/830/424 530/957/424 532/961/424
+f 499/895/425 501/899/425 500/898/425
+f 482/859/426 418/756/427 416/753/428
+f 520/938/344 456/815/344 518/933/344
+f 531/959/122 532/962/122 529/955/122
+f 421/761/212 419/758/212 422/762/212
+f 488/871/388 490/877/388 424/765/388
+f 453/809/163 454/810/163 456/816/163
+f 460/822/398 462/824/398 461/823/398
+f 478/851/345 414/751/345 476/847/345
+f 523/943/123 524/946/123 521/939/123
+f 503/903/429 504/905/429 502/902/429
+f 456/815/158 454/811/158 518/933/430
+f 489/873/431 490/876/431 487/869/431
+f 468/833/399 470/836/399 469/835/399
+f 464/828/432 462/825/432 528/954/432
+f 484/863/433 482/860/433 481/857/433
+f 480/855/434 482/859/434 416/753/434
+f 515/927/435 517/931/435 516/929/435
+f 453/809/218 451/806/218 454/810/218
+f 502/901/436 438/786/436 500/897/436
+f 412/748/437 410/744/437 474/843/437
+f 495/887/208 496/889/208 494/886/208
+f 491/879/438 493/883/438 492/882/438
+f 455/813/439 456/814/439 458/818/439
+f 474/843/440 410/744/440 536/969/440
+f 497/891/441 498/893/441 496/889/441
+f 411/746/192 412/747/192 414/750/192
+f 492/881/371 428/771/371 490/877/371
+f 451/806/382 452/807/382 454/810/382
+f 423/764/442 421/761/442 424/766/442
+f 452/808/194 450/804/194 514/925/194
+f 434/780/443 436/783/443 435/782/443
+f 488/871/130 424/765/130 422/763/130
+f 462/825/444 526/950/444 528/954/444
+f 409/743/172 410/745/172 412/747/172
+f 484/864/445 420/760/445 418/756/165
+f 410/744/446 472/840/446 536/969/446
+f 433/779/443 434/780/443 435/782/443
+f 493/883/447 494/886/447 492/882/447
+f 533/963/175 534/965/175 531/959/175
+f 535/967/448 536/970/448 533/963/448
+f 467/832/399 468/833/399 469/835/399
+f 512/922/449 448/801/449 510/918/449
+f 429/773/450 430/775/450 431/776/450
+f 494/885/372 430/774/372 492/881/372
+f 513/923/169 514/926/169 512/921/169
+f 468/834/451 466/830/452 532/961/451
+f 419/758/116 417/755/116 420/759/116
+f 480/855/111 416/753/111 414/751/111
+f 413/749/192 411/746/192 414/750/192
+f 517/931/435 518/934/435 516/929/435
+f 446/798/453 444/795/453 508/914/453
+f 522/941/454 458/819/454 520/938/454
+f 432/777/173 434/780/173 433/779/173
+f 511/919/385 512/921/385 510/917/385
+f 432/778/455 430/774/455 494/885/455
+f 469/835/178 470/836/178 472/839/178
+f 526/950/456 462/825/457 460/821/458
+f 530/958/384 528/953/384 527/951/384
+f 487/869/459 488/872/459 485/865/459
+f 527/951/383 528/953/383 525/947/383
+f 428/772/404 430/775/404 429/773/404
+f 513/923/460 515/927/460 514/926/460
+f 525/947/151 526/949/151 523/943/151
+f 528/953/171 526/949/171 525/947/171
+f 411/746/172 409/743/172 412/747/172
+f 458/819/138 456/815/461 520/938/138
+f 507/911/110 508/913/110 505/907/110
+f 491/879/155 492/882/155 490/878/155
+f 428/771/462 426/768/131 490/877/131
+f 496/890/463 432/778/463 494/885/463
+f 463/826/119 464/827/119 465/829/119
+f 505/907/132 506/909/132 504/905/132
+f 415/752/112 413/749/112 416/754/112
+f 423/764/464 424/766/464 426/769/464
+f 526/949/402 524/946/402 523/943/402
+f 481/857/209 482/860/209 479/853/209
+f 500/897/465 436/784/465 498/894/465
+f 516/930/466 452/808/466 514/925/466
+f 507/911/97 509/915/97 508/913/97
+f 497/891/139 499/895/139 498/893/139
+f 447/800/197 448/802/197 450/805/197
+f 529/955/133 530/958/133 527/951/133
+f 478/851/467 480/855/467 414/751/467
+f 460/821/348 458/819/348 524/945/348
+f 425/767/468 426/769/468 427/770/468
+f 472/839/140 410/745/140 409/743/140
+f 485/865/391 486/867/391 483/861/391
+f 509/915/385 511/919/385 510/917/385
+f 508/913/469 506/909/470 505/907/470
+f 498/894/471 434/781/472 496/890/472
+f 477/849/389 478/852/389 475/845/389
+f 501/899/393 503/903/393 502/902/393
+f 455/812/163 453/809/163 456/816/163
+f 471/838/473 472/839/473 409/743/473
+f 457/817/397 458/818/397 459/820/397
+f 449/803/474 447/800/474 450/805/474
+f 493/883/208 495/887/208 494/886/208
+f 517/931/182 519/935/182 518/934/182
+f 465/829/475 466/831/475 467/832/475
+f 489/874/155 491/879/155 490/878/155
+f 482/859/476 484/864/476 418/756/476
+f 437/785/477 438/787/477 439/788/477
+f 439/788/114 440/790/114 441/791/114
+f 466/830/346 464/828/346 530/957/346
+f 442/792/478 440/789/478 506/910/360
+f 419/758/479 420/759/479 422/762/479
+f 415/752/174 416/754/174 418/757/174
+f 445/797/181 443/794/181 444/796/181
+f 466/831/475 468/833/475 467/832/475
+f 443/794/193 441/791/193 444/796/193
+f 430/775/450 432/777/450 431/776/450
+f 482/860/209 480/856/209 479/853/209
+f 431/776/173 432/777/173 433/779/173
+f 421/761/442 422/762/442 424/766/442
+f 514/925/480 450/804/340 512/922/340
+f 479/853/156 480/856/156 477/849/156
+f 426/769/403 428/772/403 427/770/403
+f 499/895/139 500/898/139 498/893/139
+f 484/864/481 486/868/481 420/760/481
+f 509/915/482 510/917/482 508/913/482
+f 457/817/121 455/813/121 458/818/121
+f 450/804/483 448/801/484 512/922/483
+f 444/795/485 442/792/485 508/914/485
+f 503/903/486 505/907/486 504/905/486
+f 447/800/115 445/797/115 448/802/115
+f 425/767/136 423/764/136 426/769/136
+f 510/918/361 446/798/361 508/914/361
+f 446/799/487 445/797/487 444/796/487
+f 427/770/190 428/772/190 429/773/190
+f 470/837/185 534/966/185 536/969/185
+f 524/945/350 526/950/488 460/821/488
+f 534/965/489 532/962/489 531/959/489
+f 440/789/490 504/906/490 506/910/490
+f 417/755/491 418/757/491 420/759/491
+f 441/791/492 442/793/492 444/796/492
+f 468/834/493 532/961/493 534/966/494
+f 438/786/200 502/901/200 504/906/200
+f 438/786/495 436/784/495 500/897/495
+f 536/970/216 534/965/216 533/963/216
+f 458/819/219 522/941/219 524/945/219
+f 478/852/496 476/848/496 475/845/496
+f 445/797/115 446/799/115 448/802/115
+f 414/751/497 412/748/497 476/847/497
+f 438/787/186 440/790/186 439/788/186
+f 459/820/118 460/822/118 461/823/118
+f 451/806/381 449/803/381 452/807/381
+f 464/827/119 466/831/119 465/829/119
+f 464/828/498 528/954/499 530/957/498
+f 515/927/147 516/929/147 514/926/147
+f 436/784/500 434/781/500 498/894/500
+f 470/837/501 468/834/354 534/966/501
+f 488/872/459 486/867/459 485/865/459
+f 501/899/157 502/902/157 500/898/157
+f 486/868/179 422/763/179 420/760/179
+f 449/803/381 450/805/381 452/807/381
+f 472/840/377 470/837/377 536/969/377
+f 495/887/441 497/891/441 496/889/441
+f 434/781/502 432/778/502 496/890/502
+f 532/962/122 530/958/122 529/955/122
+f 524/946/123 522/942/123 521/939/123
+f 475/845/124 476/848/124 473/841/124
+f 473/841/126 474/844/126 536/970/126
+f 454/811/503 452/808/503 516/930/503
+g KsHauptsignal_Plane.006_Lights
+usemtl Lights
+f 88/188/504 28/66/504 40/90/504
+f 40/90/504 44/98/504 56/122/504
+f 88/188/505 64/138/505 72/156/505
+f 72/156/505 76/164/505 88/188/505
+f 40/90/506 28/66/506 32/74/506
+f 32/74/504 34/78/504 40/90/504
+f 56/122/504 44/98/504 48/106/504
+f 48/106/504 52/114/504 56/122/504
+f 56/122/507 58/126/507 64/138/507
+f 64/138/505 68/146/505 70/150/505
+f 88/188/505 76/164/505 80/172/505
+f 80/172/508 84/180/508 88/188/508
+f 88/188/509 26/62/509 28/66/509
+f 28/66/509 30/70/509 32/74/509
+f 40/90/506 34/78/506 36/82/506
+f 36/82/506 38/86/506 40/90/506
+f 40/90/510 42/94/510 44/98/510
+f 44/98/511 46/102/511 48/106/511
+f 48/106/512 50/110/512 52/114/512
+f 52/114/513 54/118/514 56/122/514
+f 64/138/515 58/126/516 60/130/515
+f 60/130/517 62/134/517 64/138/517
+f 64/138/516 66/142/518 68/146/518
+f 64/138/505 70/150/505 72/156/505
+f 72/156/519 74/160/519 76/164/519
+f 76/164/505 78/168/505 80/172/520
+f 80/172/508 82/176/508 84/180/508
+f 84/180/521 86/184/521 88/188/521
+f 64/138/522 88/188/522 40/90/522
+f 156/294/523 158/298/523 160/302/523
+f 200/384/524 204/392/524 216/416/524
+f 192/366/525 186/354/525 188/358/525
+f 176/334/526 180/342/526 184/350/526
+f 212/408/527 214/412/527 216/416/528
+f 188/358/529 190/362/529 192/366/529
+f 192/366/530 196/374/531 198/378/531
+f 184/350/532 172/326/532 176/334/532
+f 204/392/533 206/396/534 208/400/533
+f 200/384/535 202/388/536 204/392/535
+f 216/416/537 192/366/537 200/384/537
+f 168/318/538 162/306/538 164/310/538
+f 164/310/539 166/314/540 168/318/539
+f 160/302/541 162/306/541 168/318/541
+f 176/334/532 178/338/532 180/342/532
+f 208/400/537 210/404/537 212/408/537
+f 168/318/542 170/322/540 172/326/542
+f 168/318/543 172/326/543 184/350/543
+f 168/318/544 156/294/544 160/302/544
+f 180/342/545 182/346/546 184/350/545
+f 208/400/537 212/408/537 216/416/537
+f 216/416/537 204/392/537 208/400/537
+f 192/366/547 194/370/547 196/374/547
+f 184/350/548 186/354/548 192/366/548
+f 192/366/549 198/378/549 200/384/549
+f 216/416/550 156/294/550 168/318/550
+f 216/416/551 154/290/551 156/294/551
+f 172/326/552 174/330/552 176/334/552
+f 192/366/553 216/416/553 168/318/553
+f 56/122/554 64/138/554 40/90/554
+f 344/644/504 284/522/504 296/546/504
+f 296/546/504 300/554/504 312/578/504
+f 344/644/505 320/594/505 328/612/505
+f 328/612/505 332/620/505 344/644/505
+f 296/546/504 284/522/555 288/530/555
+f 288/530/504 290/534/504 296/546/504
+f 312/578/506 300/554/506 304/562/506
+f 304/562/511 308/570/511 312/578/511
+f 312/578/556 314/582/556 320/594/556
+f 320/594/505 324/602/505 326/606/505
+f 344/644/505 332/620/505 336/628/505
+f 336/628/557 340/636/557 344/644/557
+f 344/644/558 282/518/558 284/522/558
+f 284/522/559 286/526/559 288/530/559
+f 296/546/506 290/534/506 292/538/506
+f 292/538/560 294/542/560 296/546/560
+f 296/546/510 298/550/510 300/554/510
+f 300/554/506 302/558/506 304/562/506
+f 304/562/512 306/566/512 308/570/512
+f 308/570/561 310/574/561 312/578/561
+f 320/594/562 314/582/562 316/586/562
+f 316/586/563 318/590/563 320/594/563
+f 320/594/518 322/598/518 324/602/518
+f 320/594/518 326/606/518 328/612/518
+f 328/612/564 330/616/564 332/620/564
+f 332/620/565 334/624/565 336/628/565
+f 336/628/566 338/632/566 340/636/566
+f 340/636/567 342/640/567 344/644/567
+f 320/594/522 344/644/522 296/546/522
+f 312/578/554 320/594/554 296/546/554
+f 184/350/568 192/366/568 168/318/568
+f 509/916/546 507/912/546 505/908/546
+f 521/940/538 527/952/538 525/948/538
+f 473/842/550 533/964/550 521/940/550
+f 521/940/569 519/936/569 517/932/569
+f 517/932/570 515/928/570 513/924/570
+f 533/964/523 531/960/523 529/956/523
+f 497/892/571 495/888/571 493/884/572
+f 489/875/536 487/870/536 485/866/536
+f 477/850/573 475/846/574 473/842/573
+f 473/842/537 497/892/537 489/875/537
+f 497/892/575 503/904/575 501/900/575
+f 505/908/543 517/932/576 513/924/543
+f 473/842/577 535/968/578 533/964/578
+f 481/858/579 477/850/579 473/842/580
+f 497/892/553 473/842/553 521/940/553
+f 501/900/581 499/896/581 497/892/582
+f 497/892/583 493/884/583 491/880/583
+f 521/940/543 517/932/543 505/908/543
+f 505/908/584 497/892/584 521/940/584
+f 473/842/537 485/866/537 481/858/537
+f 497/892/549 491/880/549 489/875/549
+f 529/956/540 527/952/539 521/940/539
+f 521/940/544 533/964/544 529/956/544
+f 481/858/585 479/854/585 477/850/585
+f 505/908/312 503/904/586 497/892/312
+f 513/924/532 511/920/532 509/916/532
+f 489/875/524 485/866/524 473/842/524
+f 513/924/526 509/916/526 505/908/526
+f 525/948/544 523/944/544 521/940/550
+f 485/866/582 483/862/582 481/858/582
+l 571 572
+l 540 537
+l 572 569
+l 539 540
+l 538 539
+l 570 571
+l 543 544
+l 575 576
+l 541 542
+l 573 574
+l 542 543
+l 574 575
diff --git a/advtrains_signals_ks/models/advtrains_signals_ks_distant_smr30.obj b/advtrains_signals_ks/models/advtrains_signals_ks_distant_smr30.obj
new file mode 100644
index 0000000..c6ba424
--- /dev/null
+++ b/advtrains_signals_ks/models/advtrains_signals_ks_distant_smr30.obj
@@ -0,0 +1,3291 @@
+# Blender 3.4.1
+# www.blender.org
+mtllib advtrains_signals_ks_distant_smr30.mtl
+o KsHauptsignal_Plane.006
+v 0.244813 -0.503378 -0.366253
+v 0.195330 -0.503378 -0.394869
+v 0.223945 -0.503378 -0.444354
+v 0.273429 -0.503378 -0.415738
+v 0.187727 -0.503378 -0.267534
+v 0.159111 -0.503378 -0.218050
+v 0.109626 -0.503378 -0.246666
+v 0.138242 -0.503378 -0.296151
+v 0.244813 -0.299114 -0.366253
+v 0.244813 -0.172218 -0.366253
+v 0.195330 -0.172218 -0.394869
+v 0.195330 -0.299114 -0.394869
+v 0.223945 -0.299114 -0.444354
+v 0.223945 -0.172218 -0.444354
+v 0.273429 -0.172218 -0.415738
+v 0.273429 -0.299114 -0.415738
+v 0.187727 -0.299114 -0.267534
+v 0.187727 -0.172218 -0.267534
+v 0.159111 -0.172218 -0.218050
+v 0.159111 -0.299114 -0.218050
+v 0.109626 -0.299114 -0.246666
+v 0.109626 -0.172218 -0.246666
+v 0.138242 -0.172218 -0.296151
+v 0.138242 -0.299114 -0.296151
+v 0.176174 0.243339 -0.575544
+v 0.162678 0.242789 -0.552205
+v 0.205184 0.253858 -0.619416
+v 0.165383 0.253308 -0.550587
+v 0.209615 0.263561 -0.616804
+v 0.169812 0.263011 -0.547976
+v 0.215602 0.272074 -0.613299
+v 0.175798 0.271524 -0.544470
+v 0.222913 0.279070 -0.609035
+v 0.183111 0.278520 -0.540205
+v 0.231269 0.284280 -0.604174
+v 0.191467 0.283730 -0.535346
+v 0.240350 0.287504 -0.598908
+v 0.200547 0.286954 -0.530079
+v 0.249804 0.288619 -0.593434
+v 0.210002 0.288069 -0.524606
+v 0.259270 0.287581 -0.587965
+v 0.219468 0.287030 -0.519137
+v 0.268384 0.284430 -0.582712
+v 0.228582 0.283880 -0.513883
+v 0.276795 0.279287 -0.577874
+v 0.236993 0.278737 -0.509046
+v 0.284180 0.272350 -0.573639
+v 0.244378 0.271800 -0.504811
+v 0.290255 0.263886 -0.570170
+v 0.250453 0.263336 -0.501341
+v 0.294787 0.254219 -0.567598
+v 0.254985 0.253669 -0.498770
+v 0.271296 0.243722 -0.520535
+v 0.257800 0.243172 -0.497197
+v 0.272277 0.232316 -0.519964
+v 0.258788 0.232252 -0.496680
+v 0.271254 0.221688 -0.520373
+v 0.257913 0.221322 -0.497239
+v 0.268572 0.211344 -0.521968
+v 0.255206 0.210797 -0.498851
+v 0.264139 0.201642 -0.524576
+v 0.250773 0.201094 -0.501459
+v 0.258151 0.193129 -0.528079
+v 0.244784 0.192581 -0.504961
+v 0.250838 0.186134 -0.532340
+v 0.237471 0.185585 -0.509222
+v 0.242480 0.180924 -0.537196
+v 0.229113 0.180374 -0.514079
+v 0.233400 0.177700 -0.542462
+v 0.220032 0.177150 -0.519345
+v 0.223944 0.176585 -0.547935
+v 0.210577 0.176035 -0.524818
+v 0.214479 0.177623 -0.553405
+v 0.201111 0.177073 -0.530288
+v 0.205366 0.180774 -0.558659
+v 0.191998 0.180224 -0.535543
+v 0.196957 0.185916 -0.563499
+v 0.183587 0.185367 -0.540383
+v 0.189573 0.192853 -0.567737
+v 0.176205 0.192304 -0.544621
+v 0.183501 0.201317 -0.571210
+v 0.170131 0.200769 -0.548094
+v 0.178971 0.210983 -0.573785
+v 0.165602 0.210436 -0.550669
+v 0.176188 0.221305 -0.575349
+v 0.162790 0.220938 -0.552248
+v 0.175259 0.231924 -0.576070
+v 0.161803 0.231861 -0.552767
+v 0.174033 0.243790 -0.576777
+v 0.142249 0.243271 -0.521817
+v 0.203163 0.254782 -0.620574
+v 0.145074 0.254264 -0.520123
+v 0.207791 0.264921 -0.617843
+v 0.149702 0.264402 -0.517392
+v 0.214045 0.273816 -0.614177
+v 0.155957 0.273298 -0.513727
+v 0.221685 0.281127 -0.609720
+v 0.163597 0.280609 -0.509270
+v 0.230417 0.286572 -0.604641
+v 0.172328 0.286053 -0.504191
+v 0.239905 0.289941 -0.599137
+v 0.181815 0.289423 -0.498686
+v 0.249784 0.291105 -0.593417
+v 0.191695 0.290587 -0.492966
+v 0.259676 0.290021 -0.587702
+v 0.201587 0.289502 -0.487252
+v 0.269199 0.286728 -0.582213
+v 0.211111 0.286210 -0.481763
+v 0.277989 0.281354 -0.577159
+v 0.219900 0.280836 -0.476709
+v 0.285707 0.274105 -0.572735
+v 0.227619 0.273587 -0.472285
+v 0.292056 0.265261 -0.569112
+v 0.233967 0.264742 -0.468661
+v 0.296793 0.255160 -0.566427
+v 0.238705 0.254641 -0.465977
+v 0.273430 0.244190 -0.519295
+v 0.241647 0.243672 -0.464335
+v 0.274438 0.232775 -0.518712
+v 0.242682 0.232256 -0.463798
+v 0.273424 0.221351 -0.519127
+v 0.241770 0.220833 -0.464387
+v 0.270600 0.210359 -0.520819
+v 0.238946 0.209840 -0.466080
+v 0.265972 0.200220 -0.523551
+v 0.234318 0.199702 -0.468812
+v 0.259717 0.191324 -0.527216
+v 0.228063 0.190806 -0.472477
+v 0.252077 0.184013 -0.531673
+v 0.220423 0.183495 -0.476934
+v 0.243346 0.178569 -0.536752
+v 0.211692 0.178051 -0.482013
+v 0.233859 0.175200 -0.542257
+v 0.202204 0.174682 -0.487518
+v 0.223979 0.174035 -0.547977
+v 0.192325 0.173517 -0.493238
+v 0.214087 0.175120 -0.553691
+v 0.182433 0.174602 -0.498953
+v 0.204563 0.178413 -0.559180
+v 0.172909 0.177894 -0.504442
+v 0.195774 0.183787 -0.564234
+v 0.164120 0.183268 -0.509496
+v 0.188056 0.191035 -0.568658
+v 0.156402 0.190517 -0.513919
+v 0.181707 0.199880 -0.572282
+v 0.150052 0.199362 -0.517543
+v 0.176969 0.209981 -0.574966
+v 0.145315 0.209463 -0.520228
+v 0.174027 0.220950 -0.576608
+v 0.142372 0.220432 -0.521869
+v 0.173093 0.232366 -0.577321
+v 0.141338 0.231848 -0.522406
+v 0.377218 0.383581 -0.459284
+v 0.363721 0.383353 -0.435946
+v 0.404628 0.387932 -0.504081
+v 0.364827 0.387704 -0.435252
+v 0.406448 0.391945 -0.502979
+v 0.366646 0.391717 -0.434150
+v 0.408913 0.395466 -0.501510
+v 0.369111 0.395239 -0.432681
+v 0.411928 0.398360 -0.499730
+v 0.372126 0.398132 -0.430902
+v 0.415378 0.400515 -0.497708
+v 0.375576 0.400287 -0.428880
+v 0.419130 0.401848 -0.495522
+v 0.379327 0.401621 -0.426694
+v 0.423039 0.402309 -0.493257
+v 0.383237 0.402082 -0.424427
+v 0.426955 0.401880 -0.490996
+v 0.387154 0.401652 -0.422168
+v 0.430729 0.400577 -0.488830
+v 0.390927 0.400349 -0.420002
+v 0.434214 0.398450 -0.486841
+v 0.394413 0.398222 -0.418013
+v 0.437278 0.395581 -0.485106
+v 0.397476 0.395353 -0.416276
+v 0.439802 0.392080 -0.483689
+v 0.400000 0.391852 -0.414861
+v 0.441689 0.388081 -0.482648
+v 0.401888 0.387854 -0.413820
+v 0.416561 0.383739 -0.436531
+v 0.403065 0.383512 -0.413191
+v 0.416966 0.379021 -0.436293
+v 0.403489 0.378995 -0.413003
+v 0.416497 0.374626 -0.436382
+v 0.403140 0.374474 -0.413257
+v 0.415399 0.370347 -0.437062
+v 0.402031 0.370121 -0.413944
+v 0.413577 0.366334 -0.438160
+v 0.400209 0.366108 -0.415043
+v 0.411110 0.362813 -0.439626
+v 0.397743 0.362586 -0.416508
+v 0.408093 0.359920 -0.441402
+v 0.394726 0.359693 -0.418285
+v 0.404642 0.357765 -0.443421
+v 0.391275 0.357538 -0.420305
+v 0.400890 0.356431 -0.445606
+v 0.387522 0.356204 -0.422489
+v 0.396981 0.355970 -0.447871
+v 0.383613 0.355743 -0.424755
+v 0.393065 0.356399 -0.450132
+v 0.379696 0.356172 -0.427015
+v 0.389292 0.357703 -0.452299
+v 0.375924 0.357476 -0.429183
+v 0.385808 0.359830 -0.454290
+v 0.372438 0.359603 -0.431174
+v 0.382745 0.362699 -0.456029
+v 0.369377 0.362472 -0.432913
+v 0.380224 0.366200 -0.457449
+v 0.366854 0.365973 -0.434332
+v 0.378339 0.370198 -0.458494
+v 0.364970 0.369972 -0.435378
+v 0.377176 0.374467 -0.459122
+v 0.363795 0.374316 -0.436011
+v 0.376839 0.378860 -0.459500
+v 0.363374 0.378833 -0.436202
+v 0.376331 0.383767 -0.459792
+v 0.344548 0.383553 -0.404831
+v 0.403790 0.388314 -0.504556
+v 0.345700 0.388099 -0.404105
+v 0.405690 0.392507 -0.503401
+v 0.347601 0.392293 -0.402951
+v 0.408265 0.396187 -0.501865
+v 0.350175 0.395973 -0.401414
+v 0.411415 0.399211 -0.500003
+v 0.353326 0.398996 -0.399553
+v 0.415019 0.401463 -0.497889
+v 0.356930 0.401248 -0.397439
+v 0.418938 0.402856 -0.495604
+v 0.360850 0.402642 -0.395154
+v 0.423023 0.403338 -0.493235
+v 0.364934 0.403124 -0.392785
+v 0.427116 0.402889 -0.490875
+v 0.369027 0.402675 -0.390424
+v 0.431060 0.401527 -0.488613
+v 0.372970 0.401313 -0.388162
+v 0.434703 0.399305 -0.486535
+v 0.376614 0.399090 -0.386084
+v 0.437905 0.396306 -0.484722
+v 0.379816 0.396092 -0.384271
+v 0.440543 0.392648 -0.483245
+v 0.382454 0.392434 -0.382794
+v 0.442516 0.388470 -0.482159
+v 0.384428 0.388256 -0.381708
+v 0.417443 0.383933 -0.436016
+v 0.385660 0.383719 -0.381055
+v 0.417860 0.379211 -0.435775
+v 0.386104 0.378997 -0.380859
+v 0.417397 0.374486 -0.435870
+v 0.385742 0.374272 -0.381131
+v 0.416243 0.369939 -0.436596
+v 0.384589 0.369725 -0.381858
+v 0.414343 0.365746 -0.437750
+v 0.382688 0.365532 -0.383012
+v 0.411768 0.362067 -0.439287
+v 0.380114 0.361852 -0.384549
+v 0.408619 0.359043 -0.441148
+v 0.376965 0.358828 -0.386410
+v 0.405015 0.356791 -0.443262
+v 0.373360 0.356576 -0.388523
+v 0.401095 0.355397 -0.445547
+v 0.369441 0.355183 -0.390808
+v 0.397010 0.354916 -0.447916
+v 0.365356 0.354701 -0.393177
+v 0.392917 0.355364 -0.450277
+v 0.361263 0.355150 -0.395538
+v 0.388973 0.356726 -0.452539
+v 0.357319 0.356512 -0.397801
+v 0.385331 0.358949 -0.454617
+v 0.353677 0.358735 -0.399878
+v 0.382128 0.361947 -0.456430
+v 0.350474 0.361733 -0.401690
+v 0.379490 0.365605 -0.457907
+v 0.347835 0.365391 -0.403169
+v 0.377516 0.369783 -0.458993
+v 0.345862 0.369569 -0.404255
+v 0.376285 0.374320 -0.459646
+v 0.344630 0.374106 -0.404907
+v 0.375942 0.379042 -0.460017
+v 0.344186 0.378828 -0.405102
+v 0.349310 0.243339 -0.475424
+v 0.335814 0.242789 -0.452084
+v 0.378320 0.253858 -0.519295
+v 0.338518 0.253308 -0.450466
+v 0.382750 0.263561 -0.516683
+v 0.342947 0.263011 -0.447854
+v 0.388737 0.272074 -0.513177
+v 0.348934 0.271524 -0.444349
+v 0.396048 0.279070 -0.508913
+v 0.356246 0.278520 -0.440085
+v 0.404405 0.284280 -0.504054
+v 0.364602 0.283730 -0.435225
+v 0.413485 0.287504 -0.498786
+v 0.373682 0.286954 -0.429958
+v 0.422940 0.288619 -0.493314
+v 0.383137 0.288069 -0.424485
+v 0.432406 0.287581 -0.487845
+v 0.392603 0.287030 -0.419016
+v 0.441519 0.284430 -0.482591
+v 0.401717 0.283880 -0.413762
+v 0.449930 0.279287 -0.477754
+v 0.410128 0.278737 -0.408924
+v 0.457315 0.272350 -0.473518
+v 0.417513 0.271800 -0.404690
+v 0.463390 0.263886 -0.470049
+v 0.423588 0.263336 -0.401220
+v 0.467923 0.254219 -0.467478
+v 0.428120 0.253669 -0.398649
+v 0.444431 0.243722 -0.420415
+v 0.430935 0.243172 -0.397075
+v 0.445413 0.232316 -0.419843
+v 0.431924 0.232252 -0.396560
+v 0.444389 0.221688 -0.420253
+v 0.431048 0.221322 -0.397117
+v 0.441708 0.211344 -0.421848
+v 0.428341 0.210797 -0.398730
+v 0.437275 0.201642 -0.424456
+v 0.423908 0.201094 -0.401338
+v 0.431286 0.193129 -0.427957
+v 0.417920 0.192581 -0.404840
+v 0.423973 0.186134 -0.432219
+v 0.410606 0.185585 -0.409101
+v 0.415616 0.180924 -0.437075
+v 0.402248 0.180374 -0.413959
+v 0.406535 0.177700 -0.442341
+v 0.393168 0.177150 -0.419225
+v 0.397080 0.176585 -0.447814
+v 0.383712 0.176035 -0.424697
+v 0.387614 0.177623 -0.453283
+v 0.374246 0.177073 -0.430167
+v 0.378502 0.180774 -0.458539
+v 0.365133 0.180224 -0.435422
+v 0.370092 0.185916 -0.463378
+v 0.356723 0.185368 -0.440262
+v 0.362709 0.192853 -0.467616
+v 0.349340 0.192304 -0.444500
+v 0.356636 0.201317 -0.471089
+v 0.343266 0.200769 -0.447973
+v 0.352106 0.210983 -0.473664
+v 0.338737 0.210436 -0.450548
+v 0.349324 0.221305 -0.475229
+v 0.335925 0.220938 -0.452127
+v 0.348394 0.231924 -0.475949
+v 0.334938 0.231861 -0.452646
+v 0.347168 0.243790 -0.476657
+v 0.315385 0.243271 -0.421696
+v 0.376298 0.254782 -0.520453
+v 0.318209 0.254264 -0.420003
+v 0.380926 0.264921 -0.517721
+v 0.322838 0.264402 -0.417271
+v 0.387181 0.273816 -0.514057
+v 0.329092 0.273298 -0.413606
+v 0.394820 0.281127 -0.509599
+v 0.336732 0.280609 -0.409149
+v 0.403552 0.286572 -0.504521
+v 0.345463 0.286053 -0.404069
+v 0.413040 0.289941 -0.499016
+v 0.354951 0.289423 -0.398565
+v 0.422920 0.291105 -0.493296
+v 0.364831 0.290587 -0.392845
+v 0.432812 0.290021 -0.487582
+v 0.374722 0.289502 -0.387132
+v 0.442335 0.286728 -0.482093
+v 0.384246 0.286210 -0.381641
+v 0.451125 0.281354 -0.477039
+v 0.393035 0.280836 -0.376587
+v 0.458842 0.274105 -0.472614
+v 0.400754 0.273587 -0.372164
+v 0.465191 0.265261 -0.468991
+v 0.407103 0.264742 -0.368541
+v 0.469929 0.255160 -0.466307
+v 0.411840 0.254641 -0.365855
+v 0.446566 0.244190 -0.419175
+v 0.414782 0.243672 -0.364214
+v 0.447573 0.232775 -0.418592
+v 0.415817 0.232256 -0.363677
+v 0.446559 0.221351 -0.419005
+v 0.414905 0.220833 -0.364267
+v 0.443735 0.210359 -0.420698
+v 0.412081 0.209840 -0.365960
+v 0.439107 0.200220 -0.423429
+v 0.407453 0.199702 -0.368691
+v 0.432853 0.191324 -0.427095
+v 0.401198 0.190806 -0.372356
+v 0.425213 0.184014 -0.431553
+v 0.393558 0.183495 -0.376813
+v 0.416481 0.178569 -0.436631
+v 0.384827 0.178051 -0.381892
+v 0.406994 0.175200 -0.442136
+v 0.375340 0.174682 -0.387398
+v 0.397114 0.174035 -0.447855
+v 0.365460 0.173517 -0.393117
+v 0.387222 0.175120 -0.453570
+v 0.355568 0.174602 -0.398831
+v 0.377699 0.178413 -0.459059
+v 0.346044 0.177894 -0.404320
+v 0.368909 0.183787 -0.464113
+v 0.337255 0.183268 -0.409374
+v 0.361191 0.191035 -0.468537
+v 0.329537 0.190517 -0.413798
+v 0.354842 0.199880 -0.472160
+v 0.323187 0.199362 -0.417422
+v 0.350105 0.209981 -0.474845
+v 0.318450 0.209463 -0.420106
+v 0.347162 0.220950 -0.476487
+v 0.315507 0.220432 -0.421749
+v 0.346229 0.232366 -0.477199
+v 0.314473 0.231848 -0.422285
+v 0.344186 0.050272 -0.405102
+v 0.375942 0.050487 -0.460017
+v 0.344630 0.045551 -0.404907
+v 0.376285 0.045765 -0.459646
+v 0.345862 0.041014 -0.404255
+v 0.377516 0.041228 -0.458993
+v 0.347835 0.036836 -0.403169
+v 0.379490 0.037050 -0.457907
+v 0.350474 0.033177 -0.401690
+v 0.382128 0.033392 -0.456430
+v 0.353677 0.030179 -0.399878
+v 0.385331 0.030393 -0.454617
+v 0.357319 0.027956 -0.397801
+v 0.388973 0.028171 -0.452539
+v 0.361263 0.026594 -0.395538
+v 0.392917 0.026809 -0.450277
+v 0.365356 0.026146 -0.393177
+v 0.397010 0.026360 -0.447916
+v 0.369441 0.026627 -0.390808
+v 0.401095 0.026842 -0.445547
+v 0.373360 0.028021 -0.388523
+v 0.405015 0.028235 -0.443262
+v 0.376965 0.030273 -0.386410
+v 0.408619 0.030487 -0.441148
+v 0.380114 0.033297 -0.384549
+v 0.411768 0.033511 -0.439287
+v 0.382688 0.036976 -0.383012
+v 0.414343 0.037191 -0.437750
+v 0.384589 0.041170 -0.381858
+v 0.416243 0.041384 -0.436596
+v 0.385742 0.045716 -0.381131
+v 0.417397 0.045931 -0.435870
+v 0.386104 0.050441 -0.380859
+v 0.417860 0.050656 -0.435775
+v 0.385660 0.055163 -0.381055
+v 0.417443 0.055377 -0.436016
+v 0.384428 0.059700 -0.381708
+v 0.442516 0.059915 -0.482159
+v 0.382454 0.063878 -0.382795
+v 0.440543 0.064093 -0.483245
+v 0.379816 0.067537 -0.384271
+v 0.437905 0.067751 -0.484722
+v 0.376614 0.070535 -0.386084
+v 0.434703 0.070749 -0.486535
+v 0.372970 0.072757 -0.388162
+v 0.431060 0.072972 -0.488613
+v 0.369027 0.074119 -0.390424
+v 0.427116 0.074334 -0.490875
+v 0.364934 0.074568 -0.392785
+v 0.423023 0.074782 -0.493235
+v 0.360850 0.074086 -0.395154
+v 0.418938 0.074301 -0.495604
+v 0.356930 0.072693 -0.397439
+v 0.415019 0.072907 -0.497889
+v 0.353326 0.070441 -0.399553
+v 0.411415 0.070655 -0.500003
+v 0.350175 0.067417 -0.401414
+v 0.408265 0.067631 -0.501865
+v 0.347601 0.063738 -0.402951
+v 0.405690 0.063952 -0.503401
+v 0.345700 0.059544 -0.404105
+v 0.403790 0.059758 -0.504556
+v 0.344548 0.054997 -0.404831
+v 0.376331 0.055212 -0.459792
+v 0.363374 0.050278 -0.436202
+v 0.376839 0.050304 -0.459500
+v 0.363795 0.045760 -0.436011
+v 0.377176 0.045912 -0.459122
+v 0.364970 0.041416 -0.435378
+v 0.378339 0.041642 -0.458494
+v 0.366854 0.037418 -0.434332
+v 0.380224 0.037644 -0.457449
+v 0.369377 0.033917 -0.432913
+v 0.382745 0.034143 -0.456029
+v 0.372438 0.031047 -0.431174
+v 0.385808 0.031274 -0.454291
+v 0.375924 0.028920 -0.429183
+v 0.389292 0.029147 -0.452299
+v 0.379696 0.027617 -0.427015
+v 0.393065 0.027844 -0.450132
+v 0.383613 0.027187 -0.424755
+v 0.396981 0.027415 -0.447871
+v 0.387522 0.027648 -0.422489
+v 0.400890 0.027876 -0.445606
+v 0.391275 0.028982 -0.420305
+v 0.404642 0.029209 -0.443421
+v 0.394726 0.031137 -0.418285
+v 0.408093 0.031364 -0.441402
+v 0.397743 0.034031 -0.416508
+v 0.411110 0.034258 -0.439626
+v 0.400209 0.037552 -0.415043
+v 0.413577 0.037779 -0.438160
+v 0.402031 0.041566 -0.413944
+v 0.415399 0.041792 -0.437062
+v 0.403140 0.045919 -0.413257
+v 0.416497 0.046070 -0.436382
+v 0.403489 0.050440 -0.413003
+v 0.416966 0.050466 -0.436293
+v 0.403065 0.054956 -0.413191
+v 0.416561 0.055184 -0.436531
+v 0.401888 0.059298 -0.413820
+v 0.441689 0.059526 -0.482648
+v 0.400000 0.063296 -0.414861
+v 0.439802 0.063524 -0.483689
+v 0.397476 0.066797 -0.416276
+v 0.437278 0.067025 -0.485106
+v 0.394413 0.069667 -0.418013
+v 0.434214 0.069894 -0.486841
+v 0.390927 0.071794 -0.420002
+v 0.430729 0.072021 -0.488830
+v 0.387154 0.073097 -0.422168
+v 0.426955 0.073325 -0.490996
+v 0.383237 0.073526 -0.424427
+v 0.423039 0.073754 -0.493257
+v 0.379327 0.073065 -0.426694
+v 0.419130 0.073293 -0.495522
+v 0.375576 0.071732 -0.428880
+v 0.415378 0.071959 -0.497708
+v 0.372126 0.069577 -0.430902
+v 0.411928 0.069804 -0.499730
+v 0.369111 0.066683 -0.432681
+v 0.408913 0.066911 -0.501510
+v 0.366646 0.063162 -0.434150
+v 0.406448 0.063390 -0.502979
+v 0.364827 0.059149 -0.435252
+v 0.404628 0.059376 -0.504081
+v 0.363721 0.054798 -0.435946
+v 0.377218 0.055025 -0.459284
+v 0.244813 -0.168051 -0.366253
+v 0.195330 -0.168051 -0.394869
+v 0.223945 -0.168051 -0.444354
+v 0.273429 -0.168051 -0.415738
+v 0.187727 -0.168051 -0.267534
+v 0.159111 -0.168051 -0.218050
+v 0.109626 -0.168051 -0.246666
+v 0.138242 -0.168051 -0.296151
+v 0.109626 -0.168051 -0.246666
+v 0.159111 -0.168051 -0.218050
+v 0.273429 -0.168051 -0.415738
+v 0.223945 -0.168051 -0.444354
+v 0.138242 -0.168051 -0.296151
+v 0.195330 -0.168051 -0.394869
+v 0.187727 -0.168051 -0.267534
+v 0.244813 -0.168051 -0.366253
+v 0.244813 0.037812 -0.366253
+v 0.244813 0.165702 -0.366253
+v 0.195330 0.165702 -0.394869
+v 0.195330 0.037812 -0.394869
+v 0.223945 0.037812 -0.444354
+v 0.223945 0.165702 -0.444354
+v 0.273429 0.165702 -0.415738
+v 0.273429 0.037812 -0.415738
+v 0.187727 0.037812 -0.267534
+v 0.187727 0.165702 -0.267534
+v 0.159111 0.165702 -0.218050
+v 0.159111 0.037812 -0.218050
+v 0.109626 0.037812 -0.246666
+v 0.109626 0.165702 -0.246666
+v 0.138242 0.165702 -0.296151
+v 0.138242 0.037812 -0.296151
+v 0.244813 0.166622 -0.366253
+v 0.195330 0.166622 -0.394869
+v 0.223945 0.166622 -0.444354
+v 0.273429 0.166622 -0.415738
+v 0.187727 0.166622 -0.267534
+v 0.159111 0.166622 -0.218050
+v 0.109626 0.166622 -0.246666
+v 0.138242 0.166622 -0.296151
+v 0.109626 0.166622 -0.246666
+v 0.159111 0.166622 -0.218050
+v 0.273429 0.166622 -0.415738
+v 0.223945 0.166622 -0.444354
+v 0.138242 0.166622 -0.296151
+v 0.195330 0.166622 -0.394869
+v 0.187727 0.166622 -0.267534
+v 0.244813 0.166622 -0.366253
+v 0.244813 0.370885 -0.366253
+v 0.244813 0.497782 -0.366253
+v 0.195330 0.497782 -0.394869
+v 0.195330 0.370885 -0.394869
+v 0.223945 0.370885 -0.444354
+v 0.223945 0.497782 -0.444354
+v 0.273429 0.497782 -0.415738
+v 0.273429 0.370885 -0.415738
+v 0.187727 0.370885 -0.267534
+v 0.187727 0.497782 -0.267534
+v 0.159111 0.497782 -0.218050
+v 0.159111 0.370885 -0.218050
+v 0.109626 0.370885 -0.246666
+v 0.109626 0.497782 -0.246666
+v 0.138242 0.497782 -0.296151
+v 0.138242 0.370885 -0.296151
+v 0.507761 -0.080213 -0.339964
+v 0.074923 -0.080213 -0.590266
+v 0.507761 0.496942 -0.339963
+v 0.074922 0.496942 -0.590265
+v 0.484023 0.496942 -0.298914
+v 0.484023 -0.080213 -0.298914
+v 0.051185 -0.080213 -0.549217
+v 0.051185 0.496942 -0.549217
+vn -0.8657 -0.0000 -0.5006
+vn 0.8657 -0.0000 0.5006
+vn 0.5006 -0.0000 -0.8657
+vn -0.5006 -0.0000 0.8657
+vn -0.0000 -1.0000 -0.0000
+vn -0.0000 1.0000 -0.0000
+vn -0.0784 -0.9955 -0.0533
+vn 0.8628 -0.0946 0.4967
+vn -0.8602 -0.1015 -0.4998
+vn -0.8631 0.0944 -0.4962
+vn 0.8599 0.1014 0.5003
+vn -0.8622 0.0945 -0.4977
+vn 0.8608 0.1015 0.4987
+vn 0.8622 -0.0946 0.4977
+vn -0.8608 -0.1014 -0.4988
+vn 0.6607 0.6370 0.3972
+vn 0.5075 0.0090 -0.8616
+vn 0.5028 0.0067 -0.8644
+vn 0.5010 0.0100 -0.8654
+vn 0.4986 0.0089 -0.8668
+vn 0.4984 0.0091 -0.8669
+vn 0.5018 0.0096 -0.8649
+vn 0.5025 0.0054 -0.8646
+vn 0.4980 0.0081 -0.8671
+vn 0.5001 0.0101 -0.8659
+vn 0.5027 0.0077 -0.8644
+vn 0.4993 0.0093 -0.8664
+vn 0.5025 0.0087 -0.8646
+vn 0.4977 0.0071 -0.8673
+vn 0.4983 0.0059 -0.8670
+vn 0.5021 0.0057 -0.8648
+vn 0.4990 0.0002 -0.8666
+vn 0.4979 -0.0134 -0.8671
+vn 0.4962 -0.0144 -0.8681
+vn 0.2643 -0.9600 -0.0924
+vn 0.5083 0.0120 -0.8611
+vn 0.4929 0.0119 -0.8700
+vn 0.5079 0.0151 -0.8613
+vn 0.5066 0.0180 -0.8620
+vn 0.5044 0.0198 -0.8632
+vn 0.4990 0.0208 -0.8663
+vn 0.4965 0.0200 -0.8678
+vn 0.4944 0.0178 -0.8690
+vn 0.4933 0.0147 -0.8697
+vn -0.2109 -0.9097 -0.3576
+vn 0.4936 0.0089 -0.8696
+vn 0.4950 0.0057 -0.8689
+vn 0.8624 -0.0920 0.4978
+vn -0.5532 0.7708 -0.3160
+vn 0.5451 0.7752 0.3192
+vn -0.6650 -0.6371 -0.3897
+vn 0.4034 0.8836 0.2379
+vn -0.2445 -0.9579 -0.1505
+vn -0.2451 -0.9579 -0.1494
+vn 0.0795 0.9955 0.0511
+vn 0.0913 -0.9948 0.0449
+vn 0.2463 0.9579 0.1474
+vn 0.4138 -0.8803 0.2322
+vn 0.5542 -0.7708 0.3143
+vn -0.7604 -0.4745 -0.4435
+vn 0.7665 -0.4683 0.4395
+vn 0.8278 -0.2964 0.4763
+vn -0.8609 -0.0990 -0.4990
+vn -0.8609 0.1021 -0.4985
+vn 0.7571 0.4744 0.4491
+vn 0.5018 0.0208 -0.8647
+vn 0.8244 0.2954 0.4828
+vn 0.5014 -0.0002 -0.8652
+vn 0.5388 0.7751 0.3300
+vn 0.3962 0.8834 0.2501
+vn 0.6658 0.6371 0.3883
+vn -0.4018 -0.8835 -0.2407
+vn -0.2640 0.9557 -0.1299
+vn -0.4198 0.8801 -0.2218
+vn -0.5594 0.7707 -0.3052
+vn -0.6776 0.6316 -0.3768
+vn -0.7697 0.4683 -0.4339
+vn -0.8314 0.2888 -0.4748
+vn 0.8607 0.1091 0.4973
+vn -0.8622 0.0946 -0.4977
+vn -0.6647 -0.6370 -0.3904
+vn -0.5436 -0.7752 -0.3217
+vn 0.7610 0.4745 0.4425
+vn 0.0716 0.9953 0.0651
+vn 0.6733 -0.6317 0.3843
+vn -0.4024 -0.8835 -0.2397
+vn -0.0778 -0.9955 -0.0544
+vn -0.0900 0.9948 -0.0469
+vn -0.2563 0.9559 -0.1433
+vn -0.4127 0.8802 -0.2341
+vn -0.8239 -0.3033 -0.4789
+vn -0.6725 0.6317 -0.3856
+vn -0.7659 0.4683 -0.4405
+vn -0.8284 0.2926 -0.4776
+vn 0.8608 0.1014 0.4987
+vn 0.2385 0.9578 0.1607
+vn -0.8263 -0.2936 -0.4806
+vn -0.7601 -0.4745 -0.4440
+vn 0.8263 0.2936 0.4806
+vn -0.0981 0.9946 -0.0331
+vn -0.5442 -0.7752 -0.3209
+vn 0.2575 -0.9559 0.1413
+vn 0.0920 -0.9948 0.0438
+vn 0.2581 -0.9559 0.1402
+vn 0.4143 -0.8803 0.2313
+vn 0.5547 -0.7708 0.3135
+vn 0.6737 -0.6317 0.3836
+vn 0.7668 -0.4683 0.4390
+vn 0.8305 -0.2869 0.4775
+vn -0.8608 -0.1015 -0.4987
+vn 0.5169 0.0217 -0.8558
+vn -0.7612 -0.4745 -0.4421
+vn 0.8259 0.2956 0.4801
+vn 0.8620 -0.0946 0.4981
+vn 0.7616 0.4745 0.4414
+vn -0.5455 -0.7753 -0.3185
+vn -0.8245 -0.3031 -0.4778
+vn -0.2550 0.9560 -0.1454
+vn -0.5522 0.7708 -0.3177
+vn 0.4827 0.0359 -0.8751
+vn 0.0810 0.9955 0.0489
+vn 0.6720 -0.6317 0.3865
+vn 0.0893 -0.9948 0.0484
+vn 0.8607 0.1013 0.4990
+vn -0.7653 0.4684 -0.4415
+vn 0.8622 -0.0920 0.4982
+vn 0.4954 0.0218 -0.8684
+vn 0.4952 0.0220 -0.8685
+vn 0.5048 0.0207 -0.8630
+vn 0.5094 0.0479 -0.8592
+vn 0.4965 0.0503 -0.8666
+vn -0.8611 0.1020 -0.4982
+vn 0.5527 -0.7708 0.3170
+vn -0.0803 -0.9955 -0.0497
+vn 0.5055 0.0131 -0.8628
+vn -0.0800 -0.9955 -0.0502
+vn 0.4901 -0.0348 -0.8710
+vn 0.5015 0.0244 -0.8648
+vn -0.6738 0.6316 -0.3834
+vn -0.8620 0.0946 -0.4981
+vn 0.6721 -0.6318 0.3862
+vn -0.8623 0.0945 -0.4975
+vn 0.4016 0.8835 0.2410
+vn 0.5053 0.0160 -0.8628
+vn 0.4945 0.0192 -0.8689
+vn -0.4041 -0.8836 -0.2366
+vn -0.5457 -0.7752 -0.3181
+vn 0.5032 0.0503 -0.8627
+vn 0.5032 0.0504 -0.8627
+vn 0.4820 0.0286 -0.8757
+vn 0.2554 -0.9560 0.1446
+vn 0.0898 -0.9948 0.0480
+vn 0.5045 0.0138 -0.8633
+vn 0.2445 0.9579 0.1508
+vn -0.0921 0.9948 -0.0435
+vn 0.7600 0.4744 0.4442
+vn -0.7669 0.4683 -0.4389
+vn 0.5034 0.0229 -0.8638
+vn -0.2583 0.9559 -0.1400
+vn 0.4853 0.0428 -0.8733
+vn 0.4853 0.0429 -0.8733
+vn -0.8610 -0.1015 -0.4983
+vn 0.2476 0.9579 0.1452
+vn 0.7657 -0.4682 0.4410
+vn 0.5179 0.0364 -0.8546
+vn 0.8606 0.1089 0.4975
+vn 0.8622 -0.0945 0.4977
+vn 0.4966 0.0004 -0.8680
+vn -0.6663 -0.6371 -0.3874
+vn -0.8612 -0.0991 -0.4985
+vn 0.4119 -0.8803 0.2353
+vn -0.8610 -0.1016 -0.4983
+vn 0.5529 -0.7707 0.3167
+vn -0.6661 -0.6371 -0.3877
+vn 0.7656 -0.4683 0.4412
+vn -0.0888 0.9948 -0.0492
+vn 0.5435 0.7752 0.3220
+vn -0.8281 0.2926 -0.4782
+vn 0.5146 0.0430 -0.8563
+vn -0.5547 0.7708 -0.3132
+vn 0.8270 0.2937 0.4794
+vn -0.2470 -0.9580 -0.1460
+vn 0.4902 0.0478 -0.8703
+vn 0.4902 0.0481 -0.8703
+vn -0.2753 -0.9260 -0.2584
+vn 0.8298 -0.2869 0.4787
+vn 0.5189 0.0291 -0.8543
+vn 0.4939 -0.0325 -0.8689
+vn 0.5462 0.7752 0.3175
+vn 0.2559 -0.9559 0.1442
+vn -0.6717 0.6316 -0.3871
+vn -0.8271 -0.2935 -0.4794
+vn 0.8610 0.1015 0.4983
+vn 0.5057 0.0187 -0.8625
+vn 0.5058 0.0187 -0.8625
+vn 0.4947 0.0143 -0.8689
+vn 0.6667 0.6370 0.3869
+vn -0.4116 0.8803 -0.2361
+vn 0.4123 -0.8803 0.2349
+vn 0.4879 0.0135 -0.8728
+vn 0.4875 0.0136 -0.8730
+vn 0.4877 0.0136 -0.8729
+vn -0.7614 -0.4744 -0.4419
+vn 0.0775 0.9955 0.0546
+vn 0.2059 -0.9785 0.0156
+vn 0.4941 0.0168 -0.8692
+vn 0.4974 0.0230 -0.8672
+vn -0.4145 0.8802 -0.2311
+vn 0.6645 0.6371 0.3906
+vn 0.4841 0.0214 -0.8747
+vn 0.4844 0.0212 -0.8746
+vn -0.4040 -0.8835 -0.2371
+vn -0.8298 0.2888 -0.4774
+vn 0.5021 -0.0005 -0.8648
+vn 0.5020 -0.0006 -0.8649
+vn 0.8272 -0.2964 0.4774
+vn -0.2468 -0.9579 -0.1465
+vn 0.4045 0.8836 0.2358
+vn 0.4992 0.0244 -0.8661
+vn -0.0785 -0.9955 -0.0533
+vn 0.5055 0.0071 -0.8628
+vn 0.5033 0.0070 -0.8641
+vn 0.5010 0.0101 -0.8654
+vn 0.4987 0.0087 -0.8667
+vn 0.5019 0.0098 -0.8648
+vn 0.5060 0.0057 -0.8625
+vn 0.4983 0.0078 -0.8670
+vn 0.5002 0.0099 -0.8659
+vn 0.5032 0.0081 -0.8641
+vn 0.4991 0.0097 -0.8665
+vn 0.5027 0.0091 -0.8644
+vn 0.4987 0.0065 -0.8668
+vn 0.4990 0.0057 -0.8666
+vn 0.5032 0.0060 -0.8642
+vn 0.4999 -0.0002 -0.8661
+vn 0.5052 -0.0144 -0.8629
+vn 0.5049 -0.0144 -0.8631
+vn 0.5035 -0.0133 -0.8639
+vn 0.5032 -0.0134 -0.8641
+vn 0.4209 -0.9071 0.0078
+vn 0.5069 0.0099 -0.8620
+vn 0.4942 0.0100 -0.8693
+vn 0.5071 0.0133 -0.8618
+vn 0.5063 0.0163 -0.8622
+vn 0.5044 0.0191 -0.8633
+vn 0.4990 0.0206 -0.8663
+vn 0.4965 0.0188 -0.8678
+vn 0.4949 0.0159 -0.8688
+vn 0.4940 0.0132 -0.8694
+vn -0.0460 -0.9613 -0.2718
+vn 0.4955 0.0071 -0.8686
+vn 0.4986 0.0054 -0.8668
+vn 0.4034 0.8835 0.2379
+vn -0.2444 -0.9579 -0.1504
+vn -0.2450 -0.9580 -0.1493
+vn 0.0796 0.9955 0.0512
+vn 0.2462 0.9580 0.1473
+vn 0.4138 -0.8802 0.2322
+vn 0.5541 -0.7708 0.3143
+vn -0.7604 -0.4744 -0.4435
+vn 0.7665 -0.4682 0.4395
+vn 0.8322 -0.2869 0.4745
+vn 0.5019 0.0203 -0.8647
+vn 0.8245 0.2935 0.4838
+vn 0.5024 0.0002 -0.8646
+vn 0.3963 0.8834 0.2502
+vn 0.6659 0.6370 0.3883
+vn -0.8323 0.2868 -0.4744
+vn -0.6646 -0.6371 -0.3904
+vn -0.5437 -0.7752 -0.3217
+vn 0.0715 0.9953 0.0651
+vn -0.2564 0.9559 -0.1433
+vn -0.4127 0.8803 -0.2341
+vn -0.8245 -0.2937 -0.4837
+vn -0.8304 0.2869 -0.4775
+vn -0.8263 -0.2937 -0.4806
+vn -0.7601 -0.4744 -0.4441
+vn 0.8253 0.2995 0.4788
+vn 0.5024 0.0065 -0.8646
+vn 0.4985 0.0089 -0.8668
+vn 0.5017 0.0092 -0.8650
+vn 0.5024 0.0053 -0.8646
+vn 0.4979 0.0081 -0.8672
+vn 0.5000 0.0101 -0.8659
+vn 0.5028 0.0076 -0.8644
+vn 0.4994 0.0093 -0.8663
+vn 0.4981 0.0069 -0.8671
+vn 0.4980 0.0061 -0.8672
+vn 0.4990 0.0001 -0.8666
+vn 0.4965 -0.0144 -0.8679
+vn 0.2641 -0.9600 -0.0925
+vn 0.5080 0.0118 -0.8613
+vn 0.5065 0.0177 -0.8621
+vn 0.4944 0.0177 -0.8691
+vn 0.4932 0.0151 -0.8698
+vn 0.4940 0.0087 -0.8694
+vn 0.4953 0.0056 -0.8687
+vn -0.8609 -0.0991 -0.4990
+vn 0.5011 -0.0002 -0.8654
+vn -0.8239 -0.3032 -0.4789
+vn -0.8284 0.2927 -0.4776
+vn 0.4144 -0.8802 0.2313
+vn 0.5546 -0.7708 0.3134
+vn 0.5032 0.0070 -0.8641
+vn 0.5020 0.0098 -0.8648
+vn 0.5061 0.0057 -0.8625
+vn 0.4986 0.0075 -0.8668
+vn 0.5001 0.0099 -0.8659
+vn 0.4989 0.0097 -0.8666
+vn 0.5025 0.0088 -0.8646
+vn 0.4984 0.0067 -0.8669
+vn 0.4991 0.0057 -0.8665
+vn 0.5028 0.0059 -0.8644
+vn 0.4999 -0.0003 -0.8661
+vn 0.5051 -0.0144 -0.8630
+vn 0.5031 -0.0134 -0.8641
+vn 0.4205 -0.9073 0.0076
+vn 0.5069 0.0100 -0.8620
+vn 0.4945 0.0096 -0.8691
+vn 0.5072 0.0134 -0.8617
+vn 0.5070 0.0130 -0.8619
+vn 0.5062 0.0164 -0.8623
+vn 0.5043 0.0187 -0.8633
+vn 0.4966 0.0188 -0.8678
+vn 0.4948 0.0163 -0.8689
+vn 0.4958 0.0069 -0.8684
+vn 0.4986 0.0053 -0.8668
+vn 0.4989 0.0053 -0.8666
+vn 0.5021 0.0002 -0.8648
+vn -0.0805 -0.9955 -0.0499
+vn 0.5125 0.0170 -0.8585
+vn 0.5122 0.0169 -0.8587
+vn 0.5075 0.0173 -0.8615
+vn 0.5016 0.0245 -0.8648
+vn 0.5016 0.0247 -0.8647
+vn 0.4963 0.0206 -0.8679
+vn 0.5038 0.0238 -0.8635
+vn 0.5136 0.0138 -0.8579
+vn 0.4955 0.0184 -0.8684
+vn 0.4993 0.0244 -0.8661
+vn 0.4993 0.0241 -0.8661
+vn 0.5067 0.0195 -0.8619
+vn 0.4969 0.0238 -0.8675
+vn 0.4968 0.0241 -0.8675
+vn 0.5054 0.0215 -0.8626
+vn 0.4958 0.0159 -0.8683
+vn 0.4968 0.0137 -0.8677
+vn 0.4967 0.0137 -0.8678
+vn 0.5063 0.0144 -0.8622
+vn 0.5063 0.0144 -0.8623
+vn 0.4991 -0.0005 -0.8665
+vn 0.5108 -0.0347 -0.8590
+vn 0.5107 -0.0347 -0.8591
+vn 0.5066 -0.0323 -0.8616
+vn 0.3672 -0.9232 0.1132
+vn 0.5156 0.0239 -0.8565
+vn 0.4853 0.0238 -0.8740
+vn 0.4853 0.0237 -0.8740
+vn 0.5161 0.0320 -0.8559
+vn 0.5161 0.0320 -0.8560
+vn 0.5139 0.0394 -0.8569
+vn 0.5139 0.0391 -0.8570
+vn 0.5139 0.0396 -0.8569
+vn 0.5093 0.0455 -0.8594
+vn 0.4963 0.0496 -0.8667
+vn 0.4904 0.0459 -0.8703
+vn 0.4863 0.0394 -0.8729
+vn 0.4863 0.0393 -0.8729
+vn 0.4845 0.0316 -0.8742
+vn 0.4846 0.0314 -0.8742
+vn -0.1103 -0.9797 -0.1671
+vn 0.4889 0.0166 -0.8722
+vn 0.4956 0.0129 -0.8685
+vn -0.5521 0.7709 -0.3176
+vn 0.5461 0.7753 0.3174
+vn 0.4046 0.8836 0.2359
+vn 0.0893 -0.9948 0.0483
+vn 0.4120 -0.8803 0.2353
+vn 0.5525 -0.7709 0.3170
+vn -0.7614 -0.4745 -0.4418
+vn 0.7656 -0.4683 0.4411
+vn 0.8305 -0.2871 0.4774
+vn 0.7600 0.4743 0.4442
+vn 0.5034 0.0493 -0.8626
+vn 0.8263 0.2935 0.4807
+vn 0.5044 0.0005 -0.8635
+vn 0.4016 0.8836 0.2409
+vn -0.4145 0.8802 -0.2310
+vn -0.6739 0.6315 -0.3835
+vn -0.8305 0.2869 -0.4774
+vn 0.6721 -0.6316 0.3866
+vn -0.0802 -0.9955 -0.0503
+vn -0.0888 0.9948 -0.0493
+vn -0.2551 0.9559 -0.1455
+vn -0.7653 0.4683 -0.4416
+vn -0.8298 0.2868 -0.4787
+vn 0.8257 0.2995 0.4781
+vn -0.5457 -0.7752 -0.3182
+vn 0.2556 -0.9559 0.1447
+vn 0.0896 -0.9948 0.0479
+vn 0.2557 -0.9559 0.1441
+vn 0.6720 -0.6319 0.3862
+vn 0.7656 -0.4684 0.4409
+vn -0.8298 0.2867 -0.4787
+vn -0.2472 -0.9579 -0.1461
+vn -0.0803 -0.9955 -0.0498
+vn 0.5022 -0.0005 -0.8648
+vn 0.5023 -0.0005 -0.8647
+vn 0.5112 -0.0347 -0.8588
+vn 0.5110 -0.0347 -0.8589
+vn 0.5032 0.0502 -0.8627
+vn 0.5042 0.0137 -0.8635
+vn 0.5055 0.0218 -0.8625
+vn 0.5436 0.7751 0.3220
+vn -0.0887 0.9948 -0.0491
+vn 0.7657 -0.4683 0.4409
+vn -0.6662 -0.6370 -0.3878
+vn 0.4955 0.0129 -0.8685
+vn 0.4940 0.0168 -0.8693
+vn -0.7669 0.4682 -0.4389
+vn 0.5190 0.0294 -0.8542
+vn 0.5192 0.0297 -0.8541
+vn 0.5190 0.0293 -0.8543
+vn -0.8298 0.2889 -0.4774
+vn 0.5032 0.0226 -0.8638
+vn 0.0777 0.9955 0.0548
+vn 0.4962 0.0206 -0.8679
+vn 0.5435 0.7752 0.3219
+vn -0.4042 -0.8836 -0.2366
+vn 0.4888 0.0166 -0.8722
+vn 0.0808 0.9955 0.0489
+vn 0.4992 -0.0005 -0.8665
+vn -0.5548 0.7707 -0.3133
+vn 0.6721 -0.6317 0.3862
+vn 0.5093 0.0479 -0.8592
+vn 0.4956 0.0217 -0.8683
+vn 0.4957 0.0217 -0.8682
+vn 0.5180 0.0366 -0.8546
+vn 0.6722 -0.6317 0.3862
+vn -0.2581 0.9559 -0.1399
+vn 0.8304 -0.2871 0.4774
+vn 0.5058 0.0142 -0.8625
+vn 0.4122 -0.8803 0.2349
+vn 0.4954 0.0161 -0.8685
+vn 0.4951 0.0163 -0.8687
+vn 0.2053 -0.9786 0.0153
+vn 0.5016 0.0244 -0.8648
+vn 0.4851 0.0434 -0.8734
+vn 0.4975 0.0226 -0.8672
+vn 0.4975 0.0225 -0.8672
+vn 0.4975 0.0227 -0.8672
+vn 0.2443 0.9579 0.1507
+vn 0.4120 -0.8803 0.2354
+vn -0.5458 -0.7751 -0.3182
+vn 0.5015 0.0240 -0.8649
+vn 0.4965 0.0505 -0.8666
+vn 0.4864 0.0389 -0.8729
+vn 0.4854 0.0237 -0.8740
+vn 0.5065 0.0195 -0.8620
+vn -0.8263 -0.2936 -0.4807
+vn 0.5124 0.0171 -0.8586
+vn 0.0894 -0.9948 0.0478
+vn -0.8612 -0.0986 -0.4985
+vn -0.8612 -0.0987 -0.4985
+vn 0.4845 0.0315 -0.8742
+vn 0.4848 0.0312 -0.8741
+vn -0.8619 0.0943 -0.4982
+vn 0.6668 0.6369 0.3870
+vn -0.6717 0.6317 -0.3871
+vn 0.5163 0.0326 -0.8558
+vn 0.8297 -0.2870 0.4787
+vn 0.5069 -0.0323 -0.8614
+vn -0.4039 -0.8836 -0.2370
+vn 0.5072 0.0171 -0.8616
+vn 0.5139 0.0397 -0.8569
+vn 0.5054 0.0161 -0.8627
+vn 0.5052 0.0159 -0.8628
+vn 0.5050 0.0006 -0.8631
+vn 0.8257 0.2994 0.4781
+vn 0.4970 0.0237 -0.8674
+vn 0.7656 -0.4682 0.4412
+vn 0.4898 -0.0347 -0.8711
+vn -0.5456 -0.7752 -0.3185
+vn 0.8610 0.1016 0.4983
+vn 0.4946 0.0143 -0.8690
+vn 0.4946 0.0144 -0.8690
+vn 0.4842 0.0213 -0.8747
+vn 0.8259 0.2955 0.4801
+vn 0.5052 0.0131 -0.8629
+vn 0.4942 0.0195 -0.8691
+vn 0.4946 0.0192 -0.8689
+vn 0.4824 0.0283 -0.8755
+vn 0.4970 0.0136 -0.8676
+vn -0.1103 -0.9797 -0.1672
+vn 0.4826 0.0359 -0.8751
+vn 0.5050 0.0210 -0.8629
+vn 0.5006 0.0044 -0.8657
+vn 0.5006 0.0040 -0.8657
+vn 0.5006 0.0040 -0.8656
+vn 0.5006 0.0045 -0.8657
+vn 0.5001 0.0042 -0.8659
+vn 0.5007 0.0044 -0.8656
+vn 0.5006 0.0046 -0.8657
+vn 0.5006 0.0045 -0.8656
+vn 0.5004 0.0043 -0.8658
+vn 0.5004 0.0044 -0.8658
+vn 0.5001 0.0043 -0.8660
+vn 0.5010 0.0038 -0.8654
+vn 0.5007 0.0038 -0.8656
+vn 0.5005 0.0041 -0.8657
+vn 0.5006 0.0039 -0.8657
+vn 0.5007 0.0041 -0.8656
+vn 0.5005 0.0040 -0.8657
+vn 0.5017 0.0042 -0.8650
+vn 0.5007 0.0043 -0.8656
+vn 0.5004 0.0109 -0.8657
+vn 0.5006 0.0097 -0.8656
+vn 0.5011 0.0092 -0.8653
+vn 0.5004 0.0107 -0.8658
+vn 0.5031 0.0102 -0.8642
+vn 0.5032 0.0102 -0.8641
+vn 0.5008 0.0095 -0.8655
+vn 0.5003 0.0102 -0.8658
+vn 0.5004 0.0099 -0.8657
+vn 0.5007 0.0109 -0.8656
+vn 0.5005 0.0095 -0.8657
+vn 0.5004 0.0092 -0.8658
+vn 0.5007 0.0098 -0.8656
+vn 0.5007 0.0099 -0.8656
+vn 0.5006 0.0096 -0.8656
+vn 0.5006 0.0106 -0.8656
+vn 0.5006 0.0107 -0.8656
+vn 0.5006 0.0109 -0.8656
+vn 0.5007 0.0104 -0.8655
+vn 0.5005 0.0096 -0.8657
+vn 0.5006 0.0110 -0.8656
+vn 0.5005 0.0109 -0.8656
+vn 0.5005 0.0107 -0.8657
+vn 0.5005 0.0108 -0.8656
+vn 0.5007 0.0108 -0.8656
+vn 0.5007 0.0108 -0.8655
+vn 0.5008 0.0093 -0.8655
+vn 0.4997 0.0100 -0.8661
+vn 0.5007 0.0093 -0.8656
+vn 0.5009 0.0107 -0.8654
+vn 0.5004 0.0104 -0.8657
+vn 0.5009 0.0104 -0.8655
+vn 0.5003 0.0041 -0.8658
+vn 0.5006 0.0044 -0.8656
+vn 0.5005 0.0047 -0.8657
+vn 0.5007 0.0045 -0.8656
+vn 0.5004 0.0041 -0.8658
+vn 0.5007 0.0040 -0.8656
+vn 0.5009 0.0044 -0.8655
+vn 0.5003 0.0047 -0.8659
+vn 0.5007 0.0039 -0.8656
+vn 0.5006 0.0046 -0.8656
+vn 0.5005 0.0044 -0.8657
+vn 0.5008 0.0038 -0.8656
+vn 0.5008 0.0038 -0.8655
+vn 0.5007 0.0046 -0.8656
+vn 0.5008 0.0044 -0.8656
+vn 0.5002 0.0037 -0.8659
+vn 0.5013 0.0042 -0.8652
+vn 0.4998 0.0099 -0.8661
+vn 0.5005 0.0110 -0.8656
+vn 0.5004 0.0103 -0.8657
+vn 0.5007 0.0096 -0.8656
+vn 0.5007 0.0095 -0.8656
+vn 0.5033 0.0103 -0.8640
+vn 0.5016 0.0089 -0.8651
+vn 0.5004 0.0107 -0.8657
+vn 0.5007 0.0109 -0.8655
+vn 0.5015 0.0104 -0.8651
+vn 0.5012 0.0106 -0.8653
+vn 0.5009 0.0096 -0.8654
+vn 0.5008 0.0096 -0.8655
+vn 0.5003 0.0097 -0.8658
+vn 0.5004 0.0096 -0.8657
+vn 0.5005 0.0097 -0.8657
+vn 0.5004 0.0095 -0.8658
+vn 0.5000 0.0099 -0.8659
+vn 0.4998 0.0100 -0.8661
+vn 0.5006 0.0108 -0.8656
+vt 0.357039 0.250000
+vt 0.785559 0.750000
+vt 0.214079 0.250000
+vt 0.714079 0.750000
+vt 0.285559 0.250000
+vt 0.571118 0.250000
+vt 0.285559 0.250000
+vt 0.642599 0.250000
+vt 0.928520 0.154203
+vt 0.499638 0.250000
+vt 0.714079 0.750000
+vt 0.571118 0.250000
+vt 0.642599 0.750000
+vt 0.000000 0.250000
+vt 0.857040 0.154203
+vt 0.071480 0.250000
+vt 0.857040 0.480583
+vt 0.357039 0.095797
+vt 0.785559 0.904203
+vt 0.785559 1.000000
+vt 0.357039 0.000000
+vt 0.785559 0.043154
+vt 0.714079 1.000000
+vt 0.214079 0.000000
+vt 0.857040 0.043154
+vt 0.928520 0.480583
+vt 0.214079 0.095797
+vt 0.714079 0.904203
+vt 0.285559 0.095797
+vt 0.571118 0.095797
+vt 0.285559 0.000000
+vt 0.571118 0.000000
+vt 0.857040 0.000000
+vt 0.285559 0.000000
+vt 0.642598 0.000000
+vt 0.785559 0.000000
+vt 0.285559 0.095797
+vt 0.642599 0.095797
+vt 0.857040 0.394495
+vt 0.928520 0.000000
+vt 0.499638 0.095797
+vt 0.499638 0.000000
+vt 0.785559 0.129242
+vt 0.714079 1.000000
+vt 0.571118 0.000000
+vt 0.785559 0.172396
+vt 0.714079 0.904203
+vt 0.571118 0.095797
+vt 0.642599 0.904203
+vt 0.000000 0.095797
+vt 0.642599 1.000000
+vt 0.000000 0.000000
+vt 0.857040 0.172396
+vt 0.071480 0.000000
+vt 0.857040 0.129242
+vt 0.928520 0.394495
+vt 0.857040 0.000000
+vt 0.071480 0.095797
+vt 0.578089 0.326242
+vt 0.370472 0.723575
+vt 0.442971 0.724510
+vt 0.846857 0.620495
+vt 0.579868 0.178984
+vt 0.229113 0.694984
+vt 0.442901 0.694984
+vt 0.817898 0.626255
+vt 0.589057 0.151745
+vt 0.229049 0.665457
+vt 0.442837 0.665456
+vt 0.788371 0.626255
+vt 0.605204 0.126737
+vt 0.228992 0.635930
+vt 0.442780 0.635930
+vt 0.759412 0.620495
+vt 0.626230 0.105756
+vt 0.228946 0.606403
+vt 0.442734 0.606402
+vt 0.732132 0.609195
+vt 0.650942 0.089526
+vt 0.228912 0.576877
+vt 0.442700 0.576876
+vt 0.707581 0.592792
+vt 0.678244 0.078632
+vt 0.228890 0.547350
+vt 0.442678 0.547349
+vt 0.686702 0.571913
+vt 0.228883 0.517823
+vt 0.706979 0.073452
+vt 0.442670 0.517822
+vt 0.670298 0.547363
+vt 0.228889 0.488295
+vt 0.735929 0.074111
+vt 0.442678 0.488295
+vt 0.658999 0.520082
+vt 0.763884 0.080481
+vt 0.228910 0.458769
+vt 0.442698 0.458768
+vt 0.653238 0.491123
+vt 0.789700 0.092175
+vt 0.228944 0.429242
+vt 0.442732 0.429241
+vt 0.653238 0.461596
+vt 0.812355 0.108585
+vt 0.228990 0.399716
+vt 0.442778 0.399714
+vt 0.658998 0.432637
+vt 0.831035 0.128981
+vt 0.229046 0.370189
+vt 0.442833 0.370188
+vt 0.670298 0.405358
+vt 0.844628 0.151667
+vt 0.229109 0.340661
+vt 0.442897 0.340661
+vt 0.686702 0.380807
+vt 0.879631 0.287703
+vt 0.370467 0.312068
+vt 0.442967 0.311134
+vt 0.707581 0.359928
+vt 0.370705 0.281246
+vt 0.885704 0.319257
+vt 0.443056 0.281620
+vt 0.732120 0.343530
+vt 0.371348 0.252525
+vt 0.885423 0.348954
+vt 0.443160 0.252098
+vt 0.759394 0.332230
+vt 0.879533 0.378395
+vt 0.371442 0.223476
+vt 0.443252 0.222558
+vt 0.788366 0.326464
+vt 0.867982 0.406513
+vt 0.371495 0.193949
+vt 0.443305 0.193031
+vt 0.817893 0.326463
+vt 0.851292 0.431762
+vt 0.371495 0.164423
+vt 0.443305 0.163503
+vt 0.846854 0.332224
+vt 0.830124 0.453224
+vt 0.371427 0.134897
+vt 0.443237 0.133976
+vt 0.874134 0.343522
+vt 0.805300 0.470133
+vt 0.371280 0.105372
+vt 0.443090 0.104449
+vt 0.898684 0.359926
+vt 0.777761 0.481894
+vt 0.371043 0.075847
+vt 0.442853 0.074922
+vt 0.919564 0.380805
+vt 0.748545 0.488099
+vt 0.370694 0.989321
+vt 0.370712 0.046322
+vt 0.442504 0.990249
+vt 0.442522 0.045396
+vt 0.935969 0.405357
+vt 0.718747 0.488548
+vt 0.371029 0.959797
+vt 0.442838 0.960722
+vt 0.947270 0.432638
+vt 0.689480 0.483245
+vt 0.371269 0.930272
+vt 0.443079 0.931196
+vt 0.953029 0.461597
+vt 0.661830 0.472412
+vt 0.371421 0.900745
+vt 0.443231 0.901668
+vt 0.953029 0.491125
+vt 0.371492 0.871219
+vt 0.636826 0.456466
+vt 0.443302 0.872141
+vt 0.947269 0.520085
+vt 0.615392 0.436018
+vt 0.371495 0.841694
+vt 0.443305 0.842613
+vt 0.935968 0.547365
+vt 0.598317 0.411848
+vt 0.371443 0.812167
+vt 0.443254 0.813086
+vt 0.919563 0.571916
+vt 0.371352 0.783117
+vt 0.586362 0.385333
+vt 0.443163 0.783546
+vt 0.898673 0.592803
+vt 0.370709 0.754396
+vt 0.579568 0.357437
+vt 0.443060 0.754024
+vt 0.874125 0.609201
+vt 0.571207 0.326153
+vt 0.177644 0.253134
+vt 0.006929 0.253167
+vt 0.319029 0.223358
+vt 0.573209 0.178195
+vt 0.007025 0.222313
+vt 0.582798 0.148853
+vt 0.319126 0.192504
+vt 0.007122 0.191458
+vt 0.599888 0.122511
+vt 0.319224 0.161649
+vt 0.007220 0.160604
+vt 0.621946 0.100560
+vt 0.319322 0.130795
+vt 0.007319 0.129750
+vt 0.647824 0.083609
+vt 0.319423 0.099941
+vt 0.007419 0.098896
+vt 0.676401 0.072253
+vt 0.319524 0.069087
+vt 0.007520 0.068042
+vt 0.706468 0.066881
+vt 0.319626 0.038233
+vt 0.007623 0.037188
+vt 0.319730 0.007378
+vt 0.736753 0.067631
+vt 0.319730 0.992485
+vt 0.007727 0.006334
+vt 0.007728 0.993666
+vt 0.765981 0.074365
+vt 0.319616 0.961631
+vt 0.007613 0.962812
+vt 0.792953 0.086670
+vt 0.319502 0.930776
+vt 0.007499 0.931958
+vt 0.816609 0.103895
+vt 0.319390 0.899923
+vt 0.007387 0.901104
+vt 0.836176 0.125250
+vt 0.319278 0.869069
+vt 0.007276 0.870250
+vt 0.850517 0.149568
+vt 0.319168 0.838215
+vt 0.007165 0.839396
+vt 0.886223 0.285996
+vt 0.177771 0.808498
+vt 0.007055 0.808541
+vt 0.177512 0.777645
+vt 0.892437 0.317355
+vt 0.006941 0.777687
+vt 0.176852 0.746796
+vt 0.892437 0.349226
+vt 0.006827 0.746834
+vt 0.886228 0.380492
+vt 0.176742 0.715941
+vt 0.006716 0.715979
+vt 0.874138 0.409852
+vt 0.176640 0.685087
+vt 0.006614 0.685125
+vt 0.856681 0.436209
+vt 0.176549 0.654233
+vt 0.006524 0.654271
+vt 0.834553 0.458608
+vt 0.176474 0.623379
+vt 0.006448 0.623417
+vt 0.808610 0.476251
+vt 0.176415 0.592525
+vt 0.006390 0.592562
+vt 0.779839 0.488518
+vt 0.176377 0.561671
+vt 0.006351 0.561708
+vt 0.749318 0.494983
+vt 0.176359 0.530816
+vt 0.006334 0.530854
+vt 0.176362 0.499962
+vt 0.718195 0.495436
+vt 0.006337 0.500000
+vt 0.687628 0.489885
+vt 0.176387 0.469108
+vt 0.006361 0.469146
+vt 0.658752 0.478560
+vt 0.176430 0.438254
+vt 0.006405 0.438291
+vt 0.632640 0.461897
+vt 0.176492 0.407399
+vt 0.006467 0.407437
+vt 0.610260 0.440534
+vt 0.176568 0.376545
+vt 0.006543 0.376583
+vt 0.592437 0.415287
+vt 0.176656 0.345691
+vt 0.006631 0.345729
+vt 0.176753 0.314837
+vt 0.579817 0.387126
+vt 0.006727 0.314875
+vt 0.177399 0.283987
+vt 0.572791 0.357181
+vt 0.006828 0.284020
+vt 0.578089 0.326242
+vt 0.370472 0.723575
+vt 0.442971 0.724510
+vt 0.231212 0.903091
+vt 0.579868 0.178984
+vt 0.229113 0.694984
+vt 0.442901 0.694984
+vt 0.214996 0.906316
+vt 0.589057 0.151745
+vt 0.229049 0.665457
+vt 0.442837 0.665456
+vt 0.198460 0.906317
+vt 0.605204 0.126737
+vt 0.228992 0.635930
+vt 0.442780 0.635930
+vt 0.182243 0.903091
+vt 0.626230 0.105756
+vt 0.228946 0.606403
+vt 0.442734 0.606402
+vt 0.166967 0.896763
+vt 0.650942 0.089526
+vt 0.228912 0.576877
+vt 0.442700 0.576876
+vt 0.153218 0.887577
+vt 0.678244 0.078632
+vt 0.228890 0.547350
+vt 0.442678 0.547349
+vt 0.141526 0.875885
+vt 0.228883 0.517823
+vt 0.706979 0.073452
+vt 0.442670 0.517822
+vt 0.132340 0.862137
+vt 0.228889 0.488295
+vt 0.735929 0.074111
+vt 0.442678 0.488295
+vt 0.126012 0.846860
+vt 0.763884 0.080481
+vt 0.228910 0.458769
+vt 0.442698 0.458768
+vt 0.122786 0.830643
+vt 0.789700 0.092175
+vt 0.228944 0.429242
+vt 0.442732 0.429241
+vt 0.122786 0.814107
+vt 0.812355 0.108585
+vt 0.228990 0.399716
+vt 0.442778 0.399714
+vt 0.126012 0.797890
+vt 0.831035 0.128981
+vt 0.229046 0.370189
+vt 0.442833 0.370188
+vt 0.132339 0.782614
+vt 0.844628 0.151667
+vt 0.229109 0.340661
+vt 0.442897 0.340661
+vt 0.141526 0.768865
+vt 0.879631 0.287703
+vt 0.370467 0.312068
+vt 0.442967 0.311134
+vt 0.153218 0.757173
+vt 0.370705 0.281246
+vt 0.885704 0.319257
+vt 0.443056 0.281620
+vt 0.166960 0.747990
+vt 0.371348 0.252525
+vt 0.885423 0.348954
+vt 0.443160 0.252098
+vt 0.182233 0.741662
+vt 0.879533 0.378395
+vt 0.371442 0.223476
+vt 0.443252 0.222558
+vt 0.198458 0.738433
+vt 0.867982 0.406513
+vt 0.371495 0.193949
+vt 0.443305 0.193031
+vt 0.214993 0.738433
+vt 0.851292 0.431762
+vt 0.371495 0.164423
+vt 0.443305 0.163503
+vt 0.231211 0.741659
+vt 0.830124 0.453224
+vt 0.371427 0.134897
+vt 0.443237 0.133976
+vt 0.246487 0.747986
+vt 0.805300 0.470133
+vt 0.371280 0.105372
+vt 0.443090 0.104449
+vt 0.260236 0.757172
+vt 0.777761 0.481894
+vt 0.371043 0.075847
+vt 0.442853 0.074922
+vt 0.271928 0.768865
+vt 0.748545 0.488099
+vt 0.370694 0.989321
+vt 0.370712 0.046322
+vt 0.442504 0.990249
+vt 0.442522 0.045396
+vt 0.281116 0.782613
+vt 0.718747 0.488548
+vt 0.371029 0.959797
+vt 0.442838 0.960722
+vt 0.287444 0.797891
+vt 0.689480 0.483245
+vt 0.371269 0.930272
+vt 0.443079 0.931196
+vt 0.290669 0.814108
+vt 0.661830 0.472412
+vt 0.371421 0.900745
+vt 0.443231 0.901668
+vt 0.290669 0.830644
+vt 0.371492 0.871219
+vt 0.636826 0.456466
+vt 0.443302 0.872141
+vt 0.287443 0.846861
+vt 0.615392 0.436018
+vt 0.371495 0.841694
+vt 0.443305 0.842613
+vt 0.281115 0.862138
+vt 0.598317 0.411848
+vt 0.371443 0.812167
+vt 0.443254 0.813086
+vt 0.271928 0.875886
+vt 0.371352 0.783117
+vt 0.586362 0.385333
+vt 0.443163 0.783546
+vt 0.260230 0.887583
+vt 0.370709 0.754396
+vt 0.579568 0.357437
+vt 0.443060 0.754024
+vt 0.246483 0.896766
+vt 0.571207 0.326153
+vt 0.177644 0.253134
+vt 0.006929 0.253167
+vt 0.319029 0.223358
+vt 0.573209 0.178195
+vt 0.007025 0.222313
+vt 0.582798 0.148853
+vt 0.319126 0.192504
+vt 0.007122 0.191458
+vt 0.599888 0.122511
+vt 0.319224 0.161649
+vt 0.007220 0.160604
+vt 0.621946 0.100560
+vt 0.319322 0.130795
+vt 0.007319 0.129750
+vt 0.647824 0.083609
+vt 0.319423 0.099941
+vt 0.007419 0.098896
+vt 0.676401 0.072253
+vt 0.319524 0.069087
+vt 0.007520 0.068042
+vt 0.706468 0.066881
+vt 0.319626 0.038233
+vt 0.007623 0.037188
+vt 0.319730 0.007378
+vt 0.736753 0.067631
+vt 0.319730 0.992485
+vt 0.007727 0.006334
+vt 0.007728 0.993666
+vt 0.765981 0.074365
+vt 0.319616 0.961631
+vt 0.007613 0.962812
+vt 0.792953 0.086670
+vt 0.319502 0.930776
+vt 0.007499 0.931958
+vt 0.816609 0.103895
+vt 0.319390 0.899923
+vt 0.007387 0.901104
+vt 0.836176 0.125250
+vt 0.319278 0.869069
+vt 0.007276 0.870250
+vt 0.850517 0.149568
+vt 0.319168 0.838215
+vt 0.007165 0.839396
+vt 0.886223 0.285996
+vt 0.177771 0.808498
+vt 0.007055 0.808541
+vt 0.177512 0.777645
+vt 0.892437 0.317355
+vt 0.006941 0.777687
+vt 0.176852 0.746796
+vt 0.892437 0.349226
+vt 0.006827 0.746834
+vt 0.886228 0.380492
+vt 0.176742 0.715941
+vt 0.006716 0.715979
+vt 0.874138 0.409852
+vt 0.176640 0.685087
+vt 0.006614 0.685125
+vt 0.856681 0.436209
+vt 0.176549 0.654233
+vt 0.006524 0.654271
+vt 0.834553 0.458608
+vt 0.176474 0.623379
+vt 0.006448 0.623417
+vt 0.808610 0.476251
+vt 0.176415 0.592525
+vt 0.006390 0.592562
+vt 0.779839 0.488518
+vt 0.176377 0.561671
+vt 0.006351 0.561708
+vt 0.749318 0.494983
+vt 0.176359 0.530816
+vt 0.006334 0.530854
+vt 0.176362 0.499962
+vt 0.718195 0.495436
+vt 0.006337 0.500000
+vt 0.687628 0.489885
+vt 0.176387 0.469108
+vt 0.006361 0.469146
+vt 0.658752 0.478560
+vt 0.176430 0.438254
+vt 0.006405 0.438291
+vt 0.632640 0.461897
+vt 0.176492 0.407399
+vt 0.006467 0.407437
+vt 0.610260 0.440534
+vt 0.176568 0.376545
+vt 0.006543 0.376583
+vt 0.592437 0.415287
+vt 0.176656 0.345691
+vt 0.006631 0.345729
+vt 0.176753 0.314837
+vt 0.579817 0.387126
+vt 0.006727 0.314875
+vt 0.177399 0.283987
+vt 0.572791 0.357181
+vt 0.006828 0.284020
+vt 0.578089 0.326242
+vt 0.370472 0.723575
+vt 0.442971 0.724510
+vt 0.278498 0.614635
+vt 0.579868 0.178984
+vt 0.229113 0.694984
+vt 0.442901 0.694984
+vt 0.249539 0.620395
+vt 0.589057 0.151745
+vt 0.229049 0.665457
+vt 0.442837 0.665456
+vt 0.220011 0.620396
+vt 0.605204 0.126737
+vt 0.228992 0.635930
+vt 0.442780 0.635930
+vt 0.191052 0.614636
+vt 0.626230 0.105756
+vt 0.228946 0.606403
+vt 0.442734 0.606402
+vt 0.163773 0.603336
+vt 0.650942 0.089526
+vt 0.228912 0.576877
+vt 0.442700 0.576876
+vt 0.139222 0.586933
+vt 0.678244 0.078632
+vt 0.228890 0.547350
+vt 0.442678 0.547349
+vt 0.118343 0.566054
+vt 0.228883 0.517823
+vt 0.706979 0.073452
+vt 0.442670 0.517822
+vt 0.101939 0.541503
+vt 0.228889 0.488295
+vt 0.735929 0.074111
+vt 0.442678 0.488295
+vt 0.090640 0.514223
+vt 0.763884 0.080481
+vt 0.228910 0.458769
+vt 0.442698 0.458768
+vt 0.084879 0.485264
+vt 0.789700 0.092175
+vt 0.228944 0.429242
+vt 0.442732 0.429241
+vt 0.084879 0.455737
+vt 0.812355 0.108585
+vt 0.228990 0.399716
+vt 0.442778 0.399714
+vt 0.090639 0.426777
+vt 0.831035 0.128981
+vt 0.229046 0.370189
+vt 0.442833 0.370188
+vt 0.101938 0.399499
+vt 0.844628 0.151667
+vt 0.229109 0.340661
+vt 0.442897 0.340661
+vt 0.118343 0.374948
+vt 0.879631 0.287703
+vt 0.370467 0.312068
+vt 0.442967 0.311134
+vt 0.139222 0.354069
+vt 0.370705 0.281246
+vt 0.885704 0.319257
+vt 0.443056 0.281620
+vt 0.163760 0.337671
+vt 0.371348 0.252525
+vt 0.885423 0.348954
+vt 0.443160 0.252098
+vt 0.191035 0.326370
+vt 0.879533 0.378395
+vt 0.371442 0.223476
+vt 0.443252 0.222558
+vt 0.220007 0.320605
+vt 0.867982 0.406513
+vt 0.371495 0.193949
+vt 0.443305 0.193031
+vt 0.249534 0.320604
+vt 0.851292 0.431762
+vt 0.371495 0.164423
+vt 0.443305 0.163503
+vt 0.278494 0.326364
+vt 0.830124 0.453224
+vt 0.371427 0.134897
+vt 0.443237 0.133976
+vt 0.305774 0.337663
+vt 0.805300 0.470133
+vt 0.371280 0.105372
+vt 0.443090 0.104449
+vt 0.330325 0.354067
+vt 0.777761 0.481894
+vt 0.371043 0.075847
+vt 0.442853 0.074922
+vt 0.351205 0.374946
+vt 0.748545 0.488099
+vt 0.370694 0.989321
+vt 0.370712 0.046322
+vt 0.442504 0.990249
+vt 0.442522 0.045396
+vt 0.367610 0.399497
+vt 0.718747 0.488548
+vt 0.371029 0.959797
+vt 0.442838 0.960722
+vt 0.378910 0.426778
+vt 0.689480 0.483245
+vt 0.371269 0.930272
+vt 0.443079 0.931196
+vt 0.384670 0.455738
+vt 0.661830 0.472412
+vt 0.371421 0.900745
+vt 0.443231 0.901668
+vt 0.384670 0.485266
+vt 0.371492 0.871219
+vt 0.636826 0.456466
+vt 0.443302 0.872141
+vt 0.378909 0.514226
+vt 0.615392 0.436018
+vt 0.371495 0.841694
+vt 0.443305 0.842613
+vt 0.367609 0.541505
+vt 0.598317 0.411848
+vt 0.371443 0.812167
+vt 0.443254 0.813086
+vt 0.351204 0.566056
+vt 0.371352 0.783117
+vt 0.586362 0.385333
+vt 0.443163 0.783546
+vt 0.330314 0.586943
+vt 0.370709 0.754396
+vt 0.579568 0.357437
+vt 0.443060 0.754024
+vt 0.305766 0.603342
+vt 0.571207 0.326153
+vt 0.177644 0.253134
+vt 0.006929 0.253167
+vt 0.319029 0.223358
+vt 0.573209 0.178195
+vt 0.007025 0.222313
+vt 0.582798 0.148853
+vt 0.319126 0.192504
+vt 0.007122 0.191458
+vt 0.599888 0.122511
+vt 0.319224 0.161649
+vt 0.007220 0.160604
+vt 0.621946 0.100560
+vt 0.319322 0.130795
+vt 0.007319 0.129750
+vt 0.647824 0.083609
+vt 0.319423 0.099941
+vt 0.007419 0.098896
+vt 0.676401 0.072253
+vt 0.319524 0.069087
+vt 0.007520 0.068042
+vt 0.706468 0.066881
+vt 0.319626 0.038233
+vt 0.007623 0.037188
+vt 0.319730 0.007378
+vt 0.736753 0.067631
+vt 0.319730 0.992485
+vt 0.007727 0.006334
+vt 0.007728 0.993666
+vt 0.765981 0.074365
+vt 0.319616 0.961631
+vt 0.007613 0.962812
+vt 0.792953 0.086670
+vt 0.319502 0.930776
+vt 0.007499 0.931958
+vt 0.816609 0.103895
+vt 0.319390 0.899923
+vt 0.007387 0.901104
+vt 0.836176 0.125250
+vt 0.319278 0.869069
+vt 0.007276 0.870250
+vt 0.850517 0.149568
+vt 0.319168 0.838215
+vt 0.007165 0.839396
+vt 0.886223 0.285996
+vt 0.177771 0.808498
+vt 0.007055 0.808541
+vt 0.177512 0.777645
+vt 0.892437 0.317355
+vt 0.006941 0.777687
+vt 0.176852 0.746796
+vt 0.892437 0.349226
+vt 0.006827 0.746834
+vt 0.886228 0.380492
+vt 0.176742 0.715941
+vt 0.006716 0.715979
+vt 0.874138 0.409852
+vt 0.176640 0.685087
+vt 0.006614 0.685125
+vt 0.856681 0.436209
+vt 0.176549 0.654233
+vt 0.006524 0.654271
+vt 0.834553 0.458608
+vt 0.176474 0.623379
+vt 0.006448 0.623417
+vt 0.808610 0.476251
+vt 0.176415 0.592525
+vt 0.006390 0.592562
+vt 0.779839 0.488518
+vt 0.176377 0.561671
+vt 0.006351 0.561708
+vt 0.749318 0.494983
+vt 0.176359 0.530816
+vt 0.006334 0.530854
+vt 0.176362 0.499962
+vt 0.718195 0.495436
+vt 0.006337 0.500000
+vt 0.687628 0.489885
+vt 0.176387 0.469108
+vt 0.006361 0.469146
+vt 0.658752 0.478560
+vt 0.176430 0.438254
+vt 0.006405 0.438291
+vt 0.632640 0.461897
+vt 0.176492 0.407399
+vt 0.006467 0.407437
+vt 0.610260 0.440534
+vt 0.176568 0.376545
+vt 0.006543 0.376583
+vt 0.592437 0.415287
+vt 0.176656 0.345691
+vt 0.006631 0.345729
+vt 0.176753 0.314837
+vt 0.579817 0.387126
+vt 0.006727 0.314875
+vt 0.177399 0.283987
+vt 0.572791 0.357181
+vt 0.006828 0.284020
+vt 0.006828 0.284020
+vt 0.572791 0.357181
+vt 0.177399 0.283987
+vt 0.006727 0.314875
+vt 0.176753 0.314837
+vt 0.579817 0.387126
+vt 0.006631 0.345729
+vt 0.176656 0.345691
+vt 0.592437 0.415287
+vt 0.006543 0.376583
+vt 0.610260 0.440534
+vt 0.176568 0.376545
+vt 0.006467 0.407437
+vt 0.632640 0.461897
+vt 0.176492 0.407399
+vt 0.006405 0.438291
+vt 0.176430 0.438254
+vt 0.658752 0.478560
+vt 0.006361 0.469146
+vt 0.176387 0.469108
+vt 0.687628 0.489885
+vt 0.006337 0.500000
+vt 0.718195 0.495436
+vt 0.176362 0.499962
+vt 0.006334 0.530854
+vt 0.749318 0.494983
+vt 0.176359 0.530816
+vt 0.006351 0.561708
+vt 0.779839 0.488518
+vt 0.176377 0.561671
+vt 0.006390 0.592562
+vt 0.808610 0.476251
+vt 0.176415 0.592525
+vt 0.006448 0.623417
+vt 0.176474 0.623379
+vt 0.834553 0.458608
+vt 0.006524 0.654271
+vt 0.176549 0.654233
+vt 0.856681 0.436209
+vt 0.006614 0.685125
+vt 0.176640 0.685087
+vt 0.874138 0.409852
+vt 0.006716 0.715979
+vt 0.886228 0.380492
+vt 0.176742 0.715941
+vt 0.006827 0.746834
+vt 0.892437 0.349226
+vt 0.176852 0.746796
+vt 0.006941 0.777687
+vt 0.892437 0.317355
+vt 0.177512 0.777645
+vt 0.007055 0.808541
+vt 0.886223 0.285996
+vt 0.177771 0.808498
+vt 0.007165 0.839396
+vt 0.850517 0.149568
+vt 0.319168 0.838215
+vt 0.007276 0.870250
+vt 0.836176 0.125250
+vt 0.319278 0.869069
+vt 0.007387 0.901104
+vt 0.816609 0.103895
+vt 0.319390 0.899923
+vt 0.007499 0.931958
+vt 0.319502 0.930776
+vt 0.792953 0.086670
+vt 0.007613 0.962812
+vt 0.319616 0.961631
+vt 0.765981 0.074365
+vt 0.007728 0.993666
+vt 0.007727 0.006334
+vt 0.319730 0.007378
+vt 0.736753 0.067631
+vt 0.319730 0.992485
+vt 0.007623 0.037188
+vt 0.319626 0.038233
+vt 0.706468 0.066881
+vt 0.007520 0.068042
+vt 0.676401 0.072253
+vt 0.319524 0.069087
+vt 0.007419 0.098896
+vt 0.319423 0.099941
+vt 0.647824 0.083609
+vt 0.007319 0.129750
+vt 0.319322 0.130795
+vt 0.621946 0.100560
+vt 0.007220 0.160604
+vt 0.599888 0.122511
+vt 0.319224 0.161649
+vt 0.007122 0.191458
+vt 0.319126 0.192504
+vt 0.582798 0.148853
+vt 0.007025 0.222313
+vt 0.319029 0.223358
+vt 0.573209 0.178195
+vt 0.006929 0.253167
+vt 0.177644 0.253134
+vt 0.571207 0.326153
+vt 0.443060 0.754024
+vt 0.208826 0.204735
+vt 0.579568 0.357437
+vt 0.370709 0.754396
+vt 0.443163 0.783546
+vt 0.222573 0.195552
+vt 0.586362 0.385333
+vt 0.371352 0.783117
+vt 0.443254 0.813086
+vt 0.234272 0.183855
+vt 0.598317 0.411848
+vt 0.371443 0.812167
+vt 0.443305 0.842613
+vt 0.243459 0.170107
+vt 0.615392 0.436018
+vt 0.371495 0.841694
+vt 0.443302 0.872141
+vt 0.249787 0.154830
+vt 0.636826 0.456466
+vt 0.371492 0.871219
+vt 0.443231 0.901668
+vt 0.253013 0.138613
+vt 0.371421 0.900745
+vt 0.661830 0.472412
+vt 0.443079 0.931196
+vt 0.253013 0.122077
+vt 0.371269 0.930272
+vt 0.689480 0.483245
+vt 0.442838 0.960722
+vt 0.249787 0.105860
+vt 0.718747 0.488548
+vt 0.371029 0.959797
+vt 0.442504 0.990249
+vt 0.442522 0.045396
+vt 0.243459 0.090582
+vt 0.370694 0.989321
+vt 0.748545 0.488099
+vt 0.370712 0.046322
+vt 0.442853 0.074922
+vt 0.234272 0.076833
+vt 0.777761 0.481894
+vt 0.371043 0.075847
+vt 0.443090 0.104449
+vt 0.222579 0.065141
+vt 0.805300 0.470133
+vt 0.371280 0.105372
+vt 0.443237 0.133976
+vt 0.208831 0.055955
+vt 0.371427 0.134897
+vt 0.830124 0.453224
+vt 0.443305 0.163503
+vt 0.193554 0.049628
+vt 0.371495 0.164423
+vt 0.851292 0.431762
+vt 0.443305 0.193031
+vt 0.177337 0.046402
+vt 0.867982 0.406513
+vt 0.371495 0.193949
+vt 0.443252 0.222558
+vt 0.160801 0.046402
+vt 0.879533 0.378395
+vt 0.371442 0.223476
+vt 0.443160 0.252098
+vt 0.144577 0.049631
+vt 0.371348 0.252525
+vt 0.885423 0.348954
+vt 0.443056 0.281620
+vt 0.129303 0.055959
+vt 0.370705 0.281246
+vt 0.885704 0.319257
+vt 0.442967 0.311134
+vt 0.115562 0.065142
+vt 0.370467 0.312068
+vt 0.879631 0.287703
+vt 0.442897 0.340661
+vt 0.103870 0.076834
+vt 0.229109 0.340661
+vt 0.844628 0.151667
+vt 0.442833 0.370188
+vt 0.094683 0.090583
+vt 0.229046 0.370189
+vt 0.831035 0.128981
+vt 0.442778 0.399714
+vt 0.088355 0.105859
+vt 0.812355 0.108585
+vt 0.228990 0.399716
+vt 0.442732 0.429241
+vt 0.085130 0.122076
+vt 0.228944 0.429242
+vt 0.789700 0.092175
+vt 0.442698 0.458768
+vt 0.085130 0.138611
+vt 0.763884 0.080481
+vt 0.228910 0.458769
+vt 0.442678 0.488295
+vt 0.088356 0.154828
+vt 0.228889 0.488295
+vt 0.735929 0.074111
+vt 0.442670 0.517822
+vt 0.094683 0.170106
+vt 0.706979 0.073452
+vt 0.228883 0.517823
+vt 0.442678 0.547349
+vt 0.103870 0.183854
+vt 0.678244 0.078632
+vt 0.228890 0.547350
+vt 0.442700 0.576876
+vt 0.115562 0.195546
+vt 0.228912 0.576877
+vt 0.650942 0.089526
+vt 0.442734 0.606402
+vt 0.129310 0.204732
+vt 0.228946 0.606403
+vt 0.626230 0.105756
+vt 0.442780 0.635930
+vt 0.144587 0.211060
+vt 0.605204 0.126737
+vt 0.228992 0.635930
+vt 0.442837 0.665456
+vt 0.160804 0.214285
+vt 0.589057 0.151745
+vt 0.229049 0.665457
+vt 0.442901 0.694984
+vt 0.177339 0.214285
+vt 0.229113 0.694984
+vt 0.579868 0.178984
+vt 0.442971 0.724510
+vt 0.193556 0.211059
+vt 0.578089 0.326242
+vt 0.370472 0.723575
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.846034 0.501950
+vt 0.307543 0.501950
+vt 0.769051 0.501950
+vt 0.307542 0.501950
+vt 0.000000 0.501950
+vt 0.692068 0.501950
+vt 0.615085 0.501950
+vt 0.846034 0.996502
+vt 0.384526 0.501950
+vt 0.538102 0.501950
+vt 0.769051 0.501950
+vt 0.923017 0.996502
+vt 0.230559 0.501950
+vt 0.076983 0.501950
+vt 0.692068 0.501950
+vt 0.692068 0.192341
+vt 0.923017 0.650215
+vt 0.076983 0.192341
+vt 0.692068 0.000000
+vt 0.076983 0.000000
+vt 0.846034 0.429417
+vt 0.769051 0.000000
+vt 0.538102 0.000000
+vt 0.923017 0.429417
+vt 0.769051 0.192341
+vt 0.538102 0.192341
+vt 1.000000 0.650215
+vt 0.615085 0.192341
+vt 0.615085 0.000000
+vt 0.923017 0.343447
+vt 0.000000 0.000000
+vt 0.692068 0.000000
+vt 0.846034 0.343447
+vt 0.000000 0.192341
+vt 0.692068 0.192341
+vt 0.923017 0.686893
+vt 0.923017 0.478709
+vt 0.230559 0.192341
+vt 0.230559 0.000000
+vt 0.846034 0.600923
+vt 0.769051 0.000000
+vt 0.307542 0.000000
+vt 0.846034 0.686893
+vt 0.769051 0.192341
+vt 0.307542 0.192341
+vt 0.846034 0.192341
+vt 0.307542 0.192341
+vt 0.846034 0.000000
+vt 0.307542 0.000000
+vt 0.923017 0.686893
+vt 0.384525 0.000000
+vt 0.923017 0.600923
+vt 0.846034 0.686893
+vt 0.384525 0.192341
+vt 1.000000 0.478709
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.846034 1.000000
+vt 0.307543 1.000000
+vt 0.769051 1.000000
+vt 0.307542 1.000000
+vt 0.000000 1.000000
+vt 0.769051 1.000000
+vt 0.615085 1.000000
+vt 0.692068 1.000000
+vt 0.923017 0.307203
+vt 0.384526 1.000000
+vt 0.538102 1.000000
+vt 0.615085 0.501950
+vt 1.000000 0.307203
+vt 0.230559 1.000000
+vt 0.076983 1.000000
+vt 0.692068 0.501950
+vt 0.692068 0.809153
+vt 0.923017 0.478709
+vt 0.076983 0.692797
+vt 0.692068 1.000000
+vt 0.076983 0.501950
+vt 0.846034 0.085970
+vt 0.615085 1.000000
+vt 0.538102 0.501950
+vt 0.923017 0.085970
+vt 0.615085 0.809153
+vt 0.538102 0.692796
+vt 1.000000 0.478709
+vt 0.615085 0.692796
+vt 0.692068 0.692796
+vt 0.615085 0.501950
+vt 0.692068 0.501950
+vt 0.923017 0.000000
+vt 0.000000 0.501950
+vt 0.769051 0.501950
+vt 0.846034 0.000000
+vt 0.000000 0.692797
+vt 0.769051 0.692796
+vt 1.000000 0.000000
+vt 0.923017 0.307203
+vt 0.230559 0.692797
+vt 0.230559 0.501950
+vt 0.846034 0.257476
+vt 0.769051 0.501950
+vt 0.307542 0.501950
+vt 0.846034 0.343447
+vt 0.769051 0.692796
+vt 0.307542 0.692796
+vt 0.846034 0.692796
+vt 0.307542 0.692797
+vt 0.846034 0.501950
+vt 0.307542 0.501950
+vt 0.923017 0.343447
+vt 0.384525 0.501950
+vt 0.923017 0.257476
+vt 0.923017 0.000000
+vt 0.384525 0.692797
+vt 1.000000 0.307203
+vt 0.875462 0.748915
+vt 0.958487 0.000000
+vt 1.000000 1.000000
+vt 0.437731 0.748915
+vt 1.000000 0.500000
+vt 0.875462 0.000000
+vt 0.875462 0.000000
+vt 0.958487 0.748915
+vt 0.958487 0.500000
+vt 0.437731 0.000000
+vt 0.958487 0.000000
+vt 0.875462 0.748915
+vt 0.437731 0.748915
+vt 0.916975 0.748915
+vt 1.000000 0.500000
+vt 0.437731 0.000000
+vt 0.916975 0.000000
+vt 0.958487 1.000000
+vt 0.000000 0.000000
+vt 0.958487 0.500000
+vt 0.916974 0.000000
+vt 0.000000 0.748915
+vt 1.000000 0.000000
+vt 0.916975 0.748915
+s 0
+g KsHauptsignal_Plane.006_Mast
+usemtl Mast
+f 13/29/1 12/27/1 11/24/1 14/31/1
+f 9/18/2 16/37/2 15/34/2 10/21/2
+f 16/38/3 13/30/3 14/32/3 15/35/3
+f 12/28/4 9/19/4 10/20/4 11/23/4
+f 21/49/4 20/47/4 19/44/4 22/51/4
+f 24/58/1 21/50/1 22/52/1 23/54/1
+f 20/48/2 17/41/2 18/42/2 19/45/2
+f 24/58/1 23/54/1 11/24/1 12/27/1
+f 17/39/5 24/56/5 12/26/5 9/17/5
+f 18/42/2 17/41/2 9/18/2 10/21/2
+f 23/55/6 18/43/6 10/22/6 11/25/6
+f 19/46/6 18/43/6 23/55/6 22/53/6
+f 11/25/6 10/22/6 15/36/6 14/33/6
+f 3/5/1 2/3/1 12/27/1 13/29/1
+f 1/1/2 4/7/2 16/37/2 9/18/2
+f 4/8/3 3/6/3 13/30/3 16/38/3
+f 2/4/4 1/2/4 9/19/4 12/28/4
+f 7/13/4 6/11/4 20/47/4 21/49/4
+f 5/9/3 8/15/3 24/57/3 17/40/3
+f 8/16/1 7/14/1 21/50/1 24/58/1
+f 6/12/2 5/10/2 17/41/2 20/48/2
+f 544/974/6 541/973/6 537/971/6 538/972/6
+f 548/981/1 550/984/1 556/1000/1 557/1002/1
+f 557/1002/1 556/1000/1 555/997/1 558/1003/1
+f 552/988/2 547/979/2 560/1008/2 553/992/2
+f 553/992/2 560/1008/2 559/1005/2 554/994/2
+f 547/980/3 548/981/3 557/1002/3 560/1009/3
+f 560/1009/3 557/1002/3 558/1003/3 559/1006/3
+f 550/985/4 552/989/4 553/990/4 556/999/4
+f 556/999/4 553/990/4 554/993/4 555/996/4
+f 545/975/4 546/977/4 564/1018/4 565/1020/4
+f 565/1020/4 564/1018/4 563/1015/4 566/1022/4
+f 551/986/3 549/982/3 568/1027/3 561/1010/3
+f 549/983/1 545/976/1 565/1021/1 568/1028/1
+f 568/1028/1 565/1021/1 566/1023/1 567/1025/1
+f 546/978/2 551/987/2 561/1012/2 564/1019/2
+f 564/1019/2 561/1012/2 562/1013/2 563/1016/2
+f 568/1028/1 567/1025/1 555/997/1 556/1000/1
+f 561/1011/5 568/1029/5 556/1001/5 553/991/5
+f 562/1013/2 561/1012/2 553/992/2 554/994/2
+f 567/1026/6 562/1014/6 554/995/6 555/998/6
+f 563/1017/6 562/1014/6 567/1026/6 566/1024/6
+f 555/998/6 554/995/6 559/1007/6 558/1004/6
+f 576/1033/6 573/1032/6 569/1030/6 570/1031/6
+f 580/1040/1 582/1044/1 588/1060/1 589/1062/1
+f 589/1062/1 588/1060/1 587/1057/1 590/1064/1
+f 584/1048/2 579/1038/2 592/1070/2 585/1052/2
+f 585/1052/2 592/1070/2 591/1067/2 586/1054/2
+f 579/1039/3 580/1041/3 589/1063/3 592/1071/3
+f 592/1071/3 589/1063/3 590/1065/3 591/1068/3
+f 582/1045/4 584/1049/4 585/1050/4 588/1059/4
+f 588/1059/4 585/1050/4 586/1053/4 587/1056/4
+f 577/1034/4 578/1036/4 596/1080/4 597/1082/4
+f 597/1082/4 596/1080/4 595/1077/4 598/1084/4
+f 583/1046/3 581/1042/3 600/1089/3 593/1072/3
+f 581/1043/1 577/1035/1 597/1083/1 600/1090/1
+f 600/1090/1 597/1083/1 598/1085/1 599/1087/1
+f 578/1037/2 583/1047/2 593/1074/2 596/1081/2
+f 596/1081/2 593/1074/2 594/1075/2 595/1078/2
+f 600/1090/1 599/1087/1 587/1057/1 588/1060/1
+f 593/1073/5 600/1091/5 588/1061/5 585/1051/5
+f 594/1075/2 593/1074/2 585/1052/2 586/1054/2
+f 599/1088/6 594/1076/6 586/1055/6 587/1058/6
+f 595/1079/6 594/1076/6 599/1088/6 598/1086/6
+f 587/1058/6 586/1055/6 591/1069/6 590/1066/6
+g KsHauptsignal_Plane.006_Schirm
+usemtl Schirm
+f 601/1092/3 602/1095/3 604/1101/3 603/1098/3
+f 606/1107/4 605/1104/4 608/1113/4 607/1110/4
+f 603/1099/2 605/1105/2 606/1108/2 601/1093/2
+f 604/1102/6 608/1114/6 605/1106/6 603/1100/6
+f 601/1094/5 606/1109/5 607/1111/5 602/1096/5
+f 602/1097/1 607/1112/1 608/1115/1 604/1103/1
+g KsHauptsignal_Plane.006_Lampenrahmen
+usemtl Lampenrahmen
+f 40/89/7 39/87/7 41/91/7
+f 26/61/8 88/187/8 25/60/8
+f 54/117/9 53/116/9 56/121/9
+f 58/125/10 56/121/10 57/123/10
+f 86/183/11 85/181/11 88/187/11
+f 90/191/12 89/190/12 152/286/12
+f 118/235/13 120/238/13 117/234/13
+f 122/241/14 121/239/14 120/238/14
+f 150/283/15 152/286/15 149/281/15
+f 79/169/16 81/174/16 82/175/16
+f 81/173/17 145/275/17 147/278/17
+f 111/224/18 113/227/18 49/107/18
+f 103/210/19 105/214/19 41/92/19
+f 99/204/20 35/79/20 33/75/21
+f 105/214/22 107/218/22 43/95/22
+f 147/278/23 149/282/23 85/182/23
+f 97/201/24 33/75/24 31/71/24
+f 103/210/25 39/88/25 37/83/25
+f 109/221/26 111/224/26 47/103/26
+f 35/79/27 99/204/27 101/207/27
+f 107/218/28 109/221/28 45/99/28
+f 95/198/29 31/71/29 29/67/29
+f 93/195/30 29/67/30 27/63/30
+f 113/227/31 115/230/31 51/111/31
+f 151/285/32 89/189/32 25/59/32
+f 149/282/33 151/285/33 87/186/33
+f 121/240/34 57/124/34 55/120/34
+f 115/230/35 117/233/35 53/115/35
+f 79/170/36 143/272/36 145/275/36
+f 125/245/37 127/248/37 63/135/37
+f 77/165/38 141/269/38 143/272/38
+f 75/161/39 139/266/39 141/269/39
+f 73/157/40 137/264/40 139/266/40
+f 133/257/41 135/260/41 71/151/41
+f 131/254/42 133/257/42 69/147/42
+f 129/251/43 131/254/43 67/143/43
+f 127/248/44 129/251/44 65/139/44
+f 91/193/45 27/63/45 25/59/45
+f 123/242/46 125/245/46 61/131/46
+f 123/242/47 59/127/47 57/124/47
+f 88/187/48 87/185/48 25/60/48
+f 97/202/49 95/199/49 96/200/49
+f 112/226/50 111/225/50 109/222/50
+f 48/105/51 47/104/51 49/108/51
+f 110/223/52 109/222/52 107/219/52
+f 140/268/53 139/267/53 137/263/53
+f 42/93/54 41/91/54 43/96/54
+f 106/216/55 105/213/55 103/211/55
+f 37/84/56 39/87/56 40/89/56
+f 108/220/57 107/219/57 105/215/57
+f 33/76/58 35/80/58 36/81/58
+f 31/72/59 33/76/59 34/77/59
+f 50/109/60 49/108/60 51/112/60
+f 27/64/61 29/68/61 30/69/61
+f 25/60/62 27/64/62 28/65/62
+f 53/116/63 55/119/63 56/121/63
+f 56/121/64 55/119/64 57/123/64
+f 81/174/65 83/178/65 84/179/65
+f 71/151/66 135/260/66 137/264/66
+f 83/178/67 85/181/67 86/183/67
+f 119/237/68 55/120/68 53/115/68
+f 77/166/69 79/169/69 80/171/69
+f 75/162/70 77/166/70 78/167/70
+f 114/229/71 113/228/71 111/225/71
+f 142/271/72 141/270/72 139/267/72
+f 68/145/73 67/144/73 69/148/73
+f 66/141/74 65/140/74 67/144/74
+f 64/137/75 63/136/75 65/140/75
+f 62/133/76 61/132/76 63/136/76
+f 60/129/77 59/128/77 61/132/77
+f 58/125/78 57/123/78 59/128/78
+f 85/181/79 87/185/79 88/187/79
+f 89/190/80 151/284/80 152/286/80
+f 146/277/81 145/276/81 143/273/81
+f 144/274/82 143/273/82 141/270/82
+f 116/232/83 115/231/83 113/228/83
+f 71/152/84 73/158/84 74/159/84
+f 29/68/85 31/72/85 32/73/85
+f 44/97/86 43/96/86 45/100/86
+f 138/265/87 137/263/87 135/261/87
+f 103/211/88 101/208/88 102/209/88
+f 101/208/89 99/205/89 100/206/89
+f 99/205/90 97/202/90 98/203/90
+f 52/113/91 51/112/91 53/116/91
+f 95/199/92 93/196/92 94/197/92
+f 93/196/93 91/192/93 92/194/93
+f 92/194/94 91/192/94 89/190/94
+f 120/238/95 119/236/95 117/234/95
+f 121/239/14 119/236/14 120/238/14
+f 73/158/96 75/162/96 76/163/96
+f 150/283/97 149/281/97 147/279/97
+f 148/280/98 147/279/98 145/276/98
+f 116/232/99 118/235/99 117/234/99
+f 70/149/100 69/148/100 71/153/100
+f 46/101/101 45/100/101 47/104/101
+f 35/80/102 37/84/102 38/85/102
+f 135/261/103 133/258/103 134/259/103
+f 133/258/104 131/255/104 132/256/104
+f 131/255/105 129/252/105 130/253/105
+f 129/252/106 127/249/106 128/250/106
+f 127/249/107 125/246/107 126/247/107
+f 125/246/108 123/243/108 124/244/108
+f 123/243/109 121/239/109 122/241/109
+f 152/286/110 151/284/110 149/281/110
+f 209/401/111 273/503/111 275/506/111
+f 276/508/112 275/507/112 273/504/112
+f 211/406/113 213/409/113 214/411/113
+f 249/467/114 247/464/114 248/466/114
+f 244/460/115 243/459/115 241/456/115
+f 272/502/116 271/501/116 269/498/116
+f 180/341/117 179/340/117 181/344/117
+f 229/436/118 227/433/118 228/434/118
+f 225/430/119 223/427/119 224/428/119
+f 255/476/120 257/479/120 193/367/120
+f 234/444/121 233/441/121 231/439/121
+f 157/296/122 159/300/122 160/301/122
+f 182/345/110 181/344/110 184/349/110
+f 165/312/123 167/315/123 168/317/123
+f 214/411/124 213/409/124 216/415/124
+f 221/424/125 219/420/125 220/422/125
+f 216/415/126 215/413/126 153/288/126
+f 227/432/127 163/307/128 161/303/128
+f 235/446/129 237/449/129 173/327/129
+f 201/385/130 265/492/130 267/494/130
+f 261/485/131 263/488/131 199/379/131
+f 184/349/132 183/347/132 185/351/132
+f 159/300/133 161/304/133 162/305/133
+f 168/317/134 167/315/134 169/319/134
+f 275/506/135 277/510/135 213/410/135
+f 266/493/136 265/491/136 263/489/136
+f 249/468/137 185/352/137 183/348/137
+f 231/438/138 233/442/138 169/320/138
+f 190/361/139 189/360/139 191/364/139
+f 217/418/140 279/512/140 280/514/140
+f 255/477/141 253/474/141 254/475/141
+f 186/353/142 184/349/142 185/351/142
+f 203/390/143 205/394/143 206/395/143
+f 239/452/144 241/455/144 177/335/144
+f 225/429/145 161/303/145 159/299/145
+f 172/325/146 171/324/146 173/328/146
+f 174/329/147 173/328/147 175/332/147
+f 199/379/148 263/488/148 265/492/149
+f 253/473/150 255/476/150 191/363/150
+f 163/308/151 165/312/151 166/313/151
+f 263/489/152 261/486/152 262/487/152
+f 241/455/153 243/458/153 179/339/153
+f 201/386/154 203/390/154 204/391/154
+f 198/377/155 197/376/155 199/381/155
+f 209/402/156 211/406/156 212/407/156
+f 188/357/157 187/356/157 189/360/157
+f 233/442/158 235/446/158 171/323/158
+f 196/373/159 195/372/159 197/376/159
+f 257/479/160 259/482/161 195/371/161
+f 278/511/162 280/514/162 277/509/162
+f 236/448/163 235/447/163 233/443/163
+f 250/469/114 249/467/114 248/466/114
+f 253/474/164 251/471/164 252/472/164
+f 205/393/165 269/497/165 271/500/165
+f 213/409/166 215/413/166 216/415/166
+f 154/289/167 216/415/167 153/288/167
+f 218/419/140 217/418/140 280/514/140
+f 279/513/168 217/417/168 153/287/168
+f 176/333/169 175/332/169 177/336/169
+f 181/344/170 183/347/170 184/349/170
+f 161/304/171 163/308/171 164/309/171
+f 280/514/172 279/512/172 277/509/172
+f 257/480/173 255/477/173 256/478/173
+f 274/505/174 273/504/174 271/501/174
+f 155/292/175 157/296/175 158/297/175
+f 231/439/176 229/436/176 230/437/176
+f 205/394/177 207/397/177 208/399/177
+f 220/422/178 219/420/178 217/418/178
+f 203/389/179 267/494/179 269/497/179
+f 192/365/180 191/364/180 193/368/180
+f 244/460/181 246/463/181 245/462/181
+f 170/321/182 169/319/182 171/324/182
+f 259/482/183 261/485/184 197/375/183
+f 219/421/185 155/291/185 153/287/185
+f 251/471/186 249/467/186 250/469/186
+f 207/398/187 271/500/187 273/503/187
+f 277/510/188 279/513/188 215/414/188
+f 240/454/189 239/453/189 237/450/189
+f 261/486/190 259/483/190 260/484/190
+f 223/427/191 221/424/191 222/425/191
+f 278/511/192 277/509/192 275/507/192
+f 248/466/193 247/464/193 245/462/193
+f 237/449/194 239/452/194 175/331/195
+f 246/463/193 248/466/193 245/462/193
+f 221/423/196 157/295/196 155/291/196
+f 242/457/197 241/456/197 239/453/197
+f 227/433/198 225/430/198 226/431/198
+f 259/483/199 257/480/199 258/481/199
+f 251/470/200 187/355/201 185/352/202
+f 178/337/203 177/336/203 179/340/203
+f 199/380/204 201/386/204 202/387/204
+f 243/458/205 245/461/205 181/343/205
+f 223/426/206 159/299/206 157/295/206
+f 163/307/207 227/432/207 229/435/207
+f 194/369/208 193/368/208 195/372/208
+f 207/397/209 209/402/209 210/403/209
+f 251/470/210 253/473/211 189/359/210
+f 270/499/212 269/498/212 267/495/212
+f 186/353/213 185/351/213 187/356/213
+f 247/465/214 183/348/214 181/343/215
+f 153/288/216 155/292/216 156/293/216
+f 268/496/217 267/495/217 265/491/217
+f 238/451/218 237/450/218 235/447/218
+f 231/438/219 167/316/219 165/311/219
+f 42/93/220 40/89/220 41/91/220
+f 80/171/16 79/169/16 82/175/16
+f 83/177/221 81/173/221 147/278/221
+f 47/103/222 111/224/222 49/107/222
+f 39/88/223 103/210/223 41/92/223
+f 97/201/224 99/204/224 33/75/224
+f 41/92/225 105/214/225 43/95/225
+f 83/177/226 147/278/226 85/182/226
+f 95/198/227 97/201/227 31/71/227
+f 101/207/228 103/210/228 37/83/228
+f 45/99/229 109/221/229 47/103/229
+f 37/83/230 35/79/230 101/207/230
+f 43/95/231 107/218/231 45/99/231
+f 93/195/232 95/198/232 29/67/232
+f 91/193/233 93/195/233 27/63/233
+f 49/107/234 113/227/234 51/111/234
+f 87/186/235 151/285/235 25/59/235
+f 85/182/236 149/282/236 87/186/237
+f 119/237/238 121/240/238 55/120/239
+f 51/111/240 115/230/240 53/115/240
+f 81/173/241 79/170/241 145/275/241
+f 61/131/242 125/245/242 63/135/242
+f 79/170/243 77/165/243 143/272/243
+f 77/165/244 75/161/244 141/269/244
+f 75/161/245 73/157/245 139/266/245
+f 69/147/246 133/257/246 71/151/246
+f 67/143/247 131/254/247 69/147/247
+f 65/139/248 129/251/248 67/143/248
+f 63/135/249 127/248/249 65/139/249
+f 89/189/250 91/193/250 25/59/250
+f 59/127/251 123/242/251 61/131/251
+f 121/240/252 123/242/252 57/124/252
+f 98/203/49 97/202/49 96/200/49
+f 110/223/50 112/226/50 109/222/50
+f 50/109/51 48/105/51 49/108/51
+f 108/220/253 110/223/253 107/219/253
+f 138/265/254 140/268/254 137/263/254
+f 44/97/255 42/93/255 43/96/255
+f 104/212/256 106/216/256 103/211/256
+f 38/85/56 37/84/56 40/89/56
+f 106/217/257 108/220/257 105/215/257
+f 34/77/258 33/76/258 36/81/258
+f 32/73/259 31/72/259 34/77/259
+f 52/113/260 50/109/260 51/112/260
+f 28/65/261 27/64/261 30/69/261
+f 26/61/262 25/60/262 28/65/262
+f 82/175/65 81/174/65 84/179/65
+f 73/157/263 71/151/263 137/264/263
+f 84/179/264 83/178/264 86/183/264
+f 117/233/265 119/237/265 53/115/265
+f 78/167/69 77/166/69 80/171/69
+f 76/163/266 75/162/266 78/167/266
+f 112/226/267 114/229/267 111/225/267
+f 140/268/72 142/271/72 139/267/72
+f 70/149/73 68/145/73 69/148/73
+f 68/145/74 66/141/74 67/144/74
+f 66/141/75 64/137/75 65/140/75
+f 64/137/76 62/133/76 63/136/76
+f 62/133/77 60/129/77 61/132/77
+f 60/129/268 58/125/268 59/128/268
+f 144/274/269 146/277/269 143/273/269
+f 142/271/270 144/274/270 141/270/270
+f 114/229/83 116/232/83 113/228/83
+f 72/154/271 71/152/271 74/159/271
+f 30/69/85 29/68/85 32/73/85
+f 46/101/86 44/97/86 45/100/86
+f 136/262/87 138/265/87 135/261/87
+f 104/212/88 103/211/88 102/209/88
+f 102/209/272 101/208/272 100/206/272
+f 100/206/273 99/205/273 98/203/273
+f 54/117/274 52/113/274 53/116/274
+f 96/200/92 95/199/92 94/197/92
+f 94/197/93 93/196/93 92/194/93
+f 90/191/275 92/194/275 89/190/275
+f 74/159/96 73/158/96 76/163/96
+f 148/280/276 150/283/276 147/279/276
+f 146/277/277 148/280/277 145/276/277
+f 115/231/278 116/232/278 117/234/278
+f 72/155/100 70/149/100 71/153/100
+f 48/105/101 46/101/101 47/104/101
+f 36/81/102 35/80/102 38/85/102
+f 136/262/103 135/261/103 134/259/103
+f 134/259/104 133/258/104 132/256/104
+f 132/256/105 131/255/105 130/253/105
+f 130/253/106 129/252/106 128/250/106
+f 128/250/107 127/249/107 126/247/107
+f 126/247/108 125/246/108 124/244/108
+f 124/244/109 123/243/109 122/241/109
+f 296/545/7 295/543/7 297/547/7
+f 282/517/8 344/643/8 281/516/8
+f 310/573/9 309/572/9 312/577/9
+f 314/581/10 312/577/10 313/579/10
+f 342/639/11 341/637/11 344/643/11
+f 346/647/12 345/646/12 408/742/12
+f 374/691/13 376/694/13 373/690/13
+f 378/697/14 377/695/14 376/694/14
+f 406/739/15 408/742/15 405/737/15
+f 335/625/16 337/630/16 338/631/16
+f 337/629/17 401/731/17 403/734/17
+f 367/680/279 369/683/279 305/563/279
+f 359/666/19 361/670/19 297/548/19
+f 355/660/280 291/535/280 289/531/280
+f 361/670/281 363/674/281 299/551/281
+f 403/734/282 405/738/282 341/638/282
+f 353/657/283 289/531/283 287/527/283
+f 359/666/284 295/544/284 293/539/284
+f 365/677/285 367/680/285 303/559/285
+f 291/535/286 355/660/286 357/663/286
+f 363/674/28 365/677/28 301/555/28
+f 351/654/287 287/527/287 285/523/287
+f 349/651/288 285/523/288 283/519/288
+f 369/683/31 371/686/31 307/567/31
+f 407/741/289 345/645/289 281/515/289
+f 405/738/33 407/741/33 343/642/33
+f 377/696/290 313/580/290 311/576/290
+f 371/686/291 373/689/291 309/571/291
+f 335/626/292 399/728/292 401/731/292
+f 381/701/37 383/704/37 319/591/37
+f 333/621/38 397/725/38 399/728/38
+f 331/617/293 395/722/293 397/725/293
+f 329/613/40 393/720/40 395/722/40
+f 389/713/41 391/716/41 327/607/41
+f 387/710/42 389/713/42 325/603/42
+f 385/707/294 387/710/294 323/599/294
+f 383/704/295 385/707/295 321/595/295
+f 347/649/45 283/519/45 281/515/45
+f 379/698/296 381/701/296 317/587/296
+f 379/698/297 315/583/297 313/580/297
+f 344/643/48 343/641/48 281/516/48
+f 353/658/49 351/655/49 352/656/49
+f 368/682/50 367/681/50 365/678/50
+f 304/561/51 303/560/51 305/564/51
+f 366/679/52 365/678/52 363/675/52
+f 396/724/53 395/723/53 393/719/53
+f 298/549/54 297/547/54 299/552/54
+f 362/672/55 361/669/55 359/667/55
+f 293/540/56 295/543/56 296/545/56
+f 364/676/57 363/675/57 361/671/57
+f 289/532/58 291/536/58 292/537/58
+f 287/528/59 289/532/59 290/533/59
+f 306/565/60 305/564/60 307/568/60
+f 283/520/61 285/524/61 286/525/61
+f 281/516/62 283/520/62 284/521/62
+f 309/572/298 311/575/298 312/577/298
+f 312/577/64 311/575/64 313/579/64
+f 337/630/65 339/634/65 340/635/65
+f 327/607/66 391/716/66 393/720/66
+f 339/634/67 341/637/67 342/639/67
+f 375/693/299 311/576/299 309/571/299
+f 333/622/69 335/625/69 336/627/69
+f 331/618/70 333/622/70 334/623/70
+f 370/685/71 369/684/71 367/681/71
+f 398/727/72 397/726/72 395/723/72
+f 324/601/73 323/600/73 325/604/73
+f 322/597/74 321/596/74 323/600/74
+f 320/593/75 319/592/75 321/596/75
+f 318/589/76 317/588/76 319/592/76
+f 316/585/77 315/584/77 317/588/77
+f 314/581/78 313/579/78 315/584/78
+f 341/637/79 343/641/79 344/643/79
+f 345/646/80 407/740/80 408/742/80
+f 402/733/81 401/732/81 399/729/81
+f 400/730/82 399/729/82 397/726/82
+f 372/688/83 371/687/83 369/684/83
+f 327/608/84 329/614/84 330/615/84
+f 285/524/85 287/528/85 288/529/85
+f 300/553/86 299/552/86 301/556/86
+f 394/721/87 393/719/87 391/717/87
+f 359/667/88 357/664/88 358/665/88
+f 357/664/89 355/661/89 356/662/89
+f 355/661/90 353/658/90 354/659/90
+f 308/569/300 307/568/300 309/572/300
+f 351/655/92 349/652/92 350/653/92
+f 349/652/93 347/648/93 348/650/93
+f 348/650/301 347/648/301 345/646/301
+f 376/694/95 375/692/95 373/690/95
+f 377/695/14 375/692/14 376/694/14
+f 329/614/96 331/618/96 332/619/96
+f 406/739/97 405/737/97 403/735/97
+f 404/736/98 403/735/98 401/732/98
+f 372/688/99 374/691/99 373/690/99
+f 326/605/100 325/604/100 327/609/100
+f 302/557/101 301/556/101 303/560/101
+f 291/536/102 293/540/102 294/541/102
+f 391/717/103 389/714/103 390/715/103
+f 389/714/104 387/711/104 388/712/104
+f 387/711/302 385/708/302 386/709/302
+f 385/708/303 383/705/303 384/706/303
+f 383/705/107 381/702/107 382/703/107
+f 381/702/108 379/699/108 380/700/108
+f 379/699/109 377/695/109 378/697/109
+f 408/742/110 407/740/110 405/737/110
+f 298/549/220 296/545/220 297/547/220
+f 336/627/16 335/625/16 338/631/16
+f 339/633/221 337/629/221 403/734/221
+f 303/559/304 367/680/304 305/563/304
+f 295/544/223 359/666/223 297/548/223
+f 353/657/224 355/660/224 289/531/224
+f 297/548/305 361/670/305 299/551/305
+f 339/633/306 403/734/306 341/638/306
+f 351/654/307 353/657/307 287/527/307
+f 357/663/308 359/666/308 293/539/308
+f 301/555/229 365/677/229 303/559/229
+f 293/539/309 291/535/309 357/663/309
+f 299/551/310 363/674/310 301/555/310
+f 349/651/311 351/654/311 285/523/311
+f 347/649/312 349/651/312 283/519/312
+f 305/563/313 369/683/313 307/567/313
+f 343/642/314 407/741/314 281/515/314
+f 341/638/315 405/738/315 343/642/315
+f 375/693/316 377/696/316 311/576/316
+f 307/567/317 371/686/317 309/571/317
+f 337/629/318 335/626/318 401/731/318
+f 317/587/319 381/701/319 319/591/319
+f 335/626/243 333/621/320 399/728/321
+f 333/621/322 331/617/322 397/725/322
+f 331/617/323 329/613/323 395/722/323
+f 325/603/246 389/713/246 327/607/246
+f 323/599/324 387/710/324 325/603/324
+f 321/595/325 385/707/325 323/599/325
+f 319/591/249 383/704/249 321/595/249
+f 345/645/250 347/649/250 281/515/250
+f 315/583/326 379/698/326 317/587/326
+f 377/696/327 379/698/328 313/580/328
+f 354/659/49 353/658/49 352/656/49
+f 366/679/50 368/682/50 365/678/50
+f 306/565/51 304/561/51 305/564/51
+f 364/676/253 366/679/253 363/675/253
+f 394/721/254 396/724/254 393/719/254
+f 300/553/255 298/549/255 299/552/255
+f 360/668/256 362/672/256 359/667/256
+f 294/541/56 293/540/56 296/545/56
+f 362/673/257 364/676/257 361/671/257
+f 290/533/258 289/532/258 292/537/258
+f 288/529/259 287/528/259 290/533/259
+f 308/569/260 306/565/260 307/568/260
+f 284/521/261 283/520/261 286/525/261
+f 282/517/262 281/516/262 284/521/262
+f 338/631/65 337/630/65 340/635/65
+f 329/613/263 327/607/263 393/720/263
+f 340/635/264 339/634/264 342/639/264
+f 373/689/329 375/693/329 309/571/329
+f 334/623/69 333/622/69 336/627/69
+f 332/619/266 331/618/266 334/623/266
+f 368/682/267 370/685/267 367/681/267
+f 396/724/72 398/727/72 395/723/72
+f 326/605/73 324/601/73 325/604/73
+f 324/601/74 322/597/74 323/600/74
+f 322/597/75 320/593/75 321/596/75
+f 320/593/76 318/589/76 319/592/76
+f 318/589/77 316/585/77 317/588/77
+f 316/585/268 314/581/268 315/584/268
+f 400/730/269 402/733/269 399/729/269
+f 398/727/270 400/730/270 397/726/270
+f 370/685/83 372/688/83 369/684/83
+f 328/610/271 327/608/271 330/615/271
+f 286/525/85 285/524/85 288/529/85
+f 302/557/86 300/553/86 301/556/86
+f 392/718/87 394/721/87 391/717/87
+f 360/668/88 359/667/88 358/665/88
+f 358/665/272 357/664/272 356/662/272
+f 356/662/273 355/661/273 354/659/273
+f 310/573/274 308/569/274 309/572/274
+f 352/656/92 351/655/92 350/653/92
+f 350/653/93 349/652/93 348/650/93
+f 346/647/275 348/650/275 345/646/275
+f 330/615/96 329/614/96 332/619/96
+f 404/736/276 406/739/276 403/735/276
+f 402/733/277 404/736/277 401/732/277
+f 371/687/278 372/688/278 373/690/278
+f 328/611/100 326/605/100 327/609/100
+f 304/561/101 302/557/101 303/560/101
+f 292/537/102 291/536/102 294/541/102
+f 392/718/103 391/717/103 390/715/103
+f 390/715/104 389/714/104 388/712/104
+f 388/712/105 387/711/105 386/709/105
+f 386/709/106 385/708/106 384/706/106
+f 384/706/107 383/705/107 382/703/107
+f 382/703/108 381/702/108 380/700/108
+f 380/700/109 379/699/109 378/697/109
+f 170/321/330 168/317/330 169/319/330
+f 208/399/209 207/397/209 210/403/209
+f 211/405/331 209/401/332 275/506/331
+f 175/331/333 239/452/333 177/335/333
+f 167/316/334 231/438/335 169/320/335
+f 225/429/336 227/432/336 161/303/336
+f 169/320/337 233/442/337 171/323/337
+f 211/405/338 275/506/338 213/410/338
+f 223/426/339 225/429/339 159/299/339
+f 229/435/340 231/438/340 165/311/341
+f 173/327/342 237/449/342 175/331/342
+f 165/311/343 163/307/344 229/435/344
+f 171/323/345 235/446/345 173/327/345
+f 221/423/346 223/426/346 157/295/346
+f 219/421/347 221/423/348 155/291/347
+f 177/335/349 241/455/349 179/339/350
+f 215/414/351 279/513/351 153/287/351
+f 213/410/352 277/510/352 215/414/353
+f 247/465/354 249/468/354 183/348/354
+f 179/339/355 243/458/355 181/343/355
+f 209/401/356 207/398/356 273/503/356
+f 189/359/357 253/473/358 191/363/357
+f 207/398/359 205/393/360 271/500/360
+f 205/393/361 203/389/362 269/497/363
+f 203/389/364 201/385/364 267/494/364
+f 197/375/365 261/485/365 199/379/365
+f 195/371/366 259/482/366 197/375/366
+f 193/367/367 257/479/368 195/371/368
+f 191/363/369 255/476/369 193/367/370
+f 217/417/371 219/421/371 153/287/371
+f 187/355/372 251/470/372 189/359/372
+f 249/468/373 251/470/373 185/352/373
+f 226/431/374 225/430/374 224/428/374
+f 238/451/375 240/454/375 237/450/375
+f 178/337/169 176/333/169 177/336/169
+f 236/448/376 238/451/376 235/447/376
+f 266/493/217 268/496/217 265/491/217
+f 172/325/182 170/321/182 171/324/182
+f 232/440/121 234/444/121 231/439/121
+f 166/313/377 165/312/377 168/317/377
+f 234/445/163 236/448/163 233/443/163
+f 162/305/378 161/304/378 164/309/378
+f 160/301/379 159/300/379 162/305/379
+f 180/341/380 178/337/380 179/340/380
+f 156/293/381 155/292/381 158/297/381
+f 154/289/382 153/288/382 156/293/382
+f 210/403/383 209/402/383 212/407/383
+f 201/385/384 199/379/384 265/492/384
+f 212/407/385 211/406/385 214/411/385
+f 245/461/386 247/465/386 181/343/386
+f 206/395/177 205/394/177 208/399/177
+f 204/391/387 203/390/387 206/395/387
+f 240/454/197 242/457/197 239/453/197
+f 268/496/212 270/499/212 267/495/212
+f 198/377/159 196/373/159 197/376/159
+f 196/373/388 194/369/388 195/372/388
+f 194/369/180 192/365/180 193/368/180
+f 192/365/389 190/361/389 191/364/389
+f 190/361/157 188/357/157 189/360/157
+f 188/357/390 186/353/390 187/356/390
+f 272/502/174 274/505/174 271/501/174
+f 270/499/116 272/502/116 269/498/116
+f 242/457/115 244/460/115 241/456/115
+f 200/382/204 199/380/204 202/387/204
+f 158/297/391 157/296/391 160/301/391
+f 174/329/146 172/325/146 173/328/146
+f 264/490/392 266/493/392 263/489/392
+f 232/440/393 231/439/393 230/437/393
+f 230/437/394 229/436/394 228/434/394
+f 228/434/198 227/433/198 226/431/198
+f 182/345/97 180/341/97 181/344/97
+f 224/428/191 223/427/191 222/425/191
+f 222/425/395 221/424/395 220/422/395
+f 218/419/396 220/422/396 217/418/396
+f 202/387/154 201/386/154 204/391/154
+f 276/508/192 278/511/192 275/507/192
+f 274/505/112 276/508/112 273/504/112
+f 243/459/397 244/460/397 245/462/397
+f 200/383/155 198/377/155 199/381/155
+f 176/333/398 174/329/398 175/332/398
+f 164/309/399 163/308/399 166/313/399
+f 264/490/400 263/489/400 262/487/400
+f 262/487/401 261/486/401 260/484/401
+f 260/484/199 259/483/199 258/481/199
+f 258/481/173 257/480/173 256/478/173
+f 256/478/402 255/477/402 254/475/402
+f 254/475/403 253/474/403 252/472/403
+f 252/472/186 251/471/186 250/469/186
+f 471/838/404 469/835/404 472/839/404
+f 519/935/405 520/937/405 518/934/405
+f 519/935/406 521/939/406 520/937/406
+f 413/749/112 414/750/112 416/754/112
+f 486/868/364 488/871/364 422/763/364
+f 476/848/166 474/844/166 473/841/166
+f 535/967/167 473/841/167 536/970/167
+f 521/939/406 522/942/406 520/937/406
+f 440/790/114 442/793/114 441/791/114
+f 442/792/407 506/910/408 508/914/408
+f 476/847/409 412/748/410 474/843/410
+f 490/876/204 488/872/204 487/869/204
+f 490/877/411 426/768/411 424/765/411
+f 448/801/412 446/798/153 510/918/412
+f 462/824/198 464/827/198 463/826/198
+f 486/867/143 484/863/143 483/861/143
+f 461/823/198 462/824/198 463/826/198
+f 518/933/413 454/811/413 516/930/413
+f 483/861/414 484/863/414 481/857/414
+f 511/919/169 513/923/169 512/921/169
+f 458/818/415 460/822/415 459/820/415
+f 430/774/184 428/771/184 492/881/184
+f 436/783/164 438/787/164 437/785/164
+f 480/856/156 478/852/156 477/849/156
+f 435/782/416 436/783/416 437/785/416
+f 417/755/417 415/752/417 418/757/417
+f 440/789/418 438/786/418 504/906/418
+f 466/830/419 530/957/419 532/961/419
+f 499/895/420 501/899/420 500/898/420
+f 482/859/421 418/756/422 416/753/423
+f 520/938/337 456/815/337 518/933/337
+f 531/959/122 532/962/122 529/955/122
+f 421/761/212 419/758/212 422/762/212
+f 488/871/384 490/877/384 424/765/384
+f 453/809/163 454/810/163 456/816/163
+f 460/822/394 462/824/394 461/823/394
+f 478/851/338 414/751/338 476/847/338
+f 523/943/377 524/946/377 521/939/377
+f 503/903/424 504/905/424 502/902/424
+f 456/815/158 454/811/158 518/933/425
+f 489/873/426 490/876/426 487/869/426
+f 468/833/395 470/836/395 469/835/395
+f 464/828/427 462/825/427 528/954/427
+f 484/863/428 482/860/428 481/857/428
+f 480/855/356 482/859/356 416/753/356
+f 515/927/429 517/931/429 516/929/429
+f 453/809/218 451/806/218 454/810/218
+f 502/901/430 438/786/430 500/897/430
+f 412/748/188 410/744/188 474/843/188
+f 495/887/388 496/889/388 494/886/388
+f 491/879/159 493/883/159 492/882/159
+f 455/813/431 456/814/431 458/818/431
+f 474/843/432 410/744/432 536/969/432
+f 497/891/433 498/893/433 496/889/433
+f 411/746/192 412/747/192 414/750/192
+f 492/881/365 428/771/365 490/877/365
+f 451/806/376 452/807/376 454/810/376
+f 423/764/217 421/761/217 424/766/217
+f 452/808/194 450/804/194 514/925/194
+f 434/780/434 436/783/434 435/782/434
+f 488/871/435 424/765/435 422/763/435
+f 462/825/436 526/950/437 528/954/436
+f 409/743/172 410/745/172 412/747/172
+f 484/864/438 420/760/438 418/756/165
+f 410/744/168 472/840/168 536/969/168
+f 433/779/439 434/780/439 435/782/439
+f 493/883/440 494/886/440 492/882/440
+f 533/963/175 534/965/175 531/959/175
+f 535/967/441 536/970/441 533/963/441
+f 467/832/395 468/833/395 469/835/395
+f 512/922/442 448/801/442 510/918/442
+f 429/773/443 430/775/443 431/776/443
+f 494/885/366 430/774/366 492/881/366
+f 513/923/169 514/926/169 512/921/169
+f 468/834/444 466/830/445 532/961/444
+f 419/758/116 417/755/116 420/759/116
+f 480/855/111 416/753/111 414/751/111
+f 413/749/192 411/746/192 414/750/192
+f 517/931/429 518/934/429 516/929/429
+f 446/798/446 444/795/446 508/914/446
+f 522/941/447 458/819/447 520/938/447
+f 432/777/173 434/780/173 433/779/173
+f 511/919/380 512/921/380 510/917/380
+f 432/778/448 430/774/448 494/885/448
+f 469/835/178 470/836/178 472/839/178
+f 526/950/449 462/825/450 460/821/451
+f 530/958/379 528/953/379 527/951/379
+f 487/869/452 488/872/452 485/865/452
+f 527/951/453 528/953/453 525/947/453
+f 428/772/401 430/775/401 429/773/401
+f 513/923/454 515/927/454 514/926/454
+f 525/947/151 526/949/151 523/943/151
+f 528/953/171 526/949/171 525/947/171
+f 411/746/172 409/743/172 412/747/172
+f 458/819/138 456/815/455 520/938/138
+f 507/911/110 508/913/110 505/907/110
+f 491/879/155 492/882/155 490/878/155
+f 428/771/456 426/768/131 490/877/131
+f 496/890/457 432/778/457 494/885/457
+f 463/826/119 464/827/119 465/829/119
+f 505/907/132 506/909/132 504/905/132
+f 415/752/112 413/749/112 416/754/112
+f 423/764/392 424/766/392 426/769/392
+f 526/949/399 524/946/399 523/943/399
+f 481/857/209 482/860/209 479/853/209
+f 500/897/458 436/784/458 498/894/458
+f 516/930/459 452/808/459 514/925/459
+f 507/911/460 509/915/460 508/913/460
+f 497/891/139 499/895/139 498/893/139
+f 447/800/197 448/802/197 450/805/197
+f 529/955/133 530/958/133 527/951/133
+f 478/851/461 480/855/461 414/751/461
+f 460/821/341 458/819/341 524/945/341
+f 425/767/462 426/769/462 427/770/462
+f 472/839/140 410/745/140 409/743/140
+f 485/865/387 486/867/387 483/861/387
+f 509/915/380 511/919/380 510/917/380
+f 508/913/463 506/909/464 505/907/464
+f 498/894/465 434/781/466 496/890/466
+f 477/849/385 478/852/385 475/845/385
+f 501/899/390 503/903/390 502/902/390
+f 455/812/163 453/809/163 456/816/163
+f 471/838/467 472/839/467 409/743/467
+f 457/817/176 458/818/176 459/820/176
+f 449/803/468 447/800/468 450/805/468
+f 493/883/388 495/887/388 494/886/388
+f 517/931/182 519/935/182 518/934/182
+f 465/829/469 466/831/469 467/832/469
+f 489/874/155 491/879/155 490/878/155
+f 482/859/470 484/864/470 418/756/470
+f 437/785/471 438/787/471 439/788/471
+f 439/788/114 440/790/114 441/791/114
+f 466/830/339 464/828/339 530/957/339
+f 442/792/472 440/789/472 506/910/354
+f 419/758/473 420/759/473 422/762/473
+f 415/752/174 416/754/174 418/757/174
+f 445/797/181 443/794/181 444/796/181
+f 466/831/469 468/833/469 467/832/469
+f 443/794/193 441/791/193 444/796/193
+f 430/775/443 432/777/443 431/776/443
+f 482/860/209 480/856/209 479/853/209
+f 431/776/173 432/777/173 433/779/173
+f 421/761/217 422/762/217 424/766/217
+f 514/925/474 450/804/333 512/922/333
+f 479/853/156 480/856/156 477/849/156
+f 426/769/400 428/772/400 427/770/400
+f 499/895/139 500/898/139 498/893/139
+f 484/864/475 486/868/475 420/760/475
+f 509/915/117 510/917/117 508/913/117
+f 457/817/121 455/813/121 458/818/121
+f 450/804/476 448/801/477 512/922/476
+f 444/795/478 442/792/478 508/914/478
+f 503/903/142 505/907/142 504/905/142
+f 447/800/115 445/797/115 448/802/115
+f 425/767/136 423/764/136 426/769/136
+f 510/918/355 446/798/355 508/914/355
+f 446/799/479 445/797/479 444/796/479
+f 427/770/190 428/772/190 429/773/190
+f 470/837/185 534/966/185 536/969/185
+f 524/945/343 526/950/480 460/821/480
+f 534/965/481 532/962/481 531/959/481
+f 440/789/482 504/906/482 506/910/482
+f 417/755/483 418/757/483 420/759/483
+f 441/791/484 442/793/484 444/796/484
+f 468/834/485 532/961/485 534/966/486
+f 438/786/200 502/901/200 504/906/200
+f 438/786/487 436/784/487 500/897/487
+f 536/970/216 534/965/216 533/963/216
+f 458/819/219 522/941/219 524/945/219
+f 478/852/488 476/848/488 475/845/488
+f 445/797/115 446/799/115 448/802/115
+f 414/751/489 412/748/489 476/847/489
+f 438/787/186 440/790/186 439/788/186
+f 459/820/118 460/822/118 461/823/118
+f 451/806/375 449/803/375 452/807/375
+f 464/827/119 466/831/119 465/829/119
+f 464/828/490 528/954/491 530/957/490
+f 515/927/147 516/929/147 514/926/147
+f 436/784/492 434/781/492 498/894/492
+f 470/837/493 468/834/347 534/966/493
+f 488/872/452 486/867/452 485/865/452
+f 501/899/157 502/902/157 500/898/157
+f 486/868/179 422/763/179 420/760/179
+f 449/803/375 450/805/375 452/807/375
+f 472/840/494 470/837/494 536/969/494
+f 495/887/433 497/891/433 496/889/433
+f 434/781/495 432/778/495 496/890/495
+f 532/962/122 530/958/122 529/955/122
+f 524/946/377 522/942/377 521/939/377
+f 475/845/124 476/848/124 473/841/124
+f 473/841/126 474/844/126 536/970/126
+f 454/811/496 452/808/496 516/930/496
+g KsHauptsignal_Plane.006_Lights
+usemtl Lights
+f 88/188/497 28/66/497 40/90/497
+f 40/90/497 44/98/497 56/122/497
+f 88/188/498 64/138/498 72/156/498
+f 72/156/499 76/164/499 88/188/499
+f 40/90/500 28/66/500 32/74/500
+f 32/74/497 34/78/497 40/90/497
+f 56/122/497 44/98/497 48/106/497
+f 48/106/497 52/114/497 56/122/497
+f 56/122/501 58/126/501 64/138/501
+f 64/138/498 68/146/498 70/150/498
+f 88/188/498 76/164/498 80/172/498
+f 80/172/498 84/180/498 88/188/498
+f 88/188/502 26/62/502 28/66/502
+f 28/66/502 30/70/502 32/74/502
+f 40/90/500 34/78/500 36/82/500
+f 36/82/500 38/86/500 40/90/500
+f 40/90/503 42/94/503 44/98/503
+f 44/98/504 46/102/504 48/106/504
+f 48/106/505 50/110/505 52/114/505
+f 52/114/506 54/118/507 56/122/507
+f 64/138/508 58/126/509 60/130/508
+f 60/130/510 62/134/510 64/138/510
+f 64/138/509 66/142/511 68/146/511
+f 64/138/498 70/150/498 72/156/498
+f 72/156/512 74/160/512 76/164/512
+f 76/164/498 78/168/498 80/172/511
+f 80/172/513 82/176/513 84/180/513
+f 84/180/514 86/184/514 88/188/514
+f 64/138/515 88/188/515 40/90/515
+f 156/294/516 158/298/516 160/302/516
+f 200/384/517 204/392/517 216/416/517
+f 192/366/518 186/354/518 188/358/518
+f 176/334/519 180/342/519 184/350/519
+f 212/408/520 214/412/520 216/416/521
+f 188/358/522 190/362/522 192/366/522
+f 192/366/523 196/374/524 198/378/524
+f 184/350/525 172/326/525 176/334/525
+f 204/392/526 206/396/527 208/400/526
+f 200/384/528 202/388/529 204/392/528
+f 216/416/530 192/366/530 200/384/530
+f 168/318/531 162/306/531 164/310/531
+f 164/310/532 166/314/533 168/318/532
+f 160/302/534 162/306/534 168/318/534
+f 176/334/525 178/338/525 180/342/525
+f 208/400/535 210/404/535 212/408/535
+f 168/318/536 170/322/537 172/326/536
+f 168/318/538 172/326/538 184/350/538
+f 168/318/539 156/294/539 160/302/539
+f 180/342/540 182/346/541 184/350/540
+f 208/400/535 212/408/535 216/416/535
+f 216/416/530 204/392/530 208/400/530
+f 192/366/542 194/370/542 196/374/542
+f 184/350/543 186/354/543 192/366/543
+f 192/366/544 198/378/544 200/384/544
+f 216/416/532 156/294/532 168/318/532
+f 216/416/545 154/290/545 156/294/545
+f 172/326/546 174/330/546 176/334/546
+f 192/366/547 216/416/547 168/318/547
+f 56/122/548 64/138/548 40/90/548
+f 344/644/497 284/522/497 296/546/497
+f 296/546/497 300/554/497 312/578/497
+f 344/644/498 320/594/498 328/612/498
+f 328/612/498 332/620/498 344/644/498
+f 296/546/549 284/522/550 288/530/550
+f 288/530/497 290/534/497 296/546/497
+f 312/578/500 300/554/500 304/562/500
+f 304/562/551 308/570/551 312/578/551
+f 312/578/552 314/582/552 320/594/552
+f 320/594/498 324/602/498 326/606/498
+f 344/644/498 332/620/498 336/628/498
+f 336/628/553 340/636/553 344/644/553
+f 344/644/554 282/518/554 284/522/554
+f 284/522/555 286/526/555 288/530/555
+f 296/546/500 290/534/500 292/538/500
+f 292/538/556 294/542/556 296/546/556
+f 296/546/557 298/550/557 300/554/557
+f 300/554/500 302/558/500 304/562/500
+f 304/562/505 306/566/505 308/570/505
+f 308/570/558 310/574/558 312/578/558
+f 320/594/559 314/582/559 316/586/559
+f 316/586/560 318/590/560 320/594/560
+f 320/594/511 322/598/511 324/602/511
+f 320/594/511 326/606/511 328/612/511
+f 328/612/561 330/616/561 332/620/561
+f 332/620/562 334/624/562 336/628/562
+f 336/628/563 338/632/563 340/636/563
+f 340/636/564 342/640/564 344/644/564
+f 320/594/515 344/644/515 296/546/515
+f 312/578/548 320/594/548 296/546/548
+f 184/350/565 192/366/565 168/318/565
+f 509/916/541 507/912/541 505/908/541
+f 521/940/531 527/952/531 525/948/531
+f 473/842/532 533/964/532 521/940/532
+f 521/940/566 519/936/566 517/932/566
+f 517/932/567 515/928/567 513/924/567
+f 533/964/516 531/960/516 529/956/516
+f 497/892/568 495/888/568 493/884/569
+f 489/875/529 487/870/529 485/866/529
+f 477/850/570 475/846/521 473/842/570
+f 473/842/530 497/892/530 489/875/530
+f 497/892/571 503/904/571 501/900/571
+f 505/908/572 517/932/573 513/924/572
+f 473/842/574 535/968/575 533/964/575
+f 481/858/576 477/850/576 473/842/577
+f 497/892/547 473/842/547 521/940/547
+f 501/900/578 499/896/578 497/892/579
+f 497/892/580 493/884/580 491/880/580
+f 521/940/538 517/932/538 505/908/538
+f 505/908/565 497/892/565 521/940/565
+f 473/842/530 485/866/530 481/858/530
+f 497/892/544 491/880/544 489/875/544
+f 529/956/537 527/952/532 521/940/532
+f 521/940/539 533/964/539 529/956/539
+f 481/858/581 479/854/581 477/850/581
+f 505/908/582 503/904/583 497/892/582
+f 513/924/525 511/920/525 509/916/525
+f 489/875/517 485/866/517 473/842/517
+f 513/924/519 509/916/519 505/908/519
+f 525/948/584 523/944/584 521/940/532
+f 485/866/579 483/862/579 481/858/579
+l 570 571
+l 540 537
+l 572 569
+l 571 572
+l 538 539
+l 539 540
+l 543 544
+l 541 542
+l 573 574
+l 575 576
+l 542 543
+l 574 575
diff --git a/advtrains_signals_ks/models/advtrains_signals_ks_distant_smr45.obj b/advtrains_signals_ks/models/advtrains_signals_ks_distant_smr45.obj
new file mode 100644
index 0000000..c9a3644
--- /dev/null
+++ b/advtrains_signals_ks/models/advtrains_signals_ks_distant_smr45.obj
@@ -0,0 +1,3296 @@
+# Blender 3.4.1
+# www.blender.org
+mtllib advtrains_signals_ks_distant_smr45.mtl
+o KsHauptsignal_Plane.006
+v 0.331265 -0.503378 -0.290411
+v 0.290873 -0.503378 -0.330859
+v 0.331321 -0.503378 -0.371252
+v 0.371713 -0.503378 -0.330803
+v 0.250573 -0.503378 -0.209831
+v 0.210125 -0.503378 -0.169439
+v 0.169733 -0.503378 -0.209888
+v 0.210181 -0.503378 -0.250280
+v 0.331265 -0.299114 -0.290411
+v 0.331265 -0.172218 -0.290411
+v 0.290873 -0.172218 -0.330859
+v 0.290873 -0.299114 -0.330859
+v 0.331321 -0.299114 -0.371252
+v 0.331321 -0.172218 -0.371252
+v 0.371713 -0.172218 -0.330803
+v 0.371713 -0.299114 -0.330803
+v 0.250573 -0.299114 -0.209831
+v 0.250573 -0.172218 -0.209831
+v 0.210125 -0.172218 -0.169439
+v 0.210125 -0.299114 -0.169439
+v 0.169733 -0.299114 -0.209888
+v 0.169733 -0.172218 -0.209888
+v 0.210181 -0.172218 -0.250280
+v 0.210181 -0.299114 -0.250280
+v 0.319133 0.243339 -0.510336
+v 0.300057 0.242789 -0.491285
+v 0.358510 0.253858 -0.545204
+v 0.302250 0.253308 -0.489022
+v 0.362113 0.263561 -0.541534
+v 0.305853 0.263011 -0.485353
+v 0.366988 0.272074 -0.536599
+v 0.310727 0.271524 -0.480417
+v 0.372947 0.279070 -0.530588
+v 0.316687 0.278520 -0.474406
+v 0.379761 0.284280 -0.523731
+v 0.323501 0.283730 -0.467550
+v 0.387169 0.287504 -0.516293
+v 0.330908 0.286954 -0.460112
+v 0.394885 0.288619 -0.508559
+v 0.338624 0.288069 -0.452378
+v 0.402612 0.287581 -0.500827
+v 0.346352 0.287030 -0.444646
+v 0.410055 0.284430 -0.493394
+v 0.353796 0.283880 -0.437212
+v 0.416928 0.279287 -0.486544
+v 0.360668 0.278737 -0.430362
+v 0.422965 0.272350 -0.480542
+v 0.366706 0.271800 -0.424360
+v 0.427935 0.263886 -0.475618
+v 0.371676 0.263336 -0.419437
+v 0.431648 0.254219 -0.471961
+v 0.375387 0.253669 -0.415780
+v 0.396777 0.243722 -0.432582
+v 0.377700 0.243172 -0.413532
+v 0.397576 0.232316 -0.431776
+v 0.378521 0.232252 -0.412777
+v 0.396693 0.221688 -0.432436
+v 0.377820 0.221322 -0.413543
+v 0.394516 0.211344 -0.434671
+v 0.375622 0.210797 -0.415801
+v 0.390909 0.201642 -0.438338
+v 0.372015 0.201094 -0.419468
+v 0.386032 0.193129 -0.443270
+v 0.367137 0.192581 -0.424400
+v 0.380071 0.186134 -0.449279
+v 0.361176 0.185585 -0.430409
+v 0.373255 0.180924 -0.456133
+v 0.354360 0.180374 -0.437263
+v 0.365846 0.177700 -0.463570
+v 0.346951 0.177150 -0.444700
+v 0.358130 0.176585 -0.471304
+v 0.339235 0.176035 -0.452434
+v 0.350403 0.177623 -0.479037
+v 0.331507 0.177073 -0.460167
+v 0.342960 0.180774 -0.486470
+v 0.324064 0.180224 -0.467603
+v 0.336090 0.185916 -0.493322
+v 0.317193 0.185367 -0.474454
+v 0.330055 0.192853 -0.499326
+v 0.311159 0.192304 -0.480458
+v 0.325088 0.201317 -0.504253
+v 0.306191 0.200769 -0.485385
+v 0.321379 0.210983 -0.507913
+v 0.302483 0.210436 -0.489044
+v 0.319096 0.221305 -0.510144
+v 0.300175 0.220938 -0.491298
+v 0.318385 0.231924 -0.511081
+v 0.299357 0.231861 -0.492054
+v 0.317383 0.243790 -0.512081
+v 0.272458 0.243271 -0.467219
+v 0.356856 0.254782 -0.546846
+v 0.274748 0.254264 -0.464853
+v 0.360620 0.264921 -0.543010
+v 0.278512 0.264402 -0.461017
+v 0.365713 0.273816 -0.537851
+v 0.283605 0.273298 -0.455858
+v 0.371939 0.281127 -0.531568
+v 0.289831 0.280609 -0.449575
+v 0.379058 0.286572 -0.524402
+v 0.296950 0.286053 -0.442409
+v 0.386798 0.289941 -0.516630
+v 0.304690 0.289423 -0.434636
+v 0.394860 0.291105 -0.508548
+v 0.312753 0.290587 -0.426554
+v 0.402936 0.290021 -0.500467
+v 0.320828 0.289502 -0.418475
+v 0.410715 0.286728 -0.492701
+v 0.328607 0.286210 -0.410707
+v 0.417897 0.281354 -0.485544
+v 0.335789 0.280836 -0.403551
+v 0.424207 0.274105 -0.479273
+v 0.342099 0.273587 -0.397280
+v 0.429401 0.265261 -0.474130
+v 0.347293 0.264742 -0.392137
+v 0.433283 0.255160 -0.470311
+v 0.351175 0.254641 -0.388318
+v 0.398517 0.244190 -0.430832
+v 0.353592 0.243672 -0.385970
+v 0.399339 0.232775 -0.430008
+v 0.354453 0.232256 -0.385184
+v 0.398467 0.221351 -0.430670
+v 0.353724 0.220833 -0.385989
+v 0.396178 0.210359 -0.433036
+v 0.351434 0.209840 -0.388355
+v 0.392414 0.200220 -0.436872
+v 0.347671 0.199702 -0.392192
+v 0.387321 0.191324 -0.442031
+v 0.342578 0.190806 -0.397351
+v 0.381095 0.184013 -0.448314
+v 0.336352 0.183495 -0.403633
+v 0.373976 0.178569 -0.455480
+v 0.329233 0.178051 -0.410799
+v 0.366237 0.175200 -0.463253
+v 0.321493 0.174682 -0.418572
+v 0.358174 0.174035 -0.471335
+v 0.313431 0.173517 -0.426654
+v 0.350098 0.175120 -0.479415
+v 0.305355 0.174602 -0.434734
+v 0.342320 0.178413 -0.487182
+v 0.297577 0.177894 -0.442501
+v 0.335138 0.183787 -0.494338
+v 0.290395 0.183268 -0.449658
+v 0.328827 0.191035 -0.500609
+v 0.284084 0.190517 -0.455928
+v 0.323633 0.199880 -0.505752
+v 0.278889 0.199362 -0.461072
+v 0.319752 0.209981 -0.509571
+v 0.275009 0.209463 -0.464891
+v 0.317335 0.220950 -0.511919
+v 0.272590 0.220432 -0.467238
+v 0.316617 0.232366 -0.512849
+v 0.271731 0.231848 -0.468025
+v 0.483236 0.383581 -0.346004
+v 0.464159 0.383353 -0.326953
+v 0.521307 0.387932 -0.382179
+v 0.465047 0.387704 -0.325997
+v 0.522780 0.391945 -0.380644
+v 0.466519 0.391717 -0.324462
+v 0.524780 0.395466 -0.378587
+v 0.468520 0.395239 -0.322405
+v 0.527232 0.398360 -0.376087
+v 0.470972 0.398132 -0.319906
+v 0.530041 0.400515 -0.373241
+v 0.473781 0.400287 -0.317060
+v 0.533099 0.401848 -0.370159
+v 0.476838 0.401621 -0.313977
+v 0.536288 0.402309 -0.366959
+v 0.480028 0.402082 -0.310776
+v 0.539486 0.401880 -0.363762
+v 0.483227 0.401652 -0.307580
+v 0.542571 0.400577 -0.360693
+v 0.486311 0.400349 -0.304511
+v 0.545423 0.398450 -0.357870
+v 0.489163 0.398222 -0.301688
+v 0.547933 0.395581 -0.355400
+v 0.491673 0.395353 -0.299218
+v 0.550004 0.392080 -0.353379
+v 0.493744 0.391852 -0.297197
+v 0.551557 0.388081 -0.351885
+v 0.495298 0.387854 -0.295703
+v 0.515350 0.383739 -0.313843
+v 0.496273 0.383512 -0.294791
+v 0.515680 0.379021 -0.313508
+v 0.496634 0.378995 -0.294500
+v 0.515249 0.374626 -0.313715
+v 0.496362 0.374474 -0.294835
+v 0.514364 0.370347 -0.314656
+v 0.495469 0.370121 -0.295786
+v 0.512889 0.366334 -0.316188
+v 0.493993 0.366108 -0.297319
+v 0.510885 0.362813 -0.318243
+v 0.491990 0.362586 -0.299373
+v 0.508431 0.359920 -0.320739
+v 0.489536 0.359693 -0.301869
+v 0.505620 0.357765 -0.323583
+v 0.486726 0.357538 -0.304713
+v 0.502561 0.356431 -0.326664
+v 0.483666 0.356204 -0.307794
+v 0.499372 0.355970 -0.329864
+v 0.480476 0.355743 -0.310995
+v 0.496174 0.356399 -0.333061
+v 0.477278 0.356172 -0.314192
+v 0.493090 0.357703 -0.336131
+v 0.474195 0.357476 -0.317263
+v 0.490241 0.359830 -0.338957
+v 0.471344 0.359603 -0.320088
+v 0.487733 0.362699 -0.341428
+v 0.468837 0.362472 -0.322560
+v 0.485664 0.366200 -0.343452
+v 0.466768 0.365973 -0.324584
+v 0.484114 0.370198 -0.344950
+v 0.465218 0.369972 -0.326082
+v 0.483153 0.374467 -0.345857
+v 0.464247 0.374316 -0.326997
+v 0.482926 0.378860 -0.346310
+v 0.463890 0.378833 -0.327291
+v 0.482511 0.383767 -0.346723
+v 0.437586 0.383553 -0.301861
+v 0.520619 0.388314 -0.382855
+v 0.438511 0.388099 -0.300861
+v 0.522156 0.392507 -0.381248
+v 0.440048 0.392293 -0.299255
+v 0.524245 0.396187 -0.379098
+v 0.442137 0.395973 -0.297104
+v 0.526807 0.399211 -0.376484
+v 0.444699 0.398996 -0.294491
+v 0.529740 0.401463 -0.373509
+v 0.447632 0.401248 -0.291516
+v 0.532935 0.402856 -0.370288
+v 0.450827 0.402642 -0.288294
+v 0.536268 0.403338 -0.366942
+v 0.454160 0.403124 -0.284949
+v 0.539610 0.402889 -0.363603
+v 0.457502 0.402675 -0.281609
+v 0.542834 0.401527 -0.360397
+v 0.460725 0.401313 -0.278403
+v 0.545815 0.399305 -0.357447
+v 0.463707 0.399090 -0.275454
+v 0.548439 0.396306 -0.354867
+v 0.466331 0.396092 -0.272874
+v 0.550605 0.392648 -0.352758
+v 0.468497 0.392434 -0.270765
+v 0.552230 0.388470 -0.351198
+v 0.470122 0.388256 -0.269205
+v 0.516068 0.383933 -0.313117
+v 0.471143 0.383719 -0.268254
+v 0.516409 0.379211 -0.312776
+v 0.471521 0.378997 -0.267951
+v 0.515986 0.374486 -0.312988
+v 0.471242 0.374272 -0.268307
+v 0.515059 0.369939 -0.313988
+v 0.470317 0.369725 -0.269307
+v 0.513523 0.365746 -0.315594
+v 0.468779 0.365532 -0.270914
+v 0.511433 0.362067 -0.317745
+v 0.466690 0.361852 -0.273065
+v 0.508873 0.359043 -0.320358
+v 0.464130 0.358828 -0.275678
+v 0.505939 0.356791 -0.323333
+v 0.461196 0.356576 -0.278652
+v 0.502744 0.355397 -0.326555
+v 0.458001 0.355183 -0.281873
+v 0.499412 0.354916 -0.329900
+v 0.454668 0.354701 -0.285219
+v 0.496069 0.355364 -0.333240
+v 0.451326 0.355150 -0.288558
+v 0.492845 0.356726 -0.336446
+v 0.448102 0.356512 -0.291765
+v 0.489864 0.358949 -0.339395
+v 0.445122 0.358735 -0.294714
+v 0.487240 0.361947 -0.341975
+v 0.442497 0.361733 -0.297294
+v 0.485074 0.365605 -0.344085
+v 0.440331 0.365391 -0.299405
+v 0.483449 0.369783 -0.345645
+v 0.438706 0.369569 -0.300964
+v 0.482429 0.374320 -0.346595
+v 0.437685 0.374106 -0.301913
+v 0.482193 0.379042 -0.347041
+v 0.437307 0.378828 -0.302217
+v 0.460456 0.243339 -0.368816
+v 0.441379 0.242789 -0.349765
+v 0.499833 0.253858 -0.403684
+v 0.443572 0.253308 -0.347502
+v 0.503436 0.263561 -0.400015
+v 0.447175 0.263011 -0.343833
+v 0.508311 0.272074 -0.395079
+v 0.452051 0.271524 -0.338898
+v 0.514270 0.279070 -0.389068
+v 0.458010 0.278520 -0.332886
+v 0.521084 0.284280 -0.382211
+v 0.464823 0.283730 -0.326029
+v 0.528491 0.287504 -0.374773
+v 0.472231 0.286954 -0.318592
+v 0.536208 0.288619 -0.367039
+v 0.479947 0.288069 -0.310858
+v 0.543936 0.287581 -0.359307
+v 0.487675 0.287030 -0.303125
+v 0.551378 0.284430 -0.351873
+v 0.495118 0.283880 -0.295691
+v 0.558251 0.279287 -0.345024
+v 0.501991 0.278737 -0.288842
+v 0.564287 0.272350 -0.339022
+v 0.508028 0.271800 -0.282840
+v 0.569258 0.263886 -0.334098
+v 0.512998 0.263336 -0.277916
+v 0.572971 0.254219 -0.330442
+v 0.516710 0.253669 -0.274260
+v 0.538098 0.243722 -0.291062
+v 0.519022 0.243172 -0.272011
+v 0.538899 0.232316 -0.290256
+v 0.519844 0.232252 -0.271257
+v 0.538016 0.221688 -0.290916
+v 0.519142 0.221322 -0.272022
+v 0.535839 0.211344 -0.293152
+v 0.516945 0.210797 -0.274281
+v 0.532232 0.201642 -0.296818
+v 0.513338 0.201094 -0.277947
+v 0.527354 0.193129 -0.301750
+v 0.508460 0.192581 -0.282880
+v 0.521393 0.186134 -0.307759
+v 0.502498 0.185585 -0.288888
+v 0.514577 0.180924 -0.314613
+v 0.495683 0.180374 -0.295744
+v 0.507169 0.177700 -0.322050
+v 0.488274 0.177150 -0.303180
+v 0.499452 0.176585 -0.329783
+v 0.480558 0.176035 -0.310914
+v 0.491725 0.177623 -0.337516
+v 0.472830 0.177073 -0.318648
+v 0.484283 0.180774 -0.344951
+v 0.465387 0.180224 -0.326082
+v 0.477412 0.185916 -0.351802
+v 0.458516 0.185368 -0.332934
+v 0.471378 0.192853 -0.357807
+v 0.452481 0.192304 -0.338938
+v 0.466410 0.201317 -0.362733
+v 0.447514 0.200769 -0.343865
+v 0.462702 0.210983 -0.366392
+v 0.443806 0.210436 -0.347525
+v 0.460419 0.221305 -0.368624
+v 0.441498 0.220938 -0.349777
+v 0.459708 0.231924 -0.369560
+v 0.440679 0.231861 -0.350534
+v 0.458707 0.243790 -0.370561
+v 0.413781 0.243271 -0.325700
+v 0.498179 0.254782 -0.405326
+v 0.416072 0.254264 -0.323333
+v 0.501943 0.264921 -0.401489
+v 0.419835 0.264402 -0.319496
+v 0.507036 0.273816 -0.396331
+v 0.424927 0.273298 -0.314337
+v 0.513261 0.281127 -0.390048
+v 0.431153 0.280609 -0.308055
+v 0.520381 0.286572 -0.382883
+v 0.438273 0.286053 -0.300889
+v 0.528120 0.289941 -0.375109
+v 0.446013 0.289423 -0.293116
+v 0.536184 0.291105 -0.367028
+v 0.454075 0.290587 -0.285034
+v 0.544259 0.290021 -0.358948
+v 0.462151 0.289502 -0.276955
+v 0.552038 0.286728 -0.351181
+v 0.469929 0.286210 -0.269187
+v 0.559220 0.281354 -0.344024
+v 0.477111 0.280836 -0.262030
+v 0.565529 0.274105 -0.337753
+v 0.483421 0.273587 -0.255760
+v 0.570724 0.265261 -0.332610
+v 0.488616 0.264742 -0.250617
+v 0.574606 0.255160 -0.328791
+v 0.492497 0.254641 -0.246797
+v 0.539840 0.244190 -0.289312
+v 0.494915 0.243672 -0.244450
+v 0.540662 0.232775 -0.288488
+v 0.495775 0.232256 -0.243663
+v 0.539790 0.221351 -0.289150
+v 0.495047 0.220833 -0.244469
+v 0.537500 0.210359 -0.291516
+v 0.492757 0.209840 -0.246835
+v 0.533736 0.200220 -0.295352
+v 0.488993 0.199702 -0.250671
+v 0.528644 0.191324 -0.300511
+v 0.483901 0.190806 -0.255830
+v 0.522418 0.184014 -0.306794
+v 0.477675 0.183495 -0.262113
+v 0.515298 0.178569 -0.313960
+v 0.470556 0.178051 -0.269279
+v 0.507559 0.175200 -0.321733
+v 0.462816 0.174682 -0.277052
+v 0.499496 0.174035 -0.329814
+v 0.454753 0.173517 -0.285134
+v 0.491420 0.175120 -0.337894
+v 0.446677 0.174602 -0.293214
+v 0.483642 0.178413 -0.345661
+v 0.438899 0.177894 -0.300981
+v 0.476460 0.183787 -0.352818
+v 0.431717 0.183268 -0.308137
+v 0.470150 0.191035 -0.359089
+v 0.425407 0.190517 -0.314408
+v 0.464955 0.199880 -0.364232
+v 0.420212 0.199362 -0.319552
+v 0.461074 0.209981 -0.368051
+v 0.416331 0.209463 -0.323370
+v 0.458657 0.220950 -0.370399
+v 0.413913 0.220432 -0.325719
+v 0.457939 0.232366 -0.371329
+v 0.413053 0.231848 -0.326504
+v 0.437307 0.050272 -0.302217
+v 0.482193 0.050487 -0.347041
+v 0.437685 0.045551 -0.301913
+v 0.482429 0.045765 -0.346595
+v 0.438706 0.041014 -0.300964
+v 0.483449 0.041228 -0.345645
+v 0.440331 0.036836 -0.299405
+v 0.485074 0.037050 -0.344085
+v 0.442497 0.033177 -0.297294
+v 0.487240 0.033392 -0.341975
+v 0.445122 0.030179 -0.294714
+v 0.489864 0.030393 -0.339395
+v 0.448102 0.027956 -0.291765
+v 0.492845 0.028171 -0.336446
+v 0.451326 0.026594 -0.288558
+v 0.496069 0.026809 -0.333240
+v 0.454668 0.026146 -0.285219
+v 0.499412 0.026360 -0.329900
+v 0.458001 0.026627 -0.281873
+v 0.502744 0.026842 -0.326555
+v 0.461196 0.028021 -0.278652
+v 0.505939 0.028235 -0.323333
+v 0.464130 0.030273 -0.275678
+v 0.508873 0.030487 -0.320358
+v 0.466691 0.033297 -0.273065
+v 0.511433 0.033511 -0.317745
+v 0.468779 0.036976 -0.270914
+v 0.513523 0.037191 -0.315594
+v 0.470317 0.041170 -0.269307
+v 0.515059 0.041384 -0.313988
+v 0.471242 0.045716 -0.268307
+v 0.515986 0.045931 -0.312988
+v 0.471521 0.050441 -0.267951
+v 0.516409 0.050656 -0.312776
+v 0.471143 0.055163 -0.268254
+v 0.516068 0.055377 -0.313117
+v 0.470122 0.059700 -0.269205
+v 0.552230 0.059915 -0.351198
+v 0.468497 0.063878 -0.270765
+v 0.550605 0.064093 -0.352758
+v 0.466331 0.067537 -0.272874
+v 0.548439 0.067751 -0.354867
+v 0.463707 0.070535 -0.275454
+v 0.545815 0.070749 -0.357447
+v 0.460725 0.072757 -0.278403
+v 0.542834 0.072972 -0.360397
+v 0.457502 0.074119 -0.281609
+v 0.539610 0.074334 -0.363603
+v 0.454160 0.074568 -0.284949
+v 0.536268 0.074782 -0.366942
+v 0.450827 0.074086 -0.288294
+v 0.532935 0.074301 -0.370288
+v 0.447633 0.072693 -0.291516
+v 0.529740 0.072907 -0.373509
+v 0.444699 0.070441 -0.294491
+v 0.526807 0.070655 -0.376484
+v 0.442137 0.067417 -0.297104
+v 0.524245 0.067631 -0.379098
+v 0.440048 0.063738 -0.299255
+v 0.522156 0.063952 -0.381248
+v 0.438511 0.059544 -0.300861
+v 0.520619 0.059758 -0.382855
+v 0.437586 0.054997 -0.301861
+v 0.482511 0.055212 -0.346723
+v 0.463890 0.050278 -0.327291
+v 0.482926 0.050304 -0.346310
+v 0.464247 0.045760 -0.326997
+v 0.483153 0.045912 -0.345857
+v 0.465218 0.041416 -0.326082
+v 0.484114 0.041642 -0.344950
+v 0.466768 0.037418 -0.324584
+v 0.485664 0.037644 -0.343452
+v 0.468837 0.033917 -0.322560
+v 0.487733 0.034143 -0.341428
+v 0.471344 0.031047 -0.320088
+v 0.490241 0.031274 -0.338957
+v 0.474195 0.028920 -0.317263
+v 0.493090 0.029147 -0.336131
+v 0.477278 0.027617 -0.314192
+v 0.496174 0.027844 -0.333061
+v 0.480476 0.027187 -0.310995
+v 0.499372 0.027415 -0.329864
+v 0.483666 0.027648 -0.307794
+v 0.502561 0.027876 -0.326664
+v 0.486726 0.028982 -0.304713
+v 0.505620 0.029209 -0.323583
+v 0.489536 0.031137 -0.301870
+v 0.508431 0.031364 -0.320739
+v 0.491990 0.034031 -0.299373
+v 0.510885 0.034258 -0.318243
+v 0.493993 0.037552 -0.297319
+v 0.512889 0.037779 -0.316188
+v 0.495469 0.041566 -0.295786
+v 0.514364 0.041792 -0.314656
+v 0.496362 0.045919 -0.294835
+v 0.515249 0.046070 -0.313715
+v 0.496634 0.050440 -0.294500
+v 0.515680 0.050466 -0.313508
+v 0.496273 0.054956 -0.294791
+v 0.515350 0.055184 -0.313843
+v 0.495298 0.059298 -0.295703
+v 0.551557 0.059526 -0.351885
+v 0.493744 0.063296 -0.297197
+v 0.550004 0.063524 -0.353379
+v 0.491673 0.066797 -0.299218
+v 0.547933 0.067025 -0.355400
+v 0.489163 0.069667 -0.301688
+v 0.545423 0.069894 -0.357870
+v 0.486311 0.071794 -0.304511
+v 0.542571 0.072021 -0.360693
+v 0.483227 0.073097 -0.307580
+v 0.539486 0.073325 -0.363762
+v 0.480028 0.073526 -0.310776
+v 0.536288 0.073754 -0.366959
+v 0.476838 0.073065 -0.313977
+v 0.533099 0.073293 -0.370159
+v 0.473781 0.071732 -0.317060
+v 0.530041 0.071959 -0.373241
+v 0.470972 0.069577 -0.319906
+v 0.527232 0.069804 -0.376087
+v 0.468520 0.066683 -0.322405
+v 0.524780 0.066911 -0.378587
+v 0.466519 0.063162 -0.324462
+v 0.522780 0.063390 -0.380644
+v 0.465047 0.059149 -0.325997
+v 0.521307 0.059376 -0.382179
+v 0.464159 0.054798 -0.326953
+v 0.483236 0.055025 -0.346004
+v 0.331265 -0.168051 -0.290411
+v 0.290873 -0.168051 -0.330859
+v 0.331321 -0.168051 -0.371252
+v 0.371713 -0.168051 -0.330803
+v 0.250573 -0.168051 -0.209831
+v 0.210125 -0.168051 -0.169439
+v 0.169733 -0.168051 -0.209888
+v 0.210181 -0.168051 -0.250280
+v 0.169733 -0.168051 -0.209888
+v 0.210125 -0.168051 -0.169439
+v 0.371713 -0.168051 -0.330803
+v 0.331321 -0.168051 -0.371252
+v 0.210181 -0.168051 -0.250280
+v 0.290873 -0.168051 -0.330859
+v 0.250573 -0.168051 -0.209831
+v 0.331265 -0.168051 -0.290411
+v 0.331265 0.037812 -0.290411
+v 0.331265 0.165702 -0.290411
+v 0.290873 0.165702 -0.330859
+v 0.290873 0.037812 -0.330859
+v 0.331321 0.037812 -0.371252
+v 0.331321 0.165702 -0.371251
+v 0.371713 0.165702 -0.330803
+v 0.371713 0.037812 -0.330803
+v 0.250573 0.037812 -0.209831
+v 0.250573 0.165702 -0.209831
+v 0.210125 0.165702 -0.169439
+v 0.210125 0.037812 -0.169439
+v 0.169733 0.037812 -0.209888
+v 0.169733 0.165702 -0.209888
+v 0.210181 0.165702 -0.250280
+v 0.210181 0.037812 -0.250280
+v 0.331265 0.166622 -0.290411
+v 0.290873 0.166622 -0.330859
+v 0.331321 0.166622 -0.371251
+v 0.371713 0.166622 -0.330803
+v 0.250573 0.166622 -0.209831
+v 0.210125 0.166622 -0.169439
+v 0.169733 0.166622 -0.209888
+v 0.210181 0.166622 -0.250280
+v 0.169733 0.166622 -0.209888
+v 0.210125 0.166622 -0.169439
+v 0.371713 0.166622 -0.330803
+v 0.331321 0.166622 -0.371251
+v 0.210181 0.166622 -0.250280
+v 0.290873 0.166622 -0.330859
+v 0.250573 0.166622 -0.209831
+v 0.331265 0.166622 -0.290411
+v 0.331265 0.370885 -0.290411
+v 0.331265 0.497782 -0.290411
+v 0.290873 0.497782 -0.330859
+v 0.290873 0.370885 -0.330859
+v 0.331321 0.370885 -0.371251
+v 0.331321 0.497782 -0.371251
+v 0.371713 0.497782 -0.330803
+v 0.371713 0.370885 -0.330803
+v 0.250573 0.370885 -0.209831
+v 0.250573 0.497782 -0.209831
+v 0.210125 0.497782 -0.169439
+v 0.210125 0.370885 -0.169439
+v 0.169732 0.370885 -0.209888
+v 0.169732 0.497782 -0.209888
+v 0.210181 0.497782 -0.250280
+v 0.210181 0.370885 -0.250280
+v 0.578449 -0.080213 -0.196962
+v 0.225142 -0.080213 -0.550762
+v 0.578448 0.496942 -0.196961
+v 0.225141 0.496942 -0.550761
+v 0.544895 0.496942 -0.163455
+v 0.544895 -0.080213 -0.163455
+v 0.191588 -0.080213 -0.517255
+v 0.191588 0.496942 -0.517255
+vn -0.7066 -0.0000 -0.7076
+vn 0.7066 -0.0000 0.7076
+vn 0.7076 -0.0000 -0.7066
+vn -0.7076 -0.0000 0.7066
+vn -0.0000 -1.0000 -0.0000
+vn -0.0000 1.0000 -0.0000
+vn -0.0619 -0.9955 -0.0718
+vn 0.7048 -0.0946 0.7030
+vn -0.7015 -0.1015 -0.7054
+vn -0.7052 0.0944 -0.7027
+vn 0.7011 0.1014 0.7058
+vn -0.7040 0.0945 -0.7039
+vn 0.7024 0.1015 0.7045
+vn 0.7040 -0.0946 0.7039
+vn -0.7024 -0.1014 -0.7045
+vn 0.5353 0.6370 0.5546
+vn 0.7132 0.0090 -0.7009
+vn 0.7094 0.0067 -0.7048
+vn 0.7079 0.0100 -0.7062
+vn 0.7060 0.0089 -0.7082
+vn 0.7058 0.0091 -0.7084
+vn 0.7086 0.0096 -0.7056
+vn 0.7091 0.0054 -0.7050
+vn 0.7055 0.0081 -0.7087
+vn 0.7072 0.0101 -0.7070
+vn 0.7093 0.0077 -0.7049
+vn 0.7065 0.0093 -0.7076
+vn 0.7091 0.0087 -0.7051
+vn 0.7052 0.0071 -0.7089
+vn 0.7057 0.0059 -0.7085
+vn 0.7088 0.0057 -0.7053
+vn 0.7063 0.0002 -0.7079
+vn 0.7054 -0.0134 -0.7087
+vn 0.7040 -0.0144 -0.7101
+vn 0.2792 -0.9600 -0.0209
+vn 0.7138 0.0120 -0.7002
+vn 0.7013 0.0119 -0.7128
+vn 0.7135 0.0151 -0.7005
+vn 0.7125 0.0180 -0.7015
+vn 0.7106 0.0198 -0.7033
+vn 0.7063 0.0208 -0.7077
+vn 0.7042 0.0200 -0.7098
+vn 0.7025 0.0178 -0.7114
+vn 0.7016 0.0147 -0.7124
+vn -0.1112 -0.9097 -0.4000
+vn 0.7019 0.0089 -0.7122
+vn 0.7030 0.0057 -0.7112
+vn 0.7042 -0.0920 0.7040
+vn -0.4526 0.7708 -0.4484
+vn 0.4439 0.7752 0.4494
+vn -0.5415 -0.6371 -0.5485
+vn 0.3281 0.8836 0.3342
+vn -0.1972 -0.9579 -0.2086
+vn -0.1981 -0.9579 -0.2077
+vn 0.0636 0.9955 0.0700
+vn 0.0766 -0.9948 0.0670
+vn 0.1998 0.9579 0.2061
+vn 0.3395 -0.8803 0.3314
+vn 0.4540 -0.7708 0.4470
+vn -0.6197 -0.4745 -0.6252
+vn 0.6266 -0.4683 0.6229
+vn 0.6763 -0.2964 0.6744
+vn -0.7024 -0.0990 -0.7048
+vn -0.7025 0.1021 -0.7043
+vn 0.6151 0.4744 0.6298
+vn 0.7085 0.0208 -0.7054
+vn 0.6713 0.2954 0.6797
+vn 0.7082 -0.0002 -0.7060
+vn 0.4350 0.7751 0.4582
+vn 0.3180 0.8834 0.3442
+vn 0.5426 0.6371 0.5474
+vn -0.3258 -0.8835 -0.3365
+vn -0.2214 0.9557 -0.1938
+vn -0.3481 0.8801 -0.3229
+vn -0.4613 0.7707 -0.4395
+vn -0.5570 0.6316 -0.5393
+vn -0.6311 0.4683 -0.6184
+vn -0.6801 0.2888 -0.6738
+vn 0.7027 0.1091 0.7031
+vn -0.7040 0.0946 -0.7039
+vn -0.5410 -0.6370 -0.5491
+vn -0.4418 -0.7752 -0.4514
+vn 0.6205 0.4745 0.6244
+vn 0.0523 0.9953 0.0814
+vn 0.5509 -0.6317 0.5455
+vn -0.3266 -0.8835 -0.3357
+vn -0.0610 -0.9955 -0.0727
+vn -0.0748 0.9948 -0.0686
+vn -0.2105 0.9559 -0.2047
+vn -0.3381 0.8802 -0.3330
+vn -0.6718 -0.3033 -0.6758
+vn -0.5498 0.6317 -0.5465
+vn -0.6258 0.4683 -0.6237
+vn -0.6766 0.2926 -0.6757
+vn 0.7024 0.1014 0.7045
+vn 0.1888 0.9578 0.2170
+vn -0.6738 -0.2936 -0.6781
+vn -0.6193 -0.4745 -0.6256
+vn 0.6738 0.2936 0.6781
+vn -0.0862 0.9946 -0.0574
+vn -0.4426 -0.7752 -0.4508
+vn 0.2121 -0.9559 0.2031
+vn 0.0775 -0.9948 0.0661
+vn 0.2130 -0.9559 0.2022
+vn 0.3403 -0.8803 0.3306
+vn 0.4547 -0.7708 0.4464
+vn 0.5514 -0.6317 0.5449
+vn 0.6271 -0.4683 0.6225
+vn 0.6786 -0.2869 0.6762
+vn -0.7024 -0.1015 -0.7045
+vn 0.7208 0.0217 -0.6929
+vn -0.6209 -0.4745 -0.6240
+vn 0.6735 0.2956 0.6775
+vn 0.7037 -0.0946 0.7042
+vn 0.6214 0.4745 0.6235
+vn -0.4445 -0.7753 -0.4488
+vn -0.6728 -0.3031 -0.6749
+vn -0.2087 0.9560 -0.2064
+vn -0.4512 0.7708 -0.4498
+vn 0.6927 0.0359 -0.7203
+vn 0.0655 0.9955 0.0682
+vn 0.5491 -0.6317 0.5473
+vn 0.0737 -0.9948 0.0698
+vn 0.7022 0.1013 0.7047
+vn -0.6249 0.4684 -0.6246
+vn 0.7039 -0.0920 0.7044
+vn 0.7033 0.0218 -0.7105
+vn 0.7031 0.0220 -0.7107
+vn 0.7110 0.0207 -0.7029
+vn 0.7144 0.0479 -0.6981
+vn 0.7039 0.0503 -0.7085
+vn -0.7028 0.1020 -0.7041
+vn 0.4518 -0.7708 0.4493
+vn -0.0647 -0.9955 -0.0688
+vn 0.7115 0.0131 -0.7025
+vn -0.0643 -0.9955 -0.0692
+vn 0.6988 -0.0348 -0.7144
+vn 0.7082 0.0244 -0.7056
+vn -0.5516 0.6316 -0.5448
+vn -0.7037 0.0946 -0.7042
+vn 0.5492 -0.6318 0.5470
+vn -0.7042 0.0945 -0.7037
+vn 0.3256 0.8835 0.3367
+vn 0.7114 0.0160 -0.7026
+vn 0.7026 0.0192 -0.7113
+vn -0.3291 -0.8836 -0.3332
+vn -0.4448 -0.7752 -0.4486
+vn 0.7093 0.0503 -0.7031
+vn 0.7093 0.0504 -0.7031
+vn 0.6922 0.0286 -0.7212
+vn 0.2093 -0.9560 0.2057
+vn 0.0743 -0.9948 0.0697
+vn 0.7108 0.0138 -0.7033
+vn 0.1971 0.9579 0.2089
+vn -0.0777 0.9948 -0.0659
+vn 0.6191 0.4744 0.6258
+vn -0.6272 0.4683 -0.6224
+vn 0.7098 0.0229 -0.7040
+vn -0.2133 0.9559 -0.2021
+vn 0.6948 0.0428 -0.7179
+vn 0.6948 0.0429 -0.7179
+vn -0.7027 -0.1015 -0.7042
+vn 0.2015 0.9579 0.2043
+vn 0.6255 -0.4682 0.6241
+vn 0.7215 0.0364 -0.6915
+vn 0.7025 0.1089 0.7033
+vn 0.7040 -0.0945 0.7039
+vn 0.7043 0.0004 -0.7099
+vn -0.5433 -0.6371 -0.5467
+vn -0.7028 -0.0991 -0.7044
+vn 0.3370 -0.8803 0.3339
+vn -0.7027 -0.1016 -0.7042
+vn 0.4521 -0.7707 0.4490
+vn -0.5431 -0.6371 -0.5469
+vn 0.6253 -0.4683 0.6243
+vn -0.0730 0.9948 -0.0706
+vn 0.4417 0.7752 0.4517
+vn -0.6761 0.2926 -0.6763
+vn 0.7187 0.0430 -0.6940
+vn -0.4548 0.7708 -0.4461
+vn 0.6748 0.2937 0.6771
+vn -0.2008 -0.9580 -0.2049
+vn 0.6988 0.0478 -0.7138
+vn 0.6988 0.0481 -0.7137
+vn -0.1991 -0.9260 -0.3209
+vn 0.6776 -0.2869 0.6772
+vn 0.7223 0.0291 -0.6909
+vn 0.7019 -0.0325 -0.7115
+vn 0.4454 0.7752 0.4480
+vn 0.2098 -0.9559 0.2055
+vn -0.5487 0.6316 -0.5478
+vn -0.6748 -0.2935 -0.6771
+vn 0.7027 0.1015 0.7042
+vn 0.7117 0.0187 -0.7022
+vn 0.7118 0.0187 -0.7022
+vn 0.7028 0.0143 -0.7113
+vn 0.5439 0.6370 0.5463
+vn -0.3364 0.8803 -0.3346
+vn 0.3374 -0.8803 0.3336
+vn 0.6972 0.0135 -0.7167
+vn 0.6969 0.0136 -0.7171
+vn 0.6970 0.0136 -0.7170
+vn -0.6211 -0.4744 -0.6239
+vn 0.0608 0.9955 0.0728
+vn 0.1948 -0.9785 0.0684
+vn 0.7023 0.0168 -0.7117
+vn 0.7049 0.0230 -0.7089
+vn -0.3406 0.8802 -0.3305
+vn 0.5408 0.6371 0.5493
+vn 0.6940 0.0214 -0.7196
+vn 0.6942 0.0212 -0.7194
+vn -0.3289 -0.8835 -0.3336
+vn -0.6780 0.2888 -0.6759
+vn 0.7088 -0.0005 -0.7054
+vn 0.7088 -0.0006 -0.7054
+vn 0.6755 -0.2964 0.6752
+vn -0.2005 -0.9579 -0.2053
+vn 0.3297 0.8836 0.3324
+vn 0.7064 0.0244 -0.7074
+vn -0.0620 -0.9955 -0.0718
+vn 0.5353 0.6370 0.5547
+vn 0.7116 0.0071 -0.7026
+vn 0.7098 0.0070 -0.7044
+vn 0.7079 0.0101 -0.7062
+vn 0.7061 0.0087 -0.7081
+vn 0.7087 0.0098 -0.7055
+vn 0.7120 0.0057 -0.7022
+vn 0.7057 0.0078 -0.7085
+vn 0.7072 0.0099 -0.7069
+vn 0.7097 0.0081 -0.7044
+vn 0.7064 0.0097 -0.7078
+vn 0.7093 0.0091 -0.7048
+vn 0.7060 0.0065 -0.7082
+vn 0.7063 0.0057 -0.7079
+vn 0.7097 0.0060 -0.7045
+vn 0.7071 -0.0002 -0.7072
+vn 0.7113 -0.0144 -0.7027
+vn 0.7111 -0.0144 -0.7030
+vn 0.7099 -0.0133 -0.7041
+vn 0.7097 -0.0134 -0.7044
+vn 0.4046 -0.9071 0.1165
+vn 0.7127 0.0099 -0.7014
+vn 0.7024 0.0100 -0.7117
+vn 0.7128 0.0133 -0.7012
+vn 0.7122 0.0163 -0.7018
+vn 0.7106 0.0191 -0.7033
+vn 0.7063 0.0206 -0.7077
+vn 0.7042 0.0188 -0.7097
+vn 0.7029 0.0159 -0.7111
+vn 0.7022 0.0132 -0.7119
+vn 0.0260 -0.9613 -0.2745
+vn 0.7034 0.0071 -0.7107
+vn 0.7059 0.0054 -0.7082
+vn 0.3281 0.8835 0.3342
+vn -0.1980 -0.9580 -0.2077
+vn 0.0637 0.9955 0.0701
+vn 0.1997 0.9580 0.2060
+vn 0.3396 -0.8802 0.3314
+vn 0.4539 -0.7708 0.4470
+vn -0.6197 -0.4744 -0.6252
+vn 0.6267 -0.4682 0.6229
+vn 0.6811 -0.2869 0.6737
+vn 0.7086 0.0203 -0.7053
+vn 0.6712 0.2935 0.6807
+vn 0.7091 0.0002 -0.7051
+vn 0.5427 0.6370 0.5474
+vn -0.4614 0.7707 -0.4396
+vn -0.6312 0.4683 -0.6184
+vn -0.6811 0.2868 -0.6737
+vn -0.5410 -0.6371 -0.5491
+vn -0.4419 -0.7752 -0.4515
+vn -0.2105 0.9559 -0.2048
+vn -0.3380 0.8803 -0.3329
+vn -0.6712 -0.2937 -0.6806
+vn -0.5498 0.6317 -0.5466
+vn -0.6785 0.2869 -0.6762
+vn 0.1888 0.9578 0.2169
+vn -0.6738 -0.2937 -0.6781
+vn -0.6193 -0.4744 -0.6256
+vn 0.6732 0.2995 0.6761
+vn 0.6270 -0.4683 0.6225
+vn 0.7091 0.0065 -0.7051
+vn 0.7059 0.0089 -0.7083
+vn 0.7085 0.0092 -0.7056
+vn 0.7091 0.0053 -0.7051
+vn 0.7054 0.0081 -0.7087
+vn 0.7071 0.0101 -0.7070
+vn 0.7094 0.0076 -0.7048
+vn 0.7066 0.0093 -0.7075
+vn 0.7055 0.0069 -0.7087
+vn 0.7055 0.0061 -0.7087
+vn 0.7063 0.0001 -0.7079
+vn 0.7042 -0.0144 -0.7098
+vn 0.2791 -0.9600 -0.0210
+vn 0.7136 0.0118 -0.7004
+vn 0.7123 0.0177 -0.7016
+vn 0.7063 0.0208 -0.7076
+vn 0.7042 0.0200 -0.7097
+vn 0.7025 0.0177 -0.7115
+vn 0.7015 0.0151 -0.7125
+vn 0.7021 0.0087 -0.7120
+vn 0.7033 0.0056 -0.7109
+vn -0.7024 -0.0991 -0.7048
+vn 0.7080 -0.0002 -0.7062
+vn 0.3180 0.8834 0.3441
+vn -0.6719 -0.3032 -0.6758
+vn -0.6766 0.2927 -0.6757
+vn 0.3404 -0.8802 0.3307
+vn 0.4546 -0.7708 0.4463
+vn 0.7116 0.0071 -0.7025
+vn 0.7097 0.0070 -0.7044
+vn 0.7060 0.0087 -0.7081
+vn 0.7087 0.0098 -0.7054
+vn 0.7120 0.0057 -0.7021
+vn 0.7060 0.0075 -0.7082
+vn 0.7072 0.0099 -0.7070
+vn 0.7062 0.0097 -0.7079
+vn 0.7091 0.0088 -0.7050
+vn 0.7058 0.0067 -0.7084
+vn 0.7094 0.0059 -0.7048
+vn 0.7071 -0.0003 -0.7072
+vn 0.7112 -0.0144 -0.7028
+vn 0.7096 -0.0134 -0.7044
+vn 0.4042 -0.9073 0.1162
+vn 0.7127 0.0100 -0.7014
+vn 0.7026 0.0096 -0.7115
+vn 0.7129 0.0133 -0.7012
+vn 0.7129 0.0134 -0.7011
+vn 0.7128 0.0130 -0.7013
+vn 0.7121 0.0164 -0.7019
+vn 0.7106 0.0187 -0.7034
+vn 0.7062 0.0206 -0.7077
+vn 0.7043 0.0188 -0.7097
+vn 0.7028 0.0163 -0.7112
+vn 0.7037 0.0069 -0.7105
+vn 0.7060 0.0053 -0.7082
+vn 0.7062 0.0053 -0.7080
+vn 0.7088 0.0002 -0.7054
+vn 0.3180 0.8834 0.3443
+vn -0.4614 0.7707 -0.4395
+vn 0.4547 -0.7708 0.4463
+vn -0.0649 -0.9955 -0.0690
+vn 0.7172 0.0170 -0.6967
+vn 0.7170 0.0169 -0.6969
+vn 0.7132 0.0173 -0.7008
+vn 0.7083 0.0245 -0.7055
+vn 0.7083 0.0247 -0.7054
+vn 0.7040 0.0206 -0.7099
+vn 0.7101 0.0238 -0.7037
+vn 0.7181 0.0138 -0.6958
+vn 0.7034 0.0184 -0.7106
+vn 0.7065 0.0244 -0.7073
+vn 0.7065 0.0241 -0.7073
+vn 0.7125 0.0195 -0.7014
+vn 0.7045 0.0238 -0.7093
+vn 0.7044 0.0241 -0.7094
+vn 0.7115 0.0215 -0.7024
+vn 0.7036 0.0159 -0.7104
+vn 0.7045 0.0137 -0.7096
+vn 0.7044 0.0137 -0.7097
+vn 0.7122 0.0144 -0.7018
+vn 0.7064 -0.0005 -0.7078
+vn 0.7157 -0.0347 -0.6975
+vn 0.7156 -0.0347 -0.6976
+vn 0.7123 -0.0323 -0.7011
+vn 0.3254 -0.9232 0.2043
+vn 0.7197 0.0239 -0.6939
+vn 0.6949 0.0238 -0.7187
+vn 0.6950 0.0237 -0.7186
+vn 0.7200 0.0320 -0.6932
+vn 0.7182 0.0394 -0.6947
+vn 0.7182 0.0391 -0.6948
+vn 0.7182 0.0396 -0.6947
+vn 0.7143 0.0455 -0.6983
+vn 0.7037 0.0496 -0.7087
+vn 0.6989 0.0459 -0.7137
+vn 0.6957 0.0394 -0.7173
+vn 0.6957 0.0393 -0.7173
+vn 0.6943 0.0316 -0.7190
+vn 0.6943 0.0314 -0.7190
+vn -0.0633 -0.9797 -0.1900
+vn 0.6980 0.0166 -0.7159
+vn 0.7035 0.0129 -0.7106
+vn -0.4511 0.7709 -0.4497
+vn 0.4453 0.7753 0.4480
+vn 0.3298 0.8836 0.3325
+vn 0.2016 0.9579 0.2043
+vn 0.3371 -0.8803 0.3340
+vn 0.4517 -0.7709 0.4492
+vn -0.6211 -0.4745 -0.6238
+vn 0.6786 -0.2871 0.6761
+vn 0.6192 0.4743 0.6258
+vn 0.7095 0.0493 -0.7029
+vn 0.6737 0.2935 0.6782
+vn 0.7107 0.0005 -0.7035
+vn 0.3255 0.8836 0.3366
+vn -0.5517 0.6315 -0.5448
+vn -0.6786 0.2869 -0.6761
+vn 0.5491 -0.6316 0.5473
+vn -0.0645 -0.9955 -0.0693
+vn -0.0731 0.9948 -0.0706
+vn -0.2088 0.9559 -0.2066
+vn -0.6737 -0.2936 -0.6781
+vn -0.6249 0.4683 -0.6246
+vn -0.6776 0.2868 -0.6772
+vn 0.6738 0.2995 0.6755
+vn -0.0777 0.9948 -0.0658
+vn 0.2095 -0.9559 0.2059
+vn 0.0742 -0.9948 0.0695
+vn 0.2097 -0.9559 0.2054
+vn 0.5492 -0.6319 0.5469
+vn 0.6254 -0.4684 0.6241
+vn -0.6776 0.2867 -0.6772
+vn -0.2009 -0.9579 -0.2051
+vn 0.7089 -0.0005 -0.7053
+vn 0.7090 -0.0005 -0.7052
+vn 0.7090 -0.0005 -0.7053
+vn 0.7160 -0.0347 -0.6972
+vn 0.7158 -0.0347 -0.6974
+vn 0.7159 -0.0347 -0.6973
+vn 0.0607 0.9955 0.0728
+vn 0.7093 0.0502 -0.7031
+vn 0.7105 0.0137 -0.7035
+vn 0.7116 0.0218 -0.7023
+vn 0.4418 0.7751 0.4518
+vn -0.0729 0.9948 -0.0704
+vn 0.6255 -0.4683 0.6241
+vn -0.5432 -0.6370 -0.5470
+vn 0.7034 0.0129 -0.7107
+vn 0.7021 0.0168 -0.7119
+vn -0.6272 0.4682 -0.6224
+vn 0.7225 0.0294 -0.6908
+vn 0.7226 0.0297 -0.6906
+vn 0.7224 0.0293 -0.6909
+vn 0.7102 0.0238 -0.7036
+vn 0.7096 0.0493 -0.7029
+vn -0.6780 0.2889 -0.6759
+vn 0.7097 0.0226 -0.7042
+vn 0.0609 0.9955 0.0730
+vn 0.4417 0.7752 0.4516
+vn 0.6979 0.0166 -0.7160
+vn 0.7020 -0.0325 -0.7115
+vn -0.2133 0.9559 -0.2020
+vn 0.0654 0.9955 0.0681
+vn -0.4549 0.7707 -0.4462
+vn 0.5493 -0.6317 0.5470
+vn 0.7034 0.0217 -0.7104
+vn 0.7035 0.0217 -0.7104
+vn 0.7215 0.0366 -0.6914
+vn -0.2131 0.9559 -0.2019
+vn 0.7118 0.0142 -0.7022
+vn 0.6990 0.0459 -0.7137
+vn 0.7033 0.0161 -0.7107
+vn 0.7031 0.0163 -0.7109
+vn 0.7208 0.0217 -0.6928
+vn 0.1944 -0.9786 0.0680
+vn 0.7083 0.0244 -0.7055
+vn 0.6947 0.0434 -0.7180
+vn 0.7050 0.0226 -0.7089
+vn 0.7050 0.0225 -0.7089
+vn 0.7049 0.0227 -0.7089
+vn 0.1970 0.9579 0.2088
+vn -0.4449 -0.7751 -0.4486
+vn 0.7082 0.0240 -0.7056
+vn 0.7039 0.0505 -0.7085
+vn 0.6957 0.0389 -0.7173
+vn 0.5408 0.6371 0.5492
+vn 0.6951 0.0237 -0.7186
+vn 0.7123 0.0195 -0.7016
+vn 0.7171 0.0171 -0.6967
+vn 0.0740 -0.9948 0.0693
+vn -0.7029 -0.0986 -0.7045
+vn -0.7029 -0.0987 -0.7044
+vn 0.6943 0.0315 -0.7190
+vn 0.6945 0.0312 -0.7188
+vn -0.7036 0.0943 -0.7043
+vn 0.5439 0.6369 0.5463
+vn -0.5486 0.6317 -0.5477
+vn 0.7202 0.0326 -0.6930
+vn 0.6776 -0.2870 0.6771
+vn 0.7126 -0.0323 -0.7009
+vn -0.3288 -0.8836 -0.3335
+vn 0.7130 0.0171 -0.7010
+vn 0.7182 0.0397 -0.6947
+vn 0.7115 0.0161 -0.7025
+vn 0.7113 0.0159 -0.7027
+vn 0.7112 0.0006 -0.7030
+vn 0.3254 -0.9232 0.2044
+vn 0.6738 0.2994 0.6755
+vn 0.7046 0.0237 -0.7093
+vn 0.6253 -0.4682 0.6243
+vn 0.6986 -0.0347 -0.7147
+vn -0.4445 -0.7752 -0.4489
+vn 0.7027 0.1016 0.7042
+vn 0.7027 0.0143 -0.7113
+vn 0.7026 0.0144 -0.7114
+vn 0.6941 0.0213 -0.7196
+vn 0.6735 0.2955 0.6775
+vn 0.7114 0.0131 -0.7027
+vn 0.4453 0.7753 0.4479
+vn 0.7023 0.0195 -0.7116
+vn -0.4448 -0.7752 -0.4485
+vn 0.6925 0.0283 -0.7208
+vn 0.7046 0.0136 -0.7094
+vn 0.6927 0.0359 -0.7204
+vn 0.7111 0.0210 -0.7028
+vn 0.7076 0.0044 -0.7066
+vn 0.7076 0.0040 -0.7066
+vn 0.7076 0.0045 -0.7066
+vn 0.7072 0.0042 -0.7070
+vn 0.7077 0.0044 -0.7065
+vn 0.7076 0.0046 -0.7066
+vn 0.7075 0.0043 -0.7067
+vn 0.7074 0.0044 -0.7068
+vn 0.7072 0.0043 -0.7070
+vn 0.7079 0.0038 -0.7063
+vn 0.7077 0.0038 -0.7065
+vn 0.7075 0.0041 -0.7067
+vn 0.7076 0.0039 -0.7066
+vn 0.7076 0.0041 -0.7066
+vn 0.7075 0.0040 -0.7067
+vn 0.7085 0.0042 -0.7057
+vn 0.7077 0.0043 -0.7065
+vn 0.7074 0.0109 -0.7067
+vn 0.7076 0.0097 -0.7065
+vn 0.7080 0.0092 -0.7062
+vn 0.7074 0.0107 -0.7068
+vn 0.7096 0.0102 -0.7045
+vn 0.7097 0.0102 -0.7045
+vn 0.7077 0.0095 -0.7064
+vn 0.7073 0.0102 -0.7068
+vn 0.7074 0.0099 -0.7067
+vn 0.7077 0.0109 -0.7065
+vn 0.7075 0.0095 -0.7066
+vn 0.7074 0.0092 -0.7068
+vn 0.7076 0.0098 -0.7065
+vn 0.7076 0.0099 -0.7065
+vn 0.7076 0.0096 -0.7066
+vn 0.7076 0.0106 -0.7065
+vn 0.7076 0.0107 -0.7066
+vn 0.7075 0.0109 -0.7066
+vn 0.7077 0.0104 -0.7064
+vn 0.7076 0.0109 -0.7065
+vn 0.7075 0.0096 -0.7066
+vn 0.7075 0.0110 -0.7066
+vn 0.7075 0.0107 -0.7066
+vn 0.7075 0.0108 -0.7066
+vn 0.7076 0.0108 -0.7065
+vn 0.7077 0.0108 -0.7065
+vn 0.7077 0.0093 -0.7064
+vn 0.7069 0.0100 -0.7073
+vn 0.7077 0.0093 -0.7065
+vn 0.7076 0.0107 -0.7065
+vn 0.7078 0.0107 -0.7063
+vn 0.7074 0.0104 -0.7067
+vn 0.7078 0.0104 -0.7063
+vn 0.7074 0.0041 -0.7068
+vn 0.7075 0.0047 -0.7067
+vn 0.7077 0.0045 -0.7065
+vn 0.7077 0.0040 -0.7065
+vn 0.7078 0.0044 -0.7064
+vn 0.7073 0.0047 -0.7069
+vn 0.7077 0.0039 -0.7065
+vn 0.7075 0.0044 -0.7067
+vn 0.7078 0.0038 -0.7064
+vn 0.7077 0.0046 -0.7065
+vn 0.7073 0.0037 -0.7069
+vn 0.7082 0.0042 -0.7060
+vn 0.7074 0.0041 -0.7069
+vn 0.7070 0.0099 -0.7072
+vn 0.7074 0.0103 -0.7067
+vn 0.7076 0.0096 -0.7065
+vn 0.7077 0.0095 -0.7065
+vn 0.7098 0.0103 -0.7043
+vn 0.7097 0.0102 -0.7044
+vn 0.7084 0.0089 -0.7058
+vn 0.7075 0.0107 -0.7067
+vn 0.7083 0.0104 -0.7058
+vn 0.7081 0.0106 -0.7061
+vn 0.7078 0.0096 -0.7063
+vn 0.7077 0.0096 -0.7064
+vn 0.7073 0.0097 -0.7068
+vn 0.7074 0.0096 -0.7067
+vn 0.7075 0.0097 -0.7066
+vn 0.7076 0.0093 -0.7065
+vn 0.7074 0.0095 -0.7067
+vn 0.7071 0.0099 -0.7070
+vn 0.7069 0.0100 -0.7072
+vn 0.7075 0.0096 -0.7067
+vt 0.357039 0.250000
+vt 0.785559 0.750000
+vt 0.214079 0.250000
+vt 0.714079 0.750000
+vt 0.285559 0.250000
+vt 0.571118 0.250000
+vt 0.285559 0.250000
+vt 0.642599 0.250000
+vt 0.928520 0.154203
+vt 0.499638 0.250000
+vt 0.714079 0.750000
+vt 0.571118 0.250000
+vt 0.642599 0.750000
+vt 0.000000 0.250000
+vt 0.857040 0.154203
+vt 0.071480 0.250000
+vt 0.857040 0.480583
+vt 0.357039 0.095797
+vt 0.785559 0.904203
+vt 0.785559 1.000000
+vt 0.357039 0.000000
+vt 0.785559 0.043154
+vt 0.714079 1.000000
+vt 0.214079 0.000000
+vt 0.857040 0.043154
+vt 0.928520 0.480583
+vt 0.214079 0.095797
+vt 0.714079 0.904203
+vt 0.285559 0.095797
+vt 0.571118 0.095797
+vt 0.285559 0.000000
+vt 0.571118 0.000000
+vt 0.857040 0.000000
+vt 0.285559 0.000000
+vt 0.642598 0.000000
+vt 0.785559 0.000000
+vt 0.285559 0.095797
+vt 0.642599 0.095797
+vt 0.857040 0.394495
+vt 0.928520 0.000000
+vt 0.499638 0.095797
+vt 0.499638 0.000000
+vt 0.785559 0.129242
+vt 0.714079 1.000000
+vt 0.571118 0.000000
+vt 0.785559 0.172396
+vt 0.714079 0.904203
+vt 0.571118 0.095797
+vt 0.642599 0.904203
+vt 0.000000 0.095797
+vt 0.642599 1.000000
+vt 0.000000 0.000000
+vt 0.857040 0.172396
+vt 0.071480 0.000000
+vt 0.857040 0.129242
+vt 0.928520 0.394495
+vt 0.857040 0.000000
+vt 0.071480 0.095797
+vt 0.578089 0.326242
+vt 0.370472 0.723575
+vt 0.442971 0.724510
+vt 0.846857 0.620495
+vt 0.579868 0.178984
+vt 0.229113 0.694984
+vt 0.442901 0.694984
+vt 0.817898 0.626255
+vt 0.589057 0.151745
+vt 0.229049 0.665457
+vt 0.442837 0.665456
+vt 0.788371 0.626255
+vt 0.605204 0.126737
+vt 0.228992 0.635930
+vt 0.442780 0.635930
+vt 0.759412 0.620495
+vt 0.626230 0.105756
+vt 0.228946 0.606403
+vt 0.442734 0.606402
+vt 0.732132 0.609195
+vt 0.650942 0.089526
+vt 0.228912 0.576877
+vt 0.442700 0.576876
+vt 0.707581 0.592792
+vt 0.678244 0.078632
+vt 0.228890 0.547350
+vt 0.442678 0.547349
+vt 0.686702 0.571913
+vt 0.228883 0.517823
+vt 0.706979 0.073452
+vt 0.442670 0.517822
+vt 0.670298 0.547363
+vt 0.228889 0.488295
+vt 0.735929 0.074111
+vt 0.442678 0.488295
+vt 0.658999 0.520082
+vt 0.763884 0.080481
+vt 0.228910 0.458769
+vt 0.442698 0.458768
+vt 0.653238 0.491123
+vt 0.789700 0.092175
+vt 0.228944 0.429242
+vt 0.442732 0.429241
+vt 0.653238 0.461596
+vt 0.812355 0.108585
+vt 0.228990 0.399716
+vt 0.442778 0.399714
+vt 0.658998 0.432637
+vt 0.831035 0.128981
+vt 0.229046 0.370189
+vt 0.442833 0.370188
+vt 0.670298 0.405358
+vt 0.844628 0.151667
+vt 0.229109 0.340661
+vt 0.442897 0.340661
+vt 0.686702 0.380807
+vt 0.879631 0.287703
+vt 0.370467 0.312068
+vt 0.442967 0.311134
+vt 0.707581 0.359928
+vt 0.370705 0.281246
+vt 0.885704 0.319257
+vt 0.443056 0.281620
+vt 0.732120 0.343530
+vt 0.371348 0.252525
+vt 0.885423 0.348954
+vt 0.443160 0.252098
+vt 0.759394 0.332230
+vt 0.879533 0.378395
+vt 0.371442 0.223476
+vt 0.443252 0.222558
+vt 0.788366 0.326464
+vt 0.867982 0.406513
+vt 0.371495 0.193949
+vt 0.443305 0.193031
+vt 0.817893 0.326463
+vt 0.851292 0.431762
+vt 0.371495 0.164423
+vt 0.443305 0.163503
+vt 0.846854 0.332224
+vt 0.830124 0.453224
+vt 0.371427 0.134897
+vt 0.443237 0.133976
+vt 0.874134 0.343522
+vt 0.805300 0.470133
+vt 0.371280 0.105372
+vt 0.443090 0.104449
+vt 0.898684 0.359926
+vt 0.777761 0.481894
+vt 0.371043 0.075847
+vt 0.442853 0.074922
+vt 0.919564 0.380805
+vt 0.748545 0.488099
+vt 0.370694 0.989321
+vt 0.370712 0.046322
+vt 0.442504 0.990249
+vt 0.442522 0.045396
+vt 0.935969 0.405357
+vt 0.718747 0.488548
+vt 0.371029 0.959797
+vt 0.442838 0.960722
+vt 0.947270 0.432638
+vt 0.689480 0.483245
+vt 0.371269 0.930272
+vt 0.443079 0.931196
+vt 0.953029 0.461597
+vt 0.661830 0.472412
+vt 0.371421 0.900745
+vt 0.443231 0.901668
+vt 0.953029 0.491125
+vt 0.371492 0.871219
+vt 0.636826 0.456466
+vt 0.443302 0.872141
+vt 0.947269 0.520085
+vt 0.615392 0.436018
+vt 0.371495 0.841694
+vt 0.443305 0.842613
+vt 0.935968 0.547365
+vt 0.598317 0.411848
+vt 0.371443 0.812167
+vt 0.443254 0.813086
+vt 0.919563 0.571916
+vt 0.371352 0.783117
+vt 0.586362 0.385333
+vt 0.443163 0.783546
+vt 0.898673 0.592803
+vt 0.370709 0.754396
+vt 0.579568 0.357437
+vt 0.443060 0.754024
+vt 0.874125 0.609201
+vt 0.571207 0.326153
+vt 0.177644 0.253134
+vt 0.006929 0.253167
+vt 0.319029 0.223358
+vt 0.573209 0.178195
+vt 0.007025 0.222313
+vt 0.582798 0.148853
+vt 0.319126 0.192504
+vt 0.007122 0.191458
+vt 0.599888 0.122511
+vt 0.319224 0.161649
+vt 0.007220 0.160604
+vt 0.621946 0.100560
+vt 0.319322 0.130795
+vt 0.007319 0.129750
+vt 0.647824 0.083609
+vt 0.319423 0.099941
+vt 0.007419 0.098896
+vt 0.676401 0.072253
+vt 0.319524 0.069087
+vt 0.007520 0.068042
+vt 0.706468 0.066881
+vt 0.319626 0.038233
+vt 0.007623 0.037188
+vt 0.319730 0.007378
+vt 0.736753 0.067631
+vt 0.319730 0.992485
+vt 0.007727 0.006334
+vt 0.007728 0.993666
+vt 0.765981 0.074365
+vt 0.319616 0.961631
+vt 0.007613 0.962812
+vt 0.792953 0.086670
+vt 0.319502 0.930776
+vt 0.007499 0.931958
+vt 0.816609 0.103895
+vt 0.319390 0.899923
+vt 0.007387 0.901104
+vt 0.836176 0.125250
+vt 0.319278 0.869069
+vt 0.007276 0.870250
+vt 0.850517 0.149568
+vt 0.319168 0.838215
+vt 0.007165 0.839396
+vt 0.886223 0.285996
+vt 0.177771 0.808498
+vt 0.007055 0.808541
+vt 0.177512 0.777645
+vt 0.892437 0.317355
+vt 0.006941 0.777687
+vt 0.176852 0.746796
+vt 0.892437 0.349226
+vt 0.006827 0.746834
+vt 0.886228 0.380492
+vt 0.176742 0.715941
+vt 0.006716 0.715979
+vt 0.874138 0.409852
+vt 0.176640 0.685087
+vt 0.006614 0.685125
+vt 0.856681 0.436209
+vt 0.176549 0.654233
+vt 0.006524 0.654271
+vt 0.834553 0.458608
+vt 0.176474 0.623379
+vt 0.006448 0.623417
+vt 0.808610 0.476251
+vt 0.176415 0.592525
+vt 0.006390 0.592562
+vt 0.779839 0.488518
+vt 0.176377 0.561671
+vt 0.006351 0.561708
+vt 0.749318 0.494983
+vt 0.176359 0.530816
+vt 0.006334 0.530854
+vt 0.176362 0.499962
+vt 0.718195 0.495436
+vt 0.006337 0.500000
+vt 0.687628 0.489885
+vt 0.176387 0.469108
+vt 0.006361 0.469146
+vt 0.658752 0.478560
+vt 0.176430 0.438254
+vt 0.006405 0.438291
+vt 0.632640 0.461897
+vt 0.176492 0.407399
+vt 0.006467 0.407437
+vt 0.610260 0.440534
+vt 0.176568 0.376545
+vt 0.006543 0.376583
+vt 0.592437 0.415287
+vt 0.176656 0.345691
+vt 0.006631 0.345729
+vt 0.176753 0.314837
+vt 0.579817 0.387126
+vt 0.006727 0.314875
+vt 0.177399 0.283987
+vt 0.572791 0.357181
+vt 0.006828 0.284020
+vt 0.578089 0.326242
+vt 0.370472 0.723575
+vt 0.442971 0.724510
+vt 0.231212 0.903091
+vt 0.579868 0.178984
+vt 0.229113 0.694984
+vt 0.442901 0.694984
+vt 0.214996 0.906316
+vt 0.589057 0.151745
+vt 0.229049 0.665457
+vt 0.442837 0.665456
+vt 0.198460 0.906317
+vt 0.605204 0.126737
+vt 0.228992 0.635930
+vt 0.442780 0.635930
+vt 0.182243 0.903091
+vt 0.626230 0.105756
+vt 0.228946 0.606403
+vt 0.442734 0.606402
+vt 0.166967 0.896763
+vt 0.650942 0.089526
+vt 0.228912 0.576877
+vt 0.442700 0.576876
+vt 0.153218 0.887577
+vt 0.678244 0.078632
+vt 0.228890 0.547350
+vt 0.442678 0.547349
+vt 0.141526 0.875885
+vt 0.228883 0.517823
+vt 0.706979 0.073452
+vt 0.442670 0.517822
+vt 0.132340 0.862137
+vt 0.228889 0.488295
+vt 0.735929 0.074111
+vt 0.442678 0.488295
+vt 0.126012 0.846860
+vt 0.763884 0.080481
+vt 0.228910 0.458769
+vt 0.442698 0.458768
+vt 0.122786 0.830643
+vt 0.789700 0.092175
+vt 0.228944 0.429242
+vt 0.442732 0.429241
+vt 0.122786 0.814107
+vt 0.812355 0.108585
+vt 0.228990 0.399716
+vt 0.442778 0.399714
+vt 0.126012 0.797890
+vt 0.831035 0.128981
+vt 0.229046 0.370189
+vt 0.442833 0.370188
+vt 0.132339 0.782614
+vt 0.844628 0.151667
+vt 0.229109 0.340661
+vt 0.442897 0.340661
+vt 0.141526 0.768865
+vt 0.879631 0.287703
+vt 0.370467 0.312068
+vt 0.442967 0.311134
+vt 0.153218 0.757173
+vt 0.370705 0.281246
+vt 0.885704 0.319257
+vt 0.443056 0.281620
+vt 0.166960 0.747990
+vt 0.371348 0.252525
+vt 0.885423 0.348954
+vt 0.443160 0.252098
+vt 0.182233 0.741662
+vt 0.879533 0.378395
+vt 0.371442 0.223476
+vt 0.443252 0.222558
+vt 0.198458 0.738433
+vt 0.867982 0.406513
+vt 0.371495 0.193949
+vt 0.443305 0.193031
+vt 0.214993 0.738433
+vt 0.851292 0.431762
+vt 0.371495 0.164423
+vt 0.443305 0.163503
+vt 0.231211 0.741659
+vt 0.830124 0.453224
+vt 0.371427 0.134897
+vt 0.443237 0.133976
+vt 0.246487 0.747986
+vt 0.805300 0.470133
+vt 0.371280 0.105372
+vt 0.443090 0.104449
+vt 0.260236 0.757172
+vt 0.777761 0.481894
+vt 0.371043 0.075847
+vt 0.442853 0.074922
+vt 0.271928 0.768865
+vt 0.748545 0.488099
+vt 0.370694 0.989321
+vt 0.370712 0.046322
+vt 0.442504 0.990249
+vt 0.442522 0.045396
+vt 0.281116 0.782613
+vt 0.718747 0.488548
+vt 0.371029 0.959797
+vt 0.442838 0.960722
+vt 0.287444 0.797891
+vt 0.689480 0.483245
+vt 0.371269 0.930272
+vt 0.443079 0.931196
+vt 0.290669 0.814108
+vt 0.661830 0.472412
+vt 0.371421 0.900745
+vt 0.443231 0.901668
+vt 0.290669 0.830644
+vt 0.371492 0.871219
+vt 0.636826 0.456466
+vt 0.443302 0.872141
+vt 0.287443 0.846861
+vt 0.615392 0.436018
+vt 0.371495 0.841694
+vt 0.443305 0.842613
+vt 0.281115 0.862138
+vt 0.598317 0.411848
+vt 0.371443 0.812167
+vt 0.443254 0.813086
+vt 0.271928 0.875886
+vt 0.371352 0.783117
+vt 0.586362 0.385333
+vt 0.443163 0.783546
+vt 0.260230 0.887583
+vt 0.370709 0.754396
+vt 0.579568 0.357437
+vt 0.443060 0.754024
+vt 0.246483 0.896766
+vt 0.571207 0.326153
+vt 0.177644 0.253134
+vt 0.006929 0.253167
+vt 0.319029 0.223358
+vt 0.573209 0.178195
+vt 0.007025 0.222313
+vt 0.582798 0.148853
+vt 0.319126 0.192504
+vt 0.007122 0.191458
+vt 0.599888 0.122511
+vt 0.319224 0.161649
+vt 0.007220 0.160604
+vt 0.621946 0.100560
+vt 0.319322 0.130795
+vt 0.007319 0.129750
+vt 0.647824 0.083609
+vt 0.319423 0.099941
+vt 0.007419 0.098896
+vt 0.676401 0.072253
+vt 0.319524 0.069087
+vt 0.007520 0.068042
+vt 0.706468 0.066881
+vt 0.319626 0.038233
+vt 0.007623 0.037188
+vt 0.319730 0.007378
+vt 0.736753 0.067631
+vt 0.319730 0.992485
+vt 0.007727 0.006334
+vt 0.007728 0.993666
+vt 0.765981 0.074365
+vt 0.319616 0.961631
+vt 0.007613 0.962812
+vt 0.792953 0.086670
+vt 0.319502 0.930776
+vt 0.007499 0.931958
+vt 0.816609 0.103895
+vt 0.319390 0.899923
+vt 0.007387 0.901104
+vt 0.836176 0.125250
+vt 0.319278 0.869069
+vt 0.007276 0.870250
+vt 0.850517 0.149568
+vt 0.319168 0.838215
+vt 0.007165 0.839396
+vt 0.886223 0.285996
+vt 0.177771 0.808498
+vt 0.007055 0.808541
+vt 0.177512 0.777645
+vt 0.892437 0.317355
+vt 0.006941 0.777687
+vt 0.176852 0.746796
+vt 0.892437 0.349226
+vt 0.006827 0.746834
+vt 0.886228 0.380492
+vt 0.176742 0.715941
+vt 0.006716 0.715979
+vt 0.874138 0.409852
+vt 0.176640 0.685087
+vt 0.006614 0.685125
+vt 0.856681 0.436209
+vt 0.176549 0.654233
+vt 0.006524 0.654271
+vt 0.834553 0.458608
+vt 0.176474 0.623379
+vt 0.006448 0.623417
+vt 0.808610 0.476251
+vt 0.176415 0.592525
+vt 0.006390 0.592562
+vt 0.779839 0.488518
+vt 0.176377 0.561671
+vt 0.006351 0.561708
+vt 0.749318 0.494983
+vt 0.176359 0.530816
+vt 0.006334 0.530854
+vt 0.176362 0.499962
+vt 0.718195 0.495436
+vt 0.006337 0.500000
+vt 0.687628 0.489885
+vt 0.176387 0.469108
+vt 0.006361 0.469146
+vt 0.658752 0.478560
+vt 0.176430 0.438254
+vt 0.006405 0.438291
+vt 0.632640 0.461897
+vt 0.176492 0.407399
+vt 0.006467 0.407437
+vt 0.610260 0.440534
+vt 0.176568 0.376545
+vt 0.006543 0.376583
+vt 0.592437 0.415287
+vt 0.176656 0.345691
+vt 0.006631 0.345729
+vt 0.176753 0.314837
+vt 0.579817 0.387126
+vt 0.006727 0.314875
+vt 0.177399 0.283987
+vt 0.572791 0.357181
+vt 0.006828 0.284020
+vt 0.578089 0.326242
+vt 0.370472 0.723575
+vt 0.442971 0.724510
+vt 0.278498 0.614635
+vt 0.579868 0.178984
+vt 0.229113 0.694984
+vt 0.442901 0.694984
+vt 0.249539 0.620395
+vt 0.589057 0.151745
+vt 0.229049 0.665457
+vt 0.442837 0.665456
+vt 0.220011 0.620396
+vt 0.605204 0.126737
+vt 0.228992 0.635930
+vt 0.442780 0.635930
+vt 0.191052 0.614636
+vt 0.626230 0.105756
+vt 0.228946 0.606403
+vt 0.442734 0.606402
+vt 0.163773 0.603336
+vt 0.650942 0.089526
+vt 0.228912 0.576877
+vt 0.442700 0.576876
+vt 0.139222 0.586933
+vt 0.678244 0.078632
+vt 0.228890 0.547350
+vt 0.442678 0.547349
+vt 0.118343 0.566054
+vt 0.228883 0.517823
+vt 0.706979 0.073452
+vt 0.442670 0.517822
+vt 0.101939 0.541503
+vt 0.228889 0.488295
+vt 0.735929 0.074111
+vt 0.442678 0.488295
+vt 0.090640 0.514223
+vt 0.763884 0.080481
+vt 0.228910 0.458769
+vt 0.442698 0.458768
+vt 0.084879 0.485264
+vt 0.789700 0.092175
+vt 0.228944 0.429242
+vt 0.442732 0.429241
+vt 0.084879 0.455737
+vt 0.812355 0.108585
+vt 0.228990 0.399716
+vt 0.442778 0.399714
+vt 0.090639 0.426777
+vt 0.831035 0.128981
+vt 0.229046 0.370189
+vt 0.442833 0.370188
+vt 0.101938 0.399499
+vt 0.844628 0.151667
+vt 0.229109 0.340661
+vt 0.442897 0.340661
+vt 0.118343 0.374948
+vt 0.879631 0.287703
+vt 0.370467 0.312068
+vt 0.442967 0.311134
+vt 0.139222 0.354069
+vt 0.370705 0.281246
+vt 0.885704 0.319257
+vt 0.443056 0.281620
+vt 0.163760 0.337671
+vt 0.371348 0.252525
+vt 0.885423 0.348954
+vt 0.443160 0.252098
+vt 0.191035 0.326370
+vt 0.879533 0.378395
+vt 0.371442 0.223476
+vt 0.443252 0.222558
+vt 0.220007 0.320605
+vt 0.867982 0.406513
+vt 0.371495 0.193949
+vt 0.443305 0.193031
+vt 0.249534 0.320604
+vt 0.851292 0.431762
+vt 0.371495 0.164423
+vt 0.443305 0.163503
+vt 0.278494 0.326364
+vt 0.830124 0.453224
+vt 0.371427 0.134897
+vt 0.443237 0.133976
+vt 0.305774 0.337663
+vt 0.805300 0.470133
+vt 0.371280 0.105372
+vt 0.443090 0.104449
+vt 0.330325 0.354067
+vt 0.777761 0.481894
+vt 0.371043 0.075847
+vt 0.442853 0.074922
+vt 0.351205 0.374946
+vt 0.748545 0.488099
+vt 0.370694 0.989321
+vt 0.370712 0.046322
+vt 0.442504 0.990249
+vt 0.442522 0.045396
+vt 0.367610 0.399497
+vt 0.718747 0.488548
+vt 0.371029 0.959797
+vt 0.442838 0.960722
+vt 0.378910 0.426778
+vt 0.689480 0.483245
+vt 0.371269 0.930272
+vt 0.443079 0.931196
+vt 0.384670 0.455738
+vt 0.661830 0.472412
+vt 0.371421 0.900745
+vt 0.443231 0.901668
+vt 0.384670 0.485266
+vt 0.371492 0.871219
+vt 0.636826 0.456466
+vt 0.443302 0.872141
+vt 0.378909 0.514226
+vt 0.615392 0.436018
+vt 0.371495 0.841694
+vt 0.443305 0.842613
+vt 0.367609 0.541505
+vt 0.598317 0.411848
+vt 0.371443 0.812167
+vt 0.443254 0.813086
+vt 0.351204 0.566056
+vt 0.371352 0.783117
+vt 0.586362 0.385333
+vt 0.443163 0.783546
+vt 0.330314 0.586943
+vt 0.370709 0.754396
+vt 0.579568 0.357437
+vt 0.443060 0.754024
+vt 0.305766 0.603342
+vt 0.571207 0.326153
+vt 0.177644 0.253134
+vt 0.006929 0.253167
+vt 0.319029 0.223358
+vt 0.573209 0.178195
+vt 0.007025 0.222313
+vt 0.582798 0.148853
+vt 0.319126 0.192504
+vt 0.007122 0.191458
+vt 0.599888 0.122511
+vt 0.319224 0.161649
+vt 0.007220 0.160604
+vt 0.621946 0.100560
+vt 0.319322 0.130795
+vt 0.007319 0.129750
+vt 0.647824 0.083609
+vt 0.319423 0.099941
+vt 0.007419 0.098896
+vt 0.676401 0.072253
+vt 0.319524 0.069087
+vt 0.007520 0.068042
+vt 0.706468 0.066881
+vt 0.319626 0.038233
+vt 0.007623 0.037188
+vt 0.319730 0.007378
+vt 0.736753 0.067631
+vt 0.319730 0.992485
+vt 0.007727 0.006334
+vt 0.007728 0.993666
+vt 0.765981 0.074365
+vt 0.319616 0.961631
+vt 0.007613 0.962812
+vt 0.792953 0.086670
+vt 0.319502 0.930776
+vt 0.007499 0.931958
+vt 0.816609 0.103895
+vt 0.319390 0.899923
+vt 0.007387 0.901104
+vt 0.836176 0.125250
+vt 0.319278 0.869069
+vt 0.007276 0.870250
+vt 0.850517 0.149568
+vt 0.319168 0.838215
+vt 0.007165 0.839396
+vt 0.886223 0.285996
+vt 0.177771 0.808498
+vt 0.007055 0.808541
+vt 0.177512 0.777645
+vt 0.892437 0.317355
+vt 0.006941 0.777687
+vt 0.176852 0.746796
+vt 0.892437 0.349226
+vt 0.006827 0.746834
+vt 0.886228 0.380492
+vt 0.176742 0.715941
+vt 0.006716 0.715979
+vt 0.874138 0.409852
+vt 0.176640 0.685087
+vt 0.006614 0.685125
+vt 0.856681 0.436209
+vt 0.176549 0.654233
+vt 0.006524 0.654271
+vt 0.834553 0.458608
+vt 0.176474 0.623379
+vt 0.006448 0.623417
+vt 0.808610 0.476251
+vt 0.176415 0.592525
+vt 0.006390 0.592562
+vt 0.779839 0.488518
+vt 0.176377 0.561671
+vt 0.006351 0.561708
+vt 0.749318 0.494983
+vt 0.176359 0.530816
+vt 0.006334 0.530854
+vt 0.176362 0.499962
+vt 0.718195 0.495436
+vt 0.006337 0.500000
+vt 0.687628 0.489885
+vt 0.176387 0.469108
+vt 0.006361 0.469146
+vt 0.658752 0.478560
+vt 0.176430 0.438254
+vt 0.006405 0.438291
+vt 0.632640 0.461897
+vt 0.176492 0.407399
+vt 0.006467 0.407437
+vt 0.610260 0.440534
+vt 0.176568 0.376545
+vt 0.006543 0.376583
+vt 0.592437 0.415287
+vt 0.176656 0.345691
+vt 0.006631 0.345729
+vt 0.176753 0.314837
+vt 0.579817 0.387126
+vt 0.006727 0.314875
+vt 0.177399 0.283987
+vt 0.572791 0.357181
+vt 0.006828 0.284020
+vt 0.006828 0.284020
+vt 0.572791 0.357181
+vt 0.177399 0.283987
+vt 0.006727 0.314875
+vt 0.176753 0.314837
+vt 0.579817 0.387126
+vt 0.006631 0.345729
+vt 0.176656 0.345691
+vt 0.592437 0.415287
+vt 0.006543 0.376583
+vt 0.610260 0.440534
+vt 0.176568 0.376545
+vt 0.006467 0.407437
+vt 0.632640 0.461897
+vt 0.176492 0.407399
+vt 0.006405 0.438291
+vt 0.176430 0.438254
+vt 0.658752 0.478560
+vt 0.006361 0.469146
+vt 0.176387 0.469108
+vt 0.687628 0.489885
+vt 0.006337 0.500000
+vt 0.718195 0.495436
+vt 0.176362 0.499962
+vt 0.006334 0.530854
+vt 0.749318 0.494983
+vt 0.176359 0.530816
+vt 0.006351 0.561708
+vt 0.779839 0.488518
+vt 0.176377 0.561671
+vt 0.006390 0.592562
+vt 0.808610 0.476251
+vt 0.176415 0.592525
+vt 0.006448 0.623417
+vt 0.176474 0.623379
+vt 0.834553 0.458608
+vt 0.006524 0.654271
+vt 0.176549 0.654233
+vt 0.856681 0.436209
+vt 0.006614 0.685125
+vt 0.176640 0.685087
+vt 0.874138 0.409852
+vt 0.006716 0.715979
+vt 0.886228 0.380492
+vt 0.176742 0.715941
+vt 0.006827 0.746834
+vt 0.892437 0.349226
+vt 0.176852 0.746796
+vt 0.006941 0.777687
+vt 0.892437 0.317355
+vt 0.177512 0.777645
+vt 0.007055 0.808541
+vt 0.886223 0.285996
+vt 0.177771 0.808498
+vt 0.007165 0.839396
+vt 0.850517 0.149568
+vt 0.319168 0.838215
+vt 0.007276 0.870250
+vt 0.836176 0.125250
+vt 0.319278 0.869069
+vt 0.007387 0.901104
+vt 0.816609 0.103895
+vt 0.319390 0.899923
+vt 0.007499 0.931958
+vt 0.319502 0.930776
+vt 0.792953 0.086670
+vt 0.007613 0.962812
+vt 0.319616 0.961631
+vt 0.765981 0.074365
+vt 0.007728 0.993666
+vt 0.007727 0.006334
+vt 0.319730 0.007378
+vt 0.736753 0.067631
+vt 0.319730 0.992485
+vt 0.007623 0.037188
+vt 0.319626 0.038233
+vt 0.706468 0.066881
+vt 0.007520 0.068042
+vt 0.676401 0.072253
+vt 0.319524 0.069087
+vt 0.007419 0.098896
+vt 0.319423 0.099941
+vt 0.647824 0.083609
+vt 0.007319 0.129750
+vt 0.319322 0.130795
+vt 0.621946 0.100560
+vt 0.007220 0.160604
+vt 0.599888 0.122511
+vt 0.319224 0.161649
+vt 0.007122 0.191458
+vt 0.319126 0.192504
+vt 0.582798 0.148853
+vt 0.007025 0.222313
+vt 0.319029 0.223358
+vt 0.573209 0.178195
+vt 0.006929 0.253167
+vt 0.177644 0.253134
+vt 0.571207 0.326153
+vt 0.443060 0.754024
+vt 0.208826 0.204735
+vt 0.579568 0.357437
+vt 0.370709 0.754396
+vt 0.443163 0.783546
+vt 0.222573 0.195552
+vt 0.586362 0.385333
+vt 0.371352 0.783117
+vt 0.443254 0.813086
+vt 0.234272 0.183855
+vt 0.598317 0.411848
+vt 0.371443 0.812167
+vt 0.443305 0.842613
+vt 0.243459 0.170107
+vt 0.615392 0.436018
+vt 0.371495 0.841694
+vt 0.443302 0.872141
+vt 0.249787 0.154830
+vt 0.636826 0.456466
+vt 0.371492 0.871219
+vt 0.443231 0.901668
+vt 0.253013 0.138613
+vt 0.371421 0.900745
+vt 0.661830 0.472412
+vt 0.443079 0.931196
+vt 0.253013 0.122077
+vt 0.371269 0.930272
+vt 0.689480 0.483245
+vt 0.442838 0.960722
+vt 0.249787 0.105860
+vt 0.718747 0.488548
+vt 0.371029 0.959797
+vt 0.442504 0.990249
+vt 0.442522 0.045396
+vt 0.243459 0.090582
+vt 0.370694 0.989321
+vt 0.748545 0.488099
+vt 0.370712 0.046322
+vt 0.442853 0.074922
+vt 0.234272 0.076833
+vt 0.777761 0.481894
+vt 0.371043 0.075847
+vt 0.443090 0.104449
+vt 0.222579 0.065141
+vt 0.805300 0.470133
+vt 0.371280 0.105372
+vt 0.443237 0.133976
+vt 0.208831 0.055955
+vt 0.371427 0.134897
+vt 0.830124 0.453224
+vt 0.443305 0.163503
+vt 0.193554 0.049628
+vt 0.371495 0.164423
+vt 0.851292 0.431762
+vt 0.443305 0.193031
+vt 0.177337 0.046402
+vt 0.867982 0.406513
+vt 0.371495 0.193949
+vt 0.443252 0.222558
+vt 0.160801 0.046402
+vt 0.879533 0.378395
+vt 0.371442 0.223476
+vt 0.443160 0.252098
+vt 0.144577 0.049631
+vt 0.371348 0.252525
+vt 0.885423 0.348954
+vt 0.443056 0.281620
+vt 0.129303 0.055959
+vt 0.370705 0.281246
+vt 0.885704 0.319257
+vt 0.442967 0.311134
+vt 0.115562 0.065142
+vt 0.370467 0.312068
+vt 0.879631 0.287703
+vt 0.442897 0.340661
+vt 0.103870 0.076834
+vt 0.229109 0.340661
+vt 0.844628 0.151667
+vt 0.442833 0.370188
+vt 0.094683 0.090583
+vt 0.229046 0.370189
+vt 0.831035 0.128981
+vt 0.442778 0.399714
+vt 0.088355 0.105859
+vt 0.812355 0.108585
+vt 0.228990 0.399716
+vt 0.442732 0.429241
+vt 0.085130 0.122076
+vt 0.228944 0.429242
+vt 0.789700 0.092175
+vt 0.442698 0.458768
+vt 0.085130 0.138611
+vt 0.763884 0.080481
+vt 0.228910 0.458769
+vt 0.442678 0.488295
+vt 0.088356 0.154828
+vt 0.228889 0.488295
+vt 0.735929 0.074111
+vt 0.442670 0.517822
+vt 0.094683 0.170106
+vt 0.706979 0.073452
+vt 0.228883 0.517823
+vt 0.442678 0.547349
+vt 0.103870 0.183854
+vt 0.678244 0.078632
+vt 0.228890 0.547350
+vt 0.442700 0.576876
+vt 0.115562 0.195546
+vt 0.228912 0.576877
+vt 0.650942 0.089526
+vt 0.442734 0.606402
+vt 0.129310 0.204732
+vt 0.228946 0.606403
+vt 0.626230 0.105756
+vt 0.442780 0.635930
+vt 0.144587 0.211060
+vt 0.605204 0.126737
+vt 0.228992 0.635930
+vt 0.442837 0.665456
+vt 0.160804 0.214285
+vt 0.589057 0.151745
+vt 0.229049 0.665457
+vt 0.442901 0.694984
+vt 0.177339 0.214285
+vt 0.229113 0.694984
+vt 0.579868 0.178984
+vt 0.442971 0.724510
+vt 0.193556 0.211059
+vt 0.578089 0.326242
+vt 0.370472 0.723575
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.846034 0.501950
+vt 0.307543 0.501950
+vt 0.769051 0.501950
+vt 0.307542 0.501950
+vt 0.000000 0.501950
+vt 0.692068 0.501950
+vt 0.615085 0.501950
+vt 0.846034 0.996502
+vt 0.384526 0.501950
+vt 0.538102 0.501950
+vt 0.769051 0.501950
+vt 0.923017 0.996502
+vt 0.230559 0.501950
+vt 0.076983 0.501950
+vt 0.692068 0.501950
+vt 0.692068 0.192341
+vt 0.923017 0.650215
+vt 0.076983 0.192341
+vt 0.692068 0.000000
+vt 0.076983 0.000000
+vt 0.846034 0.429417
+vt 0.769051 0.000000
+vt 0.538102 0.000000
+vt 0.923017 0.429417
+vt 0.769051 0.192341
+vt 0.538102 0.192341
+vt 1.000000 0.650215
+vt 0.615085 0.192341
+vt 0.615085 0.000000
+vt 0.923017 0.343447
+vt 0.000000 0.000000
+vt 0.692068 0.000000
+vt 0.846034 0.343447
+vt 0.000000 0.192341
+vt 0.692068 0.192341
+vt 0.923017 0.686893
+vt 0.923017 0.478709
+vt 0.230559 0.192341
+vt 0.230559 0.000000
+vt 0.846034 0.600923
+vt 0.769051 0.000000
+vt 0.307542 0.000000
+vt 0.846034 0.686893
+vt 0.769051 0.192341
+vt 0.307542 0.192341
+vt 0.846034 0.192341
+vt 0.307542 0.192341
+vt 0.846034 0.000000
+vt 0.307542 0.000000
+vt 0.923017 0.686893
+vt 0.384525 0.000000
+vt 0.923017 0.600923
+vt 0.846034 0.686893
+vt 0.384525 0.192341
+vt 1.000000 0.478709
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.846034 1.000000
+vt 0.307543 1.000000
+vt 0.769051 1.000000
+vt 0.307542 1.000000
+vt 0.000000 1.000000
+vt 0.769051 1.000000
+vt 0.615085 1.000000
+vt 0.692068 1.000000
+vt 0.923017 0.307203
+vt 0.384526 1.000000
+vt 0.538102 1.000000
+vt 0.615085 0.501950
+vt 1.000000 0.307203
+vt 0.230559 1.000000
+vt 0.076983 1.000000
+vt 0.692068 0.501950
+vt 0.692068 0.809153
+vt 0.923017 0.478709
+vt 0.076983 0.692797
+vt 0.692068 1.000000
+vt 0.076983 0.501950
+vt 0.846034 0.085970
+vt 0.615085 1.000000
+vt 0.538102 0.501950
+vt 0.923017 0.085970
+vt 0.615085 0.809153
+vt 0.538102 0.692796
+vt 1.000000 0.478709
+vt 0.615085 0.692796
+vt 0.692068 0.692796
+vt 0.615085 0.501950
+vt 0.692068 0.501950
+vt 0.923017 0.000000
+vt 0.000000 0.501950
+vt 0.769051 0.501950
+vt 0.846034 0.000000
+vt 0.000000 0.692797
+vt 0.769051 0.692796
+vt 1.000000 0.000000
+vt 0.923017 0.307203
+vt 0.230559 0.692797
+vt 0.230559 0.501950
+vt 0.846034 0.257476
+vt 0.769051 0.501950
+vt 0.307542 0.501950
+vt 0.846034 0.343447
+vt 0.769051 0.692796
+vt 0.307542 0.692796
+vt 0.846034 0.692796
+vt 0.307542 0.692797
+vt 0.846034 0.501950
+vt 0.307542 0.501950
+vt 0.923017 0.343447
+vt 0.384525 0.501950
+vt 0.923017 0.257476
+vt 0.923017 0.000000
+vt 0.384525 0.692797
+vt 1.000000 0.307203
+vt 0.875462 0.748915
+vt 0.958487 0.000000
+vt 1.000000 1.000000
+vt 0.437731 0.748915
+vt 1.000000 0.500000
+vt 0.875462 0.000000
+vt 0.875462 0.000000
+vt 0.958487 0.748915
+vt 0.958487 0.500000
+vt 0.437731 0.000000
+vt 0.958487 0.000000
+vt 0.875462 0.748915
+vt 0.437731 0.748915
+vt 0.916975 0.748915
+vt 1.000000 0.500000
+vt 0.437731 0.000000
+vt 0.916975 0.000000
+vt 0.958487 1.000000
+vt 0.000000 0.000000
+vt 0.958487 0.500000
+vt 0.916974 0.000000
+vt 0.000000 0.748915
+vt 1.000000 0.000000
+vt 0.916975 0.748915
+s 0
+g KsHauptsignal_Plane.006_Mast
+usemtl Mast
+f 13/29/1 12/27/1 11/24/1 14/31/1
+f 9/18/2 16/37/2 15/34/2 10/21/2
+f 16/38/3 13/30/3 14/32/3 15/35/3
+f 12/28/4 9/19/4 10/20/4 11/23/4
+f 21/49/4 20/47/4 19/44/4 22/51/4
+f 24/58/1 21/50/1 22/52/1 23/54/1
+f 20/48/2 17/41/2 18/42/2 19/45/2
+f 24/58/1 23/54/1 11/24/1 12/27/1
+f 17/39/5 24/56/5 12/26/5 9/17/5
+f 18/42/2 17/41/2 9/18/2 10/21/2
+f 23/55/6 18/43/6 10/22/6 11/25/6
+f 19/46/6 18/43/6 23/55/6 22/53/6
+f 11/25/6 10/22/6 15/36/6 14/33/6
+f 3/5/1 2/3/1 12/27/1 13/29/1
+f 1/1/2 4/7/2 16/37/2 9/18/2
+f 4/8/3 3/6/3 13/30/3 16/38/3
+f 2/4/4 1/2/4 9/19/4 12/28/4
+f 7/13/4 6/11/4 20/47/4 21/49/4
+f 5/9/3 8/15/3 24/57/3 17/40/3
+f 8/16/1 7/14/1 21/50/1 24/58/1
+f 6/12/2 5/10/2 17/41/2 20/48/2
+f 544/974/6 541/973/6 537/971/6 538/972/6
+f 548/981/1 550/984/1 556/1000/1 557/1002/1
+f 557/1002/1 556/1000/1 555/997/1 558/1003/1
+f 552/988/2 547/979/2 560/1008/2 553/992/2
+f 553/992/2 560/1008/2 559/1005/2 554/994/2
+f 547/980/3 548/981/3 557/1002/3 560/1009/3
+f 560/1009/3 557/1002/3 558/1003/3 559/1006/3
+f 550/985/4 552/989/4 553/990/4 556/999/4
+f 556/999/4 553/990/4 554/993/4 555/996/4
+f 545/975/4 546/977/4 564/1018/4 565/1020/4
+f 565/1020/4 564/1018/4 563/1015/4 566/1022/4
+f 551/986/3 549/982/3 568/1027/3 561/1010/3
+f 549/983/1 545/976/1 565/1021/1 568/1028/1
+f 568/1028/1 565/1021/1 566/1023/1 567/1025/1
+f 546/978/2 551/987/2 561/1012/2 564/1019/2
+f 564/1019/2 561/1012/2 562/1013/2 563/1016/2
+f 568/1028/1 567/1025/1 555/997/1 556/1000/1
+f 561/1011/5 568/1029/5 556/1001/5 553/991/5
+f 562/1013/2 561/1012/2 553/992/2 554/994/2
+f 567/1026/6 562/1014/6 554/995/6 555/998/6
+f 563/1017/6 562/1014/6 567/1026/6 566/1024/6
+f 555/998/6 554/995/6 559/1007/6 558/1004/6
+f 576/1033/6 573/1032/6 569/1030/6 570/1031/6
+f 580/1040/1 582/1044/1 588/1060/1 589/1062/1
+f 589/1062/1 588/1060/1 587/1057/1 590/1064/1
+f 584/1048/2 579/1038/2 592/1070/2 585/1052/2
+f 585/1052/2 592/1070/2 591/1067/2 586/1054/2
+f 579/1039/3 580/1041/3 589/1063/3 592/1071/3
+f 592/1071/3 589/1063/3 590/1065/3 591/1068/3
+f 582/1045/4 584/1049/4 585/1050/4 588/1059/4
+f 588/1059/4 585/1050/4 586/1053/4 587/1056/4
+f 577/1034/4 578/1036/4 596/1080/4 597/1082/4
+f 597/1082/4 596/1080/4 595/1077/4 598/1084/4
+f 583/1046/3 581/1042/3 600/1089/3 593/1072/3
+f 581/1043/1 577/1035/1 597/1083/1 600/1090/1
+f 600/1090/1 597/1083/1 598/1085/1 599/1087/1
+f 578/1037/2 583/1047/2 593/1074/2 596/1081/2
+f 596/1081/2 593/1074/2 594/1075/2 595/1078/2
+f 600/1090/1 599/1087/1 587/1057/1 588/1060/1
+f 593/1073/5 600/1091/5 588/1061/5 585/1051/5
+f 594/1075/2 593/1074/2 585/1052/2 586/1054/2
+f 599/1088/6 594/1076/6 586/1055/6 587/1058/6
+f 595/1079/6 594/1076/6 599/1088/6 598/1086/6
+f 587/1058/6 586/1055/6 591/1069/6 590/1066/6
+g KsHauptsignal_Plane.006_Schirm
+usemtl Schirm
+f 601/1092/3 602/1095/3 604/1101/3 603/1098/3
+f 606/1107/4 605/1104/4 608/1113/4 607/1110/4
+f 603/1099/2 605/1105/2 606/1108/2 601/1093/2
+f 604/1102/6 608/1114/6 605/1106/6 603/1100/6
+f 601/1094/5 606/1109/5 607/1111/5 602/1096/5
+f 602/1097/1 607/1112/1 608/1115/1 604/1103/1
+g KsHauptsignal_Plane.006_Lampenrahmen
+usemtl Lampenrahmen
+f 40/89/7 39/87/7 41/91/7
+f 26/61/8 88/187/8 25/60/8
+f 54/117/9 53/116/9 56/121/9
+f 58/125/10 56/121/10 57/123/10
+f 86/183/11 85/181/11 88/187/11
+f 90/191/12 89/190/12 152/286/12
+f 118/235/13 120/238/13 117/234/13
+f 122/241/14 121/239/14 120/238/14
+f 150/283/15 152/286/15 149/281/15
+f 79/169/16 81/174/16 82/175/16
+f 81/173/17 145/275/17 147/278/17
+f 111/224/18 113/227/18 49/107/18
+f 103/210/19 105/214/19 41/92/19
+f 99/204/20 35/79/20 33/75/21
+f 105/214/22 107/218/22 43/95/22
+f 147/278/23 149/282/23 85/182/23
+f 97/201/24 33/75/24 31/71/24
+f 103/210/25 39/88/25 37/83/25
+f 109/221/26 111/224/26 47/103/26
+f 35/79/27 99/204/27 101/207/27
+f 107/218/28 109/221/28 45/99/28
+f 95/198/29 31/71/29 29/67/29
+f 93/195/30 29/67/30 27/63/30
+f 113/227/31 115/230/31 51/111/31
+f 151/285/32 89/189/32 25/59/32
+f 149/282/33 151/285/33 87/186/33
+f 121/240/34 57/124/34 55/120/34
+f 115/230/35 117/233/35 53/115/35
+f 79/170/36 143/272/36 145/275/36
+f 125/245/37 127/248/37 63/135/37
+f 77/165/38 141/269/38 143/272/38
+f 75/161/39 139/266/39 141/269/39
+f 73/157/40 137/264/40 139/266/40
+f 133/257/41 135/260/41 71/151/41
+f 131/254/42 133/257/42 69/147/42
+f 129/251/43 131/254/43 67/143/43
+f 127/248/44 129/251/44 65/139/44
+f 91/193/45 27/63/45 25/59/45
+f 123/242/46 125/245/46 61/131/46
+f 123/242/47 59/127/47 57/124/47
+f 88/187/48 87/185/48 25/60/48
+f 97/202/49 95/199/49 96/200/49
+f 112/226/50 111/225/50 109/222/50
+f 48/105/51 47/104/51 49/108/51
+f 110/223/52 109/222/52 107/219/52
+f 140/268/53 139/267/53 137/263/53
+f 42/93/54 41/91/54 43/96/54
+f 106/216/55 105/213/55 103/211/55
+f 37/84/56 39/87/56 40/89/56
+f 108/220/57 107/219/57 105/215/57
+f 33/76/58 35/80/58 36/81/58
+f 31/72/59 33/76/59 34/77/59
+f 50/109/60 49/108/60 51/112/60
+f 27/64/61 29/68/61 30/69/61
+f 25/60/62 27/64/62 28/65/62
+f 53/116/63 55/119/63 56/121/63
+f 56/121/64 55/119/64 57/123/64
+f 81/174/65 83/178/65 84/179/65
+f 71/151/66 135/260/66 137/264/66
+f 83/178/67 85/181/67 86/183/67
+f 119/237/68 55/120/68 53/115/68
+f 77/166/69 79/169/69 80/171/69
+f 75/162/70 77/166/70 78/167/70
+f 114/229/71 113/228/71 111/225/71
+f 142/271/72 141/270/72 139/267/72
+f 68/145/73 67/144/73 69/148/73
+f 66/141/74 65/140/74 67/144/74
+f 64/137/75 63/136/75 65/140/75
+f 62/133/76 61/132/76 63/136/76
+f 60/129/77 59/128/77 61/132/77
+f 58/125/78 57/123/78 59/128/78
+f 85/181/79 87/185/79 88/187/79
+f 89/190/80 151/284/80 152/286/80
+f 146/277/81 145/276/81 143/273/81
+f 144/274/82 143/273/82 141/270/82
+f 116/232/83 115/231/83 113/228/83
+f 71/152/84 73/158/84 74/159/84
+f 29/68/85 31/72/85 32/73/85
+f 44/97/86 43/96/86 45/100/86
+f 138/265/87 137/263/87 135/261/87
+f 103/211/88 101/208/88 102/209/88
+f 101/208/89 99/205/89 100/206/89
+f 99/205/90 97/202/90 98/203/90
+f 52/113/91 51/112/91 53/116/91
+f 95/199/92 93/196/92 94/197/92
+f 93/196/93 91/192/93 92/194/93
+f 92/194/94 91/192/94 89/190/94
+f 120/238/95 119/236/95 117/234/95
+f 121/239/14 119/236/14 120/238/14
+f 73/158/96 75/162/96 76/163/96
+f 150/283/97 149/281/97 147/279/97
+f 148/280/98 147/279/98 145/276/98
+f 116/232/99 118/235/99 117/234/99
+f 70/149/100 69/148/100 71/153/100
+f 46/101/101 45/100/101 47/104/101
+f 35/80/102 37/84/102 38/85/102
+f 135/261/103 133/258/103 134/259/103
+f 133/258/104 131/255/104 132/256/104
+f 131/255/105 129/252/105 130/253/105
+f 129/252/106 127/249/106 128/250/106
+f 127/249/107 125/246/107 126/247/107
+f 125/246/108 123/243/108 124/244/108
+f 123/243/109 121/239/109 122/241/109
+f 152/286/110 151/284/110 149/281/110
+f 209/401/111 273/503/111 275/506/111
+f 276/508/112 275/507/112 273/504/112
+f 211/406/113 213/409/113 214/411/113
+f 249/467/114 247/464/114 248/466/114
+f 244/460/115 243/459/115 241/456/115
+f 272/502/116 271/501/116 269/498/116
+f 180/341/117 179/340/117 181/344/117
+f 229/436/118 227/433/118 228/434/118
+f 225/430/119 223/427/119 224/428/119
+f 255/476/120 257/479/120 193/367/120
+f 234/444/121 233/441/121 231/439/121
+f 157/296/122 159/300/122 160/301/122
+f 182/345/110 181/344/110 184/349/110
+f 165/312/123 167/315/123 168/317/123
+f 214/411/124 213/409/124 216/415/124
+f 221/424/125 219/420/125 220/422/125
+f 216/415/126 215/413/126 153/288/126
+f 227/432/127 163/307/128 161/303/128
+f 235/446/129 237/449/129 173/327/129
+f 201/385/130 265/492/130 267/494/130
+f 261/485/131 263/488/131 199/379/131
+f 184/349/132 183/347/132 185/351/132
+f 159/300/133 161/304/133 162/305/133
+f 168/317/134 167/315/134 169/319/134
+f 275/506/135 277/510/135 213/410/135
+f 266/493/136 265/491/136 263/489/136
+f 249/468/137 185/352/137 183/348/137
+f 231/438/138 233/442/138 169/320/138
+f 190/361/139 189/360/139 191/364/139
+f 217/418/140 279/512/140 280/514/140
+f 255/477/141 253/474/141 254/475/141
+f 186/353/142 184/349/142 185/351/142
+f 203/390/143 205/394/143 206/395/143
+f 239/452/144 241/455/144 177/335/144
+f 225/429/145 161/303/145 159/299/145
+f 172/325/146 171/324/146 173/328/146
+f 174/329/147 173/328/147 175/332/147
+f 199/379/148 263/488/148 265/492/149
+f 253/473/150 255/476/150 191/363/150
+f 163/308/151 165/312/151 166/313/151
+f 263/489/152 261/486/152 262/487/152
+f 241/455/153 243/458/153 179/339/153
+f 201/386/154 203/390/154 204/391/154
+f 198/377/155 197/376/155 199/381/155
+f 209/402/156 211/406/156 212/407/156
+f 188/357/157 187/356/157 189/360/157
+f 233/442/158 235/446/158 171/323/158
+f 196/373/159 195/372/159 197/376/159
+f 257/479/160 259/482/161 195/371/161
+f 278/511/162 280/514/162 277/509/162
+f 236/448/163 235/447/163 233/443/163
+f 250/469/114 249/467/114 248/466/114
+f 253/474/164 251/471/164 252/472/164
+f 205/393/165 269/497/165 271/500/165
+f 213/409/166 215/413/166 216/415/166
+f 154/289/167 216/415/167 153/288/167
+f 218/419/140 217/418/140 280/514/140
+f 279/513/168 217/417/168 153/287/168
+f 176/333/169 175/332/169 177/336/169
+f 181/344/170 183/347/170 184/349/170
+f 161/304/171 163/308/171 164/309/171
+f 280/514/172 279/512/172 277/509/172
+f 257/480/173 255/477/173 256/478/173
+f 274/505/174 273/504/174 271/501/174
+f 155/292/175 157/296/175 158/297/175
+f 231/439/176 229/436/176 230/437/176
+f 205/394/177 207/397/177 208/399/177
+f 220/422/178 219/420/178 217/418/178
+f 203/389/179 267/494/179 269/497/179
+f 192/365/180 191/364/180 193/368/180
+f 244/460/181 246/463/181 245/462/181
+f 170/321/182 169/319/182 171/324/182
+f 259/482/183 261/485/184 197/375/183
+f 219/421/185 155/291/185 153/287/185
+f 251/471/186 249/467/186 250/469/186
+f 207/398/187 271/500/187 273/503/187
+f 277/510/188 279/513/188 215/414/188
+f 240/454/189 239/453/189 237/450/189
+f 261/486/190 259/483/190 260/484/190
+f 223/427/191 221/424/191 222/425/191
+f 278/511/192 277/509/192 275/507/192
+f 248/466/193 247/464/193 245/462/193
+f 237/449/194 239/452/194 175/331/195
+f 246/463/193 248/466/193 245/462/193
+f 221/423/196 157/295/196 155/291/196
+f 242/457/197 241/456/197 239/453/197
+f 227/433/198 225/430/198 226/431/198
+f 259/483/199 257/480/199 258/481/199
+f 251/470/200 187/355/201 185/352/202
+f 178/337/203 177/336/203 179/340/203
+f 199/380/204 201/386/204 202/387/204
+f 243/458/205 245/461/205 181/343/205
+f 223/426/206 159/299/206 157/295/206
+f 163/307/207 227/432/207 229/435/207
+f 194/369/208 193/368/208 195/372/208
+f 207/397/209 209/402/209 210/403/209
+f 251/470/210 253/473/211 189/359/210
+f 270/499/212 269/498/212 267/495/212
+f 186/353/213 185/351/213 187/356/213
+f 247/465/214 183/348/214 181/343/215
+f 153/288/216 155/292/216 156/293/216
+f 268/496/217 267/495/217 265/491/217
+f 238/451/218 237/450/218 235/447/218
+f 231/438/219 167/316/219 165/311/219
+f 42/93/220 40/89/220 41/91/220
+f 80/171/221 79/169/221 82/175/221
+f 83/177/222 81/173/222 147/278/222
+f 47/103/223 111/224/223 49/107/223
+f 39/88/224 103/210/224 41/92/224
+f 97/201/225 99/204/225 33/75/225
+f 41/92/226 105/214/226 43/95/226
+f 83/177/227 147/278/227 85/182/227
+f 95/198/228 97/201/228 31/71/228
+f 101/207/229 103/210/229 37/83/229
+f 45/99/230 109/221/230 47/103/230
+f 37/83/231 35/79/231 101/207/231
+f 43/95/232 107/218/232 45/99/232
+f 93/195/233 95/198/233 29/67/233
+f 91/193/234 93/195/234 27/63/234
+f 49/107/235 113/227/235 51/111/235
+f 87/186/236 151/285/236 25/59/236
+f 85/182/237 149/282/237 87/186/238
+f 119/237/239 121/240/239 55/120/240
+f 51/111/241 115/230/241 53/115/241
+f 81/173/242 79/170/242 145/275/242
+f 61/131/243 125/245/243 63/135/243
+f 79/170/244 77/165/244 143/272/244
+f 77/165/245 75/161/245 141/269/245
+f 75/161/246 73/157/246 139/266/246
+f 69/147/247 133/257/247 71/151/247
+f 67/143/248 131/254/248 69/147/248
+f 65/139/249 129/251/249 67/143/249
+f 63/135/250 127/248/250 65/139/250
+f 89/189/251 91/193/251 25/59/251
+f 59/127/252 123/242/252 61/131/252
+f 121/240/253 123/242/253 57/124/253
+f 98/203/49 97/202/49 96/200/49
+f 110/223/50 112/226/50 109/222/50
+f 50/109/51 48/105/51 49/108/51
+f 108/220/254 110/223/254 107/219/254
+f 138/265/53 140/268/53 137/263/53
+f 44/97/255 42/93/255 43/96/255
+f 104/212/256 106/216/256 103/211/256
+f 38/85/56 37/84/56 40/89/56
+f 106/217/257 108/220/257 105/215/257
+f 34/77/258 33/76/258 36/81/258
+f 32/73/259 31/72/259 34/77/259
+f 52/113/260 50/109/260 51/112/260
+f 28/65/261 27/64/261 30/69/261
+f 26/61/262 25/60/262 28/65/262
+f 82/175/65 81/174/65 84/179/65
+f 73/157/263 71/151/263 137/264/263
+f 84/179/264 83/178/264 86/183/264
+f 117/233/265 119/237/265 53/115/265
+f 78/167/69 77/166/69 80/171/69
+f 76/163/70 75/162/70 78/167/70
+f 112/226/266 114/229/266 111/225/266
+f 140/268/72 142/271/72 139/267/72
+f 70/149/73 68/145/73 69/148/73
+f 68/145/74 66/141/74 67/144/74
+f 66/141/267 64/137/267 65/140/267
+f 64/137/76 62/133/76 63/136/76
+f 62/133/268 60/129/268 61/132/268
+f 60/129/269 58/125/269 59/128/269
+f 144/274/270 146/277/270 143/273/270
+f 142/271/271 144/274/271 141/270/271
+f 114/229/83 116/232/83 113/228/83
+f 72/154/84 71/152/84 74/159/84
+f 30/69/85 29/68/85 32/73/85
+f 46/101/86 44/97/86 45/100/86
+f 136/262/87 138/265/87 135/261/87
+f 104/212/88 103/211/88 102/209/88
+f 102/209/272 101/208/272 100/206/272
+f 100/206/273 99/205/273 98/203/273
+f 54/117/274 52/113/274 53/116/274
+f 96/200/275 95/199/275 94/197/275
+f 94/197/93 93/196/93 92/194/93
+f 90/191/276 92/194/276 89/190/276
+f 74/159/277 73/158/277 76/163/277
+f 148/280/278 150/283/278 147/279/278
+f 146/277/279 148/280/279 145/276/279
+f 115/231/280 116/232/280 117/234/280
+f 72/155/100 70/149/100 71/153/100
+f 48/105/101 46/101/101 47/104/101
+f 36/81/102 35/80/102 38/85/102
+f 136/262/103 135/261/103 134/259/103
+f 134/259/104 133/258/104 132/256/104
+f 132/256/105 131/255/105 130/253/105
+f 130/253/106 129/252/106 128/250/106
+f 128/250/107 127/249/107 126/247/107
+f 126/247/281 125/246/281 124/244/281
+f 124/244/109 123/243/109 122/241/109
+f 296/545/7 295/543/7 297/547/7
+f 282/517/8 344/643/8 281/516/8
+f 310/573/9 309/572/9 312/577/9
+f 314/581/10 312/577/10 313/579/10
+f 342/639/11 341/637/11 344/643/11
+f 346/647/12 345/646/12 408/742/12
+f 374/691/13 376/694/13 373/690/13
+f 378/697/14 377/695/14 376/694/14
+f 406/739/15 408/742/15 405/737/15
+f 335/625/16 337/630/16 338/631/16
+f 337/629/17 401/731/17 403/734/17
+f 367/680/282 369/683/282 305/563/282
+f 359/666/19 361/670/19 297/548/19
+f 355/660/283 291/535/283 289/531/283
+f 361/670/284 363/674/284 299/551/284
+f 403/734/285 405/738/285 341/638/285
+f 353/657/286 289/531/286 287/527/286
+f 359/666/287 295/544/287 293/539/287
+f 365/677/288 367/680/288 303/559/288
+f 291/535/289 355/660/289 357/663/289
+f 363/674/28 365/677/28 301/555/28
+f 351/654/290 287/527/290 285/523/290
+f 349/651/291 285/523/291 283/519/291
+f 369/683/31 371/686/31 307/567/31
+f 407/741/292 345/645/292 281/515/292
+f 405/738/33 407/741/33 343/642/33
+f 377/696/293 313/580/293 311/576/293
+f 371/686/294 373/689/294 309/571/294
+f 335/626/295 399/728/295 401/731/295
+f 381/701/37 383/704/37 319/591/37
+f 333/621/38 397/725/38 399/728/38
+f 331/617/296 395/722/296 397/725/296
+f 329/613/40 393/720/40 395/722/40
+f 389/713/297 391/716/297 327/607/297
+f 387/710/298 389/713/298 325/603/298
+f 385/707/299 387/710/299 323/599/299
+f 383/704/300 385/707/300 321/595/300
+f 347/649/45 283/519/45 281/515/45
+f 379/698/301 381/701/301 317/587/301
+f 379/698/302 315/583/302 313/580/302
+f 344/643/48 343/641/48 281/516/48
+f 353/658/49 351/655/49 352/656/49
+f 368/682/50 367/681/50 365/678/50
+f 304/561/51 303/560/51 305/564/51
+f 366/679/52 365/678/52 363/675/52
+f 396/724/53 395/723/53 393/719/53
+f 298/549/54 297/547/54 299/552/54
+f 362/672/55 361/669/55 359/667/55
+f 293/540/56 295/543/56 296/545/56
+f 364/676/57 363/675/57 361/671/57
+f 289/532/58 291/536/58 292/537/58
+f 287/528/59 289/532/59 290/533/59
+f 306/565/60 305/564/60 307/568/60
+f 283/520/61 285/524/61 286/525/61
+f 281/516/62 283/520/62 284/521/62
+f 309/572/303 311/575/303 312/577/303
+f 312/577/64 311/575/64 313/579/64
+f 337/630/65 339/634/65 340/635/65
+f 327/607/66 391/716/66 393/720/66
+f 339/634/67 341/637/67 342/639/67
+f 375/693/304 311/576/304 309/571/304
+f 333/622/69 335/625/69 336/627/69
+f 331/618/305 333/622/305 334/623/305
+f 370/685/71 369/684/71 367/681/71
+f 398/727/72 397/726/72 395/723/72
+f 324/601/73 323/600/73 325/604/73
+f 322/597/74 321/596/74 323/600/74
+f 320/593/75 319/592/75 321/596/75
+f 318/589/76 317/588/76 319/592/76
+f 316/585/77 315/584/77 317/588/77
+f 314/581/78 313/579/78 315/584/78
+f 341/637/79 343/641/79 344/643/79
+f 345/646/80 407/740/80 408/742/80
+f 402/733/81 401/732/81 399/729/81
+f 400/730/82 399/729/82 397/726/82
+f 372/688/83 371/687/83 369/684/83
+f 327/608/84 329/614/84 330/615/84
+f 285/524/85 287/528/85 288/529/85
+f 300/553/86 299/552/86 301/556/86
+f 394/721/87 393/719/87 391/717/87
+f 359/667/88 357/664/88 358/665/88
+f 357/664/89 355/661/89 356/662/89
+f 355/661/90 353/658/90 354/659/90
+f 308/569/306 307/568/306 309/572/306
+f 351/655/92 349/652/92 350/653/92
+f 349/652/93 347/648/93 348/650/93
+f 348/650/307 347/648/307 345/646/307
+f 376/694/95 375/692/95 373/690/95
+f 377/695/14 375/692/14 376/694/14
+f 329/614/96 331/618/96 332/619/96
+f 406/739/97 405/737/97 403/735/97
+f 404/736/98 403/735/98 401/732/98
+f 372/688/99 374/691/99 373/690/99
+f 326/605/100 325/604/100 327/609/100
+f 302/557/101 301/556/101 303/560/101
+f 291/536/102 293/540/102 294/541/102
+f 391/717/103 389/714/103 390/715/103
+f 389/714/104 387/711/104 388/712/104
+f 387/711/308 385/708/308 386/709/308
+f 385/708/309 383/705/309 384/706/309
+f 383/705/107 381/702/107 382/703/107
+f 381/702/108 379/699/108 380/700/108
+f 379/699/109 377/695/109 378/697/109
+f 408/742/110 407/740/110 405/737/110
+f 298/549/220 296/545/220 297/547/220
+f 336/627/221 335/625/221 338/631/221
+f 339/633/310 337/629/310 403/734/310
+f 303/559/311 367/680/311 305/563/311
+f 295/544/224 359/666/224 297/548/224
+f 353/657/312 355/660/312 289/531/312
+f 297/548/313 361/670/313 299/551/313
+f 339/633/314 403/734/314 341/638/314
+f 351/654/315 353/657/315 287/527/315
+f 357/663/316 359/666/316 293/539/316
+f 301/555/230 365/677/230 303/559/230
+f 293/539/317 291/535/317 357/663/317
+f 299/551/318 363/674/318 301/555/318
+f 349/651/319 351/654/319 285/523/319
+f 347/649/234 349/651/234 283/519/234
+f 305/563/320 369/683/320 307/567/320
+f 343/642/321 407/741/321 281/515/321
+f 341/638/322 405/738/322 343/642/322
+f 375/693/323 377/696/323 311/576/323
+f 307/567/324 371/686/324 309/571/324
+f 337/629/325 335/626/325 401/731/325
+f 317/587/326 381/701/326 319/591/326
+f 335/626/327 333/621/328 399/728/329
+f 333/621/330 331/617/330 397/725/330
+f 331/617/331 329/613/331 395/722/331
+f 325/603/332 389/713/332 327/607/332
+f 323/599/333 387/710/333 325/603/333
+f 321/595/334 385/707/334 323/599/334
+f 319/591/250 383/704/250 321/595/250
+f 345/645/251 347/649/251 281/515/251
+f 315/583/335 379/698/335 317/587/335
+f 377/696/336 379/698/337 313/580/337
+f 354/659/49 353/658/49 352/656/49
+f 366/679/50 368/682/50 365/678/50
+f 306/565/51 304/561/51 305/564/51
+f 364/676/254 366/679/254 363/675/254
+f 394/721/53 396/724/53 393/719/53
+f 300/553/255 298/549/255 299/552/255
+f 360/668/256 362/672/256 359/667/256
+f 294/541/56 293/540/56 296/545/56
+f 362/673/257 364/676/257 361/671/257
+f 290/533/258 289/532/258 292/537/258
+f 288/529/259 287/528/259 290/533/259
+f 308/569/260 306/565/260 307/568/260
+f 284/521/261 283/520/261 286/525/261
+f 282/517/262 281/516/262 284/521/262
+f 338/631/65 337/630/65 340/635/65
+f 329/613/263 327/607/263 393/720/263
+f 340/635/264 339/634/264 342/639/264
+f 373/689/338 375/693/338 309/571/338
+f 334/623/69 333/622/69 336/627/69
+f 332/619/339 331/618/339 334/623/339
+f 368/682/266 370/685/266 367/681/266
+f 396/724/72 398/727/72 395/723/72
+f 326/605/73 324/601/73 325/604/73
+f 324/601/74 322/597/74 323/600/74
+f 322/597/340 320/593/340 321/596/340
+f 320/593/76 318/589/76 319/592/76
+f 318/589/268 316/585/268 317/588/268
+f 316/585/269 314/581/269 315/584/269
+f 400/730/270 402/733/270 399/729/270
+f 398/727/271 400/730/271 397/726/271
+f 370/685/83 372/688/83 369/684/83
+f 328/610/84 327/608/84 330/615/84
+f 286/525/85 285/524/85 288/529/85
+f 302/557/86 300/553/86 301/556/86
+f 392/718/87 394/721/87 391/717/87
+f 360/668/88 359/667/88 358/665/88
+f 358/665/272 357/664/272 356/662/272
+f 356/662/273 355/661/273 354/659/273
+f 310/573/274 308/569/274 309/572/274
+f 352/656/275 351/655/275 350/653/275
+f 350/653/93 349/652/93 348/650/93
+f 346/647/276 348/650/276 345/646/276
+f 330/615/96 329/614/96 332/619/96
+f 404/736/278 406/739/278 403/735/278
+f 402/733/279 404/736/279 401/732/279
+f 371/687/280 372/688/280 373/690/280
+f 328/611/100 326/605/100 327/609/100
+f 304/561/101 302/557/101 303/560/101
+f 292/537/102 291/536/102 294/541/102
+f 392/718/103 391/717/103 390/715/103
+f 390/715/104 389/714/104 388/712/104
+f 388/712/105 387/711/105 386/709/105
+f 386/709/341 385/708/341 384/706/341
+f 384/706/107 383/705/107 382/703/107
+f 382/703/281 381/702/281 380/700/281
+f 380/700/109 379/699/109 378/697/109
+f 170/321/342 168/317/342 169/319/342
+f 208/399/209 207/397/209 210/403/209
+f 211/405/343 209/401/344 275/506/343
+f 175/331/345 239/452/345 177/335/345
+f 167/316/346 231/438/347 169/320/347
+f 225/429/348 227/432/348 161/303/348
+f 169/320/349 233/442/349 171/323/349
+f 211/405/350 275/506/350 213/410/350
+f 223/426/351 225/429/351 159/299/351
+f 229/435/352 231/438/352 165/311/353
+f 173/327/354 237/449/354 175/331/354
+f 165/311/355 163/307/356 229/435/356
+f 171/323/357 235/446/357 173/327/357
+f 221/423/358 223/426/358 157/295/358
+f 219/421/359 221/423/360 155/291/359
+f 177/335/361 241/455/361 179/339/361
+f 215/414/362 279/513/362 153/287/362
+f 213/410/363 277/510/363 215/414/364
+f 247/465/365 249/468/365 183/348/365
+f 179/339/366 243/458/366 181/343/366
+f 209/401/367 207/398/367 273/503/367
+f 189/359/368 253/473/369 191/363/368
+f 207/398/370 205/393/370 271/500/370
+f 205/393/371 203/389/372 269/497/373
+f 203/389/374 201/385/374 267/494/374
+f 197/375/375 261/485/375 199/379/375
+f 195/371/376 259/482/376 197/375/376
+f 193/367/377 257/479/378 195/371/378
+f 191/363/379 255/476/379 193/367/380
+f 217/417/381 219/421/381 153/287/381
+f 187/355/382 251/470/382 189/359/382
+f 249/468/383 251/470/383 185/352/383
+f 226/431/384 225/430/384 224/428/384
+f 238/451/385 240/454/385 237/450/385
+f 178/337/169 176/333/169 177/336/169
+f 236/448/386 238/451/386 235/447/386
+f 266/493/217 268/496/217 265/491/217
+f 172/325/182 170/321/182 171/324/182
+f 232/440/121 234/444/121 231/439/121
+f 166/313/123 165/312/123 168/317/123
+f 234/445/387 236/448/387 233/443/387
+f 162/305/388 161/304/388 164/309/388
+f 160/301/389 159/300/389 162/305/389
+f 180/341/390 178/337/390 179/340/390
+f 156/293/175 155/292/175 158/297/175
+f 154/289/391 153/288/391 156/293/391
+f 210/403/392 209/402/392 212/407/392
+f 201/385/393 199/379/393 265/492/393
+f 212/407/394 211/406/394 214/411/394
+f 245/461/395 247/465/395 181/343/395
+f 206/395/177 205/394/177 208/399/177
+f 204/391/396 203/390/396 206/395/396
+f 240/454/197 242/457/197 239/453/197
+f 268/496/212 270/499/212 267/495/212
+f 198/377/159 196/373/159 197/376/159
+f 196/373/208 194/369/208 195/372/208
+f 194/369/180 192/365/180 193/368/180
+f 192/365/397 190/361/397 191/364/397
+f 190/361/157 188/357/157 189/360/157
+f 188/357/398 186/353/398 187/356/398
+f 272/502/174 274/505/174 271/501/174
+f 270/499/116 272/502/116 269/498/116
+f 242/457/115 244/460/115 241/456/115
+f 200/382/204 199/380/204 202/387/204
+f 158/297/399 157/296/399 160/301/399
+f 174/329/146 172/325/146 173/328/146
+f 264/490/400 266/493/400 263/489/400
+f 232/440/401 231/439/401 230/437/401
+f 230/437/402 229/436/402 228/434/402
+f 228/434/198 227/433/198 226/431/198
+f 182/345/403 180/341/403 181/344/403
+f 224/428/191 223/427/191 222/425/191
+f 222/425/404 221/424/404 220/422/404
+f 218/419/405 220/422/405 217/418/405
+f 202/387/154 201/386/154 204/391/154
+f 276/508/192 278/511/192 275/507/192
+f 274/505/112 276/508/112 273/504/112
+f 243/459/406 244/460/406 245/462/406
+f 200/383/407 198/377/407 199/381/407
+f 176/333/147 174/329/147 175/332/147
+f 164/309/408 163/308/408 166/313/408
+f 264/490/409 263/489/409 262/487/409
+f 262/487/410 261/486/410 260/484/410
+f 260/484/199 259/483/199 258/481/199
+f 258/481/173 257/480/173 256/478/173
+f 256/478/411 255/477/411 254/475/411
+f 254/475/412 253/474/412 252/472/412
+f 252/472/186 251/471/186 250/469/186
+f 471/838/413 469/835/413 472/839/413
+f 519/935/414 520/937/414 518/934/414
+f 519/935/134 521/939/134 520/937/134
+f 413/749/112 414/750/112 416/754/112
+f 486/868/374 488/871/374 422/763/374
+f 476/848/166 474/844/166 473/841/166
+f 535/967/167 473/841/167 536/970/167
+f 521/939/134 522/942/134 520/937/134
+f 440/790/114 442/793/114 441/791/114
+f 442/792/415 506/910/416 508/914/417
+f 476/847/418 412/748/419 474/843/420
+f 490/876/421 488/872/421 487/869/421
+f 490/877/422 426/768/422 424/765/422
+f 448/801/423 446/798/153 510/918/423
+f 462/824/198 464/827/198 463/826/198
+f 486/867/143 484/863/143 483/861/143
+f 461/823/198 462/824/198 463/826/198
+f 518/933/424 454/811/424 516/930/424
+f 483/861/425 484/863/425 481/857/425
+f 511/919/169 513/923/169 512/921/169
+f 458/818/426 460/822/426 459/820/426
+f 430/774/184 428/771/184 492/881/184
+f 436/783/164 438/787/164 437/785/164
+f 480/856/156 478/852/156 477/849/156
+f 435/782/427 436/783/427 437/785/427
+f 417/755/428 415/752/428 418/757/428
+f 440/789/429 438/786/429 504/906/429
+f 466/830/430 530/957/430 532/961/430
+f 499/895/431 501/899/431 500/898/431
+f 482/859/432 418/756/433 416/753/434
+f 520/938/435 456/815/435 518/933/435
+f 531/959/122 532/962/122 529/955/122
+f 421/761/212 419/758/212 422/762/212
+f 488/871/436 490/877/436 424/765/436
+f 453/809/163 454/810/163 456/816/163
+f 460/822/402 462/824/402 461/823/402
+f 478/851/350 414/751/350 476/847/350
+f 523/943/123 524/946/123 521/939/123
+f 503/903/437 504/905/437 502/902/437
+f 456/815/158 454/811/158 518/933/438
+f 489/873/439 490/876/439 487/869/439
+f 468/833/404 470/836/404 469/835/404
+f 464/828/348 462/825/348 528/954/348
+f 484/863/440 482/860/440 481/857/440
+f 480/855/367 482/859/367 416/753/367
+f 515/927/146 517/931/146 516/929/146
+f 453/809/218 451/806/218 454/810/218
+f 502/901/441 438/786/441 500/897/441
+f 412/748/442 410/744/442 474/843/442
+f 495/887/208 496/889/208 494/886/208
+f 491/879/443 493/883/443 492/882/443
+f 455/813/444 456/814/444 458/818/444
+f 474/843/362 410/744/362 536/969/362
+f 497/891/445 498/893/445 496/889/445
+f 411/746/192 412/747/192 414/750/192
+f 492/881/375 428/771/375 490/877/375
+f 451/806/386 452/807/386 454/810/386
+f 423/764/217 421/761/217 424/766/217
+f 452/808/194 450/804/194 514/925/194
+f 434/780/446 436/783/446 435/782/446
+f 488/871/130 424/765/130 422/763/130
+f 462/825/447 526/950/448 528/954/447
+f 409/743/172 410/745/172 412/747/172
+f 484/864/449 420/760/449 418/756/165
+f 410/744/168 472/840/168 536/969/168
+f 433/779/446 434/780/446 435/782/446
+f 493/883/450 494/886/450 492/882/450
+f 533/963/175 534/965/175 531/959/175
+f 535/967/391 536/970/391 533/963/391
+f 467/832/404 468/833/404 469/835/404
+f 512/922/451 448/801/451 510/918/451
+f 429/773/199 430/775/199 431/776/199
+f 494/885/452 430/774/452 492/881/452
+f 513/923/169 514/926/169 512/921/169
+f 468/834/453 466/830/454 532/961/453
+f 419/758/116 417/755/116 420/759/116
+f 480/855/455 416/753/455 414/751/455
+f 413/749/192 411/746/192 414/750/192
+f 517/931/146 518/934/146 516/929/146
+f 446/798/456 444/795/456 508/914/456
+f 522/941/457 458/819/457 520/938/457
+f 432/777/173 434/780/173 433/779/173
+f 511/919/390 512/921/390 510/917/390
+f 432/778/458 430/774/458 494/885/458
+f 469/835/178 470/836/178 472/839/178
+f 526/950/459 462/825/460 460/821/461
+f 530/958/389 528/953/389 527/951/389
+f 487/869/462 488/872/462 485/865/462
+f 527/951/388 528/953/388 525/947/388
+f 428/772/410 430/775/410 429/773/410
+f 513/923/463 515/927/463 514/926/463
+f 525/947/151 526/949/151 523/943/151
+f 528/953/171 526/949/171 525/947/171
+f 411/746/172 409/743/172 412/747/172
+f 458/819/138 456/815/464 520/938/138
+f 507/911/110 508/913/110 505/907/110
+f 491/879/407 492/882/407 490/878/407
+f 428/771/465 426/768/131 490/877/131
+f 496/890/466 432/778/466 494/885/466
+f 463/826/119 464/827/119 465/829/119
+f 505/907/132 506/909/132 504/905/132
+f 415/752/112 413/749/112 416/754/112
+f 423/764/400 424/766/400 426/769/400
+f 526/949/408 524/946/408 523/943/408
+f 481/857/467 482/860/467 479/853/467
+f 500/897/468 436/784/468 498/894/468
+f 516/930/469 452/808/469 514/925/469
+f 507/911/403 509/915/403 508/913/403
+f 497/891/139 499/895/139 498/893/139
+f 447/800/197 448/802/197 450/805/197
+f 529/955/133 530/958/133 527/951/133
+f 478/851/470 480/855/470 414/751/470
+f 460/821/353 458/819/353 524/945/353
+f 425/767/471 426/769/471 427/770/471
+f 472/839/140 410/745/140 409/743/140
+f 485/865/396 486/867/396 483/861/396
+f 509/915/390 511/919/390 510/917/390
+f 508/913/472 506/909/473 505/907/473
+f 498/894/474 434/781/475 496/890/475
+f 477/849/394 478/852/394 475/845/394
+f 501/899/398 503/903/398 502/902/398
+f 455/812/163 453/809/163 456/816/163
+f 471/838/476 472/839/476 409/743/476
+f 457/817/401 458/818/401 459/820/401
+f 449/803/477 447/800/477 450/805/477
+f 493/883/208 495/887/208 494/886/208
+f 517/931/182 519/935/182 518/934/182
+f 465/829/478 466/831/478 467/832/478
+f 489/874/407 491/879/407 490/878/407
+f 482/859/479 484/864/479 418/756/479
+f 437/785/480 438/787/480 439/788/480
+f 439/788/114 440/790/114 441/791/114
+f 466/830/351 464/828/351 530/957/351
+f 442/792/481 440/789/481 506/910/365
+f 419/758/482 420/759/482 422/762/482
+f 415/752/174 416/754/174 418/757/174
+f 445/797/181 443/794/181 444/796/181
+f 466/831/478 468/833/478 467/832/478
+f 443/794/193 441/791/193 444/796/193
+f 430/775/199 432/777/199 431/776/199
+f 482/860/467 480/856/467 479/853/467
+f 431/776/173 432/777/173 433/779/173
+f 421/761/217 422/762/217 424/766/217
+f 514/925/483 450/804/345 512/922/345
+f 479/853/156 480/856/156 477/849/156
+f 426/769/409 428/772/409 427/770/409
+f 499/895/139 500/898/139 498/893/139
+f 484/864/484 486/868/484 420/760/484
+f 509/915/117 510/917/117 508/913/117
+f 457/817/121 455/813/121 458/818/121
+f 450/804/485 448/801/486 512/922/485
+f 444/795/487 442/792/487 508/914/487
+f 503/903/142 505/907/142 504/905/142
+f 447/800/115 445/797/115 448/802/115
+f 425/767/136 423/764/136 426/769/136
+f 510/918/488 446/798/488 508/914/488
+f 446/799/489 445/797/489 444/796/489
+f 427/770/190 428/772/190 429/773/190
+f 470/837/185 534/966/185 536/969/185
+f 524/945/355 526/950/490 460/821/490
+f 534/965/491 532/962/491 531/959/491
+f 440/789/492 504/906/492 506/910/492
+f 417/755/493 418/757/493 420/759/493
+f 441/791/494 442/793/494 444/796/494
+f 468/834/495 532/961/495 534/966/496
+f 438/786/200 502/901/200 504/906/200
+f 438/786/497 436/784/497 500/897/497
+f 536/970/216 534/965/216 533/963/216
+f 458/819/219 522/941/219 524/945/219
+f 478/852/498 476/848/498 475/845/498
+f 445/797/115 446/799/115 448/802/115
+f 414/751/499 412/748/499 476/847/499
+f 438/787/186 440/790/186 439/788/186
+f 459/820/118 460/822/118 461/823/118
+f 451/806/500 449/803/500 452/807/500
+f 464/827/119 466/831/119 465/829/119
+f 464/828/501 528/954/145 530/957/501
+f 515/927/502 516/929/502 514/926/502
+f 436/784/503 434/781/503 498/894/503
+f 470/837/504 468/834/359 534/966/504
+f 488/872/462 486/867/462 485/865/462
+f 501/899/157 502/902/157 500/898/157
+f 486/868/179 422/763/179 420/760/179
+f 449/803/500 450/805/500 452/807/500
+f 472/840/381 470/837/381 536/969/381
+f 495/887/445 497/891/445 496/889/445
+f 434/781/505 432/778/505 496/890/505
+f 532/962/122 530/958/122 529/955/122
+f 524/946/123 522/942/123 521/939/123
+f 475/845/124 476/848/124 473/841/124
+f 473/841/126 474/844/126 536/970/126
+f 454/811/506 452/808/506 516/930/506
+g KsHauptsignal_Plane.006_Lights
+usemtl Lights
+f 88/188/507 28/66/507 40/90/507
+f 40/90/507 44/98/507 56/122/507
+f 88/188/508 64/138/508 72/156/508
+f 72/156/508 76/164/508 88/188/508
+f 40/90/509 28/66/509 32/74/509
+f 32/74/507 34/78/507 40/90/507
+f 56/122/507 44/98/507 48/106/507
+f 48/106/507 52/114/507 56/122/507
+f 56/122/510 58/126/510 64/138/510
+f 64/138/508 68/146/508 70/150/508
+f 88/188/508 76/164/508 80/172/508
+f 80/172/508 84/180/508 88/188/508
+f 88/188/511 26/62/511 28/66/511
+f 28/66/511 30/70/511 32/74/511
+f 40/90/509 34/78/509 36/82/509
+f 36/82/509 38/86/509 40/90/509
+f 40/90/512 42/94/512 44/98/512
+f 44/98/509 46/102/509 48/106/509
+f 48/106/513 50/110/513 52/114/513
+f 52/114/514 54/118/515 56/122/515
+f 64/138/516 58/126/517 60/130/516
+f 60/130/518 62/134/518 64/138/518
+f 64/138/517 66/142/519 68/146/519
+f 64/138/508 70/150/508 72/156/508
+f 72/156/520 74/160/520 76/164/520
+f 76/164/508 78/168/508 80/172/519
+f 80/172/521 82/176/521 84/180/521
+f 84/180/522 86/184/522 88/188/522
+f 64/138/523 88/188/523 40/90/523
+f 156/294/524 158/298/524 160/302/524
+f 200/384/525 204/392/525 216/416/525
+f 192/366/526 186/354/526 188/358/526
+f 176/334/527 180/342/527 184/350/527
+f 212/408/528 214/412/528 216/416/529
+f 188/358/530 190/362/530 192/366/530
+f 192/366/531 196/374/532 198/378/532
+f 184/350/533 172/326/533 176/334/533
+f 204/392/534 206/396/535 208/400/534
+f 200/384/536 202/388/537 204/392/536
+f 216/416/538 192/366/538 200/384/538
+f 168/318/539 162/306/539 164/310/539
+f 164/310/540 166/314/541 168/318/540
+f 160/302/542 162/306/542 168/318/542
+f 176/334/543 178/338/543 180/342/543
+f 208/400/544 210/404/544 212/408/544
+f 168/318/545 170/322/541 172/326/545
+f 168/318/546 172/326/546 184/350/546
+f 168/318/547 156/294/547 160/302/547
+f 180/342/548 182/346/549 184/350/548
+f 208/400/544 212/408/544 216/416/544
+f 216/416/538 204/392/538 208/400/538
+f 192/366/550 194/370/550 196/374/550
+f 184/350/551 186/354/551 192/366/551
+f 192/366/552 198/378/552 200/384/552
+f 216/416/553 156/294/553 168/318/553
+f 216/416/554 154/290/554 156/294/554
+f 172/326/555 174/330/555 176/334/555
+f 192/366/556 216/416/556 168/318/556
+f 56/122/557 64/138/557 40/90/557
+f 344/644/507 284/522/507 296/546/507
+f 296/546/507 300/554/507 312/578/507
+f 344/644/508 320/594/508 328/612/508
+f 328/612/508 332/620/508 344/644/508
+f 296/546/507 284/522/558 288/530/558
+f 288/530/507 290/534/507 296/546/507
+f 312/578/509 300/554/509 304/562/509
+f 304/562/559 308/570/559 312/578/559
+f 312/578/557 314/582/557 320/594/557
+f 320/594/508 324/602/508 326/606/508
+f 344/644/508 332/620/508 336/628/508
+f 336/628/560 340/636/560 344/644/560
+f 344/644/561 282/518/561 284/522/561
+f 284/522/562 286/526/562 288/530/562
+f 296/546/509 290/534/509 292/538/509
+f 292/538/563 294/542/563 296/546/563
+f 296/546/512 298/550/512 300/554/512
+f 300/554/509 302/558/509 304/562/509
+f 304/562/513 306/566/513 308/570/513
+f 308/570/564 310/574/564 312/578/564
+f 320/594/565 314/582/565 316/586/565
+f 316/586/565 318/590/565 320/594/565
+f 320/594/519 322/598/519 324/602/519
+f 320/594/519 326/606/519 328/612/519
+f 328/612/566 330/616/566 332/620/566
+f 332/620/511 334/624/511 336/628/511
+f 336/628/567 338/632/567 340/636/567
+f 340/636/568 342/640/568 344/644/568
+f 320/594/523 344/644/523 296/546/523
+f 312/578/569 320/594/569 296/546/569
+f 184/350/570 192/366/570 168/318/570
+f 509/916/549 507/912/549 505/908/549
+f 521/940/539 527/952/539 525/948/539
+f 473/842/553 533/964/553 521/940/553
+f 521/940/545 519/936/545 517/932/545
+f 517/932/571 515/928/571 513/924/571
+f 533/964/524 531/960/524 529/956/524
+f 497/892/572 495/888/572 493/884/573
+f 489/875/537 487/870/537 485/866/537
+f 477/850/574 475/846/575 473/842/574
+f 473/842/538 497/892/538 489/875/538
+f 497/892/576 503/904/576 501/900/576
+f 505/908/577 517/932/533 513/924/577
+f 473/842/578 535/968/579 533/964/579
+f 481/858/580 477/850/580 473/842/581
+f 497/892/556 473/842/556 521/940/556
+f 501/900/582 499/896/582 497/892/583
+f 497/892/584 493/884/584 491/880/584
+f 521/940/546 517/932/546 505/908/546
+f 505/908/570 497/892/570 521/940/570
+f 473/842/538 485/866/538 481/858/538
+f 497/892/585 491/880/585 489/875/585
+f 529/956/541 527/952/553 521/940/553
+f 521/940/547 533/964/547 529/956/547
+f 481/858/586 479/854/586 477/850/586
+f 505/908/587 503/904/588 497/892/587
+f 513/924/543 511/920/543 509/916/543
+f 489/875/525 485/866/525 473/842/525
+f 513/924/527 509/916/527 505/908/527
+f 525/948/548 523/944/548 521/940/553
+f 485/866/589 483/862/589 481/858/589
+l 543 544
+l 540 537
+l 572 569
+l 539 540
+l 571 572
+l 538 539
+l 570 571
+l 575 576
+l 541 542
+l 573 574
+l 542 543
+l 574 575
diff --git a/advtrains_signals_ks/models/advtrains_signals_ks_distant_smr60.obj b/advtrains_signals_ks/models/advtrains_signals_ks_distant_smr60.obj
new file mode 100644
index 0000000..a55f479
--- /dev/null
+++ b/advtrains_signals_ks/models/advtrains_signals_ks_distant_smr60.obj
@@ -0,0 +1,3293 @@
+# Blender 3.4.1
+# www.blender.org
+mtllib advtrains_signals_ks_distant_smr60.mtl
+o KsHauptsignal_Plane.006
+v 0.395141 -0.503378 -0.194778
+v 0.366595 -0.503378 -0.244301
+v 0.416119 -0.503378 -0.272849
+v 0.444666 -0.503378 -0.223325
+v 0.296343 -0.503378 -0.137828
+v 0.246819 -0.503378 -0.109281
+v 0.218272 -0.503378 -0.158806
+v 0.267797 -0.503378 -0.187353
+v 0.395141 -0.299114 -0.194778
+v 0.395141 -0.172218 -0.194778
+v 0.366595 -0.172218 -0.244301
+v 0.366595 -0.299114 -0.244301
+v 0.416119 -0.299114 -0.272849
+v 0.416119 -0.172218 -0.272849
+v 0.444666 -0.172218 -0.223325
+v 0.444666 -0.299114 -0.223325
+v 0.296343 -0.299114 -0.137828
+v 0.296343 -0.172218 -0.137828
+v 0.246819 -0.172218 -0.109281
+v 0.246819 -0.299114 -0.109281
+v 0.218272 -0.299114 -0.158806
+v 0.218272 -0.172218 -0.158806
+v 0.267797 -0.172218 -0.187353
+v 0.267797 -0.299114 -0.187353
+v 0.440343 0.243339 -0.410349
+v 0.416986 0.242789 -0.396885
+v 0.487403 0.253858 -0.433838
+v 0.418519 0.253308 -0.394131
+v 0.489934 0.263561 -0.429360
+v 0.421050 0.263011 -0.389655
+v 0.493366 0.272074 -0.423331
+v 0.424481 0.271524 -0.383625
+v 0.497566 0.279070 -0.415983
+v 0.428681 0.278520 -0.376276
+v 0.502372 0.284280 -0.407596
+v 0.433488 0.283730 -0.367890
+v 0.507603 0.287504 -0.398494
+v 0.438718 0.286954 -0.358788
+v 0.513054 0.288619 -0.389027
+v 0.444170 0.288069 -0.349321
+v 0.518517 0.287581 -0.379558
+v 0.449633 0.287030 -0.339852
+v 0.523783 0.284430 -0.370451
+v 0.454899 0.283880 -0.330745
+v 0.528648 0.279287 -0.362056
+v 0.459765 0.278737 -0.322350
+v 0.532926 0.272350 -0.354696
+v 0.464043 0.271800 -0.314990
+v 0.536453 0.263886 -0.348654
+v 0.467570 0.263336 -0.308948
+v 0.539092 0.254219 -0.344161
+v 0.470208 0.253669 -0.304455
+v 0.495217 0.243722 -0.315148
+v 0.471860 0.243172 -0.301685
+v 0.495781 0.232316 -0.314164
+v 0.472457 0.232252 -0.300744
+v 0.495099 0.221688 -0.315029
+v 0.471979 0.221322 -0.301665
+v 0.493575 0.211344 -0.317752
+v 0.470441 0.210797 -0.304415
+v 0.491039 0.201642 -0.322227
+v 0.467905 0.201094 -0.308890
+v 0.487605 0.193129 -0.328254
+v 0.464470 0.192581 -0.314917
+v 0.483402 0.186134 -0.335601
+v 0.460267 0.185585 -0.322264
+v 0.478593 0.180924 -0.343985
+v 0.455457 0.180374 -0.330649
+v 0.473361 0.177700 -0.353086
+v 0.450226 0.177150 -0.339750
+v 0.467909 0.176585 -0.362554
+v 0.444774 0.176035 -0.349217
+v 0.462447 0.177623 -0.372023
+v 0.439311 0.177073 -0.358687
+v 0.457182 0.180774 -0.381130
+v 0.434047 0.180224 -0.367795
+v 0.452319 0.185916 -0.389526
+v 0.429183 0.185367 -0.376191
+v 0.448044 0.192853 -0.396888
+v 0.424908 0.192304 -0.383553
+v 0.444521 0.201317 -0.402932
+v 0.421385 0.200769 -0.389598
+v 0.441886 0.210983 -0.407427
+v 0.418750 0.210436 -0.394092
+v 0.440258 0.221305 -0.410173
+v 0.417104 0.220938 -0.396866
+v 0.439814 0.231924 -0.411262
+v 0.416509 0.231861 -0.397809
+v 0.439105 0.243790 -0.412487
+v 0.384100 0.243271 -0.380782
+v 0.486231 0.254782 -0.435851
+v 0.385699 0.254264 -0.377903
+v 0.488874 0.264921 -0.431172
+v 0.388342 0.264402 -0.373224
+v 0.492458 0.273816 -0.424871
+v 0.391926 0.273298 -0.366922
+v 0.496845 0.281127 -0.417191
+v 0.396314 0.280609 -0.359242
+v 0.501867 0.286572 -0.408426
+v 0.401336 0.286053 -0.350478
+v 0.507332 0.289941 -0.398915
+v 0.406800 0.289423 -0.340967
+v 0.513028 0.291105 -0.389022
+v 0.412496 0.290587 -0.331073
+v 0.518737 0.290021 -0.379127
+v 0.418205 0.289502 -0.321179
+v 0.524240 0.286728 -0.369612
+v 0.423709 0.286210 -0.311663
+v 0.529325 0.281354 -0.360840
+v 0.428794 0.280836 -0.302892
+v 0.533797 0.274105 -0.353150
+v 0.433266 0.273587 -0.295201
+v 0.537484 0.265261 -0.346837
+v 0.436952 0.264742 -0.288889
+v 0.540244 0.255160 -0.342144
+v 0.439713 0.254641 -0.284195
+v 0.496445 0.244190 -0.313008
+v 0.441440 0.243672 -0.281302
+v 0.497026 0.232775 -0.311999
+v 0.442068 0.232256 -0.280320
+v 0.496356 0.221351 -0.312865
+v 0.441572 0.220833 -0.281286
+v 0.494756 0.210359 -0.315743
+v 0.439973 0.209840 -0.284164
+v 0.492114 0.200220 -0.320422
+v 0.437331 0.199702 -0.288844
+v 0.488530 0.191324 -0.326723
+v 0.433747 0.190806 -0.295146
+v 0.484142 0.184013 -0.334403
+v 0.429359 0.183495 -0.302826
+v 0.479120 0.178569 -0.343168
+v 0.424338 0.178051 -0.311590
+v 0.473656 0.175200 -0.352679
+v 0.418873 0.174682 -0.321101
+v 0.467960 0.174035 -0.362572
+v 0.413177 0.173517 -0.330994
+v 0.462250 0.175120 -0.372467
+v 0.407468 0.174602 -0.340889
+v 0.456747 0.178413 -0.381983
+v 0.401965 0.177894 -0.350405
+v 0.451662 0.183787 -0.390754
+v 0.396880 0.183268 -0.359176
+v 0.447190 0.191035 -0.398444
+v 0.392407 0.190517 -0.366866
+v 0.443504 0.199880 -0.404757
+v 0.388720 0.199362 -0.373179
+v 0.440743 0.209981 -0.409450
+v 0.385961 0.209463 -0.377873
+v 0.439016 0.220950 -0.412344
+v 0.384232 0.220432 -0.380766
+v 0.438564 0.232366 -0.413428
+v 0.383605 0.231848 -0.381748
+v 0.556323 0.383581 -0.209143
+v 0.532965 0.383353 -0.195679
+v 0.602459 0.387932 -0.234233
+v 0.533575 0.387704 -0.194526
+v 0.603484 0.391945 -0.232369
+v 0.534600 0.391717 -0.192662
+v 0.604884 0.395466 -0.229864
+v 0.536000 0.395239 -0.190157
+v 0.606605 0.398360 -0.226815
+v 0.537722 0.398132 -0.187109
+v 0.608582 0.400515 -0.223339
+v 0.539698 0.400287 -0.183633
+v 0.610738 0.401848 -0.219570
+v 0.541854 0.401621 -0.179864
+v 0.612991 0.402309 -0.215653
+v 0.544106 0.402082 -0.175946
+v 0.615252 0.401880 -0.211738
+v 0.546369 0.401652 -0.172031
+v 0.617437 0.400577 -0.207975
+v 0.548554 0.400349 -0.168269
+v 0.619461 0.398450 -0.204510
+v 0.550578 0.398222 -0.164803
+v 0.621247 0.395581 -0.201475
+v 0.552363 0.395353 -0.161768
+v 0.622724 0.392080 -0.198986
+v 0.553841 0.391852 -0.159280
+v 0.623838 0.388081 -0.197141
+v 0.554955 0.387854 -0.157434
+v 0.579018 0.383739 -0.169767
+v 0.555661 0.383512 -0.156302
+v 0.579250 0.379021 -0.169358
+v 0.555933 0.378995 -0.155927
+v 0.578888 0.374626 -0.169669
+v 0.555758 0.374474 -0.156321
+v 0.578277 0.370347 -0.170807
+v 0.555141 0.370121 -0.157471
+v 0.577248 0.366334 -0.172669
+v 0.554113 0.366108 -0.159333
+v 0.575845 0.362813 -0.175172
+v 0.552709 0.362586 -0.161835
+v 0.574120 0.359920 -0.178219
+v 0.550985 0.359693 -0.164882
+v 0.572141 0.357765 -0.181693
+v 0.549007 0.357538 -0.168357
+v 0.569984 0.356431 -0.185461
+v 0.546849 0.356204 -0.172125
+v 0.567731 0.355970 -0.189377
+v 0.544596 0.355743 -0.176042
+v 0.565470 0.356399 -0.193293
+v 0.542334 0.356172 -0.179958
+v 0.563286 0.357703 -0.197056
+v 0.540151 0.357476 -0.183721
+v 0.561265 0.359830 -0.200523
+v 0.538128 0.359603 -0.187189
+v 0.559482 0.362699 -0.203560
+v 0.536346 0.362472 -0.190225
+v 0.558008 0.366200 -0.206050
+v 0.534871 0.365973 -0.192716
+v 0.556898 0.370198 -0.207898
+v 0.533762 0.369972 -0.194563
+v 0.556205 0.374467 -0.209023
+v 0.533061 0.374316 -0.195699
+v 0.556102 0.378860 -0.209519
+v 0.532792 0.378833 -0.196075
+v 0.555808 0.383767 -0.210026
+v 0.500803 0.383553 -0.178320
+v 0.601970 0.388314 -0.235063
+v 0.501438 0.388099 -0.177115
+v 0.603038 0.392507 -0.233113
+v 0.502507 0.392293 -0.175165
+v 0.604500 0.396187 -0.230495
+v 0.503968 0.395973 -0.172547
+v 0.606297 0.399211 -0.227308
+v 0.505766 0.398996 -0.169360
+v 0.608361 0.401463 -0.223675
+v 0.507830 0.401248 -0.165727
+v 0.610613 0.402856 -0.219737
+v 0.510082 0.402642 -0.161788
+v 0.612966 0.403338 -0.215643
+v 0.512435 0.403124 -0.157694
+v 0.615331 0.402889 -0.211552
+v 0.514799 0.402675 -0.153603
+v 0.617615 0.401527 -0.207621
+v 0.517083 0.401313 -0.149673
+v 0.619731 0.399305 -0.204000
+v 0.519200 0.399090 -0.146052
+v 0.621598 0.396306 -0.200829
+v 0.521066 0.396092 -0.142880
+v 0.623144 0.392648 -0.198231
+v 0.522613 0.392434 -0.140282
+v 0.624310 0.388470 -0.196304
+v 0.523778 0.388256 -0.138355
+v 0.579524 0.383933 -0.168879
+v 0.524519 0.383719 -0.137173
+v 0.579765 0.379211 -0.168462
+v 0.524806 0.378997 -0.136782
+v 0.579411 0.374486 -0.168776
+v 0.524628 0.374272 -0.137198
+v 0.578775 0.369939 -0.169982
+v 0.523993 0.369725 -0.138404
+v 0.577707 0.365746 -0.171931
+v 0.522924 0.365532 -0.140354
+v 0.576245 0.362067 -0.174550
+v 0.521463 0.361852 -0.142972
+v 0.574448 0.359043 -0.177736
+v 0.519666 0.358828 -0.146158
+v 0.572384 0.356791 -0.181369
+v 0.517601 0.356576 -0.149791
+v 0.570132 0.355397 -0.185308
+v 0.515349 0.355183 -0.153729
+v 0.567779 0.354916 -0.189402
+v 0.512996 0.354701 -0.157823
+v 0.565415 0.355364 -0.193492
+v 0.510632 0.355150 -0.161914
+v 0.563130 0.356726 -0.197424
+v 0.508348 0.356512 -0.165846
+v 0.561015 0.358949 -0.201044
+v 0.506232 0.358735 -0.169466
+v 0.559148 0.361947 -0.204216
+v 0.504364 0.361733 -0.172637
+v 0.557601 0.365605 -0.206814
+v 0.502819 0.365391 -0.175236
+v 0.556435 0.369783 -0.208742
+v 0.501653 0.369569 -0.177164
+v 0.555696 0.374320 -0.209923
+v 0.500912 0.374106 -0.178345
+v 0.555584 0.379042 -0.210415
+v 0.500625 0.378828 -0.178736
+v 0.540223 0.243339 -0.237074
+v 0.516865 0.242789 -0.223609
+v 0.587282 0.253858 -0.260563
+v 0.518398 0.253308 -0.220856
+v 0.589813 0.263561 -0.256085
+v 0.520928 0.263011 -0.216379
+v 0.593244 0.272074 -0.250056
+v 0.524361 0.271524 -0.210351
+v 0.597444 0.279070 -0.242708
+v 0.528561 0.278520 -0.203001
+v 0.602252 0.284280 -0.234321
+v 0.533367 0.283730 -0.194615
+v 0.607481 0.287504 -0.225219
+v 0.538597 0.286954 -0.185514
+v 0.612934 0.288619 -0.215752
+v 0.544049 0.288069 -0.176046
+v 0.618397 0.287581 -0.206283
+v 0.549512 0.287030 -0.166577
+v 0.623662 0.284430 -0.197176
+v 0.554778 0.283880 -0.157470
+v 0.628528 0.279287 -0.188782
+v 0.559644 0.278737 -0.149075
+v 0.632805 0.272350 -0.181421
+v 0.563922 0.271800 -0.141715
+v 0.636332 0.263886 -0.175379
+v 0.567448 0.263336 -0.135673
+v 0.638972 0.254219 -0.170886
+v 0.570087 0.253669 -0.131180
+v 0.595096 0.243722 -0.141874
+v 0.571738 0.243172 -0.128410
+v 0.595660 0.232316 -0.140888
+v 0.572337 0.232252 -0.127469
+v 0.594979 0.221688 -0.141755
+v 0.571858 0.221322 -0.128390
+v 0.593454 0.211344 -0.144477
+v 0.570319 0.210797 -0.131140
+v 0.590919 0.201642 -0.148952
+v 0.567784 0.201094 -0.135615
+v 0.587483 0.193129 -0.154979
+v 0.564349 0.192581 -0.141642
+v 0.583281 0.186134 -0.162326
+v 0.560146 0.185585 -0.148989
+v 0.578471 0.180924 -0.170710
+v 0.555337 0.180374 -0.157374
+v 0.573240 0.177700 -0.179811
+v 0.550106 0.177150 -0.166475
+v 0.567788 0.176585 -0.189278
+v 0.544654 0.176035 -0.175942
+v 0.562325 0.177623 -0.198748
+v 0.539190 0.177073 -0.185413
+v 0.557062 0.180774 -0.207855
+v 0.533925 0.180224 -0.194520
+v 0.552198 0.185916 -0.216251
+v 0.529063 0.185368 -0.202917
+v 0.547923 0.192853 -0.223613
+v 0.524787 0.192304 -0.210278
+v 0.544400 0.201317 -0.229657
+v 0.521264 0.200769 -0.216323
+v 0.541765 0.210983 -0.234152
+v 0.518629 0.210436 -0.220818
+v 0.540138 0.221305 -0.236898
+v 0.516983 0.220938 -0.223591
+v 0.539693 0.231924 -0.237987
+v 0.516388 0.231861 -0.224534
+v 0.538985 0.243790 -0.239213
+v 0.483979 0.243271 -0.207507
+v 0.586111 0.254782 -0.262577
+v 0.485579 0.254264 -0.204628
+v 0.588752 0.264921 -0.257897
+v 0.488221 0.264402 -0.199948
+v 0.592337 0.273816 -0.251596
+v 0.491805 0.273298 -0.193647
+v 0.596724 0.281127 -0.243915
+v 0.496193 0.280609 -0.185967
+v 0.601747 0.286572 -0.235152
+v 0.501215 0.286053 -0.177203
+v 0.607211 0.289941 -0.225640
+v 0.506679 0.289423 -0.167692
+v 0.612907 0.291105 -0.215747
+v 0.512375 0.290587 -0.157798
+v 0.618617 0.290021 -0.205852
+v 0.518085 0.289502 -0.147905
+v 0.624120 0.286728 -0.196337
+v 0.523587 0.286210 -0.138388
+v 0.629205 0.281354 -0.187565
+v 0.528672 0.280836 -0.129616
+v 0.633676 0.274105 -0.179875
+v 0.533145 0.273587 -0.121926
+v 0.637363 0.265261 -0.173562
+v 0.536832 0.264742 -0.115614
+v 0.640124 0.255160 -0.168869
+v 0.539591 0.254641 -0.110920
+v 0.596325 0.244190 -0.139733
+v 0.541319 0.243672 -0.108027
+v 0.596906 0.232775 -0.138724
+v 0.541947 0.232256 -0.107045
+v 0.596234 0.221351 -0.139590
+v 0.541452 0.220833 -0.108012
+v 0.594635 0.210359 -0.142468
+v 0.539853 0.209840 -0.110890
+v 0.591992 0.200220 -0.147147
+v 0.537210 0.199702 -0.115569
+v 0.588409 0.191324 -0.153448
+v 0.533626 0.190806 -0.121871
+v 0.584022 0.184014 -0.161129
+v 0.529238 0.183495 -0.129551
+v 0.578999 0.178569 -0.169893
+v 0.524216 0.178051 -0.138315
+v 0.573535 0.175200 -0.179404
+v 0.518753 0.174682 -0.147826
+v 0.567838 0.174035 -0.189297
+v 0.513056 0.173517 -0.157719
+v 0.562129 0.175120 -0.199192
+v 0.507346 0.174602 -0.167614
+v 0.556626 0.178413 -0.208707
+v 0.501844 0.177894 -0.177130
+v 0.551541 0.183787 -0.217479
+v 0.496758 0.183268 -0.185901
+v 0.547069 0.191035 -0.225169
+v 0.492286 0.190517 -0.193591
+v 0.543382 0.199880 -0.231482
+v 0.488599 0.199362 -0.199905
+v 0.540622 0.209981 -0.236175
+v 0.485839 0.209463 -0.204597
+v 0.538895 0.220950 -0.239069
+v 0.484112 0.220432 -0.207491
+v 0.538442 0.232366 -0.240152
+v 0.483484 0.231848 -0.208473
+v 0.500625 0.050272 -0.178736
+v 0.555584 0.050487 -0.210415
+v 0.500912 0.045551 -0.178345
+v 0.555696 0.045765 -0.209923
+v 0.501653 0.041014 -0.177164
+v 0.556435 0.041228 -0.208742
+v 0.502819 0.036836 -0.175236
+v 0.557601 0.037050 -0.206814
+v 0.504364 0.033177 -0.172637
+v 0.559148 0.033392 -0.204216
+v 0.506232 0.030179 -0.169466
+v 0.561015 0.030393 -0.201044
+v 0.508348 0.027956 -0.165846
+v 0.563130 0.028171 -0.197424
+v 0.510632 0.026594 -0.161914
+v 0.565415 0.026809 -0.193492
+v 0.512996 0.026146 -0.157823
+v 0.567779 0.026360 -0.189402
+v 0.515349 0.026627 -0.153729
+v 0.570132 0.026842 -0.185308
+v 0.517601 0.028021 -0.149791
+v 0.572385 0.028235 -0.181369
+v 0.519666 0.030273 -0.146158
+v 0.574448 0.030487 -0.177736
+v 0.521463 0.033297 -0.142972
+v 0.576245 0.033511 -0.174550
+v 0.522924 0.036976 -0.140354
+v 0.577707 0.037191 -0.171931
+v 0.523993 0.041170 -0.138404
+v 0.578775 0.041384 -0.169982
+v 0.524628 0.045716 -0.137198
+v 0.579411 0.045931 -0.168776
+v 0.524806 0.050441 -0.136782
+v 0.579765 0.050656 -0.168462
+v 0.524519 0.055163 -0.137173
+v 0.579524 0.055377 -0.168879
+v 0.523778 0.059700 -0.138355
+v 0.624310 0.059915 -0.196304
+v 0.522613 0.063878 -0.140283
+v 0.623144 0.064093 -0.198231
+v 0.521066 0.067537 -0.142880
+v 0.621598 0.067751 -0.200829
+v 0.519200 0.070535 -0.146052
+v 0.619731 0.070749 -0.204000
+v 0.517083 0.072757 -0.149673
+v 0.617615 0.072972 -0.207621
+v 0.514799 0.074119 -0.153604
+v 0.615331 0.074334 -0.211552
+v 0.512435 0.074568 -0.157695
+v 0.612966 0.074782 -0.215643
+v 0.510082 0.074086 -0.161788
+v 0.610613 0.074301 -0.219737
+v 0.507830 0.072693 -0.165727
+v 0.608361 0.072907 -0.223675
+v 0.505766 0.070441 -0.169360
+v 0.606297 0.070655 -0.227308
+v 0.503968 0.067417 -0.172547
+v 0.604500 0.067631 -0.230495
+v 0.502507 0.063738 -0.175165
+v 0.603038 0.063952 -0.233113
+v 0.501438 0.059544 -0.177115
+v 0.601970 0.059758 -0.235063
+v 0.500803 0.054997 -0.178320
+v 0.555808 0.055212 -0.210026
+v 0.532792 0.050278 -0.196075
+v 0.556102 0.050304 -0.209519
+v 0.533061 0.045760 -0.195699
+v 0.556205 0.045912 -0.209023
+v 0.533762 0.041416 -0.194563
+v 0.556898 0.041642 -0.207898
+v 0.534871 0.037418 -0.192716
+v 0.558008 0.037644 -0.206050
+v 0.536346 0.033917 -0.190225
+v 0.559482 0.034143 -0.203560
+v 0.538128 0.031047 -0.187189
+v 0.561265 0.031274 -0.200523
+v 0.540151 0.028920 -0.183721
+v 0.563286 0.029147 -0.197056
+v 0.542334 0.027617 -0.179958
+v 0.565470 0.027844 -0.193293
+v 0.544596 0.027187 -0.176042
+v 0.567731 0.027415 -0.189377
+v 0.546849 0.027648 -0.172125
+v 0.569984 0.027876 -0.185461
+v 0.549007 0.028982 -0.168357
+v 0.572141 0.029209 -0.181693
+v 0.550985 0.031137 -0.164882
+v 0.574120 0.031364 -0.178219
+v 0.552709 0.034031 -0.161835
+v 0.575845 0.034258 -0.175172
+v 0.554113 0.037552 -0.159333
+v 0.577248 0.037779 -0.172669
+v 0.555141 0.041566 -0.157471
+v 0.578277 0.041792 -0.170807
+v 0.555758 0.045919 -0.156321
+v 0.578888 0.046070 -0.169669
+v 0.555933 0.050440 -0.155927
+v 0.579250 0.050466 -0.169358
+v 0.555661 0.054956 -0.156302
+v 0.579018 0.055184 -0.169767
+v 0.554955 0.059298 -0.157434
+v 0.623838 0.059526 -0.197141
+v 0.553841 0.063296 -0.159280
+v 0.622724 0.063524 -0.198986
+v 0.552363 0.066797 -0.161768
+v 0.621247 0.067025 -0.201475
+v 0.550578 0.069667 -0.164803
+v 0.619461 0.069894 -0.204510
+v 0.548554 0.071794 -0.168269
+v 0.617437 0.072021 -0.207975
+v 0.546369 0.073097 -0.172031
+v 0.615252 0.073325 -0.211738
+v 0.544106 0.073526 -0.175946
+v 0.612991 0.073754 -0.215653
+v 0.541854 0.073065 -0.179864
+v 0.610739 0.073293 -0.219570
+v 0.539698 0.071732 -0.183633
+v 0.608582 0.071959 -0.223339
+v 0.537722 0.069577 -0.187109
+v 0.606605 0.069804 -0.226815
+v 0.536000 0.066683 -0.190157
+v 0.604884 0.066911 -0.229864
+v 0.534600 0.063162 -0.192662
+v 0.603484 0.063390 -0.232369
+v 0.533575 0.059149 -0.194526
+v 0.602459 0.059376 -0.234233
+v 0.532965 0.054798 -0.195680
+v 0.556323 0.055025 -0.209143
+v 0.395141 -0.168051 -0.194778
+v 0.366595 -0.168051 -0.244301
+v 0.416119 -0.168051 -0.272849
+v 0.444666 -0.168051 -0.223325
+v 0.296343 -0.168051 -0.137828
+v 0.246819 -0.168051 -0.109281
+v 0.218272 -0.168051 -0.158806
+v 0.267797 -0.168051 -0.187353
+v 0.218272 -0.168051 -0.158806
+v 0.246819 -0.168051 -0.109281
+v 0.444666 -0.168051 -0.223325
+v 0.416119 -0.168051 -0.272849
+v 0.267797 -0.168051 -0.187353
+v 0.366595 -0.168051 -0.244301
+v 0.296343 -0.168051 -0.137828
+v 0.395141 -0.168051 -0.194778
+v 0.395141 0.037812 -0.194778
+v 0.395141 0.165702 -0.194778
+v 0.366595 0.165702 -0.244301
+v 0.366595 0.037812 -0.244301
+v 0.416119 0.037812 -0.272849
+v 0.416119 0.165702 -0.272849
+v 0.444666 0.165702 -0.223325
+v 0.444666 0.037812 -0.223325
+v 0.296343 0.037812 -0.137828
+v 0.296343 0.165702 -0.137828
+v 0.246819 0.165702 -0.109281
+v 0.246819 0.037812 -0.109281
+v 0.218272 0.037812 -0.158806
+v 0.218272 0.165702 -0.158806
+v 0.267797 0.165702 -0.187353
+v 0.267797 0.037812 -0.187353
+v 0.395141 0.166622 -0.194778
+v 0.366595 0.166622 -0.244301
+v 0.416119 0.166622 -0.272849
+v 0.444666 0.166622 -0.223325
+v 0.296343 0.166622 -0.137828
+v 0.246819 0.166622 -0.109281
+v 0.218272 0.166622 -0.158806
+v 0.267797 0.166622 -0.187353
+v 0.218272 0.166622 -0.158806
+v 0.246819 0.166622 -0.109281
+v 0.444666 0.166622 -0.223325
+v 0.416119 0.166622 -0.272849
+v 0.267797 0.166622 -0.187353
+v 0.366595 0.166622 -0.244301
+v 0.296343 0.166622 -0.137828
+v 0.395141 0.166622 -0.194778
+v 0.395141 0.370885 -0.194778
+v 0.395141 0.497782 -0.194778
+v 0.366595 0.497782 -0.244301
+v 0.366595 0.370885 -0.244301
+v 0.416119 0.370885 -0.272849
+v 0.416119 0.497782 -0.272849
+v 0.444666 0.497782 -0.223325
+v 0.444666 0.370885 -0.223325
+v 0.296343 0.370885 -0.137828
+v 0.296343 0.497782 -0.137828
+v 0.246819 0.497782 -0.109281
+v 0.246819 0.370885 -0.109281
+v 0.218272 0.370885 -0.158806
+v 0.218272 0.497782 -0.158806
+v 0.267797 0.497782 -0.187353
+v 0.267797 0.370885 -0.187353
+v 0.609716 -0.080213 -0.040537
+v 0.360018 -0.080213 -0.473724
+v 0.609715 0.496942 -0.040536
+v 0.360017 0.496942 -0.473724
+v 0.568633 0.496942 -0.016856
+v 0.568633 -0.080213 -0.016856
+v 0.318936 -0.080213 -0.450043
+v 0.318936 0.496942 -0.450043
+vn -0.4994 -0.0000 -0.8664
+vn 0.4994 -0.0000 0.8664
+vn 0.8664 -0.0000 -0.4994
+vn -0.8664 -0.0000 0.4994
+vn -0.0000 -1.0000 -0.0000
+vn -0.0000 1.0000 -0.0000
+vn -0.0412 -0.9955 -0.0853
+vn 0.4989 -0.0946 0.8615
+vn -0.4950 -0.1015 -0.8629
+vn -0.4993 0.0944 -0.8612
+vn 0.4945 0.1014 0.8632
+vn -0.4978 0.0945 -0.8621
+vn 0.4961 0.1015 0.8623
+vn 0.4978 -0.0946 0.8621
+vn -0.4961 -0.1014 -0.8623
+vn 0.3735 0.6370 0.6743
+vn 0.8703 0.0090 -0.4925
+vn 0.8676 0.0067 -0.4972
+vn 0.8666 0.0100 -0.4989
+vn 0.8652 0.0089 -0.5014
+vn 0.8651 0.0091 -0.5016
+vn 0.8670 0.0096 -0.4981
+vn 0.8675 0.0054 -0.4975
+vn 0.8648 0.0081 -0.5020
+vn 0.8661 0.0101 -0.4998
+vn 0.8676 0.0077 -0.4973
+vn 0.8656 0.0093 -0.5006
+vn 0.8674 0.0087 -0.4975
+vn 0.8647 0.0071 -0.5023
+vn 0.8650 0.0059 -0.5017
+vn 0.8672 0.0057 -0.4978
+vn 0.8655 0.0002 -0.5010
+vn 0.8648 -0.0134 -0.5020
+vn 0.8638 -0.0144 -0.5037
+vn 0.2751 -0.9600 0.0521
+vn 0.8707 0.0120 -0.4916
+vn 0.8619 0.0119 -0.5070
+vn 0.8705 0.0151 -0.4919
+vn 0.8697 0.0180 -0.4932
+vn 0.8684 0.0198 -0.4954
+vn 0.8653 0.0208 -0.5008
+vn 0.8639 0.0200 -0.5033
+vn 0.8627 0.0178 -0.5054
+vn 0.8621 0.0147 -0.5065
+vn -0.0039 -0.9097 -0.4152
+vn 0.8623 0.0089 -0.5063
+vn 0.8631 0.0057 -0.5050
+vn 0.4980 -0.0920 0.8623
+vn -0.3211 0.7708 -0.5503
+vn 0.3125 0.7752 0.5490
+vn -0.3811 -0.6371 -0.6700
+vn 0.2305 0.8836 0.4077
+vn -0.1365 -0.9579 -0.2526
+vn -0.1376 -0.9579 -0.2519
+vn 0.0433 0.9955 0.0841
+vn 0.0567 -0.9948 0.0845
+vn 0.1396 0.9579 0.2508
+vn 0.2422 -0.8803 0.4080
+vn 0.3228 -0.7708 0.5493
+vn -0.4368 -0.4745 -0.7643
+vn 0.4440 -0.4683 0.7639
+vn 0.4787 -0.2964 0.8264
+vn -0.4961 -0.0990 -0.8626
+vn -0.4963 0.1021 -0.8621
+vn 0.4311 0.4744 0.7675
+vn 0.8670 0.0208 -0.4979
+vn 0.4725 0.2954 0.8303
+vn 0.8668 -0.0002 -0.4986
+vn 0.3016 0.7751 0.5552
+vn 0.2181 0.8834 0.4147
+vn 0.3825 0.6371 0.6692
+vn -0.2276 -0.8835 -0.4094
+vn -0.1637 0.9557 -0.2445
+vn -0.2526 0.8801 -0.4020
+vn -0.3318 0.7707 -0.5440
+vn -0.3984 0.6316 -0.6651
+vn -0.4496 0.4683 -0.7606
+vn -0.4826 0.2888 -0.8269
+vn 0.4967 0.1091 0.8610
+vn -0.4978 0.0946 -0.8621
+vn -0.3804 -0.6370 -0.6704
+vn -0.3099 -0.7752 -0.5504
+vn 0.4378 0.4745 0.7637
+vn 0.0294 0.9953 0.0921
+vn 0.3909 -0.6317 0.6694
+vn -0.2286 -0.8835 -0.4088
+vn -0.0402 -0.9955 -0.0860
+vn -0.0545 0.9948 -0.0857
+vn -0.1503 0.9559 -0.2522
+vn -0.2404 0.8802 -0.4091
+vn -0.4741 -0.3033 -0.8266
+vn -0.3896 0.6317 -0.6702
+vn -0.4431 0.4683 -0.7645
+vn -0.4787 0.2926 -0.8278
+vn 0.4961 0.1014 0.8623
+vn 0.1262 0.9578 0.2584
+vn -0.4753 -0.2936 -0.8294
+vn -0.4362 -0.4745 -0.7646
+vn 0.4753 0.2936 0.8294
+vn -0.0684 0.9946 -0.0777
+vn -0.3108 -0.7752 -0.5500
+vn 0.1523 -0.9559 0.2511
+vn 0.0578 -0.9948 0.0839
+vn 0.1534 -0.9559 0.2505
+vn 0.2432 -0.8803 0.4074
+vn 0.3236 -0.7708 0.5488
+vn 0.3916 -0.6317 0.6690
+vn 0.4446 -0.4683 0.7636
+vn 0.4804 -0.2869 0.8288
+vn -0.4961 -0.1015 -0.8623
+vn 0.8755 0.0217 -0.4827
+vn -0.4382 -0.4745 -0.7635
+vn 0.4752 0.2956 0.8288
+vn 0.4974 -0.0946 0.8623
+vn 0.4388 0.4745 0.7631
+vn -0.3132 -0.7753 -0.5485
+vn -0.4752 -0.3031 -0.8261
+vn -0.1481 0.9560 -0.2534
+vn -0.3194 0.7708 -0.5512
+vn 0.8555 0.0359 -0.5165
+vn 0.0457 0.9955 0.0829
+vn 0.3887 -0.6317 0.6707
+vn 0.0532 -0.9948 0.0865
+vn 0.4959 0.1013 0.8625
+vn -0.4420 0.4684 -0.7650
+vn 0.4976 -0.0920 0.8625
+vn 0.8633 0.0218 -0.5043
+vn 0.8631 0.0220 -0.5045
+vn 0.8687 0.0207 -0.4949
+vn 0.8707 0.0479 -0.4894
+vn 0.8633 0.0503 -0.5022
+vn -0.4966 0.1020 -0.8620
+vn 0.3201 -0.7708 0.5509
+vn -0.0447 -0.9955 -0.0832
+vn 0.8691 0.0131 -0.4944
+vn -0.0442 -0.9955 -0.0835
+vn 0.8599 -0.0348 -0.5092
+vn 0.8667 0.0244 -0.4982
+vn -0.3918 0.6316 -0.6690
+vn -0.4974 0.0946 -0.8623
+vn 0.3889 -0.6318 0.6705
+vn -0.4981 0.0945 -0.8620
+vn 0.2274 0.8835 0.4095
+vn 0.8690 0.0160 -0.4945
+vn 0.8628 0.0192 -0.5053
+vn -0.2317 -0.8836 -0.4070
+vn -0.3135 -0.7752 -0.5484
+vn 0.8671 0.0503 -0.4956
+vn 0.8671 0.0504 -0.4955
+vn 0.8552 0.0286 -0.5174
+vn 0.1489 -0.9560 0.2529
+vn 0.0538 -0.9948 0.0865
+vn 0.8686 0.0138 -0.4954
+vn 0.1363 0.9579 0.2528
+vn -0.0580 0.9948 -0.0837
+vn 0.4361 0.4744 0.7647
+vn -0.4447 0.4683 -0.7635
+vn 0.8678 0.0229 -0.4964
+vn -0.1537 0.9559 -0.2504
+vn 0.8569 0.0428 -0.5136
+vn 0.8569 0.0429 -0.5137
+vn -0.4965 -0.1015 -0.8621
+vn 0.1418 0.9579 0.2495
+vn 0.4426 -0.4682 0.7647
+vn 0.8759 0.0364 -0.4812
+vn 0.4965 0.1089 0.8612
+vn 0.4978 -0.0945 0.8621
+vn 0.8640 0.0004 -0.5034
+vn -0.3833 -0.6371 -0.6687
+vn -0.4966 -0.0991 -0.8623
+vn 0.2391 -0.8803 0.4097
+vn -0.4965 -0.1016 -0.8621
+vn 0.3205 -0.7707 0.5507
+vn -0.3830 -0.6371 -0.6688
+vn 0.4424 -0.4683 0.7648
+vn -0.0523 0.9948 -0.0871
+vn 0.3097 0.7752 0.5506
+vn -0.4780 0.2926 -0.8282
+vn 0.8739 0.0430 -0.4843
+vn -0.3238 0.7708 -0.5486
+vn 0.4765 0.2937 0.8287
+vn -0.1409 -0.9580 -0.2499
+vn 0.8597 0.0478 -0.5086
+vn 0.8597 0.0481 -0.5086
+vn -0.1092 -0.9260 -0.3615
+vn 0.4792 -0.2869 0.8295
+vn 0.8765 0.0291 -0.4804
+vn 0.8622 -0.0325 -0.5056
+vn 0.3143 0.7752 0.5480
+vn 0.1495 -0.9559 0.2528
+vn -0.3882 0.6316 -0.6711
+vn -0.4765 -0.2935 -0.8287
+vn 0.4965 0.1015 0.8621
+vn 0.8692 0.0187 -0.4941
+vn 0.8692 0.0187 -0.4940
+vn 0.8629 0.0143 -0.5052
+vn 0.3839 0.6370 0.6684
+vn -0.2384 0.8803 -0.4103
+vn 0.2396 -0.8803 0.4096
+vn 0.8590 0.0135 -0.5119
+vn 0.8587 0.0136 -0.5123
+vn 0.8588 0.0136 -0.5122
+vn -0.4385 -0.4744 -0.7634
+vn 0.0398 0.9955 0.0861
+vn 0.1705 -0.9785 0.1165
+vn 0.8625 0.0168 -0.5057
+vn 0.8644 0.0230 -0.5023
+vn -0.2435 0.8802 -0.4074
+vn 0.3802 0.6371 0.6705
+vn 0.8566 0.0214 -0.5155
+vn 0.8568 0.0212 -0.5152
+vn -0.2313 -0.8835 -0.4073
+vn -0.4800 0.2888 -0.8284
+vn 0.8672 -0.0005 -0.4979
+vn 0.8672 -0.0006 -0.4980
+vn 0.4777 -0.2964 0.8270
+vn -0.1405 -0.9579 -0.2502
+vn 0.2324 0.8836 0.4064
+vn 0.8654 0.0244 -0.5005
+vn -0.0413 -0.9955 -0.0854
+vn 0.8692 0.0071 -0.4944
+vn 0.8679 0.0070 -0.4967
+vn 0.8666 0.0101 -0.4989
+vn 0.8653 0.0087 -0.5012
+vn 0.8671 0.0098 -0.4980
+vn 0.8695 0.0057 -0.4940
+vn 0.8650 0.0078 -0.5017
+vn 0.8661 0.0099 -0.4998
+vn 0.8679 0.0081 -0.4968
+vn 0.8655 0.0097 -0.5008
+vn 0.8676 0.0091 -0.4972
+vn 0.8653 0.0065 -0.5013
+vn 0.8654 0.0057 -0.5010
+vn 0.8678 0.0060 -0.4968
+vn 0.8660 -0.0002 -0.5001
+vn 0.8689 -0.0144 -0.4947
+vn 0.8688 -0.0144 -0.4950
+vn 0.8680 -0.0133 -0.4964
+vn 0.8678 -0.0134 -0.4967
+vn 0.3606 -0.9071 0.2172
+vn 0.8700 0.0099 -0.4930
+vn 0.8627 0.0100 -0.5057
+vn 0.8700 0.0133 -0.4928
+vn 0.8695 0.0163 -0.4936
+vn 0.8684 0.0191 -0.4954
+vn 0.8653 0.0206 -0.5008
+vn 0.8639 0.0188 -0.5033
+vn 0.8630 0.0159 -0.5049
+vn 0.8625 0.0132 -0.5059
+vn 0.0961 -0.9613 -0.2584
+vn 0.8634 0.0071 -0.5045
+vn 0.8652 0.0054 -0.5014
+vn 0.2305 0.8835 0.4077
+vn -0.1365 -0.9579 -0.2525
+vn -0.1375 -0.9580 -0.2518
+vn 0.0434 0.9955 0.0841
+vn 0.1396 0.9580 0.2507
+vn 0.2422 -0.8802 0.4080
+vn -0.4368 -0.4744 -0.7643
+vn 0.4441 -0.4682 0.7639
+vn 0.4835 -0.2869 0.8270
+vn 0.8670 0.0203 -0.4979
+vn 0.4721 0.2935 0.8312
+vn 0.8674 0.0002 -0.4976
+vn 0.2181 0.8834 0.4148
+vn 0.3825 0.6370 0.6692
+vn -0.2527 0.8801 -0.4020
+vn -0.3319 0.7707 -0.5440
+vn -0.4836 0.2868 -0.8270
+vn -0.3804 -0.6371 -0.6704
+vn -0.3100 -0.7752 -0.5505
+vn -0.1504 0.9559 -0.2523
+vn -0.2403 0.8803 -0.4091
+vn -0.4722 -0.2937 -0.8312
+vn -0.4431 0.4683 -0.7644
+vn -0.4804 0.2869 -0.8288
+vn -0.4753 -0.2937 -0.8294
+vn -0.4362 -0.4744 -0.7646
+vn 0.4753 0.2995 0.8273
+vn 0.1524 -0.9559 0.2511
+vn 0.8674 0.0065 -0.4975
+vn 0.8670 0.0092 -0.4982
+vn 0.8674 0.0053 -0.4976
+vn 0.8660 0.0101 -0.4999
+vn 0.8676 0.0076 -0.4972
+vn 0.8657 0.0093 -0.5005
+vn 0.8649 0.0069 -0.5019
+vn 0.8649 0.0061 -0.5020
+vn 0.8654 0.0001 -0.5010
+vn 0.8640 -0.0144 -0.5034
+vn 0.2750 -0.9600 0.0520
+vn 0.8706 0.0118 -0.4919
+vn 0.8705 0.0151 -0.4920
+vn 0.8697 0.0177 -0.4933
+vn 0.8654 0.0208 -0.5007
+vn 0.8627 0.0177 -0.5054
+vn 0.8620 0.0151 -0.5067
+vn -0.0039 -0.9097 -0.4151
+vn 0.8625 0.0087 -0.5060
+vn 0.8633 0.0056 -0.5046
+vn -0.4961 -0.0991 -0.8626
+vn 0.8669 0.0208 -0.4980
+vn 0.8667 -0.0002 -0.4989
+vn -0.4741 -0.3032 -0.8266
+vn -0.4787 0.2927 -0.8278
+vn 0.2432 -0.8802 0.4075
+vn 0.8652 0.0087 -0.5013
+vn 0.8672 0.0098 -0.4979
+vn 0.8695 0.0057 -0.4939
+vn 0.8652 0.0075 -0.5014
+vn 0.8660 0.0099 -0.4999
+vn 0.8679 0.0081 -0.4967
+vn 0.8654 0.0097 -0.5010
+vn 0.8674 0.0088 -0.4975
+vn 0.8651 0.0067 -0.5016
+vn 0.8655 0.0057 -0.5009
+vn 0.8676 0.0059 -0.4972
+vn 0.8660 -0.0003 -0.5001
+vn 0.8689 -0.0144 -0.4948
+vn 0.8678 -0.0134 -0.4968
+vn 0.3604 -0.9073 0.2168
+vn 0.8699 0.0100 -0.4931
+vn 0.8628 0.0096 -0.5054
+vn 0.8701 0.0133 -0.4928
+vn 0.8701 0.0134 -0.4927
+vn 0.8700 0.0130 -0.4929
+vn 0.8695 0.0164 -0.4936
+vn 0.8684 0.0187 -0.4955
+vn 0.8639 0.0188 -0.5032
+vn 0.8629 0.0163 -0.5051
+vn 0.8636 0.0069 -0.5042
+vn 0.8652 0.0053 -0.5014
+vn 0.8654 0.0053 -0.5011
+vn 0.8672 0.0002 -0.4979
+vn 0.2181 0.8834 0.4149
+vn -0.4835 0.2868 -0.8270
+vn -0.4722 -0.2937 -0.8311
+vn -0.0448 -0.9955 -0.0834
+vn 0.8731 0.0170 -0.4873
+vn 0.8729 0.0169 -0.4876
+vn 0.8702 0.0173 -0.4923
+vn 0.8668 0.0245 -0.4981
+vn 0.8668 0.0247 -0.4981
+vn 0.8638 0.0206 -0.5035
+vn 0.8680 0.0238 -0.4959
+vn 0.8737 0.0138 -0.4862
+vn 0.8633 0.0184 -0.5043
+vn 0.8655 0.0244 -0.5004
+vn 0.8655 0.0241 -0.5004
+vn 0.8697 0.0195 -0.4931
+vn 0.8641 0.0238 -0.5028
+vn 0.8640 0.0241 -0.5029
+vn 0.8690 0.0215 -0.4943
+vn 0.8635 0.0159 -0.5041
+vn 0.8641 0.0137 -0.5031
+vn 0.8641 0.0137 -0.5032
+vn 0.8696 0.0144 -0.4935
+vn 0.8696 0.0144 -0.4936
+vn 0.8655 -0.0005 -0.5009
+vn 0.8719 -0.0347 -0.4885
+vn 0.8718 -0.0347 -0.4886
+vn 0.8695 -0.0323 -0.4928
+vn 0.2614 -0.9232 0.2816
+vn 0.8748 0.0239 -0.4839
+vn 0.8573 0.0238 -0.5143
+vn 0.8573 0.0237 -0.5143
+vn 0.8749 0.0320 -0.4832
+vn 0.8735 0.0394 -0.4852
+vn 0.8735 0.0391 -0.4852
+vn 0.8735 0.0396 -0.4852
+vn 0.8707 0.0455 -0.4896
+vn 0.8632 0.0496 -0.5024
+vn 0.8599 0.0459 -0.5085
+vn 0.8576 0.0394 -0.5128
+vn 0.8576 0.0393 -0.5128
+vn 0.8567 0.0316 -0.5148
+vn 0.8568 0.0314 -0.5147
+vn -0.0120 -0.9797 -0.1999
+vn 0.8595 0.0166 -0.5109
+vn 0.8634 0.0129 -0.5043
+vn -0.3193 0.7709 -0.5511
+vn 0.3142 0.7753 0.5479
+vn 0.2325 0.8836 0.4066
+vn 0.0531 -0.9948 0.0865
+vn 0.2391 -0.8803 0.4098
+vn 0.3200 -0.7709 0.5508
+vn -0.4384 -0.4745 -0.7633
+vn 0.4805 -0.2871 0.8287
+vn 0.4361 0.4743 0.7647
+vn 0.8673 0.0493 -0.4954
+vn 0.4752 0.2935 0.8295
+vn 0.8686 0.0005 -0.4956
+vn 0.2273 0.8836 0.4094
+vn -0.2435 0.8802 -0.4073
+vn -0.3919 0.6315 -0.6690
+vn -0.4805 0.2869 -0.8287
+vn 0.3888 -0.6316 0.6708
+vn -0.0443 -0.9955 -0.0837
+vn -0.1482 0.9559 -0.2536
+vn -0.4420 0.4683 -0.7651
+vn -0.4793 0.2868 -0.8295
+vn -0.4382 -0.4745 -0.7634
+vn 0.4760 0.2995 0.8269
+vn 0.1490 -0.9559 0.2531
+vn 0.0536 -0.9948 0.0863
+vn 0.1494 -0.9559 0.2527
+vn 0.3889 -0.6319 0.6705
+vn 0.4426 -0.4684 0.7647
+vn -0.4793 0.2867 -0.8295
+vn -0.1410 -0.9579 -0.2501
+vn -0.0447 -0.9955 -0.0833
+vn 0.4966 0.1089 0.8612
+vn 0.8673 -0.0005 -0.4978
+vn 0.8674 -0.0005 -0.4976
+vn 0.8673 -0.0005 -0.4977
+vn 0.8721 -0.0347 -0.4881
+vn 0.8720 -0.0347 -0.4884
+vn 0.8720 -0.0347 -0.4883
+vn 0.8671 0.0502 -0.4956
+vn 0.8684 0.0137 -0.4956
+vn 0.8691 0.0218 -0.4942
+vn 0.3098 0.7751 0.5507
+vn -0.0522 0.9948 -0.0869
+vn 0.4426 -0.4683 0.7647
+vn -0.3831 -0.6370 -0.6689
+vn 0.8634 0.0129 -0.5044
+vn 0.8624 0.0168 -0.5059
+vn -0.4447 0.4682 -0.7635
+vn 0.8766 0.0294 -0.4803
+vn 0.8767 0.0297 -0.4801
+vn 0.8766 0.0293 -0.4804
+vn 0.8681 0.0238 -0.4959
+vn 0.8673 0.0493 -0.4953
+vn -0.4799 0.2889 -0.8284
+vn 0.8677 0.0226 -0.4965
+vn 0.0399 0.9955 0.0863
+vn 0.8637 0.0206 -0.5035
+vn 0.3098 0.7752 0.5506
+vn 0.8748 0.0239 -0.4840
+vn 0.8594 0.0166 -0.5110
+vn 0.8622 -0.0325 -0.5055
+vn 0.0455 0.9955 0.0827
+vn 0.8656 -0.0005 -0.5008
+vn -0.3239 0.7707 -0.5487
+vn 0.3890 -0.6317 0.6705
+vn 0.8633 0.0217 -0.5042
+vn 0.8634 0.0217 -0.5041
+vn 0.8759 0.0366 -0.4811
+vn 0.8641 0.0004 -0.5034
+vn 0.3890 -0.6317 0.6706
+vn -0.1536 0.9559 -0.2502
+vn 0.8693 0.0142 -0.4941
+vn 0.2395 -0.8803 0.4096
+vn 0.8633 0.0161 -0.5045
+vn 0.8631 0.0163 -0.5047
+vn 0.1702 -0.9786 0.1160
+vn 0.8668 0.0244 -0.4981
+vn 0.8568 0.0434 -0.5138
+vn 0.8644 0.0226 -0.5023
+vn 0.8644 0.0225 -0.5023
+vn 0.8644 0.0227 -0.5023
+vn 0.1362 0.9579 0.2526
+vn -0.3136 -0.7751 -0.5485
+vn 0.8667 0.0240 -0.4983
+vn -0.0581 0.9948 -0.0837
+vn 0.8633 0.0505 -0.5022
+vn 0.8577 0.0389 -0.5127
+vn 0.8574 0.0237 -0.5142
+vn 0.8696 0.0195 -0.4933
+vn 0.8730 0.0171 -0.4874
+vn 0.0535 -0.9948 0.0861
+vn -0.4966 -0.0986 -0.8624
+vn -0.4966 -0.0987 -0.8624
+vn 0.8567 0.0315 -0.5148
+vn 0.8569 0.0312 -0.5146
+vn -0.4974 0.0943 -0.8624
+vn 0.3840 0.6369 0.6685
+vn -0.3881 0.6317 -0.6710
+vn 0.8750 0.0326 -0.4829
+vn 0.4792 -0.2870 0.8294
+vn 0.8697 -0.0323 -0.4925
+vn -0.2313 -0.8836 -0.4072
+vn 0.8701 0.0171 -0.4926
+vn 0.8735 0.0397 -0.4851
+vn -0.4751 -0.3031 -0.8260
+vn 0.8690 0.0161 -0.4945
+vn 0.8689 0.0159 -0.4946
+vn 0.8689 0.0006 -0.4950
+vn 0.4760 0.2994 0.8269
+vn 0.8641 0.0237 -0.5027
+vn 0.4424 -0.4682 0.7649
+vn 0.8597 -0.0347 -0.5095
+vn -0.3132 -0.7752 -0.5486
+vn 0.4965 0.1016 0.8621
+vn 0.8628 0.0144 -0.5053
+vn 0.8567 0.0213 -0.5154
+vn 0.4752 0.2955 0.8288
+vn 0.8690 0.0131 -0.4946
+vn 0.8626 0.0195 -0.5056
+vn 0.8628 0.0192 -0.5052
+vn 0.8555 0.0283 -0.5170
+vn 0.8642 0.0136 -0.5029
+vn 0.8738 0.0430 -0.4843
+vn 0.8688 0.0210 -0.4948
+vn 0.8664 0.0044 -0.4994
+vn 0.8664 0.0040 -0.4994
+vn 0.8664 0.0045 -0.4994
+vn 0.8661 0.0042 -0.4999
+vn 0.8663 0.0040 -0.4994
+vn 0.8664 0.0044 -0.4993
+vn 0.8664 0.0046 -0.4994
+vn 0.8664 0.0045 -0.4993
+vn 0.8663 0.0043 -0.4995
+vn 0.8662 0.0044 -0.4996
+vn 0.8661 0.0043 -0.4999
+vn 0.8666 0.0038 -0.4990
+vn 0.8664 0.0038 -0.4993
+vn 0.8663 0.0041 -0.4995
+vn 0.8664 0.0039 -0.4994
+vn 0.8664 0.0041 -0.4993
+vn 0.8663 0.0039 -0.4994
+vn 0.8670 0.0042 -0.4983
+vn 0.8664 0.0043 -0.4993
+vn 0.8662 0.0109 -0.4996
+vn 0.8664 0.0097 -0.4993
+vn 0.8666 0.0092 -0.4989
+vn 0.8662 0.0107 -0.4996
+vn 0.8678 0.0102 -0.4969
+vn 0.8678 0.0102 -0.4968
+vn 0.8664 0.0095 -0.4992
+vn 0.8662 0.0102 -0.4997
+vn 0.8662 0.0099 -0.4995
+vn 0.8664 0.0109 -0.4992
+vn 0.8663 0.0095 -0.4994
+vn 0.8662 0.0092 -0.4996
+vn 0.8664 0.0098 -0.4993
+vn 0.8664 0.0099 -0.4993
+vn 0.8663 0.0096 -0.4994
+vn 0.8664 0.0106 -0.4993
+vn 0.8663 0.0107 -0.4994
+vn 0.8663 0.0109 -0.4994
+vn 0.8664 0.0104 -0.4992
+vn 0.8664 0.0109 -0.4993
+vn 0.8663 0.0110 -0.4994
+vn 0.8663 0.0108 -0.4994
+vn 0.8664 0.0108 -0.4993
+vn 0.8664 0.0108 -0.4992
+vn 0.8664 0.0093 -0.4992
+vn 0.8658 0.0100 -0.5002
+vn 0.8664 0.0093 -0.4993
+vn 0.8664 0.0107 -0.4993
+vn 0.8665 0.0107 -0.4990
+vn 0.8662 0.0104 -0.4995
+vn 0.8665 0.0104 -0.4991
+vn 0.8662 0.0041 -0.4997
+vn 0.8663 0.0047 -0.4995
+vn 0.8662 0.0041 -0.4996
+vn 0.8664 0.0040 -0.4993
+vn 0.8665 0.0044 -0.4991
+vn 0.8662 0.0047 -0.4997
+vn 0.8664 0.0039 -0.4993
+vn 0.8663 0.0044 -0.4995
+vn 0.8665 0.0038 -0.4992
+vn 0.8664 0.0046 -0.4993
+vn 0.8665 0.0044 -0.4992
+vn 0.8662 0.0037 -0.4998
+vn 0.8668 0.0042 -0.4987
+vn 0.8659 0.0099 -0.5001
+vn 0.8662 0.0103 -0.4996
+vn 0.8664 0.0096 -0.4993
+vn 0.8679 0.0103 -0.4966
+vn 0.8679 0.0102 -0.4967
+vn 0.8669 0.0089 -0.4984
+vn 0.8663 0.0107 -0.4995
+vn 0.8668 0.0104 -0.4985
+vn 0.8667 0.0106 -0.4987
+vn 0.8665 0.0096 -0.4990
+vn 0.8665 0.0096 -0.4992
+vn 0.8662 0.0097 -0.4996
+vn 0.8662 0.0096 -0.4995
+vn 0.8663 0.0097 -0.4994
+vn 0.8663 0.0107 -0.4993
+vn 0.8662 0.0095 -0.4996
+vn 0.8659 0.0100 -0.5002
+vn 0.8663 0.0108 -0.4993
+vn 0.8663 0.0096 -0.4995
+vt 0.357039 0.250000
+vt 0.785559 0.750000
+vt 0.214079 0.250000
+vt 0.714079 0.750000
+vt 0.285559 0.250000
+vt 0.571118 0.250000
+vt 0.285559 0.250000
+vt 0.642599 0.250000
+vt 0.928520 0.154203
+vt 0.499638 0.250000
+vt 0.714079 0.750000
+vt 0.571118 0.250000
+vt 0.642599 0.750000
+vt 0.000000 0.250000
+vt 0.857040 0.154203
+vt 0.071480 0.250000
+vt 0.857040 0.480583
+vt 0.357039 0.095797
+vt 0.785559 0.904203
+vt 0.785559 1.000000
+vt 0.357039 0.000000
+vt 0.785559 0.043154
+vt 0.714079 1.000000
+vt 0.214079 0.000000
+vt 0.857040 0.043154
+vt 0.928520 0.480583
+vt 0.214079 0.095797
+vt 0.714079 0.904203
+vt 0.285559 0.095797
+vt 0.571118 0.095797
+vt 0.285559 0.000000
+vt 0.571118 0.000000
+vt 0.857040 0.000000
+vt 0.285559 0.000000
+vt 0.642598 0.000000
+vt 0.785559 0.000000
+vt 0.285559 0.095797
+vt 0.642599 0.095797
+vt 0.857040 0.394495
+vt 0.928520 0.000000
+vt 0.499638 0.095797
+vt 0.499638 0.000000
+vt 0.785559 0.129242
+vt 0.714079 1.000000
+vt 0.571118 0.000000
+vt 0.785559 0.172396
+vt 0.714079 0.904203
+vt 0.571118 0.095797
+vt 0.642599 0.904203
+vt 0.000000 0.095797
+vt 0.642599 1.000000
+vt 0.000000 0.000000
+vt 0.857040 0.172396
+vt 0.071480 0.000000
+vt 0.857040 0.129242
+vt 0.928520 0.394495
+vt 0.857040 0.000000
+vt 0.071480 0.095797
+vt 0.578089 0.326242
+vt 0.370472 0.723575
+vt 0.442971 0.724510
+vt 0.846857 0.620495
+vt 0.579868 0.178984
+vt 0.229113 0.694984
+vt 0.442901 0.694984
+vt 0.817898 0.626255
+vt 0.589057 0.151745
+vt 0.229049 0.665457
+vt 0.442837 0.665456
+vt 0.788371 0.626255
+vt 0.605204 0.126737
+vt 0.228992 0.635930
+vt 0.442780 0.635930
+vt 0.759412 0.620495
+vt 0.626230 0.105756
+vt 0.228946 0.606403
+vt 0.442734 0.606402
+vt 0.732132 0.609195
+vt 0.650942 0.089526
+vt 0.228912 0.576877
+vt 0.442700 0.576876
+vt 0.707581 0.592792
+vt 0.678244 0.078632
+vt 0.228890 0.547350
+vt 0.442678 0.547349
+vt 0.686702 0.571913
+vt 0.228883 0.517823
+vt 0.706979 0.073452
+vt 0.442670 0.517822
+vt 0.670298 0.547363
+vt 0.228889 0.488295
+vt 0.735929 0.074111
+vt 0.442678 0.488295
+vt 0.658999 0.520082
+vt 0.763884 0.080481
+vt 0.228910 0.458769
+vt 0.442698 0.458768
+vt 0.653238 0.491123
+vt 0.789700 0.092175
+vt 0.228944 0.429242
+vt 0.442732 0.429241
+vt 0.653238 0.461596
+vt 0.812355 0.108585
+vt 0.228990 0.399716
+vt 0.442778 0.399714
+vt 0.658998 0.432637
+vt 0.831035 0.128981
+vt 0.229046 0.370189
+vt 0.442833 0.370188
+vt 0.670298 0.405358
+vt 0.844628 0.151667
+vt 0.229109 0.340661
+vt 0.442897 0.340661
+vt 0.686702 0.380807
+vt 0.879631 0.287703
+vt 0.370467 0.312068
+vt 0.442967 0.311134
+vt 0.707581 0.359928
+vt 0.370705 0.281246
+vt 0.885704 0.319257
+vt 0.443056 0.281620
+vt 0.732120 0.343530
+vt 0.371348 0.252525
+vt 0.885423 0.348954
+vt 0.443160 0.252098
+vt 0.759394 0.332230
+vt 0.879533 0.378395
+vt 0.371442 0.223476
+vt 0.443252 0.222558
+vt 0.788366 0.326464
+vt 0.867982 0.406513
+vt 0.371495 0.193949
+vt 0.443305 0.193031
+vt 0.817893 0.326463
+vt 0.851292 0.431762
+vt 0.371495 0.164423
+vt 0.443305 0.163503
+vt 0.846854 0.332224
+vt 0.830124 0.453224
+vt 0.371427 0.134897
+vt 0.443237 0.133976
+vt 0.874134 0.343522
+vt 0.805300 0.470133
+vt 0.371280 0.105372
+vt 0.443090 0.104449
+vt 0.898684 0.359926
+vt 0.777761 0.481894
+vt 0.371043 0.075847
+vt 0.442853 0.074922
+vt 0.919564 0.380805
+vt 0.748545 0.488099
+vt 0.370694 0.989321
+vt 0.370712 0.046322
+vt 0.442504 0.990249
+vt 0.442522 0.045396
+vt 0.935969 0.405357
+vt 0.718747 0.488548
+vt 0.371029 0.959797
+vt 0.442838 0.960722
+vt 0.947270 0.432638
+vt 0.689480 0.483245
+vt 0.371269 0.930272
+vt 0.443079 0.931196
+vt 0.953029 0.461597
+vt 0.661830 0.472412
+vt 0.371421 0.900745
+vt 0.443231 0.901668
+vt 0.953029 0.491125
+vt 0.371492 0.871219
+vt 0.636826 0.456466
+vt 0.443302 0.872141
+vt 0.947269 0.520085
+vt 0.615392 0.436018
+vt 0.371495 0.841694
+vt 0.443305 0.842613
+vt 0.935968 0.547365
+vt 0.598317 0.411848
+vt 0.371443 0.812167
+vt 0.443254 0.813086
+vt 0.919563 0.571916
+vt 0.371352 0.783117
+vt 0.586362 0.385333
+vt 0.443163 0.783546
+vt 0.898673 0.592803
+vt 0.370709 0.754396
+vt 0.579568 0.357437
+vt 0.443060 0.754024
+vt 0.874125 0.609201
+vt 0.571207 0.326153
+vt 0.177644 0.253134
+vt 0.006929 0.253167
+vt 0.319029 0.223358
+vt 0.573209 0.178195
+vt 0.007025 0.222313
+vt 0.582798 0.148853
+vt 0.319126 0.192504
+vt 0.007122 0.191458
+vt 0.599888 0.122511
+vt 0.319224 0.161649
+vt 0.007220 0.160604
+vt 0.621946 0.100560
+vt 0.319322 0.130795
+vt 0.007319 0.129750
+vt 0.647824 0.083609
+vt 0.319423 0.099941
+vt 0.007419 0.098896
+vt 0.676401 0.072253
+vt 0.319524 0.069087
+vt 0.007520 0.068042
+vt 0.706468 0.066881
+vt 0.319626 0.038233
+vt 0.007623 0.037188
+vt 0.319730 0.007378
+vt 0.736753 0.067631
+vt 0.319730 0.992485
+vt 0.007727 0.006334
+vt 0.007728 0.993666
+vt 0.765981 0.074365
+vt 0.319616 0.961631
+vt 0.007613 0.962812
+vt 0.792953 0.086670
+vt 0.319502 0.930776
+vt 0.007499 0.931958
+vt 0.816609 0.103895
+vt 0.319390 0.899923
+vt 0.007387 0.901104
+vt 0.836176 0.125250
+vt 0.319278 0.869069
+vt 0.007276 0.870250
+vt 0.850517 0.149568
+vt 0.319168 0.838215
+vt 0.007165 0.839396
+vt 0.886223 0.285996
+vt 0.177771 0.808498
+vt 0.007055 0.808541
+vt 0.177512 0.777645
+vt 0.892437 0.317355
+vt 0.006941 0.777687
+vt 0.176852 0.746796
+vt 0.892437 0.349226
+vt 0.006827 0.746834
+vt 0.886228 0.380492
+vt 0.176742 0.715941
+vt 0.006716 0.715979
+vt 0.874138 0.409852
+vt 0.176640 0.685087
+vt 0.006614 0.685125
+vt 0.856681 0.436209
+vt 0.176549 0.654233
+vt 0.006524 0.654271
+vt 0.834553 0.458608
+vt 0.176474 0.623379
+vt 0.006448 0.623417
+vt 0.808610 0.476251
+vt 0.176415 0.592525
+vt 0.006390 0.592562
+vt 0.779839 0.488518
+vt 0.176377 0.561671
+vt 0.006351 0.561708
+vt 0.749318 0.494983
+vt 0.176359 0.530816
+vt 0.006334 0.530854
+vt 0.176362 0.499962
+vt 0.718195 0.495436
+vt 0.006337 0.500000
+vt 0.687628 0.489885
+vt 0.176387 0.469108
+vt 0.006361 0.469146
+vt 0.658752 0.478560
+vt 0.176430 0.438254
+vt 0.006405 0.438291
+vt 0.632640 0.461897
+vt 0.176492 0.407399
+vt 0.006467 0.407437
+vt 0.610260 0.440534
+vt 0.176568 0.376545
+vt 0.006543 0.376583
+vt 0.592437 0.415287
+vt 0.176656 0.345691
+vt 0.006631 0.345729
+vt 0.176753 0.314837
+vt 0.579817 0.387126
+vt 0.006727 0.314875
+vt 0.177399 0.283987
+vt 0.572791 0.357181
+vt 0.006828 0.284020
+vt 0.578089 0.326242
+vt 0.370472 0.723575
+vt 0.442971 0.724510
+vt 0.231212 0.903091
+vt 0.579868 0.178984
+vt 0.229113 0.694984
+vt 0.442901 0.694984
+vt 0.214996 0.906316
+vt 0.589057 0.151745
+vt 0.229049 0.665457
+vt 0.442837 0.665456
+vt 0.198460 0.906317
+vt 0.605204 0.126737
+vt 0.228992 0.635930
+vt 0.442780 0.635930
+vt 0.182243 0.903091
+vt 0.626230 0.105756
+vt 0.228946 0.606403
+vt 0.442734 0.606402
+vt 0.166967 0.896763
+vt 0.650942 0.089526
+vt 0.228912 0.576877
+vt 0.442700 0.576876
+vt 0.153218 0.887577
+vt 0.678244 0.078632
+vt 0.228890 0.547350
+vt 0.442678 0.547349
+vt 0.141526 0.875885
+vt 0.228883 0.517823
+vt 0.706979 0.073452
+vt 0.442670 0.517822
+vt 0.132340 0.862137
+vt 0.228889 0.488295
+vt 0.735929 0.074111
+vt 0.442678 0.488295
+vt 0.126012 0.846860
+vt 0.763884 0.080481
+vt 0.228910 0.458769
+vt 0.442698 0.458768
+vt 0.122786 0.830643
+vt 0.789700 0.092175
+vt 0.228944 0.429242
+vt 0.442732 0.429241
+vt 0.122786 0.814107
+vt 0.812355 0.108585
+vt 0.228990 0.399716
+vt 0.442778 0.399714
+vt 0.126012 0.797890
+vt 0.831035 0.128981
+vt 0.229046 0.370189
+vt 0.442833 0.370188
+vt 0.132339 0.782614
+vt 0.844628 0.151667
+vt 0.229109 0.340661
+vt 0.442897 0.340661
+vt 0.141526 0.768865
+vt 0.879631 0.287703
+vt 0.370467 0.312068
+vt 0.442967 0.311134
+vt 0.153218 0.757173
+vt 0.370705 0.281246
+vt 0.885704 0.319257
+vt 0.443056 0.281620
+vt 0.166960 0.747990
+vt 0.371348 0.252525
+vt 0.885423 0.348954
+vt 0.443160 0.252098
+vt 0.182233 0.741662
+vt 0.879533 0.378395
+vt 0.371442 0.223476
+vt 0.443252 0.222558
+vt 0.198458 0.738433
+vt 0.867982 0.406513
+vt 0.371495 0.193949
+vt 0.443305 0.193031
+vt 0.214993 0.738433
+vt 0.851292 0.431762
+vt 0.371495 0.164423
+vt 0.443305 0.163503
+vt 0.231211 0.741659
+vt 0.830124 0.453224
+vt 0.371427 0.134897
+vt 0.443237 0.133976
+vt 0.246487 0.747986
+vt 0.805300 0.470133
+vt 0.371280 0.105372
+vt 0.443090 0.104449
+vt 0.260236 0.757172
+vt 0.777761 0.481894
+vt 0.371043 0.075847
+vt 0.442853 0.074922
+vt 0.271928 0.768865
+vt 0.748545 0.488099
+vt 0.370694 0.989321
+vt 0.370712 0.046322
+vt 0.442504 0.990249
+vt 0.442522 0.045396
+vt 0.281116 0.782613
+vt 0.718747 0.488548
+vt 0.371029 0.959797
+vt 0.442838 0.960722
+vt 0.287444 0.797891
+vt 0.689480 0.483245
+vt 0.371269 0.930272
+vt 0.443079 0.931196
+vt 0.290669 0.814108
+vt 0.661830 0.472412
+vt 0.371421 0.900745
+vt 0.443231 0.901668
+vt 0.290669 0.830644
+vt 0.371492 0.871219
+vt 0.636826 0.456466
+vt 0.443302 0.872141
+vt 0.287443 0.846861
+vt 0.615392 0.436018
+vt 0.371495 0.841694
+vt 0.443305 0.842613
+vt 0.281115 0.862138
+vt 0.598317 0.411848
+vt 0.371443 0.812167
+vt 0.443254 0.813086
+vt 0.271928 0.875886
+vt 0.371352 0.783117
+vt 0.586362 0.385333
+vt 0.443163 0.783546
+vt 0.260230 0.887583
+vt 0.370709 0.754396
+vt 0.579568 0.357437
+vt 0.443060 0.754024
+vt 0.246483 0.896766
+vt 0.571207 0.326153
+vt 0.177644 0.253134
+vt 0.006929 0.253167
+vt 0.319029 0.223358
+vt 0.573209 0.178195
+vt 0.007025 0.222313
+vt 0.582798 0.148853
+vt 0.319126 0.192504
+vt 0.007122 0.191458
+vt 0.599888 0.122511
+vt 0.319224 0.161649
+vt 0.007220 0.160604
+vt 0.621946 0.100560
+vt 0.319322 0.130795
+vt 0.007319 0.129750
+vt 0.647824 0.083609
+vt 0.319423 0.099941
+vt 0.007419 0.098896
+vt 0.676401 0.072253
+vt 0.319524 0.069087
+vt 0.007520 0.068042
+vt 0.706468 0.066881
+vt 0.319626 0.038233
+vt 0.007623 0.037188
+vt 0.319730 0.007378
+vt 0.736753 0.067631
+vt 0.319730 0.992485
+vt 0.007727 0.006334
+vt 0.007728 0.993666
+vt 0.765981 0.074365
+vt 0.319616 0.961631
+vt 0.007613 0.962812
+vt 0.792953 0.086670
+vt 0.319502 0.930776
+vt 0.007499 0.931958
+vt 0.816609 0.103895
+vt 0.319390 0.899923
+vt 0.007387 0.901104
+vt 0.836176 0.125250
+vt 0.319278 0.869069
+vt 0.007276 0.870250
+vt 0.850517 0.149568
+vt 0.319168 0.838215
+vt 0.007165 0.839396
+vt 0.886223 0.285996
+vt 0.177771 0.808498
+vt 0.007055 0.808541
+vt 0.177512 0.777645
+vt 0.892437 0.317355
+vt 0.006941 0.777687
+vt 0.176852 0.746796
+vt 0.892437 0.349226
+vt 0.006827 0.746834
+vt 0.886228 0.380492
+vt 0.176742 0.715941
+vt 0.006716 0.715979
+vt 0.874138 0.409852
+vt 0.176640 0.685087
+vt 0.006614 0.685125
+vt 0.856681 0.436209
+vt 0.176549 0.654233
+vt 0.006524 0.654271
+vt 0.834553 0.458608
+vt 0.176474 0.623379
+vt 0.006448 0.623417
+vt 0.808610 0.476251
+vt 0.176415 0.592525
+vt 0.006390 0.592562
+vt 0.779839 0.488518
+vt 0.176377 0.561671
+vt 0.006351 0.561708
+vt 0.749318 0.494983
+vt 0.176359 0.530816
+vt 0.006334 0.530854
+vt 0.176362 0.499962
+vt 0.718195 0.495436
+vt 0.006337 0.500000
+vt 0.687628 0.489885
+vt 0.176387 0.469108
+vt 0.006361 0.469146
+vt 0.658752 0.478560
+vt 0.176430 0.438254
+vt 0.006405 0.438291
+vt 0.632640 0.461897
+vt 0.176492 0.407399
+vt 0.006467 0.407437
+vt 0.610260 0.440534
+vt 0.176568 0.376545
+vt 0.006543 0.376583
+vt 0.592437 0.415287
+vt 0.176656 0.345691
+vt 0.006631 0.345729
+vt 0.176753 0.314837
+vt 0.579817 0.387126
+vt 0.006727 0.314875
+vt 0.177399 0.283987
+vt 0.572791 0.357181
+vt 0.006828 0.284020
+vt 0.578089 0.326242
+vt 0.370472 0.723575
+vt 0.442971 0.724510
+vt 0.278498 0.614635
+vt 0.579868 0.178984
+vt 0.229113 0.694984
+vt 0.442901 0.694984
+vt 0.249539 0.620395
+vt 0.589057 0.151745
+vt 0.229049 0.665457
+vt 0.442837 0.665456
+vt 0.220011 0.620396
+vt 0.605204 0.126737
+vt 0.228992 0.635930
+vt 0.442780 0.635930
+vt 0.191052 0.614636
+vt 0.626230 0.105756
+vt 0.228946 0.606403
+vt 0.442734 0.606402
+vt 0.163773 0.603336
+vt 0.650942 0.089526
+vt 0.228912 0.576877
+vt 0.442700 0.576876
+vt 0.139222 0.586933
+vt 0.678244 0.078632
+vt 0.228890 0.547350
+vt 0.442678 0.547349
+vt 0.118343 0.566054
+vt 0.228883 0.517823
+vt 0.706979 0.073452
+vt 0.442670 0.517822
+vt 0.101939 0.541503
+vt 0.228889 0.488295
+vt 0.735929 0.074111
+vt 0.442678 0.488295
+vt 0.090640 0.514223
+vt 0.763884 0.080481
+vt 0.228910 0.458769
+vt 0.442698 0.458768
+vt 0.084879 0.485264
+vt 0.789700 0.092175
+vt 0.228944 0.429242
+vt 0.442732 0.429241
+vt 0.084879 0.455737
+vt 0.812355 0.108585
+vt 0.228990 0.399716
+vt 0.442778 0.399714
+vt 0.090639 0.426777
+vt 0.831035 0.128981
+vt 0.229046 0.370189
+vt 0.442833 0.370188
+vt 0.101938 0.399499
+vt 0.844628 0.151667
+vt 0.229109 0.340661
+vt 0.442897 0.340661
+vt 0.118343 0.374948
+vt 0.879631 0.287703
+vt 0.370467 0.312068
+vt 0.442967 0.311134
+vt 0.139222 0.354069
+vt 0.370705 0.281246
+vt 0.885704 0.319257
+vt 0.443056 0.281620
+vt 0.163760 0.337671
+vt 0.371348 0.252525
+vt 0.885423 0.348954
+vt 0.443160 0.252098
+vt 0.191035 0.326370
+vt 0.879533 0.378395
+vt 0.371442 0.223476
+vt 0.443252 0.222558
+vt 0.220007 0.320605
+vt 0.867982 0.406513
+vt 0.371495 0.193949
+vt 0.443305 0.193031
+vt 0.249534 0.320604
+vt 0.851292 0.431762
+vt 0.371495 0.164423
+vt 0.443305 0.163503
+vt 0.278494 0.326364
+vt 0.830124 0.453224
+vt 0.371427 0.134897
+vt 0.443237 0.133976
+vt 0.305774 0.337663
+vt 0.805300 0.470133
+vt 0.371280 0.105372
+vt 0.443090 0.104449
+vt 0.330325 0.354067
+vt 0.777761 0.481894
+vt 0.371043 0.075847
+vt 0.442853 0.074922
+vt 0.351205 0.374946
+vt 0.748545 0.488099
+vt 0.370694 0.989321
+vt 0.370712 0.046322
+vt 0.442504 0.990249
+vt 0.442522 0.045396
+vt 0.367610 0.399497
+vt 0.718747 0.488548
+vt 0.371029 0.959797
+vt 0.442838 0.960722
+vt 0.378910 0.426778
+vt 0.689480 0.483245
+vt 0.371269 0.930272
+vt 0.443079 0.931196
+vt 0.384670 0.455738
+vt 0.661830 0.472412
+vt 0.371421 0.900745
+vt 0.443231 0.901668
+vt 0.384670 0.485266
+vt 0.371492 0.871219
+vt 0.636826 0.456466
+vt 0.443302 0.872141
+vt 0.378909 0.514226
+vt 0.615392 0.436018
+vt 0.371495 0.841694
+vt 0.443305 0.842613
+vt 0.367609 0.541505
+vt 0.598317 0.411848
+vt 0.371443 0.812167
+vt 0.443254 0.813086
+vt 0.351204 0.566056
+vt 0.371352 0.783117
+vt 0.586362 0.385333
+vt 0.443163 0.783546
+vt 0.330314 0.586943
+vt 0.370709 0.754396
+vt 0.579568 0.357437
+vt 0.443060 0.754024
+vt 0.305766 0.603342
+vt 0.571207 0.326153
+vt 0.177644 0.253134
+vt 0.006929 0.253167
+vt 0.319029 0.223358
+vt 0.573209 0.178195
+vt 0.007025 0.222313
+vt 0.582798 0.148853
+vt 0.319126 0.192504
+vt 0.007122 0.191458
+vt 0.599888 0.122511
+vt 0.319224 0.161649
+vt 0.007220 0.160604
+vt 0.621946 0.100560
+vt 0.319322 0.130795
+vt 0.007319 0.129750
+vt 0.647824 0.083609
+vt 0.319423 0.099941
+vt 0.007419 0.098896
+vt 0.676401 0.072253
+vt 0.319524 0.069087
+vt 0.007520 0.068042
+vt 0.706468 0.066881
+vt 0.319626 0.038233
+vt 0.007623 0.037188
+vt 0.319730 0.007378
+vt 0.736753 0.067631
+vt 0.319730 0.992485
+vt 0.007727 0.006334
+vt 0.007728 0.993666
+vt 0.765981 0.074365
+vt 0.319616 0.961631
+vt 0.007613 0.962812
+vt 0.792953 0.086670
+vt 0.319502 0.930776
+vt 0.007499 0.931958
+vt 0.816609 0.103895
+vt 0.319390 0.899923
+vt 0.007387 0.901104
+vt 0.836176 0.125250
+vt 0.319278 0.869069
+vt 0.007276 0.870250
+vt 0.850517 0.149568
+vt 0.319168 0.838215
+vt 0.007165 0.839396
+vt 0.886223 0.285996
+vt 0.177771 0.808498
+vt 0.007055 0.808541
+vt 0.177512 0.777645
+vt 0.892437 0.317355
+vt 0.006941 0.777687
+vt 0.176852 0.746796
+vt 0.892437 0.349226
+vt 0.006827 0.746834
+vt 0.886228 0.380492
+vt 0.176742 0.715941
+vt 0.006716 0.715979
+vt 0.874138 0.409852
+vt 0.176640 0.685087
+vt 0.006614 0.685125
+vt 0.856681 0.436209
+vt 0.176549 0.654233
+vt 0.006524 0.654271
+vt 0.834553 0.458608
+vt 0.176474 0.623379
+vt 0.006448 0.623417
+vt 0.808610 0.476251
+vt 0.176415 0.592525
+vt 0.006390 0.592562
+vt 0.779839 0.488518
+vt 0.176377 0.561671
+vt 0.006351 0.561708
+vt 0.749318 0.494983
+vt 0.176359 0.530816
+vt 0.006334 0.530854
+vt 0.176362 0.499962
+vt 0.718195 0.495436
+vt 0.006337 0.500000
+vt 0.687628 0.489885
+vt 0.176387 0.469108
+vt 0.006361 0.469146
+vt 0.658752 0.478560
+vt 0.176430 0.438254
+vt 0.006405 0.438291
+vt 0.632640 0.461897
+vt 0.176492 0.407399
+vt 0.006467 0.407437
+vt 0.610260 0.440534
+vt 0.176568 0.376545
+vt 0.006543 0.376583
+vt 0.592437 0.415287
+vt 0.176656 0.345691
+vt 0.006631 0.345729
+vt 0.176753 0.314837
+vt 0.579817 0.387126
+vt 0.006727 0.314875
+vt 0.177399 0.283987
+vt 0.572791 0.357181
+vt 0.006828 0.284020
+vt 0.006828 0.284020
+vt 0.572791 0.357181
+vt 0.177399 0.283987
+vt 0.006727 0.314875
+vt 0.176753 0.314837
+vt 0.579817 0.387126
+vt 0.006631 0.345729
+vt 0.176656 0.345691
+vt 0.592437 0.415287
+vt 0.006543 0.376583
+vt 0.610260 0.440534
+vt 0.176568 0.376545
+vt 0.006467 0.407437
+vt 0.632640 0.461897
+vt 0.176492 0.407399
+vt 0.006405 0.438291
+vt 0.176430 0.438254
+vt 0.658752 0.478560
+vt 0.006361 0.469146
+vt 0.176387 0.469108
+vt 0.687628 0.489885
+vt 0.006337 0.500000
+vt 0.718195 0.495436
+vt 0.176362 0.499962
+vt 0.006334 0.530854
+vt 0.749318 0.494983
+vt 0.176359 0.530816
+vt 0.006351 0.561708
+vt 0.779839 0.488518
+vt 0.176377 0.561671
+vt 0.006390 0.592562
+vt 0.808610 0.476251
+vt 0.176415 0.592525
+vt 0.006448 0.623417
+vt 0.176474 0.623379
+vt 0.834553 0.458608
+vt 0.006524 0.654271
+vt 0.176549 0.654233
+vt 0.856681 0.436209
+vt 0.006614 0.685125
+vt 0.176640 0.685087
+vt 0.874138 0.409852
+vt 0.006716 0.715979
+vt 0.886228 0.380492
+vt 0.176742 0.715941
+vt 0.006827 0.746834
+vt 0.892437 0.349226
+vt 0.176852 0.746796
+vt 0.006941 0.777687
+vt 0.892437 0.317355
+vt 0.177512 0.777645
+vt 0.007055 0.808541
+vt 0.886223 0.285996
+vt 0.177771 0.808498
+vt 0.007165 0.839396
+vt 0.850517 0.149568
+vt 0.319168 0.838215
+vt 0.007276 0.870250
+vt 0.836176 0.125250
+vt 0.319278 0.869069
+vt 0.007387 0.901104
+vt 0.816609 0.103895
+vt 0.319390 0.899923
+vt 0.007499 0.931958
+vt 0.319502 0.930776
+vt 0.792953 0.086670
+vt 0.007613 0.962812
+vt 0.319616 0.961631
+vt 0.765981 0.074365
+vt 0.007728 0.993666
+vt 0.007727 0.006334
+vt 0.319730 0.007378
+vt 0.736753 0.067631
+vt 0.319730 0.992485
+vt 0.007623 0.037188
+vt 0.319626 0.038233
+vt 0.706468 0.066881
+vt 0.007520 0.068042
+vt 0.676401 0.072253
+vt 0.319524 0.069087
+vt 0.007419 0.098896
+vt 0.319423 0.099941
+vt 0.647824 0.083609
+vt 0.007319 0.129750
+vt 0.319322 0.130795
+vt 0.621946 0.100560
+vt 0.007220 0.160604
+vt 0.599888 0.122511
+vt 0.319224 0.161649
+vt 0.007122 0.191458
+vt 0.319126 0.192504
+vt 0.582798 0.148853
+vt 0.007025 0.222313
+vt 0.319029 0.223358
+vt 0.573209 0.178195
+vt 0.006929 0.253167
+vt 0.177644 0.253134
+vt 0.571207 0.326153
+vt 0.443060 0.754024
+vt 0.208826 0.204735
+vt 0.579568 0.357437
+vt 0.370709 0.754396
+vt 0.443163 0.783546
+vt 0.222573 0.195552
+vt 0.586362 0.385333
+vt 0.371352 0.783117
+vt 0.443254 0.813086
+vt 0.234272 0.183855
+vt 0.598317 0.411848
+vt 0.371443 0.812167
+vt 0.443305 0.842613
+vt 0.243459 0.170107
+vt 0.615392 0.436018
+vt 0.371495 0.841694
+vt 0.443302 0.872141
+vt 0.249787 0.154830
+vt 0.636826 0.456466
+vt 0.371492 0.871219
+vt 0.443231 0.901668
+vt 0.253013 0.138613
+vt 0.371421 0.900745
+vt 0.661830 0.472412
+vt 0.443079 0.931196
+vt 0.253013 0.122077
+vt 0.371269 0.930272
+vt 0.689480 0.483245
+vt 0.442838 0.960722
+vt 0.249787 0.105860
+vt 0.718747 0.488548
+vt 0.371029 0.959797
+vt 0.442504 0.990249
+vt 0.442522 0.045396
+vt 0.243459 0.090582
+vt 0.370694 0.989321
+vt 0.748545 0.488099
+vt 0.370712 0.046322
+vt 0.442853 0.074922
+vt 0.234272 0.076833
+vt 0.777761 0.481894
+vt 0.371043 0.075847
+vt 0.443090 0.104449
+vt 0.222579 0.065141
+vt 0.805300 0.470133
+vt 0.371280 0.105372
+vt 0.443237 0.133976
+vt 0.208831 0.055955
+vt 0.371427 0.134897
+vt 0.830124 0.453224
+vt 0.443305 0.163503
+vt 0.193554 0.049628
+vt 0.371495 0.164423
+vt 0.851292 0.431762
+vt 0.443305 0.193031
+vt 0.177337 0.046402
+vt 0.867982 0.406513
+vt 0.371495 0.193949
+vt 0.443252 0.222558
+vt 0.160801 0.046402
+vt 0.879533 0.378395
+vt 0.371442 0.223476
+vt 0.443160 0.252098
+vt 0.144577 0.049631
+vt 0.371348 0.252525
+vt 0.885423 0.348954
+vt 0.443056 0.281620
+vt 0.129303 0.055959
+vt 0.370705 0.281246
+vt 0.885704 0.319257
+vt 0.442967 0.311134
+vt 0.115562 0.065142
+vt 0.370467 0.312068
+vt 0.879631 0.287703
+vt 0.442897 0.340661
+vt 0.103870 0.076834
+vt 0.229109 0.340661
+vt 0.844628 0.151667
+vt 0.442833 0.370188
+vt 0.094683 0.090583
+vt 0.229046 0.370189
+vt 0.831035 0.128981
+vt 0.442778 0.399714
+vt 0.088355 0.105859
+vt 0.812355 0.108585
+vt 0.228990 0.399716
+vt 0.442732 0.429241
+vt 0.085130 0.122076
+vt 0.228944 0.429242
+vt 0.789700 0.092175
+vt 0.442698 0.458768
+vt 0.085130 0.138611
+vt 0.763884 0.080481
+vt 0.228910 0.458769
+vt 0.442678 0.488295
+vt 0.088356 0.154828
+vt 0.228889 0.488295
+vt 0.735929 0.074111
+vt 0.442670 0.517822
+vt 0.094683 0.170106
+vt 0.706979 0.073452
+vt 0.228883 0.517823
+vt 0.442678 0.547349
+vt 0.103870 0.183854
+vt 0.678244 0.078632
+vt 0.228890 0.547350
+vt 0.442700 0.576876
+vt 0.115562 0.195546
+vt 0.228912 0.576877
+vt 0.650942 0.089526
+vt 0.442734 0.606402
+vt 0.129310 0.204732
+vt 0.228946 0.606403
+vt 0.626230 0.105756
+vt 0.442780 0.635930
+vt 0.144587 0.211060
+vt 0.605204 0.126737
+vt 0.228992 0.635930
+vt 0.442837 0.665456
+vt 0.160804 0.214285
+vt 0.589057 0.151745
+vt 0.229049 0.665457
+vt 0.442901 0.694984
+vt 0.177339 0.214285
+vt 0.229113 0.694984
+vt 0.579868 0.178984
+vt 0.442971 0.724510
+vt 0.193556 0.211059
+vt 0.578089 0.326242
+vt 0.370472 0.723575
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.846034 0.501950
+vt 0.307543 0.501950
+vt 0.769051 0.501950
+vt 0.307542 0.501950
+vt 0.000000 0.501950
+vt 0.692068 0.501950
+vt 0.615085 0.501950
+vt 0.846034 0.996502
+vt 0.384526 0.501950
+vt 0.538102 0.501950
+vt 0.769051 0.501950
+vt 0.923017 0.996502
+vt 0.230559 0.501950
+vt 0.076983 0.501950
+vt 0.692068 0.501950
+vt 0.692068 0.192341
+vt 0.923017 0.650215
+vt 0.076983 0.192341
+vt 0.692068 0.000000
+vt 0.076983 0.000000
+vt 0.846034 0.429417
+vt 0.769051 0.000000
+vt 0.538102 0.000000
+vt 0.923017 0.429417
+vt 0.769051 0.192341
+vt 0.538102 0.192341
+vt 1.000000 0.650215
+vt 0.615085 0.192341
+vt 0.615085 0.000000
+vt 0.923017 0.343447
+vt 0.000000 0.000000
+vt 0.692068 0.000000
+vt 0.846034 0.343447
+vt 0.000000 0.192341
+vt 0.692068 0.192341
+vt 0.923017 0.686893
+vt 0.923017 0.478709
+vt 0.230559 0.192341
+vt 0.230559 0.000000
+vt 0.846034 0.600923
+vt 0.769051 0.000000
+vt 0.307542 0.000000
+vt 0.846034 0.686893
+vt 0.769051 0.192341
+vt 0.307542 0.192341
+vt 0.846034 0.192341
+vt 0.307542 0.192341
+vt 0.846034 0.000000
+vt 0.307542 0.000000
+vt 0.923017 0.686893
+vt 0.384525 0.000000
+vt 0.923017 0.600923
+vt 0.846034 0.686893
+vt 0.384525 0.192341
+vt 1.000000 0.478709
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.000000 0.000000
+vt 0.846034 1.000000
+vt 0.307543 1.000000
+vt 0.769051 1.000000
+vt 0.307542 1.000000
+vt 0.000000 1.000000
+vt 0.769051 1.000000
+vt 0.615085 1.000000
+vt 0.692068 1.000000
+vt 0.923017 0.307203
+vt 0.384526 1.000000
+vt 0.538102 1.000000
+vt 0.615085 0.501950
+vt 1.000000 0.307203
+vt 0.230559 1.000000
+vt 0.076983 1.000000
+vt 0.692068 0.501950
+vt 0.692068 0.809153
+vt 0.923017 0.478709
+vt 0.076983 0.692797
+vt 0.692068 1.000000
+vt 0.076983 0.501950
+vt 0.846034 0.085970
+vt 0.615085 1.000000
+vt 0.538102 0.501950
+vt 0.923017 0.085970
+vt 0.615085 0.809153
+vt 0.538102 0.692796
+vt 1.000000 0.478709
+vt 0.615085 0.692796
+vt 0.692068 0.692796
+vt 0.615085 0.501950
+vt 0.692068 0.501950
+vt 0.923017 0.000000
+vt 0.000000 0.501950
+vt 0.769051 0.501950
+vt 0.846034 0.000000
+vt 0.000000 0.692797
+vt 0.769051 0.692796
+vt 1.000000 0.000000
+vt 0.923017 0.307203
+vt 0.230559 0.692797
+vt 0.230559 0.501950
+vt 0.846034 0.257476
+vt 0.769051 0.501950
+vt 0.307542 0.501950
+vt 0.846034 0.343447
+vt 0.769051 0.692796
+vt 0.307542 0.692796
+vt 0.846034 0.692796
+vt 0.307542 0.692797
+vt 0.846034 0.501950
+vt 0.307542 0.501950
+vt 0.923017 0.343447
+vt 0.384525 0.501950
+vt 0.923017 0.257476
+vt 0.923017 0.000000
+vt 0.384525 0.692797
+vt 1.000000 0.307203
+vt 0.875462 0.748915
+vt 0.958487 0.000000
+vt 1.000000 1.000000
+vt 0.437731 0.748915
+vt 1.000000 0.500000
+vt 0.875462 0.000000
+vt 0.875462 0.000000
+vt 0.958487 0.748915
+vt 0.958487 0.500000
+vt 0.437731 0.000000
+vt 0.958487 0.000000
+vt 0.875462 0.748915
+vt 0.437731 0.748915
+vt 0.916975 0.748915
+vt 1.000000 0.500000
+vt 0.437731 0.000000
+vt 0.916975 0.000000
+vt 0.958487 1.000000
+vt 0.000000 0.000000
+vt 0.958487 0.500000
+vt 0.916974 0.000000
+vt 0.000000 0.748915
+vt 1.000000 0.000000
+vt 0.916975 0.748915
+s 0
+g KsHauptsignal_Plane.006_Mast
+usemtl Mast
+f 13/29/1 12/27/1 11/24/1 14/31/1
+f 9/18/2 16/37/2 15/34/2 10/21/2
+f 16/38/3 13/30/3 14/32/3 15/35/3
+f 12/28/4 9/19/4 10/20/4 11/23/4
+f 21/49/4 20/47/4 19/44/4 22/51/4
+f 24/58/1 21/50/1 22/52/1 23/54/1
+f 20/48/2 17/41/2 18/42/2 19/45/2
+f 24/58/1 23/54/1 11/24/1 12/27/1
+f 17/39/5 24/56/5 12/26/5 9/17/5
+f 18/42/2 17/41/2 9/18/2 10/21/2
+f 23/55/6 18/43/6 10/22/6 11/25/6
+f 19/46/6 18/43/6 23/55/6 22/53/6
+f 11/25/6 10/22/6 15/36/6 14/33/6
+f 3/5/1 2/3/1 12/27/1 13/29/1
+f 1/1/2 4/7/2 16/37/2 9/18/2
+f 4/8/3 3/6/3 13/30/3 16/38/3
+f 2/4/4 1/2/4 9/19/4 12/28/4
+f 7/13/4 6/11/4 20/47/4 21/49/4
+f 5/9/3 8/15/3 24/57/3 17/40/3
+f 8/16/1 7/14/1 21/50/1 24/58/1
+f 6/12/2 5/10/2 17/41/2 20/48/2
+f 544/974/6 541/973/6 537/971/6 538/972/6
+f 548/981/1 550/984/1 556/1000/1 557/1002/1
+f 557/1002/1 556/1000/1 555/997/1 558/1003/1
+f 552/988/2 547/979/2 560/1008/2 553/992/2
+f 553/992/2 560/1008/2 559/1005/2 554/994/2
+f 547/980/3 548/981/3 557/1002/3 560/1009/3
+f 560/1009/3 557/1002/3 558/1003/3 559/1006/3
+f 550/985/4 552/989/4 553/990/4 556/999/4
+f 556/999/4 553/990/4 554/993/4 555/996/4
+f 545/975/4 546/977/4 564/1018/4 565/1020/4
+f 565/1020/4 564/1018/4 563/1015/4 566/1022/4
+f 551/986/3 549/982/3 568/1027/3 561/1010/3
+f 549/983/1 545/976/1 565/1021/1 568/1028/1
+f 568/1028/1 565/1021/1 566/1023/1 567/1025/1
+f 546/978/2 551/987/2 561/1012/2 564/1019/2
+f 564/1019/2 561/1012/2 562/1013/2 563/1016/2
+f 568/1028/1 567/1025/1 555/997/1 556/1000/1
+f 561/1011/5 568/1029/5 556/1001/5 553/991/5
+f 562/1013/2 561/1012/2 553/992/2 554/994/2
+f 567/1026/6 562/1014/6 554/995/6 555/998/6
+f 563/1017/6 562/1014/6 567/1026/6 566/1024/6
+f 555/998/6 554/995/6 559/1007/6 558/1004/6
+f 576/1033/6 573/1032/6 569/1030/6 570/1031/6
+f 580/1040/1 582/1044/1 588/1060/1 589/1062/1
+f 589/1062/1 588/1060/1 587/1057/1 590/1064/1
+f 584/1048/2 579/1038/2 592/1070/2 585/1052/2
+f 585/1052/2 592/1070/2 591/1067/2 586/1054/2
+f 579/1039/3 580/1041/3 589/1063/3 592/1071/3
+f 592/1071/3 589/1063/3 590/1065/3 591/1068/3
+f 582/1045/4 584/1049/4 585/1050/4 588/1059/4
+f 588/1059/4 585/1050/4 586/1053/4 587/1056/4
+f 577/1034/4 578/1036/4 596/1080/4 597/1082/4
+f 597/1082/4 596/1080/4 595/1077/4 598/1084/4
+f 583/1046/3 581/1042/3 600/1089/3 593/1072/3
+f 581/1043/1 577/1035/1 597/1083/1 600/1090/1
+f 600/1090/1 597/1083/1 598/1085/1 599/1087/1
+f 578/1037/2 583/1047/2 593/1074/2 596/1081/2
+f 596/1081/2 593/1074/2 594/1075/2 595/1078/2
+f 600/1090/1 599/1087/1 587/1057/1 588/1060/1
+f 593/1073/5 600/1091/5 588/1061/5 585/1051/5
+f 594/1075/2 593/1074/2 585/1052/2 586/1054/2
+f 599/1088/6 594/1076/6 586/1055/6 587/1058/6
+f 595/1079/6 594/1076/6 599/1088/6 598/1086/6
+f 587/1058/6 586/1055/6 591/1069/6 590/1066/6
+g KsHauptsignal_Plane.006_Schirm
+usemtl Schirm
+f 601/1092/3 602/1095/3 604/1101/3 603/1098/3
+f 606/1107/4 605/1104/4 608/1113/4 607/1110/4
+f 603/1099/2 605/1105/2 606/1108/2 601/1093/2
+f 604/1102/6 608/1114/6 605/1106/6 603/1100/6
+f 601/1094/5 606/1109/5 607/1111/5 602/1096/5
+f 602/1097/1 607/1112/1 608/1115/1 604/1103/1
+g KsHauptsignal_Plane.006_Lampenrahmen
+usemtl Lampenrahmen
+f 40/89/7 39/87/7 41/91/7
+f 26/61/8 88/187/8 25/60/8
+f 54/117/9 53/116/9 56/121/9
+f 58/125/10 56/121/10 57/123/10
+f 86/183/11 85/181/11 88/187/11
+f 90/191/12 89/190/12 152/286/12
+f 118/235/13 120/238/13 117/234/13
+f 122/241/14 121/239/14 120/238/14
+f 150/283/15 152/286/15 149/281/15
+f 79/169/16 81/174/16 82/175/16
+f 81/173/17 145/275/17 147/278/17
+f 111/224/18 113/227/18 49/107/18
+f 103/210/19 105/214/19 41/92/19
+f 99/204/20 35/79/20 33/75/21
+f 105/214/22 107/218/22 43/95/22
+f 147/278/23 149/282/23 85/182/23
+f 97/201/24 33/75/24 31/71/24
+f 103/210/25 39/88/25 37/83/25
+f 109/221/26 111/224/26 47/103/26
+f 35/79/27 99/204/27 101/207/27
+f 107/218/28 109/221/28 45/99/28
+f 95/198/29 31/71/29 29/67/29
+f 93/195/30 29/67/30 27/63/30
+f 113/227/31 115/230/31 51/111/31
+f 151/285/32 89/189/32 25/59/32
+f 149/282/33 151/285/33 87/186/33
+f 121/240/34 57/124/34 55/120/34
+f 115/230/35 117/233/35 53/115/35
+f 79/170/36 143/272/36 145/275/36
+f 125/245/37 127/248/37 63/135/37
+f 77/165/38 141/269/38 143/272/38
+f 75/161/39 139/266/39 141/269/39
+f 73/157/40 137/264/40 139/266/40
+f 133/257/41 135/260/41 71/151/41
+f 131/254/42 133/257/42 69/147/42
+f 129/251/43 131/254/43 67/143/43
+f 127/248/44 129/251/44 65/139/44
+f 91/193/45 27/63/45 25/59/45
+f 123/242/46 125/245/46 61/131/46
+f 123/242/47 59/127/47 57/124/47
+f 88/187/48 87/185/48 25/60/48
+f 97/202/49 95/199/49 96/200/49
+f 112/226/50 111/225/50 109/222/50
+f 48/105/51 47/104/51 49/108/51
+f 110/223/52 109/222/52 107/219/52
+f 140/268/53 139/267/53 137/263/53
+f 42/93/54 41/91/54 43/96/54
+f 106/216/55 105/213/55 103/211/55
+f 37/84/56 39/87/56 40/89/56
+f 108/220/57 107/219/57 105/215/57
+f 33/76/58 35/80/58 36/81/58
+f 31/72/59 33/76/59 34/77/59
+f 50/109/60 49/108/60 51/112/60
+f 27/64/61 29/68/61 30/69/61
+f 25/60/62 27/64/62 28/65/62
+f 53/116/63 55/119/63 56/121/63
+f 56/121/64 55/119/64 57/123/64
+f 81/174/65 83/178/65 84/179/65
+f 71/151/66 135/260/66 137/264/66
+f 83/178/67 85/181/67 86/183/67
+f 119/237/68 55/120/68 53/115/68
+f 77/166/69 79/169/69 80/171/69
+f 75/162/70 77/166/70 78/167/70
+f 114/229/71 113/228/71 111/225/71
+f 142/271/72 141/270/72 139/267/72
+f 68/145/73 67/144/73 69/148/73
+f 66/141/74 65/140/74 67/144/74
+f 64/137/75 63/136/75 65/140/75
+f 62/133/76 61/132/76 63/136/76
+f 60/129/77 59/128/77 61/132/77
+f 58/125/78 57/123/78 59/128/78
+f 85/181/79 87/185/79 88/187/79
+f 89/190/80 151/284/80 152/286/80
+f 146/277/81 145/276/81 143/273/81
+f 144/274/82 143/273/82 141/270/82
+f 116/232/83 115/231/83 113/228/83
+f 71/152/84 73/158/84 74/159/84
+f 29/68/85 31/72/85 32/73/85
+f 44/97/86 43/96/86 45/100/86
+f 138/265/87 137/263/87 135/261/87
+f 103/211/88 101/208/88 102/209/88
+f 101/208/89 99/205/89 100/206/89
+f 99/205/90 97/202/90 98/203/90
+f 52/113/91 51/112/91 53/116/91
+f 95/199/92 93/196/92 94/197/92
+f 93/196/93 91/192/93 92/194/93
+f 92/194/94 91/192/94 89/190/94
+f 120/238/95 119/236/95 117/234/95
+f 121/239/14 119/236/14 120/238/14
+f 73/158/96 75/162/96 76/163/96
+f 150/283/97 149/281/97 147/279/97
+f 148/280/98 147/279/98 145/276/98
+f 116/232/99 118/235/99 117/234/99
+f 70/149/100 69/148/100 71/153/100
+f 46/101/101 45/100/101 47/104/101
+f 35/80/102 37/84/102 38/85/102
+f 135/261/103 133/258/103 134/259/103
+f 133/258/104 131/255/104 132/256/104
+f 131/255/105 129/252/105 130/253/105
+f 129/252/106 127/249/106 128/250/106
+f 127/249/107 125/246/107 126/247/107
+f 125/246/108 123/243/108 124/244/108
+f 123/243/109 121/239/109 122/241/109
+f 152/286/110 151/284/110 149/281/110
+f 209/401/111 273/503/111 275/506/111
+f 276/508/112 275/507/112 273/504/112
+f 211/406/113 213/409/113 214/411/113
+f 249/467/114 247/464/114 248/466/114
+f 244/460/115 243/459/115 241/456/115
+f 272/502/116 271/501/116 269/498/116
+f 180/341/117 179/340/117 181/344/117
+f 229/436/118 227/433/118 228/434/118
+f 225/430/119 223/427/119 224/428/119
+f 255/476/120 257/479/120 193/367/120
+f 234/444/121 233/441/121 231/439/121
+f 157/296/122 159/300/122 160/301/122
+f 182/345/110 181/344/110 184/349/110
+f 165/312/123 167/315/123 168/317/123
+f 214/411/124 213/409/124 216/415/124
+f 221/424/125 219/420/125 220/422/125
+f 216/415/126 215/413/126 153/288/126
+f 227/432/127 163/307/128 161/303/128
+f 235/446/129 237/449/129 173/327/129
+f 201/385/130 265/492/130 267/494/130
+f 261/485/131 263/488/131 199/379/131
+f 184/349/132 183/347/132 185/351/132
+f 159/300/133 161/304/133 162/305/133
+f 168/317/134 167/315/134 169/319/134
+f 275/506/135 277/510/135 213/410/135
+f 266/493/136 265/491/136 263/489/136
+f 249/468/137 185/352/137 183/348/137
+f 231/438/138 233/442/138 169/320/138
+f 190/361/139 189/360/139 191/364/139
+f 217/418/140 279/512/140 280/514/140
+f 255/477/141 253/474/141 254/475/141
+f 186/353/142 184/349/142 185/351/142
+f 203/390/143 205/394/143 206/395/143
+f 239/452/144 241/455/144 177/335/144
+f 225/429/145 161/303/145 159/299/145
+f 172/325/146 171/324/146 173/328/146
+f 174/329/147 173/328/147 175/332/147
+f 199/379/148 263/488/148 265/492/149
+f 253/473/150 255/476/150 191/363/150
+f 163/308/151 165/312/151 166/313/151
+f 263/489/152 261/486/152 262/487/152
+f 241/455/153 243/458/153 179/339/153
+f 201/386/154 203/390/154 204/391/154
+f 198/377/155 197/376/155 199/381/155
+f 209/402/156 211/406/156 212/407/156
+f 188/357/157 187/356/157 189/360/157
+f 233/442/158 235/446/158 171/323/158
+f 196/373/159 195/372/159 197/376/159
+f 257/479/160 259/482/161 195/371/161
+f 278/511/162 280/514/162 277/509/162
+f 236/448/163 235/447/163 233/443/163
+f 250/469/114 249/467/114 248/466/114
+f 253/474/164 251/471/164 252/472/164
+f 205/393/165 269/497/165 271/500/165
+f 213/409/166 215/413/166 216/415/166
+f 154/289/167 216/415/167 153/288/167
+f 218/419/140 217/418/140 280/514/140
+f 279/513/168 217/417/168 153/287/168
+f 176/333/169 175/332/169 177/336/169
+f 181/344/170 183/347/170 184/349/170
+f 161/304/171 163/308/171 164/309/171
+f 280/514/172 279/512/172 277/509/172
+f 257/480/173 255/477/173 256/478/173
+f 274/505/174 273/504/174 271/501/174
+f 155/292/175 157/296/175 158/297/175
+f 231/439/176 229/436/176 230/437/176
+f 205/394/177 207/397/177 208/399/177
+f 220/422/178 219/420/178 217/418/178
+f 203/389/179 267/494/179 269/497/179
+f 192/365/180 191/364/180 193/368/180
+f 244/460/181 246/463/181 245/462/181
+f 170/321/182 169/319/182 171/324/182
+f 259/482/183 261/485/184 197/375/183
+f 219/421/185 155/291/185 153/287/185
+f 251/471/186 249/467/186 250/469/186
+f 207/398/187 271/500/187 273/503/187
+f 277/510/188 279/513/188 215/414/188
+f 240/454/189 239/453/189 237/450/189
+f 261/486/190 259/483/190 260/484/190
+f 223/427/191 221/424/191 222/425/191
+f 278/511/192 277/509/192 275/507/192
+f 248/466/193 247/464/193 245/462/193
+f 237/449/194 239/452/194 175/331/195
+f 246/463/193 248/466/193 245/462/193
+f 221/423/196 157/295/196 155/291/196
+f 242/457/197 241/456/197 239/453/197
+f 227/433/198 225/430/198 226/431/198
+f 259/483/199 257/480/199 258/481/199
+f 251/470/200 187/355/201 185/352/202
+f 178/337/203 177/336/203 179/340/203
+f 199/380/204 201/386/204 202/387/204
+f 243/458/205 245/461/205 181/343/205
+f 223/426/206 159/299/206 157/295/206
+f 163/307/207 227/432/207 229/435/207
+f 194/369/208 193/368/208 195/372/208
+f 207/397/209 209/402/209 210/403/209
+f 251/470/210 253/473/211 189/359/210
+f 270/499/212 269/498/212 267/495/212
+f 186/353/213 185/351/213 187/356/213
+f 247/465/214 183/348/214 181/343/215
+f 153/288/216 155/292/216 156/293/216
+f 268/496/217 267/495/217 265/491/217
+f 238/451/218 237/450/218 235/447/218
+f 231/438/219 167/316/219 165/311/219
+f 42/93/220 40/89/220 41/91/220
+f 80/171/16 79/169/16 82/175/16
+f 83/177/221 81/173/221 147/278/221
+f 47/103/222 111/224/222 49/107/222
+f 39/88/223 103/210/223 41/92/223
+f 97/201/224 99/204/224 33/75/224
+f 41/92/225 105/214/225 43/95/225
+f 83/177/226 147/278/226 85/182/226
+f 95/198/227 97/201/227 31/71/227
+f 101/207/228 103/210/228 37/83/228
+f 45/99/229 109/221/229 47/103/229
+f 37/83/230 35/79/230 101/207/230
+f 43/95/231 107/218/231 45/99/231
+f 93/195/232 95/198/232 29/67/232
+f 91/193/233 93/195/233 27/63/233
+f 49/107/234 113/227/234 51/111/234
+f 87/186/235 151/285/235 25/59/235
+f 85/182/236 149/282/236 87/186/237
+f 119/237/238 121/240/238 55/120/239
+f 51/111/240 115/230/240 53/115/240
+f 81/173/241 79/170/241 145/275/241
+f 61/131/242 125/245/242 63/135/242
+f 79/170/243 77/165/243 143/272/243
+f 77/165/244 75/161/244 141/269/244
+f 75/161/245 73/157/245 139/266/245
+f 69/147/246 133/257/246 71/151/246
+f 67/143/247 131/254/247 69/147/247
+f 65/139/248 129/251/248 67/143/248
+f 63/135/249 127/248/249 65/139/249
+f 89/189/250 91/193/250 25/59/250
+f 59/127/251 123/242/251 61/131/251
+f 121/240/252 123/242/252 57/124/252
+f 98/203/49 97/202/49 96/200/49
+f 110/223/50 112/226/50 109/222/50
+f 50/109/51 48/105/51 49/108/51
+f 108/220/253 110/223/253 107/219/253
+f 138/265/254 140/268/254 137/263/254
+f 44/97/255 42/93/255 43/96/255
+f 104/212/256 106/216/256 103/211/256
+f 38/85/56 37/84/56 40/89/56
+f 106/217/257 108/220/257 105/215/257
+f 34/77/258 33/76/258 36/81/258
+f 32/73/59 31/72/59 34/77/59
+f 52/113/259 50/109/259 51/112/259
+f 28/65/260 27/64/260 30/69/260
+f 26/61/261 25/60/261 28/65/261
+f 82/175/65 81/174/65 84/179/65
+f 73/157/262 71/151/262 137/264/262
+f 84/179/263 83/178/263 86/183/263
+f 117/233/264 119/237/264 53/115/264
+f 78/167/69 77/166/69 80/171/69
+f 76/163/265 75/162/265 78/167/265
+f 112/226/266 114/229/266 111/225/266
+f 140/268/72 142/271/72 139/267/72
+f 70/149/73 68/145/73 69/148/73
+f 68/145/267 66/141/267 67/144/267
+f 66/141/268 64/137/268 65/140/268
+f 64/137/76 62/133/76 63/136/76
+f 62/133/77 60/129/77 61/132/77
+f 60/129/269 58/125/269 59/128/269
+f 144/274/270 146/277/270 143/273/270
+f 142/271/271 144/274/271 141/270/271
+f 114/229/83 116/232/83 113/228/83
+f 72/154/84 71/152/84 74/159/84
+f 30/69/85 29/68/85 32/73/85
+f 46/101/86 44/97/86 45/100/86
+f 136/262/87 138/265/87 135/261/87
+f 104/212/88 103/211/88 102/209/88
+f 102/209/272 101/208/272 100/206/272
+f 100/206/273 99/205/273 98/203/273
+f 54/117/274 52/113/274 53/116/274
+f 96/200/92 95/199/92 94/197/92
+f 94/197/275 93/196/275 92/194/275
+f 90/191/276 92/194/276 89/190/276
+f 74/159/96 73/158/96 76/163/96
+f 148/280/277 150/283/277 147/279/277
+f 146/277/278 148/280/278 145/276/278
+f 115/231/279 116/232/279 117/234/279
+f 72/155/100 70/149/100 71/153/100
+f 48/105/101 46/101/101 47/104/101
+f 36/81/280 35/80/280 38/85/280
+f 136/262/103 135/261/103 134/259/103
+f 134/259/104 133/258/104 132/256/104
+f 132/256/105 131/255/105 130/253/105
+f 130/253/106 129/252/106 128/250/106
+f 128/250/107 127/249/107 126/247/107
+f 126/247/108 125/246/108 124/244/108
+f 124/244/109 123/243/109 122/241/109
+f 296/545/7 295/543/7 297/547/7
+f 282/517/8 344/643/8 281/516/8
+f 310/573/9 309/572/9 312/577/9
+f 314/581/10 312/577/10 313/579/10
+f 342/639/11 341/637/11 344/643/11
+f 346/647/12 345/646/12 408/742/12
+f 374/691/13 376/694/13 373/690/13
+f 378/697/14 377/695/14 376/694/14
+f 406/739/15 408/742/15 405/737/15
+f 335/625/16 337/630/16 338/631/16
+f 337/629/17 401/731/17 403/734/17
+f 367/680/281 369/683/281 305/563/281
+f 359/666/19 361/670/19 297/548/19
+f 355/660/20 291/535/20 289/531/20
+f 361/670/282 363/674/282 299/551/282
+f 403/734/283 405/738/283 341/638/283
+f 353/657/24 289/531/24 287/527/24
+f 359/666/284 295/544/284 293/539/284
+f 365/677/285 367/680/285 303/559/285
+f 291/535/286 355/660/286 357/663/286
+f 363/674/28 365/677/28 301/555/28
+f 351/654/287 287/527/287 285/523/287
+f 349/651/288 285/523/288 283/519/288
+f 369/683/31 371/686/31 307/567/31
+f 407/741/289 345/645/289 281/515/289
+f 405/738/33 407/741/33 343/642/33
+f 377/696/290 313/580/290 311/576/290
+f 371/686/291 373/689/291 309/571/291
+f 335/626/292 399/728/292 401/731/292
+f 381/701/37 383/704/37 319/591/37
+f 333/621/293 397/725/293 399/728/293
+f 331/617/294 395/722/294 397/725/294
+f 329/613/40 393/720/40 395/722/40
+f 389/713/295 391/716/295 327/607/295
+f 387/710/42 389/713/42 325/603/42
+f 385/707/296 387/710/296 323/599/296
+f 383/704/297 385/707/297 321/595/297
+f 347/649/298 283/519/298 281/515/298
+f 379/698/299 381/701/299 317/587/299
+f 379/698/300 315/583/300 313/580/300
+f 344/643/48 343/641/48 281/516/48
+f 353/658/49 351/655/49 352/656/49
+f 368/682/50 367/681/50 365/678/50
+f 304/561/51 303/560/51 305/564/51
+f 366/679/52 365/678/52 363/675/52
+f 396/724/53 395/723/53 393/719/53
+f 298/549/54 297/547/54 299/552/54
+f 362/672/55 361/669/55 359/667/55
+f 293/540/56 295/543/56 296/545/56
+f 364/676/57 363/675/57 361/671/57
+f 289/532/58 291/536/58 292/537/58
+f 287/528/59 289/532/59 290/533/59
+f 306/565/60 305/564/60 307/568/60
+f 283/520/61 285/524/61 286/525/61
+f 281/516/62 283/520/62 284/521/62
+f 309/572/301 311/575/301 312/577/301
+f 312/577/64 311/575/64 313/579/64
+f 337/630/65 339/634/65 340/635/65
+f 327/607/302 391/716/302 393/720/302
+f 339/634/67 341/637/67 342/639/67
+f 375/693/303 311/576/303 309/571/303
+f 333/622/69 335/625/69 336/627/69
+f 331/618/70 333/622/70 334/623/70
+f 370/685/71 369/684/71 367/681/71
+f 398/727/72 397/726/72 395/723/72
+f 324/601/73 323/600/73 325/604/73
+f 322/597/74 321/596/74 323/600/74
+f 320/593/75 319/592/75 321/596/75
+f 318/589/76 317/588/76 319/592/76
+f 316/585/77 315/584/77 317/588/77
+f 314/581/78 313/579/78 315/584/78
+f 341/637/79 343/641/79 344/643/79
+f 345/646/80 407/740/80 408/742/80
+f 402/733/81 401/732/81 399/729/81
+f 400/730/82 399/729/82 397/726/82
+f 372/688/83 371/687/83 369/684/83
+f 327/608/84 329/614/84 330/615/84
+f 285/524/85 287/528/85 288/529/85
+f 300/553/86 299/552/86 301/556/86
+f 394/721/87 393/719/87 391/717/87
+f 359/667/88 357/664/88 358/665/88
+f 357/664/89 355/661/89 356/662/89
+f 355/661/90 353/658/90 354/659/90
+f 308/569/304 307/568/304 309/572/304
+f 351/655/92 349/652/92 350/653/92
+f 349/652/93 347/648/93 348/650/93
+f 348/650/305 347/648/305 345/646/305
+f 376/694/95 375/692/95 373/690/95
+f 377/695/14 375/692/14 376/694/14
+f 329/614/96 331/618/96 332/619/96
+f 406/739/97 405/737/97 403/735/97
+f 404/736/98 403/735/98 401/732/98
+f 372/688/99 374/691/99 373/690/99
+f 326/605/100 325/604/100 327/609/100
+f 302/557/101 301/556/101 303/560/101
+f 291/536/280 293/540/280 294/541/280
+f 391/717/103 389/714/103 390/715/103
+f 389/714/104 387/711/104 388/712/104
+f 387/711/306 385/708/306 386/709/306
+f 385/708/106 383/705/106 384/706/106
+f 383/705/107 381/702/107 382/703/107
+f 381/702/108 379/699/108 380/700/108
+f 379/699/109 377/695/109 378/697/109
+f 408/742/110 407/740/110 405/737/110
+f 298/549/220 296/545/220 297/547/220
+f 336/627/16 335/625/16 338/631/16
+f 339/633/221 337/629/221 403/734/221
+f 303/559/222 367/680/222 305/563/222
+f 295/544/223 359/666/223 297/548/223
+f 353/657/307 355/660/307 289/531/307
+f 297/548/308 361/670/308 299/551/308
+f 339/633/309 403/734/309 341/638/309
+f 351/654/310 353/657/310 287/527/310
+f 357/663/311 359/666/311 293/539/311
+f 301/555/312 365/677/312 303/559/312
+f 293/539/313 291/535/313 357/663/313
+f 299/551/314 363/674/314 301/555/314
+f 349/651/315 351/654/315 285/523/315
+f 347/649/316 349/651/316 283/519/316
+f 305/563/317 369/683/317 307/567/317
+f 343/642/318 407/741/318 281/515/318
+f 341/638/319 405/738/319 343/642/319
+f 375/693/320 377/696/320 311/576/320
+f 307/567/321 371/686/321 309/571/321
+f 337/629/322 335/626/322 401/731/322
+f 317/587/323 381/701/323 319/591/323
+f 335/626/324 333/621/325 399/728/326
+f 333/621/327 331/617/327 397/725/327
+f 331/617/328 329/613/328 395/722/328
+f 325/603/246 389/713/246 327/607/246
+f 323/599/329 387/710/329 325/603/329
+f 321/595/330 385/707/330 323/599/330
+f 319/591/249 383/704/249 321/595/249
+f 345/645/250 347/649/250 281/515/250
+f 315/583/331 379/698/331 317/587/331
+f 377/696/332 379/698/333 313/580/333
+f 354/659/49 353/658/49 352/656/49
+f 366/679/50 368/682/50 365/678/50
+f 306/565/51 304/561/51 305/564/51
+f 364/676/253 366/679/253 363/675/253
+f 394/721/254 396/724/254 393/719/254
+f 300/553/255 298/549/255 299/552/255
+f 360/668/256 362/672/256 359/667/256
+f 294/541/56 293/540/56 296/545/56
+f 362/673/257 364/676/257 361/671/257
+f 290/533/258 289/532/258 292/537/258
+f 288/529/59 287/528/59 290/533/59
+f 308/569/259 306/565/259 307/568/259
+f 284/521/260 283/520/260 286/525/260
+f 282/517/261 281/516/261 284/521/261
+f 338/631/65 337/630/65 340/635/65
+f 329/613/262 327/607/262 393/720/262
+f 340/635/263 339/634/263 342/639/263
+f 373/689/334 375/693/334 309/571/334
+f 334/623/69 333/622/69 336/627/69
+f 332/619/335 331/618/335 334/623/335
+f 368/682/266 370/685/266 367/681/266
+f 396/724/72 398/727/72 395/723/72
+f 326/605/73 324/601/73 325/604/73
+f 324/601/267 322/597/267 323/600/267
+f 322/597/268 320/593/268 321/596/268
+f 320/593/76 318/589/76 319/592/76
+f 318/589/77 316/585/77 317/588/77
+f 316/585/336 314/581/336 315/584/336
+f 400/730/270 402/733/270 399/729/270
+f 398/727/271 400/730/271 397/726/271
+f 370/685/83 372/688/83 369/684/83
+f 328/610/84 327/608/84 330/615/84
+f 286/525/85 285/524/85 288/529/85
+f 302/557/86 300/553/86 301/556/86
+f 392/718/87 394/721/87 391/717/87
+f 360/668/88 359/667/88 358/665/88
+f 358/665/272 357/664/272 356/662/272
+f 356/662/273 355/661/273 354/659/273
+f 310/573/337 308/569/337 309/572/337
+f 352/656/92 351/655/92 350/653/92
+f 350/653/275 349/652/275 348/650/275
+f 346/647/276 348/650/276 345/646/276
+f 330/615/96 329/614/96 332/619/96
+f 404/736/277 406/739/277 403/735/277
+f 402/733/278 404/736/278 401/732/278
+f 371/687/279 372/688/279 373/690/279
+f 328/611/100 326/605/100 327/609/100
+f 304/561/101 302/557/101 303/560/101
+f 292/537/102 291/536/102 294/541/102
+f 392/718/103 391/717/103 390/715/103
+f 390/715/104 389/714/104 388/712/104
+f 388/712/105 387/711/105 386/709/105
+f 386/709/106 385/708/106 384/706/106
+f 384/706/107 383/705/107 382/703/107
+f 382/703/108 381/702/108 380/700/108
+f 380/700/109 379/699/109 378/697/109
+f 170/321/338 168/317/338 169/319/338
+f 208/399/209 207/397/209 210/403/209
+f 211/405/339 209/401/340 275/506/339
+f 175/331/341 239/452/341 177/335/341
+f 167/316/342 231/438/343 169/320/343
+f 225/429/344 227/432/344 161/303/344
+f 169/320/345 233/442/345 171/323/345
+f 211/405/346 275/506/346 213/410/346
+f 223/426/347 225/429/347 159/299/347
+f 229/435/348 231/438/348 165/311/349
+f 173/327/350 237/449/350 175/331/350
+f 165/311/351 163/307/352 229/435/352
+f 171/323/353 235/446/353 173/327/353
+f 221/423/354 223/426/354 157/295/354
+f 219/421/355 221/423/356 155/291/355
+f 177/335/357 241/455/357 179/339/358
+f 215/414/359 279/513/359 153/287/359
+f 213/410/360 277/510/360 215/414/361
+f 247/465/362 249/468/362 183/348/362
+f 179/339/363 243/458/363 181/343/363
+f 209/401/364 207/398/364 273/503/364
+f 189/359/365 253/473/366 191/363/365
+f 207/398/367 205/393/367 271/500/367
+f 205/393/368 203/389/369 269/497/370
+f 203/389/371 201/385/371 267/494/371
+f 197/375/372 261/485/372 199/379/372
+f 195/371/373 259/482/373 197/375/373
+f 193/367/374 257/479/375 195/371/375
+f 191/363/376 255/476/376 193/367/377
+f 217/417/378 219/421/378 153/287/378
+f 187/355/379 251/470/379 189/359/379
+f 249/468/380 251/470/380 185/352/380
+f 226/431/381 225/430/381 224/428/381
+f 238/451/382 240/454/382 237/450/382
+f 178/337/169 176/333/169 177/336/169
+f 236/448/383 238/451/383 235/447/383
+f 266/493/217 268/496/217 265/491/217
+f 172/325/182 170/321/182 171/324/182
+f 232/440/121 234/444/121 231/439/121
+f 166/313/384 165/312/384 168/317/384
+f 234/445/163 236/448/163 233/443/163
+f 162/305/385 161/304/385 164/309/385
+f 160/301/386 159/300/386 162/305/386
+f 180/341/387 178/337/387 179/340/387
+f 156/293/175 155/292/175 158/297/175
+f 154/289/388 153/288/388 156/293/388
+f 210/403/389 209/402/389 212/407/389
+f 201/385/390 199/379/390 265/492/390
+f 212/407/391 211/406/391 214/411/391
+f 245/461/392 247/465/392 181/343/392
+f 206/395/177 205/394/177 208/399/177
+f 204/391/393 203/390/393 206/395/393
+f 240/454/197 242/457/197 239/453/197
+f 268/496/212 270/499/212 267/495/212
+f 198/377/159 196/373/159 197/376/159
+f 196/373/394 194/369/394 195/372/394
+f 194/369/180 192/365/180 193/368/180
+f 192/365/395 190/361/395 191/364/395
+f 190/361/157 188/357/157 189/360/157
+f 188/357/396 186/353/396 187/356/396
+f 272/502/174 274/505/174 271/501/174
+f 270/499/116 272/502/116 269/498/116
+f 242/457/115 244/460/115 241/456/115
+f 200/382/204 199/380/204 202/387/204
+f 158/297/397 157/296/397 160/301/397
+f 174/329/146 172/325/146 173/328/146
+f 264/490/398 266/493/398 263/489/398
+f 232/440/176 231/439/176 230/437/176
+f 230/437/399 229/436/399 228/434/399
+f 228/434/198 227/433/198 226/431/198
+f 182/345/97 180/341/97 181/344/97
+f 224/428/191 223/427/191 222/425/191
+f 222/425/400 221/424/400 220/422/400
+f 218/419/401 220/422/401 217/418/401
+f 202/387/154 201/386/154 204/391/154
+f 276/508/192 278/511/192 275/507/192
+f 274/505/402 276/508/402 273/504/402
+f 243/459/403 244/460/403 245/462/403
+f 200/383/155 198/377/155 199/381/155
+f 176/333/147 174/329/147 175/332/147
+f 164/309/404 163/308/404 166/313/404
+f 264/490/405 263/489/405 262/487/405
+f 262/487/406 261/486/406 260/484/406
+f 260/484/199 259/483/199 258/481/199
+f 258/481/173 257/480/173 256/478/173
+f 256/478/407 255/477/407 254/475/407
+f 254/475/408 253/474/408 252/472/408
+f 252/472/186 251/471/186 250/469/186
+f 471/838/409 469/835/409 472/839/409
+f 519/935/410 520/937/410 518/934/410
+f 519/935/411 521/939/411 520/937/411
+f 413/749/112 414/750/112 416/754/112
+f 486/868/371 488/871/371 422/763/371
+f 476/848/412 474/844/412 473/841/412
+f 535/967/167 473/841/167 536/970/167
+f 521/939/411 522/942/411 520/937/411
+f 440/790/114 442/793/114 441/791/114
+f 442/792/413 506/910/414 508/914/415
+f 476/847/416 412/748/417 474/843/418
+f 490/876/204 488/872/204 487/869/204
+f 490/877/419 426/768/419 424/765/419
+f 448/801/420 446/798/153 510/918/420
+f 462/824/198 464/827/198 463/826/198
+f 486/867/143 484/863/143 483/861/143
+f 461/823/198 462/824/198 463/826/198
+f 518/933/421 454/811/421 516/930/421
+f 483/861/422 484/863/422 481/857/422
+f 511/919/169 513/923/169 512/921/169
+f 458/818/423 460/822/423 459/820/423
+f 430/774/184 428/771/184 492/881/184
+f 436/783/164 438/787/164 437/785/164
+f 480/856/156 478/852/156 477/849/156
+f 435/782/424 436/783/424 437/785/424
+f 417/755/425 415/752/425 418/757/425
+f 440/789/426 438/786/426 504/906/426
+f 466/830/427 530/957/427 532/961/427
+f 499/895/428 501/899/428 500/898/428
+f 482/859/429 418/756/430 416/753/431
+f 520/938/432 456/815/432 518/933/432
+f 531/959/122 532/962/122 529/955/122
+f 421/761/212 419/758/212 422/762/212
+f 488/871/433 490/877/433 424/765/433
+f 453/809/163 454/810/163 456/816/163
+f 460/822/399 462/824/399 461/823/399
+f 478/851/346 414/751/346 476/847/346
+f 523/943/123 524/946/123 521/939/123
+f 503/903/434 504/905/434 502/902/434
+f 456/815/158 454/811/158 518/933/435
+f 489/873/436 490/876/436 487/869/436
+f 468/833/400 470/836/400 469/835/400
+f 464/828/437 462/825/437 528/954/437
+f 484/863/438 482/860/438 481/857/438
+f 480/855/439 482/859/439 416/753/439
+f 515/927/146 517/931/146 516/929/146
+f 453/809/218 451/806/218 454/810/218
+f 502/901/440 438/786/440 500/897/440
+f 412/748/441 410/744/441 474/843/441
+f 495/887/394 496/889/394 494/886/394
+f 491/879/159 493/883/159 492/882/159
+f 455/813/442 456/814/442 458/818/442
+f 474/843/443 410/744/443 536/969/443
+f 497/891/444 498/893/444 496/889/444
+f 411/746/192 412/747/192 414/750/192
+f 492/881/372 428/771/372 490/877/372
+f 451/806/383 452/807/383 454/810/383
+f 423/764/217 421/761/217 424/766/217
+f 452/808/194 450/804/194 514/925/194
+f 434/780/445 436/783/445 435/782/445
+f 488/871/130 424/765/130 422/763/130
+f 462/825/446 526/950/447 528/954/446
+f 409/743/172 410/745/172 412/747/172
+f 484/864/448 420/760/448 418/756/165
+f 410/744/449 472/840/449 536/969/449
+f 433/779/450 434/780/450 435/782/450
+f 493/883/451 494/886/451 492/882/451
+f 533/963/175 534/965/175 531/959/175
+f 535/967/388 536/970/388 533/963/388
+f 467/832/400 468/833/400 469/835/400
+f 512/922/452 448/801/452 510/918/452
+f 429/773/453 430/775/453 431/776/453
+f 494/885/373 430/774/373 492/881/373
+f 513/923/169 514/926/169 512/921/169
+f 468/834/454 466/830/455 532/961/454
+f 419/758/116 417/755/116 420/759/116
+f 480/855/111 416/753/111 414/751/111
+f 413/749/192 411/746/192 414/750/192
+f 517/931/146 518/934/146 516/929/146
+f 446/798/456 444/795/456 508/914/456
+f 522/941/457 458/819/457 520/938/457
+f 432/777/173 434/780/173 433/779/173
+f 511/919/387 512/921/387 510/917/387
+f 432/778/458 430/774/458 494/885/458
+f 469/835/178 470/836/178 472/839/178
+f 526/950/459 462/825/460 460/821/461
+f 530/958/386 528/953/386 527/951/386
+f 487/869/462 488/872/462 485/865/462
+f 527/951/385 528/953/385 525/947/385
+f 428/772/406 430/775/406 429/773/406
+f 513/923/463 515/927/463 514/926/463
+f 525/947/151 526/949/151 523/943/151
+f 528/953/385 526/949/385 525/947/385
+f 411/746/172 409/743/172 412/747/172
+f 458/819/138 456/815/464 520/938/138
+f 507/911/110 508/913/110 505/907/110
+f 491/879/465 492/882/465 490/878/465
+f 428/771/466 426/768/131 490/877/131
+f 496/890/467 432/778/467 494/885/467
+f 463/826/119 464/827/119 465/829/119
+f 505/907/132 506/909/132 504/905/132
+f 415/752/402 413/749/402 416/754/402
+f 423/764/398 424/766/398 426/769/398
+f 526/949/404 524/946/404 523/943/404
+f 481/857/209 482/860/209 479/853/209
+f 500/897/468 436/784/468 498/894/468
+f 516/930/469 452/808/469 514/925/469
+f 507/911/97 509/915/97 508/913/97
+f 497/891/139 499/895/139 498/893/139
+f 447/800/197 448/802/197 450/805/197
+f 529/955/133 530/958/133 527/951/133
+f 478/851/470 480/855/470 414/751/470
+f 460/821/349 458/819/349 524/945/349
+f 425/767/471 426/769/471 427/770/471
+f 472/839/140 410/745/140 409/743/140
+f 485/865/393 486/867/393 483/861/393
+f 509/915/387 511/919/387 510/917/387
+f 508/913/472 506/909/473 505/907/473
+f 498/894/474 434/781/475 496/890/475
+f 477/849/391 478/852/391 475/845/391
+f 501/899/396 503/903/396 502/902/396
+f 455/812/163 453/809/163 456/816/163
+f 471/838/476 472/839/476 409/743/476
+f 457/817/176 458/818/176 459/820/176
+f 449/803/477 447/800/477 450/805/477
+f 493/883/394 495/887/394 494/886/394
+f 517/931/182 519/935/182 518/934/182
+f 465/829/478 466/831/478 467/832/478
+f 489/874/155 491/879/155 490/878/155
+f 482/859/479 484/864/479 418/756/479
+f 437/785/480 438/787/480 439/788/480
+f 439/788/114 440/790/114 441/791/114
+f 466/830/347 464/828/347 530/957/347
+f 442/792/481 440/789/481 506/910/362
+f 419/758/482 420/759/482 422/762/482
+f 415/752/174 416/754/174 418/757/174
+f 445/797/181 443/794/181 444/796/181
+f 466/831/478 468/833/478 467/832/478
+f 443/794/193 441/791/193 444/796/193
+f 430/775/453 432/777/453 431/776/453
+f 482/860/209 480/856/209 479/853/209
+f 431/776/173 432/777/173 433/779/173
+f 421/761/217 422/762/217 424/766/217
+f 514/925/483 450/804/341 512/922/341
+f 479/853/156 480/856/156 477/849/156
+f 426/769/405 428/772/405 427/770/405
+f 499/895/139 500/898/139 498/893/139
+f 484/864/484 486/868/484 420/760/484
+f 509/915/485 510/917/485 508/913/485
+f 457/817/121 455/813/121 458/818/121
+f 450/804/486 448/801/487 512/922/486
+f 444/795/488 442/792/488 508/914/488
+f 503/903/142 505/907/142 504/905/142
+f 447/800/115 445/797/115 448/802/115
+f 425/767/136 423/764/136 426/769/136
+f 510/918/363 446/798/363 508/914/363
+f 446/799/489 445/797/489 444/796/489
+f 427/770/190 428/772/190 429/773/190
+f 470/837/185 534/966/185 536/969/185
+f 524/945/351 526/950/490 460/821/490
+f 534/965/491 532/962/491 531/959/491
+f 440/789/492 504/906/492 506/910/492
+f 417/755/493 418/757/493 420/759/493
+f 441/791/494 442/793/494 444/796/494
+f 468/834/196 532/961/196 534/966/495
+f 438/786/200 502/901/200 504/906/200
+f 438/786/496 436/784/496 500/897/496
+f 536/970/216 534/965/216 533/963/216
+f 458/819/219 522/941/219 524/945/219
+f 478/852/497 476/848/497 475/845/497
+f 445/797/115 446/799/115 448/802/115
+f 414/751/498 412/748/498 476/847/498
+f 438/787/186 440/790/186 439/788/186
+f 459/820/118 460/822/118 461/823/118
+f 451/806/382 449/803/382 452/807/382
+f 464/827/119 466/831/119 465/829/119
+f 464/828/499 528/954/500 530/957/499
+f 515/927/147 516/929/147 514/926/147
+f 436/784/501 434/781/501 498/894/501
+f 470/837/502 468/834/355 534/966/502
+f 488/872/462 486/867/462 485/865/462
+f 501/899/157 502/902/157 500/898/157
+f 486/868/503 422/763/503 420/760/503
+f 449/803/382 450/805/382 452/807/382
+f 472/840/378 470/837/378 536/969/378
+f 495/887/444 497/891/444 496/889/444
+f 434/781/120 432/778/120 496/890/120
+f 532/962/122 530/958/122 529/955/122
+f 524/946/123 522/942/123 521/939/123
+f 475/845/124 476/848/124 473/841/124
+f 473/841/126 474/844/126 536/970/126
+f 454/811/504 452/808/504 516/930/504
+g KsHauptsignal_Plane.006_Lights
+usemtl Lights
+f 88/188/505 28/66/505 40/90/505
+f 40/90/505 44/98/505 56/122/505
+f 88/188/506 64/138/506 72/156/506
+f 72/156/506 76/164/506 88/188/506
+f 40/90/507 28/66/507 32/74/507
+f 32/74/505 34/78/505 40/90/505
+f 56/122/505 44/98/505 48/106/505
+f 48/106/505 52/114/505 56/122/505
+f 56/122/508 58/126/508 64/138/508
+f 64/138/506 68/146/506 70/150/506
+f 88/188/506 76/164/506 80/172/506
+f 80/172/509 84/180/509 88/188/509
+f 88/188/510 26/62/510 28/66/510
+f 28/66/510 30/70/510 32/74/510
+f 40/90/507 34/78/507 36/82/507
+f 36/82/507 38/86/507 40/90/507
+f 40/90/511 42/94/511 44/98/511
+f 44/98/512 46/102/512 48/106/512
+f 48/106/513 50/110/513 52/114/513
+f 52/114/514 54/118/515 56/122/515
+f 64/138/516 58/126/517 60/130/516
+f 60/130/518 62/134/518 64/138/518
+f 64/138/517 66/142/519 68/146/519
+f 64/138/506 70/150/506 72/156/506
+f 72/156/520 74/160/520 76/164/520
+f 76/164/506 78/168/506 80/172/521
+f 80/172/509 82/176/509 84/180/509
+f 84/180/522 86/184/522 88/188/522
+f 64/138/523 88/188/523 40/90/523
+f 156/294/524 158/298/524 160/302/524
+f 200/384/525 204/392/525 216/416/525
+f 192/366/526 186/354/526 188/358/526
+f 176/334/527 180/342/527 184/350/527
+f 212/408/528 214/412/528 216/416/529
+f 188/358/530 190/362/530 192/366/530
+f 192/366/531 196/374/532 198/378/532
+f 184/350/533 172/326/533 176/334/533
+f 204/392/534 206/396/535 208/400/534
+f 200/384/536 202/388/537 204/392/536
+f 216/416/538 192/366/538 200/384/538
+f 168/318/539 162/306/539 164/310/539
+f 164/310/540 166/314/541 168/318/540
+f 160/302/542 162/306/542 168/318/542
+f 176/334/543 178/338/543 180/342/543
+f 208/400/538 210/404/538 212/408/538
+f 168/318/544 170/322/541 172/326/544
+f 168/318/540 172/326/540 184/350/540
+f 168/318/545 156/294/545 160/302/545
+f 180/342/546 182/346/547 184/350/546
+f 208/400/538 212/408/538 216/416/538
+f 216/416/538 204/392/538 208/400/538
+f 192/366/548 194/370/548 196/374/548
+f 184/350/549 186/354/549 192/366/549
+f 192/366/550 198/378/550 200/384/550
+f 216/416/551 156/294/551 168/318/551
+f 216/416/552 154/290/552 156/294/552
+f 172/326/553 174/330/553 176/334/553
+f 192/366/554 216/416/554 168/318/554
+f 56/122/555 64/138/555 40/90/555
+f 344/644/505 284/522/505 296/546/505
+f 296/546/505 300/554/505 312/578/505
+f 344/644/506 320/594/506 328/612/506
+f 328/612/506 332/620/506 344/644/506
+f 296/546/505 284/522/556 288/530/556
+f 288/530/505 290/534/505 296/546/505
+f 312/578/507 300/554/507 304/562/507
+f 304/562/512 308/570/512 312/578/512
+f 312/578/557 314/582/557 320/594/557
+f 320/594/506 324/602/506 326/606/506
+f 344/644/506 332/620/506 336/628/506
+f 336/628/558 340/636/558 344/644/558
+f 344/644/559 282/518/559 284/522/559
+f 284/522/560 286/526/560 288/530/560
+f 296/546/507 290/534/507 292/538/507
+f 292/538/561 294/542/561 296/546/561
+f 296/546/511 298/550/511 300/554/511
+f 300/554/507 302/558/507 304/562/507
+f 304/562/513 306/566/513 308/570/513
+f 308/570/562 310/574/562 312/578/562
+f 320/594/563 314/582/563 316/586/563
+f 316/586/563 318/590/563 320/594/563
+f 320/594/519 322/598/519 324/602/519
+f 320/594/519 326/606/519 328/612/519
+f 328/612/564 330/616/564 332/620/564
+f 332/620/565 334/624/565 336/628/565
+f 336/628/566 338/632/566 340/636/566
+f 340/636/567 342/640/567 344/644/567
+f 320/594/523 344/644/523 296/546/523
+f 312/578/555 320/594/555 296/546/555
+f 184/350/568 192/366/568 168/318/568
+f 509/916/547 507/912/547 505/908/547
+f 521/940/539 527/952/539 525/948/539
+f 473/842/551 533/964/551 521/940/551
+f 521/940/544 519/936/544 517/932/544
+f 517/932/569 515/928/569 513/924/569
+f 533/964/524 531/960/524 529/956/524
+f 497/892/570 495/888/570 493/884/530
+f 489/875/537 487/870/537 485/866/537
+f 477/850/571 475/846/572 473/842/571
+f 473/842/538 497/892/538 489/875/538
+f 497/892/573 503/904/573 501/900/573
+f 505/908/574 517/932/533 513/924/574
+f 473/842/575 535/968/576 533/964/576
+f 481/858/577 477/850/577 473/842/578
+f 497/892/554 473/842/554 521/940/554
+f 501/900/579 499/896/579 497/892/580
+f 497/892/581 493/884/581 491/880/581
+f 521/940/540 517/932/540 505/908/540
+f 505/908/568 497/892/568 521/940/568
+f 473/842/538 485/866/538 481/858/538
+f 497/892/550 491/880/550 489/875/550
+f 529/956/541 527/952/582 521/940/582
+f 521/940/545 533/964/545 529/956/545
+f 481/858/583 479/854/583 477/850/583
+f 505/908/311 503/904/584 497/892/311
+f 513/924/543 511/920/543 509/916/543
+f 489/875/525 485/866/525 473/842/525
+f 513/924/527 509/916/527 505/908/527
+f 525/948/585 523/944/585 521/940/551
+f 485/866/586 483/862/586 481/858/586
+l 570 571
+l 540 537
+l 572 569
+l 571 572
+l 538 539
+l 539 540
+l 543 544
+l 575 576
+l 541 542
+l 573 574
+l 542 543
+l 574 575
diff --git a/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne3_smr0.obj b/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne3_smr0.obj
new file mode 100644
index 0000000..01bc666
--- /dev/null
+++ b/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne3_smr0.obj
@@ -0,0 +1,212 @@
+# Blender v3.0.0 OBJ File: ''
+# www.blender.org
+mtllib advtrains_signals_ks_sign_ne3_smr0.mtl
+o Cube.001
+v 0.100000 1.200000 -0.324543
+v 0.100000 1.200000 -0.344543
+v -0.100000 1.200000 -0.344543
+v -0.100000 1.200000 -0.324543
+v -0.100000 0.700000 -0.324543
+v 0.100000 0.700000 -0.324543
+v 0.100000 0.700000 -0.344543
+v -0.100000 0.700000 -0.344543
+v 0.024805 1.055075 -0.324528
+v 0.000000 1.055075 -0.334802
+v 0.000000 1.055075 -0.334802
+v 0.024805 1.055075 -0.324528
+v -0.035079 1.055075 -0.299723
+v -0.024805 1.055075 -0.274918
+v -0.024805 1.055075 -0.274918
+v -0.035079 1.055075 -0.299723
+v 0.024805 1.055075 -0.274918
+v 0.035079 1.055075 -0.299723
+v 0.035079 1.055075 -0.299723
+v 0.024805 1.055075 -0.274918
+v -0.024805 1.055075 -0.324528
+v -0.024805 1.055075 -0.324528
+v 0.000000 1.055075 -0.264644
+v 0.000000 1.055075 -0.264644
+v 0.024805 -0.499729 -0.274918
+v 0.035080 -0.499729 -0.299723
+v 0.000000 -0.499729 -0.264644
+v 0.024805 -0.499729 -0.324528
+v 0.000000 -0.499729 -0.334802
+v -0.024804 -0.499729 -0.274918
+v -0.024804 -0.499729 -0.324528
+v -0.035079 -0.499729 -0.299723
+v 0.100000 1.150000 -0.344543
+v 0.100000 1.100000 -0.344543
+v 0.100000 1.050000 -0.344543
+v 0.100000 1.000000 -0.344543
+v 0.100000 0.950000 -0.344543
+v 0.100000 0.900000 -0.344543
+v 0.100000 0.850000 -0.344543
+v 0.100000 0.800000 -0.344543
+v 0.100000 0.750000 -0.344543
+v -0.100000 1.150000 -0.344543
+v -0.100000 1.100000 -0.344543
+v -0.100000 1.050000 -0.344543
+v -0.100000 1.000000 -0.344543
+v -0.100000 0.950000 -0.344543
+v -0.100000 0.900000 -0.344543
+v -0.100000 0.850000 -0.344543
+v -0.100000 0.800000 -0.344543
+v -0.100000 0.750000 -0.344543
+vt 0.583333 0.750000
+vt 0.583333 0.500000
+vt 0.583333 0.250000
+vt 0.583333 1.000000
+vt 0.125000 0.500000
+vt 0.375000 0.500000
+vt 0.375000 0.750000
+vt 0.125000 0.750000
+vt 0.583333 0.000000
+vt 0.562500 0.250000
+vt 0.541666 0.250000
+vt 0.520833 0.250000
+vt 0.500000 0.250000
+vt 0.479166 0.250000
+vt 0.458333 0.250000
+vt 0.437500 0.250000
+vt 0.416667 0.250000
+vt 0.395833 0.250000
+vt 0.375000 0.250000
+vt 0.375000 0.000000
+vt 0.541666 0.500000
+vt 0.562500 0.500000
+vt 0.395833 0.500000
+vt 0.416667 0.500000
+vt 0.437500 0.500000
+vt 0.458333 0.500000
+vt 0.479166 0.500000
+vt 0.500000 0.500000
+vt 0.520833 0.500000
+vt 0.375000 1.000000
+vt 0.372863 0.578719
+vt 0.372317 0.613951
+vt 0.371666 0.613734
+vt 0.372221 0.578991
+vt 0.315877 0.640082
+vt 0.292344 0.614764
+vt 0.292986 0.614492
+vt 0.316131 0.639438
+vt 0.315346 0.553400
+vt 0.349331 0.553400
+vt 0.349077 0.554044
+vt 0.315664 0.554020
+vt 0.349861 0.640082
+vt 0.349543 0.639462
+vt 0.292889 0.579531
+vt 0.293541 0.579748
+vt 0.444102 0.636706
+vt 0.409596 0.636706
+vt 0.385196 0.612306
+vt 0.385196 0.577799
+vt 0.409596 0.553400
+vt 0.444102 0.553400
+vt 0.468502 0.577800
+vt 0.468502 0.612306
+vt 0.037160 1.987962
+vt 0.036793 0.000397
+vt 0.071486 0.000384
+vt 0.071853 1.987948
+vt 0.002468 1.987974
+vt 0.002100 0.000409
+vt 0.106546 1.987936
+vt 0.106179 0.000371
+vt 0.140872 0.000358
+vt 0.141239 1.987923
+vt 0.245318 1.987885
+vt 0.244951 0.000319
+vt 0.279643 0.000307
+vt 0.280011 1.987871
+vt 0.175932 1.987910
+vt 0.175565 0.000345
+vt 0.210258 0.000332
+vt 0.210625 1.987897
+vt 0.812500 1.000000
+vt 0.000000 1.000000
+vt 0.000000 0.937500
+vt 0.000000 0.875000
+vt 0.812500 0.875000
+vt 0.812500 0.937500
+vt 0.000000 0.812500
+vt 0.812500 0.812500
+vt 0.000000 0.750000
+vt 0.812500 0.750000
+vt 0.000000 0.687500
+vt 0.812500 0.687500
+vt 0.000000 0.625000
+vt 0.812500 0.625000
+vt 0.000000 0.562500
+vt 0.812500 0.562500
+vt 0.000000 0.500000
+vt 0.812500 0.500000
+vt 0.000000 0.437500
+vt 0.812500 0.437500
+vt 0.000000 0.375000
+vt 0.812500 0.375000
+vt 0.812500 1.000000
+vt 0.000000 0.937500
+vt 0.812500 0.937500
+vn 0.9239 0.3827 -0.0000
+vn -0.0000 0.3826 -0.9239
+vn -0.9239 0.3827 -0.0001
+vn 0.0000 0.3827 0.9239
+vn 0.7071 -0.7071 0.0000
+vn -0.7071 -0.7071 -0.0000
+vn -1.0000 0.0000 0.0000
+vn 1.0000 0.0000 0.0000
+vn 0.0000 -0.0000 1.0000
+vn 0.0000 1.0000 0.0000
+vn 0.9239 0.0000 0.3827
+vn 0.3827 -0.0000 0.9239
+vn 0.3827 0.0000 -0.9239
+vn 0.9239 0.0000 -0.3827
+vn -0.3827 -0.0000 0.9239
+vn -0.9239 -0.0000 -0.3827
+vn -0.9239 -0.0000 0.3827
+vn -0.3827 -0.0000 -0.9239
+vn -0.3952 0.8351 -0.3827
+vn 0.0000 0.0000 -1.0000
+vn 0.0000 0.7071 -0.7071
+g Cube.001_Cube.001_Default_OBJ
+usemtl Default_OBJ
+s off
+f 1/1/1 2/2/2 3/3/3 4/4/4
+f 8/5/5 7/6/6 6/7/6 5/8/5
+f 4/9/7 3/3/7 42/10/7 43/11/7 44/12/7 45/13/7 46/14/7 47/15/7 48/16/7 49/17/7 50/18/7 8/19/7 5/20/7
+f 34/21/8 33/22/8 2/2/8 1/1/8 6/7/8 7/6/8 41/23/8 40/24/8 39/25/8 38/26/8 37/27/8 36/28/8 35/29/8
+f 1/1/9 4/4/9 5/30/9 6/7/9
+s 1
+f 9/31/9 10/32/9 11/33/9 12/34/9
+f 13/35/9 14/36/9 15/37/9 16/38/9
+f 17/39/9 18/40/9 19/41/9 20/42/9
+f 10/32/9 21/43/9 22/44/9 11/33/9
+f 14/36/9 23/45/9 24/46/9 15/37/9
+f 18/40/9 9/31/9 12/34/9 19/41/9
+f 21/47/10 13/48/10 14/49/10 23/50/10 17/51/10 18/52/10 9/53/10 10/54/10
+f 20/55/11 25/56/11 26/57/11 19/58/11
+f 21/43/9 13/35/9 16/38/9 22/44/9
+f 23/45/9 17/39/9 20/42/9 24/46/9
+f 24/59/12 27/60/12 25/56/12 20/55/12
+f 12/61/13 28/62/13 29/63/13 11/64/13
+f 19/58/14 26/57/14 28/62/14 12/61/14
+f 15/65/15 30/66/15 27/67/15 24/68/15
+f 22/69/16 31/70/16 32/71/16 16/72/16
+f 16/72/17 32/71/17 30/66/17 15/65/17
+f 11/64/18 29/63/18 31/70/18 22/69/18
+g Cube.001_Cube.001_Anzeige
+usemtl Anzeige
+s off
+f 3/73/19 2/74/20 33/75/20
+f 34/76/20 42/77/20 3/78/20 33/75/20
+f 35/79/20 43/80/20 42/77/20 34/76/20
+f 36/81/20 44/82/20 43/80/20 35/79/20
+f 37/83/20 45/84/20 44/82/20 36/81/20
+f 38/85/20 46/86/20 45/84/20 37/83/20
+f 39/87/20 47/88/20 46/86/20 38/85/20
+f 40/89/20 48/90/20 47/88/20 39/87/20
+f 41/91/20 49/92/20 48/90/20 40/89/20
+f 7/93/20 50/94/20 49/92/20 41/91/20
+f 50/95/20 7/96/20 8/97/21
diff --git a/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne3_smr30.obj b/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne3_smr30.obj
new file mode 100644
index 0000000..a20c0bf
--- /dev/null
+++ b/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne3_smr30.obj
@@ -0,0 +1,213 @@
+# Blender v3.0.0 OBJ File: ''
+# www.blender.org
+mtllib advtrains_signals_ks_sign_ne3_smr30.mtl
+o Cube.001
+v 0.234304 1.200000 -0.245825
+v 0.243228 1.200000 -0.263724
+v 0.064241 1.200000 -0.352963
+v 0.055317 1.200000 -0.335064
+v 0.055317 0.700000 -0.335064
+v 0.234304 0.700000 -0.245825
+v 0.243228 0.700000 -0.263724
+v 0.064241 0.700000 -0.352963
+v 0.167003 1.055075 -0.279363
+v 0.149388 1.055075 -0.299626
+v 0.149388 1.055075 -0.299626
+v 0.167003 1.055075 -0.279363
+v 0.102342 1.055075 -0.283885
+v 0.100469 1.055075 -0.257101
+v 0.100469 1.055075 -0.257101
+v 0.102342 1.055075 -0.283885
+v 0.144867 1.055075 -0.234966
+v 0.165129 1.055075 -0.252580
+v 0.165129 1.055075 -0.252580
+v 0.144867 1.055075 -0.234966
+v 0.122605 1.055075 -0.301499
+v 0.122605 1.055075 -0.301499
+v 0.118084 1.055075 -0.236839
+v 0.118084 1.055075 -0.236839
+v 0.144867 -0.499729 -0.234966
+v 0.165130 -0.499729 -0.252580
+v 0.118084 -0.499729 -0.236839
+v 0.167003 -0.499729 -0.279363
+v 0.149388 -0.499729 -0.299626
+v 0.100470 -0.499729 -0.257101
+v 0.122606 -0.499729 -0.301499
+v 0.102342 -0.499729 -0.283885
+v 0.243228 1.150000 -0.263724
+v 0.243228 1.100000 -0.263724
+v 0.243228 1.050000 -0.263724
+v 0.243228 1.000000 -0.263724
+v 0.243228 0.950000 -0.263724
+v 0.243228 0.900000 -0.263724
+v 0.243228 0.850000 -0.263724
+v 0.243228 0.800000 -0.263724
+v 0.243228 0.750000 -0.263724
+v 0.064241 1.150000 -0.352963
+v 0.064241 1.100000 -0.352963
+v 0.064241 1.050000 -0.352963
+v 0.064241 1.000000 -0.352963
+v 0.064241 0.950000 -0.352963
+v 0.064241 0.900000 -0.352963
+v 0.064241 0.850000 -0.352963
+v 0.064241 0.800000 -0.352963
+v 0.064241 0.750000 -0.352963
+vt 0.583333 0.750000
+vt 0.583333 0.500000
+vt 0.583333 0.250000
+vt 0.583333 1.000000
+vt 0.125000 0.500000
+vt 0.375000 0.500000
+vt 0.375000 0.750000
+vt 0.125000 0.750000
+vt 0.583333 0.000000
+vt 0.562500 0.250000
+vt 0.541666 0.250000
+vt 0.520833 0.250000
+vt 0.500000 0.250000
+vt 0.479166 0.250000
+vt 0.458333 0.250000
+vt 0.437500 0.250000
+vt 0.416667 0.250000
+vt 0.395833 0.250000
+vt 0.375000 0.250000
+vt 0.375000 0.000000
+vt 0.541666 0.500000
+vt 0.562500 0.500000
+vt 0.395833 0.500000
+vt 0.416667 0.500000
+vt 0.437500 0.500000
+vt 0.458333 0.500000
+vt 0.479166 0.500000
+vt 0.500000 0.500000
+vt 0.520833 0.500000
+vt 0.375000 1.000000
+vt 0.372863 0.578719
+vt 0.372317 0.613951
+vt 0.371666 0.613734
+vt 0.372221 0.578991
+vt 0.315877 0.640082
+vt 0.292344 0.614764
+vt 0.292986 0.614492
+vt 0.316131 0.639438
+vt 0.315346 0.553400
+vt 0.349331 0.553400
+vt 0.349077 0.554044
+vt 0.315664 0.554020
+vt 0.349861 0.640082
+vt 0.349543 0.639462
+vt 0.292889 0.579531
+vt 0.293541 0.579748
+vt 0.444102 0.636706
+vt 0.409596 0.636706
+vt 0.385196 0.612306
+vt 0.385196 0.577799
+vt 0.409596 0.553400
+vt 0.444102 0.553400
+vt 0.468502 0.577800
+vt 0.468502 0.612306
+vt 0.037160 1.987962
+vt 0.036793 0.000397
+vt 0.071486 0.000384
+vt 0.071853 1.987948
+vt 0.002468 1.987974
+vt 0.002100 0.000409
+vt 0.106546 1.987936
+vt 0.106179 0.000371
+vt 0.140872 0.000358
+vt 0.141239 1.987923
+vt 0.245318 1.987885
+vt 0.244951 0.000319
+vt 0.279643 0.000307
+vt 0.280011 1.987871
+vt 0.175932 1.987910
+vt 0.175565 0.000345
+vt 0.210258 0.000332
+vt 0.210625 1.987897
+vt 0.812500 1.000000
+vt 0.000000 1.000000
+vt 0.000000 0.937500
+vt 0.000000 0.875000
+vt 0.812500 0.875000
+vt 0.812500 0.937500
+vt 0.000000 0.812500
+vt 0.812500 0.812500
+vt 0.000000 0.750000
+vt 0.812500 0.750000
+vt 0.000000 0.687500
+vt 0.812500 0.687500
+vt 0.000000 0.625000
+vt 0.812500 0.625000
+vt 0.000000 0.562500
+vt 0.812500 0.562500
+vt 0.000000 0.500000
+vt 0.812500 0.500000
+vt 0.000000 0.437500
+vt 0.812500 0.437500
+vt 0.000000 0.375000
+vt 0.812500 0.375000
+vt 0.812500 1.000000
+vt 0.000000 0.937500
+vt 0.812500 0.937500
+vn 0.8268 0.3827 0.4122
+vn 0.4122 0.3826 -0.8269
+vn -0.8268 0.3827 -0.4123
+vn -0.4122 0.3827 0.8268
+vn 0.6328 -0.7071 0.3155
+vn -0.6328 -0.7071 -0.3155
+vn -0.8949 -0.0000 -0.4462
+vn 0.8949 -0.0000 0.4462
+vn -0.4462 -0.0000 0.8949
+vn 0.0000 0.0000 1.0000
+vn 0.0000 1.0000 -0.0000
+vn 0.6561 0.0000 0.7547
+vn -0.0698 -0.0000 0.9976
+vn 0.7547 0.0000 -0.6561
+vn 0.9976 0.0000 0.0698
+vn -0.7547 -0.0000 0.6561
+vn -0.6561 -0.0000 -0.7547
+vn -0.9976 -0.0000 -0.0698
+vn 0.0698 -0.0000 -0.9976
+vn -0.1830 0.8351 -0.5188
+vn 0.4462 0.0000 -0.8949
+vn 0.3155 0.7071 -0.6328
+g Cube.001_Cube.001_Default_OBJ
+usemtl Default_OBJ
+s off
+f 1/1/1 2/2/2 3/3/3 4/4/4
+f 8/5/5 7/6/6 6/7/6 5/8/5
+f 4/9/7 3/3/7 42/10/7 43/11/7 44/12/7 45/13/7 46/14/7 47/15/7 48/16/7 49/17/7 50/18/7 8/19/7 5/20/7
+f 34/21/8 33/22/8 2/2/8 1/1/8 6/7/8 7/6/8 41/23/8 40/24/8 39/25/8 38/26/8 37/27/8 36/28/8 35/29/8
+f 1/1/9 4/4/9 5/30/9 6/7/9
+s 1
+f 9/31/10 10/32/10 11/33/10 12/34/10
+f 13/35/10 14/36/10 15/37/10 16/38/10
+f 17/39/10 18/40/10 19/41/10 20/42/10
+f 10/32/10 21/43/10 22/44/10 11/33/10
+f 14/36/10 23/45/10 24/46/10 15/37/10
+f 18/40/10 9/31/10 12/34/10 19/41/10
+f 21/47/11 13/48/11 14/49/11 23/50/11 17/51/11 18/52/11 9/53/11 10/54/11
+f 20/55/12 25/56/12 26/57/12 19/58/12
+f 21/43/10 13/35/10 16/38/10 22/44/10
+f 23/45/10 17/39/10 20/42/10 24/46/10
+f 24/59/13 27/60/13 25/56/13 20/55/13
+f 12/61/14 28/62/14 29/63/14 11/64/14
+f 19/58/15 26/57/15 28/62/15 12/61/15
+f 15/65/16 30/66/16 27/67/16 24/68/16
+f 22/69/17 31/70/17 32/71/17 16/72/17
+f 16/72/18 32/71/18 30/66/18 15/65/18
+f 11/64/19 29/63/19 31/70/19 22/69/19
+g Cube.001_Cube.001_Anzeige
+usemtl Anzeige
+s off
+f 3/73/20 2/74/21 33/75/21
+f 34/76/21 42/77/21 3/78/21 33/75/21
+f 35/79/21 43/80/21 42/77/21 34/76/21
+f 36/81/21 44/82/21 43/80/21 35/79/21
+f 37/83/21 45/84/21 44/82/21 36/81/21
+f 38/85/21 46/86/21 45/84/21 37/83/21
+f 39/87/21 47/88/21 46/86/21 38/85/21
+f 40/89/21 48/90/21 47/88/21 39/87/21
+f 41/91/21 49/92/21 48/90/21 40/89/21
+f 7/93/21 50/94/21 49/92/21 41/91/21
+f 50/95/21 7/96/21 8/97/22
diff --git a/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne3_smr45.obj b/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne3_smr45.obj
new file mode 100644
index 0000000..ed1047b
--- /dev/null
+++ b/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne3_smr45.obj
@@ -0,0 +1,213 @@
+# Blender v3.0.0 OBJ File: ''
+# www.blender.org
+mtllib advtrains_signals_ks_sign_ne3_smr45.mtl
+o Cube.001
+v 0.300197 1.200000 -0.158776
+v 0.314339 1.200000 -0.172918
+v 0.172918 1.200000 -0.314339
+v 0.158776 1.200000 -0.300197
+v 0.158776 0.700000 -0.300197
+v 0.300197 0.700000 -0.158776
+v 0.314339 0.700000 -0.172918
+v 0.172918 0.700000 -0.314339
+v 0.247016 1.055075 -0.211936
+v 0.236741 1.055075 -0.236741
+v 0.236741 1.055075 -0.236741
+v 0.247016 1.055075 -0.211936
+v 0.187132 1.055075 -0.236741
+v 0.176857 1.055075 -0.211936
+v 0.176857 1.055075 -0.211936
+v 0.187132 1.055075 -0.236741
+v 0.211936 1.055075 -0.176857
+v 0.236741 1.055075 -0.187132
+v 0.236741 1.055075 -0.187132
+v 0.211936 1.055075 -0.176857
+v 0.211936 1.055075 -0.247016
+v 0.211936 1.055075 -0.247016
+v 0.187132 1.055075 -0.187132
+v 0.187132 1.055075 -0.187132
+v 0.211936 -0.499729 -0.176857
+v 0.236741 -0.499729 -0.187131
+v 0.187132 -0.499729 -0.187132
+v 0.247016 -0.499729 -0.211936
+v 0.236741 -0.499729 -0.236741
+v 0.176857 -0.499729 -0.211935
+v 0.211937 -0.499729 -0.247015
+v 0.187132 -0.499729 -0.236741
+v 0.314339 1.150000 -0.172918
+v 0.314339 1.100000 -0.172918
+v 0.314339 1.050000 -0.172918
+v 0.314339 1.000000 -0.172918
+v 0.314339 0.950000 -0.172918
+v 0.314339 0.900000 -0.172918
+v 0.314339 0.850000 -0.172918
+v 0.314339 0.800000 -0.172918
+v 0.314339 0.750000 -0.172918
+v 0.172918 1.150000 -0.314339
+v 0.172918 1.100000 -0.314339
+v 0.172918 1.050000 -0.314339
+v 0.172918 1.000000 -0.314339
+v 0.172918 0.950000 -0.314339
+v 0.172918 0.900000 -0.314339
+v 0.172918 0.850000 -0.314339
+v 0.172918 0.800000 -0.314339
+v 0.172918 0.750000 -0.314339
+vt 0.583333 0.750000
+vt 0.583333 0.500000
+vt 0.583333 0.250000
+vt 0.583333 1.000000
+vt 0.125000 0.500000
+vt 0.375000 0.500000
+vt 0.375000 0.750000
+vt 0.125000 0.750000
+vt 0.583333 0.000000
+vt 0.562500 0.250000
+vt 0.541666 0.250000
+vt 0.520833 0.250000
+vt 0.500000 0.250000
+vt 0.479166 0.250000
+vt 0.458333 0.250000
+vt 0.437500 0.250000
+vt 0.416667 0.250000
+vt 0.395833 0.250000
+vt 0.375000 0.250000
+vt 0.375000 0.000000
+vt 0.541666 0.500000
+vt 0.562500 0.500000
+vt 0.395833 0.500000
+vt 0.416667 0.500000
+vt 0.437500 0.500000
+vt 0.458333 0.500000
+vt 0.479166 0.500000
+vt 0.500000 0.500000
+vt 0.520833 0.500000
+vt 0.375000 1.000000
+vt 0.372863 0.578719
+vt 0.372317 0.613951
+vt 0.371666 0.613734
+vt 0.372221 0.578991
+vt 0.315877 0.640082
+vt 0.292344 0.614764
+vt 0.292986 0.614492
+vt 0.316131 0.639438
+vt 0.315346 0.553400
+vt 0.349331 0.553400
+vt 0.349077 0.554044
+vt 0.315664 0.554020
+vt 0.349861 0.640082
+vt 0.349543 0.639462
+vt 0.292889 0.579531
+vt 0.293541 0.579748
+vt 0.444102 0.636706
+vt 0.409596 0.636706
+vt 0.385196 0.612306
+vt 0.385196 0.577799
+vt 0.409596 0.553400
+vt 0.444102 0.553400
+vt 0.468502 0.577800
+vt 0.468502 0.612306
+vt 0.037160 1.987962
+vt 0.036793 0.000397
+vt 0.071486 0.000384
+vt 0.071853 1.987948
+vt 0.002468 1.987974
+vt 0.002100 0.000409
+vt 0.106546 1.987936
+vt 0.106179 0.000371
+vt 0.140872 0.000358
+vt 0.141239 1.987923
+vt 0.245318 1.987885
+vt 0.244951 0.000319
+vt 0.279643 0.000307
+vt 0.280011 1.987871
+vt 0.175932 1.987910
+vt 0.175565 0.000345
+vt 0.210258 0.000332
+vt 0.210625 1.987897
+vt 0.812500 1.000000
+vt 0.000000 1.000000
+vt 0.000000 0.937500
+vt 0.000000 0.875000
+vt 0.812500 0.875000
+vt 0.812500 0.937500
+vt 0.000000 0.812500
+vt 0.812500 0.812500
+vt 0.000000 0.750000
+vt 0.812500 0.750000
+vt 0.000000 0.687500
+vt 0.812500 0.687500
+vt 0.000000 0.625000
+vt 0.812500 0.625000
+vt 0.000000 0.562500
+vt 0.812500 0.562500
+vt 0.000000 0.500000
+vt 0.812500 0.500000
+vt 0.000000 0.437500
+vt 0.812500 0.437500
+vt 0.000000 0.375000
+vt 0.812500 0.375000
+vt 0.812500 1.000000
+vt 0.000000 0.937500
+vt 0.812500 0.937500
+vn 0.6533 0.3827 0.6532
+vn 0.6533 0.3826 -0.6533
+vn -0.6532 0.3827 -0.6533
+vn -0.6532 0.3827 0.6533
+vn 0.5000 -0.7071 0.5000
+vn -0.5000 -0.7071 -0.5000
+vn -0.7071 0.0000 -0.7071
+vn 0.7071 0.0000 0.7071
+vn -0.7071 -0.0000 0.7071
+vn 0.0000 0.0000 1.0000
+vn 0.0000 1.0000 -0.0000
+vn 0.3827 0.0000 0.9239
+vn -0.3827 -0.0000 0.9239
+vn 0.9239 0.0000 -0.3827
+vn 0.9239 0.0000 0.3827
+vn -0.9239 -0.0000 0.3827
+vn -0.3827 -0.0000 -0.9239
+vn -0.9239 -0.0000 -0.3827
+vn 0.3827 -0.0000 -0.9239
+vn -0.0089 0.8351 -0.5501
+vn 0.7071 0.0000 -0.7071
+vn 0.5000 0.7071 -0.5000
+g Cube.001_Cube.001_Default_OBJ
+usemtl Default_OBJ
+s off
+f 1/1/1 2/2/2 3/3/3 4/4/4
+f 8/5/5 7/6/6 6/7/6 5/8/5
+f 4/9/7 3/3/7 42/10/7 43/11/7 44/12/7 45/13/7 46/14/7 47/15/7 48/16/7 49/17/7 50/18/7 8/19/7 5/20/7
+f 34/21/8 33/22/8 2/2/8 1/1/8 6/7/8 7/6/8 41/23/8 40/24/8 39/25/8 38/26/8 37/27/8 36/28/8 35/29/8
+f 1/1/9 4/4/9 5/30/9 6/7/9
+s 1
+f 9/31/10 10/32/10 11/33/10 12/34/10
+f 13/35/10 14/36/10 15/37/10 16/38/10
+f 17/39/10 18/40/10 19/41/10 20/42/10
+f 10/32/10 21/43/10 22/44/10 11/33/10
+f 14/36/10 23/45/10 24/46/10 15/37/10
+f 18/40/10 9/31/10 12/34/10 19/41/10
+f 21/47/11 13/48/11 14/49/11 23/50/11 17/51/11 18/52/11 9/53/11 10/54/11
+f 20/55/12 25/56/12 26/57/12 19/58/12
+f 21/43/10 13/35/10 16/38/10 22/44/10
+f 23/45/10 17/39/10 20/42/10 24/46/10
+f 24/59/13 27/60/13 25/56/13 20/55/13
+f 12/61/14 28/62/14 29/63/14 11/64/14
+f 19/58/15 26/57/15 28/62/15 12/61/15
+f 15/65/16 30/66/16 27/67/16 24/68/16
+f 22/69/17 31/70/17 32/71/17 16/72/17
+f 16/72/18 32/71/18 30/66/18 15/65/18
+f 11/64/19 29/63/19 31/70/19 22/69/19
+g Cube.001_Cube.001_Anzeige
+usemtl Anzeige
+s off
+f 3/73/20 2/74/21 33/75/21
+f 34/76/21 42/77/21 3/78/21 33/75/21
+f 35/79/21 43/80/21 42/77/21 34/76/21
+f 36/81/21 44/82/21 43/80/21 35/79/21
+f 37/83/21 45/84/21 44/82/21 36/81/21
+f 38/85/21 46/86/21 45/84/21 37/83/21
+f 39/87/21 47/88/21 46/86/21 38/85/21
+f 40/89/21 48/90/21 47/88/21 39/87/21
+f 41/91/21 49/92/21 48/90/21 40/89/21
+f 7/93/21 50/94/21 49/92/21 41/91/21
+f 50/95/21 7/96/21 8/97/22
diff --git a/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne3_smr60.obj b/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne3_smr60.obj
new file mode 100644
index 0000000..0a28ff1
--- /dev/null
+++ b/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne3_smr60.obj
@@ -0,0 +1,213 @@
+# Blender v3.0.0 OBJ File: ''
+# www.blender.org
+mtllib advtrains_signals_ks_sign_ne3_smr60.mtl
+o Cube.001
+v 0.335064 1.200000 -0.055317
+v 0.352963 1.200000 -0.064241
+v 0.263724 1.200000 -0.243228
+v 0.245825 1.200000 -0.234304
+v 0.245825 0.700000 -0.234304
+v 0.335064 0.700000 -0.055317
+v 0.352963 0.700000 -0.064241
+v 0.263724 0.700000 -0.243228
+v 0.301499 1.055075 -0.122605
+v 0.299626 1.055075 -0.149388
+v 0.299626 1.055075 -0.149388
+v 0.301499 1.055075 -0.122605
+v 0.252580 1.055075 -0.165129
+v 0.234966 1.055075 -0.144867
+v 0.234966 1.055075 -0.144867
+v 0.252580 1.055075 -0.165129
+v 0.257101 1.055075 -0.100469
+v 0.283885 1.055075 -0.102342
+v 0.283885 1.055075 -0.102342
+v 0.257101 1.055075 -0.100469
+v 0.279363 1.055075 -0.167002
+v 0.279363 1.055075 -0.167002
+v 0.236839 1.055075 -0.118084
+v 0.236839 1.055075 -0.118084
+v 0.257102 -0.499729 -0.100469
+v 0.283885 -0.499729 -0.102341
+v 0.236839 -0.499729 -0.118084
+v 0.301499 -0.499729 -0.122605
+v 0.299626 -0.499729 -0.149388
+v 0.234966 -0.499729 -0.144866
+v 0.279364 -0.499729 -0.167002
+v 0.252580 -0.499729 -0.165129
+v 0.352963 1.150000 -0.064241
+v 0.352963 1.100000 -0.064241
+v 0.352963 1.050000 -0.064241
+v 0.352963 1.000000 -0.064241
+v 0.352963 0.950000 -0.064241
+v 0.352963 0.900000 -0.064241
+v 0.352963 0.850000 -0.064241
+v 0.352963 0.800000 -0.064241
+v 0.352963 0.750000 -0.064241
+v 0.263724 1.150000 -0.243228
+v 0.263724 1.100000 -0.243228
+v 0.263724 1.050000 -0.243228
+v 0.263724 1.000000 -0.243228
+v 0.263724 0.950000 -0.243228
+v 0.263724 0.900000 -0.243228
+v 0.263724 0.850000 -0.243228
+v 0.263724 0.800000 -0.243228
+v 0.263724 0.750000 -0.243228
+vt 0.583333 0.750000
+vt 0.583333 0.500000
+vt 0.583333 0.250000
+vt 0.583333 1.000000
+vt 0.125000 0.500000
+vt 0.375000 0.500000
+vt 0.375000 0.750000
+vt 0.125000 0.750000
+vt 0.583333 0.000000
+vt 0.562500 0.250000
+vt 0.541666 0.250000
+vt 0.520833 0.250000
+vt 0.500000 0.250000
+vt 0.479166 0.250000
+vt 0.458333 0.250000
+vt 0.437500 0.250000
+vt 0.416667 0.250000
+vt 0.395833 0.250000
+vt 0.375000 0.250000
+vt 0.375000 0.000000
+vt 0.541666 0.500000
+vt 0.562500 0.500000
+vt 0.395833 0.500000
+vt 0.416667 0.500000
+vt 0.437500 0.500000
+vt 0.458333 0.500000
+vt 0.479166 0.500000
+vt 0.500000 0.500000
+vt 0.520833 0.500000
+vt 0.375000 1.000000
+vt 0.372863 0.578719
+vt 0.372317 0.613951
+vt 0.371666 0.613734
+vt 0.372221 0.578991
+vt 0.315877 0.640082
+vt 0.292344 0.614764
+vt 0.292986 0.614492
+vt 0.316131 0.639438
+vt 0.315346 0.553400
+vt 0.349331 0.553400
+vt 0.349077 0.554044
+vt 0.315664 0.554020
+vt 0.349861 0.640082
+vt 0.349543 0.639462
+vt 0.292889 0.579531
+vt 0.293541 0.579748
+vt 0.444102 0.636706
+vt 0.409596 0.636706
+vt 0.385196 0.612306
+vt 0.385196 0.577799
+vt 0.409596 0.553400
+vt 0.444102 0.553400
+vt 0.468502 0.577800
+vt 0.468502 0.612306
+vt 0.037160 1.987962
+vt 0.036793 0.000397
+vt 0.071486 0.000384
+vt 0.071853 1.987948
+vt 0.002468 1.987974
+vt 0.002100 0.000409
+vt 0.106546 1.987936
+vt 0.106179 0.000371
+vt 0.140872 0.000358
+vt 0.141239 1.987923
+vt 0.245318 1.987885
+vt 0.244951 0.000319
+vt 0.279643 0.000307
+vt 0.280011 1.987871
+vt 0.175932 1.987910
+vt 0.175565 0.000345
+vt 0.210258 0.000332
+vt 0.210625 1.987897
+vt 0.812500 1.000000
+vt 0.000000 1.000000
+vt 0.000000 0.937500
+vt 0.000000 0.875000
+vt 0.812500 0.875000
+vt 0.812500 0.937500
+vt 0.000000 0.812500
+vt 0.812500 0.812500
+vt 0.000000 0.750000
+vt 0.812500 0.750000
+vt 0.000000 0.687500
+vt 0.812500 0.687500
+vt 0.000000 0.625000
+vt 0.812500 0.625000
+vt 0.000000 0.562500
+vt 0.812500 0.562500
+vt 0.000000 0.500000
+vt 0.812500 0.500000
+vt 0.000000 0.437500
+vt 0.812500 0.437500
+vt 0.000000 0.375000
+vt 0.812500 0.375000
+vt 0.812500 1.000000
+vt 0.000000 0.937500
+vt 0.812500 0.937500
+vn 0.4123 0.3827 0.8268
+vn 0.8268 0.3826 -0.4123
+vn -0.4122 0.3827 -0.8268
+vn -0.8268 0.3827 0.4123
+vn 0.3155 -0.7071 0.6328
+vn -0.3155 -0.7071 -0.6328
+vn -0.4462 0.0000 -0.8949
+vn 0.4462 0.0000 0.8949
+vn -0.8949 -0.0000 0.4462
+vn 0.0000 0.0000 1.0000
+vn 0.0000 1.0000 0.0000
+vn 0.0698 0.0000 0.9976
+vn -0.6561 -0.0000 0.7547
+vn 0.9976 0.0000 -0.0698
+vn 0.7547 0.0000 0.6561
+vn -0.9976 -0.0000 0.0698
+vn -0.0698 -0.0000 -0.9976
+vn -0.7547 -0.0000 -0.6561
+vn 0.6561 -0.0000 -0.7547
+vn 0.1661 0.8351 -0.5245
+vn 0.8949 0.0000 -0.4462
+vn 0.6328 0.7071 -0.3155
+g Cube.001_Cube.001_Default_OBJ
+usemtl Default_OBJ
+s off
+f 1/1/1 2/2/2 3/3/3 4/4/4
+f 8/5/5 7/6/6 6/7/6 5/8/5
+f 4/9/7 3/3/7 42/10/7 43/11/7 44/12/7 45/13/7 46/14/7 47/15/7 48/16/7 49/17/7 50/18/7 8/19/7 5/20/7
+f 34/21/8 33/22/8 2/2/8 1/1/8 6/7/8 7/6/8 41/23/8 40/24/8 39/25/8 38/26/8 37/27/8 36/28/8 35/29/8
+f 1/1/9 4/4/9 5/30/9 6/7/9
+s 1
+f 9/31/10 10/32/10 11/33/10 12/34/10
+f 13/35/10 14/36/10 15/37/10 16/38/10
+f 17/39/10 18/40/10 19/41/10 20/42/10
+f 10/32/10 21/43/10 22/44/10 11/33/10
+f 14/36/10 23/45/10 24/46/10 15/37/10
+f 18/40/10 9/31/10 12/34/10 19/41/10
+f 21/47/11 13/48/11 14/49/11 23/50/11 17/51/11 18/52/11 9/53/11 10/54/11
+f 20/55/12 25/56/12 26/57/12 19/58/12
+f 21/43/10 13/35/10 16/38/10 22/44/10
+f 23/45/10 17/39/10 20/42/10 24/46/10
+f 24/59/13 27/60/13 25/56/13 20/55/13
+f 12/61/14 28/62/14 29/63/14 11/64/14
+f 19/58/15 26/57/15 28/62/15 12/61/15
+f 15/65/16 30/66/16 27/67/16 24/68/16
+f 22/69/17 31/70/17 32/71/17 16/72/17
+f 16/72/18 32/71/18 30/66/18 15/65/18
+f 11/64/19 29/63/19 31/70/19 22/69/19
+g Cube.001_Cube.001_Anzeige
+usemtl Anzeige
+s off
+f 3/73/20 2/74/21 33/75/21
+f 34/76/21 42/77/21 3/78/21 33/75/21
+f 35/79/21 43/80/21 42/77/21 34/76/21
+f 36/81/21 44/82/21 43/80/21 35/79/21
+f 37/83/21 45/84/21 44/82/21 36/81/21
+f 38/85/21 46/86/21 45/84/21 37/83/21
+f 39/87/21 47/88/21 46/86/21 38/85/21
+f 40/89/21 48/90/21 47/88/21 39/87/21
+f 41/91/21 49/92/21 48/90/21 40/89/21
+f 7/93/21 50/94/21 49/92/21 41/91/21
+f 50/95/21 7/96/21 8/97/22
diff --git a/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne4_smr0.obj b/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne4_smr0.obj
new file mode 100644
index 0000000..3c4fa17
--- /dev/null
+++ b/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne4_smr0.obj
@@ -0,0 +1,148 @@
+# Blender v3.0.0 OBJ File: ''
+# www.blender.org
+mtllib advtrains_signals_ks_sign_ne4_smr0.mtl
+o Cube.001
+v 0.100000 1.200000 -0.324543
+v 0.100000 1.200000 -0.344543
+v -0.100000 1.200000 -0.344543
+v -0.100000 1.200000 -0.324543
+v 0.100000 0.700000 -0.344543
+v 0.100000 0.700000 -0.324543
+v -0.100000 0.700000 -0.344543
+v -0.100000 0.700000 -0.324543
+v 0.024805 1.055075 -0.324528
+v 0.000000 1.055075 -0.334802
+v 0.000000 1.055075 -0.334802
+v 0.024805 1.055075 -0.324528
+v -0.035079 1.055075 -0.299723
+v -0.024805 1.055075 -0.274918
+v -0.024805 1.055075 -0.274918
+v -0.035079 1.055075 -0.299723
+v 0.024805 1.055075 -0.274918
+v 0.035079 1.055075 -0.299723
+v 0.035079 1.055075 -0.299723
+v 0.024805 1.055075 -0.274918
+v -0.024805 1.055075 -0.324528
+v -0.024805 1.055075 -0.324528
+v 0.000000 1.055075 -0.264644
+v 0.000000 1.055075 -0.264644
+v 0.024805 -0.499729 -0.274918
+v 0.035080 -0.499729 -0.299723
+v 0.000000 -0.499729 -0.264644
+v 0.024805 -0.499729 -0.324528
+v 0.000000 -0.499729 -0.334802
+v -0.024804 -0.499729 -0.274918
+v -0.024804 -0.499729 -0.324528
+v -0.035079 -0.499729 -0.299723
+vt 0.583333 0.750000
+vt 0.583333 0.500000
+vt 0.583333 0.250000
+vt 0.583333 1.000000
+vt 0.375000 1.000000
+vt 0.375000 0.750000
+vt 0.375000 0.500000
+vt 0.375000 0.000000
+vt 0.583333 0.000000
+vt 0.375000 0.250000
+vt 0.125000 0.500000
+vt 0.125000 0.750000
+vt 0.372863 0.578719
+vt 0.372317 0.613951
+vt 0.371666 0.613734
+vt 0.372221 0.578991
+vt 0.315877 0.640082
+vt 0.292344 0.614764
+vt 0.292986 0.614492
+vt 0.316131 0.639438
+vt 0.315346 0.553400
+vt 0.349331 0.553400
+vt 0.349077 0.554044
+vt 0.315664 0.554020
+vt 0.349861 0.640082
+vt 0.349543 0.639462
+vt 0.292889 0.579531
+vt 0.293541 0.579748
+vt 0.444102 0.636706
+vt 0.409596 0.636706
+vt 0.385196 0.612306
+vt 0.385196 0.577799
+vt 0.409596 0.553400
+vt 0.444102 0.553400
+vt 0.468502 0.577800
+vt 0.468502 0.612306
+vt 0.037160 1.987962
+vt 0.036793 0.000397
+vt 0.071486 0.000384
+vt 0.071853 1.987948
+vt 0.002468 1.987974
+vt 0.002100 0.000409
+vt 0.106546 1.987936
+vt 0.106179 0.000371
+vt 0.140872 0.000358
+vt 0.141239 1.987923
+vt 0.245318 1.987885
+vt 0.244951 0.000319
+vt 0.279643 0.000307
+vt 0.280011 1.987871
+vt 0.175932 1.987910
+vt 0.175565 0.000345
+vt 0.210258 0.000332
+vt 0.210625 1.987897
+vt 0.250000 0.375000
+vt 0.250000 1.000000
+vt 0.000000 1.000000
+vt 0.000000 0.375000
+vn 0.9239 0.3827 -0.0000
+vn -0.0000 0.3826 -0.9239
+vn -0.9239 0.3827 -0.0001
+vn 0.0000 0.3827 0.9239
+vn 0.0000 -0.0000 1.0000
+vn 1.0000 -0.0000 0.0000
+vn -1.0000 0.0000 0.0000
+vn 0.7071 -0.7071 0.0000
+vn -0.7071 -0.7071 -0.0000
+vn -0.9659 0.0000 0.2588
+vn 0.9659 0.0000 0.2588
+vn -0.7814 -0.5678 0.2588
+vn 0.9659 0.0000 -0.2588
+vn -0.9659 0.0000 -0.2588
+vn 0.0000 1.0000 0.0000
+vn 0.9239 0.0000 0.3827
+vn 0.3827 -0.0000 0.9239
+vn 0.3827 0.0000 -0.9239
+vn 0.9239 0.0000 -0.3827
+vn -0.3827 -0.0000 0.9239
+vn -0.9239 -0.0000 -0.3827
+vn -0.9239 -0.0000 0.3827
+vn -0.3827 -0.0000 -0.9239
+vn 0.0000 0.0000 -1.0000
+g Cube.001_Cube.001_Default_OBJ
+usemtl Default_OBJ
+s off
+f 1/1/1 2/2/2 3/3/3 4/4/4
+f 1/1/5 4/4/5 8/5/5 6/6/5
+f 5/7/6 2/2/6 1/1/6 6/6/6
+f 8/8/7 4/9/7 3/3/7 7/10/7
+f 7/11/8 5/7/9 6/6/9 8/12/8
+s 1
+f 9/13/10 10/14/5 11/15/11 12/16/12
+f 13/17/6 14/18/13 15/19/14 16/20/6
+f 17/21/13 18/22/7 19/23/7 20/24/14
+f 10/14/5 21/25/10 22/26/11 11/15/11
+f 14/18/13 23/27/5 24/28/14 15/19/14
+f 18/22/7 9/13/10 12/16/12 19/23/7
+f 21/29/15 13/30/15 14/31/15 23/32/15 17/33/15 18/34/15 9/35/15 10/36/15
+f 20/37/16 25/38/16 26/39/16 19/40/16
+f 21/25/10 13/17/6 16/20/6 22/26/11
+f 23/27/5 17/21/13 20/24/14 24/28/14
+f 24/41/17 27/42/17 25/38/17 20/37/17
+f 12/43/18 28/44/18 29/45/18 11/46/18
+f 19/40/19 26/39/19 28/44/19 12/43/19
+f 15/47/20 30/48/20 27/49/20 24/50/20
+f 22/51/21 31/52/21 32/53/21 16/54/21
+f 16/54/22 32/53/22 30/48/22 15/47/22
+f 11/46/23 29/45/23 31/52/23 22/51/23
+g Cube.001_Cube.001_Anzeige
+usemtl Anzeige
+s off
+f 7/55/24 3/56/24 2/57/24 5/58/24
diff --git a/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne4_smr30.obj b/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne4_smr30.obj
new file mode 100644
index 0000000..d6f8603
--- /dev/null
+++ b/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne4_smr30.obj
@@ -0,0 +1,151 @@
+# Blender v3.0.0 OBJ File: ''
+# www.blender.org
+mtllib advtrains_signals_ks_sign_ne4_smr30.mtl
+o Cube.001
+v 0.234304 1.200000 -0.245825
+v 0.243228 1.200000 -0.263724
+v 0.064241 1.200000 -0.352963
+v 0.055317 1.200000 -0.335064
+v 0.243228 0.700000 -0.263724
+v 0.234304 0.700000 -0.245825
+v 0.064241 0.700000 -0.352963
+v 0.055317 0.700000 -0.335064
+v 0.167003 1.055075 -0.279363
+v 0.149388 1.055075 -0.299626
+v 0.149388 1.055075 -0.299626
+v 0.167003 1.055075 -0.279363
+v 0.102342 1.055075 -0.283885
+v 0.100469 1.055075 -0.257101
+v 0.100469 1.055075 -0.257101
+v 0.102342 1.055075 -0.283885
+v 0.144867 1.055075 -0.234966
+v 0.165129 1.055075 -0.252580
+v 0.165129 1.055075 -0.252580
+v 0.144867 1.055075 -0.234966
+v 0.122605 1.055075 -0.301499
+v 0.122605 1.055075 -0.301499
+v 0.118084 1.055075 -0.236839
+v 0.118084 1.055075 -0.236839
+v 0.144867 -0.499729 -0.234966
+v 0.165130 -0.499729 -0.252580
+v 0.118084 -0.499729 -0.236839
+v 0.167003 -0.499729 -0.279363
+v 0.149388 -0.499729 -0.299626
+v 0.100470 -0.499729 -0.257101
+v 0.122606 -0.499729 -0.301499
+v 0.102342 -0.499729 -0.283885
+vt 0.583333 0.750000
+vt 0.583333 0.500000
+vt 0.583333 0.250000
+vt 0.583333 1.000000
+vt 0.375000 1.000000
+vt 0.375000 0.750000
+vt 0.375000 0.500000
+vt 0.375000 0.000000
+vt 0.583333 0.000000
+vt 0.375000 0.250000
+vt 0.125000 0.500000
+vt 0.125000 0.750000
+vt 0.372863 0.578719
+vt 0.372317 0.613951
+vt 0.371666 0.613734
+vt 0.372221 0.578991
+vt 0.315877 0.640082
+vt 0.292344 0.614764
+vt 0.292986 0.614492
+vt 0.316131 0.639438
+vt 0.315346 0.553400
+vt 0.349331 0.553400
+vt 0.349077 0.554044
+vt 0.315664 0.554020
+vt 0.349861 0.640082
+vt 0.349543 0.639462
+vt 0.292889 0.579531
+vt 0.293541 0.579748
+vt 0.444102 0.636706
+vt 0.409596 0.636706
+vt 0.385196 0.612306
+vt 0.385196 0.577799
+vt 0.409596 0.553400
+vt 0.444102 0.553400
+vt 0.468502 0.577800
+vt 0.468502 0.612306
+vt 0.037160 1.987962
+vt 0.036793 0.000397
+vt 0.071486 0.000384
+vt 0.071853 1.987948
+vt 0.002468 1.987974
+vt 0.002100 0.000409
+vt 0.106546 1.987936
+vt 0.106179 0.000371
+vt 0.140872 0.000358
+vt 0.141239 1.987923
+vt 0.245318 1.987885
+vt 0.244951 0.000319
+vt 0.279643 0.000307
+vt 0.280011 1.987871
+vt 0.175932 1.987910
+vt 0.175565 0.000345
+vt 0.210258 0.000332
+vt 0.210625 1.987897
+vt 0.250000 0.375000
+vt 0.250000 1.000000
+vt 0.000000 1.000000
+vt 0.000000 0.375000
+vn 0.8268 0.3827 0.4122
+vn 0.4122 0.3826 -0.8269
+vn -0.8268 0.3827 -0.4123
+vn -0.4122 0.3826 0.8268
+vn -0.4462 -0.0000 0.8949
+vn 0.8949 -0.0000 0.4462
+vn -0.8949 -0.0000 -0.4462
+vn 0.6328 -0.7071 0.3155
+vn -0.6328 -0.7071 -0.3155
+vn -0.9769 0.0000 0.2136
+vn 0.0000 0.0000 1.0000
+vn 0.9659 0.0000 0.2588
+vn -0.9659 0.0000 0.2588
+vn 0.9769 0.0000 -0.2136
+vn -0.7815 -0.5677 0.2588
+vn 0.9659 0.0000 -0.2588
+vn -0.9659 0.0000 -0.2588
+vn 0.0000 1.0000 -0.0000
+vn 0.6561 0.0000 0.7547
+vn -0.0698 -0.0000 0.9976
+vn 0.7547 0.0000 -0.6561
+vn 0.9976 0.0000 0.0698
+vn -0.7547 -0.0000 0.6561
+vn -0.6561 -0.0000 -0.7547
+vn -0.9976 -0.0000 -0.0698
+vn 0.0698 -0.0000 -0.9976
+vn 0.4462 0.0000 -0.8949
+g Cube.001_Cube.001_Default_OBJ
+usemtl Default_OBJ
+s off
+f 1/1/1 2/2/2 3/3/3 4/4/4
+f 1/1/5 4/4/5 8/5/5 6/6/5
+f 5/7/6 2/2/6 1/1/6 6/6/6
+f 8/8/7 4/9/7 3/3/7 7/10/7
+f 7/11/8 5/7/9 6/6/9 8/12/8
+s 1
+f 9/13/10 10/14/11 11/15/12 12/16/10
+f 13/17/13 14/18/14 15/19/14 16/20/15
+f 17/21/16 18/22/16 19/23/17 20/24/17
+f 10/14/11 21/25/13 22/26/12 11/15/12
+f 14/18/14 23/27/11 24/28/17 15/19/14
+f 18/22/16 9/13/10 12/16/10 19/23/17
+f 21/29/18 13/30/18 14/31/18 23/32/18 17/33/18 18/34/18 9/35/18 10/36/18
+f 20/37/19 25/38/19 26/39/19 19/40/19
+f 21/25/13 13/17/13 16/20/15 22/26/12
+f 23/27/11 17/21/16 20/24/17 24/28/17
+f 24/41/20 27/42/20 25/38/20 20/37/20
+f 12/43/21 28/44/21 29/45/21 11/46/21
+f 19/40/22 26/39/22 28/44/22 12/43/22
+f 15/47/23 30/48/23 27/49/23 24/50/23
+f 22/51/24 31/52/24 32/53/24 16/54/24
+f 16/54/25 32/53/25 30/48/25 15/47/25
+f 11/46/26 29/45/26 31/52/26 22/51/26
+g Cube.001_Cube.001_Anzeige
+usemtl Anzeige
+s off
+f 7/55/27 3/56/27 2/57/27 5/58/27
diff --git a/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne4_smr45.obj b/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne4_smr45.obj
new file mode 100644
index 0000000..b2c828e
--- /dev/null
+++ b/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne4_smr45.obj
@@ -0,0 +1,151 @@
+# Blender v3.0.0 OBJ File: ''
+# www.blender.org
+mtllib advtrains_signals_ks_sign_ne4_smr45.mtl
+o Cube.001
+v 0.300197 1.200000 -0.158776
+v 0.314339 1.200000 -0.172918
+v 0.172918 1.200000 -0.314339
+v 0.158776 1.200000 -0.300197
+v 0.314339 0.700000 -0.172918
+v 0.300197 0.700000 -0.158776
+v 0.172918 0.700000 -0.314339
+v 0.158776 0.700000 -0.300197
+v 0.247016 1.055075 -0.211936
+v 0.236741 1.055075 -0.236741
+v 0.236741 1.055075 -0.236741
+v 0.247016 1.055075 -0.211936
+v 0.187132 1.055075 -0.236741
+v 0.176857 1.055075 -0.211936
+v 0.176857 1.055075 -0.211936
+v 0.187132 1.055075 -0.236741
+v 0.211936 1.055075 -0.176857
+v 0.236741 1.055075 -0.187132
+v 0.236741 1.055075 -0.187132
+v 0.211936 1.055075 -0.176857
+v 0.211936 1.055075 -0.247016
+v 0.211936 1.055075 -0.247016
+v 0.187132 1.055075 -0.187132
+v 0.187132 1.055075 -0.187132
+v 0.211936 -0.499729 -0.176857
+v 0.236741 -0.499729 -0.187131
+v 0.187132 -0.499729 -0.187132
+v 0.247016 -0.499729 -0.211936
+v 0.236741 -0.499729 -0.236741
+v 0.176857 -0.499729 -0.211935
+v 0.211937 -0.499729 -0.247015
+v 0.187132 -0.499729 -0.236741
+vt 0.583333 0.750000
+vt 0.583333 0.500000
+vt 0.583333 0.250000
+vt 0.583333 1.000000
+vt 0.375000 1.000000
+vt 0.375000 0.750000
+vt 0.375000 0.500000
+vt 0.375000 0.000000
+vt 0.583333 0.000000
+vt 0.375000 0.250000
+vt 0.125000 0.500000
+vt 0.125000 0.750000
+vt 0.372863 0.578719
+vt 0.372317 0.613951
+vt 0.371666 0.613734
+vt 0.372221 0.578991
+vt 0.315877 0.640082
+vt 0.292344 0.614764
+vt 0.292986 0.614492
+vt 0.316131 0.639438
+vt 0.315346 0.553400
+vt 0.349331 0.553400
+vt 0.349077 0.554044
+vt 0.315664 0.554020
+vt 0.349861 0.640082
+vt 0.349543 0.639462
+vt 0.292889 0.579531
+vt 0.293541 0.579748
+vt 0.444102 0.636706
+vt 0.409596 0.636706
+vt 0.385196 0.612306
+vt 0.385196 0.577799
+vt 0.409596 0.553400
+vt 0.444102 0.553400
+vt 0.468502 0.577800
+vt 0.468502 0.612306
+vt 0.037160 1.987962
+vt 0.036793 0.000397
+vt 0.071486 0.000384
+vt 0.071853 1.987948
+vt 0.002468 1.987974
+vt 0.002100 0.000409
+vt 0.106546 1.987936
+vt 0.106179 0.000371
+vt 0.140872 0.000358
+vt 0.141239 1.987923
+vt 0.245318 1.987885
+vt 0.244951 0.000319
+vt 0.279643 0.000307
+vt 0.280011 1.987871
+vt 0.175932 1.987910
+vt 0.175565 0.000345
+vt 0.210258 0.000332
+vt 0.210625 1.987897
+vt 0.250000 0.375000
+vt 0.250000 1.000000
+vt 0.000000 1.000000
+vt 0.000000 0.375000
+vn 0.6533 0.3827 0.6532
+vn 0.6533 0.3826 -0.6533
+vn -0.6532 0.3826 -0.6534
+vn -0.6533 0.3827 0.6533
+vn -0.7071 -0.0000 0.7071
+vn 0.7071 -0.0000 0.7071
+vn -0.7071 0.0000 -0.7071
+vn 0.5000 -0.7071 0.5000
+vn -0.5000 -0.7071 -0.5000
+vn -1.0000 0.0000 -0.0000
+vn 0.0000 0.0000 1.0000
+vn 0.9659 0.0000 0.2588
+vn -0.9659 0.0000 0.2588
+vn 1.0000 0.0000 -0.0000
+vn -0.7815 -0.5677 0.2588
+vn 0.9659 0.0000 -0.2588
+vn -0.9659 0.0000 -0.2588
+vn 0.0000 1.0000 -0.0000
+vn 0.3827 0.0000 0.9239
+vn -0.3827 -0.0000 0.9239
+vn 0.9239 0.0000 -0.3827
+vn 0.9239 0.0000 0.3827
+vn -0.9239 -0.0000 0.3827
+vn -0.3827 -0.0000 -0.9239
+vn -0.9239 -0.0000 -0.3827
+vn 0.3827 -0.0000 -0.9239
+vn 0.7071 0.0000 -0.7071
+g Cube.001_Cube.001_Default_OBJ
+usemtl Default_OBJ
+s off
+f 1/1/1 2/2/2 3/3/3 4/4/4
+f 1/1/5 4/4/5 8/5/5 6/6/5
+f 5/7/6 2/2/6 1/1/6 6/6/6
+f 8/8/7 4/9/7 3/3/7 7/10/7
+f 7/11/8 5/7/9 6/6/9 8/12/8
+s 1
+f 9/13/10 10/14/11 11/15/12 12/16/10
+f 13/17/13 14/18/14 15/19/14 16/20/15
+f 17/21/16 18/22/16 19/23/17 20/24/17
+f 10/14/11 21/25/13 22/26/12 11/15/12
+f 14/18/14 23/27/11 24/28/17 15/19/14
+f 18/22/16 9/13/10 12/16/10 19/23/17
+f 21/29/18 13/30/18 14/31/18 23/32/18 17/33/18 18/34/18 9/35/18 10/36/18
+f 20/37/19 25/38/19 26/39/19 19/40/19
+f 21/25/13 13/17/13 16/20/15 22/26/12
+f 23/27/11 17/21/16 20/24/17 24/28/17
+f 24/41/20 27/42/20 25/38/20 20/37/20
+f 12/43/21 28/44/21 29/45/21 11/46/21
+f 19/40/22 26/39/22 28/44/22 12/43/22
+f 15/47/23 30/48/23 27/49/23 24/50/23
+f 22/51/24 31/52/24 32/53/24 16/54/24
+f 16/54/25 32/53/25 30/48/25 15/47/25
+f 11/46/26 29/45/26 31/52/26 22/51/26
+g Cube.001_Cube.001_Anzeige
+usemtl Anzeige
+s off
+f 7/55/27 3/56/27 2/57/27 5/58/27
diff --git a/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne4_smr60.obj b/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne4_smr60.obj
new file mode 100644
index 0000000..c49488b
--- /dev/null
+++ b/advtrains_signals_ks/models/advtrains_signals_ks_sign_ne4_smr60.obj
@@ -0,0 +1,151 @@
+# Blender v3.0.0 OBJ File: ''
+# www.blender.org
+mtllib advtrains_signals_ks_sign_ne4_smr60.mtl
+o Cube.001
+v 0.335064 1.200000 -0.055317
+v 0.352963 1.200000 -0.064241
+v 0.263724 1.200000 -0.243228
+v 0.245825 1.200000 -0.234304
+v 0.352963 0.700000 -0.064241
+v 0.335064 0.700000 -0.055317
+v 0.263724 0.700000 -0.243228
+v 0.245825 0.700000 -0.234304
+v 0.301499 1.055075 -0.122605
+v 0.299626 1.055075 -0.149388
+v 0.299626 1.055075 -0.149388
+v 0.301499 1.055075 -0.122605
+v 0.252580 1.055075 -0.165129
+v 0.234966 1.055075 -0.144867
+v 0.234966 1.055075 -0.144867
+v 0.252580 1.055075 -0.165129
+v 0.257101 1.055075 -0.100469
+v 0.283885 1.055075 -0.102342
+v 0.283885 1.055075 -0.102342
+v 0.257101 1.055075 -0.100469
+v 0.279363 1.055075 -0.167002
+v 0.279363 1.055075 -0.167002
+v 0.236839 1.055075 -0.118084
+v 0.236839 1.055075 -0.118084
+v 0.257102 -0.499729 -0.100469
+v 0.283885 -0.499729 -0.102341
+v 0.236839 -0.499729 -0.118084
+v 0.301499 -0.499729 -0.122605
+v 0.299626 -0.499729 -0.149388
+v 0.234966 -0.499729 -0.144866
+v 0.279364 -0.499729 -0.167002
+v 0.252580 -0.499729 -0.165129
+vt 0.583333 0.750000
+vt 0.583333 0.500000
+vt 0.583333 0.250000
+vt 0.583333 1.000000
+vt 0.375000 1.000000
+vt 0.375000 0.750000
+vt 0.375000 0.500000
+vt 0.375000 0.000000
+vt 0.583333 0.000000
+vt 0.375000 0.250000
+vt 0.125000 0.500000
+vt 0.125000 0.750000
+vt 0.372863 0.578719
+vt 0.372317 0.613951
+vt 0.371666 0.613734
+vt 0.372221 0.578991
+vt 0.315877 0.640082
+vt 0.292344 0.614764
+vt 0.292986 0.614492
+vt 0.316131 0.639438
+vt 0.315346 0.553400
+vt 0.349331 0.553400
+vt 0.349077 0.554044
+vt 0.315664 0.554020
+vt 0.349861 0.640082
+vt 0.349543 0.639462
+vt 0.292889 0.579531
+vt 0.293541 0.579748
+vt 0.444102 0.636706
+vt 0.409596 0.636706
+vt 0.385196 0.612306
+vt 0.385196 0.577799
+vt 0.409596 0.553400
+vt 0.444102 0.553400
+vt 0.468502 0.577800
+vt 0.468502 0.612306
+vt 0.037160 1.987962
+vt 0.036793 0.000397
+vt 0.071486 0.000384
+vt 0.071853 1.987948
+vt 0.002468 1.987974
+vt 0.002100 0.000409
+vt 0.106546 1.987936
+vt 0.106179 0.000371
+vt 0.140872 0.000358
+vt 0.141239 1.987923
+vt 0.245318 1.987885
+vt 0.244951 0.000319
+vt 0.279643 0.000307
+vt 0.280011 1.987871
+vt 0.175932 1.987910
+vt 0.175565 0.000345
+vt 0.210258 0.000332
+vt 0.210625 1.987897
+vt 0.250000 0.375000
+vt 0.250000 1.000000
+vt 0.000000 1.000000
+vt 0.000000 0.375000
+vn 0.4123 0.3827 0.8268
+vn 0.8268 0.3826 -0.4123
+vn -0.4122 0.3827 -0.8269
+vn -0.8268 0.3827 0.4123
+vn -0.8949 -0.0000 0.4462
+vn 0.4462 0.0000 0.8949
+vn -0.4462 0.0000 -0.8949
+vn 0.3155 -0.7071 0.6328
+vn -0.3155 -0.7071 -0.6328
+vn -0.9769 0.0000 -0.2136
+vn 0.0000 0.0000 1.0000
+vn 0.9659 0.0000 0.2588
+vn -0.9659 0.0000 0.2588
+vn 0.9769 0.0000 0.2136
+vn -0.7817 -0.5674 0.2588
+vn 0.9659 0.0000 -0.2588
+vn -0.9659 0.0000 -0.2588
+vn 0.0000 1.0000 0.0000
+vn 0.0698 0.0000 0.9976
+vn -0.6561 -0.0000 0.7547
+vn 0.9976 0.0000 -0.0698
+vn 0.7547 0.0000 0.6561
+vn -0.9976 -0.0000 0.0698
+vn -0.0698 -0.0000 -0.9976
+vn -0.7547 -0.0000 -0.6561
+vn 0.6561 -0.0000 -0.7547
+vn 0.8949 0.0000 -0.4462
+g Cube.001_Cube.001_Default_OBJ
+usemtl Default_OBJ
+s off
+f 1/1/1 2/2/2 3/3/3 4/4/4
+f 1/1/5 4/4/5 8/5/5 6/6/5
+f 5/7/6 2/2/6 1/1/6 6/6/6
+f 8/8/7 4/9/7 3/3/7 7/10/7
+f 7/11/8 5/7/9 6/6/9 8/12/8
+s 1
+f 9/13/10 10/14/11 11/15/12 12/16/10
+f 13/17/13 14/18/14 15/19/14 16/20/15
+f 17/21/16 18/22/16 19/23/17 20/24/17
+f 10/14/11 21/25/13 22/26/12 11/15/12
+f 14/18/14 23/27/11 24/28/17 15/19/14
+f 18/22/16 9/13/10 12/16/10 19/23/17
+f 21/29/18 13/30/18 14/31/18 23/32/18 17/33/18 18/34/18 9/35/18 10/36/18
+f 20/37/19 25/38/19 26/39/19 19/40/19
+f 21/25/13 13/17/13 16/20/15 22/26/12
+f 23/27/11 17/21/16 20/24/17 24/28/17
+f 24/41/20 27/42/20 25/38/20 20/37/20
+f 12/43/21 28/44/21 29/45/21 11/46/21
+f 19/40/22 26/39/22 28/44/22 12/43/22
+f 15/47/23 30/48/23 27/49/23 24/50/23
+f 22/51/24 31/52/24 32/53/24 16/54/24
+f 16/54/25 32/53/25 30/48/25 15/47/25
+f 11/46/26 29/45/26 31/52/26 22/51/26
+g Cube.001_Cube.001_Anzeige
+usemtl Anzeige
+s off
+f 7/55/27 3/56/27 2/57/27 5/58/27
diff --git a/advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x1.png b/advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x1.png
new file mode 100644
index 0000000..19f860a
--- /dev/null
+++ b/advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x1.png
Binary files differ
diff --git a/advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x2.png b/advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x2.png
new file mode 100644
index 0000000..dd86d09
--- /dev/null
+++ b/advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x2.png
Binary files differ
diff --git a/advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x3.png b/advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x3.png
new file mode 100644
index 0000000..f343fc4
--- /dev/null
+++ b/advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x3.png
Binary files differ
diff --git a/advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x4.png b/advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x4.png
new file mode 100644
index 0000000..98c0248
--- /dev/null
+++ b/advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x4.png
Binary files differ
diff --git a/advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x5.png b/advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x5.png
new file mode 100644
index 0000000..d9d21bd
--- /dev/null
+++ b/advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne3x5.png
Binary files differ
diff --git a/advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne4.png b/advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne4.png
new file mode 100644
index 0000000..4efbe57
--- /dev/null
+++ b/advtrains_signals_ks/textures/advtrains_signals_ks_sign_ne4.png
Binary files differ
diff --git a/advtrains_signals_ks/textures/advtrains_signals_ks_vs_inv.png b/advtrains_signals_ks/textures/advtrains_signals_ks_vs_inv.png
new file mode 100644
index 0000000..1e4acfd
--- /dev/null
+++ b/advtrains_signals_ks/textures/advtrains_signals_ks_vs_inv.png
Binary files differ
diff --git a/advtrains_signals_muc_ubahn/init.lua b/advtrains_signals_muc_ubahn/init.lua
new file mode 100755
index 0000000..182a3dc
--- /dev/null
+++ b/advtrains_signals_muc_ubahn/init.lua
@@ -0,0 +1,88 @@
+-- advtrains_signals_muc_ubahn
+-- Signals modeled after the Munich U-Bahn signalling system
+-- It reuses the original historic wall signal mesh, but extends it to 4 signal lamps and supports the new signal API only.
+
+-- For a reference of the signals (in German) see: https://www.u-bahn-muenchen.de/betrieb/zugsicherung/signale/
+-- Hp4 and Hp5 are not implemented because they do not make sense.
+-- Also the speed signals are not yet added (they will be added later)
+
+local all_sigs = {
+ hp0 = { asp = { main = 0 }, crea = true }, -- halt
+ hp1 = { asp = { main = -1, proceed_as_main = true } }, -- free full speed
+ hp2 = { asp = { main = 12, proceed_as_main = true } }, -- slow speed
+ hp3 = { asp = { main = 0, shunt = true } }, -- shunting
+ vr0 = { asp = { dst = 0 }, distant = true, crea = true }, -- distant halt/slow
+ vr1 = { asp = { dst = -1 }, distant = true }, -- distant free
+}
+
+local mainaspects = {
+ { name = "hp1", description = "Hp1: Full speed" },
+ { name = "hp2", description = "Hp2: Reduced Speed" },
+ { name = "hp3", description = "Hp3: Shunt" },
+}
+local dstaspects = {
+ { name = "vr1", description = "Vr1: Expect Full speed" },
+}
+
+local function applyaspect_main(loc)
+ return function(pos, node, main_aspect, rem_aspect, rem_aspinfo)
+ local ma_node = main_aspect.name
+ if all_sigs[ma_node] and not all_sigs[ma_node].distant then
+ -- ma_node is fine
+ elseif main_aspect.halt then
+ ma_node = "hp0" -- default halt aspect
+ else
+ ma_node = "hp1" -- default free aspect
+ end
+ advtrains.ndb.swap_node(pos, {name = "advtrains_signals_muc_ubahn:signal_wall_"..loc.."_"..ma_node, param2 = node.param2})
+ end
+end
+
+local function applyaspect_distant(loc)
+ return function(pos, node, main_aspect, rem_aspect, rem_aspinfo)
+ local ma_node = "vr0" -- show expect stop by default
+ if not main_aspect.halt and (not rem_aspinfo or not rem_aspinfo.main or rem_aspinfo.main>12 or rem_aspinfo.main==-1) then
+ ma_node = "vr1" -- show free when dst is at least 12
+ end
+ advtrains.ndb.swap_node(pos, {name = "advtrains_signals_muc_ubahn:signal_wall_"..loc.."_"..ma_node, param2 = node.param2})
+ end
+end
+
+for r,f in pairs(all_sigs) do
+ for loc, sbox in pairs({l={-1/2, -1/2, -1/4, 0, 1/2, 1/4}, r={0, -1/2, -1/4, 1/2, 1/2, 1/4}, t={-1/2, 0, -1/4, 1/2, 1/2, 1/4}}) do
+ minetest.register_node("advtrains_signals_muc_ubahn:signal_wall_"..loc.."_"..r, {
+ drawtype = "mesh",
+ paramtype="light",
+ paramtype2="facedir",
+ walkable = false,
+ selection_box = {
+ type = "fixed",
+ fixed = sbox,
+ },
+ mesh = "advtrains_signals_muc_ubahn_wsig_"..loc..".obj",
+ tiles = {"advtrains_signals_muc_ubahn_"..r..".png"},
+ drop = f.distant and "advtrains_signals_muc_ubahn:signal_wall_"..loc.."_vr0" or "advtrains_signals_muc_ubahn:signal_wall_"..loc.."_hp0",
+ description = f.distant and attrans("Munich U-Bahn Distant Signal ("..loc..")") or attrans("Munich U-Bahn Main Signal ("..loc..")"),
+ groups = {
+ cracky=3,
+ not_blocking_trains=1,
+ save_in_at_nodedb=1,
+ advtrains_signal = 2,
+ not_in_creative_inventory = f.crea and 0 or 1
+ },
+ light_source = 1,
+ sunlight_propagates=true,
+ on_rightclick = advtrains.interlocking.signal.on_rightclick,
+ can_dig = advtrains.interlocking.signal.can_dig,
+ after_dig_node = advtrains.interlocking.signal.after_dig,
+ -- new signal API
+ advtrains = {
+ main_aspects = f.distant and dstaspects or mainaspects, -- main aspects only for main
+ apply_aspect = f.distant and applyaspect_distant(loc) or applyaspect_main(loc),
+ pure_distant = f.distant,
+ get_aspect_info = function() return f.asp end,
+ route_role = f.distant and "distant" or "main"
+ },
+ })
+ end
+end
diff --git a/advtrains_signals_muc_ubahn/mod.conf b/advtrains_signals_muc_ubahn/mod.conf
new file mode 100644
index 0000000..05d1624
--- /dev/null
+++ b/advtrains_signals_muc_ubahn/mod.conf
@@ -0,0 +1,5 @@
+name=advtrains_signals_muc_ubahn
+title=Advtrains Interlocking Signal Set - Munich U-Bahn signals
+description=Subway signal set for the Advanced Trains Interlocking system modeled after Munich U-Bahn signals
+author=orwell96
+depends=advtrains_interlocking
diff --git a/advtrains_signals_muc_ubahn/models/advtrains_signals_muc_ubahn_wsig_l.obj b/advtrains_signals_muc_ubahn/models/advtrains_signals_muc_ubahn_wsig_l.obj
new file mode 100644
index 0000000..5e80955
--- /dev/null
+++ b/advtrains_signals_muc_ubahn/models/advtrains_signals_muc_ubahn_wsig_l.obj
@@ -0,0 +1,2137 @@
+# Blender 3.4.1
+# www.blender.org
+mtllib advtrains_signals_muc_ubahn_wsig_l.mtl
+o Cube
+v 0.500000 0.399750 0.043560
+v 0.500000 0.357794 0.043560
+v 0.500000 -0.357244 0.043560
+v 0.500000 -0.402505 0.043560
+v 0.500000 0.399750 -0.043560
+v 0.500000 0.357794 -0.043560
+v 0.500000 -0.357244 -0.043560
+v 0.500000 -0.402505 -0.043560
+v 0.302890 0.399750 0.043560
+v 0.302890 0.357794 0.043560
+v 0.302890 -0.357244 0.043560
+v 0.302890 -0.402505 0.043560
+v 0.302890 0.399750 -0.043560
+v 0.302890 0.357794 -0.043560
+v 0.302890 -0.357244 -0.043560
+v 0.302890 -0.402505 -0.043560
+v 0.218703 -0.357244 0.042354
+v 0.218703 0.357794 0.042353
+v 0.218703 -0.357244 -0.031584
+v 0.218703 0.357794 -0.031584
+v 0.450908 -0.357244 0.042353
+v 0.450908 0.357794 0.042353
+v 0.450908 -0.357244 -0.031584
+v 0.450908 0.357794 -0.031584
+v 0.265363 -0.310583 0.042353
+v 0.265363 0.311134 0.042353
+v 0.404248 0.311134 0.042353
+v 0.404247 -0.310583 0.042353
+v 0.265363 -0.310583 0.089581
+v 0.265363 0.311134 0.089581
+v 0.404248 0.311134 0.089581
+v 0.404247 -0.310583 0.089581
+v 0.257920 0.318577 -0.031584
+v 0.257920 -0.318027 -0.031584
+v 0.411691 0.318577 -0.031584
+v 0.411691 -0.318027 -0.031584
+v 0.331404 0.241338 -0.037331
+v 0.326280 0.189125 -0.037331
+v 0.316192 0.191128 -0.037331
+v 0.306689 0.195061 -0.037331
+v 0.298135 0.200771 -0.037331
+v 0.290861 0.208041 -0.037331
+v 0.285144 0.216591 -0.037331
+v 0.281205 0.226091 -0.037331
+v 0.279195 0.236177 -0.037331
+v 0.279191 0.246462 -0.037331
+v 0.281194 0.256550 -0.037331
+v 0.285127 0.266053 -0.037331
+v 0.290838 0.274606 -0.037331
+v 0.298107 0.281881 -0.037331
+v 0.306657 0.287598 -0.037331
+v 0.316157 0.291537 -0.037331
+v 0.326243 0.293547 -0.037331
+v 0.336528 0.293550 -0.037331
+v 0.346616 0.291547 -0.037331
+v 0.356119 0.287615 -0.037331
+v 0.364672 0.281904 -0.037331
+v 0.371947 0.274634 -0.037331
+v 0.377664 0.266085 -0.037331
+v 0.381603 0.256585 -0.037331
+v 0.383613 0.246498 -0.037331
+v 0.383616 0.236214 -0.037331
+v 0.381613 0.226126 -0.037331
+v 0.377681 0.216623 -0.037331
+v 0.371970 0.208069 -0.037331
+v 0.364700 0.200795 -0.037331
+v 0.356151 0.195078 -0.037331
+v 0.346651 0.191139 -0.037331
+v 0.336564 0.189129 -0.037331
+v 0.313532 0.182348 -0.031584
+v 0.325384 0.179994 -0.031584
+v 0.302367 0.186968 -0.031584
+v 0.292318 0.193677 -0.031584
+v 0.283771 0.202218 -0.031584
+v 0.277054 0.212263 -0.031584
+v 0.272426 0.223425 -0.031584
+v 0.270065 0.235275 -0.031584
+v 0.270060 0.247358 -0.031584
+v 0.272414 0.259210 -0.031584
+v 0.277034 0.270375 -0.031584
+v 0.283743 0.280424 -0.031584
+v 0.292285 0.288971 -0.031584
+v 0.302329 0.295687 -0.031584
+v 0.313491 0.300315 -0.031584
+v 0.325341 0.302677 -0.031584
+v 0.337424 0.302681 -0.031584
+v 0.349276 0.300328 -0.031584
+v 0.360441 0.295708 -0.031584
+v 0.370490 0.288998 -0.031584
+v 0.379037 0.280457 -0.031584
+v 0.385753 0.270413 -0.031584
+v 0.390381 0.259251 -0.031584
+v 0.392743 0.247401 -0.031584
+v 0.392747 0.235317 -0.031584
+v 0.390394 0.223466 -0.031584
+v 0.385774 0.212301 -0.031584
+v 0.379064 0.202252 -0.031584
+v 0.370523 0.193705 -0.031584
+v 0.360479 0.186988 -0.031584
+v 0.349317 0.182360 -0.031584
+v 0.337467 0.179999 -0.031584
+v 0.316192 0.191128 -0.054117
+v 0.326280 0.189125 -0.054117
+v 0.306689 0.195061 -0.054117
+v 0.298135 0.200771 -0.054117
+v 0.290861 0.208041 -0.054117
+v 0.285144 0.216591 -0.054117
+v 0.281205 0.226091 -0.054117
+v 0.279195 0.236177 -0.054117
+v 0.279191 0.246462 -0.083504
+v 0.281194 0.256550 -0.083504
+v 0.285127 0.266053 -0.083504
+v 0.290838 0.274606 -0.083504
+v 0.298107 0.281881 -0.083504
+v 0.306657 0.287598 -0.083504
+v 0.316157 0.291537 -0.083504
+v 0.326243 0.293547 -0.083504
+v 0.336528 0.293550 -0.083504
+v 0.346616 0.291547 -0.083504
+v 0.356119 0.287615 -0.083504
+v 0.364672 0.281904 -0.083504
+v 0.371947 0.274634 -0.083504
+v 0.377664 0.266085 -0.083504
+v 0.381603 0.256585 -0.083504
+v 0.383613 0.246498 -0.083504
+v 0.383616 0.236214 -0.054117
+v 0.381613 0.226126 -0.054117
+v 0.377681 0.216623 -0.054117
+v 0.371970 0.208069 -0.054117
+v 0.364700 0.200795 -0.054117
+v 0.356151 0.195078 -0.054117
+v 0.346651 0.191139 -0.054117
+v 0.336564 0.189129 -0.054117
+v 0.313532 0.182348 -0.054117
+v 0.325384 0.179994 -0.054117
+v 0.302367 0.186968 -0.054117
+v 0.292318 0.193677 -0.054117
+v 0.283771 0.202218 -0.054117
+v 0.277054 0.212263 -0.054117
+v 0.272426 0.223425 -0.054117
+v 0.270065 0.235275 -0.054117
+v 0.270060 0.247358 -0.083504
+v 0.272414 0.259210 -0.083504
+v 0.277034 0.270375 -0.083504
+v 0.283743 0.280424 -0.083504
+v 0.292284 0.288971 -0.083504
+v 0.302329 0.295687 -0.083504
+v 0.313491 0.300315 -0.083504
+v 0.325341 0.302677 -0.083504
+v 0.337424 0.302681 -0.083504
+v 0.349276 0.300328 -0.083504
+v 0.360441 0.295708 -0.083504
+v 0.370490 0.288998 -0.083504
+v 0.379037 0.280457 -0.083504
+v 0.385753 0.270413 -0.083504
+v 0.390381 0.259251 -0.083504
+v 0.392743 0.247401 -0.083504
+v 0.392747 0.235317 -0.054117
+v 0.390394 0.223466 -0.054117
+v 0.385774 0.212301 -0.054117
+v 0.379064 0.202252 -0.054117
+v 0.370523 0.193705 -0.054117
+v 0.360479 0.186988 -0.054117
+v 0.349317 0.182360 -0.054117
+v 0.337467 0.179999 -0.054117
+v 0.331404 0.091338 -0.037331
+v 0.326280 0.039125 -0.037331
+v 0.316192 0.041128 -0.037331
+v 0.306689 0.045061 -0.037331
+v 0.298135 0.050771 -0.037331
+v 0.290861 0.058041 -0.037331
+v 0.285144 0.066591 -0.037331
+v 0.281205 0.076091 -0.037331
+v 0.279195 0.086177 -0.037331
+v 0.279191 0.096462 -0.037331
+v 0.281194 0.106550 -0.037331
+v 0.285127 0.116053 -0.037331
+v 0.290838 0.124606 -0.037331
+v 0.298107 0.131881 -0.037331
+v 0.306657 0.137598 -0.037331
+v 0.316157 0.141537 -0.037331
+v 0.326243 0.143547 -0.037331
+v 0.336528 0.143550 -0.037331
+v 0.346616 0.141547 -0.037331
+v 0.356119 0.137615 -0.037331
+v 0.364672 0.131904 -0.037331
+v 0.371947 0.124634 -0.037331
+v 0.377664 0.116085 -0.037331
+v 0.381603 0.106585 -0.037331
+v 0.383613 0.096498 -0.037331
+v 0.383616 0.086214 -0.037331
+v 0.381613 0.076126 -0.037331
+v 0.377681 0.066623 -0.037331
+v 0.371970 0.058069 -0.037331
+v 0.364700 0.050795 -0.037331
+v 0.356151 0.045078 -0.037331
+v 0.346651 0.041139 -0.037331
+v 0.336564 0.039129 -0.037331
+v 0.313532 0.032348 -0.031584
+v 0.325384 0.029994 -0.031584
+v 0.302367 0.036968 -0.031584
+v 0.292318 0.043677 -0.031584
+v 0.283771 0.052218 -0.031584
+v 0.277054 0.062263 -0.031584
+v 0.272426 0.073425 -0.031584
+v 0.270065 0.085275 -0.031584
+v 0.270060 0.097358 -0.031584
+v 0.272414 0.109210 -0.031584
+v 0.277034 0.120375 -0.031584
+v 0.283743 0.130424 -0.031584
+v 0.292285 0.138971 -0.031584
+v 0.302329 0.145687 -0.031584
+v 0.313491 0.150315 -0.031584
+v 0.325341 0.152677 -0.031584
+v 0.337424 0.152681 -0.031584
+v 0.349276 0.150328 -0.031584
+v 0.360441 0.145708 -0.031584
+v 0.370490 0.138998 -0.031584
+v 0.379037 0.130457 -0.031584
+v 0.385753 0.120412 -0.031584
+v 0.390381 0.109251 -0.031584
+v 0.392743 0.097401 -0.031584
+v 0.392747 0.085317 -0.031584
+v 0.390394 0.073466 -0.031584
+v 0.385774 0.062301 -0.031584
+v 0.379064 0.052252 -0.031584
+v 0.370523 0.043705 -0.031584
+v 0.360479 0.036988 -0.031584
+v 0.349317 0.032360 -0.031584
+v 0.337467 0.029999 -0.031584
+v 0.316192 0.041128 -0.054117
+v 0.326280 0.039125 -0.054117
+v 0.306689 0.045061 -0.054117
+v 0.298135 0.050771 -0.054117
+v 0.290861 0.058041 -0.054117
+v 0.285144 0.066591 -0.054117
+v 0.281205 0.076091 -0.054117
+v 0.279195 0.086177 -0.054117
+v 0.279191 0.096462 -0.083504
+v 0.281194 0.106550 -0.083504
+v 0.285127 0.116053 -0.083504
+v 0.290838 0.124606 -0.083504
+v 0.298107 0.131881 -0.083504
+v 0.306657 0.137598 -0.083504
+v 0.316157 0.141537 -0.083504
+v 0.326243 0.143547 -0.083504
+v 0.336528 0.143550 -0.083504
+v 0.346616 0.141547 -0.083504
+v 0.356119 0.137615 -0.083504
+v 0.364672 0.131904 -0.083504
+v 0.371947 0.124634 -0.083504
+v 0.377664 0.116085 -0.083504
+v 0.381603 0.106585 -0.083504
+v 0.383613 0.096498 -0.083504
+v 0.383616 0.086214 -0.054117
+v 0.381613 0.076126 -0.054117
+v 0.377681 0.066623 -0.054117
+v 0.371970 0.058069 -0.054117
+v 0.364700 0.050795 -0.054117
+v 0.356151 0.045078 -0.054117
+v 0.346651 0.041139 -0.054117
+v 0.336564 0.039129 -0.054117
+v 0.313532 0.032348 -0.054117
+v 0.325384 0.029994 -0.054117
+v 0.302367 0.036968 -0.054117
+v 0.292318 0.043677 -0.054117
+v 0.283771 0.052218 -0.054117
+v 0.277054 0.062263 -0.054117
+v 0.272426 0.073425 -0.054117
+v 0.270065 0.085275 -0.054117
+v 0.270060 0.097358 -0.083504
+v 0.272414 0.109210 -0.083504
+v 0.277034 0.120375 -0.083504
+v 0.283743 0.130424 -0.083504
+v 0.292284 0.138971 -0.083504
+v 0.302329 0.145687 -0.083504
+v 0.313491 0.150315 -0.083504
+v 0.325341 0.152677 -0.083504
+v 0.337424 0.152681 -0.083504
+v 0.349276 0.150328 -0.083504
+v 0.360441 0.145708 -0.083504
+v 0.370490 0.138998 -0.083504
+v 0.379037 0.130457 -0.083504
+v 0.385753 0.120412 -0.083504
+v 0.390381 0.109251 -0.083504
+v 0.392743 0.097401 -0.083504
+v 0.392747 0.085317 -0.054117
+v 0.390394 0.073466 -0.054117
+v 0.385774 0.062301 -0.054117
+v 0.379064 0.052252 -0.054117
+v 0.370523 0.043705 -0.054117
+v 0.360479 0.036988 -0.054117
+v 0.349317 0.032360 -0.054117
+v 0.337467 0.029999 -0.054117
+v 0.331404 -0.058662 -0.037331
+v 0.326280 -0.110875 -0.037331
+v 0.316192 -0.108872 -0.037331
+v 0.306689 -0.104939 -0.037331
+v 0.298135 -0.099229 -0.037331
+v 0.290861 -0.091959 -0.037331
+v 0.285144 -0.083409 -0.037331
+v 0.281205 -0.073909 -0.037331
+v 0.279195 -0.063823 -0.037331
+v 0.279191 -0.053538 -0.037331
+v 0.281194 -0.043450 -0.037331
+v 0.285127 -0.033947 -0.037331
+v 0.290838 -0.025394 -0.037331
+v 0.298107 -0.018119 -0.037331
+v 0.306657 -0.012402 -0.037331
+v 0.316157 -0.008463 -0.037331
+v 0.326243 -0.006453 -0.037331
+v 0.336528 -0.006450 -0.037331
+v 0.346616 -0.008453 -0.037331
+v 0.356119 -0.012385 -0.037331
+v 0.364672 -0.018096 -0.037331
+v 0.371947 -0.025366 -0.037331
+v 0.377664 -0.033915 -0.037331
+v 0.381603 -0.043415 -0.037331
+v 0.383613 -0.053502 -0.037331
+v 0.383616 -0.063786 -0.037331
+v 0.381613 -0.073874 -0.037331
+v 0.377681 -0.083377 -0.037331
+v 0.371970 -0.091931 -0.037331
+v 0.364700 -0.099205 -0.037331
+v 0.356151 -0.104922 -0.037331
+v 0.346651 -0.108861 -0.037331
+v 0.336564 -0.110871 -0.037331
+v 0.313532 -0.117652 -0.031584
+v 0.325384 -0.120006 -0.031584
+v 0.302367 -0.113032 -0.031584
+v 0.292318 -0.106323 -0.031584
+v 0.283771 -0.097782 -0.031584
+v 0.277054 -0.087737 -0.031584
+v 0.272426 -0.076575 -0.031584
+v 0.270065 -0.064725 -0.031584
+v 0.270060 -0.052642 -0.031584
+v 0.272414 -0.040790 -0.031584
+v 0.277034 -0.029625 -0.031584
+v 0.283743 -0.019576 -0.031584
+v 0.292285 -0.011029 -0.031584
+v 0.302329 -0.004313 -0.031584
+v 0.313491 0.000315 -0.031584
+v 0.325341 0.002677 -0.031584
+v 0.337424 0.002681 -0.031584
+v 0.349276 0.000328 -0.031584
+v 0.360441 -0.004292 -0.031584
+v 0.370490 -0.011002 -0.031584
+v 0.379037 -0.019543 -0.031584
+v 0.385753 -0.029587 -0.031584
+v 0.390381 -0.040749 -0.031584
+v 0.392743 -0.052599 -0.031584
+v 0.392747 -0.064683 -0.031584
+v 0.390394 -0.076534 -0.031584
+v 0.385774 -0.087699 -0.031584
+v 0.379064 -0.097748 -0.031584
+v 0.370523 -0.106295 -0.031584
+v 0.360479 -0.113012 -0.031584
+v 0.349317 -0.117640 -0.031584
+v 0.337467 -0.120001 -0.031584
+v 0.316192 -0.108872 -0.054117
+v 0.326280 -0.110875 -0.054117
+v 0.306689 -0.104939 -0.054117
+v 0.298135 -0.099229 -0.054117
+v 0.290861 -0.091959 -0.054117
+v 0.285144 -0.083409 -0.054117
+v 0.281205 -0.073909 -0.054117
+v 0.279195 -0.063823 -0.054117
+v 0.279191 -0.053538 -0.083504
+v 0.281194 -0.043450 -0.083504
+v 0.285127 -0.033947 -0.083504
+v 0.290838 -0.025394 -0.083504
+v 0.298107 -0.018119 -0.083504
+v 0.306657 -0.012402 -0.083504
+v 0.316157 -0.008463 -0.083504
+v 0.326243 -0.006453 -0.083504
+v 0.336528 -0.006450 -0.083504
+v 0.346616 -0.008453 -0.083504
+v 0.356119 -0.012385 -0.083504
+v 0.364672 -0.018096 -0.083504
+v 0.371947 -0.025366 -0.083504
+v 0.377664 -0.033915 -0.083504
+v 0.381603 -0.043415 -0.083504
+v 0.383613 -0.053502 -0.083504
+v 0.383616 -0.063786 -0.054117
+v 0.381613 -0.073874 -0.054117
+v 0.377681 -0.083377 -0.054117
+v 0.371970 -0.091931 -0.054117
+v 0.364700 -0.099205 -0.054117
+v 0.356151 -0.104922 -0.054117
+v 0.346651 -0.108861 -0.054117
+v 0.336564 -0.110871 -0.054117
+v 0.313532 -0.117652 -0.054117
+v 0.325384 -0.120006 -0.054117
+v 0.302367 -0.113032 -0.054117
+v 0.292318 -0.106323 -0.054117
+v 0.283771 -0.097782 -0.054117
+v 0.277054 -0.087737 -0.054117
+v 0.272426 -0.076575 -0.054117
+v 0.270065 -0.064725 -0.054117
+v 0.270060 -0.052642 -0.083504
+v 0.272414 -0.040790 -0.083504
+v 0.277034 -0.029625 -0.083504
+v 0.283743 -0.019576 -0.083504
+v 0.292284 -0.011029 -0.083504
+v 0.302329 -0.004313 -0.083504
+v 0.313491 0.000315 -0.083504
+v 0.325341 0.002677 -0.083504
+v 0.337424 0.002681 -0.083504
+v 0.349276 0.000328 -0.083504
+v 0.360441 -0.004292 -0.083504
+v 0.370490 -0.011002 -0.083504
+v 0.379037 -0.019543 -0.083504
+v 0.385753 -0.029587 -0.083504
+v 0.390381 -0.040749 -0.083504
+v 0.392743 -0.052599 -0.083504
+v 0.392747 -0.064683 -0.054117
+v 0.390394 -0.076534 -0.054117
+v 0.385774 -0.087699 -0.054117
+v 0.379064 -0.097748 -0.054117
+v 0.370523 -0.106295 -0.054117
+v 0.360479 -0.113012 -0.054117
+v 0.349317 -0.117640 -0.054117
+v 0.337467 -0.120001 -0.054117
+v 0.331404 -0.208662 -0.037331
+v 0.326280 -0.260875 -0.037331
+v 0.316192 -0.258872 -0.037331
+v 0.306689 -0.254939 -0.037331
+v 0.298135 -0.249229 -0.037331
+v 0.290861 -0.241959 -0.037331
+v 0.285144 -0.233410 -0.037331
+v 0.281205 -0.223909 -0.037331
+v 0.279195 -0.213823 -0.037331
+v 0.279191 -0.203538 -0.037331
+v 0.281194 -0.193450 -0.037331
+v 0.285127 -0.183947 -0.037331
+v 0.290838 -0.175394 -0.037331
+v 0.298107 -0.168119 -0.037331
+v 0.306657 -0.162402 -0.037331
+v 0.316157 -0.158463 -0.037331
+v 0.326243 -0.156453 -0.037331
+v 0.336528 -0.156450 -0.037331
+v 0.346616 -0.158453 -0.037331
+v 0.356119 -0.162385 -0.037331
+v 0.364672 -0.168096 -0.037331
+v 0.371947 -0.175366 -0.037331
+v 0.377664 -0.183915 -0.037331
+v 0.381603 -0.193415 -0.037331
+v 0.383613 -0.203502 -0.037331
+v 0.383616 -0.213786 -0.037331
+v 0.381613 -0.223874 -0.037331
+v 0.377681 -0.233377 -0.037331
+v 0.371970 -0.241931 -0.037331
+v 0.364700 -0.249205 -0.037331
+v 0.356151 -0.254922 -0.037331
+v 0.346651 -0.258861 -0.037331
+v 0.336564 -0.260871 -0.037331
+v 0.313532 -0.267652 -0.031584
+v 0.325384 -0.270006 -0.031584
+v 0.302367 -0.263032 -0.031584
+v 0.292318 -0.256323 -0.031584
+v 0.283771 -0.247782 -0.031584
+v 0.277054 -0.237737 -0.031584
+v 0.272426 -0.226575 -0.031584
+v 0.270065 -0.214725 -0.031584
+v 0.270060 -0.202642 -0.031584
+v 0.272414 -0.190790 -0.031584
+v 0.277034 -0.179625 -0.031584
+v 0.283743 -0.169576 -0.031584
+v 0.292285 -0.161029 -0.031584
+v 0.302329 -0.154313 -0.031584
+v 0.313491 -0.149685 -0.031584
+v 0.325341 -0.147323 -0.031584
+v 0.337424 -0.147319 -0.031584
+v 0.349276 -0.149672 -0.031584
+v 0.360441 -0.154292 -0.031584
+v 0.370490 -0.161002 -0.031584
+v 0.379037 -0.169543 -0.031584
+v 0.385753 -0.179587 -0.031584
+v 0.390381 -0.190749 -0.031584
+v 0.392743 -0.202599 -0.031584
+v 0.392747 -0.214683 -0.031584
+v 0.390394 -0.226534 -0.031584
+v 0.385774 -0.237699 -0.031584
+v 0.379064 -0.247748 -0.031584
+v 0.370523 -0.256295 -0.031584
+v 0.360479 -0.263012 -0.031584
+v 0.349317 -0.267640 -0.031584
+v 0.337467 -0.270001 -0.031584
+v 0.316192 -0.258872 -0.054117
+v 0.326280 -0.260875 -0.054117
+v 0.306689 -0.254939 -0.054117
+v 0.298135 -0.249229 -0.054117
+v 0.290861 -0.241959 -0.054117
+v 0.285144 -0.233410 -0.054117
+v 0.281205 -0.223909 -0.054117
+v 0.279195 -0.213823 -0.054117
+v 0.279191 -0.203538 -0.083504
+v 0.281194 -0.193450 -0.083504
+v 0.285127 -0.183947 -0.083504
+v 0.290838 -0.175394 -0.083504
+v 0.298107 -0.168119 -0.083504
+v 0.306657 -0.162402 -0.083504
+v 0.316157 -0.158463 -0.083504
+v 0.326243 -0.156453 -0.083504
+v 0.336528 -0.156450 -0.083504
+v 0.346616 -0.158453 -0.083504
+v 0.356119 -0.162385 -0.083504
+v 0.364672 -0.168096 -0.083504
+v 0.371947 -0.175366 -0.083504
+v 0.377664 -0.183915 -0.083504
+v 0.381603 -0.193415 -0.083504
+v 0.383613 -0.203502 -0.083504
+v 0.383616 -0.213786 -0.054117
+v 0.381613 -0.223874 -0.054117
+v 0.377681 -0.233377 -0.054117
+v 0.371970 -0.241931 -0.054117
+v 0.364700 -0.249205 -0.054117
+v 0.356151 -0.254922 -0.054117
+v 0.346651 -0.258861 -0.054117
+v 0.336564 -0.260871 -0.054117
+v 0.313532 -0.267652 -0.054117
+v 0.325384 -0.270006 -0.054117
+v 0.302367 -0.263032 -0.054117
+v 0.292318 -0.256323 -0.054117
+v 0.283771 -0.247782 -0.054117
+v 0.277054 -0.237737 -0.054117
+v 0.272426 -0.226575 -0.054117
+v 0.270065 -0.214725 -0.054117
+v 0.270060 -0.202642 -0.083504
+v 0.272414 -0.190790 -0.083504
+v 0.277034 -0.179625 -0.083504
+v 0.283743 -0.169576 -0.083504
+v 0.292284 -0.161029 -0.083504
+v 0.302329 -0.154313 -0.083504
+v 0.313491 -0.149685 -0.083504
+v 0.325341 -0.147323 -0.083504
+v 0.337424 -0.147319 -0.083504
+v 0.349276 -0.149672 -0.083504
+v 0.360441 -0.154292 -0.083504
+v 0.370490 -0.161002 -0.083504
+v 0.379037 -0.169543 -0.083504
+v 0.385753 -0.179587 -0.083504
+v 0.390381 -0.190749 -0.083504
+v 0.392743 -0.202599 -0.083504
+v 0.392747 -0.214683 -0.054117
+v 0.390394 -0.226534 -0.054117
+v 0.385774 -0.237699 -0.054117
+v 0.379064 -0.247748 -0.054117
+v 0.370523 -0.256295 -0.054117
+v 0.360479 -0.263012 -0.054117
+v 0.349317 -0.267640 -0.054117
+v 0.337467 -0.270001 -0.054117
+vn -0.0000 -1.0000 -0.0000
+vn -0.0000 1.0000 -0.0000
+vn -1.0000 -0.0000 -0.0000
+vn -0.0000 -0.0000 -1.0000
+vn -0.0000 -0.0000 1.0000
+vn 1.0000 -0.0000 -0.0000
+vn 0.3830 -0.9237 -0.0000
+vn -0.1954 0.9807 -0.0000
+vn -0.8313 -0.5559 -0.0000
+vn -0.0003 1.0000 -0.0000
+vn 0.9237 0.3830 -0.0000
+vn -0.9807 -0.1954 -0.0000
+vn -0.1947 -0.9809 -0.0000
+vn -0.7073 0.7069 -0.0000
+vn 0.5559 -0.8313 -0.0000
+vn -0.3830 0.9237 -0.0000
+vn 0.3824 0.9240 -0.0000
+vn -1.0000 -0.0003 -0.0000
+vn 0.8313 0.5559 -0.0000
+vn -0.9237 -0.3830 -0.0000
+vn 0.9809 -0.1947 -0.0000
+vn 0.7074 -0.7069 -0.0000
+vn -0.5559 0.8313 -0.0000
+vn -0.7069 -0.7074 -0.0000
+vn 0.7069 0.7073 -0.0000
+vn 0.0003 -1.0000 -0.0000
+vn -0.8317 0.5553 -0.0000
+vn 0.8317 -0.5553 -0.0000
+vn -0.7074 0.7069 -0.0000
+vn 0.5553 0.8317 -0.0000
+vn 0.0003 -0.9346 -0.3557
+vn 0.9240 -0.3824 -0.0000
+vn -1.0000 -0.0004 -0.0000
+vn 0.9807 0.1954 -0.0000
+vn -0.9809 0.1948 -0.0000
+vn -0.1948 -0.9809 -0.0000
+vn 0.1954 -0.9807 -0.0000
+vn -0.3824 -0.9240 -0.0000
+vn 1.0000 0.0004 -0.0000
+vn 0.1947 0.9809 -0.0000
+vn -0.9240 0.3824 -0.0000
+vn -0.5553 -0.8317 -0.0000
+vn 0.7069 0.7074 -0.0000
+vn 1.0000 0.0003 -0.0000
+vn 0.7073 -0.7069 -0.0000
+vn -0.0004 1.0000 -0.0000
+vt 0.640016 0.630686
+vt 0.874116 0.630686
+vt 0.899031 0.371671
+vt 0.899031 0.605771
+vt 0.275073 0.371671
+vt 0.275073 0.605771
+vt 0.536051 0.632649
+vt 0.301951 0.632649
+vt 0.588281 0.578951
+vt 0.588281 0.344851
+vt 0.847296 0.319936
+vt 0.613196 0.319936
+vt 0.326808 0.319936
+vt 0.560909 0.319936
+vt 0.587786 0.580914
+vt 0.587786 0.346814
+vt 0.757066 0.513636
+vt 0.781981 0.488721
+vt 0.392123 0.488721
+vt 0.419001 0.515599
+vt 0.705331 0.461901
+vt 0.730246 0.436986
+vt 0.443858 0.436986
+vt 0.470736 0.463864
+vt 0.062341 0.000247
+vt 0.000247 0.062341
+vt 0.000248 0.359598
+vt 0.039909 0.319936
+vt 0.662834 0.000247
+vt 0.724927 0.062341
+vt 0.000247 0.960091
+vt 0.039910 0.999753
+vt 0.062341 0.062341
+vt 0.662834 0.062341
+vt 0.062341 0.319441
+vt 0.000247 0.257348
+vt 0.274579 0.359598
+vt 0.234916 0.319936
+vt 0.662834 0.319441
+vt 0.724927 0.257348
+vt 0.274579 0.960090
+vt 0.234917 0.999753
+vt 0.062341 0.257348
+vt 0.662834 0.257348
+vt 0.079095 0.359121
+vt 0.039433 0.398784
+vt 0.079095 0.960567
+vt 0.039433 0.920905
+vt 0.195731 0.960567
+vt 0.235393 0.920905
+vt 0.195731 0.359121
+vt 0.235393 0.398783
+vt 0.079095 0.398783
+vt 0.079095 0.920905
+vt 0.195731 0.920905
+vt 0.195731 0.398783
+vt 0.629899 0.095275
+vt 0.095275 0.095275
+vt 0.629899 0.224413
+vt 0.095275 0.224413
+vt 0.753412 0.903011
+vt 0.719354 0.930961
+vt 0.334815 0.932388
+vt 0.611202 0.932388
+vt 0.714555 0.923780
+vt 0.602565 0.932389
+vt 0.711250 0.915800
+vt 0.593928 0.932389
+vt 0.709565 0.907329
+vt 0.585291 0.932389
+vt 0.709565 0.898692
+vt 0.576654 0.932389
+vt 0.711250 0.890221
+vt 0.568017 0.932389
+vt 0.714556 0.882241
+vt 0.559379 0.932389
+vt 0.719354 0.875060
+vt 0.550742 0.932389
+vt 0.725461 0.868952
+vt 0.542105 0.932389
+vt 0.732643 0.864154
+vt 0.533468 0.932389
+vt 0.740623 0.860849
+vt 0.524831 0.932389
+vt 0.749094 0.859164
+vt 0.516194 0.932389
+vt 0.757731 0.859164
+vt 0.507557 0.932389
+vt 0.766202 0.860849
+vt 0.498920 0.932389
+vt 0.774182 0.864154
+vt 0.490283 0.932389
+vt 0.781363 0.868953
+vt 0.481646 0.932389
+vt 0.787470 0.875060
+vt 0.473009 0.932389
+vt 0.792269 0.882241
+vt 0.464371 0.932389
+vt 0.795574 0.890221
+vt 0.455734 0.932389
+vt 0.797259 0.898692
+vt 0.447097 0.932389
+vt 0.797259 0.907329
+vt 0.438460 0.932389
+vt 0.795574 0.915800
+vt 0.429823 0.932389
+vt 0.792269 0.923780
+vt 0.421186 0.932389
+vt 0.787470 0.930962
+vt 0.412549 0.932389
+vt 0.781363 0.937069
+vt 0.403912 0.932389
+vt 0.774181 0.941867
+vt 0.395275 0.932389
+vt 0.766201 0.945173
+vt 0.386637 0.932389
+vt 0.757730 0.946858
+vt 0.378001 0.932389
+vt 0.749093 0.946858
+vt 0.369363 0.932389
+vt 0.740622 0.945172
+vt 0.360726 0.932389
+vt 0.732643 0.941867
+vt 0.352089 0.932388
+vt 0.725461 0.937069
+vt 0.343452 0.932388
+vt 0.618830 0.816892
+vt 0.628978 0.816892
+vt 0.608683 0.816892
+vt 0.598535 0.816892
+vt 0.588388 0.816892
+vt 0.578240 0.816892
+vt 0.568093 0.816892
+vt 0.557945 0.816892
+vt 0.547798 0.816892
+vt 0.537650 0.816892
+vt 0.527503 0.816892
+vt 0.517355 0.816892
+vt 0.507208 0.816892
+vt 0.497060 0.816892
+vt 0.486913 0.816892
+vt 0.476765 0.816892
+vt 0.466618 0.816892
+vt 0.456470 0.816892
+vt 0.446323 0.816892
+vt 0.436175 0.816892
+vt 0.426028 0.816892
+vt 0.415880 0.816892
+vt 0.405733 0.816892
+vt 0.395585 0.816892
+vt 0.385438 0.816892
+vt 0.375290 0.816892
+vt 0.365143 0.816892
+vt 0.354995 0.816892
+vt 0.344848 0.816892
+vt 0.334700 0.816892
+vt 0.324553 0.816892
+vt 0.314405 0.816892
+vt 0.639125 0.816892
+vt 0.517095 0.669905
+vt 0.602565 0.946486
+vt 0.334815 0.946485
+vt 0.515410 0.678377
+vt 0.611202 0.946485
+vt 0.520400 0.661926
+vt 0.593928 0.946485
+vt 0.525198 0.654744
+vt 0.585291 0.946486
+vt 0.531306 0.648637
+vt 0.576654 0.946486
+vt 0.568017 0.946486
+vt 0.538487 0.643839
+vt 0.546467 0.640533
+vt 0.559379 0.946486
+vt 0.554938 0.638848
+vt 0.507247 0.729756
+vt 0.550742 0.946486
+vt 0.481149 0.728166
+vt 0.542105 0.971165
+vt 0.472678 0.726481
+vt 0.533468 0.971165
+vt 0.464698 0.723176
+vt 0.524831 0.971165
+vt 0.457517 0.718377
+vt 0.516194 0.971165
+vt 0.507557 0.971165
+vt 0.451409 0.712270
+vt 0.446611 0.705089
+vt 0.498920 0.971165
+vt 0.443305 0.697109
+vt 0.490283 0.971165
+vt 0.441620 0.688638
+vt 0.481646 0.971165
+vt 0.473008 0.971165
+vt 0.441620 0.680001
+vt 0.443305 0.671530
+vt 0.464371 0.971165
+vt 0.446610 0.663550
+vt 0.455734 0.971165
+vt 0.451409 0.656369
+vt 0.447097 0.971165
+vt 0.438460 0.971165
+vt 0.457516 0.650261
+vt 0.464698 0.645463
+vt 0.429823 0.971165
+vt 0.472677 0.642157
+vt 0.421186 0.971165
+vt 0.412549 0.971165
+vt 0.481149 0.640472
+vt 0.507247 0.638882
+vt 0.554938 0.726542
+vt 0.403912 0.946486
+vt 0.546467 0.724857
+vt 0.395275 0.946486
+vt 0.538487 0.721552
+vt 0.386637 0.946486
+vt 0.378001 0.946486
+vt 0.531306 0.716753
+vt 0.525198 0.710646
+vt 0.369363 0.946486
+vt 0.520400 0.703464
+vt 0.360726 0.946485
+vt 0.517095 0.695485
+vt 0.352089 0.946485
+vt 0.343452 0.946485
+vt 0.515410 0.687014
+vt 0.509722 0.667669
+vt 0.618830 0.797969
+vt 0.507742 0.677621
+vt 0.628978 0.797969
+vt 0.513605 0.658294
+vt 0.608683 0.797969
+vt 0.519242 0.649857
+vt 0.598535 0.797969
+vt 0.588388 0.797969
+vt 0.526418 0.642681
+vt 0.578240 0.797969
+vt 0.534855 0.637043
+vt 0.568093 0.797969
+vt 0.544230 0.633160
+vt 0.554183 0.631180
+vt 0.507028 0.737458
+vt 0.557945 0.797969
+vt 0.480393 0.735834
+vt 0.547798 0.773289
+vt 0.470441 0.733854
+vt 0.537650 0.773289
+vt 0.461066 0.729971
+vt 0.527503 0.773289
+vt 0.452629 0.724334
+vt 0.517355 0.773289
+vt 0.507208 0.773290
+vt 0.445453 0.717158
+vt 0.497060 0.773290
+vt 0.439816 0.708721
+vt 0.486913 0.773290
+vt 0.435932 0.699346
+vt 0.433953 0.689393
+vt 0.476765 0.773290
+vt 0.433953 0.679246
+vt 0.466618 0.773290
+vt 0.435932 0.669293
+vt 0.456470 0.773290
+vt 0.439815 0.659918
+vt 0.446323 0.773289
+vt 0.436175 0.773289
+vt 0.445453 0.651481
+vt 0.426028 0.773289
+vt 0.452628 0.644305
+vt 0.415880 0.773289
+vt 0.461066 0.638668
+vt 0.470441 0.634784
+vt 0.405733 0.773289
+vt 0.480393 0.632805
+vt 0.395585 0.773289
+vt 0.507028 0.631180
+vt 0.554183 0.734210
+vt 0.385438 0.797969
+vt 0.544230 0.732230
+vt 0.375290 0.797969
+vt 0.534855 0.728347
+vt 0.365143 0.797969
+vt 0.354995 0.797969
+vt 0.526418 0.722709
+vt 0.344848 0.797969
+vt 0.519243 0.715534
+vt 0.334700 0.797969
+vt 0.513605 0.707096
+vt 0.509721 0.697721
+vt 0.324553 0.797969
+vt 0.507742 0.687769
+vt 0.314405 0.797969
+vt 0.639125 0.797969
+vt 0.752599 0.742115
+vt 0.718541 0.770066
+vt 0.334815 0.932388
+vt 0.611202 0.932388
+vt 0.713742 0.762885
+vt 0.602565 0.932389
+vt 0.710437 0.754905
+vt 0.593928 0.932389
+vt 0.708752 0.746434
+vt 0.585291 0.932389
+vt 0.708752 0.737797
+vt 0.576654 0.932389
+vt 0.710437 0.729326
+vt 0.568017 0.932389
+vt 0.713743 0.721346
+vt 0.559379 0.932389
+vt 0.718541 0.714165
+vt 0.550742 0.932389
+vt 0.724649 0.708057
+vt 0.542105 0.932389
+vt 0.731830 0.703259
+vt 0.533468 0.932389
+vt 0.739810 0.699954
+vt 0.524831 0.932389
+vt 0.748281 0.698269
+vt 0.516194 0.932389
+vt 0.756918 0.698269
+vt 0.507557 0.932389
+vt 0.765389 0.699954
+vt 0.498920 0.932389
+vt 0.773369 0.703259
+vt 0.490283 0.932389
+vt 0.780550 0.708058
+vt 0.481646 0.932389
+vt 0.786658 0.714165
+vt 0.473009 0.932389
+vt 0.791456 0.721346
+vt 0.464371 0.932389
+vt 0.794761 0.729326
+vt 0.455734 0.932389
+vt 0.796446 0.737797
+vt 0.447097 0.932389
+vt 0.796446 0.746434
+vt 0.438460 0.932389
+vt 0.794761 0.754905
+vt 0.429823 0.932389
+vt 0.791456 0.762885
+vt 0.421186 0.932389
+vt 0.786657 0.770066
+vt 0.412549 0.932389
+vt 0.780550 0.776174
+vt 0.403912 0.932389
+vt 0.773368 0.780972
+vt 0.395275 0.932389
+vt 0.765389 0.784277
+vt 0.386637 0.932389
+vt 0.756917 0.785962
+vt 0.378001 0.932389
+vt 0.748280 0.785962
+vt 0.369363 0.932389
+vt 0.739809 0.784277
+vt 0.360726 0.932389
+vt 0.731830 0.780972
+vt 0.352089 0.932388
+vt 0.724648 0.776174
+vt 0.343452 0.932388
+vt 0.618830 0.816892
+vt 0.628978 0.816892
+vt 0.608683 0.816892
+vt 0.598535 0.816892
+vt 0.588388 0.816892
+vt 0.578240 0.816892
+vt 0.568093 0.816892
+vt 0.557945 0.816892
+vt 0.547798 0.816892
+vt 0.537650 0.816892
+vt 0.527503 0.816892
+vt 0.517355 0.816892
+vt 0.507208 0.816892
+vt 0.497060 0.816892
+vt 0.486913 0.816892
+vt 0.476765 0.816892
+vt 0.466618 0.816892
+vt 0.456470 0.816892
+vt 0.446323 0.816892
+vt 0.436175 0.816892
+vt 0.426028 0.816892
+vt 0.415880 0.816892
+vt 0.405733 0.816892
+vt 0.395585 0.816892
+vt 0.385438 0.816892
+vt 0.375290 0.816892
+vt 0.365143 0.816892
+vt 0.354995 0.816892
+vt 0.344848 0.816892
+vt 0.334700 0.816892
+vt 0.324553 0.816892
+vt 0.314405 0.816892
+vt 0.639125 0.816892
+vt 0.517095 0.669905
+vt 0.602565 0.946486
+vt 0.334815 0.946485
+vt 0.515410 0.678377
+vt 0.611202 0.946485
+vt 0.520400 0.661926
+vt 0.593928 0.946485
+vt 0.525198 0.654744
+vt 0.585291 0.946486
+vt 0.531306 0.648637
+vt 0.576654 0.946486
+vt 0.568017 0.946486
+vt 0.538487 0.643839
+vt 0.546467 0.640533
+vt 0.559379 0.946486
+vt 0.554938 0.638848
+vt 0.507247 0.729756
+vt 0.550742 0.946486
+vt 0.481149 0.728166
+vt 0.542105 0.971165
+vt 0.472678 0.726481
+vt 0.533468 0.971165
+vt 0.464698 0.723176
+vt 0.524831 0.971165
+vt 0.457517 0.718377
+vt 0.516194 0.971165
+vt 0.507557 0.971165
+vt 0.451409 0.712270
+vt 0.446611 0.705089
+vt 0.498920 0.971165
+vt 0.443305 0.697109
+vt 0.490283 0.971165
+vt 0.441620 0.688638
+vt 0.481646 0.971165
+vt 0.473008 0.971165
+vt 0.441620 0.680001
+vt 0.443305 0.671530
+vt 0.464371 0.971165
+vt 0.446610 0.663550
+vt 0.455734 0.971165
+vt 0.451409 0.656369
+vt 0.447097 0.971165
+vt 0.438460 0.971165
+vt 0.457516 0.650261
+vt 0.464698 0.645463
+vt 0.429823 0.971165
+vt 0.472677 0.642157
+vt 0.421186 0.971165
+vt 0.412549 0.971165
+vt 0.481149 0.640472
+vt 0.507247 0.638882
+vt 0.554938 0.726542
+vt 0.403912 0.946486
+vt 0.546467 0.724857
+vt 0.395275 0.946486
+vt 0.538487 0.721552
+vt 0.386637 0.946486
+vt 0.378001 0.946486
+vt 0.531306 0.716753
+vt 0.525198 0.710646
+vt 0.369363 0.946486
+vt 0.520400 0.703464
+vt 0.360726 0.946485
+vt 0.517095 0.695485
+vt 0.352089 0.946485
+vt 0.343452 0.946485
+vt 0.515410 0.687014
+vt 0.509722 0.667669
+vt 0.618830 0.797969
+vt 0.507742 0.677621
+vt 0.628978 0.797969
+vt 0.513605 0.658294
+vt 0.608683 0.797969
+vt 0.519242 0.649857
+vt 0.598535 0.797969
+vt 0.588388 0.797969
+vt 0.526418 0.642681
+vt 0.578240 0.797969
+vt 0.534855 0.637043
+vt 0.568093 0.797969
+vt 0.544230 0.633160
+vt 0.554183 0.631180
+vt 0.507028 0.737458
+vt 0.557945 0.797969
+vt 0.480393 0.735834
+vt 0.547798 0.773289
+vt 0.470441 0.733854
+vt 0.537650 0.773289
+vt 0.461066 0.729971
+vt 0.527503 0.773289
+vt 0.452629 0.724334
+vt 0.517355 0.773289
+vt 0.507208 0.773290
+vt 0.445453 0.717158
+vt 0.497060 0.773290
+vt 0.439816 0.708721
+vt 0.486913 0.773290
+vt 0.435932 0.699346
+vt 0.433953 0.689393
+vt 0.476765 0.773290
+vt 0.433953 0.679246
+vt 0.466618 0.773290
+vt 0.435932 0.669293
+vt 0.456470 0.773290
+vt 0.439815 0.659918
+vt 0.446323 0.773289
+vt 0.436175 0.773289
+vt 0.445453 0.651481
+vt 0.426028 0.773289
+vt 0.452628 0.644305
+vt 0.415880 0.773289
+vt 0.461066 0.638668
+vt 0.470441 0.634784
+vt 0.405733 0.773289
+vt 0.480393 0.632805
+vt 0.395585 0.773289
+vt 0.507028 0.631180
+vt 0.554183 0.734210
+vt 0.385438 0.797969
+vt 0.544230 0.732230
+vt 0.375290 0.797969
+vt 0.534855 0.728347
+vt 0.365143 0.797969
+vt 0.354995 0.797969
+vt 0.526418 0.722709
+vt 0.344848 0.797969
+vt 0.519243 0.715534
+vt 0.334700 0.797969
+vt 0.513605 0.707096
+vt 0.509721 0.697721
+vt 0.324553 0.797969
+vt 0.507742 0.687769
+vt 0.314405 0.797969
+vt 0.639125 0.797969
+vt 0.908735 0.903617
+vt 0.874676 0.931567
+vt 0.334815 0.932388
+vt 0.611202 0.932388
+vt 0.869878 0.924386
+vt 0.602565 0.932389
+vt 0.866572 0.916406
+vt 0.593928 0.932389
+vt 0.864888 0.907935
+vt 0.585291 0.932389
+vt 0.864888 0.899298
+vt 0.576654 0.932389
+vt 0.866573 0.890827
+vt 0.568017 0.932389
+vt 0.869878 0.882847
+vt 0.559379 0.932389
+vt 0.874676 0.875666
+vt 0.550742 0.932389
+vt 0.880784 0.869558
+vt 0.542105 0.932389
+vt 0.887965 0.864760
+vt 0.533468 0.932389
+vt 0.895945 0.861455
+vt 0.524831 0.932389
+vt 0.904416 0.859770
+vt 0.516194 0.932389
+vt 0.913053 0.859770
+vt 0.507557 0.932389
+vt 0.921524 0.861455
+vt 0.498920 0.932389
+vt 0.929504 0.864760
+vt 0.490283 0.932389
+vt 0.936685 0.869559
+vt 0.481646 0.932389
+vt 0.942793 0.875666
+vt 0.473009 0.932389
+vt 0.947591 0.882847
+vt 0.464371 0.932389
+vt 0.950896 0.890827
+vt 0.455734 0.932389
+vt 0.952581 0.899298
+vt 0.447097 0.932389
+vt 0.952581 0.907935
+vt 0.438460 0.932389
+vt 0.950896 0.916406
+vt 0.429823 0.932389
+vt 0.947591 0.924386
+vt 0.421186 0.932389
+vt 0.942792 0.931568
+vt 0.412549 0.932389
+vt 0.936685 0.937675
+vt 0.403912 0.932389
+vt 0.929504 0.942473
+vt 0.395275 0.932389
+vt 0.921524 0.945779
+vt 0.386637 0.932389
+vt 0.913053 0.947464
+vt 0.378001 0.932389
+vt 0.904416 0.947464
+vt 0.369363 0.932389
+vt 0.895945 0.945778
+vt 0.360726 0.932389
+vt 0.887965 0.942473
+vt 0.352089 0.932388
+vt 0.880783 0.937675
+vt 0.343452 0.932388
+vt 0.618830 0.816892
+vt 0.628978 0.816892
+vt 0.608683 0.816892
+vt 0.598535 0.816892
+vt 0.588388 0.816892
+vt 0.578240 0.816892
+vt 0.568093 0.816892
+vt 0.557945 0.816892
+vt 0.547798 0.816892
+vt 0.537650 0.816892
+vt 0.527503 0.816892
+vt 0.517355 0.816892
+vt 0.507208 0.816892
+vt 0.497060 0.816892
+vt 0.486913 0.816892
+vt 0.476765 0.816892
+vt 0.466618 0.816892
+vt 0.456470 0.816892
+vt 0.446323 0.816892
+vt 0.436175 0.816892
+vt 0.426028 0.816892
+vt 0.415880 0.816892
+vt 0.405733 0.816892
+vt 0.395585 0.816892
+vt 0.385438 0.816892
+vt 0.375290 0.816892
+vt 0.365143 0.816892
+vt 0.354995 0.816892
+vt 0.344848 0.816892
+vt 0.334700 0.816892
+vt 0.324553 0.816892
+vt 0.314405 0.816892
+vt 0.639125 0.816892
+vt 0.517095 0.669905
+vt 0.602565 0.946486
+vt 0.334815 0.946485
+vt 0.515410 0.678377
+vt 0.611202 0.946485
+vt 0.520400 0.661926
+vt 0.593928 0.946485
+vt 0.525198 0.654744
+vt 0.585291 0.946486
+vt 0.531306 0.648637
+vt 0.576654 0.946486
+vt 0.568017 0.946486
+vt 0.538487 0.643839
+vt 0.546467 0.640533
+vt 0.559379 0.946486
+vt 0.554938 0.638848
+vt 0.507247 0.729756
+vt 0.550742 0.946486
+vt 0.481149 0.728166
+vt 0.542105 0.971165
+vt 0.472678 0.726481
+vt 0.533468 0.971165
+vt 0.464698 0.723176
+vt 0.524831 0.971165
+vt 0.457517 0.718377
+vt 0.516194 0.971165
+vt 0.507557 0.971165
+vt 0.451409 0.712270
+vt 0.446611 0.705089
+vt 0.498920 0.971165
+vt 0.443305 0.697109
+vt 0.490283 0.971165
+vt 0.441620 0.688638
+vt 0.481646 0.971165
+vt 0.473008 0.971165
+vt 0.441620 0.680001
+vt 0.443305 0.671530
+vt 0.464371 0.971165
+vt 0.446610 0.663550
+vt 0.455734 0.971165
+vt 0.451409 0.656369
+vt 0.447097 0.971165
+vt 0.438460 0.971165
+vt 0.457516 0.650261
+vt 0.464698 0.645463
+vt 0.429823 0.971165
+vt 0.472677 0.642157
+vt 0.421186 0.971165
+vt 0.412549 0.971165
+vt 0.481149 0.640472
+vt 0.507247 0.638882
+vt 0.554938 0.726542
+vt 0.403912 0.946486
+vt 0.546467 0.724857
+vt 0.395275 0.946486
+vt 0.538487 0.721552
+vt 0.386637 0.946486
+vt 0.378001 0.946486
+vt 0.531306 0.716753
+vt 0.525198 0.710646
+vt 0.369363 0.946486
+vt 0.520400 0.703464
+vt 0.360726 0.946485
+vt 0.517095 0.695485
+vt 0.352089 0.946485
+vt 0.343452 0.946485
+vt 0.515410 0.687014
+vt 0.509722 0.667669
+vt 0.618830 0.797969
+vt 0.507742 0.677621
+vt 0.628978 0.797969
+vt 0.513605 0.658294
+vt 0.608683 0.797969
+vt 0.519242 0.649857
+vt 0.598535 0.797969
+vt 0.588388 0.797969
+vt 0.526418 0.642681
+vt 0.578240 0.797969
+vt 0.534855 0.637043
+vt 0.568093 0.797969
+vt 0.544230 0.633160
+vt 0.554183 0.631180
+vt 0.507028 0.737458
+vt 0.557945 0.797969
+vt 0.480393 0.735834
+vt 0.547798 0.773289
+vt 0.470441 0.733854
+vt 0.537650 0.773289
+vt 0.461066 0.729971
+vt 0.527503 0.773289
+vt 0.452629 0.724334
+vt 0.517355 0.773289
+vt 0.507208 0.773290
+vt 0.445453 0.717158
+vt 0.497060 0.773290
+vt 0.439816 0.708721
+vt 0.486913 0.773290
+vt 0.435932 0.699346
+vt 0.433953 0.689393
+vt 0.476765 0.773290
+vt 0.433953 0.679246
+vt 0.466618 0.773290
+vt 0.435932 0.669293
+vt 0.456470 0.773290
+vt 0.439815 0.659918
+vt 0.446323 0.773289
+vt 0.436175 0.773289
+vt 0.445453 0.651481
+vt 0.426028 0.773289
+vt 0.452628 0.644305
+vt 0.415880 0.773289
+vt 0.461066 0.638668
+vt 0.470441 0.634784
+vt 0.405733 0.773289
+vt 0.480393 0.632805
+vt 0.395585 0.773289
+vt 0.507028 0.631180
+vt 0.554183 0.734210
+vt 0.385438 0.797969
+vt 0.544230 0.732230
+vt 0.375290 0.797969
+vt 0.534855 0.728347
+vt 0.365143 0.797969
+vt 0.354995 0.797969
+vt 0.526418 0.722709
+vt 0.344848 0.797969
+vt 0.519243 0.715534
+vt 0.334700 0.797969
+vt 0.513605 0.707096
+vt 0.509721 0.697721
+vt 0.324553 0.797969
+vt 0.507742 0.687769
+vt 0.314405 0.797969
+vt 0.639125 0.797969
+vt 0.908268 0.746864
+vt 0.874210 0.774814
+vt 0.334815 0.932388
+vt 0.611202 0.932388
+vt 0.869411 0.767633
+vt 0.602565 0.932389
+vt 0.866106 0.759653
+vt 0.593928 0.932389
+vt 0.864421 0.751182
+vt 0.585291 0.932389
+vt 0.864421 0.742545
+vt 0.576654 0.932389
+vt 0.866106 0.734074
+vt 0.568017 0.932389
+vt 0.869412 0.726094
+vt 0.559379 0.932389
+vt 0.874210 0.718913
+vt 0.550742 0.932389
+vt 0.880318 0.712805
+vt 0.542105 0.932389
+vt 0.887499 0.708007
+vt 0.533468 0.932389
+vt 0.895479 0.704702
+vt 0.524831 0.932389
+vt 0.903950 0.703017
+vt 0.516194 0.932389
+vt 0.912587 0.703017
+vt 0.507557 0.932389
+vt 0.921058 0.704702
+vt 0.498920 0.932389
+vt 0.929038 0.708007
+vt 0.490283 0.932389
+vt 0.936219 0.712806
+vt 0.481646 0.932389
+vt 0.942326 0.718913
+vt 0.473009 0.932389
+vt 0.947125 0.726095
+vt 0.464371 0.932389
+vt 0.950430 0.734074
+vt 0.455734 0.932389
+vt 0.952115 0.742545
+vt 0.447097 0.932389
+vt 0.952115 0.751182
+vt 0.438460 0.932389
+vt 0.950430 0.759654
+vt 0.429823 0.932389
+vt 0.947125 0.767633
+vt 0.421186 0.932389
+vt 0.942326 0.774815
+vt 0.412549 0.932389
+vt 0.936219 0.780922
+vt 0.403912 0.932389
+vt 0.929037 0.785721
+vt 0.395275 0.932389
+vt 0.921058 0.789026
+vt 0.386637 0.932389
+vt 0.912586 0.790711
+vt 0.378001 0.932389
+vt 0.903949 0.790711
+vt 0.369363 0.932389
+vt 0.895478 0.789025
+vt 0.360726 0.932389
+vt 0.887499 0.785720
+vt 0.352089 0.932388
+vt 0.880317 0.780922
+vt 0.343452 0.932388
+vt 0.618830 0.816892
+vt 0.628978 0.816892
+vt 0.608683 0.816892
+vt 0.598535 0.816892
+vt 0.588388 0.816892
+vt 0.578240 0.816892
+vt 0.568093 0.816892
+vt 0.557945 0.816892
+vt 0.547798 0.816892
+vt 0.537650 0.816892
+vt 0.527503 0.816892
+vt 0.517355 0.816892
+vt 0.507208 0.816892
+vt 0.497060 0.816892
+vt 0.486913 0.816892
+vt 0.476765 0.816892
+vt 0.466618 0.816892
+vt 0.456470 0.816892
+vt 0.446323 0.816892
+vt 0.436175 0.816892
+vt 0.426028 0.816892
+vt 0.415880 0.816892
+vt 0.405733 0.816892
+vt 0.395585 0.816892
+vt 0.385438 0.816892
+vt 0.375290 0.816892
+vt 0.365143 0.816892
+vt 0.354995 0.816892
+vt 0.344848 0.816892
+vt 0.334700 0.816892
+vt 0.324553 0.816892
+vt 0.314405 0.816892
+vt 0.639125 0.816892
+vt 0.517095 0.669905
+vt 0.602565 0.946486
+vt 0.334815 0.946485
+vt 0.515410 0.678377
+vt 0.611202 0.946485
+vt 0.520400 0.661926
+vt 0.593928 0.946485
+vt 0.525198 0.654744
+vt 0.585291 0.946486
+vt 0.531306 0.648637
+vt 0.576654 0.946486
+vt 0.568017 0.946486
+vt 0.538487 0.643839
+vt 0.546467 0.640533
+vt 0.559379 0.946486
+vt 0.554938 0.638848
+vt 0.507247 0.729756
+vt 0.550742 0.946486
+vt 0.481149 0.728166
+vt 0.542105 0.971165
+vt 0.472678 0.726481
+vt 0.533468 0.971165
+vt 0.464698 0.723176
+vt 0.524831 0.971165
+vt 0.457517 0.718377
+vt 0.516194 0.971165
+vt 0.507557 0.971165
+vt 0.451409 0.712270
+vt 0.446611 0.705089
+vt 0.498920 0.971165
+vt 0.443305 0.697109
+vt 0.490283 0.971165
+vt 0.441620 0.688638
+vt 0.481646 0.971165
+vt 0.473008 0.971165
+vt 0.441620 0.680001
+vt 0.443305 0.671530
+vt 0.464371 0.971165
+vt 0.446610 0.663550
+vt 0.455734 0.971165
+vt 0.451409 0.656369
+vt 0.447097 0.971165
+vt 0.438460 0.971165
+vt 0.457516 0.650261
+vt 0.464698 0.645463
+vt 0.429823 0.971165
+vt 0.472677 0.642157
+vt 0.421186 0.971165
+vt 0.412549 0.971165
+vt 0.481149 0.640472
+vt 0.507247 0.638882
+vt 0.554938 0.726542
+vt 0.403912 0.946486
+vt 0.546467 0.724857
+vt 0.395275 0.946486
+vt 0.538487 0.721552
+vt 0.386637 0.946486
+vt 0.378001 0.946486
+vt 0.531306 0.716753
+vt 0.525198 0.710646
+vt 0.369363 0.946486
+vt 0.520400 0.703464
+vt 0.360726 0.946485
+vt 0.517095 0.695485
+vt 0.352089 0.946485
+vt 0.343452 0.946485
+vt 0.515410 0.687014
+vt 0.509722 0.667669
+vt 0.618830 0.797969
+vt 0.507742 0.677621
+vt 0.628978 0.797969
+vt 0.513605 0.658294
+vt 0.608683 0.797969
+vt 0.519242 0.649857
+vt 0.598535 0.797969
+vt 0.588388 0.797969
+vt 0.526418 0.642681
+vt 0.578240 0.797969
+vt 0.534855 0.637043
+vt 0.568093 0.797969
+vt 0.544230 0.633160
+vt 0.554183 0.631180
+vt 0.507028 0.737458
+vt 0.557945 0.797969
+vt 0.480393 0.735834
+vt 0.547798 0.773289
+vt 0.470441 0.733854
+vt 0.537650 0.773289
+vt 0.461066 0.729971
+vt 0.527503 0.773289
+vt 0.452629 0.724334
+vt 0.517355 0.773289
+vt 0.507208 0.773290
+vt 0.445453 0.717158
+vt 0.497060 0.773290
+vt 0.439816 0.708721
+vt 0.486913 0.773290
+vt 0.435932 0.699346
+vt 0.433953 0.689393
+vt 0.476765 0.773290
+vt 0.433953 0.679246
+vt 0.466618 0.773290
+vt 0.435932 0.669293
+vt 0.456470 0.773290
+vt 0.439815 0.659918
+vt 0.446323 0.773289
+vt 0.436175 0.773289
+vt 0.445453 0.651481
+vt 0.426028 0.773289
+vt 0.452628 0.644305
+vt 0.415880 0.773289
+vt 0.461066 0.638668
+vt 0.470441 0.634784
+vt 0.405733 0.773289
+vt 0.480393 0.632805
+vt 0.395585 0.773289
+vt 0.507028 0.631180
+vt 0.554183 0.734210
+vt 0.385438 0.797969
+vt 0.544230 0.732230
+vt 0.375290 0.797969
+vt 0.534855 0.728347
+vt 0.365143 0.797969
+vt 0.354995 0.797969
+vt 0.526418 0.722709
+vt 0.344848 0.797969
+vt 0.519243 0.715534
+vt 0.334700 0.797969
+vt 0.513605 0.707096
+vt 0.509721 0.697721
+vt 0.324553 0.797969
+vt 0.507742 0.687769
+vt 0.314405 0.797969
+vt 0.639125 0.797969
+s 0
+g Cube_Material
+usemtl Material
+f 6/11/1 2/3/1 10/18/1 14/22/1
+f 1/1/2 5/9/2 13/21/2 9/17/2
+f 12/20/3 11/19/3 15/23/3 16/24/3
+f 10/18/3 9/17/3 13/21/3 14/22/3
+f 3/5/2 7/13/2 15/23/2 11/19/2
+f 8/15/1 4/7/1 12/20/1 16/24/1
+f 7/14/4 8/16/4 16/24/4 15/23/4
+f 5/10/4 6/12/4 14/22/4 13/21/4
+f 4/8/5 3/6/5 11/19/5 12/20/5
+f 2/4/5 1/2/5 9/17/5 10/18/5
+f 18/29/3 20/34/3 19/33/3 17/25/3
+f 33/57/4 35/59/4 36/60/4 34/58/4
+f 24/44/6 22/39/6 21/35/6 23/43/6
+f 25/45/1 28/51/1 32/56/1 29/53/1
+f 17/26/1 19/33/1 23/43/1 21/36/1
+f 22/40/2 24/44/2 20/34/2 18/30/2
+f 25/46/5 26/48/5 18/31/5 17/27/5
+f 27/50/5 28/52/5 21/37/5 22/41/5
+f 26/47/5 27/49/5 22/42/5 18/32/5
+f 28/51/5 25/45/5 17/28/5 21/38/5
+f 31/55/5 30/54/5 29/53/5 32/56/5
+f 27/49/2 26/47/2 30/54/2 31/55/2
+f 28/52/6 27/50/6 31/55/6 32/56/6
+f 26/48/3 25/46/3 29/53/3 30/54/3
+f 33/57/4 34/58/4 19/33/4 20/34/4
+f 35/59/4 33/57/4 20/34/4 24/44/4
+f 36/60/4 35/59/4 24/44/4 23/43/4
+f 34/58/4 36/60/4 23/43/4 19/33/4
+f 37/61/4 38/62/4 39/65/4
+f 37/61/4 39/65/4 40/67/4
+f 37/61/4 40/67/4 41/69/4
+f 37/61/4 41/69/4 42/71/4
+f 37/61/4 42/71/4 43/73/4
+f 37/61/4 43/73/4 44/75/4
+f 37/61/4 44/75/4 45/77/4
+f 37/61/4 45/77/4 46/79/4
+f 37/61/4 46/79/4 47/81/4
+f 37/61/4 47/81/4 48/83/4
+f 37/61/4 48/83/4 49/85/4
+f 37/61/4 49/85/4 50/87/4
+f 37/61/4 50/87/4 51/89/4
+f 37/61/4 51/89/4 52/91/4
+f 37/61/4 52/91/4 53/93/4
+f 37/61/4 53/93/4 54/95/4
+f 37/61/4 54/95/4 55/97/4
+f 37/61/4 55/97/4 56/99/4
+f 37/61/4 56/99/4 57/101/4
+f 37/61/4 57/101/4 58/103/4
+f 37/61/4 58/103/4 59/105/4
+f 37/61/4 59/105/4 60/107/4
+f 37/61/4 60/107/4 61/109/4
+f 37/61/4 61/109/4 62/111/4
+f 37/61/4 62/111/4 63/113/4
+f 37/61/4 63/113/4 64/115/4
+f 37/61/4 64/115/4 65/117/4
+f 37/61/4 65/117/4 66/119/4
+f 37/61/4 66/119/4 67/121/4
+f 37/61/4 67/121/4 68/123/4
+f 37/61/4 68/123/4 69/125/4
+f 37/61/4 69/125/4 38/62/4
+f 99/156/7 100/157/7 164/290/7 163/287/7
+f 84/141/8 85/142/8 149/259/8 148/256/8
+f 59/106/9 58/104/9 122/202/9 123/205/9
+f 38/63/10 69/126/10 133/225/10 103/162/10
+f 44/76/11 43/74/11 107/171/11 108/174/11
+f 91/148/11 92/149/11 156/273/11 155/270/11
+f 76/133/12 77/134/12 141/243/12 140/239/12
+f 55/98/13 54/96/13 118/194/13 119/197/13
+f 66/120/14 65/118/14 129/217/14 130/220/14
+f 98/155/15 99/156/15 163/287/15 162/285/15
+f 83/140/16 84/141/16 148/256/16 147/254/16
+f 40/68/17 39/66/17 102/161/17 104/166/17
+f 51/90/15 50/88/15 114/186/15 115/189/15
+f 62/112/18 61/110/18 125/208/18 126/212/18
+f 90/147/19 91/148/19 155/270/19 154/268/19
+f 75/132/20 76/133/20 140/239/20 139/237/20
+f 47/82/21 46/80/21 110/179/21 111/181/21
+f 97/154/22 98/155/22 162/285/22 161/283/22
+f 82/139/23 83/140/23 147/254/23 146/252/23
+f 58/104/24 57/102/24 121/201/24 122/202/24
+f 69/126/8 68/124/8 132/224/8 133/225/8
+f 89/146/25 90/147/25 154/268/25 153/266/25
+f 43/74/19 42/72/19 106/170/19 107/171/19
+f 54/96/26 53/94/26 117/193/26 118/194/26
+f 74/131/9 75/132/9 139/237/9 138/235/9
+f 65/118/27 64/116/27 128/216/27 129/217/27
+f 96/153/28 97/154/28 161/283/28 160/282/28
+f 81/138/29 82/139/29 146/252/29 145/251/29
+f 50/88/22 49/86/22 113/185/22 114/186/22
+f 88/145/30 89/146/30 153/266/30 152/265/30
+f 61/110/12 60/108/12 124/207/12 125/208/12
+f 73/130/24 74/131/24 138/235/24 137/234/24
+f 102/160/4 103/163/4 135/229/4 134/227/4
+f 104/165/4 102/160/4 134/227/4 136/231/4
+f 105/167/4 104/165/4 136/231/4 137/233/4
+f 106/169/4 105/167/4 137/233/4 138/236/4
+f 107/172/4 106/169/4 138/236/4 139/238/4
+f 108/173/4 107/172/4 139/238/4 140/240/4
+f 109/175/4 108/173/4 140/240/4 141/241/4
+f 110/178/31 109/176/31 141/242/31 142/244/31
+f 111/180/4 110/178/4 142/244/4 143/246/4
+f 112/182/4 111/180/4 143/246/4 144/248/4
+f 113/184/4 112/182/4 144/248/4 145/250/4
+f 114/187/4 113/184/4 145/250/4 146/253/4
+f 115/188/4 114/187/4 146/253/4 147/255/4
+f 116/190/4 115/188/4 147/255/4 148/257/4
+f 117/192/4 116/190/4 148/257/4 149/258/4
+f 118/195/4 117/192/4 149/258/4 150/260/4
+f 119/196/4 118/195/4 150/260/4 151/262/4
+f 120/198/4 119/196/4 151/262/4 152/264/4
+f 121/200/4 120/198/4 152/264/4 153/267/4
+f 122/203/4 121/200/4 153/267/4 154/269/4
+f 123/204/4 122/203/4 154/269/4 155/271/4
+f 124/206/4 123/204/4 155/271/4 156/272/4
+f 125/209/4 124/206/4 156/272/4 157/274/4
+f 126/210/31 125/209/31 157/274/31 158/276/31
+f 127/213/4 126/211/4 158/277/4 159/279/4
+f 128/215/4 127/213/4 159/279/4 160/281/4
+f 129/218/4 128/215/4 160/281/4 161/284/4
+f 130/219/4 129/218/4 161/284/4 162/286/4
+f 131/221/4 130/219/4 162/286/4 163/288/4
+f 132/223/4 131/221/4 163/288/4 164/289/4
+f 133/226/4 132/223/4 164/289/4 165/291/4
+f 103/163/4 133/226/4 165/291/4 135/229/4
+f 48/84/32 47/82/32 111/181/32 112/183/32
+f 77/134/33 78/135/33 142/245/33 141/243/33
+f 92/149/34 93/150/34 157/275/34 156/273/34
+f 63/114/35 62/112/35 126/212/35 127/214/35
+f 52/92/7 51/90/7 115/189/7 116/191/7
+f 71/128/36 70/127/36 134/228/36 135/230/36
+f 41/70/30 40/68/30 104/166/30 105/168/30
+f 85/142/10 86/143/10 150/261/10 149/259/10
+f 100/157/37 101/158/37 165/292/37 164/290/37
+f 67/122/23 66/120/23 130/220/23 131/222/23
+f 56/100/38 55/98/38 119/197/38 120/199/38
+f 78/135/35 79/136/35 143/247/35 142/245/35
+f 93/150/39 94/151/39 158/278/39 157/275/39
+f 45/78/34 44/76/34 108/174/34 109/177/34
+f 70/127/38 72/129/38 136/232/38 134/228/38
+f 60/108/20 59/106/20 123/205/20 124/207/20
+f 86/143/40 87/144/40 151/263/40 150/261/40
+f 101/159/26 71/128/26 135/230/26 165/293/26
+f 49/86/28 48/84/28 112/183/28 113/185/28
+f 79/136/41 80/137/41 144/249/41 143/247/41
+f 94/151/21 95/152/21 159/280/21 158/278/21
+f 64/116/41 63/114/41 127/214/41 128/216/41
+f 53/94/37 52/92/37 116/191/37 117/193/37
+f 72/129/42 73/130/42 137/234/42 136/232/42
+f 42/72/43 41/70/43 105/168/43 106/170/43
+f 87/144/17 88/145/17 152/265/17 151/263/17
+f 68/124/16 67/122/16 131/222/16 132/224/16
+f 39/66/40 38/64/40 103/164/40 102/161/40
+f 57/102/42 56/100/42 120/199/42 121/201/42
+f 80/137/27 81/138/27 145/251/27 144/249/27
+f 95/152/32 96/153/32 160/282/32 159/280/32
+f 46/80/44 45/78/44 109/177/44 110/179/44
+f 166/294/4 167/295/4 168/298/4
+f 166/294/4 168/298/4 169/300/4
+f 166/294/4 169/300/4 170/302/4
+f 166/294/4 170/302/4 171/304/4
+f 166/294/4 171/304/4 172/306/4
+f 166/294/4 172/306/4 173/308/4
+f 166/294/4 173/308/4 174/310/4
+f 166/294/4 174/310/4 175/312/4
+f 166/294/4 175/312/4 176/314/4
+f 166/294/4 176/314/4 177/316/4
+f 166/294/4 177/316/4 178/318/4
+f 166/294/4 178/318/4 179/320/4
+f 166/294/4 179/320/4 180/322/4
+f 166/294/4 180/322/4 181/324/4
+f 166/294/4 181/324/4 182/326/4
+f 166/294/4 182/326/4 183/328/4
+f 166/294/4 183/328/4 184/330/4
+f 166/294/4 184/330/4 185/332/4
+f 166/294/4 185/332/4 186/334/4
+f 166/294/4 186/334/4 187/336/4
+f 166/294/4 187/336/4 188/338/4
+f 166/294/4 188/338/4 189/340/4
+f 166/294/4 189/340/4 190/342/4
+f 166/294/4 190/342/4 191/344/4
+f 166/294/4 191/344/4 192/346/4
+f 166/294/4 192/346/4 193/348/4
+f 166/294/4 193/348/4 194/350/4
+f 166/294/4 194/350/4 195/352/4
+f 166/294/4 195/352/4 196/354/4
+f 166/294/4 196/354/4 197/356/4
+f 166/294/4 197/356/4 198/358/4
+f 166/294/4 198/358/4 167/295/4
+f 228/389/7 229/390/7 293/523/7 292/520/7
+f 213/374/8 214/375/8 278/492/8 277/489/8
+f 188/339/9 187/337/9 251/435/9 252/438/9
+f 167/296/10 198/359/10 262/458/10 232/395/10
+f 173/309/11 172/307/11 236/404/11 237/407/11
+f 220/381/11 221/382/11 285/506/11 284/503/11
+f 205/366/12 206/367/12 270/476/12 269/472/12
+f 184/331/13 183/329/13 247/427/13 248/430/13
+f 195/353/14 194/351/14 258/450/14 259/453/14
+f 227/388/15 228/389/15 292/520/15 291/518/15
+f 212/373/16 213/374/16 277/489/16 276/487/16
+f 169/301/17 168/299/17 231/394/17 233/399/17
+f 180/323/15 179/321/15 243/419/15 244/422/15
+f 191/345/18 190/343/18 254/441/18 255/445/18
+f 219/380/19 220/381/19 284/503/19 283/501/19
+f 204/365/20 205/366/20 269/472/20 268/470/20
+f 176/315/21 175/313/21 239/412/21 240/414/21
+f 226/387/22 227/388/22 291/518/22 290/516/22
+f 211/372/23 212/373/23 276/487/23 275/485/23
+f 187/337/24 186/335/24 250/434/24 251/435/24
+f 198/359/8 197/357/8 261/457/8 262/458/8
+f 218/379/25 219/380/25 283/501/25 282/499/25
+f 172/307/19 171/305/19 235/403/19 236/404/19
+f 183/329/26 182/327/26 246/426/26 247/427/26
+f 203/364/9 204/365/9 268/470/9 267/468/9
+f 194/351/27 193/349/27 257/449/27 258/450/27
+f 225/386/28 226/387/28 290/516/28 289/515/28
+f 210/371/29 211/372/29 275/485/29 274/484/29
+f 179/321/22 178/319/22 242/418/22 243/419/22
+f 217/378/30 218/379/30 282/499/30 281/498/30
+f 190/343/12 189/341/12 253/440/12 254/441/12
+f 202/363/24 203/364/24 267/468/24 266/467/24
+f 231/393/4 232/396/4 264/462/4 263/460/4
+f 233/398/4 231/393/4 263/460/4 265/464/4
+f 234/400/4 233/398/4 265/464/4 266/466/4
+f 235/402/4 234/400/4 266/466/4 267/469/4
+f 236/405/4 235/402/4 267/469/4 268/471/4
+f 237/406/4 236/405/4 268/471/4 269/473/4
+f 238/408/4 237/406/4 269/473/4 270/474/4
+f 239/411/31 238/409/31 270/475/31 271/477/31
+f 240/413/4 239/411/4 271/477/4 272/479/4
+f 241/415/4 240/413/4 272/479/4 273/481/4
+f 242/417/4 241/415/4 273/481/4 274/483/4
+f 243/420/4 242/417/4 274/483/4 275/486/4
+f 244/421/4 243/420/4 275/486/4 276/488/4
+f 245/423/4 244/421/4 276/488/4 277/490/4
+f 246/425/4 245/423/4 277/490/4 278/491/4
+f 247/428/4 246/425/4 278/491/4 279/493/4
+f 248/429/4 247/428/4 279/493/4 280/495/4
+f 249/431/4 248/429/4 280/495/4 281/497/4
+f 250/433/4 249/431/4 281/497/4 282/500/4
+f 251/436/4 250/433/4 282/500/4 283/502/4
+f 252/437/4 251/436/4 283/502/4 284/504/4
+f 253/439/4 252/437/4 284/504/4 285/505/4
+f 254/442/4 253/439/4 285/505/4 286/507/4
+f 255/443/31 254/442/31 286/507/31 287/509/31
+f 256/446/4 255/444/4 287/510/4 288/512/4
+f 257/448/4 256/446/4 288/512/4 289/514/4
+f 258/451/4 257/448/4 289/514/4 290/517/4
+f 259/452/4 258/451/4 290/517/4 291/519/4
+f 260/454/4 259/452/4 291/519/4 292/521/4
+f 261/456/4 260/454/4 292/521/4 293/522/4
+f 262/459/4 261/456/4 293/522/4 294/524/4
+f 232/396/4 262/459/4 294/524/4 264/462/4
+f 177/317/32 176/315/32 240/414/32 241/416/32
+f 206/367/33 207/368/33 271/478/33 270/476/33
+f 221/382/34 222/383/34 286/508/34 285/506/34
+f 192/347/35 191/345/35 255/445/35 256/447/35
+f 181/325/7 180/323/7 244/422/7 245/424/7
+f 200/361/36 199/360/36 263/461/36 264/463/36
+f 170/303/30 169/301/30 233/399/30 234/401/30
+f 214/375/10 215/376/10 279/494/10 278/492/10
+f 229/390/37 230/391/37 294/525/37 293/523/37
+f 196/355/23 195/353/23 259/453/23 260/455/23
+f 185/333/38 184/331/38 248/430/38 249/432/38
+f 207/368/35 208/369/35 272/480/35 271/478/35
+f 222/383/39 223/384/39 287/511/39 286/508/39
+f 174/311/34 173/309/34 237/407/34 238/410/34
+f 199/360/38 201/362/38 265/465/38 263/461/38
+f 189/341/20 188/339/20 252/438/20 253/440/20
+f 215/376/40 216/377/40 280/496/40 279/494/40
+f 230/392/26 200/361/26 264/463/26 294/526/26
+f 178/319/28 177/317/28 241/416/28 242/418/28
+f 208/369/41 209/370/41 273/482/41 272/480/41
+f 223/384/21 224/385/21 288/513/21 287/511/21
+f 193/349/41 192/347/41 256/447/41 257/449/41
+f 182/327/37 181/325/37 245/424/37 246/426/37
+f 201/362/42 202/363/42 266/467/42 265/465/42
+f 171/305/43 170/303/43 234/401/43 235/403/43
+f 216/377/17 217/378/17 281/498/17 280/496/17
+f 197/357/16 196/355/16 260/455/16 261/457/16
+f 168/299/40 167/297/40 232/397/40 231/394/40
+f 186/335/42 185/333/42 249/432/42 250/434/42
+f 209/370/27 210/371/27 274/484/27 273/482/27
+f 224/385/32 225/386/32 289/515/32 288/513/32
+f 175/313/44 174/311/44 238/410/44 239/412/44
+f 295/527/4 296/528/4 297/531/4
+f 295/527/4 297/531/4 298/533/4
+f 295/527/4 298/533/4 299/535/4
+f 295/527/4 299/535/4 300/537/4
+f 295/527/4 300/537/4 301/539/4
+f 295/527/4 301/539/4 302/541/4
+f 295/527/4 302/541/4 303/543/4
+f 295/527/4 303/543/4 304/545/4
+f 295/527/4 304/545/4 305/547/4
+f 295/527/4 305/547/4 306/549/4
+f 295/527/4 306/549/4 307/551/4
+f 295/527/4 307/551/4 308/553/4
+f 295/527/4 308/553/4 309/555/4
+f 295/527/4 309/555/4 310/557/4
+f 295/527/4 310/557/4 311/559/4
+f 295/527/4 311/559/4 312/561/4
+f 295/527/4 312/561/4 313/563/4
+f 295/527/4 313/563/4 314/565/4
+f 295/527/4 314/565/4 315/567/4
+f 295/527/4 315/567/4 316/569/4
+f 295/527/4 316/569/4 317/571/4
+f 295/527/4 317/571/4 318/573/4
+f 295/527/4 318/573/4 319/575/4
+f 295/527/4 319/575/4 320/577/4
+f 295/527/4 320/577/4 321/579/4
+f 295/527/4 321/579/4 322/581/4
+f 295/527/4 322/581/4 323/583/4
+f 295/527/4 323/583/4 324/585/4
+f 295/527/4 324/585/4 325/587/4
+f 295/527/4 325/587/4 326/589/4
+f 295/527/4 326/589/4 327/591/4
+f 295/527/4 327/591/4 296/528/4
+f 357/622/7 358/623/7 422/756/7 421/753/7
+f 342/607/8 343/608/8 407/725/8 406/722/8
+f 317/572/9 316/570/9 380/668/9 381/671/9
+f 296/529/10 327/592/10 391/691/10 361/628/10
+f 302/542/11 301/540/11 365/637/11 366/640/11
+f 349/614/11 350/615/11 414/739/11 413/736/11
+f 334/599/12 335/600/12 399/709/12 398/705/12
+f 313/564/13 312/562/13 376/660/13 377/663/13
+f 324/586/14 323/584/14 387/683/14 388/686/14
+f 356/621/15 357/622/15 421/753/15 420/751/15
+f 341/606/16 342/607/16 406/722/16 405/720/16
+f 298/534/17 297/532/17 360/627/17 362/632/17
+f 309/556/15 308/554/15 372/652/15 373/655/15
+f 320/578/18 319/576/18 383/674/18 384/678/18
+f 348/613/19 349/614/19 413/736/19 412/734/19
+f 333/598/20 334/599/20 398/705/20 397/703/20
+f 305/548/21 304/546/21 368/645/21 369/647/21
+f 355/620/45 356/621/45 420/751/45 419/749/45
+f 340/605/23 341/606/23 405/720/23 404/718/23
+f 316/570/24 315/568/24 379/667/24 380/668/24
+f 327/592/8 326/590/8 390/690/8 391/691/8
+f 347/612/25 348/613/25 412/734/25 411/732/25
+f 301/540/19 300/538/19 364/636/19 365/637/19
+f 312/562/26 311/560/26 375/659/26 376/660/26
+f 332/597/9 333/598/9 397/703/9 396/701/9
+f 323/584/27 322/582/27 386/682/27 387/683/27
+f 354/619/28 355/620/28 419/749/28 418/748/28
+f 339/604/29 340/605/29 404/718/29 403/717/29
+f 308/554/22 307/552/22 371/651/22 372/652/22
+f 346/611/30 347/612/30 411/732/30 410/731/30
+f 319/576/12 318/574/12 382/673/12 383/674/12
+f 331/596/24 332/597/24 396/701/24 395/700/24
+f 360/626/4 361/629/4 393/695/4 392/693/4
+f 362/631/4 360/626/4 392/693/4 394/697/4
+f 363/633/4 362/631/4 394/697/4 395/699/4
+f 364/635/4 363/633/4 395/699/4 396/702/4
+f 365/638/4 364/635/4 396/702/4 397/704/4
+f 366/639/4 365/638/4 397/704/4 398/706/4
+f 367/641/4 366/639/4 398/706/4 399/707/4
+f 368/644/31 367/642/31 399/708/31 400/710/31
+f 369/646/4 368/644/4 400/710/4 401/712/4
+f 370/648/4 369/646/4 401/712/4 402/714/4
+f 371/650/4 370/648/4 402/714/4 403/716/4
+f 372/653/4 371/650/4 403/716/4 404/719/4
+f 373/654/4 372/653/4 404/719/4 405/721/4
+f 374/656/4 373/654/4 405/721/4 406/723/4
+f 375/658/4 374/656/4 406/723/4 407/724/4
+f 376/661/4 375/658/4 407/724/4 408/726/4
+f 377/662/4 376/661/4 408/726/4 409/728/4
+f 378/664/4 377/662/4 409/728/4 410/730/4
+f 379/666/4 378/664/4 410/730/4 411/733/4
+f 380/669/4 379/666/4 411/733/4 412/735/4
+f 381/670/4 380/669/4 412/735/4 413/737/4
+f 382/672/4 381/670/4 413/737/4 414/738/4
+f 383/675/4 382/672/4 414/738/4 415/740/4
+f 384/676/31 383/675/31 415/740/31 416/742/31
+f 385/679/4 384/677/4 416/743/4 417/745/4
+f 386/681/4 385/679/4 417/745/4 418/747/4
+f 387/684/4 386/681/4 418/747/4 419/750/4
+f 388/685/4 387/684/4 419/750/4 420/752/4
+f 389/687/4 388/685/4 420/752/4 421/754/4
+f 390/689/4 389/687/4 421/754/4 422/755/4
+f 391/692/4 390/689/4 422/755/4 423/757/4
+f 361/629/4 391/692/4 423/757/4 393/695/4
+f 306/550/32 305/548/32 369/647/32 370/649/32
+f 335/600/33 336/601/33 400/711/33 399/709/33
+f 350/615/34 351/616/34 415/741/34 414/739/34
+f 321/580/35 320/578/35 384/678/35 385/680/35
+f 310/558/7 309/556/7 373/655/7 374/657/7
+f 329/594/36 328/593/36 392/694/36 393/696/36
+f 299/536/30 298/534/30 362/632/30 363/634/30
+f 343/608/10 344/609/10 408/727/10 407/725/10
+f 358/623/37 359/624/37 423/758/37 422/756/37
+f 325/588/23 324/586/23 388/686/23 389/688/23
+f 314/566/38 313/564/38 377/663/38 378/665/38
+f 336/601/35 337/602/35 401/713/35 400/711/35
+f 351/616/39 352/617/39 416/744/39 415/741/39
+f 303/544/34 302/542/34 366/640/34 367/643/34
+f 328/593/38 330/595/38 394/698/38 392/694/38
+f 318/574/20 317/572/20 381/671/20 382/673/20
+f 344/609/40 345/610/40 409/729/40 408/727/40
+f 359/625/26 329/594/26 393/696/26 423/759/26
+f 307/552/28 306/550/28 370/649/28 371/651/28
+f 337/602/41 338/603/41 402/715/41 401/713/41
+f 352/617/21 353/618/21 417/746/21 416/744/21
+f 322/582/41 321/580/41 385/680/41 386/682/41
+f 311/560/37 310/558/37 374/657/37 375/659/37
+f 330/595/42 331/596/42 395/700/42 394/698/42
+f 300/538/43 299/536/43 363/634/43 364/636/43
+f 345/610/17 346/611/17 410/731/17 409/729/17
+f 326/590/16 325/588/16 389/688/16 390/690/16
+f 297/532/40 296/530/40 361/630/40 360/627/40
+f 315/568/42 314/566/42 378/665/42 379/667/42
+f 338/603/27 339/604/27 403/717/27 402/715/27
+f 353/618/32 354/619/32 418/748/32 417/746/32
+f 304/546/44 303/544/44 367/643/44 368/645/44
+f 424/760/4 425/761/4 426/764/4
+f 424/760/4 426/764/4 427/766/4
+f 424/760/4 427/766/4 428/768/4
+f 424/760/4 428/768/4 429/770/4
+f 424/760/4 429/770/4 430/772/4
+f 424/760/4 430/772/4 431/774/4
+f 424/760/4 431/774/4 432/776/4
+f 424/760/4 432/776/4 433/778/4
+f 424/760/4 433/778/4 434/780/4
+f 424/760/4 434/780/4 435/782/4
+f 424/760/4 435/782/4 436/784/4
+f 424/760/4 436/784/4 437/786/4
+f 424/760/4 437/786/4 438/788/4
+f 424/760/4 438/788/4 439/790/4
+f 424/760/4 439/790/4 440/792/4
+f 424/760/4 440/792/4 441/794/4
+f 424/760/4 441/794/4 442/796/4
+f 424/760/4 442/796/4 443/798/4
+f 424/760/4 443/798/4 444/800/4
+f 424/760/4 444/800/4 445/802/4
+f 424/760/4 445/802/4 446/804/4
+f 424/760/4 446/804/4 447/806/4
+f 424/760/4 447/806/4 448/808/4
+f 424/760/4 448/808/4 449/810/4
+f 424/760/4 449/810/4 450/812/4
+f 424/760/4 450/812/4 451/814/4
+f 424/760/4 451/814/4 452/816/4
+f 424/760/4 452/816/4 453/818/4
+f 424/760/4 453/818/4 454/820/4
+f 424/760/4 454/820/4 455/822/4
+f 424/760/4 455/822/4 456/824/4
+f 424/760/4 456/824/4 425/761/4
+f 486/855/7 487/856/7 551/989/7 550/986/7
+f 471/840/8 472/841/8 536/958/8 535/955/8
+f 446/805/9 445/803/9 509/901/9 510/904/9
+f 425/762/46 456/825/46 520/924/46 490/861/46
+f 431/775/11 430/773/11 494/870/11 495/873/11
+f 478/847/11 479/848/11 543/972/11 542/969/11
+f 463/832/12 464/833/12 528/942/12 527/938/12
+f 442/797/13 441/795/13 505/893/13 506/896/13
+f 453/819/14 452/817/14 516/916/14 517/919/14
+f 485/854/15 486/855/15 550/986/15 549/984/15
+f 470/839/16 471/840/16 535/955/16 534/953/16
+f 427/767/17 426/765/17 489/860/17 491/865/17
+f 438/789/15 437/787/15 501/885/15 502/888/15
+f 449/811/18 448/809/18 512/907/18 513/911/18
+f 477/846/19 478/847/19 542/969/19 541/967/19
+f 462/831/20 463/832/20 527/938/20 526/936/20
+f 434/781/21 433/779/21 497/878/21 498/880/21
+f 484/853/22 485/854/22 549/984/22 548/982/22
+f 469/838/23 470/839/23 534/953/23 533/951/23
+f 445/803/24 444/801/24 508/900/24 509/901/24
+f 456/825/8 455/823/8 519/923/8 520/924/8
+f 476/845/25 477/846/25 541/967/25 540/965/25
+f 430/773/19 429/771/19 493/869/19 494/870/19
+f 441/795/26 440/793/26 504/892/26 505/893/26
+f 461/830/9 462/831/9 526/936/9 525/934/9
+f 452/817/27 451/815/27 515/915/27 516/916/27
+f 483/852/28 484/853/28 548/982/28 547/981/28
+f 468/837/29 469/838/29 533/951/29 532/950/29
+f 437/787/22 436/785/22 500/884/22 501/885/22
+f 475/844/30 476/845/30 540/965/30 539/964/30
+f 448/809/12 447/807/12 511/906/12 512/907/12
+f 460/829/24 461/830/24 525/934/24 524/933/24
+f 489/859/4 490/862/4 522/928/4 521/926/4
+f 491/864/4 489/859/4 521/926/4 523/930/4
+f 492/866/4 491/864/4 523/930/4 524/932/4
+f 493/868/4 492/866/4 524/932/4 525/935/4
+f 494/871/4 493/868/4 525/935/4 526/937/4
+f 495/872/4 494/871/4 526/937/4 527/939/4
+f 496/874/4 495/872/4 527/939/4 528/940/4
+f 497/877/31 496/875/31 528/941/31 529/943/31
+f 498/879/4 497/877/4 529/943/4 530/945/4
+f 499/881/4 498/879/4 530/945/4 531/947/4
+f 500/883/4 499/881/4 531/947/4 532/949/4
+f 501/886/4 500/883/4 532/949/4 533/952/4
+f 502/887/4 501/886/4 533/952/4 534/954/4
+f 503/889/4 502/887/4 534/954/4 535/956/4
+f 504/891/4 503/889/4 535/956/4 536/957/4
+f 505/894/4 504/891/4 536/957/4 537/959/4
+f 506/895/4 505/894/4 537/959/4 538/961/4
+f 507/897/4 506/895/4 538/961/4 539/963/4
+f 508/899/4 507/897/4 539/963/4 540/966/4
+f 509/902/4 508/899/4 540/966/4 541/968/4
+f 510/903/4 509/902/4 541/968/4 542/970/4
+f 511/905/4 510/903/4 542/970/4 543/971/4
+f 512/908/4 511/905/4 543/971/4 544/973/4
+f 513/909/31 512/908/31 544/973/31 545/975/31
+f 514/912/4 513/910/4 545/976/4 546/978/4
+f 515/914/4 514/912/4 546/978/4 547/980/4
+f 516/917/4 515/914/4 547/980/4 548/983/4
+f 517/918/4 516/917/4 548/983/4 549/985/4
+f 518/920/4 517/918/4 549/985/4 550/987/4
+f 519/922/4 518/920/4 550/987/4 551/988/4
+f 520/925/4 519/922/4 551/988/4 552/990/4
+f 490/862/4 520/925/4 552/990/4 522/928/4
+f 435/783/32 434/781/32 498/880/32 499/882/32
+f 464/833/33 465/834/33 529/944/33 528/942/33
+f 479/848/34 480/849/34 544/974/34 543/972/34
+f 450/813/35 449/811/35 513/911/35 514/913/35
+f 439/791/7 438/789/7 502/888/7 503/890/7
+f 458/827/36 457/826/36 521/927/36 522/929/36
+f 428/769/30 427/767/30 491/865/30 492/867/30
+f 472/841/10 473/842/10 537/960/10 536/958/10
+f 487/856/37 488/857/37 552/991/37 551/989/37
+f 454/821/23 453/819/23 517/919/23 518/921/23
+f 443/799/38 442/797/38 506/896/38 507/898/38
+f 465/834/35 466/835/35 530/946/35 529/944/35
+f 480/849/39 481/850/39 545/977/39 544/974/39
+f 432/777/34 431/775/34 495/873/34 496/876/34
+f 457/826/38 459/828/38 523/931/38 521/927/38
+f 447/807/20 446/805/20 510/904/20 511/906/20
+f 473/842/40 474/843/40 538/962/40 537/960/40
+f 488/858/26 458/827/26 522/929/26 552/992/26
+f 436/785/28 435/783/28 499/882/28 500/884/28
+f 466/835/41 467/836/41 531/948/41 530/946/41
+f 481/850/21 482/851/21 546/979/21 545/977/21
+f 451/815/41 450/813/41 514/913/41 515/915/41
+f 440/793/37 439/791/37 503/890/37 504/892/37
+f 459/828/42 460/829/42 524/933/42 523/931/42
+f 429/771/43 428/769/43 492/867/43 493/869/43
+f 474/843/17 475/844/17 539/964/17 538/962/17
+f 455/823/16 454/821/16 518/921/16 519/923/16
+f 426/765/40 425/763/40 490/863/40 489/860/40
+f 444/801/42 443/799/42 507/898/42 508/900/42
+f 467/836/27 468/837/27 532/950/27 531/948/27
+f 482/851/32 483/852/32 547/981/32 546/979/32
+f 433/779/44 432/777/44 496/876/44 497/878/44
diff --git a/advtrains_signals_muc_ubahn/models/advtrains_signals_muc_ubahn_wsig_r.obj b/advtrains_signals_muc_ubahn/models/advtrains_signals_muc_ubahn_wsig_r.obj
new file mode 100644
index 0000000..6843591
--- /dev/null
+++ b/advtrains_signals_muc_ubahn/models/advtrains_signals_muc_ubahn_wsig_r.obj
@@ -0,0 +1,2137 @@
+# Blender 3.4.1
+# www.blender.org
+mtllib advtrains_signals_muc_ubahn_wsig_r.mtl
+o Cube
+v -0.500000 0.399750 0.043560
+v -0.500000 0.357794 0.043560
+v -0.500000 -0.357244 0.043560
+v -0.500000 -0.402505 0.043561
+v -0.500000 0.399750 -0.043560
+v -0.500000 0.357794 -0.043560
+v -0.500000 -0.357244 -0.043560
+v -0.500000 -0.402505 -0.043560
+v -0.302890 0.399750 0.043560
+v -0.302890 0.357794 0.043560
+v -0.302890 -0.357244 0.043560
+v -0.302890 -0.402505 0.043560
+v -0.302890 0.399750 -0.043560
+v -0.302890 0.357794 -0.043560
+v -0.302890 -0.357244 -0.043560
+v -0.302890 -0.402505 -0.043560
+v -0.218703 -0.357244 0.042354
+v -0.218703 0.357794 0.042354
+v -0.218703 -0.357244 -0.031584
+v -0.218703 0.357794 -0.031584
+v -0.450908 -0.357244 0.042354
+v -0.450908 0.357794 0.042354
+v -0.450908 -0.357244 -0.031584
+v -0.450908 0.357794 -0.031584
+v -0.265363 -0.310583 0.042354
+v -0.265363 0.311134 0.042354
+v -0.404248 0.311134 0.042354
+v -0.404247 -0.310583 0.042354
+v -0.265363 -0.310583 0.089581
+v -0.265363 0.311134 0.089581
+v -0.404247 0.311134 0.089581
+v -0.404247 -0.310583 0.089581
+v -0.257920 0.318577 -0.031584
+v -0.257920 -0.318027 -0.031584
+v -0.411691 0.318577 -0.031584
+v -0.411691 -0.318027 -0.031584
+v -0.331404 0.241338 -0.037331
+v -0.326280 0.189125 -0.037331
+v -0.316192 0.191128 -0.037331
+v -0.306689 0.195061 -0.037331
+v -0.298135 0.200771 -0.037331
+v -0.290861 0.208041 -0.037331
+v -0.285144 0.216591 -0.037331
+v -0.281205 0.226091 -0.037331
+v -0.279195 0.236177 -0.037331
+v -0.279191 0.246462 -0.037331
+v -0.281194 0.256550 -0.037331
+v -0.285127 0.266053 -0.037331
+v -0.290838 0.274606 -0.037331
+v -0.298107 0.281881 -0.037331
+v -0.306657 0.287598 -0.037331
+v -0.316157 0.291537 -0.037331
+v -0.326243 0.293547 -0.037331
+v -0.336528 0.293550 -0.037331
+v -0.346616 0.291547 -0.037331
+v -0.356119 0.287615 -0.037331
+v -0.364672 0.281904 -0.037331
+v -0.371947 0.274634 -0.037331
+v -0.377664 0.266085 -0.037331
+v -0.381603 0.256585 -0.037331
+v -0.383613 0.246498 -0.037331
+v -0.383616 0.236214 -0.037331
+v -0.381613 0.226126 -0.037331
+v -0.377681 0.216623 -0.037331
+v -0.371970 0.208069 -0.037331
+v -0.364700 0.200795 -0.037331
+v -0.356151 0.195078 -0.037331
+v -0.346651 0.191139 -0.037331
+v -0.336564 0.189129 -0.037331
+v -0.313532 0.182348 -0.031584
+v -0.325384 0.179994 -0.031584
+v -0.302367 0.186968 -0.031584
+v -0.292318 0.193677 -0.031584
+v -0.283771 0.202218 -0.031584
+v -0.277054 0.212263 -0.031584
+v -0.272426 0.223425 -0.031584
+v -0.270065 0.235275 -0.031584
+v -0.270060 0.247358 -0.031584
+v -0.272414 0.259210 -0.031584
+v -0.277034 0.270375 -0.031584
+v -0.283743 0.280424 -0.031584
+v -0.292285 0.288971 -0.031584
+v -0.302329 0.295687 -0.031584
+v -0.313491 0.300315 -0.031584
+v -0.325341 0.302677 -0.031584
+v -0.337424 0.302681 -0.031584
+v -0.349276 0.300328 -0.031584
+v -0.360441 0.295708 -0.031584
+v -0.370490 0.288998 -0.031584
+v -0.379037 0.280457 -0.031584
+v -0.385753 0.270413 -0.031584
+v -0.390381 0.259251 -0.031584
+v -0.392743 0.247401 -0.031584
+v -0.392747 0.235317 -0.031584
+v -0.390394 0.223466 -0.031584
+v -0.385774 0.212301 -0.031584
+v -0.379064 0.202252 -0.031584
+v -0.370523 0.193705 -0.031584
+v -0.360479 0.186988 -0.031584
+v -0.349317 0.182360 -0.031584
+v -0.337467 0.179999 -0.031584
+v -0.316192 0.191128 -0.054117
+v -0.326280 0.189125 -0.054117
+v -0.306689 0.195061 -0.054117
+v -0.298136 0.200771 -0.054117
+v -0.290861 0.208041 -0.054117
+v -0.285144 0.216591 -0.054117
+v -0.281205 0.226091 -0.054117
+v -0.279195 0.236177 -0.054117
+v -0.279191 0.246462 -0.083504
+v -0.281194 0.256550 -0.083504
+v -0.285127 0.266053 -0.083504
+v -0.290838 0.274606 -0.083504
+v -0.298107 0.281881 -0.083504
+v -0.306657 0.287598 -0.083504
+v -0.316157 0.291537 -0.083504
+v -0.326243 0.293547 -0.083504
+v -0.336528 0.293550 -0.083504
+v -0.346616 0.291547 -0.083504
+v -0.356119 0.287615 -0.083504
+v -0.364672 0.281904 -0.083504
+v -0.371947 0.274634 -0.083504
+v -0.377664 0.266085 -0.083504
+v -0.381603 0.256585 -0.083504
+v -0.383613 0.246498 -0.083504
+v -0.383616 0.236214 -0.054117
+v -0.381613 0.226126 -0.054117
+v -0.377681 0.216623 -0.054117
+v -0.371970 0.208069 -0.054117
+v -0.364700 0.200795 -0.054117
+v -0.356151 0.195078 -0.054117
+v -0.346651 0.191139 -0.054117
+v -0.336564 0.189129 -0.054117
+v -0.313532 0.182348 -0.054117
+v -0.325384 0.179994 -0.054117
+v -0.302367 0.186968 -0.054117
+v -0.292318 0.193677 -0.054117
+v -0.283771 0.202218 -0.054117
+v -0.277054 0.212263 -0.054117
+v -0.272426 0.223425 -0.054117
+v -0.270065 0.235275 -0.054117
+v -0.270060 0.247358 -0.083504
+v -0.272414 0.259210 -0.083504
+v -0.277034 0.270375 -0.083504
+v -0.283743 0.280424 -0.083504
+v -0.292285 0.288971 -0.083504
+v -0.302329 0.295687 -0.083504
+v -0.313491 0.300315 -0.083504
+v -0.325341 0.302677 -0.083504
+v -0.337424 0.302681 -0.083504
+v -0.349276 0.300328 -0.083504
+v -0.360441 0.295708 -0.083504
+v -0.370490 0.288998 -0.083504
+v -0.379037 0.280457 -0.083504
+v -0.385753 0.270413 -0.083504
+v -0.390381 0.259251 -0.083504
+v -0.392743 0.247401 -0.083504
+v -0.392747 0.235317 -0.054117
+v -0.390394 0.223466 -0.054117
+v -0.385774 0.212301 -0.054117
+v -0.379064 0.202252 -0.054117
+v -0.370523 0.193705 -0.054117
+v -0.360479 0.186988 -0.054117
+v -0.349317 0.182360 -0.054117
+v -0.337467 0.179999 -0.054117
+v -0.331404 0.091338 -0.037331
+v -0.326280 0.039125 -0.037331
+v -0.316192 0.041128 -0.037331
+v -0.306689 0.045061 -0.037331
+v -0.298135 0.050771 -0.037331
+v -0.290861 0.058041 -0.037331
+v -0.285144 0.066591 -0.037331
+v -0.281205 0.076091 -0.037331
+v -0.279195 0.086177 -0.037331
+v -0.279191 0.096462 -0.037331
+v -0.281194 0.106550 -0.037331
+v -0.285127 0.116053 -0.037331
+v -0.290838 0.124606 -0.037331
+v -0.298107 0.131881 -0.037331
+v -0.306657 0.137598 -0.037331
+v -0.316157 0.141537 -0.037331
+v -0.326243 0.143547 -0.037331
+v -0.336528 0.143550 -0.037331
+v -0.346616 0.141547 -0.037331
+v -0.356119 0.137615 -0.037331
+v -0.364672 0.131904 -0.037331
+v -0.371947 0.124634 -0.037331
+v -0.377664 0.116085 -0.037331
+v -0.381603 0.106585 -0.037331
+v -0.383613 0.096498 -0.037331
+v -0.383616 0.086214 -0.037331
+v -0.381613 0.076126 -0.037331
+v -0.377681 0.066623 -0.037331
+v -0.371970 0.058069 -0.037331
+v -0.364700 0.050795 -0.037331
+v -0.356151 0.045078 -0.037331
+v -0.346651 0.041139 -0.037331
+v -0.336564 0.039129 -0.037331
+v -0.313532 0.032348 -0.031584
+v -0.325384 0.029994 -0.031584
+v -0.302367 0.036968 -0.031584
+v -0.292318 0.043677 -0.031584
+v -0.283771 0.052218 -0.031584
+v -0.277054 0.062263 -0.031584
+v -0.272426 0.073425 -0.031584
+v -0.270065 0.085275 -0.031584
+v -0.270060 0.097358 -0.031584
+v -0.272414 0.109210 -0.031584
+v -0.277034 0.120375 -0.031584
+v -0.283743 0.130424 -0.031584
+v -0.292285 0.138971 -0.031584
+v -0.302329 0.145687 -0.031584
+v -0.313491 0.150315 -0.031584
+v -0.325341 0.152677 -0.031584
+v -0.337424 0.152681 -0.031584
+v -0.349276 0.150328 -0.031584
+v -0.360441 0.145708 -0.031584
+v -0.370490 0.138998 -0.031584
+v -0.379037 0.130457 -0.031584
+v -0.385753 0.120412 -0.031584
+v -0.390381 0.109251 -0.031584
+v -0.392743 0.097401 -0.031584
+v -0.392747 0.085317 -0.031584
+v -0.390394 0.073466 -0.031584
+v -0.385774 0.062301 -0.031584
+v -0.379064 0.052252 -0.031584
+v -0.370523 0.043705 -0.031584
+v -0.360479 0.036988 -0.031584
+v -0.349317 0.032360 -0.031584
+v -0.337467 0.029999 -0.031584
+v -0.316192 0.041128 -0.054117
+v -0.326280 0.039125 -0.054117
+v -0.306689 0.045061 -0.054117
+v -0.298136 0.050771 -0.054117
+v -0.290861 0.058041 -0.054117
+v -0.285144 0.066591 -0.054117
+v -0.281205 0.076091 -0.054117
+v -0.279195 0.086177 -0.054117
+v -0.279191 0.096462 -0.083504
+v -0.281194 0.106550 -0.083504
+v -0.285127 0.116053 -0.083504
+v -0.290838 0.124606 -0.083504
+v -0.298107 0.131881 -0.083504
+v -0.306657 0.137598 -0.083504
+v -0.316157 0.141537 -0.083504
+v -0.326243 0.143547 -0.083504
+v -0.336528 0.143550 -0.083504
+v -0.346616 0.141547 -0.083504
+v -0.356119 0.137615 -0.083504
+v -0.364672 0.131904 -0.083504
+v -0.371947 0.124634 -0.083504
+v -0.377664 0.116085 -0.083504
+v -0.381603 0.106585 -0.083504
+v -0.383613 0.096498 -0.083504
+v -0.383616 0.086214 -0.054117
+v -0.381613 0.076126 -0.054117
+v -0.377681 0.066623 -0.054117
+v -0.371970 0.058069 -0.054117
+v -0.364700 0.050795 -0.054117
+v -0.356151 0.045078 -0.054117
+v -0.346651 0.041139 -0.054117
+v -0.336564 0.039129 -0.054117
+v -0.313532 0.032348 -0.054117
+v -0.325384 0.029994 -0.054117
+v -0.302367 0.036968 -0.054117
+v -0.292318 0.043677 -0.054117
+v -0.283771 0.052218 -0.054117
+v -0.277054 0.062263 -0.054117
+v -0.272426 0.073425 -0.054117
+v -0.270065 0.085275 -0.054117
+v -0.270060 0.097358 -0.083504
+v -0.272414 0.109210 -0.083504
+v -0.277034 0.120375 -0.083504
+v -0.283743 0.130424 -0.083504
+v -0.292285 0.138971 -0.083504
+v -0.302329 0.145687 -0.083504
+v -0.313491 0.150315 -0.083504
+v -0.325341 0.152677 -0.083504
+v -0.337424 0.152681 -0.083504
+v -0.349276 0.150328 -0.083504
+v -0.360441 0.145708 -0.083504
+v -0.370490 0.138998 -0.083504
+v -0.379037 0.130457 -0.083504
+v -0.385753 0.120412 -0.083504
+v -0.390381 0.109251 -0.083504
+v -0.392743 0.097401 -0.083504
+v -0.392747 0.085317 -0.054117
+v -0.390394 0.073466 -0.054117
+v -0.385774 0.062301 -0.054117
+v -0.379064 0.052252 -0.054117
+v -0.370523 0.043705 -0.054117
+v -0.360479 0.036988 -0.054117
+v -0.349317 0.032360 -0.054117
+v -0.337467 0.029999 -0.054117
+v -0.331404 -0.058662 -0.037331
+v -0.326280 -0.110875 -0.037331
+v -0.316192 -0.108872 -0.037331
+v -0.306689 -0.104939 -0.037331
+v -0.298135 -0.099229 -0.037331
+v -0.290861 -0.091959 -0.037331
+v -0.285144 -0.083409 -0.037331
+v -0.281205 -0.073909 -0.037331
+v -0.279195 -0.063823 -0.037331
+v -0.279191 -0.053538 -0.037331
+v -0.281194 -0.043450 -0.037331
+v -0.285127 -0.033947 -0.037331
+v -0.290838 -0.025394 -0.037331
+v -0.298107 -0.018119 -0.037331
+v -0.306657 -0.012402 -0.037331
+v -0.316157 -0.008463 -0.037331
+v -0.326243 -0.006453 -0.037331
+v -0.336528 -0.006450 -0.037331
+v -0.346616 -0.008453 -0.037331
+v -0.356119 -0.012385 -0.037331
+v -0.364672 -0.018096 -0.037331
+v -0.371947 -0.025366 -0.037331
+v -0.377664 -0.033915 -0.037331
+v -0.381603 -0.043415 -0.037331
+v -0.383613 -0.053502 -0.037331
+v -0.383616 -0.063786 -0.037331
+v -0.381613 -0.073874 -0.037331
+v -0.377681 -0.083377 -0.037331
+v -0.371970 -0.091931 -0.037331
+v -0.364700 -0.099205 -0.037331
+v -0.356151 -0.104922 -0.037331
+v -0.346651 -0.108861 -0.037331
+v -0.336564 -0.110871 -0.037331
+v -0.313532 -0.117652 -0.031584
+v -0.325384 -0.120006 -0.031584
+v -0.302367 -0.113032 -0.031584
+v -0.292318 -0.106323 -0.031584
+v -0.283771 -0.097782 -0.031584
+v -0.277054 -0.087737 -0.031584
+v -0.272426 -0.076575 -0.031584
+v -0.270065 -0.064725 -0.031584
+v -0.270060 -0.052642 -0.031584
+v -0.272414 -0.040790 -0.031584
+v -0.277034 -0.029625 -0.031584
+v -0.283743 -0.019576 -0.031584
+v -0.292285 -0.011029 -0.031584
+v -0.302329 -0.004313 -0.031584
+v -0.313491 0.000315 -0.031584
+v -0.325341 0.002677 -0.031584
+v -0.337424 0.002681 -0.031584
+v -0.349276 0.000328 -0.031584
+v -0.360441 -0.004292 -0.031584
+v -0.370490 -0.011002 -0.031584
+v -0.379037 -0.019543 -0.031584
+v -0.385753 -0.029587 -0.031584
+v -0.390381 -0.040749 -0.031584
+v -0.392743 -0.052599 -0.031584
+v -0.392747 -0.064683 -0.031584
+v -0.390394 -0.076534 -0.031584
+v -0.385774 -0.087699 -0.031584
+v -0.379064 -0.097748 -0.031584
+v -0.370523 -0.106295 -0.031584
+v -0.360479 -0.113012 -0.031584
+v -0.349317 -0.117640 -0.031584
+v -0.337467 -0.120001 -0.031584
+v -0.316192 -0.108872 -0.054117
+v -0.326280 -0.110875 -0.054117
+v -0.306689 -0.104939 -0.054117
+v -0.298136 -0.099229 -0.054117
+v -0.290861 -0.091959 -0.054117
+v -0.285144 -0.083409 -0.054117
+v -0.281205 -0.073909 -0.054117
+v -0.279195 -0.063823 -0.054117
+v -0.279191 -0.053538 -0.083504
+v -0.281194 -0.043450 -0.083504
+v -0.285127 -0.033947 -0.083504
+v -0.290838 -0.025394 -0.083504
+v -0.298107 -0.018119 -0.083504
+v -0.306657 -0.012402 -0.083504
+v -0.316157 -0.008463 -0.083504
+v -0.326243 -0.006453 -0.083504
+v -0.336528 -0.006450 -0.083504
+v -0.346616 -0.008453 -0.083504
+v -0.356119 -0.012385 -0.083504
+v -0.364672 -0.018096 -0.083504
+v -0.371947 -0.025366 -0.083504
+v -0.377664 -0.033915 -0.083504
+v -0.381603 -0.043415 -0.083504
+v -0.383613 -0.053502 -0.083504
+v -0.383616 -0.063786 -0.054117
+v -0.381613 -0.073874 -0.054117
+v -0.377681 -0.083377 -0.054117
+v -0.371970 -0.091931 -0.054117
+v -0.364700 -0.099205 -0.054117
+v -0.356151 -0.104922 -0.054117
+v -0.346651 -0.108861 -0.054117
+v -0.336564 -0.110871 -0.054117
+v -0.313532 -0.117652 -0.054117
+v -0.325384 -0.120006 -0.054117
+v -0.302367 -0.113032 -0.054117
+v -0.292318 -0.106323 -0.054117
+v -0.283771 -0.097782 -0.054117
+v -0.277054 -0.087737 -0.054117
+v -0.272426 -0.076575 -0.054117
+v -0.270065 -0.064725 -0.054117
+v -0.270060 -0.052642 -0.083504
+v -0.272414 -0.040790 -0.083504
+v -0.277034 -0.029625 -0.083504
+v -0.283743 -0.019576 -0.083504
+v -0.292285 -0.011029 -0.083504
+v -0.302329 -0.004313 -0.083504
+v -0.313491 0.000315 -0.083504
+v -0.325341 0.002677 -0.083504
+v -0.337424 0.002681 -0.083504
+v -0.349276 0.000328 -0.083504
+v -0.360441 -0.004292 -0.083504
+v -0.370490 -0.011002 -0.083504
+v -0.379037 -0.019543 -0.083504
+v -0.385753 -0.029587 -0.083504
+v -0.390381 -0.040749 -0.083504
+v -0.392743 -0.052599 -0.083504
+v -0.392747 -0.064683 -0.054117
+v -0.390394 -0.076534 -0.054117
+v -0.385774 -0.087699 -0.054117
+v -0.379064 -0.097748 -0.054117
+v -0.370523 -0.106295 -0.054117
+v -0.360479 -0.113012 -0.054117
+v -0.349317 -0.117640 -0.054117
+v -0.337467 -0.120001 -0.054117
+v -0.331404 -0.208662 -0.037331
+v -0.326280 -0.260875 -0.037331
+v -0.316192 -0.258872 -0.037331
+v -0.306689 -0.254939 -0.037331
+v -0.298135 -0.249229 -0.037331
+v -0.290861 -0.241959 -0.037331
+v -0.285144 -0.233410 -0.037331
+v -0.281205 -0.223909 -0.037331
+v -0.279195 -0.213823 -0.037331
+v -0.279191 -0.203538 -0.037331
+v -0.281194 -0.193450 -0.037331
+v -0.285127 -0.183947 -0.037331
+v -0.290838 -0.175394 -0.037331
+v -0.298107 -0.168119 -0.037331
+v -0.306657 -0.162402 -0.037331
+v -0.316157 -0.158463 -0.037331
+v -0.326243 -0.156453 -0.037331
+v -0.336528 -0.156450 -0.037331
+v -0.346616 -0.158453 -0.037331
+v -0.356119 -0.162385 -0.037331
+v -0.364672 -0.168096 -0.037331
+v -0.371947 -0.175366 -0.037331
+v -0.377664 -0.183915 -0.037331
+v -0.381603 -0.193415 -0.037331
+v -0.383613 -0.203502 -0.037331
+v -0.383616 -0.213786 -0.037331
+v -0.381613 -0.223874 -0.037331
+v -0.377681 -0.233377 -0.037331
+v -0.371970 -0.241931 -0.037331
+v -0.364700 -0.249205 -0.037331
+v -0.356151 -0.254922 -0.037331
+v -0.346651 -0.258861 -0.037331
+v -0.336564 -0.260871 -0.037331
+v -0.313532 -0.267652 -0.031584
+v -0.325384 -0.270006 -0.031584
+v -0.302367 -0.263032 -0.031584
+v -0.292318 -0.256323 -0.031584
+v -0.283771 -0.247782 -0.031584
+v -0.277054 -0.237737 -0.031584
+v -0.272426 -0.226575 -0.031584
+v -0.270065 -0.214725 -0.031584
+v -0.270060 -0.202642 -0.031584
+v -0.272414 -0.190790 -0.031584
+v -0.277034 -0.179625 -0.031584
+v -0.283743 -0.169576 -0.031584
+v -0.292285 -0.161029 -0.031584
+v -0.302329 -0.154313 -0.031584
+v -0.313491 -0.149685 -0.031584
+v -0.325341 -0.147323 -0.031584
+v -0.337424 -0.147319 -0.031584
+v -0.349276 -0.149672 -0.031584
+v -0.360441 -0.154292 -0.031584
+v -0.370490 -0.161002 -0.031584
+v -0.379037 -0.169543 -0.031584
+v -0.385753 -0.179587 -0.031584
+v -0.390381 -0.190749 -0.031584
+v -0.392743 -0.202599 -0.031584
+v -0.392747 -0.214683 -0.031584
+v -0.390394 -0.226534 -0.031584
+v -0.385774 -0.237699 -0.031584
+v -0.379064 -0.247748 -0.031584
+v -0.370523 -0.256295 -0.031584
+v -0.360479 -0.263012 -0.031584
+v -0.349317 -0.267640 -0.031584
+v -0.337467 -0.270001 -0.031584
+v -0.316192 -0.258872 -0.054117
+v -0.326280 -0.260875 -0.054117
+v -0.306689 -0.254939 -0.054117
+v -0.298136 -0.249229 -0.054117
+v -0.290861 -0.241959 -0.054117
+v -0.285144 -0.233410 -0.054117
+v -0.281205 -0.223909 -0.054117
+v -0.279195 -0.213823 -0.054117
+v -0.279191 -0.203538 -0.083504
+v -0.281194 -0.193450 -0.083504
+v -0.285127 -0.183947 -0.083504
+v -0.290838 -0.175394 -0.083504
+v -0.298107 -0.168119 -0.083504
+v -0.306657 -0.162402 -0.083504
+v -0.316157 -0.158463 -0.083504
+v -0.326243 -0.156453 -0.083504
+v -0.336528 -0.156450 -0.083504
+v -0.346616 -0.158453 -0.083504
+v -0.356119 -0.162385 -0.083504
+v -0.364672 -0.168096 -0.083504
+v -0.371947 -0.175366 -0.083504
+v -0.377664 -0.183915 -0.083504
+v -0.381603 -0.193415 -0.083504
+v -0.383613 -0.203502 -0.083504
+v -0.383616 -0.213786 -0.054117
+v -0.381613 -0.223874 -0.054117
+v -0.377681 -0.233377 -0.054117
+v -0.371970 -0.241931 -0.054117
+v -0.364700 -0.249205 -0.054117
+v -0.356151 -0.254922 -0.054117
+v -0.346651 -0.258861 -0.054117
+v -0.336564 -0.260871 -0.054117
+v -0.313532 -0.267652 -0.054117
+v -0.325384 -0.270006 -0.054117
+v -0.302367 -0.263032 -0.054117
+v -0.292318 -0.256323 -0.054117
+v -0.283771 -0.247782 -0.054117
+v -0.277054 -0.237737 -0.054117
+v -0.272426 -0.226575 -0.054117
+v -0.270065 -0.214725 -0.054117
+v -0.270060 -0.202642 -0.083504
+v -0.272414 -0.190790 -0.083504
+v -0.277034 -0.179625 -0.083504
+v -0.283743 -0.169576 -0.083504
+v -0.292285 -0.161029 -0.083504
+v -0.302329 -0.154313 -0.083504
+v -0.313491 -0.149685 -0.083504
+v -0.325341 -0.147323 -0.083504
+v -0.337424 -0.147319 -0.083504
+v -0.349276 -0.149672 -0.083504
+v -0.360441 -0.154292 -0.083504
+v -0.370490 -0.161002 -0.083504
+v -0.379037 -0.169543 -0.083504
+v -0.385753 -0.179587 -0.083504
+v -0.390381 -0.190749 -0.083504
+v -0.392743 -0.202599 -0.083504
+v -0.392747 -0.214683 -0.054117
+v -0.390394 -0.226534 -0.054117
+v -0.385774 -0.237699 -0.054117
+v -0.379064 -0.247748 -0.054117
+v -0.370523 -0.256295 -0.054117
+v -0.360479 -0.263012 -0.054117
+v -0.349317 -0.267640 -0.054117
+v -0.337467 -0.270001 -0.054117
+vn -0.0000 -1.0000 -0.0000
+vn -0.0000 1.0000 -0.0000
+vn 1.0000 -0.0000 -0.0000
+vn -0.0000 -0.0000 -1.0000
+vn -0.0000 -0.0000 1.0000
+vn -1.0000 -0.0000 -0.0000
+vn -0.3830 -0.9237 -0.0000
+vn 0.1954 0.9807 -0.0000
+vn 0.8313 -0.5559 -0.0000
+vn 0.0003 1.0000 -0.0000
+vn -0.9237 0.3830 -0.0000
+vn 0.9807 -0.1954 -0.0000
+vn 0.1947 -0.9809 -0.0000
+vn 0.7073 0.7069 -0.0000
+vn -0.5559 -0.8313 -0.0000
+vn 0.3830 0.9237 -0.0000
+vn -0.3824 0.9240 -0.0000
+vn 1.0000 -0.0003 -0.0000
+vn -0.8313 0.5559 -0.0000
+vn 0.9237 -0.3830 -0.0000
+vn -0.9809 -0.1947 -0.0000
+vn -0.7074 -0.7069 -0.0000
+vn 0.5559 0.8313 -0.0000
+vn 0.7069 -0.7074 -0.0000
+vn -0.7069 0.7073 -0.0000
+vn -0.0003 -1.0000 -0.0000
+vn 0.8317 0.5553 -0.0000
+vn -0.8317 -0.5553 -0.0000
+vn 0.7074 0.7069 -0.0000
+vn -0.5553 0.8317 -0.0000
+vn -0.0003 -0.9346 -0.3557
+vn -0.9240 -0.3824 -0.0000
+vn 1.0000 -0.0004 -0.0000
+vn -0.9807 0.1954 -0.0000
+vn 0.9809 0.1948 -0.0000
+vn 0.1948 -0.9809 -0.0000
+vn -0.1954 -0.9807 -0.0000
+vn 0.3824 -0.9240 -0.0000
+vn -1.0000 0.0004 -0.0000
+vn -0.1947 0.9809 -0.0000
+vn 0.9240 0.3824 -0.0000
+vn 0.5553 -0.8317 -0.0000
+vn -0.7069 0.7074 -0.0000
+vn -1.0000 0.0003 -0.0000
+vn -0.7073 -0.7069 -0.0000
+vn 0.0004 1.0000 -0.0000
+vt 0.640016 0.630686
+vt 0.874116 0.630686
+vt 0.899031 0.371671
+vt 0.899031 0.605771
+vt 0.275073 0.371671
+vt 0.275073 0.605771
+vt 0.536051 0.632649
+vt 0.301951 0.632649
+vt 0.588281 0.578951
+vt 0.588281 0.344851
+vt 0.847296 0.319936
+vt 0.613196 0.319936
+vt 0.326808 0.319936
+vt 0.560909 0.319936
+vt 0.587786 0.580914
+vt 0.587786 0.346814
+vt 0.757066 0.513636
+vt 0.781981 0.488721
+vt 0.392123 0.488721
+vt 0.419001 0.515599
+vt 0.705331 0.461901
+vt 0.730246 0.436986
+vt 0.443858 0.436986
+vt 0.470736 0.463864
+vt 0.062341 0.000247
+vt 0.000247 0.062341
+vt 0.000248 0.359598
+vt 0.039909 0.319936
+vt 0.662834 0.000247
+vt 0.724927 0.062341
+vt 0.000247 0.960091
+vt 0.039910 0.999753
+vt 0.062341 0.062341
+vt 0.662834 0.062341
+vt 0.062341 0.319441
+vt 0.000247 0.257348
+vt 0.274579 0.359598
+vt 0.234916 0.319936
+vt 0.662834 0.319441
+vt 0.724927 0.257348
+vt 0.274579 0.960090
+vt 0.234917 0.999753
+vt 0.062341 0.257348
+vt 0.662834 0.257348
+vt 0.079095 0.359121
+vt 0.039433 0.398784
+vt 0.079095 0.960567
+vt 0.039433 0.920905
+vt 0.195731 0.960567
+vt 0.235393 0.920905
+vt 0.195731 0.359121
+vt 0.235393 0.398783
+vt 0.079095 0.398783
+vt 0.079095 0.920905
+vt 0.195731 0.920905
+vt 0.195731 0.398783
+vt 0.629899 0.095275
+vt 0.095275 0.095275
+vt 0.629899 0.224413
+vt 0.095275 0.224413
+vt 0.753412 0.903011
+vt 0.719354 0.930961
+vt 0.334815 0.932388
+vt 0.611202 0.932388
+vt 0.714555 0.923780
+vt 0.602565 0.932389
+vt 0.711250 0.915800
+vt 0.593928 0.932389
+vt 0.709565 0.907329
+vt 0.585291 0.932389
+vt 0.709565 0.898692
+vt 0.576654 0.932389
+vt 0.711250 0.890221
+vt 0.568017 0.932389
+vt 0.714556 0.882241
+vt 0.559379 0.932389
+vt 0.719354 0.875060
+vt 0.550742 0.932389
+vt 0.725461 0.868952
+vt 0.542105 0.932389
+vt 0.732643 0.864154
+vt 0.533468 0.932389
+vt 0.740623 0.860849
+vt 0.524831 0.932389
+vt 0.749094 0.859164
+vt 0.516194 0.932389
+vt 0.757731 0.859164
+vt 0.507557 0.932389
+vt 0.766202 0.860849
+vt 0.498920 0.932389
+vt 0.774182 0.864154
+vt 0.490283 0.932389
+vt 0.781363 0.868953
+vt 0.481646 0.932389
+vt 0.787470 0.875060
+vt 0.473009 0.932389
+vt 0.792269 0.882241
+vt 0.464371 0.932389
+vt 0.795574 0.890221
+vt 0.455734 0.932389
+vt 0.797259 0.898692
+vt 0.447097 0.932389
+vt 0.797259 0.907329
+vt 0.438460 0.932389
+vt 0.795574 0.915800
+vt 0.429823 0.932389
+vt 0.792269 0.923780
+vt 0.421186 0.932389
+vt 0.787470 0.930962
+vt 0.412549 0.932389
+vt 0.781363 0.937069
+vt 0.403912 0.932389
+vt 0.774181 0.941867
+vt 0.395275 0.932389
+vt 0.766201 0.945173
+vt 0.386637 0.932389
+vt 0.757730 0.946858
+vt 0.378001 0.932389
+vt 0.749093 0.946858
+vt 0.369363 0.932389
+vt 0.740622 0.945172
+vt 0.360726 0.932389
+vt 0.732643 0.941867
+vt 0.352089 0.932388
+vt 0.725461 0.937069
+vt 0.343452 0.932388
+vt 0.618830 0.816892
+vt 0.628978 0.816892
+vt 0.608683 0.816892
+vt 0.598535 0.816892
+vt 0.588388 0.816892
+vt 0.578240 0.816892
+vt 0.568093 0.816892
+vt 0.557945 0.816892
+vt 0.547798 0.816892
+vt 0.537650 0.816892
+vt 0.527503 0.816892
+vt 0.517355 0.816892
+vt 0.507208 0.816892
+vt 0.497060 0.816892
+vt 0.486913 0.816892
+vt 0.476765 0.816892
+vt 0.466618 0.816892
+vt 0.456470 0.816892
+vt 0.446323 0.816892
+vt 0.436175 0.816892
+vt 0.426028 0.816892
+vt 0.415880 0.816892
+vt 0.405733 0.816892
+vt 0.395585 0.816892
+vt 0.385438 0.816892
+vt 0.375290 0.816892
+vt 0.365143 0.816892
+vt 0.354995 0.816892
+vt 0.344848 0.816892
+vt 0.334700 0.816892
+vt 0.324553 0.816892
+vt 0.314405 0.816892
+vt 0.639125 0.816892
+vt 0.517095 0.669905
+vt 0.602565 0.946486
+vt 0.334815 0.946485
+vt 0.515410 0.678377
+vt 0.611202 0.946485
+vt 0.520400 0.661926
+vt 0.593928 0.946485
+vt 0.525198 0.654744
+vt 0.585291 0.946486
+vt 0.531306 0.648637
+vt 0.576654 0.946486
+vt 0.568017 0.946486
+vt 0.538487 0.643839
+vt 0.546467 0.640533
+vt 0.559379 0.946486
+vt 0.554938 0.638848
+vt 0.507247 0.729756
+vt 0.550742 0.946486
+vt 0.481149 0.728166
+vt 0.542105 0.971165
+vt 0.472678 0.726481
+vt 0.533468 0.971165
+vt 0.464698 0.723176
+vt 0.524831 0.971165
+vt 0.457517 0.718377
+vt 0.516194 0.971165
+vt 0.507557 0.971165
+vt 0.451409 0.712270
+vt 0.446611 0.705089
+vt 0.498920 0.971165
+vt 0.443305 0.697109
+vt 0.490283 0.971165
+vt 0.441620 0.688638
+vt 0.481646 0.971165
+vt 0.473008 0.971165
+vt 0.441620 0.680001
+vt 0.443305 0.671530
+vt 0.464371 0.971165
+vt 0.446610 0.663550
+vt 0.455734 0.971165
+vt 0.451409 0.656369
+vt 0.447097 0.971165
+vt 0.438460 0.971165
+vt 0.457516 0.650261
+vt 0.464698 0.645463
+vt 0.429823 0.971165
+vt 0.472677 0.642157
+vt 0.421186 0.971165
+vt 0.412549 0.971165
+vt 0.481149 0.640472
+vt 0.507247 0.638882
+vt 0.554938 0.726542
+vt 0.403912 0.946486
+vt 0.546467 0.724857
+vt 0.395275 0.946486
+vt 0.538487 0.721552
+vt 0.386637 0.946486
+vt 0.378001 0.946486
+vt 0.531306 0.716753
+vt 0.525198 0.710646
+vt 0.369363 0.946486
+vt 0.520400 0.703464
+vt 0.360726 0.946485
+vt 0.517095 0.695485
+vt 0.352089 0.946485
+vt 0.343452 0.946485
+vt 0.515410 0.687014
+vt 0.509722 0.667669
+vt 0.618830 0.797969
+vt 0.507742 0.677621
+vt 0.628978 0.797969
+vt 0.513605 0.658294
+vt 0.608683 0.797969
+vt 0.519242 0.649857
+vt 0.598535 0.797969
+vt 0.588388 0.797969
+vt 0.526418 0.642681
+vt 0.578240 0.797969
+vt 0.534855 0.637043
+vt 0.568093 0.797969
+vt 0.544230 0.633160
+vt 0.554183 0.631180
+vt 0.507028 0.737458
+vt 0.557945 0.797969
+vt 0.480393 0.735834
+vt 0.547798 0.773289
+vt 0.470441 0.733854
+vt 0.537650 0.773289
+vt 0.461066 0.729971
+vt 0.527503 0.773289
+vt 0.452629 0.724334
+vt 0.517355 0.773289
+vt 0.507208 0.773290
+vt 0.445453 0.717158
+vt 0.497060 0.773290
+vt 0.439816 0.708721
+vt 0.486913 0.773290
+vt 0.435932 0.699346
+vt 0.433953 0.689393
+vt 0.476765 0.773290
+vt 0.433953 0.679246
+vt 0.466618 0.773290
+vt 0.435932 0.669293
+vt 0.456470 0.773290
+vt 0.439815 0.659918
+vt 0.446323 0.773289
+vt 0.436175 0.773289
+vt 0.445453 0.651481
+vt 0.426028 0.773289
+vt 0.452628 0.644305
+vt 0.415880 0.773289
+vt 0.461066 0.638668
+vt 0.470441 0.634784
+vt 0.405733 0.773289
+vt 0.480393 0.632805
+vt 0.395585 0.773289
+vt 0.507028 0.631180
+vt 0.554183 0.734210
+vt 0.385438 0.797969
+vt 0.544230 0.732230
+vt 0.375290 0.797969
+vt 0.534855 0.728347
+vt 0.365143 0.797969
+vt 0.354995 0.797969
+vt 0.526418 0.722709
+vt 0.344848 0.797969
+vt 0.519243 0.715534
+vt 0.334700 0.797969
+vt 0.513605 0.707096
+vt 0.509721 0.697721
+vt 0.324553 0.797969
+vt 0.507742 0.687769
+vt 0.314405 0.797969
+vt 0.639125 0.797969
+vt 0.752599 0.742115
+vt 0.718541 0.770066
+vt 0.334815 0.932388
+vt 0.611202 0.932388
+vt 0.713742 0.762885
+vt 0.602565 0.932389
+vt 0.710437 0.754905
+vt 0.593928 0.932389
+vt 0.708752 0.746434
+vt 0.585291 0.932389
+vt 0.708752 0.737797
+vt 0.576654 0.932389
+vt 0.710437 0.729326
+vt 0.568017 0.932389
+vt 0.713743 0.721346
+vt 0.559379 0.932389
+vt 0.718541 0.714165
+vt 0.550742 0.932389
+vt 0.724649 0.708057
+vt 0.542105 0.932389
+vt 0.731830 0.703259
+vt 0.533468 0.932389
+vt 0.739810 0.699954
+vt 0.524831 0.932389
+vt 0.748281 0.698269
+vt 0.516194 0.932389
+vt 0.756918 0.698269
+vt 0.507557 0.932389
+vt 0.765389 0.699954
+vt 0.498920 0.932389
+vt 0.773369 0.703259
+vt 0.490283 0.932389
+vt 0.780550 0.708058
+vt 0.481646 0.932389
+vt 0.786658 0.714165
+vt 0.473009 0.932389
+vt 0.791456 0.721346
+vt 0.464371 0.932389
+vt 0.794761 0.729326
+vt 0.455734 0.932389
+vt 0.796446 0.737797
+vt 0.447097 0.932389
+vt 0.796446 0.746434
+vt 0.438460 0.932389
+vt 0.794761 0.754905
+vt 0.429823 0.932389
+vt 0.791456 0.762885
+vt 0.421186 0.932389
+vt 0.786657 0.770066
+vt 0.412549 0.932389
+vt 0.780550 0.776174
+vt 0.403912 0.932389
+vt 0.773368 0.780972
+vt 0.395275 0.932389
+vt 0.765389 0.784277
+vt 0.386637 0.932389
+vt 0.756917 0.785962
+vt 0.378001 0.932389
+vt 0.748280 0.785962
+vt 0.369363 0.932389
+vt 0.739809 0.784277
+vt 0.360726 0.932389
+vt 0.731830 0.780972
+vt 0.352089 0.932388
+vt 0.724648 0.776174
+vt 0.343452 0.932388
+vt 0.618830 0.816892
+vt 0.628978 0.816892
+vt 0.608683 0.816892
+vt 0.598535 0.816892
+vt 0.588388 0.816892
+vt 0.578240 0.816892
+vt 0.568093 0.816892
+vt 0.557945 0.816892
+vt 0.547798 0.816892
+vt 0.537650 0.816892
+vt 0.527503 0.816892
+vt 0.517355 0.816892
+vt 0.507208 0.816892
+vt 0.497060 0.816892
+vt 0.486913 0.816892
+vt 0.476765 0.816892
+vt 0.466618 0.816892
+vt 0.456470 0.816892
+vt 0.446323 0.816892
+vt 0.436175 0.816892
+vt 0.426028 0.816892
+vt 0.415880 0.816892
+vt 0.405733 0.816892
+vt 0.395585 0.816892
+vt 0.385438 0.816892
+vt 0.375290 0.816892
+vt 0.365143 0.816892
+vt 0.354995 0.816892
+vt 0.344848 0.816892
+vt 0.334700 0.816892
+vt 0.324553 0.816892
+vt 0.314405 0.816892
+vt 0.639125 0.816892
+vt 0.517095 0.669905
+vt 0.602565 0.946486
+vt 0.334815 0.946485
+vt 0.515410 0.678377
+vt 0.611202 0.946485
+vt 0.520400 0.661926
+vt 0.593928 0.946485
+vt 0.525198 0.654744
+vt 0.585291 0.946486
+vt 0.531306 0.648637
+vt 0.576654 0.946486
+vt 0.568017 0.946486
+vt 0.538487 0.643839
+vt 0.546467 0.640533
+vt 0.559379 0.946486
+vt 0.554938 0.638848
+vt 0.507247 0.729756
+vt 0.550742 0.946486
+vt 0.481149 0.728166
+vt 0.542105 0.971165
+vt 0.472678 0.726481
+vt 0.533468 0.971165
+vt 0.464698 0.723176
+vt 0.524831 0.971165
+vt 0.457517 0.718377
+vt 0.516194 0.971165
+vt 0.507557 0.971165
+vt 0.451409 0.712270
+vt 0.446611 0.705089
+vt 0.498920 0.971165
+vt 0.443305 0.697109
+vt 0.490283 0.971165
+vt 0.441620 0.688638
+vt 0.481646 0.971165
+vt 0.473008 0.971165
+vt 0.441620 0.680001
+vt 0.443305 0.671530
+vt 0.464371 0.971165
+vt 0.446610 0.663550
+vt 0.455734 0.971165
+vt 0.451409 0.656369
+vt 0.447097 0.971165
+vt 0.438460 0.971165
+vt 0.457516 0.650261
+vt 0.464698 0.645463
+vt 0.429823 0.971165
+vt 0.472677 0.642157
+vt 0.421186 0.971165
+vt 0.412549 0.971165
+vt 0.481149 0.640472
+vt 0.507247 0.638882
+vt 0.554938 0.726542
+vt 0.403912 0.946486
+vt 0.546467 0.724857
+vt 0.395275 0.946486
+vt 0.538487 0.721552
+vt 0.386637 0.946486
+vt 0.378001 0.946486
+vt 0.531306 0.716753
+vt 0.525198 0.710646
+vt 0.369363 0.946486
+vt 0.520400 0.703464
+vt 0.360726 0.946485
+vt 0.517095 0.695485
+vt 0.352089 0.946485
+vt 0.343452 0.946485
+vt 0.515410 0.687014
+vt 0.509722 0.667669
+vt 0.618830 0.797969
+vt 0.507742 0.677621
+vt 0.628978 0.797969
+vt 0.513605 0.658294
+vt 0.608683 0.797969
+vt 0.519242 0.649857
+vt 0.598535 0.797969
+vt 0.588388 0.797969
+vt 0.526418 0.642681
+vt 0.578240 0.797969
+vt 0.534855 0.637043
+vt 0.568093 0.797969
+vt 0.544230 0.633160
+vt 0.554183 0.631180
+vt 0.507028 0.737458
+vt 0.557945 0.797969
+vt 0.480393 0.735834
+vt 0.547798 0.773289
+vt 0.470441 0.733854
+vt 0.537650 0.773289
+vt 0.461066 0.729971
+vt 0.527503 0.773289
+vt 0.452629 0.724334
+vt 0.517355 0.773289
+vt 0.507208 0.773290
+vt 0.445453 0.717158
+vt 0.497060 0.773290
+vt 0.439816 0.708721
+vt 0.486913 0.773290
+vt 0.435932 0.699346
+vt 0.433953 0.689393
+vt 0.476765 0.773290
+vt 0.433953 0.679246
+vt 0.466618 0.773290
+vt 0.435932 0.669293
+vt 0.456470 0.773290
+vt 0.439815 0.659918
+vt 0.446323 0.773289
+vt 0.436175 0.773289
+vt 0.445453 0.651481
+vt 0.426028 0.773289
+vt 0.452628 0.644305
+vt 0.415880 0.773289
+vt 0.461066 0.638668
+vt 0.470441 0.634784
+vt 0.405733 0.773289
+vt 0.480393 0.632805
+vt 0.395585 0.773289
+vt 0.507028 0.631180
+vt 0.554183 0.734210
+vt 0.385438 0.797969
+vt 0.544230 0.732230
+vt 0.375290 0.797969
+vt 0.534855 0.728347
+vt 0.365143 0.797969
+vt 0.354995 0.797969
+vt 0.526418 0.722709
+vt 0.344848 0.797969
+vt 0.519243 0.715534
+vt 0.334700 0.797969
+vt 0.513605 0.707096
+vt 0.509721 0.697721
+vt 0.324553 0.797969
+vt 0.507742 0.687769
+vt 0.314405 0.797969
+vt 0.639125 0.797969
+vt 0.908735 0.903617
+vt 0.874676 0.931567
+vt 0.334815 0.932388
+vt 0.611202 0.932388
+vt 0.869878 0.924386
+vt 0.602565 0.932389
+vt 0.866572 0.916406
+vt 0.593928 0.932389
+vt 0.864888 0.907935
+vt 0.585291 0.932389
+vt 0.864888 0.899298
+vt 0.576654 0.932389
+vt 0.866573 0.890827
+vt 0.568017 0.932389
+vt 0.869878 0.882847
+vt 0.559379 0.932389
+vt 0.874676 0.875666
+vt 0.550742 0.932389
+vt 0.880784 0.869558
+vt 0.542105 0.932389
+vt 0.887965 0.864760
+vt 0.533468 0.932389
+vt 0.895945 0.861455
+vt 0.524831 0.932389
+vt 0.904416 0.859770
+vt 0.516194 0.932389
+vt 0.913053 0.859770
+vt 0.507557 0.932389
+vt 0.921524 0.861455
+vt 0.498920 0.932389
+vt 0.929504 0.864760
+vt 0.490283 0.932389
+vt 0.936685 0.869559
+vt 0.481646 0.932389
+vt 0.942793 0.875666
+vt 0.473009 0.932389
+vt 0.947591 0.882847
+vt 0.464371 0.932389
+vt 0.950896 0.890827
+vt 0.455734 0.932389
+vt 0.952581 0.899298
+vt 0.447097 0.932389
+vt 0.952581 0.907935
+vt 0.438460 0.932389
+vt 0.950896 0.916406
+vt 0.429823 0.932389
+vt 0.947591 0.924386
+vt 0.421186 0.932389
+vt 0.942792 0.931568
+vt 0.412549 0.932389
+vt 0.936685 0.937675
+vt 0.403912 0.932389
+vt 0.929504 0.942473
+vt 0.395275 0.932389
+vt 0.921524 0.945779
+vt 0.386637 0.932389
+vt 0.913053 0.947464
+vt 0.378001 0.932389
+vt 0.904416 0.947464
+vt 0.369363 0.932389
+vt 0.895945 0.945778
+vt 0.360726 0.932389
+vt 0.887965 0.942473
+vt 0.352089 0.932388
+vt 0.880783 0.937675
+vt 0.343452 0.932388
+vt 0.618830 0.816892
+vt 0.628978 0.816892
+vt 0.608683 0.816892
+vt 0.598535 0.816892
+vt 0.588388 0.816892
+vt 0.578240 0.816892
+vt 0.568093 0.816892
+vt 0.557945 0.816892
+vt 0.547798 0.816892
+vt 0.537650 0.816892
+vt 0.527503 0.816892
+vt 0.517355 0.816892
+vt 0.507208 0.816892
+vt 0.497060 0.816892
+vt 0.486913 0.816892
+vt 0.476765 0.816892
+vt 0.466618 0.816892
+vt 0.456470 0.816892
+vt 0.446323 0.816892
+vt 0.436175 0.816892
+vt 0.426028 0.816892
+vt 0.415880 0.816892
+vt 0.405733 0.816892
+vt 0.395585 0.816892
+vt 0.385438 0.816892
+vt 0.375290 0.816892
+vt 0.365143 0.816892
+vt 0.354995 0.816892
+vt 0.344848 0.816892
+vt 0.334700 0.816892
+vt 0.324553 0.816892
+vt 0.314405 0.816892
+vt 0.639125 0.816892
+vt 0.517095 0.669905
+vt 0.602565 0.946486
+vt 0.334815 0.946485
+vt 0.515410 0.678377
+vt 0.611202 0.946485
+vt 0.520400 0.661926
+vt 0.593928 0.946485
+vt 0.525198 0.654744
+vt 0.585291 0.946486
+vt 0.531306 0.648637
+vt 0.576654 0.946486
+vt 0.568017 0.946486
+vt 0.538487 0.643839
+vt 0.546467 0.640533
+vt 0.559379 0.946486
+vt 0.554938 0.638848
+vt 0.507247 0.729756
+vt 0.550742 0.946486
+vt 0.481149 0.728166
+vt 0.542105 0.971165
+vt 0.472678 0.726481
+vt 0.533468 0.971165
+vt 0.464698 0.723176
+vt 0.524831 0.971165
+vt 0.457517 0.718377
+vt 0.516194 0.971165
+vt 0.507557 0.971165
+vt 0.451409 0.712270
+vt 0.446611 0.705089
+vt 0.498920 0.971165
+vt 0.443305 0.697109
+vt 0.490283 0.971165
+vt 0.441620 0.688638
+vt 0.481646 0.971165
+vt 0.473008 0.971165
+vt 0.441620 0.680001
+vt 0.443305 0.671530
+vt 0.464371 0.971165
+vt 0.446610 0.663550
+vt 0.455734 0.971165
+vt 0.451409 0.656369
+vt 0.447097 0.971165
+vt 0.438460 0.971165
+vt 0.457516 0.650261
+vt 0.464698 0.645463
+vt 0.429823 0.971165
+vt 0.472677 0.642157
+vt 0.421186 0.971165
+vt 0.412549 0.971165
+vt 0.481149 0.640472
+vt 0.507247 0.638882
+vt 0.554938 0.726542
+vt 0.403912 0.946486
+vt 0.546467 0.724857
+vt 0.395275 0.946486
+vt 0.538487 0.721552
+vt 0.386637 0.946486
+vt 0.378001 0.946486
+vt 0.531306 0.716753
+vt 0.525198 0.710646
+vt 0.369363 0.946486
+vt 0.520400 0.703464
+vt 0.360726 0.946485
+vt 0.517095 0.695485
+vt 0.352089 0.946485
+vt 0.343452 0.946485
+vt 0.515410 0.687014
+vt 0.509722 0.667669
+vt 0.618830 0.797969
+vt 0.507742 0.677621
+vt 0.628978 0.797969
+vt 0.513605 0.658294
+vt 0.608683 0.797969
+vt 0.519242 0.649857
+vt 0.598535 0.797969
+vt 0.588388 0.797969
+vt 0.526418 0.642681
+vt 0.578240 0.797969
+vt 0.534855 0.637043
+vt 0.568093 0.797969
+vt 0.544230 0.633160
+vt 0.554183 0.631180
+vt 0.507028 0.737458
+vt 0.557945 0.797969
+vt 0.480393 0.735834
+vt 0.547798 0.773289
+vt 0.470441 0.733854
+vt 0.537650 0.773289
+vt 0.461066 0.729971
+vt 0.527503 0.773289
+vt 0.452629 0.724334
+vt 0.517355 0.773289
+vt 0.507208 0.773290
+vt 0.445453 0.717158
+vt 0.497060 0.773290
+vt 0.439816 0.708721
+vt 0.486913 0.773290
+vt 0.435932 0.699346
+vt 0.433953 0.689393
+vt 0.476765 0.773290
+vt 0.433953 0.679246
+vt 0.466618 0.773290
+vt 0.435932 0.669293
+vt 0.456470 0.773290
+vt 0.439815 0.659918
+vt 0.446323 0.773289
+vt 0.436175 0.773289
+vt 0.445453 0.651481
+vt 0.426028 0.773289
+vt 0.452628 0.644305
+vt 0.415880 0.773289
+vt 0.461066 0.638668
+vt 0.470441 0.634784
+vt 0.405733 0.773289
+vt 0.480393 0.632805
+vt 0.395585 0.773289
+vt 0.507028 0.631180
+vt 0.554183 0.734210
+vt 0.385438 0.797969
+vt 0.544230 0.732230
+vt 0.375290 0.797969
+vt 0.534855 0.728347
+vt 0.365143 0.797969
+vt 0.354995 0.797969
+vt 0.526418 0.722709
+vt 0.344848 0.797969
+vt 0.519243 0.715534
+vt 0.334700 0.797969
+vt 0.513605 0.707096
+vt 0.509721 0.697721
+vt 0.324553 0.797969
+vt 0.507742 0.687769
+vt 0.314405 0.797969
+vt 0.639125 0.797969
+vt 0.908268 0.746864
+vt 0.874210 0.774814
+vt 0.334815 0.932388
+vt 0.611202 0.932388
+vt 0.869411 0.767633
+vt 0.602565 0.932389
+vt 0.866106 0.759653
+vt 0.593928 0.932389
+vt 0.864421 0.751182
+vt 0.585291 0.932389
+vt 0.864421 0.742545
+vt 0.576654 0.932389
+vt 0.866106 0.734074
+vt 0.568017 0.932389
+vt 0.869412 0.726094
+vt 0.559379 0.932389
+vt 0.874210 0.718913
+vt 0.550742 0.932389
+vt 0.880318 0.712805
+vt 0.542105 0.932389
+vt 0.887499 0.708007
+vt 0.533468 0.932389
+vt 0.895479 0.704702
+vt 0.524831 0.932389
+vt 0.903950 0.703017
+vt 0.516194 0.932389
+vt 0.912587 0.703017
+vt 0.507557 0.932389
+vt 0.921058 0.704702
+vt 0.498920 0.932389
+vt 0.929038 0.708007
+vt 0.490283 0.932389
+vt 0.936219 0.712806
+vt 0.481646 0.932389
+vt 0.942326 0.718913
+vt 0.473009 0.932389
+vt 0.947125 0.726095
+vt 0.464371 0.932389
+vt 0.950430 0.734074
+vt 0.455734 0.932389
+vt 0.952115 0.742545
+vt 0.447097 0.932389
+vt 0.952115 0.751182
+vt 0.438460 0.932389
+vt 0.950430 0.759654
+vt 0.429823 0.932389
+vt 0.947125 0.767633
+vt 0.421186 0.932389
+vt 0.942326 0.774815
+vt 0.412549 0.932389
+vt 0.936219 0.780922
+vt 0.403912 0.932389
+vt 0.929037 0.785721
+vt 0.395275 0.932389
+vt 0.921058 0.789026
+vt 0.386637 0.932389
+vt 0.912586 0.790711
+vt 0.378001 0.932389
+vt 0.903949 0.790711
+vt 0.369363 0.932389
+vt 0.895478 0.789025
+vt 0.360726 0.932389
+vt 0.887499 0.785720
+vt 0.352089 0.932388
+vt 0.880317 0.780922
+vt 0.343452 0.932388
+vt 0.618830 0.816892
+vt 0.628978 0.816892
+vt 0.608683 0.816892
+vt 0.598535 0.816892
+vt 0.588388 0.816892
+vt 0.578240 0.816892
+vt 0.568093 0.816892
+vt 0.557945 0.816892
+vt 0.547798 0.816892
+vt 0.537650 0.816892
+vt 0.527503 0.816892
+vt 0.517355 0.816892
+vt 0.507208 0.816892
+vt 0.497060 0.816892
+vt 0.486913 0.816892
+vt 0.476765 0.816892
+vt 0.466618 0.816892
+vt 0.456470 0.816892
+vt 0.446323 0.816892
+vt 0.436175 0.816892
+vt 0.426028 0.816892
+vt 0.415880 0.816892
+vt 0.405733 0.816892
+vt 0.395585 0.816892
+vt 0.385438 0.816892
+vt 0.375290 0.816892
+vt 0.365143 0.816892
+vt 0.354995 0.816892
+vt 0.344848 0.816892
+vt 0.334700 0.816892
+vt 0.324553 0.816892
+vt 0.314405 0.816892
+vt 0.639125 0.816892
+vt 0.517095 0.669905
+vt 0.602565 0.946486
+vt 0.334815 0.946485
+vt 0.515410 0.678377
+vt 0.611202 0.946485
+vt 0.520400 0.661926
+vt 0.593928 0.946485
+vt 0.525198 0.654744
+vt 0.585291 0.946486
+vt 0.531306 0.648637
+vt 0.576654 0.946486
+vt 0.568017 0.946486
+vt 0.538487 0.643839
+vt 0.546467 0.640533
+vt 0.559379 0.946486
+vt 0.554938 0.638848
+vt 0.507247 0.729756
+vt 0.550742 0.946486
+vt 0.481149 0.728166
+vt 0.542105 0.971165
+vt 0.472678 0.726481
+vt 0.533468 0.971165
+vt 0.464698 0.723176
+vt 0.524831 0.971165
+vt 0.457517 0.718377
+vt 0.516194 0.971165
+vt 0.507557 0.971165
+vt 0.451409 0.712270
+vt 0.446611 0.705089
+vt 0.498920 0.971165
+vt 0.443305 0.697109
+vt 0.490283 0.971165
+vt 0.441620 0.688638
+vt 0.481646 0.971165
+vt 0.473008 0.971165
+vt 0.441620 0.680001
+vt 0.443305 0.671530
+vt 0.464371 0.971165
+vt 0.446610 0.663550
+vt 0.455734 0.971165
+vt 0.451409 0.656369
+vt 0.447097 0.971165
+vt 0.438460 0.971165
+vt 0.457516 0.650261
+vt 0.464698 0.645463
+vt 0.429823 0.971165
+vt 0.472677 0.642157
+vt 0.421186 0.971165
+vt 0.412549 0.971165
+vt 0.481149 0.640472
+vt 0.507247 0.638882
+vt 0.554938 0.726542
+vt 0.403912 0.946486
+vt 0.546467 0.724857
+vt 0.395275 0.946486
+vt 0.538487 0.721552
+vt 0.386637 0.946486
+vt 0.378001 0.946486
+vt 0.531306 0.716753
+vt 0.525198 0.710646
+vt 0.369363 0.946486
+vt 0.520400 0.703464
+vt 0.360726 0.946485
+vt 0.517095 0.695485
+vt 0.352089 0.946485
+vt 0.343452 0.946485
+vt 0.515410 0.687014
+vt 0.509722 0.667669
+vt 0.618830 0.797969
+vt 0.507742 0.677621
+vt 0.628978 0.797969
+vt 0.513605 0.658294
+vt 0.608683 0.797969
+vt 0.519242 0.649857
+vt 0.598535 0.797969
+vt 0.588388 0.797969
+vt 0.526418 0.642681
+vt 0.578240 0.797969
+vt 0.534855 0.637043
+vt 0.568093 0.797969
+vt 0.544230 0.633160
+vt 0.554183 0.631180
+vt 0.507028 0.737458
+vt 0.557945 0.797969
+vt 0.480393 0.735834
+vt 0.547798 0.773289
+vt 0.470441 0.733854
+vt 0.537650 0.773289
+vt 0.461066 0.729971
+vt 0.527503 0.773289
+vt 0.452629 0.724334
+vt 0.517355 0.773289
+vt 0.507208 0.773290
+vt 0.445453 0.717158
+vt 0.497060 0.773290
+vt 0.439816 0.708721
+vt 0.486913 0.773290
+vt 0.435932 0.699346
+vt 0.433953 0.689393
+vt 0.476765 0.773290
+vt 0.433953 0.679246
+vt 0.466618 0.773290
+vt 0.435932 0.669293
+vt 0.456470 0.773290
+vt 0.439815 0.659918
+vt 0.446323 0.773289
+vt 0.436175 0.773289
+vt 0.445453 0.651481
+vt 0.426028 0.773289
+vt 0.452628 0.644305
+vt 0.415880 0.773289
+vt 0.461066 0.638668
+vt 0.470441 0.634784
+vt 0.405733 0.773289
+vt 0.480393 0.632805
+vt 0.395585 0.773289
+vt 0.507028 0.631180
+vt 0.554183 0.734210
+vt 0.385438 0.797969
+vt 0.544230 0.732230
+vt 0.375290 0.797969
+vt 0.534855 0.728347
+vt 0.365143 0.797969
+vt 0.354995 0.797969
+vt 0.526418 0.722709
+vt 0.344848 0.797969
+vt 0.519243 0.715534
+vt 0.334700 0.797969
+vt 0.513605 0.707096
+vt 0.509721 0.697721
+vt 0.324553 0.797969
+vt 0.507742 0.687769
+vt 0.314405 0.797969
+vt 0.639125 0.797969
+s 0
+g Cube_Material
+usemtl Material
+f 6/11/1 14/22/1 10/18/1 2/3/1
+f 1/1/2 9/17/2 13/21/2 5/9/2
+f 12/20/3 16/24/3 15/23/3 11/19/3
+f 10/18/3 14/22/3 13/21/3 9/17/3
+f 3/5/2 11/19/2 15/23/2 7/13/2
+f 8/15/1 16/24/1 12/20/1 4/7/1
+f 7/14/4 15/23/4 16/24/4 8/16/4
+f 5/10/4 13/21/4 14/22/4 6/12/4
+f 4/8/5 12/20/5 11/19/5 3/6/5
+f 2/4/5 10/18/5 9/17/5 1/2/5
+f 18/29/3 17/25/3 19/33/3 20/34/3
+f 33/57/4 34/58/4 36/60/4 35/59/4
+f 24/44/6 23/43/6 21/35/6 22/39/6
+f 25/45/1 29/53/1 32/56/1 28/51/1
+f 17/26/1 21/36/1 23/43/1 19/33/1
+f 22/40/2 18/30/2 20/34/2 24/44/2
+f 25/46/5 17/27/5 18/31/5 26/48/5
+f 27/50/5 22/41/5 21/37/5 28/52/5
+f 26/47/5 18/32/5 22/42/5 27/49/5
+f 28/51/5 21/38/5 17/28/5 25/45/5
+f 31/55/5 32/56/5 29/53/5 30/54/5
+f 27/49/2 31/55/2 30/54/2 26/47/2
+f 28/52/6 32/56/6 31/55/6 27/50/6
+f 26/48/3 30/54/3 29/53/3 25/46/3
+f 33/57/4 20/34/4 19/33/4 34/58/4
+f 35/59/4 24/44/4 20/34/4 33/57/4
+f 36/60/4 23/43/4 24/44/4 35/59/4
+f 34/58/4 19/33/4 23/43/4 36/60/4
+f 37/61/4 39/65/4 38/62/4
+f 37/61/4 40/67/4 39/65/4
+f 37/61/4 41/69/4 40/67/4
+f 37/61/4 42/71/4 41/69/4
+f 37/61/4 43/73/4 42/71/4
+f 37/61/4 44/75/4 43/73/4
+f 37/61/4 45/77/4 44/75/4
+f 37/61/4 46/79/4 45/77/4
+f 37/61/4 47/81/4 46/79/4
+f 37/61/4 48/83/4 47/81/4
+f 37/61/4 49/85/4 48/83/4
+f 37/61/4 50/87/4 49/85/4
+f 37/61/4 51/89/4 50/87/4
+f 37/61/4 52/91/4 51/89/4
+f 37/61/4 53/93/4 52/91/4
+f 37/61/4 54/95/4 53/93/4
+f 37/61/4 55/97/4 54/95/4
+f 37/61/4 56/99/4 55/97/4
+f 37/61/4 57/101/4 56/99/4
+f 37/61/4 58/103/4 57/101/4
+f 37/61/4 59/105/4 58/103/4
+f 37/61/4 60/107/4 59/105/4
+f 37/61/4 61/109/4 60/107/4
+f 37/61/4 62/111/4 61/109/4
+f 37/61/4 63/113/4 62/111/4
+f 37/61/4 64/115/4 63/113/4
+f 37/61/4 65/117/4 64/115/4
+f 37/61/4 66/119/4 65/117/4
+f 37/61/4 67/121/4 66/119/4
+f 37/61/4 68/123/4 67/121/4
+f 37/61/4 69/125/4 68/123/4
+f 37/61/4 38/62/4 69/125/4
+f 99/156/7 163/287/7 164/290/7 100/157/7
+f 84/141/8 148/256/8 149/259/8 85/142/8
+f 59/106/9 123/205/9 122/202/9 58/104/9
+f 38/63/10 103/162/10 133/225/10 69/126/10
+f 44/76/11 108/174/11 107/171/11 43/74/11
+f 91/148/11 155/270/11 156/273/11 92/149/11
+f 76/133/12 140/239/12 141/243/12 77/134/12
+f 55/98/13 119/197/13 118/194/13 54/96/13
+f 66/120/14 130/220/14 129/217/14 65/118/14
+f 98/155/15 162/285/15 163/287/15 99/156/15
+f 83/140/16 147/254/16 148/256/16 84/141/16
+f 40/68/17 104/166/17 102/161/17 39/66/17
+f 51/90/15 115/189/15 114/186/15 50/88/15
+f 62/112/18 126/212/18 125/208/18 61/110/18
+f 90/147/19 154/268/19 155/270/19 91/148/19
+f 75/132/20 139/237/20 140/239/20 76/133/20
+f 47/82/21 111/181/21 110/179/21 46/80/21
+f 97/154/22 161/283/22 162/285/22 98/155/22
+f 82/139/23 146/252/23 147/254/23 83/140/23
+f 58/104/24 122/202/24 121/201/24 57/102/24
+f 69/126/8 133/225/8 132/224/8 68/124/8
+f 89/146/25 153/266/25 154/268/25 90/147/25
+f 43/74/19 107/171/19 106/170/19 42/72/19
+f 54/96/26 118/194/26 117/193/26 53/94/26
+f 74/131/9 138/235/9 139/237/9 75/132/9
+f 65/118/27 129/217/27 128/216/27 64/116/27
+f 96/153/28 160/282/28 161/283/28 97/154/28
+f 81/138/29 145/251/29 146/252/29 82/139/29
+f 50/88/22 114/186/22 113/185/22 49/86/22
+f 88/145/30 152/265/30 153/266/30 89/146/30
+f 61/110/12 125/208/12 124/207/12 60/108/12
+f 73/130/24 137/234/24 138/235/24 74/131/24
+f 102/160/4 134/227/4 135/229/4 103/163/4
+f 104/165/4 136/231/4 134/227/4 102/160/4
+f 105/167/4 137/233/4 136/231/4 104/165/4
+f 106/169/4 138/236/4 137/233/4 105/167/4
+f 107/172/4 139/238/4 138/236/4 106/169/4
+f 108/173/4 140/240/4 139/238/4 107/172/4
+f 109/175/4 141/241/4 140/240/4 108/173/4
+f 110/178/31 142/244/31 141/242/31 109/176/31
+f 111/180/4 143/246/4 142/244/4 110/178/4
+f 112/182/4 144/248/4 143/246/4 111/180/4
+f 113/184/4 145/250/4 144/248/4 112/182/4
+f 114/187/4 146/253/4 145/250/4 113/184/4
+f 115/188/4 147/255/4 146/253/4 114/187/4
+f 116/190/4 148/257/4 147/255/4 115/188/4
+f 117/192/4 149/258/4 148/257/4 116/190/4
+f 118/195/4 150/260/4 149/258/4 117/192/4
+f 119/196/4 151/262/4 150/260/4 118/195/4
+f 120/198/4 152/264/4 151/262/4 119/196/4
+f 121/200/4 153/267/4 152/264/4 120/198/4
+f 122/203/4 154/269/4 153/267/4 121/200/4
+f 123/204/4 155/271/4 154/269/4 122/203/4
+f 124/206/4 156/272/4 155/271/4 123/204/4
+f 125/209/4 157/274/4 156/272/4 124/206/4
+f 126/210/31 158/276/31 157/274/31 125/209/31
+f 127/213/4 159/279/4 158/277/4 126/211/4
+f 128/215/4 160/281/4 159/279/4 127/213/4
+f 129/218/4 161/284/4 160/281/4 128/215/4
+f 130/219/4 162/286/4 161/284/4 129/218/4
+f 131/221/4 163/288/4 162/286/4 130/219/4
+f 132/223/4 164/289/4 163/288/4 131/221/4
+f 133/226/4 165/291/4 164/289/4 132/223/4
+f 103/163/4 135/229/4 165/291/4 133/226/4
+f 48/84/32 112/183/32 111/181/32 47/82/32
+f 77/134/33 141/243/33 142/245/33 78/135/33
+f 92/149/34 156/273/34 157/275/34 93/150/34
+f 63/114/35 127/214/35 126/212/35 62/112/35
+f 52/92/7 116/191/7 115/189/7 51/90/7
+f 71/128/36 135/230/36 134/228/36 70/127/36
+f 41/70/30 105/168/30 104/166/30 40/68/30
+f 85/142/10 149/259/10 150/261/10 86/143/10
+f 100/157/37 164/290/37 165/292/37 101/158/37
+f 67/122/23 131/222/23 130/220/23 66/120/23
+f 56/100/38 120/199/38 119/197/38 55/98/38
+f 78/135/35 142/245/35 143/247/35 79/136/35
+f 93/150/39 157/275/39 158/278/39 94/151/39
+f 45/78/34 109/177/34 108/174/34 44/76/34
+f 70/127/38 134/228/38 136/232/38 72/129/38
+f 60/108/20 124/207/20 123/205/20 59/106/20
+f 86/143/40 150/261/40 151/263/40 87/144/40
+f 101/159/26 165/293/26 135/230/26 71/128/26
+f 49/86/28 113/185/28 112/183/28 48/84/28
+f 79/136/41 143/247/41 144/249/41 80/137/41
+f 94/151/21 158/278/21 159/280/21 95/152/21
+f 64/116/41 128/216/41 127/214/41 63/114/41
+f 53/94/37 117/193/37 116/191/37 52/92/37
+f 72/129/42 136/232/42 137/234/42 73/130/42
+f 42/72/43 106/170/43 105/168/43 41/70/43
+f 87/144/17 151/263/17 152/265/17 88/145/17
+f 68/124/16 132/224/16 131/222/16 67/122/16
+f 39/66/40 102/161/40 103/164/40 38/64/40
+f 57/102/42 121/201/42 120/199/42 56/100/42
+f 80/137/27 144/249/27 145/251/27 81/138/27
+f 95/152/32 159/280/32 160/282/32 96/153/32
+f 46/80/44 110/179/44 109/177/44 45/78/44
+f 166/294/4 168/298/4 167/295/4
+f 166/294/4 169/300/4 168/298/4
+f 166/294/4 170/302/4 169/300/4
+f 166/294/4 171/304/4 170/302/4
+f 166/294/4 172/306/4 171/304/4
+f 166/294/4 173/308/4 172/306/4
+f 166/294/4 174/310/4 173/308/4
+f 166/294/4 175/312/4 174/310/4
+f 166/294/4 176/314/4 175/312/4
+f 166/294/4 177/316/4 176/314/4
+f 166/294/4 178/318/4 177/316/4
+f 166/294/4 179/320/4 178/318/4
+f 166/294/4 180/322/4 179/320/4
+f 166/294/4 181/324/4 180/322/4
+f 166/294/4 182/326/4 181/324/4
+f 166/294/4 183/328/4 182/326/4
+f 166/294/4 184/330/4 183/328/4
+f 166/294/4 185/332/4 184/330/4
+f 166/294/4 186/334/4 185/332/4
+f 166/294/4 187/336/4 186/334/4
+f 166/294/4 188/338/4 187/336/4
+f 166/294/4 189/340/4 188/338/4
+f 166/294/4 190/342/4 189/340/4
+f 166/294/4 191/344/4 190/342/4
+f 166/294/4 192/346/4 191/344/4
+f 166/294/4 193/348/4 192/346/4
+f 166/294/4 194/350/4 193/348/4
+f 166/294/4 195/352/4 194/350/4
+f 166/294/4 196/354/4 195/352/4
+f 166/294/4 197/356/4 196/354/4
+f 166/294/4 198/358/4 197/356/4
+f 166/294/4 167/295/4 198/358/4
+f 228/389/7 292/520/7 293/523/7 229/390/7
+f 213/374/8 277/489/8 278/492/8 214/375/8
+f 188/339/9 252/438/9 251/435/9 187/337/9
+f 167/296/10 232/395/10 262/458/10 198/359/10
+f 173/309/11 237/407/11 236/404/11 172/307/11
+f 220/381/11 284/503/11 285/506/11 221/382/11
+f 205/366/12 269/472/12 270/476/12 206/367/12
+f 184/331/13 248/430/13 247/427/13 183/329/13
+f 195/353/14 259/453/14 258/450/14 194/351/14
+f 227/388/15 291/518/15 292/520/15 228/389/15
+f 212/373/16 276/487/16 277/489/16 213/374/16
+f 169/301/17 233/399/17 231/394/17 168/299/17
+f 180/323/15 244/422/15 243/419/15 179/321/15
+f 191/345/18 255/445/18 254/441/18 190/343/18
+f 219/380/19 283/501/19 284/503/19 220/381/19
+f 204/365/20 268/470/20 269/472/20 205/366/20
+f 176/315/21 240/414/21 239/412/21 175/313/21
+f 226/387/22 290/516/22 291/518/22 227/388/22
+f 211/372/23 275/485/23 276/487/23 212/373/23
+f 187/337/24 251/435/24 250/434/24 186/335/24
+f 198/359/8 262/458/8 261/457/8 197/357/8
+f 218/379/25 282/499/25 283/501/25 219/380/25
+f 172/307/19 236/404/19 235/403/19 171/305/19
+f 183/329/26 247/427/26 246/426/26 182/327/26
+f 203/364/9 267/468/9 268/470/9 204/365/9
+f 194/351/27 258/450/27 257/449/27 193/349/27
+f 225/386/28 289/515/28 290/516/28 226/387/28
+f 210/371/29 274/484/29 275/485/29 211/372/29
+f 179/321/22 243/419/22 242/418/22 178/319/22
+f 217/378/30 281/498/30 282/499/30 218/379/30
+f 190/343/12 254/441/12 253/440/12 189/341/12
+f 202/363/24 266/467/24 267/468/24 203/364/24
+f 231/393/4 263/460/4 264/462/4 232/396/4
+f 233/398/4 265/464/4 263/460/4 231/393/4
+f 234/400/4 266/466/4 265/464/4 233/398/4
+f 235/402/4 267/469/4 266/466/4 234/400/4
+f 236/405/4 268/471/4 267/469/4 235/402/4
+f 237/406/4 269/473/4 268/471/4 236/405/4
+f 238/408/4 270/474/4 269/473/4 237/406/4
+f 239/411/31 271/477/31 270/475/31 238/409/31
+f 240/413/4 272/479/4 271/477/4 239/411/4
+f 241/415/4 273/481/4 272/479/4 240/413/4
+f 242/417/4 274/483/4 273/481/4 241/415/4
+f 243/420/4 275/486/4 274/483/4 242/417/4
+f 244/421/4 276/488/4 275/486/4 243/420/4
+f 245/423/4 277/490/4 276/488/4 244/421/4
+f 246/425/4 278/491/4 277/490/4 245/423/4
+f 247/428/4 279/493/4 278/491/4 246/425/4
+f 248/429/4 280/495/4 279/493/4 247/428/4
+f 249/431/4 281/497/4 280/495/4 248/429/4
+f 250/433/4 282/500/4 281/497/4 249/431/4
+f 251/436/4 283/502/4 282/500/4 250/433/4
+f 252/437/4 284/504/4 283/502/4 251/436/4
+f 253/439/4 285/505/4 284/504/4 252/437/4
+f 254/442/4 286/507/4 285/505/4 253/439/4
+f 255/443/31 287/509/31 286/507/31 254/442/31
+f 256/446/4 288/512/4 287/510/4 255/444/4
+f 257/448/4 289/514/4 288/512/4 256/446/4
+f 258/451/4 290/517/4 289/514/4 257/448/4
+f 259/452/4 291/519/4 290/517/4 258/451/4
+f 260/454/4 292/521/4 291/519/4 259/452/4
+f 261/456/4 293/522/4 292/521/4 260/454/4
+f 262/459/4 294/524/4 293/522/4 261/456/4
+f 232/396/4 264/462/4 294/524/4 262/459/4
+f 177/317/32 241/416/32 240/414/32 176/315/32
+f 206/367/33 270/476/33 271/478/33 207/368/33
+f 221/382/34 285/506/34 286/508/34 222/383/34
+f 192/347/35 256/447/35 255/445/35 191/345/35
+f 181/325/7 245/424/7 244/422/7 180/323/7
+f 200/361/36 264/463/36 263/461/36 199/360/36
+f 170/303/30 234/401/30 233/399/30 169/301/30
+f 214/375/10 278/492/10 279/494/10 215/376/10
+f 229/390/37 293/523/37 294/525/37 230/391/37
+f 196/355/23 260/455/23 259/453/23 195/353/23
+f 185/333/38 249/432/38 248/430/38 184/331/38
+f 207/368/35 271/478/35 272/480/35 208/369/35
+f 222/383/39 286/508/39 287/511/39 223/384/39
+f 174/311/34 238/410/34 237/407/34 173/309/34
+f 199/360/38 263/461/38 265/465/38 201/362/38
+f 189/341/20 253/440/20 252/438/20 188/339/20
+f 215/376/40 279/494/40 280/496/40 216/377/40
+f 230/392/26 294/526/26 264/463/26 200/361/26
+f 178/319/28 242/418/28 241/416/28 177/317/28
+f 208/369/41 272/480/41 273/482/41 209/370/41
+f 223/384/21 287/511/21 288/513/21 224/385/21
+f 193/349/41 257/449/41 256/447/41 192/347/41
+f 182/327/37 246/426/37 245/424/37 181/325/37
+f 201/362/42 265/465/42 266/467/42 202/363/42
+f 171/305/43 235/403/43 234/401/43 170/303/43
+f 216/377/17 280/496/17 281/498/17 217/378/17
+f 197/357/16 261/457/16 260/455/16 196/355/16
+f 168/299/40 231/394/40 232/397/40 167/297/40
+f 186/335/42 250/434/42 249/432/42 185/333/42
+f 209/370/27 273/482/27 274/484/27 210/371/27
+f 224/385/32 288/513/32 289/515/32 225/386/32
+f 175/313/44 239/412/44 238/410/44 174/311/44
+f 295/527/4 297/531/4 296/528/4
+f 295/527/4 298/533/4 297/531/4
+f 295/527/4 299/535/4 298/533/4
+f 295/527/4 300/537/4 299/535/4
+f 295/527/4 301/539/4 300/537/4
+f 295/527/4 302/541/4 301/539/4
+f 295/527/4 303/543/4 302/541/4
+f 295/527/4 304/545/4 303/543/4
+f 295/527/4 305/547/4 304/545/4
+f 295/527/4 306/549/4 305/547/4
+f 295/527/4 307/551/4 306/549/4
+f 295/527/4 308/553/4 307/551/4
+f 295/527/4 309/555/4 308/553/4
+f 295/527/4 310/557/4 309/555/4
+f 295/527/4 311/559/4 310/557/4
+f 295/527/4 312/561/4 311/559/4
+f 295/527/4 313/563/4 312/561/4
+f 295/527/4 314/565/4 313/563/4
+f 295/527/4 315/567/4 314/565/4
+f 295/527/4 316/569/4 315/567/4
+f 295/527/4 317/571/4 316/569/4
+f 295/527/4 318/573/4 317/571/4
+f 295/527/4 319/575/4 318/573/4
+f 295/527/4 320/577/4 319/575/4
+f 295/527/4 321/579/4 320/577/4
+f 295/527/4 322/581/4 321/579/4
+f 295/527/4 323/583/4 322/581/4
+f 295/527/4 324/585/4 323/583/4
+f 295/527/4 325/587/4 324/585/4
+f 295/527/4 326/589/4 325/587/4
+f 295/527/4 327/591/4 326/589/4
+f 295/527/4 296/528/4 327/591/4
+f 357/622/7 421/753/7 422/756/7 358/623/7
+f 342/607/8 406/722/8 407/725/8 343/608/8
+f 317/572/9 381/671/9 380/668/9 316/570/9
+f 296/529/10 361/628/10 391/691/10 327/592/10
+f 302/542/11 366/640/11 365/637/11 301/540/11
+f 349/614/11 413/736/11 414/739/11 350/615/11
+f 334/599/12 398/705/12 399/709/12 335/600/12
+f 313/564/13 377/663/13 376/660/13 312/562/13
+f 324/586/14 388/686/14 387/683/14 323/584/14
+f 356/621/15 420/751/15 421/753/15 357/622/15
+f 341/606/16 405/720/16 406/722/16 342/607/16
+f 298/534/17 362/632/17 360/627/17 297/532/17
+f 309/556/15 373/655/15 372/652/15 308/554/15
+f 320/578/18 384/678/18 383/674/18 319/576/18
+f 348/613/19 412/734/19 413/736/19 349/614/19
+f 333/598/20 397/703/20 398/705/20 334/599/20
+f 305/548/21 369/647/21 368/645/21 304/546/21
+f 355/620/45 419/749/45 420/751/45 356/621/45
+f 340/605/23 404/718/23 405/720/23 341/606/23
+f 316/570/24 380/668/24 379/667/24 315/568/24
+f 327/592/8 391/691/8 390/690/8 326/590/8
+f 347/612/25 411/732/25 412/734/25 348/613/25
+f 301/540/19 365/637/19 364/636/19 300/538/19
+f 312/562/26 376/660/26 375/659/26 311/560/26
+f 332/597/9 396/701/9 397/703/9 333/598/9
+f 323/584/27 387/683/27 386/682/27 322/582/27
+f 354/619/28 418/748/28 419/749/28 355/620/28
+f 339/604/29 403/717/29 404/718/29 340/605/29
+f 308/554/22 372/652/22 371/651/22 307/552/22
+f 346/611/30 410/731/30 411/732/30 347/612/30
+f 319/576/12 383/674/12 382/673/12 318/574/12
+f 331/596/24 395/700/24 396/701/24 332/597/24
+f 360/626/4 392/693/4 393/695/4 361/629/4
+f 362/631/4 394/697/4 392/693/4 360/626/4
+f 363/633/4 395/699/4 394/697/4 362/631/4
+f 364/635/4 396/702/4 395/699/4 363/633/4
+f 365/638/4 397/704/4 396/702/4 364/635/4
+f 366/639/4 398/706/4 397/704/4 365/638/4
+f 367/641/4 399/707/4 398/706/4 366/639/4
+f 368/644/31 400/710/31 399/708/31 367/642/31
+f 369/646/4 401/712/4 400/710/4 368/644/4
+f 370/648/4 402/714/4 401/712/4 369/646/4
+f 371/650/4 403/716/4 402/714/4 370/648/4
+f 372/653/4 404/719/4 403/716/4 371/650/4
+f 373/654/4 405/721/4 404/719/4 372/653/4
+f 374/656/4 406/723/4 405/721/4 373/654/4
+f 375/658/4 407/724/4 406/723/4 374/656/4
+f 376/661/4 408/726/4 407/724/4 375/658/4
+f 377/662/4 409/728/4 408/726/4 376/661/4
+f 378/664/4 410/730/4 409/728/4 377/662/4
+f 379/666/4 411/733/4 410/730/4 378/664/4
+f 380/669/4 412/735/4 411/733/4 379/666/4
+f 381/670/4 413/737/4 412/735/4 380/669/4
+f 382/672/4 414/738/4 413/737/4 381/670/4
+f 383/675/4 415/740/4 414/738/4 382/672/4
+f 384/676/31 416/742/31 415/740/31 383/675/31
+f 385/679/4 417/745/4 416/743/4 384/677/4
+f 386/681/4 418/747/4 417/745/4 385/679/4
+f 387/684/4 419/750/4 418/747/4 386/681/4
+f 388/685/4 420/752/4 419/750/4 387/684/4
+f 389/687/4 421/754/4 420/752/4 388/685/4
+f 390/689/4 422/755/4 421/754/4 389/687/4
+f 391/692/4 423/757/4 422/755/4 390/689/4
+f 361/629/4 393/695/4 423/757/4 391/692/4
+f 306/550/32 370/649/32 369/647/32 305/548/32
+f 335/600/33 399/709/33 400/711/33 336/601/33
+f 350/615/34 414/739/34 415/741/34 351/616/34
+f 321/580/35 385/680/35 384/678/35 320/578/35
+f 310/558/7 374/657/7 373/655/7 309/556/7
+f 329/594/36 393/696/36 392/694/36 328/593/36
+f 299/536/30 363/634/30 362/632/30 298/534/30
+f 343/608/10 407/725/10 408/727/10 344/609/10
+f 358/623/37 422/756/37 423/758/37 359/624/37
+f 325/588/23 389/688/23 388/686/23 324/586/23
+f 314/566/38 378/665/38 377/663/38 313/564/38
+f 336/601/35 400/711/35 401/713/35 337/602/35
+f 351/616/39 415/741/39 416/744/39 352/617/39
+f 303/544/34 367/643/34 366/640/34 302/542/34
+f 328/593/38 392/694/38 394/698/38 330/595/38
+f 318/574/20 382/673/20 381/671/20 317/572/20
+f 344/609/40 408/727/40 409/729/40 345/610/40
+f 359/625/26 423/759/26 393/696/26 329/594/26
+f 307/552/28 371/651/28 370/649/28 306/550/28
+f 337/602/41 401/713/41 402/715/41 338/603/41
+f 352/617/21 416/744/21 417/746/21 353/618/21
+f 322/582/41 386/682/41 385/680/41 321/580/41
+f 311/560/37 375/659/37 374/657/37 310/558/37
+f 330/595/42 394/698/42 395/700/42 331/596/42
+f 300/538/43 364/636/43 363/634/43 299/536/43
+f 345/610/17 409/729/17 410/731/17 346/611/17
+f 326/590/16 390/690/16 389/688/16 325/588/16
+f 297/532/40 360/627/40 361/630/40 296/530/40
+f 315/568/42 379/667/42 378/665/42 314/566/42
+f 338/603/27 402/715/27 403/717/27 339/604/27
+f 353/618/32 417/746/32 418/748/32 354/619/32
+f 304/546/44 368/645/44 367/643/44 303/544/44
+f 424/760/4 426/764/4 425/761/4
+f 424/760/4 427/766/4 426/764/4
+f 424/760/4 428/768/4 427/766/4
+f 424/760/4 429/770/4 428/768/4
+f 424/760/4 430/772/4 429/770/4
+f 424/760/4 431/774/4 430/772/4
+f 424/760/4 432/776/4 431/774/4
+f 424/760/4 433/778/4 432/776/4
+f 424/760/4 434/780/4 433/778/4
+f 424/760/4 435/782/4 434/780/4
+f 424/760/4 436/784/4 435/782/4
+f 424/760/4 437/786/4 436/784/4
+f 424/760/4 438/788/4 437/786/4
+f 424/760/4 439/790/4 438/788/4
+f 424/760/4 440/792/4 439/790/4
+f 424/760/4 441/794/4 440/792/4
+f 424/760/4 442/796/4 441/794/4
+f 424/760/4 443/798/4 442/796/4
+f 424/760/4 444/800/4 443/798/4
+f 424/760/4 445/802/4 444/800/4
+f 424/760/4 446/804/4 445/802/4
+f 424/760/4 447/806/4 446/804/4
+f 424/760/4 448/808/4 447/806/4
+f 424/760/4 449/810/4 448/808/4
+f 424/760/4 450/812/4 449/810/4
+f 424/760/4 451/814/4 450/812/4
+f 424/760/4 452/816/4 451/814/4
+f 424/760/4 453/818/4 452/816/4
+f 424/760/4 454/820/4 453/818/4
+f 424/760/4 455/822/4 454/820/4
+f 424/760/4 456/824/4 455/822/4
+f 424/760/4 425/761/4 456/824/4
+f 486/855/7 550/986/7 551/989/7 487/856/7
+f 471/840/8 535/955/8 536/958/8 472/841/8
+f 446/805/9 510/904/9 509/901/9 445/803/9
+f 425/762/46 490/861/46 520/924/46 456/825/46
+f 431/775/11 495/873/11 494/870/11 430/773/11
+f 478/847/11 542/969/11 543/972/11 479/848/11
+f 463/832/12 527/938/12 528/942/12 464/833/12
+f 442/797/13 506/896/13 505/893/13 441/795/13
+f 453/819/14 517/919/14 516/916/14 452/817/14
+f 485/854/15 549/984/15 550/986/15 486/855/15
+f 470/839/16 534/953/16 535/955/16 471/840/16
+f 427/767/17 491/865/17 489/860/17 426/765/17
+f 438/789/15 502/888/15 501/885/15 437/787/15
+f 449/811/18 513/911/18 512/907/18 448/809/18
+f 477/846/19 541/967/19 542/969/19 478/847/19
+f 462/831/20 526/936/20 527/938/20 463/832/20
+f 434/781/21 498/880/21 497/878/21 433/779/21
+f 484/853/22 548/982/22 549/984/22 485/854/22
+f 469/838/23 533/951/23 534/953/23 470/839/23
+f 445/803/24 509/901/24 508/900/24 444/801/24
+f 456/825/8 520/924/8 519/923/8 455/823/8
+f 476/845/25 540/965/25 541/967/25 477/846/25
+f 430/773/19 494/870/19 493/869/19 429/771/19
+f 441/795/26 505/893/26 504/892/26 440/793/26
+f 461/830/9 525/934/9 526/936/9 462/831/9
+f 452/817/27 516/916/27 515/915/27 451/815/27
+f 483/852/28 547/981/28 548/982/28 484/853/28
+f 468/837/29 532/950/29 533/951/29 469/838/29
+f 437/787/22 501/885/22 500/884/22 436/785/22
+f 475/844/30 539/964/30 540/965/30 476/845/30
+f 448/809/12 512/907/12 511/906/12 447/807/12
+f 460/829/24 524/933/24 525/934/24 461/830/24
+f 489/859/4 521/926/4 522/928/4 490/862/4
+f 491/864/4 523/930/4 521/926/4 489/859/4
+f 492/866/4 524/932/4 523/930/4 491/864/4
+f 493/868/4 525/935/4 524/932/4 492/866/4
+f 494/871/4 526/937/4 525/935/4 493/868/4
+f 495/872/4 527/939/4 526/937/4 494/871/4
+f 496/874/4 528/940/4 527/939/4 495/872/4
+f 497/877/31 529/943/31 528/941/31 496/875/31
+f 498/879/4 530/945/4 529/943/4 497/877/4
+f 499/881/4 531/947/4 530/945/4 498/879/4
+f 500/883/4 532/949/4 531/947/4 499/881/4
+f 501/886/4 533/952/4 532/949/4 500/883/4
+f 502/887/4 534/954/4 533/952/4 501/886/4
+f 503/889/4 535/956/4 534/954/4 502/887/4
+f 504/891/4 536/957/4 535/956/4 503/889/4
+f 505/894/4 537/959/4 536/957/4 504/891/4
+f 506/895/4 538/961/4 537/959/4 505/894/4
+f 507/897/4 539/963/4 538/961/4 506/895/4
+f 508/899/4 540/966/4 539/963/4 507/897/4
+f 509/902/4 541/968/4 540/966/4 508/899/4
+f 510/903/4 542/970/4 541/968/4 509/902/4
+f 511/905/4 543/971/4 542/970/4 510/903/4
+f 512/908/4 544/973/4 543/971/4 511/905/4
+f 513/909/31 545/975/31 544/973/31 512/908/31
+f 514/912/4 546/978/4 545/976/4 513/910/4
+f 515/914/4 547/980/4 546/978/4 514/912/4
+f 516/917/4 548/983/4 547/980/4 515/914/4
+f 517/918/4 549/985/4 548/983/4 516/917/4
+f 518/920/4 550/987/4 549/985/4 517/918/4
+f 519/922/4 551/988/4 550/987/4 518/920/4
+f 520/925/4 552/990/4 551/988/4 519/922/4
+f 490/862/4 522/928/4 552/990/4 520/925/4
+f 435/783/32 499/882/32 498/880/32 434/781/32
+f 464/833/33 528/942/33 529/944/33 465/834/33
+f 479/848/34 543/972/34 544/974/34 480/849/34
+f 450/813/35 514/913/35 513/911/35 449/811/35
+f 439/791/7 503/890/7 502/888/7 438/789/7
+f 458/827/36 522/929/36 521/927/36 457/826/36
+f 428/769/30 492/867/30 491/865/30 427/767/30
+f 472/841/10 536/958/10 537/960/10 473/842/10
+f 487/856/37 551/989/37 552/991/37 488/857/37
+f 454/821/23 518/921/23 517/919/23 453/819/23
+f 443/799/38 507/898/38 506/896/38 442/797/38
+f 465/834/35 529/944/35 530/946/35 466/835/35
+f 480/849/39 544/974/39 545/977/39 481/850/39
+f 432/777/34 496/876/34 495/873/34 431/775/34
+f 457/826/38 521/927/38 523/931/38 459/828/38
+f 447/807/20 511/906/20 510/904/20 446/805/20
+f 473/842/40 537/960/40 538/962/40 474/843/40
+f 488/858/26 552/992/26 522/929/26 458/827/26
+f 436/785/28 500/884/28 499/882/28 435/783/28
+f 466/835/41 530/946/41 531/948/41 467/836/41
+f 481/850/21 545/977/21 546/979/21 482/851/21
+f 451/815/41 515/915/41 514/913/41 450/813/41
+f 440/793/37 504/892/37 503/890/37 439/791/37
+f 459/828/42 523/931/42 524/933/42 460/829/42
+f 429/771/43 493/869/43 492/867/43 428/769/43
+f 474/843/17 538/962/17 539/964/17 475/844/17
+f 455/823/16 519/923/16 518/921/16 454/821/16
+f 426/765/40 489/860/40 490/863/40 425/763/40
+f 444/801/42 508/900/42 507/898/42 443/799/42
+f 467/836/27 531/948/27 532/950/27 468/837/27
+f 482/851/32 546/979/32 547/981/32 483/852/32
+f 433/779/44 497/878/44 496/876/44 432/777/44
diff --git a/advtrains_signals_muc_ubahn/models/advtrains_signals_muc_ubahn_wsig_t.mtl b/advtrains_signals_muc_ubahn/models/advtrains_signals_muc_ubahn_wsig_t.mtl
new file mode 100644
index 0000000..69697a9
--- /dev/null
+++ b/advtrains_signals_muc_ubahn/models/advtrains_signals_muc_ubahn_wsig_t.mtl
@@ -0,0 +1,12 @@
+# Blender 3.4.1 MTL File: 'signal_wall_ceiling_muc_ubahn.blend'
+# www.blender.org
+
+newmtl Material
+Ns 250.000000
+Ka 1.000000 1.000000 1.000000
+Kd 0.800000 0.800000 0.800000
+Ks 0.500000 0.500000 0.500000
+Ke 0.000000 0.000000 0.000000
+Ni 1.000000
+d 1.000000
+illum 2
diff --git a/advtrains_signals_muc_ubahn/models/advtrains_signals_muc_ubahn_wsig_t.obj b/advtrains_signals_muc_ubahn/models/advtrains_signals_muc_ubahn_wsig_t.obj
new file mode 100644
index 0000000..c443b3c
--- /dev/null
+++ b/advtrains_signals_muc_ubahn/models/advtrains_signals_muc_ubahn_wsig_t.obj
@@ -0,0 +1,2138 @@
+# Blender 3.4.1
+# www.blender.org
+mtllib advtrains_signals_muc_ubahn_wsig_t.mtl
+o Cube
+v 0.399475 0.500000 0.043560
+v 0.357519 0.500000 0.043560
+v -0.357519 0.500000 0.043560
+v -0.402780 0.500000 0.043560
+v 0.399475 0.500000 -0.043560
+v 0.357519 0.500000 -0.043560
+v -0.357519 0.500000 -0.043560
+v -0.402780 0.500000 -0.043560
+v 0.399475 0.302890 0.043560
+v 0.357519 0.302890 0.043560
+v -0.357519 0.302890 0.043560
+v -0.402780 0.302890 0.043560
+v 0.399475 0.302890 -0.043561
+v 0.357519 0.302890 -0.043561
+v -0.357519 0.302890 -0.043560
+v -0.402780 0.302890 -0.043560
+v -0.357519 0.218703 0.042354
+v 0.357519 0.218703 0.042353
+v -0.357519 0.218703 -0.031584
+v 0.357519 0.218703 -0.031584
+v -0.357519 0.450908 0.042354
+v 0.357519 0.450908 0.042353
+v -0.357519 0.450908 -0.031584
+v 0.357519 0.450908 -0.031584
+v -0.310859 0.265363 0.042354
+v 0.310859 0.265363 0.042353
+v 0.310859 0.404247 0.042353
+v -0.310859 0.404247 0.042354
+v -0.310858 0.265363 0.089581
+v 0.310859 0.265363 0.089581
+v 0.310859 0.404247 0.089581
+v -0.310858 0.404247 0.089581
+v 0.318302 0.257920 -0.031584
+v -0.318302 0.257920 -0.031584
+v 0.318302 0.411691 -0.031584
+v -0.318302 0.411691 -0.031584
+v 0.241062 0.331404 -0.037331
+v 0.246186 0.279191 -0.037331
+v 0.256274 0.281194 -0.037331
+v 0.265777 0.285126 -0.037331
+v 0.274330 0.290837 -0.037331
+v 0.281605 0.298107 -0.037331
+v 0.287322 0.306656 -0.037331
+v 0.291261 0.316157 -0.037331
+v 0.293271 0.326243 -0.037331
+v 0.293275 0.336528 -0.037331
+v 0.291272 0.346615 -0.037331
+v 0.287339 0.356118 -0.037331
+v 0.281628 0.364672 -0.037331
+v 0.274359 0.371947 -0.037331
+v 0.265809 0.377664 -0.037331
+v 0.256309 0.381603 -0.037331
+v 0.246223 0.383613 -0.037331
+v 0.235938 0.383616 -0.037331
+v 0.225850 0.381613 -0.037331
+v 0.216347 0.377681 -0.037331
+v 0.207794 0.371970 -0.037331
+v 0.200519 0.364700 -0.037331
+v 0.194802 0.356151 -0.037331
+v 0.190863 0.346650 -0.037331
+v 0.188853 0.336564 -0.037331
+v 0.188850 0.326279 -0.037331
+v 0.190853 0.316192 -0.037331
+v 0.194785 0.306689 -0.037331
+v 0.200496 0.298135 -0.037331
+v 0.207766 0.290860 -0.037331
+v 0.216315 0.285144 -0.037331
+v 0.225815 0.281204 -0.037331
+v 0.235902 0.279195 -0.037331
+v 0.258934 0.272413 -0.031584
+v 0.247082 0.270060 -0.031584
+v 0.270099 0.277034 -0.031584
+v 0.280148 0.283743 -0.031584
+v 0.288695 0.292284 -0.031584
+v 0.295412 0.302329 -0.031584
+v 0.300040 0.313490 -0.031584
+v 0.302401 0.325341 -0.031584
+v 0.302406 0.337424 -0.031584
+v 0.300052 0.349275 -0.031584
+v 0.295432 0.360440 -0.031584
+v 0.288723 0.370490 -0.031584
+v 0.280181 0.379037 -0.031584
+v 0.270137 0.385753 -0.031584
+v 0.258975 0.390381 -0.031584
+v 0.247125 0.392743 -0.031584
+v 0.235042 0.392747 -0.031584
+v 0.223190 0.390394 -0.031584
+v 0.212025 0.385773 -0.031584
+v 0.201976 0.379064 -0.031584
+v 0.193429 0.370523 -0.031584
+v 0.186713 0.360478 -0.031584
+v 0.182085 0.349317 -0.031584
+v 0.179723 0.337466 -0.031584
+v 0.179719 0.325383 -0.031584
+v 0.182072 0.313532 -0.031584
+v 0.186692 0.302367 -0.031584
+v 0.193402 0.292317 -0.031584
+v 0.201943 0.283770 -0.031584
+v 0.211987 0.277054 -0.031584
+v 0.223149 0.272426 -0.031584
+v 0.234999 0.270065 -0.031584
+v 0.256274 0.281194 -0.054117
+v 0.246186 0.279191 -0.054117
+v 0.265777 0.285126 -0.054117
+v 0.274330 0.290837 -0.054117
+v 0.281605 0.298107 -0.054117
+v 0.287322 0.306656 -0.054117
+v 0.291261 0.316157 -0.054117
+v 0.293271 0.326243 -0.054117
+v 0.293275 0.336528 -0.083504
+v 0.291272 0.346615 -0.083504
+v 0.287339 0.356118 -0.083504
+v 0.281628 0.364672 -0.083504
+v 0.274359 0.371947 -0.083504
+v 0.265809 0.377664 -0.083504
+v 0.256309 0.381603 -0.083504
+v 0.246223 0.383613 -0.083504
+v 0.235938 0.383616 -0.083504
+v 0.225850 0.381613 -0.083504
+v 0.216347 0.377681 -0.083504
+v 0.207794 0.371970 -0.083504
+v 0.200519 0.364700 -0.083504
+v 0.194802 0.356151 -0.083504
+v 0.190863 0.346650 -0.083504
+v 0.188853 0.336564 -0.083504
+v 0.188850 0.326279 -0.054117
+v 0.190853 0.316192 -0.054117
+v 0.194785 0.306689 -0.054117
+v 0.200496 0.298135 -0.054117
+v 0.207766 0.290860 -0.054117
+v 0.216315 0.285144 -0.054117
+v 0.225815 0.281204 -0.054117
+v 0.235902 0.279195 -0.054117
+v 0.258934 0.272413 -0.054117
+v 0.247082 0.270060 -0.054117
+v 0.270099 0.277034 -0.054117
+v 0.280148 0.283743 -0.054117
+v 0.288695 0.292284 -0.054117
+v 0.295412 0.302329 -0.054117
+v 0.300040 0.313490 -0.054117
+v 0.302401 0.325341 -0.054117
+v 0.302406 0.337424 -0.083504
+v 0.300052 0.349275 -0.083504
+v 0.295432 0.360440 -0.083504
+v 0.288723 0.370490 -0.083504
+v 0.280181 0.379037 -0.083504
+v 0.270137 0.385753 -0.083504
+v 0.258975 0.390381 -0.083504
+v 0.247125 0.392743 -0.083504
+v 0.235042 0.392747 -0.083504
+v 0.223190 0.390394 -0.083504
+v 0.212025 0.385773 -0.083504
+v 0.201976 0.379064 -0.083504
+v 0.193429 0.370523 -0.083504
+v 0.186713 0.360478 -0.083504
+v 0.182085 0.349317 -0.083504
+v 0.179723 0.337466 -0.083504
+v 0.179719 0.325383 -0.054117
+v 0.182072 0.313532 -0.054117
+v 0.186692 0.302367 -0.054117
+v 0.193402 0.292317 -0.054117
+v 0.201943 0.283770 -0.054117
+v 0.211987 0.277054 -0.054117
+v 0.223149 0.272426 -0.054117
+v 0.234999 0.270065 -0.054117
+v 0.091062 0.331404 -0.037331
+v 0.096186 0.279191 -0.037331
+v 0.106274 0.281194 -0.037331
+v 0.115777 0.285126 -0.037331
+v 0.124331 0.290837 -0.037331
+v 0.131605 0.298107 -0.037331
+v 0.137322 0.306656 -0.037331
+v 0.141261 0.316157 -0.037331
+v 0.143271 0.326243 -0.037331
+v 0.143275 0.336528 -0.037331
+v 0.141272 0.346615 -0.037331
+v 0.137339 0.356119 -0.037331
+v 0.131628 0.364672 -0.037331
+v 0.124359 0.371947 -0.037331
+v 0.115809 0.377664 -0.037331
+v 0.106309 0.381603 -0.037331
+v 0.096223 0.383613 -0.037331
+v 0.085938 0.383616 -0.037331
+v 0.075850 0.381613 -0.037331
+v 0.066347 0.377681 -0.037331
+v 0.057794 0.371970 -0.037331
+v 0.050519 0.364700 -0.037331
+v 0.044802 0.356151 -0.037331
+v 0.040863 0.346650 -0.037331
+v 0.038853 0.336564 -0.037331
+v 0.038850 0.326279 -0.037331
+v 0.040853 0.316192 -0.037331
+v 0.044785 0.306689 -0.037331
+v 0.050496 0.298135 -0.037331
+v 0.057766 0.290860 -0.037331
+v 0.066315 0.285144 -0.037331
+v 0.075815 0.281205 -0.037331
+v 0.085902 0.279195 -0.037331
+v 0.108934 0.272414 -0.031584
+v 0.097082 0.270060 -0.031584
+v 0.120099 0.277034 -0.031584
+v 0.130148 0.283743 -0.031584
+v 0.138695 0.292284 -0.031584
+v 0.145412 0.302329 -0.031584
+v 0.150040 0.313490 -0.031584
+v 0.152401 0.325341 -0.031584
+v 0.152406 0.337424 -0.031584
+v 0.150052 0.349276 -0.031584
+v 0.145432 0.360440 -0.031584
+v 0.138723 0.370490 -0.031584
+v 0.130181 0.379037 -0.031584
+v 0.120137 0.385753 -0.031584
+v 0.108975 0.390381 -0.031584
+v 0.097125 0.392743 -0.031584
+v 0.085042 0.392747 -0.031584
+v 0.073190 0.390394 -0.031584
+v 0.062025 0.385773 -0.031584
+v 0.051976 0.379064 -0.031584
+v 0.043429 0.370523 -0.031584
+v 0.036713 0.360478 -0.031584
+v 0.032085 0.349317 -0.031584
+v 0.029723 0.337467 -0.031584
+v 0.029719 0.325383 -0.031584
+v 0.032072 0.313532 -0.031584
+v 0.036692 0.302367 -0.031584
+v 0.043402 0.292317 -0.031584
+v 0.051943 0.283770 -0.031584
+v 0.061987 0.277054 -0.031584
+v 0.073149 0.272426 -0.031584
+v 0.084999 0.270065 -0.031584
+v 0.106274 0.281194 -0.054117
+v 0.096186 0.279191 -0.054117
+v 0.115777 0.285126 -0.054117
+v 0.124331 0.290837 -0.054117
+v 0.131605 0.298107 -0.054117
+v 0.137322 0.306656 -0.054117
+v 0.141261 0.316157 -0.054117
+v 0.143271 0.326243 -0.054117
+v 0.143275 0.336528 -0.083504
+v 0.141272 0.346615 -0.083504
+v 0.137339 0.356119 -0.083504
+v 0.131628 0.364672 -0.083504
+v 0.124359 0.371947 -0.083504
+v 0.115809 0.377664 -0.083504
+v 0.106309 0.381603 -0.083504
+v 0.096223 0.383613 -0.083504
+v 0.085938 0.383616 -0.083504
+v 0.075850 0.381613 -0.083504
+v 0.066347 0.377681 -0.083504
+v 0.057794 0.371970 -0.083504
+v 0.050519 0.364700 -0.083504
+v 0.044802 0.356151 -0.083504
+v 0.040863 0.346650 -0.083504
+v 0.038853 0.336564 -0.083504
+v 0.038850 0.326279 -0.054117
+v 0.040853 0.316192 -0.054117
+v 0.044785 0.306689 -0.054117
+v 0.050496 0.298135 -0.054117
+v 0.057766 0.290860 -0.054117
+v 0.066315 0.285144 -0.054117
+v 0.075815 0.281205 -0.054117
+v 0.085902 0.279195 -0.054117
+v 0.108934 0.272414 -0.054117
+v 0.097082 0.270060 -0.054117
+v 0.120099 0.277034 -0.054117
+v 0.130148 0.283743 -0.054117
+v 0.138695 0.292284 -0.054117
+v 0.145412 0.302329 -0.054117
+v 0.150040 0.313490 -0.054117
+v 0.152401 0.325341 -0.054117
+v 0.152406 0.337424 -0.083504
+v 0.150052 0.349276 -0.083504
+v 0.145432 0.360440 -0.083504
+v 0.138723 0.370490 -0.083504
+v 0.130181 0.379037 -0.083504
+v 0.120137 0.385753 -0.083504
+v 0.108975 0.390381 -0.083504
+v 0.097125 0.392743 -0.083504
+v 0.085042 0.392747 -0.083504
+v 0.073190 0.390394 -0.083504
+v 0.062025 0.385773 -0.083504
+v 0.051976 0.379064 -0.083504
+v 0.043429 0.370523 -0.083504
+v 0.036713 0.360478 -0.083504
+v 0.032085 0.349317 -0.083504
+v 0.029723 0.337466 -0.083504
+v 0.029719 0.325383 -0.054117
+v 0.032072 0.313532 -0.054117
+v 0.036692 0.302367 -0.054117
+v 0.043402 0.292317 -0.054117
+v 0.051943 0.283770 -0.054117
+v 0.061987 0.277054 -0.054117
+v 0.073149 0.272426 -0.054117
+v 0.084999 0.270065 -0.054117
+v -0.058938 0.331404 -0.037331
+v -0.053814 0.279191 -0.037331
+v -0.043726 0.281194 -0.037331
+v -0.034223 0.285126 -0.037331
+v -0.025669 0.290837 -0.037331
+v -0.018395 0.298107 -0.037331
+v -0.012678 0.306656 -0.037331
+v -0.008739 0.316157 -0.037331
+v -0.006729 0.326243 -0.037331
+v -0.006725 0.336528 -0.037331
+v -0.008728 0.346615 -0.037331
+v -0.012661 0.356119 -0.037331
+v -0.018372 0.364672 -0.037331
+v -0.025641 0.371947 -0.037331
+v -0.034191 0.377664 -0.037331
+v -0.043691 0.381603 -0.037331
+v -0.053777 0.383613 -0.037331
+v -0.064062 0.383616 -0.037331
+v -0.074150 0.381613 -0.037331
+v -0.083653 0.377681 -0.037331
+v -0.092206 0.371970 -0.037331
+v -0.099481 0.364700 -0.037331
+v -0.105198 0.356151 -0.037331
+v -0.109137 0.346650 -0.037331
+v -0.111147 0.336564 -0.037331
+v -0.111150 0.326279 -0.037331
+v -0.109147 0.316192 -0.037331
+v -0.105215 0.306689 -0.037331
+v -0.099504 0.298135 -0.037331
+v -0.092234 0.290860 -0.037331
+v -0.083685 0.285144 -0.037331
+v -0.074185 0.281205 -0.037331
+v -0.064098 0.279195 -0.037331
+v -0.041066 0.272414 -0.031584
+v -0.052918 0.270060 -0.031584
+v -0.029901 0.277034 -0.031584
+v -0.019852 0.283743 -0.031584
+v -0.011305 0.292284 -0.031584
+v -0.004588 0.302329 -0.031584
+v 0.000040 0.313490 -0.031584
+v 0.002401 0.325341 -0.031584
+v 0.002406 0.337424 -0.031584
+v 0.000052 0.349276 -0.031584
+v -0.004568 0.360440 -0.031584
+v -0.011277 0.370490 -0.031584
+v -0.019819 0.379037 -0.031584
+v -0.029863 0.385753 -0.031584
+v -0.041025 0.390381 -0.031584
+v -0.052875 0.392743 -0.031584
+v -0.064958 0.392747 -0.031584
+v -0.076810 0.390394 -0.031584
+v -0.087975 0.385773 -0.031584
+v -0.098024 0.379064 -0.031584
+v -0.106571 0.370523 -0.031584
+v -0.113287 0.360478 -0.031584
+v -0.117915 0.349317 -0.031584
+v -0.120277 0.337467 -0.031584
+v -0.120281 0.325383 -0.031584
+v -0.117928 0.313532 -0.031584
+v -0.113308 0.302367 -0.031584
+v -0.106598 0.292317 -0.031584
+v -0.098057 0.283770 -0.031584
+v -0.088013 0.277054 -0.031584
+v -0.076851 0.272426 -0.031584
+v -0.065001 0.270065 -0.031584
+v -0.043726 0.281194 -0.054117
+v -0.053814 0.279191 -0.054117
+v -0.034223 0.285126 -0.054117
+v -0.025669 0.290837 -0.054117
+v -0.018395 0.298107 -0.054117
+v -0.012678 0.306656 -0.054117
+v -0.008739 0.316157 -0.054117
+v -0.006729 0.326243 -0.054117
+v -0.006725 0.336528 -0.083504
+v -0.008728 0.346615 -0.083504
+v -0.012661 0.356119 -0.083504
+v -0.018372 0.364672 -0.083504
+v -0.025641 0.371947 -0.083504
+v -0.034191 0.377664 -0.083504
+v -0.043691 0.381603 -0.083504
+v -0.053777 0.383613 -0.083504
+v -0.064062 0.383616 -0.083504
+v -0.074150 0.381613 -0.083504
+v -0.083653 0.377681 -0.083504
+v -0.092206 0.371970 -0.083504
+v -0.099481 0.364700 -0.083504
+v -0.105198 0.356151 -0.083504
+v -0.109137 0.346650 -0.083504
+v -0.111147 0.336564 -0.083504
+v -0.111150 0.326279 -0.054117
+v -0.109147 0.316192 -0.054117
+v -0.105215 0.306689 -0.054117
+v -0.099504 0.298135 -0.054117
+v -0.092234 0.290860 -0.054117
+v -0.083685 0.285144 -0.054117
+v -0.074185 0.281205 -0.054117
+v -0.064098 0.279195 -0.054117
+v -0.041066 0.272414 -0.054117
+v -0.052918 0.270060 -0.054117
+v -0.029901 0.277034 -0.054117
+v -0.019852 0.283743 -0.054117
+v -0.011305 0.292284 -0.054117
+v -0.004588 0.302329 -0.054117
+v 0.000040 0.313490 -0.054117
+v 0.002401 0.325341 -0.054117
+v 0.002406 0.337424 -0.083504
+v 0.000052 0.349276 -0.083504
+v -0.004568 0.360440 -0.083504
+v -0.011277 0.370490 -0.083504
+v -0.019819 0.379037 -0.083504
+v -0.029863 0.385753 -0.083504
+v -0.041025 0.390381 -0.083504
+v -0.052875 0.392743 -0.083504
+v -0.064958 0.392747 -0.083504
+v -0.076810 0.390394 -0.083504
+v -0.087975 0.385773 -0.083504
+v -0.098024 0.379064 -0.083504
+v -0.106571 0.370523 -0.083504
+v -0.113287 0.360478 -0.083504
+v -0.117915 0.349317 -0.083504
+v -0.120277 0.337466 -0.083504
+v -0.120281 0.325383 -0.054117
+v -0.117928 0.313532 -0.054117
+v -0.113308 0.302367 -0.054117
+v -0.106598 0.292317 -0.054117
+v -0.098057 0.283770 -0.054117
+v -0.088013 0.277054 -0.054117
+v -0.076851 0.272426 -0.054117
+v -0.065001 0.270065 -0.054117
+v -0.208938 0.331404 -0.037331
+v -0.203814 0.279191 -0.037331
+v -0.193726 0.281194 -0.037331
+v -0.184223 0.285126 -0.037331
+v -0.175669 0.290837 -0.037331
+v -0.168395 0.298107 -0.037331
+v -0.162678 0.306656 -0.037331
+v -0.158739 0.316157 -0.037331
+v -0.156729 0.326243 -0.037331
+v -0.156725 0.336528 -0.037331
+v -0.158728 0.346615 -0.037331
+v -0.162661 0.356119 -0.037331
+v -0.168372 0.364672 -0.037331
+v -0.175641 0.371947 -0.037331
+v -0.184191 0.377664 -0.037331
+v -0.193691 0.381603 -0.037331
+v -0.203777 0.383613 -0.037331
+v -0.214062 0.383616 -0.037331
+v -0.224150 0.381613 -0.037331
+v -0.233653 0.377681 -0.037331
+v -0.242206 0.371970 -0.037331
+v -0.249481 0.364700 -0.037331
+v -0.255198 0.356151 -0.037331
+v -0.259137 0.346650 -0.037331
+v -0.261147 0.336564 -0.037331
+v -0.261150 0.326279 -0.037331
+v -0.259147 0.316192 -0.037331
+v -0.255215 0.306689 -0.037331
+v -0.249504 0.298135 -0.037331
+v -0.242234 0.290860 -0.037331
+v -0.233685 0.285144 -0.037331
+v -0.224185 0.281205 -0.037331
+v -0.214098 0.279195 -0.037331
+v -0.191066 0.272414 -0.031584
+v -0.202918 0.270060 -0.031584
+v -0.179901 0.277034 -0.031584
+v -0.169852 0.283743 -0.031584
+v -0.161305 0.292284 -0.031584
+v -0.154588 0.302329 -0.031584
+v -0.149960 0.313490 -0.031584
+v -0.147599 0.325341 -0.031584
+v -0.147594 0.337424 -0.031584
+v -0.149948 0.349276 -0.031584
+v -0.154568 0.360440 -0.031584
+v -0.161277 0.370490 -0.031584
+v -0.169818 0.379037 -0.031584
+v -0.179863 0.385753 -0.031584
+v -0.191025 0.390381 -0.031584
+v -0.202875 0.392743 -0.031584
+v -0.214958 0.392747 -0.031584
+v -0.226810 0.390394 -0.031584
+v -0.237975 0.385774 -0.031584
+v -0.248024 0.379064 -0.031584
+v -0.256571 0.370523 -0.031584
+v -0.263287 0.360478 -0.031584
+v -0.267915 0.349317 -0.031584
+v -0.270277 0.337467 -0.031584
+v -0.270281 0.325383 -0.031584
+v -0.267928 0.313532 -0.031584
+v -0.263308 0.302367 -0.031584
+v -0.256598 0.292317 -0.031584
+v -0.248057 0.283770 -0.031584
+v -0.238013 0.277054 -0.031584
+v -0.226851 0.272426 -0.031584
+v -0.215001 0.270065 -0.031584
+v -0.193726 0.281194 -0.054117
+v -0.203814 0.279191 -0.054117
+v -0.184223 0.285126 -0.054117
+v -0.175669 0.290837 -0.054117
+v -0.168395 0.298107 -0.054117
+v -0.162678 0.306656 -0.054117
+v -0.158739 0.316157 -0.054117
+v -0.156729 0.326243 -0.054117
+v -0.156725 0.336528 -0.083504
+v -0.158728 0.346615 -0.083504
+v -0.162661 0.356119 -0.083504
+v -0.168372 0.364672 -0.083504
+v -0.175641 0.371947 -0.083504
+v -0.184191 0.377664 -0.083504
+v -0.193691 0.381603 -0.083504
+v -0.203777 0.383613 -0.083504
+v -0.214062 0.383616 -0.083504
+v -0.224150 0.381613 -0.083504
+v -0.233653 0.377681 -0.083504
+v -0.242206 0.371970 -0.083504
+v -0.249481 0.364700 -0.083504
+v -0.255198 0.356151 -0.083504
+v -0.259137 0.346650 -0.083504
+v -0.261147 0.336564 -0.083504
+v -0.261150 0.326279 -0.054117
+v -0.259147 0.316192 -0.054117
+v -0.255215 0.306689 -0.054117
+v -0.249504 0.298135 -0.054117
+v -0.242234 0.290860 -0.054117
+v -0.233685 0.285144 -0.054117
+v -0.224185 0.281204 -0.054117
+v -0.214098 0.279195 -0.054117
+v -0.191066 0.272414 -0.054117
+v -0.202918 0.270060 -0.054117
+v -0.179901 0.277034 -0.054117
+v -0.169852 0.283743 -0.054117
+v -0.161305 0.292284 -0.054117
+v -0.154588 0.302329 -0.054117
+v -0.149960 0.313490 -0.054117
+v -0.147599 0.325341 -0.054117
+v -0.147594 0.337424 -0.083504
+v -0.149948 0.349276 -0.083504
+v -0.154568 0.360440 -0.083504
+v -0.161277 0.370490 -0.083504
+v -0.169819 0.379037 -0.083504
+v -0.179863 0.385753 -0.083504
+v -0.191025 0.390381 -0.083504
+v -0.202875 0.392743 -0.083504
+v -0.214958 0.392747 -0.083504
+v -0.226810 0.390394 -0.083504
+v -0.237975 0.385774 -0.083504
+v -0.248024 0.379064 -0.083504
+v -0.256571 0.370523 -0.083504
+v -0.263287 0.360478 -0.083504
+v -0.267915 0.349317 -0.083504
+v -0.270277 0.337466 -0.083504
+v -0.270281 0.325383 -0.054117
+v -0.267928 0.313532 -0.054117
+v -0.263308 0.302367 -0.054117
+v -0.256598 0.292317 -0.054117
+v -0.248057 0.283770 -0.054117
+v -0.238013 0.277054 -0.054117
+v -0.226851 0.272426 -0.054117
+v -0.215001 0.270065 -0.054117
+vn -1.0000 -0.0000 -0.0000
+vn 1.0000 -0.0000 -0.0000
+vn -0.0000 -1.0000 -0.0000
+vn -0.0000 -0.0000 -1.0000
+vn -0.0000 -0.0000 1.0000
+vn -0.0000 1.0000 -0.0000
+vn -0.3830 -0.9237 -0.0000
+vn 0.1954 0.9807 -0.0000
+vn 0.8313 -0.5559 -0.0000
+vn 0.0003 1.0000 -0.0000
+vn -0.9237 0.3830 -0.0000
+vn 0.9807 -0.1954 -0.0000
+vn 0.1947 -0.9809 -0.0000
+vn 0.7073 0.7069 -0.0000
+vn -0.5559 -0.8313 -0.0000
+vn 0.3830 0.9237 -0.0000
+vn -0.3824 0.9240 -0.0000
+vn 1.0000 -0.0003 -0.0000
+vn -0.8313 0.5559 -0.0000
+vn 0.9237 -0.3830 -0.0000
+vn -0.9809 -0.1948 -0.0000
+vn -0.7073 -0.7069 -0.0000
+vn 0.5559 0.8313 -0.0000
+vn 0.7069 -0.7074 -0.0000
+vn -0.7069 0.7073 -0.0000
+vn -0.0003 -1.0000 -0.0000
+vn 0.8317 0.5553 -0.0000
+vn -0.8317 -0.5553 -0.0000
+vn 0.7074 0.7069 -0.0000
+vn -0.7074 -0.7069 -0.0000
+vn -0.5553 0.8317 -0.0000
+vn -0.0003 -0.9346 -0.3557
+vn -0.9240 -0.3824 -0.0000
+vn 1.0000 -0.0004 -0.0000
+vn -0.9807 0.1954 -0.0000
+vn 0.9809 0.1948 -0.0000
+vn 0.1948 -0.9809 -0.0000
+vn -0.1954 -0.9807 -0.0000
+vn 0.3824 -0.9240 -0.0000
+vn -1.0000 0.0004 -0.0000
+vn -0.1947 0.9809 -0.0000
+vn 0.9240 0.3824 -0.0000
+vn -0.9809 -0.1947 -0.0000
+vn 0.5553 -0.8317 -0.0000
+vn -0.7069 0.7074 -0.0000
+vn -1.0000 0.0003 -0.0000
+vn 0.9809 0.1947 -0.0000
+vt 0.640016 0.630686
+vt 0.874116 0.630686
+vt 0.899031 0.371671
+vt 0.899031 0.605771
+vt 0.275073 0.371671
+vt 0.275073 0.605771
+vt 0.536051 0.632649
+vt 0.301951 0.632649
+vt 0.588281 0.578951
+vt 0.588281 0.344851
+vt 0.847296 0.319936
+vt 0.613196 0.319936
+vt 0.326808 0.319936
+vt 0.560909 0.319936
+vt 0.587786 0.580914
+vt 0.587786 0.346814
+vt 0.757066 0.513636
+vt 0.781981 0.488721
+vt 0.392123 0.488721
+vt 0.419001 0.515599
+vt 0.705331 0.461901
+vt 0.730246 0.436986
+vt 0.443858 0.436986
+vt 0.470736 0.463864
+vt 0.062341 0.000247
+vt 0.000247 0.062341
+vt 0.000248 0.359598
+vt 0.039909 0.319936
+vt 0.662834 0.000247
+vt 0.724927 0.062341
+vt 0.000247 0.960091
+vt 0.039910 0.999753
+vt 0.062341 0.062341
+vt 0.662834 0.062341
+vt 0.062341 0.319441
+vt 0.000247 0.257348
+vt 0.274579 0.359598
+vt 0.234916 0.319936
+vt 0.662834 0.319441
+vt 0.724927 0.257348
+vt 0.274579 0.960090
+vt 0.234917 0.999753
+vt 0.062341 0.257348
+vt 0.662834 0.257348
+vt 0.079095 0.359121
+vt 0.039433 0.398784
+vt 0.079095 0.960567
+vt 0.039433 0.920905
+vt 0.195731 0.960567
+vt 0.235393 0.920905
+vt 0.195731 0.359121
+vt 0.235393 0.398783
+vt 0.079095 0.398783
+vt 0.079095 0.920905
+vt 0.195731 0.920905
+vt 0.195731 0.398783
+vt 0.629899 0.095275
+vt 0.095275 0.095275
+vt 0.629899 0.224413
+vt 0.095275 0.224413
+vt 0.753412 0.903011
+vt 0.719354 0.930961
+vt 0.334815 0.932388
+vt 0.611202 0.932388
+vt 0.714555 0.923780
+vt 0.602565 0.932389
+vt 0.711250 0.915800
+vt 0.593928 0.932389
+vt 0.709565 0.907329
+vt 0.585291 0.932389
+vt 0.709565 0.898692
+vt 0.576654 0.932389
+vt 0.711250 0.890221
+vt 0.568017 0.932389
+vt 0.714556 0.882241
+vt 0.559379 0.932389
+vt 0.719354 0.875060
+vt 0.550742 0.932389
+vt 0.725461 0.868952
+vt 0.542105 0.932389
+vt 0.732643 0.864154
+vt 0.533468 0.932389
+vt 0.740623 0.860849
+vt 0.524831 0.932389
+vt 0.749094 0.859164
+vt 0.516194 0.932389
+vt 0.757731 0.859164
+vt 0.507557 0.932389
+vt 0.766202 0.860849
+vt 0.498920 0.932389
+vt 0.774182 0.864154
+vt 0.490283 0.932389
+vt 0.781363 0.868953
+vt 0.481646 0.932389
+vt 0.787470 0.875060
+vt 0.473009 0.932389
+vt 0.792269 0.882241
+vt 0.464371 0.932389
+vt 0.795574 0.890221
+vt 0.455734 0.932389
+vt 0.797259 0.898692
+vt 0.447097 0.932389
+vt 0.797259 0.907329
+vt 0.438460 0.932389
+vt 0.795574 0.915800
+vt 0.429823 0.932389
+vt 0.792269 0.923780
+vt 0.421186 0.932389
+vt 0.787470 0.930962
+vt 0.412549 0.932389
+vt 0.781363 0.937069
+vt 0.403912 0.932389
+vt 0.774181 0.941867
+vt 0.395275 0.932389
+vt 0.766201 0.945173
+vt 0.386637 0.932389
+vt 0.757730 0.946858
+vt 0.378001 0.932389
+vt 0.749093 0.946858
+vt 0.369363 0.932389
+vt 0.740622 0.945172
+vt 0.360726 0.932389
+vt 0.732643 0.941867
+vt 0.352089 0.932388
+vt 0.725461 0.937069
+vt 0.343452 0.932388
+vt 0.618830 0.816892
+vt 0.628978 0.816892
+vt 0.608683 0.816892
+vt 0.598535 0.816892
+vt 0.588388 0.816892
+vt 0.578240 0.816892
+vt 0.568093 0.816892
+vt 0.557945 0.816892
+vt 0.547798 0.816892
+vt 0.537650 0.816892
+vt 0.527503 0.816892
+vt 0.517355 0.816892
+vt 0.507208 0.816892
+vt 0.497060 0.816892
+vt 0.486913 0.816892
+vt 0.476765 0.816892
+vt 0.466618 0.816892
+vt 0.456470 0.816892
+vt 0.446323 0.816892
+vt 0.436175 0.816892
+vt 0.426028 0.816892
+vt 0.415880 0.816892
+vt 0.405733 0.816892
+vt 0.395585 0.816892
+vt 0.385438 0.816892
+vt 0.375290 0.816892
+vt 0.365143 0.816892
+vt 0.354995 0.816892
+vt 0.344848 0.816892
+vt 0.334700 0.816892
+vt 0.324553 0.816892
+vt 0.314405 0.816892
+vt 0.639125 0.816892
+vt 0.517095 0.669905
+vt 0.602565 0.946486
+vt 0.334815 0.946485
+vt 0.515410 0.678377
+vt 0.611202 0.946485
+vt 0.520400 0.661926
+vt 0.593928 0.946485
+vt 0.525198 0.654744
+vt 0.585291 0.946486
+vt 0.531306 0.648637
+vt 0.576654 0.946486
+vt 0.568017 0.946486
+vt 0.538487 0.643839
+vt 0.546467 0.640533
+vt 0.559379 0.946486
+vt 0.554938 0.638848
+vt 0.507247 0.729756
+vt 0.550742 0.946486
+vt 0.481149 0.728166
+vt 0.542105 0.971165
+vt 0.472678 0.726481
+vt 0.533468 0.971165
+vt 0.464698 0.723176
+vt 0.524831 0.971165
+vt 0.457517 0.718377
+vt 0.516194 0.971165
+vt 0.507557 0.971165
+vt 0.451409 0.712270
+vt 0.446611 0.705089
+vt 0.498920 0.971165
+vt 0.443305 0.697109
+vt 0.490283 0.971165
+vt 0.441620 0.688638
+vt 0.481646 0.971165
+vt 0.473008 0.971165
+vt 0.441620 0.680001
+vt 0.443305 0.671530
+vt 0.464371 0.971165
+vt 0.446610 0.663550
+vt 0.455734 0.971165
+vt 0.451409 0.656369
+vt 0.447097 0.971165
+vt 0.438460 0.971165
+vt 0.457516 0.650261
+vt 0.464698 0.645463
+vt 0.429823 0.971165
+vt 0.472677 0.642157
+vt 0.421186 0.971165
+vt 0.412549 0.971165
+vt 0.481149 0.640472
+vt 0.507247 0.638882
+vt 0.554938 0.726542
+vt 0.403912 0.946486
+vt 0.546467 0.724857
+vt 0.395275 0.946486
+vt 0.538487 0.721552
+vt 0.386637 0.946486
+vt 0.378001 0.946486
+vt 0.531306 0.716753
+vt 0.525198 0.710646
+vt 0.369363 0.946486
+vt 0.520400 0.703464
+vt 0.360726 0.946485
+vt 0.517095 0.695485
+vt 0.352089 0.946485
+vt 0.343452 0.946485
+vt 0.515410 0.687014
+vt 0.509722 0.667669
+vt 0.618830 0.797969
+vt 0.507742 0.677621
+vt 0.628978 0.797969
+vt 0.513605 0.658294
+vt 0.608683 0.797969
+vt 0.519242 0.649857
+vt 0.598535 0.797969
+vt 0.588388 0.797969
+vt 0.526418 0.642681
+vt 0.578240 0.797969
+vt 0.534855 0.637043
+vt 0.568093 0.797969
+vt 0.544230 0.633160
+vt 0.554183 0.631180
+vt 0.507028 0.737458
+vt 0.557945 0.797969
+vt 0.480393 0.735834
+vt 0.547798 0.773289
+vt 0.470441 0.733854
+vt 0.537650 0.773289
+vt 0.461066 0.729971
+vt 0.527503 0.773289
+vt 0.452629 0.724334
+vt 0.517355 0.773289
+vt 0.507208 0.773290
+vt 0.445453 0.717158
+vt 0.497060 0.773290
+vt 0.439816 0.708721
+vt 0.486913 0.773290
+vt 0.435932 0.699346
+vt 0.433953 0.689393
+vt 0.476765 0.773290
+vt 0.433953 0.679246
+vt 0.466618 0.773290
+vt 0.435932 0.669293
+vt 0.456470 0.773290
+vt 0.439815 0.659918
+vt 0.446323 0.773289
+vt 0.436175 0.773289
+vt 0.445453 0.651481
+vt 0.426028 0.773289
+vt 0.452628 0.644305
+vt 0.415880 0.773289
+vt 0.461066 0.638668
+vt 0.470441 0.634784
+vt 0.405733 0.773289
+vt 0.480393 0.632805
+vt 0.395585 0.773289
+vt 0.507028 0.631180
+vt 0.554183 0.734210
+vt 0.385438 0.797969
+vt 0.544230 0.732230
+vt 0.375290 0.797969
+vt 0.534855 0.728347
+vt 0.365143 0.797969
+vt 0.354995 0.797969
+vt 0.526418 0.722709
+vt 0.344848 0.797969
+vt 0.519243 0.715534
+vt 0.334700 0.797969
+vt 0.513605 0.707096
+vt 0.509721 0.697721
+vt 0.324553 0.797969
+vt 0.507742 0.687769
+vt 0.314405 0.797969
+vt 0.639125 0.797969
+vt 0.752599 0.742115
+vt 0.718541 0.770066
+vt 0.334815 0.932388
+vt 0.611202 0.932388
+vt 0.713742 0.762885
+vt 0.602565 0.932389
+vt 0.710437 0.754905
+vt 0.593928 0.932389
+vt 0.708752 0.746434
+vt 0.585291 0.932389
+vt 0.708752 0.737797
+vt 0.576654 0.932389
+vt 0.710437 0.729326
+vt 0.568017 0.932389
+vt 0.713743 0.721346
+vt 0.559379 0.932389
+vt 0.718541 0.714165
+vt 0.550742 0.932389
+vt 0.724649 0.708057
+vt 0.542105 0.932389
+vt 0.731830 0.703259
+vt 0.533468 0.932389
+vt 0.739810 0.699954
+vt 0.524831 0.932389
+vt 0.748281 0.698269
+vt 0.516194 0.932389
+vt 0.756918 0.698269
+vt 0.507557 0.932389
+vt 0.765389 0.699954
+vt 0.498920 0.932389
+vt 0.773369 0.703259
+vt 0.490283 0.932389
+vt 0.780550 0.708058
+vt 0.481646 0.932389
+vt 0.786658 0.714165
+vt 0.473009 0.932389
+vt 0.791456 0.721346
+vt 0.464371 0.932389
+vt 0.794761 0.729326
+vt 0.455734 0.932389
+vt 0.796446 0.737797
+vt 0.447097 0.932389
+vt 0.796446 0.746434
+vt 0.438460 0.932389
+vt 0.794761 0.754905
+vt 0.429823 0.932389
+vt 0.791456 0.762885
+vt 0.421186 0.932389
+vt 0.786657 0.770066
+vt 0.412549 0.932389
+vt 0.780550 0.776174
+vt 0.403912 0.932389
+vt 0.773368 0.780972
+vt 0.395275 0.932389
+vt 0.765389 0.784277
+vt 0.386637 0.932389
+vt 0.756917 0.785962
+vt 0.378001 0.932389
+vt 0.748280 0.785962
+vt 0.369363 0.932389
+vt 0.739809 0.784277
+vt 0.360726 0.932389
+vt 0.731830 0.780972
+vt 0.352089 0.932388
+vt 0.724648 0.776174
+vt 0.343452 0.932388
+vt 0.618830 0.816892
+vt 0.628978 0.816892
+vt 0.608683 0.816892
+vt 0.598535 0.816892
+vt 0.588388 0.816892
+vt 0.578240 0.816892
+vt 0.568093 0.816892
+vt 0.557945 0.816892
+vt 0.547798 0.816892
+vt 0.537650 0.816892
+vt 0.527503 0.816892
+vt 0.517355 0.816892
+vt 0.507208 0.816892
+vt 0.497060 0.816892
+vt 0.486913 0.816892
+vt 0.476765 0.816892
+vt 0.466618 0.816892
+vt 0.456470 0.816892
+vt 0.446323 0.816892
+vt 0.436175 0.816892
+vt 0.426028 0.816892
+vt 0.415880 0.816892
+vt 0.405733 0.816892
+vt 0.395585 0.816892
+vt 0.385438 0.816892
+vt 0.375290 0.816892
+vt 0.365143 0.816892
+vt 0.354995 0.816892
+vt 0.344848 0.816892
+vt 0.334700 0.816892
+vt 0.324553 0.816892
+vt 0.314405 0.816892
+vt 0.639125 0.816892
+vt 0.517095 0.669905
+vt 0.602565 0.946486
+vt 0.334815 0.946485
+vt 0.515410 0.678377
+vt 0.611202 0.946485
+vt 0.520400 0.661926
+vt 0.593928 0.946485
+vt 0.525198 0.654744
+vt 0.585291 0.946486
+vt 0.531306 0.648637
+vt 0.576654 0.946486
+vt 0.568017 0.946486
+vt 0.538487 0.643839
+vt 0.546467 0.640533
+vt 0.559379 0.946486
+vt 0.554938 0.638848
+vt 0.507247 0.729756
+vt 0.550742 0.946486
+vt 0.481149 0.728166
+vt 0.542105 0.971165
+vt 0.472678 0.726481
+vt 0.533468 0.971165
+vt 0.464698 0.723176
+vt 0.524831 0.971165
+vt 0.457517 0.718377
+vt 0.516194 0.971165
+vt 0.507557 0.971165
+vt 0.451409 0.712270
+vt 0.446611 0.705089
+vt 0.498920 0.971165
+vt 0.443305 0.697109
+vt 0.490283 0.971165
+vt 0.441620 0.688638
+vt 0.481646 0.971165
+vt 0.473008 0.971165
+vt 0.441620 0.680001
+vt 0.443305 0.671530
+vt 0.464371 0.971165
+vt 0.446610 0.663550
+vt 0.455734 0.971165
+vt 0.451409 0.656369
+vt 0.447097 0.971165
+vt 0.438460 0.971165
+vt 0.457516 0.650261
+vt 0.464698 0.645463
+vt 0.429823 0.971165
+vt 0.472677 0.642157
+vt 0.421186 0.971165
+vt 0.412549 0.971165
+vt 0.481149 0.640472
+vt 0.507247 0.638882
+vt 0.554938 0.726542
+vt 0.403912 0.946486
+vt 0.546467 0.724857
+vt 0.395275 0.946486
+vt 0.538487 0.721552
+vt 0.386637 0.946486
+vt 0.378001 0.946486
+vt 0.531306 0.716753
+vt 0.525198 0.710646
+vt 0.369363 0.946486
+vt 0.520400 0.703464
+vt 0.360726 0.946485
+vt 0.517095 0.695485
+vt 0.352089 0.946485
+vt 0.343452 0.946485
+vt 0.515410 0.687014
+vt 0.509722 0.667669
+vt 0.618830 0.797969
+vt 0.507742 0.677621
+vt 0.628978 0.797969
+vt 0.513605 0.658294
+vt 0.608683 0.797969
+vt 0.519242 0.649857
+vt 0.598535 0.797969
+vt 0.588388 0.797969
+vt 0.526418 0.642681
+vt 0.578240 0.797969
+vt 0.534855 0.637043
+vt 0.568093 0.797969
+vt 0.544230 0.633160
+vt 0.554183 0.631180
+vt 0.507028 0.737458
+vt 0.557945 0.797969
+vt 0.480393 0.735834
+vt 0.547798 0.773289
+vt 0.470441 0.733854
+vt 0.537650 0.773289
+vt 0.461066 0.729971
+vt 0.527503 0.773289
+vt 0.452629 0.724334
+vt 0.517355 0.773289
+vt 0.507208 0.773290
+vt 0.445453 0.717158
+vt 0.497060 0.773290
+vt 0.439816 0.708721
+vt 0.486913 0.773290
+vt 0.435932 0.699346
+vt 0.433953 0.689393
+vt 0.476765 0.773290
+vt 0.433953 0.679246
+vt 0.466618 0.773290
+vt 0.435932 0.669293
+vt 0.456470 0.773290
+vt 0.439815 0.659918
+vt 0.446323 0.773289
+vt 0.436175 0.773289
+vt 0.445453 0.651481
+vt 0.426028 0.773289
+vt 0.452628 0.644305
+vt 0.415880 0.773289
+vt 0.461066 0.638668
+vt 0.470441 0.634784
+vt 0.405733 0.773289
+vt 0.480393 0.632805
+vt 0.395585 0.773289
+vt 0.507028 0.631180
+vt 0.554183 0.734210
+vt 0.385438 0.797969
+vt 0.544230 0.732230
+vt 0.375290 0.797969
+vt 0.534855 0.728347
+vt 0.365143 0.797969
+vt 0.354995 0.797969
+vt 0.526418 0.722709
+vt 0.344848 0.797969
+vt 0.519243 0.715534
+vt 0.334700 0.797969
+vt 0.513605 0.707096
+vt 0.509721 0.697721
+vt 0.324553 0.797969
+vt 0.507742 0.687769
+vt 0.314405 0.797969
+vt 0.639125 0.797969
+vt 0.908735 0.903617
+vt 0.874676 0.931567
+vt 0.334815 0.932388
+vt 0.611202 0.932388
+vt 0.869878 0.924386
+vt 0.602565 0.932389
+vt 0.866572 0.916406
+vt 0.593928 0.932389
+vt 0.864888 0.907935
+vt 0.585291 0.932389
+vt 0.864888 0.899298
+vt 0.576654 0.932389
+vt 0.866573 0.890827
+vt 0.568017 0.932389
+vt 0.869878 0.882847
+vt 0.559379 0.932389
+vt 0.874676 0.875666
+vt 0.550742 0.932389
+vt 0.880784 0.869558
+vt 0.542105 0.932389
+vt 0.887965 0.864760
+vt 0.533468 0.932389
+vt 0.895945 0.861455
+vt 0.524831 0.932389
+vt 0.904416 0.859770
+vt 0.516194 0.932389
+vt 0.913053 0.859770
+vt 0.507557 0.932389
+vt 0.921524 0.861455
+vt 0.498920 0.932389
+vt 0.929504 0.864760
+vt 0.490283 0.932389
+vt 0.936685 0.869559
+vt 0.481646 0.932389
+vt 0.942793 0.875666
+vt 0.473009 0.932389
+vt 0.947591 0.882847
+vt 0.464371 0.932389
+vt 0.950896 0.890827
+vt 0.455734 0.932389
+vt 0.952581 0.899298
+vt 0.447097 0.932389
+vt 0.952581 0.907935
+vt 0.438460 0.932389
+vt 0.950896 0.916406
+vt 0.429823 0.932389
+vt 0.947591 0.924386
+vt 0.421186 0.932389
+vt 0.942792 0.931568
+vt 0.412549 0.932389
+vt 0.936685 0.937675
+vt 0.403912 0.932389
+vt 0.929504 0.942473
+vt 0.395275 0.932389
+vt 0.921524 0.945779
+vt 0.386637 0.932389
+vt 0.913053 0.947464
+vt 0.378001 0.932389
+vt 0.904416 0.947464
+vt 0.369363 0.932389
+vt 0.895945 0.945778
+vt 0.360726 0.932389
+vt 0.887965 0.942473
+vt 0.352089 0.932388
+vt 0.880783 0.937675
+vt 0.343452 0.932388
+vt 0.618830 0.816892
+vt 0.628978 0.816892
+vt 0.608683 0.816892
+vt 0.598535 0.816892
+vt 0.588388 0.816892
+vt 0.578240 0.816892
+vt 0.568093 0.816892
+vt 0.557945 0.816892
+vt 0.547798 0.816892
+vt 0.537650 0.816892
+vt 0.527503 0.816892
+vt 0.517355 0.816892
+vt 0.507208 0.816892
+vt 0.497060 0.816892
+vt 0.486913 0.816892
+vt 0.476765 0.816892
+vt 0.466618 0.816892
+vt 0.456470 0.816892
+vt 0.446323 0.816892
+vt 0.436175 0.816892
+vt 0.426028 0.816892
+vt 0.415880 0.816892
+vt 0.405733 0.816892
+vt 0.395585 0.816892
+vt 0.385438 0.816892
+vt 0.375290 0.816892
+vt 0.365143 0.816892
+vt 0.354995 0.816892
+vt 0.344848 0.816892
+vt 0.334700 0.816892
+vt 0.324553 0.816892
+vt 0.314405 0.816892
+vt 0.639125 0.816892
+vt 0.517095 0.669905
+vt 0.602565 0.946486
+vt 0.334815 0.946485
+vt 0.515410 0.678377
+vt 0.611202 0.946485
+vt 0.520400 0.661926
+vt 0.593928 0.946485
+vt 0.525198 0.654744
+vt 0.585291 0.946486
+vt 0.531306 0.648637
+vt 0.576654 0.946486
+vt 0.568017 0.946486
+vt 0.538487 0.643839
+vt 0.546467 0.640533
+vt 0.559379 0.946486
+vt 0.554938 0.638848
+vt 0.507247 0.729756
+vt 0.550742 0.946486
+vt 0.481149 0.728166
+vt 0.542105 0.971165
+vt 0.472678 0.726481
+vt 0.533468 0.971165
+vt 0.464698 0.723176
+vt 0.524831 0.971165
+vt 0.457517 0.718377
+vt 0.516194 0.971165
+vt 0.507557 0.971165
+vt 0.451409 0.712270
+vt 0.446611 0.705089
+vt 0.498920 0.971165
+vt 0.443305 0.697109
+vt 0.490283 0.971165
+vt 0.441620 0.688638
+vt 0.481646 0.971165
+vt 0.473008 0.971165
+vt 0.441620 0.680001
+vt 0.443305 0.671530
+vt 0.464371 0.971165
+vt 0.446610 0.663550
+vt 0.455734 0.971165
+vt 0.451409 0.656369
+vt 0.447097 0.971165
+vt 0.438460 0.971165
+vt 0.457516 0.650261
+vt 0.464698 0.645463
+vt 0.429823 0.971165
+vt 0.472677 0.642157
+vt 0.421186 0.971165
+vt 0.412549 0.971165
+vt 0.481149 0.640472
+vt 0.507247 0.638882
+vt 0.554938 0.726542
+vt 0.403912 0.946486
+vt 0.546467 0.724857
+vt 0.395275 0.946486
+vt 0.538487 0.721552
+vt 0.386637 0.946486
+vt 0.378001 0.946486
+vt 0.531306 0.716753
+vt 0.525198 0.710646
+vt 0.369363 0.946486
+vt 0.520400 0.703464
+vt 0.360726 0.946485
+vt 0.517095 0.695485
+vt 0.352089 0.946485
+vt 0.343452 0.946485
+vt 0.515410 0.687014
+vt 0.509722 0.667669
+vt 0.618830 0.797969
+vt 0.507742 0.677621
+vt 0.628978 0.797969
+vt 0.513605 0.658294
+vt 0.608683 0.797969
+vt 0.519242 0.649857
+vt 0.598535 0.797969
+vt 0.588388 0.797969
+vt 0.526418 0.642681
+vt 0.578240 0.797969
+vt 0.534855 0.637043
+vt 0.568093 0.797969
+vt 0.544230 0.633160
+vt 0.554183 0.631180
+vt 0.507028 0.737458
+vt 0.557945 0.797969
+vt 0.480393 0.735834
+vt 0.547798 0.773289
+vt 0.470441 0.733854
+vt 0.537650 0.773289
+vt 0.461066 0.729971
+vt 0.527503 0.773289
+vt 0.452629 0.724334
+vt 0.517355 0.773289
+vt 0.507208 0.773290
+vt 0.445453 0.717158
+vt 0.497060 0.773290
+vt 0.439816 0.708721
+vt 0.486913 0.773290
+vt 0.435932 0.699346
+vt 0.433953 0.689393
+vt 0.476765 0.773290
+vt 0.433953 0.679246
+vt 0.466618 0.773290
+vt 0.435932 0.669293
+vt 0.456470 0.773290
+vt 0.439815 0.659918
+vt 0.446323 0.773289
+vt 0.436175 0.773289
+vt 0.445453 0.651481
+vt 0.426028 0.773289
+vt 0.452628 0.644305
+vt 0.415880 0.773289
+vt 0.461066 0.638668
+vt 0.470441 0.634784
+vt 0.405733 0.773289
+vt 0.480393 0.632805
+vt 0.395585 0.773289
+vt 0.507028 0.631180
+vt 0.554183 0.734210
+vt 0.385438 0.797969
+vt 0.544230 0.732230
+vt 0.375290 0.797969
+vt 0.534855 0.728347
+vt 0.365143 0.797969
+vt 0.354995 0.797969
+vt 0.526418 0.722709
+vt 0.344848 0.797969
+vt 0.519243 0.715534
+vt 0.334700 0.797969
+vt 0.513605 0.707096
+vt 0.509721 0.697721
+vt 0.324553 0.797969
+vt 0.507742 0.687769
+vt 0.314405 0.797969
+vt 0.639125 0.797969
+vt 0.908268 0.746864
+vt 0.874210 0.774814
+vt 0.334815 0.932388
+vt 0.611202 0.932388
+vt 0.869411 0.767633
+vt 0.602565 0.932389
+vt 0.866106 0.759653
+vt 0.593928 0.932389
+vt 0.864421 0.751182
+vt 0.585291 0.932389
+vt 0.864421 0.742545
+vt 0.576654 0.932389
+vt 0.866106 0.734074
+vt 0.568017 0.932389
+vt 0.869412 0.726094
+vt 0.559379 0.932389
+vt 0.874210 0.718913
+vt 0.550742 0.932389
+vt 0.880318 0.712805
+vt 0.542105 0.932389
+vt 0.887499 0.708007
+vt 0.533468 0.932389
+vt 0.895479 0.704702
+vt 0.524831 0.932389
+vt 0.903950 0.703017
+vt 0.516194 0.932389
+vt 0.912587 0.703017
+vt 0.507557 0.932389
+vt 0.921058 0.704702
+vt 0.498920 0.932389
+vt 0.929038 0.708007
+vt 0.490283 0.932389
+vt 0.936219 0.712806
+vt 0.481646 0.932389
+vt 0.942326 0.718913
+vt 0.473009 0.932389
+vt 0.947125 0.726095
+vt 0.464371 0.932389
+vt 0.950430 0.734074
+vt 0.455734 0.932389
+vt 0.952115 0.742545
+vt 0.447097 0.932389
+vt 0.952115 0.751182
+vt 0.438460 0.932389
+vt 0.950430 0.759654
+vt 0.429823 0.932389
+vt 0.947125 0.767633
+vt 0.421186 0.932389
+vt 0.942326 0.774815
+vt 0.412549 0.932389
+vt 0.936219 0.780922
+vt 0.403912 0.932389
+vt 0.929037 0.785721
+vt 0.395275 0.932389
+vt 0.921058 0.789026
+vt 0.386637 0.932389
+vt 0.912586 0.790711
+vt 0.378001 0.932389
+vt 0.903949 0.790711
+vt 0.369363 0.932389
+vt 0.895478 0.789025
+vt 0.360726 0.932389
+vt 0.887499 0.785720
+vt 0.352089 0.932388
+vt 0.880317 0.780922
+vt 0.343452 0.932388
+vt 0.618830 0.816892
+vt 0.628978 0.816892
+vt 0.608683 0.816892
+vt 0.598535 0.816892
+vt 0.588388 0.816892
+vt 0.578240 0.816892
+vt 0.568093 0.816892
+vt 0.557945 0.816892
+vt 0.547798 0.816892
+vt 0.537650 0.816892
+vt 0.527503 0.816892
+vt 0.517355 0.816892
+vt 0.507208 0.816892
+vt 0.497060 0.816892
+vt 0.486913 0.816892
+vt 0.476765 0.816892
+vt 0.466618 0.816892
+vt 0.456470 0.816892
+vt 0.446323 0.816892
+vt 0.436175 0.816892
+vt 0.426028 0.816892
+vt 0.415880 0.816892
+vt 0.405733 0.816892
+vt 0.395585 0.816892
+vt 0.385438 0.816892
+vt 0.375290 0.816892
+vt 0.365143 0.816892
+vt 0.354995 0.816892
+vt 0.344848 0.816892
+vt 0.334700 0.816892
+vt 0.324553 0.816892
+vt 0.314405 0.816892
+vt 0.639125 0.816892
+vt 0.517095 0.669905
+vt 0.602565 0.946486
+vt 0.334815 0.946485
+vt 0.515410 0.678377
+vt 0.611202 0.946485
+vt 0.520400 0.661926
+vt 0.593928 0.946485
+vt 0.525198 0.654744
+vt 0.585291 0.946486
+vt 0.531306 0.648637
+vt 0.576654 0.946486
+vt 0.568017 0.946486
+vt 0.538487 0.643839
+vt 0.546467 0.640533
+vt 0.559379 0.946486
+vt 0.554938 0.638848
+vt 0.507247 0.729756
+vt 0.550742 0.946486
+vt 0.481149 0.728166
+vt 0.542105 0.971165
+vt 0.472678 0.726481
+vt 0.533468 0.971165
+vt 0.464698 0.723176
+vt 0.524831 0.971165
+vt 0.457517 0.718377
+vt 0.516194 0.971165
+vt 0.507557 0.971165
+vt 0.451409 0.712270
+vt 0.446611 0.705089
+vt 0.498920 0.971165
+vt 0.443305 0.697109
+vt 0.490283 0.971165
+vt 0.441620 0.688638
+vt 0.481646 0.971165
+vt 0.473008 0.971165
+vt 0.441620 0.680001
+vt 0.443305 0.671530
+vt 0.464371 0.971165
+vt 0.446610 0.663550
+vt 0.455734 0.971165
+vt 0.451409 0.656369
+vt 0.447097 0.971165
+vt 0.438460 0.971165
+vt 0.457516 0.650261
+vt 0.464698 0.645463
+vt 0.429823 0.971165
+vt 0.472677 0.642157
+vt 0.421186 0.971165
+vt 0.412549 0.971165
+vt 0.481149 0.640472
+vt 0.507247 0.638882
+vt 0.554938 0.726542
+vt 0.403912 0.946486
+vt 0.546467 0.724857
+vt 0.395275 0.946486
+vt 0.538487 0.721552
+vt 0.386637 0.946486
+vt 0.378001 0.946486
+vt 0.531306 0.716753
+vt 0.525198 0.710646
+vt 0.369363 0.946486
+vt 0.520400 0.703464
+vt 0.360726 0.946485
+vt 0.517095 0.695485
+vt 0.352089 0.946485
+vt 0.343452 0.946485
+vt 0.515410 0.687014
+vt 0.509722 0.667669
+vt 0.618830 0.797969
+vt 0.507742 0.677621
+vt 0.628978 0.797969
+vt 0.513605 0.658294
+vt 0.608683 0.797969
+vt 0.519242 0.649857
+vt 0.598535 0.797969
+vt 0.588388 0.797969
+vt 0.526418 0.642681
+vt 0.578240 0.797969
+vt 0.534855 0.637043
+vt 0.568093 0.797969
+vt 0.544230 0.633160
+vt 0.554183 0.631180
+vt 0.507028 0.737458
+vt 0.557945 0.797969
+vt 0.480393 0.735834
+vt 0.547798 0.773289
+vt 0.470441 0.733854
+vt 0.537650 0.773289
+vt 0.461066 0.729971
+vt 0.527503 0.773289
+vt 0.452629 0.724334
+vt 0.517355 0.773289
+vt 0.507208 0.773290
+vt 0.445453 0.717158
+vt 0.497060 0.773290
+vt 0.439816 0.708721
+vt 0.486913 0.773290
+vt 0.435932 0.699346
+vt 0.433953 0.689393
+vt 0.476765 0.773290
+vt 0.433953 0.679246
+vt 0.466618 0.773290
+vt 0.435932 0.669293
+vt 0.456470 0.773290
+vt 0.439815 0.659918
+vt 0.446323 0.773289
+vt 0.436175 0.773289
+vt 0.445453 0.651481
+vt 0.426028 0.773289
+vt 0.452628 0.644305
+vt 0.415880 0.773289
+vt 0.461066 0.638668
+vt 0.470441 0.634784
+vt 0.405733 0.773289
+vt 0.480393 0.632805
+vt 0.395585 0.773289
+vt 0.507028 0.631180
+vt 0.554183 0.734210
+vt 0.385438 0.797969
+vt 0.544230 0.732230
+vt 0.375290 0.797969
+vt 0.534855 0.728347
+vt 0.365143 0.797969
+vt 0.354995 0.797969
+vt 0.526418 0.722709
+vt 0.344848 0.797969
+vt 0.519243 0.715534
+vt 0.334700 0.797969
+vt 0.513605 0.707096
+vt 0.509721 0.697721
+vt 0.324553 0.797969
+vt 0.507742 0.687769
+vt 0.314405 0.797969
+vt 0.639125 0.797969
+s 0
+g Cube_Material
+usemtl Material
+f 6/11/1 14/22/1 10/18/1 2/3/1
+f 1/1/2 9/17/2 13/21/2 5/9/2
+f 12/20/3 16/24/3 15/23/3 11/19/3
+f 10/18/3 14/22/3 13/21/3 9/17/3
+f 3/5/2 11/19/2 15/23/2 7/13/2
+f 8/15/1 16/24/1 12/20/1 4/7/1
+f 7/14/4 15/23/4 16/24/4 8/16/4
+f 5/10/4 13/21/4 14/22/4 6/12/4
+f 4/8/5 12/20/5 11/19/5 3/6/5
+f 2/4/5 10/18/5 9/17/5 1/2/5
+f 18/29/3 17/25/3 19/33/3 20/34/3
+f 33/57/4 34/58/4 36/60/4 35/59/4
+f 24/44/6 23/43/6 21/35/6 22/39/6
+f 25/45/1 29/53/1 32/56/1 28/51/1
+f 17/26/1 21/36/1 23/43/1 19/33/1
+f 22/40/2 18/30/2 20/34/2 24/44/2
+f 25/46/5 17/27/5 18/31/5 26/48/5
+f 27/50/5 22/41/5 21/37/5 28/52/5
+f 26/47/5 18/32/5 22/42/5 27/49/5
+f 28/51/5 21/38/5 17/28/5 25/45/5
+f 31/55/5 32/56/5 29/53/5 30/54/5
+f 27/49/2 31/55/2 30/54/2 26/47/2
+f 28/52/6 32/56/6 31/55/6 27/50/6
+f 26/48/3 30/54/3 29/53/3 25/46/3
+f 33/57/4 20/34/4 19/33/4 34/58/4
+f 35/59/4 24/44/4 20/34/4 33/57/4
+f 36/60/4 23/43/4 24/44/4 35/59/4
+f 34/58/4 19/33/4 23/43/4 36/60/4
+f 37/61/4 39/65/4 38/62/4
+f 37/61/4 40/67/4 39/65/4
+f 37/61/4 41/69/4 40/67/4
+f 37/61/4 42/71/4 41/69/4
+f 37/61/4 43/73/4 42/71/4
+f 37/61/4 44/75/4 43/73/4
+f 37/61/4 45/77/4 44/75/4
+f 37/61/4 46/79/4 45/77/4
+f 37/61/4 47/81/4 46/79/4
+f 37/61/4 48/83/4 47/81/4
+f 37/61/4 49/85/4 48/83/4
+f 37/61/4 50/87/4 49/85/4
+f 37/61/4 51/89/4 50/87/4
+f 37/61/4 52/91/4 51/89/4
+f 37/61/4 53/93/4 52/91/4
+f 37/61/4 54/95/4 53/93/4
+f 37/61/4 55/97/4 54/95/4
+f 37/61/4 56/99/4 55/97/4
+f 37/61/4 57/101/4 56/99/4
+f 37/61/4 58/103/4 57/101/4
+f 37/61/4 59/105/4 58/103/4
+f 37/61/4 60/107/4 59/105/4
+f 37/61/4 61/109/4 60/107/4
+f 37/61/4 62/111/4 61/109/4
+f 37/61/4 63/113/4 62/111/4
+f 37/61/4 64/115/4 63/113/4
+f 37/61/4 65/117/4 64/115/4
+f 37/61/4 66/119/4 65/117/4
+f 37/61/4 67/121/4 66/119/4
+f 37/61/4 68/123/4 67/121/4
+f 37/61/4 69/125/4 68/123/4
+f 37/61/4 38/62/4 69/125/4
+f 99/156/7 163/287/7 164/290/7 100/157/7
+f 84/141/8 148/256/8 149/259/8 85/142/8
+f 59/106/9 123/205/9 122/202/9 58/104/9
+f 38/63/10 103/162/10 133/225/10 69/126/10
+f 44/76/11 108/174/11 107/171/11 43/74/11
+f 91/148/11 155/270/11 156/273/11 92/149/11
+f 76/133/12 140/239/12 141/243/12 77/134/12
+f 55/98/13 119/197/13 118/194/13 54/96/13
+f 66/120/14 130/220/14 129/217/14 65/118/14
+f 98/155/15 162/285/15 163/287/15 99/156/15
+f 83/140/16 147/254/16 148/256/16 84/141/16
+f 40/68/17 104/166/17 102/161/17 39/66/17
+f 51/90/15 115/189/15 114/186/15 50/88/15
+f 62/112/18 126/212/18 125/208/18 61/110/18
+f 90/147/19 154/268/19 155/270/19 91/148/19
+f 75/132/20 139/237/20 140/239/20 76/133/20
+f 47/82/21 111/181/21 110/179/21 46/80/21
+f 97/154/22 161/283/22 162/285/22 98/155/22
+f 82/139/23 146/252/23 147/254/23 83/140/23
+f 58/104/24 122/202/24 121/201/24 57/102/24
+f 69/126/8 133/225/8 132/224/8 68/124/8
+f 89/146/25 153/266/25 154/268/25 90/147/25
+f 43/74/19 107/171/19 106/170/19 42/72/19
+f 54/96/26 118/194/26 117/193/26 53/94/26
+f 74/131/9 138/235/9 139/237/9 75/132/9
+f 65/118/27 129/217/27 128/216/27 64/116/27
+f 96/153/28 160/282/28 161/283/28 97/154/28
+f 81/138/29 145/251/29 146/252/29 82/139/29
+f 50/88/30 114/186/30 113/185/30 49/86/30
+f 88/145/31 152/265/31 153/266/31 89/146/31
+f 61/110/12 125/208/12 124/207/12 60/108/12
+f 73/130/24 137/234/24 138/235/24 74/131/24
+f 102/160/4 134/227/4 135/229/4 103/163/4
+f 104/165/4 136/231/4 134/227/4 102/160/4
+f 105/167/4 137/233/4 136/231/4 104/165/4
+f 106/169/4 138/236/4 137/233/4 105/167/4
+f 107/172/4 139/238/4 138/236/4 106/169/4
+f 108/173/4 140/240/4 139/238/4 107/172/4
+f 109/175/4 141/241/4 140/240/4 108/173/4
+f 110/178/32 142/244/32 141/242/32 109/176/32
+f 111/180/4 143/246/4 142/244/4 110/178/4
+f 112/182/4 144/248/4 143/246/4 111/180/4
+f 113/184/4 145/250/4 144/248/4 112/182/4
+f 114/187/4 146/253/4 145/250/4 113/184/4
+f 115/188/4 147/255/4 146/253/4 114/187/4
+f 116/190/4 148/257/4 147/255/4 115/188/4
+f 117/192/4 149/258/4 148/257/4 116/190/4
+f 118/195/4 150/260/4 149/258/4 117/192/4
+f 119/196/4 151/262/4 150/260/4 118/195/4
+f 120/198/4 152/264/4 151/262/4 119/196/4
+f 121/200/4 153/267/4 152/264/4 120/198/4
+f 122/203/4 154/269/4 153/267/4 121/200/4
+f 123/204/4 155/271/4 154/269/4 122/203/4
+f 124/206/4 156/272/4 155/271/4 123/204/4
+f 125/209/4 157/274/4 156/272/4 124/206/4
+f 126/210/32 158/276/32 157/274/32 125/209/32
+f 127/213/4 159/279/4 158/277/4 126/211/4
+f 128/215/4 160/281/4 159/279/4 127/213/4
+f 129/218/4 161/284/4 160/281/4 128/215/4
+f 130/219/4 162/286/4 161/284/4 129/218/4
+f 131/221/4 163/288/4 162/286/4 130/219/4
+f 132/223/4 164/289/4 163/288/4 131/221/4
+f 133/226/4 165/291/4 164/289/4 132/223/4
+f 103/163/4 135/229/4 165/291/4 133/226/4
+f 48/84/33 112/183/33 111/181/33 47/82/33
+f 77/134/34 141/243/34 142/245/34 78/135/34
+f 92/149/35 156/273/35 157/275/35 93/150/35
+f 63/114/36 127/214/36 126/212/36 62/112/36
+f 52/92/7 116/191/7 115/189/7 51/90/7
+f 71/128/37 135/230/37 134/228/37 70/127/37
+f 41/70/31 105/168/31 104/166/31 40/68/31
+f 85/142/10 149/259/10 150/261/10 86/143/10
+f 100/157/38 164/290/38 165/292/38 101/158/38
+f 67/122/23 131/222/23 130/220/23 66/120/23
+f 56/100/39 120/199/39 119/197/39 55/98/39
+f 78/135/36 142/245/36 143/247/36 79/136/36
+f 93/150/40 157/275/40 158/278/40 94/151/40
+f 45/78/35 109/177/35 108/174/35 44/76/35
+f 70/127/39 134/228/39 136/232/39 72/129/39
+f 60/108/20 124/207/20 123/205/20 59/106/20
+f 86/143/41 150/261/41 151/263/41 87/144/41
+f 101/159/26 165/293/26 135/230/26 71/128/26
+f 49/86/28 113/185/28 112/183/28 48/84/28
+f 79/136/42 143/247/42 144/249/42 80/137/42
+f 94/151/43 158/278/43 159/280/43 95/152/43
+f 64/116/42 128/216/42 127/214/42 63/114/42
+f 53/94/38 117/193/38 116/191/38 52/92/38
+f 72/129/44 136/232/44 137/234/44 73/130/44
+f 42/72/45 106/170/45 105/168/45 41/70/45
+f 87/144/17 151/263/17 152/265/17 88/145/17
+f 68/124/16 132/224/16 131/222/16 67/122/16
+f 39/66/41 102/161/41 103/164/41 38/64/41
+f 57/102/44 121/201/44 120/199/44 56/100/44
+f 80/137/27 144/249/27 145/251/27 81/138/27
+f 95/152/33 159/280/33 160/282/33 96/153/33
+f 46/80/46 110/179/46 109/177/46 45/78/46
+f 166/294/4 168/298/4 167/295/4
+f 166/294/4 169/300/4 168/298/4
+f 166/294/4 170/302/4 169/300/4
+f 166/294/4 171/304/4 170/302/4
+f 166/294/4 172/306/4 171/304/4
+f 166/294/4 173/308/4 172/306/4
+f 166/294/4 174/310/4 173/308/4
+f 166/294/4 175/312/4 174/310/4
+f 166/294/4 176/314/4 175/312/4
+f 166/294/4 177/316/4 176/314/4
+f 166/294/4 178/318/4 177/316/4
+f 166/294/4 179/320/4 178/318/4
+f 166/294/4 180/322/4 179/320/4
+f 166/294/4 181/324/4 180/322/4
+f 166/294/4 182/326/4 181/324/4
+f 166/294/4 183/328/4 182/326/4
+f 166/294/4 184/330/4 183/328/4
+f 166/294/4 185/332/4 184/330/4
+f 166/294/4 186/334/4 185/332/4
+f 166/294/4 187/336/4 186/334/4
+f 166/294/4 188/338/4 187/336/4
+f 166/294/4 189/340/4 188/338/4
+f 166/294/4 190/342/4 189/340/4
+f 166/294/4 191/344/4 190/342/4
+f 166/294/4 192/346/4 191/344/4
+f 166/294/4 193/348/4 192/346/4
+f 166/294/4 194/350/4 193/348/4
+f 166/294/4 195/352/4 194/350/4
+f 166/294/4 196/354/4 195/352/4
+f 166/294/4 197/356/4 196/354/4
+f 166/294/4 198/358/4 197/356/4
+f 166/294/4 167/295/4 198/358/4
+f 228/389/7 292/520/7 293/523/7 229/390/7
+f 213/374/8 277/489/8 278/492/8 214/375/8
+f 188/339/9 252/438/9 251/435/9 187/337/9
+f 167/296/10 232/395/10 262/458/10 198/359/10
+f 173/309/11 237/407/11 236/404/11 172/307/11
+f 220/381/11 284/503/11 285/506/11 221/382/11
+f 205/366/12 269/472/12 270/476/12 206/367/12
+f 184/331/13 248/430/13 247/427/13 183/329/13
+f 195/353/14 259/453/14 258/450/14 194/351/14
+f 227/388/15 291/518/15 292/520/15 228/389/15
+f 212/373/16 276/487/16 277/489/16 213/374/16
+f 169/301/17 233/399/17 231/394/17 168/299/17
+f 180/323/15 244/422/15 243/419/15 179/321/15
+f 191/345/18 255/445/18 254/441/18 190/343/18
+f 219/380/19 283/501/19 284/503/19 220/381/19
+f 204/365/20 268/470/20 269/472/20 205/366/20
+f 176/315/21 240/414/21 239/412/21 175/313/21
+f 226/387/22 290/516/22 291/518/22 227/388/22
+f 211/372/23 275/485/23 276/487/23 212/373/23
+f 187/337/24 251/435/24 250/434/24 186/335/24
+f 198/359/8 262/458/8 261/457/8 197/357/8
+f 218/379/25 282/499/25 283/501/25 219/380/25
+f 172/307/19 236/404/19 235/403/19 171/305/19
+f 183/329/26 247/427/26 246/426/26 182/327/26
+f 203/364/9 267/468/9 268/470/9 204/365/9
+f 194/351/27 258/450/27 257/449/27 193/349/27
+f 225/386/28 289/515/28 290/516/28 226/387/28
+f 210/371/29 274/484/29 275/485/29 211/372/29
+f 179/321/30 243/419/30 242/418/30 178/319/30
+f 217/378/31 281/498/31 282/499/31 218/379/31
+f 190/343/12 254/441/12 253/440/12 189/341/12
+f 202/363/24 266/467/24 267/468/24 203/364/24
+f 231/393/4 263/460/4 264/462/4 232/396/4
+f 233/398/4 265/464/4 263/460/4 231/393/4
+f 234/400/4 266/466/4 265/464/4 233/398/4
+f 235/402/4 267/469/4 266/466/4 234/400/4
+f 236/405/4 268/471/4 267/469/4 235/402/4
+f 237/406/4 269/473/4 268/471/4 236/405/4
+f 238/408/4 270/474/4 269/473/4 237/406/4
+f 239/411/32 271/477/32 270/475/32 238/409/32
+f 240/413/4 272/479/4 271/477/4 239/411/4
+f 241/415/4 273/481/4 272/479/4 240/413/4
+f 242/417/4 274/483/4 273/481/4 241/415/4
+f 243/420/4 275/486/4 274/483/4 242/417/4
+f 244/421/4 276/488/4 275/486/4 243/420/4
+f 245/423/4 277/490/4 276/488/4 244/421/4
+f 246/425/4 278/491/4 277/490/4 245/423/4
+f 247/428/4 279/493/4 278/491/4 246/425/4
+f 248/429/4 280/495/4 279/493/4 247/428/4
+f 249/431/4 281/497/4 280/495/4 248/429/4
+f 250/433/4 282/500/4 281/497/4 249/431/4
+f 251/436/4 283/502/4 282/500/4 250/433/4
+f 252/437/4 284/504/4 283/502/4 251/436/4
+f 253/439/4 285/505/4 284/504/4 252/437/4
+f 254/442/4 286/507/4 285/505/4 253/439/4
+f 255/443/32 287/509/32 286/507/32 254/442/32
+f 256/446/4 288/512/4 287/510/4 255/444/4
+f 257/448/4 289/514/4 288/512/4 256/446/4
+f 258/451/4 290/517/4 289/514/4 257/448/4
+f 259/452/4 291/519/4 290/517/4 258/451/4
+f 260/454/4 292/521/4 291/519/4 259/452/4
+f 261/456/4 293/522/4 292/521/4 260/454/4
+f 262/459/4 294/524/4 293/522/4 261/456/4
+f 232/396/4 264/462/4 294/524/4 262/459/4
+f 177/317/33 241/416/33 240/414/33 176/315/33
+f 206/367/34 270/476/34 271/478/34 207/368/34
+f 221/382/35 285/506/35 286/508/35 222/383/35
+f 192/347/36 256/447/36 255/445/36 191/345/36
+f 181/325/7 245/424/7 244/422/7 180/323/7
+f 200/361/37 264/463/37 263/461/37 199/360/37
+f 170/303/31 234/401/31 233/399/31 169/301/31
+f 214/375/10 278/492/10 279/494/10 215/376/10
+f 229/390/38 293/523/38 294/525/38 230/391/38
+f 196/355/23 260/455/23 259/453/23 195/353/23
+f 185/333/39 249/432/39 248/430/39 184/331/39
+f 207/368/36 271/478/36 272/480/36 208/369/36
+f 222/383/40 286/508/40 287/511/40 223/384/40
+f 174/311/35 238/410/35 237/407/35 173/309/35
+f 199/360/39 263/461/39 265/465/39 201/362/39
+f 189/341/20 253/440/20 252/438/20 188/339/20
+f 215/376/41 279/494/41 280/496/41 216/377/41
+f 230/392/26 294/526/26 264/463/26 200/361/26
+f 178/319/28 242/418/28 241/416/28 177/317/28
+f 208/369/42 272/480/42 273/482/42 209/370/42
+f 223/384/43 287/511/43 288/513/43 224/385/43
+f 193/349/42 257/449/42 256/447/42 192/347/42
+f 182/327/38 246/426/38 245/424/38 181/325/38
+f 201/362/44 265/465/44 266/467/44 202/363/44
+f 171/305/45 235/403/45 234/401/45 170/303/45
+f 216/377/17 280/496/17 281/498/17 217/378/17
+f 197/357/16 261/457/16 260/455/16 196/355/16
+f 168/299/41 231/394/41 232/397/41 167/297/41
+f 186/335/44 250/434/44 249/432/44 185/333/44
+f 209/370/27 273/482/27 274/484/27 210/371/27
+f 224/385/33 288/513/33 289/515/33 225/386/33
+f 175/313/46 239/412/46 238/410/46 174/311/46
+f 295/527/4 297/531/4 296/528/4
+f 295/527/4 298/533/4 297/531/4
+f 295/527/4 299/535/4 298/533/4
+f 295/527/4 300/537/4 299/535/4
+f 295/527/4 301/539/4 300/537/4
+f 295/527/4 302/541/4 301/539/4
+f 295/527/4 303/543/4 302/541/4
+f 295/527/4 304/545/4 303/543/4
+f 295/527/4 305/547/4 304/545/4
+f 295/527/4 306/549/4 305/547/4
+f 295/527/4 307/551/4 306/549/4
+f 295/527/4 308/553/4 307/551/4
+f 295/527/4 309/555/4 308/553/4
+f 295/527/4 310/557/4 309/555/4
+f 295/527/4 311/559/4 310/557/4
+f 295/527/4 312/561/4 311/559/4
+f 295/527/4 313/563/4 312/561/4
+f 295/527/4 314/565/4 313/563/4
+f 295/527/4 315/567/4 314/565/4
+f 295/527/4 316/569/4 315/567/4
+f 295/527/4 317/571/4 316/569/4
+f 295/527/4 318/573/4 317/571/4
+f 295/527/4 319/575/4 318/573/4
+f 295/527/4 320/577/4 319/575/4
+f 295/527/4 321/579/4 320/577/4
+f 295/527/4 322/581/4 321/579/4
+f 295/527/4 323/583/4 322/581/4
+f 295/527/4 324/585/4 323/583/4
+f 295/527/4 325/587/4 324/585/4
+f 295/527/4 326/589/4 325/587/4
+f 295/527/4 327/591/4 326/589/4
+f 295/527/4 296/528/4 327/591/4
+f 357/622/7 421/753/7 422/756/7 358/623/7
+f 342/607/8 406/722/8 407/725/8 343/608/8
+f 317/572/9 381/671/9 380/668/9 316/570/9
+f 296/529/10 361/628/10 391/691/10 327/592/10
+f 302/542/11 366/640/11 365/637/11 301/540/11
+f 349/614/11 413/736/11 414/739/11 350/615/11
+f 334/599/12 398/705/12 399/709/12 335/600/12
+f 313/564/13 377/663/13 376/660/13 312/562/13
+f 324/586/14 388/686/14 387/683/14 323/584/14
+f 356/621/15 420/751/15 421/753/15 357/622/15
+f 341/606/16 405/720/16 406/722/16 342/607/16
+f 298/534/17 362/632/17 360/627/17 297/532/17
+f 309/556/15 373/655/15 372/652/15 308/554/15
+f 320/578/18 384/678/18 383/674/18 319/576/18
+f 348/613/19 412/734/19 413/736/19 349/614/19
+f 333/598/20 397/703/20 398/705/20 334/599/20
+f 305/548/21 369/647/21 368/645/21 304/546/21
+f 355/620/22 419/749/22 420/751/22 356/621/22
+f 340/605/23 404/718/23 405/720/23 341/606/23
+f 316/570/24 380/668/24 379/667/24 315/568/24
+f 327/592/8 391/691/8 390/690/8 326/590/8
+f 347/612/25 411/732/25 412/734/25 348/613/25
+f 301/540/19 365/637/19 364/636/19 300/538/19
+f 312/562/26 376/660/26 375/659/26 311/560/26
+f 332/597/9 396/701/9 397/703/9 333/598/9
+f 323/584/27 387/683/27 386/682/27 322/582/27
+f 354/619/28 418/748/28 419/749/28 355/620/28
+f 339/604/29 403/717/29 404/718/29 340/605/29
+f 308/554/30 372/652/30 371/651/30 307/552/30
+f 346/611/31 410/731/31 411/732/31 347/612/31
+f 319/576/12 383/674/12 382/673/12 318/574/12
+f 331/596/24 395/700/24 396/701/24 332/597/24
+f 360/626/4 392/693/4 393/695/4 361/629/4
+f 362/631/4 394/697/4 392/693/4 360/626/4
+f 363/633/4 395/699/4 394/697/4 362/631/4
+f 364/635/4 396/702/4 395/699/4 363/633/4
+f 365/638/4 397/704/4 396/702/4 364/635/4
+f 366/639/4 398/706/4 397/704/4 365/638/4
+f 367/641/4 399/707/4 398/706/4 366/639/4
+f 368/644/32 400/710/32 399/708/32 367/642/32
+f 369/646/4 401/712/4 400/710/4 368/644/4
+f 370/648/4 402/714/4 401/712/4 369/646/4
+f 371/650/4 403/716/4 402/714/4 370/648/4
+f 372/653/4 404/719/4 403/716/4 371/650/4
+f 373/654/4 405/721/4 404/719/4 372/653/4
+f 374/656/4 406/723/4 405/721/4 373/654/4
+f 375/658/4 407/724/4 406/723/4 374/656/4
+f 376/661/4 408/726/4 407/724/4 375/658/4
+f 377/662/4 409/728/4 408/726/4 376/661/4
+f 378/664/4 410/730/4 409/728/4 377/662/4
+f 379/666/4 411/733/4 410/730/4 378/664/4
+f 380/669/4 412/735/4 411/733/4 379/666/4
+f 381/670/4 413/737/4 412/735/4 380/669/4
+f 382/672/4 414/738/4 413/737/4 381/670/4
+f 383/675/4 415/740/4 414/738/4 382/672/4
+f 384/676/32 416/742/32 415/740/32 383/675/32
+f 385/679/4 417/745/4 416/743/4 384/677/4
+f 386/681/4 418/747/4 417/745/4 385/679/4
+f 387/684/4 419/750/4 418/747/4 386/681/4
+f 388/685/4 420/752/4 419/750/4 387/684/4
+f 389/687/4 421/754/4 420/752/4 388/685/4
+f 390/689/4 422/755/4 421/754/4 389/687/4
+f 391/692/4 423/757/4 422/755/4 390/689/4
+f 361/629/4 393/695/4 423/757/4 391/692/4
+f 306/550/33 370/649/33 369/647/33 305/548/33
+f 335/600/34 399/709/34 400/711/34 336/601/34
+f 350/615/35 414/739/35 415/741/35 351/616/35
+f 321/580/36 385/680/36 384/678/36 320/578/36
+f 310/558/7 374/657/7 373/655/7 309/556/7
+f 329/594/37 393/696/37 392/694/37 328/593/37
+f 299/536/31 363/634/31 362/632/31 298/534/31
+f 343/608/10 407/725/10 408/727/10 344/609/10
+f 358/623/38 422/756/38 423/758/38 359/624/38
+f 325/588/23 389/688/23 388/686/23 324/586/23
+f 314/566/39 378/665/39 377/663/39 313/564/39
+f 336/601/36 400/711/36 401/713/36 337/602/36
+f 351/616/40 415/741/40 416/744/40 352/617/40
+f 303/544/35 367/643/35 366/640/35 302/542/35
+f 328/593/39 392/694/39 394/698/39 330/595/39
+f 318/574/20 382/673/20 381/671/20 317/572/20
+f 344/609/41 408/727/41 409/729/41 345/610/41
+f 359/625/26 423/759/26 393/696/26 329/594/26
+f 307/552/28 371/651/28 370/649/28 306/550/28
+f 337/602/42 401/713/42 402/715/42 338/603/42
+f 352/617/43 416/744/43 417/746/43 353/618/43
+f 322/582/42 386/682/42 385/680/42 321/580/42
+f 311/560/38 375/659/38 374/657/38 310/558/38
+f 330/595/44 394/698/44 395/700/44 331/596/44
+f 300/538/45 364/636/45 363/634/45 299/536/45
+f 345/610/17 409/729/17 410/731/17 346/611/17
+f 326/590/16 390/690/16 389/688/16 325/588/16
+f 297/532/41 360/627/41 361/630/41 296/530/41
+f 315/568/44 379/667/44 378/665/44 314/566/44
+f 338/603/27 402/715/27 403/717/27 339/604/27
+f 353/618/33 417/746/33 418/748/33 354/619/33
+f 304/546/46 368/645/46 367/643/46 303/544/46
+f 424/760/4 426/764/4 425/761/4
+f 424/760/4 427/766/4 426/764/4
+f 424/760/4 428/768/4 427/766/4
+f 424/760/4 429/770/4 428/768/4
+f 424/760/4 430/772/4 429/770/4
+f 424/760/4 431/774/4 430/772/4
+f 424/760/4 432/776/4 431/774/4
+f 424/760/4 433/778/4 432/776/4
+f 424/760/4 434/780/4 433/778/4
+f 424/760/4 435/782/4 434/780/4
+f 424/760/4 436/784/4 435/782/4
+f 424/760/4 437/786/4 436/784/4
+f 424/760/4 438/788/4 437/786/4
+f 424/760/4 439/790/4 438/788/4
+f 424/760/4 440/792/4 439/790/4
+f 424/760/4 441/794/4 440/792/4
+f 424/760/4 442/796/4 441/794/4
+f 424/760/4 443/798/4 442/796/4
+f 424/760/4 444/800/4 443/798/4
+f 424/760/4 445/802/4 444/800/4
+f 424/760/4 446/804/4 445/802/4
+f 424/760/4 447/806/4 446/804/4
+f 424/760/4 448/808/4 447/806/4
+f 424/760/4 449/810/4 448/808/4
+f 424/760/4 450/812/4 449/810/4
+f 424/760/4 451/814/4 450/812/4
+f 424/760/4 452/816/4 451/814/4
+f 424/760/4 453/818/4 452/816/4
+f 424/760/4 454/820/4 453/818/4
+f 424/760/4 455/822/4 454/820/4
+f 424/760/4 456/824/4 455/822/4
+f 424/760/4 425/761/4 456/824/4
+f 486/855/7 550/986/7 551/989/7 487/856/7
+f 471/840/8 535/955/8 536/958/8 472/841/8
+f 446/805/9 510/904/9 509/901/9 445/803/9
+f 425/762/10 490/861/10 520/924/10 456/825/10
+f 431/775/11 495/873/11 494/870/11 430/773/11
+f 478/847/11 542/969/11 543/972/11 479/848/11
+f 463/832/12 527/938/12 528/942/12 464/833/12
+f 442/797/13 506/896/13 505/893/13 441/795/13
+f 453/819/14 517/919/14 516/916/14 452/817/14
+f 485/854/15 549/984/15 550/986/15 486/855/15
+f 470/839/16 534/953/16 535/955/16 471/840/16
+f 427/767/17 491/865/17 489/860/17 426/765/17
+f 438/789/15 502/888/15 501/885/15 437/787/15
+f 449/811/18 513/911/18 512/907/18 448/809/18
+f 477/846/19 541/967/19 542/969/19 478/847/19
+f 462/831/20 526/936/20 527/938/20 463/832/20
+f 434/781/21 498/880/21 497/878/21 433/779/21
+f 484/853/30 548/982/30 549/984/30 485/854/30
+f 469/838/23 533/951/23 534/953/23 470/839/23
+f 445/803/24 509/901/24 508/900/24 444/801/24
+f 456/825/8 520/924/8 519/923/8 455/823/8
+f 476/845/25 540/965/25 541/967/25 477/846/25
+f 430/773/19 494/870/19 493/869/19 429/771/19
+f 441/795/26 505/893/26 504/892/26 440/793/26
+f 461/830/9 525/934/9 526/936/9 462/831/9
+f 452/817/27 516/916/27 515/915/27 451/815/27
+f 483/852/28 547/981/28 548/982/28 484/853/28
+f 468/837/29 532/950/29 533/951/29 469/838/29
+f 437/787/30 501/885/30 500/884/30 436/785/30
+f 475/844/31 539/964/31 540/965/31 476/845/31
+f 448/809/12 512/907/12 511/906/12 447/807/12
+f 460/829/24 524/933/24 525/934/24 461/830/24
+f 489/859/4 521/926/4 522/928/4 490/862/4
+f 491/864/4 523/930/4 521/926/4 489/859/4
+f 492/866/4 524/932/4 523/930/4 491/864/4
+f 493/868/4 525/935/4 524/932/4 492/866/4
+f 494/871/4 526/937/4 525/935/4 493/868/4
+f 495/872/4 527/939/4 526/937/4 494/871/4
+f 496/874/4 528/940/4 527/939/4 495/872/4
+f 497/877/32 529/943/32 528/941/32 496/875/32
+f 498/879/4 530/945/4 529/943/4 497/877/4
+f 499/881/4 531/947/4 530/945/4 498/879/4
+f 500/883/4 532/949/4 531/947/4 499/881/4
+f 501/886/4 533/952/4 532/949/4 500/883/4
+f 502/887/4 534/954/4 533/952/4 501/886/4
+f 503/889/4 535/956/4 534/954/4 502/887/4
+f 504/891/4 536/957/4 535/956/4 503/889/4
+f 505/894/4 537/959/4 536/957/4 504/891/4
+f 506/895/4 538/961/4 537/959/4 505/894/4
+f 507/897/4 539/963/4 538/961/4 506/895/4
+f 508/899/4 540/966/4 539/963/4 507/897/4
+f 509/902/4 541/968/4 540/966/4 508/899/4
+f 510/903/4 542/970/4 541/968/4 509/902/4
+f 511/905/4 543/971/4 542/970/4 510/903/4
+f 512/908/4 544/973/4 543/971/4 511/905/4
+f 513/909/32 545/975/32 544/973/32 512/908/32
+f 514/912/4 546/978/4 545/976/4 513/910/4
+f 515/914/4 547/980/4 546/978/4 514/912/4
+f 516/917/4 548/983/4 547/980/4 515/914/4
+f 517/918/4 549/985/4 548/983/4 516/917/4
+f 518/920/4 550/987/4 549/985/4 517/918/4
+f 519/922/4 551/988/4 550/987/4 518/920/4
+f 520/925/4 552/990/4 551/988/4 519/922/4
+f 490/862/4 522/928/4 552/990/4 520/925/4
+f 435/783/33 499/882/33 498/880/33 434/781/33
+f 464/833/34 528/942/34 529/944/34 465/834/34
+f 479/848/35 543/972/35 544/974/35 480/849/35
+f 450/813/47 514/913/47 513/911/47 449/811/47
+f 439/791/7 503/890/7 502/888/7 438/789/7
+f 458/827/37 522/929/37 521/927/37 457/826/37
+f 428/769/31 492/867/31 491/865/31 427/767/31
+f 472/841/10 536/958/10 537/960/10 473/842/10
+f 487/856/38 551/989/38 552/991/38 488/857/38
+f 454/821/23 518/921/23 517/919/23 453/819/23
+f 443/799/39 507/898/39 506/896/39 442/797/39
+f 465/834/36 529/944/36 530/946/36 466/835/36
+f 480/849/40 544/974/40 545/977/40 481/850/40
+f 432/777/35 496/876/35 495/873/35 431/775/35
+f 457/826/39 521/927/39 523/931/39 459/828/39
+f 447/807/20 511/906/20 510/904/20 446/805/20
+f 473/842/41 537/960/41 538/962/41 474/843/41
+f 488/858/26 552/992/26 522/929/26 458/827/26
+f 436/785/28 500/884/28 499/882/28 435/783/28
+f 466/835/42 530/946/42 531/948/42 467/836/42
+f 481/850/43 545/977/43 546/979/43 482/851/43
+f 451/815/42 515/915/42 514/913/42 450/813/42
+f 440/793/38 504/892/38 503/890/38 439/791/38
+f 459/828/44 523/931/44 524/933/44 460/829/44
+f 429/771/45 493/869/45 492/867/45 428/769/45
+f 474/843/17 538/962/17 539/964/17 475/844/17
+f 455/823/16 519/923/16 518/921/16 454/821/16
+f 426/765/41 489/860/41 490/863/41 425/763/41
+f 444/801/44 508/900/44 507/898/44 443/799/44
+f 467/836/27 531/948/27 532/950/27 468/837/27
+f 482/851/33 546/979/33 547/981/33 483/852/33
+f 433/779/46 497/878/46 496/876/46 432/777/46
diff --git a/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_hp0.png b/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_hp0.png
new file mode 100644
index 0000000..6e8ca57
--- /dev/null
+++ b/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_hp0.png
Binary files differ
diff --git a/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_hp1.png b/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_hp1.png
new file mode 100644
index 0000000..d9fa346
--- /dev/null
+++ b/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_hp1.png
Binary files differ
diff --git a/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_hp2.png b/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_hp2.png
new file mode 100644
index 0000000..1fd1919
--- /dev/null
+++ b/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_hp2.png
Binary files differ
diff --git a/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_hp3.png b/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_hp3.png
new file mode 100644
index 0000000..8588b12
--- /dev/null
+++ b/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_hp3.png
Binary files differ
diff --git a/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_template.png b/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_template.png
new file mode 100755
index 0000000..0bab765
--- /dev/null
+++ b/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_template.png
Binary files differ
diff --git a/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_vr0.png b/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_vr0.png
new file mode 100644
index 0000000..d327bc3
--- /dev/null
+++ b/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_vr0.png
Binary files differ
diff --git a/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_vr1.png b/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_vr1.png
new file mode 100644
index 0000000..494e890
--- /dev/null
+++ b/advtrains_signals_muc_ubahn/textures/advtrains_signals_muc_ubahn_vr1.png
Binary files differ
diff --git a/advtrains_trackmap/fsrender.lua b/advtrains_trackmap/fsrender.lua
new file mode 100644
index 0000000..340b08c
--- /dev/null
+++ b/advtrains_trackmap/fsrender.lua
@@ -0,0 +1,41 @@
+-- fsrender.lua
+-- Rendering of a grid of characters into a formspec
+
+local tm = advtrains.trackmap
+
+function tm.render_grid_formspec(formsize_x, formsize_y, gridtbl, origin_pos, width, height)
+ local grid = gridtbl.grid
+ local gcolor = gridtbl.gcolor
+ local s = {
+ "formspec_version[3]",
+ "size["..formsize_x..","..formsize_y.."]",
+ "no_prepend[]",
+ "bgcolor[white;false;white]",
+ "style_type[label;font=mono]",
+ "label[0,0;",
+ minetest.get_color_escape_sequence("black")
+ }
+ local last_color = nil
+ for z=height-1, 0, -1 do
+ -- render a row
+ for x=0,width-1 do
+ local apos_x = origin_pos.x + x
+ local apos_z = origin_pos.z + z
+ local chr = " "
+ if grid[apos_x] and grid[apos_x][apos_z] then
+ local color = gcolor[apos_x][apos_z]
+ if color ~= last_color then
+ -- change the color of the text
+ table.insert(s, minetest.get_color_escape_sequence(color or "black"))
+ last_color = color
+ end
+ chr = grid[apos_x][apos_z]
+ end
+ table.insert(s, chr)
+ end
+ table.insert(s,"\n")
+ end
+ table.insert(s, "]")
+ table.insert(s, "style_type[label;font=]") -- reset font style
+ return table.concat(s)
+end \ No newline at end of file
diff --git a/advtrains_trackmap/grid.lua b/advtrains_trackmap/grid.lua
new file mode 100644
index 0000000..4c98b2f
--- /dev/null
+++ b/advtrains_trackmap/grid.lua
@@ -0,0 +1,107 @@
+-- grid.lua
+-- routines taking the tracks and building a grid out of them
+
+local tm = advtrains.trackmap
+
+-- Maximum scan length for track iterator
+local TS_MAX_SCAN = 1000
+
+-- Get a character matching the class requested by chr, in the correct orientation for conndir
+-- The characters will be chosen from the unicode frame set if appropriate
+-- valid character classes -, =, |
+local char_equiv = {
+ ["-"] = {"│", "/", "─", "\\\\"}, -- single-line
+ ["="] = {"║", "/", "═", "\\\\"}, -- double-line
+ ["|"] = {"─", "\\\\", "│", "/"}, -- break (i.e. TCB, perpendicular to orientation)
+}
+local dir_to_charmap = {
+ [15] = 1,
+ [ 0] = 1,
+ [ 1] = 1,
+ [ 2] = 2,
+ [ 3] = 3,
+ [ 4] = 3,
+ [ 5] = 3,
+ [ 6] = 4,
+ [ 7] = 1,
+ [ 8] = 1,
+ [ 9] = 1,
+ [10] = 2,
+ [11] = 3,
+ [12] = 3,
+ [13] = 3,
+ [14] = 4,
+}
+function tm.rotate_char_class_by_conn(chr, conndir)
+ --atdebug("rotatechar", chr, conndir, "dircharmap", dir_to_charmap[conndir], "charequiv", char_equiv[chr])
+ return char_equiv[chr][dir_to_charmap[conndir]]
+end
+
+-- Generate a grid map by walking tracks starting from the given position start_pos
+-- For every track that is visited, node_callback is called.
+-- signature: node_callback(pos, conns, connid)
+-- should return a table as follows:
+-- {
+-- char = "X" -- the character to use in this place, defaults to guessing a fitting frame character
+-- color = "colorstring" or nil -- the color to render the character with
+-- stop = false -- if true, the iterator will stop following this branch
+-- }
+-- Returning nil will assume defaults.
+-- return value: a table:
+-- {
+-- grid = <grid> - access abs_grid[x][y] - contains one character per cell
+-- min_pos = <pos> - the minimum pos contained in this grid
+-- max_pos = <pos> - the maximum pos
+-- }
+function tm.generate_grid_map(start_pos, node_callback)
+ local ti = advtrains.get_track_iterator(start_pos, nil, TS_MAX_SCAN, true)
+ local grid = {} -- grid[x][y]
+ local gcolor = {} -- grid[x][y]
+ local gylev = {} -- grid[x][y]
+ local gminx, gmaxx, gminz, gmaxz = start_pos.x, start_pos.x, start_pos.z, start_pos.z
+
+ while ti:has_next_branch() do
+ pos, connid = ti:next_branch()
+ repeat
+ local ok, conns = advtrains.get_rail_info_at(pos)
+ local c = node_callback(pos, conns, connid)
+ local chr = c and c.char
+ --atdebug("init",pos.x,pos.z,chr,"")
+ if not chr then
+ chr = tm.rotate_char_class_by_conn("=", conns[connid].c)
+ end
+
+ -- add the char to the grid
+ if not grid[pos.x] then
+ grid[pos.x] = {}
+ gcolor[pos.x] = {}
+ gylev[pos.x] = {}
+ end
+
+ -- ensure that higher rails are rendered on top
+ local prev_ylev = gylev[pos.x][pos.z]
+ if not prev_ylev or prev_ylev < pos.y then
+ grid[pos.x][pos.z] = chr
+ gylev[pos.x][pos.z] = pos.y
+ if c and c.color then
+ gcolor[pos.x][pos.z] = c.color
+ end
+ end
+
+ gminx = math.min(gminx, pos.x)
+ gmaxx = math.max(gmaxx, pos.x)
+ gminz = math.min(gminz, pos.z)
+ gmaxz = math.max(gmaxz, pos.z)
+
+ if c and c.stop then break end
+ pos, connid = ti:next_track()
+ until not pos
+ end
+ return {
+ grid = grid,
+ gcolor = gcolor,
+ min_pos = {x=gminx, z=gminz},
+ max_pos = {x=gmaxx, z=gmaxz},
+ }
+end
+
diff --git a/advtrains_trackmap/init.lua b/advtrains_trackmap/init.lua
new file mode 100644
index 0000000..060e597
--- /dev/null
+++ b/advtrains_trackmap/init.lua
@@ -0,0 +1,14 @@
+-- Advtrains track map
+-- Replacement of itrainmap
+
+advtrains.trackmap = {}
+
+
+local modpath = minetest.get_modpath(minetest.get_current_modname()) .. DIR_DELIM
+
+
+dofile(modpath.."grid.lua")
+dofile(modpath.."fsrender.lua")
+dofile(modpath.."viewer.lua")
+
+
diff --git a/advtrains_trackmap/mod.conf b/advtrains_trackmap/mod.conf
new file mode 100644
index 0000000..633f1e2
--- /dev/null
+++ b/advtrains_trackmap/mod.conf
@@ -0,0 +1,6 @@
+name=advtrains_trackmap
+title=Advanced Trains Track Map
+description=Formspec-based map to display track layouts
+author=orwell96
+
+depends=advtrains
diff --git a/advtrains_trackmap/viewer.lua b/advtrains_trackmap/viewer.lua
new file mode 100644
index 0000000..b7205c8
--- /dev/null
+++ b/advtrains_trackmap/viewer.lua
@@ -0,0 +1,43 @@
+-- viewer.lua
+-- standalone chatcommand/tool trackmap viewer window
+
+local tm = advtrains.trackmap
+
+local function node_left_click(pos, pname)
+ local node_ok, conns, rail_y=advtrains.get_rail_info_at(pos)
+ if not node_ok then
+ minetest.chat_send_player(pname, "Node is not a track!")
+ return
+ end
+
+ local function node_callback(npos, conns, connid)
+ if vector.equals(pos, npos) then
+ return {color = "red"}
+ end
+ return nil
+ end
+
+ local gridtbl = tm.generate_grid_map(pos, node_callback)
+ local fslabel = tm.render_grid_formspec(20, 20, gridtbl, {x=pos.x-35, z=pos.z-22}, 70, 44)
+
+ minetest.show_formspec(pname, "advtrains_trackmap:test", fslabel)
+end
+
+
+minetest.register_craftitem("advtrains_trackmap:tool",{
+ description = "Trackmap Tool\nPunch: Show map",
+ groups = {cracky=1}, -- key=name, value=rating; rating=1..3.
+ inventory_image = "at_il_tool.png",
+ wield_image = "at_il_tool.png",
+ stack_max = 1,
+ on_use = function(itemstack, player, pointed_thing)
+ local pname = player:get_player_name()
+ if not pname then
+ return
+ end
+ if pointed_thing.type=="node" then
+ local pos=pointed_thing.under
+ node_left_click(pos, pname)
+ end
+ end
+}) \ No newline at end of file
diff --git a/advtrains_train_track/init.lua b/advtrains_train_track/init.lua
index 5403ff7..c773839 100644
--- a/advtrains_train_track/init.lua
+++ b/advtrains_train_track/init.lua
@@ -2,6 +2,181 @@ ch_base.open_mod(minetest.get_current_modname())
-- Default tracks for advtrains
-- (c) orwell96 and contributors
+local default_boxen = {
+ ["st"] = {
+ [""] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {-1/2-1/16, -1/2, -1/2, 1/2+1/16, -1/2+2/16, 1/2},
+ }
+ },
+ ["_30"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.5000, -0.5000, -1.000, 0.5000, -0.3750, 1.000},
+ {-0.8750, -0.5000, -1.000, -0.5000, -0.3750, 0.2500},
+ {0.5000, -0.5000, -0.2500, 0.8750, -0.3750, 1.000},
+ {-0.1250, -0.5000, -1.375, 0.1875, -0.3750, -1.000}
+ }
+ }
+ },
+ ["_45"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.5000, -0.5000, -0.8750, 0.5000, -0.3750, 0.8750},
+ {0.5000, -0.5000, -0.5000, 0.8750, -0.3750, 0.5000},
+ {-0.8750, -0.5000, -0.5000, -0.5000, -0.3750, 0.5000}
+ }
+ }
+ },
+ ["_60"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-1.000, -0.5000, -0.5000, 1.000, -0.3750, 0.5000},
+ {-1.000, -0.5000, -0.8750, 0.2500, -0.3750, -0.5000},
+ {-0.2500, -0.5000, 0.5000, 1.000, -0.3750, 0.8750},
+ {-1.375, -0.5000, -0.1250, -1.000, -0.3750, 0.1875}
+ }
+ }
+ },
+ },
+
+ ["cr"] = {
+ [""] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.5000, -0.5000, -0.5000, 0.6875, -0.3750, 0.5000},
+ {-0.3750, -0.5000, -1.000, 1.000, -0.3750, 0.000}
+ }
+ }
+ },
+ ["_30"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.5000, -0.5000, -0.5000, 0.7500, -0.3750, 0.8750},
+ {-0.3750, -0.5000, 0.8750, 0.2500, -0.3750, 1.188},
+ {0.7500, -0.5000, 0.2500, 1.063, -0.3750, 0.8750}
+ }
+ }
+ },
+ ["_45"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.5000, -0.5000, -1.125, 0.5000, -0.3750, 0.6875},
+ {-0.8750, -0.5000, -0.9375, -0.5000, -0.3750, 0.06250},
+ {0.5000, -0.5000, -0.5000, 0.8750, -0.3750, 0.5000}
+ }
+ }
+ },
+ ["_60"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.8125, -0.5000, -0.5000, 1.188, -0.3750, 0.5000},
+ {-0.1875, -0.5000, 0.5000, 0.8750, -0.3125, 0.8750},
+ {-0.2500, -0.5000, -0.9375, 0.3125, -0.3125, -0.5000}
+ }
+ }
+ },
+ },
+
+ ["swlst"] = {
+ [""] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.5000, -0.5000, -0.5000, 0.6250, -0.3750, 0.5000},
+ {-0.3125, -0.5000, -1.000, 0.9375, -0.3125, -0.06250}
+ }
+ }
+ },
+ ["_30"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.5000, -0.5000, -1.000, 0.5000, -0.3750, 1.000},
+ {-0.8750, -0.5000, -1.000, -0.5000, -0.3750, 0.2500},
+ {0.5000, -0.5000, -0.2500, 0.8750, -0.3750, 1.000},
+ {-0.1250, -0.5000, -1.375, 0.1875, -0.3750, -1.000}
+ }
+ }
+ },
+ ["_45"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.5000, -0.5000, -1.1875, 0.5000, -0.3750, 0.8750},
+ {0.5000, -0.5000, -0.5000, 0.8750, -0.3750, 0.5000},
+ {-0.8750, -0.5000, -0.8125, -0.5000, -0.3750, 0.5000}
+ }
+ }
+ },
+ ["_60"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-1.000, -0.5000, -0.5000, 1.000, -0.3750, 0.5000},
+ {-1.000, -0.5000, -0.8750, 0.2500, -0.3750, -0.5000},
+ {-0.2500, -0.5000, 0.5000, 1.000, -0.3750, 0.8750},
+ {-1.375, -0.5000, -0.1250, -1.000, -0.3750, 0.1875}
+ }
+ }
+ },
+ },
+
+ ["swrst"] = {
+ [""] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.5000, -0.5000, -0.5000, 0.6250, -0.3750, 0.5000},
+ {-0.8125, -0.5000, -1.000, 0.4375, -0.3125, -0.06250}
+ }
+ }
+ },
+ ["_30"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.5000, -0.5000, -1.000, 0.5000, -0.3750, 1.000},
+ {-0.8750, -0.5000, -1.000, -0.5000, -0.3750, 0.2500},
+ {0.5000, -0.5000, -0.2500, 0.8750, -0.3750, 1.000},
+ {-0.1250, -0.5000, -1.375, 0.1875, -0.3750, -1.000}
+ }
+ }
+ },
+ ["_45"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-1.1875, -0.5000, -0.5000, 0.8750, -0.3750, 0.5000},
+ {-0.5000, -0.5000, 0.5000, 0.5000, -0.3750, 0.8750},
+ {-0.8125, -0.5000, -0.8750, 0.5000, -0.3750, -0.5000}
+ }
+ }
+ },
+ ["_60"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-1.000, -0.5000, -0.5000, 1.000, -0.3750, 0.5000},
+ {-1.000, -0.5000, -0.8750, 0.2500, -0.3750, -0.5000},
+ {-0.2500, -0.5000, 0.5000, 1.000, -0.3750, 0.8750},
+ {-1.375, -0.5000, -0.1250, -1.000, -0.3750, 0.1875}
+ }
+ }
+ },
+ },
+}
+
+default_boxen["swlcr"] = default_boxen["swlst"]
+default_boxen["swrcr"] = default_boxen["swrst"]
--flat
advtrains.register_tracks("default", {
@@ -12,6 +187,14 @@ advtrains.register_tracks("default", {
shared_texture="advtrains_dtrack_shared.png",
description=attrans("Track"),
formats={},
+
+ get_additional_definiton = function(def, preset, suffix, rotation)
+ if default_boxen[suffix] ~= nil and default_boxen[suffix][rotation] ~= nil then
+ return default_boxen[suffix][rotation]
+ else
+ return {}
+ end
+ end,
}, advtrains.ap.t_30deg_flat)
minetest.register_craft({
@@ -23,6 +206,59 @@ minetest.register_craft({
},
})
+local y3_boxen = {
+ [""] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.8750, -0.5000, -1.125, 0.8750, -0.3750, 0.4375}
+ }
+ }
+ },
+
+ ["_30"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.5000, -0.5000, -0.875, 0.5000, -0.3750, 1.000},
+ {-0.8750, -0.5000, -0.4375, -0.5000, -0.3750, 0.5625},
+ {0.5000, -0.5000, -0.2500, 0.8125, -0.3750, 1.000},
+ }
+ }
+ },
+
+ --UX FIXME: - 3way - have to place straight route before l and r or the
+ --nodebox overlaps too much and can't place the straight track node.
+ ["_45"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.5000, -0.5000, -1.1250, 0.5000, -0.3750, 0.8750},
+ {0.5000, -0.5000, -0.5000, 0.8750, -0.3750, 0.5000},
+ {-1.1250, -0.5000, -0.9375, -0.5000, -0.3750, 0.5000}
+ }
+ }
+ },
+
+ ["_60"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ --{-0.5000, -0.5000, -0.875, 0.5000, -0.3750, 1.000},
+ {-0.875, -0.5000, -0.5, 1.0, -0.3750, 0.5},
+ --{-0.8750, -0.5000, -0.4375, -0.5000, -0.3750, 0.5625},
+ {-0.4375, -0.5000, -0.8750, 0.5625, -0.3750, -0.5000},
+ --{0.5000, -0.5000, -0.2500, 0.8125, -0.3750, 1.000},
+ {-0.2500, -0.5000, -0.2500, 1.0000, -0.3750, 0.8125},
+ }
+ }
+ },
+}
+
+
+local function y3_turnouts_addef(def, preset, suffix, rotation)
+ return y3_boxen[rotation] or {}
+end
-- y-turnout
advtrains.register_tracks("default", {
nodename_prefix="advtrains:dtrack_sy",
@@ -32,6 +268,7 @@ advtrains.register_tracks("default", {
shared_texture="advtrains_dtrack_shared.png",
description=attrans("Y-turnout"),
formats = {},
+ get_additional_definiton = y3_turnouts_addef,
}, advtrains.ap.t_yturnout)
minetest.register_craft({
output = 'advtrains:dtrack_sy_placer 2',
@@ -50,6 +287,7 @@ advtrains.register_tracks("default", {
shared_texture="advtrains_dtrack_shared.png",
description=attrans("3-way turnout"),
formats = {},
+ get_additional_definiton = y3_turnouts_addef,
}, advtrains.ap.t_s3way)
minetest.register_craft({
output = 'advtrains:dtrack_s3_placer 1',
@@ -61,6 +299,35 @@ minetest.register_craft({
})
-- Diamond Crossings
+
+local perp_boxen = {
+ [""] = {}, --default size
+ ["_30"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-1.000, -0.5000, -1.000, 1.000, -0.3750, 1.000}
+ }
+ }
+ },
+ ["_45"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.8125, -0.5000, -0.8125, 0.8125, -0.3750, 0.8125}
+ }
+ }
+ },
+ ["_60"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-1.000, -0.5000, -1.000, 1.000, -0.3750, 1.000}
+ }
+ }
+ },
+}
+
-- perpendicular
advtrains.register_tracks("default", {
nodename_prefix="advtrains:dtrack_xing",
@@ -69,7 +336,10 @@ advtrains.register_tracks("default", {
models_suffix=".obj",
shared_texture="advtrains_dtrack_shared.png",
description=attrans("Perpendicular Diamond Crossing Track"),
- formats = {}
+ formats = {},
+ get_additional_definiton = function(def, preset, suffix, rotation)
+ return perp_boxen[rotation] or {}
+ end
}, advtrains.ap.t_perpcrossing)
minetest.register_craft({
@@ -81,6 +351,73 @@ minetest.register_craft({
}
})
+local ninety_plus_boxen = {
+ ["30l"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.5000, -0.5000, -1.000, 0.5000, -0.3750, 1.000},
+ {-0.8750, -0.5000, -1.000, -0.5000, -0.3750, 0.2500},
+ {0.5000, -0.5000, -0.2500, 0.8750, -0.3750, 1.000},
+ {-0.1250, -0.5000, -1.375, 0.1875, -0.3750, -1.000}
+ }
+ }
+ },
+ ["30r"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {0.5000, -0.5000, -1.000, -0.5000, -0.3750, 1.000},
+ {0.8750, -0.5000, -1.000, 0.5000, -0.3750, 0.2500},
+ {-0.5000, -0.5000, -0.2500, -0.8750, -0.3750, 1.000},
+ {0.1250, -0.5000, -1.375, -0.1875, -0.3750, -1.000}
+ }
+ }
+ },
+ ["45l"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.5000, -0.5000, -0.8750, 0.5000, -0.3750, 0.8750},
+ {0.5000, -0.5000, -0.5000, 0.8750, -0.3750, 0.5000},
+ {-0.8750, -0.5000, -0.5000, -0.5000, -0.3750, 0.5000}
+ }
+ }
+ },
+ ["45r"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.5000, -0.5000, -0.8750, 0.5000, -0.3750, 0.8750},
+ {0.5000, -0.5000, -0.5000, 0.8750, -0.3750, 0.5000},
+ {-0.8750, -0.5000, -0.5000, -0.5000, -0.3750, 0.5000}
+ }
+ }
+ },
+ ["60l"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-1.000, -0.5000, -0.5000, 1.000, -0.3750, 0.5000},
+ {-1.000, -0.5000, -0.8750, 0.2500, -0.3750, -0.5000},
+ {-0.2500, -0.5000, 0.5000, 1.000, -0.3750, 0.8750},
+ {-1.375, -0.5000, -0.1250, -1.000, -0.3750, 0.1875}
+ }
+ }
+ },
+ ["60r"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {1.000, -0.5000, -0.5000, -1.000, -0.3750, 0.5000},
+ {1.000, -0.5000, -0.8750, -0.2500, -0.3750, -0.5000},
+ {0.2500, -0.5000, 0.5000, -1.000, -0.3750, 0.8750},
+ {1.375, -0.5000, -0.1250, 1.000, -0.3750, 0.1875}
+ }
+ }
+ },
+}
+
-- 90plusx
-- When you face east and param2=0, then this set of rails has a rail at 90
-- degrees to the viewer, plus another rail crossing at 30, 45 or 60 degrees.
@@ -91,7 +428,10 @@ advtrains.register_tracks("default", {
models_suffix=".obj",
shared_texture="advtrains_dtrack_shared.png",
description=attrans("90+Angle Diamond Crossing Track"),
- formats = {}
+ formats = {},
+ get_additional_definiton = function(def, preset, suffix, rotation)
+ return ninety_plus_boxen[suffix] or {}
+ end,
}, advtrains.ap.t_90plusx_crossing)
minetest.register_craft({
output = 'advtrains:dtrack_xing90plusx_placer 2',
@@ -101,6 +441,7 @@ minetest.register_craft({
{'', '', 'advtrains:dtrack_placer'}
}
})
+
-- Deprecate any rails using the old name scheme
minetest.register_lbm({
label = "Upgrade legacy 4590 rails",
@@ -121,6 +462,83 @@ minetest.register_lbm({
-- This will replace any items left in the inventory
minetest.register_alias("advtrains:dtrack_xing4590_placer", "advtrains:dtrack_xing90plusx_placer")
+local diagonal_boxen = {
+ ["30r45l"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {0.5000, -0.5000, -1.000, -0.5000, -0.3750, 1.000},
+ {0.8750, -0.5000, -1.000, 0.5000, -0.3750, 0.2500},
+ {-0.5000, -0.5000, -0.2500, -0.8750, -0.3750, 1.000},
+ {0.1250, -0.5000, -1.375, -0.1875, -0.3750, -1.000}
+ }
+ }
+ },
+ ["60l30l"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-1.000, -0.5000, -0.5000, 1.000, -0.3750, 0.5000},
+ {-1.000, -0.5000, -0.8750, 0.2500, -0.3750, -0.5000},
+ {-0.2500, -0.5000, 0.5000, 1.000, -0.3750, 0.8750},
+ {-1.375, -0.5000, -0.1250, -1.000, -0.3750, 0.1875}
+ }
+ }
+ },
+ ["60l60r"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-1.000, -0.5000, -1.000, 1.000, -0.3750, 1.000}
+ }
+ }
+ },
+ ["60r30r"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {1.000, -0.5000, -0.5000, -1.000, -0.3750, 0.5000},
+ {1.000, -0.5000, -0.8750, -0.2500, -0.3750, -0.5000},
+ {0.2500, -0.5000, 0.5000, -1.000, -0.3750, 0.8750},
+ {1.375, -0.5000, -0.1250, 1.000, -0.3750, 0.1875}
+ }
+ }
+ },
+ ["30l45r"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-0.5000, -0.5000, -1.000, 0.5000, -0.3750, 1.000},
+ {-0.8750, -0.5000, -1.000, -0.5000, -0.3750, 0.2500},
+ {0.5000, -0.5000, -0.2500, 0.8750, -0.3750, 1.000},
+ {-0.1250, -0.5000, -1.375, 0.1875, -0.3750, -1.000}
+ }
+ }
+ },
+ ["60l45r"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {-1.000, -0.5000, -0.5000, 1.000, -0.3750, 0.5000},
+ {-1.000, -0.5000, -0.8750, 0.2500, -0.3750, -0.5000},
+ {-0.2500, -0.5000, 0.5000, 1.000, -0.3750, 0.8750},
+ {-1.375, -0.5000, -0.1250, -1.000, -0.3750, 0.1875}
+ }
+ }
+ },
+ ["60r45l"] = {
+ selection_box = {
+ type = "fixed",
+ fixed = {
+ {1.000, -0.5000, -0.5000, -1.000, -0.3750, 0.5000},
+ {1.000, -0.5000, -0.8750, -0.2500, -0.3750, -0.5000},
+ {0.2500, -0.5000, 0.5000, -1.000, -0.3750, 0.8750},
+ {1.375, -0.5000, -0.1250, 1.000, -0.3750, 0.1875}
+ }
+ }
+ },
+}
+
-- Diagonal
-- This set of rail crossings is named based on the angle of each intersecting
-- direction when facing east and param2=0. Rails with l/r swapped are mirror
@@ -133,6 +551,9 @@ advtrains.register_tracks("default", {
shared_texture="advtrains_dtrack_shared.png",
description=attrans("Diagonal Diamond Crossing Track"),
formats = {},
+ get_additional_definiton = function(def, preset, suffix, rotation)
+ return diagonal_boxen[suffix] or {}
+ end,
}, advtrains.ap.t_diagonalcrossing)
minetest.register_craft({
output = 'advtrains:dtrack_xingdiag_placer 2',
@@ -212,6 +633,35 @@ advtrains.register_tracks("default", {
--bumpers still use the old texture until the models are redone.
description=attrans("Bumper"),
formats={},
+ get_additional_definiton = function(def, preset, suffix, rotation)
+ -- 2024-11-25: Bumpers get the additional feature of being both a signal and a self-contained TCB, when interlocking is used.
+ if advtrains.interlocking then
+ return {
+ -- use the special callbacks for self_tcb (see tcb_ts_ui.lua)
+ can_dig = advtrains.interlocking.self_tcb_make_can_dig_callback(true),
+ after_dig_node = advtrains.interlocking.self_tcb_make_after_dig_callback(true),
+ after_place_node = advtrains.interlocking.self_tcb_make_after_place_callback(true, true),
+ on_rightclick = advtrains.interlocking.self_tcb_make_on_rightclick_callback(false, true),
+ advtrains = {
+ main_aspects = {
+ -- No main aspects, it always shows Stop.
+ -- But we need to define the table so that signal caplevel is 3
+ },
+ apply_aspect = function(pos, node, main_aspect, rem_aspect, rem_aspinfo)
+ -- is a no-op for bumpers, it always shows Stop
+ end,
+ get_aspect_info = function(pos, main_aspect)
+ -- it always shows Stop
+ return advtrains.interlocking.signal.ASPI_HALT
+ end,
+ distant_support = false, -- not a distant
+ route_role = "end", -- the end is nigh!
+ }
+ }
+ else
+ return {} -- no additional defs when interlocking is not used
+ end
+ end,
}, advtrains.ap.t_30deg_straightonly)
minetest.register_craft({
output = 'advtrains:dtrack_bumper_placer 2',
@@ -297,8 +747,45 @@ minetest.register_on_player_receive_fields(function(player, formname, fields)
end
end)
+local function load_wagon(wagon_id, node_inv, node_fc, unload)
+ local inv_modified = false
+ local w_inv=minetest.get_inventory({type="detached", name="advtrains_wgn_"..wagon_id})
+ if w_inv and w_inv:get_list("box") then
+
+ local wagon_data = advtrains.wagons[wagon_id]
+ local wagon_fc
+ if wagon_data.fc then
+ if not wagon_data.fcind then wagon_data.fcind = 1 end
+ wagon_fc = tostring(wagon_data.fc[wagon_data.fcind]) or ""
+ end
+
+ if node_fc == "" or wagon_fc == node_fc then
+ if not unload then
+ for _, item in ipairs(node_inv:get_list("main")) do
+ if w_inv:get_list("box") and w_inv:room_for_item("box", item) then
+ w_inv:add_item("box", item)
+ node_inv:remove_item("main", item)
+ if item.name ~= "" then inv_modified = true end
+ end
+ end
+ else
+ for _, item in ipairs(w_inv:get_list("box")) do
+ if node_inv:get_list("main") and node_inv:room_for_item("main", item) then
+ w_inv:remove_item("box", item)
+ node_inv:add_item("main", item)
+ if item.name ~= "" then inv_modified = true end
+ end
+ end
+ end
+ end
+ end
+ return inv_modified
+end
-local function train_load(pos, train_id, unload)
+local function load_entire_train(pos, train_id, unload) -- flood load when not in an active area
+ if advtrains.is_node_loaded(pos) then -- leave the loading to the nodetimer if area is loaded
+ return
+ end
local train=advtrains.trains[train_id]
local below = get_far_node({x=pos.x, y=pos.y-1, z=pos.z})
if not string.match(below.name, "chest") then
@@ -311,43 +798,60 @@ local function train_load(pos, train_id, unload)
--track section is disabled
return
end
-
- local inv = minetest.get_inventory({type="node", pos={x=pos.x, y=pos.y-1, z=pos.z}})
- if inv and train.velocity < 2 then
- for k, v in ipairs(train.trainparts) do
- local i=minetest.get_inventory({type="detached", name="advtrains_wgn_"..v})
- if i and i:get_list("box") then
-
- local wagon_data = advtrains.wagons[v]
- local wagon_fc
- if wagon_data.fc then
- if not wagon_data.fcind then wagon_data.fcind = 1 end
- wagon_fc = tostring(wagon_data.fc[wagon_data.fcind]) or ""
- end
-
- if node_fc == "" or wagon_fc == node_fc then
- if not unload then
- for _, item in ipairs(inv:get_list("main")) do
- if i:get_list("box") and i:room_for_item("box", item) then
- i:add_item("box", item)
- inv:remove_item("main", item)
- end
- end
- else
- for _, item in ipairs(i:get_list("box")) do
- if inv:get_list("main") and inv:room_for_item("main", item) then
- i:remove_item("box", item)
- inv:add_item("main", item)
- end
+ local node_inv = minetest.get_inventory({type="node", pos={x=pos.x, y=pos.y-1, z=pos.z}})
+ if node_inv and train.velocity <= 2 then
+ for _, wagon_id in ipairs(train.trainparts) do
+ load_wagon(wagon_id, node_inv, node_fc, unload)
+ end
+ end
+end
+
+local function load_wagon_on_timer(pos, unload) -- loading ramp when in an active area
+ if not advtrains.is_node_loaded(pos) then -- leave the loading for the flood load function. we're out of area
+ return true -- reset the nodetimer until the node is loaded again
+ end
+ local tid, tidx = advtrains.get_train_at_pos(pos)
+ if not tid or tid == "" then
+ return true
+ end -- no train to load.
+
+ local train = advtrains.trains[tid]
+ local below = get_far_node({x=pos.x, y=pos.y-1, z=pos.z})
+ if not string.match(below.name, "chest") then
+ atprint("this is not a chest! at "..minetest.pos_to_string(pos))
+ return true
+ end
+ local node_fc = minetest.get_meta(pos):get_string("fc") or ""
+ if node_fc == "#" then
+ --track section is disabled
+ return true
+ end
+ local node_inv = minetest.get_inventory({type="node", pos={x=pos.x, y=pos.y-1, z=pos.z}})
+ if node_inv and train.velocity <= 2 then
+ local _, wagon_id, wagon_data = advtrains.get_wagon_at_index(tid, tidx)
+ if wagon_id then
+ local inv_modified = load_wagon(wagon_id, node_inv, node_fc, unload)
+ if inv_modified then
+ if advtrains.wagon_prototypes[advtrains.get_wagon_prototype(wagon_data)].set_textures then
+ local wagon_object = advtrains.wagon_objects[wagon_id]
+ if wagon_object and wagon_data then
+ local ent = wagon_object:get_luaentity()
+ if ent and ent.set_textures then
+ ent:set_textures(wagon_data)
end
end
end
end
end
end
+ return true
end
-
+local nodetimer_interval = minetest.settings:get("advtrains_loading_track_timer") or 1
+local function start_nodetimer(pos)
+ local timer = minetest.get_node_timer(pos)
+ timer:start(nodetimer_interval)
+end
advtrains.register_tracks("default", {
nodename_prefix="advtrains:dtrack_unload",
@@ -366,9 +870,16 @@ advtrains.register_tracks("default", {
on_rightclick = function(pos, node, player)
show_fc_formspec(pos, player)
end,
+ after_place_node = function(pos)
+ advtrains.ndb.update(pos)
+ start_nodetimer(pos)
+ end,
+ on_timer = function(pos)
+ return load_wagon_on_timer(pos, true)
+ end,
advtrains = {
on_train_enter = function(pos, train_id)
- train_load(pos, train_id, true)
+ load_entire_train(pos, train_id, true)
end,
},
}
@@ -391,9 +902,16 @@ advtrains.register_tracks("default", {
on_rightclick = function(pos, node, player)
show_fc_formspec(pos, player)
end,
+ after_place_node = function(pos)
+ advtrains.ndb.update(pos)
+ start_nodetimer(pos)
+ end,
+ on_timer = function(pos)
+ return load_wagon_on_timer(pos, false)
+ end,
advtrains = {
on_train_enter = function(pos, train_id)
- train_load(pos, train_id, false)
+ load_entire_train(pos, train_id, false)
end,
},
}
@@ -407,7 +925,6 @@ if minetest.get_modpath("basic_materials") then
elseif minetest.get_modpath("technic") then
loader_core = "technic:control_logic_unit"
end
---print("Loader Core: "..loader_core)
minetest.register_craft({
type="shapeless",
diff --git a/advtrains_train_track/mod.conf b/advtrains_train_track/mod.conf
index 2aece3e..a7fef4d 100644
--- a/advtrains_train_track/mod.conf
+++ b/advtrains_train_track/mod.conf
@@ -4,4 +4,4 @@ description=Default track set for Advanced Trains
author=orwell96
depends=advtrains
-optional_depends=mesecons,digtron
+optional_depends=mesecons,digtron,advtrains_interlocking
diff --git a/advtrains_train_track/settingtypes.txt b/advtrains_train_track/settingtypes.txt
new file mode 100644
index 0000000..0af0081
--- /dev/null
+++ b/advtrains_train_track/settingtypes.txt
@@ -0,0 +1,4 @@
+# Set the nodetimer delay for the loading tracks.
+# A longer delay may cause wagons to be missed if the pass over too fast.
+# A shorter delay may cause lag as wagons are checked multiple times as they pass over.
+advtrains_loading_track_timer (Loading Track Timer) int 1 \ No newline at end of file
diff --git a/assets/mm_trains_bvbg.png b/assets/mm_trains_bvbg.png
new file mode 100644
index 0000000..e1ac0ed
--- /dev/null
+++ b/assets/mm_trains_bvbg.png
Binary files differ
diff --git a/assets/muc_ubahn_uvs.png b/assets/muc_ubahn_uvs.png
new file mode 100644
index 0000000..b4292af
--- /dev/null
+++ b/assets/muc_ubahn_uvs.png
Binary files differ
diff --git a/assets/signal_wall.blend1 b/assets/signal_wall.blend1
new file mode 100644
index 0000000..7915a52
--- /dev/null
+++ b/assets/signal_wall.blend1
Binary files differ
diff --git a/assets/signal_wall_ceiling_muc_ubahn.blend b/assets/signal_wall_ceiling_muc_ubahn.blend
new file mode 100644
index 0000000..b4d8e06
--- /dev/null
+++ b/assets/signal_wall_ceiling_muc_ubahn.blend
Binary files differ
diff --git a/assets/signal_wall_muc_ubahn.blend b/assets/signal_wall_muc_ubahn.blend
new file mode 100644
index 0000000..6f40560
--- /dev/null
+++ b/assets/signal_wall_muc_ubahn.blend
Binary files differ
diff --git a/make_release.sh b/make_release.sh
new file mode 100755
index 0000000..d6ae82a
--- /dev/null
+++ b/make_release.sh
@@ -0,0 +1,53 @@
+#!/bin/sh
+
+# get version argument
+VERS="$1"
+if [ ! "$VERS" ];then
+ echo "Argument version expected: example ./make_release.sh 2.5.0"
+ exit 1
+fi
+
+# cd to root dir
+cd "$(dirname "$0")"
+AROOT="$(pwd)"
+echo "Working directory is: $AROOT"
+# create temp directory
+TMP=$(mktemp -d -t advtrains-XXXX)
+echo "Temp directory is: $TMP"
+mkdir "$TMP/advtrains"
+TDIR="$TMP/advtrains/"
+
+# copy dirs
+cp -r "advtrains" "$TDIR"
+cp -r "advtrains_interlocking" "$TDIR"
+cp -r "advtrains_line_automation" "$TDIR"
+cp -r "advtrains_luaautomation" "$TDIR"
+cp -r "advtrains_signals_ks" "$TDIR"
+cp -r "advtrains_signals_japan" "$TDIR"
+cp -r "advtrains_signals_muc_ubahn" "$TDIR"
+cp -r "advtrains_train_track" "$TDIR"
+cp -r "serialize_lib" "$TDIR"
+
+# copy files
+cp "atc_command.txt" "$TDIR"
+cp "description.txt" "$TDIR"
+cp "license.txt" "$TDIR"
+cp "license_media.txt" "$TDIR"
+cp "modpack.conf" "$TDIR"
+cp "privilege_guide.txt" "$TDIR"
+cp "README.md" "$TDIR"
+cp "screenshot.png" "$TDIR"
+
+# compress to zip archive
+ZIPNAME="$AROOT/advtrains_$VERS.zip"
+echo "Target ZIP file is: $ZIPNAME"
+cd "$TMP"
+zip -r "$ZIPNAME" "advtrains"
+cd "$AROOT"
+
+# success
+echo "Release $VERS created at: $ZIPNAME"
+
+# remove tempdir
+rm -rf "$TMP"
+
diff --git a/modpack.conf b/modpack.conf
index e01f462..5185be4 100644
--- a/modpack.conf
+++ b/modpack.conf
@@ -2,3 +2,4 @@ name=advtrains
title=Advanced Trains
description=Realistic trains and various equipment for railways, with a focus on automated train operation. No trains included, please install those separately.
author=orwell96
+min_minetest_version=5.4 \ No newline at end of file