uniapp小程序实现横屏手写签名(整理)

简介: uniapp小程序实现横屏手写签名(整理)

效果图:

方案一作为参考:

<template>
  <view class="signBox">
    <view class="topHint">请在下方空白书写区域内写出您的名字</view>
    <view class="btn">
      <view class="saveBtn" @click="save">确认</view>
      <view class="cancelBtn" @click="clear">清除</view>
    </view>
    <canvas class="canvas" disable-scroll="true" :style="{'width':width,'height':height}" canvas-id="designature"
      @touchstart="starts" @touchmove="moves" @touchend="end"></canvas>
  </view>
</template>
<script>
  export default {
    components: {},
    data() {
      return {
        dom: null,
        line: [],
        radius: 0,
        width: '0rpx',
        height: '0rpx',
      }
    },
    onLoad() {},
    computed: {},
    created() {
      uni.getSystemInfo({
        success: (res) => {
          this.width = res.windowWidth - 60 + 'rpx';
          this.height = res.windowHeight - 56 + 'rpx';
        }
      });
      this.dom = uni.createCanvasContext('designature', this);
    },
    onShow() {},
    methods: {
      end(e) {},
      distance(a, b) {
        let x = b.x - a.x;
        let y = b.y - a.y;
        return Math.sqrt(x * x + y * y);
      },
      // 开始
      starts(e) {
        this.line.push({
          points: [{
            time: new Date().getTime(),
            x: e.touches[0].x,
            y: e.touches[0].y,
            dis: 0
          }]
        })
        let currentPoint = {
          x: e.touches[0].x,
          y: e.touches[0].y
        }
        this.currentPoint = currentPoint
        this.drawer(this.line[this.line.length - 1])
      },
      // 滑动
      moves(e) {
        let point = {
          x: e.touches[0].x,
          y: e.touches[0].y
        }
        this.lastPoint = this.currentPoint,
          this.currentPoint = point
        this.line[this.line.length - 1].points.push({
          time: new Date().getTime(),
          x: e.touches[0].x,
          y: e.touches[0].y,
          dis: this.distance(this.currentPoint, this.lastPoint)
        })
        this.drawer(this.line[this.line.length - 1])
      },
      // 书写
      drawer(item) {
        let x1, x2, y1, y2, len, radius, r, cx, cy, t = 0.5,
          x, y;
        var time = 0;
        if (item.points.length > 2) {
          let lines = item.points[item.points.length - 3];
          let line = item.points[item.points.length - 2];
          let end = item.points[item.points.length - 1];
          x = line.x;
          y = line.y;
          x1 = lines.x;
          y1 = lines.y;
          x2 = end.x;
          y2 = end.y;
          var dis = 0;
          time = (line.time - lines.time) + (end.time - line.time)
          dis = line.dis + lines.dis + end.dis;
          var dom = this.dom;
          var or = Math.min(time / dis * this.linePressure + this.lineMin, this.lineMax);
          cx = (x - (Math.pow(1 - t, 2) * x1) - Math.pow(t, 2) * x2) / (2 * t * (1 - t))
          cy = (y - (Math.pow(1 - t, 2) * y1) - Math.pow(t, 2) * y2) / (2 * t * (1 - t))
          dom.setLineCap('round')
          dom.beginPath();
          dom.setStrokeStyle('black')
          dom.setLineWidth(5)
          dom.moveTo(x1, y1);
          dom.quadraticCurveTo(cx, cy, x2, y2);
          dom.stroke();
          dom.draw(true)
        }
      },
      // 清除
      clear() {
        this.dom.clearRect(0, 0, 1000, 1000)
        this.dom.draw()
      },
      // 保存图片
      save() {
        var t = this;
        uni.canvasToTempFilePath({
          canvasId: 'designature',
          fileType: 'png',
          quality: 1, //图片质量
          success: function(res) {
            t.$emit('getImg', res.tempFilePath)
            console.log(res.tempFilePath, '148')
            // uni.navigateBack({
            //     delta:1
            // })
          },
          fail(e) {
            console.log(e)
          }
        }, this)
      }
    }
  }
