Monitoring and Mapping Wildfires Using Satellite Data
Contents
Monitoring and Mapping Wildfires Using Satellite Data#
Content creators: Brittany Engle, Diana Cadillo
Content reviewers: Yuhan Douglas Rao, Abigail Bodner, Jenna Pearson, Chi Zhang, Ohad Zivan
Content editors: Zane Mitrevica, Natalie Steinemann, Jenna Pearson, Chi Zhang, Ohad Zivan
Production editors: Wesley Banfield, Jenna Pearson, Chi Zhang, Ohad Zivan
Our 2023 Sponsors: NASA TOPS and Google DeepMind
# @title Project Background
from ipywidgets import widgets
from IPython.display import YouTubeVideo
from IPython.display import IFrame
from IPython.display import display
class PlayVideo(IFrame):
def __init__(self, id, source, page=1, width=400, height=300, **kwargs):
self.id = id
if source == "Bilibili":
src = f"https://player.bilibili.com/player.html?bvid={id}&page={page}"
elif source == "Osf":
src = f"https://mfr.ca-1.osf.io/render?url=https://osf.io/download/{id}/?direct%26mode=render"
super(PlayVideo, self).__init__(src, width, height, **kwargs)
def display_videos(video_ids, W=400, H=300, fs=1):
tab_contents = []
for i, video_id in enumerate(video_ids):
out = widgets.Output()
with out:
if video_ids[i][0] == "Youtube":
video = YouTubeVideo(
id=video_ids[i][1], width=W, height=H, fs=fs, rel=0
)
print(f"Video available at https://youtube.com/watch?v={video.id}")
else:
video = PlayVideo(
id=video_ids[i][1],
source=video_ids[i][0],
width=W,
height=H,
fs=fs,
autoplay=False,
)
if video_ids[i][0] == "Bilibili":
print(
f"Video available at https://www.bilibili.com/video/{video.id}"
)
elif video_ids[i][0] == "Osf":
print(f"Video available at https://osf.io/{video.id}")
display(video)
tab_contents.append(out)
return tab_contents
video_ids = [('Youtube', '2CDVn6O_q34'), ('Bilibili', 'BV1GP411k7XA')]
tab_contents = display_videos(video_ids, W=730, H=410)
tabs = widgets.Tab()
tabs.children = tab_contents
for i in range(len(tab_contents)):
tabs.set_title(i, video_ids[i][0])
display(tabs)
# @title Tutorial slides
# @markdown These are the slides for the videos in all tutorials today
from IPython.display import IFrame
link_id = "8w5da"
In this project, you will be working with wildfire remote sensing data from Sentinel 2 to extract Burn Perimeters using multiple Burn Indices and other relevant information related to wildfires. By integrating this data with information from other global databases, you will have the opportunity to explore the connections between wildfires and their impact on both the ecosystem and society.
Project Template#
Note: The dashed boxes are socio-economic questions.
Data Exploration Notebook#
Project Setup#
# installs
# !pip install gdal
# !pip install pandas
# !pip install geopandas
# imports
import random
import numpy
import matplotlib.pyplot as plt
import pandas as pd
import geopandas as gpd
import os
import matplotlib
/tmp/ipykernel_419586/2929139239.py:6: DeprecationWarning: Shapely 2.0 is installed, but because PyGEOS is also installed, GeoPandas still uses PyGEOS by default. However, starting with version 0.14, the default will switch to Shapely. To force to use Shapely 2.0 now, you can either uninstall PyGEOS or set the environment variable USE_PYGEOS=0. You can do this before starting the Python process, or in your code before importing geopandas:
import os
os.environ['USE_PYGEOS'] = '0'
import geopandas
In the next release, GeoPandas will switch to using Shapely by default, even if PyGEOS is installed. If you only have PyGEOS installed to get speed-ups, this switch should be smooth. However, if you are using PyGEOS directly (calling PyGEOS functions on geometries from GeoPandas), this will then stop working and you are encouraged to migrate from PyGEOS to Shapely 2.0 (https://shapely.readthedocs.io/en/latest/migration_pygeos.html).
import geopandas as gpd
Satellite Data#
Global Wildfire Information System (GWIS) is a global, joint initiative created by the GEO and the Copernicus Work Programs. The goal of this program is to bring together sources at a regional and international level to provide a global, comprehensive understanding of fire regimes and their effects.
The Globfire dataset uses the MODIS burned area product (MCD64A1) to determine the amount of burnt area per event (Artés et al., 2019). The MCD64A1 product that Globfire is built on top of combines imagery from the Terra and Aqua satellites with thermal anomalies to provide burnt area information (Website, MODIS C6 BA User Guide & User Guide).
Sentinel 2 Launched by the European Space Agency in June of 2015 (S2-A) and March of 2017 (S2-B), the Copernicus Sentinel-2 mission combines two polar-orbiting satellites to monitor variability in land surface conditions. Together, they have a global revisit time of roughly 5 days.
In the provided Project Template, we load the following datasets:
Global Wildfire Information Systems: Climate Action Large Wildfire Dataset#
The Climate Action Large Wildfire Dataset is a filtered version of the Globfire dataset (created by GWIS, which you can access here).
The resolution of this dataset is 500m. It has been pre-filtered to include only fires that are Class F or greater. Per the National Wildfire Coordinating Group, a Class F fire is defined as a fire that is 1,000 acres or greater, but less than 5,000 acres (note that a 2000m square region is roughly 1000 acres). For this dataset, all fires greater than 1,000 acres are included. Additional columns indicating area (acres), landcover number and landcover description, and country in which they are located within, were added. The landcover number and description were added using the Copernicus Global Land Cover Layers: CGLS-LC100 Collection 3 User Guide.
Table Information: ID: Globfire fire ID (unique to each fire)
IDate: Globfire Initial (start) date of the fire
FDate: Globfire Final (end) date of the fire
Continent: Location of the fire
Area_acres: size of fire in acres
Landcover: land cover value from ESA, if the landcover of the fire cover is greater than 51%, then it is labeled as that landcover
LC_descrip: land cover description from ESA
Country: country the fire is located in
# code to retrieve and load the data
url_climateaction = "~/shared/Data/Projects/Wildfires/ClimateAction_countries.shp"
Dataset = gpd.read_file(url_climateaction) # need to update to OSF and pooch.retrieve
---------------------------------------------------------------------------
CPLE_OpenFailedError Traceback (most recent call last)
File fiona/ogrext.pyx:136, in fiona.ogrext.gdal_open_vector()
File fiona/_err.pyx:291, in fiona._err.exc_wrap_pointer()
CPLE_OpenFailedError: /home/wesley/shared/Data/Projects/Wildfires/ClimateAction_countries.shp: No such file or directory
During handling of the above exception, another exception occurred:
DriverError Traceback (most recent call last)
Cell In[5], line 3
1 # code to retrieve and load the data
2 url_climateaction = "~/shared/Data/Projects/Wildfires/ClimateAction_countries.shp"
----> 3 Dataset = gpd.read_file(url_climateaction) # need to update to OSF and pooch.retrieve
File ~/miniconda3/envs/climatematch/lib/python3.10/site-packages/geopandas/io/file.py:281, in _read_file(filename, bbox, mask, rows, engine, **kwargs)
278 else:
279 path_or_bytes = filename
--> 281 return _read_file_fiona(
282 path_or_bytes, from_bytes, bbox=bbox, mask=mask, rows=rows, **kwargs
283 )
285 else:
286 raise ValueError(f"unknown engine '{engine}'")
File ~/miniconda3/envs/climatematch/lib/python3.10/site-packages/geopandas/io/file.py:322, in _read_file_fiona(path_or_bytes, from_bytes, bbox, mask, rows, where, **kwargs)
319 reader = fiona.open
321 with fiona_env():
--> 322 with reader(path_or_bytes, **kwargs) as features:
323 crs = features.crs_wkt
324 # attempt to get EPSG code
File ~/miniconda3/envs/climatematch/lib/python3.10/site-packages/fiona/env.py:457, in ensure_env_with_credentials.<locals>.wrapper(*args, **kwds)
454 session = DummySession()
456 with env_ctor(session=session):
--> 457 return f(*args, **kwds)
File ~/miniconda3/envs/climatematch/lib/python3.10/site-packages/fiona/__init__.py:292, in open(fp, mode, driver, schema, crs, encoding, layer, vfs, enabled_drivers, crs_wkt, allow_unsupported_drivers, **kwargs)
289 path = parse_path(fp)
291 if mode in ("a", "r"):
--> 292 colxn = Collection(
293 path,
294 mode,
295 driver=driver,
296 encoding=encoding,
297 layer=layer,
298 enabled_drivers=enabled_drivers,
299 allow_unsupported_drivers=allow_unsupported_drivers,
300 **kwargs
301 )
302 elif mode == "w":
303 colxn = Collection(
304 path,
305 mode,
(...)
314 **kwargs
315 )
File ~/miniconda3/envs/climatematch/lib/python3.10/site-packages/fiona/collection.py:243, in Collection.__init__(self, path, mode, driver, schema, crs, encoding, layer, vsi, archive, enabled_drivers, crs_wkt, ignore_fields, ignore_geometry, include_fields, wkt_version, allow_unsupported_drivers, **kwargs)
241 if self.mode == "r":
242 self.session = Session()
--> 243 self.session.start(self, **kwargs)
244 elif self.mode in ("a", "w"):
245 self.session = WritingSession()
File fiona/ogrext.pyx:588, in fiona.ogrext.Session.start()
File fiona/ogrext.pyx:143, in fiona.ogrext.gdal_open_vector()
DriverError: /home/wesley/shared/Data/Projects/Wildfires/ClimateAction_countries.shp: No such file or directory
We can now visualize the content of the dataset.
# code to print the shape, array names, etc of the dataset
Dataset.head()
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[6], line 2
1 # code to print the shape, array names, etc of the dataset
----> 2 Dataset.head()
NameError: name 'Dataset' is not defined
# plot the dataset
Dataset.plot()
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[7], line 2
1 # plot the dataset
----> 2 Dataset.plot()
NameError: name 'Dataset' is not defined
Great work! Now you are all set to address the questions you are interested in! Good luck!
Sentinel-2#
Each of the two satellites contain a Multi-Spectral Instrument (MSI) that houses 13 spectral bands, at various resolutions. The MSI uses a push-broom acquisition technique and measures in the Visible and Near Infrared (NIR) and the Short Wave Infrared (SWIR) domains. There are:
Four bands with 10m resolution: B2-Blue (490 nm), B3- Green (560 nm), B4- Red (665 nm) and Band 8- Near Infrared (NIR) (842 nm).
Six bands with 20m resolution: B5, B6, B7 and B8A (705 nm, 740 nm, 783 nm and 865 nm respectively) Vegetation Red Edge bands, along with B11 and B12 (1610 nm and 2190 nm) SWIR bands. These bands are mostly used for vegetation characterisation, vegetation stress assessment and ice/cloud/snow detection.
Three additional bands: B1 - Coastal Aerosol (443 nm), B9- Water Vapor (945 nm), and B10-SWIR-Cirrus (1375 nm) which are typically used for corrections of the atmosphere and clouds.
# view image
def showImage(Output):
plt.imshow(Output)
plt.show()
# data source-specific imports
# root folder location of where the imagery is currently saved
rootFolder = '/home/jovyan/shared/Data/Projects/Wildfires'
continet = "Asia"
# import pre images
# pre_fire_paths = glob.glob(rootFolder + continet + id +"/pre_fire_*.npy")
pre_fire_paths = [
fileName
for fileName in os.listdir(os.path.join(rootFolder, continet))
if (fileName.endswith(".npy") & fileName.startswith("pre_fire_"))
]
pre_fires_numpy = [
numpy.load(os.path.join(rootFolder, continet, x)).astype(int)
for x in pre_fire_paths
]
# import post images
post_fire_paths = [
fileName
for fileName in os.listdir(os.path.join(rootFolder, continet))
if (fileName.endswith(".npy") & fileName.startswith("post_fire_"))
]
post_fires_numpy = [
numpy.load(os.path.join(rootFolder, continet, x)).astype(int)
for x in post_fire_paths
]
# import Pre-SCL Mask for cloud coverage
scl_fire_paths = [
fileName
for fileName in os.listdir(os.path.join(rootFolder, continet))
if (fileName.endswith(".npy") & fileName.startswith("scl_pre_fire_"))
]
scl_fires_numpy = [
numpy.load(os.path.join(rootFolder, continet, x)) for x in scl_fire_paths
]
# import Post-SCL Mask for cloud coverage
scl_fires_post_paths = [
fileName
for fileName in os.listdir(os.path.join(rootFolder, continet))
if (fileName.endswith(".npy") & fileName.startswith("scl_post_fire_"))
]
scl_fires_post_numpy = [
numpy.load(os.path.join(rootFolder, continet, x)) for x in scl_fires_post_paths
]
---------------------------------------------------------------------------
FileNotFoundError Traceback (most recent call last)
Cell In[9], line 12
6 continet = "Asia"
8 # import pre images
9 # pre_fire_paths = glob.glob(rootFolder + continet + id +"/pre_fire_*.npy")
10 pre_fire_paths = [
11 fileName
---> 12 for fileName in os.listdir(os.path.join(rootFolder, continet))
13 if (fileName.endswith(".npy") & fileName.startswith("pre_fire_"))
14 ]
15 pre_fires_numpy = [
16 numpy.load(os.path.join(rootFolder, continet, x)).astype(int)
17 for x in pre_fire_paths
18 ]
20 # import post images
FileNotFoundError: [Errno 2] No such file or directory: '/home/jovyan/shared/Data/Projects/Wildfires/Asia'
# print list of pre_fires
print("\n".join(pre_fire_paths))
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[10], line 2
1 # print list of pre_fires
----> 2 print("\n".join(pre_fire_paths))
NameError: name 'pre_fire_paths' is not defined
# print list of post_fire
print("\n".join(post_fire_paths))
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[11], line 2
1 # print list of post_fire
----> 2 print("\n".join(post_fire_paths))
NameError: name 'post_fire_paths' is not defined
# print list of SCL
print("\n".join(scl_fire_paths))
print("\n".join(scl_fires_post_paths))
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[12], line 2
1 # print list of SCL
----> 2 print("\n".join(scl_fire_paths))
3 print("\n".join(scl_fires_post_paths))
NameError: name 'scl_fire_paths' is not defined
# view pre-fire satellite image that was taken right before the fire start date
showImage(
numpy.clip(pre_fires_numpy[2][:, :, [3, 2, 1]] / 10000 * 3.5, 0, 1)
) # RGB bands for Sentinel 2 are Bands: 4,3,2
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[13], line 3
1 # view pre-fire satellite image that was taken right before the fire start date
2 showImage(
----> 3 numpy.clip(pre_fires_numpy[2][:, :, [3, 2, 1]] / 10000 * 3.5, 0, 1)
4 ) # RGB bands for Sentinel 2 are Bands: 4,3,2
NameError: name 'pre_fires_numpy' is not defined
# view post-fire satellite image that was taken right before the fire start date
showImage(
numpy.clip(post_fires_numpy[0][:, :, [3, 2, 1]] / 10000 * 3.5, 0, 1)
) # RGB bands for Sentinel 2 are Bands: 4,3,2
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[14], line 3
1 # view post-fire satellite image that was taken right before the fire start date
2 showImage(
----> 3 numpy.clip(post_fires_numpy[0][:, :, [3, 2, 1]] / 10000 * 3.5, 0, 1)
4 ) # RGB bands for Sentinel 2 are Bands: 4,3,2
NameError: name 'post_fires_numpy' is not defined
Sentinel-2: Vegetation Health & Burnt Areas#
Sentinel-2 Imagery - Vegetation Health & Burnt Areas
Continents included:
Asia
Africa
Austrailia
Europe
North America
South America
Vegetation Health: Vegetation indices define and monitor the health of vegetation using the radiance values of the visible and near-infrared channels. The Normalized Difference Vegetation Index (NDVI) is used to measure “greenness” of vegetation. As one of the most widely used vegetation indexes, the NDVI takes advantage of how strongly Chlorophyll absorbs visible light and how well the leafs cellular structure reflects near-infrared. Its values range from +1.0 to -1.0, with areas of sand, rock, and snow typically having a value of 0.1 or less. Shrubs and spare vegetation are roughly 0.2 to 0.5, while higher NDVI values (0.6 to 0.9) indicate dense vegetation that is typically found in tropical forests.
The NDVI can also be used to create the Vegetation Condition Index (VCI). The VCI depends on the current NDVI along with the extreme NDVI values within the dataset (NDVImax and NDVImin). Specifically, $\(VCI = \frac{NDVI-NDVImin}{NDVImax-NDVImin}\times 100\%\)$
This number is then compared to the threshold to determine the drought severity. For this project, up to 3 months of pre-fire imagery will be used to determine current drought conditions.
Burnt Area Severity: Burn severity describes how the intensity of the fire affects the functioning of the ecosystem in which it occurred. The degree to which it alters the ecosystem is typically found using a burn index, which then (typically) classes the severity of the fire as: unburned, low severity, moderate severity, or high severity. One of the most common burn indices is the Normalized Burn Ratio (NBR). This index is designed to highlight burnt areas in fire zones. The formula is similar to NDVI, except that the formula combines the use of both near infrared (NIR) and shortwave infrared (SWIR) wavelengths. Healthy vegetation shows a very high reflectance in the NIR, and low reflectance in the SWIR portion of the spectrum - the opposite of what is seen in areas devastated by fire. Recently burnt areas demonstrate low reflectance in the NIR and high reflectance in the SWIR. The difference between the spectral responses of healthy vegetation and burnt areas reach their peak in the NIR and the SWIR regions of the spectrum. The difference between normalized burn ratios before and after a fire is called the dNBR, and is one of many useful indices.
Specifically, the dNBR isolates the burned areas from the unburned areas, and subtracts the pre-fire imagery from the post-fire imagery. This removes any unchanged, and thus unburned, pixels as their values result in near zero. The results of the dNBR are based on burn severity, and correspond to the gradient of burn severity for every pixel. The dNBR has an unscaled range of -2.0 to +2.0 with burned areas tending to show more positively.
SCL Mask Importing the SCL in order to mask out clouds form imagery
# compute SCL Mask to 0s and 1s, masking out clouds and bad pixels
def computeSCLMask(image):
rImage, cImage = image.shape
sclOutput = numpy.zeros((rImage, cImage))
for x in range(cImage):
for y in range(rImage):
sclOutput[y, x] = 1 if image[y, x] in [0, 1, 3, 8, 9, 11] else 0
return sclOutput
# create Pre-fire and post-fire SCL masks
print(scl_fires_numpy[5])
pre_SCL_Mask = computeSCLMask(scl_fires_numpy[5])
post_SCL_Mask = computeSCLMask(scl_fires_post_numpy[0])
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[16], line 2
1 # create Pre-fire and post-fire SCL masks
----> 2 print(scl_fires_numpy[5])
3 pre_SCL_Mask = computeSCLMask(scl_fires_numpy[5])
4 post_SCL_Mask = computeSCLMask(scl_fires_post_numpy[0])
NameError: name 'scl_fires_numpy' is not defined
# view SCL imagery for image closests to the start fire date
# ------- Save SCL as colored image based on SCL classification
# No Data (0) = black
# Saturated / Defective (1) = red
# Dark Area Pixels (2) = chocolate
# Cloud Shadows (3) = brown
# Vegetation (4) = lime
# Bare Soils (5) = yellow
# Water (6) = blue
# Clouds low probability / Unclassified (7) = aqua
# clouds medium probability (8) = darkgrey
# Clouds high probability (9) light grey
# Cirrus (10) = deepskyblue
# Snow / Ice (11) = magenta
# colors: https://matplotlib.org/3.1.1/gallery/color/named_colors.html#sphx-glr-gallery-color-named-colors-py
def showSCL(image):
cmap = matplotlib.colors.ListedColormap(
[
"black",
"red",
"chocolate",
"brown",
"lime",
"yellow",
"blue",
"aqua",
"darkgrey",
"lightgrey",
"deepskyblue",
"magenta",
]
)
plt.imshow(image, cmap=cmap)
plt.show()
showSCL(scl_fires_numpy[5])
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[17], line 40
36 plt.imshow(image, cmap=cmap)
37 plt.show()
---> 40 showSCL(scl_fires_numpy[5])
NameError: name 'scl_fires_numpy' is not defined
Dataset-specific Functions: NDVI#
# this code computes the NDVI for an image
"The NDVI can be computed on any image (pre or post)."
"Compute the NDVI on the pre-fire image"
# compute NDVI
def computeNDVI(image, mask):
r, c, ch = image.shape
ndviOutput = numpy.zeros((r, c))
for x in range(c):
for y in range(r):
if (image[y, x, 7] == 0 and image[y, x, 3] == 0) or mask[y, x] == 1:
ndviOutput[y, x] = numpy.nan
else:
ndviOutput[y, x] = (image[y, x, 7] - image[y, x, 3]) / (
image[y, x, 7] + image[y, x, 3]
)
return ndviOutput
# TA Code
computeNDVI_value = computeNDVI(pre_fires_numpy[2], pre_SCL_Mask)
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[19], line 2
1 # TA Code
----> 2 computeNDVI_value = computeNDVI(pre_fires_numpy[2], pre_SCL_Mask)
NameError: name 'pre_fires_numpy' is not defined
# plot NDVI without remap
showImage(computeNDVI_value)
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[20], line 2
1 # plot NDVI without remap
----> 2 showImage(computeNDVI_value)
NameError: name 'computeNDVI_value' is not defined
# use this code to remap the NDVI to specific colors for values
def remapNDVI(NDVI):
remapped = numpy.zeros((NDVI.shape[0], NDVI.shape[1]))
for x in range(remapped.shape[0]):
for y in range(remapped.shape[1]):
if numpy.isnan(NDVI[x, y]):
remapped[x, y] = numpy.nan
elif NDVI[x, y] <= -0.2:
remapped[x, y] = 1
elif NDVI[x, y] <= 0:
remapped[x, y] = 2
elif NDVI[x, y] <= 0.1:
remapped[x, y] = 3
elif NDVI[x, y] <= 0.2:
remapped[x, y] = 4
elif NDVI[x, y] <= 0.3:
remapped[x, y] = 5
elif NDVI[x, y] <= 0.4:
remapped[x, y] = 6
elif NDVI[x, y] <= 0.5:
remapped[x, y] = 7
elif NDVI[x, y] <= 0.6:
remapped[x, y] = 8
elif NDVI[x, y] <= 0.7:
remapped[x, y] = 9
elif NDVI[x, y] <= 0.8:
remapped[x, y] = 10
elif NDVI[x, y] <= 0.9:
remapped[x, y] = 11
elif NDVI[x, y] <= 1:
remapped[x, y] = 12
else:
remapped[x, y] = 13
return remapped
# TA Code
NDVI_remap = remapNDVI(computeNDVI_value)
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[22], line 2
1 # TA Code
----> 2 NDVI_remap = remapNDVI(computeNDVI_value)
NameError: name 'computeNDVI_value' is not defined
# view remapped NDVI
def showNDVI(image):
cmap = matplotlib.colors.ListedColormap(
[
"#000000",
"#a50026",
"#d73027",
"#f46d43",
"#fdae61",
"#fee08b",
"#ffffbf",
"#d9ef8b",
"#a6d96a",
"#66bd63",
"#1a9850",
"#006837",
]
)
plt.imshow(image, cmap=cmap)
plt.show()
showNDVI(NDVI_remap)
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[23], line 23
19 plt.imshow(image, cmap=cmap)
20 plt.show()
---> 23 showNDVI(NDVI_remap)
NameError: name 'NDVI_remap' is not defined
Dataset-specific Functions: VCI#
# compute the SCL mask for all the SCLs and apply it to the pre_fire_NDVIs
pre_fires_scl = [computeSCLMask(x) for x in scl_fires_numpy]
pre_fires_NDVI = [computeNDVI(x[0], x[1]) for x in zip(pre_fires_numpy, pre_fires_scl)]
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[24], line 2
1 # compute the SCL mask for all the SCLs and apply it to the pre_fire_NDVIs
----> 2 pre_fires_scl = [computeSCLMask(x) for x in scl_fires_numpy]
3 pre_fires_NDVI = [computeNDVI(x[0], x[1]) for x in zip(pre_fires_numpy, pre_fires_scl)]
NameError: name 'scl_fires_numpy' is not defined
# compute for VCI
def computeVCI(for_ndvi_image, ndvi_dataset):
rImage, cImage = for_ndvi_image.shape
vciOutput = numpy.zeros((rImage, cImage))
ndvi_dataset.append(for_ndvi_image)
for x in range(cImage):
for y in range(rImage):
pixels = [z[y, x] for z in ndvi_dataset]
filtered_pixels = [f for f in pixels if not numpy.isnan(f)]
if len(filtered_pixels) == 0 or len(filtered_pixels) == 1:
vciOutput[y, x] = 1
elif numpy.isnan(for_ndvi_image[y, x]):
vciOutput[y, x] = 1
else:
max_val = max(filtered_pixels)
min_val = min(filtered_pixels)
if max_val == min_val:
vciOutput[y, x] = 1
else:
vciOutput[y, x] = (for_ndvi_image[y, x] - min_val) / (
max_val - min_val
)
return vciOutput
VCI Drought Threshold#
image and more information: Spatial and Temporal Variation of Drought Based onSatellite Derived Vegetation Condition Index in Nepal from 1982–2015
![droughtthreshold.png]()
# compute the VCI for the last pre-fire to view the drought over the time period
last_pre_fire_NDVI = pre_fires_NDVI.pop(2)
last_pre_fire_vci = computeVCI(last_pre_fire_NDVI, pre_fires_NDVI)
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[26], line 2
1 # compute the VCI for the last pre-fire to view the drought over the time period
----> 2 last_pre_fire_NDVI = pre_fires_NDVI.pop(2)
3 last_pre_fire_vci = computeVCI(last_pre_fire_NDVI, pre_fires_NDVI)
NameError: name 'pre_fires_NDVI' is not defined
# view the non-thresholded VCI
showImage(last_pre_fire_vci)
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[27], line 2
1 # view the non-thresholded VCI
----> 2 showImage(last_pre_fire_vci)
NameError: name 'last_pre_fire_vci' is not defined
# map specific colors to values to show the severity of the droughts
def remapVCI(vci):
remapped = numpy.zeros(vci.shape)
for x in range(remapped.shape[0]):
for y in range(remapped.shape[1]):
if vci[x, y] < 0.35:
remapped[x, y] = 1
elif vci[x, y] <= 0.50:
remapped[x, y] = 2
else:
remapped[x, y] = 3
return remapped
# define the VCI mapped/thresholded values
def showVCI(vci_image):
cmap = matplotlib.colors.ListedColormap(["red", "yellow", "green"])
plt.imshow(remapVCI(vci_image), cmap=cmap)
plt.show()
# view the mapped VCI values
showVCI(last_pre_fire_vci)
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[30], line 2
1 # view the mapped VCI values
----> 2 showVCI(last_pre_fire_vci)
NameError: name 'last_pre_fire_vci' is not defined
Dataset-specific Functions: NBR and dNBR Code#
# this code creates the NBR for each image then uses the NBR to create the dNBR. It can easily be updated for other burnt area indices
def computeFireMasks(pre_fire, post_fire):
rows, columns, channels = pre_fire.shape
nbrPost = numpy.zeros((rows, columns))
nbrPre = numpy.zeros((rows, columns))
dnbr = numpy.zeros((rows, columns))
for x in range(columns):
for y in range(rows):
nbrPost[y, x] = (post_fire[y, x, 7] - post_fire[y, x, 11]) / (
post_fire[y, x, 7] + post_fire[y, x, 11]
)
nbrPre[y, x] = (pre_fire[y, x, 7] - pre_fire[y, x, 11]) / (
pre_fire[y, x, 7] + pre_fire[y, x, 11]
)
dnbr[y, x] = nbrPre[y, x] - nbrPost[y, x]
return dnbr
# TA Code
# run computeFireMasks
dnbr = computeFireMasks(pre_fires_numpy[2], post_fires_numpy[0])
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[32], line 3
1 # TA Code
2 # run computeFireMasks
----> 3 dnbr = computeFireMasks(pre_fires_numpy[2], post_fires_numpy[0])
NameError: name 'pre_fires_numpy' is not defined
# this code applies a threshold to the dNBR to show the level of burn intensity (unburned, low severity, moderate severity, or high severity)
def remapDNBR(dnbr):
remapped = numpy.zeros((dnbr.shape[0], dnbr.shape[1]))
for x in range(remapped.shape[0]):
for y in range(remapped.shape[1]):
if numpy.isnan(dnbr[x, y]):
remapped[x, y] = numpy.nan
elif dnbr[x, y] <= -0.251:
remapped[x, y] = 1
elif dnbr[x, y] <= -0.101:
remapped[x, y] = 2
elif dnbr[x, y] <= 0.099:
remapped[x, y] = 3
elif dnbr[x, y] <= 0.269:
remapped[x, y] = 4
elif dnbr[x, y] <= 0.439:
remapped[x, y] = 5
elif dnbr[x, y] <= 0.659:
remapped[x, y] = 6
elif dnbr[x, y] <= 1.3:
remapped[x, y] = 7
else:
remapped[x, y] = 8
return remapped
# TA Code
dnbr_remapped = remapDNBR(dnbr)
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[34], line 2
1 # TA Code
----> 2 dnbr_remapped = remapDNBR(dnbr)
NameError: name 'dnbr' is not defined
dNBR Threshold#
image and more information: Normalized Burn Ratio (NBR) UN-SPIDER Knowledge Portal
![threshold.png]()
# this code takes the above function (remapDNBR) where the dNBR threshold has been applied to the image'
# and applies a color coded map to each threshold as shown in the image above'
def showDNBR(dnbr):
cmap = matplotlib.colors.ListedColormap(
[
"blue",
"teal",
"green",
"yellow",
"orange",
"red",
"purple",
]
)
plt.imshow(remapDNBR(dnbr), cmap=cmap)
showDNBR(dnbr_remapped)
---------------------------------------------------------------------------
NameError Traceback (most recent call last)
Cell In[35], line 20
6 cmap = matplotlib.colors.ListedColormap(
7 [
8 "blue",
(...)
15 ]
16 )
17 plt.imshow(remapDNBR(dnbr), cmap=cmap)
---> 20 showDNBR(dnbr_remapped)
NameError: name 'dnbr_remapped' is not defined
Further Reading#
A Project for Monitoring Trends in Burn Severity | Fire Ecology | Full Text
The unequal vulnerability of communities of color to wildfire | PLOS ONE
Fire Frequency, Area Burned, and Severity: A Quantitative Approach to Defining a Normal Fire Year
Science, technology, and human factors in fire danger rating: the Canadian experience
Socio-economic effects of wildfires
Zhao, J. et al. (2020). Quantifying the Effects of Vegetation Restorations on the Soil Erosion Export and Nutrient Loss on the Loess Plateau. Front. Plant Sci. 11 https://doi.org/10.3389/fpls.2020.573126
Amanda K. Hohner, Charles C. Rhoades, Paul Wilkerson, Fernando L. Rosario-Ortiz (2019). Wildfires alter forest watersheds and threaten drinking water quality. Accounts of Chemical Research. 52: 1234-1244. https://doi.org/10.1021/acs.accounts.8b00670
Alan Buis (2021). The Climate Connections of a Record Fire Year in the U.S. West. Link
Ian P. Davies ,Ryan D. Haugo,James C. Robertson,Phillip S. Levin (2018). The unequal vulnerability of communities of color to wildfire.PLoS ONE 13(11): e0205825. https://doi.org/10.1371/journal.pone.0205825
Wildfires and forest ecosystems