一個簡易的 LED 數字時鐘實現方法

来源:https://www.cnblogs.com/zml1023/archive/2020/01/17/12206007.html
-Advertisement-
Play Games

這個應該是已經有很多人做過的東西,我應該只是算手癢,想寫一下,所以,花了點時間折騰了這個,順便把 Dark Mode 的處理也加上了。 首先可以很明確的一點,這個真沒技術含量存在,只是需要點耐心。 LED 數字包含了左右各兩條線,中間三條線,一共 7 條線。所以,為了能夠更容易辨識,在寫 demo ...


這個應該是已經有很多人做過的東西,我應該只是算手癢,想寫一下,所以,花了點時間折騰了這個,順便把 Dark Mode 的處理也加上了。

首先可以很明確的一點,這個真沒技術含量存在,只是需要點耐心。

LED 數字包含了左右各兩條線,中間三條線,一共 7 條線。所以,為了能夠更容易辨識,在寫 demo 的時候,我直接這樣寫了。

<div class="digital digital_0">
  <span class="c1"></span>
  <span class="c2"></span>
  <span class="c3"></span>
  <span class="l1"></span>
  <span class="l2"></span>
  <span class="r1"></span>
  <span class="r2"></span>
</div>

至於 digital_0 這個的作用,其實很簡單,就是控制從 0 ~ 9 這十個數字的變化而存在的。

.digital_1 .c1,
.digital_1 .c2,
.digital_1 .c3,
.digital_1 .l1,
.digital_1 .l2,
.digital_2 .l1,
.digital_2 .r2,
.digital_3 .l1,
.digital_3 .l2,
.digital_4 .c1,
.digital_4 .c3,
.digital_4 .l2,
.digital_5 .l2,
.digital_5 .r1,
.digital_6 .r1,
.digital_7 .c2,
.digital_7 .c3,
.digital_7 .l1,
.digital_7 .l2,
.digital_9 .l2,
.digital_0 .c2 {
  animation: changeDigital 200ms 0ms 1 ease-in forwards;
}

這裡使用了一個 animation 動畫,並且時間是 200ms,主要是為了讓某些部分消失的時候,有一個過渡效果。

@keyframes changeDigital {
  form {
    opacity: 1;
  }
  to {
    opacity: 0;
  }
}

那麼剩下來的就是最需要耐心的地方了,調整控制那 7 條線的位置,同時還要考慮每條線是帶有斜角的,而且還稍微有點圓弧的感覺。帶點圓弧的感覺那麼直接使用 border-radius 就可以了,至於那個斜角的話,如果各位知道 CSS 中用邊框畫三角的方法,那麼就應該明白怎麼做這個斜角了。

當我們要畫三角的時候,width 和 height 都是為 0,然後用過控制 border-width 而得到最終的三角的大小,那麼這裡如果我們根據線條的位置,適當選擇 width 和 height 有具體的值,是不是就有斜角的感覺了呢。

接著,需要註意橫著的中間那根線,也就是 .c2 這個元素,線條兩邊是三角突出的,可以實現的方式也很多,這裡我選擇使用 :after 和 :before 相疊加。

最後就是對每個元素的位置做調整,控制好大小位置就可以了,因為是使用定位的方式來操作,所以,調整起來還是很簡單的。

.digital span {
  position: absolute;
  border-radius: 50vh;
  box-sizing: border-box;
}
.digital .c1,
.digital .c2,
.digital .c3 {
  height: 0;
  width: 26px;
  border-left: 4px solid transparent;
  border-right: 4px solid transparent;
}
.digital .c1 {
  top: 0;
  left: 0;
  border-top: 4px solid currentColor;
}
.digital .c2 {
  top: 50%;
  left: 0;
  margin-top: -2px;
}
.digital .c2:before,
.digital .c2:after {
  content: "";
  height: 0;
  width: 24px;
  border-left: 2px solid transparent;
  border-right: 2px solid transparent;
  box-sizing: border-box;
}
.digital .c2:before {
  position: absolute;
  top: 0;
  left: -3px;
  border-bottom: 2px solid currentColor;
}
.digital .c2:after {
  position: absolute;
  top: 2px;
  left: -3px;
  border-top: 2px solid currentColor;
}
.digital .c3 {
  bottom: 0;
  left: 0;
  border-bottom: 4px solid currentColor;
}
.digital .l1,
.digital .l2 {
  height: 21px;
  width: 0;
  left: 0;
  border-top: 2px solid transparent;
  border-bottom: 2px solid transparent;
  border-left: 4px solid currentColor;
}
.digital .l1 {
  top: 1px;
  border-top-width: 4px;
}
.digital .l2 {
  top: 24px;
  border-bottom-width: 4px;
}
.digital .r1,
.digital .r2 {
  height: 21px;
  width: 0;
  right: 0;
  border-top: 2px solid transparent;
  border-bottom: 2px solid transparent;
  border-right: 4px solid currentColor;
}
.digital .r1 {
  top: 1px;
  border-top-width: 4px;
}
.digital .r2 {
  top: 24px;
  border-bottom-width: 4px;
}

