移動端視網膜(Retina)屏幕下1px邊框線 解決方案


原因:

因為Retine屏的分辨率始終是普通屏幕的2倍,1px的邊框在devicePixelRatio=2的retina屏下會顯示成2px。

但在IOS8中,已經支持0.5px了,那就意味着, 在devicePixelRatio=2的時候,我們可以使用如下的css:

 

div{
     border:1px solid #000;
}    

@media (-webkit-min-device-pixel-ratio: 2) {
    div{
          border:0.5px solid #000;
    }
}

但在ios7以下,android等其他系統里,0.5px會被顯示為0px,所以需要寫hack來兼容舊版本的系統。

有兩種方案:

1、JS判斷瀏覽器版本,是否是IOS8+,是的話則加上hairlines的類名,加在head里即可。

CSS代碼:

div{
      border:1px solid #000
}

.hairlines div{
      border-width:0.5px
}

JS代碼:

if (window.devicePixelRatio && devicePixelRatio >= 2) {
      document.querySelector('body').classList.add('hairlines')
}

  

2、JS判斷是否支持0.5px的邊框,是的話,則加上hairlines的類名。(代碼放在body內)

if (window.devicePixelRatio && devicePixelRatio >= 2) {
      var testElem = document.createElement('div');
      testElem.style.border = '.5px solid #000';
      document.body.appendChild(testElem);

      //當div存在
      if (testElem.offsetHeight == 1){
        document.querySelector('html').classList.add('hairlines');
      }

      //添加完hairlines類名后,則刪除div
      document.body.removeChild(testElem);
}

將代碼放在body內會有一些重繪,第一種方法會更好一些。

 

當然這些方法,只能兼容IOS7包括IOS7以上的機器,但如果想兼容其他機器怎么辦呢?這塊我就要細細道來...

在retina屏下面,如果你寫了這樣的meta <meta name="viewport" content="initial-scale=1, maximum-scale=1, minimum-scale=1, user-scalable=no"> 你將永遠無法寫出1px寬度的東西。

關於CSS像素和物理像素的定義,這個文章里有詳細的介紹  走向視網膜(Retina)的Web時代

在 viewport 中, 因為設置了initial-scale(表示初始時的縮放比例),minimum-scale(最小縮放比例)和maximum-scale(最大縮放比例)都為1, 因此整個頁面都不能縮放, retina 屏幕下1個 css 像素對應 2個(多個)物理像素, 因此我們永遠寫不出1px(物理)寬度的東西. 然而我們其實可以這樣寫:

<meta name="viewport" content="initial-scale=0.5, maximum-scale=0.5, minimum-scale=0.5, user-scalable=no">

淘寶M端 就是這樣的方案,這樣一個css的像素就能完美對應1個物理像素,我們就可以寫出1px的東西了。

 

其他方案:

1. transform: scale(0.5)

方法1:

 CSS代碼:

div{
        height:1px;
        -webkit-transform: scaleY(0.5);
        -webkit-transform-origin:0 0;
        overflow: hidden;
        background: #000;
}
  

  

HTML代碼:

<body>
   1px邊框線
   <div></div>
</body>

 

缺點

  圓角無法實現,實現4條邊框比較麻煩,並且只能單獨實現,如果嵌套,會對包含的效果產生不想要的效果,所以此方案配合:after和before獨立使用較多。

  比如畫一個商品的邊框四條線,容器的after和before可以畫2條線,利用容器的父元素的after、before再畫2條線。

div{
   position: relative;
}

div:after{
   content:"";
   position: absolute;
   bottom:0px;
   left:0px;
   right:0px;
   border-bottom:1px solid #000;
   -webkit-transform:scaleY(.5);
   -webkit-transform-origin:0 0;
}

  

2. box-shadow

實現方法:

利用CSS對陰影處理的方式實現0.5px的效果。

-webkit-box-shadow:0 1px 1px -1px rgba(0, 0, 0, 0.5);

優點:

基本所有場景都能滿足,包含圓角的button,單條,多條線。

缺點:

  • 顏色不好處理, 黑色 rgba(0,0,0,1) 最深的情況了。有陰影出現,不好用。  
  • 大量使用box-shadow可能會導致性能瓶頸。
  • 四條邊框實現效果不理想。

3. 使用background-image

使用 background-image 實現1px有兩種方式: 漸變 linear-gradient 或直接使用圖片(base64)。

 

漸變 linear-gradient   (50%有顏色,50%透明)

單條線:

 

div {
     height: 1px;
     background-image: -webkit-linear-gradient(top,transparent 50%,#000 50%);
     background-position: top left;
     background-repeat: no-repeat
     background-size: 100% 1px;
}

 

多條線:  

div {
        background-image:-webkit-linear-gradient(top, transparent 50%, #000 50%),-webkit-linear-gradient(bottom, transparent 50%, #000 50%),-webkit-linear-gradient(left, transparent 50%, #000 50%),-webkit-linear-gradient(right, transparent 50%, #000 50%);
        background-size: 100% 1px,100% 1px,1px 100%,1px 100%;
        background-repeat: no-repeat;
        background-position: top left, bottom left, left top, right top;
}

優點:

  • 可以設置單條,多條邊框
  • 可以設置顏色

缺點:

  • 大量使用漸變可能導致性能瓶頸
  • 代碼量大
  • 多背景圖片有兼容性問題

  

用圖片(base64):

div {
        border-image:url("data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAAECAYAAABP2FU6AAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAB5JREFUeNpiPnH8zH/G////MzAxAAHTyRNn/wMEGABpvQm9g9TJ1QAAAABJRU5ErkJggg==") 2 0 stretch;
        border-width: 0px 0px 1px;
}

優點:

  • 可以設置單條,多條邊框
  • 沒有性能瓶頸的問題

缺點:

  • 修改顏色麻煩, 需要替換圖片
  • 需要用到兩張圖片
  • 多背景圖片有兼容性問題

  

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM