You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

559 lines
66 KiB

3 months ago
  1. <template>
  2. <view style="position: relative">
  3. <view class="verify-image-out" v-show="showImage">
  4. <view class="verify-image-panel" :style="{'width': '100%',
  5. 'height': imgSize.height,
  6. 'margin-bottom': vSpace + 'px'}">
  7. <view class="verify-refresh" style="z-index:3" @click="refresh" v-show="showRefresh">
  8. <text class="iconfont icon-refresh"></text>
  9. </view>
  10. <image :src="pointBackImgBase?('data:image/png;base64,'+pointBackImgBase):defaultImg" id="image"
  11. ref="canvas" style="width:100%;height:100%;display:block"
  12. @click=" bindingClick? canvasClick($event): undefined"></image>
  13. <view v-for="(tempPoint, index) in tempPoints" :key="index" class="point-area" :style="{
  14. 'background-color':'#1abd6c',
  15. color:'#fff',
  16. 'z-index':9999,
  17. width:'20px',
  18. height:'20px',
  19. 'text-align':'center',
  20. 'line-height':'20px',
  21. 'border-radius': '50%',
  22. position:'absolute',
  23. top:parseInt(tempPoint.y-10) + 'px',
  24. left:parseInt(tempPoint.x-10) + 'px'
  25. }">
  26. {{index + 1}}
  27. </view>
  28. </view>
  29. </view>
  30. <!-- 'height': this.barSize.height, -->
  31. <view class="verify-bar-area" :style="{'width': '100%',
  32. 'color': barAreaColor,
  33. 'border-color': barAreaBorderColor,
  34. 'line-height':'40px'}">
  35. <text class="verify-msg">{{text}}</text>
  36. </view>
  37. </view>
  38. </template>
  39. <script type="text/babel">
  40. /**
  41. * VerifyPoints
  42. * @description 点选
  43. * */
  44. import {aesEncrypt} from "@/utils/ase.js"
  45. import {getAjcaptcha,ajcaptchaCheck} from '../api/index.js';
  46. export default {
  47. name: 'VerifyPoints',
  48. props: {
  49. //弹出式pop,固定fixed
  50. mode: {
  51. type: String,
  52. default: 'fixed'
  53. },
  54. captchaType:{
  55. type:String,
  56. },
  57. //间隔
  58. vSpace: {
  59. type: Number,
  60. default: 5
  61. },
  62. imgSize: {
  63. type: Object,
  64. default() {
  65. return {
  66. width: '310px',
  67. height: '155px'
  68. }
  69. }
  70. },
  71. barSize: {
  72. type: Object,
  73. default() {
  74. return {
  75. width: '310px',
  76. height: '40px'
  77. }
  78. }
  79. },
  80. defaultImg: {
  81. type: String,
  82. default: ''
  83. }
  84. },
  85. data() {
  86. return {
  87. secretKey:'', //后端返回的加密秘钥 字段
  88. checkNum:3, //
  89. fontPos: [], // 选中的坐标信息
  90. checkPosArr: [], //用户点击的坐标
  91. num: 1, //点击的记数
  92. pointBackImgBase:'', //后端获取到的背景图片
  93. poinTextList:[], //后端返回的点击字体顺序
  94. backToken:'', //后端返回的token值
  95. imgRand: 0, //随机的背景图片
  96. setSize: {
  97. imgHeight: 0,
  98. imgWidth: 0,
  99. barHeight: 0,
  100. barWidth: 0
  101. },
  102. showImage: true,
  103. tempPoints: [],
  104. text: '',
  105. barAreaColor: '#fff',
  106. barAreaBorderColor: "#fff",
  107. showRefresh: true,
  108. bindingClick: true,
  109. imgLeft:'' ,
  110. imgTop:'',
  111. }
  112. },
  113. methods: {
  114. init() {
  115. //加载页面
  116. this.fontPos.splice(0, this.fontPos.length)
  117. this.checkPosArr.splice(0, this.checkPosArr.length)
  118. this.num = 1
  119. this.$nextTick(() => {
  120. this.refresh();
  121. this.$parent.$emit('ready', this)
  122. })
  123. },
  124. canvasClick(e) {
  125. const query = uni.createSelectorQuery().in(this);
  126. query.select('#image').boundingClientRect(data => {
  127. this.imgLeft =Math.ceil(data.left)
  128. this.imgTop =Math.ceil(data.top)
  129. this.checkPosArr.push(this.getMousePos(this.$refs.canvas, e));
  130. if (this.num == this.checkNum) {
  131. this.num = this.createPoint(this.getMousePos(this.$refs.canvas, e));
  132. //按比例转换坐标值
  133. this.checkPosArr = this.pointTransfrom(this.checkPosArr,this.imgSize);
  134. //等创建坐标执行完
  135. setTimeout(() => {
  136. //发送后端请求
  137. var captchaVerification =this.secretKey? aesEncrypt(this.backToken+'---'+JSON.stringify(this.checkPosArr),this.secretKey):this.backToken+'---'+JSON.stringify(this.checkPosArr)
  138. let data = {
  139. captchaType:this.captchaType,
  140. "pointJson":this.secretKey? aesEncrypt(JSON.stringify(this.checkPosArr),this.secretKey):JSON.stringify(this.checkPosArr),
  141. "token":this.backToken
  142. }
  143. ajcaptchaCheck(data).then(r => {
  144. let res = r.data;
  145. if (res.repCode == '0000') {
  146. this.barAreaColor = '#4cae4c';
  147. this.barAreaBorderColor = '#5cb85c';
  148. this.text = '验证成功';
  149. this.bindingClick = false;
  150. if (this.mode == 'pop') {
  151. setTimeout(() => {
  152. this.$parent.clickShow = false;
  153. this.refresh();
  154. }, 1500);
  155. }
  156. setTimeout(() => {
  157. this.tipWords = ""
  158. this.$emit('success', {
  159. captchaVerification
  160. })
  161. }, 1000)
  162. // this.$parent.$emit('success', { captchaVerification });
  163. } else {
  164. this.$parent.$emit('error', this);
  165. this.barAreaColor = '#d9534f';
  166. this.barAreaBorderColor = '#d9534f';
  167. this.text = '验证失败';
  168. setTimeout(() => {
  169. this.refresh();
  170. }, 700);
  171. }
  172. })
  173. }, 400);
  174. }
  175. if (this.num < this.checkNum) {
  176. this.num = this.createPoint(this.getMousePos(this.$refs.canvas, e));
  177. }
  178. }).exec();
  179. },
  180. //获取坐标
  181. getMousePos: function (obj, e) {
  182. let position = {
  183. x:Math.ceil(e.detail.x)-this.imgLeft,
  184. y:Math.ceil(e.detail.y)-this.imgTop,
  185. }
  186. return position
  187. },
  188. //创建坐标点
  189. createPoint: function (pos) {
  190. this.tempPoints.push(Object.assign({}, pos))
  191. return ++this.num;
  192. },
  193. refresh: function () {
  194. this.tempPoints.splice(0, this.tempPoints.length)
  195. this.barAreaColor = '#000'
  196. this.barAreaBorderColor = '#ddd'
  197. this.bindingClick = true
  198. this.fontPos.splice(0, this.fontPos.length)
  199. this.checkPosArr.splice(0, this.checkPosArr.length)
  200. this.num = 1
  201. this.getPictrue();
  202. // this.text = '验证失败'
  203. this.showRefresh = true
  204. },
  205. // 请求背景图片和验证图片
  206. getPictrue(){
  207. let data = {
  208. captchaType:this.captchaType,
  209. clientUid: uni.getStorageSync('point'),
  210. ts: Date.now(), // 现在的时间戳
  211. }
  212. getAjcaptcha(data).then((res) => {
  213. let data = res.data
  214. if (data.repCode == '0000') {
  215. this.pointBackImgBase = data.repData.originalImageBase64;
  216. this.backToken = data.repData.token;
  217. this.secretKey = data.repData.secretKey;
  218. this.poinTextList = data.repData.wordList;
  219. this.text = '请依次点击【' + this.poinTextList.join(',') + '】';
  220. } else {
  221. this.text = data.repMsg;
  222. }
  223. // 判断接口请求次数是否失效
  224. if (res.repCode == '6201') {
  225. this.pointBackImgBase = null;
  226. }
  227. }).catch(()=>{
  228. this.pointBackImgBase = null
  229. })
  230. },
  231. //坐标转换函数
  232. pointTransfrom(pointArr,imgSize){
  233. var newPointArr = pointArr.map(p=>{
  234. let x = Math.round(310 * p.x/parseInt(imgSize.width))
  235. let y =Math.round(155 * p.y/parseInt(imgSize.height))
  236. return {x,y}
  237. })
  238. return newPointArr
  239. },
  240. },
  241. watch: {
  242. // type变化则全面刷新
  243. type: {
  244. immediate: true,
  245. handler() {
  246. this.init()
  247. }
  248. }
  249. },
  250. mounted() {
  251. }
  252. }
  253. </script>
  254. <style scoped>
  255. .verifybox {
  256. position: relative;
  257. box-sizing: border-box;
  258. border-radius: 2px;
  259. border: 1px solid #e4e7eb;
  260. background-color: #fff;
  261. box-shadow: 0 0 10px rgba(0, 0, 0, .3);
  262. left: 50%;
  263. top: 50%;
  264. transform: translate(-50%, -50%);
  265. }
  266. .verifybox-top {
  267. padding: 0 15px;
  268. height: 50px;
  269. line-height: 50px;
  270. text-align: left;
  271. font-size: 16px;
  272. color: #45494c;
  273. border-bottom: 1px solid #e4e7eb;
  274. box-sizing: border-box;
  275. }
  276. .verifybox-bottom {
  277. /* padding: 15px; */
  278. box-sizing: border-box;
  279. }
  280. .verifybox-close {
  281. position: absolute;
  282. top: 13px;
  283. right: 9px;
  284. width: 24px;
  285. height: 24px;
  286. text-align: center;
  287. cursor: pointer;
  288. }
  289. .mask {
  290. position: fixed;
  291. top: 0;
  292. left: 0;
  293. z-index: 1001;
  294. width: 100%;
  295. height: 100vh;
  296. background: rgba(0, 0, 0, .3);
  297. /* display: none; */
  298. transition: all .5s;
  299. }
  300. .verify-tips {
  301. position: absolute;
  302. left: 0px;
  303. bottom: 0px;
  304. width: 100%;
  305. height: 30px;
  306. background-color: rgb(231, 27, 27, .5);
  307. line-height: 30px;
  308. color: #fff;
  309. }
  310. .tips-enter,
  311. .tips-leave-to {
  312. bottom: -30px;
  313. }
  314. .tips-enter-active,
  315. .tips-leave-active {
  316. transition: bottom .5s;
  317. }
  318. /* ---------------------------- */
  319. /*常规验证码*/
  320. .verify-code {
  321. font-size: 20px;
  322. text-align: center;
  323. cursor: pointer;
  324. margin-bottom: 5px;
  325. border: 1px solid #ddd;
  326. }
  327. .cerify-code-panel {
  328. height: 100%;
  329. overflow: hidden;
  330. }
  331. .verify-code-area {
  332. float: left;
  333. }
  334. .verify-input-area {
  335. float: left;
  336. width: 60%;
  337. padding-right: 10px;
  338. }
  339. .verify-change-area {
  340. line-height: 30px;
  341. float: left;
  342. }
  343. .varify-input-code {
  344. display: inline-block;
  345. width: 100%;
  346. height: 25px;
  347. }
  348. .verify-change-code {
  349. color: #337AB7;
  350. cursor: pointer;
  351. }
  352. .verify-btn {
  353. width: 200px;
  354. height: 30px;
  355. background-color: #337AB7;
  356. color: #FFFFFF;
  357. border: none;
  358. margin-top: 10px;
  359. }
  360. /*滑动验证码*/
  361. .verify-bar-area {
  362. position: relative;
  363. background: #FFFFFF;
  364. text-align: center;
  365. -webkit-box-sizing: content-box;
  366. -moz-box-sizing: content-box;
  367. box-sizing: content-box;
  368. border: 1px solid #ddd;
  369. -webkit-border-radius: 4px;
  370. }
  371. .verify-bar-area .verify-move-block {
  372. position: absolute;
  373. top: 0px;
  374. left: 0;
  375. background: #fff;
  376. cursor: pointer;
  377. -webkit-box-sizing: content-box;
  378. -moz-box-sizing: content-box;
  379. box-sizing: content-box;
  380. box-shadow: 0 0 2px #888888;
  381. -webkit-border-radius: 1px;
  382. }
  383. .verify-bar-area .verify-move-block:hover {
  384. background-color: #337ab7;
  385. color: #FFFFFF;
  386. }
  387. .verify-bar-area .verify-left-bar {
  388. position: absolute;
  389. top: -1px;
  390. left: -1px;
  391. background: #f0fff0;
  392. cursor: pointer;
  393. -webkit-box-sizing: content-box;
  394. -moz-box-sizing: content-box;
  395. box-sizing: content-box;
  396. border: 1px solid #ddd;
  397. }
  398. .verify-image-panel {
  399. margin: 0;
  400. -webkit-box-sizing: content-box;
  401. -moz-box-sizing: content-box;
  402. box-sizing: content-box;
  403. border-top: 1px solid #ddd;
  404. border-bottom: 1px solid #ddd;
  405. border-radius: 3px;
  406. position: relative;
  407. }
  408. .verify-image-panel .verify-refresh {
  409. width: 25px;
  410. height: 25px;
  411. text-align: center;
  412. padding: 5px;
  413. cursor: pointer;
  414. position: absolute;
  415. top: 0;
  416. right: 0;
  417. z-index: 2;
  418. }
  419. .verify-image-panel .icon-refresh {
  420. font-size: 20px;
  421. color: #fff;
  422. }
  423. .verify-image-panel .verify-gap {
  424. background-color: #fff;
  425. position: relative;
  426. z-index: 2;
  427. border: 1px solid #fff;
  428. }
  429. .verify-bar-area .verify-move-block .verify-sub-block {
  430. position: absolute;
  431. text-align: center;
  432. z-index: 3;
  433. /* border: 1px solid #fff; */
  434. }
  435. .verify-bar-area .verify-move-block .verify-icon {
  436. font-size: 18px;
  437. }
  438. .verify-bar-area .verify-msg {
  439. z-index: 3;
  440. }
  441. /*字体图标的css*/
  442. /*@font-face {font-family: "iconfont";*/
  443. /*src: url('../fonts/iconfont.eot?t=1508229193188'); !* IE9*!*/
  444. /*src: url('../fonts/iconfont.eot?t=1508229193188#iefix') format('embedded-opentype'), !* IE6-IE8 *!*/
  445. /*url('data:application/x-font-woff;charset=utf-8;base64,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') format('woff'),*/
  446. /*url('../fonts/iconfont.ttf?t=1508229193188') format('truetype'), !* chrome, firefox, opera, Safari, Android, iOS 4.2+*!*/
  447. /*url('../fonts/iconfont.svg?t=1508229193188#iconfont') format('svg'); !* iOS 4.1- *!*/
  448. /*}*/
  449. .iconfont {
  450. font-family: "iconfont" !important;
  451. font-size: 16px;
  452. font-style: normal;
  453. -webkit-font-smoothing: antialiased;
  454. -moz-osx-font-smoothing: grayscale;
  455. }
  456. .icon-check:before {
  457. content: " ";
  458. display: block;
  459. width: 16px;
  460. height: 16px;
  461. position: absolute;
  462. margin: auto;
  463. left: 0;
  464. right: 0;
  465. top: 0;
  466. bottom: 0;
  467. z-index: 9999;
  468. background-image: url("
  469. background-size: contain;
  470. }
  471. .icon-close:before {
  472. content: " ";
  473. display: block;
  474. width: 16px;
  475. height: 16px;
  476. position: absolute;
  477. margin: auto;
  478. left: 0;
  479. right: 0;
  480. top: 0;
  481. bottom: 0;
  482. z-index: 9999;
  483. background-image: url("
  484. background-size: contain;
  485. }
  486. .icon-right:before {
  487. content: " ";
  488. display: block;
  489. width: 16px;
  490. height: 16px;
  491. position: absolute;
  492. margin: auto;
  493. left: 0;
  494. right: 0;
  495. top: 0;
  496. bottom: 0;
  497. background-size: cover;
  498. z-index: 9999;
  499. background-image: url("
  500. background-size: contain;
  501. }
  502. .icon-refresh:before {
  503. content: " ";
  504. display: block;
  505. width: 16px;
  506. height: 16px;
  507. position: absolute;
  508. margin: auto;
  509. left: 0;
  510. right: 0;
  511. top: 0;
  512. bottom: 0;
  513. z-index: 9999;
  514. background-image: url("
  515. background-size: contain;
  516. }
  517. </style>