Loading...
Logo
Processing Forum
Hi,

I have written a small program that allows you to manipulate objects with the mouse and some key strokes. I want to start with a blank canvas with a button. When you click on the button a new ShapeMove is added to the canvas. I feel like this should be pretty simple, but I'm stumped.

Here's my code so far. Sorry it's so long. The button class is at the very bottom. I'm using PImages but in this example it's all rectangles. 

Thanks
Copy code
  1. int x; // shape x
  2. int y; // shape y
  3. int h; // shape height
  4. int w; // shape width
  5. int b = 0; //counts button clicks
  6. boolean captured;
  7. boolean isOverSomething;
  8. ShapeMove[] shapeMove = new ShapeMove[3];
  9. Button button;

  10. void setup()
  11. {
  12.   size(800, 500);
  13.   for (int i = 0; i <= 2; i++) 
  14.   {
  15.     shapeMove[i] = new ShapeMove(0, 0, 200, 200);
  16.   }
  17.   button = new Button(200, 300);
  18. }

  19. void draw()
  20. {
  21.   background(255);
  22.   button.display();
  23.   if (b > 0)
  24.   {
  25.     fill(120, 250,0);
  26.     shapeMove[b-1].update();
  27.     shapeMove[b-1].display();
  28.   }
  29. }

  30. //////////////////////////////////////////////////////////////////////////
  31. class ShapeMove
  32. {
  33.   int x;
  34.   int y;
  35.   int h;
  36.   int w;
  37.   int origx;
  38.   int origy;
  39.   int deltax;
  40.   int deltay;
  41.   int posXdiff;
  42.   int posYdiff;
  43.   int xdragdelta;
  44.   int ydragdelta;
  45.   boolean over;
  46.   boolean locked = false;
  47.   boolean imCaptured = false;
  48.   boolean overMe = false;


  49.   ShapeMove(int _x, int _y, int _h, int _w)
  50.   {
  51.     x = _x;
  52.     y = _y;
  53.     h = _h;
  54.     w = _w;
  55.   }

  56.   void isover()
  57.   {
  58.     if (mouseX >= x && mouseX <=x+w &&
  59.       mouseY >= y && mouseY <= y+h)
  60.     {
  61.       over = true;
  62.       isOverSomething = true;
  63.     }
  64.     else 
  65.     {
  66.       over = false;
  67.       isOverSomething = false;
  68.     }
  69.   }
  70.   
  71.   void overme() 
  72.   {
  73.     if(isOverSomething) // over something
  74.     {
  75.       if(overMe) // over me!
  76.       {
  77.         if(!over)
  78.         {
  79.           overMe = false;
  80.           isOverSomething = false;
  81.         }
  82.       }
  83.     }
  84.     else // not over anything at the moment
  85.     {
  86.       if(over) // over me!
  87.       {
  88.         overMe = true;
  89.         isOverSomething = true;
  90.       }
  91.     }
  92.   }

  93.   void update()
  94.   {
  95.     if (over && mousePressed && !imCaptured && !captured)
  96.     {
  97.       deltax = x - mouseX;
  98.       deltay = y - mouseY;
  99.       origx = x;
  100.       origy = y;
  101.       captured = true;
  102.       imCaptured = true;
  103.     }
  104.     if (!mousePressed && imCaptured)
  105.     {
  106.       captured = false;
  107.       imCaptured = false;
  108.     }
  109.     if (imCaptured)
  110.     {
  111.       xdragdelta = x - (mouseX + posXdiff);
  112.       ydragdelta = y - (mouseY + posYdiff);
  113.       x = mouseX + deltax;
  114.       y = mouseY + deltay;
  115.     }
  116.   } 

  117.   void changesize() 
  118.   {
  119.     if (over && keyPressed)
  120.     {
  121.       if (key == 'b')
  122.       {
  123.         h += 2;
  124.         w += 2;
  125.       }
  126.       if (key == 's')
  127.       {
  128.         h -= 2;
  129.         w -= 2;
  130.       }
  131.     }
  132.   }
  133.   
  134.   void display()
  135.   {
  136.     rect(x, y, h, w);
  137.     changesize(); 
  138.   }
  139. }
  140. /////////////////////////////////////////////////////////////////////////
  141. class Button
  142. {
  143.   int rectX, rectY;      // Position of square button
  144.   int rectSize = 50;     // Diameter of rect
  145.   color rectColor;
  146.   color rectHighlight;
  147.   boolean rectOver = false;
  148.   
  149.   Button(int _rectX, int _rectY)
  150.   {
  151.     smooth();
  152.     rectColor = color(0);
  153.     rectHighlight = color(51);
  154.     rectX = _rectX;
  155.     rectY = _rectY;
  156.   }
  157.   
  158.   void display()
  159.   {
  160.     update(mouseX, mouseY);    
  161.     if(rectOver) {
  162.       fill(rectHighlight);
  163.     } else {
  164.       fill(rectColor);
  165.     }
  166.     stroke(255);
  167.     rect(rectX, rectY, rectSize, rectSize);
  168.   }
  169.   
  170.   void update(int x, int y)
  171.   {
  172.     if ( overRect(rectX, rectY, rectSize, rectSize) ) 
  173.     {
  174.       rectOver = true;
  175.     } else {
  176.       rectOver = false;
  177.     }
  178.   }
  179.   
  180.   void mousePressed()
  181.   {
  182.     if(rectOver) 
  183.     {
  184.     b++;  
  185.     }
  186.   }
  187.   
  188.   boolean overRect(int x, int y, int width, int height) 
  189.   {
  190.     if (mouseX >= x && mouseX <= x+width && 
  191.         mouseY >= y && mouseY <= y+height) {
  192.       return true;
  193.     } else {
  194.       return false;
  195.     }
  196.   }
  197. }

