Loading...
Logo
Processing Forum
There are surely a million different problems in this sketch, but go easy since it's my first try at processing (and java). If you find more than what this post is about, let me know. I'll appreciate any help.

Like a previous post, this sketch is meant to plot the position of a high altitude balloon transmitting it's position from an xbee. The string the xbee receives is saved using a serial capture to a file, and this sketch reads in each line using BufferedReader. The problem is the sketch tries to load the file when there's a partial string, leading to an error. Is there a way to force it to wait until the string has loaded?

Here's a sample of the transmitted data:
!!!47.6696,-71.4853,0.16,249.91,162.50,6
!!!47.6696,-71.4853,0.45,249.91,162.50,6

Here's the sketch, and the throw message is below:
Copy code
  1. BufferedReader reader;
  2. import googlemapper.*;
  3. import controlP5.*;
  4. double mapCenterLat = 42.66656;
  5. double mapCenterLon = -72.4831;
  6. int zoomLevel =10;
  7. String mapType = GoogleMapper.MAPTYPE_ROADMAP;
  8. int mapWidth=640;
  9. int mapHeight=480;
  10. String inBuffer;
  11. Boolean isGPSActive;
  12. ArrayList<Double> lat = new ArrayList<Double>();
  13. ArrayList<Double> lng = new ArrayList<Double>();
  14. Double newLat;
  15. Double newLng;
  16. float cent_lat = 0;
  17. float cent_lng = 0;
  18. int x;

  19. ControlP5 cp5;
  20. int zoomSlider=15;
  21. Slider abc;

  22. PImage map;
  23. GoogleMapper gMapper;

  24. public void setup() {
  25.   //start reading the file with fake info
  26.   reader = createReader("serialcapture.txt");    
  27.   //sets the window size to 640x480
  28.   size(640, 480);
  29.   smooth();
  30.   //creates new controls using the P5 Library
  31.   cp5 = new ControlP5(this);
  32.   cp5.addSlider("zoomSlider")
  33.     .setPosition(50, 100)
  34.       .setSize(40, 200)
  35.         .setRange(3, 21)
  36.           .setNumberOfTickMarks(18);
  37.   cp5.addButton("refresh")
  38.     .setValue(0)
  39.       .setPosition(50, 80)
  40.         .setSize(40, 19);
  41.   //sets the zoomlevel
  42.   zoomLevel=zoomSlider;
  43. }

  44. long interval = millis();

  45. //function to run when refresh button is pressed
  46. void refresh() {
  47.   loop();
  48.   println("refreshing map");
  49.   try {
  50.     interval = millis();
  51.     zoomLevel=zoomSlider;
  52.     if (cent_lat != 0) {
  53.       mapCenterLat = gMapper.y2lat(cent_lat);
  54.       mapCenterLon = gMapper.x2lon(cent_lng);
  55.     }
  56.     gMapper = new GoogleMapper(mapCenterLat, mapCenterLon, zoomLevel, mapType, mapWidth, mapHeight);
  57.     map = gMapper.getMap();
  58.     image(map, 0, 0);
  59.     lat.clear();
  60.     lng.clear();
  61.     reader.close();
  62.     reader = createReader("serialcapture.txt");    
  63.     inBuffer = reader.readLine();
  64.     drawcourse(inBuffer);
  65.     cent_lat = 0; 
  66.     cent_lng = 0;
  67.   } 
  68.   catch(Exception e) {
  69.     e.printStackTrace();
  70.   }
  71.   x=0;
  72. }

  73. void draw() {
  74.   inBuffer = null;
  75.   //reads the data from the serial and stores it in a buffer
  76.   if (mousePressed && keyPressed) {
  77.     if (key == 'c' || key == 'C') {
  78.       cent_lat = mouseY;
  79.       cent_lng = mouseX;
  80.     }
  81.   }

  82.     try {
  83.       inBuffer = reader.readLine();
  84.     }
  85.     catch (IOException e) {
  86.       e.printStackTrace();
  87.       inBuffer = null;
  88.       delay(250);
  89.     }
  90.   
  91.   if (inBuffer != null) {
  92.     drawcourse(inBuffer);
  93.   }
  94.   else return;
  95. }


  96. void drawcourse(String inBuffer) {
  97.   //*****display iteration in top left
  98.   background(0);
  99.   image(map, 0, 0);
  100.   fill(0);
  101.   textSize(16);
  102.   text("position: "+x, 10, 20);
  103.   x++;

  104.   inBuffer = inBuffer.substring(3);
  105.   //println(inBuffer);//for debugging inBuffer read
  106.   String[] op = inBuffer.split(",");

  107.   //only when we get valid data, split it to get coordinates
  108.   if (op.length==6) {
  109.     newLat = Double.valueOf(op[0]);  //get latitude
  110.     newLng = Double.valueOf(op[1]);  //get longitude
  111.     lat.add(newLat.doubleValue());
  112.     lng.add(newLng.doubleValue());
  113.     isGPSActive=true;
  114.   }
  115.   for (int i=1; i < lat.size()-1 ; i++)
  116.   {
  117.     if (lat.get(i)==null) return;
  118.     else {
  119.       float lat1 = (float) gMapper.lat2y(lat.get(i));
  120.       float lng1 = (float) gMapper.lon2x(lng.get(i));
  121.       float lat2 = (float) gMapper.lat2y(lat.get(i+1));
  122.       float lng2 = (float) gMapper.lon2x(lng.get(i+1));
  123.       stroke(0);
  124.       strokeWeight(2);
  125.       line(lng1, lat1, lng2, lat2);//draws the latest position on the screen
  126.       //ellipse(lng2, lat2, 5, 5);
  127.       //println(lat1+"-"+lng1+"|"+lat2+"-"+lng2);//debug output to see line positions
  128.     }
  129.   }
  130. }