做完這些,剩下的就是簡單的對外層元素做點優化,比如稍微傾斜一點。

.digital {
  position: relative;
  width: 26px;
  height: 46px;
  margin-left: 10px;
  transform: skew(-6deg);
}

哦,忘了說了,這裡的邊框顏色我選擇使用 currentColor 的方式,這樣可以在暗黑模式切換的過程中,只需要修改 body 中的文字顏色和背景色就可以了。currentColor 會選擇文字顏色來使用。

樣式處理完之後,接著就是添加點 JS 來顯示時鐘了,只要把兩個數字整齣來放到不同的 div 中就好了。這裡我取個位數的時候,是用 m%10 的方式取餘;取十位數的時候是通過 parseInt(m/10) 的方式取整。取出來之後分別放到想對應的 div 中就 ok 了。

完整代碼如下:

樣式:

body {
  color: #393e4d;
  background-color: #f5f5f5;
}
@media screen and (prefers-color-scheme: dark) {
  body {
    color: #59f6fb;
    background-color: #1c1b1e;
  }
}
.digital {
    position: relative;
    width: 26px;
    height: 46px;
  margin-left: 10px;
    transform: skew(-6deg);
}
.digital span {
    position: absolute;
    border-radius: 50vh;
    box-sizing: border-box;
}
.digital .c1,
.digital .c2,
.digital .c3 {
    height: 0;
    width: 26px;
    border-left: 4px solid transparent;
    border-right: 4px solid transparent;
}
.digital .c1 {
    top: 0;
    left: 0;
    border-top: 4px solid currentColor;
}
.digital .c2 {
    top: 50%;
    left: 0;
    margin-top: -2px;
}
.digital .c2:before,
.digital .c2:after {
    content: "";
    height: 0;
    width: 24px;
    border-left: 2px solid transparent;
    border-right: 2px solid transparent;
    box-sizing: border-box;
}
.digital .c2:before {
    position: absolute;
    top: 0;
    left: -3px;
    border-bottom: 2px solid currentColor;
}
.digital .c2:after {
    position: absolute;
    top: 2px;
    left: -3px;
    border-top: 2px solid currentColor;
}
.digital .c3 {
    bottom: 0;
    left: 0;
    border-bottom: 4px solid currentColor;
}
.digital .l1,
.digital .l2 {
    height: 21px;
    width: 0;
    left: 0;
    border-top: 2px solid transparent;
    border-bottom: 2px solid transparent;
    border-left: 4px solid currentColor;
}
.digital .l1 {
    top: 1px;
    border-top-width: 4px;
}
.digital .l2 {
    top: 24px;
    border-bottom-width: 4px;
}
.digital .r1,
.digital .r2 {
    height: 21px;
    width: 0;
    right: 0;
    border-top: 2px solid transparent;
    border-bottom: 2px solid transparent;
    border-right: 4px solid currentColor;
}
.digital .r1 {
    top: 1px;
    border-top-width: 4px;
}
.digital .r2 {
    top: 24px;
    border-bottom-width: 4px;
}

.digital_1 .c1,
.digital_1 .c2,
.digital_1 .c3,
.digital_1 .l1,
.digital_1 .l2,
.digital_2 .l1,
.digital_2 .r2,
.digital_3 .l1,
.digital_3 .l2,
.digital_4 .c1,
.digital_4 .c3,
.digital_4 .l2,
.digital_5 .l2,
.digital_5 .r1,
.digital_6 .r1,
.digital_7 .c2,
.digital_7 .c3,
.digital_7 .l1,
.digital_7 .l2,
.digital_9 .l2,
.digital_0 .c2 {
    animation: changeDigital 200ms 0ms 1 ease-in forwards;
}

@keyframes changeDigital {
    form {
        opacity: 1;
    }
    to {
        opacity: 0;
    }
}

