【CSS】Houdini, CSS的成人礼

前情提要

  • CSS:老板,你看ES9,ES10都出来了,您看我的事情什么时候...
  • W3C: 这不是正在走着流程嘛!小C你不要心急!
  • W3C:(语重心长)你看啊,我们先(1)提个开发提案章程, 然后再批准成立工作组(2)紧接着工作组建立标准和指南,然后再修改修改。(3)再然后啊,给顾问委员会技术报告审核一下(4)最后浏览器厂商再实现一下 这不就完事了吗?大概不到10年咱们就全部搞定了吧
  • CSS: 。。。。。。 (难过)
  • W3C:(顿了顿):但是这些年啊,你的刻苦努力我们都是看在眼里的!
  • CSS:所以!!??(兴奋加期待)
  • W3C:所以啊,我们要感谢你为前端社区,所作出的贡献!
  • CSS:。。。。。。(沮丧)
  • W3C:但是呢!考虑到你设备老旧,是该更新一下啦。
  • CSS:请问是!?(兴奋加期待,还有微微的迟疑)
  • W3C:没错,这是老板我送给你的步入成年的礼物—— Houdini

什么是Houdini

Houdini包含6组API

S1. Paint API

简单地讲,Paint API就是允许你通过JavaScript注册一个背景函数,类似于linear-gradient()那种,在定义时候可以提供Canvas的2Dcontext给你自行绘制,你通过JS注册这个背景函数后呢,就可以自由地在CSS中通过以下方式使用该背景

background: paint(背景函数名);

如果想了解Paint API的MDN入门指导请点击这里 ,因为这个MDN上是有入门文档的,所以就不放草案了

S2.Layout API

可以允许你自行定义布局,并通过以下方式使用:

display:layout(自定义布局名)

这意味着什么呢?我们常用的flex布局, 栅格布局都可以重新通过display进行定义和使用,非常方便。甚至,如果/假如Layout API早出来10年的话,我们甚至可以为这些类似于flex等的新兴属性定义polyfill!, 就像我们为ES6中的proxy对象,promise对象分别添加proxy-polyfill,promise-polyfill一样,实现对低级浏览器的兼容,这无疑让人感到高兴。

Layout的定义虽然是为布局使用的,但实际上,根据W3C草案,它还有控制定位和overflow等的能力,总之,除了动画意外的疑难杂症,Layout几乎都能够涉足。如果你想仔细了解:layout的W3C草案

S3. AnimationWorklet

这么关键的场合怎么能少得了动画呢?它可以控制动画的效果。该 API 使用户可以在专有的线程中去运行动画,从而大大降低了主线程的压力。如果想了解AnimationWorklet的W3C草案内容请点击这里,这个草案在今年的6月底刚刚发布

S4. Properties & Values

用于自定义CSS属性并为其约定类型,行为和默认值。可以看作less的@式定义和Sass的$式定义的2.0加强版。顺便一提,它经常和前3个API搭配使用。

S5. Parser API

允许开发者自由扩展 CSS 词法分析器,比如新的媒体规则、新的伪类、嵌套等等。时至今日,它还没有形成一个完整的草案,只是初具模型,大家可以阅读 Parser API草案(W3C)

S6.Typed OM

用人话解释就是,原本我们不是可以通过element.style.cssText去修改CSS样式嘛,大佬们觉得这样的字符串操作麻烦而且速度很慢,所以定义了一组CSS专有的接口对象去进行操作,Typed OM提供了与底层值交互的接口,通过使用专门的JS对象来表示它们。除此外还增加了一些其他的API 点击这里查看MDN的Typed OM的入门指导

为什么说Houdini是CSS的成人礼?

这个问题等同于:

问题:结合CSS的产生背景和历史渊源,请问如何客观评价Houdini的历史地位?(本小题10分,请考生答题时不要超出装订线) ( 难道我们在考历史题吗?逃Σ( ° △ °|||)︴)。

A. 为CSS新特性提供polyfill

它的主要作用在于给予开发者更多开发CSS的自由度,推动CSS新特性发布的进程,同时为未来的那些像flexbox这样优秀的特性提供polyfill,让我们可以不再需要顾忌兼容性,而能够尽快地使用新发布的CSS特性,当然,CSS的生态也会因此更为繁荣。

