Compare commits

..

3 Commits

Author SHA1 Message Date
98b05e435e Add SCircle 2019-03-19 21:56:00 +01:00
5be59b37f0 Avoid use of try-catch as control flow 2019-03-19 21:37:11 +01:00
51885d8c53 Base java event POC 2019-03-19 21:29:26 +01:00
10 changed files with 139 additions and 97 deletions

View File

@ -15,13 +15,6 @@
<version>1.0-SNAPSHOT</version> <version>1.0-SNAPSHOT</version>
<dependencies> <dependencies>
<!-- https://mvnrepository.com/artifact/io.reactivex.rxjava2/rxjava -->
<dependency>
<groupId>io.reactivex.rxjava2</groupId>
<artifactId>rxjava</artifactId>
<version>2.2.7</version>
</dependency>
<dependency> <dependency>
<groupId>org.slf4j</groupId> <groupId>org.slf4j</groupId>
<artifactId>slf4j-api</artifactId> <artifactId>slf4j-api</artifactId>

View File

@ -1,5 +1,6 @@
package ovh.gasser.newshapes; package ovh.gasser.newshapes;
import ovh.gasser.newshapes.shapes.SCircle;
import ovh.gasser.newshapes.shapes.SCollection; import ovh.gasser.newshapes.shapes.SCollection;
import ovh.gasser.newshapes.shapes.SRectangle; import ovh.gasser.newshapes.shapes.SRectangle;
import ovh.gasser.newshapes.shapes.Shape; import ovh.gasser.newshapes.shapes.Shape;
@ -31,7 +32,8 @@ public class App {
SRectangle.create(70, 10, 40, 60), SRectangle.create(70, 10, 40, 60),
SCollection.of( SCollection.of(
SRectangle.create(100, 200, 40, 60, Color.MAGENTA), SRectangle.create(100, 200, 40, 60, Color.MAGENTA),
SRectangle.create(150, 200, 40, 60, Color.MAGENTA) SRectangle.create(150, 200, 40, 60, Color.MAGENTA),
SCircle.create(200, 200, 60)
) )
); );
} }

View File

@ -1,5 +1,6 @@
package ovh.gasser.newshapes; package ovh.gasser.newshapes;
import ovh.gasser.newshapes.shapes.SCircle;
import ovh.gasser.newshapes.shapes.SCollection; import ovh.gasser.newshapes.shapes.SCollection;
import ovh.gasser.newshapes.shapes.SRectangle; import ovh.gasser.newshapes.shapes.SRectangle;
@ -7,4 +8,6 @@ public interface ShapeVisitor {
void visitRectangle(SRectangle sRectangle); void visitRectangle(SRectangle sRectangle);
void visitCollection(SCollection collection); void visitCollection(SCollection collection);
void visitCircle(SCircle sCircle);
} }

View File

@ -0,0 +1,31 @@
package ovh.gasser.newshapes.shapes;
import ovh.gasser.newshapes.ShapeVisitor;
import ovh.gasser.newshapes.attributes.SelectionAttributes;
import java.awt.*;
public class SCircle extends AbstractShape {
private int radius;
private SCircle(int x, int y, int radius) {
super(new Rectangle(x, y, radius, radius));
this.radius = radius;
}
@Override
public void accept(ShapeVisitor visitor) {
visitor.visitCircle(this);
}
public int getRadius() {
return radius;
}
public static SCircle create(int x, int y, int radius) {
var circle = new SCircle(x, y, radius);
circle.addAttributes(new SelectionAttributes());
return circle;
}
}

View File

