001package jmri.jmrit.vsdecoder; 002 003import java.beans.PropertyChangeEvent; 004import java.beans.PropertyChangeListener; 005import java.io.File; 006import java.util.ArrayList; 007import java.util.Collection; 008import java.util.HashMap; 009import java.util.List; 010import java.util.Set; 011import jmri.Audio; 012import jmri.Block; 013import jmri.IdTag; 014import jmri.LocoAddress; 015import jmri.Manager; 016import jmri.NamedBean; 017import jmri.Path; 018import jmri.PhysicalLocationReporter; 019import jmri.Reporter; 020import jmri.implementation.DefaultIdTag; 021import jmri.jmrit.display.layoutEditor.*; 022import jmri.jmrit.roster.Roster; 023import jmri.jmrit.roster.RosterEntry; 024import jmri.jmrit.operations.trains.Train; 025import jmri.jmrit.operations.trains.TrainManager; 026import jmri.jmrit.vsdecoder.listener.ListeningSpot; 027import jmri.jmrit.vsdecoder.listener.VSDListener; 028import jmri.jmrit.vsdecoder.swing.VSDManagerFrame; 029import jmri.util.FileUtil; 030import jmri.util.JmriJFrame; 031import jmri.util.MathUtil; 032import jmri.util.PhysicalLocation; 033import java.awt.event.ActionEvent; 034import java.awt.event.ActionListener; 035import java.awt.geom.Point2D; 036import java.awt.GraphicsEnvironment; 037import javax.swing.Timer; 038import org.jdom2.Element; 039 040/** 041 * VSDecoderFactory, builds VSDecoders as needed, handles loading from XML if needed. 042 * 043 * <hr> 044 * This file is part of JMRI. 045 * <p> 046 * JMRI is free software; you can redistribute it and/or modify it under 047 * the terms of version 2 of the GNU General Public License as published 048 * by the Free Software Foundation. See the "COPYING" file for a copy 049 * of this license. 050 * <p> 051 * JMRI is distributed in the hope that it will be useful, but WITHOUT 052 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 053 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License 054 * for more details. 055 * 056 * @author Mark Underwood Copyright (C) 2011 057 * @author Klaus Killinger Copyright (C) 2018-2024 058 */ 059public class VSDecoderManager implements PropertyChangeListener { 060 061 //private static final ResourceBundle rb = VSDecoderBundle.bundle(); 062 private static final String vsd_property_change_name = "VSDecoder Manager"; // NOI18N 063 064 // Array-pointer for blockParameter 065 private static final int RADIUS = 0; 066 private static final int SLOPE = 1; 067 private static final int ROTATE_XPOS_I = 2; 068 private static final int ROTATE_YPOS_I = 3; 069 private static final int LENGTH = 4; 070 071 // Array-pointer for locoInBlock 072 private static final int ADDRESS = 0; 073 private static final int BLOCK = 1; 074 private static final int DISTANCE_TO_GO = 2; 075 private static final int DIR_FN = 3; 076 private static final int DIRECTION = 4; 077 078 protected jmri.NamedBeanHandleManager nbhm = jmri.InstanceManager.getDefault(jmri.NamedBeanHandleManager.class); 079 080 private HashMap<String, VSDListener> listenerTable; // list of listeners 081 private HashMap<String, VSDecoder> decodertable; // list of active decoders by System ID 082 private HashMap<String, VSDecoder> decoderAddressMap; // List of active decoders by address 083 private HashMap<Integer, VSDecoder> decoderInBlock; // list of active decoders by LocoAddress.getNumber() 084 private HashMap<String, String> profiletable; // list of loaded profiles key = profile name, value = path 085 HashMap<VSDecoder, Block> currentBlock; // list of active blocks by decoders 086 public HashMap<Block, LayoutEditor> possibleStartBlocks; // list of possible start blocks and their LE panel 087 private HashMap<String, Timer> timertable; // list of active timers by decoder System ID 088 089 private int locoInBlock[][]; // Block status for locos 090 private float blockParameter[][][]; 091 private List<List<PhysicalLocation>> blockPositionlists; 092 private List<List<Integer>> reporterlists; 093 private List<Boolean> circlelist; 094 private PhysicalLocation newPosition; 095 private PhysicalLocation models_origin; 096 private ArrayList<Block> blockList; 097 098 // List of registered event listeners 099 protected javax.swing.event.EventListenerList listenerList = new javax.swing.event.EventListenerList(); 100 101 //private static VSDecoderManager instance = null; // sole instance of this class 102 private volatile static VSDecoderManagerThread thread = null; // thread for running the manager 103 104 private VSDecoderPreferences vsdecoderPrefs; // local pointer to the preferences object 105 106 private JmriJFrame managerFrame = null; 107 108 private int vsdecoderID = 0; 109 private int locorow = -1; // Will be increased before first use 110 111 private int check_time; // Time interval in ms for track following updates 112 private float layout_scale; 113 private float distance_rest = 0.0f; // Block distance to go 114 private float distance_rest_old = 0.0f; // Block distance to go, copy 115 private float distance_rest_new = 0.0f; // Block distance to go, copy 116 117 private float xPosi; 118 public static final int max_decoder = 8; 119 boolean geofile_ok = false; 120 int num_setups; 121 private int lf_version; 122 int alf_version; 123 124 // constructor - for kicking off by the VSDecoderManagerThread... 125 // WARNING: Should only be called from static instance() 126 public VSDecoderManager() { 127 // Setup the decoder table 128 listenerTable = new HashMap<>(); 129 decodertable = new HashMap<>(); 130 decoderAddressMap = new HashMap<>(); 131 timertable = new HashMap<>(); 132 decoderInBlock = new HashMap<>(); // Key = decoder number 133 profiletable = new HashMap<>(); // key = profile name, value = path 134 currentBlock = new HashMap<>(); // key = decoder, value = block 135 possibleStartBlocks = new HashMap<>(); 136 locoInBlock = new int[max_decoder][5]; // Loco address number, current block, distance in cm to go in block, dirfn, direction 137 // Setup lists 138 reporterlists = new ArrayList<>(); 139 blockPositionlists = new ArrayList<>(); 140 circlelist = new ArrayList<>(); 141 // Get preferences 142 String dirname = FileUtil.getUserFilesPath() + "vsdecoder" + File.separator; // NOI18N 143 FileUtil.createDirectory(dirname); 144 vsdecoderPrefs = new VSDecoderPreferences(dirname + VSDecoderPreferences.VSDPreferencesFileName); 145 // Listen to ReporterManager for Report List changes 146 setupReporterManagerListener(); 147 // Get a Listener 148 VSDListener t = new VSDListener(); 149 listenerTable.put(t.getSystemName(), t); 150 // Update JMRI "Default Audio Listener" 151 setListenerLocation(t.getSystemName(), vsdecoderPrefs.getListenerPosition()); 152 // Look for additional layout geometry data 153 VSDGeoFile gf = new VSDGeoFile(); 154 if (gf.geofile_ok) { 155 geofile_ok = true; 156 alf_version = gf.alf_version; 157 num_setups = gf.getNumberOfSetups(); 158 reporterlists = gf.getReporterList(); 159 blockParameter = gf.getBlockParameter(); 160 blockPositionlists = gf.getBlockPosition(); 161 circlelist = gf.getCirclingList(); 162 check_time = gf.check_time; 163 layout_scale = gf.layout_scale; 164 models_origin = gf.models_origin; 165 possibleStartBlocks = gf.possibleStartBlocks; 166 blockList = gf.blockList; 167 } else { 168 geofile_ok = false; 169 if (gf.lf_version > 0) { 170 lf_version = gf.lf_version; 171 log.debug("assume location following"); 172 } 173 } 174 } 175 176 /** 177 * Provide the VSdecoderManager instance. 178 * @return the manager 179 */ 180 public static VSDecoderManager instance() { 181 if (thread == null) { 182 thread = VSDecoderManagerThread.instance(true); 183 } 184 return VSDecoderManagerThread.manager(); 185 } 186 187 /** 188 * Get a reference to the VSD Preferences. 189 * @return the preferences reference 190 */ 191 public VSDecoderPreferences getVSDecoderPreferences() { 192 return vsdecoderPrefs; 193 } 194 195 /** 196 * Get the master volume of all VSDecoders. 197 * @return the master volume 198 */ 199 public int getMasterVolume() { 200 return getVSDecoderPreferences().getMasterVolume(); 201 } 202 203 /** 204 * Set the master volume for all VSDecoders. 205 * @param mv The new master volume 206 */ 207 public void setMasterVolume(int mv) { 208 getVSDecoderPreferences().setMasterVolume(mv); 209 } 210 211 /** 212 * Get the VSD GUI. 213 * @return the VSD frame 214 */ 215 public JmriJFrame provideManagerFrame() { 216 if (managerFrame == null) { 217 if (GraphicsEnvironment.isHeadless()) { 218 String vsdRosterGroup = "VSD"; 219 if (Roster.getDefault().getRosterGroupList().contains(vsdRosterGroup)) { 220 List<RosterEntry> rosterList; 221 rosterList = Roster.getDefault().getEntriesInGroup(vsdRosterGroup); 222 // Allow <max_decoder> roster entries 223 int entry_counter = 0; 224 for (RosterEntry entry : rosterList) { 225 if (entry_counter < max_decoder) { 226 VSDConfig config = new VSDConfig(); 227 config.setLocoAddress(entry.getDccLocoAddress()); 228 log.info("Loading Roster Entry \"{}\", VSDecoder {} ...", entry.getId(), config.getLocoAddress()); 229 String path = entry.getAttribute("VSDecoder_Path"); 230 String profile = entry.getAttribute("VSDecoder_Profile"); 231 if (path != null && profile != null) { 232 if (LoadVSDFileAction.loadVSDFile(path)) { 233 // config.xml OK 234 log.info(" VSD path: {}", FileUtil.getExternalFilename(path)); 235 config.setProfileName(profile); 236 log.debug(" entry VSD profile: {}", profile); 237 if (entry.getAttribute("VSDecoder_Volume") != null) { 238 config.setVolume(Float.parseFloat(entry.getAttribute("VSDecoder_Volume"))); 239 } else { 240 config.setVolume(0.8f); 241 } 242 VSDecoder newDecoder = VSDecoderManager.instance().getVSDecoder(config); 243 if (newDecoder != null) { 244 log.info("VSD {}, profile \"{}\" ready.", config.getLocoAddress(), config.getProfileName()); 245 entry_counter++; 246 } else { 247 log.warn("VSD {} failed", config.getProfileName()); 248 } 249 } 250 } else { 251 log.error("Cannot load VSD File - path or profile missing - check your Roster Media"); 252 } 253 } else { 254 log.warn("Only {} roster entries allowed. Disgarded {}", max_decoder, rosterList.size() - max_decoder); 255 } 256 } 257 if (entry_counter == 0) { 258 log.warn("No Roster entry found in Roster Group {}", vsdRosterGroup); 259 } 260 } else { 261 log.warn("Roster group \"{}\" not found", vsdRosterGroup); 262 } 263 } else { 264 // Run VSDecoder with GUI 265 managerFrame = new VSDManagerFrame(); 266 } 267 } else { 268 log.warn("Virtual Sound Decoder Manager is already running"); 269 } 270 return managerFrame; 271 } 272 273 private String getNextVSDecoderID() { 274 // vsdecoderID initialized to zero, pre-incremented before return... 275 // first returned ID value is 1. 276 return "IAD:VSD:VSDecoderID" + (++vsdecoderID); // NOI18N 277 } 278 279 private Integer getNextlocorow() { 280 // locorow initialized to -1, pre-incremented before return... 281 // first returned value is 0. 282 return ++locorow; 283 } 284 285 /** 286 * Provide or build a VSDecoder based on a provided configuration. 287 * 288 * @param config previous configuration, not null. 289 * @return vsdecoder, or null on error. 290 */ 291 public VSDecoder getVSDecoder(VSDConfig config) { 292 String path; 293 String profile_name = config.getProfileName(); 294 // First, check to see if we already have a VSDecoder on this Address 295 if (decoderAddressMap.containsKey(config.getLocoAddress().toString())) { 296 return decoderAddressMap.get(config.getLocoAddress().toString()); 297 } 298 if (profiletable.containsKey(profile_name)) { 299 path = profiletable.get(profile_name); 300 log.debug("Profile {} is in table. Path: {}", profile_name, path); 301 302 config.setVSDPath(path); 303 config.setId(getNextVSDecoderID()); 304 VSDecoder vsd = new VSDecoder(config); 305 decodertable.put(vsd.getId(), vsd); 306 decoderAddressMap.put(vsd.getAddress().toString(), vsd); 307 decoderInBlock.put(vsd.getAddress().getNumber(), vsd); 308 locoInBlock[getNextlocorow()][ADDRESS] = vsd.getAddress().getNumber(); 309 310 // set volume for this decoder 311 vsd.setDecoderVolume(vsd.getDecoderVolume()); 312 313 if (geofile_ok) { 314 if (vsd.topspeed == 0) { 315 log.info("Top-speed not defined. No advanced location following possible."); 316 } else { 317 initSoundPositionTimer(vsd); 318 } 319 } 320 return vsd; 321 } else { 322 // Don't have enough info to try to load from file. 323 log.error("Requested profile not loaded: {}", profile_name); 324 return null; 325 } 326 } 327 328 /** 329 * Get a VSDecoder by its Id. 330 * 331 * @param id The Id of the VSDecoder 332 * @return vsdecoder, or null on error. 333 */ 334 public VSDecoder getVSDecoderByID(String id) { 335 VSDecoder v = decodertable.get(id); 336 if (v == null) { 337 log.debug("No decoder in table! ID: {}", id); 338 } 339 return decodertable.get(id); 340 } 341 342 /** 343 * Get a VSDecoder by its address. 344 * 345 * @param sa The address of the VSDecoder 346 * @return vsdecoder, or null on error. 347 */ 348 public VSDecoder getVSDecoderByAddress(String sa) { 349 if (sa == null) { 350 log.debug("Decoder Address is Null"); 351 return null; 352 } 353 log.debug("Decoder Address: {}", sa); 354 VSDecoder rv = decoderAddressMap.get(sa); 355 if (rv == null) { 356 log.debug("Not found."); 357 } else { 358 log.debug("Found: {}", rv.getAddress()); 359 } 360 return rv; 361 } 362 363 /** 364 * Get a list of all profiles. 365 * 366 * @return sl The profiles list. 367 */ 368 public ArrayList<String> getVSDProfileNames() { 369 ArrayList<String> sl = new ArrayList<>(); 370 for (String p : profiletable.keySet()) { 371 sl.add(p); 372 } 373 return sl; 374 } 375 376 /** 377 * Get a list of all VSDecoders. 378 * 379 * @return the VSDecoder list. 380 */ 381 public Collection<VSDecoder> getVSDecoderList() { 382 return decodertable.values(); 383 } 384 385 /** 386 * Get the VSD listener system name. 387 * 388 * @return the system name. 389 */ 390 public String getDefaultListenerName() { 391 return VSDListener.ListenerSysName; 392 } 393 394 /** 395 * Get the VSD listener location. 396 * 397 * @return the location or null. 398 */ 399 public ListeningSpot getDefaultListenerLocation() { 400 VSDListener l = listenerTable.get(getDefaultListenerName()); 401 if (l != null) { 402 return l.getLocation(); 403 } else { 404 return null; 405 } 406 } 407 408 public void setListenerLocation(String id, ListeningSpot sp) { 409 VSDListener l = listenerTable.get(id); 410 log.debug("Set listener location {} listener: {}", sp, l); 411 if (l != null) { 412 l.setLocation(sp); 413 } 414 } 415 416 public void setDecoderPositionByID(String id, PhysicalLocation p) { 417 VSDecoder d = decodertable.get(id); 418 if (d != null) { 419 d.setPosition(p); 420 } 421 } 422 423 public void setDecoderPositionByAddr(LocoAddress a, PhysicalLocation l) { 424 // Find the addressed decoder 425 // This is a bit hokey. Need a better way to index decoder by address 426 // OK, this whole LocoAddress vs. DccLocoAddress thing has rendered this SUPER HOKEY. 427 if (a == null) { 428 log.warn("Decoder Address is Null"); 429 return; 430 } 431 if (l == null) { 432 log.warn("PhysicalLocation is Null"); 433 return; 434 } 435 if (l.equals(PhysicalLocation.Origin)) { 436 log.info("Location: {} ... ignoring", l); 437 // Physical location at origin means it hasn't been set. 438 return; 439 } 440 log.debug("Decoder Address: {}", a.getNumber()); 441 for (VSDecoder d : decodertable.values()) { 442 // Get the Decoder's address protocol. If it's a DCC_LONG or DCC_SHORT, convert to DCC 443 // since the LnReporter can't tell the difference and will always report "DCC". 444 if (d == null) { 445 log.debug("VSdecoder null pointer!"); 446 return; 447 } 448 LocoAddress pa = d.getAddress(); 449 if (pa == null) { 450 log.info("Vsdecoder {} address null!", d); 451 return; 452 } 453 LocoAddress.Protocol p = d.getAddress().getProtocol(); 454 if (p == null) { 455 log.debug("Vsdecoder {} address = {} protocol null!", d, pa); 456 return; 457 } 458 if ((p == LocoAddress.Protocol.DCC_LONG) || (p == LocoAddress.Protocol.DCC_SHORT)) { 459 p = LocoAddress.Protocol.DCC; 460 } 461 if ((d.getAddress().getNumber() == a.getNumber()) && (p == a.getProtocol())) { 462 d.setPosition(l); 463 // Loop through all the decoders (assumes N will be "small"), in case 464 // there are multiple decoders with the same address. This will be somewhat broken 465 // if there's a DCC_SHORT and a DCC_LONG decoder with the same address number. 466 //return; 467 } 468 } 469 // decoder not found. Do nothing. 470 return; 471 } 472 473 // VSDecoderManager Events 474 public void addEventListener(VSDManagerListener listener) { 475 listenerList.add(VSDManagerListener.class, listener); 476 } 477 478 public void removeEventListener(VSDManagerListener listener) { 479 listenerList.remove(VSDManagerListener.class, listener); 480 } 481 482 void fireMyEvent(VSDManagerEvent evt) { 483 //Object[] listeners = listenerList.getListenerList(); 484 485 for (VSDManagerListener l : listenerList.getListeners(VSDManagerListener.class)) { 486 l.eventAction(evt); 487 } 488 } 489 490 /** 491 * Retrieve the Path for a given Profile name. 492 * 493 * @param profile the profile to get the path for 494 * @return the path for the profile 495 */ 496 public String getProfilePath(String profile) { 497 return profiletable.get(profile); 498 } 499 500 protected void registerReporterListener(String sysName) { 501 Reporter r = jmri.InstanceManager.getDefault(jmri.ReporterManager.class).getReporter(sysName); 502 if (r == null) { 503 return; 504 } 505 jmri.NamedBeanHandle<Reporter> h = nbhm.getNamedBeanHandle(sysName, r); 506 507 // Make sure we aren't already registered. 508 java.beans.PropertyChangeListener[] ll = r.getPropertyChangeListenersByReference(h.getName()); 509 if (ll.length == 0) { 510 r.addPropertyChangeListener(this, h.getName(), vsd_property_change_name); 511 } 512 } 513 514 protected void registerBeanListener(Manager<Block> beanManager, String sysName) { 515 NamedBean b = beanManager.getBySystemName(sysName); 516 if (b == null) { 517 log.debug("No bean by name {}", sysName); 518 return; 519 } 520 jmri.NamedBeanHandle<NamedBean> h = nbhm.getNamedBeanHandle(sysName, b); 521 522 // Make sure we aren't already registered. 523 java.beans.PropertyChangeListener[] ll = b.getPropertyChangeListenersByReference(h.getName()); 524 if (ll.length == 0) { 525 b.addPropertyChangeListener(this, h.getName(), vsd_property_change_name); 526 log.debug("Added listener to bean {} type {}", b.getDisplayName(), b.getClass().getName()); 527 } 528 } 529 530 protected void registerReporterListeners() { 531 // Walk through the list of reporters 532 Set<Reporter> reporterSet = jmri.InstanceManager.getDefault(jmri.ReporterManager.class).getNamedBeanSet(); 533 for (Reporter r : reporterSet) { 534 if (r != null) { 535 registerReporterListener(r.getSystemName()); 536 } 537 } 538 539 Set<Block> blockSet = jmri.InstanceManager.getDefault(jmri.BlockManager.class).getNamedBeanSet(); 540 for (Block b : blockSet) { 541 if (b != null) { 542 registerBeanListener(jmri.InstanceManager.getDefault(jmri.BlockManager.class), b.getSystemName()); 543 } 544 } 545 } 546 547 // This listener listens to the ReporterManager for changes to the list of Reporters. 548 // Need to trap list length (name="length") changes and add listeners when new ones are added. 549 private void setupReporterManagerListener() { 550 // Register ourselves as a listener for changes to the Reporter list. For now, we won't do this. Just force a 551 // save and reboot after reporters are added. We'll fix this later. 552 // jmri.InstanceManager.getDefault(jmri.ReporterManager.class).addPropertyChangeListener(new PropertyChangeListener() { 553 // public void propertyChange(PropertyChangeEvent event) { 554 // log.debug("property change name {}, old: {}, new: {}", event.getPropertyName(), event.getOldValue(), event.getNewValue()); 555 // reporterManagerPropertyChange(event); 556 // } 557 // }); 558 jmri.InstanceManager.getDefault(jmri.ReporterManager.class).addPropertyChangeListener(this); 559 560 // Now, the Reporter Table might already be loaded and filled out, so we need to get all the Reporters and list them. 561 // And add ourselves as a listener to them. 562 Set<Reporter> reporterSet = jmri.InstanceManager.getDefault(jmri.ReporterManager.class).getNamedBeanSet(); 563 for (Reporter r : reporterSet) { 564 if (r != null) { 565 registerReporterListener(r.getSystemName()); 566 } 567 } 568 569 Set<Block> blockSet = jmri.InstanceManager.getDefault(jmri.BlockManager.class).getNamedBeanSet(); 570 for (Block b : blockSet) { 571 if (b != null) { 572 registerBeanListener(jmri.InstanceManager.getDefault(jmri.BlockManager.class), b.getSystemName()); 573 } 574 } 575 } 576 577 /** 578 * Delete a VSDecoder 579 * 580 * @param address The DCC address of the VSDecoder 581 */ 582 public void deleteDecoder(String address) { 583 log.debug("delete Decoder called, VSDecoder DCC address: {}", address); 584 if (this.getVSDecoderByAddress(address) == null) { 585 log.warn("VSDecoder not found"); 586 } else { 587 removeVSDecoder(address); 588 } 589 } 590 591 private void removeVSDecoder(String sa) { 592 VSDecoder d = this.getVSDecoderByAddress(sa); 593 jmri.InstanceManager.getDefault(jmri.ThrottleManager.class).removeListener(d.getAddress(), d); 594 stopSoundPositionTimer(d); 595 d.shutdown(); 596 d.disable(); 597 598 decodertable.remove(d.getId()); 599 decoderAddressMap.remove(sa); 600 currentBlock.remove(d); 601 decoderInBlock.remove(d.getAddress().getNumber()); 602 locoInBlockRemove(d.getAddress().getNumber()); 603 timertable.remove(d.getId()); // Remove timer 604 locorow--; // prepare array index for eventually adding a new decoder 605 606 d.sound_list.clear(); 607 d.event_list.clear(); 608 609 jmri.AudioManager am = jmri.InstanceManager.getDefault(jmri.AudioManager.class); 610 ArrayList<Audio> sources = new ArrayList<>(am.getNamedBeanSet(Audio.SOURCE)); 611 ArrayList<Audio> buffers = new ArrayList<>(am.getNamedBeanSet(Audio.BUFFER)); 612 // wait until audio threads are finished and then run audio cleanup via dispose() 613 jmri.util.ThreadingUtil.newThread(new Runnable() { 614 @Override 615 public void run() { 616 try { 617 Thread.sleep(200); 618 } catch (InterruptedException ex) { 619 } 620 for (Audio source: sources) { 621 if (source.getSystemName().contains(d.getId())) { 622 source.dispose(); 623 } 624 } 625 for (Audio buffer: buffers) { 626 if (buffer.getSystemName().contains(d.getId())) { 627 buffer.dispose(); 628 } 629 } 630 } 631 }).start(); 632 } 633 634 /** 635 * Prepare the start of a VSDecoder on the layout 636 * 637 * @param blk The current Block of the VSDecoder 638 */ 639 public void atStart(Block blk) { 640 // blk could be the start block or a current block for an existing VSDecoder 641 int locoAddress = getLocoAddr(blk); 642 if (locoAddress != 0) { 643 // look for an existing and configured VSDecoder 644 if (decoderInBlock.containsKey(locoAddress)) { 645 VSDecoder d = decoderInBlock.get(locoAddress); 646 if (geofile_ok) { 647 if (alf_version == 2 && blockList.contains(blk)) { 648 handleAlf2(d, locoAddress, blk); 649 } else { 650 log.debug("Block {} not valid for panel {}", blk, d.getModels()); 651 } 652 } else { 653 d.savedSound.setTunnel(blk.getPhysicalLocation().isTunnel()); 654 d.setPosition(blk.getPhysicalLocation()); 655 } 656 } else { 657 log.warn("Block value \"{}\" is not a valid VSDecoder address", blk.getValue()); 658 } 659 } 660 } 661 662 /** 663 * Get the loco address from a Block 664 * 665 * @param blk The current Block of the VSDecoder 666 * @return The number of the loco address 667 */ 668 public int getLocoAddr(Block blk) { 669 if (blk == null || blk.getValue() == null) { 670 return 0; 671 } 672 673 var blkVal = blk.getValue(); 674 int locoAddress = 0; 675 676 // handle different formats or objects to get the address 677 if (blkVal instanceof String) { 678 String val = blkVal.toString(); 679 RosterEntry entry = Roster.getDefault().getEntryForId(val); 680 if (entry != null) { 681 locoAddress = Integer.parseInt(entry.getDccAddress()); // numeric RosterEntry Id 682 } else if (org.apache.commons.lang3.StringUtils.isNumeric(val)) { 683 locoAddress = Integer.parseInt(val); 684 } else if (jmri.InstanceManager.getDefault(TrainManager.class).getTrainByName(val) != null) { 685 // Operations Train 686 Train selected_train = jmri.InstanceManager.getDefault(TrainManager.class).getTrainByName(val); 687 if (selected_train.getLeadEngineDccAddress().isEmpty()) { 688 locoAddress = 0; 689 } else { 690 locoAddress = Integer.parseInt(selected_train.getLeadEngineDccAddress()); 691 } 692 } 693 } else if (blkVal instanceof jmri.BasicRosterEntry) { 694 locoAddress = Integer.parseInt(((RosterEntry) blkVal).getDccAddress()); 695 } else if (blkVal instanceof jmri.implementation.DefaultIdTag) { 696 // Covers TranspondingTag also 697 String val = ((DefaultIdTag) blkVal).getTagID(); 698 if (org.apache.commons.lang3.StringUtils.isNumeric(val)) { 699 locoAddress = Integer.parseInt(val); 700 } 701 } else { 702 log.warn("Block Value \"{}\" found - unsupported object!", blkVal); 703 } 704 log.debug("loco address: {}", locoAddress); 705 return locoAddress; 706 } 707 708 @Override 709 public void propertyChange(PropertyChangeEvent evt) { 710 log.debug("property change type {} name {} old {} new {}", 711 evt.getSource().getClass().getName(), evt.getPropertyName(), evt.getOldValue(), evt.getNewValue()); 712 if (evt.getSource() instanceof jmri.ReporterManager) { 713 reporterManagerPropertyChange(evt); 714 } else if (evt.getSource() instanceof jmri.Reporter) { 715 reporterPropertyChange(evt); // Location Following 716 } else if (evt.getSource() instanceof jmri.Block) { 717 log.debug("Block property change! name: {} old: {} new = {}", evt.getPropertyName(), evt.getOldValue(), evt.getNewValue()); 718 blockPropertyChange(evt); 719 } else if (evt.getSource() instanceof VSDManagerFrame) { 720 if (evt.getPropertyName().equals(VSDManagerFrame.REMOVE_DECODER)) { 721 // Shut down the requested decoder and remove it from the manager's hash maps. 722 // Unless there are "illegal" handles, this should put the decoder on the garbage heap. I think. 723 removeVSDecoder((String) evt.getOldValue()); 724 } else if (evt.getPropertyName().equals(VSDManagerFrame.CLOSE_WINDOW)) { 725 // Note this assumes there is only one VSDManagerFrame open at a time. 726 if (managerFrame != null) { 727 managerFrame = null; 728 } 729 } 730 } else { 731 // Un-Handled source. Does nothing ... yet... 732 } 733 return; 734 } 735 736 public void blockPropertyChange(PropertyChangeEvent event) { 737 // Needs to check the ID on the event, look up the appropriate VSDecoder, 738 // get the location of the event source, and update the decoder's location. 739 String eventName = event.getPropertyName(); 740 if (event.getSource() instanceof PhysicalLocationReporter) { 741 Block blk = (Block) event.getSource(); 742 String repVal = null; 743 // Depending on the type of Block Event, extract the needed report info from 744 // the appropriate place... 745 // "state" => Get loco address from Block's Reporter if present 746 // "value" => Get loco address from event's newValue. 747 if ("state".equals(eventName)) { // NOI18N 748 // Need to decide which reporter it is, so we can use different methods 749 // to extract the address and the location. 750 if ((Integer) event.getNewValue() == Block.OCCUPIED) { 751 // Is there a Block's Reporter? 752 var blockReporter = blk.getReporter(); 753 if ( blockReporter == null) { 754 log.debug("Block {} has no reporter! Skipping state-type report", blk.getSystemName()); 755 return; 756 } 757 // Get this Block's Reporter's current/last report value 758 if (blk.isReportingCurrent()) { 759 Object currentReport = blockReporter.getCurrentReport(); 760 if ( currentReport != null) { 761 if(currentReport instanceof jmri.Reportable) { 762 repVal = ((jmri.Reportable)currentReport).toReportString(); 763 } else { 764 repVal = currentReport.toString(); 765 } 766 } 767 } else { 768 Object lastReport = blockReporter.getLastReport(); 769 if ( lastReport != null) { 770 if(lastReport instanceof jmri.Reportable) { 771 repVal = ((jmri.Reportable)lastReport).toReportString(); 772 } else { 773 repVal = lastReport.toString(); 774 } 775 } 776 } 777 } else { 778 log.debug("Ignoring report. not an OCCUPIED event."); 779 return; 780 } 781 log.debug("block repVal: {}", repVal); 782 } else if ("value".equals(eventName)) { // NOI18N 783 if (event.getNewValue() == null ) { 784 return; // block value was cleared, nothing to do 785 } 786 atStart(blk); 787 } else { 788 log.debug("Not a supported Block event type. Ignoring."); 789 return; 790 } 791 792 // Set the decoder's position due to the report. 793 if (repVal == null) { 794 log.debug("Report from Block {} is null!", blk.getSystemName()); 795 } 796 if (repVal != null && blk.getDirection(repVal) == PhysicalLocationReporter.Direction.ENTER) { 797 setDecoderPositionByAddr(blk.getLocoAddress(repVal), blk.getPhysicalLocation()); 798 } 799 800 } else { 801 log.debug("Reporter doesn't support physical location reporting."); 802 } 803 804 } 805 806 public void reporterPropertyChange(PropertyChangeEvent event) { 807 // Needs to check the ID on the event, look up the appropriate VSDecoder, 808 // get the location of the event source, and update the decoder's location. 809 String eventName = event.getPropertyName(); 810 if (lf_version == 1 || (geofile_ok && alf_version == 1)) { 811 if ((event.getSource() instanceof PhysicalLocationReporter) && (eventName.equals("currentReport"))) { // NOI18N 812 PhysicalLocationReporter arp = (PhysicalLocationReporter) event.getSource(); 813 // Need to decide which reporter it is, so we can use different methods 814 // to extract the address and the location. 815 if (event.getNewValue() instanceof IdTag) { 816 // RFID-tag, Digitrax Transponding tags, RailCom tags 817 if (event.getNewValue() instanceof jmri.jmrix.loconet.TranspondingTag) { 818 String repVal = ((jmri.Reportable) event.getNewValue()).toReportString(); 819 int locoAddress = arp.getLocoAddress(repVal).getNumber(); 820 log.debug("Reporter repVal: {}, number: {}", repVal, locoAddress); 821 // Check: is loco address valid? 822 if (decoderInBlock.containsKey(locoAddress)) { 823 VSDecoder d = decoderInBlock.get(locoAddress); 824 // look for additional geometric layout information 825 if (geofile_ok) { 826 Reporter rp = (Reporter) event.getSource(); 827 int new_rp = 0; 828 try { 829 new_rp = Integer.parseInt(Manager.getSystemSuffix(rp.getSystemName())); 830 } catch (java.lang.NumberFormatException e) { 831 log.warn("Invalid Reporter system name '{}'", rp.getSystemName()); 832 } 833 // Check: Reporter must be valid for GeoData processing 834 // use the current Reporter list as a filter (changeable by a Train selection) 835 if (reporterlists.get(d.setup_index).contains(new_rp)) { 836 if (arp.getDirection(repVal) == PhysicalLocationReporter.Direction.ENTER) { 837 handleAlf(d, locoAddress, new_rp); // Advanced Location Following version 1 838 } 839 } else { 840 log.info("Reporter {} not valid for {} setup {}", new_rp, VSDGeoFile.VSDGeoDataFileName, d.setup_index + 1); 841 } 842 } else { 843 if (arp.getDirection(repVal) == PhysicalLocationReporter.Direction.ENTER) { 844 d.savedSound.setTunnel(arp.getPhysicalLocation(repVal).isTunnel()); 845 d.setPosition(arp.getPhysicalLocation(repVal)); 846 log.debug("position set to: {}", arp.getPhysicalLocation(repVal)); 847 } 848 } 849 } else { 850 log.info(" decoder address {} is not valid!", locoAddress); 851 } 852 return; 853 } else { 854 // newValue is of IdTag type. 855 // Dcc4Pc, Ecos, 856 // Assume Reporter "arp" is the most recent seen location 857 IdTag newValue = (IdTag) event.getNewValue(); 858 decoderInBlock.get(arp.getLocoAddress(newValue.getTagID()).getNumber()).savedSound.setTunnel(arp.getPhysicalLocation(null).isTunnel()); 859 setDecoderPositionByAddr(arp.getLocoAddress(newValue.getTagID()), arp.getPhysicalLocation(null)); 860 } 861 } else { 862 log.info("Reporter's return type is not supported."); 863 } 864 } else { 865 log.debug("Reporter doesn't support physical location reporting or isn't reporting new info."); 866 } 867 } 868 return; 869 } 870 871 public void reporterManagerPropertyChange(PropertyChangeEvent event) { 872 String eventName = event.getPropertyName(); 873 874 log.debug("VSDecoder received Reporter Manager Property Change: {}", eventName); 875 if (eventName.equals("length")) { // NOI18N 876 877 // Re-register for all the reporters. The registerReporterListener() will skip 878 // any that we're already registered for. 879 for (Reporter r : jmri.InstanceManager.getDefault(jmri.ReporterManager.class).getNamedBeanSet()) { 880 registerReporterListener(r.getSystemName()); 881 } 882 883 // It could be that we lost a Reporter. But since we aren't keeping a list anymore 884 // we don't care. 885 } 886 } 887 888 // handle Advanced Location Following version 1 889 private void handleAlf(VSDecoder d, int locoAddress, int new_rp) { 890 int new_rp_index = reporterlists.get(d.setup_index).indexOf(new_rp); 891 int old_rp = -1; // set to "undefined" 892 int old_rp_index = -1; // set to "undefined" 893 int ix = getArrayIndex(locoAddress); 894 if (ix < locoInBlock.length) { 895 old_rp = locoInBlock[ix][BLOCK]; 896 if (old_rp == 0) old_rp = -1; // set to "undefined" 897 old_rp_index = reporterlists.get(d.setup_index).indexOf(old_rp); // -1 if not found (undefined) 898 } else { 899 log.warn(" Array locoInBlock INDEX {} IS NOT VALID! Set to 0.", ix); 900 ix = 0; 901 } 902 log.debug("new_rp: {}, old_rp: {}, new index: {}, old index: {}", new_rp, old_rp, new_rp_index, old_rp_index); 903 // Validation check: don't proceed when it's the same reporter 904 if (new_rp != old_rp) { 905 // Validation check: reporter must be a new or a neighbour reporter or must rotating in a circle 906 int lastrepix = reporterlists.get(d.setup_index).size() - 1; // Get the index of the last Reporter 907 if ((old_rp == -1) // Loco can be in any section, if it's the first reported section; old rp is "undefined" 908 || (old_rp_index + d.dirfn == new_rp_index) // Loco is running forward or reverse 909 || (circlelist.get(d.setup_index) && d.dirfn == -1 && old_rp_index == 0 && new_rp_index == lastrepix) // Loco is running reverse and circling 910 || (circlelist.get(d.setup_index) && d.dirfn == 1 && old_rp_index == lastrepix && new_rp_index == 0)) { // Loco is running forward and circling 911 // Validation check: OK 912 locoInBlock[ix][BLOCK] = new_rp; // Set new block number (int) 913 log.debug(" distance rest (old) to go in block {}: {} cm", old_rp, locoInBlock[ix][DISTANCE_TO_GO]); 914 locoInBlock[ix][DISTANCE_TO_GO] = Math.round(blockParameter[d.setup_index][new_rp_index][LENGTH] * 100.0f); // block distance init: block length in cm 915 log.debug(" distance rest (new) to go in block {}: {} cm", new_rp, locoInBlock[ix][DISTANCE_TO_GO]); 916 // get the new sound position point (depends on the loco traveling direction) 917 if (d.dirfn == 1) { 918 d.posToSet = blockPositionlists.get(d.setup_index).get(new_rp_index); // Start position 919 } else { 920 d.posToSet = blockPositionlists.get(d.setup_index).get(new_rp_index + 1); // End position 921 } 922 if (old_rp == -1 && d.startPos != null) { // Special case start position: first choice; if found, overwrite it. 923 d.posToSet = d.startPos; 924 } 925 d.savedSound.setTunnel(blockPositionlists.get(d.setup_index).get(new_rp_index).isTunnel()); // set the tunnel status 926 log.debug("address {}: position to set: {}", d.getAddress(), d.posToSet); 927 d.setPosition(d.posToSet); // Sound set position 928 changeDirection(d, locoAddress, new_rp_index); 929 stopSoundPositionTimer(d); 930 startSoundPositionTimer(d); // timer restart 931 } else { 932 log.info(" Validation failed! Last reporter: {}, new reporter: {}, dirfn: {} for {}", old_rp, new_rp, d.dirfn, locoAddress); 933 } 934 } else { 935 log.info(" Same PhysicalLocationReporter, position not set!"); 936 } 937 } 938 939 // handle Advanced Location Following version 2 940 private void handleAlf2(VSDecoder d, int locoAddress, Block newBlock) { 941 if (currentBlock.get(d) != newBlock) { 942 int ix = getArrayIndex(locoAddress); // ix = decoder number 0 - max_decoder-1 943 if (locoInBlock[ix][DIR_FN] == 0) { // at start 944 if (d.getLayoutTrack() == null) { 945 if (possibleStartBlocks.get(newBlock) != null) { 946 d.setModels(possibleStartBlocks.get(newBlock)); // get the models from the HashMap via block 947 log.debug("Block: {}, models: {}", newBlock, d.getModels()); 948 TrackSegment ts = null; 949 for (LayoutTrack lt : d.getModels().getLayoutTracks()) { 950 if (lt instanceof TrackSegment) { 951 ts = (TrackSegment) lt; 952 if (ts.getLayoutBlock() != null && ts.getLayoutBlock().getBlock() == newBlock) { 953 break; 954 } 955 } 956 } 957 if (ts != null) { 958 TrackSegmentView tsv = d.getModels().getTrackSegmentView(ts); 959 d.setLayoutTrack(ts); 960 d.setReturnTrack(d.getLayoutTrack()); 961 d.setReturnLastTrack(tsv.getConnect2()); 962 d.setLastTrack(tsv.getConnect1()); 963 d.setReturnDistance(MathUtil.distance(d.getModels().getCoords(tsv.getConnect1(), tsv.getType1()), 964 d.getModels().getCoords(tsv.getConnect2(), tsv.getType2()))); 965 d.setDistance(0); 966 d.distanceOnTrack = 0.5d * d.getReturnDistance(); // halved to get starting position (mid or centre of the track) 967 if (d.dirfn == -1) { // in case the loco is running in reverse direction 968 d.setLayoutTrack(d.getReturnTrack()); 969 d.setLastTrack(d.getReturnLastTrack()); 970 } 971 locoInBlock[ix][DIR_FN] = d.dirfn; 972 currentBlock.put(d, newBlock); 973 // prepare navigation 974 d.posToSet = new PhysicalLocation(0.0f, 0.0f, 0.0f); 975 log.info("at start - TS: {}, block: {}, loco: {}, panel: {}", ts.getName(), newBlock, locoAddress, d.getModels().getTitle()); 976 } 977 } else { 978 log.warn("block {} is not a valid start block; valid start blocks are: {}", newBlock, possibleStartBlocks); 979 } 980 } 981 982 } else { 983 984 currentBlock.put(d, newBlock); 985 // new block; if end point is already reached, d.distanceOnTrack is zero 986 if (d.distanceOnTrack > 0) { 987 // it's still on this track 988 // handle a block change, if the loco reaches the next block before the calculated end 989 boolean result = true; // new block, so go to the next track 990 d.distanceOnTrack = 0; 991 // go to next track 992 LayoutTrack last = d.getLayoutTrack(); 993 if (d.getLayoutTrack() instanceof TrackSegment) { 994 TrackSegmentView tsv = d.getModels().getTrackSegmentView((TrackSegment) d.getLayoutTrack()); 995 log.debug(" true - layout track: {}, last track: {}, connect1: {}, connect2: {}, last block: {}", 996 d.getLayoutTrack().getName(), d.getLastTrack().getName(), tsv.getConnect1(), tsv.getConnect2(), tsv.getBlockName()); 997 if (tsv.getConnect1().equals(d.getLastTrack())) { 998 d.setLayoutTrack(tsv.getConnect2()); 999 } else if (tsv.getConnect2().equals(d.getLastTrack())) { 1000 d.setLayoutTrack(tsv.getConnect1()); 1001 } else { // OOPS! we're lost! 1002 log.info(" TS lost, c1: {}, c2: {}, last track: {}", tsv.getConnect1(), tsv.getConnect2(), d.getLastTrack()); 1003 result = false; 1004 } 1005 if (result) { 1006 d.setLastTrack(last); 1007 d.setReturnTrack(d.getLayoutTrack()); 1008 d.setReturnLastTrack(d.getLayoutTrack()); 1009 log.debug(" next track (layout track): {}, last track: {}", d.getLayoutTrack(), d.getLastTrack()); 1010 } 1011 } else if (d.getLayoutTrack() instanceof LayoutTurnout 1012 || d.getLayoutTrack() instanceof LayoutSlip 1013 || d.getLayoutTrack() instanceof LevelXing 1014 || d.getLayoutTrack() instanceof LayoutTurntable) { 1015 // go to next track 1016 if (d.nextLayoutTrack != null) { 1017 d.setLayoutTrack(d.nextLayoutTrack); 1018 } else { // OOPS! we're lost! 1019 result = false; 1020 } 1021 if (result) { 1022 d.setLastTrack(last); 1023 d.setReturnTrack(d.getLayoutTrack()); 1024 d.setReturnLastTrack(d.getLayoutTrack()); 1025 } 1026 } 1027 } 1028 } 1029 startSoundPositionTimer(d); 1030 } else { 1031 log.warn(" Same PhysicalLocationReporter, position not set!"); 1032 } 1033 } 1034 1035 private void changeDirection(VSDecoder d, int locoAddress, int new_rp_index) { 1036 PhysicalLocation point1 = blockPositionlists.get(d.setup_index).get(new_rp_index); 1037 PhysicalLocation point2 = blockPositionlists.get(d.setup_index).get(new_rp_index + 1); 1038 Point2D coords1 = new Point2D.Double(point1.x, point1.y); 1039 Point2D coords2 = new Point2D.Double(point2.x, point2.y); 1040 int direct; 1041 if (d.dirfn == 1) { 1042 direct = Path.computeDirection(coords1, coords2); 1043 } else { 1044 direct = Path.computeDirection(coords2, coords1); 1045 } 1046 locoInBlock[getArrayIndex(locoAddress)][DIRECTION] = direct; 1047 log.debug("direction: {} ({})", Path.decodeDirection(direct), direct); 1048 } 1049 1050 /** 1051 * Get index of a decoder. 1052 * @param number The loco address number. 1053 * @return the index of a decoder's loco address number 1054 * in the array or the length of the array. 1055 */ 1056 public int getArrayIndex(int number) { 1057 for (int i = 0; i < locoInBlock.length; i++) { 1058 if (locoInBlock[i][ADDRESS] == number) { 1059 return i; 1060 } 1061 } 1062 return locoInBlock.length; 1063 } 1064 1065 public void locoInBlockRemove(int numb) { 1066 // Works only for <locoInBlock.length> rows 1067 // find index first 1068 int remove_index = 0; 1069 for (int i = 0; i < locoInBlock.length; i++) { 1070 if (locoInBlock[i][ADDRESS] == numb) { 1071 remove_index = i; 1072 } 1073 } 1074 for (int i = remove_index; i < locoInBlock.length - 1; i++) { 1075 for (int k = 0; k < locoInBlock[i].length; k++) { 1076 locoInBlock[i][k] = locoInBlock[i + 1][k]; 1077 } 1078 } 1079 // Delete last row 1080 int il = locoInBlock.length - 1; 1081 for (int k = 0; k < locoInBlock[il].length; k++) { 1082 locoInBlock[il][k] = 0; 1083 } 1084 } 1085 1086 public void loadProfiles(VSDFile vf) { 1087 Element root; 1088 String pname; 1089 root = vf.getRoot(); 1090 if (root == null) { 1091 return; 1092 } 1093 1094 ArrayList<String> new_entries = new ArrayList<>(); 1095 1096 java.util.Iterator<Element> i = root.getChildren("profile").iterator(); // NOI18N 1097 while (i.hasNext()) { 1098 Element e = i.next(); 1099 pname = e.getAttributeValue("name"); 1100 log.debug("Profile name: {}", pname); 1101 if ((pname != null) && !(pname.isEmpty())) { // NOI18N 1102 profiletable.put(pname, vf.getName()); 1103 new_entries.add(pname); 1104 } 1105 } 1106 1107 if (!GraphicsEnvironment.isHeadless()) { 1108 fireMyEvent(new VSDManagerEvent(this, VSDManagerEvent.EventType.PROFILE_LIST_CHANGE, new_entries)); 1109 } 1110 } 1111 1112 void initSoundPositionTimer(VSDecoder d) { 1113 if (geofile_ok) { 1114 Timer t = new Timer(check_time, new ActionListener() { 1115 @Override 1116 public void actionPerformed(ActionEvent e) { 1117 if (alf_version == 1) { 1118 calcNewPosition(d); 1119 } else if (alf_version == 2) { 1120 int ix = getArrayIndex(d.getAddress().getNumber()); // ix = decoder number 0 - max_decoder-1 1121 float actualspeed = d.getEngineSound().getActualSpeed(); 1122 if (locoInBlock[ix][DIR_FN] != d.dirfn) { 1123 // traveling direction has changed 1124 if (d.getEngineSound().isEngineStarted()) { 1125 locoInBlock[ix][DIR_FN] = d.dirfn; // save traveling direction info 1126 if (d.distanceOnTrack <= d.getReturnDistance()) { 1127 d.distanceOnTrack = d.getReturnDistance() - d.distanceOnTrack; 1128 } else { 1129 d.distanceOnTrack = d.getReturnDistance(); 1130 } 1131 d.setLayoutTrack(d.getReturnTrack()); 1132 d.setLastTrack(d.getReturnLastTrack()); 1133 log.debug("direction changed to {}, layout: {}, last: {}, return: {}, d.getReturnDistance: {}, d.distanceOnTrack: {}, d.getDistance: {}", 1134 d.dirfn, d.getLayoutTrack(), d.getLastTrack(), d.getReturnTrack(), d.getReturnDistance(), d.distanceOnTrack, d.getDistance()); 1135 d.setDistance(0); 1136 d.navigate(); 1137 } 1138 } 1139 if ((d.getEngineSound().isEngineStarted() && actualspeed > 0.0f) || d.getLayoutTrack() instanceof LayoutTurntable) { 1140 float speed_ms = actualspeed * (d.dirfn == 1 ? d.topspeed : d.topspeed_rev) * 0.44704f / layout_scale; // calculate the speed 1141 d.setDistance(d.getDistance() + speed_ms * check_time / 10.0); // d.getDistance() normally is 0, but can content an overflow 1142 d.navigate(); 1143 Point2D loc = d.getLocation(); 1144 Point2D loc2 = new Point2D.Double(((float) loc.getX() - models_origin.x) * 0.01f, (models_origin.y - (float) loc.getY()) * 0.01f); 1145 d.posToSet.x = (float) loc2.getX(); 1146 d.posToSet.y = (float) loc2.getY(); 1147 d.posToSet.z = 0.0f; 1148 log.debug("address {} position to set: {}, location: {}", d.getAddress(), d.posToSet, loc); 1149 d.setPosition(d.posToSet); 1150 } 1151 } 1152 } 1153 }); 1154 t.setRepeats(true); 1155 timertable.put(d.getId(), t); 1156 log.debug("timer {} created for decoder {}, id: {}", t, d, d.getId()); 1157 } else { 1158 log.debug("No timer created, GeoData not available"); 1159 } 1160 } 1161 1162 void startSoundPositionTimer(VSDecoder d) { 1163 Timer t = timertable.get(d.getId()); 1164 if (t != null) { 1165 t.setInitialDelay(check_time); 1166 t.start(); 1167 log.debug("timer {} started for decoder id {}, {}, check time: {}", t, d.getId(), d, check_time); 1168 } 1169 } 1170 1171 void stopSoundPositionTimer(VSDecoder d) { 1172 Timer t = timertable.get(d.getId()); 1173 if (t != null) { 1174 if (t.isRunning()) { 1175 t.stop(); 1176 log.debug("timer {} stopped for {}", t, d); 1177 } else { 1178 log.debug("timer {} was not running", t); 1179 } 1180 } 1181 } 1182 1183 // Simple way to calulate loco positions within a block 1184 // train route is described by a combination of two types of geometric elements: line track or curve track 1185 // the train route data is provided by a xml file and gathered by method getBlockValues 1186 public void calcNewPosition(VSDecoder d) { 1187 float actualspeed = d.getEngineSound().getActualSpeed(); 1188 if (actualspeed > 0.0f && d.topspeed > 0) { // proceed only, if the loco is running and if a topspeed value is available 1189 int dadr = d.getAddress().getNumber(); 1190 int dadr_index = getArrayIndex(dadr); // check, if the decoder is in "Block status for locos" - remove this check? 1191 if (dadr_index < locoInBlock.length) { 1192 // decoder is valid 1193 int dadr_block = locoInBlock[dadr_index][BLOCK]; // get block number for current decoder/loco 1194 if (reporterlists.get(d.setup_index).contains(dadr_block)) { 1195 int dadr_block_index = reporterlists.get(d.setup_index).indexOf(dadr_block); 1196 newPosition = new PhysicalLocation(0.0f, 0.0f, 0.0f, d.savedSound.getTunnel()); 1197 // calculate actual speed in meter/second; support topspeed forward or reverse 1198 // JMRI speed is 0-1; actual speed is speed after speedCurve(float); in steam1 it is calculated from actual RPM; convert MPH to meter/second; regard layout scale 1199 float speed_ms = actualspeed * (d.dirfn == 1 ? d.topspeed : d.topspeed_rev) * 0.44704f / layout_scale; 1200 d.distanceMeter = speed_ms * check_time / 1000; // distance in Meter 1201 if (locoInBlock[dadr_index][DIR_FN] == 0) { // at start 1202 locoInBlock[dadr_index][DIR_FN] = d.dirfn; 1203 } 1204 distance_rest_old = locoInBlock[dadr_index][DISTANCE_TO_GO] / 100.0f; // Distance to go in meter 1205 if (locoInBlock[dadr_index][DIR_FN] == d.dirfn) { // Last traveling direction 1206 distance_rest = distance_rest_old; 1207 } else { 1208 // traveling direction has changed 1209 distance_rest = blockParameter[d.setup_index][dadr_block_index][LENGTH] - distance_rest_old; 1210 locoInBlock[dadr_index][DIR_FN] = d.dirfn; 1211 changeDirection(d, dadr, dadr_block_index); 1212 log.debug("direction changed to {}", locoInBlock[dadr_index][DIRECTION]); 1213 } 1214 distance_rest_new = distance_rest - d.distanceMeter; // Distance to go in Meter 1215 log.debug(" distance_rest_old: {}, distance_rest: {}, distance_rest_new: {} (all in Meter)", distance_rest_old, distance_rest, distance_rest_new); 1216 // Calculate and set sound position only, if loco would be still inside the block 1217 if (distance_rest_new > 0.0f) { 1218 // Which geometric element? RADIUS = 0 means "line" 1219 if (blockParameter[d.setup_index][dadr_block_index][RADIUS] == 0.0f) { 1220 // Line 1221 if (locoInBlock[dadr_index][DIRECTION] == Path.SOUTH) { 1222 newPosition.x = d.lastPos.x; 1223 newPosition.y = d.lastPos.y - d.distanceMeter; 1224 } else if (locoInBlock[dadr_index][DIRECTION] == Path.NORTH) { 1225 newPosition.x = d.lastPos.x; 1226 newPosition.y = d.lastPos.y + d.distanceMeter; 1227 } else { 1228 xPosi = d.distanceMeter * (float) Math.sqrt(1.0f / (1.0f + 1229 blockParameter[d.setup_index][dadr_block_index][SLOPE] * blockParameter[d.setup_index][dadr_block_index][SLOPE])); 1230 if (locoInBlock[dadr_index][DIRECTION] == Path.SOUTH_WEST || locoInBlock[dadr_index][DIRECTION] == Path.WEST || locoInBlock[dadr_index][DIRECTION] == Path.NORTH_WEST) { 1231 newPosition.x = d.lastPos.x - xPosi; 1232 newPosition.y = d.lastPos.y - xPosi * blockParameter[d.setup_index][dadr_block_index][SLOPE]; 1233 } else { 1234 newPosition.x = d.lastPos.x + xPosi; 1235 newPosition.y = d.lastPos.y + xPosi * blockParameter[d.setup_index][dadr_block_index][SLOPE]; 1236 } 1237 } 1238 newPosition.z = 0.0f; 1239 } else { 1240 // Curve 1241 float anglePos = d.distanceMeter / blockParameter[d.setup_index][dadr_block_index][RADIUS] * (-d.dirfn); // distanceMeter / RADIUS * (-loco direction) 1242 float rotate_xpos = blockParameter[d.setup_index][dadr_block_index][ROTATE_XPOS_I]; 1243 float rotate_ypos = blockParameter[d.setup_index][dadr_block_index][ROTATE_YPOS_I]; // rotation center point y 1244 newPosition.x = rotate_xpos + (float) Math.cos(anglePos) * (d.lastPos.x - rotate_xpos) - (float) Math.sin(anglePos) * (d.lastPos.y - rotate_ypos); 1245 newPosition.y = rotate_ypos + (float) Math.sin(anglePos) * (d.lastPos.x - rotate_xpos) + (float) Math.cos(anglePos) * (d.lastPos.y - rotate_ypos); 1246 newPosition.z = 0.0f; 1247 } 1248 log.debug("position to set: {}", newPosition); 1249 d.setPosition(newPosition); // Sound set position 1250 log.debug(" distance rest to go in block: {} of {} cm", Math.round(distance_rest_new * 100.0f), 1251 Math.round(blockParameter[d.setup_index][dadr_block_index][LENGTH] * 100.0f)); 1252 locoInBlock[dadr_index][DISTANCE_TO_GO] = Math.round(distance_rest_new * 100.0f); // Save distance rest in cm 1253 log.debug(" saved distance rest: {}", locoInBlock[dadr_index][DISTANCE_TO_GO]); 1254 } else { 1255 log.debug(" new position not set due to less distance"); 1256 } 1257 } else { 1258 log.warn(" block for loco address {} not yet identified. May be there is another loco in the same block", dadr); 1259 } 1260 } else { 1261 log.warn(" decoder {} not found", dadr); 1262 } 1263 } 1264 } 1265 1266 private static final org.slf4j.Logger log = org.slf4j.LoggerFactory.getLogger(VSDecoderManager.class); 1267 1268}