举个例子,我们在使用ES6的Promise的时候,怎么考虑让它在低版本浏览器不报“Promise is undefined”的错误呢?很简单,只要加个promise-polyfill就可以了,如果你想一本万利,那么导入个babel-polyfill,所有(严格的说是大部分)ES6的新对象都可以放心用了。

试着想一下,假如Houdini比flex早出来10年,这个时候还需要担心flex兼容性吗?不需要了,我们可以从NPM社区上下一个包,这个前人写好的npm包注册了一个layout方法,你只要下载这个包,在index引用运行一下,然后可以肆无忌怛的通过使用flex了,想怎么浪就怎么浪

B. Houdini的作用是为CSS提供进一步的完善

Houdini出来是16年的事情了,当时的话大家都觉得个这个新东西挺有发展潜力的,但是我觉得嘛,要客观看待,Houdini做的事情,其实很多时候,原本我们就可以做了,举个例子:

  1. Paint API: 在JS领域里直接使用Canvas标签和包装起来的相关函数也是可以做的
  2. Layout API:提供的是大幅度的自定义布局的功能,但问题是。。。flexbox和grid已经把大多数场景给肝了。Layout要是早出来还好,这比flex完了这么多年,就感觉有点尴尬
  3. Typed OM:提供了直接操作CSS属性的对象接口,但问题是CSSOM的标准出来也不少时间了呀,相比之下Typed OM的功能好像就失了一些新意。。。(某个表情包老头估计会说:别笔笔,一梭子ele.style.cssText不就完事了吗!)
  4. 其他,想到再更

我们可以理解为,Houdini在为一开始CSS没有设计想要,并设计进去的一些东西做一些补充(XX可能会迟到!但永远不会缺席。但问题是没打到卡是要扣工资的呀~)

  • CSS:大师!我感觉我在传承了Houdini的灵力加持后功力大增!
  • 大师:那尼玛是因为你一开始的内力。。。emmm

啊~下面又到了我最喜欢的互怼环节了呀

  1. 告别CSS
  2. 为什么CSS一开始这么难学
  3. 前端工程师讨厌写CSS是什么心态?

为什么说Houdini是CSS漫长的成人礼?

至今为止,刚刚在can i use上查到的结果显示,目前houdini的七大特性,除了Paint API和TypedOM外,其余几个API在浏览器上可以说都是全军覆没的状态(或者说刚刚萌芽更合适一些?),就算是Paint API也好,它也仅仅只在Edge76,Chrome66和Opera52以上实现了技术扎根, 其余浏览器,哪怕是IE,Firefox,Safari的最新版本,也都尚未实现

下面这张是谷歌上搜到的,2018年底制作,现在可以认为绝大部分仍然适用

此图制作于2018年,到现在为止也仍然使用

可能有同学会问了,上面写的Chrome-Canary是什么意思呢?额。。。canary直译为“金丝雀”,属于和dev,beta相类似的版本的概念范畴,(你就理解为内测版吧,逃~)根据相关资料提示,Layout API在Chrome-Canary上实现部分支持,但是我试用了一下发现还是用不了(心塞),控制台CSS.layoutWorklet打出来是undefined,所以评论区有高人还请指教下啦~

Chrome-Canary中国区下载链接

实战Houdini之Paint API

那咱们就写一个咱们可以在Chrome上跑的demo好了

首先,我们需要编写一个JavaScript文件,我们命名为paint.js,然后通过registerPaint方法注册一个paint方法,paint方法可以绘制div的背景,例如下面我们把这个paint方法命名为circle,这意味着它可以通过background: paint(circle)去使用绘制的画布。我们上面也说过,我们注册paint方法时系统会提供ctx作为参数,这个ctx是HTML5/Canvas的2Dcontext的子集,实现了除了文字操作外的大多数方法和属性。(文字操作指的是ctx.fillText或者ctx.strokeText这一类方法)。

下面我们来通过JS注册一个paint方法,来为div添加一个background,这个paint方法命名为circle,它的功能是为div提供一个半径为Math.min(长,宽)的实心圆,圆的背景色可在CSS的上下文代码块中通过--color这个属性名指定。

paint.js的全部代码如下

