CSS
jQuery
flexbox

Flexbox で全体を中央に配置しつつ最後の行を左揃えにする

More than 1 year has passed since last update.


目的

以下のように、親パネルの中に子パネルが 12 枚並んでいます。

スクリーンショット 2016-02-05 22.30.08.png

Flexbox (CSS Flexible Box Layout) を使って子パネルをこの画像の通りに並べたいです。

要件は次の 2 つです。

1. 子パネル全体は親パネルの左右中央に配置しつつ、子パネルは左から順番にならべたい。

2. それぞれの子パネルの行と列を整列させたい。

HTML は以下の通りです。

<li class="grid">

<ul class="cell">ゆの</ul>
<ul class="cell">宮子</ul>
<ul class="cell">沙英</ul>
<ul class="cell">ヒロ</ul>
<ul class="cell">乃莉</ul>
<ul class="cell">なずな</ul>
<ul class="cell">茉里</ul>
<ul class="cell">まどか</ul>
<ul class="cell">さやか</ul>
<ul class="cell">マミ</ul>
<ul class="cell">杏子</ul>
<ul class="cell">ほむら</ul>
</li>


試行錯誤

まず、以下のように CSS を書きます。

/* 親パネル */

.grid {
display: flex;
flex-flow: row wrap;
justify-content: center;
}

スクリーンショット 2016-02-05 22.30.41.png

これは justify-content: center; を指定することで、子パネルを中央に配置しています。しかし、最後の行の 2 つの子パネルが中央に寄ってしまうことで、列が揃わなくなってしまいます。:cold_sweat:

では justify-content: flex-start; (justify-content の初期値) を指定するとどうなるでしょうか?

/* 親パネル */

.grid {
display: flex;
flex-flow: row wrap;
justify-content: flex-start;
}

スクリーンショット 2016-02-05 22.41.52.png

行と列はきれいに揃うのですが、もちろん左寄せになってしまいます。:cold_sweat:


解決策

Flexbox でなんとかこの目的を達成できないかとネットの海をさまよい続け、ようやく解答を得ました。

ただし、Flexbox あるいは CSS だけでもどうにもなりませんでした。

発見したのは Stack Overflow の以下の投稿です。

Flex-box: Align last row to grid

ここで提示されている解決策のひとつが、


One technique would be inserting a number of extra elements (as many as the max number of elements you ever expect to have in a row) that are given zero height. Space is still divided, but superfluous rows collapse to nothing


という、最後の行の残りの子要素を JavaScript (jQuery) を使って空の子要素で埋める方法です。空の子要素を実際の子要素の数だけ末尾に追加することで、たとえ親要素の width が変わって 1 列の子要素の数が変動したとしても、きちんと実現できるようになっています。

僕の挙げた例でも、実際に空の子パネルを親パネルに追加することで対応してみました。

/* 親パネル */

.grid {
display: flex;
flex-flow: row wrap;
justify-content: center;
}

/*
* 空の子パネル
* padding, margin の左右も 0 に指定してしまうと、
* 最後の行のレイアウトが崩れるので注意。
*/

.cell.is-empty {
height: 0;
padding-top: 0;
padding-bottom: 0;
margin-top: 0;
margin-bottom: 0;
}

var $grid = $('.grid'),   

emptyCells = [],
i;

// 子パネル (ul.cell) の数だけ空の子パネル (ul.cell.is-empty) を追加する。
for (i = 0; i < $grid.find('.cell').length; i++) {
emptyCells.push($('<ul>', { class: 'cell is-empty' }));
}

$grid.append(emptyCells);

すると結果はこのようになりました。期待通りに並んでいます。 :smiley:

スクリーンショット 2016-02-05 22.57.00.png

親パネルの幅が変わっても期待通りに表示されます。:blush:

スクリーンショット 2016-02-05 22.58.25.png


感想

CSS だけで完結しなかったり、JavaScript で無駄な要素を追加する必要があったりしたのは残念です。正直、ワークアラウンド感が否めない対応ですね。しかし、JavaScript で要素の位置や幅、余白を計算したり調整したりすることなく実現できたのはよかったです。

また「JavaScript を使って末尾に透明な子要素を追加すればよさそう」ということはなんとなくイメージできていました。しかし「高さや余白が 0 の子要素を実際の子要素数だけ用意すれば、親要素の幅に依存せずに期待通りに表示される」というところまで考えが至りませんでした。そのため、件の Stack Overflow の投稿内容が非常に有用でした。


おまけ

今回の検証に用いたコードを CodePen に上げています。

See the Pen Flexbox (justify-content: center AND align left last row) by QUANON (@quanon) on CodePen.