Техніка CSS для горизонтальної лінії зі словами в середині


286

Я намагаюся скласти горизонтальне правило з деяким текстом посередині. Наприклад:

----------------------------------- моя назва тут ------------ -----------------

Чи можна це зробити в CSS? Без усіх тире "-" очевидно.


Це всередині formтегу?
тридцятьдот

у такому випадку, як вказував тридцять у відповідь на мою відповідь, ви можете поглянути fieldsetі на legendелементи
Стефан Мюллер


Для вирішення питання про прозорий фон див. Рядок до та після центру тексту над прозорим фоном
web-tiki

Відповіді:


404

Це приблизно так, як я це зробив: рядок створюється шляхом встановлення a border-bottomна містити, h2потім надаючи h2менший line-height. Потім текст поміщається в вкладене spanз непрозорим фоном.

h2 {
   width: 100%; 
   text-align: center; 
   border-bottom: 1px solid #000; 
   line-height: 0.1em;
   margin: 10px 0 20px; 
} 

h2 span { 
    background:#fff; 
    padding:0 10px; 
}
<h2><span>THIS IS A TEST</span></h2>
<p>this is some content other</p>

Я протестував лише в Chrome, але немає причин, щоб він не працював у інших браузерах.

JSFiddle: http://jsfiddle.net/7jGHS/


3
Це моє улюблене рішення. Він працює і на OSX, а деякі інші не роблять. Якщо ви використовуєте це рішення, не забудьте встановити фон прольоту таким же кольором, як фон вашої сторінки, буде особливо очевидно, що я маю на увазі, якщо ваш фон не білий. ;)
DrLazer

1
будь-який спосіб зробити це приблизно на чверть шляху поперек?
Троя Косентіно

1
Отримайте текст з відступом зліва, використовуючи "text-align: left; text-indent: 40px;" у стилі h2.
Matt__C

14
Це не є гнучким, оскільки ви фіксуєте колір фону як білий.
Чао

2
@NateBarbettini Якщо тільки рядок, який ви намагаєтеся зробити це надмірно, і текст повинен мати прозорі фони для інших речей за ним.
Coburn

266

Спробувавши різні рішення, я придумав одне дійсне для різної ширини тексту, будь-якого можливого фону та без додавання додаткової розмітки.

h1 {
  overflow: hidden;
  text-align: center;
}

h1:before,
h1:after {
  background-color: #000;
  content: "";
  display: inline-block;
  height: 1px;
  position: relative;
  vertical-align: middle;
  width: 50%;
}

h1:before {
  right: 0.5em;
  margin-left: -50%;
}

h1:after {
  left: 0.5em;
  margin-right: -50%;
}
<h1>Heading</h1>
<h1>This is a longer heading</h1>

Я перевірив його в IE8, IE9, Firefox та Chrome. Перевірити це можна тут http://jsfiddle.net/Puigcerber/vLwDf/1/


