通过 hack 的方式实现卡片式超链接

本贴最后更新于 1035 天前,其中的信息可能已经时移世易

本来不是很想把这种比较蛋疼的实现方式贴出来的,但是看到好像很多人需要这个,就简单说一下实现的思路吧

使用 shadowdom 是为了防止 hack 的链接影响编辑器运行。

import { DOM监听器 } from "/script/public/DOMwatcher.js"; //DOM监听器只是一个observer的封装 export class linkCard extends naive.plugin { constructor() { super({ name: "linkCard" }); window.requestAnimationFrame(() => this.hackLink()); this.icons = naive.fs.readdirSync( naive.pathConstructor.naivePath() + "/script/plugin/corePlugins/linkCard/icon4Tsundoku" ); window.siyuan.ws.ws.addEventListener("message", () => { this.hackLink(); }); document.addEventListener("mouseover", () => { this.hackLink(); }); let 监听选项1 = { 监听目标: `[data-node-id]`, //监听目标这里是指定了一个css选择器 监听器回调: () => this.hackLink(), //hacklink函数才是的功能代码 }; this.DOM监听器1 = new DOM监听器(监听选项1); } hackLink() { try{ let links = document.querySelectorAll( '.protyle-wysiwyg.protyle-wysiwyg--attr [data-node-id] span[data-type="a"]' ); links.forEach((link) => { if (link.dataset.title && link.dataset.title.indexOf("card:") >= 0) { //意思是标题以card:开头的超链接才会显示成卡片形式 if (!link.shadowRoot||!link.shadowRoot.innerHTML||!link.shadowRoot.querySelector("img")) { //已经有shadowDOM的元素不重复添加 this.attachLinkShadow(link); } else { //已经有shadowDOM元素,修改之后需要确保图标和URL等正确 let src = this.getImg(link.dataset.href); if ( link.shadowRoot.querySelector("img").getAttribute("src") !== src ) { link.shadowRoot.querySelector("img").setAttribute("src", src); } if (link.dataset.title) { link.shadowRoot.querySelector(".LinkCard-title").innerText = link.dataset.title.slice(5); } if (link.dataset.href) { link.shadowRoot.querySelector(".LinkCard-href").innerText = link.dataset.href } } } else{ link.shadowRoot?link.shadowRoot.innerHTML=`<span>${link.innerHTML}</span>`:null } }); }catch(e){} //出错时直接忽略不是好习惯,不过这里好像没有太大问题,因为触发频率比较高,如果在控制台输出错误信息很可能会卡死 } attachLinkShadow(a) { a.shadowRoot?a.shadowRoot.innerHTML='':null //下面这段添加了卡片链接样式 let linkStyle = document.createElement("style"); linkStyle.innerHTML = ` .LinkCard:hover{ background-color:var(--b3-theme-primary-lightest) !important } .LinkCard{ position: relative; display: -webkit-box; display: -webkit-flex; display: -ms-flexbox; display: flex; box-sizing: border-box; -webkit-flex-direction: row; -ms-flex-direction: row; flex-direction: row; -webkit-align-items: center; -webkit-box-align: center; -ms-flex-align: center; align-items: center; width: 60%; min-height: 84px; border-radius: 8px; max-width: 100%; overflow: hidden; margin: 16px auto; padding: 12px 12px 9px 12px; background-color: #F6F6F6; text-decoration:none } .LinkCard-contents { display: block; -webkit-flex: 1 1 auto; -ms-flex: 1 1 auto; flex: 1 1 auto; position: relative; white-space:normal !important; } .LinkCard-image { -webkit-flex: 0 0 auto; -ms-flex: 0 0 auto; flex: 0 0 auto; background-color:var(--b3-theme-surface); background-size: cover; background-position: center; position: relative; display: block; width: 60px; height: 60px; margin-left: 20px; object-fit: cover; border-radius: inherit; overflow: hidden; } .LinkCard-image img { width: 100%; height: 100%; object-fit: cover; top: 0; position: absolute; } .LinkCard-title { line-height: 20px; display: -webkit-box; text-overflow: ellipsis; overflow: hidden; color:var(--b3-theme-on-surface); -webkit-box-orient: vertical; -webkit-line-clamp: 2; } .LinkCard-href{ line-height: 15px; font-size:15px; display: -webkit-box; text-overflow: ellipsis; overflow: hidden; color:var(--b3-theme-on-surface); -webkit-box-orient: vertical; -webkit-line-clamp: 1; } `; let shadow = a.shadowRoot?a.shadowRoot:a.attachShadow({ mode: "open" }); shadow.innerHTML += `<div class="LinkCardContainer"> <a class='LinkCard' href='${a.dataset.href}'> <span class='LinkCard-contents'> <span class='LinkCard-title'>${a.dataset.title.slice( 5 )}</span> <span class='LinkCard-href'>${a.dataset.href }</span> </span> <span class="LinkCard-image"> <img src="${this.getImg(a.dataset.href)}"></img> </span> </a> </div>`; shadow.appendChild(linkStyle); } getImg(href) { //这里使用了icon4Tsundoku中的图标,故名思意,这些图标是从tsundoku主题偷来的.... //你也可以通过其他方式请求网站的图标,或者仅仅在远程请求失败的时候使用本地图标 let iconNameRes = naive.pathConstructor.naivePath() + "/script/plugin/corePlugins/linkCard/icon4Tsundoku/" + "link2.svg"; this.icons.forEach((iconName) => { if (iconName && iconName.split) { let iconURL = iconName.split(".")[0].replace("_", "."); if (href && href.indexOf(iconURL) >= 0) { //这里的URL可能有有点怪,因为为了方便这里是跟naive另外拉起的服务器请求图标的,在其它地方实现的时候可以忽略 iconNameRes = naive.pathConstructor.naivePath() + "/script/plugin/corePlugins/linkCard/icon4Tsundoku/" + iconName; } } }); return iconNameRes; } } //这段代码可以忽略,因为这个本来是naive主题的一个插件,所以会有这个指定生效环境的内容,实际上不仅仅可以在APP环境生效,我这里是为了对比性能在其他环境关闭了这个插件 export const environments = ["APP"];

