| """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 |
|
|
| |
| |
| |
| TCLPROC = '''\ |
| proc setcolor {canv colors} { |
| set i 1 |
| foreach c $colors { |
| $canv itemconfigure $i -fill $c -outline $c |
| incr i |
| } |
| } |
| ''' |
|
|
| |
| 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 |
|
|
| |
| def constant_red_generator(numchips, red, green, blue): |
| seq = constant(numchips) |
| return list(zip([red] * numchips, seq, seq)) |
|
|
| |
| def constant_green_generator(numchips, red, green, blue): |
| seq = constant(numchips) |
| return list(zip(seq, [green] * numchips, seq)) |
|
|
| |
| def constant_blue_generator(numchips, red, green, blue): |
| seq = constant(numchips) |
| return list(zip(seq, seq, [blue] * numchips)) |
|
|
| |
| def constant_cyan_generator(numchips, red, green, blue): |
| seq = constant(numchips) |
| return list(zip(seq, [green] * numchips, [blue] * numchips)) |
|
|
| |
| def constant_magenta_generator(numchips, red, green, blue): |
| seq = constant(numchips) |
| return list(zip([red] * numchips, seq, [blue] * numchips)) |
|
|
| |
| 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, |
| 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): |
| |
| self.__generator = generator |
| self.__axis = axis |
| self.__numchips = numchips |
| assert self.__axis in (0, 1, 2) |
| self.__uwd = uwdvar |
| self.__hexp = hexvar |
| |
| self.__lastchip = None |
| self.__sb = switchboard |
|
|
| canvaswidth = numchips * (chipwidth + 1) |
| canvasheight = chipheight + 43 |
|
|
| |
| canvas = self.__canvas = Canvas(master, |
| width=canvaswidth, |
| height=canvasheight, |
| |
| |
| ) |
|
|
| canvas.pack() |
| canvas.bind('<ButtonPress-1>', self.__select_chip) |
| canvas.bind('<ButtonRelease-1>', self.__select_chip) |
| canvas.bind('<B1-Motion>', self.__select_chip) |
|
|
| |
| |
| canvas.tk.eval(TCLPROC) |
|
|
| |
| 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 |
| chips.append(color) |
|
|
| |
| self.__label = canvas.create_text( |
| 3, y + chipheight + 8, |
| text=label, |
| anchor=W) |
|
|
| |
| 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 |
|
|
| |
| |
| 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()): |
| |
| self.__sb.update_views(red, green, blue) |
| else: |
| |
| self.__trackarrow(chip[0], (red, green, blue)) |
|
|
| def __trackarrow(self, chip, rgbtuple): |
| |
| if self.__lastchip is not None: |
| color = self.__canvas.itemcget(self.__lastchip, 'fill') |
| self.__canvas.itemconfigure(self.__lastchip, outline=color) |
| self.__lastchip = chip |
| |
| coloraxis = rgbtuple[self.__axis] |
| if self.__hexp.get(): |
| |
| text = hex(coloraxis) |
| else: |
| |
| text = repr(coloraxis) |
| |
| if coloraxis <= 128: |
| |
| self.__leftarrow.set_text(text) |
| self.__leftarrow.move_to(self.__arrow_x(chip-1)) |
| self.__rightarrow.move_to(-100) |
| else: |
| |
| self.__rightarrow.set_text(text) |
| self.__rightarrow.move_to(self.__arrow_x(chip-1)) |
| self.__leftarrow.move_to(-100) |
| |
| 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 |
| |
| 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 |
| |
| colors = SPACE.join(chips) |
| tk.eval('setcolor %s {%s}' % (self.__canvas._w, colors)) |
| |
| 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() |
| |
| frame = Frame(master, relief=RAISED, borderwidth=1) |
| frame.grid(row=1, column=0, columnspan=2, sticky='NSEW') |
| |
| uwd = self.__uwdvar = BooleanVar() |
| uwd.set(optiondb.get('UPWHILEDRAG', 0)) |
| hexp = self.__hexpvar = BooleanVar() |
| hexp.set(optiondb.get('HEXSTRIP', 0)) |
| |
| 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) |
|
|
| |
| frame2 = Frame(frame) |
| frame2.pack(expand=YES, fill=BOTH) |
| frame2.columnconfigure(0, weight=20) |
| frame2.columnconfigure(2, weight=20) |
|
|
| padx = 8 |
|
|
| |
| blackbtn = Button(frame2, |
| text='Black', |
| command=self.__toblack) |
| blackbtn.grid(row=0, column=0, rowspan=2, sticky=W, padx=padx) |
|
|
| |
| 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) |
|
|
| |
| 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 __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() |
|
|