微信小程序实现带刻度尺滑块

效果图

输入图片说明
输入图片说明

场景

当一屏显示不下,例如年龄体重选择,金额选择等大区间需要的选择器,相比自带的picker要直观一些。

思路:

  1. 先画一个scrollView
    2 装进canvas
  2. lineTo画刻度线段,lineTo+fill画出三角形游标,fillText描绘文本标签
  3. 通过bindscroll监听刻度尺触摸事件
  4. 渲染取值到页面

基本布局

<scroll-view scroll-x="true" bindscroll="bindscroll">
    <canvas canvas-id="canvas" id="canvas"></canvas>
</scroll-view>

实现bindscroll方法

bindscroll: function (e) {
    // deltaX 水平位置偏移位,每次滑动一次触发一次,所以需要记录从第一次触发滑动起,一共滑动了多少距离
    deltaX += e.detail.deltaX;
    console.log(deltaX)
}

描绘刻度

const context = wx.createCanvasContext('canvas-ruler');
// 移动到原点
context.moveTo(origion.x, origion.y);
// 画线到刻度高度
context.lineTo(origion.x, origion.y - heightDecimal);
// 设置属性
context.setLineWidth(1);
// 描线
context.stroke();
// 描绘文本标签
context.setFontSize(fontSize);
context.fillText('0', origion.x - fontSize / 2, fontSize);
context.draw();

遍历刻度

for (var i = 0; i <= maxValue; i++) {
    // 开始一个路径,这条非常重要,否则会重复绘制之前的刻度n次,效果表现为页面加载很卡,lineWidth得到的线很粗
    context.beginPath();
    // 绘制同上,不再赘述
    ...
    // 关闭一个路径,它是可选的,调用过了beginPath,不关闭也没有影响,保险起见,加上它
    context.closePath();
}

切记要调用context.beginPath();

描绘游标

drawCursor: function () {
        /* 定义变量 */
        // 定义三角形顶点 TODO x
        var center = {x: app.screenWidth / 2, y: 5};
        // 定义三角形边长
        var length = 20;
        // 左端点
        var left = {x: center.x - length / 2, y: center.y + length / 2 * Math.sqrt(3)};
        // 右端点
        var right = {x: center.x + length / 2, y: center.y + length / 2 * Math.sqrt(3)};
        // 初始化context
        const context = wx.createCanvasContext('canvas-cursor');
        context.moveTo(center.x, center.y);
        context.lineTo(left.x, left.y);
        context.lineTo(right.x, right.y);
        // fill()填充而不是stroke()描边,于是省去手动回归原点,context.lineTo(center.x, center.y);
        context.setFillStyle('#48c23d');
        context.fill();
        context.draw();
    }

画带一个绿色的正三角形作为游标,注意游标是悬浮不动的,所以另起一个cancas来装它。当然它不是必须的,偷个懒ps一张三角形的png代替也无妨,甚至刻度其实也可以用<view style="background: gray; width: 2px;">加绝对定位来生成的。

定义刻度默认初值

that.setData({
        scrollLeft: (currentValue - minValue) * ratio
});

<scroll-view scroll-x="true" bindscroll="bindscroll" scroll-left="{{scrollLeft}}">

绑定scroll-left参数,相当于iOS里了UIScrollView的contentOffset,手动让偏移到默认初值对应的坐标位置。

适配最小值

当业务场景需要做数据验证,例如金额要>0,年龄要大于18岁等,就得适配极值。

that.setData({
        amount: Math.floor(- deltaX / 10 + minValue)
});

同时要修正�刻度线的x轴坐标

// 2.2 画刻度线
context.moveTo(origion.x + (i - minValue) * ratio, origion.y);
// 画线到刻度高度,10的位数就加高
context.lineTo(origion.x + (i - minValue) * ratio, origion.y - (i % ratio == 0 ? heightDecimal : heightDigit));
// 2.3 描绘文本标签
context.fillText(i == 0 ? ' ' + i : i, origion.x + (i - minValue) * ratio - fontSize / 2, fontSize);

最终js代码

