Sunday, January 13, 2013

JavaFX Tree Visualization Part 2

In this blog post I'll post a follow up on my post about painting random generated trees.

My intention was first to improve the code a bit and use a more functional way to express the recursion, now I'm using case classes and more pattern matching to build up the tree. I think now it is far more understandable what is going on.



Moreover I've added logic for 'growing' the trees, albeit it doesn't look very natural how they grow ;-)

What I've discovered however is that by using timelines you can parallelize animations - if you look at the video you'll see that more than one tree grows without disturbing the others doing their thing. I don't know if it is the "correct" way to do stuff like this in JavaFX, but it seems to work.

Still I'm amazed how easy and intuitive it is to create such graphic stuff, and my processor doesn't seem to bother much that I'm painting thousands of Nodes in this little program.

Here is the source code, full source available as a standalone maven project.


package net.ladstatt.apps
import scala.collection.JavaConversions.seqAsJavaList
import scala.math.Pi
import scala.math.cos
import scala.math.sin
import scala.util.Random
import javafx.animation.Animation
import javafx.animation.KeyFrame
import javafx.animation.Timeline
import javafx.application.Application
import javafx.event.ActionEvent
import javafx.event.EventHandler
import javafx.scene.Group
import javafx.scene.Scene
import javafx.scene.effect.Glow
import javafx.scene.input.MouseEvent
import javafx.scene.paint.Color
import javafx.scene.paint.CycleMethod
import javafx.scene.paint.LinearGradient
import javafx.scene.paint.Paint
import javafx.scene.paint.Stop
import javafx.scene.shape.Line
import javafx.scene.shape.Rectangle
import javafx.scene.shape.Shape
import javafx.stage.Stage
import javafx.util.Duration
/**
* See video and some comments on http://ladstatt.blogspot.com/
*/
object PlantSomeTrees {
def main(args: Array[String]): Unit = {
Application.launch(classOf[PlantSomeTrees], args: _*)
}
}
class PlantSomeTrees extends javafx.application.Application {
val canvasWidth = 800
val canvasHeight = 600
val treeDepth = 5
val trunkWidth = 7
val (minTreeSize, maxTreeSize) = (50, 80)
//val treeColor = Color.CHOCOLATE
def treeColor = mkRandColor
val initialDirection = (3 * Pi / 2)
val (minDegree, maxDegree) = (0, Pi / 4)
val growingSpeed = 96
val branchSlices = 10
override def start(stage: Stage): Unit = {
stage.setTitle("A growing forrest")
val root = new Group()
val background = {
val b = new Rectangle(0, 0, canvasWidth, canvasHeight)
val stops = List(new Stop(0, Color.BLACK), new Stop(1, Color.WHITESMOKE))
val g = new LinearGradient(0.0, 1.0, 0.0, 0.0, true, CycleMethod.NO_CYCLE, stops)
b.setFill(g)
b
}
root.getChildren.add(background)
root.addEventHandler(MouseEvent.MOUSE_CLICKED, new EventHandler[MouseEvent] {
def handle(event: MouseEvent) {
val broot = Branch(event.getX, event.getY, minTreeSize + (maxTreeSize - minTreeSize) * Random.nextDouble,
initialDirection, treeColor, trunkWidth, treeDepth)
var lines2Paint = traverse(mkRandomTree(broot)).toList
val tree = new Group()
tree.addEventHandler(MouseEvent.MOUSE_ENTERED, new EventHandler[MouseEvent] {
def handle(event: MouseEvent) {
tree.setEffect(new Glow(1.0))
}
})
tree.addEventHandler(MouseEvent.MOUSE_EXITED, new EventHandler[MouseEvent] {
def handle(event: MouseEvent) {
tree.setEffect(new Glow(0))
}
})
root.getChildren.add(tree)
val growTimeline = new Timeline
growTimeline.setRate(growingSpeed)
growTimeline.setCycleCount(Animation.INDEFINITE)
growTimeline.getKeyFrames().add(
new KeyFrame(Duration.seconds(1),
new EventHandler[ActionEvent]() {
def handle(event: ActionEvent) {
if (!lines2Paint.isEmpty) {
val (hd :: tail) = lines2Paint
tree.getChildren.add(hd)
lines2Paint = tail
} else {
growTimeline.stop
}
}
}))
growTimeline.play()
}
})
stage.setScene(new Scene(root, canvasWidth, canvasHeight))
stage.show()
}
def mkRandDegree = (maxDegree - minDegree) * Random.nextDouble
def mkRandColor = {
def randInt = (Random.nextFloat * 255).toInt
Color.rgb(randInt, randInt, randInt)
}
sealed trait ATree
case class Branch(x: Double, y: Double, length: Double, degree: Double, color: Color, width: Int, ord: Int) extends ATree
case class SubTree(center: ATree, left: ATree, right: ATree) extends ATree
def mkRandomTree(root: Branch): ATree = {
def mkRandTree(tree: ATree): ATree =
tree match {
case Branch(x0, y0, length, d0, color, width, ord) => {
ord match {
case 0 => tree
case _ => {
val l0 = length * (1 - Random.nextDouble * 0.3)
val l1 = length * (1 - Random.nextDouble * 0.5)
val l2 = length * (1 - Random.nextDouble * 0.5)
// val l0 = length * 0.7
// val l1 = length * 0.5
// val l2 = length * 0.5
// startpoint of left and right branch
val (xm, ym) = (x0 + l0 * cos(d0), y0 + l0 * sin(d0))
val (d1, d2) = (d0 + mkRandDegree, d0 - mkRandDegree)
// val (d1, d2) = (d0 + Pi / 4, d0 - Pi / 4)
mkRandTree(SubTree(
Branch(x0, y0, length, d0, color, width, ord - 1), // trunk
Branch(xm, ym, l1, d1, color.darker, width - 1, ord - 1), // leftbranch
Branch(xm, ym, l2, d2, color.darker, width - 1, ord - 1))) // rightbranch
}
}
}
case SubTree(center, left, right) => SubTree(mkRandTree(center), mkRandTree(left), mkRandTree(right))
}
mkRandTree(root)
}
def traverse(tree: ATree): List[Shape] = {
tree match {
case Branch(x, y, l, d, c, width, ord) => mkLine(x, y, l, d, c, if (width < 1) 1 else width)
case SubTree(center, left, right) => traverse(center) ++ traverse(left) ++ traverse(right)
}
}
def mkLine(x: Double, y: Double, length: Double, degree: Double, paint: Paint, width: Int): List[Shape] = {
val (dc0, ds0) = (length / branchSlices * cos(degree), length / branchSlices * sin(degree))
(for (i <- 1 to branchSlices) yield {
val (x2, y2) = (x + i * dc0, y + i * ds0)
val l = new Line(x + (i - 1) * dc0, y + (i - 1) * ds0, x2, y2)
l.setStrokeWidth(width)
l.setStroke(paint)
l
}).toList
}
}


No comments:

Post a Comment