Formik & Yup:基于数组字段值的条件验证

formik & yup:基于数组字段值的条件验证

本文介绍了如何在使用 Formik 和 Yup 进行表单验证时,根据数组字段的值进行条件验证。针对 lessonType 字段包含 “video” 或 “document” 时,分别验证 videoFile 和 documentFile 是否存在的场景,提供了详细的解决方案和代码示例,帮助开发者实现更复杂的表单验证逻辑。

在使用 Formik 和 Yup 构建表单时,经常会遇到需要根据某些字段的值来动态验证其他字段的情况。当涉及到数组类型的字段时,这种条件验证会变得更加复杂。本文将介绍如何使用 Yup 的 when 方法,结合 test 方法,来实现基于数组字段值的条件验证。

问题场景

假设我们有一个包含模块和课程的表单,每个模块包含多个课程,每个课程可以有不同的类型(例如:视频、文档)。我们需要验证:

如果课程类型 lessonType 包含 “video”,则必须上传视频文件 videoFile。如果课程类型 lessonType 包含 “document”,则必须上传文档文件 documentFile。

解决方案

Yup 提供了 when 方法来实现条件验证,但由于 lessonType 是一个数组,我们需要使用 test 方法来处理数组的判断逻辑。以下是具体的实现步骤:

定义 Yup 验证 Schema

首先,定义一个 Yup 验证 Schema,包含 modules 数组,每个 module 包含 title 和 lessons 数组,每个 lesson 包含 lessonTitle、lessonType、videoFile 和 documentFile 字段。

import * as Yup from 'yup';let validationSchema = Yup.object({  modules: Yup.array(    Yup.object({      title: Yup.string()        .required("A module title is required")        .min(1, "A module title is required"),      lessons: Yup.array(        Yup.object({          lessonTitle: Yup.string().required("A lesson title is required"),          lessonType: Yup.array()            .min(1, "Lesson type is required")            .required("Lesson type is required")            .test('file-test', 'File required', function (value) {              const { videoFile, documentFile } = this.parent;              if (value && value.includes('video') && (!videoFile || videoFile.length === 0)) {                return this.createError({ message: 'Video file required when lesson type is video' });              }              if (value && value.includes('document') && (!documentFile || documentFile.length === 0)) {                return this.createError({ message: 'Document file required when lesson type is document' });              }              return true;            }),          videoFile: Yup.mixed(),          documentFile: Yup.mixed(),        })      ),    })  )    .min(1, "You must add at least one module")    .required("You must add at least one module"),});

使用 test 方法进行条件验证

在 lessonType 字段的验证中,我们使用了 test 方法,它允许我们自定义验证逻辑。test 方法接收三个参数:

name: 验证名称,用于标识该验证。message: 验证失败时的错误消息。test: 验证函数,接收当前字段的值作为参数,并返回 true 表示验证通过,返回 false 或调用 this.createError 表示验证失败。

在验证函数中,我们通过 this.parent 访问父对象,也就是当前的 lesson 对象,从而获取 videoFile 和 documentFile 的值。然后,我们根据 lessonType 数组是否包含 “video” 或 “document” 来判断是否需要验证 videoFile 或 documentFile 是否存在。

定义 videoFile 和 documentFile 字段

需要定义 videoFile 和 documentFile 字段,类型可以设置为 Yup.mixed(),表示可以接收任何类型的值,例如文件对象。

示例代码

以下是一个完整的示例代码,展示了如何使用 Formik 和 Yup 来实现上述条件验证:

import React from 'react';import { Formik, Form, Field } from 'formik';import * as Yup from 'yup';// Yup validation schema (as defined above)const MyForm = () => {  const initialValues = {    modules: [{      title: '',      lessons: [{        lessonTitle: '',        lessonType: [],        videoFile: null,        documentFile: null,      }]    }]  };  const onSubmit = (values, { setSubmitting }) => {    setTimeout(() => {      alert(JSON.stringify(values, null, 2));      setSubmitting(false);    }, 400);  };  return (          {({ isSubmitting }) => (                                                  Video            Document                                                      )}      );};export default MyForm;

注意事项

this.parent 属性在 test 方法中非常有用,它可以访问到父对象,从而获取其他字段的值。错误消息的显示位置可能需要根据实际情况进行调整,因为默认情况下错误消息会显示在 lessonType 字段下方。可以根据实际需求,自定义验证逻辑和错误消息。

总结

通过使用 Yup 的 when 方法和 test 方法,我们可以实现基于数组字段值的条件验证,从而构建更加灵活和强大的表单。在实际开发中,可以根据具体的需求,灵活运用这些方法,来实现各种复杂的表单验证逻辑。

