Skip to content

Commit 9acd594

Browse files
committed
Version 2.0.0-rc1. Shapes (partially addresses #4, #34 and #37), sizes, providers, annotations and minor improvements.
1 parent ee09a7f commit 9acd594

29 files changed

+1317
-135
lines changed

README.md

+19-11
Original file line numberDiff line numberDiff line change
@@ -23,20 +23,27 @@ through a [force-directed algorithm](https://en.wikipedia.org/wiki/Force-directe
2323

2424
### What's new?
2525

26-
- (1.1.0) Automatic layout is now performed through an instantiated *strategy*. There are two available (but the pattern allows for the user to devise others):
27-
28-
- `ForceDirectedSpringSystemLayoutStrategy`: this is the original implementation for the automatic placement, through a spring system;
29-
- `ForceDirectedSpringGravityLayoutStrategy`: (**new**) this is a variant of the spring system implementation, but with a gravity pull towards the center of the panel. This is now the default strategy and has the advantage of not repelling isolated vertices and/or bipartite graphs to the edges of the panel.
26+
This is a *release candidate* version, as it hasn't been thoroughly tested yet. Feedback is appreciated.
3027

31-
- (1.0.0) Package now available through [Maven Central](https://central.sonatype.com/namespace/com.brunomnsilva). The library seems stable, after dozens of college projects of my students have used it. Hence, the version was bumped to 1.0.0.
28+
Although one full version number higher than the previous *stable* version (1.1.0), existing applications are expected to work with this library version without significant changes.
3229

33-
- (0.9.4) You can now annotate a method with `@SmartLabelSource` within a model class to provide the displayed label for a vertex/edge; see the example at `com.brunomnsilva.smartgraph.example`. If no annotation is present, then the `toString()` method is used to obtain the label's text.
30+
:warning: The only exception is the necessary use of `SmartStylableNode.setStyleInline(...)` instead of `SmartStylableNode.setStyle(...)` to correctly apply inline styles to nodes (vertices and edges). Css classes are set the same way as before.
3431

35-
- (0.9.4) You can manually alter a vertex position on the panel at anytime, through `SmartGraphPanel.setVertexPosition(Vertex<V> v)`; see the example at `com.brunomnsilva.smartgraph.example`.
32+
- (2.0.0-rc1) Shapes, sizes, providers, annotations and minor improvements:
3633

37-
- (0.9.4) You can override specific default properties by using a *String* parameter to the `SmartGraphProperties` constructor; see the example at `com.brunomnsilva.smartgraph.example`. This is useful if you want to display visually different graphs within the same application.
34+
- Different shapes can be used to represent vertices, namely circles, stars and regular polygons (from triangles to dodecagons);
35+
- The default shape can be specified with the `vertex.shape` property in `smartgraph.properties`
36+
- Can be set/changed at runtime through a `SmartShapeTypeProvider` or `SmartShapeTypeSource` annotation.
3837

39-
- (0.9.4) You can now style labels and arrows individually.
38+
- The radius of the shape (enclosing circle) used to represent a vertex can be set/changed at runtime through a `SmartRadiusProvider` or `SmartRadiusSource` annotation.
39+
40+
- Updated shapes and radii are only reflected in the visualization after calling `SmartGraphPanel.update()` or `SmartGraphPanel.updateAndWait()`.
41+
42+
- Improvements:
43+
- When dragging nodes, they will be kept within the panel's bounds.
44+
- The look of curved edges has been improved.
45+
46+
See the [wiki](https://github.com/brunomnsilva/JavaFXSmartGraph/wiki) for the complete changelist.
4047

4148
### Using the library
4249

@@ -216,7 +223,7 @@ graphView.setVertexDoubleClickAction(graphVertex -> {
216223
graphView.setEdgeDoubleClickAction(graphEdge -> {
217224
System.out.println("Edge contains element: " + graphEdge.getUnderlyingEdge().element());
218225
//dynamically change the style, can also be done for a vertex
219-
graphEdge.setStyle("-fx-stroke: black; -fx-stroke-width: 2;");
226+
graphEdge.setStyleInline("-fx-stroke: black; -fx-stroke-width: 2;");
220227
});
221228
```
222229

@@ -234,7 +241,8 @@ You can set the graph visualization properties in the `smartgraph.properties` fi
234241
# Vertex related configurations
235242
#
236243
vertex.allow-user-move = true
237-
vertex.radius = 15
244+
vertex.radius = 15
245+
vertex.shape = circle
238246
vertex.tooltip = true
239247
vertex.label = false
240248

pom.xml

+1-1
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,7 @@
3030

3131
<groupId>com.brunomnsilva</groupId>
3232
<artifactId>smartgraph</artifactId>
33-
<version>1.1.1</version>
33+
<version>2.0.0-rc1</version>
3434
<packaging>jar</packaging>
3535

3636
<name>${project.groupId}:${project.artifactId}</name>

smartgraph.properties

+31-4
Original file line numberDiff line numberDiff line change
@@ -25,14 +25,40 @@
2525
################################################################################
2626
# javafxgraph.properties
2727
#
28-
# ATTENTION: do not leave any trailing spaces after 'true' or 'false' values
28+
# These properties can be used to override the default values, which are:
29+
#
30+
# vertex.allow-user-move = true (boolean)
31+
# vertex.radius = 15 (in pixels)
32+
# vertex.shape = circle (string)
33+
# vertex.tooltip = true (boolean)
34+
# vertex.label = true (boolean)
35+
# edge.tooltip = true (boolean)
36+
# edge.label = true (boolean)
37+
# edge.arrow = true (boolean)
38+
# edge.arrowsize = 5 (in pixels)
39+
#
40+
# Allowed/available vertex shape names (string without quotes) are:
41+
# - circle
42+
# - star
43+
# - triangle
44+
# - square
45+
# - pentagon
46+
# - hexagon
47+
# - heptagon
48+
# - octagon
49+
# - nonagon
50+
# - decagon
51+
# - hendecagon
52+
# - dodecagon
2953
#
54+
# ATTENTION: do not leave any trailing spaces after 'true' or 'false' values
3055
################################################################################
3156

3257
# Vertex related configurations
3358
#
3459
vertex.allow-user-move = true
35-
vertex.radius = 15
60+
vertex.radius = 20
61+
vertex.shape = star
3662
vertex.tooltip = true
3763
vertex.label = true
3864

@@ -46,12 +72,13 @@ edge.arrow = true
4672
# size in pixels (side of a triangle); only for directed graphs
4773
edge.arrowsize = 5
4874

49-
# Force-directed layout related configurations
50-
#
5175
# Notice: deprecated since version 1.1.0 Force directed layout strategies are now
5276
# instantiated and can be swapped at runtime, per the Strategy design pattern.
5377
# The parameters are passed as arguments or one can use the default ones described
5478
# in the javadoc documentation.
79+
#
80+
# Force-directed layout related configurations
81+
#
5582
# -- You should experiment with different values for your
5683
# -- particular problem, knowing that not all will achieve
5784
# -- a stable state

src/main/java/com/brunomnsilva/smartgraph/Main.java

+4-4
Original file line numberDiff line numberDiff line change
@@ -64,7 +64,7 @@ public void start(Stage ignored) {
6464
This can be done at any time afterwards.
6565
*/
6666
if (g.numVertices() > 0) {
67-
graphView.getStylableVertex("A").setStyle("-fx-fill: gold; -fx-stroke: brown;");
67+
graphView.getStylableVertex("A").setStyleInline("-fx-fill: gold; -fx-stroke: brown;");
6868
}
6969

7070
/*
@@ -110,9 +110,9 @@ public void start(Stage ignored) {
110110
graphView.setEdgeDoubleClickAction(graphEdge -> {
111111
System.out.println("Edge contains element: " + graphEdge.getUnderlyingEdge().element());
112112
//dynamically change the style when clicked
113-
graphEdge.setStyle("-fx-stroke: black; -fx-stroke-width: 3;");
113+
graphEdge.setStyleInline("-fx-stroke: black; -fx-stroke-width: 3;");
114114

115-
graphEdge.getStylableArrow().setStyle("-fx-stroke: black; -fx-stroke-width: 3;");
115+
graphEdge.getStylableArrow().setStyleInline("-fx-stroke: black; -fx-stroke-width: 3;");
116116

117117
//uncomment to see edges being removed after click
118118
//Edge<String, String> underlyingEdge = graphEdge.getUnderlyingEdge();
@@ -250,7 +250,7 @@ private void continuously_test_adding_elements(Graph<String, String> g, SmartGra
250250
//color new vertices
251251
SmartStylableNode stylableVertex = graphView.getStylableVertex(vertexId);
252252
if(stylableVertex != null) {
253-
stylableVertex.setStyle("-fx-fill: orange;");
253+
stylableVertex.setStyleInline("-fx-fill: orange;");
254254
}
255255
} else {
256256
Vertex<String> existing1 = get_random_vertex(g);

src/main/java/com/brunomnsilva/smartgraph/example/City.java

+31-2
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,8 @@
2525
package com.brunomnsilva.smartgraph.example;
2626

2727
import com.brunomnsilva.smartgraph.graphview.SmartLabelSource;
28+
import com.brunomnsilva.smartgraph.graphview.SmartRadiusSource;
29+
import com.brunomnsilva.smartgraph.graphview.SmartShapeTypeSource;
2830

2931
/**
3032
* A simple class to represent a city in an example usage of the library.
@@ -53,7 +55,6 @@ public void setName(String name) {
5355
this.name = name;
5456
}
5557

56-
5758
public float getPopulation() {
5859
return population;
5960
}
@@ -67,5 +68,33 @@ public String toString() {
6768
return "City{" + "name=" + name + ", population=" + population + '}';
6869
}
6970

70-
71+
@SmartShapeTypeSource
72+
public String modelShape() {
73+
if(this.name == "Tokyo") {
74+
return "star";
75+
}
76+
77+
return "circle";
78+
}
79+
80+
@SmartRadiusSource
81+
public Double modelRadius() {
82+
return convertToLogScale(Double.valueOf(String.valueOf(this.population)));
83+
}
84+
85+
private static double convertToLogScale(double value) {
86+
// Define input range
87+
double minValue = 1;
88+
double maxValue = 40;
89+
90+
// Define output range
91+
double minOutputValue = 15;
92+
double maxOutputValue = 40;
93+
94+
// Map the input value to the output range using logarithmic function
95+
double mappedValue = (Math.log(value) - Math.log(minValue)) / (Math.log(maxValue) - Math.log(minValue));
96+
97+
// Map the mapped value to the output range
98+
return minOutputValue + mappedValue * (maxOutputValue - minOutputValue);
99+
}
71100
}

src/main/java/com/brunomnsilva/smartgraph/example/ExampleMain.java

+12-6
Original file line numberDiff line numberDiff line change
@@ -25,12 +25,13 @@
2525
package com.brunomnsilva.smartgraph.example;
2626

2727
import com.brunomnsilva.smartgraph.containers.SmartGraphDemoContainer;
28-
import com.brunomnsilva.smartgraph.graph.Graph;
29-
import com.brunomnsilva.smartgraph.graph.GraphEdgeList;
28+
import com.brunomnsilva.smartgraph.graph.Digraph;
29+
import com.brunomnsilva.smartgraph.graph.DigraphEdgeList;
3030
import com.brunomnsilva.smartgraph.graph.Vertex;
3131
import com.brunomnsilva.smartgraph.graphview.SmartCircularSortedPlacementStrategy;
3232
import com.brunomnsilva.smartgraph.graphview.SmartGraphPanel;
3333
import com.brunomnsilva.smartgraph.graphview.SmartGraphProperties;
34+
import com.brunomnsilva.smartgraph.graphview.SmartGraphVertex;
3435
import javafx.application.Application;
3536
import javafx.scene.Scene;
3637
import javafx.stage.Stage;
@@ -45,7 +46,7 @@ public class ExampleMain extends Application {
4546
@Override
4647
public void start(Stage ignored) {
4748

48-
Graph<City, Distance> distances = new GraphEdgeList<>();
49+
Digraph<City, Distance> distances = new DigraphEdgeList<>();
4950

5051
Vertex<City> prague = distances.insertVertex(new City("Prague", 1.3f));
5152
Vertex<City> tokyo = distances.insertVertex(new City("Tokyo", 37.5f));
@@ -64,7 +65,7 @@ public void start(Stage ignored) {
6465
distances.insertEdge(beijing, london, new Distance(8132));
6566

6667
/* Only Java 15 allows for multi-line strings. */
67-
String customProps = "edge.label = true" + "\n" + "edge.arrow = false";
68+
String customProps = "edge.label = true" + "\n" + "edge.arrow = true";
6869

6970
SmartGraphProperties properties = new SmartGraphProperties(customProps);
7071

@@ -92,10 +93,15 @@ public void start(Stage ignored) {
9293
graphView.setVertexPosition(helsinky, 924, 100);
9394
graphView.setVertexPosition(london, 200, 668);
9495
graphView.setVertexPosition(prague, 824, 668);
95-
graphView.setVertexPosition(tokyo, 512, 300);
96+
graphView.setVertexPosition(tokyo, 512, 200);
9697
graphView.setVertexPosition(newYork, 512, 400);
9798

98-
graphView.getStylableLabel(tokyo).setStyle("-fx-stroke: red; -fx-fill: red;");
99+
graphView.getStylableLabel(tokyo).setStyleInline("-fx-stroke: green; -fx-fill: green;");
100+
101+
graphView.setVertexDoubleClickAction((SmartGraphVertex<City> graphVertex) -> {
102+
graphVertex.addStyleClass("myVertex");
103+
});
104+
99105
}
100106

101107
public static void main(String[] args) {

src/main/java/com/brunomnsilva/smartgraph/graphview/ForceDirectedSpringSystemLayoutStrategy.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -97,7 +97,7 @@ protected Point2D computeForceBetween(SmartGraphVertexNode<V> v, SmartGraphVerte
9797

9898
Point2D vPosition = v.getUpdatedPosition();
9999
Point2D wPosition = w.getUpdatedPosition();
100-
double distance = vPosition.distance(wPosition);
100+
double distance = vPosition.distance(wPosition) - (v.getRadius() + w.getRadius());
101101
Point2D forceDirection = wPosition.subtract(vPosition).normalize();
102102

103103
if (distance < 1) {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,88 @@
1+
/*
2+
* The MIT License
3+
*
4+
* JavaFXSmartGraph | Copyright 2024 [email protected]
5+
*
6+
* Permission is hereby granted, free of charge, to any person obtaining a copy
7+
* of this software and associated documentation files (the "Software"), to deal
8+
* in the Software without restriction, including without limitation the rights
9+
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10+
* copies of the Software, and to permit persons to whom the Software is
11+
* furnished to do so, subject to the following conditions:
12+
*
13+
* The above copyright notice and this permission notice shall be included in
14+
* all copies or substantial portions of the Software.
15+
*
16+
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17+
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18+
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19+
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20+
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21+
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
22+
* THE SOFTWARE.
23+
*/
24+
25+
package com.brunomnsilva.smartgraph.graphview;
26+
27+
import javafx.beans.property.DoubleProperty;
28+
import javafx.scene.shape.Circle;
29+
import javafx.scene.shape.Shape;
30+
31+
/**
32+
* This class represents a circle shape with a specified radius.
33+
*
34+
* @author brunomnsilva
35+
*/
36+
public class ShapeCircle implements ShapeWithRadius<Circle> {
37+
38+
private final Circle surrogate;
39+
40+
/**
41+
* Creates a circle shape.
42+
* @param x the x-center coordinate
43+
* @param y the y-center coordinate
44+
* @param radius the radius of the circle
45+
*/
46+
public ShapeCircle(double x, double y, double radius) {
47+
Args.requireNonNegative(x, "x");
48+
Args.requireNonNegative(y, "y");
49+
Args.requireNonNegative(radius, "radius");
50+
51+
this.surrogate = new Circle(x, y, radius);
52+
}
53+
54+
@Override
55+
public Shape getShape() {
56+
return surrogate;
57+
}
58+
59+
@Override
60+
public DoubleProperty centerXProperty() {
61+
return surrogate.centerXProperty();
62+
}
63+
64+
@Override
65+
public DoubleProperty centerYProperty() {
66+
return surrogate.centerYProperty();
67+
}
68+
69+
@Override
70+
public DoubleProperty radiusProperty() {
71+
return surrogate.radiusProperty();
72+
}
73+
74+
@Override
75+
public double getRadius() {
76+
return surrogate.getRadius();
77+
}
78+
79+
@Override
80+
public void setRadius(double radius) {
81+
Args.requireNonNegative(radius, "radius");
82+
83+
// Only update if different
84+
if(Double.compare(this.getRadius(), radius) != 0) {
85+
surrogate.setRadius(radius);
86+
}
87+
}
88+
}

0 commit comments

Comments
 (0)