clrpick.tcl
clrpick.tcl

clrpick.tcl

  • clrpick.tcl
    • color::
    • GetOptions color::GetOptions : Gets current settings of options. Returns a list of selected color options: moveall - value of moveall option fgcolor - value of foreground (inverse) color bgcolor - value of background (selected) color
    • ShowAtXY color::ShowAtXY : Shows the picker at X,Y coordinates, checking the coordinates against the screen, incl. decors win - window's path x - window's X coordinate y - window's Y coordinate
    • InitValues color::InitValues
    • Config color::Config
    • BuildDialog color::BuildDialog
    • SetRGBValue color::SetRGBValue
    • XToRgb color::XToRgb
    • RgbToX color::RgbToX
    • DrawColorScale color::DrawColorScale
    • CreateSelector color::CreateSelector
    • InvertBg color::InvertBg
    • ReverseFinalFg color::ReverseFinalFg
    • RedrawFinalColor color::RedrawFinalColor
    • RedrawColorBars color::RedrawColorBars
    • DrawOthers color::DrawOthers
    • StickSelectors color::StickSelectors
    • LittleSwatch color::LittleSwatch
    • StartMove color::StartMove
    • MoveSelector color::MoveSelector
    • ReleaseMouse color::ReleaseMouse
    • ResizeColorBars color::ResizeColorBars
    • HandleSelEntry color::HandleSelEntry
    • HandleRGBEntry color::HandleRGBEntry
    • EnterLeaveColorBar color::EnterLeaveColorBar
    • OkCmd0 color::OkCmd0
    • Loupe color::Loupe
    • OkCmd color::OkCmd
    • CancelCmd color::CancelCmd
  • EOF
# clrpick.tcl --
#
#    Color selection dialog for platforms that do not support a
#    standard color selection dialog.
#
# Copyright (c) 1996 Sun Microsystems, Inc.
#
# See the file "license.terms" for information on usage and redistribution
# of this file, and for a DISCLAIMER OF ALL WARRANTIES.
#
# ToDo:
#
#    (1): Find out how many free colors are left in the colormap and
#         don't allocate too many colors.
#    (2): Implement HSV color selection.
#

# Make sure namespaces exist
namespace eval ::tk {}
namespace eval ::tk::dialog {}
namespace eval ::tk::dialog::color {
  namespace import ::tk::msgcat::*
}
::msgcat::mcload [file join [file dirname [info script]] msgs]

