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
英文地址,請注明出處:http://en.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;大幅降低開發門檻。本文…