oicontest
Version:
OI Contest Management Tool
269 lines (267 loc) • 11.4 kB
JavaScript
;
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __importStar = (this && this.__importStar) || (function () {
var ownKeys = function(o) {
ownKeys = Object.getOwnPropertyNames || function (o) {
var ar = [];
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
return ar;
};
return ownKeys(o);
};
return function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
__setModuleDefault(result, mod);
return result;
};
})();
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.convertMarkdownToHtml = convertMarkdownToHtml;
const fs = __importStar(require("fs"));
const marked_1 = require("marked");
const marked_highlight_1 = require("marked-highlight");
const highlight_js_1 = __importDefault(require("highlight.js"));
const path_1 = __importDefault(require("path"));
const katex_1 = __importDefault(require("katex"));
// 只在模块顶部配置一次 marked
marked_1.marked.use((0, marked_highlight_1.markedHighlight)({
langPrefix: 'hljs language-',
highlight: (code, lang) => {
if (lang && highlight_js_1.default.getLanguage(lang)) {
return highlight_js_1.default.highlight(code, { language: lang }).value;
}
return highlight_js_1.default.highlightAuto(code).value;
}
}));
// 自定义数学公式渲染函数
function renderMathInText(text) {
// 先渲染块级数学公式,避免与行内公式冲突
text = text.replace(/\$\$([\s\S]*?)\$\$/g, (match, formula) => {
try {
return katex_1.default.renderToString(formula.trim(), {
displayMode: true,
throwOnError: false,
errorColor: '#cc0000'
});
}
catch (error) {
console.error('KaTeX渲染错误:', error);
return `<span style="color: #cc0000;">[数学公式渲染错误: ${formula}]</span>`;
}
});
// 然后渲染行内数学公式
text = text.replace(/\$([^\$\n]+?)\$/g, (match, formula) => {
// 跳过已经被渲染的块级公式
if (match.includes('katex')) {
return match;
}
try {
return katex_1.default.renderToString(formula.trim(), {
displayMode: false,
throwOnError: false,
errorColor: '#cc0000'
});
}
catch (error) {
console.error('KaTeX渲染错误:', error);
return `<span style="color: #cc0000;">[数学公式渲染错误: $${formula}$]</span>`;
}
});
return text;
}
marked_1.marked.setOptions({
breaks: true,
gfm: true
});
// 动态生成KaTeX字体@font-face,使用file://绝对路径
function getKatexFontFaceCssAbs(fontsDir) {
const fontMap = [
{ file: 'KaTeX_Main-Regular', family: 'KaTeX_Main', weight: 'normal', style: 'normal' },
{ file: 'KaTeX_Main-Bold', family: 'KaTeX_Main', weight: 'bold', style: 'normal' },
{ file: 'KaTeX_Main-Italic', family: 'KaTeX_Main', weight: 'normal', style: 'italic' },
{ file: 'KaTeX_Main-BoldItalic', family: 'KaTeX_Main', weight: 'bold', style: 'italic' },
{ file: 'KaTeX_Math-Italic', family: 'KaTeX_Math', weight: 'normal', style: 'italic' },
{ file: 'KaTeX_Math-BoldItalic', family: 'KaTeX_Math', weight: 'bold', style: 'italic' },
{ file: 'KaTeX_AMS-Regular', family: 'KaTeX_AMS', weight: 'normal', style: 'normal' },
{ file: 'KaTeX_SansSerif-Regular', family: 'KaTeX_SansSerif', weight: 'normal', style: 'normal' },
{ file: 'KaTeX_SansSerif-Bold', family: 'KaTeX_SansSerif', weight: 'bold', style: 'normal' },
{ file: 'KaTeX_SansSerif-Italic', family: 'KaTeX_SansSerif', weight: 'normal', style: 'italic' },
{ file: 'KaTeX_Script-Regular', family: 'KaTeX_Script', weight: 'normal', style: 'normal' },
{ file: 'KaTeX_Fraktur-Regular', family: 'KaTeX_Fraktur', weight: 'normal', style: 'normal' },
{ file: 'KaTeX_Fraktur-Bold', family: 'KaTeX_Fraktur', weight: 'bold', style: 'normal' },
{ file: 'KaTeX_Caligraphic-Regular', family: 'KaTeX_Caligraphic', weight: 'normal', style: 'normal' },
{ file: 'KaTeX_Caligraphic-Bold', family: 'KaTeX_Caligraphic', weight: 'bold', style: 'normal' },
{ file: 'KaTeX_Typewriter-Regular', family: 'KaTeX_Typewriter', weight: 'normal', style: 'normal' },
{ file: 'KaTeX_Size1-Regular', family: 'KaTeX_Size1', weight: 'normal', style: 'normal' },
{ file: 'KaTeX_Size2-Regular', family: 'KaTeX_Size2', weight: 'normal', style: 'normal' },
{ file: 'KaTeX_Size3-Regular', family: 'KaTeX_Size3', weight: 'normal', style: 'normal' },
{ file: 'KaTeX_Size4-Regular', family: 'KaTeX_Size4', weight: 'normal', style: 'normal' },
];
return fontMap.map(f => {
const woff2 = path_1.default.resolve(fontsDir, f.file + '.woff2');
const woff = path_1.default.resolve(fontsDir, f.file + '.woff');
const ttf = path_1.default.resolve(fontsDir, f.file + '.ttf');
let src = [];
if (fs.existsSync(woff2))
src.push(`url('file://${woff2}') format('woff2')`);
if (fs.existsSync(woff))
src.push(`url('file://${woff}') format('woff')`);
if (fs.existsSync(ttf))
src.push(`url('file://${ttf}') format('truetype')`);
if (src.length === 0)
return '';
return `@font-face {
font-family: '${f.family}';
src: ${src.join(',\n ')};
font-weight: ${f.weight};
font-style: ${f.style};
}`;
}).join('\n');
}
// 获取当前日期时间作为转换时间戳
const conversionDate = new Date().toLocaleString('zh-CN', {
year: 'numeric',
month: 'long',
day: 'numeric',
hour: '2-digit',
minute: '2-digit',
second: '2-digit'
});
function convertMarkdownToHtml(markdownContent, contestTitle) {
return __awaiter(this, void 0, void 0, function* () {
try {
// 直接解析
// 然后渲染数学公式
const contentWithMath = renderMathInText(markdownContent);
// 最后渲染markdown(包含代码高亮)
//const htmlContent = marked.parse(contentWithMath);
const staticPath = path_1.default.resolve(__dirname, "../templates/html");
const fontPath = path_1.default.resolve(__dirname, "../fonts/katex");
const fontFaceCss = getKatexFontFaceCssAbs(fontPath);
const htmlContent = marked_1.marked.parse(contentWithMath);
// 生成完整的HTML页面,引用html目录下的本地css和字体
const fullHtml = `<!DOCTYPE html>
<html lang="zh-CN">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>${contestTitle}</title>
<!-- GitHub风格CSS -->
<link rel="stylesheet" href="${staticPath}/github-markdown-light.min.css">
<!-- 代码高亮样式 -->
<link rel="stylesheet" href="${staticPath}/github.min.css">
<!-- KaTeX样式 -->
<link rel="stylesheet" href="${staticPath}/katex.min.css">
<style>
.markdown-body {
box-sizing: border-box;
max-width: 980px;
margin: 0 auto;
padding: 45px;
}
@media (max-width: 767px) {
.markdown-body {
padding: 15px;
}
}
/* 图片样式 */
.markdown-image {
max-width: 100%;
height: auto;
display: block;
margin: 1em auto;
border-radius: 4px;
box-shadow: 0 2px 4px rgba(0,0,0,0.1);
}
/* 页脚样式 */
.page-footer {
margin-top: 4em;
padding-top: 2em;
border-top: 1px solid #e1e4e8;
color: #6a737d;
font-size: 0.9em;
text-align: center;
}
.footer-separator {
margin: 0 0.5em;
opacity: 0.5;
}
${fontFaceCss}
@media print {
#print-btn {
display: none !important;
}
.copy-btn { display: none !important; }
}
.copy-btn { position: absolute; right: 8px; top: 8px; padding: 2px 8px; font-size: 12px; cursor: pointer; background: #eee; border: 1px solid #ccc; border-radius: 4px; z-index: 10; }
.copy-btn:active { background: #ddd; }
.pre-block { position: relative; }
</style>
</head>
<body>
<article class="markdown-body">
${htmlContent}
<!-- 页脚 -->
<footer class="page-footer">
<p>
转换时间: ${conversionDate}
<span class="footer-separator">|</span>
由 oicontent 工具生成
</p>
</footer>
</article>
<button id="print-btn" onclick="window.print()" style="position:fixed;top:20px;right:20px;z-index:999;">打印/导出PDF</button>
<script>
document.querySelectorAll('pre > code').forEach(function(codeBlock) {
var pre = codeBlock.parentElement;
pre.classList.add('pre-block');
var btn = document.createElement('button');
btn.innerText = '复制';
btn.className = 'copy-btn';
btn.onclick = function() {
navigator.clipboard.writeText(codeBlock.innerText);
btn.innerText = '已复制!';
setTimeout(() => btn.innerText = '复制', 1000);
};
pre.appendChild(btn);
});
</script>
</body>
</html>`;
return fullHtml;
}
catch (error) {
console.error('转换失败:', error);
process.exit(1);
}
});
}