for Dummies

Part II

Earth Engine

Michael A Menarguez

mamenarguez@ou.edu

Contents

  1. GEE Objects and tools (basics)

  2. Sample application exercise for tornado tracking

GEE Objects and tools  (Part 1)

  1. ee.Map
  2. ee.Image
  3. ee.ImageCollection
  4. ee.Geometry
  5. ee.Feature
  6. ee.FeatureCollection
  7. Two exercises (Tornado track, MODIS paddy rice algorithm part 1)

GEE Objects and tools

ee.Map

  1. Object that can display Image / Features / Geometries

//Prints the viewport of the map
print(Map.getBounds())
//Center map at lon,lat,zoom level
Map.setCenter(-97.4390,35.2058,17)
// Create a rectangle, center it and display it on map
var rectangle = ee.Geometry.Rectangle(-97.4438, 35.2048,-97.4409, 35.2070)
Map.centerObject(rectangle)
Map.addLayer(rectangle)

GEE Objects and tools

Image (I)

  1. Used to show result in map or to download results
  2. Can have one or many bands
  3. An image can cover the whole world!

 

var myImage = ee.Image(0); // Image with zeros on the whole world
Map.addLayer(myImage,'dumb layer')
var img1 = ee.Image('MOD09GA/MOD09GA_005_2012_03_09'); // From name
var img2 = ee.Image(1) //Constant value
var img3 = ee.Image() //Empty transparent Image
print (img1)
Map.addLayer(img2,{},'equal_to_1');
Map.addLayer(img1.select(['sur_refl_b01','sur_refl_b04','sur_refl_b03'])
            ,{gain:"0.0255,0.0255,0.0255"},
            'MOD09GA');
Map.addLayer(img3,'transparent');

Constructors (just several)

GEE Objects and tools

Image (II)

Some OPERATIONS

Algebraic: add, Subtract, Multiply, Divide

Trigonometric: sin, cos, tan, asin, acos, atan

Binary: eq, gt, gte, lte, and, or, not, leftShift, rightShift

Type: toInt16, toInt32, toLong, toString, toFloat, ...

Other: where, mask, normalizeddifference

Much more in the documentation!

GEE Objects and tools

Image (III)

Example: Load two MOD09A1 world images, calculate ndvi and show their difference in map

var image1 = ee.Image('MODIS/MOD09A1/MOD09A1_005_2010_02_02') //Load image 1
var image2 = ee.Image('MODIS/MOD09A1/MOD09A1_005_2011_02_10') //load image 2
var ndvi1 = image1.normalizedDifference(['sur_refl_b01','sur_refl_b02']) //Shortcut
var ndvi2 = image2.expression('(RED-NIR)/(RED+NIR)',{ //Alternative way
  RED:image2.select(['sur_refl_b01']),
  NIR:image2.select(['sur_refl_b02'])
  });
var ndviDifference = ndvi2.subtract(ndvi1) //Calculate their difference
//This palette will help giving colors to visualization
var palette = ['FFFFFF', 'CE7E45', 'DF923D', 'F1B555', 'FCD163', '99B718',
               '74A901', '66A000', '529400', '3E8601', '207401', '056201',
               '004C00', '023B01', '012E01', '011D01', '011301'];
Map.addLayer(ndviDifference,{min:-1,max:.1,palette:palette})

GEE Objects and tools

Image (IV)

var image = ee.Image('USGS/SRTMGL1_003') //SRTM Digital Elevation Data 30m

var altNaiveClass = ee.Image(0) //Start with an image of zeros

altNaiveClass = altNaiveClass.where(image.lt(500),1) //1: <500m
altNaiveClass = altNaiveClass.where(image.gte(500).and(image.lt(1000)),2) //2: [500m-100m)
altNaiveClass = altNaiveClass.where(image.gte(1000),3) // 3: >=1000m

altNaiveClass=altNaiveClass.mask(altNaiveClass) //Removes all Zeros

Map.addLayer(altNaiveClass)

Example: Load DEM and get image with three classes based on its height

GEE Objects and tools

ee.ImageCollection (I)

  1. Used to organize sets of images
  2. Useful if you want to filter a collection by metadata (date, path and row, ...)
  3. Can map a function over every single image separately (vegetation index, filtering... )

In order to display in map, you need to convert the ImageCollection a single Image! (mean, median, max, ...)

GEE Objects and tools

ee.ImageCollection (II)

Some OPERATIONS

