425 lines
14 KiB
Python
Raw Permalink Normal View History

2012-03-30 16:12:24 +00:00
# -*- coding: utf8 -*-
2012-01-28 14:52:09 +00:00
"""
2012-03-30 16:12:24 +00:00
.. module:: lesscpy.lessc.color
:synopsis: Lesscpy Color functions
2013-07-19 11:21:51 +02:00
2012-01-28 14:52:09 +00:00
Copyright (c)
See LICENSE for details.
2012-07-18 17:28:04 -04:00
.. moduleauthor:: Johann T. Mariusson <jtm@robot.is>
2012-01-28 14:52:09 +00:00
"""
import operator
2012-01-28 14:52:09 +00:00
import colorsys
import re
2012-03-10 16:27:14 +00:00
from . import utility
2012-06-25 08:38:12 +00:00
from lesscpy.lib import colors
2012-01-28 14:52:09 +00:00
2013-07-19 11:21:51 +02:00
2012-03-10 16:27:14 +00:00
class Color():
2013-07-19 11:21:51 +02:00
2012-01-28 14:52:09 +00:00
def process(self, expression):
""" Process color expression
2012-03-30 16:12:24 +00:00
args:
expression (tuple): color expression
returns:
str
2012-01-28 14:52:09 +00:00
"""
a, o, b = expression
2012-03-10 16:27:14 +00:00
c1 = self._hextorgb(a)
c2 = self._hextorgb(b)
2012-01-28 14:52:09 +00:00
r = ['#']
for i in range(3):
v = self.operate(c1[i], c2[i], o)
2013-07-19 11:21:51 +02:00
if v > 0xff:
v = 0xff
if v < 0:
v = 0
2012-01-28 14:52:09 +00:00
r.append("%02x" % v)
return ''.join(r)
2013-07-19 11:21:51 +02:00
2012-03-30 16:12:24 +00:00
def operate(self, left, right, operation):
2012-01-28 14:52:09 +00:00
""" Do operation on colors
2012-03-30 16:12:24 +00:00
args:
left (str): left side
right (str): right side
operation (str): Operation
returns:
str
2012-01-28 14:52:09 +00:00
"""
operation = {
'+': operator.add,
'-': operator.sub,
'*': operator.mul,
'/': operator.truediv
2012-03-30 16:12:24 +00:00
}.get(operation)
return operation(left, right)
2013-07-19 11:21:51 +02:00
2012-03-10 16:27:14 +00:00
def rgb(self, *args):
2012-03-30 16:12:24 +00:00
""" Translate rgb(...) to color string
raises:
ValueError
returns:
str
2012-03-10 16:27:14 +00:00
"""
if len(args) == 4:
args = args[:3]
if len(args) == 3:
2012-03-10 16:27:14 +00:00
try:
2013-07-19 11:53:00 +02:00
return self._rgbatohex(list(map(int, args)))
2012-03-10 16:27:14 +00:00
except ValueError:
2013-07-19 11:21:51 +02:00
if all((a for a in args
if a[-1] == '%'
2012-03-10 16:27:14 +00:00
and 100 >= int(a[:-1]) >= 0)):
return self._rgbatohex([int(a[:-1]) * 255 / 100.0
for a in args])
raise ValueError('Illegal color values')
2013-07-19 11:21:51 +02:00
2012-03-10 16:27:14 +00:00
def rgba(self, *args):
2012-03-30 16:12:24 +00:00
""" Translate rgba(...) to color string
raises:
ValueError
returns:
str
2012-03-10 16:27:14 +00:00
"""
if len(args) == 4:
try:
falpha = float(list(args)[3])
if falpha > 1:
args = args[:3]
if falpha == 0:
values = self._rgbatohex_raw(list(map(int, args)))
return "rgba(%s)" % ','.join([str(a) for a in values])
2013-07-19 11:53:00 +02:00
return self._rgbatohex(list(map(int, args)))
2012-03-10 16:27:14 +00:00
except ValueError:
2013-07-19 11:21:51 +02:00
if all((a for a in args
if a[-1] == '%'
2012-03-10 16:27:14 +00:00
and 100 >= int(a[:-1]) >= 0)):
alpha = list(args)[3]
if alpha[-1] == '%' and float(alpha[:-1]) == 0:
values = self._rgbatohex_raw([int(a[:-1]) * 255 / 100.0
for a in args])
return "rgba(%s)" % ','.join([str(a) for a in values])
2012-03-10 16:27:14 +00:00
return self._rgbatohex([int(a[:-1]) * 255 / 100.0
for a in args])
raise ValueError('Illegal color values')
2013-07-19 11:21:51 +02:00
def argb(self, *args):
""" Translate argb(...) to color string
Creates a hex representation of a color in #AARRGGBB format (NOT
#RRGGBBAA!). This format is used in Internet Explorer, and .NET
and Android development.
raises:
ValueError
returns:
str
"""
if len(args) == 1 and type(args[0]) is str:
match = re.match(r'rgba\((.*)\)', args[0])
if match:
# NOTE(saschpe): Evil hack to cope with rgba(.., .., .., 0.5) passed through untransformed
rgb = re.sub(r'\s+', '', match.group(1)).split(',')
else:
rgb = list(self._hextorgb(args[0]))
else:
rgb = list(args)
if len(rgb) == 3:
return self._rgbatohex([255] + list(map(int, rgb)))
elif len(rgb) == 4:
rgb = [rgb.pop()] + rgb # Move Alpha to front
try:
fval = float(list(rgb)[0])
if fval > 1:
rgb = [255] + rgb[1:] # Clip invalid integer/float values
elif 1 >= fval >= 0:
rgb = [fval * 256] + rgb[1:] # Convert 0-1 to 0-255 range for _rgbatohex
else:
rgb = [0] + rgb[1:] # Clip lower bound
return self._rgbatohex(list(map(int, rgb)))
except ValueError:
if all((a for a in rgb
if a[-1] == '%'
and 100 >= int(a[:-1]) >= 0)):
return self._rgbatohex([int(a[:-1]) * 255 / 100.0
for a in rgb])
raise ValueError('Illegal color values')
2012-03-10 16:27:14 +00:00
def hsl(self, *args):
2012-03-30 16:12:24 +00:00
""" Translate hsl(...) to color string
raises:
ValueError
returns:
str
2012-03-10 16:27:14 +00:00
"""
if len(args) == 4:
return self.hsla(*args)
elif len(args) == 3:
h, s, l = args
rgb = colorsys.hls_to_rgb(int(h) / 360.0, utility.pc_or_float(l), utility.pc_or_float(s))
color = (utility.convergent_round(c * 255) for c in rgb)
2012-03-10 16:27:14 +00:00
return self._rgbatohex(color)
raise ValueError('Illegal color values')
2013-07-19 11:21:51 +02:00
2012-03-10 16:27:14 +00:00
def hsla(self, *args):
2012-03-30 16:12:24 +00:00
""" Translate hsla(...) to color string
raises:
ValueError
returns:
str
2012-03-10 16:27:14 +00:00
"""
if len(args) == 4:
h, s, l, a = args
rgb = colorsys.hls_to_rgb(int(h) / 360.0, utility.pc_or_float(l), utility.pc_or_float(s))
color = [float(utility.convergent_round(c * 255)) for c in rgb]
color.append(utility.pc_or_float(a))
2012-03-10 16:27:14 +00:00
return "rgba(%s,%s,%s,%s)" % tuple(color)
raise ValueError('Illegal color values')
2013-07-19 11:21:51 +02:00
2012-03-30 16:12:24 +00:00
def hue(self, color, *args):
""" Return the hue value of a color
args:
color (str): color
raises:
ValueError
returns:
float
"""
if color:
h, l, s = self._hextohls(color)
return utility.convergent_round(h * 360.0, 3)
2012-03-11 14:47:58 +00:00
raise ValueError('Illegal color values')
2013-07-19 11:21:51 +02:00
2012-03-30 16:12:24 +00:00
def saturation(self, color, *args):
""" Return the saturation value of a color
args:
color (str): color
raises:
ValueError
returns:
float
"""
if color:
h, l, s = self._hextohls(color)
return s * 100.0
2012-03-11 14:47:58 +00:00
raise ValueError('Illegal color values')
2013-07-19 11:21:51 +02:00
2012-03-30 16:12:24 +00:00
def lightness(self, color, *args):
""" Return the lightness value of a color
args:
color (str): color
raises:
ValueError
returns:
float
"""
if color:
h, l, s = self._hextohls(color)
return l * 100.0
2012-03-11 14:47:58 +00:00
raise ValueError('Illegal color values')
2013-07-19 11:21:51 +02:00
2012-03-10 16:27:14 +00:00
def opacity(self, *args):
2012-01-28 14:52:09 +00:00
"""
"""
2012-03-10 16:27:14 +00:00
pass
2013-07-19 11:21:51 +02:00
2012-03-30 16:12:24 +00:00
def lighten(self, color, diff, *args):
""" Lighten a color
args:
color (str): color
diff (str): percentage
returns:
str
"""
if color and diff:
return self._ophsl(color, diff, 1, operator.add)
2012-03-10 16:27:14 +00:00
raise ValueError('Illegal color values')
2013-07-19 11:21:51 +02:00
2012-03-30 16:12:24 +00:00
def darken(self, color, diff, *args):
""" Darken a color
args:
color (str): color
diff (str): percentage
returns:
str
"""
if color and diff:
return self._ophsl(color, diff, 1, operator.sub)
2012-03-10 16:27:14 +00:00
raise ValueError('Illegal color values')
2013-07-19 11:21:51 +02:00
2012-03-30 16:12:24 +00:00
def saturate(self, color, diff, *args):
""" Saturate a color
args:
color (str): color
diff (str): percentage
returns:
str
"""
if color and diff:
return self._ophsl(color, diff, 2, operator.add)
2012-03-10 16:27:14 +00:00
raise ValueError('Illegal color values')
2013-07-19 11:21:51 +02:00
2012-03-30 16:12:24 +00:00
def desaturate(self, color, diff, *args):
""" Desaturate a color
args:
color (str): color
diff (str): percentage
returns:
str
"""
if color and diff:
return self._ophsl(color, diff, 2, operator.sub)
2012-03-10 16:27:14 +00:00
raise ValueError('Illegal color values')
2013-07-19 11:21:51 +02:00
2012-03-30 16:12:24 +00:00
def _clamp(self, value):
# Clamp value
return min(1, max(0, value))
2013-07-19 11:21:51 +02:00
def greyscale(self, color, *args):
2012-03-30 16:12:24 +00:00
""" Simply 100% desaturate.
args:
color (str): color
returns:
str
"""
if color:
return self.desaturate(color, 100.0)
2012-03-10 16:27:14 +00:00
raise ValueError('Illegal color values')
def grayscale(self, color, *args):
"""Wrapper for greyscale, other spelling
2012-01-28 14:52:09 +00:00
"""
return self.greyscale(color, *args)
2013-07-19 11:21:51 +02:00
2012-03-30 16:12:24 +00:00
def spin(self, color, degree, *args):
""" Spin color by degree. (Increase / decrease hue)
args:
color (str): color
degree (str): percentage
raises:
ValueError
returns:
str
"""
if color and degree:
2013-07-19 11:21:51 +02:00
if isinstance(degree, str):
degree = float(degree.strip('%'))
2012-03-10 16:27:14 +00:00
h, l, s = self._hextohls(color)
2012-03-30 16:12:24 +00:00
h = ((h * 360.0) + degree) % 360.0
h = 360.0 + h if h < 0 else h
rgb = colorsys.hls_to_rgb(h / 360.0, l, s)
color = (utility.convergent_round(c * 255) for c in rgb)
2012-03-10 16:27:14 +00:00
return self._rgbatohex(color)
raise ValueError('Illegal color values')
2013-07-19 11:21:51 +02:00
2012-03-30 16:12:24 +00:00
def mix(self, color1, color2, weight=50, *args):
"""This algorithm factors in both the user-provided weight
2012-03-11 14:47:58 +00:00
and the difference between the alpha values of the two colors
to decide how to perform the weighted average of the two RGB values.
2013-07-19 11:21:51 +02:00
2012-03-11 14:47:58 +00:00
It works by first normalizing both parameters to be within [-1, 1],
where 1 indicates "only use color1", -1 indicates "only use color 0",
and all values in between indicated a proportionately weighted average.
2013-07-19 11:21:51 +02:00
2012-03-11 14:47:58 +00:00
Once we have the normalized variables w and a,
we apply the formula (w + a)/(1 + w*a)
to get the combined weight (in [-1, 1]) of color1.
This formula has two especially nice properties:
2013-07-19 11:21:51 +02:00
2012-03-11 14:47:58 +00:00
* When either w or a are -1 or 1, the combined weight is also that number
(cases where w * a == -1 are undefined, and handled as a special case).
2013-07-19 11:21:51 +02:00
2012-03-11 14:47:58 +00:00
* When a is 0, the combined weight is w, and vice versa
2013-07-19 11:21:51 +02:00
2012-03-11 14:47:58 +00:00
Finally, the weight of color1 is renormalized to be within [0, 1]
and the weight of color2 is given by 1 minus the weight of color1.
2013-07-19 11:21:51 +02:00
2012-03-10 16:27:14 +00:00
Copyright (c) 2006-2009 Hampton Catlin, Nathan Weizenbaum, and Chris Eppstein
http://sass-lang.com
2012-03-30 16:12:24 +00:00
args:
color1 (str): first color
color2 (str): second color
weight (int/str): weight
raises:
ValueError
returns:
str
"""
if color1 and color2:
2013-07-19 11:21:51 +02:00
if isinstance(weight, str):
weight = float(weight.strip('%'))
2012-03-30 16:12:24 +00:00
weight = ((weight / 100.0) * 2) - 1
rgb1 = self._hextorgb(color1)
rgb2 = self._hextorgb(color2)
alpha = 0
2013-07-19 11:21:51 +02:00
w1 = (((weight if weight * alpha == -1
2012-03-30 16:12:24 +00:00
else weight + alpha) / (1 + weight * alpha)) + 1)
2012-03-11 14:47:58 +00:00
w1 = w1 / 2.0
2012-03-10 16:27:14 +00:00
w2 = 1 - w1
rgb = [
rgb1[0] * w1 + rgb2[0] * w2,
rgb1[1] * w1 + rgb2[1] * w2,
rgb1[2] * w1 + rgb2[2] * w2,
]
return self._rgbatohex(rgb)
raise ValueError('Illegal color values')
2013-07-19 11:21:51 +02:00
2012-03-10 16:27:14 +00:00
def fmt(self, color):
2012-03-30 16:12:24 +00:00
""" Format CSS Hex color code.
uppercase becomes lowercase, 3 digit codes expand to 6 digit.
args:
color (str): color
raises:
ValueError
returns:
str
2012-01-28 14:52:09 +00:00
"""
2012-03-10 16:27:14 +00:00
if utility.is_color(color):
color = color.lower().strip('#')
if len(color) in [3, 4]:
2012-01-28 14:52:09 +00:00
color = ''.join([c * 2 for c in color])
return '#%s' % color
raise ValueError('Cannot format non-color')
2013-07-19 11:21:51 +02:00
def _rgbatohex_raw(self, rgba):
values = ["%x" % v for v in [0xff
if h > 0xff else
0 if h < 0 else h
for h in rgba]]
return values
2012-03-10 16:27:14 +00:00
def _rgbatohex(self, rgba):
2013-07-19 11:21:51 +02:00
return '#%s' % ''.join(["%02x" % v for v in
[0xff
if h > 0xff else
0 if h < 0 else h
2012-03-10 16:27:14 +00:00
for h in rgba]
2012-01-28 14:52:09 +00:00
])
2013-07-19 11:21:51 +02:00
def _hextorgb(self, hex):
2012-06-25 08:38:12 +00:00
if hex.lower() in colors.lessColors:
hex = colors.lessColors[hex.lower()]
2012-03-10 16:27:14 +00:00
hex = hex.strip()
if hex[0] == '#':
hex = hex.strip('#').strip(';')
if len(hex) == 3:
hex = [c * 2 for c in hex]
else:
2013-07-19 11:21:51 +02:00
hex = [hex[i:i + 2] for i in range(0, len(hex), 2)]
2012-03-10 16:27:14 +00:00
return tuple(int(c, 16) for c in hex)
return [int(hex, 16)] * 3
2013-07-19 11:21:51 +02:00
2012-03-10 16:27:14 +00:00
def _hextohls(self, hex):
rgb = self._hextorgb(hex)
return colorsys.rgb_to_hls(*[c / 255.0 for c in rgb])
2013-07-19 11:21:51 +02:00
def _ophsl(self, color, diff, idx, operation):
2013-07-19 11:21:51 +02:00
if isinstance(diff, str):
diff = float(diff.strip('%'))
2012-03-10 16:27:14 +00:00
hls = list(self._hextohls(color))
hls[idx] = self._clamp(operation(hls[idx], diff / 100.0))
2012-03-10 16:27:14 +00:00
rgb = colorsys.hls_to_rgb(*hls)
color = (utility.away_from_zero_round(c * 255) for c in rgb)
2012-03-10 16:27:14 +00:00
return self._rgbatohex(color)