md/src/utils/index.js

387 lines
10 KiB
JavaScript
Raw Normal View History

2024-08-18 19:49:16 +08:00
import juice from 'juice'
import prettier from 'prettier/standalone'
import prettierCss from 'prettier/parser-postcss'
import prettierMarkdown from 'prettier/parser-markdown'
2020-02-15 21:44:42 +08:00
2024-08-20 12:39:32 +08:00
import { prefix } from '@/config'
2024-08-18 19:49:16 +08:00
export function addPrefix(str) {
return `${prefix}__${str}`
}
function createCustomTheme(theme, color, isDefault = true) {
const customTheme = JSON.parse(JSON.stringify(theme))
customTheme.block.h1[`border-bottom`] = `2px solid ${color}`
2024-08-18 19:49:16 +08:00
customTheme.block.h2.background = color
customTheme.block.h3[`border-left`] = `3px solid ${color}`
2024-08-18 19:49:16 +08:00
customTheme.block.h4.color = color
customTheme.inline.strong.color = color
if (!isDefault) {
customTheme.block.h3[`border-bottom`] = `1px dashed ${color}`
customTheme.block.blockquote[`border-left`] = `4px solid ${color}`
}
return customTheme
}
2020-01-13 22:16:04 +08:00
// 设置自定义颜色
export function setColorWithTemplate(theme) {
return (color) => {
return createCustomTheme(theme, color)
}
}
export function setColorWithCustomTemplate(theme, color, isDefault = true) {
return createCustomTheme(theme, color, isDefault)
}
2020-01-13 22:16:04 +08:00
// 设置自定义字体大小
export function setFontSizeWithTemplate(template) {
return function (fontSize, isDefault = true) {
const customTheme = JSON.parse(JSON.stringify(template))
if (isDefault) {
customTheme.block.h1[`font-size`] = `${fontSize * 1.2}px`
customTheme.block.h2[`font-size`] = `${fontSize * 1.2}px`
customTheme.block.h3[`font-size`] = `${fontSize * 1.1}px`
customTheme.block.h4[`font-size`] = `${fontSize}px`
}
else {
customTheme.block.h1[`font-size`] = `${fontSize * 1.4}px`
customTheme.block.h2[`font-size`] = `${fontSize * 1.3}px`
customTheme.block.h3[`font-size`] = `${fontSize * 1.2}px`
customTheme.block.h4[`font-size`] = `${fontSize * 1.1}px`
}
return customTheme
}
2020-01-13 22:16:04 +08:00
}
export function setTheme(theme, fontSize, color, isDefault) {
return setColorWithCustomTemplate(setFontSizeWithTemplate(theme)(fontSize, isDefault), color, isDefault)
}
2020-01-13 22:16:04 +08:00
export function customCssWithTemplate(jsonString, color, theme) {
// block
const customTheme = createCustomTheme(theme, color)
customTheme.block.h1 = Object.assign(customTheme.block.h1, jsonString.h1)
customTheme.block.h2 = Object.assign(customTheme.block.h2, jsonString.h2)
customTheme.block.h3 = Object.assign(customTheme.block.h3, jsonString.h3)
customTheme.block.h4 = Object.assign(customTheme.block.h4, jsonString.h4)
customTheme.block.code = Object.assign(
customTheme.block.code,
2024-08-18 19:49:16 +08:00
jsonString.code,
)
customTheme.block.p = Object.assign(customTheme.block.p, jsonString.p)
customTheme.block.hr = Object.assign(customTheme.block.hr, jsonString.hr)
customTheme.block.blockquote = Object.assign(
customTheme.block.blockquote,
2024-08-18 19:49:16 +08:00
jsonString.blockquote,
)
customTheme.block.blockquote_p = Object.assign(
customTheme.block.blockquote_p,
2024-08-18 19:49:16 +08:00
jsonString.blockquote_p,
)
customTheme.block.image = Object.assign(
customTheme.block.image,
2024-08-18 19:49:16 +08:00
jsonString.image,
)
// inline
customTheme.inline.strong = Object.assign(
customTheme.inline.strong,
2024-08-18 19:49:16 +08:00
jsonString.strong,
)
customTheme.inline.codespan = Object.assign(
customTheme.inline.codespan,
2024-08-18 19:49:16 +08:00
jsonString.codespan,
)
customTheme.inline.link = Object.assign(
customTheme.inline.link,
2024-08-18 19:49:16 +08:00
jsonString.link,
)
customTheme.inline.wx_link = Object.assign(
customTheme.inline.wx_link,
2024-08-18 19:49:16 +08:00
jsonString.wx_link,
)
customTheme.block.ul = Object.assign(customTheme.block.ul, jsonString.ul)
customTheme.block.ol = Object.assign(customTheme.block.ol, jsonString.ol)
customTheme.inline.listitem = Object.assign(
customTheme.inline.listitem,
2024-08-18 19:49:16 +08:00
jsonString.li,
)
return customTheme
2020-01-13 22:16:04 +08:00
}
/**
2024-08-20 19:01:44 +08:00
* CSS 字符串转换为 JSON 对象
2020-01-13 22:16:04 +08:00
*
2024-08-20 19:01:44 +08:00
* @param {string} css - CSS 字符串
* @returns {object} - JSON 格式的 CSS
2020-01-13 22:16:04 +08:00
*/
export function css2json(css) {
2024-08-20 19:01:44 +08:00
// 去除所有 CSS 注释
css = css.replace(/\/\*[\s\S]*?\*\//g, ``)
2024-08-18 19:49:16 +08:00
const json = {}
2024-08-20 19:01:44 +08:00
// 辅助函数:将声明数组转换为对象
const toObject = array =>
array.reduce((obj, item) => {
const [property, value] = item.split(`:`).map(part => part.trim())
if (property)
obj[property] = value
return obj
}, {})
while (css.includes(`{`) && css.includes(`}`)) {
const lbracket = css.indexOf(`{`)
const rbracket = css.indexOf(`}`)
2024-08-20 19:01:44 +08:00
// 获取声明块并转换为对象
const declarations = css.substring(lbracket + 1, rbracket)
.split(`;`)
2024-08-18 19:49:16 +08:00
.map(e => e.trim())
2024-08-20 19:01:44 +08:00
.filter(Boolean)
2024-08-20 19:01:44 +08:00
// 获取选择器并去除空格
const selectors = css.substring(0, lbracket)
.split(`,`)
2024-08-18 19:49:16 +08:00
.map(selector => selector.trim())
2024-08-20 19:01:44 +08:00
const declarationObj = toObject(declarations)
// 将声明对象关联到相应的选择器
selectors.forEach((selector) => {
2024-08-20 19:01:44 +08:00
json[selector] = { ...(json[selector] || {}), ...declarationObj }
})
2020-01-13 22:16:04 +08:00
2024-08-20 19:01:44 +08:00
// 处理下一个声明块
css = css.slice(rbracket + 1).trim()
}
return json
2020-01-13 22:16:04 +08:00
}
2020-05-01 21:30:25 +08:00
/**
* 将编辑器内容保存到 LocalStorage
2020-10-20 11:43:11 +00:00
* @param {*} editor
* @param {*} name
2020-05-01 21:30:25 +08:00
*/
export function saveEditorContent(editor, name) {
const content = editor.getValue(0)
if (content) {
localStorage.setItem(name, content)
2024-08-18 19:49:16 +08:00
}
else {
localStorage.removeItem(name)
}
2020-05-01 21:30:25 +08:00
}
2020-11-21 09:33:45 +08:00
/**
* 格式化文档
* @param {string} content - 文档内容
2020-11-21 09:33:45 +08:00
*/
2020-05-31 17:30:44 +08:00
export function formatDoc(content) {
return prettier.format(content, {
parser: `markdown`,
plugins: [prettierMarkdown],
})
}
2020-07-04 00:55:40 +08:00
2020-12-03 00:51:39 +08:00
/**
* 格式化css
* @param {string} content - css内容
2020-12-03 00:51:39 +08:00
*/
export function formatCss(content) {
return prettier.format(content, {
parser: `css`,
plugins: [prettierCss],
})
2020-12-03 00:51:39 +08:00
}
2020-11-21 09:33:45 +08:00
/**
* 导出原始 Markdown 文档
* @param {string} doc - 文档内容
2020-11-21 09:33:45 +08:00
*/
2020-11-22 22:36:35 +08:00
export function downloadMD(doc) {
const downLink = document.createElement(`a`)
2020-07-13 00:26:29 +08:00
downLink.download = `content.md`
downLink.style.display = `none`
const blob = new Blob([doc])
2020-07-13 00:26:29 +08:00
downLink.href = URL.createObjectURL(blob)
document.body.appendChild(downLink)
downLink.click()
document.body.removeChild(downLink)
2020-08-31 20:48:22 +08:00
}
2020-10-13 00:07:14 +08:00
2021-10-14 20:12:19 +08:00
/**
* 导出 HTML 生成内容
*/
export function exportHTML() {
const element = document.querySelector(`#output`)
setStyles(element)
const htmlStr = element.innerHTML
const downLink = document.createElement(`a`)
2021-10-14 20:12:19 +08:00
downLink.download = `content.html`
downLink.style.display = `none`
2024-08-18 19:49:16 +08:00
const blob = new Blob([
`<html><head><meta charset="utf-8" /></head><body><div style="width: 750px; margin: auto;">${htmlStr}</div></body></html>`,
])
2021-10-14 20:12:19 +08:00
downLink.href = URL.createObjectURL(blob)
document.body.appendChild(downLink)
downLink.click()
document.body.removeChild(downLink)
function setStyles(element) {
/**
* 获取一个 DOM 元素的所有样式
* @param {DOM 元素} element DOM 元素
* @param {排除的属性} excludes 如果某些属性对结果有不良影响可以使用这个参数来排除
* @returns 行内样式拼接结果
*/
function getElementStyles(element, excludes = [`width`, `height`]) {
const styles = getComputedStyle(element, null)
return Object.entries(styles)
.filter(
2024-08-18 19:49:16 +08:00
([key]) => styles.getPropertyValue(key) && !excludes.includes(key),
)
.map(([key, value]) => `${key}:${value};`)
.join(``)
}
switch (true) {
case isPre(element):
case isCode(element):
case isSpan(element):
element.setAttribute(`style`, getElementStyles(element))
// eslint-disable-next-line no-fallthrough
default:
}
if (element.children.length) {
2024-08-18 19:49:16 +08:00
Array.from(element.children).forEach(child => setStyles(child))
}
// 判断是否是包裹代码块的 pre 元素
function isPre(element) {
return (
2024-08-18 19:49:16 +08:00
element.tagName === `PRE`
&& Array.from(element.classList).includes(`code__pre`)
)
}
// 判断是否是包裹代码块的 code 元素
function isCode(element) {
return element.tagName === `CODE`
}
// 判断是否是包裹代码字符的 span 元素
function isSpan(element) {
return (
2024-08-18 19:49:16 +08:00
element.tagName === `SPAN`
&& (isCode(element.parentElement)
|| isCode(element.parentElement.parentElement))
)
}
}
2021-10-14 20:12:19 +08:00
}
2020-10-13 00:07:14 +08:00
/**
2024-08-20 12:39:32 +08:00
* 根据数据生成 Markdown 表格
*
* @param {object} options - 选项
* @param {object} options.data - 表格数据
* @param {number} options.rows - 行数
* @param {number} options.cols - 列数
* @returns {string} 生成的 Markdown 表格
2020-10-13 00:07:14 +08:00
*/
2020-10-20 11:43:11 +00:00
export function createTable({ data, rows, cols }) {
let table = ``
for (let i = 0; i < rows + 2; ++i) {
table += `| `
const currRow = []
for (let j = 0; j < cols; ++j) {
const rowIdx = i > 1 ? i - 1 : i
currRow.push(i === 1 ? `---` : data[`k_${rowIdx}_${j}`] || ` `)
2020-10-13 00:07:14 +08:00
}
table += currRow.join(` | `)
table += ` |\n`
}
2020-10-13 00:07:14 +08:00
return table
2020-10-20 11:43:11 +00:00
}
2020-12-04 00:57:46 +08:00
export function toBase64(file) {
return new Promise((resolve, reject) => {
const reader = new FileReader()
reader.readAsDataURL(file)
reader.onload = () => resolve(reader.result.split(`,`).pop())
2024-08-18 19:49:16 +08:00
reader.onerror = error => reject(error)
})
}
2020-12-05 21:16:09 +08:00
export function checkImage(file) {
2024-08-20 12:39:32 +08:00
// 检查文件名后缀
2024-08-20 19:01:44 +08:00
const isValidSuffix = /\.(?:gif|jpe?g|png)$/i.test(file.name)
if (!isValidSuffix) {
return {
ok: false,
msg: `请上传 JPG/PNG/GIF 格式的图片`,
}
}
2024-08-20 12:39:32 +08:00
// 检查文件大小
const maxSizeMB = 10
if (file.size > maxSizeMB * 1024 * 1024) {
return {
ok: false,
2024-08-20 12:39:32 +08:00
msg: `由于公众号限制,图片大小不能超过 ${maxSizeMB}M`,
}
}
2024-08-20 12:39:32 +08:00
return { ok: true }
2020-12-05 21:16:09 +08:00
}
/**
* 移除左边多余空格
2021-11-26 23:42:56 +08:00
* @param {*} str
2024-08-18 19:49:16 +08:00
* @returns string
*/
export function removeLeft(str) {
const lines = str.split(`\n`)
// 获取应该删除的空白符数量
const minSpaceNum = lines
2024-08-18 19:49:16 +08:00
.filter(item => item.trim())
.map(item => item.match(/(^\s+)?/)[0].length)
.sort((a, b) => a - b)[0]
// 删除空白符
2024-08-18 19:49:16 +08:00
return lines.map(item => item.slice(minSpaceNum)).join(`\n`)
}
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]
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) {
return juice(html, {
inlinePseudoElements: true,
preserveImportant: true,
})
2021-11-26 23:42:56 +08:00
}