Replies(14)

First: You really should avoid re-using keywords defined by Processing. I'm mostly thinking about your function "void mousePressed()" inside your button class.

You could use an array for your "ShapeMove" collection, but I would strongly suggest using a list instead. This will simplify the addition or removal of ShapeMove objects!



Copy code
  1. int x; // shape x
  2. int y; // shape y
  3. int h; // shape height
  4. int w; // shape width
  5. int b = 0; //counts button clicks
  6. boolean captured;
  7. boolean isOverSomething;
  8. //ShapeMove[] shapeMove = new ShapeMove[3];
  9. ArrayList shapeMove = new ArrayList();
  10. Button button;
  11. void setup()
  12. {
  13.   size(800, 500);
  14.   button = new Button(200, 300);
  15. }
  16. void draw()
  17. {
  18.   background(255);
  19.   button.display();
  20.  
  21.   for( int i = 0; i < shapeMove.size(); i++ ){
  22.     ((ShapeMove)shapeMove.get(i)).update();
  23.     ((ShapeMove)shapeMove.get(i)).display();
  24.   }
  25. }
  26. void mousePressed(){
  27.   if( button.pressed(mouseX, mouseY) ){
  28.     shapeMove.add( new ShapeMove((int)random(0,width),(int)random(0,height), (int)random(10,20), (int)random(10,20)) );
  29.   }
  30. }
  31. //////////////////////////////////////////////////////////////////////////
  32. class ShapeMove
  33. {
  34.   int x;
  35.   int y;
  36.   int h;
  37.   int w;
  38.   int origx;
  39.   int origy;
  40.   int deltax;
  41.   int deltay;
  42.   int posXdiff;
  43.   int posYdiff;
  44.   int xdragdelta;
  45.   int ydragdelta;
  46.   boolean over;
  47.   boolean locked = false;
  48.   boolean imCaptured = false;
  49.   boolean overMe = false;
  50.   ShapeMove(int _x, int _y, int _h, int _w)
  51.   {
  52.     x = _x;
  53.     y = _y;
  54.     h = _h;
  55.     w = _w;
  56.   }
  57.   void isover()
  58.   {
  59.     if (mouseX >= x && mouseX <=x+w &&
  60.       mouseY >= y && mouseY <= y+h)
  61.     {
  62.       over = true;
  63.       isOverSomething = true;
  64.     }
  65.     else
  66.     {
  67.       over = false;
  68.       isOverSomething = false;
  69.     }
  70.   }
  71.   void overme()
  72.   {
  73.     if (isOverSomething) // over something
  74.     {
  75.       if (overMe) // over me!
  76.       {
  77.         if (!over)
  78.         {
  79.           overMe = false;
  80.           isOverSomething = false;
  81.         }
  82.       }
  83.     }
  84.     else // not over anything at the moment
  85.     {
  86.       if (over) // over me!
  87.       {
  88.         overMe = true;
  89.         isOverSomething = true;
  90.       }
  91.     }
  92.   }
  93.   void update()
  94.   {
  95.     if (over && mousePressed && !imCaptured && !captured)
  96.     {
  97.       deltax = x - mouseX;
  98.       deltay = y - mouseY;
  99.       origx = x;
  100.       origy = y;
  101.       captured = true;
  102.       imCaptured = true;
  103.     }
  104.     if (!mousePressed && imCaptured)
  105.     {
  106.       captured = false;
  107.       imCaptured = false;
  108.     }
  109.     if (imCaptured)
  110.     {
  111.       xdragdelta = x - (mouseX + posXdiff);
  112.       ydragdelta = y - (mouseY + posYdiff);
  113.       x = mouseX + deltax;
  114.       y = mouseY + deltay;
  115.     }
  116.   }
  117.   void changesize()
  118.   {
  119.     if (over && keyPressed)
  120.     {
  121.       if (key == 'b')
  122.       {
  123.         h += 2;
  124.         w += 2;
  125.       }
  126.       if (key == 's')
  127.       {
  128.         h -= 2;
  129.         w -= 2;
  130.       }
  131.     }
  132.   }
  133.   void display()
  134.   {
  135.     rect(x, y, h, w);
  136.     changesize();
  137.   }
  138. }
  139. /////////////////////////////////////////////////////////////////////////
  140. class Button
  141. {
  142.   int rectX, rectY;      // Position of square button
  143.   int rectSize = 50;     // Diameter of rect
  144.   color rectColor;
  145.   color rectHighlight;
  146.   boolean rectOver = false;
  147.   Button(int _rectX, int _rectY)
  148.   {
  149.     smooth();
  150.     rectColor = color(0);
  151.     rectHighlight = color(51);
  152.     rectX = _rectX;
  153.     rectY = _rectY;
  154.   }
  155.   void display()
  156.   {
  157.     update(mouseX, mouseY);   
  158.     if (rectOver) {
  159.       fill(rectHighlight);
  160.     }
  161.     else {
  162.       fill(rectColor);
  163.     }
  164.     stroke(255);
  165.     rect(rectX, rectY, rectSize, rectSize);
  166.   }
  167.   void update(int x, int y)
  168.   {
  169.     if ( overRect(rectX, rectY, rectSize, rectSize) )
  170.     {
  171.       rectOver = true;
  172.     }
  173.     else {
  174.       rectOver = false;
  175.     }
  176.   }
  177.   boolean overRect(int x, int y, int width, int height)
  178.   {
  179.     if (mouseX >= x && mouseX <= x+width &&
  180.       mouseY >= y && mouseY <= y+height) {
  181.       return true;
  182.     }
  183.     else {
  184.       return false;
  185.     }
  186.   }
  187.  
  188.   boolean pressed(int _x, int _y){
  189.     if( rectOver )
  190.       return true;
  191.    
  192.     return false;
  193.   }
  194. }

