001package jmri.jmrit.operations.trains.gui;
002
003import java.awt.*;
004import java.beans.PropertyChangeEvent;
005import java.beans.PropertyChangeListener;
006import java.util.Hashtable;
007import java.util.List;
008
009import javax.swing.*;
010import javax.swing.table.DefaultTableCellRenderer;
011import javax.swing.table.TableCellEditor;
012
013import jmri.InstanceManager;
014import jmri.jmrit.beantable.EnablingCheckboxRenderer;
015import jmri.jmrit.operations.locations.Track;
016import jmri.jmrit.operations.setup.Control;
017import jmri.jmrit.operations.setup.Setup;
018import jmri.jmrit.operations.trains.*;
019import jmri.util.swing.JmriJOptionPane;
020import jmri.util.swing.XTableColumnModel;
021import jmri.util.table.ButtonEditor;
022import jmri.util.table.ButtonRenderer;
023
024/**
025 * Table Model for edit of trains used by operations
026 *
027 * @author Daniel Boudreau Copyright (C) 2008, 2012
028 */
029public class TrainsTableModel extends javax.swing.table.AbstractTableModel implements PropertyChangeListener {
030
031    TrainManager trainManager = InstanceManager.getDefault(TrainManager.class); // There is only one manager
032    volatile List<Train> sysList = trainManager.getTrainsByTimeList();
033    JTable _table = null;
034    TrainsTableFrame _frame = null;
035    
036    // Defines the columns
037    private static final int ID_COLUMN = 0;
038    private static final int TIME_COLUMN = ID_COLUMN + 1;
039    private static final int BUILDBOX_COLUMN = TIME_COLUMN + 1;
040    private static final int BUILD_COLUMN = BUILDBOX_COLUMN + 1;
041    private static final int NAME_COLUMN = BUILD_COLUMN + 1;
042    private static final int DESCRIPTION_COLUMN = NAME_COLUMN + 1;
043    private static final int BUILT_COLUMN = DESCRIPTION_COLUMN + 1;
044    private static final int CAR_ROAD_COLUMN = BUILT_COLUMN + 1;
045    private static final int CABOOSE_ROAD_COLUMN = CAR_ROAD_COLUMN + 1;
046    private static final int LOCO_ROAD_COLUMN = CABOOSE_ROAD_COLUMN + 1;
047    private static final int LOAD_COLUMN = LOCO_ROAD_COLUMN + 1;
048    private static final int OWNER_COLUMN = LOAD_COLUMN + 1;
049    private static final int ROUTE_COLUMN = OWNER_COLUMN + 1;
050    private static final int DEPARTS_COLUMN = ROUTE_COLUMN + 1;
051    private static final int TERMINATES_COLUMN = DEPARTS_COLUMN + 1;
052    private static final int CURRENT_COLUMN = TERMINATES_COLUMN + 1;
053    private static final int CARS_COLUMN = CURRENT_COLUMN + 1;
054    private static final int STATUS_COLUMN = CARS_COLUMN + 1;
055    private static final int ACTION_COLUMN = STATUS_COLUMN + 1;
056    private static final int EDIT_COLUMN = ACTION_COLUMN + 1;
057
058    private static final int HIGHESTCOLUMN = EDIT_COLUMN + 1;
059
060    public TrainsTableModel() {
061        super();
062        trainManager.addPropertyChangeListener(this);
063        Setup.getDefault().addPropertyChangeListener(this);
064        updateList();
065    }
066
067    public final int SORTBYTIME = 2;
068    public final int SORTBYID = 7;
069
070    private int _sort = SORTBYTIME;
071
072    public void setSort(int sort) {
073        _sort = sort;
074        updateList();
075        updateColumnVisible();
076    }
077
078    private boolean _showAll = true;
079
080    public void setShowAll(boolean showAll) {
081        _showAll = showAll;
082        updateList();
083        fireTableDataChanged();
084    }
085
086    public boolean isShowAll() {
087        return _showAll;
088    }
089
090    private void updateList() {
091        // first, remove listeners from the individual objects
092        removePropertyChangeTrains();
093
094        List<Train> tempList;
095        if (_sort == SORTBYID) {
096            tempList = trainManager.getTrainsByIdList();
097        } else {
098            tempList = trainManager.getTrainsByTimeList();
099        }
100
101        if (!isShowAll()) {
102            // filter out trains not checked
103            for (int i = tempList.size() - 1; i >= 0; i--) {
104                if (!tempList.get(i).isBuildEnabled()) {
105                    tempList.remove(i);
106                }
107            }
108        }
109        sysList = tempList;
110
111        // and add listeners back in
112        addPropertyChangeTrains();
113    }
114
115    private Train getTrainByRow(int row) {
116        return sysList.get(row);
117    }
118
119    void initTable(JTable table, TrainsTableFrame frame) {
120        _table = table;
121        _frame = frame;
122        // allow row color to be controlled
123        table.setDefaultRenderer(Object.class, new MyTableCellRenderer());
124        table.setDefaultRenderer(Integer.class, new MyTableCellRenderer());
125        initTable();
126    }
127
128    // Train frame table column widths, starts with id column and ends with edit
129    private final int[] _tableColumnWidths =
130            {50, 50, 50, 72, 100, 140, 50, 50, 50, 50, 50, 50, 120, 120, 120, 120, 50, 120, 90,
131            70};
132
133    void initTable() {
134        // Use XTableColumnModel so we can control which columns are visible
135        XTableColumnModel tcm = new XTableColumnModel();
136        _table.setColumnModel(tcm);
137        _table.createDefaultColumnsFromModel();
138
139        // Install the button handlers
140        ButtonRenderer buttonRenderer = new ButtonRenderer();
141        ButtonRenderer buttonRenderer2 = new ButtonRenderer(); // for tool tips
142        TableCellEditor buttonEditor = new ButtonEditor(new javax.swing.JButton());
143        tcm.getColumn(EDIT_COLUMN).setCellRenderer(buttonRenderer);
144        tcm.getColumn(EDIT_COLUMN).setCellEditor(buttonEditor);
145        tcm.getColumn(ACTION_COLUMN).setCellRenderer(buttonRenderer);
146        tcm.getColumn(ACTION_COLUMN).setCellEditor(buttonEditor);
147        tcm.getColumn(BUILD_COLUMN).setCellRenderer(buttonRenderer2);
148        tcm.getColumn(BUILD_COLUMN).setCellEditor(buttonEditor);
149        _table.setDefaultRenderer(Boolean.class, new EnablingCheckboxRenderer());
150
151        // set column preferred widths
152        for (int i = 0; i < tcm.getColumnCount(); i++) {
153            tcm.getColumn(i).setPreferredWidth(_tableColumnWidths[i]);
154        }
155        _frame.loadTableDetails(_table);
156
157        // turn off column
158        updateColumnVisible();
159    }
160
161    private void updateColumnVisible() {
162        XTableColumnModel tcm = (XTableColumnModel) _table.getColumnModel();
163        tcm.setColumnVisible(tcm.getColumnByModelIndex(ID_COLUMN), _sort == SORTBYID);
164        tcm.setColumnVisible(tcm.getColumnByModelIndex(TIME_COLUMN), _sort == SORTBYTIME);
165        tcm.setColumnVisible(tcm.getColumnByModelIndex(BUILT_COLUMN), trainManager.isBuiltRestricted());
166        tcm.setColumnVisible(tcm.getColumnByModelIndex(CAR_ROAD_COLUMN), trainManager.isCarRoadRestricted());
167        tcm.setColumnVisible(tcm.getColumnByModelIndex(CABOOSE_ROAD_COLUMN), trainManager.isCabooseRoadRestricted());
168        tcm.setColumnVisible(tcm.getColumnByModelIndex(LOCO_ROAD_COLUMN), trainManager.isLocoRoadRestricted());
169        tcm.setColumnVisible(tcm.getColumnByModelIndex(LOAD_COLUMN), trainManager.isLoadRestricted());
170        tcm.setColumnVisible(tcm.getColumnByModelIndex(OWNER_COLUMN), trainManager.isOwnerRestricted());
171    }
172
173    @Override
174    public int getRowCount() {
175        return sysList.size();
176    }
177
178    @Override
179    public int getColumnCount() {
180        return HIGHESTCOLUMN;
181    }
182
183    public static final String IDCOLUMNNAME = Bundle.getMessage("Id");
184    public static final String TIMECOLUMNNAME = Bundle.getMessage("Time");
185    public static final String BUILDBOXCOLUMNNAME = Bundle.getMessage("Build");
186    public static final String BUILDCOLUMNNAME = Bundle.getMessage("Function");
187    public static final String NAMECOLUMNNAME = Bundle.getMessage("Name");
188    public static final String DESCRIPTIONCOLUMNNAME = Bundle.getMessage("Description");
189    public static final String ROUTECOLUMNNAME = Bundle.getMessage("Route");
190    public static final String DEPARTSCOLUMNNAME = Bundle.getMessage("Departs");
191    public static final String CURRENTCOLUMNNAME = Bundle.getMessage("Current");
192    public static final String TERMINATESCOLUMNNAME = Bundle.getMessage("Terminates");
193    public static final String STATUSCOLUMNNAME = Bundle.getMessage("Status");
194    public static final String ACTIONCOLUMNNAME = Bundle.getMessage("Action");
195    public static final String EDITCOLUMNNAME = Bundle.getMessage("ButtonEdit");
196
197    @Override
198    public String getColumnName(int col) {
199        switch (col) {
200            case ID_COLUMN:
201                return IDCOLUMNNAME;
202            case TIME_COLUMN:
203                return TIMECOLUMNNAME;
204            case BUILDBOX_COLUMN:
205                return BUILDBOXCOLUMNNAME;
206            case BUILD_COLUMN:
207                return BUILDCOLUMNNAME;
208            case NAME_COLUMN:
209                return NAMECOLUMNNAME;
210            case DESCRIPTION_COLUMN:
211                return DESCRIPTIONCOLUMNNAME;
212            case BUILT_COLUMN:
213                return Bundle.getMessage("Built");
214            case CAR_ROAD_COLUMN:
215                return Bundle.getMessage("RoadsCar");
216            case CABOOSE_ROAD_COLUMN:
217                return Bundle.getMessage("RoadsCaboose");
218            case LOCO_ROAD_COLUMN:
219                return Bundle.getMessage("RoadsLoco");
220            case LOAD_COLUMN:
221                return Bundle.getMessage("Load");
222            case OWNER_COLUMN:
223                return Bundle.getMessage("Owner");
224            case ROUTE_COLUMN:
225                return ROUTECOLUMNNAME;
226            case DEPARTS_COLUMN:
227                return DEPARTSCOLUMNNAME;
228            case CURRENT_COLUMN:
229                return CURRENTCOLUMNNAME;
230            case TERMINATES_COLUMN:
231                return TERMINATESCOLUMNNAME;
232            case CARS_COLUMN:
233                return Bundle.getMessage("Cars");
234            case STATUS_COLUMN:
235                return STATUSCOLUMNNAME;
236            case ACTION_COLUMN:
237                return ACTIONCOLUMNNAME;
238            case EDIT_COLUMN:
239                return EDITCOLUMNNAME;
240            default:
241                return "unknown"; // NOI18N
242        }
243    }
244
245    @Override
246    public Class<?> getColumnClass(int col) {
247        switch (col) {
248            case BUILDBOX_COLUMN:
249                return Boolean.class;
250            case ID_COLUMN:
251            case CARS_COLUMN:
252                return Integer.class;
253            case TIME_COLUMN:
254            case NAME_COLUMN:
255            case DESCRIPTION_COLUMN:
256            case BUILT_COLUMN:
257            case CAR_ROAD_COLUMN:
258            case CABOOSE_ROAD_COLUMN:
259            case LOCO_ROAD_COLUMN:
260            case LOAD_COLUMN:
261            case OWNER_COLUMN:
262            case ROUTE_COLUMN:
263            case DEPARTS_COLUMN:
264            case CURRENT_COLUMN:
265            case TERMINATES_COLUMN:
266            case STATUS_COLUMN:
267                return String.class;
268            case BUILD_COLUMN:
269            case ACTION_COLUMN:
270            case EDIT_COLUMN:
271                return JButton.class;
272            default:
273                return null;
274        }
275    }
276
277    @Override
278    public boolean isCellEditable(int row, int col) {
279        switch (col) {
280            case BUILD_COLUMN:
281            case BUILDBOX_COLUMN:
282            case ACTION_COLUMN:
283            case EDIT_COLUMN:
284                return true;
285            default:
286                return false;
287        }
288    }
289
290    @Override
291    public Object getValueAt(int row, int col) {
292        if (row >= getRowCount()) {
293            return "ERROR row " + row; // NOI18N
294        }
295        Train train = getTrainByRow(row);
296        if (train == null) {
297            return "ERROR train unknown " + row; // NOI18N
298        }
299        switch (col) {
300            case ID_COLUMN:
301                return Integer.parseInt(train.getId());
302            case TIME_COLUMN:
303                return train.getDepartureTime();
304            case NAME_COLUMN:
305                return train.getIconName();
306            case DESCRIPTION_COLUMN:
307                return train.getDescription();
308            case BUILDBOX_COLUMN:
309                return Boolean.valueOf(train.isBuildEnabled());
310            case BUILT_COLUMN:
311                return getBuiltString(train);
312            case CAR_ROAD_COLUMN:
313                return getModifiedString(train.getCarRoadNames().length, train.getCarRoadOption().equals(Train.ALL_ROADS),
314                        train.getCarRoadOption().equals(Train.INCLUDE_ROADS));
315            case CABOOSE_ROAD_COLUMN:
316                return getModifiedString(train.getCabooseRoadNames().length,
317                        train.getCabooseRoadOption().equals(Train.ALL_ROADS),
318                        train.getCabooseRoadOption().equals(Train.INCLUDE_ROADS));
319            case LOCO_ROAD_COLUMN:
320                return getModifiedString(train.getLocoRoadNames().length, train.getLocoRoadOption().equals(Train.ALL_ROADS),
321                        train.getLocoRoadOption().equals(Train.INCLUDE_ROADS));
322            case LOAD_COLUMN:
323                return getModifiedString(train.getLoadNames().length, train.getLoadOption().equals(Train.ALL_LOADS),
324                        train.getLoadOption().equals(Train.INCLUDE_LOADS));
325            case OWNER_COLUMN:
326                return getModifiedString(train.getOwnerNames().length, train.getOwnerOption().equals(Train.ALL_OWNERS),
327                        train.getOwnerOption().equals(Train.INCLUDE_OWNERS));
328            case ROUTE_COLUMN:
329                return train.getTrainRouteName();
330            case DEPARTS_COLUMN: {
331                if (train.getDepartureTrack() == null) {
332                    return train.getTrainDepartsName();
333                } else {
334                    return train.getTrainDepartsName() + " (" + train.getDepartureTrack().getName() + ")";
335                }
336            }
337            case CURRENT_COLUMN:
338                return train.getCurrentLocationName();
339            case TERMINATES_COLUMN: {
340                if (train.getTerminationTrack() == null) {
341                    return train.getTrainTerminatesName();
342                } else {
343                    return train.getTrainTerminatesName() + " (" + train.getTerminationTrack().getName() + ")";
344                }
345            }
346            case CARS_COLUMN:
347                return train.getNumberCarsInTrain();
348            case STATUS_COLUMN:
349                return train.getStatus();
350            case BUILD_COLUMN: {
351                if (train.isBuilt()) {
352                    if (Setup.isGenerateCsvManifestEnabled() && trainManager.isOpenFileEnabled()) {
353                        setToolTip(Bundle.getMessage("OpenTrainTip",
354                                train.getName()), row, col);
355                        return Bundle.getMessage("OpenFile");
356                    }
357                    if (Setup.isGenerateCsvManifestEnabled() && trainManager.isRunFileEnabled()) {
358                        setToolTip(Bundle.getMessage("RunTrainTip",
359                                train.getName()), row, col);
360                        return Bundle.getMessage("RunFile");
361                    }
362                    setToolTip(Bundle.getMessage("PrintTrainTip"), row, col);
363                    if (trainManager.isPrintPreviewEnabled()) {
364                        return Bundle.getMessage("Preview");
365                    } else if (train.isPrinted()) {
366                        return Bundle.getMessage("Printed");
367                    } else {
368                        return Bundle.getMessage("Print");
369                    }
370                }
371                setToolTip(Bundle.getMessage("BuildTrainTip", train.getName()),
372                        row, col);
373                return Bundle.getMessage("Build");
374            }
375            case ACTION_COLUMN: {
376                if (train.isBuildFailed()) {
377                    return Bundle.getMessage("Report");
378                }
379                if (train.getCurrentRouteLocation() == train.getTrainTerminatesRouteLocation() &&
380                        trainManager.getTrainsFrameTrainAction().equals(TrainsTableFrame.MOVE)) {
381                    return Bundle.getMessage("Terminate");
382                }
383                return trainManager.getTrainsFrameTrainAction();
384            }
385            case EDIT_COLUMN:
386                return Bundle.getMessage("ButtonEdit");
387            default:
388                return "unknown " + col; // NOI18N
389        }
390    }
391
392    private void setToolTip(String text, int row, int col) {
393        XTableColumnModel tcm = (XTableColumnModel) _table.getColumnModel();
394        ButtonRenderer buttonRenderer = (ButtonRenderer) tcm.getColumnByModelIndex(col).getCellRenderer();
395        if (buttonRenderer != null) {
396            buttonRenderer.setToolTipText(text);
397        }
398    }
399
400    private String getBuiltString(Train train) {
401        if (!train.getBuiltStartYear().equals(Train.NONE) && train.getBuiltEndYear().equals(Train.NONE)) {
402            return "A " + train.getBuiltStartYear();
403        }
404        if (train.getBuiltStartYear().equals(Train.NONE) && !train.getBuiltEndYear().equals(Train.NONE)) {
405            return "B " + train.getBuiltEndYear();
406        }
407        if (!train.getBuiltStartYear().equals(Train.NONE) && !train.getBuiltEndYear().equals(Train.NONE)) {
408            return "R " + train.getBuiltStartYear() + ":" + train.getBuiltEndYear();
409        }
410        return "";
411    }
412
413    private String getModifiedString(int number, boolean all, boolean accept) {
414        if (all) {
415            return "";
416        }
417        if (accept) {
418            return "A " + Integer.toString(number); // NOI18N
419        }
420        return "E " + Integer.toString(number); // NOI18N
421    }
422
423    @Override
424    public void setValueAt(Object value, int row, int col) {
425        switch (col) {
426            case EDIT_COLUMN:
427                editTrain(row);
428                break;
429            case BUILD_COLUMN:
430                buildTrain(row);
431                break;
432            case ACTION_COLUMN:
433                actionTrain(row);
434                break;
435            case BUILDBOX_COLUMN: {
436                Train train = getTrainByRow(row);
437                train.setBuildEnabled(((Boolean) value).booleanValue());
438                break;
439            }
440            default:
441                break;
442        }
443    }
444
445    public Color getRowColor(int row) {
446        Train train = getTrainByRow(row);
447        return train.getTableRowColor();
448    }
449
450    TrainEditFrame tef = null;
451
452    private void editTrain(int row) {
453        if (tef != null) {
454            tef.dispose();
455        }
456        // use invokeLater so new window appears on top
457        SwingUtilities.invokeLater(() -> {
458            Train train = getTrainByRow(row);
459            log.debug("Edit train ({})", train.getName());
460            tef = new TrainEditFrame(train);
461        });
462    }
463
464    Thread build;
465
466    private void buildTrain(int row) {
467        final Train train = getTrainByRow(row);
468        if (!train.isBuilt()) {
469            // only one train build at a time
470            if (build != null && build.isAlive()) {
471                return;
472            }
473            // use a thread to allow table updates during build
474            build = jmri.util.ThreadingUtil.newThread(new Runnable() {
475                @Override
476                public void run() {
477                    train.build();
478                }
479            });
480            build.setName("Build Train (" + train.getName() + ")"); // NOI18N
481            build.start();
482            // print build report, print manifest, run or open file
483        } else {
484            if (trainManager.isBuildReportEnabled()) {
485                train.printBuildReport();
486            }
487            if (Setup.isGenerateCsvManifestEnabled() && trainManager.isOpenFileEnabled()) {
488                train.openFile();
489            } else if (Setup.isGenerateCsvManifestEnabled() && trainManager.isRunFileEnabled()) {
490                train.runFile();
491            } else {
492                if (!train.printManifestIfBuilt()) {
493                    log.debug("Manifest file for train ({}) not found", train.getName());
494                    int result = JmriJOptionPane.showConfirmDialog(null,
495                            Bundle.getMessage("TrainManifestFileMissing",
496                                    train.getName()),
497                            Bundle.getMessage("TrainManifestFileError"), JmriJOptionPane.YES_NO_OPTION);
498                    if (result == JmriJOptionPane.YES_OPTION) {
499                        train.setModified(true);
500                        if (!train.printManifestIfBuilt()) {
501                            log.error("Unable to create manifest for train ({})", train.getName());
502                        }
503                    }
504                }
505            }
506        }
507    }
508
509    // one of five buttons: Report, Move, Reset, Conductor or Terminate
510    private void actionTrain(int row) {
511        // no actions while a train is being built
512        if (build != null && build.isAlive()) {
513            return;
514        }
515        Train train = getTrainByRow(row);
516        // move button becomes report if failure
517        if (train.isBuildFailed()) {
518            train.printBuildReport();
519        } else if (trainManager.getTrainsFrameTrainAction().equals(TrainsTableFrame.RESET)) {
520            log.debug("Reset train ({})", train.getName());
521            // check to see if departure track was reused
522            if (checkDepartureTrack(train)) {
523                log.debug("Train is departing staging that already has inbound cars");
524                JmriJOptionPane.showMessageDialog(null,
525                        Bundle.getMessage("StagingTrackUsed",
526                                train.getDepartureTrack().getName()),
527                        Bundle.getMessage("CanNotResetTrain"), JmriJOptionPane.INFORMATION_MESSAGE);
528            } else if (!train.reset()) {
529                JmriJOptionPane.showMessageDialog(null,
530                        Bundle.getMessage("TrainIsInRoute",
531                                train.getTrainTerminatesName()),
532                        Bundle.getMessage("CanNotResetTrain"), JmriJOptionPane.ERROR_MESSAGE);
533            }
534        } else if (!train.isBuilt()) {
535            int reply = JmriJOptionPane.showOptionDialog(null,
536                    Bundle.getMessage("TrainNeedsBuild", train.getName()),
537                    Bundle.getMessage("CanNotPerformAction"), JmriJOptionPane.NO_OPTION,
538                    JmriJOptionPane.INFORMATION_MESSAGE, null,
539                    new Object[]{Bundle.getMessage("ButtonOK"), Bundle.getMessage("Build")}, null);
540            if (reply == 1) {
541                train.build();
542            }
543        } else if (train.isBuilt() && trainManager.getTrainsFrameTrainAction().equals(TrainsTableFrame.MOVE)) {
544            log.debug("Move train ({})", train.getName());
545            train.move();
546        } else if (train.isBuilt() && trainManager.getTrainsFrameTrainAction().equals(TrainsTableFrame.TERMINATE)) {
547            log.debug("Terminate train ({})", train.getName());
548            int status = JmriJOptionPane.showConfirmDialog(null,
549                    Bundle.getMessage("TerminateTrain",
550                            train.getName(), train.getDescription()),
551                    Bundle.getMessage("DoYouWantToTermiate", train.getName()),
552                    JmriJOptionPane.YES_NO_OPTION);
553            if (status == JmriJOptionPane.YES_OPTION) {
554                train.terminate();
555            }
556        } else if (train.isBuilt() && trainManager.getTrainsFrameTrainAction().equals(TrainsTableFrame.CONDUCTOR)) {
557            log.debug("Enable conductor for train ({})", train.getName());
558            launchConductor(train);
559        }
560    }
561
562    /*
563     * Check to see if the departure track in staging has been taken by another
564     * train. return true if track has been allocated to another train.
565     */
566    private boolean checkDepartureTrack(Train train) {
567        return (Setup.isStagingTrackImmediatelyAvail() &&
568                !train.isTrainEnRoute() &&
569                train.getDepartureTrack() != null &&
570                train.getDepartureTrack().isStaging() &&
571                train.getDepartureTrack() != train.getTerminationTrack() &&
572                train.getDepartureTrack().getIgnoreUsedLengthPercentage() == Track.IGNORE_0 &&
573                train.getDepartureTrack().getDropRS() > 0);
574    }
575
576    private static Hashtable<String, TrainConductorFrame> _trainConductorHashTable = new Hashtable<>();
577
578    private void launchConductor(Train train) {
579        // use invokeLater so new window appears on top
580        SwingUtilities.invokeLater(() -> {
581            TrainConductorFrame f = _trainConductorHashTable.get(train.getId());
582            // create a copy train frame
583            if (f == null || !f.isVisible()) {
584                f = new TrainConductorFrame(train);
585                _trainConductorHashTable.put(train.getId(), f);
586            } else {
587                f.setExtendedState(Frame.NORMAL);
588            }
589            f.setVisible(true); // this also brings the frame into focus
590        });
591    }
592
593    @Override
594    public void propertyChange(PropertyChangeEvent e) {
595        if (Control.SHOW_PROPERTY) {
596            log.debug("Property change {} old: {} new: {}", e.getPropertyName(), e.getOldValue(), e.getNewValue()); // NOI18N
597        }
598        if (e.getPropertyName().equals(Train.BUILT_YEAR_CHANGED_PROPERTY) ||
599                e.getPropertyName().equals(Train.ROADS_CHANGED_PROPERTY) ||
600                e.getPropertyName().equals(Train.LOADS_CHANGED_PROPERTY) ||
601                e.getPropertyName().equals(Train.OWNERS_CHANGED_PROPERTY)) {
602            updateColumnVisible();
603        }
604        if (e.getPropertyName().equals(TrainManager.LISTLENGTH_CHANGED_PROPERTY) ||
605                e.getPropertyName().equals(TrainManager.PRINTPREVIEW_CHANGED_PROPERTY) ||
606                e.getPropertyName().equals(TrainManager.OPEN_FILE_CHANGED_PROPERTY) ||
607                e.getPropertyName().equals(TrainManager.RUN_FILE_CHANGED_PROPERTY) ||
608                e.getPropertyName().equals(Setup.MANIFEST_CSV_PROPERTY_CHANGE) ||
609                e.getPropertyName().equals(TrainManager.TRAIN_ACTION_CHANGED_PROPERTY) ||
610                e.getPropertyName().equals(Train.DEPARTURETIME_CHANGED_PROPERTY) ||
611                (e.getPropertyName().equals(Train.BUILD_CHANGED_PROPERTY) && !isShowAll())) {
612            SwingUtilities.invokeLater(() -> {
613                updateList();
614                fireTableDataChanged();
615            });
616        } else if (e.getSource().getClass().equals(Train.class)) {
617            Train train = ((Train) e.getSource());
618            SwingUtilities.invokeLater(() -> {
619                int row = sysList.indexOf(train);
620                if (row >= 0 && _table != null) {
621                    fireTableRowsUpdated(row, row);
622                    int viewRow = _table.convertRowIndexToView(row);
623                    log.debug("Scroll table to row: {}, property: {}", viewRow, e.getPropertyName());
624                    _table.scrollRectToVisible(_table.getCellRect(viewRow, 0, true));
625                }
626            });
627        }
628    }
629
630    private void removePropertyChangeTrains() {
631        for (Train train : trainManager.getTrainsByIdList()) {
632            train.removePropertyChangeListener(this);
633        }
634    }
635
636    private void addPropertyChangeTrains() {
637        for (Train train : trainManager.getTrainsByIdList()) {
638            train.addPropertyChangeListener(this);
639        }
640    }
641
642    public void dispose() {
643        if (tef != null) {
644            tef.dispose();
645        }
646        trainManager.removePropertyChangeListener(this);
647        Setup.getDefault().removePropertyChangeListener(this);
648        removePropertyChangeTrains();
649    }
650
651    class MyTableCellRenderer extends DefaultTableCellRenderer {
652
653        @Override
654        public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus,
655                int row, int column) {
656            Component component = super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);
657            if (!isSelected) {
658                int modelRow = table.convertRowIndexToModel(row);
659                // log.debug("View row: {} Column: {} Model row: {}", row, column, modelRow);
660                Color background = getRowColor(modelRow);
661                component.setBackground(background);
662                component.setForeground(getForegroundColor(background));
663            }
664            return component;
665        }
666
667        Color[] darkColors = { Color.BLACK, Color.BLUE, Color.GRAY, Color.RED, Color.MAGENTA };
668
669        /**
670         * Dark colors need white lettering
671         *
672         */
673        private Color getForegroundColor(Color background) {
674            if (background == null) {
675                return null;
676            }
677            for (Color color : darkColors) {
678                if (background == color) {
679                    return Color.WHITE;
680                }
681            }
682            return Color.BLACK; // all others get black lettering
683        }
684    }
685
686    private static final org.slf4j.Logger log = org.slf4j.LoggerFactory.getLogger(TrainsTableModel.class);
687}