diff --git a/autotest/test_export.py b/autotest/test_export.py index c7f84661da..5a06989110 100644 --- a/autotest/test_export.py +++ b/autotest/test_export.py @@ -139,6 +139,44 @@ def disu_sim(name, tmpdir, missing_arrays=False): return sim +@pytest.fixture +def unstructured_grid(example_data_path): + ws = example_data_path / "unstructured" + + # load vertices + verts = load_verts(ws / "ugrid_verts.dat") + + # load the index list into iverts, xc, and yc + iverts, xc, yc = load_iverts(ws / "ugrid_iverts.dat", closed=True) + + # create a 3 layer model grid + ncpl = np.array(3 * [len(iverts)]) + nnodes = np.sum(ncpl) + + top = np.ones(nnodes) + botm = np.ones(nnodes) + + # set top and botm elevations + i0 = 0 + i1 = ncpl[0] + elevs = [100, 0, -100, -200] + for ix, cpl in enumerate(ncpl): + top[i0:i1] *= elevs[ix] + botm[i0:i1] *= elevs[ix + 1] + i0 += cpl + i1 += cpl + + return UnstructuredGrid( + vertices=verts, + iverts=iverts, + xcenters=xc, + ycenters=yc, + top=top, + botm=botm, + ncpl=ncpl, + ) + + @requires_pkg("shapefile") @pytest.mark.parametrize("pathlike", (True, False)) def test_output_helper_shapefile_export( @@ -613,7 +651,7 @@ def test_export_array2(function_tmpdir): @requires_pkg("shapefile", "shapely") -def test_export_array_contours(function_tmpdir): +def test_export_array_contours_structured(function_tmpdir): nrow = 7 ncol = 11 crs = 4431 @@ -648,6 +686,61 @@ def test_export_array_contours(function_tmpdir): assert os.path.isfile(filename), "did not create contour shapefile" +@requires_pkg("shapefile", "shapely") +def test_export_array_contours_unstructured( + function_tmpdir, unstructured_grid +): + from shapefile import Reader + + grid = unstructured_grid + fname = function_tmpdir / "myarraycontours1.shp" + export_array_contours(grid, fname, np.arange(grid.nnodes)) + assert fname.is_file(), "did not create contour shapefile" + + # visual debugging + grid.plot(alpha=0.2) + with Reader(fname) as r: + shapes = r.shapes() + for s in shapes: + x = [i[0] for i in s.points[:]] + y = [i[1] for i in s.points[:]] + plt.plot(x, y) + + # plt.show() + + +from autotest.test_gridgen import sim_disu_diff_layers + + +@requires_pkg("shapefile", "shapely") +def test_export_array_contours_unstructured_diff_layers( + function_tmpdir, sim_disu_diff_layers +): + from shapefile import Reader + + gwf = sim_disu_diff_layers.get_model() + grid = gwf.modelgrid + a = np.arange(grid.nnodes) + for layer in range(3): + fname = function_tmpdir / f"contours.{layer}.shp" + export_array_contours(grid, fname, a, layer=layer) + assert fname.is_file(), "did not create contour shapefile" + + # visual debugging + fig, axes = plt.subplots(1, 3, subplot_kw={"aspect": "equal"}) + for layer, ax in enumerate(axes): + fname = function_tmpdir / f"contours.{layer}.shp" + with Reader(fname) as r: + shapes = r.shapes() + for s in shapes: + x = [i[0] for i in s.points[:]] + y = [i[1] for i in s.points[:]] + ax.plot(x, y) + grid.plot(ax=ax, alpha=0.2, layer=layer) + + # plt.show() + + @requires_pkg("shapefile", "shapely") def test_export_contourf(function_tmpdir, example_data_path): from shapefile import Reader @@ -1331,52 +1424,18 @@ def test_vtk_vector(function_tmpdir, example_data_path): @requires_pkg("vtk") -def test_vtk_unstructured(function_tmpdir, example_data_path): +def test_vtk_unstructured(function_tmpdir, unstructured_grid): from vtkmodules.util.numpy_support import vtk_to_numpy from vtkmodules.vtkIOLegacy import vtkUnstructuredGridReader - u_data_ws = example_data_path / "unstructured" - - # load vertices - verts = load_verts(u_data_ws / "ugrid_verts.dat") - - # load the index list into iverts, xc, and yc - iverts, xc, yc = load_iverts(u_data_ws / "ugrid_iverts.dat", closed=True) - - # create a 3 layer model grid - ncpl = np.array(3 * [len(iverts)]) - nnodes = np.sum(ncpl) - - top = np.ones(nnodes) - botm = np.ones(nnodes) - - # set top and botm elevations - i0 = 0 - i1 = ncpl[0] - elevs = [100, 0, -100, -200] - for ix, cpl in enumerate(ncpl): - top[i0:i1] *= elevs[ix] - botm[i0:i1] *= elevs[ix + 1] - i0 += cpl - i1 += cpl - - # create the modelgrid - modelgrid = UnstructuredGrid( - vertices=verts, - iverts=iverts, - xcenters=xc, - ycenters=yc, - top=top, - botm=botm, - ncpl=ncpl, - ) + grid = unstructured_grid outfile = function_tmpdir / "disu_grid.vtu" vtkobj = Vtk( - modelgrid=modelgrid, vertical_exageration=2, binary=True, smooth=False + modelgrid=grid, vertical_exageration=2, binary=True, smooth=False ) - vtkobj.add_array(modelgrid.top, "top") - vtkobj.add_array(modelgrid.botm, "botm") + vtkobj.add_array(grid.top, "top") + vtkobj.add_array(grid.botm, "botm") vtkobj.write(outfile) assert is_binary_file(outfile) @@ -1390,7 +1449,9 @@ def test_vtk_unstructured(function_tmpdir, example_data_path): top2 = vtk_to_numpy(data.GetCellData().GetArray("top")) - assert np.allclose(np.ravel(top), top2), "Field data not properly written" + assert np.allclose( + np.ravel(grid.top), top2 + ), "Field data not properly written" @requires_pkg("vtk", "pyvista") diff --git a/autotest/test_gridgen.py b/autotest/test_gridgen.py index 0703a274eb..4063e86ff0 100644 --- a/autotest/test_gridgen.py +++ b/autotest/test_gridgen.py @@ -164,13 +164,11 @@ def test_mf6disv(function_tmpdir): plt.close("all") -@pytest.mark.slow -@requires_exe("mf6", "gridgen") -@requires_pkg("shapely", "shapefile") -def test_mf6disu(function_tmpdir): +@pytest.fixture +def sim_disu_diff_layers(function_tmpdir): from shapely.geometry import Polygon - name = "dummy" + name = "disu_diff_layers" nlay = 3 nrow = 10 ncol = 10 @@ -232,6 +230,17 @@ def test_mf6disu(function_tmpdir): head_filerecord=head_file, saverecord=[("HEAD", "ALL"), ("BUDGET", "ALL")], ) + + return sim + + +@pytest.mark.slow +@requires_exe("mf6", "gridgen") +@requires_pkg("shapely", "shapefile") +def test_mf6disu(sim_disu_diff_layers): + sim = sim_disu_diff_layers + ws = sim.sim_path + gwf = sim.get_model() sim.write_simulation() gwf.modelgrid.set_coord_info(angrot=15) @@ -243,9 +252,9 @@ def test_mf6disu(function_tmpdir): assert np.allclose(gwf.modelgrid.ncpl, np.array([436, 184, 112])) # write grid and model shapefiles - fname = function_tmpdir / "grid.shp" + fname = ws / "grid.shp" gwf.modelgrid.write_shapefile(fname) - fname = function_tmpdir / "model.shp" + fname = ws / "model.shp" gwf.export(fname) sim.run_simulation(silent=True) @@ -272,7 +281,7 @@ def test_mf6disu(function_tmpdir): ax.set_title(f"Layer {ilay + 1}") pmv.plot_vector(spdis["qx"], spdis["qy"], color="white") fname = "results.png" - fname = function_tmpdir / fname + fname = ws / fname plt.savefig(fname) plt.close("all") @@ -307,7 +316,7 @@ def test_mf6disu(function_tmpdir): sim_name=name, version="mf6", exe_name="mf6", - sim_ws=function_tmpdir, + sim_ws=ws, ) gwf = sim.get_model(name) diff --git a/flopy/export/utils.py b/flopy/export/utils.py index 0277e55d43..ff77baadda 100644 --- a/flopy/export/utils.py +++ b/flopy/export/utils.py @@ -1948,14 +1948,15 @@ def export_array_contours( assert nlevels < maxlevels, msg levels = np.arange(imin, imax, interval) ax = plt.subplots()[-1] - ctr = contour_array(modelgrid, ax, a, levels=levels) + layer = kwargs.pop("layer", 0) + ctr = contour_array(modelgrid, ax, a, layer, levels=levels) kwargs["mg"] = modelgrid export_contours(filename, ctr, fieldname, **kwargs) plt.close() -def contour_array(modelgrid, ax, a, **kwargs): +def contour_array(modelgrid, ax, a, layer=0, **kwargs): """ Create a QuadMesh plot of the specified array using pcolormesh @@ -1967,6 +1968,8 @@ def contour_array(modelgrid, ax, a, **kwargs): ax to add the contours a : np.ndarray array to contour + layer : int, optional + layer to contour Returns ------- @@ -1976,7 +1979,7 @@ def contour_array(modelgrid, ax, a, **kwargs): from ..plot import PlotMapView kwargs["ax"] = ax - pmv = PlotMapView(modelgrid=modelgrid) + pmv = PlotMapView(modelgrid=modelgrid, layer=layer) contour_set = pmv.contour_array(a=a, **kwargs) return contour_set diff --git a/flopy/plot/map.py b/flopy/plot/map.py index 98ecdd44d8..de09071224 100644 --- a/flopy/plot/map.py +++ b/flopy/plot/map.py @@ -156,8 +156,9 @@ def plot_array(self, a, masked_values=None, **kwargs): def contour_array(self, a, masked_values=None, **kwargs): """ - Contour an array. If the array is three-dimensional, then the method - will contour the layer tied to this class (self.layer). + Contour an array on the grid. By default the top layer + is contoured. To select a different layer, specify the + layer in the class constructor. Parameters ----------