前端代码

<template>
    <view class="takePhotoBeforView">
    <!--  活体提示 -->
    <div class="face-tips" v-if="step === 1">
      <div class="title">多动作活体验证</div>
      <div class="img">
        <div class="img-bg"></div>
        <div class="img-line"></div>
        <img src="../static/images/face/face_tips4.png" alt="">
      </div>
      <div class="content">
        <p>请根据动作提示,按动作顺序</p>
        <p>录制 <span class="red-text">1-3秒动作</span> 视频</p>
        <p>请正对手机摄像头,确保面部光源充足</p>
      </div>
      <div class="btn" @click="nextStep">开始验证</div>
    </div>
    <!--  活体视频录制 -->
    <div class="face-box" v-show="step === 2">
      <div class="title">多动作活体验证</div>
      <div class="action-box">
        <div class="action-box-p" v-if="actionVal.length > 1"><span class="txt">先{{ actionList[actionVal[0]].text }}、后{{ actionList[actionVal[1]].text }}</span></div>
        <div class="action-box-list">
          <div class="item" v-for="(item, index) in actionVal" :key="index">
            <img :src="actionList[item].icon" alt="">
          </div>
        </div>
      </div>
      <div class="face-success">
      <div class="face-success-imgs">
        <video  ref="videoRef"  style="object-fit:fill" :show-center-play-btn="false" muted preload></video>
      </div>
      </div>
      <div class="btn" @click="saveVideo">{{ recordStatus === 0 ? '开始录制' : '结束录制(' + recordCount +  's)' }}</div>
    </div>
    <!--  成功 -->
    <div class="face-tips" v-if="step === 3">
      <div class="title">多动作活体验证</div>
      <div class="face-success">
        <div class="face-success-img">
          <img :src="videoImg" alt="">
        </div>
      </div>
      <div class="btn" @click="successBack">确定</div>
    </div>
    <!--   失败 -->
    <div class="face-tips" v-if="step === 4">
      <div class="title">多动作活体验证</div>
      <div class="error-wrap">
        <div class="error-tips">{{ errorTips }}</div>
        <div class="error-p">请确保本人操作,并避免以下问题</div>
        <ul class="error-list">
          <li>
            <img src="../static/images/face/face_error1.jpg" alt="">
            <p>没有对焦</p>
          </li>
          <li>
            <img src="../static/images/face/face_error2.jpg" alt="">
            <p>遮挡人脸</p>
          </li>
          <li>
            <img src="../static/images/face/face_error3.jpg" alt="">
            <p>光照不均</p>
          </li>
          <li>
            <img src="../static/images/face/face_error4.jpg" alt="">
            <p>手机晃动</p>
          </li>
        </ul>
      </div>
      <div class="btn" @click="again">重新检测</div>
    </div>
  </view>
  
</template

js代码

