-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathangletree.py
113 lines (71 loc) · 2.49 KB
/
angletree.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
from math import *
from operator import concat
from random import randint, random, choice, uniform
import colorsys
import pysvg.structure
import pysvg.builders
img_side = 1024
img_xmid = img_side/2
img_ymid = img_side/2
starting_rad = 128
class Circle:
__slots__ = ('x', 'y', 'rad', 'depth')
def __init__(self, x, y, rad, depth):
self.x = x
self.y = y
self.rad = rad
self.depth = depth
circles = []
def rgb_to_hex(rgb):
return "#" + reduce(concat, map(lambda x: "%02x" % x, rgb))
def make_child(circle):
# ang = random() * (1 * pi)
ang = uniform(1.9 * pi, 2.1 * pi) if random() < 0.10 else uniform(.9 * pi, 1.1 * pi)
px = (2 * circle.rad) * cos(ang) + circle.x
py = (2 * circle.rad) * sin(ang) + circle.y
return Circle(px, py, circle.rad * 0.5, circle.depth + 1)
def make_tree(root, branch, depth):
if depth == 0:
return
children = []
for _ in range(branch):
child = make_child(root)
children.append(child)
circles.append(child)
for child in children:
make_tree(child, branch, depth - 1)
# Add the root
root = Circle(img_xmid, img_ymid, starting_rad, 0)
circles.append(root)
# Make the tree
depth = 6
branching_factor = 7
make_tree(root, branching_factor, depth)
# Make the SVG
svg = pysvg.structure.svg()
sb = pysvg.builders.ShapeBuilder()
bot_y = min(circles, key = lambda circ: circ.y).y
top_y = max(circles, key = lambda circ: circ.y).y
bot_x_circ = min(circles, key = lambda circ: circ.x - circ.rad)
bot_x = bot_x_circ.x - bot_x_circ.rad
bot_y_circ = min(circles, key = lambda circ: circ.y - circ.rad)
bot_y = bot_y_circ.y - bot_y_circ.rad
highest_dist_circ = max(circles, key = lambda circ: sqrt((circ.x - bot_x)**2 + (circ.y-bot_y)**2))
highest_dist = sqrt((highest_dist_circ.x - bot_x)**2 + (highest_dist_circ.y - bot_y)**2)
for circ in circles:
# darkness = (float(circ.depth) / depth) * 255
# light = float(circ.depth) / depth
light = 0.5
# hue = float(circ.y - bot_y) / (top_y - bot_y)
hue = sqrt((circ.x - bot_x)**2 + (circ.y - bot_y)**2) / highest_dist
hue += choice((-1, 1)) * random() * 0.25
# sat = float(circ.depth) / depth
sat = 0.5
rgb = map(lambda x: int(255 * x), colorsys.hls_to_rgb(hue, light, sat))
color = rgb_to_hex(rgb)
if(circ.depth > 2):
svg.addElement(sb.createCircle(circ.x - bot_x, circ.y - bot_y, circ.rad, strokewidth = 0, fill = color))
# Hack fill opacity in because PySVG doesn't have it :(
xml = svg.getXML().replace("; \"", "; fill-opacity:0.75; \"")
with open("angletree.svg", "w") as f:
f.write(xml)