Mesh#
See Mesh in MIKE IO Documentation
import numpy as np
import matplotlib.pyplot as plt
import mikeio
A simple mesh#
Let’s consider a simple mesh consisting of 2 triangular elements.
fn = "data/two_elements.mesh"
with open(fn, "r") as f:
print(f.read())
100079 1000 4 UTM-31
1 0.0 0.0 -10.0 1
2 3.0 0.0 -10.0 2
3 3.0 3.0 -10.0 2
4 0.0 3.0 -10.0 1
2 3 21
1 1 2 4
2 2 3 4
msh = mikeio.open(fn)
msh
<Mesh>
number of elements: 2
number of nodes: 4
projection: UTM-31
msh.plot(show_mesh=True);
data:image/s3,"s3://crabby-images/4d7d1/4d7d1240e957a1fa36e5b2953acf8feacddba8e4" alt="_images/23e3ab3146f26c019f8d095b9bfc733324e3cb0c50e162534c84430c19000e35.png"
msh.geometry
Flexible Mesh Geometry: Dfsu2D
number of nodes: 4
number of elements: 2
projection: UTM-31
msh.node_coordinates
array([[ 0., 0., -10.],
[ 3., 0., -10.],
[ 3., 3., -10.],
[ 0., 3., -10.]])
msh.element_table
[array([0, 1, 3], dtype=int32), array([1, 2, 3], dtype=int32)]
msh.element_coordinates
array([[ 1., 1., -10.],
[ 2., 2., -10.]])
msh.geometry.get_element_area()
array([4.5, 4.5])
Let’s plot the node and element coordinates:
xn, yn = msh.node_coordinates[:,0], msh.node_coordinates[:,1]
xe, ye = msh.element_coordinates[:,0], msh.element_coordinates[:,1]
ax = msh.plot(show_mesh=True)
ax.plot(xn, yn, 'ro', markersize=10)
ax.plot(xe, ye, 'bx', markersize=10)
[<matplotlib.lines.Line2D at 0x7f66d1ee3ad0>]
data:image/s3,"s3://crabby-images/f0073/f0073adf0c123f670e72301befb4d9380ba9d649" alt="_images/30fafd57a25387f48eecb9c1625330b082ea1d07c413ca19e87a04ad4260cf1f.png"
Boundary polylines#
It can sometimes be convenient to have mesh boundary as a polyline (or multiple in case of more complex meshes).
bxy = msh.geometry.boundary_polylines.exteriors[0].xy
plt.plot(bxy[:,0], bxy[:,1])
plt.axis("equal");
data:image/s3,"s3://crabby-images/bcf31/bcf3129c150e6ab70e23546276549834f9f40a52" alt="_images/583e426200a69825e80db59b02a2a9e113711d192ad5bd983aade20d89ddea00.png"
Inside domain?#
MIKE IO has a method for determining if a point (or a list of points) is inside the domain:
contains()
pt_1 = [2.0, 1.2]
msh.geometry.contains(pt_1)[0]
np.True_
# or multiple points at the same time
pt_2 = [4.0, 1.2]
pts = np.array([pt_1, pt_2])
msh.geometry.contains(pts)
array([ True, False])
plt.plot(bxy[:,0], bxy[:,1], label='boundary')
plt.plot(xe[0], ye[0], 'b*', markersize=10, label="center, elem 0")
plt.plot(xe[1], ye[1], 'c*', markersize=10, label="center, elem 1")
plt.plot(*pt_1, 'go', markersize=10, label="pt_1")
plt.plot(*pt_2, 'rs', markersize=10, label="pt_2")
plt.axis("equal")
plt.legend(loc="upper right");
data:image/s3,"s3://crabby-images/b498c/b498c55daa38a66fb47799d7591fd4ced2d5a345" alt="_images/b76c83a52fe457375ae2fc7bea12380e0ef630186a80a5d996ff66ebc9d6380a.png"
Find element containing point#
MIKE IO has a method for obtaining the index of the element containing a point:
find_index()
g = msh.geometry
g.find_index(coords=pt_1)[0]
np.int64(1)
MIKE IO also has a method for obtaining a list of the n closest element centers:
find_nearest_elements()
g.find_nearest_elements(pt_1)
1
g.find_nearest_elements(pt_1, return_distances=True)
(1, 0.8)
g.find_nearest_elements(pt_1, n_nearest=2)
array([1, 0])
# for multiple points
g.find_nearest_elements(pts, return_distances=True)
(array([1, 1]), array([0.8 , 2.15406592]))
A larger mesh#
dfs = mikeio.open("data/FakeLake.dfsu")
g = dfs.geometry
g
Flexible Mesh Geometry: Dfsu2D
number of nodes: 798
number of elements: 1011
projection: PROJCS["UTM-17",GEOGCS["Unused",DATUM["UTM Projections",SPHEROID["WGS 1984",6378137,298.257223563]],PRIMEM["Greenwich",0],UNIT["Degree",0.0174532925199433]],PROJECTION["Transverse_Mercator"],PARAMETER["False_Easting",500000],PARAMETER["False_Northing",0],PARAMETER["Central_Meridian",-81],PARAMETER["Scale_Factor",0.9996],PARAMETER["Latitude_Of_Origin",0],UNIT["Meter",1]]
g.plot();
data:image/s3,"s3://crabby-images/6be61/6be61bb512f8aab2b9efe807216575b3497e3a6c" alt="_images/51332fede9da62b6a6af4f243cf90e548d29864fdab14c985bc3b99b695c0bb6.png"
Inline Exercise#
please check if the point A: (x,y)=(-0.5, 0.0) is inside the mesh
please check if the point B: (x,y)=(-0.5, 0.4) is inside the mesh
find index of the 5 closest points to B
# insert code here
g.max_nodes_per_element
4
Change depth#
msh = mikeio.open("data/FakeLake.dfsu").geometry
msh.plot();
data:image/s3,"s3://crabby-images/6be61/6be61bb512f8aab2b9efe807216575b3497e3a6c" alt="_images/51332fede9da62b6a6af4f243cf90e548d29864fdab14c985bc3b99b695c0bb6.png"
msh.node_coordinates[:,2] = np.clip(msh.node_coordinates[:,2], -15, 0) # clip depth to interval [-15,0]
msh.plot(title="No change??")
<Axes: title={'center': 'No change??'}, xlabel='Easting [m]', ylabel='Northing [m]'>
data:image/s3,"s3://crabby-images/3c19f/3c19f4145023c855fc69cbf15e1e628473c87c9e" alt="_images/5f45f9953874b350d5338ede058f113d283175e43a7a4c41814860ef1e9481ad.png"
del msh.element_coordinates # remove cached element coords calculated based on original node coords)
msh.plot(title="Updated")
<Axes: title={'center': 'Updated'}, xlabel='Easting [m]', ylabel='Northing [m]'>
data:image/s3,"s3://crabby-images/67d12/67d12f5523bd0f9731d8137c44c5ecf4ac8bab7b" alt="_images/3c6595bc45e98b1005f5b41286324fde9de4c825bdb1534f576afc0321cd7dd7.png"
msh.to_mesh('Fake_lake_clip15.mesh') # save to a new file
Visualisation#
msh = mikeio.open("data/southern_north_sea.mesh")
msh
<Mesh>
number of elements: 958
number of nodes: 570
projection: LONG/LAT
The default is to plot the elements and color them according to the bathymetry.
msh.plot();
data:image/s3,"s3://crabby-images/12055/1205506ee8c2451e122d9df576b73ae3f9e17337" alt="_images/3f4d0464b246e9583c5cb5a78f60ff77658bc52d3ecd8ba7fc725dae18444e62.png"
msh.plot.outline();
data:image/s3,"s3://crabby-images/d85af/d85afca5abf3bbac4229685acc3917271f8c8d93" alt="_images/280c1a1f25eba0150693238c527b6b8b8f40253a604eb5984154d65a6cf26171.png"
msh.plot.mesh();
data:image/s3,"s3://crabby-images/c1bd9/c1bd910a319bd2aecb01d43cd73fb44a032f1dce" alt="_images/3b743ab1e95203958d37604ae697d39df10e8067a70f3ca4eab8ebd4450cd988.png"
Maybe we would like to higlight the bathymetric variations in some range, in this case in the -40, -20m range.
msh.plot(vmin=-40, vmax=-20);
data:image/s3,"s3://crabby-images/ba9f3/ba9f3a7612890b6f85a0d134c703fa315155389d" alt="_images/ec51456a62d5749c2275c963bb10f10a7cc629c6218af39c0509e78dcc3c3a89.png"
There are other options as well, such as explicit specification of which contour lines to show or choosing a specific colormap (matplotlib colormaps)
msh.plot.contour(show_mesh=True,
levels=[-50,-30,-20,-10,-5], cmap="tab10",
figsize=(12,12), title="Coarse North Sea model");
data:image/s3,"s3://crabby-images/6dc6a/6dc6a0ab662808d971b3a6546492bddad8a41135" alt="_images/b56a40601f9b8b289f3aacd00d1964713be8f9b6cd6942432114caffc8ac9e7c.png"
See more in the MIKE IO User guide section on Mesh for more Mesh operations (including shapely operations).