前端每日實戰:130# 視頻演示如何用 CSS 在線字體和 D3 創做一個 Google & googol 信息圖

圖片描述

效果預覽

按下右側的「點擊預覽」按鈕能夠在當前頁面預覽,點擊連接能夠全屏預覽。javascript

https://codepen.io/comehope/pen/xaPZyecss

可交互視頻

此視頻是能夠交互的,你能夠隨時暫停視頻,編輯視頻中的代碼。html

請用 chrome, safari, edge 打開觀看。前端

https://scrimba.com/p/pEgDAM/cwkpGf9java

源代碼下載

每日前端實戰系列的所有源代碼請從 github 下載:git

https://github.com/comehope/front-end-daily-challengesgithub

代碼解讀

定義 dom,只有 1 個空元素,其中不包含任何文本:web

<div class="logo"></div>

引入字體文件,Product Sans 是 Google 專門爲品牌推廣建立的無襯線字體:chrome

@import url("https://fonts.googleapis.com/css?family=Product+Sans");

居中顯示:api

body {
    margin: 0;
    height: 100vh;
    display: flex;
    align-items: center;
    justify-content: center;
}

用僞元素製做 logo,注意 content 的內容不是 "Google",而是 "google_logo"

.logo::before {
    content: 'google_logo';
    font-size: 10vw;
}

設置字體,採用剛纔引入的在線字體,剛纔頁面上的 "google_logo" 文字被替換成了單色的 logo 圖案:

body {
    font-family: 'product sans';
}

定義顏色變量:

:root {
    --blue: #4285f4;
    --red: #ea4335;
    --yellow: #fbbc05;
    --green: #34a853;
}

設置文字遮罩效果,爲文字上色:

.logo::before {
    background-image: linear-gradient(
        to right,
        var(--blue) 0%, var(--blue) 26.5%, 
        var(--red) 26.5%, var(--red) 43.5%, 
        var(--yellow) 43.5%, var(--yellow) 61.5%,
        var(--blue) 61.5%, var(--blue) 78.5%, 
        var(--green) 78.5%, var(--green) 84.5%, 
        var(--red) 84.5%, var(--red) 100%
    );
    -webkit-background-clip: text;
    -webkit-text-fill-color: transparent;
}

至此,Google logo 製做完成,接下來製做 googol 信息,說明 Google 的名字來源於含義是 1 後面跟 100 個零的大數的單詞 googol。

在 dom 中添加一行說明文本和容納數字的容器,容器中包含 5 個數字,在每一個數字的內聯樣式中指定了顏色變量:

<p class="desc">The name of Google originated from a misspelling of the word "googol", the number 1 followed by 100 zeros.</p> 
<div class="zeros">
    <span style="--c:var(--blue);">1</span>
    <span style="--c:var(--red);">0</span>
    <span style="--c:var(--yellow);">0</span>
    <span style="--c:var(--blue);">0</span>
    <span style="--c:var(--green);">0</span>
</div>

設置說明文本的樣式:

.desc {
    font-size: 1.5vw;
    font-weight: normal;
    color: dimgray;
    margin-top: 2em;
}

設置數字的樣式:

.zeros {
    font-size: 3vw;
    font-weight: bold;
    margin-top: 0.2em;
    text-align: center;
    width: 25.5em;
    word-wrap: break-word;
}

爲數字上色:

.zeros span {
    color: var(--c);
}

微調數字 "1" 的邊距,讓它不要和後面的 "0" 靠得太緊:

.zeros span:nth-child(1) {
    margin-right: 0.2em;
}

至此,靜態佈局完成,接下來用 d3 批量處理數字。

引入 d3 庫,並刪除掉 dom 中 .zeros 的數字子元素:

<script src="https://d3js.org/d3.v5.min.js"></script>

最終咱們會在頁面上顯示 100 個 0,每一個 0 的顏色都不一樣,而且爲了美觀,相鄰數字的顏色也要不一樣。
因此,先定義一個獲取顏色的函數,它能夠從 Google logo 配色的 4 種顏色中取任意一個顏色,而且有一個表示被排除顏色的參數,當指定的此參數時,就從 4 個可選的顏色中去掉這個顏色,而後從剩下的 3 個顏色中隨機取一個顏色:

function getColor(excludedColor) {
    let colors = new Set(['blue', 'red', 'yellow', 'green'])
    colors.delete(excludedColor)
    return Array.from(colors)[Math.floor(d3.randomUniform(0, colors.size)())]
}

而後,定義 2 個常量,ZEROS 是存儲 100 個 0 的數組,ONE 是存儲數字 1 的對象,它有 2 個屬性,number 表示它的數值是 1,color 表示它的顏色是藍色:

const ZEROS = d3.range(100).map(x=>0)
const ONE = {number: 1, color: 'blue'}

接下來,經過用 reduce 函數遍歷 ZEROS 數組,返回一個新的數組 numbers,它有 101 個元素(1 以及跟隨它的 100 個 0),每一個元素都是 1 個包含 numbercolor 屬性的對象:

let numbers = ZEROS.reduce(function (numberObjects, d) {
    numberObjects.push({
        number: d,
        color: getColor(numberObjects[numberObjects.length - 1].color)
    })
    return numberObjects
}, [ONE])

而後,以 numbers 爲數據源,用 d3 批量建立出 dom 元素,而且把顏色信息寫在行內樣式中:

d3.select('.zeros')
    .selectAll('span')
    .data(numberObjects)
    .enter()
    .append('span')
    .style('--c', (d)=>`var(--${d.color})`)
    .text((d)=>d.number)

最後,微調一下內容的邊距,使整個內容居中:

.logo {
    margin-top: -10vh;
}

大功告成!

相關文章
相關標籤/搜索