如何實現同等間隙的卡片佈局

来源:https://www.cnblogs.com/imwtr/archive/2018/12/02/10053640.html
-Advertisement-
Play Games

在列表展示中,經常會使用卡片的內容展示形式,為了美觀,常常要求各卡片間的間隙是一致的。 卡片內容不一樣可能高度不等,但一般來說為了整體的一致性,會限制每個卡片的寬高都相等。 本文就基於寬高一致的多個卡片,在不同屏幕大小下,每行卡片數量可能有調整,考量如何實現等間隙的佈局。 點我預覽 放置一張張卡片項 ...


在列表展示中,經常會使用卡片的內容展示形式,為了美觀,常常要求各卡片間的間隙是一致的。

卡片內容不一樣可能高度不等,但一般來說為了整體的一致性,會限制每個卡片的寬高都相等。

本文就基於寬高一致的多個卡片,在不同屏幕大小下,每行卡片數量可能有調整,考量如何實現等間隙的佈局。

點我預覽

 

放置一張張卡片項,為了設置間距,最常見的就是直接使用一個特定的margin值了,這種方式雖然可以(通過精確計算後確實也可以)

直接設置一個間距,比如統一 margin-left 和 margin-bottom都為 20px ,並不能保證每行最後一個卡片之後的間距是20px

關於如何定這個 margin的值,需要通過一個規則來計算,這個後文再說明

 

設置同等間距,常用的還有 flex佈局中的 justify-content: space-between,可以定義各子項目以相同間距佈局,但不好處理左右子項目與邊框的間距。 space-around這個就更用不得了,會使得左右子項目右margin == 左margin * 2

所以最終還是回到使用margin值來設置,通過一個可用的規則,來保證間距是一致的。

 

先把基本結構搭上

<div class="container">
    <h2>項目列表</h2>
    <ul class="proj-items"></ul>
</div>

<!-- 模板結構 -->
<script type="text/template" id="proj-item-tpl">
    <li class="proj-item">
        <a href="#/p/{{projectID}}">
            <h3 class="proj-item__title">{{projectName}}</h3>
            <p class="proj-item__author">{{author}}</p>
        </a>
    </li>
</script>

JS生成N個項目

        function addEvent(elem, type, handler) {
            elem.addEventListener(type, handler, false);
        }

        function qs(selector) {
            return document.querySelector(selector);
        }

        function qsa(selectors) {
            return document.querySelectorAll(selectors);
        }

        var mockData = (function(num) {
            var data = [];
            
            for (var i = 1; i <= num; ++i) {
                data.push({
                    projectID: i,
                    projectName: '項目' + i,
                    author: '張大大'
                });
            }
            
            return data;
        })(8);
        
        var itemTpl = qs('#proj-item-tpl').innerHTML;
        var itemsDOM = qs('.proj-items');
        
        /**
         * 渲染數據
         * @param  {[type]} data [description]
         * @return {[type]}      [description]
         */
        function renderList(data) {
            var html = '';
            var fragment = document.createDocumentFragment();

            data.forEach(function(item) {
                var divTemp = document.createElement('div');

                // 模板替換
                divTemp.innerHTML = itemTpl.replace(/{{(\w+)}}/g, function(input, match) {
                    return match ? item[match] || '' : '';
                });

                fragment.appendChild(divTemp.firstElementChild);
            });

            // 渲染
            itemsDOM.appendChild(fragment);
        }
       
        renderList(mockData);

把基礎樣式放上,這裡我們先指定一個特定的itemMargin值為20px

$itemMargin: 20px;
$itemWidth: 130px;
$itemHeight: 150px;

.container {
    margin: 20px auto;
    width: 450px;
    background-color: #f2f2f2;
    color: #666;
    
    h2 {
        margin: 20px;
        padding-top: 20px;
        font-size: 20px;
    }
}

.proj-items {
    display: flex;
    flex-wrap: wrap;
    /* justify-content: space-between; */
    padding: 0;
    list-style: none;
    
    &:after {
        content: "";
        display: block;
        flex-grow: 99999;
    }
}