Hmm, a bit better maybe... ?

Copy code
  1.   boolean pressed(int _x, int _y){
  2.     return rectOver;
  3.   }


yeah , I just wrote the same code than a__g  

there's a lot of stuff in both classes that are not needed and come from your experiments probably...






Lists are fun! =)
Thanks a__g, that solves part of my problem. I'm pretty new to programming, so any tips are greatly appreciated.

I want to be able to grab the shapes with a mouse click and drag them around the screen. With the setup and draw that you suggest I can't manipulate the shapes. I think this is because the ints x, y, h, and w are called inside the button.

I might have somehow broken the code in the first example. Anyway, if you run the following code you'll get two squares that you can resize with 's' and 'b' and move with the mouse. What I want is what you suggested, but with the ability to manipulate the shapes.

Thanks so much for the help.

Copy code
  1. //PImage clv;
  2. int x; // x, y, height and width of shapeMove
  3. int y;
  4. int h;
  5. int w;
  6. boolean captured;
  7. boolean selected = false;
  8. ShapeMove[] shapeMove = new ShapeMove[2];
  9. Button button;

  10. void setup()
  11. {
  12.   size(800, 500);
  13.   //clv = loadImage("clv.png");
  14.   int i;
  15.   i = 0;
  16.   shapeMove[0] = new ShapeMove(0, 0, 200, 200);
  17.   shapeMove[1] = new ShapeMove(250, y, 250, 250);
  18.   button = new Button(200, 300);
  19. }

  20. void draw()
  21. {
  22.   background(255);
  23.   button.display();
  24.   fill(120, 250,0);
  25.   for (int i = shapeMove.length-1; i >=0; i--) 
  26.   {
  27.     shapeMove[i].update();
  28.     shapeMove[i].display();
  29.   }
  30. }

  31. //////////////////////////////////////////////////////////////////////////
  32. class ShapeMove
  33. {
  34.   int x;
  35.   int y;
  36.   int h;
  37.   int w;
  38.   int origx;
  39.   int origy;
  40.   int deltax;
  41.   int deltay;
  42.   int posXdiff;
  43.   int posYdiff;
  44.   boolean over;
  45.   boolean locked = false;
  46.   boolean imCaptured = false;
  47.   int xdragdelta, ydragdelta;

  48.   ShapeMove(int _x, int _y, int _h, int _w)
  49.   {
  50.     x = _x;
  51.     y = _y;
  52.     h = _h;
  53.     w = _w;
  54.   }

  55.   boolean over()
  56.   {
  57.     if (mouseX >= x && mouseX <=x+w &&
  58.       mouseY >= y && mouseY <= y+h)
  59.     {
  60.       over = true;
  61.       return true;
  62.     }
  63.     else 
  64.     {
  65.       over = false;
  66.       return false;
  67.     }
  68.   }

  69.   void update()
  70.   {
  71.     if (over() && mousePressed && !imCaptured && !captured)
  72.     {
  73.       deltax = x - mouseX;
  74.       deltay = y - mouseY;
  75.       origx = x;
  76.       origy = y;
  77.       captured = true;
  78.       imCaptured = true;
  79.     }
  80.     if (!mousePressed && imCaptured)
  81.     {
  82.       captured = false;
  83.       imCaptured = false;
  84.     }
  85.     if (imCaptured)
  86.     {
  87.       xdragdelta = x - (mouseX + posXdiff);
  88.       ydragdelta = y - (mouseY + posYdiff);
  89.       x = mouseX + deltax;
  90.       y = mouseY + deltay;
  91.     }
  92.   } 

  93.   void changesize() 
  94.   {
  95.     if (over() && keyPressed)
  96.     {
  97.       if (key == 'b')
  98.       {
  99.         h += 2;
  100.         w += 2;
  101.       }
  102.       if (key == 's')
  103.       {
  104.         h -= 2;
  105.         w -= 2;
  106.       }
  107.     }
  108.   }
  109.   
  110.   void display()
  111.   {
  112.     rect(x, y, h, w);
  113.     changesize(); 
  114.   }
  115. }
  116. /////////////////////////////////////////////////////////////////////////
  117. class Button
  118. {
  119.   int rectX, rectY;      // Position of square button
  120.   int rectSize = 50;     // Diameter of rect
  121.   color rectColor;
  122.   color rectHighlight;
  123.   boolean rectOver = false;
  124.   
  125.   Button(int _rectX, int _rectY)
  126.   {
  127.     smooth();
  128.     rectColor = color(0);
  129.     rectHighlight = color(51);
  130.     rectX = _rectX;
  131.     rectY = _rectY;
  132.   }
  133.   
  134.   void display()
  135.   {
  136.     update(mouseX, mouseY);    
  137.     if(rectOver) {
  138.       fill(rectHighlight);
  139.     } else {
  140.       fill(rectColor);
  141.     }
  142.     stroke(255);
  143.     rect(rectX, rectY, rectSize, rectSize);
  144.   }
  145.   
  146.   void update(int x, int y)
  147.   {
  148.     if ( overRect(rectX, rectY, rectSize, rectSize) ) {
  149.       rectOver = true;
  150.     } else {
  151.       rectOver = false;
  152.     }
  153.   }
  154.   
  155.   void mousePressed()
  156.   {
  157.     if(rectOver) {
  158. //      currentColor = rectColor;
  159.     }
  160.   }
  161.   
  162.   boolean overRect(int x, int y, int width, int height) 
  163.   {
  164.     if (mouseX >= x && mouseX <= x+width && 
  165.         mouseY >= y && mouseY <= y+height) {
  166.       return true;
  167.     } else {
  168.       return false;
  169.     }
  170.   }
  171. }
