ColorContrastCalc
is a utility that helps you choose colors with
sufficient contrast, WCAG 2.0 in mind.
With this library, you can do following things:
- Check the contrast ratio between two colors
- Find (if exists) a color that has sufficient contrast to a given color
- Create a new color from a given color by adjusting properties of the latter
- Sort colors
Add this line to your application's Gemfile:
gem 'color_contrast_calc'
And then execute:
$ bundle install
Or install it yourself as:
$ gem install color_contrast_calc
Here are some examples that will give you a brief overview of the library.
The full documentation is available at http://www.rubydoc.info/gems/color_contrast_calc
To represent a color, class ColorContrastCalc::Color
is provided.
And most of the operations in this utility use this class.
As an illustration, if you want to create an instance of Color
for red,
you may use a method ColorContrastCalc.color_from
Save the following code as color_instance.rb
:
require 'color_contrast_calc'
# Create an instance of Color from a hex code
# (You can pass 'red', [255, 0, 0], 'rgb(255, 0, 0)', 'hsl(0deg, 100%, 50%)' or
# hwb(60deg 0% 0%) instead of '#ff0000')
red = ColorContrastCalc.color_from('#ff0000')
puts red.class
puts red.name
puts red.hex
puts red.rgb.to_s
puts red.hsl.to_s
Then execute the script:
$ ruby color_instance.rb
ColorContrastCalc::Color
red
#ff0000
[255, 0, 0]
[0.0, 100.0, 50.0]
The following formats are supported for the first argument of ColorContrastCalc.color_from()
.
- RGB values in hexadecimal notation: #ff0, #ffff00, #FF0, etc.
- RGB values in functional notation: rgb(255, 255, 0), rgb(255 255 0), etc.
- RGB values as an Array of Integers: [255, 255, 0], etc.
- HSL colors in functional notation: hsl(60deg, 100%, 50%), hsl(60 100% 50%), etc.
- [Experimental] HWB colors in functional notation: hwb(60deg 0% 0%), hwb(60 0% 0%), etc.
- Extended color keywords: white, black, red, etc.
To calculate the contrast ratio between two colors, a class method of ColorContrastCalc
, .contrast_ratio()
is available.
For example, if you want to calculate the contrast ratio between yellow and black at the command line, you can do as follows:
$ ruby -rcolor_contrast_calc -e 'puts ColorContrastCalc.contrast_ratio("#ff0", "#000")'
19.555999999999997
Or
$ ruby -rcolor_contrast_calc -e 'puts ColorContrastCalc.contrast_ratio("rgb(255, 255, 0)", "black")'
19.555999999999997
(To represent a yellow, you can also use "hsl(60deg, 100%, 50%)", "#FFFF00", [255, 255, 0]...)
If you want to calculate the contrast ratio between yellow and black,
save the following code as yellow_black_contrast.rb
:
require 'color_contrast_calc'
yellow = ColorContrastCalc.color_from('yellow')
black = ColorContrastCalc.color_from('black')
contrast_ratio = yellow.contrast_ratio_against(black)
report = 'The contrast ratio between %s and %s is %2.4f'
puts(format(report, yellow.name, black.name, contrast_ratio))
puts(format(report, yellow.hex, black.hex, contrast_ratio))
Then execute the script:
$ ruby yellow_black_contrast.rb
The contrast ratio between yellow and black is 19.5560
The contrast ratio between #ffff00 and #000000 is 19.5560
It is also possible to calculate the contrast ratio of two colors from their hex color codes or RGB values.
Save the following code as yellow_black_hex_contrast.rb
:
require 'color_contrast_calc'
yellow, black = %w[#ff0 #000000]
# or
# yellow, black = [[255, 255, 0], [0, 0, 0]]
ratio = ColorContrastCalc::Checker.contrast_ratio(yellow, black)
level = ColorContrastCalc::Checker.ratio_to_level(ratio)
puts "Contrast ratio between yellow and black: #{ratio}"
puts "Contrast level: #{level}"
Then execute the script:
$ ruby yellow_black_hex_contrast.rb
Contrast ratio between yellow and black: 19.555999999999997
Contrast level: AAA
ColorContrastCalc.contrast_ratio_with_opacity()
is provided for the
calculation.
The method takes three arguments, foreground color, background color and an optional base color.
Please note that the third color is placed below the other two colors and expects to be fully opaque.
For example:
irb -r color_contrast_calc
irb(main):001:0> ColorContrastCalc.contrast_ratio_with_opacity('rgb(255 255 0 / 1.0)', 'rgb(0 255 0 / 0.5)', 'white')
=> 1.1828076947731336
irb(main):002:0> ColorContrastCalc.contrast_ratio_with_opacity('rgb(255 255 0 / 1.0)', 'rgb(0 255 0 / 0.5)') # The default value for the third parameter is white.
=> 1.1828076947731336
irb(main):003:0> ColorContrastCalc.contrast_ratio_with_opacity('rgb(255 255 0 / 1.0)', 'rgb(0 255 0 / 0.5)', 'black')
=> 4.78414518008597
irb(main):004:0> ColorContrastCalc.contrast_ratio_with_opacity('rgb(255 255 0)', 'rgb(0 255 0 / 0.5)', 'black') # For a fully opaque color, you don't need to specify the opacity.
=> 4.78414518008597
If you want to find a combination of colors with sufficient contrast
by changing the brightness/lightness of one of those colors, save the
following code as yellow_orange_contrast.rb
:
require 'color_contrast_calc'
yellow = ColorContrastCalc.color_from('yellow')
orange = ColorContrastCalc.color_from('orange')
report = 'The contrast ratio between %s and %s is %2.4f'
# Find brightness adjusted colors.
a_orange = yellow.find_brightness_threshold(orange, 'A')
a_contrast_ratio = yellow.contrast_ratio_against(a_orange)
aa_orange = yellow.find_brightness_threshold(orange, 'AA')
aa_contrast_ratio = yellow.contrast_ratio_against(aa_orange)
puts('# Brightness adjusted colors')
puts(format(report, yellow.hex, a_orange.hex, a_contrast_ratio))
puts(format(report, yellow.hex, aa_orange.hex, aa_contrast_ratio))
# Find lightness adjusted colors.
a_orange = yellow.find_lightness_threshold(orange, 'A')
a_contrast_ratio = yellow.contrast_ratio_against(a_orange)
aa_orange = yellow.find_lightness_threshold(orange, 'AA')
aa_contrast_ratio = yellow.contrast_ratio_against(aa_orange)
puts('# Lightness adjusted colors')
puts(format(report, yellow.hex, a_orange.hex, a_contrast_ratio))
puts(format(report, yellow.hex, aa_orange.hex, aa_contrast_ratio))
Then execute the script:
$ ruby yellow_orange_contrast.rb
# Brightness adjusted colors
The contrast ratio between #ffff00 and #c68000 is 3.0138
The contrast ratio between #ffff00 and #9d6600 is 4.5121
# Lightness adjusted colors
The contrast ratio between #ffff00 and #c78000 is 3.0012
The contrast ratio between #ffff00 and #9d6600 is 4.5121
For getting grayscale, ColorContrastCalc::Color
has an instance method
with_grayscale
.
For example, save the following code as grayscale.rb
:
require 'color_contrast_calc'
yellow = ColorContrastCalc.color_from('yellow')
orange = ColorContrastCalc.color_from('orange')
report = 'The grayscale of %s is %s.'
puts(format(report, yellow.hex, yellow.with_grayscale))
puts(format(report, orange.hex, orange.with_grayscale))
Then execute the script:
$ ruby grayscale.rb
The grayscale of #ffff00 is #ededed.
The grayscale of #ffa500 is #acacac.
And other than with_grayscale
, following instance methods
are available for ColorContrastCalc::Color
:
with_brightness
with_contrast
with_hue_rotate
with_invert
with_saturate
Please note the following methods are deprecated:
new_grayscale_color
new_brightness_color
new_contrast_color
new_hue_rotate_color
new_invert_color
new_saturate_color
You can sort colors using a method ColorContrastCalc::Sorter.sort
or
its alias ColorContrastCalc.sort
.
And by passing the second argument to this method, you can also specify the sort order.
For example, save the following code as sort_colors.rb
:
require 'color_contrast_calc'
color_names = ['red', 'lime', 'cyan', 'yellow', 'fuchsia', 'blue']
# Sort by hSL order. An uppercase for a component of color means
# that component should be sorted in descending order.
hsl_ordered = ColorContrastCalc.sort(color_names, 'hSL')
puts("Colors sorted in the order of hSL: #{hsl_ordered}")
# Sort by RGB order.
rgb_ordered = ColorContrastCalc.sort(color_names, 'RGB')
puts("Colors sorted in the order of RGB: #{rgb_ordered}")
# You can also change the precedence of components.
grb_ordered = ColorContrastCalc.sort(color_names, 'GRB')
puts("Colors sorted in the order of GRB: #{grb_ordered}")
# And you can directly sort hex color codes.
## Hex color codes that correspond to the color_names given above.
hex_codes = ['#ff0000', '#00ff00', '#0ff', '#ff0', '#f0f', '#0000FF']
hsl_ordered = ColorContrastCalc.sort(hex_codes, 'hSL')
puts("Colors sorted in the order of hSL: #{hsl_ordered}")
# If you want to sort colors in different notations,
# you should specify a key_mapper.
colors = ['rgb(255 0 0)', 'hsl(120 100% 50%)', '#0ff', 'hwb(60 0% 0%)', [255, 0, 255], '#0000ff']
key_mapper = proc {|c| ColorContrastCalc.color_from(c) }
colors_in_hsl_order = ColorContrastCalc.sort(colors, 'hSL', key_mapper)
puts("Colors sorted in the order of hSL: #{colors_in_hsl_order}")
Then execute the script:
$ ruby sort_colors.rb
Colors sorted in the order of hSL: ["red", "yellow", "lime", "cyan", "blue", "fuchsia"]
Colors sorted in the order of RGB: ["yellow", "fuchsia", "red", "cyan", "lime", "blue"]
Colors sorted in the order of GRB: ["yellow", "cyan", "lime", "fuchsia", "red", "blue"]
Colors sorted in the order of hSL: ["#ff0000", "#ff0", "#00ff00", "#0ff", "#0000FF", "#f0f"]
Colors sorted in the order of hSL: ["rgb(255 0 0)", "hwb(60 0% 0%)", "hsl(120 100% 50%)", "#0ff", "#0000ff", [255, 0, 255]]
Two lists of colors are provided, one is for named colors and the other for the web safe colors.
And there is a method ColorContrastCalc::Color::List.hsl_colors
that
generates a list of HSL colors that share same saturation and lightness.
For example, save the following code as color_lists.rb
:
require 'color_contrast_calc'
# Named colors
named_colors = ColorContrastCalc.named_colors
puts("The number of named colors: #{named_colors.size}")
puts("The first of named colors: #{named_colors[0].name}")
puts("The last of named colors: #{named_colors[-1].name}")
# Web safe colors
web_safe_colors = ColorContrastCalc.web_safe_colors
puts("The number of web safe colors: #{web_safe_colors.size}")
puts("The first of web safe colors: #{web_safe_colors[0].name}")
puts("The last of web safe colors: #{web_safe_colors[-1].name}")
# HSL colors
hsl_colors = ColorContrastCalc.hsl_colors
puts("The number of HSL colors: #{hsl_colors.size}")
puts("The first of HSL colors: #{hsl_colors[0].name}")
puts("The 60th of HSL colors: #{hsl_colors[60].name}")
puts("The 120th of HSL colors: #{hsl_colors[120].name}")
puts("The last of HSL colors: #{hsl_colors[-1].name}")
Then execute the script:
$ ruby color_lists.rb
The number of named colors: 147
The first of named colors: aliceblue
The last of named colors: yellowgreen
The number of web safe colors: 216
The first of web safe colors: black
The last of web safe colors: white
The number of HSL colors: 361
The first of HSL colors: #ff0000
The 60th of HSL colors: #ffff00
The 120th of HSL colors: #00ff00
The last of HSL colors: #ff0000
After checking out the repo, run bin/setup
to install dependencies. Then, run rake spec
to run the tests. You can also run bin/console
for an interactive prompt that will allow you to experiment.
To install this gem onto your local machine, run bundle exec rake install
. To release a new version, update the version number in version.rb
, and then run bundle exec rake release
, which will create a git tag for the version, push git commits and tags, and push the .gem
file to rubygems.org.
Bug reports and pull requests are welcome on GitHub at https://github.com/nico-hn/color_contrast_calc_rb.
The gem is available as open source under the terms of the MIT License.