.proj-item {
    margin-left: $itemMargin;
    margin-bottom: $itemMargin;
    width: $itemWidth;
    height: $itemHeight;
    background-color: #fff;
    border-radius: 3px;
    text-align: center;
    
    &:hover {
        box-shadow: 0 0 20px #ddd;
    }
    
    a {
        display: block;
        padding: 15px;
        height: 100%;
        color: #666;
        text-decoration: none;
    }
    
    &__title {
        margin-top: 0;
        font-size: 16px;
    }
    
    &__author {
        font-size: 12px;
    }
}

 

可以看到,每行最後一個間距不一致了,所以不能簡單的寫個margin值

再來看看設置 space-between的時候

.proj-items {
    justify-content: space-between;
    ...
}

.proj-item {
    /* margin-left: $itemMargin; */
    margin-bottom: $itemMargin;
    ...
}

 

 看來並不夠強大

如果看得仔細,應該能看到項目7和8是挨在一起的,為何沒有間距呢

其一是因為沒有margin-left值,其二是在項目列表後放了一個坑來占位,防止最後一行項目過少時 space-between的值太大了

把這個撤掉看看這個影響

    &:after {
        content: "";
        display: block;
        flex-grow: 99999;
    }

 

 

還是把目光投向margin值的設定規則吧

在設計一個頁面佈局時,至少已經確定了XX頁面大小的情況下,容器寬度應該設置為多少(比如為1200px),每行放n個項目,項目的寬高是多少

有了這些指標(也必須有這些指標),我們就可以用來計算margin值了

containerWidth == n * itemWidth + (n + 1) * itemMargin

得出

itemMargin = (containerWidth - n * itemWidth) / (n + 1)

代入這裡的情況,containerWidth 450px,itemWidth 130px,每行 3個,即可得出 itemMargin 正好為 15px 

 

有了某種特定情況下的佈局規則之後,接下來還要考慮不同屏幕大小的情況下,怎麼調整這個margin值

這個需要結合媒體查詢來設定,同時相應的計算規則也可以通過scss來處理

第一種情況是每行3個,n只可能為整數,即可推算出需要處理的臨界值為1 2 3 4 5 6 ... 這些整數值

加入n為4,如果要保證 itemMargin值15px在各種情況下都相等,計算可得 容器寬度containerWidth值 為 595px

同理求得 n是5時為 740px ,n是2時為 305px

當然,如果覺得這個containerWidth值不太好看,也可以自己定義,比如 n是4的時候設置為 600px,代入公式那麼 itemMargin值為16px。

為了保證各種請下間距都相等,我個人就不推薦這麼幹了

 

通過上述的規則計算,我們可以得出每行項目數量遞增時的容器寬度臨界值。把這些臨界值放在媒體查詢裡面配置,即可方便地實現這種佈局的自適應。

/* 這兩個為初始就確定的基準值 */
$containerWidth: 305px;
$itemMargin: 15px;

$itemWidth: 130px;
$itemHeight: 150px;

/* 每行項目數量為itemNum時的容器寬度 */
@function getContainerWidth($itemNum) {
    @return $itemNum * $itemWidth + ($itemNum + 1) * $itemMargin; 
}

/* 配置各個頁面寬度下的容器寬度(應用) */
@mixin adjustContainerWidth(
    $from: 2,
    $to: 5
) {
    @for $i from $from through $to {
        $minWidth: getContainerWidth($i);
        $maxWidth: getContainerWidth($i + 1);

        @media only screen and (min-width: $minWidth) and (max-width: $maxWidth) {
            .container {
                width: $minWidth;
            }
        }
    }
}

.container {
    margin: 20px auto;
    width: $containerWidth;
    background-color: #f2f2f2;
    color: #666;
    
    h2 {
        margin: 20px;
        padding-top: 20px;
        font-size: 20px;
    }
}

@include adjustContainerWidth(
    $from: 1,
    $to: 7
);

即可實現各個頁面大小下的自適應效果