.clock {
  display: flex;
  justify-content: center;
  padding-top: 20px;
}
.gap {
  height: 46px;
  padding-left: 10px;
  font-size: 50px;
  font-weight: bold;
  line-height: 0.8;
  transform: skew(-6deg);
}

結構:

<div class="clock">
  <div class="digital digital_0">
    <span class="c1"></span>
    <span class="c2"></span>
    <span class="c3"></span>
    <span class="l1"></span>
    <span class="l2"></span>
    <span class="r1"></span>
    <span class="r2"></span>
  </div>
  <div class="digital digital_0">
    <span class="c1"></span>
    <span class="c2"></span>
    <span class="c3"></span>
    <span class="l1"></span>
    <span class="l2"></span>
    <span class="r1"></span>
    <span class="r2"></span>
  </div>
  <div class="gap">:</div>
  <div class="digital digital_0">
    <span class="c1"></span>
    <span class="c2"></span>
    <span class="c3"></span>
    <span class="l1"></span>
    <span class="l2"></span>
    <span class="r1"></span>
    <span class="r2"></span>
  </div>
  <div class="digital digital_0">
    <span class="c1"></span>
    <span class="c2"></span>
    <span class="c3"></span>
    <span class="l1"></span>
    <span class="l2"></span>
    <span class="r1"></span>
    <span class="r2"></span>
  </div>
  <div class="gap">:</div>
  <div class="digital digital_0">
    <span class="c1"></span>
    <span class="c2"></span>
    <span class="c3"></span>
    <span class="l1"></span>
    <span class="l2"></span>
    <span class="r1"></span>
    <span class="r2"></span>
  </div>
  <div class="digital digital_0">
    <span class="c1"></span>
    <span class="c2"></span>
    <span class="c3"></span>
    <span class="l1"></span>
    <span class="l2"></span>
    <span class="r1"></span>
    <span class="r2"></span>
  </div>
</div>

JS:

var s1 = document.getElementsByClassName("digital")[5],
    s2 = document.getElementsByClassName("digital")[4],
    m1 = document.getElementsByClassName("digital")[3],
    m2 = document.getElementsByClassName("digital")[2],
    h1 = document.getElementsByClassName("digital")[1],
    h2 = document.getElementsByClassName("digital")[0];

setInterval(function(){
  var date = new Date();
  var h = date.getHours();
  var m = date.getMinutes();
  var s = date.getSeconds();
  if (h < 10) {
    h1.className = 'digital digital_' + h;
    h2.className = 'digital digital_0';
  } else {
    h1.className = 'digital digital_' + parseInt(h%10);
    h2.className = 'digital digital_' + parseInt(h/10);
  }
  if (m < 10) {
    m1.className = 'digital digital_' + m;
    m2.className = 'digital digital_0';
  } else {
    m1.className = 'digital digital_' + parseInt(m%10);
    m2.className = 'digital digital_' + parseInt(m/10);
  }
  if (s < 10) {
    s1.className = 'digital digital_' + s;
    s2.className = 'digital digital_0';
  } else {
    s1.className = 'digital digital_' + parseInt(s%10);
    s2.className = 'digital digital_' + parseInt(s/10);
  }
}, 1000)

這樣一個簡易的 LED 數字時鐘就大功告成啦。


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

-Advertisement-
Play Games
更多相關文章
  • 背景: 項目基於原生js,沒用到任何腳手架和框架,但也需要打包壓縮。 項目的js中聲明瞭一些全局變數 供其他js調用。 這時候如果用webpack打包,基於webpack特性,會嵌套一層大函數,會將js中的變數變成局部,不能供其他js調用。 因此棄用了webpack。選用了uglifyjs。 原因: ...
  • 手風琴案例+stop的使用(解決動畫隊列的問題) stop();// 停止當前正在執行的動畫 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> <style> * { margin: ...
  • title: '[react]深入 - 一等公民 props & onChange'date: 2017-08-23 10:05:07tags:reactreactjspropsonChange框架reduxcategories:技術寫在前面這篇博客的前身是 《React 新手必須知道的10件事》,... ...
  • 自定義動畫 animate(); 第一個參數:{對象},裡面可以傳需要動畫的樣式 第二個參數:speed 動畫的執行時間 第三個參數:easing 動畫的執行效果 第四個參數:callback 回調函數 //第一個參數:對象,裡面可以傳需要動畫的樣式 $("#box1").animate({ lef ...
  • https://www.zhangxinxu.com/wordpress/2019/10/a-href-target-window-blank-refresh/ ...
  • 京東輪播圖 有個計數的,點右邊,計數增加,判斷計數是否超過總的長度,超過設置計數為0,再設置當前的圖片動畫,兄弟的圖片動畫 左邊點擊同理,計數是--,判斷計數是否等於-1,等於則reset計數為總長度,在設置當前的圖片動畫,兄弟的圖片動畫 <!DOCTYPE html> <html lang="zh ...
  • 路由的使用 "react router" 其中一個組件 獲取路由參數 "Ant design" 使用在上面的代碼有體現 按需導入配置 .baelrc "electron技術" ...
  • 三組基本動畫 show hide //show不傳參數,沒有動畫效果 $("div").show(); //show(speed) //speed:動畫的持續時間 可以是毫秒值 還可以是固定字元串 //fast:200ms normal:400ms slow:600ms $("div").show( ...
