CSS 世界中的方位与顺序

在 CSS 中,咱们常常会与各类方向方位打交道。css

譬如 marginpadding,它们就会有 margin-leftmargin-right 或者是 padding-leftpadding-right。还有定位中的 lefttoprightbottom,它们表示了上下左右不一样的方位。html

还有一种状况是从x方位到x方位,譬如 writing-modedirection,它表明了一种顺序,表示块流动方向,或者文字书写的方向等。前端

本文将捋一捋 CSS 世界中的方位与顺序,探寻其中一些有意思的点。git

writing-mode & direction & unicode-bidi

在 CSS 世界中,这 3 个属性都与排版顺序相关,互有关联但做用各异。github

  • writing-mode:定义了文本水平或垂直排布以及在块级元素中文本的行进方向。
  • direction:设置文本排列的方向。 rtl 表示从右到左 (相似希伯来语或阿拉伯语), ltr 表示从左到右。
  • unicode-bidi:它与 direction 很是相似,两个会常常一块儿出现。在现代计算机应用中,最经常使用来处理双向文字的算法是Unicode 双向算法。而 unicode-bidi 这个属性是用来重写这个算法的。

单纯看定义有点懵逼,咱们简单的看几个应用示意图:web

writing-mode 示意

writing-mode 基本只须要留意最多见的 horizontal-tbvertical-lrvertical-rl 。表示文本的行进方向,下图表示浏览器对 writing-mode 的支持完整的状况下输出的外观:算法

direction 示意

OK,那 direction 又为什么呢?它表示文本排列的方向浏览器

  • direction: ltr:默认属性。可设置文本和其余元素的默认方向是从左到右。
  • direction: rtl:可设置文本和其余元素的默认方向是从右到左。

有点绕,因此上 Demo 最为直观。假设,咱们有以下结构:ide

<ul class="wrap">
    <li>1</li>
    <li>2</li>
    <li>3</li>
    <li>4</li>
</ul>
<p>这是一段正常顺序的文本</p>

简单的 CSS 以下:wordpress

p, ul {
    background: #ff00ff50;
    padding: 10px;
}
ul {
    display: flex;
    justify-content: space-between;
    
    & > li {
        border: 1px solid #333;
    }
}

正常状况下的样式以下:

咱们分别给两组元素的父容器 <p><ul> 加上 direction: ltrdirection: rtl,则最终效果以下:

能够看到,direction 能够改变子元素的排列方向,可是它确没法改变单段文本内(或是内联元素内),每个文字的书写顺序

那若是,我但愿 这是一段正常顺序的文本 这段文字,不是从左向右进行书写,而是反过来,从右到左进行书写,又该如何设置呢?

unicode-bidi 示意

这就须要请出 unicode-bidi 了。

单独使用 direction: rtl 没法使单段文本内(或是内联元素内),文字的书写顺序改成从右至左。须要配合 unicode-bidi

CSS 中的 unicode-bidi 属性,和 direction 属性,共同决定如何处理文档中的双书写方向文本。

仍是上述的代码,咱们改造一下:

<p>这是一段正常顺序的文本</p>
p {
    direction: rtl;
    unicode-bidi: bidi-override;
}

结果以下:

放到一块儿比较:

这里除了 unicode-bidi: bidi-overrideunicode-bidi: isolate-override 也能获得一样的效果。

这里涉及了一个很是重要的知识 -- Unicode 双向算法

Unicode 双向算法

双向文字就是一个字符串中包含了两种文字,既包含从左到右的文字又包含从右到左的文字。

对于文字书写习惯,分为:

  1. 大多数文字都是从左到右的书写习惯:好比拉丁文字(英文字母)和汉字;
  2. 少数文字是从右到左的书写方式好比阿拉伯文(ar)跟希伯来文(he)。

在现代计算机应用中,最经常使用来处理双向文字的算法是 Unicode 双向算法(Unicode Bidirectional Algorithm)。

一个区域内有整体方向,决定从这个区域的哪边开始书写文字,一般称为基础方向。浏览器会根据你的默认语言来设置默认的基础方向,如英语、汉语的基础方向为从左到右,阿拉伯语的基础方向为从右到左

