前端每日實戰:116# 視頻演示如何用 CSS 和原生 JS 開發一個監控網絡鏈接狀態的頁面

圖片描述

效果預覽

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

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

可交互視頻

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

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

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

源代碼下載

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

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

代碼解讀

navigator.onLine 屬性用於獲取在線狀態,再配合相應的事件觸發,就能夠開發一個在線檢測工具了。整個過程分紅兩部分,先畫出視覺效果,再檢測在線/離線狀態。chrome

定義 dom,容器中包含客戶端、信號和服務器:服務器

<div class="detector">
    <div class="client"></div>
    <div class="signal"></div>
    <div class="server"></div>
</div>

居中顯示:dom

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

在頂部增長一個橫條,用於顯示當前狀態是在線仍是離線,用綠色表示在線:

:root {
    --status-color: green;
}

body {
    background: linear-gradient(var(--status-color) 5vh, #ccc 5vh);
}

定義容器尺寸:

.detector {
    width: 40em;
    height: 14em;
    font-size: 10px;
}

定義子元素(客戶端、信號、服務器)的總體佈局和主色:

.detector {
    display: flex;
    justify-content: space-between;
    align-items: center;
    color: #333;
}

設置子元素(客戶端、信號、服務器)和它們的僞元素的共有屬性:

.detector > * {
    position: relative;
    box-sizing: border-box;
}

.detector > *::before,
.detector > *::after {
    content: '';
    position: absolute;
    box-sizing: border-box;
}

畫出客戶端的顯示器:

.client {
    width: 17em;
    height: 10em;
    border: 0.5em solid;
    border-radius: 0.5em;
}

用僞元素畫出顯示器的底座:

.client {
    display: flex;
    flex-direction: column;
    align-items: center;
    margin-top: -4em;
}

.client::before {
    width: 1.5em;
    height: 3em;
    background-color: currentColor;
    top: 9.5em;
}

.client::after {
    width: 5em;
    height: 1em;
    background-color: currentColor;
    border-radius: 0.3em;
    top: 12.5em;
}

畫出服務器的機箱:

.server {
    width: 7em;
    height: 14em;
    border: 0.5em solid;
    border-radius: 0.5em;
}

用僞元素畫出硬盤,留意此處陰影的用法,用陰影畫出了第二塊硬盤:

.server::before {
    width: 5em;
    height: 1em;
    background-color: currentColor;
    border-radius: 0.2em;
    top: 8em;
    left: 0.5em;
    box-shadow: 0 1.5em 0;
}

用僞元素畫出按鈕,和上面陰影一樣的用法,此次用陰影畫出了第二個按鈕:

.server::after {
    width: 0.6em;
    height: 0.6em;
    background-color: currentColor;
    border-radius: 50%;
    right: 1.5em;
    bottom: 0.5em;
    box-shadow: 1em 0 0 0.1em;
}

畫出信號,注意配色用的是表明在線/離線的顏色,目前是綠色:

.signal,
.signal::before,
.signal::after {
    width: 1em;
    height: 1em;
    background-color: var(--status-color);
    border-radius: 50%;
}

.signal::before {
    right: 2.5em;
}

.signal::after {
    left: 2.5em;
}

給信號增長動畫效果:

.signal,
.signal::before,
.signal::after {
    animation: blink 0.6s infinite;
}

@keyframes blink {
    50% {
        filter: opacity(0.1);
    }
}

爲第 2 個信號和第 3 個信號設置動畫延時,延時的值用變量定義:

:root {
    --second-signal-delay: 0.2s;
    --third-signal-delay: 0.4s;
}

.signal::before {
    animation-delay: var(--second-signal-delay);
}

.signal::after {
    animation-delay: var(--third-signal-delay);
}

至此,視覺效果已經完成,目前是在線狀態的效果,在 :root 中一共定義了 3 個變量,頂部橫條和信號是綠色,信號燈依次閃爍表示正在傳輸數據:

:root {
    --status-color: green;
    --second-signal-delay: 0.2s;
    --third-signal-delay: 0.4s;
}

經過修改這 3 個變量的值,能夠獲得離線狀態的視覺效果,頂部橫條和信號變爲紅色,信號燈一塊兒閃爍表示線路不通:

:root {
    --status-color: orangered;
    --second-signal-delay: 0s;
    --third-signal-delay: 0s;
}

接下來經過檢測在線/離線狀態,動態應用這 2 種效果。

定義在線狀態主題:

const ONLINE_THEME = {
    statusColor: 'green',
    secondSignalDelay: '0.2s',
    thirdSignalDelay: '0.4s'
}

相似地,定義離線狀態主題:

const OFFLINE_THEME = {
    statusColor: 'orangered',
    secondSignalDelay: '0s',
    thirdSignalDelay: '0s'
}

建立一個函數,用於根據在線/離線狀態顯示不一樣的主題:

function detectOnlineStatus() {
    let theme = navigator.onLine ? ONLINE_THEME : OFFLINE_THEME
    let root = document.documentElement
    root.style.setProperty('--status-color', theme.statusColor)
    root.style.setProperty('--second-signal-delay', theme.secondSignalDelay)
    root.style.setProperty('--third-signal-delay', theme.thirdSignalDelay)
}

detectOnlineStatus()

如今,關掉 wifi 鏈接,而後刷新頁面,頁面會採用紅色主題;再打開 wifi 鏈接,而後刷新頁面,頁面會採用綠色主題。

接下來把檢測函數與系統事件綁定,當鏈接斷開或從新鏈接時,頁面會自動設置主題,不用手動刷新頁面了:

window.addEventListener('online', detectOnlineStatus)
window.addEventListener('offline', detectOnlineStatus)

大功告成!

相關文章
相關標籤/搜索