Loading...
Logo
Processing Forum
Hi everyone, I home someone here can point out what I am missing.  I have written 2 processing applets that can control led cubes ( youtube ledcube if you aren't familiar with them ) ,  and am trying to combine them into one java application. I have been trying test if this is possible for the past week and am completely stuck. I don't know if this is the best place to post this, but I figure people in the java forums might just complain about the differences between processing and java. I figured I'd try here first. 

Here is what I am doing:
I have two applets (Applet1 and Applet2)  inside Frames (Frame1, Frame2) a circle class and my main class. The problem is that the Circle Object created in the main class is out of scope in regards to the applets. I have tried using java's clonable, writing my own copyable method and even overwriting the Circle object, but it will not compile. Any refrence to the main circle object  " aNewCircle" is highlighted because it is an unknown variable. 
My goal is to have two applets each drawing the same circle. Changing the color or position attribute of an circle in one applet should instantly show the same change in the other applet. 
I have tried every scope imaginable for aNewCircle. static, public, protected ect.. 

I am using proclipsing  http://code.google.com/p/proclipsing/
I have completely exhausted all my java books and google on this issue, any tips would be helpful. 


Copy code
  1. package ellipseproblem;
  2. public class Applet1 extends processing.core.PApplet
  3. {
  4. Circle aGreenCircle;
  5. public void setup()
  6. {
  7. background(255,0,0);
  8. aGreenCircle = new Circle(); 
  9. }
  10. public void draw()
  11. {
  12. background(255,0,0);
  13. aGreenCircle.display(this);
  14. }
  15. public void mouseReleased()
  16. {
  17. aGreenCircle.setXLocation(100);
  18. }
  19. }

//=================================================

Copy code
  1. package ellipseproblem;
  2. public class Applet2 extends processing.core.PApplet
  3. {
  4. //public Circle aGreenCircle;
  5. Circle aGreenCircle = new Circle();
  6. //Circle aGreenCircle = aNewCircle;
  7. public void setup()
  8. {
  9. background(0,0,255);
  10. aGreenCircle = new Circle();
  11. }

  12. public void draw()
  13. {
  14. background(0,0,255);
  15. aGreenCircle.display(this);
  16. //aNewCircle.display(this);
  17. }
  18. public void mouseReleased()
  19. {
  20. aGreenCircle.setXLocation(100);

  21. }
  22. }

//=================================================

Copy code
  1. package ellipseproblem;

  2. public class Frame1 extends java.awt.Frame 
  3. {
  4. Applet1 anApplet1;
  5.    Frame1()
  6.    {
  7.      setBounds (0,0, 300, 300);
  8.        anApplet1 = new Applet1();
  9.      add(anApplet1);
  10.       anApplet1.init();
  11.      show();     
  12.    }
  13.   
  14. }


//=================================================

Copy code
  1. package ellipseproblem;

  2. public class Frame2 extends java.awt.Frame 
  3. {
  4. Applet2 anApplet2;
  5.    Frame2()
  6.    {
  7.      setBounds (0,400, 400, 400);
  8.       anApplet2 = new Applet2();
  9.      add(anApplet2);
  10.          anApplet2.init();
  11.      show();  
  12.    }
  13.     
  14. }

//=================================================
Copy code
  1. package ellipseproblem;

  2. import processing.core.PApplet;

  3. public class EllipseProblem extends PApplet 
  4. {
  5. // Frame1 aFrame1;
  6. // Frame2 aFrame2;
  7. Applet1 anApplet1;
  8. Applet2 anApplet2;
  9. public static Circle aNewCircle;

  10. public void draw() 
  11.         {
  12. anApplet1.redraw(); 
  13. anApplet2.redraw();
  14. }
  15. public static void createGui()
  16. {
  17. Frame1 aFrame1 = new Frame1();
  18. Frame2 aFrame2 = new Frame2();
  19. aFrame1.setVisible( true );//not needed
  20. }
  21. public static void main(String _args[]) 
  22.         {
  23. PApplet.main(new String[] { ellipseproblem.EllipseProblem.class.getName() });
  24. createGui();
  25. aNewCircle = new Circle();
  26. }
  27. }


Replies(5)

I am no experience in controlling several applets from a common applet, but I am not sure if that's a good idea. Particularly, the idea of having the draw() of the main applet controlling those in the secondary applets is disturbing...
Perhaps the main class should not be a PApplet and should let the secondary applets draw at their own rhythm.

These secondary applets have no knowledge of the main class. So they cannot reference something from there*.
You can either pass the main class (this) or a reference to the Circle to the constructor (for example) of each sub-applet.

* Well, not exactly, true in the general case, but since the Circle object is static and public, you can reference it directly like:
Circle aGreenCircle = EllipseProblem.aNewCircle;

But passing the reference to the object is probably the cleanest version, particularly for more complex programs.
I agree with everything that PhiLho has said and I am also concerned about the classes you have created.

For instance the two classes called Applet1 and Applet2 are the same effectivley the same the only difference being that value stored in a attribute. Same problem with the classes Frame1 and Frame2. This is poor OO design which will cause loads of problems - just think of the number of classes needed if you wanted 8 linked Windows.

Having said all that there is a solution to your problem.

The code below works I have tested it, but it uses classes from the G4P library. Download the G4P library from here, extract the library from the zip file and locate the file called guicomponents.jar

Since you are using Proclipsing there will be a folder called lib inside the Eclipse Project, copy guicomponents.jar into this folder, if Eclipse is open then you need to refresh the project.

Now you have to modify the project's build path, right click on the project name and select Build Path | Configure Build Path then in the dialog box select the Libraries tab. Click on Add External JARs then browse to the guicomponents.jar file and click OK

Now your project can make use of anything in the G4P library including its support of multiple Windows.

You have not posted your Circle class so I had to create my own which I called CircleQ so it would not conflict with your class. Add this class to the project.

Copy code
  1.     public class CircleQ {
  2.         public int x,y,r,col;

  3.         public CircleQ(int x, int y, int r, int col) {
  4.             this.x = x;
  5.             this.y = y;
  6.             this.r = r;
  7.             this.col = col;
  8.         }
  9.     }

The next class is the core of the solution and holds all the dtata to be shared between the Windows. The key here is that shared attributes should be public static

Copy code
  1.     public class SharedData {
  2.         public static CircleQ cq;
  3.     }


The final class is the main PApplet and when you run it it will display 4 Windows (with different background colours). Click on the one with the gray background to make it active. The wasd keys will move the circle and the zx keys decrease/increase the circles radius. Note that the changes are shown in all 4 Windows.

HTH

Copy code
  1.     import guicomponents.GWinApplet;
  2.     import guicomponents.GWinData;
  3.     import guicomponents.GWindow;
  4.     import processing.core.PApplet;

  5.     public class LinkedWindows extends PApplet {
  6.         private GWindow[] window;

  7.         public void setup(){
  8.             size(200,200);
  9.             SharedData.cq = new CircleQ(100,100,60,color(128,0,128));
  10.             createWindows();
  11.         }

  12.         public void createWindows(){
  13.             int col;
  14.             window = new GWindow[3];
  15.             for(int i = 0; i < 3; i++){
  16.                 col = (200 << (i * 8)) | 0xff000000;
  17.                 window[i] = new GWindow(this, "Window "+i, 130+i*210, 100+i*100, 200, 200, false, JAVA2D);
  18.                 window[i].setBackground(col);
  19.                 window[i].addDrawHandler(this, "windowDraw"+i);
  20.             }
  21.         }

  22.         /**
  23.          * Draw for the main window
  24.          */
  25.         public void draw(){
  26.             pushMatrix();
  27.             background(192);
  28.             fill(0);
  29.             text("Window 0", 20,20);
  30.             if(SharedData.cq != null){
  31.                 fill(SharedData.cq.col);
  32.                 ellipseMode(CENTER);
  33.                 translate(SharedData.cq.x, SharedData.cq.y);
  34.                 ellipse(0, 0, SharedData.cq.r, SharedData.cq.r);
  35.             }
  36.             popMatrix();
  37.         }

  38.         public void keyTyped(){
  39.             switch(key){
  40.             case 'w':
  41.                 SharedData.cq.y -= 4;
  42.                 break;
  43.             case 's':
  44.                 SharedData.cq.y += 4;
  45.                 break;
  46.             case 'a':
  47.                 SharedData.cq.x -= 4;
  48.                 break;
  49.             case 'd':
  50.                 SharedData.cq.x += 4;
  51.                 break;
  52.             case 'z':
  53.                 SharedData.cq.r = max(SharedData.cq.r - 4, 40);
  54.                 break;
  55.             case 'x':
  56.                 SharedData.cq.r = min(SharedData.cq.r + 4, 80);
  57.                 break;
  58.             }
  59.         }

  60.         public void windowDraw0(GWinApplet appc, GWinData data){
  61.             appc.pushMatrix();
  62.             appc.fill(255);
  63.             appc.text("Window 0", 20,20);
  64.             if(SharedData.cq != null){
  65.                 appc.fill(SharedData.cq.col);
  66.                 appc.ellipseMode(CENTER);
  67.                 appc.translate(SharedData.cq.x, SharedData.cq.y);
  68.                 appc.ellipse(0, 0, SharedData.cq.r, SharedData.cq.r);
  69.             }
  70.             appc.popMatrix();
  71.         }  

  72.         public void windowDraw1(GWinApplet appc, GWinData data){
  73.             appc.pushMatrix();
  74.             appc.fill(0);
  75.             appc.text("Window 1", 20,20);
  76.             if(SharedData.cq != null){
  77.                 appc.fill(SharedData.cq.col);
  78.                 appc.ellipseMode(CENTER);
  79.                 appc.translate(SharedData.cq.x, SharedData.cq.y);
  80.                 appc.ellipse(0, 0, SharedData.cq.r, SharedData.cq.r);
  81.             }
  82.             appc.popMatrix();
  83.         }
  84.     
  85.         public void windowDraw2(GWinApplet appc, GWinData data){
  86.             appc.pushMatrix();
  87.             appc.fill(255,255,0);
  88.             appc.text("Window 2", 20,20);
  89.             if(SharedData.cq != null){
  90.                 appc.fill(SharedData.cq.col);
  91.                 appc.ellipseMode(CENTER);
  92.                 appc.translate(SharedData.cq.x, SharedData.cq.y);
  93.                 appc.ellipse(0, 0, SharedData.cq.r, SharedData.cq.r);
  94.             }
  95.             appc.popMatrix();
  96.         }  
  97.     }
Thanks so much quarks for your help. Your code worked perfectly. A static class is needed for two applets to communicate with each other. 
The logic behind my class structure in this sample code is that my real code uses two complete and different applets. One of them is using the 2d engine, and the other is using opengl. I am looking into your approach of using the gcomponents library. 

For those interested, here is how I got the code working. It is still using multiple frame classes, which now I know is not the best way to do it. At the very least 1 frame bulder class that has overridden constructors, one for Applet1 and one for Applet2, would be better code. quarks' gcomponents library looks to be an even better approach. 




Copy code
  1. package ellipseproblem;

  2. import processing.core.PApplet;

  3. public class Circle
  4. {
  5.  // PApplet parent;

  6.   int xLocation;
  7.   int yLocation;
  8.   int width;
  9.   int height;
  10.   
  11.    Circle( int xLocation, int yLocation, int width, int height )
  12.    {    
  13.      this.xLocation = xLocation;
  14.      this.yLocation = yLocation;
  15.      this.width = width;
  16.      this.height = height;
  17.    }

  18. public int getxLocation() {
  19. return xLocation;
  20. }

  21. public void setxLocation(int xLocation) {
  22. this.xLocation = xLocation;
  23. }

  24. public int getyLocation() {
  25. return yLocation;
  26. }

  27. public void setyLocation(int yLocation) {
  28. this.yLocation = yLocation;
  29. }

  30. public int getWidth() {
  31. return width;
  32. }

  33. public void setWidth(int width) {
  34. this.width = width;
  35. }

  36. public int getHeight() {
  37. return height;
  38. }

  39. public void setHeight(int height) {
  40. this.height = height;
  41. }

  42. }


Copy code

  1. package ellipseproblem;

  2. import java.awt.Dimension;
  3. import java.awt.Toolkit;

  4. public class EllipseProblem {

  5. Applet1 anApplet1;
  6. Applet2 anApplet2;

  7. public static void main(String _args[]) 
  8. {

  9. SharedData.aFixedCircle = new Circle(40, 40, 40, 40);

  10. Frame1  aFrame1 = new Frame1();
  11. aFrame1.setTitle("Frame 1");
  12. aFrame1.setLocation(0, 0);
  13. Frame2  aFrame2 = new Frame2();
  14. aFrame2.setTitle("Frame 2");
  15. aFrame2.setLocation(400, 400);

  16. }// end main
  17. }// end class EllipseProblem


Copy code

  1. package ellipseproblem;

  2. public class Frame1 extends java.awt.Frame 
  3. {
  4. Applet1 anApplet1;
  5.    Frame1()
  6.    {
  7.      setBounds (0,0, 300, 300);
  8.        anApplet1 = new Applet1();
  9.      add(anApplet1);
  10.       anApplet1.init();
  11.      //show();//deprecated  
  12.       this.setVisible(true);   
  13.    }
  14.   
  15. }



Copy code
  1. package ellipseproblem;

  2. public class Frame2 extends java.awt.Frame 
  3. {
  4. Applet2 anApplet2;
  5.    Frame2()
  6.    {
  7.      setBounds (0,400, 400, 400);
  8.       anApplet2 = new Applet2();
  9.      add(anApplet2);
  10.          anApplet2.init();
  11.       //show();//deprecated  
  12.       this.setVisible(true); 
  13.    }
  14.     
  15. }


Copy code

  1. package ellipseproblem;
  2. public class Applet1 extends processing.core.PApplet
  3. {

  4. public void setup()
  5. {

  6. }

  7. public void draw()
  8. {
  9. background(255,0,0);
  10. //aGreenCircle.display(this);
  11. int localX = SharedData.aFixedCircle.getxLocation();
  12. int localY = SharedData.aFixedCircle.getyLocation();
  13. int localWidth = SharedData.aFixedCircle.getWidth();
  14. int localHeight = SharedData.aFixedCircle.getHeight();
  15. this.ellipse( localX, localY, localWidth, localHeight );
  16. }

  17. public void mouseReleased()
  18. {
  19. SharedData.aFixedCircle.setxLocation( SharedData.aFixedCircle.getxLocation() + 10 );
  20. }

  21. }



Copy code

  1. package ellipseproblem;
  2. public class Applet2 extends processing.core.PApplet
  3. {

  4. public void setup()
  5. {
  6. background(0,0,255);

  7. }

  8. public void draw()
  9. {
  10. background(0,0,255);
  11. int localX = SharedData.aFixedCircle.getxLocation();
  12. int localY = SharedData.aFixedCircle.getyLocation();
  13. int localWidth = SharedData.aFixedCircle.getWidth();
  14. int localHeight = SharedData.aFixedCircle.getHeight();
  15. this.ellipse(localX, localY, localWidth, localHeight);
  16. }
  17. public void mouseReleased()
  18. {
  19. SharedData.aFixedCircle.setxLocation( SharedData.aFixedCircle.getxLocation() - 10 );

  20. }
  21. }

" A static class is needed for two applets to communicate with each other."
No.
As I wrote, you can pass references of an object to another object / class. So it also applies to applets.

The logic behind my class structure in this sample code is that my real code uses two complete and different applets. One of them is using the 2d engine, and the other is using opengl
In G4P the GWindow objects created can use any renderer (P2D, JAVA2D, P3D, OPENGL), they do not all need to be the same.

In the Shapes3D library some of the examples use G4P to create multiple windows with different renderers, OPENGL for 3D display and JAVA2D for the control windows. In this example sketch click on the button in the 3D display to open the 2D control window.

P.S. I used the static class as a convenience, there are other ways to do this as PhiLho pointed out.