Loading...
Logo
Processing Forum
I want to publish two 3D graphs on one segmented window but the problems is that the end sketch seems to be distorted. This method seems to work with 2D graphs but I cant get it to work for 3D graphs. I cant seem to figure out what the problem is so any help would be much appreciated. This is what I am trying to generate:-

but this is what I end up with:-



The first part of the sketch;
Copy code
  1.  
  2. import processing.opengl.*;
  3.  
  4. PVector[] vecs = new PVector[160];
  5. int dim = 250;
  6. float x;
  7. float y;
  8. float z;
  9. float maxX, maxY, maxZ;
  10. float minX, minY, minZ;
  11. float rotY;      // Amount of rotation around y-axis.
  12. float rotX;      // Amount of rotation around the x-axis.
  13. PFont font;

  14. void setup()
  15. {
  16.   size(500,500,P3D);
  17.   smooth();
  18.   setupSketch1();
  19.   setupSketch2();
  20. }

  21. void draw()
  22. {  
  23.   scale(.5,.5);
  24.   drawSketch1();
  25.   
  26.   translate(width,0);
  27.   drawSketch2();

  28. }


The second bit of the sketch;
Copy code
  1. void setupSketch1()
  2. {
  3.   font = createFont( "arial", 14 );
  4.   textFont(font);
  5.   
  6. Random gauss = new Random(0);

  7.  for (int i=0; i<vecs.length; i++)
  8.    {
  9.      x = (float)gauss.nextGaussian()*118.9533 + 261.3167;
  10.      y = (float)gauss.nextGaussian()*48.88477 + 133.6271;
  11.      z = (float)gauss.nextGaussian()*3318.683 + 906.6409;
  12.      vecs[i] = new PVector(x, y, z);
  13.      println(vecs[i]);
  14.      
  15.      maxX = max(maxX,vecs[i].x);
  16.      maxY = max(maxY,vecs[i].y);
  17.      maxZ = max(maxZ,vecs[i].z);
  18.      
  19.      minX = min(minX,vecs[i].x);
  20.      minY = min(minY,vecs[i].y);
  21.      minZ = min(minZ,vecs[i].z);
  22.    }

  23. }

  24. void drawSketch1()
  25. {
  26.   background(255);
  27.    
  28.   translate(width/2,height/2);
  29.   scale(1,-1,1); // so Y is up, which makes more sense in plotting
  30.   
  31.   rotateY(rotY);
  32.   rotateX(rotX);
  33.  
  34.   noFill();
  35.   strokeWeight(1);
  36.   stroke(0,10);
  37.   box(dim);
  38.  
  39.   translate(-dim/2,-dim/2,-dim/2);
  40.   
  41.   strokeWeight(2);
  42.   stroke(0);
  43.   line(0,0,0,dim,0,0);
  44.   line(0,0,0,0,dim,0);
  45.   line(0,0,0,0,0,dim);
  46.   
  47.   // Draw axis labels.
  48.   textSize(25);
  49.   String axisLabel;
  50.   fill(50);
  51.   pushMatrix();
  52.   
  53.   scale(1,-1,1);
  54.   axisLabel = "x-axis";
  55.   text(axisLabel,dim-textWidth(axisLabel),0,0);
  56.   
  57.   axisLabel = "y-axis";
  58.   rotateZ(HALF_PI);
  59.   text(axisLabel,-dim,0,0);
  60.   
  61.   axisLabel = "z-axis";
  62.   rotateZ(-HALF_PI);
  63.   rotateY(HALF_PI);
  64.   text(axisLabel,-dim,0,0);
  65.  
  66.   popMatrix();
  67.   
  68.   // Draw cubes.
  69.   for (int i=0; i<vecs.length; i++) 
  70.   {
  71.     PVector v = vecs[i];
  72.     
  73.     // cubes
  74.     noStroke();
  75.     fill( 255, 100, 100 );
  76.     pushMatrix();
  77.     
  78.     // translate(v.x,v.y,v.z);
  79.     
  80.     translate( map(v.x,minX,maxX,0,dim),
  81.                map(v.y,minY,maxY,0,dim),
  82.                map(v.z,minZ,maxZ,0,dim));
  83.     
  84.     box(5);
  85.     popMatrix();
  86.     
  87.   }
  88. }

Replies(6)

Merging two graphs is never easy. I don't know what was throwing off your view, so I started with what you had, slimmed it down a bit, and got this:

Copy code
  1. PFont font;

  2. void setup()
  3. {
  4.   size(500,500,P3D);
  5.   smooth();
  6.   font = createFont( "arial", 25 );
  7.   textFont(font);
  8. }

  9. void draw()
  10. {  
  11.  background(255);
  12.    
  13.   translate(width/2.0,height/2.0);
  14.   scale(1,-1,1); // so Y is up, which makes more sense in plotting
  15.   
  16.   
  17.   rotateX(map(mouseY,0,width,-PI,PI));
  18.   rotateY(map(mouseX,0,height,-PI,PI));
  19.  
  20.   noFill();
  21.   strokeWeight(1);
  22.   stroke(0,10);
  23.   box(250);
  24.   
  25.    // Draw axis labels.
  26.   textSize(25);
  27.   String axisLabel;
  28.   fill(50);
  29.   pushMatrix();
  30.   translate(-125,-125,-125);
  31.   
  32.   scale(1,-1,1);
  33.   axisLabel = "x-axis";
  34.   text(axisLabel,250 - textWidth(axisLabel),0,0);
  35.   
  36.   axisLabel = "y-axis";
  37.   rotateZ(HALF_PI);
  38.   text(axisLabel,-250,0,0);
  39.   
  40.   axisLabel = "z-axis";
  41.   rotateZ(-HALF_PI);
  42.   rotateY(HALF_PI);
  43.   text(axisLabel,-250,0,0);

  44.   
  45.   strokeWeight(2);
  46.   stroke(0);
  47.   line(0,0,0,-250,0,0);
  48.   line(0,0,0,0,-250,0);
  49.   line(0,0,0,0,0,250);
  50.   
  51.   popMatrix();
  52. }

It's leaving out any code that renders little red squares, and it doesn't muck about with scale(1,-1,1) calls, but at least the perspective is right!
Oh! I'm thick! I thought the problem was the skewed perspective! After reading what you actually posted, the solution is simple: Just render the second 3D view to a separate PGraphics object, then draw it's image next to the first one!

Here's the demo code snippet I have about doing that. This has three separate possible views, but you only need an always-on second one:

Copy code
  1. /// "ThreeViews" by TfGuy44 (TfGuy44@gmail.com)
  2. PGraphics BC, GC, RC;
  3. float rot = 0.0;

  4. void setup(){
  5.   size( 400, 200, P2D);
  6.   BC = createGraphics(200, 200, P3D);
  7.   RC = createGraphics(200, 200, P3D);
  8.   GC = createGraphics(200, 200, P3D);
  9. }

  10. void draw(){
  11.   rot=(rot+0.01)%360;
  12.   // Render the "main" scene.
  13.   background(0);
  14.   noStroke();
  15.   fill(color(255,0,0));
  16.   rect(10,40,20,20);
  17.   fill(color(0,0,255));
  18.   rect(70,70,20,20);
  19.   fill(color(0,255,0));
  20.   rect(70,10,20,20);
  21.   // Render to things off-screen!
  22.   RC.beginDraw();
  23.   RC.background(color(128,0,0));
  24.   RC.fill(color(255,0,0));
  25.   RC.lights();
  26.   RC.translate(100,100,0);
  27.   RC.rotateX(rot);
  28.   RC.rotateY(2*rot);
  29.   RC.box(20);
  30.   RC.endDraw();
  31.   // And the green one...
  32.   GC.beginDraw();
  33.   GC.background(color(0,128,0));
  34.   GC.fill(color(0,255,0));
  35.   GC.lights();
  36.   GC.translate(100,100,0);
  37.   GC.rotateX(rot);
  38.   GC.rotateY(2*rot);
  39.   GC.box(50);
  40.   GC.endDraw();
  41.   // And the blue one...
  42.   BC.beginDraw();
  43.   BC.background(color(0,0,128));
  44.   BC.fill(color(0,0,255));
  45.   BC.lights();
  46.   BC.translate(100,100,0);
  47.   BC.rotateX(rot);
  48.   BC.rotateY(2*rot);
  49.   BC.box(80);
  50.   BC.endDraw();
  51.   // Display one of those buffers? 
  52.   if( get( mouseX, mouseY ) == color(255,0,0) ){
  53.     image( RC.get(), 200, 0);
  54.   }
  55.   if( get( mouseX, mouseY ) == color(0,255,0) ){
  56.     image( GC.get(), 200, 0);
  57.   }
  58.   if( get( mouseX, mouseY ) == color(0,0,255) ){
  59.     image( BC.get(), 200, 0);
  60.   } 
  61. }

Thank you for responding mate. I did try out your method in my sketch but it doesn't seem to resolve the problem for some weird reason. 
@tfguy44: thanks for the tip

@nighrage: think you could look for Napplet library