闂傚倸鍊搁崐鐑芥嚄閼哥數浠氱紓鍌欒兌缁垶宕归崜浣瑰床婵炴垶鐟х弧鈧梺绋款儓婵倝鎯勯鐐叉瀬闁瑰墽绮弲鎼佹煥閻曞倹瀚�
80KM婵犵數濮烽弫鍛婃叏娴兼潙鍨傞柣鎾崇岸閺嬫牗绻涢幋鐐╂(婵炲樊浜濋弲鎻掝熆鐠虹尨榫氶柛鈺冨仱濮婃椽妫冨☉姘暫濠碘槅鍋呴悷锕傚箞閵娿儮鍫柛鏇楁櫃缁ㄥ姊洪崫鍕犻柛鏂块叄楠炲﹪宕熼鍙ョ盎闂佽濯介崺鏍偓姘炬嫹
闂傚倸鍊搁崐鐑芥嚄閸洖鍌ㄧ憸鏃堝Υ閸愨晜鍎熼柕蹇嬪焺濞茬ǹ鈹戦悩璇у伐閻庢凹鍙冨畷锝堢疀濞戞瑧鍘撻梺鍛婄箓鐎氼剟鍩€椤掆偓閹芥粌鈻庨姀銈嗗€烽柣鎴炨缚閸橀亶姊洪棃娑辨▓闁搞劍濞婇幃楣冩焼瀹ュ棛鍘告繛杈剧到濠€閬嶆儗閹烘鐓涢悘鐐额嚙婵″ジ鏌嶉挊澶樻Ц閾伙綁鏌涢…鎴濇珮濞寸厧鍟村缁樻媴妞嬪簼瑕嗙紓鍌氱С閼冲墎鍒掓繝姘唨鐟滄粓宕甸弴鐔翠簻闁哄啫鍊告禍鍓р偓瑙勬礀椤︻垶濡撮幒鎴僵闁绘挸娴锋禒顓㈡煛瀹ヤ讲鍋撻弬銉︽杸闂佺粯鍔曞Ο濠囧吹閻斿皝鏀芥い鏍ㄧ⊕鐏忥附顨ラ悙鑼闁轰焦鎹囬弫鎾绘晸閿燂拷
闂傚倸鍊峰ù鍥х暦閸偅鍙忕€规洖娲︽刊濂告煛鐏炶鍔氶柣銈囧亾缁绘盯宕卞Ο铏逛患缂備讲鍋撳┑鐘插暞閸欏繑淇婇悙棰濆殭濞存粓绠栧铏规嫚閳ュ磭浠╅梺鍝ュ枑濞兼瑩鎮鹃悜鑺ュ亜缁炬媽椴搁弲銏$箾鏉堝墽绉い鏇熺墵瀹曨垶鍩€椤掑嫭鈷掗柛灞剧懆閸忓本銇勯姀鐙呰含妞ゃ垺宀稿浠嬵敇閻愮數宕舵繝寰锋澘鈧劙宕戦幘娣簻闁宠桨闄嶉崑銏⑩偓瑙勬礀閵堟悂骞冮姀銏″仒闁斥晛鍟版禍娆撴⒑鐠囨煡顎楃紒鐘茬Ч瀹曟洘娼忛埞鎯т壕婵ḿ鍘ч獮妤冪磼鐎n亶妯€濠殿喒鍋撻梺缁樼憿閸嬫捇鏌i弬鎸庮棦闁诡喛顫夐幏鍛驳鐎n偆绉烽梺璇插閻旑剟骞忛敓锟�

教你在微信小程序中使用canvas绘制天气折线图(附代码)-小程序开发

首页 2024-07-09 09:50:08

如何在微信applet中绘制天气折线图?下面的文章将介绍在微信applet中使用canvas绘制天气折线图的方法,以及使用三阶贝塞尔曲线拟合温度点使其光滑。曲线底部有背景色。我希望它能帮助你!

折线

效果图:

自定义组件 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() {
      // 这里监听 width 变化重绘 canvas
      // 动态传入 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 的宽度, 高度按一定规则平分
      const startX = width / (data.length * 2), // 起点的横坐标 X
        baseY = height * 0.9, // 基线纵坐标 Y
        diffX = width / data.length,
        diffY = (height * 0.7) / (Max - Min); // 高度预留 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 就是温度数组,比如 [1, 2, ...]

因为我不知道有多少温度值,所以这里 width 动态传入

有一个小问题,就是宽度太大,真机不会显示。...

 // 获取 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,不知道为什么会出现绝对定位偏移的问题

.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;
}

在这里使用绝对定位实际上是为了模拟墨天气的折线图和每天在一个块内的效果,所以 hourly 要和 scroll-view 等高,canvas 需要定位

我不知道如何实现墨迹天气,所以我只能暂时这样做

贝塞尔曲线的三阶

效果图

emmm,好像不是很光滑

计算控制点

首先写一个点类

class Point {
  constructor(x, y) {
    this.x = x;
    this.y = y;
  }
}
Canvas贝塞尔曲线绘制工具 (karlew.com)http://wx.karlew.com/canvas/bezier/

通过以上网站,我们可以知道三阶贝塞尔曲线各参数的意义

也就是使用 bezierCurveTo 最后一点是下一点,前两点是控制点

控制点的计算参考: 确定贝塞尔曲线控制点的方法 - 百度文库

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

浓缩就是

这里的 a 和 b 可以是任何正数

因此,定义计算某一点的控制点 A 和 B 的方法

/**
 * 计算当前贝塞尔曲线控制点
 * @param {Point} previousPoint: 前一个点
 * @param {Point} currentPoint: 当前点
 * @param {Point} nextpointt1: 下一个点
 * @param {Point} nextpoint2: 下下个点
 * @param {Number} scale: 系数
 */
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 = nextpointt1.x - scale * (nextPoint2.x - currentPoint.x);
  y = nextpointt1.y - scale * (nextPoint2.y - currentPoint.y);

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

  return { controlPointA, controlPointB };
}