Statistical aggregation: min, max, mean, median, mode, count

Binary:  and, or, not

Algebraic Aggregation: Sum, product, ...

Other: merge, map, reduce, filterDate, filterBounds, cast, ...

Much more in the documentation!

GEE Objects and tools

ee.ImageCollection (III)

var myCollection = ee.ImageCollection('MODIS/MOD11A2')
                    .filterDate(new Date(2000,1,1),new Date(2001,1,1))
var medianLST = myCollection.select(['LST_Night_1km'])
      .median() // Select median value, now we have an ee.Image object
      .toFloat() // Transform to float
      .multiply(ee.Image(0.02)) //this band has a 0.02 scale
      .subtract(ee.Image(273.15)) //Transform Kelvin to Celsius
print(myCollection)  //Just for debugging
Map.addLayer(medianLST)

Example: load all MOD11A2 images from 2000 to 2001 and show the median value in map in Celsius

GEE Objects and tools

ee.ImageCollection (IV)

Mapping functions

Example: Get NDVI from the collection MOD09A1 (apply to all images) and extract the maximum ndvi for 2008 for each pixel 

function getNDVI (image){
  return image.normalizedDifference(['sur_refl_b02','sur_refl_b01'])
              .select([0],['ndvi']) //Give name to the band and return its value
}

var year = 2008;
var collection = ee.ImageCollection("MODIS/MOD09A1")
  .filterDate(new Date(year,1,1),new Date(year+1,1,1))
print(collection)
collection = collection.map(getNDVI);
print(collection)
var maxNDVI = collection.max()
print(maxNDVI)
Map.addLayer(maxNDVI)

GEE Objects and tools

ee.ImageCollection (V)

Mapping functions with extra parameters

Example: Get NDVI from the collection MOD09A1 and filter observations out of the range [-0.2,1.0] , then return the min

function getNdviFromThreshold(min,max){
  return function(image){
    image =  image.normalizedDifference(['sur_refl_b02','sur_refl_b01'])
    image =  image.where(image.gt(ee.Image(max)),max) //Put zeros in values out of range
    image =  image.where(image.lt(ee.Image(min)),min) //Put zeros in values out of range
    return  image.select([0],['ndvi']) //Give name to the band and return its value
  }
}
var year = 2008, min = -0.2, max = 1;
var collection = ee.ImageCollection("MODIS/MOD09A1")
  .filterDate(new Date(year,1,1),new Date(year+1,1,1));
print(collection);
collection = collection.map(getNdviFromThreshold(min,max));
print(collection);
var maxNDVI = collection.max();
print(maxNDVI);
Map.addLayer(maxNDVI,{max:max,min:min});

GEE Objects and tools

ee.Geometry (I)

Generalization of a geometric shape

Types:

  1. Point / MultiPoint
  2. Polygon, Multipolygon
  3. Rectangle ( useful shortcut)
  4. from geoJSON
  5. LinearRing

GEE Objects and tools

ee.Geometry (II)

Example: From landsat 7, calculate ndvi on norman and crop resulting image 

var image = ee.Image('LEDAPS/LE7_L1T_SR/LE70270362000024EDC00') //Load image 1
var ndvi = image.normalizedDifference(['B4','B3'])
var rectangle = ee.Geometry(ee.Geometry.Rectangle(-97.4982, 35.1752,-97.3859,35.2624),'EPSG:4326',false)
ndvi = ndvi.clip(rectangle)
var palette = ['FFFFFF', 'CE7E45', 'DF923D', 'F1B555', 'FCD163', '99B718',
              '74A901', '66A000', '529400', '3E8601', '207401', '056201',
              '004C00', '023B01', '012E01', '011D01', '011301'];
Map.addLayer(ndvi,{min:0,max:.1,palette:palette},'ndvi in the polygon')

Example: How i got the Image name?

var rectangle = ee.Geometry(ee.Geometry.Rectangle(-97.4982, 35.1752,-97.3859,35.2624),'EPSG:4326',false)
var imCol = ee.ImageCollection('LEDAPS/LE7_L1T_SR')
          .filterDate('2000-01-01','2001-01-01') // If we do not filter, this takes a long time and resources
          .filterBounds(rectangle) // Need to filter the region we want data too for the same reason
print(imCol)

Prevents Geodesic approximation

GEE Objects and tools

ee.Feature (I)

Geometry associated with an Object (null, Number, List, ...)

Example: Get value of pixels for NLCD water / ice pixels

