Compare commits
12 Commits
bf6636da3b
...
fix/issue-
| Author | SHA1 | Date | |
|---|---|---|---|
| 1322742b07 | |||
| 1db8ade18e | |||
| 1cabbd0224 | |||
| 8635770755 | |||
| 2ccea4b107 | |||
| 3623928bb6 | |||
| 8d56e941fc | |||
| d3c993eb2d | |||
| 4f4eb00baa | |||
| bcbbcef321 | |||
| 692a03ee88 | |||
| e83a6ba7d1 |
2
TODO.md
2
TODO.md
@@ -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
|
||||||
|
|||||||
@@ -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);
|
||||||
|
|||||||
@@ -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);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,129 @@
|
|||||||
|
package ovh.gasser.newshapes.attributes;
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
|
import java.awt.*;
|
||||||
|
|
||||||
|
import static org.junit.jupiter.api.Assertions.*;
|
||||||
|
|
||||||
|
class ColorAttributesTest {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testConstructorStoresFilledFlag() {
|
||||||
|
ColorAttributes attrs = new ColorAttributes(true, false, Color.RED, Color.BLACK);
|
||||||
|
assertTrue(attrs.filled, "filled flag should be true when constructed with true");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testConstructorStoresStrokedFlag() {
|
||||||
|
ColorAttributes attrs = new ColorAttributes(false, true, Color.RED, Color.BLACK);
|
||||||
|
assertTrue(attrs.stroked, "stroked flag should be true when constructed with true");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testConstructorStoresFilledColor() {
|
||||||
|
ColorAttributes attrs = new ColorAttributes(true, false, Color.BLUE, Color.BLACK);
|
||||||
|
assertEquals(Color.BLUE, attrs.filledColor, "filledColor should match the constructor argument");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testConstructorStoresStrokedColor() {
|
||||||
|
ColorAttributes attrs = new ColorAttributes(false, true, Color.RED, Color.GREEN);
|
||||||
|
assertEquals(Color.GREEN, attrs.strokedColor, "strokedColor should match the constructor argument");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testFilledAndStrokedBothTrue() {
|
||||||
|
ColorAttributes attrs = new ColorAttributes(true, true, Color.RED, Color.BLUE);
|
||||||
|
assertTrue(attrs.filled);
|
||||||
|
assertTrue(attrs.stroked);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testFilledAndStrokedBothFalse() {
|
||||||
|
ColorAttributes attrs = new ColorAttributes(false, false, Color.RED, Color.BLUE);
|
||||||
|
assertFalse(attrs.filled);
|
||||||
|
assertFalse(attrs.stroked);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testNullFilledColor() {
|
||||||
|
ColorAttributes attrs = new ColorAttributes(true, false, null, Color.BLACK);
|
||||||
|
assertNull(attrs.filledColor, "filledColor should accept null");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testNullStrokedColor() {
|
||||||
|
ColorAttributes attrs = new ColorAttributes(false, true, Color.RED, null);
|
||||||
|
assertNull(attrs.strokedColor, "strokedColor should accept null");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testBothColorsNull() {
|
||||||
|
ColorAttributes attrs = new ColorAttributes(false, false, null, null);
|
||||||
|
assertNull(attrs.filledColor);
|
||||||
|
assertNull(attrs.strokedColor);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testGetIDReturnsCorrectValue() {
|
||||||
|
ColorAttributes attrs = new ColorAttributes(false, false, Color.RED, Color.BLACK);
|
||||||
|
assertEquals(ColorAttributes.ID, attrs.getID());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testIDConstant() {
|
||||||
|
assertEquals("COLOR_ATTRS", ColorAttributes.ID);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testImplementsAttributes() {
|
||||||
|
ColorAttributes attrs = new ColorAttributes(false, false, Color.RED, Color.BLACK);
|
||||||
|
assertInstanceOf(Attributes.class, attrs);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testToStringContainsAllFields() {
|
||||||
|
ColorAttributes attrs = new ColorAttributes(true, false, Color.RED, Color.BLUE);
|
||||||
|
String str = attrs.toString();
|
||||||
|
assertTrue(str.contains("filled=true"), "toString should contain filled value");
|
||||||
|
assertTrue(str.contains("stroked=false"), "toString should contain stroked value");
|
||||||
|
assertTrue(str.contains("filledColor"), "toString should contain filledColor");
|
||||||
|
assertTrue(str.contains("strokedColor"), "toString should contain strokedColor");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testToStringWithNullColors() {
|
||||||
|
ColorAttributes attrs = new ColorAttributes(false, false, null, null);
|
||||||
|
String str = attrs.toString();
|
||||||
|
assertNotNull(str, "toString should not throw with null colors");
|
||||||
|
assertTrue(str.contains("filledColor=null"), "toString should show null filledColor");
|
||||||
|
assertTrue(str.contains("strokedColor=null"), "toString should show null strokedColor");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTwoInstancesAreIndependent() {
|
||||||
|
ColorAttributes attrs1 = new ColorAttributes(true, false, Color.RED, Color.BLACK);
|
||||||
|
ColorAttributes attrs2 = new ColorAttributes(false, true, Color.BLUE, Color.GREEN);
|
||||||
|
|
||||||
|
assertTrue(attrs1.filled);
|
||||||
|
assertFalse(attrs2.filled);
|
||||||
|
assertFalse(attrs1.stroked);
|
||||||
|
assertTrue(attrs2.stroked);
|
||||||
|
assertEquals(Color.RED, attrs1.filledColor);
|
||||||
|
assertEquals(Color.BLUE, attrs2.filledColor);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testFieldsAreImmutable() {
|
||||||
|
Color fillColor = Color.RED;
|
||||||
|
Color strokeColor = Color.BLACK;
|
||||||
|
ColorAttributes attrs = new ColorAttributes(true, true, fillColor, strokeColor);
|
||||||
|
|
||||||
|
// Since fields are final, verify they retain their values
|
||||||
|
assertEquals(Color.RED, attrs.filledColor);
|
||||||
|
assertEquals(Color.BLACK, attrs.strokedColor);
|
||||||
|
assertTrue(attrs.filled);
|
||||||
|
assertTrue(attrs.stroked);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,72 @@
|
|||||||
|
package ovh.gasser.newshapes.attributes;
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
|
import static org.junit.jupiter.api.Assertions.*;
|
||||||
|
|
||||||
|
class SelectionAttributesTest {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testDefaultConstructorIsUnselected() {
|
||||||
|
SelectionAttributes attrs = new SelectionAttributes();
|
||||||
|
assertFalse(attrs.selected, "Default constructor should create unselected state");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testConstructorWithTrue() {
|
||||||
|
SelectionAttributes attrs = new SelectionAttributes(true);
|
||||||
|
assertTrue(attrs.selected, "Constructor with true should create selected state");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testConstructorWithFalse() {
|
||||||
|
SelectionAttributes attrs = new SelectionAttributes(false);
|
||||||
|
assertFalse(attrs.selected, "Constructor with false should create unselected state");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testGetIDReturnsCorrectValue() {
|
||||||
|
SelectionAttributes attrs = new SelectionAttributes();
|
||||||
|
assertEquals(SelectionAttributes.ID, attrs.getID());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testIDConstant() {
|
||||||
|
assertEquals("SELECTION_ATTRS", SelectionAttributes.ID);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testImplementsAttributes() {
|
||||||
|
SelectionAttributes attrs = new SelectionAttributes();
|
||||||
|
assertInstanceOf(Attributes.class, attrs);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testSelectedFieldCanBeToggled() {
|
||||||
|
SelectionAttributes attrs = new SelectionAttributes(false);
|
||||||
|
assertFalse(attrs.selected);
|
||||||
|
|
||||||
|
attrs.selected = true;
|
||||||
|
assertTrue(attrs.selected);
|
||||||
|
|
||||||
|
attrs.selected = false;
|
||||||
|
assertFalse(attrs.selected);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTwoInstancesAreIndependent() {
|
||||||
|
SelectionAttributes attrs1 = new SelectionAttributes(true);
|
||||||
|
SelectionAttributes attrs2 = new SelectionAttributes(false);
|
||||||
|
|
||||||
|
assertNotEquals(attrs1.selected, attrs2.selected);
|
||||||
|
|
||||||
|
attrs2.selected = true;
|
||||||
|
assertTrue(attrs1.selected);
|
||||||
|
assertTrue(attrs2.selected);
|
||||||
|
|
||||||
|
// Mutating one doesn't affect the other
|
||||||
|
attrs1.selected = false;
|
||||||
|
assertFalse(attrs1.selected);
|
||||||
|
assertTrue(attrs2.selected);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -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");
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -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);
|
||||||
|
|||||||
@@ -0,0 +1,409 @@
|
|||||||
|
package ovh.gasser.newshapes.ui.visitors;
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
|
import org.junit.jupiter.api.Nested;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
import ovh.gasser.newshapes.attributes.ColorAttributes;
|
||||||
|
import ovh.gasser.newshapes.shapes.*;
|
||||||
|
|
||||||
|
import java.awt.*;
|
||||||
|
import java.io.PrintWriter;
|
||||||
|
import java.io.StringWriter;
|
||||||
|
|
||||||
|
import static org.junit.jupiter.api.Assertions.*;
|
||||||
|
|
||||||
|
class HTMLDraftmanTest {
|
||||||
|
|
||||||
|
private StringWriter htmlBuffer;
|
||||||
|
private StringWriter cssBuffer;
|
||||||
|
private PrintWriter htmlWriter;
|
||||||
|
private PrintWriter cssWriter;
|
||||||
|
private HTMLDraftman draftman;
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
void setUp() {
|
||||||
|
htmlBuffer = new StringWriter();
|
||||||
|
cssBuffer = new StringWriter();
|
||||||
|
htmlWriter = new PrintWriter(htmlBuffer);
|
||||||
|
cssWriter = new PrintWriter(cssBuffer);
|
||||||
|
draftman = new HTMLDraftman(htmlWriter, cssWriter);
|
||||||
|
}
|
||||||
|
|
||||||
|
private String html() {
|
||||||
|
htmlWriter.flush();
|
||||||
|
return htmlBuffer.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
private String css() {
|
||||||
|
cssWriter.flush();
|
||||||
|
return cssBuffer.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── visitRectangle ──────────────────────────────────────────────
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
class VisitRectangleTests {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testRectangleProducesDivWithId() {
|
||||||
|
SRectangle rect = SRectangle.create(10, 20, 100, 50);
|
||||||
|
draftman.visitRectangle(rect);
|
||||||
|
|
||||||
|
assertTrue(html().contains("<div id=\"rec"), "Should produce a div with rec id");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testRectangleCssContainsCorrectPosition() {
|
||||||
|
SRectangle rect = SRectangle.create(10, 20, 100, 50);
|
||||||
|
draftman.visitRectangle(rect);
|
||||||
|
|
||||||
|
String cssOut = css();
|
||||||
|
assertTrue(cssOut.contains("top:20px"), "CSS should contain correct top");
|
||||||
|
assertTrue(cssOut.contains("left:10px"), "CSS should contain correct left");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testRectangleCssContainsCorrectDimensions() {
|
||||||
|
SRectangle rect = SRectangle.create(10, 20, 100, 50);
|
||||||
|
draftman.visitRectangle(rect);
|
||||||
|
|
||||||
|
String cssOut = css();
|
||||||
|
assertTrue(cssOut.contains("width:100px"), "CSS should contain correct width");
|
||||||
|
assertTrue(cssOut.contains("height:50px"), "CSS should contain correct height");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testRectangleCssPositionIsAbsolute() {
|
||||||
|
SRectangle rect = SRectangle.create(0, 0, 30, 30);
|
||||||
|
draftman.visitRectangle(rect);
|
||||||
|
|
||||||
|
assertTrue(css().contains("position:absolute"), "CSS should use absolute positioning");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testRectangleHtmlIdMatchesCssSelector() {
|
||||||
|
SRectangle rect = SRectangle.create(0, 0, 30, 30);
|
||||||
|
draftman.visitRectangle(rect);
|
||||||
|
|
||||||
|
String id = "rec" + rect.hashCode();
|
||||||
|
assertTrue(html().contains("id=\"" + id + "\""), "HTML should contain div with correct id");
|
||||||
|
assertTrue(css().contains("#" + id), "CSS should reference same id");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testRectangleFilledAndStroked() {
|
||||||
|
SRectangle rect = SRectangle.create(0, 0, 30, 30);
|
||||||
|
rect.addAttributes(new ColorAttributes(true, true, Color.RED, Color.BLUE));
|
||||||
|
draftman.visitRectangle(rect);
|
||||||
|
|
||||||
|
String cssOut = css();
|
||||||
|
assertTrue(cssOut.contains("background:"), "CSS should contain background");
|
||||||
|
assertTrue(cssOut.contains("border:"), "CSS should contain border");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testRectangleFilledOnly() {
|
||||||
|
SRectangle rect = SRectangle.create(0, 0, 30, 30);
|
||||||
|
rect.addAttributes(new ColorAttributes(true, false, Color.RED, Color.BLACK));
|
||||||
|
draftman.visitRectangle(rect);
|
||||||
|
|
||||||
|
String cssOut = css();
|
||||||
|
assertTrue(cssOut.contains("background:"), "CSS should contain background for filled");
|
||||||
|
assertFalse(cssOut.contains("border:"), "CSS should not contain border when not stroked");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testRectangleStrokedOnly() {
|
||||||
|
SRectangle rect = SRectangle.create(0, 0, 30, 30);
|
||||||
|
rect.addAttributes(new ColorAttributes(false, true, Color.RED, Color.BLUE));
|
||||||
|
draftman.visitRectangle(rect);
|
||||||
|
|
||||||
|
String cssOut = css();
|
||||||
|
assertTrue(cssOut.contains("border:"), "CSS should contain border for stroked");
|
||||||
|
assertTrue(cssOut.contains("background:#ffffff"), "CSS should have white background when not filled");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testRectangleNeitherFilledNorStroked() {
|
||||||
|
SRectangle rect = SRectangle.create(0, 0, 30, 30);
|
||||||
|
rect.addAttributes(new ColorAttributes(false, false, Color.RED, Color.BLUE));
|
||||||
|
draftman.visitRectangle(rect);
|
||||||
|
|
||||||
|
String cssOut = css();
|
||||||
|
assertFalse(cssOut.contains("background:"), "CSS should not contain background");
|
||||||
|
assertFalse(cssOut.contains("border:"), "CSS should not contain border");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── visitCircle ─────────────────────────────────────────────────
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
class VisitCircleTests {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testCircleProducesDivWithClass() {
|
||||||
|
SCircle circle = SCircle.create(50, 60, 30);
|
||||||
|
draftman.visitCircle(circle);
|
||||||
|
|
||||||
|
assertTrue(html().contains("circle"), "HTML should contain circle class div");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testCircleCssContainsBorderRadius() {
|
||||||
|
SCircle circle = SCircle.create(50, 60, 30);
|
||||||
|
draftman.visitCircle(circle);
|
||||||
|
|
||||||
|
String cssOut = css();
|
||||||
|
assertTrue(cssOut.contains("border-radius:"), "CSS should contain border-radius");
|
||||||
|
assertTrue(cssOut.contains("-webkit-border-radius:"), "CSS should contain webkit prefix");
|
||||||
|
assertTrue(cssOut.contains("-moz-border-radius:"), "CSS should contain moz prefix");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testCircleCssContainsCorrectPosition() {
|
||||||
|
SCircle circle = SCircle.create(50, 60, 30);
|
||||||
|
draftman.visitCircle(circle);
|
||||||
|
|
||||||
|
String cssOut = css();
|
||||||
|
assertTrue(cssOut.contains("top:60px"), "CSS should contain correct top");
|
||||||
|
assertTrue(cssOut.contains("left:50px"), "CSS should contain correct left");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testCircleHtmlClassMatchesCssSelector() {
|
||||||
|
SCircle circle = SCircle.create(0, 0, 20);
|
||||||
|
draftman.visitCircle(circle);
|
||||||
|
|
||||||
|
String className = "circle" + circle.hashCode();
|
||||||
|
assertTrue(html().contains(className), "HTML should have circle class");
|
||||||
|
assertTrue(css().contains("." + className), "CSS should reference same class");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── visitTriangle ───────────────────────────────────────────────
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
class VisitTriangleTests {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTriangleProducesDivWithClass() {
|
||||||
|
STriangle tri = STriangle.create(10, 20, 40, Color.RED, Color.BLACK);
|
||||||
|
draftman.visitTriangle(tri);
|
||||||
|
|
||||||
|
assertTrue(html().contains("triangle"), "HTML should contain triangle class div");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTriangleCssContainsBorderTrick() {
|
||||||
|
STriangle tri = STriangle.create(10, 20, 40, Color.RED, Color.BLACK);
|
||||||
|
draftman.visitTriangle(tri);
|
||||||
|
|
||||||
|
String cssOut = css();
|
||||||
|
assertTrue(cssOut.contains("border-left-width:"), "CSS should use border-left-width");
|
||||||
|
assertTrue(cssOut.contains("border-right-width:"), "CSS should use border-right-width");
|
||||||
|
assertTrue(cssOut.contains("border-bottom:"), "CSS should use border-bottom");
|
||||||
|
assertTrue(cssOut.contains("width: 0px"), "CSS should set width to 0");
|
||||||
|
assertTrue(cssOut.contains("height: 0px"), "CSS should set height to 0");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTriangleCssContainsCorrectPosition() {
|
||||||
|
STriangle tri = STriangle.create(10, 20, 40, Color.RED, Color.BLACK);
|
||||||
|
draftman.visitTriangle(tri);
|
||||||
|
|
||||||
|
String cssOut = css();
|
||||||
|
assertTrue(cssOut.contains("top: 20px"), "CSS should contain correct top");
|
||||||
|
assertTrue(cssOut.contains("left: 10px"), "CSS should contain correct left");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTriangleCssContainsFillColor() {
|
||||||
|
STriangle tri = STriangle.create(0, 0, 20, Color.BLUE, Color.BLACK);
|
||||||
|
tri.addAttributes(new ColorAttributes(true, false, Color.BLUE, Color.BLACK));
|
||||||
|
draftman.visitTriangle(tri);
|
||||||
|
|
||||||
|
assertTrue(css().contains("#0000ff"), "CSS should contain the triangle fill color");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTriangleHtmlClassMatchesCssSelector_regressionHashCode() {
|
||||||
|
// Regression test: visitTriangle previously used this.hashCode()
|
||||||
|
// instead of sTriangle.hashCode(), causing HTML class and CSS
|
||||||
|
// selector mismatch when rendered by a different draftman instance.
|
||||||
|
STriangle tri = STriangle.create(5, 5, 30, Color.RED, Color.BLACK);
|
||||||
|
draftman.visitTriangle(tri);
|
||||||
|
|
||||||
|
String htmlOut = html();
|
||||||
|
String cssOut = css();
|
||||||
|
|
||||||
|
// Extract the class name from HTML: class="triangleNNN"
|
||||||
|
int classStart = htmlOut.indexOf("triangle");
|
||||||
|
assertNotEquals(-1, classStart, "HTML should contain triangle class");
|
||||||
|
int classEnd = htmlOut.indexOf("\"", classStart);
|
||||||
|
String htmlClassName = htmlOut.substring(classStart, classEnd);
|
||||||
|
|
||||||
|
// CSS should use the same class name
|
||||||
|
assertTrue(cssOut.contains("." + htmlClassName),
|
||||||
|
"CSS selector should match HTML class — " +
|
||||||
|
"if this fails, visitTriangle may be using this.hashCode() instead of sTriangle.hashCode()");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── visitText ───────────────────────────────────────────────────
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
class VisitTextTests {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTextProducesDivWithContent() {
|
||||||
|
SText text = SText.create(10, 20, "Hello");
|
||||||
|
draftman.visitText(text);
|
||||||
|
|
||||||
|
assertTrue(html().contains(">Hello</div>"), "HTML should contain text content in div");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTextHtmlIdMatchesCssSelector() {
|
||||||
|
SText text = SText.create(0, 0, "Test");
|
||||||
|
draftman.visitText(text);
|
||||||
|
|
||||||
|
String id = "txt" + text.hashCode();
|
||||||
|
assertTrue(html().contains("id=\"" + id + "\""), "HTML should contain text div with correct id");
|
||||||
|
assertTrue(css().contains("#" + id), "CSS should reference same text id");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTextCssContainsPosition() {
|
||||||
|
SText text = SText.create(15, 25, "Pos");
|
||||||
|
draftman.visitText(text);
|
||||||
|
|
||||||
|
String cssOut = css();
|
||||||
|
assertTrue(cssOut.contains("top:25px"), "CSS should contain correct top");
|
||||||
|
assertTrue(cssOut.contains("left:15px"), "CSS should contain correct left");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTextCssContainsDefaultFontAttributes() {
|
||||||
|
SText text = SText.create(0, 0, "Font");
|
||||||
|
draftman.visitText(text);
|
||||||
|
|
||||||
|
String cssOut = css();
|
||||||
|
assertTrue(cssOut.contains("font-family:"), "CSS should contain font-family");
|
||||||
|
assertTrue(cssOut.contains("font-size:16px"), "CSS should contain default font size");
|
||||||
|
assertTrue(cssOut.contains("font-style:normal"), "CSS should contain normal font style");
|
||||||
|
assertTrue(cssOut.contains("font-weight:normal"), "CSS should contain normal font weight");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTextCssContainsColor() {
|
||||||
|
SText text = SText.create(0, 0, "Colored");
|
||||||
|
text.addAttributes(new ColorAttributes(true, false, Color.RED, Color.BLACK));
|
||||||
|
draftman.visitText(text);
|
||||||
|
|
||||||
|
assertTrue(css().contains("color:#ff0000"), "CSS should contain fill color as text color");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTextCssWhiteSpaceNowrap() {
|
||||||
|
SText text = SText.create(0, 0, "NoWrap");
|
||||||
|
draftman.visitText(text);
|
||||||
|
|
||||||
|
assertTrue(css().contains("white-space: nowrap"), "CSS should prevent text wrapping");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTextWithNullColorAttributesFallsBackToBlack() {
|
||||||
|
SText text = SText.create(0, 0, "Fallback");
|
||||||
|
text.addAttributes(new ColorAttributes(false, false, null, null));
|
||||||
|
draftman.visitText(text);
|
||||||
|
|
||||||
|
assertTrue(css().contains("color:#000000"), "Should fall back to black with null colors");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── visitCollection ─────────────────────────────────────────────
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
class VisitCollectionTests {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testEmptyCollection() {
|
||||||
|
SCollection empty = SCollection.of();
|
||||||
|
draftman.visitCollection(empty);
|
||||||
|
|
||||||
|
assertEquals("", html(), "Empty collection should produce no HTML");
|
||||||
|
assertEquals("", css(), "Empty collection should produce no CSS");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testCollectionVisitsAllChildren() {
|
||||||
|
SCollection coll = SCollection.of(
|
||||||
|
SRectangle.create(0, 0, 10, 10),
|
||||||
|
SCircle.create(20, 20, 5)
|
||||||
|
);
|
||||||
|
draftman.visitCollection(coll);
|
||||||
|
|
||||||
|
assertTrue(html().contains("rec"), "Should visit rectangle");
|
||||||
|
assertTrue(html().contains("circle"), "Should visit circle");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testNestedCollectionVisitsAllDescendants() {
|
||||||
|
SCollection inner = SCollection.of(
|
||||||
|
SRectangle.create(0, 0, 5, 5)
|
||||||
|
);
|
||||||
|
SCollection outer = SCollection.of(
|
||||||
|
inner,
|
||||||
|
SCircle.create(10, 10, 3)
|
||||||
|
);
|
||||||
|
draftman.visitCollection(outer);
|
||||||
|
|
||||||
|
assertTrue(html().contains("rec"), "Should visit nested rectangle");
|
||||||
|
assertTrue(html().contains("circle"), "Should visit circle at outer level");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── generateHTML ────────────────────────────────────────────────
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
class GenerateHTMLTests {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testGenerateHTMLIncludesDoctype() {
|
||||||
|
SCollection model = SCollection.of();
|
||||||
|
draftman.generateHTML(model);
|
||||||
|
|
||||||
|
assertTrue(html().contains("<!DOCTYPE html>"), "Should include DOCTYPE");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testGenerateHTMLIncludesHtmlStructure() {
|
||||||
|
SCollection model = SCollection.of();
|
||||||
|
draftman.generateHTML(model);
|
||||||
|
|
||||||
|
String htmlOut = html();
|
||||||
|
assertTrue(htmlOut.contains("<html"), "Should include html tag");
|
||||||
|
assertTrue(htmlOut.contains("<head>"), "Should include head tag");
|
||||||
|
assertTrue(htmlOut.contains("<body>"), "Should include body tag");
|
||||||
|
assertTrue(htmlOut.contains("</body>"), "Should close body tag");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testGenerateHTMLReferencesStylesheet() {
|
||||||
|
SCollection model = SCollection.of();
|
||||||
|
draftman.generateHTML(model);
|
||||||
|
|
||||||
|
assertTrue(html().contains("style.css"), "Should reference CSS stylesheet");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testGenerateHTMLIncludesShapeContent() {
|
||||||
|
SCollection model = SCollection.of(
|
||||||
|
SRectangle.create(1, 2, 3, 4)
|
||||||
|
);
|
||||||
|
draftman.generateHTML(model);
|
||||||
|
|
||||||
|
assertTrue(html().contains("<div id=\"rec"), "Should include shape elements");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,370 @@
|
|||||||
|
package ovh.gasser.newshapes.ui.visitors;
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
|
import org.junit.jupiter.api.Nested;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
import ovh.gasser.newshapes.attributes.ColorAttributes;
|
||||||
|
import ovh.gasser.newshapes.shapes.*;
|
||||||
|
|
||||||
|
import java.awt.*;
|
||||||
|
import java.io.PrintWriter;
|
||||||
|
import java.io.StringWriter;
|
||||||
|
|
||||||
|
import static org.junit.jupiter.api.Assertions.*;
|
||||||
|
|
||||||
|
class SVGDraftmanTest {
|
||||||
|
|
||||||
|
private StringWriter buffer;
|
||||||
|
private PrintWriter writer;
|
||||||
|
private SVGDraftman draftman;
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
void setUp() {
|
||||||
|
buffer = new StringWriter();
|
||||||
|
writer = new PrintWriter(buffer);
|
||||||
|
draftman = new SVGDraftman(writer);
|
||||||
|
}
|
||||||
|
|
||||||
|
private String output() {
|
||||||
|
writer.flush();
|
||||||
|
return buffer.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── visitRectangle ──────────────────────────────────────────────
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
class VisitRectangleTests {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testRectangleElementWithCorrectDimensions() {
|
||||||
|
SRectangle rect = SRectangle.create(10, 20, 100, 50);
|
||||||
|
draftman.visitRectangle(rect);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("<rect"), "Should produce a <rect> element");
|
||||||
|
assertTrue(svg.contains("width=\"100\""), "Should have correct width");
|
||||||
|
assertTrue(svg.contains("height=\"50\""), "Should have correct height");
|
||||||
|
assertTrue(svg.contains("x=\"10\""), "Should have correct x");
|
||||||
|
assertTrue(svg.contains("y=\"20\""), "Should have correct y");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testRectangleFilledOnly() {
|
||||||
|
SRectangle rect = SRectangle.create(0, 0, 30, 30);
|
||||||
|
rect.addAttributes(new ColorAttributes(true, false, Color.RED, Color.BLACK));
|
||||||
|
draftman.visitRectangle(rect);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("fill=\"#ff0000\""), "Should contain fill color");
|
||||||
|
assertFalse(svg.contains("stroke="), "Should not contain stroke when stroked=false");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testRectangleStrokedOnly() {
|
||||||
|
SRectangle rect = SRectangle.create(0, 0, 30, 30);
|
||||||
|
rect.addAttributes(new ColorAttributes(false, true, Color.RED, Color.BLUE));
|
||||||
|
draftman.visitRectangle(rect);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("stroke=\"#0000ff\""), "Should contain stroke color");
|
||||||
|
assertTrue(svg.contains("stroke-width=\"1\""), "Should contain stroke-width");
|
||||||
|
assertTrue(svg.contains("fill=\"none\""), "Should have fill=none when not filled");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testRectangleFilledAndStroked() {
|
||||||
|
SRectangle rect = SRectangle.create(0, 0, 30, 30);
|
||||||
|
rect.addAttributes(new ColorAttributes(true, true, Color.RED, Color.BLUE));
|
||||||
|
draftman.visitRectangle(rect);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("fill=\"#ff0000\""), "Should contain fill color");
|
||||||
|
assertTrue(svg.contains("stroke=\"#0000ff\""), "Should contain stroke color");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testRectangleNeitherFilledNorStroked() {
|
||||||
|
SRectangle rect = SRectangle.create(0, 0, 30, 30);
|
||||||
|
rect.addAttributes(new ColorAttributes(false, false, Color.RED, Color.BLUE));
|
||||||
|
draftman.visitRectangle(rect);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("fill=\"none\""), "Should have fill=none");
|
||||||
|
assertFalse(svg.contains("stroke="), "Should not have stroke attribute");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── visitCircle ─────────────────────────────────────────────────
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
class VisitCircleTests {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testCircleElementWithCorrectAttributes() {
|
||||||
|
SCircle circle = SCircle.create(50, 60, 30);
|
||||||
|
draftman.visitCircle(circle);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("<circle"), "Should produce a <circle> element");
|
||||||
|
// cx = x + r = 50 + 30 = 80, cy = y + r = 60 + 30 = 90
|
||||||
|
assertTrue(svg.contains("cx=\"80\""), "cx should be x + radius");
|
||||||
|
assertTrue(svg.contains("cy=\"90\""), "cy should be y + radius");
|
||||||
|
assertTrue(svg.contains("r=\"30\""), "Should have correct radius");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testCircleFilledOnly() {
|
||||||
|
SCircle circle = SCircle.create(0, 0, 20);
|
||||||
|
circle.addAttributes(new ColorAttributes(true, false, Color.GREEN, Color.BLACK));
|
||||||
|
draftman.visitCircle(circle);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("fill=\"#00ff00\""), "Should contain fill color");
|
||||||
|
assertFalse(svg.contains("stroke="), "Should not contain stroke");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testCircleStrokedOnly() {
|
||||||
|
SCircle circle = SCircle.create(0, 0, 20);
|
||||||
|
circle.addAttributes(new ColorAttributes(false, true, Color.RED, Color.MAGENTA));
|
||||||
|
draftman.visitCircle(circle);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("stroke=\"#ff00ff\""), "Should contain stroke color");
|
||||||
|
assertTrue(svg.contains("fill=\"none\""), "Should have fill=none");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testCircleFilledAndStroked() {
|
||||||
|
SCircle circle = SCircle.create(0, 0, 15);
|
||||||
|
circle.addAttributes(new ColorAttributes(true, true, Color.YELLOW, Color.BLACK));
|
||||||
|
draftman.visitCircle(circle);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("fill=\"#ffff00\""), "Should contain fill color");
|
||||||
|
assertTrue(svg.contains("stroke=\"#000000\""), "Should contain stroke color");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── visitTriangle ───────────────────────────────────────────────
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
class VisitTriangleTests {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTriangleProducesPolygonElement() {
|
||||||
|
STriangle tri = STriangle.create(10, 20, 40, Color.RED, Color.BLACK);
|
||||||
|
draftman.visitTriangle(tri);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("<polygon"), "Should produce a <polygon> element");
|
||||||
|
assertTrue(svg.contains("points="), "Should have points attribute");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTrianglePointsAreCorrect() {
|
||||||
|
STriangle tri = STriangle.create(10, 20, 40, Color.RED, Color.BLACK);
|
||||||
|
draftman.visitTriangle(tri);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
// bottom-left: (x, y+size) = (10, 60)
|
||||||
|
assertTrue(svg.contains("10,60"), "Should contain bottom-left point");
|
||||||
|
// top-center: (x + size/2, y) = (30, 20)
|
||||||
|
assertTrue(svg.contains("30,20"), "Should contain top-center point");
|
||||||
|
// bottom-right: (x + size, y + size) = (50, 60)
|
||||||
|
assertTrue(svg.contains("50,60"), "Should contain bottom-right point");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTriangleFilledOnly() {
|
||||||
|
STriangle tri = STriangle.create(0, 0, 20, Color.BLUE, Color.BLACK);
|
||||||
|
tri.addAttributes(new ColorAttributes(true, false, Color.BLUE, Color.BLACK));
|
||||||
|
draftman.visitTriangle(tri);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("fill:#0000ff"), "Should contain fill style");
|
||||||
|
assertFalse(svg.contains("stroke:"), "Should not contain stroke style when not stroked");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTriangleStrokedAndFilled() {
|
||||||
|
STriangle tri = STriangle.create(0, 0, 20, Color.RED, Color.GREEN);
|
||||||
|
tri.addAttributes(new ColorAttributes(true, true, Color.RED, Color.GREEN));
|
||||||
|
draftman.visitTriangle(tri);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("fill:#ff0000"), "Should contain fill style");
|
||||||
|
assertTrue(svg.contains("stroke:#00ff00"), "Should contain stroke style");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTriangleNotFilled() {
|
||||||
|
STriangle tri = STriangle.create(0, 0, 20, Color.RED, Color.BLACK);
|
||||||
|
tri.addAttributes(new ColorAttributes(false, true, Color.RED, Color.BLACK));
|
||||||
|
draftman.visitTriangle(tri);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("fill:none"), "Should have fill:none when not filled");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── visitText ───────────────────────────────────────────────────
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
class VisitTextTests {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTextElementWithCorrectContent() {
|
||||||
|
SText text = SText.create(10, 20, "Hello");
|
||||||
|
draftman.visitText(text);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("<text"), "Should produce a <text> element");
|
||||||
|
assertTrue(svg.contains(">Hello</text>"), "Should contain the text content");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTextPositionIncludesFontSizeOffset() {
|
||||||
|
SText text = SText.create(10, 20, "Hi");
|
||||||
|
draftman.visitText(text);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("x=\"10\""), "x should match bounds.x");
|
||||||
|
// y = bounds.y + fontSize = 20 + 16 = 36
|
||||||
|
assertTrue(svg.contains("y=\"36\""), "y should be bounds.y + fontSize");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTextDefaultFontAttributes() {
|
||||||
|
SText text = SText.create(0, 0, "Test");
|
||||||
|
draftman.visitText(text);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("font-family=\"SansSerif\""), "Should use default font family");
|
||||||
|
assertTrue(svg.contains("font-size=\"16\""), "Should use default font size");
|
||||||
|
assertTrue(svg.contains("font-style=\"normal\""), "Default style should be normal");
|
||||||
|
assertTrue(svg.contains("font-weight=\"normal\""), "Default weight should be normal");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTextDefaultColorIsBlack() {
|
||||||
|
SText text = SText.create(0, 0, "Test");
|
||||||
|
// default ColorAttributes: filled=true, filledColor=BLACK
|
||||||
|
draftman.visitText(text);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("fill=\"#000000\""), "Default text color should be black");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTextWithCustomFillColor() {
|
||||||
|
SText text = SText.create(0, 0, "Colored");
|
||||||
|
text.addAttributes(new ColorAttributes(true, false, Color.RED, Color.BLACK));
|
||||||
|
draftman.visitText(text);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("fill=\"#ff0000\""), "Text should use filledColor");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTextWithNullColorAttributesFallsBackToBlack() {
|
||||||
|
SText text = SText.create(0, 0, "NoColor");
|
||||||
|
text.addAttributes(new ColorAttributes(false, false, null, null));
|
||||||
|
draftman.visitText(text);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("fill=\"#000000\""), "Should fall back to black with null colors");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── visitCollection ─────────────────────────────────────────────
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
class VisitCollectionTests {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testEmptyCollection() {
|
||||||
|
SCollection empty = SCollection.of();
|
||||||
|
draftman.visitCollection(empty);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertEquals("", svg, "Empty collection should produce no output");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testCollectionVisitsAllChildren() {
|
||||||
|
SCollection coll = SCollection.of(
|
||||||
|
SRectangle.create(0, 0, 10, 10),
|
||||||
|
SCircle.create(20, 20, 5)
|
||||||
|
);
|
||||||
|
draftman.visitCollection(coll);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("<rect"), "Should visit rectangle");
|
||||||
|
assertTrue(svg.contains("<circle"), "Should visit circle");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testNestedCollectionVisitsAllDescendants() {
|
||||||
|
SCollection inner = SCollection.of(
|
||||||
|
SRectangle.create(0, 0, 5, 5)
|
||||||
|
);
|
||||||
|
SCollection outer = SCollection.of(
|
||||||
|
inner,
|
||||||
|
SCircle.create(10, 10, 3)
|
||||||
|
);
|
||||||
|
draftman.visitCollection(outer);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("<rect"), "Should visit nested rectangle");
|
||||||
|
assertTrue(svg.contains("<circle"), "Should visit circle at outer level");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── generateSVG ─────────────────────────────────────────────────
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
class GenerateSVGTests {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testGenerateSVGIncludesXmlDeclaration() {
|
||||||
|
SCollection model = SCollection.of();
|
||||||
|
draftman.generateSVG(model);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("<?xml version=\"1.0\" encoding=\"utf-8\"?>"),
|
||||||
|
"Should include XML declaration");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testGenerateSVGIncludesSvgNamespace() {
|
||||||
|
SCollection model = SCollection.of();
|
||||||
|
draftman.generateSVG(model);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("xmlns=\"http://www.w3.org/2000/svg\""),
|
||||||
|
"Should include SVG namespace");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testGenerateSVGClosesSvgTag() {
|
||||||
|
SCollection model = SCollection.of();
|
||||||
|
draftman.generateSVG(model);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("</svg>"), "Should close <svg> tag");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testGenerateSVGIncludesShapeContent() {
|
||||||
|
SCollection model = SCollection.of(
|
||||||
|
SRectangle.create(1, 2, 3, 4)
|
||||||
|
);
|
||||||
|
draftman.generateSVG(model);
|
||||||
|
|
||||||
|
String svg = output();
|
||||||
|
assertTrue(svg.contains("<rect"), "Should include shape elements");
|
||||||
|
assertTrue(svg.contains("</svg>"), "Should close SVG after shapes");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user