Skip to content

Commit 4b3fa33

Browse files
authored
Merge pull request #322 from willrayeo/planetscope-veg-indices
Planetscope veg indices
2 parents f17f358 + 312e540 commit 4b3fa33

File tree

11 files changed

+490
-0
lines changed

11 files changed

+490
-0
lines changed

planet_scope/max_ndvi/README.md

+45
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
---
2+
title: 'Max Multitemporal NDVI'
3+
parent: Planetscope
4+
grand_parent: Planet
5+
layout: script
6+
permalink: /planet_scope/max_ndvi/
7+
nav_exclude: true
8+
scripts:
9+
- - Visualization
10+
- script.js
11+
- - EO Browser
12+
- eob.js
13+
- - Raw Values
14+
- raw.js
15+
examples:
16+
- zoom: '15'
17+
lat: '-16.60556'
18+
lng: '-48.80419'
19+
datasetId: 'ccb1f8f0-e5bf-4c31-afe5-d8803bcbde2a'
20+
fromTime: '2022-11-01T00:00:00.000Z'
21+
toTime: '2023-11-19T23:59:59.999Z'
22+
platform:
23+
- EOB
24+
evalscripturl: https://raw.githubusercontent.com/sentinel-hub/custom-scripts/master/planet_scope/max_ndvi/script.js
25+
additionalQueryParams:
26+
- - themeId
27+
- PLANET_SANDBOX
28+
---
29+
30+
## General description
31+
32+
The script evaluates the NDVI for each scene of the past month and returns the highest NDVI value for every pixel. In short, it returns the highest NDVI values of the past month for every pixel. The script runs on-the-fly, since
33+
it doesn't require preprocessing. It can be used as a cloud free background or an input for further analysis, such as change detection and classification. [Find out more.](https://www.sentinel-hub.com/max_service){:target="_blank"}
34+
Note that multi-temporal processing needs to be enabled for this script to run.
35+
36+
## Author of the script
37+
38+
William Ray
39+
40+
## Description of representative images
41+
42+
![figure 1](fig/fig1.png)
43+
44+
45+

planet_scope/max_ndvi/eob.js

+84
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,84 @@
1+
//VERSION=3
2+
3+
//Basic initialization setup function
4+
function setup() {
5+
return {
6+
//List of all bands, that will be used in the script, either for visualization or for choosing best pixel
7+
input: [{
8+
bands: [
9+
"Red",
10+
"NIR",
11+
]
12+
}],
13+
//This can always be the same if one is doing RGB images
14+
output: [
15+
{ id: "default", bands: 4 },
16+
{ id: "index", bands: 1, sampleType: "FLOAT32" },
17+
{ id: "eobrowserStats", bands: 2, sampleType: "FLOAT32" },
18+
{ id: "dataMask", bands: 1 }
19+
],
20+
mosaicking: "ORBIT"
21+
}
22+
}
23+
24+
/*
25+
In this function we limit the scenes, which are used for processing.
26+
These are based also on input variables.
27+
E.g. if one sets date "2017-03-01" ("TO date") and cloud coverage filter 30%,
28+
all scenes older than 2017-03-01 with cloud coverage 30% will be checked against
29+
further conditions in this function (in this function it is currently limited to 3 months).
30+
The more scenes there are, longer it will take to process the data.
31+
After 60 seconds of processing, there will be a timeout.
32+
*/
33+
34+
function preProcessScenes(collections) {
35+
collections.scenes.orbits = collections.scenes.orbits.filter(function (orbit) {
36+
var orbitDateFrom = new Date(orbit.dateFrom)
37+
return orbitDateFrom.getTime() >= (collections.to.getTime() - 3 * 31 * 24 * 3600 * 1000);
38+
})
39+
return collections
40+
}
41+
42+
function calcNDVI(sample) {
43+
var denom = sample.Red + sample.NIR;
44+
return ((denom != 0) ? (sample.NIR - sample.Red) / denom : NaN);
45+
}
46+
function evaluatePixel(samples) {
47+
var max = Number.NEGATIVE_INFINITY;
48+
for (let i = 0; i < samples.length; i++) {
49+
var ndvi = calcNDVI(samples[i]);
50+
max = ndvi > max ? ndvi : max;
51+
}
52+
let max_exists = 0;
53+
if (isFinite(max)) {
54+
max_exists = 1;
55+
}
56+
let imgVals;
57+
if (max < -1.1) { imgVals = [0, 0, 0]; }
58+
else if (max < - 0.2) { imgVals = [0.75, 0.75, 0.75]; }
59+
else if (max < - 0.1) { imgVals = [0.86, 0.86, 0.86]; }
60+
else if (max < 0) { imgVals = [1, 1, 0.88]; }
61+
else if (max < 0.025) { imgVals = [1, 0.98, 0.8]; }
62+
else if (max < 0.05) { imgVals = [0.93, 0.91, 0.71]; }
63+
else if (max < 0.075) { imgVals = [0.87, 0.85, 0.61]; }
64+
else if (max < 0.1) { imgVals = [0.8, 0.78, 0.51]; }
65+
else if (max < 0.125) { imgVals = [0.74, 0.72, 0.42]; }
66+
else if (max < 0.15) { imgVals = [0.69, 0.76, 0.38]; }
67+
else if (max < 0.175) { imgVals = [0.64, 0.8, 0.35]; }
68+
else if (max < 0.2) { imgVals = [0.57, 0.75, 0.32]; }
69+
else if (max < 0.25) { imgVals = [0.5, 0.7, 0.28]; }
70+
else if (max < 0.3) { imgVals = [0.44, 0.64, 0.25]; }
71+
else if (max < 0.35) { imgVals = [0.38, 0.59, 0.21]; }
72+
else if (max < 0.4) { imgVals = [0.31, 0.54, 0.18]; }
73+
else if (max < 0.45) { imgVals = [0.25, 0.49, 0.14]; }
74+
else if (max < 0.5) { imgVals = [0.19, 0.43, 0.11]; }
75+
else if (max < 0.55) { imgVals = [0.13, 0.38, 0.07]; }
76+
else if (max < 0.6) { imgVals = [0.06, 0.33, 0.04]; }
77+
else { imgVals = [0, 0.27, 0]; }
78+
return {
79+
default: imgVals.concat(max_exists),
80+
index: [max],
81+
eobrowserStats: [max, max_exists],
82+
dataMask: [max_exists]
83+
}
84+
}

