Вертикальное выравнивание элементов с помощью CSS
28.11.2017 в 22:52, Евгений
Проблема вертикального центрирования элементов в CSS имеет ряд готовых решений. Выбор способа выравнивания в каждом отдельном случае зависит от типа, размеров, позиционирования элементов и других заданных им свойств.
Ниже представлены популярные у верстальщиков приемы выравнивания по вертикали. Показывается, как они работают, и для каких случаев наиболее подходит каждый из них.
Вот два элемента div:
<div class="container">
<div class="inner"></div>
</div>
Каждый из способов будет применен для выравнивания внутреннего блока по центру наружного.
line-height для одной строки
Когда родитель занимает одну строку текста и высота потомка известна, можно применить свойство line-height. Значение свойства должно быть равным высоте внешнего блока. Это работает только для одной строки, поэтому потомку полезно добавить overflow: hidden и запрет переноса строки white-space: nowrap.
Не получится воспользоваться процентной записью line-height=100%, потому что 100% в этом случае — высота шрифта.
.container {
height: 300px;
line-height: 300px;
}
.inner {
white-space: nowrap;
overflow: hidden;
}
Способ применим только при известной высоте внешнего блока.
Таблица с vertical-align
Для вертикального выравнивания элементов как нельзя лучше подходит таблица. Чтобы не нарушать семантику, табличные элементы лучше создавать средствами CSS. Положением содержимого ячейки управляет vertical-align. В таблицах работают четыре значения этого свойства:
• baseline — по умолчанию;
• bottom — содержимое внизу ячейки;
• middle — содержимое посередине ячейки;
• top — содержимое вверху ячейки.
В первом примере внешним блоком становится одинокая ячейка таблицы.
.container {
display: table-cell;
vertical-align: middle;
}
Способ хорош тем, что работает для элементов без заданной высоты, но в некоторых случаях его использованию мешает тот факт, что внешний блок, как любая ячейка таблицы, по ширине подстраивается под свое содержимое, а растянуть его можно только задав ширину для width явно. Для ячейки без таблицы проценты работают неадекватно.
Этот недостаток исправляется обертыванием ячейки в родителя со свойством display:table. Этому элементу размер можно задать в процентах. Окончательно код будет выглядеть так:
.outer-wrapper {
display: table;
}
.container {
display: table-cell;
vertical-align: middle;
}
<div class="outer-wrapper">
<div class="container">
<div class="inner"></div>
</div>
</div>
Position: absolute + отрицательный margin
Способ применяется, когда известна высота внутреннего элемента. Для внешнего блока она может быть неизвестной. Родителю задается относительное позиционирование, а потомку внутри него абсолютное.
Значение свойства top, равное 50%, заставляет вложенный элемент расположиться верхним краем посередине внешнего блока. Остается поднять его отрицательным margin-top на половину собственной высоты, чтобы он встал точно по центру вертикали.
.container {
position: relative;
}
.inner {
height: 140px;
position: absolute;
top: 50%;
margin-top: -70px;
}
Недостаток этого способа в необходимости знать высоту потомка.
Выравнивание в строке с vertical-align
Выравнивание строчных и строчно-блочных элементов, включая изображения и иконки, в окружающем их тексте осуществляется свойством vertical-align. В отличие от таблицы, в этом случае работает весь набор его значений, указанных в спецификации.
При известной высоте родителя этим свойством можно воспользоваться для центрирования многострочного текста.
Для внешнего блока прописывается центрирование одной строки.
.container {
height: 140px;
line-height: 140px;
}
Значение line-height для внутреннего блока переопределяется на normal или на любую требуемую величину. Ему же задается выравнивание vertical-align: middle и преобразование в строчно-блочный тип — display: inline-block.
.inner {
display: inline-block;
line-height: normal;
vertical-align: middle;
}
Недостаток способа — надо знать высоту родителя.
Выравнивание с помощью transform
Функция translateY свойства transform позволяет отцентрировать внутренний блок с неизвестной высотой. Для этого родитель должен быть спозиционирован относительно, а потомок — абсолютно.
Свойство top со значением 50% опускает внутренний блок так, что его верхний край располагается посередине родителя. Значение translateY: -50%, поднимает элемент на половину его собственной высоты и тем самым совмещает вертикальные центры блоков.
.container {
position: relative;
}
.inner {
position: absolute;
top: 50%;
transform: translateY(-50%);
}
Недостаток приема в ограниченной поддержке функций transform браузером IE.
Выравнивание через псевдоэлемент
Способ работает, когда высота неизвестна ни для первого, ни для второго блока. Внутрь родителя с помощью before или after добавляется строчный псевдоэлемент inline-block. Высота добавленного элемента должна равняться высоте родителя — height: 100%. Вертикальное выравнивание содержимого задается посредством vertical-align: middle.
С помощью vertical-align: middle относительно этого псевдоэлемента выравнивается внутренний блок. Так как родитель и потомок имеют одинаковую высоту, внутренний элемент, выравниваясь по вертикали с псевдоэлементом, центрируется также и с внешним блоком.
.container:before {
content: "";
height: 100%;
vertical-align: middle;
display: inline-block;
}
.inner {
display: inline-block;
vertical-align: middle;
}
Универсальный способ. Не работает если внутреннему блоку задано абсолютное позиционирование.
Flexbox
Самый новый и простой способ выравнивания элементов по вертикали. Flexbox позволяет расставлять элементы Web-страницы как угодно. Чтобы выровнять блок по центру, достаточно родителю прописать display: flex, а потомку margin: auto.
.container {
display: flex;
width: 320px;
height: 140px;
}
.inner {
width: 120px;
margin: auto;
}
Flexbox работает только в современных браузерах.
Выбор способа
Какой техникой вертикального выравнивания воспользоваться зависит от задачи и тех, ограничений, которые могут присутствовать в каком конкретном случае.
Высота элементов неизвестна
В этой ситуации можно воспользоваться одним из четырех универсальных способов:
1. Таблица. Родителем становится ячейка таблицы, созданная в HTML или через display-table/display-cell. Этому элементу-родителю присваивается vertical-align: middle.
2. Псевдоэлемент. Для внешнего блока создается строчно-блочный псевдоэлемент с width=100% и vertical-align: middle. Потомку прописываются display: inline-block и vertical-align: middle. Способ не работает только когда внутреннему блоку задано абсолютное позиционирование.
3. Transform. Родитель получает position: relative. Потомку присваивается position: absolute, top: 50% и transform: translateY(-50%);
4. Flexbox. Внешнему блоку ставится display: flex, внутреннему — margin: auto.
Известна только высота потомка
Внешний блок позиционируется относительно; внутреннему элементу прописываются абсолютное позиционирование, top: 50% и margin-top, равный половине его высоты.
Одна строка в блоке с известной высотой
Внешнему блоку задается свойство line-height со значением, равным его высоте.
Высота внешнего блока известна, а внутреннего элемента – нет.
Родителю ставится line-height равный его высоте. У потомка значение line-height переопределяется на normal или на любую требуемую величину, и задаются ему display: inline-block и vertical-align: middle.