// paint.js
registerPaint(‘circle‘, class {
    // 决定了paint方法中props中能获取的CSS属性值
    static get inputProperties() { return [‘--color‘]; }

    // 绘制一个半径为长或宽的最小值的圆形作为背景
    // ctx是Canvas的ctx的子集,实现了除文字操作外的大多数方法和属性
    paint(ctx, size, props) {
        // size表示使用该paint方法的div的长和宽
        const width = size.width / 2;
        const height = size.height / 2;
        const radius = Math.min(width, height);
        // props.get表示将根据inputProperties返回的键值从CSS代码块中获取相应属性
        const color = props.get(‘--color‘);
        // 给画笔着色
        ctx.fillStyle = color;
        // 开始动笔绘制
        ctx.beginPath();
        // 以width,height为圆心,radius为半径画圆圈,从0度画到360度
        ctx.arc(width, height, radius, 0, 2 * Math.PI);
        // 用fillstyle把圆圈轨迹内部进行颜色填充
        ctx.fill();
        // 搞定!
    }
});

(代码中的具体API我们过会再仔细解释,我们先把代码跑通再说)

然后,仅仅这样做是不够的,我们还需要在主线程里,例如main.html的脚本里。通过下面这个API去加载我们刚刚定义的paint.js

CSS.paintWorklet.addModule(‘paint.js‘);

注意,paint.js内部会形成一个封闭而独立的,叫worklet的作用域,它和全局Window是不一样的!不要在这里尝试使用fetch等方法。

上面两步做完了,我们就可以使用CSS去使用我们刚刚定义的paint函数了

<!-- HTML -->
<div class="foo"></div>
<!-- CSS -->
<style>
.foo {
   border: 1px solid blue;
   width: 200px;
   height: 200px;
   /*指定背景色为红色*/
   --color: red;
   /*使用刚刚注册的paint方法*/
   background-image: paint(circle);
}
</style>

我们看下效果

github项目代码如下

https://github.com/penghuwan/houdini-module

好,我们回过头来解释下paint方法里面的逻辑

(1)size 它是一个对象,保存了使用这个定义paint方法的div的长和宽,可以分别通过size.width和size.height去访问

registerPaint(‘circle‘, class {
    // ctx是Canvas的ctx的子集,实现了除文字操作外的大多数方法和属性
    paint(ctx, size, props) {
        // size表示使用该paint方法的div的长和宽
        const width = size.width / 2;
        const height = size.height / 2;
        // ...
    }
});

(2)props:我们可以通过props参数获取CSS上下文代码块(指的是paint(circle)在的那个代码块)的其他CSS属性,但是这个属性需要在inputProperties函数中进行声明,声明的方法是在函数中返回一个数组,数组项为属性名称

// paint.js
registerPaint(‘circle‘, class {
    // 决定了paint方法中props中能获取的CSS属性值
    static get inputProperties() { return [‘--color‘]; }
    // ctx是Canvas的ctx的子集,实现了除文字操作外的大多数方法和属性
    paint(ctx, size, props) {
        const color = props.get(‘--color‘);
    }
});
// style
.foo {
   /*指定背景色为红色*/
   --color: red;
   /*使用刚刚注册的paint方法*/
   background-image: paint(circle);
}

(3)ctx:这个参数上文已经多次介绍,这里不再赘述

警告,下面讲的都是当前没有任何stable浏览器可以运行的代码,我是根据W3C的草案和示范代码要求来的,正所谓——姜太公写代码,愿者Debug

Layout

<抱歉写的太累了,休息下,本部分待会再更>

Animation

<抱歉写的太累了,休息下,本部分待会再更>

Propertis和values

<抱歉写的太累了,休息下,本部分待会再更>

Typed OM

<抱歉写的太累了,休息下,本部分待会再更>

CSS!你的Houdini来啦

但是你用不了,哈哈哈哈!

本文完

参考资料

  1. Houdini:CSS 领域最令人振奋的革新
  2. Chrome提供的Houdini Demo
  3. 1中Demo的在线运行地址(注意很多是暂时跑不了的)
  4. layout的API草案(W3C)
  5. Painting API的入门指导
  6. Parser API草案(W3C)
  7. AnimationWorklet工作草案(W3C)

给大家一点点建议,谷歌英文资料吧。。。很多干货

原文地址:https://www.cnblogs.com/penghuwan/p/11407053.html

时间: 2024-08-27 00:18:52

【CSS】Houdini, CSS的成人礼的相关文章

HTML+CSS笔记 CSS中级 颜色&长度值

