惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

爱范儿
爱范儿
Security Latest
Security Latest
NISL@THU
NISL@THU
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
C
Cybersecurity and Infrastructure Security Agency CISA
Cloudbric
Cloudbric
T
Threat Research - Cisco Blogs
大猫的无限游戏
大猫的无限游戏
C
CXSECURITY Database RSS Feed - CXSecurity.com
阮一峰的网络日志
阮一峰的网络日志
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
雷峰网
雷峰网
C
Cisco Blogs
V
Vulnerabilities – Threatpost
S
Security Archives - TechRepublic
V
Visual Studio Blog
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
cs.AI updates on arXiv.org
cs.AI updates on arXiv.org
J
Java Code Geeks
D
Darknet – Hacking Tools, Hacker News & Cyber Security
Know Your Adversary
Know Your Adversary
博客园 - 叶小钗
腾讯CDC
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
P
Privacy International News Feed
P
Palo Alto Networks Blog
博客园_首页
V
V2EX
WordPress大学
WordPress大学
Schneier on Security
Schneier on Security
月光博客
月光博客
博客园 - 司徒正美
Google DeepMind News
Google DeepMind News
TaoSecurity Blog
TaoSecurity Blog
博客园 - 聂微东
酷 壳 – CoolShell
酷 壳 – CoolShell
人人都是产品经理
人人都是产品经理
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
博客园 - 【当耐特】
The Cloudflare Blog
罗磊的独立博客
美团技术团队
N
News | PayPal Newsroom
K
KPMG report finds enterprise disconnect between AI and its ROI | CIO
Last Week in AI
Last Week in AI
K
Kaspersky official blog
Google Online Security Blog
Google Online Security Blog
S
SegmentFault 最新的问题
Application and Cybersecurity Blog
Application and Cybersecurity Blog
T
Tailwind CSS Blog

优世界

OpenWrt 路由器改纯 AP 模式记录(Cudy TR3000 + 中兴 F50) Ubuntu 24.04 安装 Zabbix 8.0 全记录 我的静态博客动态化方案 Claude Code 启动脚本 我的5G路由器方案:中兴F50+cudy tr3000 256MB 还是弃用了使用多年的全拼输入法 临走前一份襄阳牛肉面 EdgeOne Pages 部署 Twikoo 评论 一个朋友圈风格 Hugo 主题 搓了一个仿朋友圈的Hugo主题 一个人烧烤被拒单了 周末闲暇时间翻修了一下博客 我把Vercel换成了EdgeOne Pages 博客友链实时健康监测方案 Artalk评论区接入AI摘要的尝试 给Hugo博客添加瀑布流相册功能 Hugo静态博客如何实现搜索功能 为什么我觉得网页昼夜切换那么重要 博客六周年:从折腾到回归平淡 如何hugo静态实现友联朋友圈功能 脂溢性皮炎的烦恼 还是放弃了iPhone16e 米环勿扰同步问题 除草日记 Hugo使用GitHub Action自动刷新多吉云CDN缓存 迁移博客至hugo 添加ikun摆件 parsec远程软件报6023错误 win11右键菜单改回win10方法 主板电流声 主机配置单 一次点亮 访客体验优化 博客除草 老爷机升级 完善主题 家乡随拍 襄阳唐城 进厂日记 苏州一日游 四月随笔 评论置顶 Time Taker 近来二三事 感谢哥哥给的网站 又又又双叒叕换主题啦 生活篇:疫情放开前后的这些日子 .cc域名后缀续费即将涨价 Apple Watch Series7 三个月的使用体验 2022年·襄阳第一场雪,谨以此片记录 记一次莫名其妙ddos攻击,致谢天御云高防cdn 毕业篇:不出意外,这是我最后几个月的大学生活啦 毕业纪念篇:图书馆 生活篇:我的三年封校生活 没错,我回来了,湖北管局一天不到通过备案,强的! 因需要更换备案主体,临时关闭博客通知 记录篇:将笔记本联想小新pro13网卡由螃蟹网卡换成AX210网卡 主力机由荣耀20切换到iPhone13使用半年的感受 生活篇:温馨提示假期余额不足,浅浅总结一下这个暑假 typecho实现QQ头像用户评论加密,注意:pigeon,twitter用户有彩蛋哦! 生活篇:离校,暑假,租房,面试,致那些日子的琐碎事 生活篇:我跳绳的那些日子,谈谈自己的变化 去除typecho1.2.0正式版的后台提示更新bug 利用fontspider压缩博客字体大小,达到加快访问速度 iphone快捷指令发布动态说说,支持大部分typecho主题 寒假二三事 2022,除夕过后的那些事 2022,致我的春节回忆录 祝大家元旦快乐,给自己的博客加一个对联和灯笼 更换掉jsdelivr,改用腾讯云静态网站托管,网站速度比之前提升了不少 这次的落日比较有特色,西边摇摇欲坠的咸鸭蛋 盘点一下建站以来所注册的域名,我居然注册了一堆学费米 可惜不能一直做小孩子,总要长大,不知道实习生活会是什么样子 这组照片的主题,咱就叫它光吧 双十一已经变味了,不知道从什么时候反感双十一了 第一次尝试ai画简笔画,本是给自己设计logo误打误撞画起了简笔画 不用改变图片原地址,实现图片自动转webp格式,速看,一会删 让typecho支持webp格式的图片,告别阿里云oss和腾讯云cos被恶意刷流量的风险 运动会闲暇这几天,简单给大家分享一下日常吧 我又双叒叕换主题,我发现我好像一直在折腾 记录人生第一次洗牙,不得不说感觉真的特别好 我为什么要写博客?这位博主给了我答案 我想,这是一个我人生中最特殊的中秋国庆节 理工的晚霞,拿起相机记录青春的样子 别让抖音支配了你的美好大学生活,尝试做一些自己感兴趣的事 一岁一礼,一寸欢喜,生日快乐,致我的二十岁 2021暑假总结,记录一下这个充实的暑假 Twitter主题设置仿mac UI 语法高亮代码方法 Twitter主题加入加载耗时,访问总量功能 一把过,科二结束! 再次投资科目二,科目二花了400,希望19号一把过吧 宝塔面板设置Typecho伪静态去掉index.php教程 开学倒计时,再见,老家 或许不是没有年味了,只是快乐已经不属于我们这一辈人了 五福开奖,你好,2021 新春快乐!牛年大吉! 祝母校越办越好 军训太痛苦了,希望早点结束 高考加油,相信自己 高考即将来临,老师为我们加油打气
Artalk评论系统实现段落评论功能
2026-01-24 · via 优世界

