CSS 實現微信朋友圈的九宮格圖片自適應

大家好,我是 Peter,微信朋友圈都發過吧?它支持最多發 9 張圖片,所以我們也稱之爲九宮格

要知道發 1~9 任意數量的圖片時,這些圖片的排列方式可能有所不同。大體有這四種排列情況:

情況一:只有一張圖,完整展示整張圖

1 張圖

情況二:2~3 張圖的情況,無法完整展示每張圖,則在第一排依次排開(一排最多 3 張)

2 張圖 & 3 張圖

情況三:4 張圖的情況,無法完整展示每張圖,每排 2 張圖,共 2 排

4 張圖

情況四:5~9 張圖的情況跟情況二類似,也無法完整展示每張圖,每排最多 3 張圖,依次排開

5 張圖

其實就是對不同數量的圖片做了一個排版的自適應,原理也很簡單,我們一起來用 CSS 實現一下吧~

創建容器

先創建一個元素作爲容器

<div class="pictures-adaptive">
  <!-- 裏面放 img 圖片 -->
</div>

並將其設置爲 Flex 佈局,且可換行

.pictures-adaptive {
  display: flex;
  flex-wrap: wrap;
}

1 張圖片

只有 1 張圖片的情況最簡單,就讓圖片直接展示就好,但大於 1 張圖時,所有圖片都是無法完全展示的,它們是以一個正方形的形式展現的,所以我們需要給 img 外側包一個容器,便於後續控制圖片的展示

<div class="pictures-adaptive">
  <div class="wrap">
    <img src="demo.png"/>
  </div>
</div>

樣式也不用設置,最終效果是這樣的

1 張圖片

多張圖片

大於 1 張圖片時,圖片需要在正方形容器中非完整居中展示,可以讓 img 標籤相對於外部容器元素相對定位

/* 相對定位,img相對於.wrap相對定位 */
.wrap {
  position: relative;
  overflow: hidden;
  margin-bottom: 2%;
}

/* 大於1張圖片時,img標籤的樣式 */
.wrap:not(:nth-child(1):nth-last-child(1)) img {
  position: absolute;
  top: 0;
  left: 0;
  width: 100%;
  height: 100%;
  object-fit: cover;
}

分析了一下,多張圖片的情況可以分爲 2、4 張圖片 或者 3 張圖片、5~9 張圖片

2 或 4 張圖片

2 或 4 張圖片時都是一行只有兩個圖片,且兩張圖片各佔一半左右的寬度

/*  2張圖片  */
.wrap:nth-child(1):nth-last-child(2),
.wrap:nth-child(2):nth-last-child(1),
/*  4張圖片  */
.wrap:nth-child(1):nth-last-child(4),
.wrap:nth-child(2):nth-last-child(3),
.wrap:nth-child(3):nth-last-child(2),
.wrap:nth-child(4):nth-last-child(1)
{
  width: 49%;
  padding-bottom: 49%;
}


/* 每行的兩張圖片中間間隔2%的寬度 */
/*  2張圖片  */
.wrap:nth-child(2):nth-last-child(1),
/*  4張圖片  */
.wrap:nth-child(2):nth-last-child(3),
.wrap:nth-child(4):nth-last-child(1)
{
  margin-left: 2%;
}

最終效果是這樣的

2 張圖片

4 張圖片

3 張圖片

3 張圖片的情況是:三張圖排列在同一行

/*  3張圖片  */
.wrap:nth-child(1):nth-last-child(3),
.wrap:nth-child(2):nth-last-child(2),
.wrap:nth-child(3):nth-last-child(1) 
{
  width: 32%;
  padding-bottom: 32%;
}

/*  間隔  */
.wrap:nth-child(2):nth-last-child(2),
.wrap:nth-child(3):nth-last-child(1) 
{
  margin-left: 2%;
}

最終效果是這樣的

3 張圖片

5~9 張圖片

5~9 張圖片的情況其實跟 3 張圖片的排版是一樣的,不過可以簡化寫法

/*  5張以上圖片  */
.wrap:nth-child(n + 5),
.wrap:nth-child(1):nth-last-child(n + 5),
.wrap:nth-child(1):nth-last-child(n + 5) ~ .wrap
{
  width: 32%;
  padding-bottom: 32%;
}

/*  間隔  */
.wrap:nth-child(n + 5):not(:nth-child(3n + 1)),
.wrap:nth-child(1):nth-last-child(n + 5) ~ .wrap:not(:nth-child(3n + 1))
{
  margin-left: 2%;
}

最終效果是這樣的

9 張圖片

完整代碼

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <meta http-equiv="X-UA-Compatible" content="IE=edge">
  <meta >
  <title>微信朋友圈圖片九宮格排版自適應(改編版)</title>
  <style>
    .pictures-adaptive {
      display: flex;
      flex-wrap: wrap;
    }

    .wrap {
      position: relative;
      overflow: hidden;
      margin-bottom: 2%;
    }

    /*  3張圖片  */
    .wrap:nth-child(1):nth-last-child(3),
    .wrap:nth-child(2):nth-last-child(2),
    .wrap:nth-child(3):nth-last-child(1) 
    {
      width: 32%;
      padding-bottom: 32%;
    }

    /*  間隔  */
    .wrap:nth-child(2):nth-last-child(2),
    .wrap:nth-child(3):nth-last-child(1) 
    {
      margin-left: 2%;
    }

    .wrap:not(:nth-child(1):nth-last-child(1)) img {
      position: absolute;
      top: 0;
      left: 0;
      width: 100%;
      height: 100%;
      object-fit: cover;
    }

    /*  2張圖片  */
    .wrap:nth-child(1):nth-last-child(2),
    .wrap:nth-child(2):nth-last-child(1),
    /*  4張圖片  */
    .wrap:nth-child(1):nth-last-child(4),
    .wrap:nth-child(2):nth-last-child(3),
    .wrap:nth-child(3):nth-last-child(2),
    .wrap:nth-child(4):nth-last-child(1)
    {
      width: 49%;
      padding-bottom: 49%;
    }

    /* 每行的兩張圖片中間間隔2%的寬度 */
    /*  2張圖片  */
    .wrap:nth-child(2):nth-last-child(1),
    /*  4張圖片  */
    .wrap:nth-child(2):nth-last-child(3),
    .wrap:nth-child(4):nth-last-child(1)
    {
      margin-left: 2%;
    }

    /*  5張以上圖片  */
    .wrap:nth-child(n + 5),
    .wrap:nth-child(1):nth-last-child(n + 5),
    .wrap:nth-child(1):nth-last-child(n + 5) ~ .wrap
    {
      width: 32%;
      padding-bottom: 32%;
    }

    .wrap:nth-child(n + 5):not(:nth-child(3n + 1)),
    .wrap:nth-child(1):nth-last-child(n + 5) ~ .wrap:not(:nth-child(3n + 1))
    {
      margin-left: 2%;
    }
  </style>
</head>
<body>
  <div class="pictures-adaptive">
    <div class="wrap">
      <img src="">
      <!-- More imgs -->
    </div>
  </div>
</body>
</html>
本文由 Readfog 進行 AMP 轉碼,版權歸原作者所有。
來源https://mp.weixin.qq.com/s/4BjqngAs_Y9HFfCV4pMquA