export default {
  name: "VideoVerify",
  data() {
    return {
      show: false,
      step: 1, // 1 提示 2 录制 3 成功 4 失败
      name: '',
      idnum: '',
      actionList: {
        1: {
          text: '眨眼',
          icon: require('../static/images/face/face_tips1.png')
        },
        2: {
          text: '张嘴',
          icon: require('../static/images/face/face_tips2.png')
        },
        5: {
          text: '转头',
          icon: require('../static/images/face/face_tips3.png')
        },
      },
      actionVal: [],
      mediaRecorder: null,
      MediaStreamTrack: null,
      recordStatus: 0, // 0 未开始 1 正在录制
      recordCount: 5,
      recordedBlobs: [],

      errorTips: '',
      videoImg: '' ,// 人像识别图片
      canvas: null,
    }
  },
  mounted() {},
  beforeDestroy() {
    this.MediaStreamTrack && this.MediaStreamTrack.stop()
    this.countTimer && clearTimeout(this.countTimer)
  },
  methods: {
    showChoose(data) {
      this.again()
      this.show = true
      this.name = data.name
      this.idnum = data.idnum
    },
    cancelChoose() {
      this.show = false
      this.name = ''
      this.idnum = ''
    },
    // 下一步
    nextStep() {
       //this.getCamera()
    this.getActionData()
    },
    // 调用摄像头 开始录制
    getCamera () {
      let that = this
      // 注意本例需要在HTTPS协议网站中运行,新版本Chrome中getUserMedia接口在http下不再支持。
      let constraints = {
        audio: false,
        video: {
          facingMode: 'user' // 优先调前置摄像头
        }
      }

      // 老的浏览器可能根本没有实现 mediaDevices,所以我们可以先设置一个空的对象
      if (navigator.mediaDevices === undefined) {
        navigator.mediaDevices = {}
      }

      // 一些浏览器部分支持 mediaDevices。我们不能直接给对象设置 getUserMedia
      // 因为这样可能会覆盖已有的属性。这里我们只会在没有getUserMedia属性的时候添加它。
      if (navigator.mediaDevices.getUserMedia === undefined) {
        navigator.mediaDevices.getUserMedia = function (constraints) {
          // 首先,如果有getUserMedia的话,就获得它
          //   var getUserMedia = navigator.getUserMedia || navigator.webkitGetUserMedia || navigator.mozGetUserMedia
          var getUserMedia = navigator.getUserMedia ||
              navigator.webkitGetUserMedia ||
              navigator.mozGetUserMedia

          // 一些浏览器根本没实现它 - 那么就返回一个error到promise的reject来保持一个统一的接口
          if (!getUserMedia) {
              that.$modal.confirm('摄像头开启失败,请检查摄像头是否授权或是否可用!').then(() => {
                that.cancelChoose()
              })
            return Promise.reject(new Error('getUserMedia is not implemented in this browser'))
          }

          // 否则,为老的navigator.getUserMedia方法包裹一个Promise
          return new Promise(function (resolve, reject) {
            getUserMedia.call(navigator, constraints, resolve, reject)
          })
        }
      }

      navigator.mediaDevices.getUserMedia(constraints)
          .then((stream) => {
            that.MediaStreamTrack = typeof stream.stop === 'function' ? stream : stream.getTracks()[0]
            console.log(stream)
            console.log(that.MediaStreamTrack)
            let winURL = window.URL || window.webkitURL
            const video = document.querySelector("video");
            if ('srcObject' in video) {
              video.srcObject = stream
            } else {
             video.src = winURL.createObjectURL(stream)
            }
            video.onloadedmetadata = e => {
              // 播放视频
              video.play()
            }

            let options = {
              videoBitsPerSecond: 2500000
            }
            that.mediaRecorder = new MediaRecorder(stream, options)

            this.getActionData()
          })
          .catch((err) => {
            that.$modal.confirm('摄像头开启失败,请检查摄像头是否授权或是否可用!').then(() => {
              that.cancelChoose()
            })
          })
    },
    // 生成随机动作组
    getActionData() {
      this.actionVal = [1, 5]
      // 显示录制框
      this.step = 2
      this.recordStatus = 0
    },
    // 录制倒计时
    countDown() {
      let that = this
      let sendTime = Math.round(+new Date() / 1000)

      return function walk() {
        that.countTimer = setTimeout(function() {
          that.countTimer && clearTimeout(that.countTimer)
          let diff = sendTime + 5 - Math.round(+new Date() / 1000)
          if (diff > 0) {
            that.recordCount = diff
            walk()
          } else {
            console.log('倒计时结束')
            that.recordStatus = 1
            that.saveVideo()
          }
        }, 1000)
      }
    },
    // 保存录制视频
    saveVideo() {
        this.canvas = document.createElement("canvas");
        let context = this.canvas.getContext("2d");
        const video = document.querySelector("video");
        this.canvas.width = 720;
        this.canvas.height = 864;
        context.drawImage(video, 0, 0, this.canvas.width,this.canvas.height);
        let canvas = this.canvas.toDataURL("image/png");
        this.videoImg=canvas;
        
      if (this.recordStatus === 1) {
        this.countTimer && clearTimeout(this.countTimer)
        //当录制的数据可用时
        this.mediaRecorder.ondataavailable = (e) => {
          console.log(e)
          if (e.data && e.data.size > 0) {
            this.recordedBlobs.push(e.data)
          }
        }
        this.mediaRecorder.stop()

        setTimeout(() => {
          var blob = new Blob(this.recordedBlobs, {type: 'video/mp4'})
          console.log(blob)
          this.isAlreadyRecord = false
          this.MediaStreamTrack && this.MediaStreamTrack.stop()

          var reader = new FileReader();
          reader.readAsDataURL(blob, 'utf-8')
          reader.onload = () => {
            console.log(reader.result); // base64格式
            let result = reader.result.replace(/^data:video\/\w+;base64,/, '')
            this.faceReco(result)
          }
        }, 1000)


      } else {
        this.count = 8
        this.recordStatus = 1
        this.mediaRecorder.start(5000)
        this.countDown()()
      }
    },
    // 视频识别活检比对
    faceReco(video) {
        console.log("视频识别活检比对中")
        console.log(video)
      let that = this
      that.$modal.loading("验证中...")
      
      setTimeout(() => {
        let random_boolean = Math.random() < 0.5;
        this.$modal.closeLoading()
        this.step = 3
        // 随机返回结果
       /* if (random_boolean) {
          this.$modal.closeLoading()
          this.step = 3
          //this.videoImg = video
          } else {
          this.$modal.closeLoading()
          this.step = 4
          this.errorTips = '人像不完整'
        } */
        this.recordedBlobs = []
        this.recordStatus = 0
        this.MediaStreamTrack && this.MediaStreamTrack.stop()
        this.countTimer && clearTimeout(this.countTimer)
      }, 2000)

    },
    // 人脸失败、重试
    again() {
      this.step = 1
      this.actionVal = []
      this.errorTips = ''
      this.videoImg = ''
    },
    // 人脸成功返回
    successBack() {
      this.$emit('on-success', {
        photo: this.videoImg
      })
      this.$emit("openTakePhoto")
      this.cancelChoose()
    }
  }
}