一周排行
    -Advertisement-
    Play Games
  • C#TMS系統代碼-基礎頁面BaseCity學習 本人純新手,剛進公司跟領導報道,我說我是java全棧,他問我會不會C#,我說大學學過,他說這個TMS系統就給你來管了。外包已經把代碼給我了,這幾天先把增刪改查的代碼背一下,說不定後面就要趕鴨子上架了 Service頁面 //using => impo ...
  • 委托與事件 委托 委托的定義 委托是C#中的一種類型,用於存儲對方法的引用。它允許將方法作為參數傳遞給其他方法,實現回調、事件處理和動態調用等功能。通俗來講,就是委托包含方法的記憶體地址,方法匹配與委托相同的簽名,因此通過使用正確的參數類型來調用方法。 委托的特性 引用方法:委托允許存儲對方法的引用, ...
  • 前言 這幾天閑來沒事看看ABP vNext的文檔和源碼,關於關於依賴註入(屬性註入)這塊兒產生了興趣。 我們都知道。Volo.ABP 依賴註入容器使用了第三方組件Autofac實現的。有三種註入方式,構造函數註入和方法註入和屬性註入。 ABP的屬性註入原則參考如下: 這時候我就開始疑惑了,因為我知道 ...
  • C#TMS系統代碼-業務頁面ShippingNotice學習 學一個業務頁面,ok,領導開完會就被裁掉了,很突然啊,他收拾東西的時候我還以為他要旅游提前請假了,還在尋思為什麼回家連自己買的幾箱飲料都要叫跑腿帶走,怕被偷嗎?還好我在他開會之前拿了兩瓶芬達 感覺感覺前面的BaseCity差不太多,這邊的 ...
  • 概述:在C#中,通過`Expression`類、`AndAlso`和`OrElse`方法可組合兩個`Expression<Func<T, bool>>`,實現多條件動態查詢。通過創建表達式樹,可輕鬆構建複雜的查詢條件。 在C#中,可以使用AndAlso和OrElse方法組合兩個Expression< ...
  • 閑來無聊在我的Biwen.QuickApi中實現一下極簡的事件匯流排,其實代碼還是蠻簡單的,對於初學者可能有些幫助 就貼出來,有什麼不足的地方也歡迎板磚交流~ 首先定義一個事件約定的空介面 public interface IEvent{} 然後定義事件訂閱者介面 public interface I ...
  • 1. 案例 成某三甲醫預約系統, 該項目在2024年初進行上線測試,在正常運行了兩天後,業務系統報錯:The connection pool has been exhausted, either raise MaxPoolSize (currently 800) or Timeout (curren ...
  • 背景 我們有些工具在 Web 版中已經有了很好的實踐,而在 WPF 中重新開發也是一種費時費力的操作,那麼直接集成則是最省事省力的方法了。 思路解釋 為什麼要使用 WPF?莫問為什麼,老 C# 開發的堅持,另外因為 Windows 上已經裝了 Webview2/edge 整體打包比 electron ...
  • EDP是一套集組織架構,許可權框架【功能許可權,操作許可權,數據訪問許可權,WebApi許可權】,自動化日誌,動態Interface,WebApi管理等基礎功能於一體的,基於.net的企業應用開發框架。通過友好的編碼方式實現數據行、列許可權的管控。 ...
  • .Net8.0 Blazor Hybird 桌面端 (WPF/Winform) 實測可以完整運行在 win7sp1/win10/win11. 如果用其他工具打包,還可以運行在mac/linux下, 傳送門BlazorHybrid 發佈為無依賴包方式 安裝 WebView2Runtime 1.57 M ...