《uniApp-车牌号选择组件(包含键盘)》

《uniApp-车牌号选择组件(包含键盘)》
一清三白使用 UniApp 实现车牌号选择组件(含键盘和新能源支持)
在移动应用开发中,车牌号输入是一个常见需求,尤其是在与交通、停车、违章查询等相关的应用中。与普通文本输入不同,车牌号输入有着特定的规则和格式,需要专门的键盘和交互设计。本文将详细介绍如何使用 UniApp 框架实现一个功能完善的车牌号选择组件,支持常规车牌和新能源车牌。
效果预览
功能特点
- 支持普通车牌和新能源车牌输入
- 自定义键盘,分为省份简称键盘和字母数字键盘
- 智能切换键盘类型
- 支持车牌号规则限制(如第二位只能输入字母或数字等)
- 集成删除和关闭功能
- 支持初始化已有车牌号
- 优雅的样式和流畅的交互体验
实现思路
- 创建车牌号输入框组件
- 实现省份简称键盘
- 实现字母数字键盘
- 添加车牌号输入规则和限制
- 实现键盘切换和交互逻辑
代码实现
完整代码
- 在components下建立car-number-input文件 复制以下代码
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
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316<template>
<view>
<view class="car-input-container">
<view class="car-input-box"
v-for="(item,index) in inputList" :key="index" @click="plateInput(index)">
<view class="car-input-item"
:class="[curInput == index?'sel-item':'',(maxNum-1) == index?'last-item':'']" >
<view :class="curInput == index?'sel-item-line':''"></view>
<img :src="xnyImgBase64" class="new-item-img" v-if="(maxNum-1) == index"/>
{{item}}
</view>
</view>
</view>
<view class="car-number-container" v-if="showKeyPop1">
<view class="plate-close" @click="closeKeyboard"><text class="plate-close-btn">关闭</text></view>
<view class="plate-popup-list">
<view class="plate-popup-item province-item" v-for="(item,index) in keyProvince1" :key="index" @click="tapKeyboard(item)">{{item}}</view>
</view>
<view class="plate-popup-list">
<view class="plate-popup-item province-item" v-for="(item,index) in keyProvince2" :key="index" @click="tapKeyboard(item)">{{item}}</view>
</view>
<view class="plate-popup-list">
<view class="plate-popup-item province-item" v-for="(item,index) in keyProvince3" :key="index" @click="tapKeyboard(item)">{{item}}</view>
</view>
<view class="plate-popup-list">
<view class="plate-popup-item province-item" v-for="(item,index) in keyProvince4" :key="index" @click="tapKeyboard(item)">{{item}}</view>
<!-- 删除 -->
<view class="plate-popup-item province-item del" @click="onPlateDelTap">
<image :src="deleteImgBase64" />
</view>
</view>
</view>
<view class="car-number-container" v-if="showKeyPop2">
<view class="plate-close" @click="closeKeyboard"><text class="plate-close-btn">关闭</text></view>
<view class="plate-popup-list">
<view class="plate-popup-item" :class="lockInput.includes(item)?'lock-item':''"
v-for="(item,index) in keyEnInput1" :key="index" @click="tapKeyboard(item)">
{{item}}
</view>
</view>
<view class="plate-popup-list">
<view class="plate-popup-item" :class="lockInput.includes(item)?'lock-item':''"
v-for="(item,index) in keyEnInput2" :key="index" @click="tapKeyboard(item)">
{{item}}
</view>
</view>
<view class="plate-popup-list">
<view class="plate-popup-item" :class="lockInput.includes(item)?'lock-item':''"
v-for="(item,index) in keyEnInput3" :key="index" @click="tapKeyboard(item)">
{{item}}
</view>
</view>
<view class="plate-popup-list">
<view class="plate-popup-item" :class="lockInput.includes(item)?'lock-item':''"
v-for="(item,index) in keyEnInput4" :key="index" @click="tapKeyboard(item)">
{{item}}
</view>
<!-- 删除 -->
<view class="plate-popup-item del" @click="onPlateDelTap">
<image :src="deleteImgBase64" />
</view>
</view>
</view>
</view>
</template>
<script>
export default {
name: 'car-number-input',
emits: ['numberInputResult'],
props: {
defaultStr:{
type: String,
default: ''
},
plateNum: {
type: String,
default: ''
},
// maxNum: {
// type: Number,
// default: 8
// },
},
data() {
return {
inputList:[" "," "," "," "," "," "," "," "],
curInput:-1,
maxNum:8,
showKeyPop1:false,
showKeyPop2:false,
keyProvince1: ['京', '津', '晋', '冀', '蒙', '辽', '吉', '黑', '沪'],
keyProvince2: ['苏', '浙', '皖', '闽', '赣', '鲁', '豫', '鄂', '湘'],
keyProvince3: ['粤', '桂', '琼', '渝', '川', '贵', '云', '藏'],
keyProvince4: ['陕', '甘', '青', '宁', '新', 'W' ],
keyEnInput1: ["1", "2", "3", "4", "5", "6", "7", "8", "9", "0"],
keyEnInput2: ["Q", "W", "E", "R", "T", "Y", "U", "P", "学", "军"],
keyEnInput3: ["A", "S", "D", "F", "G", "H", "J", "K", "L", "警"],
keyEnInput4: ["Z", "X", "C", "V", "B", "N", "M", "港", "澳"],
lockInput: ["1", "2", "3", "4", "5", "6", "7", "8", "9", "0"],
xnyImgBase64:"",
deleteImgBase64:""
}
},
watch: {
defaultStr(val) {
if(val != "" && val != null){
const valList = val.split("")
for (let i in valList) {
this.inputList[i] = valList[i]
}
}
},
curInput(val){
this.showOrHidePop(val)
this.keyEnInput2 = ["Q", "W", "E", "R", "T", "Y", "U", "O", "P", "军"]
switch(val){
case 1:
this.lockInput = ["1", "2", "3", "4", "5", "6", "7", "8", "9", "0", "学", "军", "警", "港", "澳"]
break
case 2:
this.lockInput = ["O", "学", "军", "警", "港", "澳"]
break
case 3:
this.lockInput = ["O", "学", "军", "警", "港", "澳"]
break
case 4:
this.lockInput = ["O", "学", "军", "警", "港", "澳"]
break
case 5:
this.lockInput = ["O", "学", "军", "警", "港", "澳"]
break
case 6:
this.lockInput = ["O"]
this.keyEnInput2 = ["Q", "W", "E", "R", "T", "Y", "U", "P", "学", "军"]
break
case 7:
this.lockInput = ["O", "学", "军", "警", "港", "澳"]
break
default:
this.lockInput = []
break
}
}
},
created() {
if(this.defaultStr != "" && this.defaultStr != null){
const valList = this.defaultStr.split("")
for (let i in valList) {
this.inputList[i] = valList[i]
}
}
},
methods: {
plateInput(e){
this.curInput = e
this.showOrHidePop(e)
},
showOrHidePop(val){
if(val == -1){
this.showKeyPop1 = false
this.showKeyPop2 = false
}else if(val == 0){
this.showKeyPop1 = true
this.showKeyPop2 = false
}else{
this.showKeyPop1 = false
this.showKeyPop2 = true
}
},
tapKeyboard(e){
if(this.lockInput.includes(e)){
return
}
this.inputList[this.curInput] = e
if(this.curInput < this.maxNum-2){
this.curInput++
}else{
this.curInput = -1
}
this.emitResult()
},
closeKeyboard(){
this.curInput = -1
},
onPlateDelTap(){
if(this.inputList[this.curInput] == " "){
this.curInput--
}
this.inputList[this.curInput] = " "
this.emitResult()
},
emitResult(){
const returnResult = this.inputList.join("")
this.$emit('numberInputResult', returnResult);
}
}
};
</script>
<style scoped lang="scss">
.car-input-container{
position: relative;
padding: 0 5px;
height: 44px;
.car-input-box{
display: inline-block;
width: 12.5%;
height: 44px;
vertical-align: middle;
.car-input-item{
position: relative;
border: 1px solid #E2E2E2;
border-radius: 10rpx;
height: 40px;
line-height: 40px;
width: 80%;
margin-left: 10%;
text-align: center;
font-size: 17px;
.sel-item-line{
position: absolute;
bottom: 3px;
left: 15%;
height: 2px;
background-color: #2979ff;
width: 70%;
}
.new-item-img{
position: absolute;
top: -6px;
left: 50%;
margin-left: -15px;
height: 13px;
width: 30px;
z-index: 9;
}
}
.sel-item{
color: #2979ff;
}
.last-item{
border: 1px solid #18bc37;
}
}
}
.car-number-container{
position: fixed;
z-index: 999;
bottom: 0;
left: 0;
width: 100%;
height: 254px;
background-color: #E3E2E7;
-webkit-box-shadow: 0 0 30upx rgba(0, 0, 0, 0.1);
box-shadow: 0 0 30upx rgba(0, 0, 0, 0.1);
overflow: hidden;
text-align: center;
.plate-close{
height: 40px;
line-height: 40px;
text-align: right;
background-color: #FFF;
.plate-close-btn{
font-size: 13.5px;
color: #555;
margin-right: 15px;
}
}
//键盘主体内容-单行
.plate-popup-list {
margin: 0 auto;
overflow: hidden;
display: inline-block;
display: table;
&:last-child {
margin-bottom: 2px;
}
}
//键盘主体内容-单个
.plate-popup-item {
float: left;
font-size: 16px;
width: 8vw;
margin: 0 1vw;
margin-top: 8px;
height: 40px;
line-height: 40px;
background: #FFFFFF;
border-radius: 5px;
color: #4A4A4A;
image {
width: 16px;
height: 16px;
margin: 12px auto;
}
}
.plate-popup-item:active{
background-color: #EAEAEA;
}
.province-item{
width: 8.8vw;
}
.lock-item{
color: #AAA;
}
}
</style>
使用方法
在父组件中引入并使用该车牌号选择组件:
1 | <template> |
组件结构
组件主要分为两部分:车牌号输入框和自定义键盘。
template 部分
1 | <template> |
script 部分
1 | <script> |
style 部分
1 | <style scoped lang="scss"> |
功能解析
1. 车牌格式与输入框
车牌号的输入格式为:
- 普通车牌:7位字符(1位省份简称 + 1位字母 + 5位字母或数字)
- 新能源车牌:8位字符(1位省份简称 + 1位字母 + 6位字母或数字,最后一位用绿色框标识)
在组件中,使用了一个长度为8的数组来存储车牌号各位的字符:
1 | inputList:[" "," "," "," "," "," "," "," "] |
并使用 v-for
指令遍历数组,创建对应数量的输入框:
1 | <view class="car-input-box" |
2. 自定义键盘设计
组件实现了两种不同的键盘:
- 省份简称键盘:用于输入车牌的第一位,包含全国各省市自治区的简称。
- 字母数字键盘:用于输入车牌的其他位置,包含字母、数字以及特殊标识(如”学”、”警”等)。
键盘布局采用了多行设计,使用者可以快速找到并点击所需的字符:
1 | <view class="plate-popup-list"> |
3. 输入规则与限制
车牌号的不同位置有不同的输入规则。例如:
- 第一位只能是省份简称
- 第二位通常只能是字母
- 其他位置可以是字母或数字
组件通过 watch
监听当前输入位置 curInput
的变化,动态设置当前位置可用和禁用的字符:
1 | watch: { |
禁用的字符在键盘上以灰色显示,并且不响应点击事件:
1 | <view class="plate-popup-item" :class="lockInput.includes(item)?'lock-item':''" |
4. 新能源车牌支持
新能源车牌比普通车牌多一位,且最后一位通常用绿色边框标识。组件通过条件渲染和特殊样式来实现这一特性:
1 | <view class="car-input-item" |
其中 last-item
类应用了绿色边框,而 new-item-img
则显示了新能源车标识。
5. 事件处理与交互
组件实现了多种交互方式:
- 点击输入框:激活对应位置的输入并弹出相应键盘
- 点击键盘字符:输入字符并自动跳转到下一位
- 点击删除按钮:删除当前位置的字符
- 点击关闭按钮:收起键盘
当输入完成或变更时,组件会通过事件向父组件传递当前的车牌号:
1 | emitResult(){ |
组件优化方向
- 支持自定义车牌长度:目前组件固定为8位,可以通过props动态设置长度以支持更多车牌类型。
- 增加输入验证:可以添加车牌格式的实时验证功能,提示用户输入是否符合规范。
- 改进键盘布局:可根据用户习惯优化键盘布局,提高输入效率。
- 增加动画效果:为键盘弹出/收起和输入框切换添加平滑过渡动画。
总结
通过本文,我们实现了一个功能完善的车牌号选择组件,支持普通车牌和新能源车牌的输入。该组件封装了复杂的交互逻辑和输入规则,为用户提供了便捷的车牌号输入体验。在实际项目中,可以根据具体需求对组件进行进一步定制和优化。
希望这个组件能够帮助到各位开发者,如有任何问题或建议,欢迎在评论区留言交流!
评论
匿名评论隐私政策
✅ 你无需删除空行,直接评论以获取最佳展示效果