River Catchment – Tasks
In this exercise, a river catchment area is analyzed regarding its slope and curvature. The following tasks are performed:
- Slope Visualization, Statistics and Histogram for the whole catchment area
- Analyze the slope and it’s statistics for different elevation zones within the catchment area
- Comparing slope statistics for 10m and 100m spatial resolution
- Calculating curvature without the “curvature” function in ArcGIS Pro
Study Area
As a study area, the Neubach catchment area was chosen. This little river originates in the mountains of the Dachstein massif and its slopes, before feeding into the Lammer which eventually feeds into the Salzach near Golling an der Salzach.
data:image/s3,"s3://crabby-images/1e412/1e412fbc40e95c444acc7508374eba024bd5fe42" alt="Unbenannt3 Image 1: Location of the study area in relation to Salzburg. Study area is already visualized as DEM."
data:image/s3,"s3://crabby-images/bec44/bec442fb65f7d7024bf6c32b95ef2d8d86f13537" alt="hillshade_catchment Image 2: 3D Hillshide visualization of the catchment area. The red zones deliniate the multiple catchment areas that eventually drain into the Neubach."
For the analysis, seven separated catchment areas were unified. Those areas all have the same exit point, where the Neubach drains into the Lammer. The Catchment Code (HZB_CODE) unifies those areas under the code 2-8272231-14.
The study area ranges from 850m in elevation at the drainage point until just over 2000m on the flank of the Dachstein massif.
Slope – complete Catchment Area
From the DEM, the slope per pixel is calculated in degrees. Also, a histogram is created in Python (the code can be found at the end of the post) that shows the slope of the whole study area in 5° steps as well as the mean value.
Click right on the slideshow to see the Histogram
data:image/s3,"s3://crabby-images/911f4/911f40d9b95251b498e5f38f6e5bf0931653ca1e" alt="Image 3: Slope visualization of catchment area."
data:image/s3,"s3://crabby-images/3494a/3494a836d35a16b90bf6ca732dd44c42d78b86b1" alt="Graph 1: Slope Histogram of 10x10m pixels for the whole study area and average slope line."
Elevation Zone Creation
Next up, the catchment area is devided into the following four elevation zones:
- <1000m
- 1000 – 1250m
- 1250 – 1500m
- 1500 – 1750m
- > 1750m
data:image/s3,"s3://crabby-images/528b1/528b1eeb04a36bcb8958a7e16c479e62ea92be2d" alt="Image 4: Map of Elevation Zones."
data:image/s3,"s3://crabby-images/51769/51769080766d01519ef9050b8cb8ae76b47aff5c" alt="Image 5: 3D-Rendering of elevation zones."
Slope per Elevation Zone
Now, the slope average and histogram are calculated for each elevation zone.
data:image/s3,"s3://crabby-images/8d24a/8d24a704ae5d2b48b81788d2f311e3b413c45902" alt="Image 6: Slope Visualization and Histogram for"
data:image/s3,"s3://crabby-images/c316d/c316d6dd8d6ea21eea753aed7f7d603db26c9b1b" alt="Image 7: Slope Visualization and Histogram for 1000-1250m Elevation Zone."
data:image/s3,"s3://crabby-images/dee4d/dee4df8be269f47cd950d4be1381251dd8c7a49c" alt="Image 8: Slope Visualization and Histogram for 1250-1500m Elevation Zone."
data:image/s3,"s3://crabby-images/653c2/653c2e1dd51c2e2d8af266e7afd95ad2486dcd58" alt="Image 9: Slope Visualization and Histogram for 1500-1750m Elevation Zone."
data:image/s3,"s3://crabby-images/7046e/7046ed6a2b88735785f913b31708e660b27348fc" alt="Image 10: Slope Visualization and Histogram for >1750m Elevation Zone."
The average slope for the different elevation zones are as follows:
- <1000m: 19.3°
- 1000 – 1250m: 25.6°
- 1250 – 1500m: 32.3°
- 1500 – 1750m: 25.5
- > 1750m: 34.9
Interestingly, the slope does not always increase the higher up the mountain you go. Zone 4 (1500 – 1750m) has a lower average slope with 25.5°, most likely doe to the flatter area in the northern zone as well as the mountain ridge in the southern area of this zone.
This observation is supported if a profile graph is drawn across all elevation zones.
data:image/s3,"s3://crabby-images/78689/786899129544839090fb76ef4405f4af31bf3834" alt="Image 11: Profile Graph, background colorcoded to represent elevation zones."
data:image/s3,"s3://crabby-images/88a5c/88a5c1ac1ffaa3791b3fdfcb8401a6e824cff3a4" alt="Image 12: Location of profile graph line."
Resampling to 100x100m Slope Pixels
Now, the slope .tiff is resampled to a pixel size of 100 by 100 meters to see if the slope averages change depending on the spatial resolution of the underlaying data.
data:image/s3,"s3://crabby-images/d2386/d238689a871a82ea2878989506ff96f2092a1918" alt="slope_resampled Image 13: 100x100m Resampled Slope Visualization."
data:image/s3,"s3://crabby-images/99ced/99ced390836d1444f3d5842529b7fe57382c7ee7" alt="Image 14: Histogram of"
data:image/s3,"s3://crabby-images/d591c/d591c2565f9d75fb23ef6ada1dece69c471ab8f3" alt="Image 15: Histogram of 1000 - 1250m Elevation Zone, 100x100m pixel size."
data:image/s3,"s3://crabby-images/8a87b/8a87b6f02008f3a8f2f55687c4c8aaa248d58b08" alt="Image 16: Histogram of 1250 - 1500m Elevation Zone, 100x100m pixel size."
data:image/s3,"s3://crabby-images/23a3f/23a3fb1a1bbd2f153296fb2e6497defd369288f7" alt="Image 17: Histogram of 1500 - 1750m Elevation Zone, 100x100m pixel size."
data:image/s3,"s3://crabby-images/98b6d/98b6d49d85d3ec47ae0225753337fbbe56d6bb5a" alt="Image 18: Histogram of >1750m Elevation Zone, 100x100m pixel size."
data:image/s3,"s3://crabby-images/e5f09/e5f09f6e189c7a744617bdfbd195ecb11a09f4e9" alt="spatial_res_avg_slope_comp Table 1: Comparison of average slope per Elevation Zone and spatial resolution."
Comparing the average slopes of the different elevation zones and spatial resolutions shows that the average slope does not drastically change. As a resampling algorithm to downsample the images, the bilinear method was chosen. The low difference indicates that this method yielded good results. Surely, if another resampling method such as Nearest Neighbor would have been chosen, the result would have been much less acurate.
Curvature calculation without ArcGIS curvature function
Calculating the curvature of an area is very simple with the curvature function in ArcGIS Pro, but there is another easy workaround. Since we already have the slope layer, which indicates the slope of each pixel, we can run the slope function again on the same dataset.
After the first execution, the values show the slope of the pixel. Since the curvature is the second derivative of the slope (slope-of-the-sope), the result is similar to the curvature function.
The curvature function results in pixel values from ca. -4 to 4, while the slope of the slope is still in degrees. The degrees can be scaled to the same values, which results in the same output. Unfortunately, the visualization is different for the following two curvature representations, but the areas of conformity are still visible. Clearly, the two images show the same real-life phenomenon.
data:image/s3,"s3://crabby-images/9e786/9e786c44054aa30a0e17bce7e36f18d2b8c0484e" alt="Image 19: Curvature as calculated as a derivative of the slope."
data:image/s3,"s3://crabby-images/dc945/dc94535b88425ef73f6099ee7656b08786145692" alt="Image 20: Curvature as calculated by the ArcGIS Pro function."
Finally, as anounced, here is the code used to create all histograms in this post.
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
import numpy as np
import matplotlib.pyplot as plt
from PIL import Image
# Also open large images
Image.MAX_IMAGE_PIXELS = None
paths_slopes_resample=[
"slope_1000_resampled.tif",
"slope_1250_resampled.tif",
"slope_1500_resampled.tif",
"slope_1750_resampled.tif",
"slope_2050_resampled.tif",
]
paths_slopes=[
"slope_1000.tif",
"slope_1250.tif",
"slope_1500.tif",
"slope_1750.tif",
"slope_2050.tif",
]
path_slope_10 = ["slope_complete.tif"]
path_slope_100 = ["slope_complete_100.tif"]
# create clean title names from path
def extract_title(path):
name = path[:-4]
if "resample" in name:
name = name.replace("_resampled",", 100x100m")
else:
name = name+", 10x10m"
name = name.replace("_"," ")
name = name.replace("s","S")
name = name.replace("r","R")
name = name.replace("Slope","Slope in Degrees,")
name2 = name[:22]+"m Elevation Zone"+name[22:]
return(name2)
# Opening Image path, converting to numpy array, deleting nan values
def to_np(path):
im = Image.open(path)
im_array = np.array(im)
# round values to 2 decimal places
im_array = np.round(im_array, 2)
#Remove 0 (null) values
im_array[im_array == 0] = np.nan
# Remove nan values
im_array = im_array[~np.isnan(im_array)]
im_array = im_array[np.logical_not(np.isnan(im_array))]
#print((im_array))
return im_array
def graph(im_array,path,name):
# setting bns in 5 degree steps
bins=[0,5,10,15,20,25,30,35,40,45,50,55,60,65,70,75,80,85,90]
# draw histogram
n, bins, patches = plt.hist(im_array[~np.isnan(im_array)], bins=bins,color="royalblue", alpha=0.75, rwidth=0.85)
#set y grid
#plt.grid(axis='y', alpha=0.75)
# calculate mean
im_mean = im_array[~np.isnan(im_array)].mean()
# add mean line
plt.axvline(im_mean, color='red', linestyle='solid', linewidth=1)
# Set ticks, labels and title
plt.xticks(bins)
plt.xlabel("Slope (°)")
plt.ylabel('Frequency')
plt.title('Histogram\n'+name+"\nAverage Slope (°): "+str(round(im_mean,1)))
#set output name to sth computer readable and save
output_name = "histograms/"+path[:-4]+"_histogram.png"
plt.savefig(output_name,dpi=200)
print("saved: ",output_name)
# clear figure after saving
plt.clf()
plt.close("all")
for path in paths_slopes:
graph(to_np(path),path,extract_title(path))
for path in paths_slopes_resample:
graph(to_np(path),path,extract_title(path))
#Warning: Overflow in mean calculation possible
for path in path_slope_10:
graph(to_np(path),path,"Slope of full catchment area, 10x10m")
for path in path_slope_full:
graph(to_np(path),path,"Slope of full catchment area, 100x100m")