6 次代码提交

修改 9 个文件,包含 382 行新增34 行删除
+1 -1
查看文件
@@ -1,7 +1,7 @@
{ {
"name": "theme-terminal", "name": "theme-terminal",
"private": true, "private": true,
"version": "1.1.8", "version": "1.1.9a",
"description": "A terminal like theme for Halo.", "description": "A terminal like theme for Halo.",
"scripts": { "scripts": {
"dev": "vite build --watch", "dev": "vite build --watch",
+231
查看文件
@@ -0,0 +1,231 @@
// ============ 常量配置 ============
const TEXT_ELEMENTS = ['p', 'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'blockquote', 'li'];
const HEADING_ELEMENTS = ['h1', 'h2', 'h3', 'h4', 'h5', 'h6'];
const NON_TEXT_ELEMENTS = ['img', 'table', 'video', 'audio', 'canvas', 'svg', 'iframe', 'pre'];
const ALL_ELEMENTS = [...TEXT_ELEMENTS, ...NON_TEXT_ELEMENTS];
const DIRECT_CHILD_SELECTOR = ALL_ELEMENTS.map(tag => `div > ${tag}`).join(', ');
const DEFAULT_LINE_HEIGHT_RATIO = 1.54;
// ============ 类型定义 ============
interface LineSpanOptions {
lineNum: number;
top: number;
height: number;
isGap?: boolean;
isBreakpoint?: boolean;
}
interface ElementMetrics {
top: number;
height: number;
paddingTop: number;
lineHeight: number;
lines: number;
isHeading: boolean;
}
interface GapLinesOptions {
fragment: DocumentFragment;
gapStart: number;
gapEnd: number;
lineHeight: number;
startLine: number;
}
interface LineNumbersState {
$el?: HTMLElement;
totalLines: number;
resizeObserver: ResizeObserver | null;
init(): void;
calculateLineNumbers(): void;
destroy(): void;
}
// ============ 工具函数 ============
const isNonTextElement = (tagName: string): boolean =>
NON_TEXT_ELEMENTS.includes(tagName.toLowerCase());
const isHeadingElement = (tagName: string): boolean =>
HEADING_ELEMENTS.includes(tagName.toLowerCase());
const isEmptyElement = (element: HTMLElement): boolean => {
if (isNonTextElement(element.tagName.toLowerCase())) return false;
return !element.textContent?.trim();
};
const getStyleValue = (style: CSSStyleDeclaration, prop: 'paddingTop' | 'paddingBottom' | 'lineHeight' | 'fontSize'): number =>
parseFloat(style[prop]) || 0;
const getDefaultLineHeight = (element: HTMLElement): number => {
const style = getComputedStyle(element);
return getStyleValue(style, 'lineHeight') || getStyleValue(style, 'fontSize') * DEFAULT_LINE_HEIGHT_RATIO;
};
// 获取元素度量信息(合并多次 getComputedStyle 调用)
const getElementMetrics = (element: HTMLElement, containerRect: DOMRect, defaultLineHeight: number): ElementMetrics => {
const tagName = element.tagName.toLowerCase();
const style = getComputedStyle(element);
const paddingTop = getStyleValue(style, 'paddingTop');
const paddingBottom = getStyleValue(style, 'paddingBottom');
const height = element.offsetHeight;
const contentHeight = height - paddingTop - paddingBottom;
const isNonText = isNonTextElement(tagName);
const lineHeight = isNonText ? contentHeight : (getStyleValue(style, 'lineHeight') || defaultLineHeight);
const lines = isNonText ? 1 : Math.max(1, Math.round(contentHeight / lineHeight));
return {
top: element.getBoundingClientRect().top - containerRect.top,
height,
paddingTop,
lineHeight: contentHeight / lines,
lines,
isHeading: isHeadingElement(tagName)
};
};
// ============ DOM 创建 ============
const createLineSpan = (options: LineSpanOptions): HTMLSpanElement => {
const { lineNum, top, height, isGap = false, isBreakpoint = false } = options;
const span = document.createElement('span');
span.className = `line-number${isGap ? ' line-number--gap' : ''}${isBreakpoint ? ' line-number--breakpoint' : ''}`;
span.style.cssText = `top:${top}px;height:${height}px;line-height:${height}px`;
span.textContent = String(lineNum);
if (isBreakpoint) {
const breakpoint = document.createElement('span');
breakpoint.className = 'line-breakpoint';
breakpoint.textContent = '●';
span.appendChild(breakpoint);
}
return span;
};
// ============ 行号生成 ============
const appendGapLines = (options: GapLinesOptions): number => {
const { fragment, gapStart, gapEnd, lineHeight, startLine } = options;
const gapHeight = gapEnd - gapStart;
if (gapHeight < lineHeight) return startLine;
const gapLines = Math.round(gapHeight / lineHeight);
for (let i = 0; i < gapLines; i++) {
fragment.appendChild(createLineSpan({
lineNum: startLine + i,
top: gapStart + i * lineHeight,
height: lineHeight,
isGap: true
}));
}
return startLine + gapLines;
};
const appendElementLines = (fragment: DocumentFragment, metrics: ElementMetrics, startLine: number): number => {
const { top, paddingTop, lineHeight, lines, isHeading } = metrics;
const startTop = top + paddingTop;
for (let i = 0; i < lines; i++) {
fragment.appendChild(createLineSpan({
lineNum: startLine + i,
top: startTop + i * lineHeight,
height: lineHeight,
isBreakpoint: i === 0 && isHeading
}));
}
return startLine + lines;
};
// ============ 主组件 ============
export const lineNumbers = (): LineNumbersState => ({
totalLines: 0,
resizeObserver: null,
init() {
const content = this.$el?.querySelector('.post-content') as HTMLElement;
if (!content) return;
const recalculate = () => this.calculateLineNumbers();
// 延迟计算确保 DOM 渲染完成
requestAnimationFrame(() => setTimeout(recalculate, 50));
// 监听内容区域大小变化(带防抖)
if (typeof ResizeObserver !== 'undefined') {
let resizeTimer: number;
this.resizeObserver = new ResizeObserver(() => {
clearTimeout(resizeTimer);
resizeTimer = window.setTimeout(recalculate, 100);
});
this.resizeObserver.observe(content);
}
// 窗口大小变化时重新计算
let windowResizeTimer: number;
window.addEventListener('resize', () => {
clearTimeout(windowResizeTimer);
windowResizeTimer = window.setTimeout(recalculate, 200);
});
// 监听图片加载(使用 once 避免重复触发)
content.querySelectorAll('img').forEach((img: HTMLImageElement) => {
if (!img.complete) {
img.addEventListener('load', recalculate, { once: true });
img.addEventListener('error', recalculate, { once: true });
}
});
},
destroy() {
this.resizeObserver?.disconnect();
},
calculateLineNumbers() {
const content = this.$el?.querySelector('.post-content') as HTMLElement;
const gutter = this.$el?.querySelector('.post-line-gutter') as HTMLElement;
if (!content || !gutter) return;
const container = (content.querySelector(':scope > div') || content) as HTMLElement;
const containerRect = container.getBoundingClientRect();
const defaultLineHeight = getDefaultLineHeight(content);
const elements = Array.from(container.querySelectorAll(DIRECT_CHILD_SELECTOR))
.filter(el => !isEmptyElement(el as HTMLElement)) as HTMLElement[];
const fragment = document.createDocumentFragment();
let currentLine = 1;
let lastBottom = 0;
for (const element of elements) {
const metrics = getElementMetrics(element, containerRect, defaultLineHeight);
if (lastBottom > 0 && metrics.top > lastBottom) {
currentLine = appendGapLines({
fragment,
gapStart: lastBottom,
gapEnd: metrics.top,
lineHeight: defaultLineHeight,
startLine: currentLine
});
}
currentLine = appendElementLines(fragment, metrics, currentLine);
lastBottom = metrics.top + metrics.height;
}
gutter.innerHTML = '';
gutter.appendChild(fragment);
gutter.style.height = `${container.scrollHeight}px`;
this.totalLines = currentLine - 1;
}
});
+2
查看文件
@@ -6,6 +6,7 @@ import Alpine from 'alpinejs'
import {upvote} from './alpine/upvote' import {upvote} from './alpine/upvote'
import {themeMode} from './alpine/themeMode' import {themeMode} from './alpine/themeMode'
import {menu} from './alpine/menu' import {menu} from './alpine/menu'
import {lineNumbers} from './alpine/lineNumbers'
import {typewriterEffect} from './utils' import {typewriterEffect} from './utils'
window.Alpine = Alpine window.Alpine = Alpine
@@ -13,6 +14,7 @@ window.Alpine = Alpine
Alpine.data('upvote', upvote) Alpine.data('upvote', upvote)
Alpine.data('themeMode', themeMode) Alpine.data('themeMode', themeMode)
Alpine.data('menu', menu) Alpine.data('menu', menu)
Alpine.data('lineNumbers', lineNumbers)
Alpine.start() Alpine.start()
+6 -9
查看文件
@@ -158,7 +158,7 @@
&.open { &.open {
display: block; display: block;
animation: terminal-frame 0.2s ease-out; animation: terminal-frame-reveal 0.3s ease-out forwards;
// 逐行渲染效果 - 为每个子项添加延迟动画 // 逐行渲染效果 - 为每个子项添加延迟动画
li { li {
@@ -182,17 +182,14 @@
padding: 0; padding: 0;
} }
// 菜单框架渲染动画 - 模拟终端窗口从上到下打开 // 菜单框架渲染动画 - 从上到下逐层渲染边框和内容
@keyframes terminal-frame { @keyframes terminal-frame-reveal {
0% { 0% {
max-height: 0; clip-path: inset(-10px -10px calc(100% + 10px) -10px);
opacity: 0;
}
50% {
opacity: 1; opacity: 1;
} }
100% { 100% {
max-height: 800px; clip-path: inset(-10px -10px -10px -10px);
opacity: 1; opacity: 1;
} }
} }
@@ -201,7 +198,7 @@
@keyframes terminal-line { @keyframes terminal-line {
0% { 0% {
opacity: 0; opacity: 0;
transform: translateX(-10px); transform: translateX(-2px);
} }
50% { 50% {
opacity: 0.5; opacity: 0.5;
-9
查看文件
@@ -173,15 +173,6 @@ blockquote {
padding-right: 0; padding-right: 0;
} }
&:before {
content: '';
font-family: Georgia, serif;
font-size: 3.875rem;
position: absolute;
left: -40px;
top: -20px;
}
p:first-of-type { p:first-of-type {
margin-top: 0; margin-top: 0;
} }
+116 -1
查看文件
@@ -38,6 +38,8 @@
} }
%meta { %meta {
display: flex;
align-items: center;
font-size: 1rem; font-size: 1rem;
margin-bottom: 10px; margin-bottom: 10px;
color: var(--brightBlue); color: var(--brightBlue);
@@ -53,6 +55,30 @@
display: inline; display: inline;
} }
&-separator {
position: relative;
top: 1px;
display: inline-grid;
grid-template-columns: repeat(2, 3px);
grid-template-rows: repeat(2, 3px);
gap: 2px;
margin: 0 6px;
span {
width: 2px;
height: 2px;
background-color: var(--brightBlue);
border-radius: 50%;
opacity: 0.2;
}
// 每个点使用不同动画和时长,形成随机效果
span:nth-child(1) { animation: dot-blink-1 2.8s ease-in-out infinite; }
span:nth-child(2) { animation: dot-blink-2 2.2s ease-in-out infinite 0.4s; }
span:nth-child(3) { animation: dot-blink-3 3.4s ease-in-out infinite 1s; }
span:nth-child(4) { animation: dot-blink-4 2.6s ease-in-out infinite 0.2s; }
}
&-title { &-title {
--border: 3px dotted var(--blue); --border: 3px dotted var(--blue);
@@ -102,9 +128,68 @@
} }
} }
&-content { &-body {
position: relative;
margin-top: 30px; margin-top: 30px;
}
&-line-gutter {
position: absolute;
left: -80px;
top: 0;
width: 60px;
user-select: none;
opacity: 0.5;
border-right: 1px solid color-mix(in srgb, var(--foreground) 15%, transparent);
@media (max-width: $tablet-max-width) {
display: none;
}
.line-number {
position: absolute;
right: 8px;
display: flex;
align-items: center;
justify-content: flex-end;
padding-right: 10px;
font-size: 0.75rem;
color: var(--foreground);
font-family: Hack, Monaco, Consolas, monospace;
transition: opacity 0.15s ease;
opacity: 0.8;
&:hover {
opacity: 1;
}
}
.line-number--gap {
opacity: 0.5;
}
.line-number--breakpoint {
cursor: pointer;
.line-breakpoint {
position: absolute;
top: -1px;
right: -3px;
color: #e51400;
font-size: 0.875rem;
transition: transform 0.15s ease, filter 0.15s ease;
}
&:hover .line-breakpoint {
transform: scale(1.2);
filter: brightness(1.2);
}
}
}
&-content {
font-family: Hack, Monaco, Consolas, 'Ubuntu Mono', PingHei, 'PingFang SC', 'Microsoft YaHei', monospace; font-family: Hack, Monaco, Consolas, 'Ubuntu Mono', PingHei, 'PingFang SC', 'Microsoft YaHei', monospace;
line-height: 1.54;
} }
&-cover { &-cover {
@@ -173,3 +258,33 @@
position: relative; position: relative;
z-index: 2; z-index: 2;
} }
// 4个不同的随机动画模式
@keyframes dot-blink-1 {
0%, 100% { opacity: 0.2; }
25% { opacity: 1; }
50% { opacity: 0.3; }
75% { opacity: 0.8; }
}
@keyframes dot-blink-2 {
0%, 100% { opacity: 0.3; }
30% { opacity: 0.9; }
60% { opacity: 0.2; }
85% { opacity: 0.7; }
}
@keyframes dot-blink-3 {
0%, 100% { opacity: 0.2; }
20% { opacity: 0.6; }
45% { opacity: 1; }
70% { opacity: 0.4; }
}
@keyframes dot-blink-4 {
0%, 100% { opacity: 0.4; }
15% { opacity: 0.2; }
40% { opacity: 0.8; }
65% { opacity: 1; }
90% { opacity: 0.3; }
}
+10 -7
查看文件
@@ -12,18 +12,21 @@
<div class="posts"> <div class="posts">
<div class="post on-list" th:each="post : ${posts.items}"> <div class="post on-list" th:each="post : ${posts.items}">
<h1 class="post-title"> <h1 class="post-title">
<a th:text="${post.spec.title}" th:href="${post.status.permalink}">Post Title</a> <a th:text="'< ' + ${post.spec.title} + ' >'" th:href="${post.status.permalink}">Post Title</a>
</h1> </h1>
<div class="post-meta"> <div class="post-meta">
<span class="post-date" th:text="${#dates.format(post.spec.publishTime,'yyyy-MM-dd')}"> <span class="post-date" th:text="${#dates.format(post.spec.publishTime,'yyyy-MM-dd')}">
Post CreateTime Post CreateTime
</span> </span>
<span <span class="post-separator">
class="post-author" <span></span>
th:with="contributor = ${post.contributors[0]}" <span></span>
th:text="${':: '+contributor.displayName}" <span></span>
>:: Author</span <span></span>
> </span>
<span class="post-author" th:with="contributor = ${post.contributors[0]}" th:text="${contributor.displayName}">
Author
</span>
</div> </div>
<span class="post-tags-inline" th:each="tag : ${post.tags}"> <span class="post-tags-inline" th:each="tag : ${post.tags}">
<a <a
+14 -5
查看文件
@@ -4,11 +4,17 @@
th:replace="~{modules/layout :: html(title = |${post.spec.title} - ${site.title}|, header = null, content = ~{::content}, footer = null)}" th:replace="~{modules/layout :: html(title = |${post.spec.title} - ${site.title}|, header = null, content = ~{::content}, footer = null)}"
> >
<th:block th:fragment="content"> <th:block th:fragment="content">
<div class="post"> <div class="post" x-data="lineNumbers" x-init="init()">
<h1 class="post-title" th:text="${post.spec.title}">Post Title</h1> <h1 class="post-title" th:text="'< ' + ${post.spec.title} + ' >'">Post Title</h1>
<div class="post-meta"> <div class="post-meta">
<span class="post-date" th:text="${#dates.format(post.spec.publishTime,'yyyy-MM-dd')}"> publishTime </span> <span class="post-date" th:text="${#dates.format(post.spec.publishTime,'yyyy-MM-dd')}"> publishTime </span>
<span class="post-author" th:text="${':: '+post.owner.displayName}">:: Author</span> <span class="post-separator">
<span></span>
<span></span>
<span></span>
<span></span>
</span>
<span class="post-author" th:text="${post.owner.displayName}">Author</span>
</div> </div>
<span class="post-tags-inline" th:each="tag : ${post.tags}"> <span class="post-tags-inline" th:each="tag : ${post.tags}">
<a <a
@@ -19,8 +25,11 @@
>#Tag</a >#Tag</a
> >
</span> </span>
<div class="post-content"> <div class="post-body">
<div th:utext="${post.content.content}">Post Content</div> <div class="post-line-gutter"></div>
<div class="post-content">
<div th:utext="${post.content.content}">Post Content</div>
</div>
</div> </div>
</div> </div>
<div class="comment-wrap" th:if="${haloCommentEnabled}"> <div class="comment-wrap" th:if="${haloCommentEnabled}">
+1 -1
查看文件
@@ -13,5 +13,5 @@ spec:
repo: https://git.dev.cm/theme-terminal repo: https://git.dev.cm/theme-terminal
settingName: "theme-terminal-setting" settingName: "theme-terminal-setting"
configMapName: "theme-terminal-configMap" configMapName: "theme-terminal-configMap"
version: 1.1.8 version: 1.1.10
require: ">=2.22.0" require: ">=2.22.0"