国产av日韩一区二区三区精品,成人性爱视频在线观看,国产,欧美,日韩,一区,www.成色av久久成人,2222eeee成人天堂

目錄
折線
三階貝塞爾曲線
首頁 微信小程序 小程序開發(fā) 手把手教你在微信小程序中使用canvas繪制天氣折線圖(附代碼)

手把手教你在微信小程序中使用canvas繪制天氣折線圖(附代碼)

Feb 11, 2022 pm 07:59 PM
canvas 微信小程序

微信小程序中如何繪制天氣折線圖?下面本篇文章就來給大家介紹一下在微信小程序中使用canvas繪制天氣折線圖的方法,以及使用三階貝塞爾曲線擬合溫度點,使之變得圓滑,曲線底部有背景色,希望對大家有所幫助!

手把手教你在微信小程序中使用canvas繪制天氣折線圖(附代碼)

折線

效果圖:

1.gif

自定義組件 line-chart

<canvas type="2d" id="line" class="line-class" style="width:{{width}}px;height:{{height}}px" />
Component({
  externalClasses: [&#39;line-class&#39;],
  properties: {
    width: String,
    height: String,
    data: Array,
  },
  observers: {
    width() {
      // 這里監(jiān)聽 width 變化重繪 canvas
      // 動態(tài)傳入 width 好像只能這樣了..
      const query = this.createSelectorQuery();
      query
        .select(&#39;#line&#39;)
        .fields({ node: true, size: true })
        .exec(res => {
          const canvas = res[0].node;
          const ctx = canvas.getContext(&#39;2d&#39;);
          const width = res[0].width; // 畫布寬度
          const height = res[0].height; // 畫布高度

          console.log(`寬度: ${width}, 高度: ${height}`);

          const dpr = wx.getSystemInfoSync().pixelRatio;
          canvas.width = width * dpr;
          canvas.height = height * dpr;
          ctx.scale(dpr, dpr);

          // 開始繪圖
          this.drawLine(ctx, width, height, this.data.data);
        });
    },
  },
  methods: {
    drawLine(ctx, width, height, data) {
      const Max = Math.max(...data);
      const Min = Math.min(...data);

      // 把 canvas 的寬度, 高度按一定規(guī)則平分
      const startX = width / (data.length * 2), // 起始點的橫坐標(biāo) X
        baseY = height * 0.9, // 基線縱坐標(biāo) Y
        diffX = width / data.length,
        diffY = (height * 0.7) / (Max - Min); // 高度預(yù)留 0.2 寫溫度

      ctx.beginPath();
      ctx.textAlign = &#39;center&#39;;
      ctx.font = &#39;13px Microsoft YaHei&#39;;
      ctx.lineWidth = 2;
      ctx.strokeStyle = &#39;#ABDCFF&#39;;

      // 畫折線圖的線
      data.forEach((item, index) => {
        const x = startX + diffX * index,
          y = baseY - (item - Min) * diffY;

        ctx.fillText(`${item}°`, x, y - 10);
        ctx.lineTo(x, y);
      });
      ctx.stroke();

      // 畫折線圖背景
      ctx.lineTo(startX + (data.length - 1) * diffX, baseY); // 基線終點
      ctx.lineTo(startX, baseY); // 基線起點
      const lingrad = ctx.createLinearGradient(0, 0, 0, height * 0.7);
      lingrad.addColorStop(0, &#39;rgba(255,255,255,0.9)&#39;);
      lingrad.addColorStop(1, &#39;rgba(171,220,255,0)&#39;);
      ctx.fillStyle = lingrad;
      ctx.fill();

      // 畫折線圖上的小圓點
      ctx.beginPath();
      data.forEach((item, index) => {
        const x = startX + diffX * index,
          y = baseY - (item - Min) * diffY;

        ctx.moveTo(x, y);
        ctx.arc(x, y, 3, 0, 2 * Math.PI);
      });
      ctx.fillStyle = &#39;#0396FF&#39;;
      ctx.fill();
    },
  },
});

data 就是溫度數(shù)組,如 [1, 2, ...]

因為不知道溫度數(shù)值有多少個,因此這里的 width 動態(tài)傳入

有個小問題,就是寬度過大的話真機不會顯示...

 // 獲取 scroll-view 的總寬度
 wx.createSelectorQuery()
      .select(&#39;.hourly&#39;)
      .boundingClientRect(rect => {
        this.setData({
          scrollWidth: rect.right - rect.left,
        });
      })
      .exec();
<view class="title">小時概述</view>
<scroll-view scroll-x scroll-y class="scroll" show-scrollbar="{{false}}" enhanced="{{true}}">
    <view class="hourly">
      <view wx:for="{{time}}" wx:key="index">{{item}}</view>
    </view>
    <line-chart line-class="line" width="{{scrollWidth}}" height="100" data="{{temp}}" />
</scroll-view>

這里寫 scroll-x 和 scroll-y,要不會出現(xiàn)絕對定位偏移的問題,也不知道為什么

2.gif

.scroll {
  position: relative;
  height: 150px;
  width: 100%;
}

.hourly {
  display: flex;
  height: 150px;
  position: absolute;
  top: 0;
}

.hourly > view {
  min-width: 3.5em;
  text-align: center;
}

.line { // 折線圖絕對定位到底部
  position: absolute;
  bottom: 0;
}

這里使用絕對定位其實是想模擬墨跡天氣這種折線圖和每一天在一個塊內(nèi)的效果,所以 hourly 要和 scroll-view 等高,canvas 需要定位一下

主要是不知道墨跡天氣怎么實現(xiàn)的,只能暫時這樣

3.gif

三階貝塞爾曲線

效果圖

4.gif

emmm,好像并不怎么圓滑

計算控制點

首先寫一個點類

class Point {
  constructor(x, y) {
    this.x = x;
    this.y = y;
  }
}

Canvas貝塞爾曲線繪制工具 (karlew.com)

http://wx.karlew.com/canvas/bezier/

通過上面這個網(wǎng)站可以知道三階貝塞爾曲線各個參數(shù)的意義

5.gif

也就是使用 bezierCurveTo 的時候最后一個點是下一個點,前兩個是控制點

控制點的計算參考: 貝塞爾曲線控制點確定的方法 - 百度文庫

https://wenku.baidu.com/view/c790f8d46bec0975f565e211.html

濃縮一下就是

6.gif

這里的 a 和 b 可以是任意正數(shù)

因此定義一個計算某點的控制點 A 和 B 的方法

/**
 * 計算當(dāng)前點的貝塞爾曲線控制點
 * @param {Point} previousPoint: 前一個點
 * @param {Point} currentPoint: 當(dāng)前點
 * @param {Point} nextPoint1: 下一個點
 * @param {Point} nextPoint2: 下下個點
 * @param {Number} scale: 系數(shù)
 */
calcBezierControlPoints(
  previousPoint,
  currentPoint,
  nextPoint1,
  nextPoint2,
  scale = 0.25
) {
  let x = currentPoint.x + scale * (nextPoint1.x - previousPoint.x);
  let y = currentPoint.y + scale * (nextPoint1.y - previousPoint.y);

  const controlPointA = new Point(x, y); // 控制點 A

  x = nextPoint1.x - scale * (nextPoint2.x - currentPoint.x);
  y = nextPoint1.y - scale * (nextPoint2.y - currentPoint.y);

  const controlPointB = new Point(x, y); // 控制點 B

  return { controlPointA, controlPointB };
}

這里 scale 就是 a 和 b,不過將它們的取值相等

但是第一個點沒有 previousPoint,倒數(shù)第二個點沒有 nextPoint2

因此當(dāng)點是第一個的時候,使用 currentPoint 代替 previousPoint

當(dāng)?shù)箶?shù)第二個點的時候,使用 nextPoint1 代替 nextPoint2

7.gif

至于最后一個點,不需要做任何事,因為 bezierCurveTo 第三個參數(shù)就是下一個點,只需要提供坐標(biāo)就能連起來,不需要計算控制點

因此繪制三階貝塞爾曲線的方法:

/**
 * 繪制貝塞爾曲線
 * ctx.bezierCurveTo(控制點1, 控制點2, 當(dāng)前點);
 */
drawBezierLine(ctx, data, options) {
  const { startX, diffX, baseY, diffY, Min } = options;

  ctx.beginPath();
  // 先移動到第一個點
  ctx.moveTo(startX, baseY - (data[0] - Min) * diffY);

  data.forEach((e, i) => {
    let curPoint, prePoint, nextPoint1, nextPoint2, x, y;

    // 當(dāng)前點
    x = startX + diffX * i;
    y = baseY - (e - Min) * diffY;
    curPoint = new Point(x, y);

    // 前一個點
    x = startX + diffX * (i - 1);
    y = baseY - (data[i - 1] - Min) * diffY;
    prePoint = new Point(x, y);

    // 下一個點
    x = startX + diffX * (i + 1);
    y = baseY - (data[i + 1] - Min) * diffY;
    nextPoint1 = new Point(x, y);

    // 下下個點
    x = startX + diffX * (i + 2);
    y = baseY - (data[i + 2] - Min) * diffY;
    nextPoint2 = new Point(x, y);

    if (i === 0) {
      // 如果是第一個點, 則前一個點用當(dāng)前點代替
      prePoint = curPoint;
    } else if (i === data.length - 2) {
      // 如果是倒數(shù)第二個點, 則下下個點用下一個點代替
      nextPoint2 = nextPoint1;
    } else if (i === data.length - 1) {
      // 最后一個點直接退出
      return;
    }

    const { controlPointA, controlPointB } = this.calcBezierControlPoints(
      prePoint,
      curPoint,
      nextPoint1,
      nextPoint2
    );

    ctx.bezierCurveTo(
      controlPointA.x,
      controlPointA.y,
      controlPointB.x,
      controlPointB.y,
      nextPoint1.x,
      nextPoint1.y
    );
  });

  ctx.stroke();
},

【相關(guān)學(xué)習(xí)推薦:小程序開發(fā)教程

以上是手把手教你在微信小程序中使用canvas繪制天氣折線圖(附代碼)的詳細內(nèi)容。更多信息請關(guān)注PHP中文網(wǎng)其他相關(guān)文章!

本站聲明
本文內(nèi)容由網(wǎng)友自發(fā)貢獻,版權(quán)歸原作者所有,本站不承擔(dān)相應(yīng)法律責(zé)任。如您發(fā)現(xiàn)有涉嫌抄襲侵權(quán)的內(nèi)容,請聯(lián)系admin@php.cn

熱AI工具

Undress AI Tool

Undress AI Tool

免費脫衣服圖片

Undresser.AI Undress

Undresser.AI Undress

人工智能驅(qū)動的應(yīng)用程序,用于創(chuàng)建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用于從照片中去除衣服的在線人工智能工具。

Clothoff.io

Clothoff.io

AI脫衣機

Video Face Swap

Video Face Swap

使用我們完全免費的人工智能換臉工具輕松在任何視頻中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的代碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

功能強大的PHP集成開發(fā)環(huán)境

Dreamweaver CS6

Dreamweaver CS6

視覺化網(wǎng)頁開發(fā)工具

SublimeText3 Mac版

SublimeText3 Mac版

神級代碼編輯軟件(SublimeText3)

閑魚微信小程序正式上線 閑魚微信小程序正式上線 Feb 10, 2024 pm 10:39 PM

閑魚官方微信小程序悄然上線,在小程序中可以發(fā)布閑置與買家/賣家私信交流、查看個人資料及訂單、搜索物品等,有用好奇閑魚微信小程序叫什么,現(xiàn)在快來看一下。閑魚微信小程序叫什么答案:閑魚,閑置交易二手買賣估價回收。1、在小程序中可以發(fā)布閑置、與買家/賣家私信交流、查看個人資料及訂單、搜索指定物品等功能;2、在小程序的頁面中有首頁、附近、發(fā)閑置、消息、我的5項功能;3、想要使用的話必要要開通微信支付才可以購買;

實現(xiàn)微信小程序中的圖片濾鏡效果 實現(xiàn)微信小程序中的圖片濾鏡效果 Nov 21, 2023 pm 06:22 PM

實現(xiàn)微信小程序中的圖片濾鏡效果隨著社交媒體應(yīng)用的流行,人們越來越喜歡在照片中應(yīng)用濾鏡效果,以增強照片的藝術(shù)效果和吸引力。在微信小程序中也可以實現(xiàn)圖片濾鏡效果,為用戶提供更多有趣和創(chuàng)造性的照片編輯功能。本文將介紹如何在微信小程序中實現(xiàn)圖片濾鏡效果,并提供具體的代碼示例。首先,我們需要在微信小程序中使用canvas組件來加載和編輯圖片。canvas組件可以在頁面

實現(xiàn)微信小程序中的下拉菜單效果 實現(xiàn)微信小程序中的下拉菜單效果 Nov 21, 2023 pm 03:03 PM

實現(xiàn)微信小程序中的下拉菜單效果,需要具體代碼示例隨著移動互聯(lián)網(wǎng)的普及,微信小程序成為了互聯(lián)網(wǎng)開發(fā)的重要一環(huán),越來越多的人開始關(guān)注和使用微信小程序。微信小程序的開發(fā)相比傳統(tǒng)的APP開發(fā)更加簡便快捷,但也需要掌握一定的開發(fā)技巧。在微信小程序的開發(fā)中,下拉菜單是一個常見的UI組件,實現(xiàn)了更好的用戶操作體驗。本文將詳細介紹如何在微信小程序中實現(xiàn)下拉菜單效果,并提供具

閑魚微信小程序叫什么 閑魚微信小程序叫什么 Feb 27, 2024 pm 01:11 PM

閑魚官方微信小程序已經(jīng)悄然上線,它為用戶提供了一個便捷的平臺,讓你可以輕松地發(fā)布和交易閑置物品。在小程序中,你可以與買家或賣家進行私信交流,查看個人資料和訂單,以及搜索你想要的物品。那么閑魚在微信小程序中究竟叫什么呢,這篇教程攻略將為您詳細介紹,想要了解的用戶們快來跟著本文繼續(xù)閱讀吧!閑魚微信小程序叫什么答案:閑魚,閑置交易二手買賣估價回收。1、在小程序中可以發(fā)布閑置、與買家/賣家私信交流、查看個人資料及訂單、搜索指定物品等功能;2、在小程序的頁面中有首頁、附近、發(fā)閑置、消息、我的5項功能;3、

微信小程序?qū)崿F(xiàn)圖片上傳功能 微信小程序?qū)崿F(xiàn)圖片上傳功能 Nov 21, 2023 am 09:08 AM

微信小程序?qū)崿F(xiàn)圖片上傳功能隨著移動互聯(lián)網(wǎng)的發(fā)展,微信小程序已經(jīng)成為了人們生活中不可或缺的一部分。微信小程序不僅提供了豐富的應(yīng)用場景,還支持開發(fā)者自定義功能,其中包括圖片上傳功能。本文將介紹如何在微信小程序中實現(xiàn)圖片上傳功能,并提供具體的代碼示例。一、前期準(zhǔn)備工作在開始編寫代碼之前,我們需要先下載并安裝微信開發(fā)者工具,并注冊成為微信開發(fā)者。同時,還需要了解微信

使用微信小程序?qū)崿F(xiàn)輪播圖切換效果 使用微信小程序?qū)崿F(xiàn)輪播圖切換效果 Nov 21, 2023 pm 05:59 PM