这里 scale 就是 a 和 b,然而,相等于它们的值

但是第一点没有 previousPoint,倒数第二点没有 nextpoint2

所以当点是第一次使用时, currentPoint 代替 previousPoint

使用倒数第二点时 nextpointt1 代替 nextpoint2

至于最后一点,没有必要做任何事情,因为 bezierCurveTo 第三个参数是下一点,只需提供坐标即可连接,无需计算控制点

因此,绘制三阶贝塞尔曲线的方法:

/**
 * 绘制贝塞尔曲线
 * ctx.bezierCurveTo(控制点1, 控制点2, 当前点);
 */
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;

    // 当前点
    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;
    nextpointt1 = new Point(x, y);

    // 下下个点
    x = startX   diffX * (i   2);
    y = baseY - (data[i   2] - Min) * diffY;
    nextpoint2 = new Point(x, y);

    if (i === 0) {
      // 假如是第一点, 前一点用当前点代替
      prePoint = curPoint;
    } else if (i === data.length - 2) {
      // 如果是倒数第二点, 下一个点用下一个点代替
      nextpoint2 = nextpointt1;
    } 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,
      nextpointt1.x,
      nextpointt1.y
    );
  });

  ctx.stroke();
},

【相关学习推荐:小程序开发教程】

以上是微信小程序中使用canvas绘制天气折线图(附代码)的详细内容。请多关注其他相关文章!


