View Javadoc
1   package org.opentrafficsim.road.gtu.lane.plan.operational;
2   
3   import java.io.Serializable;
4   
5   import org.djunits.value.vdouble.scalar.Acceleration;
6   import org.djunits.value.vdouble.scalar.Duration;
7   import org.djunits.value.vdouble.scalar.Length;
8   import org.djutils.exceptions.Throw;
9   import org.opentrafficsim.core.gtu.GTUException;
10  import org.opentrafficsim.core.gtu.TurnIndicatorIntent;
11  import org.opentrafficsim.core.gtu.TurnIndicatorStatus;
12  import org.opentrafficsim.core.network.LateralDirectionality;
13  import org.opentrafficsim.road.gtu.lane.LaneBasedGTU;
14  
15  import nl.tudelft.simulation.dsol.logger.SimLogger;
16  
17  /**
18   * Simplified plan containing an acceleration value and possible lane change direction.
19   * <p>
20   * Copyright (c) 2013-2020 Delft University of Technology, PO Box 5, 2600 AA, Delft, the Netherlands. All rights reserved. <br>
21   * BSD-style license. See <a href="http://opentrafficsim.org/docs/current/license.html">OpenTrafficSim License</a>.
22   * <p>
23   * @version $Revision$, $LastChangedDate$, by $Author$, initial version Jul 26, 2016 <br>
24   * @author <a href="http://www.tbm.tudelft.nl/averbraeck">Alexander Verbraeck</a>
25   * @author <a href="http://www.tudelft.nl/pknoppers">Peter Knoppers</a>
26   * @author <a href="http://www.transport.citg.tudelft.nl">Wouter Schakel</a>
27   */
28  
29  public class SimpleOperationalPlan implements Serializable
30  {
31  
32      /** */
33      private static final long serialVersionUID = 20160811L;
34  
35      /** Acceleration. */
36      private Acceleration acceleration;
37  
38      /** Lane change direction. */
39      private final LateralDirectionality laneChangeDirection;
40  
41      /** Indicator intent. */
42      private TurnIndicatorIntent indicatorIntent = TurnIndicatorIntent.NONE;
43  
44      /** Distance to object causing turn indicator intent. */
45      private Length indicatorObjectDistance = null;
46  
47      /** Duration of the plan. */
48      private final Duration duration;
49  
50      /**
51       * @param acceleration Acceleration; acceleration
52       * @param duration Duration; duration
53       */
54      public SimpleOperationalPlan(final Acceleration acceleration, final Duration duration)
55      {
56          this(acceleration, duration, LateralDirectionality.NONE);
57      }
58  
59      /**
60       * @param acceleration Acceleration; acceleration
61       * @param duration Duration; duration
62       * @param laneChangeDirection LateralDirectionality; lane change direction, may be {@code null}.
63       */
64      public SimpleOperationalPlan(final Acceleration acceleration, final Duration duration,
65              final LateralDirectionality laneChangeDirection)
66      {
67          Throw.whenNull(acceleration, "Acceleration may not be null.");
68          Throw.whenNull(duration, "Duration may not be null.");
69          Throw.whenNull(laneChangeDirection, "Lane change direction may not be null.");
70          checkAcceleration(acceleration);
71          this.acceleration = Acceleration.max(Acceleration.instantiateSI(-100.0), acceleration);
72          this.duration = duration;
73          this.laneChangeDirection = laneChangeDirection;
74      }
75  
76      /**
77       * @return acceleration.
78       */
79      public final Acceleration getAcceleration()
80      {
81          return this.acceleration;
82      }
83      
84      /**
85       * Sets acceleration. 
86       * @param acceleration Acceleration; acceleration
87       */
88      public final void setAcceleration(final Acceleration acceleration)
89      {
90          checkAcceleration(acceleration);
91          this.acceleration = acceleration;
92      }
93  
94      /**
95       * @return duration.
96       */
97      public Duration getDuration()
98      {
99          return this.duration;
100     }
101 
102     /**
103      * @return if lane change.
104      */
105     public final boolean isLaneChange()
106     {
107         return this.laneChangeDirection != LateralDirectionality.NONE;
108     }
109 
110     /**
111      * @return laneChangeDirection, may be NONE if no lane change.
112      */
113     public final LateralDirectionality getLaneChangeDirection()
114     {
115         return this.laneChangeDirection;
116     }
117 
118     /**
119      * Set minimum of current and given acceleration.
120      * @param a Acceleration; acceleration to set if lower than current acceleration
121      */
122     public final void minimizeAcceleration(final Acceleration a)
123     {
124         checkAcceleration(a);
125         this.acceleration = Acceleration.max(Acceleration.instantiateSI(-100.0), Acceleration.min(this.acceleration, a)); // XXX: AV
126     }
127 
128     /**
129      * Check acceleration level.
130      * @param a Acceleration; acceleration
131      */
132     private void checkAcceleration(final Acceleration a)
133     {
134         if (a.equals(Acceleration.NEGATIVE_INFINITY) || a.equals(Acceleration.NEG_MAXVALUE))
135         {
136             SimLogger.always().error("Model has calculated a negative infinite or negative max value acceleration."); // XXX: AV
137         }
138     }
139 
140     /**
141      * @return indicatorIntent.
142      */
143     public final TurnIndicatorIntent getIndicatorIntent()
144     {
145         return this.indicatorIntent;
146     }
147 
148     /**
149      * Set left indicator intent. Any intent given with distance overrules this intent.
150      */
151     public final void setIndicatorIntentLeft()
152     {
153         if (this.indicatorObjectDistance != null)
154         {
155             return;
156         }
157         if (this.indicatorIntent.isRight())
158         {
159             this.indicatorIntent = TurnIndicatorIntent.CONFLICTING;
160         }
161         else
162         {
163             this.indicatorIntent = TurnIndicatorIntent.LEFT;
164         }
165     }
166 
167     /**
168      * Set right indicator intent. Any intent given with distance overrules this intent.
169      */
170     public final void setIndicatorIntentRight()
171     {
172         if (this.indicatorObjectDistance != null)
173         {
174             return;
175         }
176         if (this.indicatorIntent.isLeft())
177         {
178             this.indicatorIntent = TurnIndicatorIntent.CONFLICTING;
179         }
180         else
181         {
182             this.indicatorIntent = TurnIndicatorIntent.RIGHT;
183         }
184     }
185 
186     /**
187      * Set left indicator intent. Intent with smallest provided distance has priority.
188      * @param distance Length; distance to object pertaining to the turn indicator intent
189      */
190     public final void setIndicatorIntentLeft(final Length distance)
191     {
192         if (compareAndIgnore(distance))
193         {
194             return;
195         }
196         if (this.indicatorIntent.isRight())
197         {
198             this.indicatorIntent = TurnIndicatorIntent.CONFLICTING;
199         }
200         else
201         {
202             this.indicatorIntent = TurnIndicatorIntent.LEFT;
203         }
204 
205     }
206 
207     /**
208      * Set right indicator intent. Intent with smallest provided distance has priority.
209      * @param distance Length; distance to object pertaining to the turn indicator intent
210      */
211     public final void setIndicatorIntentRight(final Length distance)
212     {
213         if (compareAndIgnore(distance))
214         {
215             return;
216         }
217         if (this.indicatorIntent.isLeft())
218         {
219             this.indicatorIntent = TurnIndicatorIntent.CONFLICTING;
220         }
221         else
222         {
223             this.indicatorIntent = TurnIndicatorIntent.RIGHT;
224         }
225     }
226 
227     /**
228      * Compares distances and returns whether the given distance (and intent) can be ignored.
229      * @param distance Length; distance to object of intent
230      * @return whether the given distance can be ignored
231      */
232     private boolean compareAndIgnore(final Length distance)
233     {
234         if (this.indicatorObjectDistance != null)
235         {
236             if (this.indicatorObjectDistance.lt(distance))
237             {
238                 // disregard input; the intent from larger distance
239                 return true;
240             }
241             if (this.indicatorObjectDistance.gt(distance))
242             {
243                 // disregard existing; the intent from larger distance
244                 this.indicatorIntent = TurnIndicatorIntent.NONE; // prevents a set to CONFLICTING
245             }
246         }
247         else
248         {
249             // disregard existing; the intent without distance
250             this.indicatorIntent = TurnIndicatorIntent.NONE; // prevents a set to CONFLICTING
251         }
252         return false;
253     }
254 
255     /** {@inheritDoc} */
256     @Override
257     @SuppressWarnings("checkstyle:designforextension")
258     public String toString()
259     {
260         return "SimpleOperationalPlan [Acceleration=" + this.acceleration + ", change=" + this.laneChangeDirection
261                 + ", indicator intent=" + this.indicatorIntent + "]";
262     }
263 
264     /**
265      * @param gtu LaneBasedGTU; LaneBasedGTU to set the indicator on
266      * @throws GTUException if GTU does not support the indicator
267      */
268     public final void setTurnIndicator(final LaneBasedGTU gtu) throws GTUException
269     {
270         if (this.indicatorIntent.isLeft())
271         {
272             gtu.setTurnIndicatorStatus(TurnIndicatorStatus.LEFT);
273         }
274         else if (this.indicatorIntent.isRight())
275         {
276             gtu.setTurnIndicatorStatus(TurnIndicatorStatus.RIGHT);
277         }
278         else
279         {
280             gtu.setTurnIndicatorStatus(TurnIndicatorStatus.NONE);
281         }
282     }
283 
284 }