最近在番茄小说上看书,发现它的段落评论功能是真的好用。看到有共鸣的句子,选中就能直接看别人的评论或者自己吐槽两句。

Artalk评论系统实现段落评论功能

我就在想,我的博客是不是也能整一个?毕竟现在还在坚持写博客的,谁还没点表达欲呢?

下午没事折腾了一下,试着给我的 Artalk 评论系统加上了这个功能。效果还凑合,选中文字后会弹出一个“引用评论”的按钮,点击就能自动跳转到评论区并引用选中的内容。

Artalk评论系统实现段落评论功能

稍微整理了一下代码,分享给有需要的朋友。我是个前端小白,代码写得比较粗糙,大佬们见笑了。无论你用什么主题,只要是 Artalk 评论系统,理论上都能适配。

实现思路

逻辑其实挺简单的,主要是以下几步:

  1. 监听选择:监听用户的鼠标选取操作 (mouseup)。
  2. 计算位置:获取选中范围的坐标,把按钮定位到鼠标或者选区上方。
  3. 点击交互:点击按钮后,获取选中的文字,平滑滚动到评论区,把文字填进输入框。
  4. 细节优化:做了一些防抖处理,简单适配了一下移动端和暗黑模式(这个必须有!)。

代码实现

1. HTML 结构

不需要手动写 HTML,直接用 JS 动态生成插入到 body 里就行,这样省事点。

2. CSS 样式

这块我调了半天。一开始用了毛玻璃效果 (backdrop-filter),结果发现页面滚动的时候有点掉帧,为了性能我把它去掉了,改成了更实用的深色/浅色背景,顺便加了 will-change 属性开启硬件加速。

把下面这段代码加到你的自定义 CSS 里:

/* 选中弹出按钮样式 */
#selection-popup {
    position: fixed;
    display: none;
    background: rgba(30, 30, 30, 0.9);
    color: #fff;
    padding: 8px 16px;
    border-radius: 8px;
    font-size: 14px;
    cursor: pointer;
    z-index: 2147483647; /* 堆叠层级拉满 */
    transform: translate(-50%, -100%);
    pointer-events: auto;
    box-shadow: 0 4px 12px rgba(0,0,0,0.15);
    transition: opacity 0.2s, transform 0.2s;
    font-weight: 500;
    line-height: 1.4;
    user-select: none;
    -webkit-user-select: none;
    align-items: center;
    gap: 6px;
    white-space: nowrap;
    will-change: transform, opacity; /* 性能优化关键 */
    animation: popIn 0.2s cubic-bezier(0.18, 0.89, 0.32, 1.28);
}

/* 弹出动画 */
@keyframes popIn {
    0% { transform: translate(-50%, -80%) scale(0.9); opacity: 0; }
    100% { transform: translate(-50%, -100%) scale(1); opacity: 1; }
}

/* 小三角箭头 */
#selection-popup::after {
    content: '';
    position: absolute;
    top: 100%;
    left: 50%;
    margin-left: -6px;
    border-width: 6px;
    border-style: solid;
    border-color: rgba(30, 30, 30, 0.9) transparent transparent transparent;
}

/* 悬停效果 */
#selection-popup:hover {
    transform: translate(-50%, -110%) scale(1.05);
    background: #000;
    box-shadow: 0 6px 16px rgba(0,0,0,0.25);
}

/* 暗黑模式适配 */
[data-theme="dark"] #selection-popup {
    background: rgba(255, 255, 255, 0.95);
    color: #000;
    box-shadow: 0 4px 12px rgba(0,0,0,0.3);
}

