-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpyvips-vari-ndvi.py
111 lines (80 loc) · 2.89 KB
/
pyvips-vari-ndvi.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
import sys
# install with 'pip install numpy'
import numpy
# install with 'pip install pyvips'
import pyvips
numpy.set_printoptions(threshold = numpy.inf)
# find min/max and histogram
def result_histogram(result):
np_2d = numpy.ndarray(
buffer = result.write_to_memory(),
dtype = numpy.float32,
shape = [result.height, result.width]
)
return numpy.histogram(result, bins = 256)[0]
# Remaps value (that has an expected range of in_low to in_high) into a target range of to_low to to_high
def math_map_value(value, in_low, in_high, to_low, to_high):
return to_low + (value - in_low) * (to_high - to_low) / (in_high - in_low)
def find_clipped_min_max(histogram, nmin, nmax):
# histogram
summed = sum(histogram)
three_percent = summed * 0.03
lower_sum = 0
upper_sum = 0
last_lower_i = 0
last_upper_i = 0
histogram_len = len(histogram)
for i in range(histogram_len):
# summing up values in lower_sum til the sum is >= 3% of the sum of all data values
# last_lower_i will be the data position right before lower_sum summed equal to 3% of the sum of all data values
if lower_sum < three_percent:
lower_sum += histogram[i]
last_lower_i = i
# // the same with last_upper_i
if upper_sum < three_percent:
upper_sum += histogram[histogram_len - 1 - i]
last_upper_i = histogram_len - 1 - i
return {
'nmin': math_map_value(last_lower_i, 0, 255, nmin, nmax),
'nmax': math_map_value(last_upper_i, 0, 255, nmin, nmax)
}
# return image bands depending of band_order
def bandsplit(image, band_order):
if band_order == 'GRN':
second, first, third, alpha = image.bandsplit()
else:
first, second, third, alpha = image.bandsplit()
return [first, second, third, alpha]
# apply image manipulation algebra
# VARI
def vari(image, band_order):
r, g, b, alpha = bandsplit(image, band_order)
index = (g - r) / (g + r - b)
return [alpha, index]
# NDVI
def ndvi(image, band_order):
r, g, nir, alpha = bandsplit(image, band_order)
index = (nir - r) / (nir + r)
return [alpha, index]
def apply_index(input_file, index):
# load image
image = pyvips.Image.new_from_file('./' + input_file)
# call index method
if index == 'NDVI':
alpha, result = ndvi(image, 'RGN')
elif index == 'VARI':
alpha, result = vari(image, 'RGB')
# it will be used to 'normalize' results
histogram = result_histogram(result)
clip_min_max = find_clipped_min_max(histogram, result.min(), result.max())
# normalize min/max
nmin = clip_min_max['nmin']
nmax = clip_min_max['nmax']
# apply image manipulation algebra to 'normalize' results
result = ((result - nmin) / (nmax - nmin)) * 256
# apply Look-Up-Table (LUT)
rdylgn_image = pyvips.Image.new_from_file('./rdylgn.png')
rgb = result.maplut(rdylgn_image)
# save to file
rgb[0:3].bandjoin(alpha).write_to_file('./' + index.lower() + '.png')
apply_index(sys.argv[1], sys.argv[2])