Episode 40: Fractal Trees
author |
Steve Losh <steve@stevelosh.com> |
date |
Sat, 23 Jul 2016 12:43:03 +0000 (2016-07-23) |
parents |
9cf73a5adcb5
|
children |
6b92f156e83b
|
branches/tags |
(none) |
files |
package.lisp src/2d/demo.lisp src/2d/vectors.lisp |
Changes
--- a/package.lisp Sun Jul 17 22:35:16 2016 +0000
+++ b/package.lisp Sat Jul 23 12:43:03 2016 +0000
@@ -111,11 +111,13 @@
#:vec-sub
#:vec-mul
#:vec-div
+ #:vec-rotate
#:vec-lerp
#:vec-add!
#:vec-sub!
#:vec-mul!
#:vec-div!
+ #:vec-rotate!
#:vec-to-string
#:with-vec
#:with-vecs
--- a/src/2d/demo.lisp Sun Jul 17 22:35:16 2016 +0000
+++ b/src/2d/demo.lisp Sat Jul 23 12:43:03 2016 +0000
@@ -72,67 +72,36 @@
(sketch::draw-shape :triangles vertices vertices)))
-(defstruct (point
- (:constructor make-point (pos old-pos pinned)))
- pos old-pos pinned)
-
-(defstruct (stick
- (:constructor make-stick (a b length &key
- (hidden nil)
- (color (gray 0.0))
- (width 1))))
- a b length hidden color width)
-
-
-(defun make-random-point ()
- (let ((v (make-random-vec *width* *height*)))
- (make-point v v nil)))
-
-(defun update-point (point)
- (with-slots (pos old-pos pinned) point
- (when (not pinned)
- (let* ((friction 0.999)
- (gravity (make-vec 0 -0.2))
- (vel (vec-mul (vec-sub pos old-pos) friction)))
- (setf old-pos pos
- pos (vec-add gravity (vec-add pos vel)))))))
+(defun draw-tree (p0 p1 branch-angle-a branch-angle-b trunk-ratio limit)
+ (if (zerop limit)
+ (draw-line p0 p1)
+ (let* ((d (vec-mul (vec-sub p1 p0) trunk-ratio))
+ (midpoint (vec-add d p0)))
+ (draw-line p0 midpoint)
+ (draw-tree midpoint (vec-add midpoint (vec-rotate d branch-angle-a))
+ branch-angle-a branch-angle-b trunk-ratio (1- limit))
+ (draw-tree midpoint (vec-add midpoint (vec-rotate d branch-angle-b))
+ branch-angle-a branch-angle-b trunk-ratio (1- limit)))))
-(defun constrain-point (point)
- (with-slots (pos old-pos pinned) point
- (when (not pinned)
- (let* ((bounce 0.9)
- (vel (vec-sub pos old-pos)))
- (macrolet ((wrap ((cur old vel) comp bound)
- `(when (,comp ,cur ,bound)
- (setf ,cur ,bound
- ,old (+ ,cur (* bounce ,vel))))))
- (wrap ((vec-x pos) (vec-x old-pos) (vec-x vel)) > *width*)
- (wrap ((vec-x pos) (vec-x old-pos) (vec-x vel)) < 0)
- (wrap ((vec-y pos) (vec-y old-pos) (vec-y vel)) > *height*)
- (wrap ((vec-y pos) (vec-y old-pos) (vec-y vel)) < 0))))))
+(defun draw-pytree (size angle limit)
+ (rect 0 0 size size)
+ (when (not (zerop limit))
+ (let ((a-size (* size (cos angle)))
+ (b-size (* size (sin angle))))
+ (in-context
+ (translate 0 size)
+ (rotate (degrees angle))
+ (draw-pytree a-size angle (1- limit)))
+ (in-context
+ (translate size size)
+ (rotate (- (- 180 90 (degrees angle))))
+ (translate (- b-size) 0)
+ (draw-pytree b-size angle (1- limit))
+ )
+ )
-(defun update-stick (stick)
- (with-slots (a b length) stick
- (let* ((pos-a (point-pos a))
- (pos-b (point-pos b))
- (between (vec-sub pos-a pos-b))
- (distance (vec-magnitude between))
- (change (/ (- length distance) 2))
- (correction (vec-set-magnitude between change)))
- (when (not (point-pinned a))
- (setf (point-pos a) (vec-add pos-a correction)))
- (when (not (point-pinned b))
- (setf (point-pos b) (vec-sub pos-b correction))))))
-
-(defun render-point (point)
- (draw-circle (point-pos point) 5))
-
-(defun render-stick (stick)
- (with-slots (a b hidden color width) stick
- (unless hidden
- (with-pen (make-pen :stroke color :weight width)
- (draw-line (point-pos a) (point-pos b))))))
-
+ )
+ )
(defsketch demo
((width *width*) (height *height*) (y-axis :up) (title "Coding Math 2D")
@@ -144,21 +113,15 @@
(previous-time 0)
(total-time 0)
;; Data
- (points (iterate (for i :from 0 :to 6)
- (for p = (make-random-point))
- (when (= i 6)
- (setf (point-pinned p) t))
- (collect p)))
- (pin (nth 6 points))
- (angle 0.0)
- (speed 0.02)
- (sticks (append
- (iterate (for (a . b) :pairs-of-list (subseq points 0 4))
- (collect (make-stick a b (random-range 50 200) :width 5)))
- (list (make-stick (nth 0 points) (nth 2 points) 100 :hidden t)
- (make-stick (nth 3 points) (nth 4 points) 50)
- (make-stick (nth 4 points) (nth 5 points) 50)
- (make-stick (nth 5 points) (nth 6 points) 50))))
+ (p0 (make-vec *center-x* 50))
+ (p1 (make-vec *center-x* (- *height* 200)))
+ (branch-angle-a (random-range (- (/ tau 4)) (/ tau 4)))
+ (branch-angle-b (random-range (- (/ tau 4)) (/ tau 4)))
+ (trunk-ratio 1/2)
+
+ (py-angle (/ tau 8))
+
+ (a 0.0)
;; Pens
(particle-pen (make-pen :fill (gray 0.9) :stroke (gray 0.4)))
(black-pen (make-pen :stroke (rgb 0 0 0) :fill (rgb 0.4 0.4 0.4) :weight 1 :curve-steps 50))
@@ -171,19 +134,21 @@
(incf total-time (- current-time previous-time))
(incf frame)
;;
+ (incf a 0.02)
+ (wrapf a 0 tau)
+
+ (setf trunk-ratio (map-range -1 1 1/4 3/4 (sin a)))
+
+ (setf py-angle (map-range 0 tau 0 (/ tau 4) a))
+
(with-setup
(in-context
- (incf angle speed)
- (setf (vec-x (point-pos pin))
- (map-range 0 1 0 *width* (expt (sin angle) 10)))
-
- (mapc #'update-point points)
- (iterate (repeat 3)
- (mapc #'update-stick sticks)
- (mapc #'constrain-point points))
- (mapc #'render-stick sticks)
+ (with-pen black-pen
+ (in-context
+ (translate (- *center-x* 40) 0)
+ (draw-pytree 80 py-angle 5)))
(with-pen red-pen
- (mapc #'render-point points))
+ (draw-tree p0 p1 branch-angle-a branch-angle-b trunk-ratio 8))
))
;;
@@ -202,7 +167,7 @@
(defun mousedown-left (instance x y)
(declare (ignorable instance x y))
- (zap% (point-pinned (nth 6 (slot-value instance 'points))) #'not %))
+ )
(defun mousedown-right (instance x y)
(declare (ignorable instance x y))
--- a/src/2d/vectors.lisp Sun Jul 17 22:35:16 2016 +0000
+++ b/src/2d/vectors.lisp Sat Jul 23 12:43:03 2016 +0000
@@ -100,6 +100,9 @@
(make-vec (/ (vec-x v) s)
(/ (vec-y v) s)))
+(defun vec-rotate (v angle)
+ (vec-set-angle v (+ (vec-angle v) angle)))
+
(defun vec-add! (v1 v2)
(incf (vec-x v1) (vec-x v2))
@@ -117,6 +120,9 @@
(setf (vec-x v) (/ (vec-x v) s)
(vec-y v) (/ (vec-y v) s)))
+(defun vec-rotate! (v angle)
+ (setf (vec-angle v)
+ (+ (vec-angle v) angle)))
(defun vec-lerp (v1 v2 n)
(with-vecs ((x1 y1) v1