Pillow/src/PIL/GimpGradientFile.py

143 lines
3.3 KiB
Python
Raw Normal View History

2010-07-31 06:52:47 +04:00
#
# Python Imaging Library
# $Id$
#
# stuff to read (and render) GIMP gradient files
#
# History:
# 97-08-23 fl Created
#
# Copyright (c) Secret Labs AB 1997.
# Copyright (c) Fredrik Lundh 1997.
#
# See the README file for information on usage and redistribution.
#
"""
Stuff to translate curve segments to palette values (derived from
the corresponding code in GIMP, written by Federico Mena Quintero.
See the GIMP distribution for more information.)
"""
from math import log, pi, sin, sqrt
from ._binary import o8
2010-07-31 06:52:47 +04:00
EPSILON = 1e-10
"""""" # Enable auto-doc for data member
2010-07-31 06:52:47 +04:00
2014-08-26 13:16:32 +04:00
2010-07-31 06:52:47 +04:00
def linear(middle, pos):
if pos <= middle:
if middle < EPSILON:
return 0.0
else:
return 0.5 * pos / middle
else:
pos = pos - middle
middle = 1.0 - middle
if middle < EPSILON:
return 1.0
else:
return 0.5 + 0.5 * pos / middle
2014-08-26 13:16:32 +04:00
2010-07-31 06:52:47 +04:00
def curved(middle, pos):
return pos ** (log(0.5) / log(max(middle, EPSILON)))
2014-08-26 13:16:32 +04:00
2010-07-31 06:52:47 +04:00
def sine(middle, pos):
return (sin((-pi / 2.0) + pi * linear(middle, pos)) + 1.0) / 2.0
2014-08-26 13:16:32 +04:00
2010-07-31 06:52:47 +04:00
def sphere_increasing(middle, pos):
return sqrt(1.0 - (linear(middle, pos) - 1.0) ** 2)
2014-08-26 13:16:32 +04:00
2010-07-31 06:52:47 +04:00
def sphere_decreasing(middle, pos):
return 1.0 - sqrt(1.0 - linear(middle, pos) ** 2)
2018-03-03 12:54:00 +03:00
2014-08-26 13:16:32 +04:00
SEGMENTS = [linear, curved, sine, sphere_increasing, sphere_decreasing]
"""""" # Enable auto-doc for data member
2014-08-26 13:16:32 +04:00
2010-07-31 06:52:47 +04:00
class GradientFile:
2010-07-31 06:52:47 +04:00
gradient = None
2014-08-26 13:16:32 +04:00
def getpalette(self, entries=256):
2010-07-31 06:52:47 +04:00
palette = []
ix = 0
x0, x1, xm, rgb0, rgb1, segment = self.gradient[ix]
for i in range(entries):
x = i / (entries - 1)
2010-07-31 06:52:47 +04:00
while x1 < x:
ix += 1
2010-07-31 06:52:47 +04:00
x0, x1, xm, rgb0, rgb1, segment = self.gradient[ix]
w = x1 - x0
if w < EPSILON:
scale = segment(0.5, 0.5)
else:
scale = segment((xm - x0) / w, (x - x0) / w)
# expand to RGBA
r = o8(int(255 * ((rgb1[0] - rgb0[0]) * scale + rgb0[0]) + 0.5))
g = o8(int(255 * ((rgb1[1] - rgb0[1]) * scale + rgb0[1]) + 0.5))
b = o8(int(255 * ((rgb1[2] - rgb0[2]) * scale + rgb0[2]) + 0.5))
a = o8(int(255 * ((rgb1[3] - rgb0[3]) * scale + rgb0[3]) + 0.5))
2010-07-31 06:52:47 +04:00
# add to palette
palette.append(r + g + b + a)
return b"".join(palette), "RGBA"
2010-07-31 06:52:47 +04:00
2014-08-26 13:16:32 +04:00
2019-03-21 16:28:20 +03:00
class GimpGradientFile(GradientFile):
"""File handler for GIMP's gradient format."""
2010-07-31 06:52:47 +04:00
def __init__(self, fp):
if fp.readline()[:13] != b"GIMP Gradient":
msg = "not a GIMP gradient file"
raise SyntaxError(msg)
2010-07-31 06:52:47 +04:00
2014-08-26 13:14:58 +04:00
line = fp.readline()
# GIMP 1.2 gradient files don't contain a name, but GIMP 1.3 files do
2014-08-26 13:23:12 +04:00
if line.startswith(b"Name: "):
2014-08-26 13:14:58 +04:00
line = fp.readline().strip()
count = int(line)
2010-07-31 06:52:47 +04:00
gradient = []
for i in range(count):
s = fp.readline().split()
w = [float(x) for x in s[:11]]
2010-07-31 06:52:47 +04:00
2014-08-26 13:16:32 +04:00
x0, x1 = w[0], w[2]
xm = w[1]
rgb0 = w[3:7]
rgb1 = w[7:11]
2010-07-31 06:52:47 +04:00
segment = SEGMENTS[int(s[11])]
2014-08-26 13:16:32 +04:00
cspace = int(s[12])
2010-07-31 06:52:47 +04:00
if cspace != 0:
msg = "cannot handle HSV colour space"
raise OSError(msg)
2010-07-31 06:52:47 +04:00
gradient.append((x0, x1, xm, rgb0, rgb1, segment))
self.gradient = gradient