2
Мені це також подобається найкраще. Щоб отримати стиль, де текст з відступом зліва замість центру, використовуйте цю зміну: h1 { overflow: hidden; `вирівнювання тексту: зліва; `текст-відступ: 40px; `}
Matt__C

9
Це дійсно хороше рішення; особливо, оскільки він працює з будь-яким фоном і не вимагає заданої ширини елемента заголовка.
Лука

2
Це чудово. Я трохи адаптував ваш код, щоб вирівняти текст зліва і зробити :afterелемент шириною блоку. Мені цікаво: у чому саме полягає роль margin-right: -50%? Коли я роздувався з кодом, пропускаючи цю властивість, це призвело б до того, щоб прикраса перейшло на інший рядок. І навіть коли :afterприсвоєно 100% ширини, мені все одно потрібне від'ємне обмеження в 50%, щоб воно підходило. Чому?
BeetleTheNeato

Як я розумію, додавання негативних маржів в обидві сторони тягне елемент в обох напрямках, тому він стає центром.
Puigcerber

2
Одне з речей, яке мені подобається в цьому рішенні, - це те, що воно працює в ситуаціях, коли ви не знаєте, якого кольору тло. Легко найкраще рішення до цього часу.
JoeMoe1984

54

Добре, цей складніше, але він працює у всьому, крім IE <8

<div><span>text TEXT</span></div>

div {
    text-align: center;
    position: relative;
}
span {
    display: inline-block;    
}
span:before,
span:after {
    border-top: 1px solid black;
    display: block;
    height: 1px;
    content: " ";
    width: 40%;
    position: absolute;
    left: 0;
    top: 1.2em;
}
span:after {
   right: 0;  
   left: auto; 
}

Елементи: до і після: розміщуються абсолютно, щоб ми могли потягнути один вліво і один вправо. Крім того, ширина (40% в цьому випадку) дуже залежить від ширини тексту всередині. Доводиться думати про рішення для цього. Принаймні, top: 1.2emпереконайтеся, що рядки залишаються більш-менш у центрі тексту, навіть якщо у вас різний розмір шрифту.

Це, здається, працює добре: http://jsfiddle.net/tUGrf/3/


3
Вони розумні, але я впевнений, що ОП не хоче цього для formтегів. Якщо він це зробить, як я впевнений, ви знаєте, він міг би просто використати fieldsetі legend.
тридцятьдот

1
Хе, ти маєш рацію з цього приводу. Можна так чи інакше піти, якщо це не працює .. не семантично правильно, але краще, ніж вдаватися до JavaScript для чогось подібного imo.
Стефан Мюллер

8
Найкраще рішення, яке я знайшов, що працює добре, коли розміщується екран і не встановлюється колір тла тексту - jsfiddle.net/vLwDf/268
Xavier John

Так @ xavier-john, це моя відповідь ;)
Puigcerber

43

Найкоротший і найкращий метод:

span:after,
span:before{
    content:"\00a0\00a0\00a0\00a0\00a0";
    text-decoration:line-through;
}
<span> your text </span>


4
Ви повинні додати опис того, що робить ваш код, якщо ви не хочете його видалити.
inf3rno

У моєму випадку рядок виявився трохи товстим, тому я зламав його, призначивши інше сімейство шрифтів: font-family: monospace;добре працював.
Михайло Васін

38

Ось рішення на основі Flex.

Заголовок з центральною горизонтальною лінією до його сторін

h1 {
  display: flex;
  flex-direction: row;
}
h1:before, h1:after{
  content: "";
  flex: 1 1;
  border-bottom: 1px solid #000;
  margin: auto;
}
h1:before {
  margin-right: 10px
}
h1:after {
  margin-left: 10px
}
<h1>Today</h1>

JSFiddle: https://jsfiddle.net/yoshiokatsuneo/3h1fmj29/


Як отримати прокладки?
Силар

швидко і брудно, ви можете додати & nbsp;
chrismarx

1
Я використовую ще один елемент: <h1><span>Today</span></h1>з запасом / накладкою на проліт.
Джессі Буйтенхуа

19

для більш пізнього (сьогодні) браузера, display:flexandd pseudo-elements дозволяє легко малювати. border-style, box-shadowі навіть backgroundдопомагає для макіяжу. демонстрація нижче

h1 {margin-top:50px;
  display:flex;
  background:linear-gradient(to left,gray,lightgray,white,yellow,turquoise);;
}
h1:before, h1:after {
  color:white;
  content:'';
  flex:1;
  border-bottom:groove 2px;
  margin:auto 0.25em;
  box-shadow: 0 -1px ;/* ou 0 1px si border-style:ridge */
}
<h1>side lines via flex</h1>


Дякую за це, дуже струнка!
Jimmy Obonyo Abor

Ідеальне, єдине рішення, яке працювало для мене, коли фоном було зображення
cjohansson

12

.hr-sect {
	display: flex;
	flex-basis: 100%;
	align-items: center;
	color: rgba(0, 0, 0, 0.35);
	margin: 8px 0px;
}
.hr-sect::before,
.hr-sect::after {
	content: "";
	flex-grow: 1;
	background: rgba(0, 0, 0, 0.35);
	height: 1px;
	font-size: 0px;
	line-height: 0px;
	margin: 0px 8px;
}
<div class="hr-sect">Text</div>


9
<div><span>text TEXT</span></div>

div { 
  height: 1px; 
  border-top: 1px solid black; 
  text-align: center; 
  position: relative; 
}
span { 
  position: relative; 
  top: -.7em; 
  background: white; 
  display: inline-block; 
}

Дайте прольоту прокладку, щоб зробити більше місця між текстом і рядком.

Приклад: http://jsfiddle.net/tUGrf/


1
Приємно, єдина проблема полягає в тому, що ви повинні встановити фон білим і не можете встановити його прозорим.
Marnix

у такому випадку вам доведеться працювати з двома лініями, по одній з кожного боку. Це буде важко досягти чистим способом ..
Стефан Мюллер

Мій улюблений підхід поки що, але, на жаль, він чомусь не працює в електронних клієнтах (Gmail для мобільних пристроїв і веб)
reedwolf

9

Я роздивлявся кілька рішень для цього простого декору, і знайшов досить багато, деякі дивні, деякі навіть з JS, щоб обчислити висоту шрифту і bla, bla, bla, тоді я прочитав один на цю посаду і прочитав коментар від thirtydot кажучи про fieldsetі legendя подумав , що це було.

Я переосмислюю ці 2 стильові елементи, я думаю, ви можете скопіювати стандарти W3C для них і включити їх у свій .middle-line-textклас (або як би ви хочете назвати його), але ось що я зробив:

<fieldset class="featured-header">
    <legend>Your text goes here</legend>
</fieldset>

<style>
.featured-header{
    border-bottom: none;
    border-left: none;
    border-right: none;
    text-align: center;
 }

.featured-header legend{
    -webkit-padding-start: 8px; /* It sets the whitespace between the line and the text */
    -webkit-padding-end: 8px; 
    background: transparent; /** It's cool because you don't need to fill your bg-color as you would need to in some of the other examples that you can find (: */
    font-weight: normal; /* I preffer the text to be regular instead of bold  */
    color: YOU_CHOOSE;
}   

</style>

Ось загадка: http://jsfiddle.net/legnaleama/3t7wjpa2/

Я грав зі стилями межі, і він також працює в Android;) (Тестовано на kitkat 4.XX)

Редагувати:

Слідом за ідеєю Бекерова Артура, що також є хорошим варіантом, я змінив .png base64-образ, щоб створити обведення за допомогою .SVG, щоб ви могли відображати будь-яку роздільну здатність, а також змінювати колір елемента без будь-якого іншого програмного забезпечення :)

/* SVG solution based on Bekerov Artur */
/* Flexible solution, scalable, adaptable and also color customizable*/
.stroke {
background-image: url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' xmlns:xlink='http://www.w3.org/1999/xlink' x='0px' y='0px' width='1px' height='1px' viewBox='0 0 1 1' enable-background='new 0 0 1 1' fill='%23ff6600' xml:space='preserve'><rect width='1' height='1'/></svg>");
background-repeat: repeat-x;
background-position: left;
text-align: center;
}
.stroke h3 {
background-color: #ffffff;
margin: 0 auto;
padding:0 10px;
display: inline-block;
font-size: 66px;
}

5

Рішення для IE8 та новіших ...

Питання, які варто відзначити:

Використання background-color для маскування кордону може бути не найкращим рішенням. Якщо у вас складний (або невідомий) колір фону (або зображення), маскування в кінцевому рахунку не вдасться. Крім того, якщо ви зміните розмір тексту, ви помітите, що білий колір фону (або все, що ви встановите) почне прикривати текст у рядку вгорі (або внизу).

Ви також не хочете "здогадуватися" про те, наскільки широкими є розділи, оскільки це робить стилі дуже негнучкими і майже неможливими для реалізації на чуйному сайті, де змінюється ширина вмісту.

Рішення:

( Перегляд JSFiddle )

Замість того, щоб "маскувати" кордон з значком background-color, використовуйте свою displayвласність.

HTML

<div class="group">
    <div class="item line"></div>
    <div class="item text">This is a test</div>
    <div class="item line"></div>
</div>

CSS

.group { display: table; width: 100%; }
.item { display: table-cell; }
.text { white-space: nowrap; width: 1%; padding: 0 10px; }
.line { border-bottom: 1px solid #000; position: relative; top: -.5em; }

Змініть розмір тексту, розмістивши font-sizeвластивість на .groupелементі.

Обмеження:

  • Немає багаторядкового тексту. Тільки поодинокі лінії
  • Розмітка HTML не така елегантна
  • topвластивість .lineелемента має бути половиною line-height. Таким чином, якщо у вас є line-heightз 1.5em, то topповинно бути -.75em. Це обмеження, оскільки воно не є автоматизованим, і якщо ви застосовуєте ці стилі до елементів з різною висотою ліній, то вам може знадобитися повторно застосувати свій line-heightстиль.

Для мене ці обмеження переважають "питання", які я зазначив на початку своєї відповіді на більшість реалізацій.


Працює з хрому. Здається, семантично і досить просто. Працює з завантажувальним пристроєм.
Джефф Анчел

Мені довелося додати границю-крапку: окремо до .group, тому що я використовую її у можливости, яка має інше налаштування. Чудово працює.
Брайан Маккей

5

Це дає фіксовану довжину для ліній, але чудово працює. Довжини ліній керуються додаванням або взяттям \ \ 00a0 '(пробіл унікоду).

введіть тут опис зображення

h1:before, h1:after {
  content:'\00a0\00a0\00a0\00a0';
  text-decoration: line-through;
  margin: auto 0.5em;
}
<h1>side lines</h1>


2

Якщо хтось задається питанням, як встановити заголовок таким чином, щоб він відображався з фіксованою відстані до лівої сторони (а не в центрі, як представлено вище), я зрозумів це, змінивши код @ Puigcerber.

h1 {
  white-space: nowrap;
  overflow: hidden;
}

h1:before, 
h1:after {
  background-color: #000;
  content: "";
  display: inline-block;
  height: 1px;
  position: relative;
  vertical-align: middle;
}

h1:before {
  right: 0.3em;
  width: 50px;
}

h1:after {
  left: 0.3em;
  width: 100%;
}

Тут JSFiddle .


2
h6 {
    font: 14px sans-serif;
    margin-top: 20px;
    text-align: center;
    text-transform: uppercase;
    font-weight: 900;
}

    h6.background {
        position: relative;
        z-index: 1;
        margin-top: 0%;
        width:85%;
        margin-left:6%;
    }

        h6.background span {
            background: #fff;
            padding: 0 15px;
        }

        h6.background:before {
            border-top: 2px solid #dfdfdf;
            content: "";
            margin: 0 auto; /* this centers the line to the full width specified */
            position: absolute; /* positioning must be absolute here, and relative positioning must be applied to the parent */
            top: 50%;
            left: 0;
            right: 0;
            bottom: 0;
            width: 95%;
            z-index: -1;
        }

це вам допоможе


між рядком

<h6 class = "background"> ​​<span> - НАШІ ПОСЛУГИ </span> </h6> це код html
Домінік Амаль Джо F

Джо, ви можете редагувати власну відповідь, щоб додати HTML-код - набагато краще, ніж ставити його в коментарі. До відповіді слід також додати трохи пояснень.
Могсдад

Вибачте, Могсдад, я забув поставити html-код, тому і поклав його сюди.
Домінік Амаль Джо Ф

2

Я використовую макет таблиці, щоб динамічно заповнювати сторони та 0-висотові знаки абсолютного положення для динамічного вертикального позиціонування:

введіть тут опис зображення

  • жодних жорстких розмірів
  • немає зображень
  • відсутні псевдоелементи
  • поважає тло
  • зовнішній вигляд панелі управління

https://jsfiddle.net/eq5gz5xL/18/

Я виявив, що трохи нижче справжнього центру найкраще виглядає текст; це можна регулювати там, де 55%є (більша висота робить планку нижче). Зовнішній вигляд рядка можна змінити там, деborder-bottom є.

HTML:

<div class="title">
  <div class="title-row">
    <div class="bar-container">
      <div class="bar"></div>
    </div>
    <div class="text">
      Title
    </div>
    <div class="bar-container">
      <div class="bar"></div>
    </div>
  </div>
</div>

CSS:

.title{
  display: table;
  width: 100%
  background: linear-gradient(to right, white, lightgray);
}
.title-row{
  display: table-row;
}
.bar-container {
  display: table-cell;
  position: relative;
  width: 50%;
}
.bar {
  position: absolute;
  width: 100%;
  top: 55%;
  border-bottom: 1px solid black;
}
.text {
  display: table-cell;
  padding-left: 5px;
  padding-right: 5px;
  font-size: 36px;
}

2

Не бити мертвого коня, але я шукав рішення, опинився тут, і сам не був задоволений варіантами, в останню чергу чомусь мені не вдалося отримати запропоновані рішення тут, щоб добре працювати на мене. (Ймовірно, через помилки з мого боку ...) Але я грав у флексбоксі, і ось я щось взяв на роботу для себе.

Деякі налаштування є провідними, але лише для демонстрації. Я думаю, що це рішення має працювати в будь-якому сучасному браузері. Просто видаліть / відрегулюйте фіксовані параметри для .flex-батьківського класу, відрегулюйте кольори / текст / речі та (сподіваюся), ви будете такі ж щасливі, як і я з цим підходом.

HTML:

.flex-parent {
  display: flex;
  width: 300px;
  height: 20px;
  align-items: center;
}

.flex-child-edge {
  flex-grow: 2;
  height: 1px;
  background-color: #000;
  border: 1px #000 solid;
}
.flex-child-text {
  flex-basis: auto;
  flex-grow: 0;
  margin: 0px 5px 0px 5px;
  text-align: center;
}
<div class="flex-parent">
  <div class="flex-child-edge"></div>
  <div class="flex-child-text">I found this simpler!</div>
  <div class="flex-child-edge"></div>
</div>

Я також зберегла своє рішення тут: https://jsfiddle.net/Wellspring/wupj1y1a/1/


1

На всякий випадок, коли хто хоче, IMHO найкращим рішенням із використанням CSS є flexbox.

Ось приклад:

.kw-dvp-HorizonalButton {
    color: #0078d7;
    display:flex;
    flex-wrap:nowrap;
    align-items:center;
}
.kw-dvp-HorizonalButton:before, .kw-dvp-HorizonalButton:after {
    background-color: #0078d7;
    content: "";
    display: inline-block;
    float:left;
    height:1px;
}
.kw-dvp-HorizonalButton:before {
    order:1;
    flex-grow:1;
    margin-right:8px;
}
.kw-dvp-HorizonalButton:after {
    order: 3;
    flex-grow: 1;
    margin-left: 8px;
}
.kw-dvp-HorizonalButton * {
    order: 2;
}
    <div class="kw-dvp-HorizonalButton">
        <span>hello</span>
    </div>

Це завжди має призвести до ідеально орієнтованого вирівнювання вмісту з лівою та правою лінією, з простою контрольною межею між рядком та вмістом.

Він створює лінійний елемент до та після вашого верхнього керування та встановлює їх для порядку 1,3 у вашому гнучкому контейнері, встановлюючи вміст як порядок 2 (переходьте посередині). якщо до / після зростання на 1 зробить їх однаковою мірою споживати найбільш вільне місце, зберігаючи вміст у центрі.

Сподіваюсь, це допомагає!


0

Я не надто впевнений, але ви можете спробувати скористатися горизонтальним правилом і просунути текст вище його верхнього поля. Вам знадобиться фіксована ширина на тезі абзацу і на тлі. Це трохи хакі, і я не знаю, чи буде він працювати у всіх браузерах, і вам потрібно встановити від'ємну маржу залежно від розміру шрифту. Хоча працює на хромі.

<style>   
 p{ margin-top:-20px; background:#fff; width:20px;}
</style>

<hr><p>def</p>

0

Я спробував більшість запропонованих способів, але закінчується такими проблемами, як повна ширина або Не підходить для динамічного вмісту. Нарешті я змінив код і працює чудово. Цей приклад коду буде малювати ці рядки до і після, і він є гнучким у зміні вмісту. Центр також вирівняний.

HTML

        <div style="text-align:center"> 
            <h1>
                <span >S</span>
            </h1> 
        </div> 

        <div style="text-align:center"> 
            <h1>
                <span >Long text</span>
            </h1> 
        </div> 

CSS

      h1 {
            display: inline-block;
            position: relative;
            text-align: center;
        }

        h1 span {
            background: #fff;
            padding: 0 10px;
            position: relative;
            z-index: 1;
        }

        h1:before {
            background: #ddd;
            content: "";
            height: 1px;
            position: absolute;
            top: 50%;
            width: calc(100% + 50px);//Support in modern browsers
            left: -25px;
        }

        h1:before {
            left: ;
        }

Результат: https://jsfiddle.net/fasilkk/vowqfnb9/


0

Для мене це рішення працює чудово ...

HTML

<h2 class="strikethough"><span>Testing Text</span></h2>

CSS

.strikethough {
 width:100%; 
 text-align:left; 
 border-bottom: 1px solid #bcbcbc; 
 overflow: inherit;
 margin:0px 0 30px;
 font-size: 16px;
 color:#757575;
 }
.strikethough span {
 background:#fff; 
 padding:0 20px 0px 0px;
 position: relative;
 top: 10px;
}

0

Люди, які використовують Bootstrap 4, можуть досягти цього методу. Класи, згадані в коді HTML, походять з Bootstrap 4.

h1 {
    position: relative;
    flex-grow: 1;
    margin: 0;
}

h1:before {
   content: "";
   display: block;
   border-top: solid 2px blue;
   width: 100%;
   height: 1px;
   position: absolute;
   top: 50%;
   z-index: 1;
 }
 h1 span {
   background: #fff;
   left: 12%;
   padding: 0 15px;
   position: relative;
   z-index: 5;
 }

І напишіть свій HTML так

<div class="d-flex flex-row align-items-center">
  <h1><span> Title </span> </h1>
</div>

0

Горизонтальна та вертикальна лінія зі словами в середині

.box{
    background-image: url("https://i.stack.imgur.com/N39wV.jpg");
    width: 350px;
    padding: 10px;
}

/*begin first box*/
.first{
    width: 300px;
    height: 100px;
    margin: 10px;
    border-width: 0 2px 0 2px;
    border-color: red;
    border-style: solid;
    position: relative;
}

.first span {
    position: absolute;
    display: flex;
    right: 0;
    left: 0;
    align-items: center;
}
.first .foo{
    top: -8px;
}
.first .bar{
    bottom: -8.5px;
}
.first span:before{
    margin-right: 15px;
}
.first span:after {
    margin-left: 15px;
}
.first span:before , .first span:after {
    content: ' ';
    height: 2px;
    background: red;
    display: block;
    width: 50%;
}


/*begin second box*/
.second{
    width: 300px;
    height: 100px;
    margin: 10px;
    border-width: 2px 0 2px 0;
    border-color: red;
    border-style: solid;
    position: relative;
}

.second span {
    position: absolute;
    top: 0;
    bottom: 0;
    display: flex;
    flex-direction: column;
    align-items: center;
}
.second .foo{
    left: -15px;
}
.second .bar{
    right: -15.5px;
}
.second span:before{
    margin-bottom: 15px;
}
.second span:after {
    margin-top: 15px;
}
.second span:before , .second span:after {
    content: ' ';
    width: 2px;
    background: red;
    display: block;
    height: 50%;
}
<div class="box">
    <div class="first">
        <span class="foo">FOO</span>
        <span class="bar">BAR</span>
    </div>

   <br>

    <div class="second">
        <span class="foo">FOO</span>
        <span class="bar">BAR</span>
    </div>
</div>

На це також відповідають у https://stackoverflow.com/a/57279326/6569224


-1
  • Створено невелике зображення 1x18 з однією крапкою кольору # e0e0e0
  • Перетворив зображення на код base64

Результат:

body
{
  background: #c0c0c0;
}

#block_with_line
{
  width: 100%;
  position: relative;
  height: 18px;
  background: url('');
}

#block_with_line span
{
  background: #e4e4e4;
  width: 150px;
  display: block;
  margin: 0 auto;
  position: absolute;
  left: 50%;
  margin-left: -75px;
  text-align: center
}
<body>
  <div id="block_with_line">
    <span>text</span>
  </div>
</body>

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.