@ -1,8 +1,10 @@
package ovh.gasser.newshapes.shapes; package ovh.gasser.newshapes.shapes;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import ovh.gasser.newshapes.App; import ovh.gasser.newshapes.App;
import ovh.gasser.newshapes.attributes.SelectionAttributes;
import ovh.gasser.newshapes.ShapeVisitor; import ovh.gasser.newshapes.ShapeVisitor;
import ovh.gasser.newshapes.attributes.SelectionAttributes;
import ovh.gasser.newshapes.util.Streamable; import ovh.gasser.newshapes.util.Streamable;
import java.awt.*; import java.awt.*;
@ -11,6 +13,7 @@ import java.util.List;
import java.util.Spliterator; import java.util.Spliterator;
public class SCollection extends AbstractShape implements Streamable<Shape> { public class SCollection extends AbstractShape implements Streamable<Shape> {
private final static Logger logger = LoggerFactory.getLogger(SCollection.class);
private final List<Shape> children; private final List<Shape> children;
private SCollection(Shape... shapes) { private SCollection(Shape... shapes) {
@ -25,12 +28,17 @@ public class SCollection extends AbstractShape implements Streamable<Shape> {
@Override @Override
public Rectangle getBounds() { public Rectangle getBounds() {
try { try {
if (children.isEmpty()) {
// If the SCollection is empty, set the bounds to fill the window
return new Rectangle(App.WIN_SIZE);
}
Rectangle bounds = children.get(0).getBounds(); Rectangle bounds = children.get(0).getBounds();
for (Shape s : children) bounds = bounds.union(s.getBounds()); for (Shape s : children) bounds = bounds.union(s.getBounds());
return bounds; return bounds;
} catch (IndexOutOfBoundsException e){ } catch (IndexOutOfBoundsException e){
// If the SCollection is empty, set the bounds to fill the window logger.error("getBounds(): {}");
return new Rectangle(App.WIN_SIZE); throw new RuntimeException(e);
} }
} }

View File

@ -1,47 +1,64 @@
package ovh.gasser.newshapes.ui; package ovh.gasser.newshapes.ui;
import io.reactivex.disposables.Disposable;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
import ovh.gasser.newshapes.Selection; import ovh.gasser.newshapes.Selection;
import ovh.gasser.newshapes.util.EventSource;
import ovh.gasser.newshapes.shapes.Shape;
import ovh.gasser.newshapes.shapes.SCollection; import ovh.gasser.newshapes.shapes.SCollection;
import ovh.gasser.newshapes.shapes.Shape;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent; import java.awt.event.MouseEvent;
import java.util.Optional; import java.util.Optional;
public class Controller { public class Controller {
private final Logger logger = LoggerFactory.getLogger(ShapesView.class); private final static Logger logger = LoggerFactory.getLogger(ShapesView.class);
private final Disposable mouseSub; private final ShapesView view;
private final Shape model;
private Selection selection; private Selection selection;
Controller(ShapesView view, Shape model) { Controller(ShapesView view, Shape model) {
mouseSub = EventSource this.view = view;
.fromMouseEventsOf(view) this.model = model;
.subscribe(evt -> { var adapter = new MouseAdapter() {
assert model instanceof SCollection; @Override
SCollection sc = (SCollection) model; public void mousePressed(MouseEvent evt) {
switch (evt.getID()) { handleMousePressed(evt);
case MouseEvent.MOUSE_PRESSED: }
getTarget(evt, sc)
.ifPresentOrElse( @Override
s -> selection = new Selection(s, true), public void mouseDragged(MouseEvent evt) {
() -> { handleMouseDragged(evt);
if (selection != null) { }
selection.unselect(); };
selection = null; this.view.addMouseMotionListener(adapter);
} this.view.addMouseListener(adapter);
} }
);
break; private void handleMouseDragged(MouseEvent evt) {
case MouseEvent.MOUSE_DRAGGED: if (selection != null) selection.shape.setLoc(evt.getPoint());
handleMouseMoved(evt); view.repaint();
break; }
}
view.repaint(); private void handleMousePressed(MouseEvent evt) {
}, err -> logger.error("{}", err) var sc = (SCollection) this.model;
getTarget(evt, sc)
.ifPresentOrElse(
s -> {
if (selection != null) resetSelection();
selection = new Selection(s, true);
logger.debug("Selecting {}", selection.shape);
},
() -> {
if (selection != null) resetSelection();
}
); );
view.repaint();
}
private void resetSelection() {
logger.debug("Un-selecting {}", selection.shape);
selection.unselect();
selection = null;
} }
private Optional<Shape> getTarget(MouseEvent evt, SCollection sc) { private Optional<Shape> getTarget(MouseEvent evt, SCollection sc) {
@ -50,12 +67,4 @@ public class Controller {
.findFirst(); .findFirst();
} }
private void handleMouseMoved(MouseEvent evt) {
if (selection != null) selection.shape.setLoc(evt.getPoint());
}
public void dispose() {
logger.info("Cleaning subscriptions...");
mouseSub.dispose();
}
} }

View File

@ -1,11 +1,12 @@
package ovh.gasser.newshapes.ui; package ovh.gasser.newshapes.ui;
import ovh.gasser.newshapes.ShapeVisitor; import ovh.gasser.newshapes.ShapeVisitor;
import ovh.gasser.newshapes.shapes.Shape;
import ovh.gasser.newshapes.attributes.ColorAttributes; import ovh.gasser.newshapes.attributes.ColorAttributes;
import ovh.gasser.newshapes.attributes.SelectionAttributes; import ovh.gasser.newshapes.attributes.SelectionAttributes;
import ovh.gasser.newshapes.shapes.SCircle;
import ovh.gasser.newshapes.shapes.SCollection; import ovh.gasser.newshapes.shapes.SCollection;
import ovh.gasser.newshapes.shapes.SRectangle; import ovh.gasser.newshapes.shapes.SRectangle;
import ovh.gasser.newshapes.shapes.Shape;
import java.awt.*; import java.awt.*;
@ -46,6 +47,23 @@ public class ShapeDraftman implements ShapeVisitor {
drawHandlerIfSelected(collection); drawHandlerIfSelected(collection);
} }
@Override
public void visitCircle(SCircle circle) {
ColorAttributes colAttrs = (ColorAttributes) circle.getAttributes(ColorAttributes.ID);
final Rectangle bounds = circle.getBounds();
if (colAttrs == null) {
colAttrs = DEFAULT_COLOR_ATTRIBUTES;
}
if (colAttrs.filled) {
this.g2d.setColor(colAttrs.filledColor);
this.g2d.fillOval(bounds.x, bounds.y, circle.getRadius(), circle.getRadius());
}
if (colAttrs.stroked) this.g2d.setColor(colAttrs.strokedColor);
this.g2d.drawOval(bounds.x, bounds.y, circle.getRadius(), circle.getRadius());
drawHandlerIfSelected(circle);
}
private void drawHandlerIfSelected(Shape s) { private void drawHandlerIfSelected(Shape s) {
SelectionAttributes selAttrs = (SelectionAttributes) s.getAttributes(SelectionAttributes.ID); SelectionAttributes selAttrs = (SelectionAttributes) s.getAttributes(SelectionAttributes.ID);
if ((selAttrs != null) && (selAttrs.selected)){ if ((selAttrs != null) && (selAttrs.selected)){

View File

@ -2,8 +2,10 @@ package ovh.gasser.newshapes.ui.html;
import ovh.gasser.newshapes.ShapeVisitor; import ovh.gasser.newshapes.ShapeVisitor;
import ovh.gasser.newshapes.attributes.ColorAttributes; import ovh.gasser.newshapes.attributes.ColorAttributes;
import ovh.gasser.newshapes.shapes.SCircle;
import ovh.gasser.newshapes.shapes.SCollection; import ovh.gasser.newshapes.shapes.SCollection;
import ovh.gasser.newshapes.shapes.SRectangle; import ovh.gasser.newshapes.shapes.SRectangle;
import ovh.gasser.newshapes.shapes.Shape;
import java.awt.*; import java.awt.*;
import java.io.PrintWriter; import java.io.PrintWriter;
@ -29,6 +31,11 @@ public class HTMLDraftman implements ShapeVisitor {
this.cssOutput = cssOutput; this.cssOutput = cssOutput;
} }
@Override
public void visitCollection(SCollection collection) {
collection.stream().forEach(shape -> shape.accept(this));
}
@Override @Override
public void visitRectangle(SRectangle rect) { public void visitRectangle(SRectangle rect) {
htmlOutput.println("<div id=\"rec" + rect.hashCode() + "\"></div>"); htmlOutput.println("<div id=\"rec" + rect.hashCode() + "\"></div>");
@ -41,8 +48,25 @@ public class HTMLDraftman implements ShapeVisitor {
cssOutput.println(this.attributesToCss(rect) + " }"); cssOutput.println(this.attributesToCss(rect) + " }");
} }
private String attributesToCss(SRectangle rect) { @Override
ColorAttributes attrs = (ColorAttributes) rect.getAttributes(ColorAttributes.ID); public void visitCircle(SCircle circle) {
htmlOutput.println("<div class=\"circle"+circle.hashCode()+"\"></div>");
cssOutput.println(".circle" + circle.hashCode() + "{ ");
cssOutput.println("position: absolute;");
cssOutput.println("top:" + circle.getBounds().y + "px;");
cssOutput.println("left:" + circle.getBounds().x + "px;");
cssOutput.println("width:" + circle.getRadius() + "px;");
cssOutput.println("height:" + circle.getRadius() + "px;");
cssOutput.println("border-radius:" + circle.getRadius() / 2 + "px;");
cssOutput.println("-webkit-border-radius:" + circle.getRadius() / 2 + "px;");
cssOutput.println("-o-border-radius:" + circle.getRadius() / 2 + "px;");
cssOutput.println("-moz-border-radius:" + circle.getRadius() / 2 + "px;");
cssOutput.println("position: absolute;");
cssOutput.println(this.attributesToCss(circle) + " }");
}
private String attributesToCss(Shape shape) {
ColorAttributes attrs = (ColorAttributes) shape.getAttributes(ColorAttributes.ID);
String strokedColor = "#ffffff"; String strokedColor = "#ffffff";
String filledColor = "#ffffff"; String filledColor = "#ffffff";
@ -72,11 +96,6 @@ public class HTMLDraftman implements ShapeVisitor {
return String.format("#%02x%02x%02x", r, g, b); return String.format("#%02x%02x%02x", r, g, b);
} }
@Override
public void visitCollection(SCollection collection) {
collection.stream().forEach(shape -> shape.accept(this));
}
public void generateHTML(SCollection model) { public void generateHTML(SCollection model) {
htmlOutput.println(HEADER_TEMPLATE); htmlOutput.println(HEADER_TEMPLATE);
visitCollection(model); visitCollection(model);

View File

@ -1,28 +0,0 @@
package ovh.gasser.newshapes.util;
import io.reactivex.Observable;
import java.awt.*;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
public class EventSource {
public static Observable<MouseEvent> fromMouseEventsOf(final Component component) {
return Observable.create(emitter -> {
final MouseAdapter adapter = new MouseAdapter() {
@Override
public void mousePressed(MouseEvent e) {
emitter.onNext(e);
}
@Override
public void mouseDragged(MouseEvent e) {
emitter.onNext(e);
}
};
component.addMouseMotionListener(adapter);
component.addMouseListener(adapter);
emitter.setCancellable(() -> component.removeMouseListener(adapter));
});
}
}

View File

@ -1,13 +0,0 @@
<configuration>
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
<withJansi>true</withJansi>
<encoder>
<pattern>%d{HH:mm:ss.SSS} %highlight(%-5level) [%10thread] %cyan(%-40logger{36}) - %msg%n</pattern>
</encoder>
</appender>
<root level="trace">
<appender-ref ref="STDOUT" />
</root>
</configuration>