Sorry about the bug. How embarrassing. Especially because I've learned over
the years to trust my gut feelings when something doesn't feel quite right,
and when I was testing the function, I remember thinking "surely there a
better matching named color than 'magenta'".

Thanks for the fix.

Kevin



On Sat, Jun 1, 2013 at 11:30 AM, John Fox <j...@mcmaster.ca> wrote:

> Hi Michael,
>
> This has become a bit of a comedy of errors.
>
> The bug is in Kevin Wright's code, which I adapted, and you too in your
> version, which uses local() rather than function() to produce the closure.
> The matrix which.col contains character data, as a consequence of binding
> the minimum squared distances to colour names, and thus the comparison
> cols.near[2,] < near^2 doesn't work properly when, ironically, the distance
> is small enough so that it's rendered in scientific notation.
>
> Converting to numeric appears to work:
>
> > rgb2col2 <- local({
> +     all.names <- colors()
> +     all.hsv <- rgb2hsv(col2rgb(all.names))
> +     find.near <- function(x.hsv) {
> +         # return the nearest R color name and distance
> +         sq.dist <- colSums((all.hsv - x.hsv)^2)
> +         rbind(all.names[which.min(sq.dist)], min(sq.dist))
> +     }
> +     function(cols.hex, near=.25){
> +         cols.hsv <- rgb2hsv(col2rgb(cols.hex))
> +         cols.near <- apply(cols.hsv, 2, find.near)
> +         ifelse(as.numeric(cols.near[2,]) <= near^2, cols.near[1,],
> cols.hex)
> +     }
> + })
>
> > rgb2col2(c("#010101", "#EEEEEE", "#AA0000", "#00AA00", "#0000AA",
> +     "#AAAA00", "#AA00AA", "#00AAAA"))
>
> [1] "black"         "gray93"        "darkred"       "green4"        "blue4"
> "darkgoldenrod"
> [7] "darkmagenta"   "cyan4"
>
> The same bug is in the code that I just posted using Lab colours, so (for
> posterity) here's a fixed version of that, using local():
>
> > rgb2col <- local({
> +     all.names <- colors()
> +     all.lab <- t(convertColor(t(col2rgb(all.names)), from = "sRGB",
> +         to = "Lab", scale.in = 255))
> +     find.near <- function(x.lab) {
> +         sq.dist <- colSums((all.lab - x.lab)^2)
> +         rbind(all.names[which.min(sq.dist)], min(sq.dist))
> +     }
> +     function(cols.hex, near = 2.3) {
> +         cols.lab <- t(convertColor(t(col2rgb(cols.hex)), from = "sRGB",
> +             to = "Lab", scale.in = 255))
> +         cols.near <- apply(cols.lab, 2, find.near)
> +         ifelse(as.numeric(cols.near[2, ]) < near^2, cols.near[1, ],
> toupper(cols.hex))
> +     }
> + })
>
> > rgb2col(c("#010101", "#EEEEEE", "#AA0000", "#00AA00", "#0000AA",
> "#AAAA00", "#AA00AA", "#00AAAA"))
>
> [1] "black"   "gray93"  "#AA0000" "#00AA00" "#0000AA" "#AAAA00"
> [7] "#AA00AA" "#00AAAA"
>
> > rgb2col(c("#010101", "#EEEEEE", "#AA0000", "#00AA00", "#0000AA",
> "#AAAA00", "#AA00AA", "#00AAAA"), near=15)
>
> [1] "black"         "gray93"        "firebrick3"    "limegreen"
> [5] "blue4"         "#AAAA00"       "darkmagenta"   "lightseagreen"
>
> So with Lab colours, setting near to the JND of 2.3 leaves many of these
> colours unmatched. I experimented a bit, and using 15 (as above) produces
> matches that appear reasonably "close" to me.
>
> I used squared distances to avoid taking the square-roots of all the
> distances. Since the criterion for "near" colours, which is on the distance
> scale, is squared to make the comparison, this shouldn't be problematic.
>
> I hope that finally this will be a satisfactory solution.
>
> Best,
>  John
>
> > -----Original Message-----
> > From: r-help-boun...@r-project.org [mailto:r-help-bounces@r-
> > project.org] On Behalf Of Michael Friendly
> > Sent: Saturday, June 01, 2013 11:33 AM
> > To: John Fox
> > Cc: 'r-help'; 'Martin Maechler'
> > Subject: Re: [R] measuring distances between colours?
> >
> > Just a quick note:  The following two versions of your function don't
> > give the same results.  I'm not sure why, and also not sure why the
> > criterion for 'near' should be expressed in squared distance.
> >
> > # version 1
> > rgb2col <- local({
> >      hex2dec <- function(hexnums) {
> >          # suggestion of Eik Vettorazzi
> >          sapply(strtoi(hexnums, 16L), function(x) x %/% 256^(2:0) %%
> > 256)
> >      }
> >      findMatch <- function(dec.col) {
> >          sq.dist <- colSums((hsv - dec.col)^2)
> >          rbind(which.min(sq.dist), min(sq.dist))
> >      }
> >      colors <- colors()
> >      hsv <- rgb2hsv(col2rgb(colors))
> >
> >      function(cols, near=0.25) {
> >          cols <- sub("^#", "", toupper(cols))
> >          dec.cols <- rgb2hsv(hex2dec(cols))
> >          which.col <- apply(dec.cols, 2, findMatch)
> >          matches <- colors[which.col[1, ]]
> >          unmatched <- which.col[2, ] > near^2
> >          matches[unmatched] <- paste("#", cols[unmatched], sep="")
> >          matches
> >      }
> > })
> >
> > # version 2
> > rgb2col2 <- local({
> >        all.names <- colors()
> >        all.hsv <- rgb2hsv(col2rgb(all.names))
> >        find.near <- function(x.hsv) {
> >            # return the nearest R color name and distance
> >            sq.dist <- colSums((all.hsv - x.hsv)^2)
> >            rbind(all.names[which.min(sq.dist)], min(sq.dist))
> >        }
> >        function(cols.hex, near=.25){
> >            cols.hsv <- rgb2hsv(col2rgb(cols.hex))
> >            cols.near <- apply(cols.hsv, 2, find.near)
> >            ifelse(cols.near[2,] < near^2, cols.near[1,], cols.hex)
> >        }
> > })
> >
> > # tests
> >  > rgb2col(c("#010101", "#EEEEEE", "#AA0000", "#00AA00", "#0000AA",
> > "#AAAA00", "#AA00AA", "#00AAAA"))
> > [1] "black"         "gray93"        "darkred"       "green4"
> > [5] "blue4"         "darkgoldenrod" "darkmagenta"   "cyan4"
> >  > rgb2col2(c("#010101", "#EEEEEE", "#AA0000", "#00AA00", "#0000AA",
> > "#AAAA00", "#AA00AA", "#00AAAA"))
> > [1] "#010101"       "#EEEEEE"       "darkred"       "green4"
> > [5] "blue4"         "darkgoldenrod" "darkmagenta"   "cyan4"
> >  >
> >
> >
> > On 5/31/2013 7:42 PM, John Fox wrote:
> > > Dear Kevin,
> > >
> > > I generally prefer your solution. I didn't realize that col2rgb()
> > worked
> > > with hex-colour input (as opposed to named colours), so my code
> > converting
> > > hex numbers to decimal is unnecessary; and using ifelse() is clearer
> > than
> > > replacing the non-matches.
> > >
> > > I'm not so sure about avoiding the closure, since for converting
> > small
> > > numbers of colours, your function will spend most of its time
> > constructing
> > > the local function find.near() and building all.hsv. Here's an
> > example,
> > > using your rgb2col() and a comparable function employing a closure,
> > with one
> > > of your examples executed 100 times:
> > >
> > >> r2c <- function(){
> > > +     all.names <- colors()
> > > +     all.hsv <- rgb2hsv(col2rgb(all.names))
> > > +     find.near <- function(x.hsv) {
> > > +         # return the nearest R color name and distance
> > > +         sq.dist <- colSums((all.hsv - x.hsv)^2)
> > > +         rbind(all.names[which.min(sq.dist)], min(sq.dist))
> > > +     }
> > > +     function(cols.hex, near=.25){
> > > +         cols.hsv <- rgb2hsv(col2rgb(cols.hex))
> > > +         cols.near <- apply(cols.hsv, 2, find.near)
> > > +         ifelse(cols.near[2,] < near^2, cols.near[1,], cols.hex)
> > > +     }
> > > + }
> > >
> > >> mycols <- c("#010101", "#EEEEEE", "#AA0000", "#00AA00", "#0000AA",
> > > +     "#AAAA00", "#AA00AA", "#00AAAA")
> > >
> > >> system.time(for (i in 1:100) oldnew <- c(mycols, rgb2col(mycols,
> > > near=.25)))
> > >     user  system elapsed
> > >     1.97    0.00    1.97
> > >
> > >> system.time({rgb2col2 <- r2c()
> > > +     for (i in 1:100) oldnew2 <- c(mycols, rgb2col2(mycols,
> > near=.25))
> > > +     })
> > >     user  system elapsed
> > >     0.08    0.00    0.08
> > >
> > >> rbind(oldnew, oldnew2)
> > >          [,1]      [,2]      [,3]      [,4]      [,5]      [,6]
> > > oldnew  "#010101" "#EEEEEE" "#AA0000" "#00AA00" "#0000AA" "#AAAA00"
> > > oldnew2 "#010101" "#EEEEEE" "#AA0000" "#00AA00" "#0000AA" "#AAAA00"
> > >          [,7]      [,8]      [,9]      [,10]     [,11]     [,12]
> > > oldnew  "#AA00AA" "#00AAAA" "#010101" "#EEEEEE" "darkred" "green4"
> > > oldnew2 "#AA00AA" "#00AAAA" "#010101" "#EEEEEE" "darkred" "green4"
> > >          [,13]   [,14]           [,15]         [,16]
> > > oldnew  "blue4" "darkgoldenrod" "darkmagenta" "cyan4"
> > > oldnew2 "blue4" "darkgoldenrod" "darkmagenta" "cyan4"
> > >
> > > Does this really make a difference? Frankly, it wouldn't for my
> > application
> > > (for colour selection in the Rcmdr) where a user is likely to perform
> > at
> > > most one or two conversions of a small number of colours in a
> > session. The
> > > time advantage of the second approach will depend upon the number of
> > times
> > > the function is invoked and the number of colours converted each
> > time.
> > >
> > > Best,
> > >   John
> > >
> > >> -----Original Message-----
> > >> From: r-help-boun...@r-project.org [mailto:r-help-bounces@r-
> > >> project.org] On Behalf Of Kevin Wright
> > >> Sent: Friday, May 31, 2013 3:39 PM
> > >> To: Martin Maechler
> > >> Cc: r-help; John Fox
> > >> Subject: Re: [R] measuring distances between colours?
> > >>
> > >> Thanks for the discussion.  I've also wanted to be able to find
> > nearest
> > >> colors.  I took the code and comments in this thread and simplified
> > the
> > >> function even further.  (Personally, I think using closures results
> > in
> > >> Rube-Goldberg code.  YMMV.)  The first example below is what I use
> > for
> > >> 'group' colors in lattice.
> > >>
> > >> Kevin Wright
> > >>
> > >> rgb2col <- function(cols.hex, near=.25){
> > >>    # Given a vector of hex colors, find the nearest 'named' R colors
> > >>    # If no color closer than 'near' is found, return the hex color
> > >>    # Authors: John Fox, Martin Maechler, Kevin Wright
> > >>    # From r-help discussion 5.30.13
> > >>
> > >>    find.near <- function(x.hsv) {
> > >>      # return the nearest R color name and distance
> > >>      sq.dist <- colSums((all.hsv - x.hsv)^2)
> > >>      rbind(all.names[which.min(sq.dist)], min(sq.dist))
> > >>    }
> > >>    all.names <- colors()
> > >>    all.hsv <- rgb2hsv(col2rgb(all.names))
> > >>    cols.hsv <- rgb2hsv(col2rgb(cols.hex))
> > >>    cols.near <- apply(cols.hsv, 2, find.near)
> > >>    ifelse(cols.near[2,] < near^2, cols.near[1,], cols.hex)
> > >> }
> > >>
> > >> mycols <- c("royalblue", "red", "#009900", "dark orange", "#999999",
> > >> "#a6761d", "#aa00da")
> > >> mycols <- c("#010101", "#EEEEEE", "#AA0000", "#00AA00", "#0000AA",
> > >> "#AAAA00", "#AA00AA", "#00AAAA")
> > >> mycols <- c("#010101", "#090909", "#090000", "#000900", "#000009",
> > >> "#090900", "#090009", "#000909")
> > >> oldnew <- c(mycols, rgb2col(mycols, near=.25)) # Also try near=10
> > >> pie(rep(1,2*length(mycols)), labels=oldnew, col=oldnew)
> > >>
> > >>    [[alternative HTML version deleted]]
> > >>
> > >> ______________________________________________
> > >> R-help@r-project.org mailing list
> > >> https://stat.ethz.ch/mailman/listinfo/r-help
> > >> PLEASE do read the posting guide http://www.R-project.org/posting-
> > >> guide.html
> > >> and provide commented, minimal, self-contained, reproducible code.
> > >
> >
> >
> > --
> > Michael Friendly     Email: friendly AT yorku DOT ca
> > Professor, Psychology Dept. & Chair, Quantitative Methods
> > York University      Voice: 416 736-2100 x66249 Fax: 416 736-5814
> > 4700 Keele Street    Web:   http://www.datavis.ca
> > Toronto, ONT  M3J 1P3 CANADA
> >
> > ______________________________________________
> > R-help@r-project.org mailing list
> > https://stat.ethz.ch/mailman/listinfo/r-help
> > PLEASE do read the posting guide http://www.R-project.org/posting-
> > guide.html
> > and provide commented, minimal, self-contained, reproducible code.
>
> ______________________________________________
> R-help@r-project.org mailing list
> https://stat.ethz.ch/mailman/listinfo/r-help
> PLEASE do read the posting guide
> http://www.R-project.org/posting-guide.html
> and provide commented, minimal, self-contained, reproducible code.
>



-- 
Kevin Wright

        [[alternative HTML version deleted]]

______________________________________________
R-help@r-project.org mailing list
https://stat.ethz.ch/mailman/listinfo/r-help
PLEASE do read the posting guide http://www.R-project.org/posting-guide.html
and provide commented, minimal, self-contained, reproducible code.

Reply via email to