使用微信小程序?qū)崿F(xiàn)輪播圖切換效果微信小程序是一種輕量級的應(yīng)用程序,具有簡單、高效的開發(fā)和使用特點。在微信小程序中,實現(xiàn)輪播圖切換效果是常見的需求。本文將介紹如何使用微信小程序?qū)崿F(xiàn)輪播圖切換效果,并給出具體的代碼示例。首先,在微信小程序的頁面文件中,添加一個輪播圖組件。例如,可以使用&lt;swiper&gt;標(biāo)簽來實現(xiàn)輪播圖的切換效果。在該組件中,可以通過b

實現(xiàn)微信小程序中的圖片旋轉(zhuǎn)效果 實現(xiàn)微信小程序中的圖片旋轉(zhuǎn)效果 Nov 21, 2023 am 08:26 AM

實現(xiàn)微信小程序中的圖片旋轉(zhuǎn)效果,需要具體代碼示例微信小程序是一種輕量級的應(yīng)用程序,為用戶提供了豐富的功能和良好的用戶體驗。在小程序中,開發(fā)者可以利用各種組件和API來實現(xiàn)各種效果。其中,圖片旋轉(zhuǎn)效果是一種常見的動畫效果,可以為小程序增添趣味性和視覺效果。在微信小程序中實現(xiàn)圖片旋轉(zhuǎn)效果,需要使用小程序提供的動畫API。下面是一個具體的代碼示例,展示了如何在小程

學(xué)習(xí)canvas框架 詳解常用的canvas框架 學(xué)習(xí)canvas框架 詳解常用的canvas框架 Jan 17, 2024 am 11:03 AM

探索Canvas框架:了解常用的Canvas框架有哪些,需要具體代碼示例引言:Canvas是HTML5中提供的一個繪圖API,通過它我們可以實現(xiàn)豐富的圖形和動畫效果。為了提高繪圖的效率和便捷性,許多開發(fā)者開發(fā)了不同的Canvas框架。本文將介紹一些常用的Canvas框架,并提供具體代碼示例,以幫助讀者更深入地了解這些框架的使用方法。一、EaselJS框架Ea

See all articles