hml 2 лет назад
Родитель
Сommit
68cfdbf23e

Разница между файлами не показана из-за своего большого размера
+ 0 - 8
purchase_ao/public/cdn/tracking.min.js


+ 0 - 1
purchase_ao/public/index.html

@@ -7,7 +7,6 @@
   <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1, user-scalable=no">
   <link rel="icon" href="<%= BASE_URL %>favicon.ico">
   <script src="cdn/jsencrypt.min.js"></script>
-  <script src="cdn/tracking.min.js"></script>
   <title><%= webpackConfig.name %></title>
 </head>
 

Разница между файлами не показана из-за своего большого размера
+ 8 - 0
purchase_ao/src/api/face-min.js


Разница между файлами не показана из-за своего большого размера
+ 8 - 0
purchase_ao/src/api/tracking-min.js


+ 2 - 22
purchase_ao/src/views/login/index.vue

@@ -14,8 +14,8 @@
           LOGIN
         </div>
         <div class="tabs">
-          <div class="tabLeft" :class="tabNum==0?'active':''" @click="tabClick(0)">账号登录</div>
-          <!-- <div class="tabRight" :class="tabNum==1?'active':''" @click="tabClick(1)">面部登录</div> -->
+          <div class="tabLeft" :class="tabNum==0?'active':''" @click="tabNum = 0">账号登录</div>
+          <div class="tabRight" :class="tabNum==1?'active':''" @click="tabNum = 1">面部登录</div>
         </div>
         <div v-if="tabNum == 1">
           <renLian />
@@ -57,7 +57,6 @@
         </div>
       </div>
     </div>
-
     <el-dialog title="获取验证码" width="450px" :visible.sync="codeShow" :close-on-press-escape="false" :close-on-click-modal="false">
       <div class="getCode">
         <input type="text" v-model="vcode" placeholder="验证码" maxlength="6" />
@@ -130,14 +129,6 @@ export default {
       vcode: "",
     };
   },
