DrGrizz's blog

Jul 21, 2024

Adding support for open-color in rainbow-mode

Like many fellow software engineers, I'm bad at picking colors, so I was quite excited when I discovered open-color, less thinking to do ! Moreover, I use rainbow-mode once in a while, which allows to set the background of a string representing a color in a buffer to the corresponding color (i.e. "#ff0000" is displayed with a red background).

When I saw open-color had in its TODO list an item to add support for rainbow-mode, I wanted to give it a try.

Implementation

My original plan was to add support directly into rainbow-mode. But then I saw the package was offering a hook rainbow-keywords-hook for adding new keywords, my original plan changed to something less intrusive.

The work can be broken into 3 chunks.

  • The hook function itself, which be called by rainbow-mode: Its job is to add / remove keywords with font-lock-{add-keywords/remove-keywords} when the mode is enabled / disabled.
  • A list of keywords of things to highlight
  • Something to specify how to colorize what was matched

The hook is fairly straightforward. To check if rainbow-mode is enabled, we can simply check with (if rainbow-mode ...), such that it boils down to

(defun add-open-color-hook()
   (if rainbow-mode
         (font-lock-add-keywords nil open-color-rainbow-font-lock-keywords 'end)
     (font-lock-remove-keywords nil open-color-rainbow-font-lock-keywords)
     )
   )

The only gotcha here is to put our keywords at the end, otherwise when coloring something like open-color-red-1, the sub expression red would be colored in red, which is not what we want.

Each element in the list of keywords open-color-rainbow-font-lock-keywords can have one of 6 forms, as described in font-lock-keywords documentation. I took direct inspiration from rainbow-html-rgb-colors-font-lock-keywords to come with the following function

(defvar open-color-rainbow-font-lock-keywords
       '(
         ("\\<\\(open-color-[a-z]*-[0-9]\\)\\>" 1 (rainbow-colorize-open-color) )
         ("\\<\\(open-color-[a-z]*\\)\\>" 1 (rainbow-colorize-open-color) ) ;; handle open-color-black
         )
       "Font-lock keywords to add for open-color colors.")

In broad terms, if something matches the regex, then first capture group is getting colorized by rainbow-colorize-open-color (which we have to define).

The hardest part (in my opinion) is that last function. its goal to retrieve the string that was matched by font-lock and return a face for it.

  • For that, the first option is simply build a map, simple but unsatisfactory.
  • A more satisfactory solution would be from the string "X" to read the value of the variable X, such that for instance "open-color-red-1" background will be defined by the variable open-color-red-1 !

By looking on google, I found this StackOverflow question, where I learn about intern. That function returns the canonical symbol whose name is given, or creating one if any. From there, we can use symbol-value to read the content of the symbol (i.e. the actual value as defined by the open-color package). All that is left is to colorize the string using ainbow-colorize-match.

At the end, the colorization function looks like this

(defun rainbow-colorize-open-color()
  (let* (
         (color-name (match-string-no-properties 1))
         (color-object (intern color-name))
         )
    (if (boundp color-object)
        (let ((color-value (symbol-value color-object)))
          (rainbow-colorize-match color-value)
          )
      ))
  )

For some reasons, the regex doesn't always match the number at the end (such that color-name can be open-color-red), so we have to check if if the object is actually bound. If someone knows why, drop me a mail.

The result

two columns of texts, whose background is highlighted according to their name

The whole demo has been proposed in open-color README, but there has not been any activity on the PR since.

During the development, I found very useful to focus on quick (visual) feedback, and build gradually: - The first version of the hook was simply printing something in the minibuffer - The first version of open-color-rainbow-font-lock-keywords was highlighting the keyword "TEST" with with font-lock-warning-face. - A second iteration changed "TEST" to the regex, using re-builder.