faceIdentify.vue 10 KB

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