# ::tk::dialog::color:: --
#
#    Create a color dialog and let the user choose a color. This function
#    should not be called directly. It is called by the tk_chooseColor
#    function when a native color selector widget does not exist
#
proc ::tk::dialog::color:: {args} {

  variable ::tk::Priv
  set dataName __tk__color
  upvar ::tk::dialog::color::$dataName data
  set w .$dataName

  # The lines variables track the start and end indices of the line
  # elements in the colorbar canvases.
  set data(lines,red,start)   0
  set data(lines,red,last)   -1
  set data(lines,green,start) 0
  set data(lines,green,last) -1
  set data(lines,blue,start)  0
  set data(lines,blue,last)  -1

  # This is the actual number of lines that are drawn in each color strip.
  # Note that the bars may be of any width.
  # However, NUM_COLORBARS must be a number that evenly divides 256.
  # Such as 256, 128, 64, etc.
  set data(NUM_COLORBARS) 32 ;#16

  # BARS_WIDTH is the number of pixels wide the color bar portion of the
  # canvas is. This number must be a multiple of NUM_COLORBARS
  set data(BARS_WIDTH) [expr $data(NUM_COLORBARS)*14]

  # PLGN_WIDTH is the number of pixels wide of the triangular selection
  # polygon. This also results in the definition of the padding on the
  # left and right sides which is half of PLGN_WIDTH. Make this number even.
  set data(PLGN_HEIGHT) 16 ;#10

  # PLGN_HEIGHT is the height of the selection polygon and the height of the
  # selection rectangle at the bottom of the color bar. No restrictions.
  set data(PLGN_WIDTH) 12 ;#10

  Config $dataName $args
  InitValues $dataName

  set sc [winfo screen $data(-parent)]
  set winExists [winfo exists $w]
  if {!$winExists || $sc ne [winfo screen $w]} {
    if {$winExists} {
      destroy $w
    }
    toplevel $w -class TkColorDialog -screen $sc
    if {[tk windowingsystem] eq "x11"} {wm attributes $w -type dialog}
    BuildDialog $w
  }

  # Dialog boxes should be transient with respect to their parent,
  # so that they will always stay on top of their parent window.  However,
  # some window managers will create the window as withdrawn if the parent
  # window is withdrawn or iconified.  Combined with the grab we put on the
  # window, this can hang the entire application.  Therefore we only make
  # the dialog transient if the parent is viewable.

  if {[winfo viewable [winfo toplevel $data(-parent)]] } {
    wm transient $w $data(-parent)
  }

  # 5. Withdraw the window, then update all the geometry information
  # so we know how big it wants to be, then center the window in the
  # display (Motif style) and de-iconify it.

  wm title $w $data(-title)
  if {[info exists data(-geometry)] && $data(-geometry) ne {-}} {
    if {[set geo $data(-geometry)] eq {}} {
      set X [winfo rootx $data(-parent)]
      set Y [expr {[winfo rooty $data(-parent)]+32}]
      set geo +$X+$Y
    } elseif {[string match pointer+*+* $geo]} {  ;# e.g.  pointer+10+10
      lassign [split $geo +] -> X Y
      set geo +[expr {[winfo pointerx .]+$X}]+[expr {[winfo pointery .]+$Y}]
    }
    lassign [lrange [split $geo +] end-1 end] X Y
    after idle [list after 0 [list ::tk::dialog::color::ShowAtXY $w $X $Y]]
    unset data(-geometry)
  } else {
    ::tk::PlaceWindow $w widget $data(-parent)
    wm deiconify $w
  }
  if {$data(-ontop)} {wm attributes $w -topmost 1}

  # 6. Set a grab and claim the focus too.

  ::tk::SetFocusGrab $w $data(okBtn)

  # 7. Wait for the user to respond, then restore the focus and
  # return the index of the selected button.  Restore the focus
  # before deleting the window, since otherwise the window manager
  # may take the focus away so we can't redirect it.  Finally,
  # restore any grab that was in effect.

  vwait ::tk::Priv(selectColor)
  set result $Priv(selectColor)
  set ::tk::dialog::color::fgsel [$data(finalCanvas) cget -text]
  set ::tk::dialog::color::bgsel $result
  ::tk::RestoreFocusGrab $w $data(okBtn)
  unset data

  # for backward compatibility:
  # if any of new options (-tonemoves, -moveall) present
  # then return a list: color + values of these options
  if {[lsearch -regexp $args "(\[-\]tonemoves|\[-\]moveall)"] > -1} {
    return [list $result $::tk::dialog::color::moveall \
      $::tk::dialog::color::tonemoves]
  }
  return $result
}

proc ::tk::dialog::color::GetOptions {} {
  # Gets current settings of options.
  # Returns a list of selected color options:
  #   moveall - value of moveall option
  #   fgcolor - value of foreground (inverse) color
  #   bgcolor - value of background (selected) color

  return [list $::tk::dialog::color::moveall $::tk::dialog::color::fgsel $::tk::dialog::color::bgsel]
}

proc ::tk::dialog::color::ShowAtXY {win x y} {
  # Shows the picker at X,Y coordinates, checking the coordinates against the screen, incl. decors
  #   win - window's path
  #   x - window's X coordinate
  #   y - window's Y coordinate

  set w [winfo reqwidth $win]
  set h [winfo reqheight $win]
  set scrw [expr {[winfo vrootwidth .] - 12}]
  set scrh [expr {[winfo vrootheight .] - 36}]
  if {($x + $w) > $scrw } {
    set x [expr {$scrw - $w}]
  }
  if {($y + $h) > $scrh } {
    set y [expr {$scrh - $h}]
  }
  if {![string match -* $x]} {set x +[string trimleft $x +]}
  if {![string match -* $y]} {set y +[string trimleft $y +]}
  wm geometry $win $x$y
  wm deiconify $win
}