-  watch: {
-    // $route: {
-    //   handler: function(route) {
-    //     this.redirect = route.query && route.query.redirect
-    //   },
-    //   immediate: true
-    // }
-  },
   created () {
     puk().then((res) => {
       const encrypt = new JSEncrypt();
@@ -149,17 +140,6 @@ export default {
     }
   },
   methods: {
-    tabClick (index) {
-      if (index == 1) {
-        this.$message({
-          message: '功能升级中',
-          type: 'warning'
-        });
-      } else {
-        this.tabNum = index
-      }
-
-    },
     getCode () {
       this.imgurl = `${process.env.VUE_APP_BASE_API}${urls.getCode}?userName=${this.loginForm.username
         }&refrech=${this.imgCodeRefrech}&times=${Date.now()}`;

+ 76 - 206
purchase_ao/src/views/login/renlian.vue

@@ -1,244 +1,114 @@
 <template>
-  <el-form class="login-form" status-icon :rules="loginRules" ref="loginForm" :model="loginForm" label-width="0">
-    <el-form-item prop="username" style="padding-left: 44px">
-      <div class="video" style="z-index: 1">
-        <video id="video" width="274" height="265" preload autoplay loop muted style="z-index: 1;"></video>
-      </div>
-      <!-- <div class="renlian" style="z-index: 999">
+  <div class="renlianBox">
+    <div class="videoBox">
+      <video id="video" width="290" height="300" preload autoplay loop muted></video>
+      <div class="renlian" style="z-index: 999">
         <div class="box">
           <div class="line"></div>
           <div class="bottom"></div>
         </div>
-      </div> -->
+      </div>
       <canvas id="canvas" width="500" height="500" style="position: fixed;top: -10000px"></canvas>
-    </el-form-item>
-    <!--    <a href="#" style="color: red;font-weight: bold">提示:如果未录入人脸请联系管理员进行录入</a>-->
-  </el-form>
+    </div>
+
+  </div>
 </template>
 
 <script>
-import { mapGetters } from "vuex";
-// import { info } from "@/api/system/tenant";
-// import * as user from "@/api/user";
-// import { getTopUrl } from "@/util/util";
+import tracking from '@/api/tracking-min'
+import '@/api/face-min'
 export default {
   name: "renlian",
   data () {
     return {
-      tenantMode: false,
-      loginForm: {
-        //租户ID
-        tenantId: "000000",
-        //用户名
-        username: "",
-        //密码
-        password: "",
-        //账号类型
-        type: "account",
-        //验证码的值
-        code: "",
-        //验证码的索引
-        key: "",
-        //预加载白色背景
-        image: "",
-      },
-      loginRules: {
-        tenantId: [
-          { required: false, message: "请输入租户ID", trigger: "blur" }
-        ],
-        username: [
-          { required: true, message: "请输入用户名", trigger: "blur" }
-        ],
-        password: [
-          { required: true, message: "请输入密码", trigger: "blur" },
-          { min: 1, message: "密码长度最少为6位", trigger: "blur" }
-        ]
-      },
-      passwordType: "password"
+      video: null,
+      canvas: null,
+      uploadLock: true
     };
   },
-  created () {
-
-  },
   mounted () {
-    let that = this
-    setTimeout(function () {
-      that.openCamera()
-      that.getTenant();
-      that.refreshCode();
-    }, 1000)
+    this.openCamera()
   },
-  destroyed () {
-    // console.log(1111111111)
-    // // 停止侦测
-    this.trackerTask.stop()
-    // // 关闭摄像头
-    // window.tracking.closeCamera()
-  },
-  // computed: {
-  //   ...mapGetters(["tagWel"])
-  // },
-  props: [],
   methods: {
-    closeshexiangtou () {
-      this.trackerTask.stop()
-    },
-    refreshCode () {
-      user.getCaptcha().then(res => {
-        const data = res.data;
-        this.loginForm.key = data.key;
-        this.loginForm.image = data.image;
-      })
-    },
-    showPassword () {
-      this.passwordType === ""
-        ? (this.passwordType = "password")
-        : (this.passwordType = "");
-    },
-    handleLogin () {
-      this.$refs.loginForm.validate(valid => {
-        if (valid) {
-          const loading = this.$loading({
-            lock: true,
-            text: '登录中,请稍后。。。',
-            spinner: "el-icon-loading"
-          });
-          this.$store.dispatch("LoginByUsername", this.loginForm).then(() => {
-            // this.$router.push({ path: this.tagWel.value });
-            loading.close();
-          }).catch(() => {
-            loading.close();
-            this.refreshCode();
-          });
-        }
-      });
-    },
-    getTenant () {
-      let domain = getTopUrl();
-      info(domain).then(res => {
-        const data = res.data;
-        if (data.success && data.data.tenantId) {
-          this.tenantMode = false;
-          this.loginForm.tenantId = data.data.tenantId;
-          this.$parent.$refs.login.style.backgroundImage = `url(${data.data.backgroundUrl})`;
-        }
-      })
-    },
-
-
+    // 初始化设置
     openCamera () {
-
-      console.log(new tracking)
-      
-      var video = document.getElementById('video');
-      var canvas = document.getElementById('canvas');
-      var context = canvas.getContext('2d');
-
-
-      var tracker = new tracking.ObjectTracker('face');  // 引入第三方 库
-      tracker.setInitialScale(1);
+      this.video = document.getElementById('video');
+      this.canvas = document.getElementById('canvas');
+      let canvas = document.getElementById('canvas');
+      let context = canvas.getContext('2d');
+
+      // 固定写法
+      let tracker = new window.tracking.ObjectTracker('face');
+      tracker.setInitialScale(4);
       tracker.setStepSize(2);
       tracker.setEdgesDensity(0.1);
-
-      this.trackerTask = tracking.track('#video', tracker, { camera: true });
-      //-------  指定 canvas 的宽高 ,auto 自动播放
-      let constraints = {
-        video: { width: 300, height: 300 },
-        audio: true
-      };
-
-      let promise = navigator.mediaDevices.getUserMedia(constraints);   // h5 新的API
-
-      promise.then(function (MediaStream) {
-        video.srcObject = MediaStream;
-        video.play();
-      }).catch(function (PermissionDeniedError) {
-        console.log(PermissionDeniedError);
-      })
-      //--------------
-      let that = this;
-      that.tracker_fun(tracker, video, context, canvas); //open 摄像头,执行tracker_fun( 传入参数 )
-
-    },
-    tracker_fun (tracker, video, context, canvas) {
+      window.tracking.track('#video', tracker, {
+        camera: true
+      });
       let that = this;
-      let set_clear;
-      set_clear = setInterval(function () { // 每秒 检测人脸 结果
-        tracker.on('track', function (event) {     // 视频流
-          if (!that.first) {     // if  --- > else  检测到人脸 success() =>{}
-            event.data.forEach(function (rect) {
-              if (rect.x) {
-                that.first = rect.x;
-                // video.pause();    // success  将暂停 video
-                context.drawImage(video, 0, 0, 500, 500);   // 绘制到 canvas
-                context.font = '11px Helvetica';
-                context.fillText("", 100, 40);
-                context.strokeStyle = '#a64ceb';
-                context.strokeRect(rect.x, rect.y, rect.width, rect.height);
-                var data_url = canvas.toDataURL('image/png'); //base64 request
-                that.$store.dispatch("LoginByrenlian", data_url).then((res) => {
-                  if (res.error_code == 400) {
-                    setTimeout(function () {
-                      that.first = null;
-                    }, 3000)
-                    this.$message.error(res.error_msg);
-                  } else {
-                    const loading = that.$loading({
-                      lock: true,
-                      text: '登录中,请稍后。。。',
-                      spinner: "el-icon-loading"
-                    });
-                    clearInterval(set_clear)
-                    setTimeout(function () {
-                      loading.close();
-                      location.reload();
-                      // that.$router.resolve({path: that.tagWel.value});
-                    }, 3000)
-                  }
-
-
-                }).catch(() => {
-                  setTimeout(function () {
-                    that.first = null;
-                  }, 3000)
-                });
-                // return;
-              }
-            });
-          }
+      tracker.on('track', (event) => {
+        // 检测出人脸 绘画人脸位置
+        context.clearRect(0, 0, canvas.width, canvas.height);
+        event.data.forEach((rect) => {
+          context.strokeStyle = '#0764B7';
+          context.strokeRect(rect.x, rect.y, rect.width, rect.height);
+          // 上传图片
+          that.uploadLock && that.screenshotAndUpload();
         });
-        // clearTimeout(set_clear)
-        // console.log('-------')
-      }, 3000)
-
-    }
+      });
+    },
+    // 上传图片
+    screenshotAndUpload () {
+      // 上锁避免重复发送请求
+      this.uploadLock = false;
+      // 绘制当前帧图片转换为base64格式
+      let canvas = this.canvas;
+      let video = this.video;
+      let ctx = canvas.getContext('2d');
+      ctx.clearRect(0, 0, canvas.width, canvas.height);
+      ctx.drawImage(video, 0, 0, canvas.width, canvas.height);
+      let base64Img = canvas.toDataURL('image/jpeg');
+      // 使用 base64Img 请求接口即可
+      console.log('base64Img:', base64Img)
+      // 三秒后重新检测人脸
+      // setTimeout(()=>{
+      //   this.uploadLock = true
+      // },3000)
+    },
   }
-};
+
+}
 </script>
 
-<style>
+<style lang="scss">
+.renlianBox {
+  width: 100%;
+  height: 100%;
+  .videoBox {
+    width: 290px;
+    height: 290px;
+    margin: 30px auto;
+    position: relative;
+  }
+}
 .renlian {
-  position: relative;
-  height: 90px;
-  width: 274px;
+  position: absolute;
+  left: 0;
+  top: 0;
   background-size: 100% 100%;
 }
 .video {
   position: relative;
-  height: 90px;
-  width: 274px;
+  // height: 90px;
+  // width: 274px;
   background-size: 100% 100%;
 }
 
 .renlian .box {
-  width: 30vw;
-  height: 30vw;
-  max-height: 30vh;
-  max-width: 30vh;
-  position: relative;
-  left: 50%;
-  top: 50%;
-  transform: translate(-50%, -50%);
+  width: 300px;
+  height: 300px;
+  // max-height: 30vh;
+  // max-width: 30vh;
   overflow: hidden;
   border: 0.1rem solid rgba(3, 169, 244, 0.2);
 }