JS两种复制到剪贴板的方法

两种复制到剪贴板的方法

方法一:navigator.clipboard.writeText()(现代 Clipboard API)

原理

通过浏览器提供的异步 Clipboard API 实现,基于 Promise 操作剪贴板。

特点

  • 异步操作不阻塞主线程
  • 需要 HTTPS 或 localhost 安全上下文
  • API 简洁现代,代码量少
  • 必须在用户交互事件中触发

代码实现

javascript 复制代码
// Promise 写法
navigator.clipboard.writeText('要复制的文本')
  .then(() => console.log('复制成功'))
  .catch(err => console.error('复制失败:', err))

// async/await 写法
async function copyText(text) {
  try {
    await navigator.clipboard.writeText(text)
    console.log('复制成功')
  } catch (err) {
    console.error('复制失败:', err)
  }
}

浏览器支持

Chrome 66+、Firefox 63+、Safari 13.1+、Edge 79+

方法二:execCommand(传统实现)

原理

通过创建临时 <textarea> 元素,使用已废弃但广泛兼容的 document.execCommand('copy') 实现。

实现流程

  1. 创建隐藏的文本域元素
  2. 设置文本内容并添加到 DOM
  3. 选中文本内容
  4. 执行复制命令
  5. 清理 DOM 并恢复焦点

代码实现

javascript 复制代码
function copyTextToClipboard(text) {
  // 1. 创建临时 textarea 元素
  const element = document.createElement('textarea')
  const previouslyFocusedElement = document.activeElement

  // 2. 设置文本内容
  element.value = text
  element.setAttribute('readonly', '')
  
  // 3. 隐藏元素(不显示在页面上)
  element.style.contain = 'strict'
  element.style.position = 'absolute'
  element.style.left = '-9999px'
  element.style.fontSize = '12pt' // 防止 iOS 缩放

  // 4. 保存当前选择状态
  const selection = document.getSelection()
  const originalRange = selection.rangeCount > 0 && selection.getRangeAt(0)

  // 5. 添加到 DOM 并选中
  document.body.append(element)
  element.select()
  element.selectionStart = 0
  element.selectionEnd = text.length

  // 6. 执行复制命令
  let isSuccess = false
  try {
    isSuccess = document.execCommand('copy')
  } catch (err) {
    console.error('复制失败:', err)
  }

  // 7. 清理:移除元素,恢复选择状态和焦点
  element.remove()
  if (originalRange) {
    selection.removeAllRanges()
    selection.addRange(originalRange)
  }
  if (previouslyFocusedElement) {
    previouslyFocusedElement.focus()
  }

  return isSuccess
}

特点

  • 同步执行立即返回结果
  • 兼容 IE 等老旧浏览器
  • 实现复杂度较高
  • 已被现代标准标记为废弃
渐进增强策略

推荐实现方案

javascript 复制代码
function safeCopyText(text) {
  if (navigator.clipboard?.writeText) {
    return navigator.clipboard.writeText(text)
      .catch(() => copyTextToClipboard(text))
  }
  return copyTextToClipboard(text)
}

优势

  • 现代浏览器优先使用标准 API
  • 自动降级保证兼容性
  • 统一的错误处理机制

注意事项

  • 所有复制操作必须由用户交互触发
  • 建议添加成功/失败状态提示
  • 测试需覆盖不同浏览器环境
相关推荐
一点一木6 小时前
深度体验TRAE SOLO移动端7天:作为独立开发者,我把工作流揣进了兜里
前端·人工智能·trae
天外飞雨道沧桑6 小时前
TypeScript 中 omit 和 record 用法
前端·javascript·typescript
Lee川6 小时前
mini-cursor 揭秘:从 Tool 定义到 Agent 循环的完整实现
前端·人工智能·后端
canonical_entropy7 小时前
从 Spec-Driven Development 到 Attractor-Guided Engineering
前端·aigc·ai编程
研☆香7 小时前
聊聊前端页面的三种长度单位
前端
给钱,谢谢!8 小时前
React + PixiJS 实现果园成长页:从状态机到浇水动画
前端·react.js·前端框架
暗冰ཏོ9 小时前
VUE面试题大全
前端·javascript·vue.js·面试
次元工程师!9 小时前
LangFlow开发(三)—Bundles组件架构设计(3W+字详细讲解)
java·前端·python·低代码·langflow
Bug-制造者10 小时前
现代Web应用全栈开发:从架构设计到部署落地实战
前端
青春喂了后端10 小时前
IntelliGit 前端状态层重构:把一个全局 Store 拆成清晰的状态边界
前端·重构·状态模式