5 Commits

Author SHA1 Message Date
1322742b07 Fix #31: NPE when resizing SCollection due to null inherited bounds
All checks were successful
CI / build-and-test (pull_request) Successful in 19s
2026-03-28 00:55:43 +01:00
1db8ade18e docs: mark box selection as completed in TODO
All checks were successful
CI / build-and-test (push) Successful in 18s
2026-03-28 00:10:25 +01:00
1cabbd0224 Merge pull request 'test: add ResizeHandle enum unit tests' (#22) from issue-12/resize-handle-tests into master
All checks were successful
CI / build-and-test (push) Successful in 18s
Reviewed-on: #22
2026-03-27 18:06:55 +00:00
8635770755 test: add ResizeHandle enum unit tests
All checks were successful
CI / build-and-test (pull_request) Successful in 18s
Add 10 tests covering the ResizeHandle enum:
- Verify all 8 handles exist
- Verify each handle name (NW, N, NE, E, SE, S, SW, W)
- Verify each handle maps to the correct java.awt.Cursor constant
- Verify valueOf() round-trip for all handles
- Verify valueOf() with invalid name throws IllegalArgumentException

Closes #12
2026-03-27 18:06:49 +00:00
2ccea4b107 Merge pull request 'test: add SVGDraftman direct visitor tests' (#18) from issue-9/svg-draftman-tests into master
All checks were successful
CI / build-and-test (push) Successful in 17s
Reviewed-on: #18
2026-03-27 17:19:14 +00:00
6 changed files with 278 additions and 2 deletions

View File

@@ -1,6 +1,6 @@
# TODO # TODO
- [ ] Box selection (drag to select multiple shapes) - [X] Box selection (drag to select multiple shapes)
- [ ] Undo/redo stack - [ ] Undo/redo stack
- [ ] Copy/paste functionality - [ ] Copy/paste functionality
- [ ] Group/Ungroup shapes - [ ] Group/Ungroup shapes

View File

@@ -77,6 +77,14 @@ public abstract class AbstractShape implements Shape {
return new Rectangle(this.bounds); return new Rectangle(this.bounds);
} }
public void setBounds(Rectangle newBounds) {
this.bounds.setBounds(newBounds);
onBoundsChanged();
}
protected void onBoundsChanged() {
}
@Override @Override
public String toString() { public String toString() {
return String.format("x=%d, y=%d, width=%d, height=%d", bounds.x, bounds.y, bounds.width, bounds.height); return String.format("x=%d, y=%d, width=%d, height=%d", bounds.x, bounds.y, bounds.width, bounds.height);

View File

@@ -53,6 +53,14 @@ public class SCircle extends AbstractShape {
return radius; return radius;
} }
@Override
protected void onBoundsChanged() {
int diameter = Math.max(bounds.width, bounds.height);
bounds.width = diameter;
bounds.height = diameter;
this.radius = diameter / 2;
}
public static SCircle create(int x, int y, int radius) { public static SCircle create(int x, int y, int radius) {
return create(x, y, radius, Color.BLACK); return create(x, y, radius, Color.BLACK);
} }

View File

@@ -9,7 +9,6 @@ import ovh.gasser.newshapes.attributes.ColorAttributes;
import ovh.gasser.newshapes.attributes.SelectionAttributes; import ovh.gasser.newshapes.attributes.SelectionAttributes;
import ovh.gasser.newshapes.util.Streamable; import ovh.gasser.newshapes.util.Streamable;
import javax.swing.text.html.Option;
import java.awt.*; import java.awt.*;
import java.util.*; import java.util.*;
import java.util.List; import java.util.List;
@@ -19,6 +18,7 @@ public class SCollection extends AbstractShape implements Streamable<Shape> {
private final List<Shape> children; private final List<Shape> children;
private SCollection(Shape... shapes) { private SCollection(Shape... shapes) {
super(new Rectangle());
this.children = new ArrayList<>(List.of(shapes)); this.children = new ArrayList<>(List.of(shapes));
} }
@@ -57,6 +57,29 @@ public class SCollection extends AbstractShape implements Streamable<Shape> {
children.forEach(s -> s.translate(dx, dy)); children.forEach(s -> s.translate(dx, dy));
} }
@Override
public void resize(ResizeHandle handle, int dx, int dy) {
if (children.isEmpty()) {
return;
}
Rectangle currentBounds = getBounds();
Rectangle resizedBounds = resizeBounds(currentBounds, handle, dx, dy);
double scaleX = resizedBounds.width / (double) currentBounds.width;
double scaleY = resizedBounds.height / (double) currentBounds.height;
for (Shape child : children) {
Rectangle childBounds = child.getBounds();
Rectangle targetChildBounds = new Rectangle(
resizedBounds.x + (int) Math.round((childBounds.x - currentBounds.x) * scaleX),
resizedBounds.y + (int) Math.round((childBounds.y - currentBounds.y) * scaleY),
Math.max(1, (int) Math.round(childBounds.width * scaleX)),
Math.max(1, (int) Math.round(childBounds.height * scaleY))
);
resizeChild(child, targetChildBounds);
}
}
@Override @Override
public Iterator<Shape> iterator() { public Iterator<Shape> iterator() {
return children.iterator(); return children.iterator();
@@ -71,12 +94,45 @@ public class SCollection extends AbstractShape implements Streamable<Shape> {
children.add(s); children.add(s);
} }
public void clear() {
children.clear();
}
public void replaceWith(SCollection other) {
if (other == this) {
return;
}
clear();
other.stream().forEach(children::add);
}
public void add(int index, Shape s) {
children.add(index, s);
}
public void insert(int index, Shape s) {
children.add(Math.max(0, Math.min(index, children.size())), s);
}
public void remove(Shape s) { public void remove(Shape s) {
if (!children.remove(s)) { if (!children.remove(s)) {
logger.error("Unable to delete shape: {}", s); logger.error("Unable to delete shape: {}", s);
} }
} }
public int indexOf(Shape s) {
return children.indexOf(s);
}
public int size() {
return children.size();
}
public boolean contains(Shape s) {
return children.contains(s);
}
@Override @Override
public Attributes getAttributes(String key) { public Attributes getAttributes(String key) {
if (key.equals(ColorAttributes.ID)) { if (key.equals(ColorAttributes.ID)) {
@@ -110,4 +166,69 @@ public class SCollection extends AbstractShape implements Streamable<Shape> {
collection.addAttributes(new SelectionAttributes()); collection.addAttributes(new SelectionAttributes());
return collection; return collection;
} }
private static Rectangle resizeBounds(Rectangle bounds, ResizeHandle handle, int dx, int dy) {
Rectangle resizedBounds = new Rectangle(bounds);
switch (handle) {
case E -> resizedBounds.width += dx;
case W -> {
resizedBounds.x += dx;
resizedBounds.width -= dx;
}
case S -> resizedBounds.height += dy;
case N -> {
resizedBounds.y += dy;
resizedBounds.height -= dy;
}
case SE -> {
resizedBounds.width += dx;
resizedBounds.height += dy;
}
case SW -> {
resizedBounds.x += dx;
resizedBounds.width -= dx;
resizedBounds.height += dy;
}
case NE -> {
resizedBounds.width += dx;
resizedBounds.y += dy;
resizedBounds.height -= dy;
}
case NW -> {
resizedBounds.x += dx;
resizedBounds.width -= dx;
resizedBounds.y += dy;
resizedBounds.height -= dy;
}
}
if (resizedBounds.width < 1) {
resizedBounds.width = 1;
}
if (resizedBounds.height < 1) {
resizedBounds.height = 1;
}
return resizedBounds;
}
private static void resizeChild(Shape child, Rectangle targetChildBounds) {
if (child instanceof SCollection collection) {
Rectangle currentBounds = collection.getBounds();
collection.translate(targetChildBounds.x - currentBounds.x, targetChildBounds.y - currentBounds.y);
Rectangle translatedBounds = collection.getBounds();
collection.resize(
ResizeHandle.SE,
targetChildBounds.width - translatedBounds.width,
targetChildBounds.height - translatedBounds.height
);
return;
}
if (child instanceof AbstractShape abstractShape) {
abstractShape.setBounds(targetChildBounds);
}
}
} }

View File

@@ -0,0 +1,94 @@
package ovh.gasser.newshapes.shapes;
import org.junit.jupiter.api.Test;
import java.awt.Cursor;
import static org.junit.jupiter.api.Assertions.*;
class ResizeHandleTest {
@Test
void testEightHandlesExist() {
assertEquals(8, ResizeHandle.values().length,
"ResizeHandle enum should define exactly 8 handles");
}
@Test
void testAllHandleNamesExist() {
assertNotNull(ResizeHandle.valueOf("NW"), "Handle NW should exist");
assertNotNull(ResizeHandle.valueOf("N"), "Handle N should exist");
assertNotNull(ResizeHandle.valueOf("NE"), "Handle NE should exist");
assertNotNull(ResizeHandle.valueOf("E"), "Handle E should exist");
assertNotNull(ResizeHandle.valueOf("SE"), "Handle SE should exist");
assertNotNull(ResizeHandle.valueOf("S"), "Handle S should exist");
assertNotNull(ResizeHandle.valueOf("SW"), "Handle SW should exist");
assertNotNull(ResizeHandle.valueOf("W"), "Handle W should exist");
}
@Test
void testNwMapsToCorrectCursorType() {
assertEquals(Cursor.NW_RESIZE_CURSOR, ResizeHandle.NW.getCursorType(),
"NW handle should map to Cursor.NW_RESIZE_CURSOR");
}
@Test
void testNMapsToCorrectCursorType() {
assertEquals(Cursor.N_RESIZE_CURSOR, ResizeHandle.N.getCursorType(),
"N handle should map to Cursor.N_RESIZE_CURSOR");
}
@Test
void testNeMapsToCorrectCursorType() {
assertEquals(Cursor.NE_RESIZE_CURSOR, ResizeHandle.NE.getCursorType(),
"NE handle should map to Cursor.NE_RESIZE_CURSOR");
}
@Test
void testEMapsToCorrectCursorType() {
assertEquals(Cursor.E_RESIZE_CURSOR, ResizeHandle.E.getCursorType(),
"E handle should map to Cursor.E_RESIZE_CURSOR");
}
@Test
void testSeMapsToCorrectCursorType() {
assertEquals(Cursor.SE_RESIZE_CURSOR, ResizeHandle.SE.getCursorType(),
"SE handle should map to Cursor.SE_RESIZE_CURSOR");
}
@Test
void testSMapsToCorrectCursorType() {
assertEquals(Cursor.S_RESIZE_CURSOR, ResizeHandle.S.getCursorType(),
"S handle should map to Cursor.S_RESIZE_CURSOR");
}
@Test
void testSwMapsToCorrectCursorType() {
assertEquals(Cursor.SW_RESIZE_CURSOR, ResizeHandle.SW.getCursorType(),
"SW handle should map to Cursor.SW_RESIZE_CURSOR");
}
@Test
void testWMapsToCorrectCursorType() {
assertEquals(Cursor.W_RESIZE_CURSOR, ResizeHandle.W.getCursorType(),
"W handle should map to Cursor.W_RESIZE_CURSOR");
}
@Test
void testValueOfReturnsCorrectConstant() {
assertSame(ResizeHandle.NW, ResizeHandle.valueOf("NW"), "valueOf(\"NW\") should return ResizeHandle.NW");
assertSame(ResizeHandle.N, ResizeHandle.valueOf("N"), "valueOf(\"N\") should return ResizeHandle.N");
assertSame(ResizeHandle.NE, ResizeHandle.valueOf("NE"), "valueOf(\"NE\") should return ResizeHandle.NE");
assertSame(ResizeHandle.E, ResizeHandle.valueOf("E"), "valueOf(\"E\") should return ResizeHandle.E");
assertSame(ResizeHandle.SE, ResizeHandle.valueOf("SE"), "valueOf(\"SE\") should return ResizeHandle.SE");
assertSame(ResizeHandle.S, ResizeHandle.valueOf("S"), "valueOf(\"S\") should return ResizeHandle.S");
assertSame(ResizeHandle.SW, ResizeHandle.valueOf("SW"), "valueOf(\"SW\") should return ResizeHandle.SW");
assertSame(ResizeHandle.W, ResizeHandle.valueOf("W"), "valueOf(\"W\") should return ResizeHandle.W");
}
@Test
void testValueOfWithInvalidNameThrowsIllegalArgumentException() {
assertThrows(IllegalArgumentException.class,
() -> ResizeHandle.valueOf("INVALID"),
"valueOf() with an unknown name should throw IllegalArgumentException");
}
}

View File

@@ -114,6 +114,51 @@ class SCollectionTest {
assertEquals(30, bounds.y); assertEquals(30, bounds.y);
} }
@Test
void testResizeEmptyCollectionIsNoOp() {
SCollection collection = SCollection.of();
assertDoesNotThrow(() -> collection.resize(ResizeHandle.SE, 25, 25));
assertEquals(new Rectangle(App.WIN_SIZE), collection.getBounds());
}
@Test
void testResizeSingleChildResizesChildAndCollection() {
SRectangle rect = SRectangle.create(10, 20, 100, 50);
SCollection collection = SCollection.of(rect);
collection.resize(ResizeHandle.E, 20, 0);
assertEquals(new Rectangle(10, 20, 120, 50), rect.getBounds());
assertEquals(new Rectangle(10, 20, 120, 50), collection.getBounds());
}
@Test
void testResizeMultipleChildrenScalesChildrenProportionally() {
SRectangle rect1 = SRectangle.create(0, 0, 10, 10);
SRectangle rect2 = SRectangle.create(20, 10, 20, 10);
SCollection collection = SCollection.of(rect1, rect2);
collection.resize(ResizeHandle.SE, 20, 10);
assertEquals(new Rectangle(0, 0, 15, 15), rect1.getBounds());
assertEquals(new Rectangle(30, 15, 30, 15), rect2.getBounds());
assertEquals(new Rectangle(0, 0, 60, 30), collection.getBounds());
}
@Test
void testResizeFromNorthWestRepositionsAndScalesChildren() {
SRectangle rect1 = SRectangle.create(0, 0, 10, 10);
SRectangle rect2 = SRectangle.create(10, 10, 10, 10);
SCollection collection = SCollection.of(rect1, rect2);
collection.resize(ResizeHandle.NW, 10, 10);
assertEquals(new Rectangle(10, 10, 5, 5), rect1.getBounds());
assertEquals(new Rectangle(15, 15, 5, 5), rect2.getBounds());
assertEquals(new Rectangle(10, 10, 10, 10), collection.getBounds());
}
@Test @Test
void testClone() { void testClone() {
SRectangle rect = SRectangle.create(10, 20, 100, 50); SRectangle rect = SRectangle.create(10, 20, 100, 50);