//Sample layer NLCD
var a = ee.Image('USGS/NLCD/NLCD2001')
//Select only water and snow/ice pixels (11 or 12)
var b = ee.Image(0).where(
        a.select('landcover').eq(11).or(a.select('landcover').eq(12)),
        1
)
Map.addLayer(b.mask(b),{palette:'0000FF'})

var point1 = ee.Geometry.Point([-87.863438, 43.393651])
var point2 = ee.Geometry.Point([-87.8637768, 43.3936994])
Map.addLayer(point1,{color:"00ff00"},'Point1')
Map.addLayer(point2,{},'Point2')
Map.centerObject(point1)
print('Point1 (water)');print(b.sample(point1,30))
print('Point2 (no_water)');print(b.sample(point2,30))

GEE Objects and tools

ee.FeatureCollection(I)

Group of features

//Sample layer NLCD
var a = ee.Image('USGS/NLCD/NLCD2001')
//Select only water and snow/ice pixels (11 or 12)
var b = ee.Image(0).where(
        a.select('landcover').eq(11).or(a.select('landcover').eq(12)),
        1
)
Map.addLayer(b.mask(b),{palette:'0000FF'})
var point1 = ee.Geometry.Point([-87.863438, 43.393651])
var point2 = ee.Geometry.Point([-87.8637768, 43.3936994])
Map.addLayer(point1,{color:"00ff00"},'Point1')
Map.addLayer(point2,{},'Point2')
Map.centerObject(point1)
print('Point1 (water)');print(b.sample(point1,30))
print('Point2 (no_water)');print(b.sample(point2,30))
var featureColelction = ee.FeatureCollection([b.sample(point1,30),b.sample(point2,30)])
print(featureColelction)

GEE Objects and tools

ee.FeatureCollection(II)

How to load your own kml, txt, csv, tsv ?

Sol: Upload to a fusion table! Click for More info

  1. Prepare your data (csv, kml, kmz, ...)
  2. Click on create new fusion table
  3. Upload your fusion table
  4. Click on share (top right corner)
  5. Change who can view to public
  6. Copy the text of the link after docid=
  7. Import to GEE workspace using:

"ft:docid"

var fusionTableFC = ee.FeatureCollection("ft:1edO826LFvRG_H8A4_TXg6E6pbDtSHei8b5ycDoUK")
print (fusionTableFC)
Map.centerObject(fusionTableFC);
Map.addLayer(fusionTableFC)

Exercises

It is your turn!

Problem: Pepito wants to track June 1st 2011 New England's devastating tornado effects. In order to do so, he is planning to use GEE and changes in vegetation greenness (NDVI).

He thinks that landsat 5 can be used for this task. Code a solution to this problem. There is no need to filter cloud artifacts.

 

Rectangle coords = [-72.46,42.042,-72.037,42.176]

 

Tips: Use previous year data from October to compare

 

Landsat 5 ndvi median composite from 2010-08-01 to 2010-11-01

Landsat 5 ndvi median composite from 2011-06-01 to 2011-06-30

Landsat 5 ndvi median composite difference from images above. The tornado devastating is marked in red (NDVI diff < - 0.2)

var tornadoRectangle =  ee.Geometry(ee.Geometry.Rectangle(-72.460412,42.041943,-72.03708,42.175940),'EPSG:4326' )

var collectionBefore = ee.ImageCollection("LEDAPS/LT5_L1T_SR")
  .filterDate('2010-08-01','2010-11-01')
  .filterBounds(tornadoRectangle)
  .median().normalizedDifference(['B4','B3'])

var collectionAfter = ee.ImageCollection("LEDAPS/LT5_L1T_SR")
  .filterDate('2011-06-01','2011-06-30')
  .filterBounds(tornadoRectangle)
  .median().normalizedDifference(['B4','B3'])

var palette = ['FFFFFF', 'CE7E45', 'DF923D', 'F1B555', 'FCD163', '99B718',
              '74A901', '66A000', '529400', '3E8601', '207401', '056201',
              '004C00', '023B01', '012E01', '011D01', '011301'];
              
var composite1 = collectionBefore.clip(tornadoRectangle);
var composite2 = collectionAfter.clip(tornadoRectangle);
var difference  = collectionAfter.subtract(collectionBefore).clip(tornadoRectangle);
var tornadoTrail = ee.Image(0).where(difference.lte(-0.2),1)
tornadoTrail = tornadoTrail.mask(difference.lte(-0.2))

