339 lines
9.0 KiB
Vue
339 lines
9.0 KiB
Vue
<template>
|
||
<view class="u-form-item" :class="{'u-border-bottom': borderBottom, 'u-form-item__border-bottom--error': validateState === 'error' && showError('border-bottom')}">
|
||
<view class="u-form-item__body" :style="{
|
||
flexDirection: labelPosition == 'left' ? 'row' : 'column'
|
||
}">
|
||
<view class="u-form-item--left" :style="{
|
||
width: labelPosition == 'left' ? labelWidth + 'rpx' : '100%',
|
||
flex: `0 0 ${labelPosition == 'left' ? labelWidth + 'rpx' : '100%'}`,
|
||
marginBottom: labelPosition == 'left' ? 0 : '10rpx',
|
||
|
||
}">
|
||
<!-- 为了块对齐 -->
|
||
<view class="u-form-item--left__content">
|
||
<!-- nvue不支持伪元素before -->
|
||
<text v-if="isRequired" class="u-form-item--left__content--required">*</text>
|
||
<view class="u-form-item--left__content__icon" v-if="leftIcon">
|
||
<u-icon :name="leftIcon" :custom-style="leftIconStyle"></u-icon>
|
||
</view>
|
||
<view class="u-form-item--left__content__label" :style="[labelStyle, {
|
||
'justify-content': labelAlign == 'left' ? 'flex-star' : labelAlign == 'center' ? 'center' : 'flex-end'
|
||
}]">
|
||
{{label}}
|
||
</view>
|
||
</view>
|
||
</view>
|
||
<view class="u-form-item--right u-flex">
|
||
<view class="u-form-item--right__content">
|
||
<view class="u-form-item--right__content__slot ">
|
||
<slot />
|
||
</view>
|
||
<view class="u-form-item--right__content__icon u-flex" v-if="$slots.right || rightIcon">
|
||
<u-icon :custom-style="rightIconStyle" v-if="rightIcon" :name="rightIcon"></u-icon>
|
||
<slot name="right" />
|
||
</view>
|
||
</view>
|
||
</view>
|
||
</view>
|
||
<view class="u-form-item__message" v-if="validateState === 'error' && showError('message')" :style="{
|
||
paddingLeft: labelPosition == 'left' ? `${labelWidth}rpx` : '0',
|
||
}">{{validateMessage}}</view>
|
||
</view>
|
||
</template>
|
||
|
||
<script>
|
||
import Emitter from '../../libs/util/emitter.js';
|
||
import schema from '../../libs/util/async-validator';
|
||
// 去除警告信息
|
||
schema.warning = function(){};
|
||
export default {
|
||
name: 'u-form-item',
|
||
mixins: [Emitter],
|
||
inject: {
|
||
uForm: {
|
||
default() {
|
||
return null
|
||
}
|
||
}
|
||
},
|
||
props: {
|
||
// input的label提示语
|
||
label: {
|
||
type: String,
|
||
default: ''
|
||
},
|
||
// 绑定的值
|
||
prop: {
|
||
type: String,
|
||
default: ''
|
||
},
|
||
// 是否显示表单域的下划线边框
|
||
borderBottom: {
|
||
type: Boolean,
|
||
default: true
|
||
},
|
||
// label的位置,left-左边,top-上边
|
||
labelPosition: {
|
||
type: String,
|
||
default: 'left'
|
||
},
|
||
// label的宽度,单位rpx
|
||
labelWidth: {
|
||
type: [String, Number],
|
||
default: 90
|
||
},
|
||
// lable的样式,对象形式
|
||
labelStyle: {
|
||
type: Object,
|
||
default() {
|
||
return {}
|
||
}
|
||
},
|
||
// lable字体的对齐方式
|
||
labelAlign: {
|
||
type: String,
|
||
default: 'left'
|
||
},
|
||
// 右侧图标
|
||
rightIcon: {
|
||
type: String,
|
||
default: ''
|
||
},
|
||
// 左侧图标
|
||
leftIcon: {
|
||
type: String,
|
||
default: ''
|
||
},
|
||
// 左侧图标的样式
|
||
leftIconStyle: {
|
||
type: Object,
|
||
default() {
|
||
return {}
|
||
}
|
||
},
|
||
// 左侧图标的样式
|
||
rightIconStyle: {
|
||
type: Object,
|
||
default() {
|
||
return {}
|
||
}
|
||
}
|
||
},
|
||
data() {
|
||
return {
|
||
initialValue: '', // 存储的默认值
|
||
isRequired: false, // 是否必填
|
||
validateState: '', // 是否校验成功
|
||
validateMessage: '' ,// 校验失败的提示语
|
||
// 有错误时的提示方式,message-提示信息,border-如果input设置了边框,变成呈红色,
|
||
// border-bottom-下边框呈现红色,none-无提示
|
||
errorType: ['message']
|
||
};
|
||
},
|
||
watch: {
|
||
validateState(val) {
|
||
this.broadcastInputError();
|
||
},
|
||
// 监听u-form组件的errorType的变化
|
||
"uForm.errorType"(val) {
|
||
this.errorType = val;
|
||
this.broadcastInputError();
|
||
}
|
||
},
|
||
computed: {
|
||
fieldValue() {
|
||
return this.uForm.model[this.prop];
|
||
},
|
||
showError() {
|
||
return type => {
|
||
// 如果errorType数组中含有none,就不提示错误信息
|
||
if(this.errorType.indexOf('none') >= 0) return false;
|
||
else if(this.errorType.indexOf(type) >= 0) return true;
|
||
else return false;
|
||
}
|
||
}
|
||
},
|
||
methods: {
|
||
broadcastInputError() {
|
||
// 子组件发出事件,第三个参数为true或者false,true代表有错误
|
||
this.broadcast('u-input', 'on-form-item-error', this.validateState === 'error' && this.showError('border'));
|
||
},
|
||
// 判断是否需要required校验
|
||
setRules() {
|
||
let that = this;
|
||
// 从父组件u-form拿到当前u-form-item需要验证 的规则
|
||
let rules = this.getRules();
|
||
if (rules.length) {
|
||
this.isRequired = rules.some(rule => {
|
||
// 如果有必填项,就返回,没有的话,就是undefined
|
||
return rule.required;
|
||
});
|
||
}
|
||
|
||
// blur事件
|
||
this.$on('on-form-blur', that.onFieldBlur);
|
||
// change事件
|
||
this.$on('on-form-change', that.onFieldChange);
|
||
},
|
||
|
||
// 从u-form的rules属性中,取出当前u-form-item的校验规则
|
||
getRules() {
|
||
// 父组件的所有规则
|
||
let rules = this.uForm.rules;
|
||
rules = rules ? rules[this.prop] : [];
|
||
// 保证返回的是一个数组形式
|
||
return [].concat(rules || []);
|
||
},
|
||
|
||
// blur事件时进行表单校验
|
||
onFieldBlur() {
|
||
this.validation('blur');
|
||
},
|
||
|
||
// change事件进行表单校验
|
||
onFieldChange() {
|
||
this.validation('change');
|
||
},
|
||
|
||
// 过滤出符合要求的rule规则
|
||
getFilteredRule(triggerType = '') {
|
||
let rules = this.getRules();
|
||
// 整体验证表单时,triggerType为空字符串,此时返回所有规则进行验证
|
||
if(!triggerType) return rules;
|
||
// 历遍判断规则是否有对应的事件,比如blur,change触发等的事件
|
||
// return rules.filter(res => res.trigger == triggerType);
|
||
// 使用indexOf判断,是因为某些时候设置的验证规则的trigger属性可能为多个,比如"blur,change"
|
||
return rules.filter(res => !res.trigger || res.trigger.indexOf(triggerType) !== -1);
|
||
},
|
||
|
||
// 校验数据
|
||
validation(trigger, callback = () => {}) {
|
||
// blur和change是否有当前方式的校验规则
|
||
let rules = this.getFilteredRule(trigger);
|
||
// 判断是否有验证规则,如果没有规则,也调用回调方法,否则父组件u-form会因为
|
||
// 对count变量的统计错误而无法进入上一层的回调
|
||
if (!rules || rules.length === 0) {
|
||
return callback('');
|
||
}
|
||
// 设置当前的装填,标识为校验中
|
||
this.validateState = 'validating';
|
||
// 调用async-validator的方法
|
||
let validator = new schema({ [this.prop]: rules });
|
||
validator.validate({ [this.prop]: this.fieldValue }, { firstFields: true }, (errors, fields) => {
|
||
// 记录状态和报错信息
|
||
this.validateState = !errors ? 'success' : 'error';
|
||
this.validateMessage = errors ? errors[0].message : '';
|
||
// 调用回调方法
|
||
callback(this.validateMessage);
|
||
});
|
||
},
|
||
|
||
// 清空当前的u-form-item
|
||
resetField() {
|
||
this.uForm.model[this.prop] = this.initialValue;
|
||
// 设置为`success`状态,只是为了清空错误标记
|
||
this.validateState = 'success';
|
||
}
|
||
},
|
||
|
||
// 组件创建完成时,将当前实例保存到u-form中
|
||
mounted() {
|
||
// 如果没有传入prop,或者uForm为空(如果u-form-input单独使用,就不会有uForm注入),就不进行校验
|
||
if (!this.prop || this.uForm === null) return;
|
||
// 发出事件,让父组件将本实例加入到管理数组中
|
||
this.dispatch('u-form', 'on-form-item-add', this);
|
||
this.errorType = this.uForm.errorType;
|
||
// 设置初始值
|
||
this.initialValue = this.fieldValue;
|
||
// 添加表单校验
|
||
this.setRules();
|
||
},
|
||
|
||
// 组件销毁前,将实例从 Form 的缓存中移除
|
||
beforeDestroy() {
|
||
this.dispatch('u-form', 'on-form-item-remove', this);
|
||
}
|
||
};
|
||
</script>
|
||
|
||
<style lang="scss" scoped>
|
||
.u-form-item {
|
||
display: flex;
|
||
// align-items: flex-start;
|
||
padding: 20rpx 0;
|
||
font-size: 28rpx;
|
||
color: $u-main-color;
|
||
box-sizing: border-box;
|
||
line-height: $u-form-item-height;
|
||
flex-direction: column;
|
||
|
||
&__border-bottom--error:after {
|
||
border-color: $u-type-error;
|
||
}
|
||
|
||
&__body {
|
||
display: flex;
|
||
}
|
||
|
||
&--left {
|
||
display: flex;
|
||
align-items: center;
|
||
|
||
&__content {
|
||
position: relative;
|
||
display: flex;
|
||
align-items: center;
|
||
padding-right: 10rpx;
|
||
flex: 1;
|
||
|
||
&__icon {
|
||
margin-right: 8rpx;
|
||
}
|
||
|
||
&--required {
|
||
position: absolute;
|
||
left: -16rpx;
|
||
vertical-align: middle;
|
||
color: $u-type-error;
|
||
padding-top: 6rpx;
|
||
}
|
||
|
||
&__label {
|
||
display: flex;
|
||
align-items: center;
|
||
flex: 1;
|
||
}
|
||
}
|
||
}
|
||
|
||
&--right {
|
||
flex: 1;
|
||
|
||
&__content {
|
||
display: flex;
|
||
align-items: center;
|
||
flex: 1;
|
||
|
||
&__slot {
|
||
flex: 1;
|
||
/* #ifndef MP */
|
||
display: flex;
|
||
align-items: center;
|
||
/* #endif */
|
||
}
|
||
|
||
&__icon {
|
||
margin-left: 10rpx;
|
||
}
|
||
}
|
||
}
|
||
|
||
&__message {
|
||
font-size: 24rpx;
|
||
line-height: 24rpx;
|
||
color: $u-type-error;
|
||
margin-top: 12rpx;
|
||
}
|
||
}
|
||
</style>
|