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>
相关推荐
梁山好汉(Ls_man)9 分钟前
JS_使用脚本填充基于Vue的用户名密码输入框并触发登录
javascript·elementui·vue
大飞哥~BigFei18 分钟前
新版chrome浏览器安全限制及解决办法
java·前端·chrome·安全·跨域
hepingfly19 分钟前
SEO 如何寻找关键词?
前端
IT_陈寒21 分钟前
SpringBoot 3.2实战:5个性能优化技巧让你的应用提速50%
前端·人工智能·后端
扶苏100228 分钟前
前端js高频面试点汇总
开发语言·前端·javascript
firstacui29 分钟前
Keepalived 双主热备和三主热备
前端·chrome
小二·1 小时前
Python Web 开发进阶实战:微前端架构初探 —— 基于 Webpack Module Federation 的 Vue 微应用体系
前端·python·架构
阿呆5911 小时前
html前端开发注释的写法
前端·html
pusheng20251 小时前
守护能源与数据的安全防线:从UL 2075标准解析储能及数据中心氢探技术的演进
前端·安全
.又是新的一天.1 小时前
【前端Web开发HTML5+CSS3+移动web视频教程】02 html - 列表、表格、表单
前端·html·html5