Error message:
Copy code
  1. Exception in thread "Animation Thread" java.lang.StringIndexOutOfBoundsException: String index out of range: -1
  2. at java.lang.String.substring(String.java:1937)
  3. at java.lang.String.substring(String.java:1904)
  4. at googlemapper_testing__01.drawcourse(googlemapper_testing__01.java:135)
  5. at googlemapper_testing__01.draw(googlemapper_testing__01.java:120)
  6. at processing.core.PApplet.handleDraw(PApplet.java:2266)
  7. at processing.core.PGraphicsJava2D.requestDraw(PGraphicsJava2D.java:243)
  8. at processing.core.PApplet.run(PApplet.java:2140)
  9. at java.lang.Thread.run(Thread.java:680)

Replies(3)

General remark: put size() at the top of your setup().
Avoid doing I/O before it!

Are you sure it is an issue with reading a "partial string" (whatever it can be!).
Which line is highlighted? (since we cannot run your code. BTW, moved from Programming Questions, because of the requirements to run the sketch.)
Looks like you are trying to do a substring() with a negative index, so you better check the values you pass to this function.
Thanks PhiLho for the reply and moderation.

Line 115 was the offending line, which is what lead me to believe that it was bringing in a partial string before the capture was finished. I changed line 99 to the following and it seems to have solved the issue:
Copy code
  1. if (inBuffer != null && inBuffer.length()>20)
The sketch now seems to run fine, but I have one other general question. Does refresh automatically run before the draw loop starts because it's placed before draw? This could be a dumb question, but I'm learning.

Does refresh() automatically run before the draw() loop starts because it's placed before draw()?
Nope! All of your functions belong to the sketch itself.
And that's not what determines their order of execution call.

Processing always calls draw() 1st. Then any events triggered while draw() was running 
are queued up to happen only after it's finished and the screen refreshed!

Function refresh() is registered to library controlP5.
And it's also callbacked after draw(); just like any other event functions!