色のコントラスト比は重要だけどどうやって求めるんだっけ?
はじめに
背景色とテキスト色のコントラスト比はWCAG 2.1においてAA基準とAAA基準の2つの達成基準によって定められています。 AA基準における大文字のテキストの最小コントラスト比は4.5:1、小文字のテキストの最小コントラスト比は3:1です。AAA基準においては、大文字のテキストの最小コントラスト比は7:1、小文字のテキストの最小コントラスト比は4.5:1です。 大文字のテキストは18pt(24px)以上、太字の場合は14pt(18.66px)以上です。小文字のテキストは14pt(18.66px)以上、太字の場合は12pt(16px)以上です。 これらの基準はコントラストに対する支援技術を用いない中程度の弱視の人がテキストを読めることを目的として設けられています。
この基準はWebAIMのチェッカーや、storybookのa11yに関するアドオンのようなツールを用いて確認できます。 ツールを使って簡単に求められるので、その背後にある計算方法を理解せずに使うこともできます。しかし、計算方法を理解した上で活用することはデザインの質を向上させるうえで有益です。私も以前はツールに頼りっぱなしでしたが、最近、色のコントラスト比を計算する方法を知り、その重要性を理解しました。この記事では色のコントラスト比の計算方法について紹介します。
計算式
コントラスト比はお互いの色の相対輝度を用いて表されます(輝度(きど、英: luminance)とは、広がりを持つ光源からある方向へ射出される光(可視光)の面積あたりの明るさを表す物理量である。^1)。相対輝度をで表した場合、コントラスト比はその比率を取るように計算されます。
はよりも相対的に明るい色となるようにします。輝度の定義より明るい色は暗い色よりも輝度が大きいので、与式は以上になります。 さらに定数として分母と分子に、周囲光の影響としてを加えています。これによって与式の分子と分母がになることを防いでくれています。 これから計算方法を紹介しますが、相対輝度はの値をとります。さらに、計算結果は1以上の値になることを踏まえるとコントラスト比はの値をとります。
相対輝度は色空間内の任意の点の相対的な明るさです。最小値が#000000
で、最大値が#ffffff
でとなるように正規化されます。
色の表現法は様々ありますが、計算ではで赤・緑・青のそれぞれの数値で表したものを使います。#4080c0
であればそれぞれを16進数で表したをで割ったを使います。
そしてそれぞれの成分を以下のルールに基づいて変形します。
F(x)
は単調増加関数で、の最小値は0、最大値は1です。そのため、の最小値も、最大値はとなりの値を取るようになります。
これらの計算は人間の視覚が色の明るさを非線形に捉える特性を補正するために行います。
明るい色は暗い色と同じ変化でも輝度を大きく感じることが見て取れます。与式ではそれが乗によって表されています(下図はをプロットしたものです。軸が逆なことに注意してください)。
また、以下ではその寄与が過度になってしまうので単純にで割るだけの式になっています。
計算した結果を赤・緑・青それぞれの見え方に合わせて重みをつけて足し合わせると相対輝度が求められます。
重みの合計は1です。も同じくの値を取るようになっています。 重みの付け方も人間の視覚で捉えやすい色を基準に設けています。人間は緑色に対する感受性が最も高く、次に赤色、最後に青色の順で感じるので緑が最も相対輝度に寄与して、青が最も寄与しないような重みづけになっています。
コード
計算式を学んだのそれをコードに落とし込んでいきましょう。 まずは補正を行う関数を作っていきます。
color
にはの範囲外の数値を渡せますが、この記事ではそのような数値を考えないものとして記述します。
次に相対輝度を求める関数です。
number
型の3つ要素を持つタプル型をRGB
という名前で作りました。
最後にコントラスト比を求める関数です。
相対輝度の計算が終わるまでどちらが明るいか分からないので、計算後に大きい方が分子に、小さい方が分母になるように調整しています。
さいごに
コントラスト比の計算方法について紹介しました。人間の視覚で感じやすい色についての情報や、コントラスト比を高めるためには緑の要素を強めると青の要素を強めるのに比べて10倍の効果があるなどツールで調整するだけでは分からなかったことが読み取れるようになりました。 今後もツールを使って結果を鵜呑みにするだけではなく、結果がどのように導かれるかを理解していきたいです。