planet_scope/max_ndvi/fig/fig1.png

1.2 MB
Loading

planet_scope/max_ndvi/raw.js

+48
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,48 @@
1+
//VERSION=3
2+
3+
//Basic initialization setup function
4+
function setup() {
5+
return {
6+
//List of all bands, that will be used in the script, either for visualization or for choosing best pixel
7+
input: [{
8+
bands: [
9+
"Red",
10+
"NIR"
11+
]
12+
}],
13+
//This can always be the same if one is doing RGB images
14+
output: { bands: 1 },
15+
mosaicking: "ORBIT"
16+
}
17+
}
18+
19+
/*
20+
In this function we limit the scenes, which are used for processing.
21+
These are based also on input variables.
22+
E.g. if one sets date "2017-03-01" ("TO date") and cloud coverage filter 30%,
23+
all scenes older than 2017-03-01 with cloud coverage 30% will be checked against
24+
further conditions in this function (in this function it is currently limited to 3 months).
25+
The more scenes there are, longer it will take to process the data.
26+
After 60 seconds of processing, there will be a timeout.
27+
*/
28+
29+
function preProcessScenes(collections) {
30+
collections.scenes.orbits = collections.scenes.orbits.filter(function (orbit) {
31+
var orbitDateFrom = new Date(orbit.dateFrom)
32+
return orbitDateFrom.getTime() >= (collections.to.getTime() - 3 * 31 * 24 * 3600 * 1000);
33+
})
34+
return collections
35+
}
36+
37+
function calcNDVI(sample) {
38+
var denom = sample.Red + sample.NIR;
39+
return ((denom != 0) ? (sample.NIR - sample.Red) / denom : NaN);
40+
}
41+
function evaluatePixel(samples) {
42+
var max = Number.NEGATIVE_INFINITY;
43+
for (let i = 0; i < samples.length; i++) {
44+
var ndvi = calcNDVI(samples[i]);
45+
max = ndvi > max ? ndvi : max;
46+
}
47+
return [max];
48+
}

planet_scope/max_ndvi/script.js