var that;
var deltaX = 0;
var minValue = 1;
var app = getApp();
Page({
    data: {
        value: 0,
        canvasHeight: 80
    },
    onLoad: function (options) {
        that = this;
        // 绘制标尺
        that.drawRuler();
        // 绘制三角形游标
        that.drawCursor();
    },
    drawRuler: function() {

        /* 1.定义变量 */

        // 1.1 定义原点与终点,x轴方向起点与终点各留半屏空白
        var origion = {x: app.screenWidth / 2, y: that.data.canvasHeight};
        var end = {x: app.screenWidth / 2, y: that.data.canvasHeight};
        // 1.2 定义刻度线高度
        var heightDecimal = 50;
        var heightDigit = 25;
        // 1.3 定义文本标签字体大小
        var fontSize = 20;
        // 1.4 最小刻度值
        // 已经定义在全局,便于bindscroll访问
        // 1.5 总刻度值
        var maxValue = 200;
        // 1.6 当前刻度值
        var currentValue = 20;
        // 1.7 每个刻度所占位的px
        var ratio = 10;
        // 1.8 画布宽度
        var canvasWidth = maxValue * ratio + app.screenWidth - minValue * ratio;
        // 设定scroll-view初始偏移
        that.setData({
            canvasWidth: canvasWidth,
            scrollLeft: (currentValue - minValue) * ratio
        });

        /* 2.绘制 */

        // 2.1初始化context
        const context = wx.createCanvasContext('canvas-ruler');
        // 遍历maxValue
        for (var i = 0; i <= maxValue; i++) {
            context.beginPath();
            // 2.2 画刻度线
            context.moveTo(origion.x + (i - minValue) * ratio, origion.y);
            // 画线到刻度高度,10的位数就加高
            context.lineTo(origion.x + (i - minValue) * ratio, origion.y - (i % ratio == 0 ? heightDecimal : heightDigit));
            // 设置属性
            context.setLineWidth(2);
            // 10的位数就加深
            context.setStrokeStyle(i % ratio == 0 ? 'gray' : 'darkgray');
            // 描线
            context.stroke();
            // 2.3 描绘文本标签
            context.setFillStyle('gray');
            if (i % ratio == 0) {
                context.setFontSize(fontSize);
                // 为零补一个空格,让它看起来2位数,页面更整齐
                context.fillText(i == 0 ? ' ' + i : i, origion.x + (i - minValue) * ratio - fontSize / 2, fontSize);
            }
            context.closePath();
        }

        // 2.4 绘制到context
        context.draw();
    },
    drawCursor: function () {
        /* 定义变量 */
        // 定义三角形顶点 TODO x
        var center = {x: app.screenWidth / 2, y: 5};
        // 定义三角形边长
        var length = 20;
        // 左端点
        var left = {x: center.x - length / 2, y: center.y + length / 2 * Math.sqrt(3)};
        // 右端点
        var right = {x: center.x + length / 2, y: center.y + length / 2 * Math.sqrt(3)};
        // 初始化context
        const context = wx.createCanvasContext('canvas-cursor');
        context.moveTo(center.x, center.y);
        context.lineTo(left.x, left.y);
        context.lineTo(right.x, right.y);
        // fill()填充而不是stroke()描边,于是省去手动回归原点,context.lineTo(center.x, center.y);
        context.setFillStyle('#48c23d');
        context.fill();
        context.draw();
    },
    bindscroll: function (e) {
        // deltaX 水平位置偏移位,每次滑动一次触发一次,所以需要记录从第一次触发滑动起,一共滑动了多少距离
        deltaX += e.detail.deltaX;
        // 数据绑定
        that.setData({
            value: Math.floor(- deltaX / 10 + minValue)
        });
        console.log(deltaX)
    }
});

源码下载:http://git.oschina.net/dotton/lendoo-wx,本文涉及代码存于/pages/member/donate文件夹中。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 210,978评论 6 490
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 89,954评论 2 384
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 156,623评论 0 345
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,324评论 1 282
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,390评论 5 384
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,741评论 1 289
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,892评论 3 405
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,655评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,104评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,451评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,569评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,254评论 4 328
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,834评论 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,725评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,950评论 1 264
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,260评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,446评论 2 348

推荐阅读更多精彩内容