forked from fish2000/ColorPy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathplots.py
executable file
·501 lines (447 loc) · 17.3 KB
/
plots.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
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
'''
plots.py - Various types of plots.
Description:
Functions to draw various types of plots for light spectra.
Functions:
log_interpolate (y0, y1, num_values) -
Return a list of values, num_values in size, logarithmically interpolated
between y0 and y1. The first value will be y0, the last y1.
tighten_x_axis (x_list) -
Tighten the x axis (only) of the current plot to match the given range of x values.
The y axis limits are not affected.
General plots:
rgb_patch_plot (
rgb_colors,
color_names,
title,
filename = None,
patch_gap = 0.05,
num_across = 6) -
Draw a set of color patches, specified as linear rgb colors.
xyz_patch_plot (
xyz_colors,
color_names,
title,
filename = None,
patch_gap = 0.05,
num_across = 6) -
Draw a set of color patches specified as xyz colors.
spectrum_subplot (spectrum) -
Plot a spectrum, with x-axis the wavelength, and y-axis the intensity.
The curve is colored at that wavelength by the (approximate) color of a
pure spectral color at that wavelength, with intensity constant over wavelength.
(This means that dark looking colors here mean that wavelength is poorly viewed by the eye.
This is not a complete plotting function, e.g. no file is saved, etc.
It is assumed that this function is being called by one that handles those things.
spectrum_plot (
spectrum,
title,
filename = None,
xlabel = 'Wavelength ($nm$)',
ylabel = 'Intensity ($W/m^2$)') -
Plot for a single spectrum -
In a two part graph, plot:
top: color of the spectrum, as a large patch.
low: graph of spectrum intensity vs wavelength (x axis).
The graph is colored by the (approximated) color of each wavelength.
Each wavelength has equal physical intensity, so the variation in
apparent intensity (e.g. 400, 800 nm are very dark, 550 nm is bright),
is due to perceptual factors in the eye. This helps show how much
each wavelength contributes to the percieved color.
spectrum - spectrum to plot
title - title for plot
filename - filename to save plot to (or None to not save it)
xlabel - label for x axis
ylabel - label for y axis
color_vs_param_plot (
param_list,
rgb_colors,
title,
filename,
tight = False,
plotfunc = pylab.plot,
xlabel = 'param',
ylabel = 'RGB Color') -
Plot for a color that varies with a parameter -
In a two part figure, draw:
top: color as it varies with parameter (x axis)
low: r,g,b values, as linear 0.0-1.0 values, of the attempted color.
param_list - list of parameters (x axis)
rgb_colors - numpy array, one row for each param in param_list
title - title for plot
filename - filename to save plot to (or None to not save it)
plotfunc - optional plot function to use (default pylab.plot)
xlabel - label for x axis
ylabel - label for y axis (default 'RGB Color')
Specialized plots:
visible_spectrum_plot () -
Plot the visible spectrum, as a plot vs wavelength.
cie_matching_functions_plot () -
Plot the CIE XYZ matching functions, as three spectral subplots.
shark_fin_plot () -
Draw the 'shark fin' CIE chromaticity diagram of the pure spectral lines (plus purples) in xy space.
License:
Copyright (C) 2008 Mark Kness
Author - Mark Kness - [email protected]
This file is part of ColorPy.
ColorPy is free software: you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as
published by the Free Software Foundation, either version 3 of
the License, or (at your option) any later version.
ColorPy is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with ColorPy. If not, see <http://www.gnu.org/licenses/>.
'''
from __future__ import division, absolute_import, print_function
import math, random
import numpy, pylab
from . import colormodels, ciexyz
# Miscellaneous utilities for plots
def log_interpolate (y0, y1, num_values):
'''Return a list of values, num_values in size, logarithmically interpolated
between y0 and y1. The first value will be y0, the last y1.'''
rtn = []
if num_values <= 0:
raise ValueError('Invalid number of divisions %s in log_interpolate' % (str (num_values)))
if num_values == 1:
# can't use both endpoints, too constrained
yi = math.sqrt (y0 * y1)
rtn.append (yi)
else:
# normal case
beta = math.log (y1 / y0) / float (num_values - 1)
for i in range (0, num_values):
yi = y0 * math.exp (beta * float (i))
rtn.append (yi)
return rtn
def tighten_x_axis (x_list):
'''Tighten the x axis (only) of the current plot to match the given range of x values.
The y axis limits are not affected.'''
x_min = min (x_list)
x_max = max (x_list)
pylab.xlim ((x_min, x_max))
#
# Patch plots - Plots with each color value as a solid patch, with optional labels.
#
def rgb_patch_plot (
rgb_colors,
color_names,
title,
filename = None,
patch_gap = 0.05,
num_across = 6):
'''Draw a set of color patches, specified as linear rgb colors.'''
def draw_patch (x0, y0, color, name, patch_gap):
'''Draw a patch of color.'''
# patch relative vertices
m = patch_gap
omm = 1.0 - m
poly_dx = [m, m, omm, omm]
poly_dy = [m, omm, omm, m]
# construct vertices
poly_x = [ x0 + dx_i for dx_i in poly_dx ]
poly_y = [ y0 + dy_i for dy_i in poly_dy ]
pylab.fill (poly_x, poly_y, color)
if name != None:
dtext = 0.1
pylab.text (x0+dtext, y0+dtext, name, size=8.0)
# make plot with each color with one patch
pylab.clf()
num_colors = len (rgb_colors)
for i in range (0, num_colors):
(iy, ix) = divmod (i, num_across)
# get color as a displayable string
colorstring = colormodels.irgb_string_from_rgb (rgb_colors [i])
if color_names != None:
name = color_names [i]
else:
name = None
draw_patch (float (ix), float (-iy), colorstring, name, patch_gap)
pylab.axis ('off')
pylab.title (title)
if filename is not None:
print('Saving plot %s' % str (filename))
pylab.savefig (filename)
def xyz_patch_plot (
xyz_colors,
color_names,
title,
filename = None,
patch_gap = 0.05,
num_across = 6):
'''Draw a set of color patches specified as xyz colors.'''
rgb_colors = []
for xyz in xyz_colors:
rgb = colormodels.rgb_from_xyz (xyz)
rgb_colors.append (rgb)
rgb_patch_plot (rgb_colors, color_names, title, filename, patch_gap=patch_gap, num_across=num_across)
#
# Spectrum plots
#
def spectrum_subplot (spectrum):
'''Plot a spectrum, with x-axis the wavelength, and y-axis the intensity.
The curve is colored at that wavelength by the (approximate) color of a
pure spectral color at that wavelength, with intensity constant over wavelength.
(This means that dark looking colors here mean that wavelength is poorly viewed by the eye.
This is not a complete plotting function, e.g. no file is saved, etc.
It is assumed that this function is being called by one that handles those things.'''
(num_wl, num_cols) = spectrum.shape
# get rgb colors for each wavelength
rgb_colors = numpy.empty ((num_wl, 3))
for i in range (0, num_wl):
wl_nm = spectrum [i][0]
xyz = ciexyz.xyz_from_wavelength (wl_nm)
rgb_colors [i] = colormodels.rgb_from_xyz (xyz)
# scale to make brightest rgb value = 1.0
rgb_max = numpy.max (rgb_colors)
scaling = 1.0 / rgb_max
rgb_colors *= scaling
# draw color patches (thin vertical lines matching the spectrum curve) in color
for i in range (0, num_wl-1): # skipping the last one here to stay in range
x0 = spectrum [i][0]
x1 = spectrum [i+1][0]
y0 = spectrum [i][1]
y1 = spectrum [i+1][1]
poly_x = [x0, x1, x1, x0]
poly_y = [0.0, 0.0, y1, y0]
color_string = colormodels.irgb_string_from_rgb (rgb_colors [i])
pylab.fill (poly_x, poly_y, color_string, edgecolor=color_string)
# plot intensity as a curve
pylab.plot (
spectrum [:,0], spectrum [:,1],
color='k', linewidth=2.0, antialiased=True)
def spectrum_plot (
spectrum,
title,
filename = None,
xlabel = 'Wavelength ($nm$)',
ylabel = 'Intensity ($W/m^2$)'):
'''Plot for a single spectrum -
In a two part graph, plot:
top: color of the spectrum, as a large patch.
low: graph of spectrum intensity vs wavelength (x axis).
The graph is colored by the (approximated) color of each wavelength.
Each wavelength has equal physical intensity, so the variation in
apparent intensity (e.g. 400, 800 nm are very dark, 550 nm is bright),
is due to perceptual factors in the eye. This helps show how much
each wavelength contributes to the percieved color.
spectrum - spectrum to plot
title - title for plot
filename - filename to save plot to
xlabel - label for x axis
ylabel - label for y axis
'''
pylab.clf ()
# upper plot - solid patch of color that matches the spectrum color
pylab.subplot (2,1,1)
pylab.title (title)
color_string = colormodels.irgb_string_from_rgb (
colormodels.rgb_from_xyz (ciexyz.xyz_from_spectrum (spectrum)))
poly_x = [0.0, 1.0, 1.0, 0.0]
poly_y = [0.0, 0.0, 1.0, 1.0]
pylab.fill (poly_x, poly_y, color_string)
# draw a solid line around the patch to look nicer
pylab.plot (poly_x, poly_y, color='k', linewidth=2.0)
pylab.axis ('off')
# lower plot - spectrum vs wavelength, with colors of the associated spectral lines below
pylab.subplot (2,1,2)
spectrum_subplot (spectrum)
tighten_x_axis (spectrum [:,0])
pylab.xlabel (xlabel)
pylab.ylabel (ylabel)
# done
if filename is not None:
print('Saving plot %s' % str (filename))
pylab.savefig (filename)
#
# Color vs param plot
#
def color_vs_param_plot (
param_list,
rgb_colors,
title,
filename = None,
tight = False,
plotfunc = pylab.plot,
xlabel = 'param',
ylabel = 'RGB Color'):
'''Plot for a color that varies with a parameter -
In a two part figure, draw:
top: color as it varies with parameter (x axis)
low: r,g,b values, as linear 0.0-1.0 values, of the attempted color.
param_list - list of parameters (x axis)
rgb_colors - numpy array, one row for each param in param_list
title - title for plot
filename - filename to save plot to
plotfunc - optional plot function to use (default pylab.plot)
xlabel - label for x axis
ylabel - label for y axis (default 'RGB Color')
'''
pylab.clf ()
# draw color bars in upper plot
pylab.subplot (2,1,1)
pylab.title (title)
# no xlabel, ylabel in upper plot
num_points = len (param_list)
for i in range (0, num_points-1):
x0 = param_list [i]
x1 = param_list [i+1]
y0 = 0.0
y1 = 1.0
poly_x = [x0, x1, x1, x0]
poly_y = [y0, y0, y1, y1]
color_string = colormodels.irgb_string_from_rgb (rgb_colors [i])
pylab.fill (poly_x, poly_y, color_string, edgecolor=color_string)
if tight:
tighten_x_axis (param_list)
# draw rgb curves in lower plot
pylab.subplot (2,1,2)
# no title in lower plot
plotfunc (param_list, rgb_colors [:,0], color='r', label='Red')
plotfunc (param_list, rgb_colors [:,1], color='g', label='Green')
plotfunc (param_list, rgb_colors [:,2], color='b', label='Blue')
if tight:
tighten_x_axis (param_list)
pylab.xlabel (xlabel)
pylab.ylabel (ylabel)
if filename is not None:
print('Saving plot %s' % str (filename))
pylab.savefig (filename)
#
# Some specialized plots
#
def visible_spectrum_plot ():
'''Plot the visible spectrum, as a plot vs wavelength.'''
spectrum = ciexyz.empty_spectrum()
(num_wl, num_cols) = spectrum.shape
# get rgb colors for each wavelength
rgb_colors = numpy.empty ((num_wl, 3))
for i in range (0, num_wl):
xyz = ciexyz.xyz_from_wavelength (spectrum [i][0])
rgb = colormodels.rgb_from_xyz (xyz)
rgb_colors [i] = rgb
# scale to make brightest rgb value = 1.0
rgb_max = numpy.max (rgb_colors)
scaling = 1.0 / rgb_max
rgb_colors *= scaling
# plot colors and rgb values vs wavelength
color_vs_param_plot (
spectrum [:,0],
rgb_colors,
'The Visible Spectrum',
'VisibleSpectrum',
tight = True,
xlabel = r'Wavelength (nm)',
ylabel = r'RGB Color')
def cie_matching_functions_plot ():
'''Plot the CIE XYZ matching functions, as three spectral subplots.'''
# get 'spectra' for x,y,z matching functions
spectrum_x = ciexyz.empty_spectrum()
spectrum_y = ciexyz.empty_spectrum()
spectrum_z = ciexyz.empty_spectrum()
(num_wl, num_cols) = spectrum_x.shape
for i in range (0, num_wl):
wl_nm = spectrum_x [i][0]
xyz = ciexyz.xyz_from_wavelength (wl_nm)
spectrum_x [i][1] = xyz [0]
spectrum_y [i][1] = xyz [1]
spectrum_z [i][1] = xyz [2]
# Plot three separate subplots, with CIE X in the first, CIE Y in the second, and CIE Z in the third.
# Label appropriately for the whole plot.
pylab.clf ()
# X
pylab.subplot (3,1,1)
pylab.title ('1931 CIE XYZ Matching Functions')
pylab.ylabel ('CIE $X$')
spectrum_subplot (spectrum_x)
tighten_x_axis (spectrum_x [:,0])
# Y
pylab.subplot (3,1,2)
pylab.ylabel ('CIE $Y$')
spectrum_subplot (spectrum_y)
tighten_x_axis (spectrum_x [:,0])
# Z
pylab.subplot (3,1,3)
pylab.xlabel ('Wavelength (nm)')
pylab.ylabel ('CIE $Z$')
spectrum_subplot (spectrum_z)
tighten_x_axis (spectrum_x [:,0])
# done
filename = 'CIEXYZ_Matching'
print('Saving plot %s' % str (filename))
pylab.savefig (filename)
def shark_fin_plot ():
'''Draw the 'shark fin' CIE chromaticity diagram of the pure spectral lines (plus purples) in xy space.'''
# get array of (approximate) colors for the boundary of the fin
xyz_list = ciexyz.get_normalized_spectral_line_colors (brightness=1.0, num_purples=200, dwl_angstroms=2)
# get normalized colors
xy_list = xyz_list.copy()
(num_colors, num_cols) = xy_list.shape
for i in range (0, num_colors):
colormodels.xyz_normalize (xy_list [i])
# get phosphor colors and normalize
red = colormodels.PhosphorRed
green = colormodels.PhosphorGreen
blue = colormodels.PhosphorBlue
white = colormodels.PhosphorWhite
colormodels.xyz_normalize (red)
colormodels.xyz_normalize (green)
colormodels.xyz_normalize (blue)
colormodels.xyz_normalize (white)
def get_direc_to_white (xyz):
'''Get unit vector (xy plane) in direction of the white point.'''
direc = white - xyz
mag = math.hypot (direc [0], direc [1])
if mag != 0.0:
direc /= mag
return (direc[0], direc[1])
# plot
pylab.clf ()
# draw color patches for point in xy_list
s = 0.025 # distance in xy plane towards white point
for i in range (0, len (xy_list)-1):
x0 = xy_list [i][0]
y0 = xy_list [i][1]
x1 = xy_list [i+1][0]
y1 = xy_list [i+1][1]
# get unit vectors in direction of white point
(dir_x0, dir_y0) = get_direc_to_white (xy_list [i])
(dir_x1, dir_y1) = get_direc_to_white (xy_list [i+1])
# polygon vertices
poly_x = [x0, x1, x1 + s*dir_x1, x0 + s*dir_x0]
poly_y = [y0, y1, y1 + s*dir_y1, y0 + s*dir_y0]
# draw (using full color, not normalized value)
color_string = colormodels.irgb_string_from_rgb (
colormodels.rgb_from_xyz (xyz_list [i]))
pylab.fill (poly_x, poly_y, color_string, edgecolor=color_string)
# draw the curve of the xy values of the spectral lines and purples
pylab.plot (xy_list [:,0], xy_list [:,1], color='#808080', linewidth=3.0)
# draw monitor gamut and white point
pylab.plot ([red [0], green[0]], [red [1], green[1]], 'o-', color='k')
pylab.plot ([green[0], blue [0]], [green[1], blue [1]], 'o-', color='k')
pylab.plot ([blue [0], red [0]], [blue [1], red [1]], 'o-', color='k')
pylab.plot ([white[0], white[0]], [white[1], white[1]], 'o-', color='k')
# label phosphors
dx = 0.01
dy = 0.01
pylab.text (red [0] + dx, red [1], 'Red', ha='left', va='center')
pylab.text (green [0], green [1] + dy, 'Green', ha='center', va='bottom')
pylab.text (blue [0] - dx, blue [1], 'Blue', ha='right', va='center')
pylab.text (white [0], white [1] + dy, 'White', ha='center', va='bottom')
# titles etc
pylab.axis ([0.0, 0.85, 0.0, 0.85])
pylab.xlabel (r'CIE $x$')
pylab.ylabel (r'CIE $y$')
pylab.title (r'CIE Chromaticity Diagram')
filename = 'ChromaticityDiagram'
print('Saving plot %s' % (str (filename)))
pylab.savefig (filename)
# Special figures
def figures ():
'''Draw specific figures not used anywhere else.'''
visible_spectrum_plot()
cie_matching_functions_plot()
shark_fin_plot()