画像の色相回転の変化をアニメーションにする
CSSではimg要素で読み込んだ画像の色相回転をfilterプロパティのhue-rotateメソッドで調整することができます。
このプロパティはtransitionプロパティやanimationプロパティと組み合わせることで、変化の様子をアニメーションにすることも可能です。
今回はこれらのプロパティとJavaScriptによるclass属性の編集操作を組み合わせることで、色相回転をアニメーションさせていきます。
transitionプロパティを使うパターンとanimationプロパティを使うパターンの2種類解説していきますが、まずはtransitionプロパティを使ったパターンから解説していきます。
以下の例は「デフォルト」「90deg」「180deg」「270deg」の4つのボタンからいずれかを押すと、表示している画像のうち2枚目のimg要素のclass属性を切り替えて、色相回転が変化する様子をアニメーションとして再生します。
サンプルページはこちら
HTML コード例
<article id="content1">
<h1>JavaScriptレシピ</h1>
<ul id="btn_change">
<li><a href="#" class="active">デフォルト</a></li>
<li><a href="#">90deg</a></li>
<li><a href="#">180deg</a></li>
<li><a href="#">270deg</a></li>
</ul>
<div>
<figure><img id="pic1" src="./image/bear.png" alt="元画像"><figcaption>元の状態</figcaption></figure>
<figure><img id="pic2" src="./image/bear.png" alt="加工画像"><figcaption>調整後</figcaption></figure>
</div>
</article>
CSSは下の方の「色相回転の調整」部分に記述している箇所が非常に重要です。
ここではfilterプロパティによる色相回転に加えて、transitionプロパティを使ったアニメーションの設定も行います。
CSS コード例
article {
padding: 20px;
background-color: #bedef7;
}
h1 {
margin: 0 0 20px;
padding: 0;
font-size: 100%;
}
/* 色相回転の切り替えボタン
------------------------------ */
ul {
overflow: hidden;
margin: 0;
padding: 0;
}
ul li {
float: left;
margin: 0 20px 0 0;
display: inline-block;
padding: 0;
}
ul li a {
display: inline-block;
margin-bottom: 20px;
padding: 10px 20px;
color: #fff;
font-size: 86%;
text-decoration: none;
border-radius: 5px;
background-color: #4a93cb;
}
ul li a:hover,
ul li a.active {
background-color: #71b2e4;
}
/* 画像エリア
------------------------------ */
div {
overflow: hidden;
}
figure {
float: left;
margin: 0 20px 0 0;
padding: 0;
}
figure img {
width: 400px;
height: auto;
}
/* 色相回転
------------------------------ */
#pic2 {
filter: hue-rotate(0);
transition-property: filter;
transition-duration: .5s;
transition-timing-function: ease;
}
#pic2.deg90 {
filter: hue-rotate(90deg);
}
#pic2.deg180 {
filter: hue-rotate(180deg);
}
#pic2.deg270 {
filter: hue-rotate(270deg);
}
hue-rotateメソッドの0は初期状態で色相回転する前の元画像の状態です。
それ以外のクラスで指定している値「90deg」などは回転させる角度の指定になり、ボタンと設定する値との関係は以下のようになります。
クラス名 | 対応するボタン | hue-rotateメソッドの値 |
---|---|---|
.deg90 | 50% | 90deg |
.deg180 | 100% | 180deg |
.deg270 | 100% | 270deg |
最後にJavaScriptのコードです。
JavaScript コード例
window.addEventListener('DOMContentLoaded', function(){
const pic2 = document.getElementById('pic2');
const btn_change = document.getElementById('btn_change');
const btns = document.querySelectorAll('#btn_change li a');
for(var btn of btns) {
btn.addEventListener('click', (e)=> {
e.preventDefault();
e.target.classList.add('active');
const btns2 = document.querySelectorAll('#btn_change li a');
for(var i_btn of btns2) {
i_btn.classList.remove('active');
}
switch(e.target.textContent) {
case '90deg':
pic2.setAttribute('class','deg90');
break;
case '180deg':
pic2.setAttribute('class','deg180');
break;
case '270deg':
pic2.setAttribute('class','deg270');
break;
default:
pic2.removeAttribute('class');
break;
}
});
}
});
switch文の条件式にある「e.target.textContent」は押されたボタンのテキストを取得しています。
このテキストをcase文で判定して、適用するクラス名を決定します。
case文ではimg要素に対してsetAttributeメソッドやremoveAttributeメソッドを使ってclass属性を書き換えます。
適用するクラスの切り替わりと同時に適用するhue-rotateメソッドも変更となるため、そこでtransitionプロパティによるアニメーションが発生します。
animationプロパティを使った色相回転アニメーション
上記のコードではtransitionプロパティを使ってアニメーションの設定を行いましたが、animationプロパティに置き換えるとCSSは以下の内容になります。
変更はCSSのみとなり、HTMLとJavaScriptについては変更の必要はありません。
アニメーションの動作がtransitionプロパティと異なる点があります。
animationプロパティのアニメーションを設定する@keyframesは現在適用されているプロパティの値を引き継ぐことができません。
そのため、何かボタンを押してアニメーションを実行した後の状態で、他のボタンが押されたときに次のアニメーションを開始するときは初期値「hue-rotate(0)」からのスタートになります。
CSS コード例
article {
padding: 20px;
background-color: #bedef7;
}
h1 {
margin: 0 0 20px;
padding: 0;
font-size: 100%;
}
/* 色相回転の切り替えボタン
------------------------------ */
ul {
overflow: hidden;
margin: 0;
padding: 0;
}
ul li {
float: left;
margin: 0 20px 0 0;
display: inline-block;
padding: 0;
}
ul li a {
display: inline-block;
margin-bottom: 20px;
padding: 10px 20px;
color: #fff;
font-size: 86%;
text-decoration: none;
border-radius: 5px;
background-color: #4a93cb;
}
ul li a:hover,
ul li a.active {
background-color: #71b2e4;
}
/* 画像エリア
------------------------------ */
div {
overflow: hidden;
}
figure {
float: left;
margin: 0 20px 0 0;
padding: 0;
}
figure img {
width: 400px;
height: auto;
}
/* 色相回転
------------------------------ */
#pic2 {
filter: hue-rotate(0);
}
#pic2.deg90 {
animation: a1 .5s ease forwards;
}
#pic2.deg180 {
animation: a2 .5s ease forwards;
}
#pic2.deg270 {
animation: a3 .5s ease forwards;
}
@keyframes a1 {
100% {filter: hue-rotate(90deg);}
}
@keyframes a2 {
100% {filter: hue-rotate(180deg);}
}
@keyframes a3 {
100% {filter: hue-rotate(270deg);}
}