001package jmri.implementation;
002
003import java.util.*;
004
005import javax.annotation.Nonnull;
006
007import jmri.NamedBeanHandle;
008import jmri.Turnout;
009
010/**
011 * SignalMast implemented via Turnout objects.
012 * <p>
013 * A SignalMast that is built up using turnouts to control a specific
014 * appearance. System name specifies the creation information:
015 * <pre>
016 * IF$tsm:basic:one-searchlight(IT1)(IT2)
017 * </pre> The name is a colon-separated series of terms:
018 * <ul>
019 * <li>IF$tsm - defines signal masts of this type
020 * <li>basic - name of the signaling system
021 * <li>one-searchlight - name of the particular aspect map
022 * <li>(IT1)(IT2) - colon-separated list of names for Turnouts
023 * </ul>
024 *
025 * @author Bob Jacobsen Copyright (C) 2009, 2014
026 */
027public class TurnoutSignalMast extends AbstractSignalMast {
028
029    public TurnoutSignalMast(String systemName, String userName) {
030        super(systemName, userName);
031        configureFromName(systemName);
032    }
033
034    public TurnoutSignalMast(String systemName) {
035        super(systemName);
036        configureFromName(systemName);
037    }
038
039    private static final String THE_MAST_TYPE = "IF$tsm";
040
041    private void configureFromName(String systemName) {
042        // split out the basic information
043        String[] parts = systemName.split(":");
044        if (parts.length < 3) {
045            log.error("SignalMast system name needs at least three parts: {}", systemName);
046            throw new IllegalArgumentException("System name needs at least three parts: " + systemName);
047        }
048        if (!parts[0].equals(THE_MAST_TYPE)) {
049            log.warn("SignalMast system name should start with {} but is {}", THE_MAST_TYPE, systemName);
050        }
051        String system = parts[1];
052        String mast = parts[2];
053
054        mast = mast.substring(0, mast.indexOf("("));
055        setMastType(mast);
056
057        String tmp = parts[2].substring(parts[2].indexOf("($") + 2, parts[2].indexOf(")"));
058        try {
059            int autoNumber = Integer.parseInt(tmp);
060            if (autoNumber > getLastRef()) {
061                setLastRef(autoNumber);
062            }
063        } catch (NumberFormatException e) {
064            log.warn("Auto generated SystemName {} is not in the correct format", systemName);
065        }
066
067        configureSignalSystemDefinition(system);
068        configureAspectTable(system, mast);
069    }
070
071    @Override
072    public void setAspect(@Nonnull String aspect) {
073        // check it's a choice
074        if (!map.checkAspect(aspect)) {
075            // not a valid aspect
076            log.warn("attempting to set invalid aspect: {} on mast: {}", aspect, getDisplayName());
077            throw new IllegalArgumentException("attempting to set invalid aspect: "
078                + aspect + " on mast: " + getDisplayName());
079        } else if (disabledAspects.contains(aspect)) {
080            log.warn("attempting to set an aspect that has been disabled: {} on mast: {}", aspect, getDisplayName());
081            throw new IllegalArgumentException("attempting to set an aspect that has been disabled: "
082                + aspect + " on mast: " + getDisplayName());
083        }
084
085
086        if (getLit()) { // If the signalmast is lit, then send the commands to change the aspect.
087
088            // reset all states before setting this one, including this one
089            if (resetPreviousStates) {
090                // Clear all the current states, this will result in the signalmast going blank for a very short time.
091                for (Map.Entry<String, TurnoutAspect> entry : turnouts.entrySet()) {
092                    String appearance = entry.getKey();
093                    TurnoutAspect aspt = entry.getValue();
094                    if (!isAspectDisabled(appearance)) {
095                        int setState = Turnout.CLOSED;
096                        if (aspt.getTurnoutState() == Turnout.CLOSED) {
097                            setState = Turnout.THROWN;
098                        }
099                        if (aspt.getTurnout() != null ) {
100                            if (aspt.getTurnout().getKnownState() != setState) {
101                                aspt.getTurnout().setCommandedState(setState);
102                            }
103                        } else {
104                            log.error("Trying to reset \"{}\" on signal mast \"{}\" which has not been configured",
105                                appearance, getDisplayName());
106                        }
107                    }
108                }
109            }
110
111            // set the finel state if possible
112            if (turnouts.get(aspect) != null && turnouts.get(aspect).getTurnout() != null) {
113                Turnout turnToSet = turnouts.get(aspect).getTurnout();
114                int stateToSet = turnouts.get(aspect).getTurnoutState();
115                turnToSet.setCommandedState(stateToSet);
116            } else {
117                log.error("Trying to set \"{}\" on signal mast \"{}\" which has not been configured",
118                    aspect, getDisplayName());
119            }
120
121        } else {
122            log.debug("Mast set to unlit, will not send aspect change to hardware");
123        }
124        super.setAspect(aspect);
125    }
126
127    private TurnoutAspect unLit = null;
128
129    public void setUnLitTurnout(String turnoutName, int turnoutState) {
130        unLit = new TurnoutAspect(turnoutName, turnoutState);
131    }
132
133    public String getUnLitTurnoutName() {
134        if (unLit != null) {
135            return unLit.getTurnoutName();
136        }
137        return null;
138    }
139
140    public Turnout getUnLitTurnout() {
141        if (unLit != null) {
142            return unLit.getTurnout();
143        }
144        return null;
145    }
146
147    public int getUnLitTurnoutState() {
148        if (unLit != null) {
149            return unLit.getTurnoutState();
150        }
151        return -1;
152    }
153
154    @Override
155    public void setLit(boolean newLit) {
156        if (!allowUnLit() || newLit == getLit()) {
157            return;
158        }
159        super.setLit(newLit);
160        if (newLit) {
161            // This will force the signalmast to send out the commands to set the aspect again.
162            String litAspect = getAspect();
163            if (litAspect != null ) {
164                setAspect(litAspect);
165            }
166        } else {
167            if (unLit != null) {
168                // there is a specific unlit output defined
169                Turnout t = unLit.getTurnout();
170                if (t != null && t.getKnownState() != getUnLitTurnoutState()) {
171                    t.setCommandedState(getUnLitTurnoutState());
172                }
173            } else {
174                // turn everything off
175                for (TurnoutAspect tAspect : turnouts.values()) {
176                    int setState = Turnout.CLOSED;
177                    if (tAspect.getTurnoutState() == Turnout.CLOSED) {
178                        setState = Turnout.THROWN;
179                    }
180                    if (tAspect.getTurnout().getKnownState() != setState) {
181                        tAspect.getTurnout().setCommandedState(setState);
182                    }
183                }
184            }
185        }
186    }
187
188    public String getTurnoutName(String appearance) {
189        TurnoutAspect tAspect = turnouts.get(appearance);
190        if (tAspect != null) {
191            return tAspect.getTurnoutName();
192        }
193        return "";
194    }
195
196    public int getTurnoutState(String appearance) {
197        TurnoutAspect tAspect = turnouts.get(appearance);
198        if (tAspect != null) {
199            return tAspect.getTurnoutState();
200        }
201        return -1;
202    }
203
204    public void setTurnout(String appearance, String turn, int state) {
205        if (turnouts.containsKey(appearance)) {
206            log.debug("Appearance {} is already defined so will override", appearance);
207            turnouts.remove(appearance);
208        }
209        turnouts.put(appearance, new TurnoutAspect(turn, state));
210    }
211
212    private final HashMap<String, TurnoutAspect> turnouts = new HashMap<>();
213
214    private boolean resetPreviousStates = false;
215
216    /**
217     * If the signal mast driver requires the previous state to be cleared down
218     * before the next state is set.
219     *
220     * @param boo true if prior states should be cleared; false otherwise
221     */
222    public void resetPreviousStates(boolean boo) {
223        resetPreviousStates = boo;
224    }
225
226    public boolean resetPreviousStates() {
227        return resetPreviousStates;
228    }
229
230    private static class TurnoutAspect {
231
232        NamedBeanHandle<Turnout> namedTurnout;
233        int state;
234
235        TurnoutAspect(String turnoutName, int turnoutState) {
236            if (turnoutName != null && !turnoutName.isEmpty()) {
237                state = turnoutState;
238                Turnout turn = jmri.InstanceManager.turnoutManagerInstance().getTurnout(turnoutName);
239                if (turn == null) {
240                    log.error("TurnoutAspect couldn't locate turnout {}", turnoutName);
241                    return;
242                }
243                namedTurnout = jmri.InstanceManager.getDefault(jmri.NamedBeanHandleManager.class)
244                    .getNamedBeanHandle(turnoutName, turn);
245            }
246        }
247
248        Turnout getTurnout() {
249            if (namedTurnout == null) {
250                return null;
251            }
252            return namedTurnout.getBean();
253        }
254
255        String getTurnoutName() {
256            if (namedTurnout == null) {
257                return null;
258            }
259            return namedTurnout.getName();
260        }
261
262        int getTurnoutState() {
263            return state;
264        }
265    }
266
267    public boolean isTurnoutUsed(Turnout t) {
268        for (TurnoutAspect ta : turnouts.values()) {
269            if (t.equals(ta.getTurnout())) {
270                return true;
271            }
272        }
273        return t.equals(getUnLitTurnout());
274    }
275
276    public List<NamedBeanHandle<Turnout>> getHeadsUsed() {
277        return new ArrayList<>();
278    }
279
280    /**
281     *
282     * @param newVal for ordinal of all TurnoutSignalMasts in use
283     */
284    protected static void setLastRef(int newVal) {
285        lastRef = newVal;
286    }
287
288    /**
289     * @return highest ordinal of all TurnoutSignalMasts in use
290     */
291    public static int getLastRef() {
292        return lastRef;
293    }
294
295    /**
296     * Ordinal of all TurnoutSignalMasts to create unique system name.
297     */
298    protected static volatile int lastRef = 0;
299    // TODO narrow access to static, once jmri/jmrit/beantable/signalmast/TurnoutSignalMastAddPane uses setLastRef(n)
300    //private static volatile int lastRef = 0;
301
302    @Override
303    public void vetoableChange(java.beans.PropertyChangeEvent evt) throws java.beans.PropertyVetoException {
304        if (jmri.Manager.PROPERTY_CAN_DELETE.equals(evt.getPropertyName())
305                && (evt.getOldValue() instanceof Turnout) && (isTurnoutUsed((Turnout) evt.getOldValue()))) {
306
307            java.beans.PropertyChangeEvent e = new java.beans.PropertyChangeEvent(this,
308                jmri.Manager.PROPERTY_DO_NOT_DELETE, null, null);
309            throw new java.beans.PropertyVetoException(Bundle.getMessage("InUseTurnoutSignalMastVeto",
310                getDisplayName()), e);
311        }
312    }
313
314    private static final org.slf4j.Logger log = org.slf4j.LoggerFactory.getLogger(TurnoutSignalMast.class);
315
316}