# ::tk::dialog::color::InitValues --
#
#    Get called during initialization or when user resets NUM_COLORBARS
#
proc ::tk::dialog::color::InitValues {dataName} {
  upvar ::tk::dialog::color::$dataName data

  # IntensityIncr is the difference in color intensity between a colorbar
  # and its neighbors.
  set data(intensityIncr) [expr {256 / $data(NUM_COLORBARS)}]

  # ColorbarWidth is the width of each colorbar
  set data(colorbarWidth) [expr {$data(BARS_WIDTH) / $data(NUM_COLORBARS)}]

  # Indent is the width of the space at the left and right side of the
  # colorbar. It is always half the selector polygon width, because the
  # polygon extends into the space.
  set data(indent) [expr {$data(PLGN_WIDTH) / 2}]

  set data(colorPad) 2
  set data(selPad)   [expr {$data(PLGN_WIDTH) / 2}]

  #
  # minX is the x coordinate of the first colorbar
  #
  set data(minX) $data(indent)

  #
  # maxX is the x coordinate of the last colorbar
  #
  set data(maxX) [expr {$data(BARS_WIDTH) + $data(indent)-1}]

  #
  # canvasWidth is the width of the entire canvas, including the indents
  #
  set data(canvasWidth) [expr {$data(BARS_WIDTH) + $data(PLGN_WIDTH)}]

  # Set the initial color, specified by -initialcolor, or the
  # color chosen by the user the last time.
  set data(selection) $data(-initialcolor)
  set data(finalColor)  $data(-initialcolor)
  set rgb [winfo rgb . $data(selection)]

  set data(red,intensity)   [expr {[lindex $rgb 0]/0x100}]
  set data(green,intensity) [expr {[lindex $rgb 1]/0x100}]
  set data(blue,intensity)  [expr {[lindex $rgb 2]/0x100}]
}

# ::tk::dialog::color::Config  --
#
#    Parses the command line arguments to tk_chooseColor
#
proc ::tk::dialog::color::Config {dataName argList} {
  variable ::tk::Priv
  upvar ::tk::dialog::color::$dataName data

  # 1: the configuration specs
  #
  if {[info exists Priv(selectColor)] && $Priv(selectColor) ne ""} {
    set defaultColor $Priv(selectColor)
  } else {
    set defaultColor [. cget -background]
  }

  set specs [list \
    [list -moveall {} {} 0] \
    [list -tonemoves {} {} 1] \
    [list -geometry {} {} -] \
    [list -initialcolor {} {} $defaultColor] \
    [list -parent {} {} .] \
    [list -title {} {} [mc "Color"]] \
    [list -inifile {} {} {}] \
    [list -ontop {} {} 0] \
    ]

  # 2: parse the arguments
  #
  tclParseConfigSpec ::tk::dialog::color::$dataName $specs "" $argList

  if {$data(-title) eq ""} {
    set data(-title) " "
  }
  if {[catch {winfo rgb . $data(-initialcolor)} err]} {
    return -code error -errorcode \
      [list TK LOOKUP COLOR $data(-initialcolor)] $err
  }

  if {![winfo exists $data(-parent)]} {
    return -code error -errorcode [list TK LOOKUP WINDOW $data(-parent)] \
      "bad window path name \"$data(-parent)\""
  }
}

