faceIdentify.vue 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328
  1. <template>
  2. <view class="page-content">
  3. <web-view :src="webSrc" />
  4. </view>
  5. </template>
  6. <script>
  7. export default {
  8. name: "index",
  9. components: {},
  10. data() {
  11. return {
  12. tipsText: "", // 错误文案提示
  13. tempImg: "", // 本地图片路径
  14. cameraEngine: null, // 相机引擎
  15. devicePosition: true, // 摄像头朝向
  16. isAuthCamera: true, // 是否拥有相机权限
  17. webSrc: "",
  18. };
  19. },
  20. onLoad(options) {
  21. this.webSrc = options.src;
  22. // this.initData();
  23. },
  24. methods: {
  25. // 初始化相机引擎
  26. initData() {
  27. // #ifdef MP-WEIXIN
  28. // 1、初始化人脸识别
  29. wx.initFaceDetect();
  30. // 2、创建 camera 上下文 CameraContext 对象
  31. this.cameraEngine = wx.createCameraContext();
  32. // 3、获取 Camera 实时帧数据
  33. const listener = this.cameraEngine.onCameraFrame((frame) => {
  34. if (this.tempImg) {
  35. return;
  36. }
  37. // 4、人脸识别,使用前需要通过 wx.initFaceDetect 进行一次初始化,推荐使用相机接口返回的帧数据
  38. wx.faceDetect({
  39. frameBuffer: frame.data,
  40. width: frame.width,
  41. height: frame.height,
  42. enablePoint: true,
  43. enableConf: true,
  44. enableAngle: true,
  45. enableMultiFace: true,
  46. success: (faceData) => {
  47. let face = faceData.faceInfo[0];
  48. if (faceData.x == -1 || faceData.y == -1) {
  49. this.tipsText = "检测不到人";
  50. }
  51. if (faceData.faceInfo.length > 1) {
  52. this.tipsText = "请保证只有一个人";
  53. } else {
  54. const { pitch, roll, yaw } = face.angleArray;
  55. const standard = 0.5;
  56. if (
  57. Math.abs(pitch) >= standard ||
  58. Math.abs(roll) >= standard ||
  59. Math.abs(yaw) >= standard
  60. ) {
  61. this.tipsText = "请平视摄像头";
  62. } else if (
  63. face.confArray.global <= 0.8 ||
  64. face.confArray.leftEye <= 0.8 ||
  65. face.confArray.mouth <= 0.8 ||
  66. face.confArray.nose <= 0.8 ||
  67. face.confArray.rightEye <= 0.8
  68. ) {
  69. this.tipsText = "请勿遮挡五官";
  70. } else {
  71. this.tipsText = "请拍照";
  72. // 这里可以写自己的逻辑了
  73. }
  74. }
  75. },
  76. fail: (err) => {
  77. if (err.x == -1 || err.y == -1) {
  78. this.tipsText = "检测不到人";
  79. } else {
  80. this.tipsText = err.errMsg || "网络错误,请退出页面重试";
  81. }
  82. },
  83. });
  84. });
  85. // 5、开始监听帧数据
  86. listener.start();
  87. // #endif
  88. },
  89. // 切换设备镜头
  90. handleChangeCameraClick() {
  91. this.devicePosition = !this.devicePosition;
  92. },
  93. // 图片上传
  94. handleChooseImage() {
  95. uni.chooseImage({
  96. count: 1,
  97. sizeType: ["original", "compressed"],
  98. sourceType: ["album"],
  99. success: (res) => {
  100. if (res.errMsg === "chooseImage:ok") {
  101. uni.showLoading({
  102. title: "照片上传中...",
  103. });
  104. console.log("===========:", res.tempFilePaths[0]);
  105. this.handleOkClick();
  106. }
  107. },
  108. fail: (res) => {},
  109. });
  110. },
  111. // 拍照点击
  112. handleTakePhotoClick() {
  113. if (this.tipsText != "" && this.tipsText != "请拍照") {
  114. return;
  115. }
  116. uni.getSetting({
  117. success: (res) => {
  118. if (!res.authSetting["scope.camera"]) {
  119. this.isAuthCamera = false;
  120. uni.openSetting({
  121. success: (res) => {
  122. if (res.authSetting["scope.camera"]) {
  123. this.isAuthCamera = true;
  124. }
  125. },
  126. });
  127. }
  128. },
  129. });
  130. this.cameraEngine.takePhoto({
  131. quality: "high",
  132. success: ({ tempImagePath }) => {
  133. this.tempImg = tempImagePath;
  134. console.log("=======tempImg:", this.tempImg);
  135. },
  136. });
  137. },
  138. // 点击确定上传
  139. handleOkClick() {
  140. uni.showLoading({
  141. mask: true,
  142. title: "校验中...",
  143. });
  144. // 更新人脸识别图片请求协议:传七牛图片链接
  145. setTimeout(function () {
  146. uni.hideLoading();
  147. uni.showToast({
  148. icon: "none",
  149. title: "假装图片上传成功",
  150. duration: 2000,
  151. });
  152. }, 3000);
  153. },
  154. // 点击 - 取消上传
  155. handleCancelClick() {
  156. this.tempImg = "";
  157. },
  158. // 点击 - 人脸安全保障按钮
  159. handleJumpSecurityClick() {
  160. uni.showToast({
  161. icon: "none",
  162. title: "假装跳转人脸安全保障",
  163. duration: 2000,
  164. });
  165. },
  166. },
  167. };
  168. </script>
  169. <style lang="scss">
  170. .page-content {
  171. width: 100%;
  172. height: 100%;
  173. .containerV {
  174. width: 100%;
  175. height: 100%;
  176. .headerV {
  177. .top-tips1 {
  178. margin-top: 60rpx;
  179. color: #1c1c1c;
  180. font-size: 36rpx;
  181. text-align: center;
  182. }
  183. .top-tips2 {
  184. margin-top: 20rpx;
  185. color: #00aaff;
  186. font-size: 28rpx;
  187. text-align: center;
  188. }
  189. }
  190. .contentV {
  191. position: relative;
  192. display: flex;
  193. flex-direction: column;
  194. align-items: center;
  195. justify-content: center;
  196. height: 661rpx;
  197. margin-top: 30rpx;
  198. .tipV {
  199. bottom: 30rpx;
  200. position: absolute;
  201. line-height: 90rpx;
  202. padding-left: 24rpx;
  203. padding-right: 24rpx;
  204. max-width: calc(100vw - 50rpx * 2);
  205. text-align: center;
  206. font-size: 30rpx;
  207. background: #000000;
  208. opacity: 0.75;
  209. color: #ffffff;
  210. border-radius: 16rpx;
  211. overflow: hidden;
  212. white-space: nowrap;
  213. text-overflow: ellipsis;
  214. z-index: 5;
  215. }
  216. .camera {
  217. width: 100%;
  218. height: 100%;
  219. }
  220. .mark {
  221. position: absolute;
  222. left: 0;
  223. top: 0;
  224. z-index: 2;
  225. width: 750rpx;
  226. height: 100%;
  227. background: url("@/static/mine/view_face_background.png") no-repeat
  228. center bottom;
  229. background-size: 750rpx 661rpx;
  230. }
  231. image {
  232. position: absolute;
  233. width: 100%;
  234. height: 100%;
  235. z-index: 3;
  236. }
  237. }
  238. .footerV {
  239. width: 100%;
  240. display: flex;
  241. flex-direction: row;
  242. align-items: center;
  243. justify-content: center;
  244. .privacyV {
  245. padding-top: 30rpx;
  246. display: flex;
  247. flex-direction: row;
  248. align-items: center;
  249. justify-content: center;
  250. .text {
  251. font-size: 30rpx;
  252. color: #1c1c1c;
  253. text-align: center;
  254. line-height: 42rpx;
  255. margin-left: 15rpx;
  256. text {
  257. font-size: 30rpx;
  258. color: #00aaff;
  259. text-align: center;
  260. line-height: 42rpx;
  261. }
  262. }
  263. .icon {
  264. width: 40rpx;
  265. height: 47rpx;
  266. // background: url("@/static/face/icon_face_security.png") no-repeat;
  267. background-size: 100% auto;
  268. }
  269. }
  270. .bottom-tips-2 {
  271. margin-top: 20rpx;
  272. color: #999999;
  273. text-align: center;
  274. font-size: 26rpx;
  275. }
  276. .take-photo-bgV {
  277. width: 100%;
  278. margin-top: 30rpx;
  279. display: flex;
  280. flex-direction: row;
  281. align-items: center;
  282. justify-content: center;
  283. .btn-take-photo {
  284. // 由于左边没有按钮,所以左边要便宜更大,以便是拍照按钮居中
  285. margin: 0rpx 80rpx 0rpx 80rpx;
  286. width: 196rpx;
  287. height: 196rpx;
  288. background: url("https://pro-file-qn.ztjy61.com/1003020211103145058685NNR9vlTm.png")
  289. no-repeat;
  290. background-size: 100% auto;
  291. }
  292. .btn-change-upload {
  293. left: 130rpx;
  294. width: 80rpx;
  295. height: 80rpx;
  296. // background: url("@/static/face/icon_face_upload_picture.png")
  297. // no-repeat;
  298. background-size: 100% auto;
  299. }
  300. .btn-change-camera {
  301. right: 130rpx;
  302. width: 80rpx;
  303. height: 80rpx;
  304. background: url("@/static/mine/358.png") no-repeat;
  305. background-size: 100% auto;
  306. }
  307. }
  308. .confirmV {
  309. margin: 200rpx 100rpx 0rpx 100rpx;
  310. display: flex;
  311. flex-direction: row;
  312. align-items: center;
  313. justify-content: space-between;
  314. .btn-cancel {
  315. font-size: 32rpx;
  316. color: #1c1c1c;
  317. }
  318. .btn-ok {
  319. font-size: 32rpx;
  320. color: #00aaff;
  321. }
  322. }
  323. }
  324. }
  325. }
  326. </style>