/java/src/edu/berkeley/c3uv/trackers/impl/LeadingSensorPoint.java

https://bitbucket.org/c3uv_admin/c3uvorca · Java · 148 lines · 112 code · 25 blank · 11 comment · 34 complexity · 3c310b0ea2c763dfe6495b0cb769e569 MD5 · raw file

  1. package edu.berkeley.c3uv.trackers.impl;
  2. import java.util.Map;
  3. import org.apache.log4j.Logger;
  4. import org.joda.time.Instant;
  5. import edu.berkeley.c3uv.PPLI;
  6. import edu.berkeley.c3uv.Path;
  7. import edu.berkeley.c3uv.Point;
  8. import edu.berkeley.c3uv.TelemetryStore;
  9. import edu.berkeley.c3uv.math.Geo;
  10. import edu.berkeley.c3uv.trackers.SensorTracker;
  11. public class LeadingSensorPoint extends SensorTracker {
  12. private static Logger logger = Logger.getLogger(LeadingSensorPoint.class);
  13. private final double timeStep;
  14. private final double distanceThreshold;
  15. private final double leadDistance;
  16. private long lastSendingTime = 0;
  17. private double leadAngle = 0;
  18. public LeadingSensorPoint(TelemetryStore store,Map<String, String> properties) {
  19. super(store,properties);
  20. String timeStepString = properties.get("Tracker.Sensor.Lead.TimeStep");
  21. if (timeStepString!=null && !timeStepString.isEmpty()) {
  22. timeStep = Double.parseDouble(timeStepString);
  23. } else {
  24. timeStep = 1;
  25. }
  26. String distanceThresholdString = properties.get("Tracker.Sensor.Lead.DistanceThreshold");
  27. if (distanceThresholdString!=null && !distanceThresholdString.isEmpty()) {
  28. distanceThreshold = Double.parseDouble(distanceThresholdString);
  29. } else {
  30. distanceThreshold = 200;
  31. }
  32. String leadDistanceThresholdString = properties.get("Tracker.Sensor.Lead.LeadDistance");
  33. if (leadDistanceThresholdString!=null && !leadDistanceThresholdString.isEmpty()) {
  34. leadDistance = Double.parseDouble(leadDistanceThresholdString);
  35. } else {
  36. leadDistance = 50;
  37. }
  38. }
  39. @Override
  40. public Point getSensorTarget(Path path) {
  41. if (path == null || path.getWaypoints().isEmpty()) {
  42. return null;
  43. }
  44. if (new Instant().getMillis() - lastSendingTime <= minSendingInterval) {
  45. // System.out.println("SKIPPING");
  46. return null;
  47. }
  48. if (telemStore.getTelemetry() == null
  49. || telemStore.getTelemetry().getPpli() == null
  50. || telemStore.getTelemetry().getPpli().getPoint() == null) {
  51. logger.warn("No telemetry");
  52. return path.getWaypoints().get(0);
  53. }
  54. double altitude;
  55. if (telemStore.getTelemetry() != null
  56. && telemStore.getTelemetry().getPpli() != null
  57. && telemStore.getTelemetry().getPpli().getPoint() != null) {
  58. altitude = telemStore.getTelemetry().getPpli().getPoint().getAltitudeMSL()
  59. - telemStore.getTelemetry().getPpli().getAltitudeAGL();
  60. } else {
  61. altitude = path.getWaypoints().get(0).getAltitudeMSL();
  62. }
  63. PPLI ppli = telemStore.getTelemetry().getPpli();
  64. Point aircraftPoint = ppli.getPoint();
  65. Point myLocation = new Point(aircraftPoint.getLatitude(),
  66. aircraftPoint.getLongitude(),altitude,
  67. aircraftPoint.getCircularError(),aircraftPoint.getLinearError());
  68. double groundSpeed = ppli.getGroundSpeed();
  69. double turnRate = 9.81/groundSpeed * Math.tan(ppli.getAttitude().getRoll());
  70. double zeta = ppli.getGroundTrack() + ppli.getAttitude().getYaw();
  71. double zetanew = zeta + turnRate*timeStep;
  72. double n = groundSpeed/turnRate * (Math.sin(zetanew) - Math.sin(zeta));
  73. double e = groundSpeed/turnRate * (Math.cos(zeta) - Math.cos(zetanew));
  74. Geo.NorthEastDown ned = new Geo.NorthEastDown(n,e,0);
  75. // Geo.DistBearing db = new Geo.DistBearing(ppli.getGroundSpeed()*timeStep,ppli.getGroundTrack());
  76. Point origin = Geo.nedToLatLon(ned,myLocation);
  77. // First, try the closest point to me one timestep in the future
  78. double minDist = 45000 * 1000; //greater than circumference of the earth
  79. int minIndex = 0;
  80. for (int i=path.getWaypoints().size()-1;i>=0;i--) {
  81. double dist = Geo.getDistance(origin, path.getWaypoints().get(i));
  82. if (dist < minDist) {
  83. minIndex = i;
  84. minDist = dist;
  85. }
  86. }
  87. // If the closest point then is too far away, fall back to closest to me now
  88. if (minDist > distanceThreshold) {
  89. //System.err.println("Using my point");
  90. origin = myLocation;
  91. minDist = 45000 * 1000; //greater than circumference of the earth
  92. minIndex = 0;
  93. for (int i=0;i<path.getWaypoints().size();i++) {
  94. double dist = Geo.getDistance(origin, path.getWaypoints().get(i));
  95. if (dist < minDist) {
  96. minIndex = i;
  97. minDist = dist;
  98. }
  99. }
  100. } else {
  101. //System.err.println("Using lead point");
  102. }
  103. Point target;
  104. // If still too far away, look a little bit ahead
  105. if (minDist <= distanceThreshold) {
  106. target = path.getWaypoints().get(minIndex);
  107. } else {
  108. //System.err.println("Too far away");
  109. double rollFactor = ppli.getAttitude().getRoll() / (Math.PI/6.);
  110. //System.out.println("roll factor: "+rollFactor);
  111. Geo.DistBearing db2 = new Geo.DistBearing(leadDistance,ppli.getGroundTrack()+leadAngle*rollFactor);
  112. target = Geo.bearingDistToLatLon(myLocation, db2);
  113. }
  114. Geo.DistBearing finalDb = Geo.latLonToDistBearing(myLocation, target);
  115. logger.trace("\n Me: "+myLocation+" Target: "+target + "\n" +
  116. " Direction: "+Math.toDegrees(ppli.getGroundTrack())+" DB: "+finalDb);
  117. //TODO: remove
  118. // logger.warn("Setting SensorTracker.target");
  119. SensorTracker.setTarget(target);
  120. lastSendingTime = new Instant().getMillis();
  121. return target;
  122. }
  123. }