forked from rougier/matplotlib-3d
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathelevation.py
116 lines (89 loc) · 3.63 KB
/
elevation.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
# -----------------------------------------------------------------------------
# Copyright (c) 2020 Nicolas P. Rougier. All rights reserved.
# Distributed under the (new) BSD License.
# -----------------------------------------------------------------------------
# This example shows how to display a mesh
# -----------------------------------------------------------------------------
import numpy as np
from mpl3d import glm
from mpl3d.mesh import Mesh
from mpl3d.camera import Camera
class Surface(Mesh):
def __init__(self, ax, transform, Z, border=True, *args, **kwargs):
if border:
n2, n1 = Z.shape
Z_ = np.zeros((n2+2, n1+2))
Z_[1:-1,1:-1] = Z
Z = Z_
x = np.zeros(n1+2)
x[1:-1] = np.linspace(-0.5, +0.5, n1)
x[0], x[-1] = x[1], x[-2]
y = np.zeros(n2+2)
y[1:-1] = np.linspace(-0.5, +0.5, n2)
y[0], y[-1] = y[1], y[-2]
F = kwargs["facecolors"]
F_ = np.zeros((F.shape[0]+2,F.shape[1]+2,F.shape[2]))
F_[1:-1,1:-1] = F
F_[0,:] = F_[1,:]
F_[-1,:] = F_[-2,:]
F_[:,0] = F_[:,1]
F_[:,-1] = F_[:,-2]
kwargs["facecolors"] = F_
else:
n2, n1 = Z.shape
x = np.linspace(-0.5, +0.5, n1)
y = np.linspace(-0.5, +0.5, n2)
n2, n1 = Z.shape
X, Y = np.meshgrid(x, y)
vertices = np.c_[X.ravel(), Y.ravel(), Z.ravel()]
F = (np.arange((n2-1)*(n1)).reshape(n2-1,n1))[:,:-1].T
F = np.repeat(F.ravel(),6).reshape(n2-1,n1-1,6)
F[:,:] += 0,n1+1,1, 0,n1,n1+1
faces = F.reshape(-1,3)
# Recompute colors for triangles based on vertices color
facecolors = kwargs["facecolors"].reshape(-1,4)
facecolors = facecolors[faces].mean(axis=-2)
facecolors = facecolors.reshape(-1,4)[:,:3]
F = facecolors.reshape(n1-1, n2-1, 2, 3)
F[0] = F[-1] = F[:,0] = F[:,-1] = .75,.75,.75
F = vertices[faces]
# Light direction
direction = glm.normalize([1.5,1.5,-1])
# Faces center
C = F.mean(axis=1)
# Faces normal
N = glm.normalize(np.cross(F[:,2]-F[:,0], F[:,1]-F[:,0]))
# Relative light direction
D = glm.normalize(C - direction)
# Diffuse term
diffuse = glm.clip((N*D).sum(-1).reshape(-1,1))
facecolors = (1-diffuse)*facecolors
kwargs["facecolors"] = facecolors
Mesh.__init__(self, ax, transform, vertices, faces, *args, **kwargs)
# --- main --------------------------------------------------------------------
if __name__ == "__main__":
import time
import matplotlib as mpl
import matplotlib.pyplot as plt
fig = plt.figure(figsize=(8,8))
ax = fig.add_axes([0,0,1,1], xlim=[-1,+1], ylim=[-1,+1], aspect=1)
ax.axis("off")
Z = np.load("data/st-helens-after.npy")
Z[Z==-32767.] = 2231
stride = 2
Z = Z[20:-20:stride, 20:-20:stride]
Z = (Z-Z.min())/(Z.max()-Z.min())
camera = Camera("perspective", 45, 45, scale=1.25)
cmap = plt.get_cmap("viridis")
facecolors = cmap(Z)
start = time.time()
surface = Surface(ax, camera.transform, 0.3*Z, mode="all",
facecolors=facecolors, linewidths=0)
elapsed = time.time() - start
text = "{0} vertices, rendered in {1:.2f} second(s) with matplotlib"
text = text.format(Z.size, elapsed)
ax.text(0, 0, text, va="bottom", ha="left",
transform=ax.transAxes, size="x-small")
camera.connect(ax, surface.update)
plt.savefig("elevation.png", dpi=600)
plt.show()