contentEditable 实现可编辑区域

背景

最近看豆包的时候,发现他的代码写AI编程模块 如下

借助的是contentEditable 来实现的,操作比较友好,可以随意输入可编辑数据,有类似于 placeholder等属性

如果我们开发大模型,免不了东西借鉴一下,下面基于react 来封装 一个 使用contentEditable 来实现的可编辑区域组件

实现重点

  1. 开启 contentEditable 后 如何模拟 placeholder
  2. 如何判断何时展示 placeholder 何时展示真实内容
  3. 调整优化以及受控组件封装

代码如下

js 复制代码
import React, { useState, useRef, useEffect } from 'react'

interface ContentEditablePromptProps {
  placeholder?: string
  className?: string
  value?: string
  onChange?: (value: string) => void
}

const ContentEditablePrompt: React.FC<ContentEditablePromptProps> = ({
  placeholder = '请输入',
  className = '',
  value = '',
  onChange
}) => {
  const contentEditableRef = useRef<HTMLDivElement>(null)
  const [isEmpty, setIsEmpty] = useState(true)
  const [internalValue, setInternalValue] = useState(value)

  // 更新内容并处理空状态
  const updateContent = (newValue: string) => {
    if (!contentEditableRef.current) return

    // 更新内部状态
    setInternalValue(newValue)

    // 调用父组件的onChange回调
    if (onChange) {
      onChange(newValue)
    }

    // 更新空状态
    const isNowEmpty = newValue.trim() === ''
    setIsEmpty(isNowEmpty)

    // 处理空状态下的placeholder显示
    if (contentEditableRef.current) {
      if (isNowEmpty && contentEditableRef.current.innerHTML === '<br>') {
        contentEditableRef.current.innerHTML = ''
      }
    }
  }

  // 处理输入事件
  const handleInput = () => {
    if (!contentEditableRef.current) return
    updateContent(contentEditableRef.current.textContent || '')
  }

  // 处理失去焦点事件
  const handleBlur = () => {
    if (!contentEditableRef.current) return

    // 规范化内容
    const newValue = contentEditableRef.current.textContent || ''
    updateContent(newValue)

    // 确保DOM内容与内部状态一致
    if (contentEditableRef.current.textContent !== internalValue) {
      contentEditableRef.current.textContent = internalValue
    }
  }

  // 初始化内容和响应外部value变化
  useEffect(() => {
    if (!contentEditableRef.current) return

    // 只有当外部value与内部值不同时才更新
    if (value !== internalValue) {
      setInternalValue(value)
      contentEditableRef.current.textContent = value
      setIsEmpty(value.trim() === '')
    }
  }, [value])

  return (
    <div
      ref={contentEditableRef}
      className={`p-2 px-3 w-fit rounded-2xl flex items-center ${
        isEmpty
          ? 'after:content-[attr(data-placeholder)] after:tracking-[1px] bg-[#EEF6FF] font-bold text-[#007DFA] word-spacing-2'
          : 'bg-[#EEF6FF] font-bold text-[#007DFA] word-spacing-2'
      } ${className}`}
      contentEditable
      data-placeholder={placeholder}
      onInput={handleInput}
      onBlur={handleBlur}
      suppressContentEditableWarning
    />
  )
}

export default ContentEditablePrompt

使用如下:

js 复制代码
<EditableArea placeholder='预设提取项' value='' onChange={onChangeValue}></EditableArea>

结束

相关推荐
拾光拾趣录12 分钟前
for..in 和 Object.keys 的区别:从“遍历对象属性的坑”说起
前端·javascript
OpenTiny社区23 分钟前
把 SearchBox 塞进项目,搜索转化率怒涨 400%?
前端·vue.js·github
编程猪猪侠1 小时前
Tailwind CSS 自定义工具类与主题配置指南
前端·css
qhd吴飞1 小时前
mybatis 差异更新法
java·前端·mybatis
YGY Webgis糕手之路1 小时前
OpenLayers 快速入门(九)Extent 介绍
前端·经验分享·笔记·vue·web
患得患失9491 小时前
【前端】【vueDevTools】使用 vueDevTools 插件并修改默认打开编辑器
前端·编辑器
ReturnTrue8681 小时前
Vue路由状态持久化方案,优雅实现记住表单历史搜索记录!
前端·vue.js
UncleKyrie1 小时前
一个浏览器插件帮你查看Figma设计稿代码图片和转码
前端
遂心_1 小时前
深入解析前后端分离中的 /api 设计:从路由到代理的完整指南
前端·javascript·api
你听得到112 小时前
Flutter - 手搓一个日历组件,集成单日选择、日期范围选择、国际化、农历和节气显示
前端·flutter·架构