以上就是Formik & Yup:基于数组字段值的条件验证的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1511164.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月20日 06:23:51
下一篇 2025年12月20日 06:24:00

相关推荐

  • 如何构建一个支持实时数据同步的协作编辑器?

    采用CRDTs实现数据一致性,以Yjs+WebSocket+ProseMirror构建协作编辑器,通过增量同步与presence消息实现实时协作与状态感知。 要构建一个支持实时数据同步的协作编辑器,核心在于解决多个用户同时编辑时的数据一致性问题。主流方案是采用 操作转换(OT) 或 冲突-free …

    2025年12月20日
    000
  • 如何用JavaScript实现一个支持并发修改的文档模型?

    答案:实现支持并发修改的文档模型需结合前端与后端协同处理冲突。前端通过OT或CRDT技术检测和转换操作,如使用ShareDB库实现操作同步;后端利用数据库存储数据并借助消息队列处理编辑操作,同时维护操作历史以支持撤销/重做功能;通过实时同步、光标共享、冲突提示、离线编辑及性能优化等手段提升用户体验,…

    2025年12月20日
    000
  • 如何构建一个使用 GraphQL 订阅实现实时数据更新的前端应用?

    答案:使用 Apollo Client 配置 WebSocketLink 实现 GraphQL 订阅,通过 useSubscription 监听实时数据,需前后端协同支持。 要构建一个使用 GraphQL 订阅实现实时数据更新的前端应用,核心是通过 WebSocket 与支持订阅的 GraphQL …

    2025年12月20日
    000
  • React组件性能优化:深入理解React.memo如何避免不必要的重渲染

    本文深入探讨React应用中常见的性能瓶颈——组件不必要的重渲染问题。通过一个具体案例,我们详细解析了父组件状态更新如何导致子组件冗余渲染,并重点讲解了如何利用React.memo这一高阶组件,结合其浅比较机制,有效阻止子组件在props未改变时进行重复渲染,从而显著提升应用性能和用户体验。 1. …

    2025年12月20日
    000
  • 精准控制页面卸载:区分刷新与关闭以优化LocalStorage管理

    本文深入探讨如何在Web应用中精确区分页面刷新与关闭事件,利用 window.onbeforeunload 结合 Performance Timing API 的 navigation.type 属性,实现仅在所有相关页面或标签页关闭时才清除 localStorage,从而优化跨标签页数据管理策略,…

    2025年12月20日
    000
  • 根据匹配的键值对从一个数组中筛选并返回另一个数组

    本教程旨在演示如何根据一个数组中元素的匹配值,从另一个包含对象的数组中筛选并提取特定属性。我们将探讨使用JavaScript的forEach、find、filter和map等方法实现此功能的多种策略,并提供代码示例及性能考量,帮助开发者高效处理数据筛选任务。 问题阐述 在前端开发中,我们经常需要处理…

    2025年12月20日
    000
  • 如何用Node.js实现一个支持JWT的认证中间件?

    答案:通过jsonwebtoken库实现JWT认证中间件,验证Authorization头中的Bearer Token合法性。首先安装express和jsonwebtoken,登录时用jwt.sign生成带过期时间的Token;中间件authenticateToken解析请求头,提取并用jwt.ve…

    2025年12月20日
    000
  • JavaScript中的位运算符在性能优化中如何应用?

    位运算符在JavaScript中通过操作二进制提升性能,适用于取整、乘除优化、奇偶判断、标志位管理等场景,尤其在高频计算和底层逻辑中仍具优势。 JavaScript中的位运算符常被忽视,但在特定场景下能有效提升性能。它们直接操作数字的二进制表示,执行速度通常快于常规数学或逻辑操作。虽然现代JavaS…

    2025年12月20日
    000
  • 如何构建一个支持微前端间共享状态的治理方案?

    答案是建立统一的共享状态治理机制。需明确共享范围与责任归属,仅公共状态如登录信息、主题配置等可共享,并由所属团队维护;通过注册中心公开状态清单,禁止未声明的读写操作;采用标准化接入方式如全局事件总线或中央store,封装为统一API;实施变更评审、版本共存与依赖校验,结合权限控制与监控告警,将共享状…

    2025年12月20日
    000
  • JavaScript中的模块联邦(Module Federation)如何实现微前端资源共享?

    模块联邦通过Webpack 5实现微前端架构,支持运行时共享代码。1. 核心机制:配置ModuleFederationPlugin,Host应用引入Remote应用暴露的模块,通过remoteEntry.js注册并按需加载。2. 基本配置:Remote应用使用exposes导出组件(如Header)…

    2025年12月20日
    000
  • 如何编写可复用的JavaScript自定义表单验证逻辑?

    答案:通过封装通用验证函数、配置驱动规则绑定、编写通用验证器,实现表单验证逻辑解耦与复用,提升灵活性和维护性。 编写可复用的JavaScript自定义表单验证逻辑,关键在于解耦验证规则与具体表单元素,通过函数封装和配置驱动的方式提升灵活性和维护性。下面是一些实用方法。 1. 定义通用验证规则函数 将…

    2025年12月20日
    000
  • JavaScript中根据键值匹配筛选数组并提取特定字段

    本教程旨在指导如何在JavaScript中,依据一个字符串数组的匹配项,从另一个包含对象的数组中筛选并提取特定字段。文章将详细介绍使用forEach结合find进行遍历查找,以及更现代、函数式的filter与map组合方法,并探讨如何通过Set优化查找性能,帮助开发者高效处理数组数据转换需求。 问题…

    2025年12月20日
    000
  • JavaScript:高效筛选对象数组并提取匹配键值

    本教程旨在指导如何在JavaScript中根据一个字符串数组的匹配值,从一个包含对象的数组中筛选出符合条件的对象,并从中提取特定的键值(如label),最终生成一个新的数组。文章将通过多种方法,包括forEach结合find以及更现代的filter和map组合,详细阐述实现过程,并提供代码示例及实践…

    2025年12月20日
    000
  • 如何在 Next.js 13 中为带客户端交互的静态页面读取本地数据

    本文旨在解决 Next.js 13 App Router 环境下,如何为需要客户端搜索和过滤功能的静态页面读取本地 Markdown 数据的问题。核心方案是利用服务器组件在构建时(或请求时)处理本地文件系统(fs)操作,将处理后的数据作为 props 传递给客户端组件,从而实现静态页面生成与客户端交…

    2025年12月20日
    000
  • JavaScript:根据另一数组匹配值过滤对象数组并提取特定属性

    本文将指导如何在JavaScript中根据一个字符串数组的匹配值,高效地过滤一个包含对象的数组,并从中提取出特定属性(如label),最终生成一个符合需求的新数组。我们将探讨使用forEach和find等数组方法,以及更推荐的filter和map组合实现此功能,并讨论性能优化,以应对数据处理中的常见…

    2025年12月20日
    000
  • 如何设计一个可扩展的表单验证引擎?

    答案:设计可扩展表单验证引擎需解耦验证逻辑,通过规则注册机制支持自定义校验。定义统一规则结构(名称、校验函数、错误提示),以配置驱动执行;构建规则管理模块,支持动态注册与查找;实现字段级和表单级验证调度,按序执行并收集错误;支持同步异步规则,返回结构化结果;提供简洁调用接口,传入数据与规则配置即可验…

    2025年12月20日
    000
  • 如何基于另一个数组的匹配值筛选并提取JavaScript对象数组中的特定字段

    本教程详细介绍了如何在JavaScript中根据一个简单值数组来筛选并提取另一个复杂对象数组中的特定字段。我们将通过示例数据,演示如何利用数组的forEach、find以及更现代的filter和map方法,高效地实现数据匹配和转换,最终生成所需的目标数组,帮助开发者掌握灵活处理数组数据的技巧。 引言…

    2025年12月20日
    000
  • 微前端架构中如何解决JavaScript沙箱隔离难题?

    微前端沙箱隔离核心是防止子应用间全局污染,主要方案包括:1. 用Proxy代理window实现运行时隔离,支持状态回滚但不兼容IE;2. 快照机制在加载前后保存恢复window状态,兼容好但性能开销大;3. Webpack模块联邦在构建时隔离依赖,适合多团队协作;4. iframe提供强隔离但通信复…

    2025年12月20日
    000
  • 如何实现一个基于规则的前端业务逻辑引擎?

    答案:基于规则的前端业务逻辑引擎通过分离决策与代码提升灵活性,核心结构包含条件与动作,支持动态解析执行、数据监听及动作响应,适用于复杂多变场景。 实现一个基于规则的前端业务逻辑引擎,核心是把业务决策从代码中剥离出来,通过可配置的规则来驱动行为。这种方式能提升系统的灵活性,降低维护成本,尤其适合多变、…

    2025年12月20日
    000
  • JavaScript中的前端架构模式(如MVC、MVVM)如何选择?

    MVC适合逻辑集中、结构清晰的项目,由Controller处理输入与更新,View不直接绑定数据,适用于传统命令式编程场景;MVVM通过ViewModel实现双向绑定,减少DOM操作,提升开发效率,适合数据驱动的交互型应用;现代框架如Vue倾向MVVM,React结合状态管理形成组件化架构,选型应根…

    2025年12月20日
    000

发表回复

登录后才能评论
关注微信