</script>
<style scoped lang="less">
  .signBox {
    position: relative;
    overflow: hidden;
    background-color: #F6F6F6;
    height: 100vh;
    width: 100vw;
    .canvas {
      width: 100% !important;
      background-color: #FFFFFF;
      position: absolute;
      z-index: 9999;
      left: 0rpx;
      top: 25rpx;
      // border: 1rpx solid #d6d6d6;
    }
    .topHint{
      width: 100%;
      height: 25rpx;
      line-height: 25rpx;
      font-size: 15rpx;
      color: #999999;
      text-align: center;
      background: ;
    }
    .btn {
      width: 100%;
      height: 60rpx;
      position: fixed;
      bottom: 0;
      // background-color: red;
      display: flex;
      align-items: center;
      justify-content: center;
      .saveBtn {
        width: 190rpx;
        height: 40rpx;
        line-height: 40rpx;
        background: linear-gradient(306deg, #4281EA 0%, #6363F2 100%);
        border-radius: 10rpx;
        text-align: center;
        font-size: 18rpx;
        color: #FFFFFF;
        margin-right: 10rpx;
      }
      .cancelBtn {
        width: 190rpx;
        height: 40rpx;
        line-height: 40rpx;
        background: #FFFFFF;
        border-radius: 10rpx;
        text-align: center;
        font-size: 18rpx;
        color: #222222;
        margin-left: 10rpx;
      }
    }
  }
</style>
json文件
{
  "path": "pages/clientDitch/sign",
  "style": {
    "navigationBarTitleText": "签字手签",
    "pageOrientation": "landscape", //控制横屏
    // "navigationStyle": "custom"
  }
}, 

方案二是demo-用这个:

<template>
  <view class="signBox column-me">
    <!-- 这个是自定义的title-可根据自己封装的title的作为调整 -->
    <status-bar title="电子签名" :bgColor="null"></status-bar>
    <view class="topHint">请绘制清晰可辨的签名并保存</view>
    <canvas class="canvas flex1" :canvas-id="cid" :id="cid" @touchstart="touchstart" @touchmove="touchmove"
      @touchend="touchend" :disable-scroll="true"></canvas>
    <view class="btn margin-top10 margin-bottom10">
      <view class="cancelBtn" @click="reWrite">重写</view>
      <view class="saveBtn margin-left30" @click="save">保存</view>
    </view>
  </view>
</template>
<script>
  export default {
    components: {},
    data() {
      return {
        line: [],
        radius: 0,
        taskId: '',
        //以下与签名有关参数
        dom: null,
        cid: 'canvas', //画布id
        Strokes: [],
        showCanvasDialog: false,
        canvasImg: '', //签名图片
      }
    },
    onLoad(e) {
      this.taskId = e.taskId
    },
    computed: {},
    mounted: function() {
      let that = this
      this.initCanvas()
    },
    methods: {
      initCanvas() {
        let that = this
        this.$nextTick(function() {
          this.dom = uni.createCanvasContext(this.cid, this);
          var query = wx.createSelectorQuery();
          query.select('#canvas').boundingClientRect();
          query.exec(function(res) {
            let widths = res[0].width
            let heights = res[0].height
            that.dom.rect(0, 0, widths, heights)
            that.dom.setFillStyle('#FFFFFF')
            that.dom.fill()
            that.dom.draw()
          })
        });
      },
      touchstart(e) {
        this.Strokes.push({
          imageData: null,
          style: {
            color: '#000000',
            lineWidth: '3',
          },
          points: [{
            x: e.touches[0].x,
            y: e.touches[0].y,
            type: e.type,
          }]
        })
        this.drawLine(this.Strokes[this.Strokes.length - 1], e.type);
      },
      touchmove(e) {
        this.Strokes[this.Strokes.length - 1].points.push({
          x: e.touches[0].x,
          y: e.touches[0].y,
          type: e.type,
        })
        this.drawLine(this.Strokes[this.Strokes.length - 1], e.type);
      },
      touchend(e) {
        if (this.Strokes[this.Strokes.length - 1].points.length < 2) { //当此路径只有一个点的时候
          this.Strokes.pop();
        }
      },
      drawLine(StrokesItem, type) {
        if (StrokesItem.points.length > 1) {
          this.dom.setLineCap('round')
          this.dom.setStrokeStyle(StrokesItem.style.color);
          this.dom.setLineWidth(StrokesItem.style.lineWidth);
          this.dom.moveTo(StrokesItem.points[StrokesItem.points.length - 2].x, StrokesItem.points[StrokesItem
            .points.length -
            2].y);
          this.dom.lineTo(StrokesItem.points[StrokesItem.points.length - 1].x, StrokesItem.points[StrokesItem
            .points.length -
            1].y);
          this.dom.stroke();
          this.dom.draw(true);
        }
      },
      //重写签名
      reWrite() {
        this.Strokes = [];
        this.dom.draw();
        this.initCanvas()
      },
      // 保存图片
      save() {
        let that = this
        uni.canvasToTempFilePath({
          canvasId: 'canvas',
          fileType: 'png',
          quality: 1, //图片质量
          success: function(res) {
            let imgs = [res.tempFilePath]
            that.$.upload_img(imgs, 0, res => {
              let imgUrl = res.data //签名图片
              let mediaUrl = that.$.get_data('mediaUrl') //采集图片
              if (that.$.isEmpty(mediaUrl)) {
                mediaUrl = ''
              }
              that.$.ajax("POST", "/customer/user/checkTask", {
                taskId: that.taskId,
                status: 1, //状态:1同意2拒绝  
                signImage: imgUrl,
                userVideo: mediaUrl,
              }, (res) => {
                if (res.code === 1000) {
                  that.$.ti_shi(res.message)
                  setTimeout(() => {
                    uni.$emit('signOk')
                    that.$.back()
                  }, 1000)
                } else {
                  that.$.ti_shi(res.message)
                }
              });
            })
          },
          fail(e) {
            console.log(e)
          }
        })
      }
    }
  }
</script>
<style scoped lang="less">
  .signBox {
    position: relative;
    overflow: hidden;
    background-color: #F6F6F6;
    height: 100vh;
    width: 100vw;
    .canvas {
      width: 100%;
      background: #FFFFFF;
    }
    .topHint {
      width: 100%;
      height: 60rpx;
      line-height: 60rpx;
      font-size: 28rpx;
      color: #6D7984;
      text-align: center;
      background: ;
    }
    .btn {
      width: 100%;
      height: 132rpx;
      display: flex;
      align-items: center;
      justify-content: center;
      .saveBtn {
        width: 300rpx;
        height: 88rpx;
        line-height: 88rpx;
        background: #215AA0;
        border-radius: 20rpx;
        text-align: center;
        font-size: 32rpx;
        color: #FFFFFF;
      }
      .cancelBtn {
        width: 298rpx;
        height: 86rpx;
        line-height: 86rpx;
        background: #FFFFFF;
        border-radius: 20rpx;
        text-align: center;
        font-size: 32rpx;
        color: #202233;
        border: 1px solid #BBC4CC;
      }
    }
  }
</style>
相关文章
|
6天前
|
开发框架 小程序 前端开发
圈子社交app前端+后端源码,uniapp社交兴趣圈子开发,框架php圈子小程序安装搭建
本文介绍了圈子社交APP的源码获取、分析与定制,PHP实现的圈子框架设计及代码编写,以及圈子小程序的安装搭建。涵盖环境配置、数据库设计、前后端开发与接口对接等内容,确保平台的安全性、性能和功能完整性。通过详细指导,帮助开发者快速搭建稳定可靠的圈子社交平台。
66 17
|
2天前
|
移动开发 小程序 前端开发
超详细攻略!uniapp陪玩系统,打包陪玩小程序、H5需要注意什么?
陪玩系统的打包过程涵盖APP、小程序和H5平台。APP打包需使用uni-app开发工具,配置项目信息并选择云打包;小程序打包需在微信公众平台注册账号并提交审核;H5打包则直接通过uni-app生成文件并上传至服务器。各平台需注意权限配置、代码规范及充分测试,确保应用稳定性和兼容性。
|
2月前
|
人工智能 小程序 搜索推荐
uni app下开发AI运动小程序解决方案
本文介绍了在小程序中实现AI运动识别的解决方案。该方案依托于UNI平台,通过高效便捷的插件形式,实现包括相机抽帧控制、人体识别、姿态识别等在内的多项功能,无需依赖后台服务器,大幅提高识别效率和用户体验。方案内置多种运动模式,支持自定义扩展,适用于AI健身、云上赛事、AI体测等多场景,适合新开发和存量改造项目。
|
2月前
|
小程序 前端开发 JavaScript
在线课堂+工具组件小程序uniapp移动端源码
在线课堂+工具组件小程序uniapp移动端源码
43 0
在线课堂+工具组件小程序uniapp移动端源码
|
3月前
|
移动开发 小程序 数据可视化
基于npm CLI脚手架的uniapp项目创建、运行与打包全攻略(微信小程序、H5、APP全覆盖)
基于npm CLI脚手架的uniapp项目创建、运行与打包全攻略(微信小程序、H5、APP全覆盖)
409 3
|
3月前
|
移动开发 小程序 数据可视化
一招学会DIY官网可视化设计支持导出微擎、UNIAPP、H5、微信小程序源码
一招学会DIY官网可视化设计支持导出微擎、UNIAPP、H5、微信小程序源码
59 2
|
5月前
|
小程序 前端开发 Java
SpringBoot+uniapp+uview打造H5+小程序+APP入门学习的聊天小项目
JavaDog Chat v1.0.0 是一款基于 SpringBoot、MybatisPlus 和 uniapp 的简易聊天软件,兼容 H5、小程序和 APP,提供丰富的注释和简洁代码,适合初学者。主要功能包括登录注册、消息发送、好友管理及群组交流。
123 0
SpringBoot+uniapp+uview打造H5+小程序+APP入门学习的聊天小项目
|
3月前
|
小程序 数据可视化 API
低代码可视化-uniapp商城首页小程序-代码生成器
低代码可视化-uniapp商城首页小程序-代码生成器
36 0
|
3月前
|
存储 移动开发 小程序
uniapp富文本editor输入二次扩展兼容微信小程序
uniapp富文本editor输入二次扩展兼容微信小程序
103 0
|
3月前
|
小程序
uniapp实现微信小程序隐私协议组件封装
uniapp实现微信小程序隐私协议组件封装
60 0