Map.addLayer(composite1,{palette:palette},'before'); 
Map.addLayer(composite2,{palette:palette},'after'); 
Map.addLayer(difference,{palette:palette},'difference'); 
Map.addLayer(tornadoTrail,{palette:'FF0000'},'tornado trail')

Map.centerObject(tornadoRectangle)

Exercises (II)

Problem: MODIS Paddy rice mapping algorithm (2015 Zhang et. al) requires different masks and products derived from MOD09A1 8-days surface reflectance and other MOD11A2 Land Surface Temperate.

 

  1. Get 8-days NDVI, EVI, and LSWI water indices (MOD09A1)
  2. Get 8-days LST in Celsius (MOD11A2)
  3. For the four products, visualize them in the map
//################## Aux. functions #############################
//This function receives a MOD09A1 image and returns ndvi, evi, and lswi
function getVegetationIndices(img){
  var ndvi = img.normalizedDifference(['sur_refl_b02','sur_refl_b01']) 
  var lswi = img.normalizedDifference(['sur_refl_b04','sur_refl_b02']) 
  var evi = img.expression(
      '2.5 * (nir - red) / (nir + 6 * red - 7.5 * blue + 1)',
      {
          red: img.select('sur_refl_b01'),    // 620-670nm, RED
          nir: img.select('sur_refl_b02'),    // 841-876nm, NIR
          blue: img.select('sur_refl_b03')    // 459-479nm, BLUE
      });
  return img.select([]) // Implementation improvement. Need to return same image
              .addBands([ndvi,evi,lswi]) // Add bands to image
              .select([0,1,2],['ndvi','evi','lswi']) //Rename bands
}
//################## Script starts here #############################
var year = 2014 //Easy way to change year in the script
var modisCollection = ee.ImageCollection('MODIS/MOD09A1') //Load MOD09A1 Collecion
                        .filterDate(year+'-01-01',(year+1)+'-01-01') //Filter by year
print('Modis collection');print(modisCollection);
var modisViCollection = modisCollection.map(getVegetationIndices)
print('Modis VI collection');print(modisViCollection);

// Visual representation using median values for each vegetation index
var palette = ['FFFFFF', 'CE7E45', 'DF923D', 'F1B555', 'FCD163', '99B718',
              '74A901', '66A000', '529400', '3E8601', '207401', '056201',
              '004C00', '023B01', '012E01', '011D01', '011301'];
Map.addLayer(modisViCollection.select('ndvi').median(),{palette:palette},'ndvi')
Map.addLayer(modisViCollection.select('evi').median(),{palette:palette},'evi')
Map.addLayer(modisViCollection.select('lswi').median(),{palette:palette},'lswi')
// ################## Aux. functions #############################
//This function returns an image containing the celsius value from a MOD11A2 collection
function convertToCelsius(image){
    var propertyNames = image.propertyNames()
    return image.toFloat()
        // Re-scale (https://lpdaac.usgs.gov/dataset_discovery/modis/modis_products_table/mod11a2)
                .multiply(0.02)   
        // Convert Kelvin to Celsius
                .subtract(273.15) 
        // Copy properties from original image (timestart, ...)
                .copyProperties(image,propertyNames) 
}

// ################## Script starts here #############################
var year = 2014 //Easy way to change year in the script
var collection =  ee.ImageCollection('MODIS/MOD11A2')
                    .filterDate(year+'-01-01',(year+1)+'01-01') //Filter by year
                    .select(['LST_Night_1km']) //Select only the band we want
                    .sort('system:time_start') //Not necessary as of now

var collectionCelsius = collection.map(convertToCelsius) //Convert Collection to celsius
print (collectionCelsius)
// Visual representation using median values
var palette = ['011301', '011D01', '012E01', '023B01', '004C00', '056201',
               '207401', '3E8601', '529400', '66A000', '74A901', '99B718',
               'FCD163', 'F1B555', 'DF923D', 'CE7E45', 'FFFFFF'];
Map.addLayer(collectionCelsius.median(),{palette:palette,min:-20,max:30})

In the next chapter

Can't wait for it, right!

  1. List
  2. Array
  3. Reducing and Filtering
  4. Charts
  5. More things!

See you on next class!

Google Earth Engine for Dummies (II)

By Michael a Menarguez

Google Earth Engine for Dummies (II)

This seminar will present the basic concepts required to code using Google Earth engine for Remote Sensing purposes. Part II: GEE basic objects with examples

  • 10,878