在 Web 中,咱们有 3 种方式能够控制文字方向:

  1. html实体 - &lrm;&rlm;
  2. <bid><bdo> 标签 与 dir 属性
  3. CSS 属性 direction + unicode-bidi

本文介绍的就是 CSS 中的 direction + unicode-bidi 方式控制文字的书写方向。关于 Unicode 双向算法(Unicode Bidirectional Algorithm)自己仍是很是复杂的,本文也仅仅只是简单说起,更为详尽的内容,你能够参考 UNICODE BIDIRECTIONAL ALGORITHM

writing-mode & direction & unicode-bidi 的一些应用

除去自己的功能,下面咱们来看看它们其它的一些应用场景。

使用 writing-mode 进行创意排布布局

writing-mode 很是适合用于进行一些创意排版。

基础的相似中国古诗词的一些竖向展现:

<div class="g-wrap">
    <h2>凉州词</h2>
    <p>葡萄美酒夜光杯,</p>
    <p>欲饮琵琶立刻催。</p>
    <p>醉卧沙场君莫笑,</p>
    <p>古来征战几人回。</p>
</div>

.g-wrap 分别添加 writing-mode: vertical-rl 或者 writing-mode: vertical-lr 获得不一样的效果:

.rl {
    writing-mode: vertical-rl;
}
.lr {
    writing-mode: vertical-lr;
}

CodePen Demo -- display poems by writing-mode

又或者像是这样,利用 writing-mode:vertical-rl 实现标题的竖向排列,搭配内容造成有意思的报纸排版:

<div>
  <h2>Title Loomings</h2>
  <p>Call me Ishmael. Some years ago- never mind ho....
  </p>
</div>
div {
  width: 750px;
  padding-left: 150px;
}
h2 {
  position: absolute;
  writing-mode: vertical-rl;
}

获得这样的排版布局:

CodePen Demo -- writing-mode Layout Demo

改变文本溢出省略位置,使之在头部进行省略

咱们都知道,本文超长溢出的省略,经过都是在文本的最末尾。像是这样:

<p>Make CSS Ellipsis Beginning of String</p>
p {
    overflow: hidden;
    text-overflow: ellipsis;
    white-space: nowrap;
}

这里,咱们能够经过 direction,将省略打点的位置,从尾部移动至头部:

p {
    direction: rtl;
}

结果以下:

尝试了下运用在多行省略中,多行省略的打点会出如今最后一行的左侧,不符合需求。

CodePen Demo -- CSS Ellipsis Beginning of String

使用 writing-mode 改变元素方位

这个小技巧是在张老师的博客中学到的:改变CSS世界纵横规则的writing-mode属性

咱们能够经过 writing-mode: vertical-rl,将元素转一个 90° 角:

<div>➤</div>
div:hover {
    writing-mode: vertical-rl;
}

看看效果,当 hover 的时候,将箭头从向右➡️改成向下🔽 :

固然,如今这个功能彻底能够用 transform 替代,可是在以前须要兼容 IE 系列的时候,不失为一个有意思的小技巧。

CSS 中的逻辑属性

下面一个章节,咱们聊聊 CSS 中的逻辑位置。

咱们知道,在咱们使用相似 marginpadding 的时候,能够单独控制每一个方向,例如 margin-toppadding-left

然而,这种使用了 top/left/bottom/right 物理方向维度定义的属性,在不一样的排版规则下,就很是容易出问题。

思考以下这样一个 DEMO,咱们但愿给古诗的题目的上方,添加一个 padding 值:

<div class="g-wrap pt">
    <h2>凉州词</h2>
    <p>葡萄美酒夜光杯,</p>
    <p>欲饮琵琶立刻催。</p>
    <p>醉卧沙场君莫笑,</p>
    <p>古来征战几人回。</p>
</div>
<div class="g-wrap pt rl">
    <h2>凉州词</h2>
    <p>葡萄美酒夜光杯,</p>
    <p>欲饮琵琶立刻催。</p>
    <p>醉卧沙场君莫笑,</p>
    <p>古来征战几人回。</p>
</div>
.pt {
    padding-top: 100px;
}
.rl {
    writing-mode: vertical-rl;
}