样式

<style scoped lang="less">
.face-popup {
  top: 0;
  left: 0;
  width: 100%;
  height: 100%;
  background-color: #fff;
  transform: none;
  box-sizing: border-box;
}
.face-box,
.face-tips {
  display: flex;
  flex-direction: column;
  height: 100%;
  text-align: center;
  box-sizing: border-box;
}
.face-tips {
  padding: 30px;
}
.face-box {
  padding: 20px 0;
}
.title {
  text-align: center;
  line-height: 1.2;
  font-size: 20px;
  color: #333;
  font-weight: 600;
}
.img {
  position: relative;
  margin: 7vh auto;
  width: 260px;
  height: 260px;
}
@keyframes bgmove {
  from {
    height: 0;
  }
  to {
    height: 225px;
  }
}
.img-bg,
.img-line {
  position: absolute;
  top: 17px;
  left: 17px;
  width: 225px;
  overflow: hidden;
}
.img-bg {
  animation: bgmove 4s linear infinite;
  animation-direction: alternate;
  z-index: 1;
}
.img-bg:before {
  content: '';
  position: absolute;
  top: 0;
  left: 0;
  width: 225px;
  height: 225px;
  background: linear-gradient(0deg, transparent 0%, rgba(104, 24, 133, .3) 50%, transparent 100%);
  border-radius: 50%;
}
.img-line {
  z-index: 6;
  height: 225px;
  border-radius: 50%;
}
.img-line:before {
  content: '';
  position: absolute;
  top: 0;
  left: 0;
  width: 100%;
  height: 2px;
  border-bottom: 2px solid #007aff;
  animation: bgmove 4s linear infinite;
  animation-direction: alternate;
}
.img img {
  position: relative;
  width: 100%;
  height: 100%;
  z-index: 2;
}
.content {
  flex: 1;
  min-height: 0;
  text-align: center;
  font-size: 15px;
  line-height: 1.8;
  color: #999;
  font-weight: 300;
}
.red-text {
  color: #f00;
}
.btn {
  height: 45px;
  line-height: 45px;
  text-align: center;
  font-size: 18px;
  color: #fff;
  background-color: #007aff;
  border-radius: 45px;
}
.face-success {
  //flex: 1;
  min-height: 0;
  display: flex;
  align-items: center;
  justify-content: center;
}
.face-success-imgs {
  position: relative;
  width: 250px;
  height: 250px;
  //padding: 20px;
  border-radius: 50%;
  overflow: hidden;
}
.face-success-img {
  position: relative;
  width: 225px;
  height: 225px;
  padding: 20px;
}
.face-success-img:before {
  content: '';
  position: absolute;
  top: 0;
  left: 0;
  width: 100%;
  height: 100%;
  z-index: 2;
  background: url('') center center / 100% 100% no-repeat;
}
.face-success-img img {
  width: 100%;
  height: 100%;
  border-radius: 50%;
  overflow: hidden;
}
.error-wrap {
  flex: 1;
  min-height: 0;
  display: flex;
  align-items: center;
  justify-content: center;
  flex-direction: column;
}
.error-tips {
  margin: 25px 0;
  line-height: 1.4;
  font-size: 20px;
  color: #f54d55;
  letter-spacing: 2px;
  width: 100%;
}
.error-p {
  line-height: 1.6;
  font-size: 16px;
  color: #d05b63;
  letter-spacing: 2px;
  width: 100%;
}
.error-list {
  display: flex;
  flex-wrap: wrap;
  width: 100%;
  li {
    padding: 20px;
    width: 50%;
    box-sizing: border-box;
  }
  img {
    width: 75px;
    height: 75px;
    margin: 0 auto 8px;
  }
  p {
    line-height: 1.4;
    font-size: 14px;
    color: #d05b63;
  }
}
.action-box {
  margin-top: 20px;
  padding: 10px 60px;
  margin-bottom: 20px;
  background-color: #ededed;
  &-p {
    margin-bottom: 10px;
    font-size: 14px;
    overflow: hidden;
    color: #e84968;
    .text {
      position: relative;
      &:before,
      &:after {
        content: '';
        position: absolute;
        top: 50%;
        margin-top: -8px;
        width: 170px;
        height: 170px;
        background: url('') center center / contain no-repeat;
      }
      &:before {
        right: calc(100% + 25rpx);
      }
      &:after {
        left: calc(100% + 25rpx)
      }
    }
  }
  &-list {
    display: flex;
    justify-content: center;
    .item {
      width: 60px;
      height: 60px;
      margin: 0 20px;
      img {
        display: block;
        height: 100%;
        width: auto;
      }
    }
  }
}
.face-box {
  video {
    flex: 1;
    min-height: 0;
    object-fit: fill;
  }
  .btn {
    margin: 20px 20px 0;
  }
}
.takePhotoBeforView {
      width: 100vw;
      height: 100vh;
      position: fixed;
      top:0;
      left: 0;
      display: flex;
      justify-content: center;
      align-items: center;
      z-index: 9999;
      background-color: #FFFFFF;
  }
  uni-video {
      width: 100%;
      height: 100%;
  }
</style>
 

参考地址:作者结合GitHub - reknown/videoPersonVerify进行优化改造

注意事项

对照后台算法,可以结合上传video文件进行活体检测,或者是结合进行截图对指定动作图片进行算法核验,具体实现看个人操作,具体还是要对比后台活体检测算法进行实现

效果展示

 

 

Logo

为开发者提供学习成长、分享交流、生态实践、资源工具等服务,帮助开发者快速成长。

更多推荐