1、前言

  本文是前文《Vue Element-ui表单校验规则,你掌握了哪些?》针对多字段联合校验的典型应用。

  在修改密码时,一般需要确认两次密码一致,涉及2个属性字段。类似的涉及2个属性字段的情况有:

  • 日期时间范围,如果两者都有值,则要求:结束时间>=开始时间。
  • 数量关系:数量下限<=数量上限。

  特点是两个属性值都是可变的。本文以校验两次密码的一致性应用,给出两个可变属性值的字段之间的联合校验的典型解决方案。

2、方案实现

2.1、实现代码

  先给出表单的代码:

<template>
<div id="contentwrapper">
<h5 class="heading" align=left>用户管理 / 修改密码</h5>
<!-- 分隔线 -->
<el-divider></el-divider>
<el-form ref="form" :model="form" :rules="rules" label-width="100px">
<el-form-item label="原 密 码:" prop="oldPasswd">
<el-input v-model="form.oldPasswd" :type="password">
<!-- input中加图标必须要有slot="suffix"属性,不然无法显示图标 -->
<i slot="suffix" :class="icon" @click="showPassword"></i>
</el-input>
</el-form-item>
<el-form-item label="新 密 码:" prop="newPasswd">
<el-input v-model="form.newPasswd" :type="password">
<i slot="suffix" :class="icon" @click="showPassword"></i>
</el-input>
</el-form-item>
<el-form-item label="确认密码:" prop="confirmPasswd">
<el-input v-model="form.confirmPasswd" :type="password">
<i slot="suffix" :class="icon" @click="showPassword"></i>
</el-input>
</el-form-item>
<el-form-item>
<el-button type="primary" style="width:160px" size="small" @click="submit()">确定</el-button>
<el-button type="primary" style="width:160px" size="small" @click="cancel()">取消</el-button>
</el-form-item>
</el-form>
</div>
</template>
<script>
import {passwordValidator} from '@/common/validator.js'
export default {
data() {
// 比较两次密码是否相同
const comparePasswdValidator = (rule, value, callback) =>{
// 获取获取值的方法
var getvaluesMethod = rule.getValuesMethod;
// 调用getvaluesMethod方法,获取对象值
var formData = getvaluesMethod(); // 有一个为空,可能还没有输入值,此时不比较
if (formData.newPasswd == '' || formData.confirmPasswd == ''){
return callback();
} // ===========================================================
// 比较两次密码 // 两个都有值,比较
if (formData.newPasswd == formData.confirmPasswd){
// 新密码与确认密码一致
// 先清除两个密码的校验告警提示,目前是清除另一个密码的不一致的提示
this.$refs['form'].clearValidate(['newPasswd','confirmPasswd']);
callback();
}else{
callback(new Error('两次密码不一致'));
}
} return {
form : {
oldPasswd : '',
newPasswd : '',
confirmPasswd : ''
},
//用于改变Input类型
password:"password",
//用于更换Input中的图标
icon:"el-input__icon el-icon-view",
// 校验规则配置
rules: {
oldPasswd : [
{required: true, message: "密码不能为空", trigger: 'blur'}
],
newPasswd : [
{required: true, message: "新密码不能为空", trigger: 'blur'},
{min: 6, max: 18, message: "新密码6-18位", trigger: 'blur'},
{validator: passwordValidator, trigger: 'blur'},
{validator: comparePasswdValidator,trigger: 'blur','getValuesMethod':this.getValuesMethod},
],
confirmPasswd : [
{required: true, message: "确认密码不能为空", trigger: 'blur'},
{min: 6, max: 18, message: "确认密码6-18位", trigger: 'blur'},
{validator: passwordValidator, trigger: 'blur'},
{validator: comparePasswdValidator,trigger: 'blur','getValuesMethod':this.getValuesMethod},
],
},
}
},
methods: {
// 获取值的方法,为所有需要多字段联合校验的校验器使用
getValuesMethod(){
return this.form;
}, // 密码的隐藏和显示
showPassword(){
//点击图标是密码隐藏或显示
if( this.password=="text"){
this.password="password";
//更换图标
this.icon="el-input__icon el-icon-view";
}else {
this.password="text";
this.icon="el-input__icon el-icon-stopwatch";
}
}, // 提交
submit(){
let _this = this;
this.$refs['form'].validate(valid => {
// 验证通过为true,有一个不通过就是false
if (valid) {
_this.instance.changePasswd(_this.$baseUrl,_this.form).then(res => {
if (res.data.code == _this.global.SucessRequstCode){
// 提示修改成功
alert("重新设置密码已成功!");
// 跳转到首页
this.$router.push({
path: '/home',
});
}else{
if (!_this.commonFuncs.isInterceptorCode(res.data.code)){
alert(res.data.message);
}
}
}).catch(error => {
console.log(error);
});
}
})
}, // 取消
cancel(){
// 跳转到首页
this.$router.push({
path: '/home',
});
}
}
}
</script>

  导入的外部校验器passwordValidator在/src/common/validator.js文件中,代码如下:

/* 密码校验 */
export function passwordValidator(rule, value, callback) {
const reg =/^[_a-zA-Z0-9@.#%&*!\-\$^]+$/;
if(value == '' || value == undefined || value == null){
callback();
} else {
if (!reg.test(value)){
callback(new Error('密码由英文字母、数字以及下列字符组成:@.#%&*!_-$^'));
} else {
callback();
}
}
}

2.2、代码说明

2.2.1、规则配置:

    // 校验规则配置
rules: {
oldPasswd : [
{required: true, message: "密码不能为空", trigger: 'blur'}
],
newPasswd : [
{required: true, message: "新密码不能为空", trigger: 'blur'},
{min: 6, max: 18, message: "新密码6-18位", trigger: 'blur'},
{validator: passwordValidator, trigger: 'blur'},
{validator: comparePasswdValidator,trigger: 'blur','getValuesMethod':this.getValuesMethod},
],
confirmPasswd : [
{required: true, message: "确认密码不能为空", trigger: 'blur'},
{min: 6, max: 18, message: "确认密码6-18位", trigger: 'blur'},
{validator: passwordValidator, trigger: 'blur'},
{validator: comparePasswdValidator,trigger: 'blur','getValuesMethod':this.getValuesMethod},
],
},

  重点是newPasswd和confirmPasswd属性,两个配置了相同的规则集,校验规则都为:

  • 值不能为空。
  • 长度为6-18位。
  • 符合密码校验器passwordValidator的规则,即密码由英文字母、数字以及下列字符组成:@.#%&*!_-$^。
  • 比较密码校验器comparePasswdValidator,这个校验器添加了一个自定义属性getValuesMethod,属性值为this的getValuesMethod方法,注意是方法,不是方法名。该条规则,要求methods中有一个getValuesMethod方法,并且实现comparePasswdValidator校验器。

  这些规则一起作用,所有规则都通过校验,属性校验才通过,并且检测次序按照数组的先后次序执行。

  校验规则中,newPasswd和confirmPasswd属性,都配置相同的comparePasswdValidator,是因为两个字段属性值都是可变的。comparePasswdValidator排在规则的最后一条,即需要先满足前面的校验规则。

2.2.2、getValuesMethod方法

  // 获取值的方法,为所有需要多字段联合校验的校验器使用
getValuesMethod(){
return this.form;
},

  getValuesMethod方法,返回data中form数据对象。这是一个很犀利的操作,相当于提供了全局的数据探针,可以在校验器中访问data的form数据对象,并且由于form的v-modal模型,确保数据的实时性,即无需担心获取不到其它属性的最新取值。getValuesMethod方法,提供了数据绑定的另类思路。

2.2.3、comparePasswdValidator校验器

  // 比较两次密码是否相同
const comparePasswdValidator = (rule, value, callback) =>{
// 获取获取值的方法
var getvaluesMethod = rule.getValuesMethod;
// 调用getvaluesMethod方法,获取对象值
var formData = getvaluesMethod(); // 有一个为空,可能还没有输入值,此时不比较
if (formData.newPasswd == '' || formData.confirmPasswd == ''){
return callback();
} // ===========================================================
// 比较两次密码 // 两个都有值,比较
if (formData.newPasswd == formData.confirmPasswd){
// 新密码与确认密码一致
// 先清除两个密码的校验告警提示,目前是清除另一个密码的不一致的提示
this.$refs['form'].clearValidate(['newPasswd','confirmPasswd']);
callback();
}else{
callback(new Error('两次密码不一致'));
}
}

  因为配置的自定义规则属性'getValuesMethod'是一个指向this.getValuesMethod的方法,因此该属性可以看作方法来调用:

    // 获取获取值的方法
var getvaluesMethod = rule.getValuesMethod;
// 调用getvaluesMethod方法,获取对象值
var formData = getvaluesMethod();

  方法属性的调用结果,返回了指向this.form的数据对象,于是就可以随意访问该对象的属性。

  在比较两者之前,如果发现有一者为空,则返回。因为当前属性输入值之后,对端属性可能还没有输入值,此时不应该比较。

    // 有一个为空,可能还没有输入值,此时不比较
if (formData.newPasswd == '' || formData.confirmPasswd == ''){
return callback();
}

  两次密码比较:

    // ===========================================================
// 比较两次密码 // 两个都有值,比较
if (formData.newPasswd == formData.confirmPasswd){
// 新密码与确认密码一致
// 先清除两个密码的校验告警提示,目前是清除另一个密码的不一致的提示
this.$refs['form'].clearValidate(['newPasswd','confirmPasswd']);
callback();
}else{
callback(new Error('两次密码不一致'));
}

  两次密码比较,如果两个密码一致,就输出告警提示。如果一致,则需要先清除对端的告警提示,因为此时对端可能有”两次密码不一致“的提示。

      // 先清除两个密码的校验告警提示,目前是清除另一个密码的不一致的提示
this.$refs['form'].clearValidate(['newPasswd','confirmPasswd']);

  clearValidate方法,是element-form的方法,作用是清除一个或多个校验规则属性的异常提示。

  这时,是否会发生”误杀“情况呢?即清除操作将对端的其它异常提示也清除了。考虑到此时两次密码相同,且两者校验规则是相同的,由于此校验规则排在最后,从检测次序来说,是最后执行的,也就是说,执行到本校验器时,其它校验都通过了。因此,这种”误杀“的情况不会发生。实际执行效果也是如此。

  另外,需要注意的,此时不能用下列代码代替上面clearValidate调用语句:

          // 执行对端的校验
if (rule.field == 'newPasswd')
{
// 如果当前属性为newPasswd
this.$refs['form'].validateField('confirmPasswd');
}else{
this.$refs['form'].validateField('newPasswd');
}

  因为,此时正在执行校验,再调用对端校验,会导致对端调用comparePasswdValidator,而对端校验结果发现两次密码一致,将再次调用对端(对端的对端,即本身)校验,于是死循环了,导致调用堆栈溢出。因此,校验器代码中,尽量不要再调用validateField方法。

2.3、校验效果

  下面是一些效果图:

  初始状态:

  修改确认密码,离开输入焦点:

  进入新密码输入框,不输入,离开输入焦点:

  修改确认密码,删除尾部的字符"8",离开输入焦点,此时又回到了初始状态。

Vue 两个字段联合校验典型例子--修改密码的更多相关文章

  1. 1) 上传多张图片时 ,对 $_FILES 的处理. upload ; 2)fileinput 上传多张图片. 3) 修改,删除的时候删除原来的资源,图片 update, delete , 删除 4)生成器中两个字段上传图片的时候,要修改生成器生成的代码

    1上传多张图片, 要对 $_FILES进行 重新处理. //添加 public function addCourseAlbumAction() { $CourseAlbumModel = new Co ...

  2. 两个字段联合约束(mysql)

    联合约束:ALTER TABLE `lywl_provider_package` ADD unique(providerId,packCode) 给一个表建唯一约束

  3. 解决多字段联合逻辑校验问题【享学Spring MVC】

    每篇一句 不要像祥林嫂一样,天天抱怨着生活,日日思考着辞职.得罪点说一句:"沦落"到要跟这样的人共事工作,难道自己身上就没有原因? 前言 本以为洋洋洒洒的把Java/Spring数 ...

  4. Python学习/复习神器-->各种方法/技巧在哪用和典型例子(一)

    就我个人在学习Python的过程中,经常会出现学习了新方法后,如果隔上几天不用,就忘了的情况,或者刚学习的更好的方法没有得到应用,还是沿用已有的方法,这样很不利于学习和掌握新姿势,从而拉长学习时间,增 ...

  5. Anaconda安装Graphviz, mac下Graphviz安装, pcharm中调用pycharm, Graphviz典型例子

    mac下的Graphviz安装及使用 2017年10月13日 13:30:07 阅读数:7495 一.安装 Graphviz http://www.graphviz.org/ mac用户建议直接用ho ...

  6. SQL 建立多个字段唯一性校验

    由于在做压力测试,同一时间占用的问题. 两个用户同时下同一时间的订单,需要增加校验,第一个能保存的用户保存,第二个就不能让保存了. 问题是通过代码,怎么都做不到毫秒级校验,所以解决办法就只能是通过数据 ...

  7. Python--方法/技巧在哪用的典型例子

    就我个人在学习Python的过程中,经常会出现学习了新方法后,如果隔上几天不用,就忘了的情况,或者刚学习的更好的方法没有得到应用,还是沿用已有的方法,这样很不利于学习和掌握新姿势,从而拉长学习时间,增 ...

  8. RENIX报文两个字段嵌套变化——网络测试仪实操

    RENIX软件如何实现报文中两个字段嵌套变化,以下为您实操讲解详细步骤. 1.打开Renix软件,连接机框并预约测试端口: 2.创建一条RAW流量(Binding流量也可以,这里用RAW流做例子) 3 ...

  9. [每日电路图] 10、两种MOS管的典型开关电路

    下图是两种MOS管的典型应用:其中第一种NMOS管为高电平导通,低电平截断,Drain端接后面电路的接地端:第二种为PMOS管典型开关电路,为高电平断开,低电平导通,Drain端接后面电路的VCC端. ...

