成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

canvas && CSS 兩種實現(xiàn)儀表盤的方式

番茄西紅柿 / 2745人閱讀

摘要:先上效果圖這種圖形大家應該都見過,俗稱儀表盤,當然,上圖只是個最基本的儀表盤架子,可能在實際場景中還會其他很多花里胡哨的點綴,那些暫且不管,不是關(guān)鍵,這東西經(jīng)常見到,但還沒親自上手在代碼層面實現(xiàn)過,最近做的一個需求恰好有這個場景,這里歸

先上效果圖:

這種圖形大家應該都見過,俗稱儀表盤,當然,上圖只是個最基本的儀表盤架子,可能在實際場景中還會其他很多花里胡哨的點綴,那些暫且不管,不是關(guān)鍵,這東西經(jīng)常見到,但還沒親自上手在代碼層面實現(xiàn)過,最近做的一個需求恰好有這個場景,這里歸納一下

canvas實現(xiàn)

大部分情況下,對于這種偏可視化的元素,一般都選擇使用 canvas來進行繪制,現(xiàn)在已經(jīng) 9120年了,線上使用 canvas完全沒問題

儀表盤整體是一個復雜圖形,而復雜圖形是由簡單圖形組合而成,只要把所有組成這個儀表盤的簡單圖形繪制出來,再進行組合,整個儀表盤自然也就繪制出來了

所以,首先對儀表盤進行分解,分解成 canvas能繪制出的基本圖形,其主體其實就兩個圓弧,一個是底部藍色的半圓軌道,一個是代表進度的紅色圓弧,其實都是圓弧,canvas剛好有繪制圓弧的能力,即:

ctx.arc

至于動態(tài)繪制,只需要配合 requestAnimationFrame即可

const trackW = 6
const rx = 500
const ry = 500
const radius = 400
const innerLineW = 20
const canvas = document.getElementById(canvas)
const ctx = canvas.getContext(2d)
function draw (toAngle, currentAngle = Math.PI) {
  ctx.clearRect(0, 0, canvas.width, canvas.height)
  // 半圓軌道
  ctx.beginPath()
  ctx.strokeStyle = #ad80fc
  ctx.lineWidth = trackW
  ctx.arc(rx, ry, radius, Math.PI, 0, false)
  ctx.stroke()
  // 圓弧
  ctx.beginPath()
  ctx.lineCap = round
  ctx.strokeStyle = #fe4d55
  ctx.lineWidth = innerLineW
  ctx.arc(rx, ry, radius, Math.PI, currentAngle, false)
  ctx.stroke()
  if (currentAngle < toAngle) {
    currentAngle += 0.02
    if (currentAngle > toAngle) currentAngle = toAngle
    requestAnimationFrame(() => {
      draw(toAngle, currentAngle)
    })
  }
}
draw(1.5 * Math.PI)

加上變量定義,花括號等幾十行代碼即可完成,由此可見,canvas繪圖還是很方便的,所以在可視化領域,例如一些庫或者UI組件基本上都是以canvas進行構(gòu)建

css繪制

canvas實質(zhì)上就是借助 js操縱瀏覽器 API進行渲染,然而 UI渲染這種事情本應該交給 CSS來做才是,感覺用 js直接畫多影響性能?。▽嶋H上并不),哪有 css來的流暢,實際上,css完全可以做到

css的角度對儀表盤進行分解,同樣還是兩個圓弧,通過設置 border-radius屬性即可讓元素呈現(xiàn)整圓效果,然后再用一個矩形元素進行遮罩,決定展現(xiàn)出來的部分,即為圓弧,通過控制遮罩的面積來呈現(xiàn)動態(tài)繪制的效果

<div class="arc-wrapper">
  <p class="track-arc">p>
  <div class="round-box">
    <p class="round">p>
  div>
div>
:root {
  --arcRadius: 200px;
  --rectWidth: calc(var(--arcRadius) * 2);
  --trackWidth: 4px;
  --roundWidth: 10px;
}
.arc-wrapper {
  position: relative;
  margin: 0 auto;
  width: var(--rectWidth);
  height: var(--arcRadius);
  overflow: hidden;
  background-color: pink;
}
.track-arc {
  width: 100%;
  height: var(--rectWidth);
  box-sizing: border-box;
  border-radius: 50%;
  border: var(--trackWidth) solid #ad80fc;
}
.round-box {
  position: absolute;
  left: 0;
  top: 0;
  width: 100%;
  height: 100%;
  overflow: hidden;
  transform-origin: 50% 100%;
  transform: rotate(-45deg);
  z-index: 20;
}
.round {
  width: 100%;
  height: var(--rectWidth);
  box-sizing: border-box;
  border-radius: 50%;
  border: var(--roundWidth) solid #fe4d55;
}

其實沒多少代碼,也沒什么難以理解的,只不過效果似乎微調(diào):

由于圓弧的線是存在寬度的,并不是數(shù)學意義上的可以忽略,canvas繪制圓弧,是根據(jù)圓心坐標和半徑進行繪制的,繪制出來的圓弧會自動根據(jù)圓弧 line的寬度進行調(diào)整,即圓弧的半徑是圓弧線的中心位置與圓心坐標距離

而通過 css繪制的圓弧,此圓弧的半徑則是圓弧最外層邊線與圓心的坐標距離:

知道了問題其實就好解決了,只要縮減軌道半圓的半徑,并對其進行一定的偏移即可:

.track-arc {
  --trackArcSize: calc(var(--rectWidth) - var(--roundWidth) + var(--trackWidth));
  /* 尺寸改變 */
  width: var(--trackArcSize);
  height: var(--trackArcSize);
  box-sizing: border-box;
  border-radius: 50%;
  border: var(--trackWidth) solid #ad80fc;
  /* 位置偏移 */
  transform: translate(calc(var(--roundWidth) / 2 - var(--trackWidth) / 2), calc(var(--roundWidth) / 2 - var(--trackWidth) / 2));
}

