Loading...
Logo
Processing Forum
I can't post all code so i hope someone still have a clue what it might be.
I have a path and a selection tool to select anchors.
If i hold shift i want a toggle selection.

For debugging i print A, B, C etc. at serveral steps.
Copy code
  1. public void handleReleased(float x, float y) {
  2.         if(startX != x || startY != y) {
  3.             // rectangular selection
  4.            
  5.             float x1 = PApplet.min(x, startX);
  6.             float y1 = PApplet.min(y, startY);
  7.             float x2 = PApplet.max(x, startX);
  8.             float y2 = PApplet.max(y, startY);
  9.            
  10.             boolean toggle = checkToggle();
  11.            
  12.             if(toggle == false) {
  13.                 toolHandler.layerStack.clearAnchorSelection();
  14.             }
  15.            
  16.             lastAnchorSelection = new ArrayList<CVector>(toolHandler.layerStack.anchorSelection);
  17.             newAnchorSelection.clear();
  18.            
  19.             System.out.println(lastAnchorSelection.size());
  20.            
  21.             for (Layer layer : toolHandler.layerStack.layers) {
  22.                 rectangularSelection(x1, y1, x2, y2, layer, toggle);
  23.             }
  24.            
  25.             toolHandler.layerStack.anchorSelection.clear();
  26.             toolHandler.layerStack.anchorSelection.addAll(newAnchorSelection);
  27.         }
  28.     }


Copy code
  1. public void rectangularSelection(float x1, float y1, float x2, float y2, Layer layer, boolean toggle) {
  2.        
  3.         for (BezierVertexPath p : layer.paths) {
  4.            
  5.             if(p.locked()) continue;
  6.            
  7.             for(CVector v : p.getAnchors()) {
  8.                 if(x1 <= v.x() && y1 <= v.y() && x2 >= v.x() && y2 >= v.y()) {
  9.                     // CVectors can be shared, there for with toggle we don't want to turn them on and off
  10.                     // every time we meat them. So we look if they where present in the old selection.
  11.                     // And based on we add them to the new selection.
  12.                     System.out.println("A");
  13.                     if(toggle) {
  14.                         System.out.println("B");
  15.                         // ->
  16.                         if(lastAnchorSelection.contains(v) == false) {
  17.                             System.out.println("lastAnchorSelection.size(): "+lastAnchorSelection.size());
  18.                             System.out.println("C");
  19.                             // we only want to add it once
  20.                             if(newAnchorSelection.contains(v) == false) {
  21.                                 System.out.println("D");
  22.                                 newAnchorSelection.add(v);
  23.                             }
  24.                         }
  25.                         else {
  26.                             System.out.println("E");
  27.                             if(newAnchorSelection.contains(v) == false) {
  28.                                 System.out.println("F");
  29.                                 newAnchorSelection.add(v);
  30.                             }
  31.                         }
  32.                     }
  33.                     else {
  34.                         System.out.println("G");
  35.                         if(newAnchorSelection.contains(v) == false) {
  36.                             System.out.println("H");
  37.                             newAnchorSelection.add(v);
  38.                         }
  39.                     }
  40.                 }
  41.             }
  42.            
  43.         }   
  44.        
  45.         // now the subLayers
  46.         if(layer.subLayers != null) {
  47.             for(Layer l : layer.subLayers) {
  48.                 rectangularSelection(x1, y1, x2, y2, l, toggle);
  49.             }
  50.         }
  51.     }

If i select a anchor, and then select it again while holding shift then it prints


A
B
lastAnchorSelection.size(): 1
C
D

A and B are correct.
howver, there is this line:

Copy code
  1. if(lastAnchorSelection.contains(v) == false) {
lastAnchorSelection holds 1 anchor, the one i select again, but still lastAnchorSelection.contains(v) is false.

I only don't understand how that can be.
An ArrayList holds a reference right...

anyone a clue?




Replies(6)

I added 2 new lines to show how strange this is.
It prints 1 for the second line, so lastAnchorSelection contains 1 item.
But since it passed the if value, you would assume that item can not be v.
Then if i test the item in the array against v it equals true :S
Copy code
  1. if(lastAnchorSelection.contains(v) == false) {
  2.                             System.out.println("lastAnchorSelection.size(): "+lastAnchorSelection.size());
  3.                             CVector test = lastAnchorSelection.get(0);
  4.                             System.out.println(test == v);
  5.                             System.out.println("C");

If someone wants to look into this then please drop me a message and i will parse all the files needed.

got it!
Posted the problem on stackOverFlow and someone said v has probably it's own equal method.

my v is basicly a copy of PVector and PVector has it's own equal:

Copy code
  1. PVector a = new PVector();
  2. PVector b = new PVector();
  3. println(a.equals(b));
  4. PGraphics pg = createGraphics(100, 100);
  5. PGraphics pg2 = createGraphics(100, 100);
  6. println(pg.equals(pg2));

the above will print true and then false.
The equal method of PVector just checks if xyz are matching.

Wouldn't it be better if the method was called matches() ?
We have
  • a class called CVector
  • an ArrayList of CVector object references

Now consider the following code
Copy code
  1. CVector v1, v2, v3;
  2. v1 = new CVector();
  3. v2 = new CVector();
  4. v3 = v2;

Although we have 3 CVector references we only have 2 CVector objects. The references v2 and v3 refer to the same object. SInce we have used the default constructor then the attributes of the 2 objects created will be the same.

If you have not implemented the Comparable interface in the CVector class then comparisons are based on the object reference so we get
In the Java collections (ArrayList, LinkedList, HashMap etc) comparisons are done with the equals() method so if we use it with our objects we get

v1.equals(v2) is false
v1.equals(v3) is false
v2.equals(v3) is true


even though the internal states of v1 and v2 are the same.

The contains method compares the object with each one in the ArrayList using the equals() method so if you want don't want to compare references but rather compare objects based on their internal state your class must implement the Comparable interface and it must have its implementation of the equals() method.

HTH
Well just like PVector, i didn't suspect to have it's own equal method, it's also confusing

Copy code
  1. PVector a = new PVector();
  2. PVector b = new PVector();
  3. ArrayList<PVector> list = new ArrayList<PVector>();
  4. list.add(a);
  5. println(list.contains(b));

this will print true

It looks like the PVector equals(...) method (which gets called by ArrayList's contains(...)) returns true if both represent the same coordinate. When you use the no-args constructor they are both 0,0,0 so a.equals(b) is true.
You can overwrite the equals method, to make sure comparison is done as you want it to be:
Copy code
  1. public class Thing { public int value; public Thing (int x) { this.value = x; }  public boolean equals(Object object) { boolean sameSame = false; if (object != null && object instanceof Thing) { sameSame = this.value == ((Thing) object).value; } return sameSame; } }
(Taken from Stack Overflow user  ChristopheCVB on this discussion)