完整的CSS部分

 1 /* 這兩個為初始就確定的基準值 */
 2 $containerWidth: 305px;
 3 $itemMargin: 15px;
 4 
 5 $itemWidth: 130px;
 6 $itemHeight: 150px;
 7 
 8 /* 每行項目數量為itemNum時的容器寬度 */
 9 @function getContainerWidth($itemNum) {
10     @return $itemNum * $itemWidth + ($itemNum + 1) * $itemMargin; 
11 }
12 
13 /* 配置各個頁面寬度下的容器寬度(應用) */
14 @mixin adjustContainerWidth(
15     $from: 2,
16     $to: 5
17 ) {
18     @for $i from $from through $to {
19         $minWidth: getContainerWidth($i);
20         $maxWidth: getContainerWidth($i + 1);
21 
22         @media only screen and (min-width: $minWidth) and (max-width: $maxWidth) {
23             .container {
24                 width: $minWidth;
25             }
26         }
27     }
28 }
29 
30 .container {
31     margin: 20px auto;
32     width: $containerWidth;
33     background-color: #f2f2f2;
34     color: #666;
35     
36     h2 {
37         margin: 20px;
38         padding-top: 20px;
39         font-size: 20px;
40     }
41 }
42 
43 @include adjustContainerWidth(
44     $from: 1,
45     $to: 7
46 );
47 
48 .proj-items {
49     display: flex;
50     flex-wrap: wrap;
51     padding: 0;
52     list-style: none;
53 }
54 
55 .proj-item {
56     margin-left: $itemMargin;
57     margin-bottom: $itemMargin;
58     width: $itemWidth;
59     height: $itemHeight;
60     background-color: #fff;
61     border-radius: 3px;
62     text-align: center;
63     
64     &:hover {
65         box-shadow: 0 0 20px #ddd;
66     }
67     
68     a {
69         display: block;
70         padding: 15px;
71         height: 100%;
72         color: #666;
73         text-decoration: none;
74     }
75     
76     &__title {
77         margin-top: 0;
78         font-size: 16px;
79     }
80     
81     &__author {
82         font-size: 12px;
83     }
84 }
View Code

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 0.準備工作: 關閉itunes 在任務管理器中殺掉iTunes開頭的服務 1,找到iTunes預設備份路徑:C:\Users\xxx\AppData\Roaming\Apple Computer\MobileSync xxx為里的用戶名 2,刪掉預設路徑下的MobileSync文件夾,以前有備份的 ...
  • frameset 定義 W3C是這樣定義frameset框架的,通過使用框架,你可以在同一個瀏覽器視窗中顯示不止一個頁面。每份HTML文檔稱為一個框架,並且每個框架都獨立於其他的框架。註意,這是HTML框架,不是前端框架,與node.js,vue.js等不同。 垂直切割 屬性為cols。例如: <f ...
  • 先說幾個概念: 1、js代碼從上往下執行 2、變數提升: 變數提升是瀏覽器的一個功能,在運行js代碼之前,瀏覽器會給js一個全局作用域叫window,window分兩個模塊,一個叫記憶體模塊,一個叫運行模塊,記憶體模塊找到當前作用域下的所有帶var和function的關鍵字,執行模塊執行js代碼,從上到 ...
  • 聲明 本系列文章內容全部梳理自以下幾個來源: 《JavaScript權威指南》 "MDN web docs" "Github:smyhvae/web" "Github:goddyZhao/Translation/JavaScript" 作為一個前端小白,入門跟著這幾個來源學習,感謝作者的分享,在其基 ...
  • 什麼是作用域:瀏覽器給js的生存環境叫作用域。 什麼是變數提升: Js代碼執行前,瀏覽器會給一個全局作用域window Window分兩個模塊一個是存儲模塊一個是執行模塊 存儲模塊找到所有的var和function 關鍵字給這些變數添加記憶體地址 執行模塊,代碼從上到下執行,遇到變數就會去存儲模塊查找 ...
  • 1.事件 瀏覽器客戶端上客戶觸發的行為都稱為事件 所有的事件都是天生自帶的,不需要我們去綁定,只需要我們去觸發。 通過 obj.事件名=function(){} 事件名:onmouseover 滑鼠懸浮 onmouseout 滑鼠移除 onmousedown滑鼠按下 onmouseup 滑鼠抬起 o ...
  • 元素的屬性 div.attributes 是所有標簽屬性構成的數據集合 div.classList 是所有class名構成的數組集合 在classList的原型鏈上看以看到add()和remove() 1.client系列 (1) clientWidth/clientHeight 是我們設置的寬和高 ...
  • @[toc] .val()實例方法的三種用法 1. $('xxx').val() : 獲取匹配的元素集合中第一個元素的當前value屬性值(property) 2. $('xxx').val(value) : 設置匹配的元素集合中每個元素的value屬性值(property) 3. $('xxx') ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...