今回はお題として、
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/db137/db1374ef6a1bc962cd585e5eda2c062b7acc0bb2" 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/093c9/093c99aeb92ab3331bfb8ad5a129668a495110c3" 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/f2c10/f2c10740331ac40ac1dc5a605496dd12375f2ec5" 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/2db74/2db74d95ebc26eb9b40a0c28c9b5263c522ce8e4" 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/894ef/894ef0f4f4176f0c027601adf155031d33b5f4dc" 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/870f0/870f063f2c197180cb6a362e78c5197ec5448e1f" 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/19efe/19efe48e4e0ccdc939919d4c2ed17313db2fa7fc" 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/693a7/693a7598bada890a66292b8905a3d7897786c0c2" 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%);
}