434 lines
15 KiB
Python
434 lines
15 KiB
Python
"""Strip viewer and related widgets.
|
||
|
||
The classes in this file implement the StripViewer shown in the top two thirds
|
||
of the main Pynche window. It consists of three StripWidgets which display
|
||
the variations in red, green, and blue respectively of the currently selected
|
||
r/g/b color value.
|
||
|
||
Each StripWidget shows the color variations that are reachable by varying an
|
||
axis of the currently selected color. So for example, if the color is
|
||
|
||
(R,G,B)=(127,163,196)
|
||
|
||
then the Red variations show colors from (0,163,196) to (255,163,196), the
|
||
Green variations show colors from (127,0,196) to (127,255,196), and the Blue
|
||
variations show colors from (127,163,0) to (127,163,255).
|
||
|
||
The selected color is always visible in all three StripWidgets, and in fact
|
||
each StripWidget highlights the selected color, and has an arrow pointing to
|
||
the selected chip, which includes the value along that particular axis.
|
||
|
||
Clicking on any chip in any StripWidget selects that color, and updates all
|
||
arrows and other windows. By toggling on Update while dragging, Pynche will
|
||
select the color under the cursor while you drag it, but be forewarned that
|
||
this can be slow.
|
||
"""
|
||
|
||
from tkinter import *
|
||
import ColorDB
|
||
|
||
# Load this script into the Tcl interpreter and call it in
|
||
# StripWidget.set_color(). This is about as fast as it can be with the
|
||
# current _tkinter.c interface, which doesn't support Tcl Objects.
|
||
TCLPROC = '''\
|
||
proc setcolor {canv colors} {
|
||
set i 1
|
||
foreach c $colors {
|
||
$canv itemconfigure $i -fill $c -outline $c
|
||
incr i
|
||
}
|
||
}
|
||
'''
|
||
|
||
# Tcl event types
|
||
BTNDOWN = 4
|
||
BTNUP = 5
|
||
BTNDRAG = 6
|
||
|
||
SPACE = ' '
|
||
|
||
|
||
|
||
def constant(numchips):
|
||
step = 255.0 / (numchips - 1)
|
||
start = 0.0
|
||
seq = []
|
||
while numchips > 0:
|
||
seq.append(int(start))
|
||
start = start + step
|
||
numchips = numchips - 1
|
||
return seq
|
||
|
||
# red variations, green+blue = cyan constant
|
||
def constant_red_generator(numchips, red, green, blue):
|
||
seq = constant(numchips)
|
||
return list(zip([red] * numchips, seq, seq))
|
||
|
||
# green variations, red+blue = magenta constant
|
||
def constant_green_generator(numchips, red, green, blue):
|
||
seq = constant(numchips)
|
||
return list(zip(seq, [green] * numchips, seq))
|
||
|
||
# blue variations, red+green = yellow constant
|
||
def constant_blue_generator(numchips, red, green, blue):
|
||
seq = constant(numchips)
|
||
return list(zip(seq, seq, [blue] * numchips))
|
||
|
||
# red variations, green+blue = cyan constant
|
||
def constant_cyan_generator(numchips, red, green, blue):
|
||
seq = constant(numchips)
|
||
return list(zip(seq, [green] * numchips, [blue] * numchips))
|
||
|
||
# green variations, red+blue = magenta constant
|
||
def constant_magenta_generator(numchips, red, green, blue):
|
||
seq = constant(numchips)
|
||
return list(zip([red] * numchips, seq, [blue] * numchips))
|
||
|
||
# blue variations, red+green = yellow constant
|
||
def constant_yellow_generator(numchips, red, green, blue):
|
||
seq = constant(numchips)
|
||
return list(zip([red] * numchips, [green] * numchips, seq))
|
||
|
||
|
||
|
||
class LeftArrow:
|
||
_ARROWWIDTH = 30
|
||
_ARROWHEIGHT = 15
|
||
_YOFFSET = 13
|
||
_TEXTYOFFSET = 1
|
||
_TAG = ('leftarrow',)
|
||
|
||
def __init__(self, canvas, x):
|
||
self._canvas = canvas
|
||
self.__arrow, self.__text = self._create(x)
|
||
self.move_to(x)
|
||
|
||
def _create(self, x):
|
||
arrow = self._canvas.create_line(
|
||
x, self._ARROWHEIGHT + self._YOFFSET,
|
||
x, self._YOFFSET,
|
||
x + self._ARROWWIDTH, self._YOFFSET,
|
||
arrow='first',
|
||
width=3.0,
|
||
tags=self._TAG)
|
||
text = self._canvas.create_text(
|
||
x + self._ARROWWIDTH + 13,
|
||
self._ARROWHEIGHT - self._TEXTYOFFSET,
|
||
tags=self._TAG,
|
||
text='128')
|
||
return arrow, text
|
||
|
||
def _x(self):
|
||
coords = list(self._canvas.coords(self._TAG))
|
||
assert coords
|
||
return coords[0]
|
||
|
||
def move_to(self, x):
|
||
deltax = x - self._x()
|
||
self._canvas.move(self._TAG, deltax, 0)
|
||
|
||
def set_text(self, text):
|
||
self._canvas.itemconfigure(self.__text, text=text)
|
||
|
||
|
||
class RightArrow(LeftArrow):
|
||
_TAG = ('rightarrow',)
|
||
|
||
def _create(self, x):
|
||
arrow = self._canvas.create_line(
|
||
x, self._YOFFSET,
|
||
x + self._ARROWWIDTH, self._YOFFSET,
|
||
x + self._ARROWWIDTH, self._ARROWHEIGHT + self._YOFFSET,
|
||
arrow='last',
|
||
width=3.0,
|
||
tags=self._TAG)
|
||
text = self._canvas.create_text(
|
||
x - self._ARROWWIDTH + 15, # BAW: kludge
|
||
self._ARROWHEIGHT - self._TEXTYOFFSET,
|
||
justify=RIGHT,
|
||
text='128',
|
||
tags=self._TAG)
|
||
return arrow, text
|
||
|
||
def _x(self):
|
||
coords = list(self._canvas.coords(self._TAG))
|
||
assert coords
|
||
return coords[0] + self._ARROWWIDTH
|
||
|
||
|
||
|
||
class StripWidget:
|
||
_CHIPHEIGHT = 50
|
||
_CHIPWIDTH = 10
|
||
_NUMCHIPS = 40
|
||
|
||
def __init__(self, switchboard,
|
||
master = None,
|
||
chipwidth = _CHIPWIDTH,
|
||
chipheight = _CHIPHEIGHT,
|
||
numchips = _NUMCHIPS,
|
||
generator = None,
|
||
axis = None,
|
||
label = '',
|
||
uwdvar = None,
|
||
hexvar = None):
|
||
# instance variables
|
||
self.__generator = generator
|
||
self.__axis = axis
|
||
self.__numchips = numchips
|
||
assert self.__axis in (0, 1, 2)
|
||
self.__uwd = uwdvar
|
||
self.__hexp = hexvar
|
||
# the last chip selected
|
||
self.__lastchip = None
|
||
self.__sb = switchboard
|
||
|
||
canvaswidth = numchips * (chipwidth + 1)
|
||
canvasheight = chipheight + 43 # BAW: Kludge
|
||
|
||
# create the canvas and pack it
|
||
canvas = self.__canvas = Canvas(master,
|
||
width=canvaswidth,
|
||
height=canvasheight,
|
||
## borderwidth=2,
|
||
## relief=GROOVE
|
||
)
|
||
|
||
canvas.pack()
|
||
canvas.bind('<ButtonPress-1>', self.__select_chip)
|
||
canvas.bind('<ButtonRelease-1>', self.__select_chip)
|
||
canvas.bind('<B1-Motion>', self.__select_chip)
|
||
|
||
# Load a proc into the Tcl interpreter. This is used in the
|
||
# set_color() method to speed up setting the chip colors.
|
||
canvas.tk.eval(TCLPROC)
|
||
|
||
# create the color strip
|
||
chips = self.__chips = []
|
||
x = 1
|
||
y = 30
|
||
tags = ('chip',)
|
||
for c in range(self.__numchips):
|
||
color = 'grey'
|
||
canvas.create_rectangle(
|
||
x, y, x+chipwidth, y+chipheight,
|
||
fill=color, outline=color,
|
||
tags=tags)
|
||
x = x + chipwidth + 1 # for outline
|
||
chips.append(color)
|
||
|
||
# create the strip label
|
||
self.__label = canvas.create_text(
|
||
3, y + chipheight + 8,
|
||
text=label,
|
||
anchor=W)
|
||
|
||
# create the arrow and text item
|
||
chipx = self.__arrow_x(0)
|
||
self.__leftarrow = LeftArrow(canvas, chipx)
|
||
|
||
chipx = self.__arrow_x(len(chips) - 1)
|
||
self.__rightarrow = RightArrow(canvas, chipx)
|
||
|
||
def __arrow_x(self, chipnum):
|
||
coords = self.__canvas.coords(chipnum+1)
|
||
assert coords
|
||
x0, y0, x1, y1 = coords
|
||
return (x1 + x0) / 2.0
|
||
|
||
# Invoked when one of the chips is clicked. This should just tell the
|
||
# switchboard to set the color on all the output components
|
||
def __select_chip(self, event=None):
|
||
x = event.x
|
||
y = event.y
|
||
canvas = self.__canvas
|
||
chip = canvas.find_overlapping(x, y, x, y)
|
||
if chip and (1 <= chip[0] <= self.__numchips):
|
||
color = self.__chips[chip[0]-1]
|
||
red, green, blue = ColorDB.rrggbb_to_triplet(color)
|
||
etype = int(event.type)
|
||
if (etype == BTNUP or self.__uwd.get()):
|
||
# update everyone
|
||
self.__sb.update_views(red, green, blue)
|
||
else:
|
||
# just track the arrows
|
||
self.__trackarrow(chip[0], (red, green, blue))
|
||
|
||
def __trackarrow(self, chip, rgbtuple):
|
||
# invert the last chip
|
||
if self.__lastchip is not None:
|
||
color = self.__canvas.itemcget(self.__lastchip, 'fill')
|
||
self.__canvas.itemconfigure(self.__lastchip, outline=color)
|
||
self.__lastchip = chip
|
||
# get the arrow's text
|
||
coloraxis = rgbtuple[self.__axis]
|
||
if self.__hexp.get():
|
||
# hex
|
||
text = hex(coloraxis)
|
||
else:
|
||
# decimal
|
||
text = repr(coloraxis)
|
||
# move the arrow, and set its text
|
||
if coloraxis <= 128:
|
||
# use the left arrow
|
||
self.__leftarrow.set_text(text)
|
||
self.__leftarrow.move_to(self.__arrow_x(chip-1))
|
||
self.__rightarrow.move_to(-100)
|
||
else:
|
||
# use the right arrow
|
||
self.__rightarrow.set_text(text)
|
||
self.__rightarrow.move_to(self.__arrow_x(chip-1))
|
||
self.__leftarrow.move_to(-100)
|
||
# and set the chip's outline
|
||
brightness = ColorDB.triplet_to_brightness(rgbtuple)
|
||
if brightness <= 128:
|
||
outline = 'white'
|
||
else:
|
||
outline = 'black'
|
||
self.__canvas.itemconfigure(chip, outline=outline)
|
||
|
||
|
||
def update_yourself(self, red, green, blue):
|
||
assert self.__generator
|
||
i = 1
|
||
chip = 0
|
||
chips = self.__chips = []
|
||
tk = self.__canvas.tk
|
||
# get the red, green, and blue components for all chips
|
||
for t in self.__generator(self.__numchips, red, green, blue):
|
||
rrggbb = ColorDB.triplet_to_rrggbb(t)
|
||
chips.append(rrggbb)
|
||
tred, tgreen, tblue = t
|
||
if tred <= red and tgreen <= green and tblue <= blue:
|
||
chip = i
|
||
i = i + 1
|
||
# call the raw tcl script
|
||
colors = SPACE.join(chips)
|
||
tk.eval('setcolor %s {%s}' % (self.__canvas._w, colors))
|
||
# move the arrows around
|
||
self.__trackarrow(chip, (red, green, blue))
|
||
|
||
def set(self, label, generator):
|
||
self.__canvas.itemconfigure(self.__label, text=label)
|
||
self.__generator = generator
|
||
|
||
|
||
class StripViewer:
|
||
def __init__(self, switchboard, master=None):
|
||
self.__sb = switchboard
|
||
optiondb = switchboard.optiondb()
|
||
# create a frame inside the master.
|
||
frame = Frame(master, relief=RAISED, borderwidth=1)
|
||
frame.grid(row=1, column=0, columnspan=2, sticky='NSEW')
|
||
# create the options to be used later
|
||
uwd = self.__uwdvar = BooleanVar()
|
||
uwd.set(optiondb.get('UPWHILEDRAG', 0))
|
||
hexp = self.__hexpvar = BooleanVar()
|
||
hexp.set(optiondb.get('HEXSTRIP', 0))
|
||
# create the red, green, blue strips inside their own frame
|
||
frame1 = Frame(frame)
|
||
frame1.pack(expand=YES, fill=BOTH)
|
||
self.__reds = StripWidget(switchboard, frame1,
|
||
generator=constant_cyan_generator,
|
||
axis=0,
|
||
label='Red Variations',
|
||
uwdvar=uwd, hexvar=hexp)
|
||
|
||
self.__greens = StripWidget(switchboard, frame1,
|
||
generator=constant_magenta_generator,
|
||
axis=1,
|
||
label='Green Variations',
|
||
uwdvar=uwd, hexvar=hexp)
|
||
|
||
self.__blues = StripWidget(switchboard, frame1,
|
||
generator=constant_yellow_generator,
|
||
axis=2,
|
||
label='Blue Variations',
|
||
uwdvar=uwd, hexvar=hexp)
|
||
|
||
# create a frame to contain the controls
|
||
frame2 = Frame(frame)
|
||
frame2.pack(expand=YES, fill=BOTH)
|
||
frame2.columnconfigure(0, weight=20)
|
||
frame2.columnconfigure(2, weight=20)
|
||
|
||
padx = 8
|
||
|
||
# create the black button
|
||
blackbtn = Button(frame2,
|
||
text='Black',
|
||
command=self.__toblack)
|
||
blackbtn.grid(row=0, column=0, rowspan=2, sticky=W, padx=padx)
|
||
|
||
# create the controls
|
||
uwdbtn = Checkbutton(frame2,
|
||
text='Update while dragging',
|
||
variable=uwd)
|
||
uwdbtn.grid(row=0, column=1, sticky=W)
|
||
hexbtn = Checkbutton(frame2,
|
||
text='Hexadecimal',
|
||
variable=hexp,
|
||
command=self.__togglehex)
|
||
hexbtn.grid(row=1, column=1, sticky=W)
|
||
|
||
# XXX: ignore this feature for now; it doesn't work quite right yet
|
||
|
||
## gentypevar = self.__gentypevar = IntVar()
|
||
## self.__variations = Radiobutton(frame,
|
||
## text='Variations',
|
||
## variable=gentypevar,
|
||
## value=0,
|
||
## command=self.__togglegentype)
|
||
## self.__variations.grid(row=0, column=1, sticky=W)
|
||
## self.__constants = Radiobutton(frame,
|
||
## text='Constants',
|
||
## variable=gentypevar,
|
||
## value=1,
|
||
## command=self.__togglegentype)
|
||
## self.__constants.grid(row=1, column=1, sticky=W)
|
||
|
||
# create the white button
|
||
whitebtn = Button(frame2,
|
||
text='White',
|
||
command=self.__towhite)
|
||
whitebtn.grid(row=0, column=2, rowspan=2, sticky=E, padx=padx)
|
||
|
||
def update_yourself(self, red, green, blue):
|
||
self.__reds.update_yourself(red, green, blue)
|
||
self.__greens.update_yourself(red, green, blue)
|
||
self.__blues.update_yourself(red, green, blue)
|
||
|
||
def __togglehex(self, event=None):
|
||
red, green, blue = self.__sb.current_rgb()
|
||
self.update_yourself(red, green, blue)
|
||
|
||
## def __togglegentype(self, event=None):
|
||
## which = self.__gentypevar.get()
|
||
## if which == 0:
|
||
## self.__reds.set(label='Red Variations',
|
||
## generator=constant_cyan_generator)
|
||
## self.__greens.set(label='Green Variations',
|
||
## generator=constant_magenta_generator)
|
||
## self.__blues.set(label='Blue Variations',
|
||
## generator=constant_yellow_generator)
|
||
## elif which == 1:
|
||
## self.__reds.set(label='Red Constant',
|
||
## generator=constant_red_generator)
|
||
## self.__greens.set(label='Green Constant',
|
||
## generator=constant_green_generator)
|
||
## self.__blues.set(label='Blue Constant',
|
||
## generator=constant_blue_generator)
|
||
## else:
|
||
## assert 0
|
||
## self.__sb.update_views_current()
|
||
|
||
def __toblack(self, event=None):
|
||
self.__sb.update_views(0, 0, 0)
|
||
|
||
def __towhite(self, event=None):
|
||
self.__sb.update_views(255, 255, 255)
|
||
|
||
def save_options(self, optiondb):
|
||
optiondb['UPWHILEDRAG'] = self.__uwdvar.get()
|
||
optiondb['HEXSTRIP'] = self.__hexpvar.get()
|