SliderCaptcha.vue 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483
  1. <template>
  2. <div class="slider-captcha-container">
  3. <div v-if="dialogVisible" class="lang-dialog">
  4. <div class="lang-dialog__header">
  5. <div class="lang-dialog__title">
  6. <slot name="title">{{ title }}</slot>
  7. </div>
  8. <div type="button" class="lang-dialog-header-btn" @click="close"> <i class="lang-icon-close"></i> </div>
  9. </div>
  10. <div class="lang-dialog__body">
  11. <div v-show="failed" class="tips">
  12. <slot name="errorText">{{ errorText }}</slot>
  13. </div>
  14. <div v-show="!failed" class="tips">
  15. <slot name="tips">{{ tips }}</slot>
  16. </div>
  17. <div class="slider-body" :class="{ 'slider-shock': shock }">
  18. <div v-show="mask" class="loading-transparent-mask"></div>
  19. <div v-if="loading" class="loading-body">
  20. <div class="loading slider-loading"></div>
  21. </div>
  22. <div
  23. v-show="!loading" class="slider-bg slider-img"
  24. :style="{ 'background-image': 'url(data:image/png;base64,' + shadeImage + ')' }"
  25. >
  26. </div>
  27. <div
  28. v-show="!loading" class="slider-draw slider-move-draw slider-img"
  29. :style="{ 'background-image': 'url(data:image/png;base64,' + cutoutImage + ')', 'top': sliderY + 'px', 'left': sliderMoveDrawLeft }"
  30. @touchstart.stop="sliderTouchStart" @touchmove.stop="sliderTouchMove" @touchend.stop="sliderEnd"
  31. @mousedown="sliderDown"
  32. >
  33. </div>
  34. <div class="slider-progress"></div>
  35. <div v-show="success" class="slider-success">
  36. <div class="slider-success-icon">
  37. 成功
  38. </div>
  39. <div class="slider-success-text">
  40. <slot name="successText">{{ successText }}</slot>
  41. </div>
  42. </div>
  43. <div
  44. class="slider-btn slider-move-btn" :style="{ 'left': sliderMoveLeft }"
  45. :class="{ 'slider-shock': shock }" @touchstart.stop="sliderTouchStart" @touchmove.stop="sliderTouchMove"
  46. @touchend.stop="sliderEnd" @mousedown="sliderDown"
  47. >
  48. <i>&nbsp;</i>
  49. <span style="width: 100%;color: #ffffff;">● ● ●</span>
  50. </div>
  51. </div>
  52. <div style="position:relative;display: flex;align-items: center;width: 280px;margin: 0 auto;">
  53. <div class="slider-refresh" @click.stop="questionMessage = !questionMessage">
  54. 说明
  55. </div>
  56. <div class="slider-refresh" @click.stop="refresh">
  57. 刷新
  58. </div>
  59. </div>
  60. <div v-if="questionMessage">
  61. <slot name="question">
  62. <div v-if="question">{{ question }}</div>
  63. <div v-else>无注意事项</div>
  64. </slot>
  65. </div>
  66. </div>
  67. </div>
  68. </div>
  69. </template>
  70. <script>
  71. export default {
  72. name: 'SliderCaptcha',
  73. props:
  74. {
  75. value: {
  76. type: Boolean,
  77. defalut: false
  78. },
  79. loading: {
  80. type: Boolean,
  81. defalut: false
  82. },
  83. title: {
  84. type: String,
  85. default: '滑块安全验证'
  86. },
  87. tips: {
  88. type: String,
  89. default: '拖动下方滑块完成拼图'
  90. },
  91. successText: {
  92. type: String,
  93. default: '验证成功'
  94. },
  95. errorText: {
  96. type: String,
  97. default: '是不是太难了,咱换一个'
  98. },
  99. question: {
  100. type: String,
  101. default: ''
  102. },
  103. options: {
  104. type: Object,
  105. default() {
  106. return {
  107. cutoutImage: '',
  108. shadeImage: '',
  109. sliderKey: '',
  110. sliderY: ''
  111. }
  112. }
  113. }
  114. },
  115. data() {
  116. return {
  117. dialogVisible: false,
  118. mask: false,
  119. success: false,
  120. failed: false,
  121. cutoutImage: '',
  122. shadeImage: '',
  123. sliderKey: '',
  124. sliderX: 26,
  125. sliderY: '',
  126. sliderMoveDrawLeft: '26px',
  127. sliderMoveLeft: '20px',
  128. shock: false,
  129. tipEvents: {},
  130. sliderMoveX: 0,
  131. questionMessage: false
  132. }
  133. },
  134. watch: {
  135. value: {
  136. handler(val) {
  137. this.init(val)
  138. },
  139. immediate: true
  140. },
  141. options: {
  142. handler(option) {
  143. this.clear()
  144. this.cutoutImage = option.cutoutImage
  145. this.shadeImage = option.shadeImage
  146. this.sliderKey = option.sliderKey
  147. this.sliderY = option.sliderY
  148. },
  149. deep: true,
  150. immediate: true
  151. }
  152. },
  153. methods: {
  154. init(open) {
  155. this.dialogVisible = open
  156. if (open) {
  157. this.clear()
  158. }
  159. },
  160. clear() {
  161. this.mask = false
  162. this.success = false
  163. this.failed = false
  164. this.cutoutImage = ''
  165. this.shadeImage = ''
  166. this.sliderKey = ''
  167. this.sliderX = 26
  168. this.sliderMoveDrawLeft = '26px',
  169. this.sliderMoveLeft = '20px',
  170. this.sliderY = ''
  171. },
  172. check(sliderKey, sliderX) {
  173. this.$emit('check', { sliderKey, sliderX, done: this.done, error: this.error })
  174. },
  175. done() {
  176. this.success = true
  177. },
  178. error() {
  179. this.failed = true
  180. this.mask = true
  181. this.shock = true
  182. setTimeout(() => {
  183. this.shock = false
  184. this.$emit('error')
  185. }, 1000)
  186. },
  187. close() {
  188. this.dialogVisible = false
  189. this.$emit('input', this.dialogVisible)
  190. this.$emit('close')
  191. },
  192. refresh() {
  193. this.$emit('refresh')
  194. },
  195. sliderTouchStart(e) {
  196. // 移动触摸移动
  197. const that = this
  198. const slider = e.target
  199. that.sliderMoveX = e.touches[0].clientX - slider.offsetLeft
  200. console.log(e, e.touches[0].clientX, slider.offsetLeft)
  201. },
  202. sliderTouchMove(e) {
  203. const that = this
  204. const left = e.touches[0].clientX - that.sliderMoveX
  205. if (left >= 20 && left <= 280) {
  206. that.sliderMoveDrawLeft = 5 + left + 'px'
  207. that.sliderMoveLeft = left + 'px'
  208. that.sliderX = 5 + left
  209. }
  210. },
  211. sliderEnd() {
  212. this.check(this.sliderKey, this.sliderX)
  213. },
  214. sliderDown(e) {
  215. console.log(e)
  216. const that = this
  217. const slider = e.target // 获取目标元素
  218. // 算出鼠标相对元素的位置
  219. that.sliderMoveX = e.clientX - slider.offsetLeft
  220. document.onmousemove = (e) => {
  221. const left = e.clientX - that.sliderMoveX
  222. if (left >= 20 && left <= 280) {
  223. // slider.style.left = left + 'px'
  224. that.sliderMoveDrawLeft = 5 + left + 'px'
  225. that.sliderMoveLeft = left + 'px'
  226. that.sliderX = 5 + left
  227. }
  228. }
  229. document.onmouseup = () => {
  230. document.onmousemove = null
  231. document.onmouseup = null
  232. this.check(this.sliderKey, this.sliderX)
  233. }
  234. }
  235. }
  236. }
  237. </script>
  238. <style lang="scss" scoped>
  239. .slider-captcha-container {
  240. box-sizing: border-box;
  241. .slider-shock {
  242. animation-delay: 0s;
  243. animation-name: shock;
  244. animation-duration: .1s;
  245. animation-iteration-count: 5;
  246. animation-direction: normal;
  247. animation-timing-function: linear;
  248. }
  249. @keyframes shock {
  250. 0% {
  251. transform: translateX(2px);
  252. }
  253. 100% {
  254. transform: translateX(-2px);
  255. }
  256. }
  257. /* 弹窗开始 */
  258. .lang-dialog {
  259. position: relative;
  260. border-radius: 2px;
  261. box-shadow: 0 1px 3px rgba(0, 0, 0, 30%);
  262. box-sizing: border-box;
  263. background: #FFF;
  264. }
  265. .lang-dialog__header {
  266. padding: 20px 20px 10px;
  267. }
  268. .lang-dialog__title {
  269. line-height: 20px;
  270. font-size: 18px;
  271. color: #525252;
  272. }
  273. .lang-dialog-header-btn {
  274. position: absolute;
  275. top: 20px;
  276. right: 20px;
  277. padding: 0;
  278. background: 0 0;
  279. border: none;
  280. outline: 0;
  281. cursor: pointer;
  282. font-size: 16px
  283. }
  284. .lang-icon-close {
  285. color: rgba(0, 0, 0, 0.68);
  286. font-size: 20px;
  287. }
  288. .lang-icon-close:before {
  289. content: '×';
  290. }
  291. .lang-icon-close:hover {
  292. color: rgba(0, 0, 0, 0.88);
  293. }
  294. .lang-dialog__body {
  295. padding: 0 0 6px;
  296. color: #828282;
  297. font-size: 14px;
  298. text-align: center;
  299. word-break: break-all;
  300. }
  301. .loading-transparent-mask {
  302. position: absolute;
  303. top: 0;
  304. left: 0;
  305. width: 100%;
  306. height: 100%;
  307. background: transparent;
  308. z-index: 999;
  309. }
  310. /* 弹窗结束 */
  311. /* loading start */
  312. .loading-body {
  313. position: absolute;
  314. top: 0;
  315. left: 0;
  316. width: 100%;
  317. height: 100%;
  318. background: rgba(255, 255, 255, 0.6);
  319. z-index: 999;
  320. }
  321. .loading {
  322. position: relative;
  323. display: inline-block;
  324. width: 30px;
  325. height: 30px;
  326. border-radius: 50%;
  327. border-top: 2px solid transparent;
  328. border-bottom: 2px solid transparent;
  329. border-left: 2px solid #409eff;
  330. border-right: 2px solid #409eff;
  331. animation: loading 1s infinite linear;
  332. }
  333. @keyframes loading {
  334. to {
  335. transform: rotate(360deg);
  336. }
  337. }
  338. .slider-loading {
  339. position: absolute;
  340. top: calc(50% - 15px);
  341. left: calc(50% - 15px);
  342. z-index: 999;
  343. }
  344. /* loading end */
  345. .lang-dialog__header {
  346. padding: 20px 10px 10px;
  347. }
  348. .tips {
  349. color: #525252;
  350. line-height: 36px;
  351. font-size: 18px;
  352. }
  353. .slider-body {
  354. position: relative;
  355. width: 280px;
  356. height: 230px;
  357. margin: 0 auto;
  358. }
  359. .slider-bg {
  360. position: relative;
  361. width: 280px;
  362. height: 171px;
  363. overflow: hidden;
  364. background-position: top left;
  365. // background-size: cover;
  366. background-size: auto auto;
  367. background-repeat: no-repeat;
  368. }
  369. .slider-draw {
  370. position: absolute;
  371. z-index: 1;
  372. left: 26px;
  373. width: 50px;
  374. height: 50px;
  375. background-position: top left;
  376. background-size: cover;
  377. background-repeat: no-repeat;
  378. cursor: pointer;
  379. opacity: 1;
  380. box-shadow: 0px 0px 10px 2px #000000;
  381. }
  382. .slider-progress {
  383. position: relative;
  384. z-index: 1;
  385. width: 280px;
  386. height: 16px;
  387. margin-top: 22px;
  388. background-color: #c8c8c8;
  389. border-radius: 8px;
  390. opacity: 1;
  391. }
  392. .slider-btn {
  393. position: absolute;
  394. top: 184px;
  395. z-index: 1;
  396. width: 65px;
  397. height: 35px;
  398. line-height: 35px;
  399. background-color: rgb(0, 87, 212);
  400. box-shadow: rgba(0, 87, 212, 50%) 0px 0px 5.05952px 0.505952px;
  401. text-align: center;
  402. border-radius: 999px;
  403. cursor: pointer;
  404. opacity: 1;
  405. -moz-user-select: none;
  406. /* 禁止双击节点被选中 */
  407. -o-user-select: none;
  408. -khtml-user-select: none;
  409. -webkit-user-select: none;
  410. -ms-user-select: none;
  411. user-select: none;
  412. -moz-user-drag: none;
  413. /* 禁止被拖动 */
  414. -webkit-user-drag: none;
  415. /* 禁止被拖动 */
  416. // pointer-events: none;
  417. // display: inline-block;
  418. // vertical-align: middle;
  419. }
  420. .slider-btn i {
  421. display: inline-block;
  422. width: 0;
  423. vertical-align: middle;
  424. }
  425. .slider-success {
  426. position: absolute;
  427. top: 0;
  428. left: 0;
  429. width: 100%;
  430. height: 100%;
  431. z-index: 4;
  432. background: hsla(0, 0%, 100%, .8);
  433. }
  434. .slider-success .slider-success-icon {
  435. width: 64px;
  436. height: 64px;
  437. margin: 15px auto 0;
  438. }
  439. .slider-success .slider-success-text {
  440. color: #1bc300;
  441. text-align: center;
  442. margin-top: 16px;
  443. font-size: 18px;
  444. }
  445. .slider-refresh {
  446. margin: 0 16px 0 0;
  447. cursor: pointer;
  448. font-size: 18px;
  449. height: 30px;
  450. line-height: 30px;
  451. }
  452. }
  453. </style>