Coverage for /usr/local/lib/python3.11/dist-packages/pyrocko/dataset/topo/tile.py: 58%
150 statements
« prev ^ index » next coverage.py v6.5.0, created at 2023-10-06 15:01 +0000
« prev ^ index » next coverage.py v6.5.0, created at 2023-10-06 15:01 +0000
1# http://pyrocko.org - GPLv3
2#
3# The Pyrocko Developers, 21st Century
4# ---|P------/S----------~Lg----------
6'''
7Topography tile data model.
8'''
10import math
11import numpy as num
12import scipy.signal
14from pyrocko.orthodrome import positive_region
16km = 1e3
19class OutOfBounds(Exception):
20 pass
23class Tile(object):
24 '''
25 2D array of data values which are regularly gridded in lon/lat.
26 '''
28 def __init__(self, xmin, ymin, dx, dy, data):
29 self.xmin = float(xmin)
30 self.ymin = float(ymin)
31 self.dx = float(dx)
32 self.dy = float(dy)
33 self.data = data
34 self._set_maxes()
36 def _set_maxes(self):
37 self.ny, self.nx = self.data.shape
38 self.xmax = self.xmin + (self.nx-1) * self.dx
39 self.ymax = self.ymin + (self.ny-1) * self.dy
41 def x(self):
42 return self.xmin + num.arange(self.nx) * self.dx
44 def y(self):
45 return self.ymin + num.arange(self.ny) * self.dy
47 def decimate(self, ndeci):
48 assert ndeci % 2 == 0
49 kernel = num.ones((ndeci+1, ndeci+1))
50 kernel /= num.sum(kernel)
51 data = scipy.signal.convolve2d(
52 self.data.astype(float), kernel, mode='valid')
54 self.data = data[::ndeci, ::ndeci].astype(self.data.dtype)
55 self.xmin += ndeci/2
56 self.ymin += ndeci/2
57 self.dx *= ndeci
58 self.dy *= ndeci
59 self._set_maxes()
61 def yextend_with_repeat(self, ymin, ymax):
62 assert ymax >= self.ymax
63 assert ymin <= self.ymin
65 nlo = int(round((self.ymin - ymin) / self.dy))
66 nhi = int(round((ymax - self.ymax) / self.dy))
68 nx, ny = self.nx, self.ny
69 data = num.zeros((ny+nlo+nhi, nx), dtype=self.data.dtype)
70 data[:nlo, :] = self.data[nlo, :]
71 data[nlo:nlo+ny, :] = self.data
72 data[nlo+ny:, :] = self.data[-1, :]
74 self.ymin = ymin
75 self.data = data
76 self._set_maxes()
78 def get(self, x, y):
79 ix = int(round((x - self.xmin) / self.dx))
80 iy = int(round((y - self.ymin) / self.dy))
81 if 0 <= ix < self.nx and 0 <= iy < self.ny:
82 return self.data[iy, ix]
83 else:
84 raise OutOfBounds()
86 @classmethod
87 def from_grd(cls, path):
88 '''Load from GMT .grd'''
89 from pyrocko.plot import gmtpy
90 lon, lat, z = gmtpy.loadgrd(path)
91 return cls(lon[0], lat[0], lon[1] - lon[0], lat[1] - lat[0], z)
94def multiple_of(x, dx, eps=1e-5):
95 return abs(int(round(x / dx))*dx - x) < dx * eps
98def combine(tiles, region=None):
99 if not tiles:
100 return None
102 dx = tiles[0].dx
103 dy = tiles[0].dy
104 dtype = tiles[0].data.dtype
106 assert all(t.dx == dx for t in tiles)
107 assert all(t.dy == dy for t in tiles)
108 assert all(t.data.dtype == dtype for t in tiles)
109 assert all(multiple_of(t.xmin, dx) for t in tiles)
110 assert all(multiple_of(t.ymin, dy) for t in tiles)
112 if region is None:
113 xmin = min(t.xmin for t in tiles)
114 xmax = max(t.xmax for t in tiles)
115 ymin = min(t.ymin for t in tiles)
116 ymax = max(t.ymax for t in tiles)
117 else:
118 xmin, xmax, ymin, ymax = positive_region(region)
120 if not multiple_of(xmin, dx):
121 xmin = math.floor(xmin / dx) * dx
122 if not multiple_of(xmax, dx):
123 xmax = math.ceil(xmax / dx) * dx
124 if not multiple_of(ymin, dy):
125 ymin = math.floor(ymin / dy) * dy
126 if not multiple_of(ymax, dy):
127 ymax = math.ceil(ymax / dy) * dy
129 nx = int(round((xmax - xmin) / dx)) + 1
130 ny = int(round((ymax - ymin) / dy)) + 1
132 data = num.zeros((ny, nx), dtype=dtype)
133 data[:, :] = 0
135 for t in tiles:
136 for txmin in (t.xmin, t.xmin + 360.):
137 ix = int(round((txmin - xmin) / dx))
138 iy = int(round((t.ymin - ymin) / dy))
139 ixlo = max(ix, 0)
140 ixhi = min(ix+t.nx, nx)
141 iylo = max(iy, 0)
142 iyhi = min(iy+t.ny, ny)
143 jxlo = ixlo-ix
144 jxhi = jxlo + max(0, ixhi - ixlo)
145 jylo = iylo-iy
146 jyhi = jylo + max(0, iyhi - iylo)
147 if iyhi > iylo and ixhi > ixlo:
148 data[iylo:iyhi, ixlo:ixhi] = t.data[jylo:jyhi, jxlo:jxhi]
150 if not num.any(num.isfinite(data)):
151 return None
153 return Tile(xmin, ymin, dx, dy, data)
156def tile_export(tle, path, format='stl', exaggeration=3., socket_scale=1.,
157 scale=100):
158 '''Export DEM to 3D printable formats'''
159 import trimesh
161 from pyrocko import geometry
162 from pyrocko.cake import earthradius
164 x = tle.x()
165 x = x - x.min()
167 y = tle.y()
168 y = y - y.min()
170 data = tle.data * exaggeration
172 vertices, faces = geometry.topo_to_mesh(
173 y, x, data, earthradius)
175 vertices[:, 0] -= vertices[:, 0].min()
177 nlat = y.size
178 nlon = x.size
179 socket_level = -vertices[:, 0].max() * socket_scale
180 nvertices = vertices.shape[0]
182 south_indices = num.arange(0, nlon)
183 north_indices = num.arange(nvertices - nlon, nvertices)
184 east_indices = num.arange(0, nlat) * nlon
185 west_indices = num.arange(0, nlat) * nlon + nlon - 1
187 north_indices = north_indices[::-1]
188 east_indices = east_indices[::-1]
190 socket_bottom_faces = []
191 for border_indices in [north_indices, east_indices,
192 south_indices, west_indices]:
193 nvertices = vertices.shape[0]
194 nfaces = border_indices.size - 1
196 socket_vertices = vertices[border_indices.astype(int)].copy()
197 socket_vertices[:, 0] = socket_level
199 socket_faces = num.empty((nfaces, 4), dtype=int)
200 for iface in range(nfaces):
201 socket_faces[iface, 0] = iface + nvertices
202 socket_faces[iface, 1] = iface + nvertices + 1
203 socket_faces[iface, 2] = border_indices[iface+1]
204 socket_faces[iface, 3] = border_indices[iface]
206 vertices = num.vstack((vertices, socket_vertices))
207 faces = num.vstack((faces, socket_faces))
209 socket_bottom_faces.append(vertices.shape[0] - nfaces)
211 faces = num.vstack((faces, socket_bottom_faces[::-1]))
213 mesh = trimesh.Trimesh(
214 vertices, faces)
215 mesh.fix_normals()
216 mesh.fill_holes()
218 mesh.apply_scale(scale)
220 with open(path, 'wb') as f:
221 mesh.export(f, file_type=format)
223 return mesh