# ::tk::dialog::color::BuildDialog --
#
#    Build the dialog.
#
proc ::tk::dialog::color::BuildDialog {w} {

  upvar ::tk::dialog::color::[winfo name $w] data

  wm withdraw $w

  # let the dialog's background = the app's
  $w configure -background [ttk::style lookup "." -background]

  # TopFrame contains the color strips and the color selection
  #
  set topFrame [ttk::frame $w.top -relief flat]

  # StripsFrame contains the colorstrips and the individual RGB entries
  set stripsFrame [ttk::frame $topFrame.colorStrip]

  set maxWidth [::tk::mcmaxamp &Red &Green &Blue]
  set maxWidth [expr {$maxWidth<6 ? 6 : $maxWidth}]
  set colorList {
    red   "Red"
    green "Green"
    blue  "Blue"
  }
  foreach {color l} $colorList {
    # each f frame contains an [R|G|B] entry and the equiv. color strip.
    set f [ttk::frame $stripsFrame.$color]

    # The box frame contains the label and entry widget for an [R|G|B]
    set box [ttk::frame $f.box]
    ttk::label $box.label {*}[ttk::style config TLabel] -text "[mc $l]:" \
      -width $maxWidth -anchor ne
    bind $box.label <<AltUnderlined>> [list focus $box.entry]

    ttk::entry $box.entry -textvariable \
      ::tk::dialog::color::[winfo name $w]($color,intensity) -width 4
    pack $box.label -side left -fill both -pady 6
    pack $box.entry -side left -anchor n -pady 4 -padx 2
    pack $box -side left -fill both

    set height [expr { [winfo reqheight $box.entry] + 1}]

    catch {::apave::obj untouchWidgets $f.sel}
    canvas $f.color -height [expr $height-2] \
      -width $data(BARS_WIDTH) -relief sunken -bd 3
    canvas $f.sel -height [expr $data(PLGN_HEIGHT)+2] \
      -width $data(canvasWidth) -highlightthickness 0 \
      -bg [ttk::style lookup . -background]
    pack $f.color -expand yes -fill both
    pack $f.sel -expand yes -fill both

    pack $f -side top -fill x -pady 1

    set data($color,entry) $box.entry
    set data($color,col) $f.color
    set data($color,sel) $f.sel

    set data(cbc1) red
    set data(cbc2) [ttk::style lookup . -bordercolor]
    if {$data(cbc2) eq {}} {set data(cbc2) grey}
    bind $data($color,col) <Configure> \
      [list tk::dialog::color::DrawColorScale $w $color 1]
    bind $data($color,col) <Enter> \
      [list tk::dialog::color::EnterLeaveColorBar $w $color $data(cbc1)]
    bind $data($color,col) <Leave> \
      [list tk::dialog::color::EnterLeaveColorBar $w $color $data(cbc2)]

    bind $data($color,sel) <Enter> \
      [list tk::dialog::color::EnterLeaveColorBar $w $color $data(cbc1)]
    bind $data($color,sel) <Leave> \
      [list tk::dialog::color::EnterLeaveColorBar $w $color $data(cbc2)]

    bind $box.entry <Return> [list tk::dialog::color::HandleRGBEntry $w]
  }

  set mainentry ::tk::dialog::color::[winfo name $w](selection)

  # for setting the mutual move of selectors:
  set ::tk::dialog::color::tonemoves $data(-tonemoves)
  set ::tk::dialog::color::moveall $data(-moveall)
  set ::tk::dialog::color::fgsel {}
  set ::tk::dialog::color::bgsel {}
  pack [ttk::checkbutton $stripsFrame.moveallColorSel -text "[mc {Move all}]    " \
    -variable ::tk::dialog::color::moveall -command \
    "::tk::dialog::color::StickSelectors $w 1"] -padx 2 -side left
  set foc [ttk::frame $stripsFrame.foc]
  foreach l { 804000 004000 004080 008080 800080 808000 \
              ffff00 ff00ff 00ffff 0000ff 00ff00 ff0000} {
    set fl [frame $stripsFrame.foc.$l -bd 3 -relief raised -height 24 -width 24]
    $fl configure -bg #$l
    bind $fl <Button-1> [list tk::dialog::color::LittleSwatch $w $mainentry #$l]
    pack $fl -expand yes -anchor nw -padx 2 -side left
  }
  pack $foc -expand yes -anchor nw -fill both -pady 4 -side right

  pack $stripsFrame -side left -fill both -pady 4

  # The selFrame contains a frame that demonstrates the currently
  # selected color
  #
  set selFrame [ttk::frame $topFrame.sel]
  set lab [ttk::label $selFrame.lab {*}[ttk::style config TLabel] \
    -text [mc "Selection:"]]
  set ent [ttk::entry $selFrame.ent -textvariable $mainentry -width 14]
  set f1  [ttk::frame $selFrame.f1 -relief sunken]
  set data(finalCanvas) [label $f1.demo -bd 1 -width 6 -height 6]

  pack $lab -side top -padx 4 -anchor sw
  pack $ent -side top -padx 4 -pady 2 -anchor n
  pack $f1 -expand yes -anchor nw -fill both -padx 4 -pady 4
  pack $data(finalCanvas) -expand yes -fill both -anchor s

  bind $ent <Return> [list tk::dialog::color::HandleSelEntry $w]
  bind $data(finalCanvas) <Button-1> [list tk::dialog::color::ReverseFinalFg $w]

  pack $selFrame -side left -fill none -anchor nw
  pack $topFrame -side top -expand yes -fill both -anchor nw

  # the botFrame frame contains the buttons
  #
  set botFrame [ttk::frame $w.bot -relief raised]

  ttk::button $botFrame.ok0 -text [mc "From clipboard"] \
      -command [list tk::dialog::color::OkCmd0 $w]
  if {[set aloupe [info commands ::aloupe::run]] ne ""} {
    ttk::button $botFrame.loupe -text [mc "Loupe"] \
        -command [list tk::dialog::color::Loupe $w]
  }
  ttk::button $botFrame.ok -text [string map {& ""} [mc "OK"]] \
      -command [list tk::dialog::color::OkCmd $w]
  ttk::button $botFrame.cancel -text [string map {& ""} [mc "Cancel"]] \
      -command [list tk::dialog::color::CancelCmd $w]

  if {![catch {set clb [clipboard get]}] && \
  [regexp {^(#*\d{3,6}|[[:alnum:] ]{3,})$} $clb]} {
    set data(okBtn) $botFrame.ok0
  } else {
    set data(okBtn) $botFrame.ok
  }
  set data(cancelBtn)  $botFrame.cancel

  if {$aloupe eq ""} {
    grid $botFrame.ok0 x x x $botFrame.ok $botFrame.cancel -sticky ew
  } else {
    grid $botFrame.ok0 $botFrame.loupe x x x $botFrame.ok $botFrame.cancel -sticky ew -padx 2
  }
  grid configure $botFrame.ok $botFrame.cancel -padx 2 -pady 4
  grid columnconfigure $botFrame 2 -weight 2 -uniform space
  pack $botFrame -side bottom -fill x

  # Accelerator bindings
  bind $lab <<AltUnderlined>> [list focus $ent]
  bind $w <KeyPress-Escape> [list $data(cancelBtn) invoke]
  bind $w <Alt-Key> [list tk::AltKeyInDialog $w %A]

  wm protocol $w WM_DELETE_WINDOW [list tk::dialog::color::CancelCmd $w]
  bind $lab <Destroy> [list tk::dialog::color::CancelCmd $w]

  ::tk::dialog::color::StickSelectors $w 1
  wm resizable $w 0 0
}

# ::tk::dialog::color::SetRGBValue --
#
#    Sets the current selection of the dialog box
#
proc ::tk::dialog::color::SetRGBValue {w color} {
  upvar ::tk::dialog::color::[winfo name $w] data

  set data(red,intensity)   [lindex $color 0]
  set data(green,intensity) [lindex $color 1]
  set data(blue,intensity)  [lindex $color 2]

  RedrawColorBars $w all

  # Now compute the new x value of each colorbars pointer polygon
  foreach color {red green blue} {
    set x [RgbToX $w $data($color,intensity)]
    MoveSelector $w $data($color,sel) $color $x 0
  }
}

# ::tk::dialog::color::XToRgb --
#
#    Converts a screen coordinate to intensity
#
proc ::tk::dialog::color::XToRgb {w x} {
  upvar ::tk::dialog::color::[winfo name $w] data

  set x [expr {round(1.0 * ($x * $data(intensityIncr))/ $data(colorbarWidth))}]
  if {$x > 255} {set x 255}
  return $x
}

# ::tk::dialog::color::RgbToX
#
#    Converts an intensity to screen coordinate.
#
proc ::tk::dialog::color::RgbToX {w color} {
  upvar ::tk::dialog::color::[winfo name $w] data
  return [expr {($color * $data(colorbarWidth)/ $data(intensityIncr))}]
}

# ::tk::dialog::color::DrawColorScale --
#
#    Draw color scale is called whenever the size of one of the color
#    scale canvases is changed.
#
proc ::tk::dialog::color::DrawColorScale {w c {create 0}} {
  upvar ::tk::dialog::color::[winfo name $w] data

  # col: color bar canvas
  # sel: selector canvas
  set col $data($c,col)
  set sel $data($c,sel)

  # First handle the case that we are creating everything for the first time.
  if {$create} {
    # First remove all the lines that already exist.
    if { $data(lines,$c,last) > $data(lines,$c,start)} {
      for {set i $data(lines,$c,start)} {$i <= $data(lines,$c,last)} {incr i} {
        $sel delete $i
      }
    }
    # Delete the selector if it exists
    if {[info exists data($c,index)]} {
      $sel delete $data($c,index)
    }

    # Draw the selection polygons
    CreateSelector $w $sel $c
    $sel bind $data($c,index) <ButtonPress-1> \
      [list tk::dialog::color::StartMove $w $sel $c %x $data(selPad) 1]
    $sel bind $data($c,index) <B1-Motion> \
      [list tk::dialog::color::MoveSelector $w $sel $c %x $data(selPad)]
    $sel bind $data($c,index) <ButtonRelease-1> \
      [list tk::dialog::color::ReleaseMouse $w $sel $c %x $data(selPad)]

    set height [winfo height $col]
    # Create an invisible region under the colorstrip to catch mouse clicks
    # that aren't on the selector.
    set data($c,clickRegion) [$sel create rectangle 0 0 \
      $data(canvasWidth) $height -fill {} -outline {}]

    bind $col <ButtonPress-1> \
      [list tk::dialog::color::StartMove $w $sel $c %x $data(colorPad)]
    bind $col <B1-Motion> \
      [list tk::dialog::color::MoveSelector $w $sel $c %x $data(colorPad)]
    bind $col <ButtonRelease-1> \
      [list tk::dialog::color::ReleaseMouse $w $sel $c %x $data(colorPad)]

    $sel bind $data($c,clickRegion) <ButtonPress-1> \
      [list tk::dialog::color::StartMove $w $sel $c %x $data(selPad)]
    $sel bind $data($c,clickRegion) <B1-Motion> \
      [list tk::dialog::color::MoveSelector $w $sel $c %x $data(selPad)]
    $sel bind $data($c,clickRegion) <ButtonRelease-1> \
      [list tk::dialog::color::ReleaseMouse $w $sel $c %x $data(selPad)]
  } else {
    # l is the canvas index of the first colorbar.
    set l $data(lines,$c,start)
  }

  # Draw the color bars.
  set highlightW [expr {[$col cget -highlightthickness] + [$col cget -bd]}]
  for {set i 0} { $i < $data(NUM_COLORBARS)} { incr i} {
    set intensity [expr {$i * $data(intensityIncr)}]
    set startx [expr {$i * $data(colorbarWidth) + $highlightW}]
    if {$c eq "red"} {
      set color [format "#%02x%02x%02x" \
        $intensity $data(green,intensity) $data(blue,intensity)]
    } elseif {$c eq "green"} {
      set color [format "#%02x%02x%02x" \
        $data(red,intensity) $intensity $data(blue,intensity)]
    } else {
      set color [format "#%02x%02x%02x" \
        $data(red,intensity) $data(green,intensity) $intensity]
    }

    if {$create} {
      incr startx 4
      set index [$col create rect $startx $highlightW \
        [expr {$startx +$data(colorbarWidth)}] \
        [expr {[winfo height $col] + $highlightW}] \
        -fill $color -outline $color]
    } else {
      $col itemconfigure $l -fill $color -outline $color
      incr l
    }
  }
  $sel raise $data($c,index)

  if {$create} {
    set data(lines,$c,last) $index
    set data(lines,$c,start) [expr {$index - $data(NUM_COLORBARS) + 1}]
  }

  RedrawFinalColor $w
}

# ::tk::dialog::color::CreateSelector --
#
#    Creates and draws the selector polygon at the position
#    $data($c,intensity).
#
proc ::tk::dialog::color::CreateSelector {w sel c } {
  upvar ::tk::dialog::color::[winfo name $w] data
  set data($c,index) [$sel create polygon \
    0 $data(PLGN_HEIGHT) \
    $data(PLGN_WIDTH) $data(PLGN_HEIGHT) \
    $data(indent) 0]
  set data($c,x) [RgbToX $w $data($c,intensity)]
  $sel move $data($c,index) $data($c,x) 0
  EnterLeaveColorBar $w $c $data(cbc2)
}

# Gets fg color (white/black) for a bg color.
#  color - bg color

proc ::tk::dialog::color::InvertBg {color} {
  lassign [winfo rgb . $color] r g b
  if {($r%256+$b%256)<15 && ($g%256)>180} {
    set res black
  } elseif {$r+1.5*$g+0.5*$b > 100000} {
    set res black
  } else {
    set res white
  }
  return $res
}

proc ::tk::dialog::color::ReverseFinalFg {w} {
  upvar ::tk::dialog::color::[winfo name $w] data
  set testcolors [list 000000 FFFFFF 804000 004000 004080 008080 800080 808000 \
    ffff00 ff00ff 00ffff 0000ff 00ff00 ff0000]
  if {[incr data(idxFinalColor)]>=[llength $testcolors]} {set data(idxFinalColor) 0}
  set fg #[lindex $testcolors $data(idxFinalColor)]
  $data(finalCanvas) configure -fg $fg -text $fg
}

# ::tk::dialog::color::RedrawFinalColor
#
#    Combines the intensities of the three colors into the final color
#
proc ::tk::dialog::color::RedrawFinalColor {w} {
  upvar ::tk::dialog::color::[winfo name $w] data

  set color [format "#%02x%02x%02x" $data(red,intensity) \
    $data(green,intensity) $data(blue,intensity)]

  #$data(finalCanvas) configure -bg $color
  set fg [InvertBg $color]
  $data(finalCanvas) configure -bg $color -fg $fg -text $fg -font {-weight bold}
  set data(idxFinalColor) [expr {$fg ne "#000000"}]
  set data(finalColor) $color
  set data(selection) $color
  set data(finalRGB) [list \
    $data(red,intensity) \
    $data(green,intensity) \
    $data(blue,intensity)]
}

# ::tk::dialog::color::RedrawColorBars --
#
# Only redraws the colors on the color strips that were not manipulated.
# Params: color of colorstrip that changed. If color is not [red|green|blue]
#         Then all colorstrips will be updated
#
proc ::tk::dialog::color::RedrawColorBars {w colorChanged} {
  upvar ::tk::dialog::color::[winfo name $w] data

  switch $colorChanged {
    red {
      DrawColorScale $w green
      DrawColorScale $w blue
    }
    green {
      DrawColorScale $w red
      DrawColorScale $w blue
    }
    blue {
      DrawColorScale $w red
      DrawColorScale $w green
    }
    default {
      DrawColorScale $w red
      DrawColorScale $w green
      DrawColorScale $w blue
    }
  }
  RedrawFinalColor $w
}

# ::tk::dialog::color::DrawOthers
#
# Draws other selectors if the "Move all" mode is set
#
proc ::tk::dialog::color::DrawOthers {w color} {
  upvar ::tk::dialog::color::[winfo name $w] data
  if {$::tk::dialog::color::moveall} {
    set dif1 [expr {$data($color,intensity) - $data($color,stickypos)}]
    foreach c {red green blue} {
      if {$c!=$color && [set pos1 $data($c,stickypos)] > -1} {
        set intens [expr {$pos1 + $dif1}]
        set data($c,intensity) [expr {max(0,min(255,$intens))}]
        RedrawColorBars $w $c
      }
    }
  }
}

# ::tk::dialog::color::StickSelectors
#
# Saves (sticks) all selectors' positions
# in order to restore (unstick) them if possible
#
proc ::tk::dialog::color::StickSelectors {w {clear 0}} {
  upvar ::tk::dialog::color::[winfo name $w] data
  if {$::tk::dialog::color::moveall} {
    foreach c {red green blue} {
      set data($c,stickypos) [expr {$clear ? -1 : $data($c,intensity)}]
    }
  }
}

# ::tk::dialog::color::LittleSwatch
#
# Swatching a color
#
proc ::tk::dialog::color::LittleSwatch {w e c} {
  upvar ::tk::dialog::color::[winfo name $w] data
  set $e $c
  HandleSelEntry $w
}

#----------------------------------------------------------------------
#            Event handlers
#----------------------------------------------------------------------

# ::tk::dialog::color::StartMove --
#
#    Handles a mousedown button event over the selector polygon.
#    Adds the bindings for moving the mouse while the button is
#    pressed.  Sets the binding for the button-release event.
#
# Params: sel is the selector canvas window, color is the color of the strip.
#
proc ::tk::dialog::color::StartMove {w sel color x delta {dontMove 0}} {
  upvar ::tk::dialog::color::[winfo name $w] data
  StickSelectors $w
  if {!$dontMove} {
    MoveSelector $w $sel $color $x $delta
  }
}

# ::tk::dialog::color::MoveSelector --
#
# Moves the polygon selector so that its middle point has the same
# x value as the specified x. If x is outside the bounds [0,255],
# the selector is set to the closest endpoint.
#
# Params: sel is the selector canvas, c is [red|green|blue]
#         x is a x-coordinate.
#
proc ::tk::dialog::color::MoveSelector {w sel color x delta} {
  upvar ::tk::dialog::color::[winfo name $w] data

  incr x -$delta

  if { $x < 0 } {
    set x 0
  } elseif { $x > $data(BARS_WIDTH)} {
    set x $data(BARS_WIDTH)
  }
  set diff [expr {$x - $data($color,x) + 1}]
  $sel move $data($color,index) $diff 0
  set data($color,x) [expr {$data($color,x) + $diff}]

  if {$::tk::dialog::color::tonemoves} {
    set data($color,intensity) [XToRgb $w $x]
    RedrawColorBars $w $color
    # Move other selectors if "Move all" mode is set
    DrawOthers $w $color
  }

  # Return the x value that it was actually set at
  return $x
}

# ::tk::dialog::color::ReleaseMouse
#
# Removes mouse tracking bindings, updates the colorbars.
#
# Params: sel is the selector canvas, color is the color of the strip,
#         x is the x-coord of the mouse.
#
proc ::tk::dialog::color::ReleaseMouse {w sel color x delta} {
  upvar ::tk::dialog::color::[winfo name $w] data

  set x [MoveSelector $w $sel $color $x $delta]

  # Determine exactly what color we are looking at.
  set data($color,intensity) [XToRgb $w $x]
  RedrawColorBars $w $color

  if {!$::tk::dialog::color::tonemoves} {
    DrawOthers $w $color
  }
  StickSelectors $w

  HandleSelEntry $w
}
# ::tk::dialog::color::ResizeColorbars --
#
#    Completely redraws the colorbars, including resizing the
#    colorstrips
#
proc ::tk::dialog::color::ResizeColorBars {w} {
  upvar ::tk::dialog::color::[winfo name $w] data

  if {($data(BARS_WIDTH) < $data(NUM_COLORBARS)) ||
  (($data(BARS_WIDTH) % $data(NUM_COLORBARS)) != 0)} {
    set data(BARS_WIDTH) $data(NUM_COLORBARS)
  }
  InitValues [winfo name $w]
  foreach color {red green blue} {
    $data($color,col) configure -width $data(canvasWidth)
    DrawColorScale $w $color 1
  }
}

# ::tk::dialog::color::HandleSelEntry --
#
#    Handles the return keypress event in the "Selection:" entry
#
proc ::tk::dialog::color::HandleSelEntry {w} {
  upvar ::tk::dialog::color::[winfo name $w] data

  StickSelectors $w 1
  set text [string trim $data(selection)]
  # Check to make sure that the color is valid
  if {[catch {set color [winfo rgb . $text]} ]} {
    set data(selection) $data(finalColor)
    return
  }

  set R [expr {[lindex $color 0]/0x100}]
  set G [expr {[lindex $color 1]/0x100}]
  set B [expr {[lindex $color 2]/0x100}]

  SetRGBValue $w "$R $G $B"
  set data(selection) $text
}

# ::tk::dialog::color::HandleRGBEntry --
#
#    Handles the return keypress event in the R, G or B entry
#
proc ::tk::dialog::color::HandleRGBEntry {w} {
  upvar ::tk::dialog::color::[winfo name $w] data

  StickSelectors $w 1
  foreach c {red green blue} {
    if {[catch {
      set data($c,intensity) [expr {int($data($c,intensity))}]
    }]} {
      set data($c,intensity) 0
    }
    if {$data($c,intensity) < 0} {
      set data($c,intensity) 0
    }
    if {$data($c,intensity) > 255} {
      set data($c,intensity) 255
    }
  }
  SetRGBValue $w "$data(red,intensity) \
  $data(green,intensity) $data(blue,intensity)"
}

# mouse cursor enters/leaves a color bar
#
proc ::tk::dialog::color::EnterLeaveColorBar {w color fill} {
  upvar ::tk::dialog::color::[winfo name $w] data
  $data($color,sel) itemconfigure $data($color,index) -fill $fill
}

# user hits "From clipboard" button
#
proc ::tk::dialog::color::OkCmd0 {w {clb ""}} {
  set mainentry ".[winfo name $w].top.sel.ent"
  if {$clb eq ""} {
    if {[catch {set clb [clipboard get]}]} return
  }
  # try #RGB and RGB
  foreach c {"" "#"} {
    catch {
      $mainentry delete 0 end
      $mainentry insert 0 "$c$clb"
      tk::dialog::color::HandleSelEntry $w
    }
  }
}

# user hits "Loupe" button
#
proc ::tk::dialog::color::Loupe {w} {
  upvar ::tk::dialog::color::[winfo name $w] data
  if {$data(-inifile) ne {}} {
    set inifile [list -inifile $data(-inifile)]
  } else {
    set inifile []
  }
  ::aloupe::run -exit no -parent $w -commandname [mc "Color"] \
    -command "::tk::dialog::color::OkCmd0 $w %c" {*}$inifile
}

# user hits OK button
#
proc ::tk::dialog::color::OkCmd {w} {
  variable ::tk::Priv
  upvar ::tk::dialog::color::[winfo name $w] data
  set Priv(selectColor) $data(finalColor)
}

# user hits Cancel button or destroys window
#
proc ::tk::dialog::color::CancelCmd {w} {
  variable ::tk::Priv
  set Priv(selectColor) ""
}
# _________________________________ EOF _________________________________ #
#RUNF1: ~/PG/github/pave/tests/test2_pave.tcl 23 9 12 "small icons"

clrpick.tcl