p
SEO闂傚倸鍊搁崐椋庣矆娴h櫣绀婂┑鐘插€寸紓姘辨喐韫囨洘顫曢柣鎰嚟缁♀偓闂佹悶鍎滈崶顭掔船濠电姷鏁搁崑娑樜熸繝鍐洸婵犲﹤鐗婄€氬懘鏌i弬鍨倯闁绘挶鍎甸弻锝夊即閻愭祴鍋撻崷顓涘亾濮樼偓瀚�
闂傚倸鍊搁崐椋庣矆娓氣偓楠炴牠顢曢敂钘変罕闂佺硶鍓濋悷褔鎯岄幘缁樺€垫繛鎴烆伆閹达箑鐭楅煫鍥ㄧ⊕閻撶喖鏌¢崘銊モ偓鍝ユ暜閸洘鈷掗柛灞诲€曢悘锕傛煛鐏炵偓绀冪紒缁樼椤︽煡鏌¢崼顐㈠⒋鐎规洜濞€閹晝绱掑Ο閿嬪婵犵數鍋犵亸娆戝垝椤栨粍顐芥繛鎴欏灪閻撴瑩鏌涢幋娆忊偓鏍偓姘炬嫹
闂傚倸鍊风粈渚€骞栭位鍥敃閿曗偓閻ょ偓绻濇繝鍌涘櫣闁搞劍绻堥獮鏍庨鈧俊濂告煟閹惧绠撻柍瑙勫灴瀹曟帒鈹冮幘铏础闁逞屽墯閼归箖藝闁秴鐒垫い鎺嗗亾缂佺姴绉瑰畷鏇㈡焼瀹ュ懐鐤囬柟鍏兼儗閻撳绱為弽顓熺厪闁割偅绻嶅Σ褰掓煟閹惧瓨绀嬮柡灞诲妼閳规垿宕卞Δ浣诡唲濠电姷顣介崜婵嬪箖閸岀偛钃熺€广儱鐗滃銊╂⒑缁嬭法绠茬紒瀣灴濠€渚€姊洪幖鐐插姉闁哄懏绮岄悾鐑藉矗婢跺瞼顔曢梺绯曞墲閿氶柣蹇婃櫊閺岋綁顢橀悢鐑樺櫑闂佸疇顫夐崹鍧椼€佸☉妯滄棃鍩€椤掍胶顩茬紓宥囧瘲闂傚倷娴囬褍顫濋敃鍌︾稏濠㈣埖鍔曠粻鏍煕椤愶絾绀€缁炬儳娼″娲敆閳ь剛绮旈幘顔藉剹婵°倕鎳忛悡銉╂煟閺囩偛鈧湱鈧熬鎷�
婵犵數濮烽弫鎼佸磻閻愬搫鍨傞柛顐f礀缁犱即鏌熺紒銏犳灈缁炬儳顭烽弻鐔煎礈瑜忕敮娑㈡煃闁垮鐏︾紒缁樼洴瀹曞崬螣閸濆嫬袘闂備礁鎼鍡涙偡閳哄懎钃熼柣鏂挎憸閻熷綊鏌涢…鎴濇灈妞ゎ偄娲幃妤€鈻撻崹顔界亖闂佸憡鏌ㄦ鎼佸煡婢舵劖鍋ㄧ紒瀣仢缁愭稑顪冮妶鍡欏缂侇喚濞€瀹曨垰鐣濋埀顒傛閹捐纾兼繛鍡樺焾濡差喖顪冮妶鍡楃仴闁硅櫕锕㈤妴渚€寮介鐐靛€炲銈嗗笒椤︿即寮插⿰鍐炬富闁靛牆妫楃粭鎺楁倵濮樼厧澧撮柟顖氳嫰铻栭柛娑卞枤閸欏棝姊虹紒妯荤闁稿﹤婀遍埀顒佺啲閹凤拷
C闂傚倸鍊搁崐鐑芥嚄閸洖纾块柣銏⑶归悿鐐節婵犲倸鏆熸鐐存崌閺屾稖顦虫い銊ユ嚇瀹曞綊宕掗悙鑼啇闁哄鐗嗘晶浠嬪箖閸忛棿绻嗘い鎰靛亜閻忥繝鏌曢崶褍顏い銏℃礋椤㈡洟濮€閿涘嫪澹曠紓鍌氬€风拋鏌ュ磻閹炬剚鐔嗛悹杞拌閸庢垹绱掗悩鑽ょ暫闁哄瞼鍠栭獮鎴﹀箛椤撶姴娑ч梻渚€娼荤徊鑲╁垝濞嗘挸钃熼柣鏃傗拡閺佸﹦鐥鐐叉Щ濞村吋鍔曢—鍐Χ閸℃ḿ鍙嗙紓浣虹帛钃卞ǎ鍥э躬閹粓鎸婃竟鈹垮姂閺屾洘寰勯崼婵嗗Б濠碘槅鍨介幏锟�

最新文章

  • 高效稳定:文件服务器搭建全面解决方案

  • 教你在微信小程序中使用canvas绘制天气折线图(附代码)-小程序开发

  • 远程控制电脑,轻松保持屏幕常亮解锁!

  • 高效应对:大数据粘贴难题的解决之道

  • 远程掌控,网页版一键连接,高效无忧

  • 如何快速升级剑与远征——手游攻略

  • 服务器端口设置:关键步骤,高效配置

  • 相关文章

  • 如何快速升级剑与远征——手游攻略

  • 实现微信发布文章信息收集-微信开发

  • 转发和合并转发的区别是什么——微信开发

  • behaviors在小程序中是什么?如何创建和使用?-小程序开发

  • 总结和分享微信小程序开发步骤-小程序开发

  • 本文为您带来了微信小程序的相关知识,主要介绍了微信小程序的开发步骤和主要流程,希望对您有所帮助。

    【相关学习推荐:小程序学习教程】

    目前,applet行业已成为互联网营销的热门黑马之一。applet行业依托主流平台,具有自然的用户基础和独特的资源优势。它以其方便快捷的操作和简单流行的模式,在短短一年内迎来了爆炸性的增长。目前市场上开发的小程序从几千到几万不等。以腾讯云为例,腾讯官方团队设

  • 谈谈小程序如何实现“全文收起”功能-小程序开发

  • 微信小程序开发的底部导航-小程序开发

  • 总结整理微信小程序常用表单组件-小程序开发

  • 总结分享微信小程序常见面试题-小程序开发

  • 简单分析微信小程序元素-小程序开发

  • 总结整理微信小程序权限界面-小程序开发