identify.vue 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. <template>
  2. <div @click="handleClick" class="s-canvas">
  3. <canvas id="s-canvas" :width="contentWidth" :height="contentHeight"></canvas>
  4. </div>
  5. </template>
  6. <script>
  7. export default {
  8. name: 'SIdentify',
  9. props: {
  10. identifyCode: { // 默认注册码
  11. type: String,
  12. default: '1234'
  13. },
  14. fontSizeMin: { // 字体最小值
  15. type: Number,
  16. default: 25
  17. },
  18. fontSizeMax: { // 字体最大值
  19. type: Number,
  20. default: 35
  21. },
  22. backgroundColorMin: { // 验证码图片背景色最小值
  23. type: Number,
  24. default: 200
  25. },
  26. backgroundColorMax: { // 验证码图片背景色最大值
  27. type: Number,
  28. default: 220
  29. },
  30. dotColorMin: { // 背景干扰点最小值
  31. type: Number,
  32. default: 60
  33. },
  34. dotColorMax: { // 背景干扰点最大值
  35. type: Number,
  36. default: 120
  37. },
  38. contentWidth: { // 容器宽度
  39. type: Number,
  40. default: 116
  41. },
  42. contentHeight: { // 容器高度
  43. type: Number,
  44. default: 40
  45. }
  46. },
  47. methods: {
  48. // 生成一个随机数
  49. randomNum (min, max) {
  50. return Math.floor(Math.random() * (max - min) + min)
  51. },
  52. // 生成一个随机的颜色
  53. randomColor (min, max) {
  54. let r = this.randomNum(min, max)
  55. let g = this.randomNum(min, max)
  56. let b = this.randomNum(min, max)
  57. return 'rgb(' + r + ',' + g + ',' + b + ')'
  58. },
  59. drawPic () {
  60. let canvas = document.getElementById('s-canvas')
  61. let ctx = canvas.getContext('2d')
  62. ctx.textBaseline = 'bottom'
  63. // 绘制背景
  64. ctx.fillStyle = this.randomColor(this.backgroundColorMin, this.backgroundColorMax)
  65. ctx.fillRect(0, 0, this.contentWidth, this.contentHeight)
  66. // 绘制文字
  67. for (let i = 0; i < this.identifyCode.length; i++) {
  68. this.drawText(ctx, this.identifyCode[i], i)
  69. }
  70. this.drawLine(ctx)
  71. this.drawDot(ctx)
  72. },
  73. drawText (ctx, txt, i) {
  74. ctx.fillStyle = this.randomColor(50, 160) // 随机生成字体颜色
  75. ctx.font = this.randomNum(this.fontSizeMin, this.fontSizeMax) + 'px SimHei' // 随机生成字体大小
  76. let x = (i + 1) * (this.contentWidth / (this.identifyCode.length + 1))
  77. let y = this.randomNum(this.fontSizeMax, this.contentHeight - 5)
  78. var deg = this.randomNum(-30, 30)
  79. // 修改坐标原点和旋转角度
  80. ctx.translate(x, y)
  81. ctx.rotate(deg * Math.PI / 180)
  82. ctx.fillText(txt, 0, 0)
  83. // 恢复坐标原点和旋转角度
  84. ctx.rotate(-deg * Math.PI / 180)
  85. ctx.translate(-x, -y)
  86. },
  87. drawLine (ctx) {
  88. // 绘制干扰线
  89. for (let i = 0; i < 4; i++) {
  90. ctx.strokeStyle = this.randomColor(100, 200)
  91. ctx.beginPath()
  92. ctx.moveTo(this.randomNum(0, this.contentWidth), this.randomNum(0, this.contentHeight))
  93. ctx.lineTo(this.randomNum(0, this.contentWidth), this.randomNum(0, this.contentHeight))
  94. ctx.stroke()
  95. }
  96. },
  97. drawDot (ctx) {
  98. // 绘制干扰点
  99. for (let i = 0; i < 30; i++) {
  100. ctx.fillStyle = this.randomColor(0, 255)
  101. ctx.beginPath()
  102. ctx.arc(this.randomNum(0, this.contentWidth), this.randomNum(0, this.contentHeight), 1, 0, 2 * Math.PI)
  103. ctx.fill()
  104. }
  105. },
  106. //更新验证码
  107. handleClick () {
  108. this.$emit('changeCode')
  109. }
  110. },
  111. watch: {
  112. identifyCode () {
  113. this.drawPic()
  114. }
  115. },
  116. // mounted () {
  117. // this.drawPic()
  118. // }
  119. }
  120. </script>
  121. <style lang="scss" scoped>
  122. .s-canvas {
  123. cursor: pointer;
  124. }
  125. </style>