1:導入
例の「CSS完全に理解した」Tシャツ。
色々な方が話している通り、あれを意図的にやろうとすると実はなかなか難しい。
むしろ、「CSS完全に理解した」を再現するコードを書くたびに我々はCSSの完全理解に近づけるんじゃ無いだろうか?
CSS完全に理解したでCSS完全に理解すればもっとCSS完全に理解したできるんじゃ無いか
とにかく何かよくわからない追求心にとらわれた我々(約1名)は「CSS完全に理解した」を理解するために色々な方法でこれを再現することにしました。
以下に「CSS完全に理解した」するための数々の方法について書いていこうと思います。
2:前提
前提条件が限定されていないと無限の「CSS完全に理解した」状態を作り出せることになってしまうので、
今回は以下の画像のように、まず、「CSS完全に理解したわけではない」状態を再現し、ここに手を加えることで「CSS完全に理解した」状態を作り出すことにしました。
この状態でのHTML、CSSは以下の通りです。
つまり、テキストが入ったP要素を、CSSで枠を指定したDIVに入れたシンプルなものです。
HTMLはいじらず、CSSだけいじることでCSS完全に理解する状態にしていきます。
<div class="box">
<p class="box__text">
CSS<br>
完全に理解した
</p>
</div>
.box {
font-size: 16px;
position: relative;
border: 1px solid #000;
border-radius: 5px;
width: 140px;
height: 62px;
}
.box__text {
margin: 10px;
}
※以下のCSSコードでは、付け足す部分のみを記述し、上に書かれた根幹の部分は省略します。
3:実践
基本:P要素の左側に単純な隙間を空けるパターン
まず単純にテキスト部分の左側に隙間を開けることを考えてみます。
これは、padding-left
やmargin-left
使えば可能です。
(※padding-inline-start
やmargin-inline-start
も同様)
.box__text {
margin-left: 60px;
}
.box__text {
padding-left: 50px;
}
しかし、ただ隙間を空けただけだと下の画像のように勝手に文字が途中で改行されてしまいます。
また、position:absolute
とleft
を使用して右側にテキストを移動した場合も、途中で改行されてしまいます。
.box__text {
position: absolute;
left: 50px;
}
これは親要素であるDIVの幅が固定であるためで、それに引っ張られて下層のPタグも幅が制限され、改行されてしまうということです。
そこで、このテキストの自動折り返しをキャンセルする方法を考えなければなりません。
折り返しキャンセル方法1:white-space: nowrap;
文字通り改行を禁止するために使うCSSなので一番ふさわしいと思われます。line-height
などをいじればwhite-space: pre;
でも不可能ではないが、今回は再現性が低いので割愛します。
また、white-space: nowrap;
の代わりにword-break: keep-all;
でも同じ結果を得られます。
折り返しキャンセル方法2:ネガティブマージン
左側に隙間を空けた分だけの幅をmargin-right
のマイナス値で指定すれば、結果全体の幅は保たれるので、自動折り返しをキャンセルすることができます。
折り返しキャンセル方法3:幅を指定する
親の幅に引っ張られているので、自身が幅を持てばそちらを優先することができ、結果折り返しが起こらなくなります。
width
の代わりにinline-size
を利用しても同じ効果を得られます。
(もちろんmin-width
やmin-inline-size
も同様)
以上、「左側に単純な隙間を空ける方法」と「折り返しをキャンセルする方法」の組み合わせでここまでで9種類の「CSS完全に理解した」ができます。
応用:折り返しをキャンセルする必要がないパターン
position:relative
を使う
.box__text {
position: relative;
left: 50px;
}
position: relative;
は元の高さや幅を保ったまま見た目を移動させるため、left
(あるいはinset-inline-start
)を使用しても自動折り返しも起こりません。
position: absolute;
とright
を使用する
position: absolute
の時にleft
を使うと上記のように折り返しが発生しますが、right
(あるいはinset-inline-end
)をマイナス値で指定する方法ならばそれは起こりません。ただ基準がテキストの幅(フォントの種類)に依存するため、正確な位置合わせは難しくなります。
上記のサンプルでmargin-left
とmargin-right
を同時に指定したように、left: 50px;
とright: -50px
を同時に指定したり、あるいはinset-inline: 50px -50px;
を指定したりすればテキストの幅に対する依存は無くなります(そこまでする必要があるかどうかは別として)。
.box__text {
position: absolute;
right: -42px;
}
transform
を使う
transform
は「移動」ではなく「変形」なので、transformでの変形結果は幅や位置などに(見た目以上の)影響を与ええません。このためもちろん自動折り返しも起こりません。
.box__text {
transform: translateX(50px);
}
/* あるいは */
.box__text {
transform: translate(50px, 0);
}
/* またあるいは */
.box__text {
transform: translate3d(50px, 0, 0);
}
ちなみに上記は下記と同義となります。
.box__text {
transform: matrix(1, 0, 0, 1, 50, 0);
}
/* あるいは */
.box__text {
transform: matrix3d(1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 50, 0, 0, 1);
}
また、translateXやmatrix以外のtransformでも、下記のように無理やり右に50px移動させることができます。
/* Y軸回転→Z軸移動→Y軸回転 */
.box__text {
transform: rotateY(90deg) translateZ(50px) rotateY(-90deg);
}
/* Z軸回転→Y軸移動→Z軸回転 */
.box__text {
transform: rotateZ(90deg) translateY(-50px) rotateZ(-90deg);
}
応用:flexを無理やり使う。
DIVにdisplay: flex;
を指定しP要素をフレックスアイテムにした場合も自動折り返しが働きますが、この場合は上にあげた方法に加えてflex-shrink: 0;
でも折り返しをキャンセルすることができます。
.box {
display: flex;
}
.box__text {
padding-left: 50px;
flex-shrink: 0;
}
変態:常識にとらわれないパターン
ボックスを一旦消したあと書き直してCSS完全に理解する
とりあえず、枠の幅をautoに指定し、P要素の左に隙間を空けてみましょう。
.box {
width: auto;
}
.box__text {
margin-left: 60px;
}
当然P要素の指定に引っ張られて枠の長さが伸びてしまいます。
そこで、伸びた枠を背景と同じ色に指定した擬似要素で塗りつぶし、新しい枠を作ってしまいます。
その上でP要素が一番上に来るようにz-index
で調整します。
枠を上書きして塗りつぶす以外にborder-color: transparent;
で枠を透明にする方法もあります。
.box {
width: auto;
}
.box::after {
content: '';
position: absolute;
top: -1px;
left: -1px;
background-color: #fff;
width: 221px;
height: 64px;
z-index: 1;
}
.box::before {
content: '';
position: absolute;
top: -1px;
left: -1px;
border: 1px solid #000;
border-radius: 5px;
width: 140px;
height: 62px;
z-index: 2;
}
.box__text {
position: relative;
margin-left: 60px;
z-index: 3;
}
インデントを二重に指定してCSS完全に理解する
親のDIV要素にtext-indent
を指定し、P要素をインラインブロック要素にすると、親要素のtext-indentの影響をブロック全体で受けるようになるため、P要素はブロックごと右にずれることになります。しかしtext-indentはP要素の中にまで継承されるので、このままでは一行目の「CSS」がさらに右へずれてしまいます。そこでP要素でtext-indent
をゼロに指定してこれを打ち消せば、思惑通りの見た目を得ることができます。
.box {
text-indent: 50px;
}
.box__text {
text-indent: 0;
display: inline-block;
}
影文字で見た目を騙してCSS完全に理解する
本来の文字をcolor: transparent;
で透明にし、text-shadowを使ってぼかしの無い影文字を50px右に表示させることにより、スマートに枠からはみ出した文字を再現することができます。
.box__text {
color: transparent;
text-shadow: 50px 0 0 #000;
}
transformのscaleで幅を突破しCSS完全に理解する
前述したようにtransformの変形結果は幅や位置などに影響を与えません。
そこでまずは、font-size
を半分のサイズにすることで、折り返しが起こらない状況を作ります。
こうした上で、transform: scale(2);
で大きさを戻すことで折り返しを回避することができます。
.box__text {
font-size: 8px;
padding-left: 25px;
transform-origin: 0 0;
transform: scale(2);
}
この時transform-origin
で始点を左上にしてもpadding-left
はその範囲に入るので、移動させる距離である50pxの半分である25pxを指定します。
transformのrotateYで2回反転しCSS完全に理解する
まず枠全体を180度Y軸回転させます。
当然文字が裏返りますが、この状態で中のP要素のみを中心軸を変えてもう一回反転させれば、文字の裏返りが直り、なおかつ枠の外にはみ出させることができます。
.box {
transform: rotateY(180deg);
}
.box__text {
transform-origin: 35px 0;
transform: rotateY(180deg);
}
裏返っていることでtransform-origin: 0 0;
は右上をさすことになり、幅いっぱい(120px)分右にずれるので、ここから動かす50pxを引いた上での半分((120-50)/2)つまり35pxを指定すればちょうどの位置にずらすことができます。
グリッドシステム完全に理解したからCCSも完全に理解する
せっかくなのでグリッドシステムを利用した方法も考えてみました。
グリッドセルを異なるグリッドアイテムで共有できることを利用した方法です。
親の枠を消し、before疑似要素と.box__text
要素をグリッドアイテムとします。
縦方向のグリッドは
grid-template-rows: 11px auto 11px;
となります。11px
は枠線の太さと枠線と.box__text
要素間の隙間を足した数値です。
横方向のグリッドは
grid-template-columns: 11px 50px 68px 11px auto;
となります。50px
が右にずらす数値、68px
は余り、auto
ははみ出す部分です。
こうして作ったグリッドにbefore疑似要素と.box__text
要素を指定の位置に当て込めば完成です。
.box {
width: auto;
display: grid;
grid-template: 11px auto 11px / 11px 50px 68px 11px auto;
border: 0;
}
.box::before {
content: '';
grid-area: 1 / 1 / 4 / 5;
border: 1px solid #000;
border-radius: 5px;
}
.box__text {
margin: 0;
grid-area: 2 / 3 / 3 / 6;
}
鬼畜の所業!SVGをCSSに入れ込んでCSS?完全に理解する
まず、「CSS完全に理解した」と全く同じ見た目になるようにSVGを作ります。
<svg xmlns="http://www.w3.org/2000/svg" width="202" height="64" viewbox="0 0 202 64">
<style>
text {
font-family: sans-serif;
}
</style>
<rect x="0.5" y="0.5" width="141" height="63" rx="4.5" ry="4.5" stroke="#000" stroke-width="1" fill="transparent"/>
<text x="0" y="50" font-size="16">
<tspan x="61" y="24">CSS</tspan>
<tspan x="61" y="43">完全に理解した</tspan>
</text>
</svg>
cssのborderとsvgのstrokeでは線の中央位置が違うのでうまく調整しないと同じ見た目になりません。
SVGができたら、これをbase64に変換し、疑似要素のcontentとして指定します。
あとは元からあった枠を透明に、P要素は非表示にすれば完成!
.box::before {
content: url();
position: absolute;
top: -1px;
left: -1px;
border: none;
}
.box__text {
display: none;
}
これはCSSと言えるのか。。。。。?
4: まとめ
以下に上記で解説した、数々の「CSS完全に理解した」解法をのせました。
See the Pen CSS完全に理解したでCSS完全理解する by ichimonzi (@ichimonzi) on CodePen.
おそらくこれ以上の変態の解法もあるものと思われますので、変態の皆さんはよろしければ教えてください。
皆さんのさらなるCSSの完全理解をお祈りしています。