摘要:本期主要介紹以下幾個插件和幾個坑和坑這個就不用多說了,必裝插件之一。看似寫法沒有任何問題。編譯后的結果那么這樣就沒有問題了。
PostCSS常用插件介紹
繼上一次PostCSS學習指南(一)后,漸漸開始在項目中應用。
這次決定主要講解一些個人認為非常有幫助的PostCSS插件。
autoprefixer本期主要介紹以下幾個插件和幾個坑
autoprefixer
postcss-partial-import
postcss-advanced-variables
cssnano
postcss-px2rem
precss
postcss-nesting和postcss-nested
坑( ??? )?
這個就不用多說了,必裝插件之一。方便的寫規范的css,它會為你提供非常完整的hack兼容方案的。當然這里需要了解一下的是,它的大部分兼容數據來源Can I Use,另外一個稍微需要了解的插件配置參數就是browsers,比如這樣寫:
module.exports = { plugins: [ require("autoprefixer")({ browsers: "last 2 versions" }) ] }
關于這個參數的詳細介紹可以看看Browserslist Queries,文中說了,強烈建議將查詢寫入package.json(后面會告訴你為何要寫在這里),而非配置postcss.config.js中autoprefixer的browsers參數。所以此處建議寫法如下:
postcss.config.js
module.exports = { plugins: [ require("autoprefixer"); ] }
package.json內增加如下示例
"browserslist": [ "> 1%", "last 2 versions" ]
這里我對著官方文檔簡單說一下數組內的值對應的含義:
last 2 versions: 每個瀏覽器中最新的兩個版本。
> 5% or >= 5%: 全球瀏覽器使用率大于5%或大于等于5%(上例中則是1%)。
其他的一些參數簡單介紹:
ie 6-8: 選擇包含ie6-8的版本。
Firefox > 20: 火狐版本號大于20。
還有很多不一一列舉,這里的配置還是很詳細的,一般來說最省事的就是不加參數,按照默認即可。
需要配置的話,就在package.json里面添加browserslist參數,這樣其他插件也能夠從中獲取到項目將要兼容的版本,目前包含以下幾個插件會讀取改配置:
Autoprefixer
babel-preset-env (no config support, only tool option)
eslint-plugin-compat
stylelint-no-unsupported-browser-features
postcss-normalize
關于autoprefixer的介紹差不多就這些了~
postcss-partial-import這個沒啥好說的,也是很容易理解的插件,就是讓你的css文件支持@import,支持W3C的寫法也支持SASS那種寫法,這里就不多說啦。
postcss-advanced-variables同樣的,像SASS那樣可以自定義變量并進行引用,用法也十分簡單,相信大家一定不用點開官方文檔也會用的~(你肯定還是點開了哈哈哈~你這叫胡開鏈接綜合癥,生怕錯過了啥內容。)
cssnano很顯然這個插件作者比較高調,github的cssnano上面是沒有什么說明和介紹的,當然官方也寫得很詳細了。這個插件給我的感覺就是css代碼壓縮工具(實際上它集成了很多強化的功能,表面上看起來是壓縮實際上進行了相當多處理,可以看看這個表格Optimisations),他的配置建議采用默認配置,除非你知道你在做什么。
當然我在測試使用中遇到了一點點問題,關于cssnano和autoprefixer結合使用,或者說是postcss.config.js插件引入順序有關的造成輸出不同的問題?我暫時還在研究,先看代碼:
testcssnano.css
.test { -moz-border-radius: 10px; border-radius: 10px; display: flex; }
postcss.config.js(第一種)
module.exports = { plugins: [ require("cssnano")({ preset: "default", }), require("autoprefixer") ] }
輸出結果:
.test{border-radius:10px;display:flex}
postcss.config.js(第二種)
module.exports = { plugins: [ require("autoprefixer"), require("cssnano")({ preset: "default", }) ] }
輸出結果:
.test{border-radius:10px;display:-webkit-box;display:-ms-flexbox;display:flex}
問題一,如果手寫-moz-,cssnano會清除,可見示例,當然或許這個屬性在火狐已經支持不需前綴所以,就被去掉了。
問題二,如果在配置文件中,cssnano在autoprefixer之前引用,假設根據webpack的loader執行順序規則相同的話,大概postcss.config.js中的插件也是這樣由下而上依次執行,因此,在第一種例子中,css代碼先被autoprefixer處理,然后再執行cssnano清除了那些多余的前綴代碼?大致可分解為:
第一步autoprefixer處理結果:
.test { -moz-border-radius: 10px; border-radius: 10px; display: flex; display: -webkit-box; display: -ms-flexbox; }
第二步cssnano處理結果:
.test{border-radius:10px;display:flex}
可是我要遺憾的告訴大家。。。這個可能是錯誤的結論!!!
因為我將第一步的結果作為初始css,將postcss.config.js中僅引用一個cssnano插件來執行的結果如下:
.test{border-radius:10px;display:flex;display:-webkit-box;display:-ms-flexbox}
因此,這個問題就來了。。。cssnano確實將-moz-去掉,但是他并沒有處理其他的關于display的兼容代碼。
于是我再做了一個測試:
我將css代碼改為:
.test { -webkit-transform: scale(.5) translate(10, 20); -moz-transform: scale(.5) translate(10, 20); -ms-transform: scale(.5) translate(10, 20); -o-transform: scale(.5) translate(10, 20); transform: scale(.5) translate(10, 20); }
而postcss.config.js還是同時引入cssnano和autoprefixer,經過測試,無論誰先誰后輸出結果均為:
.test{-webkit-transform:scale(.5) translate(10,20);transform:scale(.5) translate(10,20)}
如果css代碼改為:
.test { transform: scale(.5) translate(10, 20); }
postcss.config.js同時引入cssnano和autoprefixer,
第一種cssnano在前結果:
.test{transform:scale(.5) translate(10,20)}
第二種autoprefixer在前結果:
.test{-webkit-transform:scale(.5) translate(10,20);transform:scale(.5) translate(10,20)}
結果又不一樣,不知道看到這里大家暈了沒有。
根據觀察,postcss.config.js的插件執行順序是有要求的,至少在同時使用cssnano和autoprefixer的時候,cssnano一定要在autoprefixer之后,否則autoprefixer可能會失效!
另外cssnano和autoprefixer都會對很舊的兼容寫法進行精簡,例如上文有一段有很多個transform屬性的css代碼,autoprefixer會(根據browserslist)剔除其他的。
至于順序問題,我后面繼續進行研究!
postcss-px2rem做移動端,適配是個頭疼的問題,不過我目前還是使用想對穩定的方案flexible,那么就需要用rem來做主要的單位,這里不說適配問題只說這個插件,一般移動端,設計師給的設計稿都是750px寬,你只需要下面這樣設置,就可以直接在代碼里面寫你在PSD量的像素值了,這真是太令人激動了!
require("postcss-px2rem")({ remUnit: 75, threeVersion: true })
因為這個postcss-px2rem又是來源于px2rem的,所以詳細的說明見px2rem,我這里寫了兩個參數,一個remUnit,這個對應的是每rem對應的px值,既然750px,就寫75啦,不知道這樣理解對不對。另一個threeVersion則對應三個不同dpr下的大小,這個比較少用,需要注意的是處理這些參數,是否轉換rem都和注釋有關,這里就不細說了,看看文檔就好~當然這里也埋下一個坑,等下會提到。
precss這就是個大雜燴,主要是為了滿足SASS開發者的習慣,繼承了很多插件,本篇前后文都有提及precss內的部分插件,如果并不是全部用到,我建議還是一個個手動安裝所需插件來進行配置,這東西,配好了以后也就不會經常改動了,而且個人認為用大雜燴很容易出現一些坑,又很難排查,例如下面兩個插件,大家仔細看看。
另外還有幾個插件,建議安裝(當然如果你完全不知道干啥的可以忽略):
postcss-mixins 一個和SASS的mixins用法相同的插件
postcss-atroot 讓你的嵌套css處于根部,官方有個bar.css的@import的例子很棒,可以看看舉一反三
postcss-extend 有相同結構卻有那么一點點不同的區別,用這個可以方便的統一管理相同部分樣式代碼
其他的一些我個人覺得不常用或者說實用意義不大所以就沒有寫出來了。
postcss-nesting和postcss-nested這兩個也是從precss里面拿出來的,就是仿SASS的嵌套css寫法用。為啥把這兩個放一起寫,因為他們長的太像了。只看名字鬼知道他們的區別,然而他們都被加入到precss里面去了。據precss介紹,他們兩個的區別是:
postcss-nesting: W3C nested selectors
postcss-nested: Sass-like nested selectors
光看這兩行我是看不懂到底是個啥玩意,難道第一個是符合W3C規范的嵌套選擇?粗略看了下兩個插件的說明文檔,沒看出沒啥區別。行,那手動寫代碼來一個個試試。先安裝postcss-nesting,編譯試試,嘩嚓一片紅。。。咋回事,我們先看看代碼片段。
.catis-list { padding: 0 50px; overflow: hidden; li { list-style: none; float: left; margin-top: 38px; width: 113px; &:not(:nth-child(4n)) { margin-right: 66px; } .iconfont { font-size: 100px; line-height: 100px; color: #506071; display: block; text-align: center; } .cati-name { font-size: 28px; line-height: 40px; display: block; color: #999; text-align: center; } } }
看似SASS寫法沒有任何問題。可是它提示的報錯信息讓人看不大懂
ERROR in ./src/style/index.css Module build failed: ModuleBuildError: Module build failed: Error: undefined:783:6: property missing ":" at error (D:webProjectsmobileweb ode_modulescsslibparseindex.js:62:15) at declaration (D:webProjectsmobileweb ode_modulescsslibparseindex.js:223:33) at declarations (D:webProjectsmobileweb ode_modulescsslibparseindex.js:252:19)
啥玩意missing了個冒號嘛。。。改來改去都不對。索性拿來官方的實例。才看清楚。原來每個嵌套的樣式前面都需要一個&(注意符號后面有個空格),實際上應該如下才對:
.catis-list { padding: 0 50px; overflow: hidden; & li { list-style: none; float: left; margin-top: 38px; width: 113px; &:not(:nth-child(4n)) { margin-right: 66px; } & .iconfont { font-size: 100px; line-height: 100px; color: #506071; display: block; text-align: center; } & .cati-name { font-size: 28px; line-height: 40px; display: block; color: #999; text-align: center; } } }
這里面有一段用到偽類其中&符號后面是沒有空格的,是正確的。編譯后的結果:
.catis-list { padding: 0 0.666667rem; overflow: hidden; } .catis-list li { list-style: none; float: left; margin-top: 0.506667rem; width: 1.506667rem; } .catis-list li:not(:nth-child(4n)) { margin-right: 0.88rem; } .catis-list li .iconfont { font-size: 1.333333rem; line-height: 1.333333rem; color: #506071; display: block; text-align: center; } .catis-list li .cati-name { font-size: 0.373333rem; line-height: 0.533333rem; display: block; color: #999; text-align: center; }
那么這樣就沒有問題了。和官方的說明的是一樣的,但是另一方面,如果每次都要寫&加空格,那豈不是很麻煩,習慣寫SASS的兄弟們肯定不愿意這樣做啦。
那么我要說的就是另一個插件postcss-nested,如precss所述,這個的確是SASS-LIKE了。當然我暫時還不太明白為何precss要收入兩個nest插件(為了滿足不同開發人員的習慣?)。我們修改postcss.config.js使用postcss-nested,并重新修改樣式代碼之前第一段,再次編譯執行,一切OK,那么結論就是,如果僅習慣于SASS的嵌套寫法,安裝postcss-nested插件即可~
坑( ??? )?最后來說說我遇到的坑,除了剛才說的cssnano和autoprefixer同時使用需要注意順序問題。還有另一個順序研究,就是確定好將要使用的插件后,在postcss.config.js中配置插件require順序還是有講究的,這里我個人觀察的確還是從上往下的(至于是不是每個插件輪流處理完文件后挨個執行我尚不確定),比如說,postcss-partial-import這個理應是第一個引入的,你若是把它放在最后面,而css代碼中第一行就用了@import那肯定會報錯!所以,建議根據css代碼的寫法來決定你的執行順序。
postcss-nested插件是大部分SASS開發者所喜愛的,但是你在css文件中用sass寫法會遇到以下幾個問題:
csslint,css文件不支持嵌套,變量等寫法,如果你將文件模式改為sass,注釋的方式會變成//,而非/* comments */,當然你可以手寫/* ... */這樣的注釋,但是用快捷鍵進行注釋會很痛苦。
這里有個小技巧,讓項目所有css文件均為sass模式下編輯,在項目settings.json添加:
"files.associations": { "*.css": "scss" }
當然你若是想要支持//注釋也是可以的,請再安裝postcss-scss插件,我這里不多說這個了,因為我已經決定手寫注釋了?
你手寫注釋沒有問題,然而編譯出來的東西會出問題,你樣式中最后一行如果將注釋寫在花括號內部,它轉換出來的代碼,注釋會在外部,這是個大坑,因為在使用postcss-px2rem的時候,那注釋來控制是否轉換的功能就失效了。我文字描述可能讓人迷糊,所以看看代碼:
例如CSS部分代碼:
.test { color:#999; border:1px solid #ffffd; /* no */ .inner { color:#333; } }
轉換后:
.test { color:#999; border:0.13333rem solid #ffffd; } /* no */ .test .inner { color:#333; }
而不是我想要的:
.test { color:#999; border:1px solid #ffffd; } .test .inner { color:#333; }
理論上是應該輸出我想要的結果,卻沒有輸出正確,錯誤的將1px的border轉成rem,原因就是先執行的postcss-nested將注釋弄在外部后,postcss-px2rem無法識別到它的規則了。
我一開始也是裝了precss后發現該問題,后來查了很久才發現是這個插件的問題。同時也發現了這個人其實已經提過issue了Wrong location of comment of the last declaration in a nested rule definition只是沒人解決。
而我臨時處理的方法就是只好將要注釋的那段代碼不要寫在最后一行了。如下:
.test { border:1px solid #ffffd; /* no */ color:#999; .inner { color:#333; } }
可能有時候并不會有兩個樣式給border墊底,如果只有一行border的樣式,就只能這樣:
.test { .inner { color:#333; } border:1px solid #ffffd; /* no */ }
這樣倒是可以了,不過看起來很奇怪,所以如果你使用和我相同的處理方式時,務必注意此點(寫完本文時我似乎發現了更好的解決方案,等確認了沒有問題后,放在下一期來寫~)。
(我記得好像還有坑的,硬是想了半天想不起來了。。。那就先這樣吧)
本期結語最后如果你還想探索更多好玩的好用的插件,你可以看看這里PostCSS Plugins List還有這個可以搜索的分類插件列表,如果你發現了更棒的插件,也歡迎留言喔~
另外上面的例子如果我沒有寫的很明白,或者你只是想直接拿來快速測試使用的話,可以看看這個mobileweb
(關于第三期PostCSS的內容還在考慮。。。可能是對第二期的補充和填坑~敬請期待?)
其他關于我個人的PostCSS一系列學習, 介紹及總結, 有興趣可以參閱:
PostCSS自學筆記(一)【安裝使用篇】
PostCSS自學筆記(二)【插件篇】
PostCSS自學筆記(二)【番外篇一】
PostCSS自學筆記(二)【番外篇二】
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/112501.html
摘要:之前有研究過做過假設,在插件列表中,的插件執行順序自上而下,一切看起來似乎是沒有任何問題的。再有摘自深入設計摘自寫的姿勢這兩張圖則應該是說明了我之前的假設,插件中的執行順序自上而下。先來看看一片來自的這段會不會跟這個有關呢,我先埋個伏筆。 圖解PostCSS的插件執行順序 文章其實是一系列的早就寫完了. 才發現忘了發在SegmentFault上面, 最早發布于https://gitee...
摘要:通過配置規則和單位使用或來解決。其他關于我個人的一系列學習介紹及總結有興趣可以參閱自學筆記一安裝使用篇自學筆記二插件篇自學筆記二番外篇一自學筆記二番外篇二 利用PostCSS解決移動端REM適配問題 上一期有提到結合postcss-px2rem插件來處理移動端適配的方案,以及相關的避坑方法,之后總覺得這個解決方案問題太多,也就誕生了另一套方案運用postcss-pxtorem插件來進行...
摘要:而則可制定個人需求的一套解決方案僅安裝需要的插件。迫不及待的你已經等不及安裝使用了吧。安裝及使用一般是結合自動化工具使用,如果要單獨使用可以安裝,這里我先對的安裝使用講解下。接下來說點實際的,如何利用結合自動化工作在項目中使用。 PostCSS介紹 PostCSS是一個利用JS插件來對CSS進行轉換的工具,這些插件非常強大,強大到無所不能。其中,Autoprefixer就是眾多Post...
摘要:從再到目前當紅明星,前端模塊打包技術日新月異,在今年月份和月份左右接連更新了和版本為了減少冗余模塊,縮減文件大小,中也加入了關于的特征,可以查看知乎如何評價新引入的代碼優化技術的討論。 從Grunt->gulp->webpack,再到目前當紅明星rollup,前端模塊打包技術日新月異,webpack在今年1月份和6月份左右接連更新了v2和v3版本,為了減少冗余模塊,縮減bundle文件...
摘要:單元測試秉承測試驅動開發的開發理念,單元測試的任務是必不可少的。維護一份按照建議,也將更新歷史等數據放在了一個名為文件上,并采用語義化的版本號。 本文原始來源:http://devework.com/postcss-p...。轉載請提供原始來源,謝謝! showImg(https://segmentfault.com/img/bVHtqu?w=2028&h=612); 前陣子為了滿足工...
閱讀 486·2019-08-30 15:44
閱讀 902·2019-08-30 10:55
閱讀 2734·2019-08-29 15:16
閱讀 934·2019-08-29 13:17
閱讀 2806·2019-08-26 13:27
閱讀 576·2019-08-26 11:53
閱讀 2125·2019-08-23 18:31
閱讀 1892·2019-08-23 18:23