js实现输入高亮@和#后面的内容

效果如下:支持连续输入

html 复制代码
<!DOCTYPE html>
<html lang="zh-CN">

<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <title></title>
  <style>
    * {
      box-sizing: border-box;
      margin: 0;
      padding: 0;
    }

    body {
      min-height: 100vh;
      display: flex;
      justify-content: center;
      align-items: center;
      padding: 20px;
    }

    .container {
      width: 100%;
      max-width: 800px;
      background: white;
      border-radius: 16px;
      box-shadow: 0 10px 30px rgba(0, 0, 0, 0.2);
      overflow: hidden;
    }

    .content {
      padding: 30px;
    }

    .editor-container {
      position: relative;
      margin-bottom: 30px;
    }

    #editor {
      min-height: 200px;
      border: 2px solid #e0e0e0;
      border-radius: 8px;
      padding: 15px;
      font-size: 16px;
      line-height: 1.6;
      outline: none;
      transition: border-color 0.3s;
      background: white;
      overflow-y: auto;
    }

    #editor:focus {
      border-color: #3498db;
      box-shadow: 0 0 0 2px rgba(52, 152, 219, 0.2);
    }

    .highlight {
      background-color: #fff9c4;
      border-radius: 3px;
      padding: 0 2px;
      color: #e74c3c;
      font-weight: 500;
    }
  </style>
</head>

<body>
  <div class="container">
    <div class="content">
      <div class="editor-container">
        <div id="editor" contenteditable="true" placeholder="输入内容..."></div>
      </div>
    </div>
  </div>

  <script>
    const editor = document.getElementById('editor');
    function saveCursorPosition() {
      const selection = window.getSelection();
      if (selection.rangeCount === 0) return null;
      const range = selection.getRangeAt(0);
      const preCaretRange = range.cloneRange();
      preCaretRange.selectNodeContents(editor);
      preCaretRange.setEnd(range.endContainer, range.endOffset);
      return {
        startOffset: preCaretRange.toString().length,
        range: range
      };
    }
    function restoreCursorPosition(startOffset) {
      const selection = window.getSelection();
      selection.removeAllRanges();
      let charCount = 0;
      let nodeStack = [editor];
      let node, foundStart = false;
      let range = document.createRange();
      range.setStart(editor, 0);
      range.collapse(true);
      while (!foundStart && (node = nodeStack.pop())) {
        if (node.nodeType === Node.TEXT_NODE) {
          const nextCharCount = charCount + node.length;
          if (!foundStart && startOffset >= charCount && startOffset <= nextCharCount) {
            range.setStart(node, startOffset - charCount);
            foundStart = true;
          }
          charCount = nextCharCount
        } else {
          const children = node.childNodes;
          for (let i = children.length - 1; i >= 0; i--) {
            nodeStack.push(children[i])
          }
        }
      }

      if (foundStart) {
        range.collapse(true);
        selection.addRange(range);
      }
    }

    function applyHighlighting() {
      const cursorPosition = saveCursorPosition();
      const text = editor.textContent;
      const tags = [];
      editor.innerHTML = text.replace(/([@#][^\s]+)/g, (match) => {
        tags.push(match);
        return `<span class="highlight">${match}</span>`;
      });
      // 恢复光标位置
      if (cursorPosition) {
        restoreCursorPosition(cursorPosition.startOffset);
      }
    }

    let isComposing = false;
    editor.addEventListener('compositionstart', () => {
      isComposing = true;
    });

    editor.addEventListener('compositionend', () => {
      isComposing = false;
      applyHighlighting();
    });

    editor.addEventListener('input', () => {
      if (!isComposing) {
        applyHighlighting();
      }
    });
    applyHighlighting();
  </script>
</body>

</html>
相关推荐
夏幻灵8 分钟前
HTML5里最常用的十大标签
前端·html·html5
冰暮流星9 分钟前
javascript之二重循环练习
开发语言·javascript·数据库
Mr Xu_22 分钟前
Vue 3 中 watch 的使用详解:监听响应式数据变化的利器
前端·javascript·vue.js
未来龙皇小蓝26 分钟前
RBAC前端架构-01:项目初始化
前端·架构
程序员agions34 分钟前
2026年,微前端终于“死“了
前端·状态模式
万岳科技系统开发34 分钟前
食堂采购系统源码库存扣减算法与并发控制实现详解
java·前端·数据库·算法
程序员猫哥_42 分钟前
HTML 生成网页工具推荐:从手写代码到 AI 自动生成网页的进化路径
前端·人工智能·html
龙飞0542 分钟前
Systemd -systemctl - journalctl 速查表:服务管理 + 日志排障
linux·运维·前端·chrome·systemctl·journalctl
我爱加班、、1 小时前
Websocket能携带token过去后端吗
前端·后端·websocket
AAA阿giao1 小时前
从零拆解一个 React + TypeScript 的 TodoList:模块化、数据流与工程实践
前端·react.js·ui·typescript·前端框架