uniapp实现六个格子输入框

本文介绍两种方法

第一种:用一个input加上六个span

<template>
	<view class="content">
		  <view class="pc_in">
				<view class="pc_on">
					<span class="g_hx"></span>
					<span class="g_hx"></span>
					<span class="g_hx"></span>
					<span class="g_hx"></span>
					<span class="g_hx"></span>
					<span class="g_hx"></span>
				</view>
				<p class="pp">
					<input :class="smsCode.length==6 ? 'input current' : 'input'" type="number" maxlength="6" v-model="smsCode" />
				</p>
			</view>
	</view>
</template>

<script>
	export default {
		data() {
			return {
				title: 'Hello'
			}
		},
		onLoad() {

		},
		methods: {

		}
	}
</script>

<style>
.pc_in {
		position: relative;
		height: 60upx;
		line-height: 60upx;
		width: 70%;
		margin: 30upx auto 0;
	}
	.pc_on {
		display: flex;
		height: 60upx;
		line-height: 60upx;
	}
	.g_hx {
		flex: 1;
		border: #E6E8ED solid 4upx;
		border-radius: 10upx;
		margin: 0 10upx;
	}
	.pp {
		position: absolute;
		width: 120%;
		height: 60upx;
		line-height: 60upx;
		top: 0;
		left: 0;
		background: none;
		overflow: hidden;
	}
	.input {
		font-family: '';
		font-size: 30upx;
		width: 120%;
		height: 60upx;
		line-height: 60upx;
		letter-spacing: 70upx;
		background: none;
		text-align: left;
		text-indent: 34upx;
	}
	.input.current{
		caret-color: transparent;
	}
</style>

效果展示

在这里插入图片描述
在这里插入图片描述
第二种:不用input,自定义键盘
首先创建number-keyboard.vue组件

<template>
  <view :class="['KeyboarBody','bottomMove', 'bodMove', bodMove]" v-if="KeyboarHid">
    <view class="KeyboarBx">
      <view v-for="(num , index) in keyboardNum " :key='index'@tap.stop="" @touchstart="clickBoard(num[0])" hover-class="hover"
        :hover-stay-time='20' class="keyboar">
		<view class="">
			<view>{{num[0]}}</view>
			<view style="font-size: 16upx;">{{num[1]}}</view>
		</view>
      </view>
      <view @tap.stop="" class="keyboar empty" > </view>
      <view @tap.stop="" @touchstart="clickBoard('0')" hover-class="hover" :hover-stay-time='20' class="keyboar zero">0</view>
      <view @tap.stop="" @touchstart="deleteKeyboar()" class="keyboar keyboarflex" :hover-stay-time='20' hover-class="hover">
        <view class="keyboarDel"></view>
      </view>
    </view>
  </view>
</template>

<script>
export default {
	props: {
		//限制输入框的长度 空值不限制
		psdLength: {
			type: [Number, String],
			default: ''
		},
		//键盘码
		keyboardNum: {
			type: [Array, Object],
			default: () => {
				return [[1], [2,'ABC'], [3,'DEF'], [4,'GHI'], [5,'JKL'], [6,'MNO'], [7,'PQRS'], [8,'TUV'], [9,'WXYZ']]
			},
			
		},
		value: {
			type: String,
			default: ''
		}
    },
    data() {
		return {
			bodMove: '',
			password: '', //要返回的结果
			iptNum: [], //输入的内容
			KeyboarHid: false, //键盘影藏和显示
		}
	},
	watch: {
		iptNum(newVal, oldVal) {
			this.$emit('input', newVal.join(''))
		},
		value(newVal, oldVal) {
			this.iptNum = newVal.split('')
		}
	},
    methods: {
		open() {
			this.KeyboarHid = true;
			if (this.tabBar) {
				uni.hideTabBar()
			}
			this.$nextTick(() => {
				setTimeout(() => {
					this.bodMove = 'moveShow'
				}, 30)
			})
		},
		close() {
			if (this.tabBar) {
				uni.showTabBar()
			}
			this.bodMove = '';
			this.$nextTick(() => {
				setTimeout(() => {
					this.KeyboarHid = false
				}, 300)
			})
		},
		// 点击键盘
		clickBoard(num) {
			if (num == '') return;
			let iptNum = this.iptNum.filter(item => item != '');
			//判断是否限制长度
			if (this.psdLength != '') {
				if (iptNum.length >= this.psdLength) {
					return
				};
				this.iptNum.push(num);
			} else {
				this.iptNum.push(num);
			}
		},
		//重置 清空
		reset() {
			this.iptNum = [];
		},
		//删除
		deleteKeyboar() {
			this.iptNum.pop();
		}
	}
}
</script>

