今回はお題として、
2つの項目の静的なデザイン
<body>
要素には、<div>
要素class
属性"item")<a>
要素class
属性"link")<span>
要素で加えている。アニメーションは、<a>
要素にclass
属性<span>
要素には、data-
グローバル属性を加えた。これらの値は、<div>
要素class
属性"container")
<div class="container">
<div class="item">
<a class="link animation-1" href="http://gihyo.jp/design/serial/01/createjs">
<span data-letters-l="Crea" data-letters-r="teJS">CreateJS</span>
</a>
</div>
<div class="item">
<a class="link animation-3" href="http://gihyo.jp/design/serial/01/css-animation">
<span>CSS</span>
<span>3</span>
</a>
</div>
</div>
Google Fontsはつぎの3つを使った<script>
要素で-prefix-freeを読み込んである
<link href="https://fonts.googleapis.com/css?family=Raleway:400,900" rel="stylesheet" type="text/css">
<link href="https://fonts.googleapis.com/css?family=Dosis:400,800" rel="stylesheet" type="text/css">
<link href="https://fonts.googleapis.com/css?family=Playfair+Display" rel="stylesheet" type="text/css">
<script src="lib/prefixfree.min.js"></script>
前掲<body>
要素の記述に対しては、<style>
要素を与える。まだアニメーションもマウスインタラクションもない。ただし、
data:image/s3,"s3://crabby-images/183e0/183e02246ff11320f43927039dc787fde910b5a5" alt="図1 はじめの項目テキストはふたつ重なって上下に線が加わった 図1 はじめの項目テキストはふたつ重なって上下に線が加わった"
body {
font-family: Arial, sans-serif;
background-color: Skyblue;
}
.item {
padding: 25px 0;
z-index: 1;
display: flex;
align-items: center;
justify-content: center;
}
.link {
text-decoration: none;
position: relative;
font-size: 5em;
line-height: 1;
}
.animation-1 {
font-family: 'Playfair Display', serif;
font-weight: 400;
color: Steelblue;
padding: 0 0 0.125em;
margin-top: -0.05em;
}
.animation-1::before,
.animation-1::after {
content: '';
width: 100%;
height: 3px;
background: Darkblue;
position: absolute;
}
.animation-1::before {
right: 0;
top: 0;
}
.animation-1::after {
left: 0;
bottom: -0.05em;
}
.animation-1 span::before,
.animation-1 span::after {
position: absolute;
color: white;
}
.animation-1 span::before {
content: attr(data-letters-l);
left: 0;
transform: translateX(-5px);
}
.animation-1 span::after {
content: attr(data-letters-r);
right: 0;
transform: translateX(5px);
}
.animation-3 {
font-weight: 900;
font-family: 'Raleway', Arial, sans-serif;
line-height: 1em;
margin-top: 0;
color: Steelblue;
}
.animation-3 span {
position: relative;
display: inline-block;
}
はじめの項目を擬似要素で飾りつける
このお題は、::before
および::after
でテキストの飾りつけやアニメーションを加えることがひとつの鍵となる。前掲コード1では、
data:image/s3,"s3://crabby-images/c8522/c8522c40fb1cb703d3f5918f49759a79732acaf2" alt="図2 図2"
まず、<a>
要素にアニメーションのため定めたclass
属性::before
と::after
で以下のように加えよう
<a class="link animation-1" href="http://gihyo.jp/design/serial/01/createjs">
</a>
.animation-1::before,
.animation-1::after {
content: '';
width: 100%;
height: 3px;
background: Darkblue;
position: absolute;
}
.animation-1::before {
right: 0;
top: 0;
}
.animation-1::after {
left: 0;
bottom: -0.05em;
}
data:image/s3,"s3://crabby-images/f537a/f537aa7c98d63b7456937eb0748d46f756f55728" alt="図3 擬似要素でテキストの上下に加えた線 図3 擬似要素でテキストの上下に加えた線"
つぎに、<span>
要素)::before
と::after
)content
プロパティに定める。その文字列の値は、data-
グローバル属性に与えた。値を取り出すには、attr()
式を用いればよい。そして、translateX()
関数
<span data-letters-l="Crea" data-letters-r="teJS">CreateJS</span>
.animation-1 span::before,
.animation-1 span::after {
position: absolute;
color: white;
}
.animation-1 span::before {
content: attr(data-letters-l);
left: 0;
transform: translateX(-5px);
}
.animation-1 span::after {
content: attr(data-letters-r);
right: 0;
transform: translateX(5px);
}
data:image/s3,"s3://crabby-images/b2d6e/b2d6e409fbba587d8398f0e958cbb262bbedd32e" alt="図4 擬似要素で左右外側にずらして重ねたテキスト 図4 擬似要素で左右外側にずらして重ねたテキスト"
擬似要素をアニメーションさせる
はじめの項目にマウスポインタを重ねたら、opacity
プロパティで透明にしておく。そして、:hover
擬似クラス)、translateX()
関数でもとのテキストと合わせた。これで、transition
プロパティに時間を定めれば、
.animation-1 span::before,
.animation-1 span::after {
opacity: 0;
transition: 0.5s;
}
.animation-1:hover span::before,
.animation-1:hover span::after {
opacity: 1;
transform: translateX(0);
}
data:image/s3,"s3://crabby-images/66cef/66cef91624349299987f1f26168a7fcdd673d3cf" alt="図5 マウスポインタを重ねると左右のテキストがフェードインしながらもとの位置に戻る 図5 マウスポインタを重ねると左右のテキストがフェードインしながらもとの位置に戻る"
つぎに、scaleX()
関数で幅を0に縮めておく。そして、:hover
擬似クラスでもとの幅に戻す。ただ、transform-origin
プロパティにより、
.animation-1::before,
.animation-1::after {
transform: scaleX(0);
transition: 0.5s;
}
.animation-1::before {
transform-origin: right;
}
.animation-1::after {
transform-origin: left;
}
.animation-1:hover::before,
.animation-1:hover::after {
transform: scaleX(1);
}
data:image/s3,"s3://crabby-images/e4fe4/e4fe460003a6e2fc8221778c74fec390644f1bfd" alt="図6 マウスポインタを重ねると上下の線がそれぞれ右と左から伸びる 図6 マウスポインタを重ねると上下の線がそれぞれ右と左から伸びる"
終わりの項目をアニメーションさせる
終わりの項目のテキストは、<span>
要素に分けておいた。それぞれに異なったアニメーションを加えるためだ。
<a class="link animation-3" href="http://gihyo.jp/design/serial/01/css-animation">
<span>CSS</span>
<span>3</span>
</a>
テキストは、class
属性"animation-3"):hover
擬似クラスで違う色に変えよう。2つの<span>
要素は、:first-of-type
と:last-of-type
で選び分けられる。transition
プロパティを定めれば、
.animation-3 span {
transition: 0.5s;
}
.animation-3:hover span:first-of-type {
color: white;
}
.animation-3:hover span:last-of-type {
color: Midnightblue;
}
data:image/s3,"s3://crabby-images/a4768/a47686692fe1d23386777a8661aff806cdad5df4" alt="図7 マウスポインタを重ねるとテキストの色が変わる 図7 マウスポインタを重ねるとテキストの色が変わる"
さらに、<span>
要素に、::before
擬似要素で矩形を加える。領域は、translateY()
関数でテキストの領域の上下外側に大きくずらしてある
.animation-3 span::before {
content: '';
position: absolute;
width: 100%;
height: 100%;
background: white;
}
.animation-3 span:last-of-type::before {
background: Midnightblue;
}
.animation-3 span:first-of-type::before {
transform: translateY(-150%);
}
.animation-3 span:last-of-type::before {
transform: translateY(150%);
}
data:image/s3,"s3://crabby-images/f6201/f6201ccb86256392e7605fea2374b003bf40c53e" alt="図8 テキスト領域の上下外側に擬似要素の矩形が置かれた 図8 テキスト領域の上下外側に擬似要素の矩形が置かれた"
2つの矩形は、overflow
プロパティプロパティの値をhidden
に定めれば、
.animation-3 {
overflow: hidden;
}
.animation-3 span::before {
transition: 0.5s;
}
.animation-3:hover span:last-of-type::before,
.animation-3 span:first-of-type::before {
transform: translateY(-150%);
}
.animation-3:hover span:first-of-type::before,
.animation-3 span:last-of-type::before {
transform: translateY(150%);
}
これで、
body {
font-family: Arial, sans-serif;
background-color: Skyblue;
}
.item {
padding: 25px 0;
z-index: 1;
position: relative;
display: flex;
align-items: center;
justify-content: center;
}
.link {
text-decoration: none;
position: relative;
font-size: 5em;
line-height: 1;
}
.animation-1 {
font-family: 'Playfair Display', serif;
font-weight: 400;
color: Steelblue;
padding: 0 0 0.125em;
margin-top: -0.05em;
}
.animation-1::before,
.animation-1::after {
content: '';
width: 100%;
height: 3px;
background: Darkblue;
position: absolute;
transform: scaleX(0);
transition: 0.5s;
}
.animation-1::before {
right: 0;
top: 0;
transform-origin: right;
}
.animation-1::after {
left: 0;
bottom: -0.05em;
transform-origin: left;
}
.animation-1:hover::before,
.animation-1:hover::after {
transform: scaleX(1);
}
.animation-1 span::before,
.animation-1 span::after {
position: absolute;
color: white;
opacity: 0;
transition: 0.5s;
}
.animation-1 span::before {
content: attr(data-letters-l);
left: 0;
transform: translateX(-5px);
}
.animation-1 span::after {
content: attr(data-letters-r);
right: 0;
transform: translateX(5px);
}
.animation-1:hover span::before,
.animation-1:hover span::after {
opacity: 1;
transform: translateX(0);
}
.animation-3 {
font-weight: 900;
font-family: 'Raleway', Arial, sans-serif;
line-height: 1em;
margin-top: 0;
overflow: hidden;
color: Steelblue;
}
.animation-3 span {
position: relative;
display: inline-block;
transition: 0.5s;
}
.animation-3:hover span:first-of-type {
color: white;
}
.animation-3:hover span:last-of-type {
color: Midnightblue;
}
.animation-3 span::before {
content: '';
position: absolute;
width: 100%;
height: 100%;
background: white;
transition: 0.5s;
}
.animation-3 span:last-of-type::before {
background: Midnightblue;
}
.animation-3:hover span:last-of-type::before,
.animation-3 span:first-of-type::before {
transform: translateY(-150%);
}
.animation-3:hover span:first-of-type::before,
.animation-3 span:last-of-type::before {
transform: translateY(150%);
}