能够看到,不管 writing-mode 如何,padding-top 始终指代物理方向的上方。

基于这种不一样排版规则,物理方向可能会带来必定的困扰这个问题,CSS 在 CSS Logical Properties and Values Level 1 规范中,推出了 CSS 逻辑属性。

CSS 逻辑属性与值是 CSS 的一个新的模块,其引入的属性与值能作到从逻辑角度控制布局,而不是从物理、方向或维度来控制。

仍是上述的 DEMO,咱们可使用 padding-block-start 替代 padding-top

重点:使用 padding-block-start 替代 padding-top

.pt {
-   padding-top: 100px;
+   padding-block-start: 100px;
}
.rl {
    writing-mode: vertical-rl;
}

此次再看看效果:

padding 的位置由物理上的上方,变成了逻辑上的上方。

完整的 Demo 你能够戳这里:CodePen Demo-- 物理方向与逻辑方向展现

margin、padding、border、relative 物理属性到逻辑属性的映射

相似这样的属性,在规范中定义了挺多的,简单罗列一下具体的映射规则:

margin 物理属性到逻辑属性的映射:

Property 属性 Logical Property 逻辑属性
margin-top margin-block-start
margin-left margin-inline-start
margin-right margin-inline-end
margin-bottom margin-block-end

padding 物理属性到逻辑属性的映射:

Property 属性 Logical Property 逻辑属性
padding-top padding-block-start
padding-left padding-inline-start
padding-right padding-inline-end
padding-bottom padding-block-end

border 物理属性到逻辑属性的映射:

Property 属性 Logical Property 逻辑属性
border-top{-size|style|color} border-block-start{-size|style|color}
border-left{-size|style|color} border-inline-start{-size|style|color}
border-right{-size|style|color} border-inline-end{-size|style|color}
border-bottom{-size|style|color} border-block-end{-size|style|color}

relative 物理属性到逻辑属性的映射:

Property 属性 Logical Property 逻辑属性
top inset-block-start
left inset-inline-start
right inset-inline-end
bottom inset-block-end

在逻辑属性中没有方向性的概念,只有开始(start)和结束(end)、块(block)和内联(inline)的概念。好比说,在从左到右(LTR)中,start 是 left,但在从右到左(RTL),它是 right。

逻辑属性下的盒子模型

考虑到不一样排版带来的逻辑问题,整个盒子模型也能够随之进行改变。

下图,左边是物理盒子模型,右边是逻辑属性下的盒子模型。

左:物理盒子 | 右:逻辑盒子

物理方向与逻辑方向重叠

固然,还有这样一种状况,就是设置的逻辑方向和物理方向重叠,譬如咱们给一个正常从左往右,从上至下的元素同时设置 padding-toppadding-block-start,看看会发生什么:

div {
    padding-top: 120px;
    padding-block-start: 100px;
}

这里若是物理方向与逻辑方向设置的 padding 重叠,将会取两个值中后面定义的那个。这里因为 padding-block-start 后于 padding-top 定义,因此 padding 的值为 100px

marginborder 同理。这里个人理解是,同个方向上仍是只能存在一个 margin\padding\border,不管是逻辑方向仍是物理方向,取后定义的值为准。

CodePen Demo-- 物理方向与逻辑方向重叠 DEMO 展现

总结一下

总结一下,当项目开始国际化,当国内更多的业务开始出海,国际化兼容适配也会愈来愈重要。好在 CSS 也一直在紧跟时代,推陈出新,当你的排版布局须要考虑不一样的 writing-mode 的时,你须要开始考虑使用逻辑属性替代物理属性!

最后

好了,本文到此结束,但愿对你有帮助 😃

想 Get 到最有意思的 CSS 资讯,千万不要错过个人公众号 -- iCSS前端趣闻 😄

更多精彩 CSS 技术文章汇总在个人 Github -- iCSS ,持续更新,欢迎点个 star 订阅收藏。

若是还有什么疑问或者建议,能够多多交流,原创文章,文笔有限,才疏学浅,文中如有不正之处,万望告知。

参考文章

相关文章
相关标签/搜索