[data-theme="dark"] #selection-popup::after {
    border-color: rgba(255, 255, 255, 0.95) transparent transparent transparent;
}

[data-theme="dark"] #selection-popup:hover {
    background: #fff;
}

3. JavaScript 逻辑

把这段 JS 加到你的 main.js 或者 footer 的 <script> 标签里。

注意:为了性能,我这里用了 requestAnimationFrame,而且去掉了 getBoundingClientRect 这种耗性能的操作,直接用鼠标坐标定位,感觉会流畅一些。

document.addEventListener('DOMContentLoaded', function() {
    initSelectionPopup();
});

let selectionPopup = null;
let selectionTimeout;

function initSelectionPopup() {
    // 避免重复创建
    if (document.getElementById('selection-popup')) return;

    // 动态创建按钮
    selectionPopup = document.createElement('div');
    selectionPopup.id = 'selection-popup';
    // 这里用了 RemixIcon,你可以换成你自己的图标库
    selectionPopup.innerHTML = '<i class="ri-chat-quote-line"></i> <span>引用评论</span>';
    document.body.appendChild(selectionPopup);

    // 防止在按钮上点击时触发选区清除
    selectionPopup.addEventListener('mousedown', function(e) {
        e.preventDefault();
        e.stopPropagation();
    });

    // 点击按钮的逻辑
    selectionPopup.addEventListener('click', function(e) {
        e.preventDefault();
        e.stopPropagation();
        
        let selection = window.getSelection().toString();
        if (selection) {
            // 简单的文本清理
            selection = selection.split('\n')
                .map(line => line.trim())
                .filter(line => line !== '')
                .join('\n');
                
            // 限制长度,防止太长刷屏
            if (selection.length > 500) {
                 selection = selection.substring(0, 500) + '...';
            }

            scrollToCommentsAndQuote(selection);
            hideSelectionPopup();
            window.getSelection().removeAllRanges(); // 清除选区
        }
    });

    // 监听全局选择事件
    document.addEventListener('mouseup', handleSelectionChange);
    document.addEventListener('keyup', handleSelectionChange);
    
    // 页面滚动时隐藏
    document.addEventListener('scroll', hideSelectionPopup, { passive: true });
}

function hideSelectionPopup() {
    if (selectionPopup) {
        selectionPopup.style.display = 'none';
    }
}

function handleSelectionChange(e) {
    // 防抖处理,别频繁触发
    clearTimeout(selectionTimeout);
    selectionTimeout = setTimeout(() => {
        const selection = window.getSelection();
        const text = selection.toString().trim();
        // 记得改成你文章容器的类名,我这里是 .post-content
        const content = document.querySelector('.post-content'); 
        
        if (!selectionPopup || !content) return;

        // 没选中文字就隐藏
        if (!text || selection.rangeCount === 0) {
            hideSelectionPopup();
            return;
        }

        const range = selection.getRangeAt(0);
        
        // 确保选中的是文章里的内容,别把侧边栏也选进去了
        if (!content.contains(range.commonAncestorContainer)) {
            hideSelectionPopup();
            return;
        }

        // 使用 requestAnimationFrame 优化 UI 更新
        requestAnimationFrame(() => {
            selectionPopup.style.display = 'flex'; 

            // 定位逻辑:鼠标抬起时跟随鼠标,键盘选择时跟随选区
            if (e && e.type === 'mouseup') {
                // 直接用鼠标坐标,比 getBoundingClientRect 性能好多了
                selectionPopup.style.top = `${e.clientY - 40}px`; 
                selectionPopup.style.left = `${e.clientX}px`;
            } else {
                // 键盘操作没办法,只能计算矩形了
                const rect = range.getBoundingClientRect();
                if (rect.width === 0 || rect.height === 0) {
                    hideSelectionPopup();
                    return;
                }
                selectionPopup.style.top = `${rect.top}px`; 
                selectionPopup.style.left = `${rect.left + rect.width / 2}px`;
            }
        });
    }, 150);
}

// 跳转评论区并填充内容
function scrollToCommentsAndQuote(text) {
    const comments = document.getElementById('Comments'); // 评论区 ID
    if (comments) {
        comments.scrollIntoView({ behavior: 'smooth' });
        
        // 等滚动完了再填内容
        setTimeout(() => {
             const textarea = document.querySelector('.atk-textarea'); // Artalk 输入框 class
             if (textarea) {
                 const quote = `> ${text}\n\n`;
                 textarea.value = quote;
                 textarea.focus();
                 
                 // 触发 input 事件,适配 Vue/React 框架的数据绑定
                textarea.dispatchEvent(new Event('input', { bubbles: true }));
                textarea.dispatchEvent(new Event('change', { bubbles: true }));
            }
       }, 500);
   }
}

简单总结

代码复制进去,刷新页面(如果有缓存记得清一下),选中这段文字试试?

这个功能不仅能提高读者的互动率,还能让评论更有针对性。我也就是瞎折腾,如果有更好的实现方式,欢迎大家在评论区指教!