编辑器、代码块、大模型AI对话中代码复制功能实现

一、效果图

二、navigator.clipboard API

直奔主题。代码块复制使用现代浏览器原生 Clipboard API

typescript 复制代码
	  const [copied, setCopied] = useState(false);

      // 处理复制逻辑
      const handleCopy = async (code: string) => {
        try {
          await navigator.clipboard.writeText(code);
          setCopied(true);
          setTimeout(() => setCopied(false), 2000);
        } catch (err) {
          console.error('复制失败:', err);
        }
      };
     const codeContent = String(children).replace(/\n$/, '')
     return(
      		<button
              onClick={() => handleCopy(codeContent)}
              style={{
                position: 'absolute',
                right: 8,
                top: 8,
                padding: '4px 8px',
                background: '#f5f5f5',
                border: '1px solid #ddd',
                borderRadius: 4,
                cursor: 'pointer',
                zIndex: 1,
                opacity: copied ? 1 : 0.6,
                 transition: "opacity 0.2s",
              }}
            >
              {copied ? '复制成功' : '⎘ 复制'}
           </button>
     )

关键点说明

  1. 状态管理:

使用 useState 跟踪复制状态,提供视觉反馈

2秒后自动重置复制状态提示

  1. 样式定位

绝对定位到代码块右上角

使用:

typescript 复制代码
 <div style={{position: 'relative'}}>//外层相对定位
 //复制
  {children && handCopyRender(children)}//复制块绝对定位
 //代码块dom
 //例:<SyntaxHighlighter/>
 </div>
  1. 内容处理:
typescript 复制代码
const codeContent = String(children).replace(/\n$/, "");

移除代码结尾的多余换行符 确保复制内容格式正确

三、浏览器兼容处理

  1. 旧版浏览器不支持 Clipboard API
  2. HTTP环境下无法生效
  3. 浏览器安全限制:Clipboard API 要确保在 HTTPS 环境或 localhost 下使用

兼容性处理后:

typescript 复制代码
const handCopyRender = (value: React.ReactNode) => {
  /** 复制功能组件 */
  const [copied, setCopied] = useState(false);
  
  // 兼容性复制方法
  const copyToClipboard = (text: string) => {
    return new Promise<boolean>((resolve) => {
      try {
        // 现代浏览器方案
        if (navigator.clipboard && window.isSecureContext) {
          navigator.clipboard.writeText(text)
            .then(() => resolve(true))
            .catch(() => resolve(false));
        } 
        // 旧浏览器兼容方案
        else {
          // 创建临时文本域
          const textArea = document.createElement('textarea');
          textArea.value = text;
          textArea.style.position = 'fixed';  // 避免滚动跳转
          textArea.style.opacity = '0';        // 透明不可见
          
          document.body.appendChild(textArea);
          textArea.focus();//文本处于可编辑区域
          textArea.select();//且被选中

          // 执行复制命令
          const success = document.execCommand('copy');
          document.body.removeChild(textArea);
          resolve(success);
        }
      } catch (err) {
        console.error("复制失败:", err);
        resolve(false);
      }
    });
  };

  const handleCopy = async () => {
    try {
      // 转换内容为字符串
      const codeContent = String(value).replace(/\n$/, "");
      
      // 执行复制操作
      const success = await copyToClipboard(codeContent);
      
      if (success) {
        setCopied(true);
        setTimeout(() => setCopied(false), 2000);
      } else {
        // 降级提示
        alert("自动复制失败,请手动选择文本后按 Ctrl+C");
      }
    } catch (err) {
      console.error("复制异常:", err);
    }
  };

  return (
    		<button
              onClick={handleCopy}
              style={{
                position: 'absolute',
                right: 8,
                top: 8,
                padding: '4px 8px',
                background: '#f5f5f5',
                border: '1px solid #ddd',
                borderRadius: 4,
                cursor: 'pointer',
                zIndex: 1,
                opacity: copied ? 1 : 0.6,
                transition: "opacity 0.2s",
              }}
            >
              {copied ? '复制成功' : '⎘ 复制'}
           </button>
  );
};

关键点说明

  • 环境检测 navigator.clipboard && window.isSecureContext
typescript 复制代码
window.isSecureContext // 检测 HTTPS本地环境

window.isSecureContext是一个浏览器 API,用于检查当前页面是否在安全上下文中运行。如果页面在安全上下文中运行(即通过HTTPS加载),则该函数返回true,否则返回false。

  • 关键思路:
  • 降级使用document.execCommand('copy')的旧方法。同时,还创建了一个临时的textarea元素来执行复制操作,并处理了相关的用户体验问题,比如提示信息、样式调整等。
  • 具体到复制功能,临时创建textarea元素的原因是为了避免界面干扰,并确保文本可以被正确选中。使用execCommand('copy')需要文本处于可编辑区域且被选中,所以创建隐藏的textarea是实现这一点的常用方法 。同时,处理完复制操作后需要立即移除该元素,避免DOM污染。
相关推荐
前端Hardy1 小时前
Wails v3 正式发布:用 Go 写桌面应用,体积仅 12MB,性能飙升 40%!
前端·javascript·go
Highcharts.js1 小时前
Highcharts React v4 迁移指南(下):分步代码示例与常见问题解决
javascript·react.js·typescript·react·highcharts·代码示例·v4迁移
Laurence1 小时前
Qt 前后端通信(QWebChannel Js / C++ 互操作):原理、示例、步骤解说
前端·javascript·c++·后端·交互·qwebchannel·互操作
Pu_Nine_91 小时前
JavaScript 字符串与数组核心方法详解
前端·javascript·ecmascript
这是个栗子1 小时前
前端开发中的常用工具函数(六)
javascript·every
kyriewen2 小时前
异步编程:从“回调地狱”到“async/await”的救赎之路
前端·javascript·面试
前端Hardy2 小时前
别再手动写 loading 了!封装一个自动防重提交的 Hook
前端·javascript·vue.js
前端Hardy2 小时前
前端如何实现“无感刷新”Token?90% 的人都做错了
前端·javascript·vue.js
SuperEugene2 小时前
Vue Router 实战规范:path/name/meta 配置 + 动态 / 嵌套路由,统一团队标准|状态管理与路由规范篇
开发语言·前端·javascript·vue.js·前端框架
张一凡932 小时前
easy-model -- "小而美"的React状态管理方案
前端·javascript·react.js