本文介绍一种基于 svg 的高效方案,通过 viewbox 和 preserveaspectratio 实现字符网格的等比缩放,使终端界面自动填满屏幕且保持清晰可读。
在构建类终端(terminal-like)Web 应用时,常见的 DOM 方案(如大量 元素)虽便于单字符样式控制,却难以实现响应式等比缩放——直接修改 font-size 或容器尺寸往往导致布局错乱、行高失衡或字符裁切。根本原因在于:CSS 中文本流的渲染机制不具备原生的“整体缩放锚点”,而 SVG 的 viewBox + preserveAspectRatio 组合恰好为此类像素级网格场景提供了理想解。
SVG 是矢量图形标准,其 viewBox 定义逻辑坐标系,width/height 控制视口尺寸,二者结合即可实现无损缩放。关键优势包括:
以下为可直接运行的核心实现:
* { margin: 0; padding: 0; }
html, body { height: 100%; }
#game { height: 100%; 
}
.cell {
font-family: 'Cascadia Code', 'Fira Code', monospace;
font-size: 12px; /* 基准字号(仅作参考,实际由 viewBox 缩放决定) */
}
svg {
display: block;
width: 100%;
height: 100%;
}const CHAR_WIDTH = 10; // 逻辑单位宽度(px)
const CHAR_HEIGHT = 12; // 逻辑单位高度(px)
const COLS = 100;
const ROWS = 25;
function draw() {
const game = document.getElementById('game');
const svg = document.createElementNS('http://www.w3.org/2000/svg', 'svg');
// 定义逻辑画布:宽 = COLS × CHAR_WIDTH,高 = ROWS × CHAR_HEIGHT
svg.setAttribute('viewBox', `0 0 ${COLS * CHAR_WIDTH} ${ROWS * CHAR_HEIGHT}`);
svg.style.display = 'block';
// 逐字符生成 元素
for (let y = 0; y < ROWS; y++) {
for (let x = 0; x < COLS; x++) {
const text = document.createElementNS('http://www.w3.org/2000/svg', 'text');
text.setAttribute('x', x * CHAR_WIDTH);
text.setAttribute('y', y * CHAR_HEIGHT);
text.setAttribute('class', 'cell');
text.setAttribute('alignment-baseline', 'hanging'); // 顶部对齐,避免基线偏移
text.textContent = '1'; // 替换为你的动态字符
svg.appendChild(text);
}
}
game.appendChild(svg);
}
document.addEventListener('DOMContentLoaded', draw); 该方案彻底规避了传统 span 网格的缩放陷阱,以声明式矢量方式达成「字体尽可能大、始终填满屏幕、严格保持宽高比」三大目标,是现代终端 Web 应用的推荐架构。