feat(typesetting): Wave 2 - 流式渲染、表格字号独立控制、作者声明

- 缩略图改为流式渲染,截图一页即显示一页
- 新增tableFontScale独立控制表格字号,含后端持久化
- 内容页顶部添加作者声明(文/作者名),空值隐藏
This commit is contained in:
augushong
2026-05-16 00:35:03 +08:00
committed by Atlas
parent 2b9bfb179f
commit 3ea3a6dbe3
4 changed files with 130 additions and 44 deletions

View File

@@ -20,6 +20,7 @@ class PhoneImage implements PostOutputManagerInterface
'watermark' => ['type' => 'text', 'default' => ''],
'pageAlignments' => ['type' => 'json', 'default' => '{}'],
'fontScale' => ['type' => 'number', 'default' => 1, 'min' => 0.5, 'max' => 2.0],
'tableFontScale' => ['type' => 'number', 'default' => 1, 'min' => 0.5, 'max' => 2.0],
];
}
@@ -45,6 +46,12 @@ class PhoneImage implements PostOutputManagerInterface
return false;
}
}
if (isset($config['tableFontScale'])) {
$scale = floatval($config['tableFontScale']);
if ($scale < 0.5 || $scale > 2.0) {
return false;
}
}
// watermark 是文本类型,无需特殊验证
// pageAlignments 是 JSON 类型,存储时由框架自动序列化

View File

