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

使用 UniApp 实现车牌号选择组件(含键盘和新能源支持)

在移动应用开发中,车牌号输入是一个常见需求,尤其是在与交通、停车、违章查询等相关的应用中。与普通文本输入不同,车牌号输入有着特定的规则和格式,需要专门的键盘和交互设计。本文将详细介绍如何使用 UniApp 框架实现一个功能完善的车牌号选择组件,支持常规车牌和新能源车牌。

效果预览

车牌号选择组件效果图
车牌号选择组件效果图
车牌号选择组件效果图

功能特点

  • 支持普通车牌和新能源车牌输入
  • 自定义键盘,分为省份简称键盘和字母数字键盘
  • 智能切换键盘类型
  • 支持车牌号规则限制(如第二位只能输入字母或数字等)
  • 集成删除和关闭功能
  • 支持初始化已有车牌号
  • 优雅的样式和流畅的交互体验

实现思路

  1. 创建车牌号输入框组件
  2. 实现省份简称键盘
  3. 实现字母数字键盘
  4. 添加车牌号输入规则和限制
  5. 实现键盘切换和交互逻辑

代码实现

完整代码

  • 在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
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
<template>
<view>
<car-number-input @numberInputResult="getPlateNumber" :defaultStr="plateNumber"></car-number-input>
<view class="result">
当前车牌号:{{plateNumber}}
</view>
</view>
</template>

<script>
import carNumberInput from '@/components/car-number-input/car-number-input.vue'

export default {
components: {
carNumberInput
},
data() {
return {
plateNumber: '京A12345'
}
},
methods: {
getPlateNumber(val) {
this.plateNumber = val.trim();
console.log('当前车牌号:', this.plateNumber);
}
}
}
</script>

组件结构

组件主要分为两部分:车牌号输入框和自定义键盘。

template 部分

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
<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 部分

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>
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:"data:image/png;base64,...", // 新能源标识图片base64编码(已省略)
deleteImgBase64:"data:image/png;base64,..." // 删除按钮图片base64编码(已省略)
}
},
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 部分

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
<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. 车牌格式与输入框

车牌号的输入格式为:

  • 普通车牌:7位字符(1位省份简称 + 1位字母 + 5位字母或数字)
  • 新能源车牌:8位字符(1位省份简称 + 1位字母 + 6位字母或数字,最后一位用绿色框标识)

在组件中,使用了一个长度为8的数组来存储车牌号各位的字符:

1
inputList:[" "," "," "," "," "," "," "," "]

并使用 v-for 指令遍历数组,创建对应数量的输入框:

1
2
3
4
<view class="car-input-box" 
v-for="(item,index) in inputList" :key="index" @click="plateInput(index)">
<!-- 内容省略 -->
</view>

2. 自定义键盘设计

组件实现了两种不同的键盘:

  1. 省份简称键盘:用于输入车牌的第一位,包含全国各省市自治区的简称。
  2. 字母数字键盘:用于输入车牌的其他位置,包含字母、数字以及特殊标识(如”学”、”警”等)。

键盘布局采用了多行设计,使用者可以快速找到并点击所需的字符:

1
2
3
<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>

3. 输入规则与限制

车牌号的不同位置有不同的输入规则。例如:

  • 第一位只能是省份简称
  • 第二位通常只能是字母
  • 其他位置可以是字母或数字

组件通过 watch 监听当前输入位置 curInput 的变化,动态设置当前位置可用和禁用的字符:

1
2
3
4
5
6
7
8
9
10
11
12
13
watch: {
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
// 其他位置的规则...
}
}
}

禁用的字符在键盘上以灰色显示,并且不响应点击事件:

1
2
3
4
<view class="plate-popup-item" :class="lockInput.includes(item)?'lock-item':''" 
v-for="(item,index) in keyEnInput1" :key="index" @click="tapKeyboard(item)">
{{item}}
</view>

4. 新能源车牌支持

新能源车牌比普通车牌多一位,且最后一位通常用绿色边框标识。组件通过条件渲染和特殊样式来实现这一特性:

1
2
3
4
5
6
<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>

其中 last-item 类应用了绿色边框,而 new-item-img 则显示了新能源车标识。

5. 事件处理与交互

组件实现了多种交互方式:

  • 点击输入框:激活对应位置的输入并弹出相应键盘
  • 点击键盘字符:输入字符并自动跳转到下一位
  • 点击删除按钮:删除当前位置的字符
  • 点击关闭按钮:收起键盘

当输入完成或变更时,组件会通过事件向父组件传递当前的车牌号:

1
2
3
4
emitResult(){
const returnResult = this.inputList.join("")
this.$emit('numberInputResult', returnResult);
}

组件优化方向

  1. 支持自定义车牌长度:目前组件固定为8位,可以通过props动态设置长度以支持更多车牌类型。
  2. 增加输入验证:可以添加车牌格式的实时验证功能,提示用户输入是否符合规范。
  3. 改进键盘布局:可根据用户习惯优化键盘布局,提高输入效率。
  4. 增加动画效果:为键盘弹出/收起和输入框切换添加平滑过渡动画。

总结

通过本文,我们实现了一个功能完善的车牌号选择组件,支持普通车牌和新能源车牌的输入。该组件封装了复杂的交互逻辑和输入规则,为用户提供了便捷的车牌号输入体验。在实际项目中,可以根据具体需求对组件进行进一步定制和优化。

希望这个组件能够帮助到各位开发者,如有任何问题或建议,欢迎在评论区留言交流!