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.
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.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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