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