<style scoped>
  .bodMove {
    transition: all .3s
  }

  .bottomMove {
    bottom: 0;
    left: 0;
    width: 100%;
    transform: translateY(100%)
  }

  .moveShow {
    transform: translateY(0)
  }

  .KeyboarBody {
    position: fixed;
    bottom: 70upx;
    left: 0;
    right: 0;
    z-index: 99;
    background-color: #fcfcfc;
	width: 90%;
	margin: 0 auto;
  }

  .KeyboarBx {
    display: flex;
    flex-wrap: wrap;
    text-align: center;
    padding: 10rpx 6rpx 0rpx 6rpx;
    margin-left: -12rpx;
  }

  .keyboar {
    width: 30%;
    flex-grow: 1;
    padding: 2upx 20upx;
    font-size: 40rpx;
    background-color: #FFFFFF;
    border-radius: 10rpx;
    margin-left: 12rpx;
    margin-bottom: 16rpx;
	min-height: 74upx;
	box-shadow: 0px 1px 1px #000;
  }
  .keyboar.zero{
	  line-height: 74upx;
  }

  .dian {
    margin-top: -10rpx;
  }

  .keyboarBtn {
    background: linear-gradient(45deg, rgba(242, 131, 9, 1) 0%, rgba(230, 36, 15, 1) 100%);
    color: #fff;
  }

  .hover {
    background: #ebedf0;
  }


  .bot {
    bottom: 0;
  }

  .empty {
    box-shadow: none;
	background-color: #fcfcfc;
  }

  .dowmImgBx {
    display: flex;
    justify-content: center;
    border-top: 1rpx solid A1A1A1;
  }

  .dowmImg {
    width: 35rpx;
    height: 35rpx;
    margin-bottom: 10rpx;
    background: url('data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAACAAAAAgCAYAAABzenr0AAACT0lEQVRYR+2UPWgTYRjH/8/dpdZCpVhU/BoEB8FBEBFt7tJe7i5CbAXhrg6CQ8HBQXAQHBzq5CA4CA4OgoNgr0GQIhQTUkJqCPEDBEEQBAdBQSoVlIakXvLIofmopMnlizjkxuN9nt/v/b/v+xB6/FGP+egL9BPoJ/D/JLC8/Hp33sl9AXDOCMqRbsyH6NLzGwLR7BapuDMQCKy4jHICi5nMNinrpAk4DOC8EZQfdVKiBGemxyHNb5V6bziCaDRxSPT5bAaOCAJd0Cb8DzshUYIT4cnqit+anqZCTQH357NE+qDIhXmAjoJpxtD8D9qRqOwcT538DyscDuer+9W8hNFo8oDok2wGHxcIFzVVvt+KRAkOYDG3NmhNTR3L/ttn01eQTL7cv+6sz4NwkhmXQpp8rxmJ8s6B2NAATFmWf9aqr/sMY7HUHsEHmxkKGJcNTb7rRaKyc1rKZ4vW5KTyfbO6hnMgHo/vYmHQBjBBxFd0VblTT6IC56REjqmq6rd66xsKuMXx+ItRFn/ZYOhMfDWkKrdrNa068xQVJVPXT3xtlJgnAbdJIpEYcViyATpFoGt60H+runkVPCOwYGra2OdG8A2DyMvihYXU8NAwzQF8moDrelC+6daV4YRXcNg0DOWTl35NC7gF6XR661qu6N6JMwzMMrPgjlcw3hCzqevKR6/wlgTcokjk3cDI6KpNRGf/wt6yWDBD4+MfmoG3LPBHIiJu37F3jsH7JAgzqjr2vll4WwKtwJoeRJ2CtD0HuinieQ50S6Iv0E+g5wn8Bo+vyyFXaYw2AAAAAElFTkSuQmCC');
    background-repeat: no-repeat;
    background-position: center center;
    background-size: 100%;
  }

  .keyboarDel {
    width: 50rpx;
    height: 36rpx;
    margin-bottom: 10rpx;
    background-image: url('data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABgAAAASCAYAAABB7B6eAAABMklEQVQ4T62UbVFDMRBFTx0gAQdFAihocQAKAAc4aKugRUFxUHAACgAHdUDnzLzthL68lx0gP/NxT3LvbiaMj0vgDpg39pXLn8AGWAH7ycjBG2ANvALPwFsCcgZ4Kc9+AFdDgBB/6jYntH9sEeRLljXAX8WDtARmp4D/EhdibtsS0BI/BxbAreEVhmiHWT10tsSSWewC0BL3kEIvwLfhdRDndoA6CpbgIyAjHrcqIddaMCDu/iNAqmUoKDMCMgXeKzfvWSTAm+htZoQtF11vhF2nZ3sW2X0tSOm5VeLLy0xKSDXkMUgt0Frw1SpyMsIeglimj8B9pUxtKtfs3kFABpLJKPb0Gi0WWi/JQnzVvPXZZYKvAa0wG3CV+a7t3t9811/2yBggutFQZ1lfAIV9uRbtD3WVXOgkzB+KAAAAAElFTkSuQmCC');
    background-repeat: no-repeat;
    background-position: center center;
    background-size: 80%;
    margin-top: 11rpx;
  }

  .keyboarflex {
    display: flex;
    justify-content: center;
    align-items: center;
	box-shadow: none;
  }