+72
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
//VERSION=3
2+
3+
//Basic initialization setup function
4+
function setup() {
5+
return {
6+
//List of all bands, that will be used in the script, either for visualization or for choosing best pixel
7+
input: [{
8+
bands: [
9+
"Red",
10+
"NIR"
11+
]
12+
}],
13+
//This can always be the same if one is doing RGB images
14+
output: { bands: 4 },
15+
mosaicking: "ORBIT"
16+
}
17+
}
18+
19+
/*
20+
In this function we limit the scenes, which are used for processing.
21+
These are based also on input variables.
22+
E.g. if one sets date "2017-03-01" ("TO date") and cloud coverage filter 30%,
23+
all scenes older than 2017-03-01 with cloud coverage 30% will be checked against
24+
further conditions in this function (in this function it is currently limited to 3 months).
25+
The more scenes there are, longer it will take to process the data.
26+
After 60 seconds of processing, there will be a timeout.
27+
*/
28+
29+
function preProcessScenes(collections) {
30+
collections.scenes.orbits = collections.scenes.orbits.filter(function (orbit) {
31+
var orbitDateFrom = new Date(orbit.dateFrom)
32+
return orbitDateFrom.getTime() >= (collections.to.getTime() - 3 * 31 * 24 * 3600 * 1000);
33+
})
34+
return collections
35+
}
36+
37+
function calcNDVI(sample) {
38+
var denom = sample.Red + sample.NIR;
39+
return ((denom != 0) ? (sample.NIR - sample.Red) / denom : NaN);
40+
}
41+
function evaluatePixel(samples) {
42+
var max = Number.NEGATIVE_INFINITY;
43+
for (let i = 0; i < samples.length; i++) {
44+
var ndvi = calcNDVI(samples[i]);
45+
max = ndvi > max ? ndvi : max;
46+
}
47+
let max_exists = 0;
48+
if (isFinite(max)) {
49+
max_exists = 1;
50+
}
51+
if (max < -1.1) { return [0, 0, 0, max_exists]; }
52+
else if (max < - 0.2) { return [0.75, 0.75, 0.75, max_exists]; }
53+
else if (max < - 0.1) { return [0.86, 0.86, 0.86, max_exists]; }
54+
else if (max < 0) { return [1, 1, 0.88, max_exists]; }
55+
else if (max < 0.025) { return [1, 0.98, 0.8, max_exists]; }
56+
else if (max < 0.05) { return [0.93, 0.91, 0.71, max_exists]; }
57+
else if (max < 0.075) { return [0.87, 0.85, 0.61, max_exists]; }
58+
else if (max < 0.1) { return [0.8, 0.78, 0.51, max_exists]; }
59+
else if (max < 0.125) { return [0.74, 0.72, 0.42, max_exists]; }
60+
else if (max < 0.15) { return [0.69, 0.76, 0.38, max_exists]; }
61+
else if (max < 0.175) { return [0.64, 0.8, 0.35, max_exists]; }
62+
else if (max < 0.2) { return [0.57, 0.75, 0.32, max_exists]; }
63+
else if (max < 0.25) { return [0.5, 0.7, 0.28, max_exists]; }
64+
else if (max < 0.3) { return [0.44, 0.64, 0.25, max_exists]; }
65+
else if (max < 0.35) { return [0.38, 0.59, 0.21, max_exists]; }
66+
else if (max < 0.4) { return [0.31, 0.54, 0.18, max_exists]; }
67+
else if (max < 0.45) { return [0.25, 0.49, 0.14, max_exists]; }
68+
else if (max < 0.5) { return [0.19, 0.43, 0.11, max_exists]; }
69+
else if (max < 0.55) { return [0.13, 0.38, 0.07, max_exists]; }
70+
else if (max < 0.6) { return [0.06, 0.33, 0.04, max_exists]; }
71+
else { return [0, 0.27, 0, max_exists]; }
72+
}
+43
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
---
2+
title: NDVI difference between two dates
3+
parent: Planetscope
4+
grand_parent: Planet
5+
layout: script
6+
permalink: /planet_scope/ndvi_difference/
7+
nav_exclude: true
8+
scripts:
9+
- - Visualization
10+
- script.js
11+
- - EO Browser
12+
- eob.js
13+
- - Raw Values
14+
- raw.js
15+
examples:
16+
- zoom: '14'
17+
lat: '52.24714'
18+
lng: '9.20886'
19+
datasetId: Planetscope
20+
fromTime: '2023-04-02T00:00:00.000Z'
21+
toTime: '2023-06-01T23:59:59.999Z'
22+
platform:
23+
- EOB
24+
evalscripturl: https://raw.githubusercontent.com/sentinel-hub/custom-scripts/master/planet_scope/ndvi_difference/script.js
25+
additionalQueryParams:
26+
- - themeId
27+
- PLANET_SANDBOX
28+
---
29+
30+
## General description
31+
This script aims to obtain the diffence of NDVI between the latest acquisition and the acquisition 10-day prior to the latest on within a specified time period. Multi-temporal analysis is common in the Earth Observation field. Here we take NDVI as an example and demonstrate how to calculate the difference of NDVI between two acquisitions using [`mosaicking: ORBIT`](https://docs.sentinel-hub.com/api/latest/evalscript/v3/#mosaicking) and [`preProcessScenes`](https://docs.sentinel-hub.com/api/latest/evalscript/v3/#preprocessscenes-function-optional) in one single request.
32+
33+
To implement multi-temporal analysis in the Evalscript, we apply `ORBIT` mosaicking to query daily mosaic in the specified time period. Then, by using the optional `preProcessScenes` function, we find out the acquisition acquired on the date closest to the date 10-day prior to the latest acquisition and filter the out the other unused acquisitions to save Processing Units. Last but not least, in the `evaluatePixel` function we initialise a combined mask to ensure the difference of NDVI between two acquisitions exists only if there is data on both dates.
34+
35+
**Note**: The example script is used to obtain the raw value of NDVI difference. For visualisation purpose, please follow the EO Browser link in the [Evaluate and visualize](#evaluate-and-visualize) section. The visualisation script contains 4 outputs: default, index, eobrowserStats and dataMask. The default layer is a visualisation layer to visualise NDVI difference in EO Browser. The index layer is the actual value of the NDVI difference. The eobrowserStats and the dataMask layer is configured to activate statistical features on EO Browser. Please see the [FAQ](https://www.sentinel-hub.com/faq/#how-configure-your-layers-statistical-info-eo-browser) for more details.
36+
37+
## Author of the script
38+
39+
William Ray
40+
41+
## Description of representative images
42+
The following image shows the NDVI difference between the latest acquisition and the acquisition 10-day prior to the latest one during the time period from 2nd of April, 2023 to 2nd June, 2023.
43+
![NDVI difference example](fig/fig1.png)

planet_scope/ndvi_difference/eob.js

+77
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,77 @@
1+
//VERSION=3
2+
// Script to extract NDVI difference between the latest acquisition and the acquisition 10-day prior to the latest within a specified time range
3+
// To be used on EO Browser. Makes statistics work in EOB
4+
function setup() {
5+
return {
6+
input: [{
7+
bands: ["Red", "NIR", "dataMask"],
8+
units: "DN"
9+
}],
10+
output: [
11+
{ id: "default", bands: 4 },
12+
{ id: "index", bands: 1, sampleType: "FLOAT32" },
13+
{ id: "eobrowserStats", bands: 2, sampleType: "FLOAT32" },
14+
{ id: "dataMask", bands: 1 }
15+
],
16+
mosaicking: Mosaicking.ORBIT
17+
}
18+
19+
}
20+
21+
function evaluatePixel(samples) {
22+
// ndvi difference
23+
let latest = samples[0];
24+
let prior = samples[1];
25+
let dataMask = latest.dataMask * prior.dataMask;
26+
const diff = dataMask === 1 ? index(latest.NIR, latest.Red) - index(prior.NIR, prior.Red) : NaN;
27+
28+
// visualisation
29+
const ramps = [
30+
[-2, 0x8e0152],
31+
[-1, 0xc51b7d],
32+
[-0.5, 0xde77ae],
33+
[0, 0xf7f7f7],
34+
[0.5, 0x7fbc41],
35+
[1, 0x4d9221],
36+
[2, 0x276419]
37+
]
38+
const visualizer = new ColorRampVisualizer(ramps);
39+
let imgVals = visualizer.process(diff);
40+
return {
41+
default: imgVals.concat(dataMask),
42+
index: [diff],
43+
eobrowserStats: [diff, dataMask],
44+
dataMask: [dataMask]
45+
};
46+
}
47+
48+
function preProcessScenes(collections) {
49+
// sort from most recent to least recent
50+
collections.scenes.orbits = collections.scenes.orbits.sort(
51+
(s1, s2) => new Date(s2.dateFrom) - new Date(s1.dateFrom)
52+
);
53+
54+
let scenes = collections.scenes.orbits;
55+
let latest;
56+
let closest;
57+
latest = closest = scenes[0];
58+
59+
// timestamp of 10-day prior to latest acquisition
60+
let target = new Date(new Date(latest.dateFrom).getTime() - 10 * 24 * 3600 * 1000);
61+
62+
// find closet timestamp to the target
63+
let diff = Number.POSITIVE_INFINITY;
64+
for (let i = 1; i < scenes.length; i++) {
65+
current = new Date(scenes[i].dateFrom);
66+
if (Math.abs(current - target) >= diff) { break; }
67+
diff = Math.abs(current - target);
68+
closest = scenes[i];
69+
}
70+
71+
// filter collections to keep the latest acquisition and the closest acquisitions to the target
72+
collections.scenes.orbits = collections.scenes.orbits.filter(function (orbit) {
73+
var orbitDateFrom = orbit.dateFrom;
74+
return [latest.dateFrom, closest.dateFrom].includes(orbitDateFrom);
75+
})
76+
return collections
77+
}
3.65 MB
Loading

0 commit comments

Comments
 (0)