效果类似这样

image.png

想要其他样式就改一下 style 部分的代码。

image.png

貌似是不影响链接编辑的,就这样吧。

因为实现得比较随便,老哥们实际用的时候自己改改代码吧

  • 思源笔记

    思源笔记是一款隐私优先的个人知识管理系统,支持完全离线使用,同时也支持端到端加密同步。

    融合块、大纲和双向链接,重构你的思维。

    26093 引用 • 108342 回帖

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • WebSocket

    WebSocket 是 HTML5 中定义的一种新协议,它实现了浏览器与服务器之间的全双工通信(full-duplex)。

    48 引用 • 206 回帖 • 285 关注
  • React

    React 是 Facebook 开源的一个用于构建 UI 的 JavaScript 库。

    192 引用 • 291 回帖 • 367 关注
  • 脑图

    脑图又叫思维导图,是表达发散性思维的有效图形思维工具 ,它简单却又很有效,是一种实用性的思维工具。

    32 引用 • 99 回帖
  • 心情

    心是产生任何想法的源泉,心本体会陷入到对自己本体不能理解的状态中,因为心能产生任何想法,不能分出对错,不能分出自己。

    59 引用 • 369 回帖 • 1 关注
  • Notion

    Notion - The all-in-one workspace for your notes, tasks, wikis, and databases.

    10 引用 • 77 回帖
  • 京东

    京东是中国最大的自营式电商企业,2015 年第一季度在中国自营式 B2C 电商市场的占有率为 56.3%。2014 年 5 月,京东在美国纳斯达克证券交易所正式挂牌上市(股票代码:JD),是中国第一个成功赴美上市的大型综合型电商平台,与腾讯、百度等中国互联网巨头共同跻身全球前十大互联网公司排行榜。

    14 引用 • 102 回帖 • 312 关注
  • CAP

    CAP 指的是在一个分布式系统中, Consistency(一致性)、 Availability(可用性)、Partition tolerance(分区容错性),三者不可兼得。

    12 引用 • 5 回帖 • 635 关注
  • SMTP

    SMTP(Simple Mail Transfer Protocol)即简单邮件传输协议,它是一组用于由源地址到目的地址传送邮件的规则,由它来控制信件的中转方式。SMTP 协议属于 TCP/IP 协议簇,它帮助每台计算机在发送或中转信件时找到下一个目的地。

    4 引用 • 18 回帖 • 631 关注
  • B3log

    B3log 是一个开源组织,名字来源于“Bulletin Board Blog”缩写,目标是将独立博客与论坛结合,形成一种新的网络社区体验,详细请看 B3log 构思。目前 B3log 已经开源了多款产品:SymSoloVditor思源笔记

    1063 引用 • 3455 回帖 • 150 关注
  • 创业

    你比 99% 的人都优秀么?

    82 引用 • 1395 回帖
  • Vim

    Vim 是类 UNIX 系统文本编辑器 Vi 的加强版本,加入了更多特性来帮助编辑源代码。Vim 的部分增强功能包括文件比较(vimdiff)、语法高亮、全面的帮助系统、本地脚本(Vimscript)和便于选择的可视化模式。

    29 引用 • 66 回帖
  • Access
    1 引用 • 3 回帖
  • OAuth

    OAuth 协议为用户资源的授权提供了一个安全的、开放而又简易的标准。与以往的授权方式不同之处是 oAuth 的授权不会使第三方触及到用户的帐号信息(如用户名与密码),即第三方无需使用用户的用户名与密码就可以申请获得该用户资源的授权,因此 oAuth 是安全的。oAuth 是 Open Authorization 的简写。

    36 引用 • 103 回帖 • 36 关注
  • 倾城之链
    23 引用 • 66 回帖 • 167 关注
  • Solo

    Solo 是一款小而美的开源博客系统,专为程序员设计。Solo 有着非常活跃的社区,可将文章作为帖子推送到社区,来自社区的回帖将作为博客评论进行联动(具体细节请浏览 B3log 构思 - 分布式社区网络)。

    这是一种全新的网络社区体验,让热爱记录和分享的你不再感到孤单!

    1443 引用 • 10082 回帖 • 498 关注
  • abitmean

    有点意思就行了

    34 关注
  • Flume

    Flume 是一套分布式的、可靠的,可用于有效地收集、聚合和搬运大量日志数据的服务架构。

    9 引用 • 6 回帖 • 662 关注
  • 学习

    “梦想从学习开始,事业从实践起步” —— 习近平

    172 引用 • 534 回帖
  • golang

    Go 语言是 Google 推出的一种全新的编程语言,可以在不损失应用程序性能的情况下降低代码的复杂性。谷歌首席软件工程师罗布派克(Rob Pike)说:我们之所以开发 Go,是因为过去 10 多年间软件开发的难度令人沮丧。Go 是谷歌 2009 发布的第二款编程语言。

    500 引用 • 1396 回帖 • 244 关注
  • Vditor

    Vditor 是一款浏览器端的 Markdown 编辑器,支持所见即所得、即时渲染(类似 Typora)和分屏预览模式。它使用 TypeScript 实现,支持原生 JavaScript、Vue、React 和 Angular。

    372 引用 • 1857 回帖 • 1 关注
  • SQLServer

    SQL Server 是由 [微软] 开发和推广的关系数据库管理系统(DBMS),它最初是由 微软、Sybase 和 Ashton-Tate 三家公司共同开发的,并于 1988 年推出了第一个 OS/2 版本。

    21 引用 • 31 回帖
  • 印象笔记
    3 引用 • 16 回帖
  • 创造

    你创造的作品可能会帮助到很多人,如果是开源项目的话就更赞了!

    186 引用 • 1021 回帖
  • Vue.js

    Vue.js(读音 /vju ː/,类似于 view)是一个构建数据驱动的 Web 界面库。Vue.js 的目标是通过尽可能简单的 API 实现响应的数据绑定和组合的视图组件。

    268 引用 • 666 回帖 • 1 关注
  • 游戏

    沉迷游戏伤身,强撸灰飞烟灭。

    187 引用 • 831 回帖
  • BND

    BND(Baidu Netdisk Downloader)是一款图形界面的百度网盘不限速下载器,支持 Windows、Linux 和 Mac,详细介绍请看这里

    107 引用 • 1281 回帖 • 36 关注
  • Wide

    Wide 是一款基于 Web 的 Go 语言 IDE。通过浏览器就可以进行 Go 开发,并有代码自动完成、查看表达式、编译反馈、Lint、实时结果输出等功能。

    欢迎访问我们运维的实例: https://wide.b3log.org

    30 引用 • 218 回帖 • 644 关注