{
"title": "Obsidian|给分享内容快速生成卡片",
"tags": [
"post",
"Obsidian"
],
"sources": [
"xlog"
],
"external_urls": [
"https://weiyexing.xlog.app/Obsidian-content-to-twee-card"
],
"date_published": "2023-05-21T03:28:37.000Z",
"content": "---\ntitle: \"Obsidian|给分享内容快速生成卡片\"\ndate: 2023-05-21T11:28:37+08:00\nslug: \"Obsidian content to tweet card\"\ntags: [Obsidian]\ndescription: \"简单快速,实用美观。\"\ndraft: false\n---\n\n![](https://wesson-image.oss-cn-beijing.aliyuncs.com/img/202305211133366.png)\n\n在之前的推送中,分享了一个快速把分享内容变成卡片的方式,现在把完整流程分享下。\n\n### 前置需求\n\n- Templater:安装社区市场的 [Templater](https://github.com/SilentVoid13/Templater) 插件并打开\n- 保存模板文件(tp-生成文字卡片)到指定文件夹下:点击下载[模板示例](https://note.ms/cardmb),或在下方附录复制\n- 保存脚本 js 文件(get_tweet_card)到指定文件夹下:点击下载 [js 文件完整代码](https://note.ms/cardjs),或在下方附录复制\n\n注:\n1. 在 Templater 插件里设置好模板文件夹、脚本文件夹,位置分别在 Template folder location、Script files folder location。\n2. 可设置不同模板,配置不同的头像以及昵称,生成不同的卡片\n\n### 操作步骤\n\n以上工作准备就绪后,就可以愉快地使用了,具体步骤如下:\n\n1. 在 `设置-快捷键` 配置好唤出 Templater 模板的快捷键,我设置的是 `⌘+/`\n2. 在 Obsidian 任意文件内,选中一段文字,按下 `⌘+/`,选择模板「tp-生成文字卡片」,回车,卡片就复制到粘贴板了\n\n![](https://wesson-image.oss-cn-beijing.aliyuncs.com/img/202305211143549.gif)\n\n---\n\n### 附录\n\n模板示例\n\n```\n<% tp.user.get_tweet_card(tp, {\n width: 1800,\n fontSize: 62,\n margin: 140,\n padding: 100,\n writeToClipboard: true,\n downloadToDisk: false,\n logo: `这里放卡片里你的头像 base64代码,例如可以在这样的网站转换 https://c.runoob.com/front-end/59/` ,\n name: '你的昵称',\n userId: '你的 ID'\n}) %>\n```\n\njs 文件代码\n\n```JavaScript\n/** @type {object} 设置项 */\nlet opt = {}\n\n/**\n * 初始化选项\n *\n * @param {object} input\n */\nconst initOpt = (input, tp) => {\n opt = Object.assign({\n size: 'M',\n logo: AppLogo,\n appLogo: AppLogo,\n name: '这里是用户名',\n userId: '@User_ID or anything',\n bgColors: [\"#ffafbd\", \"#ffc3a0\"],\n cardBgColor: 'rgba(255, 255, 255, .8)',\n contetnColor: '#333336',\n nameColor: '#333336',\n userIdColor: '#333336',\n timeColor: 'rgba(0, 0, 0, .5)',\n writeToClipboard: true,\n writeToDocument: false,\n downloadToDisk: false,\n }, input ? input : {})\n /** ==== 如未设定,则计算默认值 ==== */\n /**\n * 如果属性不存在,则计算默认值\n *\n * @param {*} key\n * @param {*} defVal\n */\n const setSubOpt = (key, defVal) => {\n if (!opt[key]) opt[key] = defVal\n }\n /** 图片宽度 */\n if (!opt.width) {\n switch (opt.size) {\n case 'S':\n opt.width = 480\n break;\n case 'M':\n opt.width = 700\n break;\n case 'L':\n opt.width = 960\n break;\n\n default:\n opt.width = 700\n break;\n }\n }\n /** 文字大小 */\n setSubOpt('fontSize', Math.round(opt.width / 30))\n setSubOpt('smallFontSize', Math.round(opt.fontSize * 0.6))\n /** 行高 */\n setSubOpt('lineHeight', 1.6)\n /** 段首缩进 */\n setSubOpt('indent', opt.fontSize * 2) /** 设置为0则不缩进 */\n /** 字体 */\n setSubOpt('fontFamily', 'Menlo, SFMono-Regular, Consolas, \"Roboto Mono\", \"Source Code Pro\", ui-sans-serif, -apple-system, BlinkMacSystemFont, \"Segoe UI\", Roboto, \"Inter\", \"Apple Color Emoji\", \"Segoe UI Emoji\", \"Segoe UI Symbol\", \"Microsoft YaHei\", sans-serif')\n /** 卡片外补 */\n setSubOpt('margin', Math.round(opt.width / 15))\n setSubOpt('marginLR', opt.margin)\n setSubOpt('marginTB', opt.margin)\n /** 卡片内补 */\n setSubOpt('padding', Math.round(opt.width / 12))\n setSubOpt('paddingLR', opt.padding)\n setSubOpt('paddingTB', opt.padding)\n /** Logo 尺寸 */\n setSubOpt('logoSize', 2 * opt.fontSize)\n /** 卡片圆角 */\n setSubOpt('cardRadius', Math.round(opt.fontSize / 2))\n\n /** ==== 必须通过计算得出的值 ==== */\n\n opt.cardWidth = opt.width - opt.marginLR * 2\n opt.contentWidth = opt.cardWidth - opt.paddingLR * 2\n opt.contentMarginLR = opt.marginLR + opt.paddingLR\n opt.contentMarginTB = opt.marginTB + opt.paddingTB\n opt.paragraphsMarginBottom = Math.round(opt.fontSize / 2)\n}\n\n/**\n * 数字两位化\n *\n * @param {number} num 0~99 的整数\n * @returnn {string}\n */\nconst dbNum = num => (num > 9 ? String(num) : '0' + num);\n/** @type {array} */\nconst daysName = ['Sun.', 'Mon.', 'Tues.', 'Wed.', 'Thur.', 'Fri.', 'Sat.']\n/**\n * 获取当前时间字符串\n *\n * @return {string} \n */\nconst getNowTime = () => {\n const now = new Date()\n const t = {\n YYYY: now.getFullYear(),\n MM: dbNum(now.getMonth() + 1),\n DD: dbNum(now.getDate()),\n hh: dbNum(now.getHours()),\n mm: dbNum(now.getMinutes()),\n ss: dbNum(now.getSeconds()),\n EE: daysName[now.getDay()]\n }\n return `${t.YYYY}-${t.MM}-${t.DD} ${t.EE} ${t.hh}:${t.mm}:${t.ss}`\n}\n// 创建画布对象\nconst canvas = document.createElement('canvas')\nconst ctx = canvas.getContext('2d')\n\n/**\n * 画布文字逐行分割\n *\n * @param {object} ctx 画布上下文对象\n * @param {string} text 要写入的文字内容\n * @param {number} width 文字内容在画布中占据的宽度\n * @return {array} 二维数组,第1层是段落,第2层是段落中的每一行\n */\nconst canvasTextSplit = (text, width) => {\n text = text.trim()\n if (text.length === 0) return []\n const result = []\n // 先进行段落的分割\n const paragraphArray = text.replace(/(\\r?\\n\\s*)+/g, '\\n').split(/\\s*\\r?\\n\\s*/g)\n for (const p of paragraphArray) {\n const linesInParagraph = []\n let nowLetter = 0\n for (let i = 0; i <= p.length; i++) {\n const thisLineWidth = linesInParagraph.length ? width : width - opt.indent\n if (ctx.measureText(p.substring(nowLetter, i)).width > thisLineWidth) {\n linesInParagraph.push(p.substring(nowLetter, i - 1))\n nowLetter = i - 1\n } else if (i === p.length) {\n linesInParagraph.push(p.substring(nowLetter, i))\n }\n }\n result.push(linesInParagraph)\n }\n return result\n}\n/**\n * 将段落数组中的文字绘制到画布\n *\n * @param {object} ctx 画布上下文对象\n * @param {array} paragraphs 二维数组,第1层是段落,第2层是段落中的每一行\n * @param {number} startX 起始的横坐标\n * @param {number} startY 起始的纵坐标\n * @param {number} opt.lineHeight 行高\n * @return {number} 结束位置的纵坐标\n */\nconst drawText = async (paragraphs, startX, startY) => {\n let thisLineY = startY\n paragraphs.forEach((p, pIndex) => {\n p.forEach((line, lIndex) => {\n const thisLineX = lIndex ? startX : startX + opt.indent\n thisLineY += opt.lineHeight * opt.fontSize\n ctx.fillText(line, thisLineX, thisLineY)\n })\n thisLineY += opt.paragraphsMarginBottom\n })\n return thisLineY\n}\n/**\n * 计算绘制文字所需要占据的高度\n *\n * @param {array} paragraphs 二维数组,第1层是段落,第2层是段落中的每一行\n * @param {number} opt.lineHeight 行高\n * @return {number} 文字内容所占据的高度\n */\nconst textNeedHeight = (paragraphs) => {\n return (paragraphs.length - 1) * opt.paragraphsMarginBottom\n + paragraphs.flat().length * opt.lineHeight * opt.fontSize\n}\n/**\n * 将 base64 格式的图片转换为 Blob 格式数据\n *\n * @param {string} dataUrl base64 格式的数据地址\n * @return {object} Blob 格式的图片数据\n */\nconst dataURLtoBlob = dataUrl => {\n const dataArr = dataUrl.split(',');\n const mime = dataArr[0].match(/:(.*?);/)[1];\n const bStr = atob(dataArr[1]);\n let n = bStr.length;\n const uint8Arr = new Uint8Array(n);\n while (n--) {\n uint8Arr[n] = bStr.charCodeAt(n);\n }\n return new Blob([uint8Arr], { type: mime });\n}\n/**\n * 将画布保存为图片并自动进行下载\n *\n * @param {object} canvas 画布对象\n * @param {string} name 保存的文件名\n * @param {string} [type=\"png\"] 文件图片的格式: png、jpeg、gif\n */\nconst downloadImgFromCanvas = (name) => {\n // const imgDataUrl = canvas.toDataURL('image/'+type)\n const imgDataUrl = canvas.toDataURL({ format: 'png', quality: 1 })\n const blob = dataURLtoBlob(imgDataUrl)\n const blobUrl = URL.createObjectURL(blob)\n const imgDownloadLink = document.createElement('a')\n imgDownloadLink.download = name + '.png'\n imgDownloadLink.href = blobUrl\n imgDownloadLink.click();\n}\n\n/**\n * 设置填充色\n *\n * @param {string|array} colors\n */\nconst setFillColor = colors => {\n let fillColor\n if (typeof (colors) === 'string') {\n fillColor = colors\n } else if (colors.length === 1) {\n fillColor = colors[0]\n } else {\n fillColor = ctx.createLinearGradient(0, 0, opt.width, opt.width / 8);\n const pointStep = 1 / (colors.length - 1)\n colors.forEach((c, i) => {\n fillColor.addColorStop(i * pointStep, c);\n })\n }\n ctx.fillStyle = fillColor\n}\n/**\n * 画布字体设置\n *\n * @param {string|number} size\n * @param {string} color\n * @param {string} [weight='normal']\n * @param {string} [align='left']\n */\nconst setFont = (size, color, weight = 'normal', align = 'left') => {\n ctx.font = weight + ' ' + size + 'px ' + opt.fontFamily\n ctx.textAlign = align\n ctx.fillStyle = color\n}\n/**\n * 设置画布阴影\n *\n * @param {number} x\n * @param {number} y\n * @param {number} blur\n * @param {string} [color='rgba(0, 0, 0, 0)']\n */\nconst setShadow = (x, y, blur, color = 'rgba(0, 0, 0, 0)') => {\n ctx.shadowOffsetX = x\n ctx.shadowOffsetY = y\n ctx.shadowBlur = blur\n ctx.shadowColor = color\n}\n/**\n * 重置画布对象\n *\n * @param {number} height 画布的高度\n * @param {string} fillColor 画布填充的背景颜色\n */\nconst canvasRest = height => {\n canvas.width = opt.width\n canvas.height = height\n setShadow(0, 0, 0)\n setFillColor(opt.bgColors)\n ctx.fillRect(0, 0, canvas.width, canvas.height)\n}\n\n/**\n * 绘制圆角矩形\n *\n * @param {number} x\n * @param {number} y\n * @param {number} w\n * @param {number} h\n * @param {number} r\n */\nconst drawRoundedRect = (x, y, w, h, r) => {\n var ptA = { x: x + r, y: y }\n var ptB = { x: x + w, y: y }\n var ptC = { x: x + w, y: y + h }\n var ptD = { x: x, y: y + h }\n var ptE = { x: x, y: y }\n\n ctx.beginPath();\n\n ctx.moveTo(ptA.x, ptA.y);\n ctx.arcTo(ptB.x, ptB.y, ptC.x, ptC.y, r);\n ctx.arcTo(ptC.x, ptC.y, ptD.x, ptD.y, r);\n ctx.arcTo(ptD.x, ptD.y, ptE.x, ptE.y, r);\n ctx.arcTo(ptE.x, ptE.y, ptA.x, ptA.y, r);\n\n ctx.closePath()\n // ctx.stroke();\n ctx.fill()\n}\n\n/**\n * 同步载入图片\n *\n * @param {string} url\n * @param {number} l\n * @param {number} t\n */\nconst loadImage = async (url, l, t) => new Promise(resolve => {\n const img = new Image()\n img.onload = () => {\n ctx.drawImage(img, l, t, opt.logoSize, opt.logoSize)\n return resolve(true)\n }\n img.src = url\n});\n\n/**\n * \n *\n * @param {*} tp\n * @return {*} \n */\nasync function get_tweet_card(tp, input) {\n let selectedText = window.getSelection().toLocaleString() // 获取选中的文字\n\n /** @type {string} 获取输入 */\n const inputContent = await tp.system.prompt('输入内容', selectedText, false, true)\n if (!inputContent) return selectedText\n\n /** 初始化选项 */\n initOpt(input, tp)\n\n /** 整理内容,计算尺寸 */\n setFont(opt.fontSize, opt.contetnColor)\n const contentArr = canvasTextSplit(inputContent, opt.contentWidth)\n opt.contentHeight = textNeedHeight(contentArr)\n opt.cardHeight = opt.contentHeight\n + opt.paddingTB * 2\n + opt.logoSize\n + opt.lineHeight * opt.fontSize /** 用来书写时间 */\n + 2 * opt.paragraphsMarginBottom /** 放在内容上下 */\n opt.height = opt.cardHeight + 2 * opt.marginTB\n /** 初始化画布 */\n canvasRest(opt.height)\n /** 绘制卡片 */\n setShadow(0, 0, opt.margin * 0.6, 'rgba(0, 0, 0, .3)')\n ctx.fillStyle = opt.cardBgColor\n drawRoundedRect(opt.marginLR, opt.marginTB, opt.cardWidth, opt.cardHeight, opt.cardRadius)\n\n /** 绘制内容文字 */\n setFont(opt.fontSize, opt.contetnColor)\n setShadow(0, 0, 0)\n drawText(contentArr, opt.contentMarginLR, opt.contentMarginTB + opt.logoSize + opt.paragraphsMarginBottom)\n /** 绘制用户名 */\n setFont(opt.smallFontSize, opt.nameColor, '700')\n ctx.fillText(opt.name, opt.contentMarginLR + opt.logoSize + opt.smallFontSize, opt.contentMarginTB + Math.round(opt.logoSize / 2));\n /** 绘制 UserID */\n setFont(opt.smallFontSize, opt.userIdColor, '200')\n ctx.fillText(opt.userId, opt.contentMarginLR + opt.logoSize + opt.smallFontSize, opt.contentMarginTB + Math.round(opt.logoSize * 0.98));\n\n /** 绘制时间 */\n setFont(opt.smallFontSize, opt.timeColor, '200', 'right')\n const nowTime = getNowTime()\n ctx.fillText(nowTime, opt.width - opt.marginLR - opt.paddingLR, canvas.height - opt.marginTB - opt.paddingTB);\n\n /** 绘制头像 */\n await loadImage(opt.logo, opt.contentMarginLR, opt.contentMarginTB)\n await loadImage(opt.appLogo, canvas.width - opt.marginLR - opt.paddingLR / 2 - opt.logoSize, opt.marginTB + opt.paddingTB / 2)\n\n /** 输出 */\n // 1. 输出到剪贴板\n if (opt.writeToClipboard) {\n await new Promise(async (reslove) => {\n canvas.toBlob(async (blob) => {\n // debugger\n let res = await navigator.clipboard.write([new ClipboardItem({\n [blob.type]: blob\n })]).then(() => {\n // 提示框\n let notice = new tp.obsidian.Notice()\n notice.setMessage(\"picture copied ~\")\n setTimeout(notice.hide, 2000)\n }).catch(err => {\n let notice = new tp.obsidian.Notice()\n notice.setMessage(\"picture write to clipboard fail\")\n setTimeout(notice.hide, 2000)\n \n throw new Error(err)\n })\n\n reslove()\n })\n })\n }\n\n // 2. 下载到本地\n if (opt.downloadToDisk) {\n downloadImgFromCanvas(nowTime)\n }\n\n // 3. 直接写到文档中\n if (opt.writeToDocument) {\n return selectedText + '\\n\\n' + '![](' + canvas.toDataURL('image/png') + ')'\n }\n\n return selectedText\n}\n/** Obsidian Logo 256*256 */\nconst AppLogo = ``\nmodule.exports = get_tweet_card;\n```",
"attributes": [
{
"value": "Obsidian-content-to-twee-card",
"trait_type": "xlog_slug"
}
]
}