-
Notifications
You must be signed in to change notification settings - Fork 115
/
Copy pathmultiplot3d.py
executable file
·132 lines (111 loc) · 4.49 KB
/
multiplot3d.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
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
#!/usr/bin/env python3
from __future__ import print_function
# Adapted from
# https://stackoverflow.com/questions/13240633/matplotlib-plot-pulse-propagation-in-3d
# and rewritten to make it clearer how to use it on real data.
import numpy
from mpl_toolkits.mplot3d import Axes3D
import matplotlib.pyplot as plt
from matplotlib.collections import PolyCollection
from matplotlib import colors as mcolors
import random
def gen_data(xbins, numplots, lintest=False):
"""Generate a list of random histograms"""
data = []
ymin = 9999999999999
ymax = -ymin
for plot in range(numplots):
plotpoints = []
y = random.randint(0, 5)
for x in range(xbins):
# Optional: instead of random data, make each plot a constant
# to make it easier to tell which plot is which.
# Even if lintest isn't set, make the last 20% of the
# data predictable, to test whether matplotlib3d is
# re-ordering the plots (it isn't).
if lintest or x > xbins * .8:
y = plot
else:
y += random.uniform(-.8, 1)
ymin = min(ymin, y)
ymax = max(ymax, y)
plotpoints.append((x, y))
data.append(plotpoints)
return data, ymin, ymax
def draw_3d(verts, ymin, ymax, line_at_zero=True, colors=True):
"""Given verts as a list of plots, each plot being a list
of (x, y) vertices, generate a 3-d figure where each plot
is shown as a translucent polygon.
If line_at_zero, a line will be drawn through the zero point
of each plot, otherwise the baseline will be at the bottom of
the plot regardless of where the zero line is.
"""
# add_collection3d() wants a collection of closed polygons;
# each polygon needs a base and won't generate it automatically.
# So for each subplot, add a base at ymin.
if line_at_zero:
zeroline = 0
else:
zeroline = ymin
for p in verts:
p.insert(0, (p[0][0], zeroline))
p.append((p[-1][0], zeroline))
if colors:
# All the matplotlib color sampling examples I can find,
# like cm.rainbow/linspace, make adjacent colors similar,
# the exact opposite of what most people would want.
# So cycle hue manually.
hue = 0
huejump = .27
facecolors = []
edgecolors = []
for v in verts:
hue = (hue + huejump) % 1
c = mcolors.hsv_to_rgb([hue, 1, 1])
# random.uniform(.8, 1),
# random.uniform(.7, 1)])
edgecolors.append(c)
# Make the facecolor translucent:
facecolors.append(mcolors.to_rgba(c, alpha=.7))
else:
facecolors = (1, 1, 1, .8)
edgecolors = (0, 0, 1, 1)
poly = PolyCollection(verts,
facecolors=facecolors, edgecolors=edgecolors)
zs = range(len(data))
# zs = range(len(data)-1, -1, -1)
fig = plt.figure()
ax = fig.add_subplot(1,1,1, projection='3d')
plt.tight_layout(pad=2.0, w_pad=10.0, h_pad=3.0)
ax.add_collection3d(poly, zs=zs, zdir='y')
ax.set_xlabel('X')
ax.set_ylabel('Y')
ax.set_zlabel('Z')
ax.set_xlim3d(0, len(data[1]))
ax.set_ylim3d(-1, len(data))
ax.set_zlim3d(ymin, ymax)
if __name__ == '__main__':
import argparse
import sys
parser = argparse.ArgumentParser()
parser.add_argument('-l', "--lintest", dest="lintest", default=False,
action="store_true",
help="Ultra simple sample data for testing")
parser.add_argument('-c', "--color", dest="colors", default=False,
action="store_true", help="Plot in multiple colors")
parser.add_argument('-x', action="store", dest="xbins",
type=int, default=50,
help='Number of points on the X axis')
parser.add_argument('-n', action="store", dest="numplots",
type=int, default=5,
help='Number of plots')
args = parser.parse_args(sys.argv[1:])
data, ymin, ymax = gen_data(args.xbins, args.numplots, lintest=args.lintest)
draw_3d(data, ymin, ymax, colors=args.colors)
plt.show()
"""
1st green: 150, 100, 100 = .59, 1, 1 matplotlib hue .17 -> 0. 1. 0.04
2nd green: 152, 100, 94 = .60, 1, .94 hue .34 -> 0. 0.94 1.
1st green claims [0. 1. 0.04], GIMP says 0 1 . 5
2dn green claims [0. 0.94 1. ], GIMP says 0 .94 .5
"""