</style>

在页面中引入

<template>
	<view class="page container" @click="KeyboarClose">
			<view class="right" @tap.stop='KeyboarOpen' data-key='1'>
				<view class="psdIptBx">
				  <block v-for="(item , index) in 6" :key='index'>
					<view class="psdTtem">
					  <text class="psdTtemTxt">
						<text>{{valArr.val[index]}}</text>
					  </text>
					  <text v-if="valArr.val.length == index && isFoucs" class="focus_move">|</text>
					</view>
				  </block>
				</view>
			</view>
			<numberKeyboard tabBar ref='KeyboarHid' @input='clickInput' :psdLength='valArr.length'></numberKeyboard>
	</view>
</template>
<script>
	import numberKeyboard from '@/components/input-number-keyboard/number-keyboard.vue'
	export default{
		components: {
			numberKeyboard
		},
		data(){
			return {
				smsCode: '',
				valArr: {
					val: '',
					length: 6
				},
				isFoucs: false
			}
		},
		methods: {
			//打开键盘
			KeyboarOpen(e) {
				this.$refs.KeyboarHid.open()
				this.isFoucs = true
			},
			//输入的值
			clickInput(val) {
				this.valArr.val = val
				this.smsCode = val
			},
			KeyboarClose(e) {
				this.$refs.KeyboarHid.close();
			}
		}
	}
</script>

<style lang='scss' scoped>
	@import 'login-new.scss';
	.psdIptBx {
	  display: flex;
	  width: 70%;
	  text-align: center;
	  margin: 20upx auto 0;
	  font-family: '';
	}
	
	.psdTtem {
		flex: 1;
		height: 60rpx;
		border: 2px solid #ccc;
		line-height: 1;
		border-radius: 10upx;
		margin: 0 10upx;
	}
	
	.psdTtemTxt {
	  text-align: center;
	  line-height: 54upx;
	  font-size: 34upx;
	}
	
	.focus_move {
	  color: #666;
	  animation: focus 1s infinite;
	}
	
	@keyframes focus {
	  from {
	    opacity: 1;
	  }
	
	  to {
	    opacity: 0;
	  }
	}
</style>

效果展示
在这里插入图片描述

  • 3
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值