@@ -50,6 +50,7 @@
--pi-radius: 12px;
/* 字体 */
--pi-font-family: 'Source Han Sans', 'SourceHanSans-Normal', sans-serif;
--pi-table-font-scale: 1;
}
/* ============================================
@@ -251,6 +252,16 @@
margin-top: var(--pi-spacing-sm);
}
/* 内容页 - 作者声明 */
.author-credit {
font-size: 11px;
color: #999;
padding-bottom: 8px;
margin-bottom: 12px;
border-bottom: 1px solid #eee;
text-indent: 0;
}
/* 内容页 - 正文区域 */
.page-content {
flex: 1;
@@ -600,7 +611,7 @@
============================================ */
/* 表格美化样式html2canvas兼容 */
.page-content table { width: 100%; border-collapse: collapse; margin: 10px 0; font-size: calc(13px * var(--pi-font-scale, 1)); }
.page-content table { width: 100%; border-collapse: collapse; margin: 10px 0; font-size: calc(13px * var(--pi-table-font-scale, 1)); }
.page-content th { background-color: #f0f0f0; font-weight: bold; padding: 4px 10px; border: 1px solid #ddd; text-align: left; }
.page-content td { padding: 4px 10px; border: 1px solid #ddd; }
.page-content tr:nth-child(even) td { background-color: #f9f9f9; }
@@ -628,7 +639,7 @@
}
/* 中间栏表格样式 */
.content-flow-block table { width: 100%; border-collapse: collapse; margin: 10px 0; font-size: calc(13px * var(--pi-font-scale, 1)); }
.content-flow-block table { width: 100%; border-collapse: collapse; margin: 10px 0; font-size: calc(13px * var(--pi-table-font-scale, 1)); }
.content-flow-block th { background-color: #f0f0f0; font-weight: bold; padding: 4px 10px; border: 1px solid #ddd; text-align: left; }
.content-flow-block td { padding: 4px 10px; border: 1px solid #ddd; }
.content-flow-block tr:nth-child(even) td { background-color: #f9f9f9; }

View File

@@ -99,7 +99,8 @@ var PhoneImageEngine = (function () {
douyin: { width: 540, height: 960 }
},
contentPadding: 20,
fontScale: 1
fontScale: 1,
tableFontScale: 1
};
// ===== 文章数据 =====
@@ -159,10 +160,17 @@ var PhoneImageEngine = (function () {
*/
function applyFontScale() {
var scale = config.fontScale || 1;
var tableScale = config.tableFontScale || 1;
var preview = document.getElementById('render-preview');
if (preview) preview.style.setProperty('--pi-font-scale', scale);
if (preview) {
preview.style.setProperty('--pi-font-scale', scale);
preview.style.setProperty('--pi-table-font-scale', tableScale);
}
var staging = document.getElementById('render-staging');
if (staging) staging.style.setProperty('--pi-font-scale', scale);
if (staging) {
staging.style.setProperty('--pi-font-scale', scale);
staging.style.setProperty('--pi-table-font-scale', tableScale);
}
}
/**
@@ -914,6 +922,11 @@ var PhoneImageEngine = (function () {
html += '</div>';
}
// 作者声明(所有内容页)
if (postData.author_name) {
html += '<div class="author-credit">文/' + escapeHtml(postData.author_name) + '</div>';
}
// 正文内容区
html += '<div class="page-content">';
for (var i = 0; i < blocks.length; i++) {
@@ -1125,17 +1138,19 @@ var PhoneImageEngine = (function () {
// 保存模式: 高质量绘制原图到canvas
renderPureImageToCanvas(pureSrc, opts.sizeConfig.width, opts.sizeConfig.height).then(function (canvas) {
results.push(canvas);
if (opts.streaming && opts.onPageReady) opts.onPageReady(canvas, idx);
if (pageProgressCallback) pageProgressCallback(idx + 1, total);
idx++;
captureNext();
}).catch(function () {
// 加载失败回退到html2canvas
capturePageViaHtml2Canvas($elem, $staging, opts, results, deferred, function () { if (pageProgressCallback) pageProgressCallback(idx + 1, total); idx++; captureNext(); });
capturePageViaHtml2Canvas($elem, $staging, opts, results, deferred, function () { if (opts.streaming && opts.onPageReady) opts.onPageReady(results[results.length - 1], idx); if (pageProgressCallback) pageProgressCallback(idx + 1, total); idx++; captureNext(); });
});
return;
} else {
// 缩略图模式: 直接用src
results.push(pureSrc);
if (opts.streaming && opts.onPageReady) opts.onPageReady(pureSrc, idx);
if (pageProgressCallback) pageProgressCallback(idx + 1, total);
idx++;
captureNext();
@@ -1144,7 +1159,7 @@ var PhoneImageEngine = (function () {
}
}
capturePageViaHtml2Canvas($elem, $staging, opts, results, deferred, function () { idx++; captureNext(); });
capturePageViaHtml2Canvas($elem, $staging, opts, results, deferred, function () { if (opts.streaming && opts.onPageReady) opts.onPageReady(results[results.length - 1], idx); idx++; captureNext(); });
}
captureNext();
@@ -1183,15 +1198,22 @@ var PhoneImageEngine = (function () {
function renderThumbnails(sizeConfig) {
var deferred = $.Deferred();
// 流式渲染:先清空容器,再逐页追加缩略图
initThumbnails(sizeConfig);
doCapturePages({
scale: 1,
outputCanvas: false,
quality: 0.85,
sizeConfig: sizeConfig
sizeConfig: sizeConfig,
streaming: true,
onPageReady: function (dataUrl, pageIndex) {
appendThumbnail(dataUrl, pageIndex, sizeConfig, pages[pageIndex]);
}
}, function (current, total) {
PhoneImageLogPanel.log('截图: 第 ' + current + '/' + total + ' 页', 'info');
}).then(function (dataUrls) {
displayThumbnails(dataUrls, sizeConfig);
updateThumbnailPageNumbers(dataUrls.length);
deferred.resolve(pages);
}).catch(function (err) {
deferred.reject(err);
@@ -1201,44 +1223,52 @@ var PhoneImageEngine = (function () {
}
/**
* 在右侧显示缩略图
* @param {Array} dataUrls - base64 图片数据
* 初始化缩略图容器(流式渲染第一步:清空容器)
* @param {Object} sizeConfig
*/
function displayThumbnails(dataUrls, sizeConfig) {
function initThumbnails(sizeConfig) {
var $preview = $('#paginated-preview');
if (!$preview.length) return;
$preview.empty();
if (dataUrls.length === 0) return;
}
/**
* 追加单个缩略图到预览区(流式渲染:每页截图完成后立即追加)
* @param {string} dataUrl - base64 图片数据或图片src
* @param {number} pageIndex - 页面索引(0-based)
* @param {Object} sizeConfig
* @param {Object} pageData - 页面对象 { type, pageNum, ... }
*/
function appendThumbnail(dataUrl, pageIndex, sizeConfig, pageData) {
var $preview = $('#paginated-preview');
if (!$preview.length) return;
// 计算缩放比例:缩略图高度 = 容器高度 - 40(页码标签) - 40(上下padding)
var containerHeight = $preview.parent().height() || 700;
var scaleRatio = (containerHeight - 80) / sizeConfig.height;
var thumbWidth = Math.round(sizeConfig.width * scaleRatio);
for (var i = 0; i < dataUrls.length; i++) {
var $item = $('<div class="preview-thumb-item" data-page-index="' + i + '">');
var $item = $('<div class="preview-thumb-item" data-page-index="' + pageIndex + '">');
$item.css('width', thumbWidth + 'px');
var $img = $('<img>');
$img.attr('src', dataUrls[i]);
$img.attr('src', dataUrl);
$img.css('width', thumbWidth + 'px');
$item.append($img);
// 页码
var $pageNum = $('<span class="preview-thumb-page-num">' + (i + 1) + '/' + dataUrls.length + '</span>');
// 页码(初始只显示序号,全部完成后更新为 N/M
var $pageNum = $('<span class="preview-thumb-page-num">' + (pageIndex + 1) + '</span>');
$item.append($pageNum);
// 对齐按钮(仅内容页)
if (pages[i] && pages[i].type === 'content') {
var pageNum = pages[i].pageNum || (i);
if (pageData && pageData.type === 'content') {
var pageNum = pageData.pageNum || (pageIndex);
var currentAlign = (config.pageAlignments && config.pageAlignments[pageNum]) || 'top';
var isActiveCenter = currentAlign === 'center';
var isActiveBottom = currentAlign === 'bottom';
var activeClass = isActiveCenter ? ' active-center' : (isActiveBottom ? ' active-bottom' : '');
var symbol = isActiveCenter ? '\u2195' : (isActiveBottom ? '\u2193' : '\u2191');
var $toggle = $('<button class="thumb-alignment-toggle' + activeClass + '" data-page-index="' + i + '" data-page-num="' + pageNum + '">' +
var $toggle = $('<button class="thumb-alignment-toggle' + activeClass + '" data-page-index="' + pageIndex + '" data-page-num="' + pageNum + '">' +
symbol +
'</button>');
$item.append($toggle);
@@ -1246,6 +1276,16 @@ var PhoneImageEngine = (function () {
$preview.append($item);
}
/**
* 更新所有缩略图页码为 N/M 格式(流式渲染全部完成后调用)
* @param {number} totalPages
*/
function updateThumbnailPageNumbers(totalPages) {
var $items = $('#paginated-preview .preview-thumb-item');
$items.each(function (i) {
$(this).find('.preview-thumb-page-num').text((i + 1) + '/' + totalPages);
});
}
/**

View File

@@ -285,7 +285,8 @@
var currentConfig = {
size: 'xiaohongshu',
watermark: '',
fontScale: 1
fontScale: 1,
tableFontScale: 1
};
var postData = {
@@ -306,13 +307,15 @@
size: savedConfig.size || 'xiaohongshu',
watermark: savedConfig.watermark || '',
pageAlignments: savedConfig.pageAlignments || {},
fontScale: savedConfig.fontScale || 1
fontScale: savedConfig.fontScale || 1,
tableFontScale: savedConfig.tableFontScale || 1
};
// 同步当前配置
currentConfig.size = initConfig.size;
currentConfig.watermark = initConfig.watermark;
currentConfig.fontScale = initConfig.fontScale;
currentConfig.tableFontScale = initConfig.tableFontScale;
// ========== wangeditor 初始化 ==========
var E = window.wangEditor;
@@ -467,6 +470,7 @@
size: currentConfig.size,
watermark: currentConfig.watermark,
fontScale: currentConfig.fontScale || 1,
tableFontScale: currentConfig.tableFontScale || 1,
content_html: PhoneImageEngine.getContentHtml()
}, saveConfigUrl).then(function(data) {
if (data.output_id) lastOutputId = data.output_id;
@@ -533,13 +537,27 @@
settingsHtml += '<input type="text" name="s_watermark" value="' + (currentConfig.watermark || '') + '" placeholder="可选水印文字" class="layui-input">';
settingsHtml += '</div>';
settingsHtml += '</div>';
settingsHtml += '<div class="layui-form-item">';
settingsHtml += '<label class="layui-form-label">表格字号</label>';
settingsHtml += '<div class="layui-input-block">';
settingsHtml += '<select name="s_table_font_scale">';
var tblScaleVal = currentConfig.tableFontScale || 1;
var tblPresets = [0.5, 0.8, 1.0, 1.2, 1.5, 2.0];
var tblIsPreset = tblPresets.indexOf(tblScaleVal) !== -1;
for (var pi = 0; pi < tblPresets.length; pi++) {
settingsHtml += '<option value="' + tblPresets[pi] + '"' + (tblScaleVal === tblPresets[pi] ? ' selected' : '') + '>' + tblPresets[pi] + 'x</option>';
}
settingsHtml += '<option value="custom"' + (!tblIsPreset ? ' selected' : '') + '>自定义</option>';
settingsHtml += '</select>';
settingsHtml += '</div>';
settingsHtml += '</div>';
settingsHtml += '</form>';
settingsHtml += '</div>';
layer.open({
type: 1,
title: '排版设置',
area: ['400px', '220px'],
area: ['400px', '280px'],
content: settingsHtml,
btn: ['确定', '取消'],
success: function (layero) {
@@ -548,8 +566,10 @@
yes: function (index, layero) {
currentConfig.size = layero.find('[name="s_size"]').val();
currentConfig.watermark = layero.find('[name="s_watermark"]').val();
var tblScaleSel = layero.find('[name="s_table_font_scale"]').val();
currentConfig.tableFontScale = tblScaleSel === 'custom' ? (currentConfig.tableFontScale || 1) : parseFloat(tblScaleSel);
layer.close(index);
PhoneImageLogPanel.log('应用新设置: 尺寸=' + currentConfig.size, 'info');
PhoneImageLogPanel.log('应用新设置: 尺寸=' + currentConfig.size + ', 表格字号=' + (currentConfig.tableFontScale || 1) + 'x', 'info');
doRender();
}
});
@@ -562,7 +582,8 @@
renderTimer = setTimeout(function () {
var newConfig = {
size: currentConfig.size,
watermark: currentConfig.watermark
watermark: currentConfig.watermark,
tableFontScale: currentConfig.tableFontScale || 1
};
if (extraConfig) {
$.extend(newConfig, extraConfig);
@@ -634,6 +655,7 @@
size: currentConfig.size,
watermark: currentConfig.watermark,
fontScale: currentConfig.fontScale || 1,
tableFontScale: currentConfig.tableFontScale || 1,
content_html: PhoneImageEngine.getContentHtml()
}, saveConfigUrl).then(function (data) {
if (data.output_id) lastOutputId = data.output_id;
@@ -662,6 +684,7 @@
size: currentConfig.size,
watermark: currentConfig.watermark,
fontScale: currentConfig.fontScale || 1,
tableFontScale: currentConfig.tableFontScale || 1,
content_html: PhoneImageEngine.getContentHtml()
}, saveConfigUrl).then(function (data) {
if (data.output_id) lastOutputId = data.output_id;
@@ -688,6 +711,7 @@
size: currentConfig.size,
watermark: currentConfig.watermark,
fontScale: currentConfig.fontScale || 1,
tableFontScale: currentConfig.tableFontScale || 1,
content_html: PhoneImageEngine.getContentHtml()
}, function(current, total, canvas) {
if (total > 0) {
@@ -779,6 +803,7 @@
if (cfg.watermark !== undefined) historyConfig.watermark = cfg.watermark;
if (cfg.pageAlignments) historyConfig.pageAlignments = cfg.pageAlignments;
if (cfg.fontScale !== undefined) historyConfig.fontScale = cfg.fontScale;
if (cfg.tableFontScale !== undefined) historyConfig.tableFontScale = cfg.tableFontScale;
// 同步当前配置
if (cfg.size) currentConfig.size = cfg.size;
@@ -787,6 +812,9 @@
currentConfig.fontScale = cfg.fontScale;
setFontScaleUI(cfg.fontScale);
}
if (cfg.tableFontScale !== undefined) {
currentConfig.tableFontScale = cfg.tableFontScale;
}
lastOutputId = outputId;
PhoneImageLogPanel.log('历史配置已加载', 'success');