I was writing my response before I saw yours. I'll see if that helps.

Chrisir, yeah, new to programming and a little overwhelmed. I'm going to try clean it up once I get this button working.

Thanks!
Here is a version using processing.js that has images rather than shapes. This way you might be able to better understand what I'm talking about.


I want the button to add trees and then I want to be able to move them around and resize them.

Thanks for all the help. I really appreciate it.


" You really should avoid re-using keywords defined by Processing. I'm mostly thinking about your function "void mousePressed()" inside your button class."
In this case, I disagree. I can have mousePressed() and draw() in my classes, there is no ambiguity there. But there can be issues with variables names, like width or keyCode.


this one works

  • drag around
  • resize



Copy code
  1. int x; // shape x
  2. int y; // shape y
  3. int h; // shape height
  4. int w; // shape width
  5. int b = 0; //counts button clicks
  6. boolean captured;
  7. boolean isOverSomething;
  8. ShapeMove[] shapeMove = new ShapeMove[30300];
  9. Button button;
  10. int i=0;
  11. //
  12. void setup()
  13. {
  14.   size(800, 500);
  15.   i=0;
  16.   /*
  17.   for (int i = 0; i <= 2; i++) 
  18.    {
  19.    shapeMove[i] = new ShapeMove(i*33, 0, 20, 20);
  20.    } */
  21.   button = new Button(200, 300);
  22. }
  23. //
  24. void draw()
  25. {
  26.   background(255);
  27.   button.display();
  28.   fill(120, 250, 0);
  29.   for (int myShapeNum=0; myShapeNum<i;myShapeNum++) {
  30.     shapeMove[myShapeNum].isover();
  31.     shapeMove[myShapeNum].changesize();
  32.     shapeMove[myShapeNum].update();
  33.     shapeMove[myShapeNum].display();
  34.   }
  35. }
  36. //
  37. void mousePressed()
  38. {
  39.   if (button.rectOver) 
  40.   {
  41.     println("Hurray!");
  42.     shapeMove[i] = new ShapeMove(i*33+33, 0, 20, 20);   
  43.     i++;
  44.     b++;
  45.   }
  46.   else
  47.   {
  48.     println("Nope.");
  49.   }
  50. }
  51. //
  52. //////////////////////////////////////////////////////////////////////////
  53. //
  54. class ShapeMove
  55. {
  56.   int x;
  57.   int y;
  58.   int h;
  59.   int w;
  60.   int origx;
  61.   int origy;
  62.   int deltax;
  63.   int deltay;
  64.   int posXdiff;
  65.   int posYdiff;
  66.   int xdragdelta;
  67.   int ydragdelta;
  68.   boolean over;
  69.   boolean locked = false;
  70.   boolean imCaptured = false;
  71.   boolean overMe = false;
  72.   //
  73.   ShapeMove(int _x, int _y, int _h, int _w)
  74.   {
  75.     x = _x;
  76.     y = _y;
  77.     h = _h;
  78.     w = _w;
  79.   }
  80.   void isover()
  81.   {
  82.     if (mouseX >= x && mouseX <=x+w &&
  83.       mouseY >= y && mouseY <= y+h)
  84.     {
  85.       over = true;
  86.       isOverSomething = true;
  87.     }
  88.     else 
  89.     {
  90.       over = false;
  91.       isOverSomething = false;
  92.     }
  93.   }
  94.   void overme() 
  95.   {
  96.     if (isOverSomething) // over something
  97.     {
  98.       if (overMe) // over me!
  99.       {
  100.         if (!over)
  101.         {
  102.           overMe = false;
  103.           isOverSomething = false;
  104.         }
  105.       }
  106.     }
  107.     else // not over anything at the moment
  108.     {
  109.       if (over) // over me!
  110.       {
  111.         overMe = true;
  112.         isOverSomething = true;
  113.       }
  114.     }
  115.   }
  116.   void update()
  117.   {
  118.     if (over && mousePressed && !imCaptured && !captured)
  119.     {
  120.       deltax = x - mouseX;
  121.       deltay = y - mouseY;
  122.       origx = x;
  123.       origy = y;
  124.       captured = true;
  125.       imCaptured = true;
  126.     }
  127.     if (!mousePressed && imCaptured)
  128.     {
  129.       captured = false;
  130.       imCaptured = false;
  131.     }
  132.     if (imCaptured)
  133.     {
  134.       xdragdelta = x - (mouseX + posXdiff);
  135.       ydragdelta = y - (mouseY + posYdiff);
  136.       x = mouseX + deltax;
  137.       y = mouseY + deltay;
  138.     }
  139.   } 
  140.   void changesize() 
  141.   {
  142.     if (over && keyPressed)
  143.     {
  144.       if (key == 'b')
  145.       {
  146.         h += 2;
  147.         w += 2;
  148.       }
  149.       if (key == 's')
  150.       {
  151.         h -= 2;
  152.         w -= 2;
  153.       }
  154.     }
  155.   }
  156.   void display()
  157.   {
  158.     rect(x, y, h, w);
  159.     changesize();
  160.   }
  161. }
  162. // ======================================================================
  163. /////////////////////////////////////////////////////////////////////////
  164. class Button
  165. {
  166.   int rectX, rectY;      // Position of square button
  167.   int rectSize = 50;     // Diameter of rect
  168.   color rectColor;
  169.   color rectHighlight;
  170.   boolean rectOver = false;
  171.   Button(int _rectX, int _rectY)
  172.   {
  173.     smooth();
  174.     rectColor = color(0);
  175.     rectHighlight = color(51);
  176.     rectX = _rectX;
  177.     rectY = _rectY;
  178.   }
  179.   void display()
  180.   {
  181.     update(mouseX, mouseY);    
  182.     if (rectOver) {
  183.       fill(rectHighlight);
  184.     }
  185.     else {
  186.       fill(rectColor);
  187.     }
  188.     stroke(255);
  189.     rect(rectX, rectY, rectSize, rectSize);
  190.   }
  191.   void update(int x, int y)
  192.   {
  193.     if ( overRect(rectX, rectY, rectSize, rectSize) ) 
  194.     {
  195.       rectOver = true;
  196.     }
  197.     else {
  198.       rectOver = false;
  199.     }
  200.   }
  201.   boolean overRect(int x, int y, int width, int height) 
  202.   {
  203.     if (mouseX >= x && mouseX <= x+width && 
  204.       mouseY >= y && mouseY <= y+height) {
  205.       return true;
  206.     }
  207.     else {
  208.       return false;
  209.     }
  210.   }
  211. }
  212. // ---------------------------------------------------------------





there is an error: you had swapped w and h for shape in the constructor and in the method display of the shape.

w=width and h=height, w is before h in the convention, e.g. with rect.

in   void isover() you got it right.

It is still in the code I posted.




Wow, perfect. That's brilliant. 

Thanks for the "Hurray!" line too. That made my day.


I haven't run the code, but it seems to me that isOverSomething should only be set to false when it is true, as it is a global variable. If one is not over button A one should not set isOverSomething = false as one might be over button B.

One should only set isOverSomething to false when one transitions from being over to not over for the item that set isOverSomething to true.
jas0501, I realize I've muddled up the example you gave me earlier for isOver and isOverSomething. If the mouse is hovering over two objects, they both resize. I follow your logic, but I just can't seem to translate it to code. 
I was commenting on line 91 (line 10 below). I think is should be removed.
Copy code

  1.     if (mouseX >= x && mouseX <=x+w &&
  2.       mouseY >= y && mouseY <= y+h)
  3.     {
  4.       over = true;
  5.       isOverSomething = true;
  6.     }
  7.     else 
  8.     {
  9.       over = false;
  10.       isOverSomething = false;
  11.     }