Pillow/src/PIL/ImageStat.py

149 lines
3.8 KiB
Python
Raw Normal View History

2010-07-31 06:52:47 +04:00
#
# The Python Imaging Library.
# $Id$
#
# global image statistics
#
# History:
# 1996-04-05 fl Created
# 1997-05-21 fl Added mask; added rms, var, stddev attributes
# 1997-08-05 fl Added median
# 1998-07-05 hk Fixed integer overflow error
#
# Notes:
# This class shows how to implement delayed evaluation of attributes.
# To get a certain value, simply access the corresponding attribute.
# The __getattr__ dispatcher takes care of the rest.
#
# Copyright (c) Secret Labs AB 1997.
# Copyright (c) Fredrik Lundh 1996-97.
#
# See the README file for information on usage and redistribution.
#
import functools
2014-08-26 17:47:10 +04:00
import math
import operator
2010-07-31 06:52:47 +04:00
class Stat:
2014-08-26 17:47:10 +04:00
def __init__(self, image_or_list, mask=None):
2010-07-31 06:52:47 +04:00
try:
if mask:
self.h = image_or_list.histogram(mask)
else:
self.h = image_or_list.histogram()
except AttributeError:
2014-08-26 17:47:10 +04:00
self.h = image_or_list # assume it to be a histogram list
if not isinstance(self.h, list):
msg = "first argument must be image or list"
raise TypeError(msg)
self.bands = list(range(len(self.h) // 256))
2010-07-31 06:52:47 +04:00
def __getattr__(self, id):
2019-02-03 07:58:24 +03:00
"""Calculate missing attribute"""
2010-07-31 06:52:47 +04:00
if id[:4] == "_get":
raise AttributeError(id)
2010-07-31 06:52:47 +04:00
# calculate missing attribute
v = getattr(self, "_get" + id)()
setattr(self, id, v)
return v
def _getextrema(self):
2019-02-03 07:58:24 +03:00
"""Get min/max values for each band in the image"""
2010-07-31 06:52:47 +04:00
def minmax(histogram):
n = 255
x = 0
for i in range(256):
if histogram[i]:
n = min(n, i)
x = max(x, i)
2014-08-26 17:47:10 +04:00
return n, x # returns (255, 0) if there's no data in the histogram
2010-07-31 06:52:47 +04:00
v = []
for i in range(0, len(self.h), 256):
v.append(minmax(self.h[i:]))
return v
def _getcount(self):
2019-02-03 07:58:24 +03:00
"""Get total number of pixels in each layer"""
2010-07-31 06:52:47 +04:00
v = []
for i in range(0, len(self.h), 256):
2019-03-21 16:28:20 +03:00
v.append(functools.reduce(operator.add, self.h[i : i + 256]))
2010-07-31 06:52:47 +04:00
return v
def _getsum(self):
2019-02-03 07:58:24 +03:00
"""Get sum of all pixels in each layer"""
2010-07-31 06:52:47 +04:00
v = []
for i in range(0, len(self.h), 256):
layer_sum = 0.0
2010-07-31 06:52:47 +04:00
for j in range(256):
layer_sum += j * self.h[i + j]
v.append(layer_sum)
2010-07-31 06:52:47 +04:00
return v
def _getsum2(self):
2019-02-03 07:58:24 +03:00
"""Get squared sum of all pixels in each layer"""
2010-07-31 06:52:47 +04:00
v = []
for i in range(0, len(self.h), 256):
sum2 = 0.0
for j in range(256):
2022-03-04 08:42:24 +03:00
sum2 += (j**2) * float(self.h[i + j])
2010-07-31 06:52:47 +04:00
v.append(sum2)
return v
def _getmean(self):
2019-02-03 07:58:24 +03:00
"""Get average pixel level for each layer"""
2010-07-31 06:52:47 +04:00
v = []
for i in self.bands:
v.append(self.sum[i] / self.count[i])
return v
def _getmedian(self):
2019-02-03 07:58:24 +03:00
"""Get median pixel level for each layer"""
2010-07-31 06:52:47 +04:00
v = []
for i in self.bands:
s = 0
2019-03-21 16:28:20 +03:00
half = self.count[i] // 2
2010-07-31 06:52:47 +04:00
b = i * 256
for j in range(256):
2019-03-21 16:28:20 +03:00
s = s + self.h[b + j]
2018-06-15 16:44:22 +03:00
if s > half:
2010-07-31 06:52:47 +04:00
break
v.append(j)
return v
def _getrms(self):
2019-02-03 07:58:24 +03:00
"""Get RMS for each layer"""
2010-07-31 06:52:47 +04:00
v = []
for i in self.bands:
v.append(math.sqrt(self.sum2[i] / self.count[i]))
return v
def _getvar(self):
2019-02-03 07:58:24 +03:00
"""Get variance for each layer"""
2010-07-31 06:52:47 +04:00
v = []
for i in self.bands:
n = self.count[i]
2019-03-21 16:28:20 +03:00
v.append((self.sum2[i] - (self.sum[i] ** 2.0) / n) / n)
2010-07-31 06:52:47 +04:00
return v
def _getstddev(self):
2019-02-03 07:58:24 +03:00
"""Get standard deviation for each layer"""
2010-07-31 06:52:47 +04:00
v = []
for i in self.bands:
v.append(math.sqrt(self.var[i]))
return v
2018-03-03 12:54:00 +03:00
2014-08-26 17:47:10 +04:00
Global = Stat # compatibility