From 03b0e73acb3531626adc070f51432d4db8392075 Mon Sep 17 00:00:00 2001 From: Libin YANG Date: Tue, 24 Dec 2024 08:46:21 +0800 Subject: [PATCH] chore: update copy func (#496) --- .../CodemirrorEditor/EditorHeader/index.vue | 294 ++++++++++-------- src/utils/index.ts | 74 ++++- 2 files changed, 241 insertions(+), 127 deletions(-) diff --git a/src/components/CodemirrorEditor/EditorHeader/index.vue b/src/components/CodemirrorEditor/EditorHeader/index.vue index 009dbc1..3555871 100644 --- a/src/components/CodemirrorEditor/EditorHeader/index.vue +++ b/src/components/CodemirrorEditor/EditorHeader/index.vue @@ -14,11 +14,26 @@ import { themeOptions, } from '@/config' import { useDisplayStore, useStore } from '@/stores' -import { addPrefix, mergeCss, solveWeChatImage } from '@/utils' -import { ChevronDownIcon, Moon, PanelLeftClose, PanelLeftOpen, Settings, Sun } from 'lucide-vue-next' +import { + addPrefix, + processClipboardContent, +} from '@/utils' +import { + ChevronDownIcon, + Moon, + PanelLeftClose, + PanelLeftOpen, + Settings, + Sun, +} from 'lucide-vue-next' import PickColors from 'vue-pick-colors' -const emit = defineEmits([`addFormat`, `formatContent`, `startCopy`, `endCopy`]) +const emit = defineEmits([ + `addFormat`, + `formatContent`, + `startCopy`, + `endCopy`, +]) const formatItems = [ { @@ -64,34 +79,10 @@ const copyMode = useStorage(addPrefix(`copyMode`), `txt`) const source = ref(``) const { copy: copyContent } = useClipboard({ source }) -function creatEmptyNode() { - const node = document.createElement(`p`) - node.style.fontSize = `0` - node.style.lineHeight = `0` - node.style.margin = `0` - node.innerHTML = ` ` - return node -} - // 复制到微信公众号 function copy() { emit(`startCopy`) setTimeout(() => { - function modifyHtmlStructure(htmlString: string) { - // 创建一个 div 元素来暂存原始 HTML 字符串 - const tempDiv = document.createElement(`div`) - tempDiv.innerHTML = htmlString - - const originalItems = tempDiv.querySelectorAll(`li > ul, li > ol`) - - originalItems.forEach((originalItem) => { - originalItem.parentElement!.insertAdjacentElement(`afterend`, originalItem) - }) - - // 返回修改后的 HTML 字符串 - return tempDiv.innerHTML - } - // 如果是深色模式,复制之前需要先切换到白天模式 const isBeforeDark = isDark.value if (isBeforeDark) { @@ -99,49 +90,11 @@ function copy() { } nextTick(async () => { - solveWeChatImage() - + processClipboardContent(primaryColor.value) const clipboardDiv = document.getElementById(`output`)! - clipboardDiv.innerHTML = mergeCss(clipboardDiv.innerHTML) - clipboardDiv.innerHTML = modifyHtmlStructure(clipboardDiv.innerHTML) - clipboardDiv.innerHTML = clipboardDiv.innerHTML - // 公众号不支持 position, 转换为等价的 translateY - .replace(/top:(.*?)em/g, `transform: translateY($1em)`) - // 适配主题中的颜色变量 - .replace(/hsl\(var\(--foreground\)\)/g, `#3f3f3f`) - .replace(/var\(--blockquote-background\)/g, `#f7f7f7`) - .replace(/var\(--md-primary-color\)/g, primaryColor.value) - .replace(/--md-primary-color:.+?;/g, ``) - .replace(/]*)>]*>(.*?)<\/p><\/span>/g, `$2`) - clipboardDiv.focus() - - // 由于 svg 无法复制, 在前后各插入一个空白节点 - const beforeNode = creatEmptyNode() - const afterNode = creatEmptyNode() - clipboardDiv.insertBefore(beforeNode, clipboardDiv.firstChild) - clipboardDiv.appendChild(afterNode) - - // 兼容 Mermaid - const nodes = clipboardDiv.querySelectorAll(`.nodeLabel`) - nodes.forEach((node) => { - const parent = node.parentElement! - const xmlns = parent.getAttribute(`xmlns`)! - const style = parent.getAttribute(`style`)! - const section = document.createElement(`section`) - section.setAttribute(`xmlns`, xmlns) - section.setAttribute(`style`, style) - section.innerHTML = parent.innerHTML - - const grand = parent.parentElement! - grand.innerHTML = `` - grand.appendChild(section) - }) - window.getSelection()!.removeAllRanges() - const temp = clipboardDiv.innerHTML - if (copyMode.value === `txt`) { const range = document.createRange() range.setStartBefore(clipboardDiv.firstChild!) @@ -150,19 +103,20 @@ function copy() { document.execCommand(`copy`) window.getSelection()!.removeAllRanges() } - clipboardDiv.innerHTML = output.value - if (isBeforeDark) { nextTick(() => toggleDark()) } - if (copyMode.value === `html`) { await copyContent(temp) } // 输出提示 - toast.success(copyMode.value === `html` ? `已复制 HTML 源码,请进行下一步操作。` : `已复制渲染后的文章到剪贴板,可直接到公众号后台粘贴。`) + toast.success( + copyMode.value === `html` + ? `已复制 HTML 源码,请进行下一步操作。` + : `已复制渲染后的文章到剪贴板,可直接到公众号后台粘贴。`, + ) editorRefresh() emit(`endCopy`) @@ -177,7 +131,9 @@ function customStyle() { }, 50) } -const pickColorsContainer = useTemplateRef(`pickColorsContainer`) +const pickColorsContainer = useTemplateRef( + `pickColorsContainer`, +) const format = ref(`rgb`) const formatOptions = ref([`rgb`, `hex`, `hsl`, `hsv`]) @@ -191,18 +147,30 @@ const formatOptions = ref([`rgb`, `hex`, `hsl`, `hsv`]) 格式 {{ label }} - + {{ item }} - + 微信外链转底部引用 @@ -212,10 +180,20 @@ const formatOptions = ref([`rgb`, `hex`, `hsl`, `hsv`]) - - @@ -230,9 +208,14 @@ const formatOptions = ref([`rgb`, `hex`, `hsl`, `hsv`])

主题

@@ -242,8 +225,13 @@ const formatOptions = ref([`rgb`, `hex`, `hsl`, `hsv`])

字体

@@ -266,12 +259,19 @@ const formatOptions = ref([`rgb`, `hex`, `hsl`, `hsv`])

主题色

@@ -325,16 +338,22 @@ const formatOptions = ref([`rgb`, `hex`, `hsl`, `hsv`])

Mac 代码块

@@ -344,16 +363,22 @@ const formatOptions = ref([`rgb`, `hex`, `hsl`, `hsv`])

微信外链转底部引用

@@ -363,16 +388,22 @@ const formatOptions = ref([`rgb`, `hex`, `hsl`, `hsv`])

段落首行缩进

@@ -382,16 +413,23 @@ const formatOptions = ref([`rgb`, `hex`, `hsl`, `hsv`])

自定义 CSS 面板

@@ -401,16 +439,22 @@ const formatOptions = ref([`rgb`, `hex`, `hsl`, `hsv`])

编辑区位置

@@ -420,16 +464,22 @@ const formatOptions = ref([`rgb`, `hex`, `hsl`, `hsv`])

模式

@@ -445,7 +495,9 @@ const formatOptions = ref([`rgb`, `hex`, `hsl`, `hsv`]) -
+
@@ -456,11 +508,7 @@ const formatOptions = ref([`rgb`, `hex`, `hsl`, `hsv`]) - + 公众号格式 diff --git a/src/utils/index.ts b/src/utils/index.ts index 8996087..f354bd0 100644 --- a/src/utils/index.ts +++ b/src/utils/index.ts @@ -340,20 +340,86 @@ export function removeLeft(str: string) { export function solveWeChatImage() { const clipboardDiv = document.getElementById(`output`)! const images = clipboardDiv.getElementsByTagName(`img`) - for (let i = 0; i < images.length; i++) { - const image = images[i] + + Array.from(images).forEach((image) => { const width = image.getAttribute(`width`)! const height = image.getAttribute(`height`)! image.removeAttribute(`width`) image.removeAttribute(`height`) image.style.width = width image.style.height = height - } + }) } -export function mergeCss(html: string) { +export function mergeCss(html: string): string { return juice(html, { inlinePseudoElements: true, preserveImportant: true, }) } + +export function createEmptyNode(): HTMLElement { + const node = document.createElement(`p`) + node.style.fontSize = `0` + node.style.lineHeight = `0` + node.style.margin = `0` + node.innerHTML = ` ` + return node +} + +export function modifyHtmlStructure(htmlString: string): string { + const tempDiv = document.createElement(`div`) + tempDiv.innerHTML = htmlString + + // 移动 `li > ul` 和 `li > ol` 到 `li` 后面 + tempDiv.querySelectorAll(`li > ul, li > ol`).forEach((originalItem) => { + originalItem.parentElement!.insertAdjacentElement(`afterend`, originalItem) + }) + + return tempDiv.innerHTML +} + +export function processClipboardContent(primaryColor: string) { + const clipboardDiv = document.getElementById(`output`)! + + // 先合并 CSS 和修改 HTML 结构 + clipboardDiv.innerHTML = modifyHtmlStructure(mergeCss(clipboardDiv.innerHTML)) + + // 处理样式和颜色变量 + clipboardDiv.innerHTML = clipboardDiv.innerHTML + .replace(/top:(.*?)em/g, `transform: translateY($1em)`) + .replace(/hsl\(var\(--foreground\)\)/g, `#3f3f3f`) + .replace(/var\(--blockquote-background\)/g, `#f7f7f7`) + .replace(/var\(--md-primary-color\)/g, primaryColor) + .replace(/--md-primary-color:.+?;/g, ``) + .replace( + /]*)>]*>(.*?)<\/p><\/span>/g, + `$2`, + ) + + // 处理图片大小 + solveWeChatImage() + + // 添加空白节点用于兼容 SVG 复制 + const beforeNode = createEmptyNode() + const afterNode = createEmptyNode() + clipboardDiv.insertBefore(beforeNode, clipboardDiv.firstChild) + clipboardDiv.appendChild(afterNode) + + // 兼容 Mermaid + const nodes = clipboardDiv.querySelectorAll(`.nodeLabel`) + nodes.forEach((node) => { + const parent = node.parentElement! + const xmlns = parent.getAttribute(`xmlns`)! + const style = parent.getAttribute(`style`)! + const section = document.createElement(`section`) + section.setAttribute(`xmlns`, xmlns) + section.setAttribute(`style`, style) + section.innerHTML = parent.innerHTML + + const grand = parent.parentElement! + // 清空父元素 + grand.innerHTML = `` + grand.appendChild(section) + }) +}