然后就順眼多了:

然而,還有個問題,一般為了呈現(xiàn)更加圓潤的效果,設計稿上圓弧的斷點處一般都是圓頭:

而上述呈現(xiàn)出來的效果是直接截斷的:

一般人之所以不使用 css來繪制儀表盤,基本都是因為這個原因,canvas簡簡單單通過設置一個 ctx.lineCap = round就能解決的問題。似乎 css無解了

乍一看好像確實沒什么好辦法,但稍微思考下,這不就是一個圓角嗎,完全在css能力范圍內(nèi)啊,只不過實現(xiàn)的方式不太那么直接罷了

方法很簡單,就是使用一個圓角矩形覆蓋在圓弧的頂端,將圓弧本身的矩形頂端覆蓋住,圓角矩形當做是圓弧的頂端,這樣視覺上看起來不就是圓頭了嗎

<div class="round-box">
  <p class="round">p>
  <p class="dot-r-box">
    <span class="dot-r">span>
  p>
div>
.dot-r-box {
  position: absolute;
  right: 0;
  bottom: 0;
  width: var(--roundWidth);
  height: var(--dotHeight);
  background-color: var(--backColor);
}
.dot-r {
  display: inline-block;
  width: 100%;
  height: 100%;
  /* 這里的100px只是為了呈現(xiàn)出最大限度的圓角 */
  border-bottom-left-radius: 100px;
  border-bottom-right-radius: 100px;
  background-color: var(--roundColor);
}

效果如下:

圓角頂端 get

同樣的,圓弧左邊的頂端也可以這么做

不過左邊這個頂端有個稍微需要注意的地方,因為其存在的目的是為了當做圓弧的左斷點,但是當進度為 0的時候,圓弧應該是完全不展現(xiàn)的,或者當進度很小的時候,圓弧應該展現(xiàn)的長度還沒有 dot-l的高度大,這樣就露餡了:

不過呢轉(zhuǎn)而又一想,一般實際場景中,就算進度為 0,我們其實為了看起來更符合常識直覺,也會讓圓弧展現(xiàn)一點點出來,只要 dot-l的高度不是太大,或者說只要圓弧的寬度不要太寬,其實預留的這點圓弧完全就可以 cover住了,不至于露餡

小結(jié)

css的新特性 varcalc還挺好用的嘛

本文示例的 Live Demo已經(jīng)上傳,感興趣的可以親自試下

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/7088.html

相關(guān)文章

  • canvas &amp;&amp; CSS 兩種實現(xiàn)表盤方式

    摘要:先上效果圖這種圖形大家應該都見過,俗稱儀表盤,當然,上圖只是個最基本的儀表盤架子,可能在實際場景中還會其他很多花里胡哨的點綴,那些暫且不管,不是關(guān)鍵,這東西經(jīng)常見到,但還沒親自上手在代碼層面實現(xiàn)過,最近做的一個需求恰好有這個場景,這里歸先上效果圖: showImg(https://user-gold-cdn.xitu.io/2019/5/23/16ae28a94cb51d3e); 這種圖形大...

    stormjun 評論0 收藏0
  • 如何使用Python pyecharts繪制表盤

      小編寫這篇文章的主要目的,主要還是去進行講解一些關(guān)于Python pyecharts繪制儀表盤的一些講解,具體怎么去進行操作呢?下面就給大家詳細解答下?! x表盤  儀表盤的效果我只能說炫酷而已,如果想要運用在實際的場景中,我其實也不清楚那個場景比較適合,但是pyecharts畢竟是炫酷可視化的利器,炫酷自然也就有它了?! ⌒∑噧x表盤是長這樣的,下面我們來看看pyecharts的儀表盤是怎么...

    89542767 評論0 收藏0
  • vue項目中canvas實現(xiàn)截圖功能

      在vue項目中canvas實現(xiàn)截圖功能是常用的,下面是具體代碼:  實現(xiàn)效果:  在vue項目中做的一個截圖功能(只能夠截取圖片),只用鼠標就可以在畫面中進行框選截取?! 崿F(xiàn):做一個彈窗,打開彈窗的時候傳入要截的圖,接下來在這個窗口里面,點擊截圖按鈕,開始截圖;點擊取消按鈕,取消截圖?! 〈翱诶锩娴膆tml主要是三個部分,一個是可截圖區(qū)域,一個是截取圖片的回顯,一個是操作按鈕(截圖按鈕和取消...

    3403771864 評論0 收藏0
  • JS前端首屏優(yōu)化技巧

      訪問時間超過3S對于用戶就十分痛苦,為考慮到用戶訪問效果。為此我,首屏加載時間一頓操作,基本都在2s左右,這樣的首屏加載時間,對于用戶來說,算是可以接受的?! ∧俏叶寄男┎僮??下面就為大家展示  打包分析  在 package.json 中添加命令  "report":"vue-cli-servicebuild--report"  然后命令行執(zhí)行npm ...

    3403771864 評論0 收藏0
  • Vue+canvas實現(xiàn)視頻截圖功能

      上傳視頻要提供視頻封面(視頻封面必填),這是在開發(fā)中實際問題。封面可以用戶自己制作并上傳,但這樣脫離網(wǎng)站,體驗不好,常見的處理方案就是用戶未選擇或上傳封面時,自動截取視頻第一幀作為封面,但這樣并不友好。因此考慮視頻上傳后,在播放中由人員自行截取畫面作為視頻封面。  簡單效果如圖:  前端代碼如下:  <template>   <div>   <videosrc=&...

    3403771864 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<