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
|
|
|
"""
|
2013-08-12 14:46:23 +02:00
|
|
|
|
|
|
|
import operator
|
|
|
|
|
2012-01-28 14:52:09 +00:00
|
|
|
import colorsys
|
2013-12-19 13:25:42 +01:00
|
|
|
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 = {
|
2013-08-12 14:46:23 +02:00
|
|
|
'+': operator.add,
|
|
|
|
'-': operator.sub,
|
|
|
|
'*': operator.mul,
|
|
|
|
'/': operator.truediv
|
2012-03-30 16:12:24 +00:00
|
|
|
}.get(operation)
|
2013-08-12 14:46:23 +02:00
|
|
|
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:
|
2013-12-19 14:56:16 +01:00
|
|
|
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:
|
2013-12-20 14:32:31 +01:00
|
|
|
falpha = float(list(args)[3])
|
|
|
|
if falpha > 1:
|
2013-12-19 15:56:32 +01:00
|
|
|
args = args[:3]
|
2013-12-20 14:32:31 +01:00
|
|
|
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)):
|
2013-12-20 14:32:31 +01:00
|
|
|
alpha = list(args)[3]
|
|
|
|
if alpha[-1] == '%' and float(alpha[:-1]) == 0:
|
|
|
|
values = self._rgbatohex_raw([int(a[:-1]) * 255 / 100.0
|
2014-02-02 14:24:16 +01:00
|
|
|
for a in args])
|
2013-12-20 14:32:31 +01:00
|
|
|
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
|
|
|
|
2013-12-19 13:25:42 +01: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:
|
2014-02-02 14:24:16 +01:00
|
|
|
rgb = [rgb.pop()] + rgb # Move Alpha to front
|
2013-12-19 13:25:42 +01:00
|
|
|
try:
|
|
|
|
fval = float(list(rgb)[0])
|
|
|
|
if fval > 1:
|
2014-02-02 14:24:16 +01:00
|
|
|
rgb = [255] + rgb[1:] # Clip invalid integer/float values
|
2013-12-19 13:25:42 +01:00
|
|
|
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
|
2014-02-18 12:00:07 +00:00
|
|
|
rgb = colorsys.hls_to_rgb(int(h) / 360.0, utility.pc_or_float(l), utility.pc_or_float(s))
|
2013-08-08 17:56:36 +02:00
|
|
|
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
|
2014-02-18 12:00:07 +00:00
|
|
|
rgb = colorsys.hls_to_rgb(int(h) / 360.0, utility.pc_or_float(l), utility.pc_or_float(s))
|
2013-08-08 17:56:36 +02:00
|
|
|
color = [float(utility.convergent_round(c * 255)) for c in rgb]
|
2014-02-18 12:00:07 +00:00
|
|
|
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)
|
2013-08-08 17:56:36 +02:00
|
|
|
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:
|
2013-08-12 14:46:23 +02:00
|
|
|
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:
|
2013-08-12 14:46:23 +02:00
|
|
|
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:
|
2013-08-12 14:46:23 +02:00
|
|
|
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:
|
2013-08-12 14:46:23 +02:00
|
|
|
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
|
|
|
|
2013-12-20 13:27:13 +01: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')
|
|
|
|
|
2013-12-20 13:27:13 +01:00
|
|
|
def grayscale(self, color, *args):
|
|
|
|
"""Wrapper for greyscale, other spelling
|
2012-01-28 14:52:09 +00:00
|
|
|
"""
|
2013-12-20 13:27:13 +01: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):
|
2013-12-20 13:21:51 +01:00
|
|
|
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)
|
2013-08-08 17:56:36 +02:00
|
|
|
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):
|
2013-12-20 13:21:51 +01:00
|
|
|
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
|
|
|
|
2013-12-20 14:32:31 +01: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
|
|
|
|
2013-08-12 14:46:23 +02:00
|
|
|
def _ophsl(self, color, diff, idx, operation):
|
2013-07-19 11:21:51 +02:00
|
|
|
if isinstance(diff, str):
|
2013-12-20 13:21:51 +01:00
|
|
|
diff = float(diff.strip('%'))
|
2012-03-10 16:27:14 +00:00
|
|
|
hls = list(self._hextohls(color))
|
2013-08-12 14:46:23 +02:00
|
|
|
hls[idx] = self._clamp(operation(hls[idx], diff / 100.0))
|
2012-03-10 16:27:14 +00:00
|
|
|
rgb = colorsys.hls_to_rgb(*hls)
|
2013-08-08 18:06:10 +02:00
|
|
|
color = (utility.away_from_zero_round(c * 255) for c in rgb)
|
2012-03-10 16:27:14 +00:00
|
|
|
return self._rgbatohex(color)
|