颜色值 在网页中的颜色设置是非常重要,有字体颜色(color).背景颜色(background-color).边框颜色(border)等,设置颜色的方法也有很多种: 1.英文命令颜色 语法: p{color:red;} 2.RGB颜色 这个与 photoshop 中的 RGB 颜色是一致的,由 R(red).G(green).B(blue) 三种颜色的比例来配色 p{color:rgb(133,45,200);} 每一项的值可以是 0~255 之间的整数,也可以是 0%~100% 的百分数. 每

CSS 背景-CSS background

这里有个很好的样式学习网站:http://www.divcss5.com/rumen/r125.shtml 一.Css background背景语法   -   TOP CSS背景基础知识 CSS 背景这里指通过CSS对对象设置背景属性,如通过CSS设置背景各种样式. 背景语法: background: background-color || background-image || background-repeat || background-attachment || background-

CSS Sprites (CSS图像拼合技术)教程工具

什么是CSS Sprites? “Sprite”(精灵)这个词在计算机图形学中有它独特的定义,由于游戏.视频等画质越来越高,必须有一种技术可以智能的处理材质和贴图,并且要 同时保持画面流畅.“Sprite”就是这样一种技术,它将许多图片组合到一个网格上,然后通过程序将每个网格的内容定位到画面上. Sprite被定位到一副静态图片上,并且通过简单的程序或硬件即可正确定位到画面上,一幅幅图片就像是被“变”出来的,他们并没有单独占用内存,所以被取名为“Sprite精灵”. 上图是口袋妖怪的组合图片,可

#CSS# 初识CSS

CSS:Cascading Style Sheet (层叠样式表) CSS的内容可以分开写入CSS文件中,PHP文件中加入以下语句即可: <link herf="css/style.css" type="text/css" rel="stylesheet"/> 其中,style.css写入定义的样式,如: ul.menu{                       list-style-type:none;    backgroun

[29/May/2017 10:56:52] &quot;GET /static/css/stylee.css HTTP/1.1&quot; 404 1859

今天遇到了一个棘手的问题,也非常奇怪. 出现了一个找不到css样式表的问题,之前配置好static路径之后也做了很小的测试:运行没有问题,css也能够正常加载. 问题描述:style.css文件的名字不能改变.仿佛系统只能识别style.css文件名.新建其他文件也不行,结果都是404错误,css文件不能正常加载. 我的环境是Django 1.10,目录和配置是按照官方Doc上来创建和配置的. 文件目录如下: setting.py: STATIC_URL = '/static/' STATIC_

HTML中使用CSS -- 定义CSS样式

CSS:Cascading Style Sheets(层叠样式表) 有三种方式可以在XHTML页面中嵌入CSS.参见CSS教程. 1.在head标签中加载一个CSS文件 示例 <head> <link rel="stylesheet" type="text/css" href="style.css" /> </head> 此例使用了link标签. abbr { font-size:12px; } .text10

CSS学习------CSS定位(position)与浮动(float)

position属性:用来对元素进行定位 定位的意义: 定位允许你定义元素框相对于其正常位置应该出现的位置. 相对于父元素. 相对于另一个元素. 相对于浏览器窗口本身的位置. 定位分为:绝对定位和相对定位. 相对定位: 如果对一个元素进行相对定位,它将出现在它所在的位置上.然后,可以通过设置垂直或水平位置,让这个元素"相对于"它的起点进行移动. 在使用相对定位时,无论是否进行移动,元素仍然占据原来的空间.因此,移动元素会导致它覆盖其它框. 示例: <html>   <

【CSS】CSS样式的三种引用方法

什么是CSS?CSS是 Cascading Style Sheets 的首字母缩写,意思是层叠样式表.有了CSS,html中大部分表现样式的标签就废弃不用了,html只负责文档的结构和内容,表现形式完全交给CSS,html文档变得更加简洁.为了让网页元素的样式更加丰富,也为了让网页的内容和样式能拆分开. 那么,怎么在HTML中引入Css? 第一个方法是:内联式(行间样式): <body> <!--内联式(行间样式)--> <div style="color:red&

HTML/CSS从零开始-CSS基础(二)

一.css定义 (cascading style sheets) 层叠样式表 表现标准语言,对网页信息显示进行控制 二.css的建立 2.1内部建立 (建立在head里用标签<style>)语法:      <head>      <style type="text/css">      div{widtn:300px;height:300px;border:1px solid red;}      </style>      说明:(1