随机推荐

  1. k8s管理机密信息(12)

    一.启动应用安全信息的保护 1.Secret介绍 应用启动过程中可能需要一些敏感信息,比如访问数据库的用户名密码或者秘钥.将这些信息直接保存在容器镜像中显然不妥,Kubernetes 提供的解决方案是 ...

  2. 戴尔 R730xd 服务器更改管理口密码 图文教程

    一.开机根据提示按F2进入配置界面 - 选择中间的iDRAC Setting选项,回车确认 二.进入之后选择 user configuration 选项 三.在change password 处键入新 ...

  3. KVM虚拟化存储管理(4)

    一.KVM存储虚拟化介绍 KVM 的存储虚拟化是通过存储池(Storage Pool)和卷(Volume)来管理的. Storage Pool 是宿主机上可以看到的一片存储空间,可以是多种型: Vol ...

  4. Linux服务之nginx服务篇三(反向代理、负载均衡)

    一.Nginx实现反向代理 概念 反向代理:在收到客户端请求之后,会修目标IP地址和端口 正向代理:在收到客户端请求之后,会修源IP地址和端口 上游服务器:代理服务器后端的哪些真正给客户端提供服务的节 ...

  5. rpm命令的简介-(转自jb51.net )

    在Linux操作系统中,有一个系统软件包,它的功能类似于Windows里面的"添加/删除程序",但是功能又比"添加/删除程序"强很多,它就是Red Hat Pa ...

  6. 【三】Kubernetes学习笔记-Pod 生命周期与 Init C 介绍

    一.容器生命周期 Init C(初始化容器)只是用于 Pod 初始化的,不会一直随着 Pod 生命周期存在,Init C 在初始化完成之后就会死亡. 一个 Pod 可以有多个 Init C,也可以不需 ...

  7. GO语言面向对象01---封装属性与创建对象的方法与工厂模式

    package main import "fmt" /* 面向过程编程:调度大量的变量.函数 ---------- 面向对象编程(OOP=Object Oriented Progr ...

  8. win10家庭中文版CUDA+CUDNN+显卡GPU使用tensorflow-gpu训练模型安装过程(精华帖汇总+重新修改多次复现)

    查看安装包 pip list 本帖提供操作过程,具体操作网上有好多了,不赘述.红色字体为后来复现出现的问题以及批注 题外话: (1)python 的环境尽量保持干净,尽量单一,否则容易把自己搞晕,不知 ...

  9. 多边形游戏——区间dp

    题目描述 多边形(Polygon)游戏是单人玩的游戏,开始的时候给定一个由N个顶点构成的多边形(图1所示的例子中,N=4),每个顶点被赋予一个整数值,而每条边则被赋予一个符号:+(加法运算)或者*(乘 ...

  10. 深度学习调用TensorFlow、PyTorch等框架

    深度学习调用TensorFlow.PyTorch等框架 一.开发目标目标 提供统一接口的库,它可以从C++和Python中的多个框架中运行深度学习模型.欧米诺使研究人员能够在自己选择的框架内轻松建立模 ...