5 次代码提交

修改 17 个文件,包含 598 行新增162 行删除
第三方依赖
+2
查看文件
@@ -1,10 +1,12 @@
/// <reference types="vite/client" /> /// <reference types="vite/client" />
import type { Alpine } from "alpinejs"; import type { Alpine } from "alpinejs";
import type { TocManager } from "./src/alpine/toc";
export {}; export {};
declare global { declare global {
interface Window { interface Window {
Alpine: Alpine; Alpine: Alpine;
tocManager?: TocManager;
} }
} }
+35
查看文件
@@ -0,0 +1,35 @@
export const headerDividing = () => ({
isSticky: false,
$el: null as HTMLElement | null,
_observer: null as IntersectionObserver | null,
init() {
const centerEl = this.$el as HTMLElement;
if (!centerEl) return;
const header = centerEl.closest('.header') as HTMLElement;
if (!header) return;
// 直接观察 header 元素本身
this._observer = new IntersectionObserver(
(entries) => {
const entry = entries[0];
this.isSticky = entry.boundingClientRect.top < 0 && !entry.isIntersecting;
},
{
threshold: 0, // 只要有任何部分离开就触发
rootMargin: '0px' // 无偏移
}
);
this._observer.observe(header);
},
destroy() {
if (this._observer) {
this._observer.disconnect();
this._observer = null;
}
}
});
@@ -3,7 +3,7 @@ interface MenuState {
handleToggleMenu(): void; handleToggleMenu(): void;
} }
export const menu = (): MenuState => ({ export const headerMenu = (): MenuState => ({
isOpen: false, isOpen: false,
handleToggleMenu() { handleToggleMenu() {
@@ -20,6 +20,8 @@ interface LineSpanOptions {
interface ElementMetrics { interface ElementMetrics {
top: number; top: number;
height: number; height: number;
marginTop: number;
marginBottom: number;
paddingTop: number; paddingTop: number;
lineHeight: number; lineHeight: number;
lines: number; lines: number;
@@ -56,7 +58,7 @@ const isEmptyElement = (element: HTMLElement): boolean => {
return !element.textContent?.trim(); return !element.textContent?.trim();
}; };
const getStyleValue = (style: CSSStyleDeclaration, prop: 'paddingTop' | 'paddingBottom' | 'lineHeight' | 'fontSize'): number => const getStyleValue = (style: CSSStyleDeclaration, prop: 'paddingTop' | 'paddingBottom' | 'lineHeight' | 'fontSize' | 'marginTop' | 'marginBottom'): number =>
parseFloat(style[prop]) || 0; parseFloat(style[prop]) || 0;
const getDefaultLineHeight = (element: HTMLElement): number => { const getDefaultLineHeight = (element: HTMLElement): number => {
@@ -70,6 +72,8 @@ const getElementMetrics = (element: HTMLElement, containerRect: DOMRect, default
const style = getComputedStyle(element); const style = getComputedStyle(element);
const paddingTop = getStyleValue(style, 'paddingTop'); const paddingTop = getStyleValue(style, 'paddingTop');
const paddingBottom = getStyleValue(style, 'paddingBottom'); const paddingBottom = getStyleValue(style, 'paddingBottom');
const marginTop = getStyleValue(style, 'marginTop');
const marginBottom = getStyleValue(style, 'marginBottom');
const height = element.offsetHeight; const height = element.offsetHeight;
const contentHeight = height - paddingTop - paddingBottom; const contentHeight = height - paddingTop - paddingBottom;
const isNonText = isNonTextElement(tagName); const isNonText = isNonTextElement(tagName);
@@ -80,6 +84,8 @@ const getElementMetrics = (element: HTMLElement, containerRect: DOMRect, default
return { return {
top: element.getBoundingClientRect().top - containerRect.top, top: element.getBoundingClientRect().top - containerRect.top,
height, height,
marginTop,
marginBottom,
paddingTop, paddingTop,
lineHeight: contentHeight / lines, lineHeight: contentHeight / lines,
lines, lines,
@@ -147,7 +153,7 @@ const appendElementLines = (fragment: DocumentFragment, metrics: ElementMetrics,
// ============ 主组件 ============ // ============ 主组件 ============
export const lineNumbers = (): LineNumbersState => ({ export const postLineNum = (): LineNumbersState => ({
totalLines: 0, totalLines: 0,
resizeObserver: null, resizeObserver: null,
@@ -206,23 +212,44 @@ export const lineNumbers = (): LineNumbersState => ({
let currentLine = 1; let currentLine = 1;
let lastBottom = 0; let lastBottom = 0;
// 派发清空目录事件
document.dispatchEvent(new CustomEvent('post-toc:clear'));
for (const element of elements) { for (const element of elements) {
const metrics = getElementMetrics(element, containerRect, defaultLineHeight); const metrics = getElementMetrics(element, containerRect, defaultLineHeight);
const tagName = element.tagName.toLowerCase();
if (lastBottom > 0 && metrics.top > lastBottom) { // 如果是标题元素,派发添加目录事件
if (metrics.isHeading) {
const level = parseInt(tagName.charAt(1), 10);
const title = element.textContent?.trim() || '';
document.dispatchEvent(new CustomEvent('post-toc:add', {
detail: { level, title, element, tagName }
}));
}
// 计算实际的内容起始位置(考虑 margin-top
const contentStart = metrics.top - metrics.marginTop;
// 只有当间隙足够大时才填充行号
if (lastBottom > 0 && contentStart > lastBottom) {
currentLine = appendGapLines({ currentLine = appendGapLines({
fragment, fragment,
gapStart: lastBottom, gapStart: lastBottom,
gapEnd: metrics.top, gapEnd: contentStart,
lineHeight: defaultLineHeight, lineHeight: defaultLineHeight,
startLine: currentLine startLine: currentLine
}); });
} }
currentLine = appendElementLines(fragment, metrics, currentLine); currentLine = appendElementLines(fragment, metrics, currentLine);
lastBottom = metrics.top + metrics.height; // 更新 lastBottom,包含 margin-bottom
lastBottom = metrics.top + metrics.height + metrics.marginBottom;
} }
// 派发初始化滚动高亮事件
document.dispatchEvent(new CustomEvent('post-toc:init-highlight'));
gutter.innerHTML = ''; gutter.innerHTML = '';
gutter.appendChild(fragment); gutter.appendChild(fragment);
gutter.style.height = `${container.scrollHeight}px`; gutter.style.height = `${container.scrollHeight}px`;
+110
查看文件
@@ -0,0 +1,110 @@
// ============ 类型定义 ============
export interface PostTocItem {
id: string;
level: number;
title: string;
element: HTMLElement;
tagName: string;
}
// ============ 工具函数 ============
const generateId = (counter: number): string =>
`post-toc-${counter}-${Date.now()}`;
const ensureElementId = (element: HTMLElement, fallbackId: string): string => {
if (!element.id) element.id = fallbackId;
return element.id;
};
const findTopVisibleEntry = (entries: IntersectionObserverEntry[]): IntersectionObserverEntry | null => {
const visible = entries.filter(e => e.isIntersecting);
if (visible.length === 0) return null;
return visible.reduce((prev, curr) =>
curr.boundingClientRect.top < prev.boundingClientRect.top ? curr : prev
);
};
const createScrollObserver = (onActiveChange: (id: string) => void): IntersectionObserver =>
new IntersectionObserver(
(entries) => {
const topEntry = findTopVisibleEntry(entries);
if (topEntry) onActiveChange(topEntry.target.id);
},
{ rootMargin: '-80px 0px -70% 0px', threshold: [0, 0.5, 1] }
);
// ============ 事件常量 ============
export const TOC_EVENTS = {
CLEAR: 'post-toc:clear',
ADD: 'post-toc:add',
HIGHLIGHT: 'post-toc:init-highlight'
} as const;
// ============ Alpine 组件 ============
export const postToc = () => ({
items: [] as PostTocItem[],
activeId: '',
observer: null as IntersectionObserver | null,
// 私有属性
_isInitialized: false,
_counter: 0,
_minLevel: 6,
init() {
if (this._isInitialized) return;
document.addEventListener(TOC_EVENTS.CLEAR, () => this.clear());
document.addEventListener(TOC_EVENTS.ADD, (e) => {
const { level, title, element, tagName } = (e as CustomEvent).detail;
this.addItem(level, title, element, tagName);
});
document.addEventListener(TOC_EVENTS.HIGHLIGHT, () => this.initScrollHighlight());
this._isInitialized = true;
},
addItem(level: number, title: string, element: HTMLElement, tagName: string) {
this._counter++;
const id = ensureElementId(element, generateId(this._counter));
if (level < this._minLevel) this._minLevel = level;
this.items = [...this.items, { id, level, title, element, tagName }];
},
clear() {
this.items = [];
this.activeId = '';
this._counter = 0;
this._minLevel = 6;
this.observer?.disconnect();
},
initScrollHighlight() {
this.observer?.disconnect();
this.observer = createScrollObserver((id) => { this.activeId = id; });
this.items.forEach(item => this.observer?.observe(item.element));
},
scrollTo(id: string) {
const element = document.getElementById(id);
if (!element) return;
element.scrollIntoView({ behavior: 'smooth', block: 'start' });
this.activeId = id;
},
getIndent(level: number): string {
return `${(level - this._minLevel) * 16}px`;
}
});
+9 -5
查看文件
@@ -4,17 +4,21 @@ import "./styles/font-pixel.scss";
import Alpine from 'alpinejs' import Alpine from 'alpinejs'
import {upvote} from './alpine/upvote' import {upvote} from './alpine/upvote'
import {themeMode} from './alpine/themeMode' import {themeMode} from './alpine/theme-mode'
import {menu} from './alpine/menu' import {postLineNum} from './alpine/post-line-num'
import {lineNumbers} from './alpine/lineNumbers' import {postToc} from './alpine/post-toc'
import {headerDividing} from './alpine/header-dividing'
import {headerMenu} from './alpine/header-menu'
import {typewriterEffect} from './utils' import {typewriterEffect} from './utils'
window.Alpine = Alpine 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('postLineNum', postLineNum)
Alpine.data('lineNumbers', lineNumbers) Alpine.data('postToc', postToc)
Alpine.data('headerMenu', headerMenu)
Alpine.data('headerDividing', headerDividing)
Alpine.start() Alpine.start()
+129
查看文件
@@ -0,0 +1,129 @@
@import './variables.scss';
.dividing {
flex: 1;
display: block;
width: 100%;
height: 25px;
position: relative;
transition: opacity 0.3s ease-in-out;
--dash-spacing: 10px;
--dash-width: 2px;
// 虚线背景层 - 带遮罩
&::before {
content: '';
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
background: repeating-linear-gradient(
90deg,
var(--foreground),
var(--foreground) var(--dash-width),
transparent 0,
transparent var(--dash-spacing)
);
background-size: var(--dash-spacing) 100%;
animation: line-flow 3s linear infinite;
// 遮罩层实现虚线淡入淡出效果
mask-image: linear-gradient(
90deg,
transparent 0%,
black 3%,
black 97%,
transparent 100%
);
-webkit-mask-image: linear-gradient(
90deg,
transparent 0%,
black 3%,
black 97%,
transparent 100%
);
}
// 扫描光效 - 模拟终端扫描线
&::after {
content: '';
position: absolute;
left: -30%;
top: 50%;
transform: translateY(-50%);
width: 30%;
height: 120%;
filter: blur(4px);
background: linear-gradient(
90deg,
transparent 0%,
var(--green) 80%,
transparent 100%
);
opacity: 0.3;
animation: scan-line 5s ease-in-out infinite;
pointer-events: none;
z-index: 10;
transition: all 0.5s cubic-bezier(0.4, 0, 0.2, 1);
}
&.sticky {
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 5px;
z-index: 100;
--dash-spacing: 6px;
--dash-width: 2px;
background: var(--background);
opacity: 0.8;
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
&::before {
animation: line-flow 1.5s linear infinite;
mask-image: none;
-webkit-mask-image: none;
top: calc(50% - 1px);
height: 2px;
}
&::after {
animation: scan-line 2.5s ease-in-out infinite;
opacity: 0.6;
filter: blur(3px);
// 调整扫描线高度以适应 5px 容器
height: 200%;
width: 25%;
}
}
}
// 虚线流动动画 - 模拟加载进度
@keyframes line-flow {
0% {
background-position: 0 0;
}
100% {
background-position: var(--dash-spacing) 0;
}
}
// 扫描线动画 - 模拟终端扫描效果
@keyframes scan-line {
0% {
left: -20%;
opacity: 0;
}
30% {
opacity: .3;
}
70% {
opacity: .3;
}
100% {
left: 80%;
opacity: 0;
}
}
+4 -90
查看文件
@@ -11,97 +11,11 @@
justify-content: space-between; justify-content: space-between;
} }
.dividing { &__center {
display: flex;
align-items: center;
justify-content: center;
flex: 1; flex: 1;
display: block;
width: 100%;
height: 25px;
position: relative;
// 虚线背景层 - 带遮罩
&::before {
content: '';
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
background: repeating-linear-gradient(
90deg,
var(--foreground),
var(--foreground) 2px,
transparent 0,
transparent 10px
);
background-size: 10px 100%;
animation: line-flow 3s linear infinite;
// 遮罩层实现虚线淡入淡出效果
mask-image: linear-gradient(
90deg,
transparent 0%,
black 3%,
black 97%,
transparent 100%
);
-webkit-mask-image: linear-gradient(
90deg,
transparent 0%,
black 3%,
black 97%,
transparent 100%
);
}
// 扫描光效 - 模拟终端扫描线
&::after {
content: '';
position: absolute;
left: -30%;
top: 50%;
transform: translateY(-50%);
width: 30%;
height: 120%;
filter: blur(4px);
background: linear-gradient(
90deg,
transparent 0%,
var(--green) 80%,
transparent 100%
);
opacity: 0.3;
animation: scan-line 5s ease-in-out infinite;
pointer-events: none;
z-index: 10;
}
}
// 虚线流动动画 - 模拟加载进度
@keyframes line-flow {
0% {
background-position: 0 0;
}
100% {
background-position: 10px 0;
}
}
// 扫描线动画 - 模拟终端扫描效果
@keyframes scan-line {
0% {
left: -20%;
opacity: 0;
}
30% {
opacity: .3;
}
70% {
opacity: .3;
}
100% {
left: 80%;
opacity: 0;
}
} }
.menu { .menu {
+54
查看文件
@@ -0,0 +1,54 @@
@import "variables";
.post-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.85rem;
color: var(--foreground);
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;
right: -3px;
color: var(--brightRed);
font-size: 0.875rem;
font-family: Hack, Monaco, Consolas, monospace;
transition: transform 0.15s ease, filter 0.15s ease;
}
&:hover .line-breakpoint {
transform: scale(1.2);
filter: brightness(1.2);
}
}
}
+188
查看文件
@@ -0,0 +1,188 @@
@import "variables";
.post-toc {
position: absolute;
left: -280px;
top: 85px;
width: 200px;
max-height: calc(100vh - 200px);
overflow-y: auto;
overflow-x: hidden;
z-index: 100;
font-size: 0.8rem;
line-height: 1.6;
color: var(--foreground);
opacity: 0.7;
transition: opacity 0.2s ease;
// 隐藏滚动条但保持可滚动
scrollbar-width: thin;
scrollbar-color: var(--foreground) transparent;
&:hover {
opacity: 1;
}
&::-webkit-scrollbar {
width: 4px;
}
&::-webkit-scrollbar-track {
background: transparent;
}
&::-webkit-scrollbar-thumb {
background-color: color-mix(in srgb, var(--foreground) 30%, transparent);
border-radius: 2px;
}
// 移动端和平板隐藏
@media (max-width: 1200px) {
display: none;
}
}
.toc-tree {
list-style: none;
margin: 0;
padding: 0;
padding-left: 10px;
position: relative;
// 左侧边框线(不连接顶部)
&::before {
content: '';
position: absolute;
left: 0;
top: 0;
bottom: 0;
width: 1px;
background-color: color-mix(in srgb, var(--foreground) 20%, transparent);
}
}
.toc-item {
position: relative;
display: flex;
align-items: flex-start;
padding: 2px 0;
cursor: pointer;
transition: background-color 0.15s ease;
border-radius: 3px;
&:hover {
background-color: color-mix(in srgb, var(--foreground) 8%, transparent);
}
// 折叠箭头
&__toggle {
flex-shrink: 0;
width: 16px;
height: 20px;
display: flex;
align-items: center;
justify-content: center;
color: var(--foreground);
opacity: 0.5;
font-size: 0.6rem;
transition: transform 0.2s ease;
user-select: none;
&--expanded {
transform: rotate(0deg);
}
&--collapsed {
transform: rotate(-90deg);
}
&--empty {
visibility: hidden;
}
}
// 标签图标 <>
&__icon {
flex-shrink: 0;
color: var(--yellow);
margin-right: 4px;
opacity: 0.7;
}
// 标签名 (h1, h2, div, etc.)
&__tag {
flex-shrink: 0;
color: var(--blue);
margin-right: 6px;
}
// 标题文字
&__title {
color: var(--foreground);
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
opacity: 0.8;
transition: color 0.15s ease, opacity 0.15s ease;
}
// 活跃状态
&--active {
background-color: color-mix(in srgb, var(--brightBlue) 15%, transparent);
.toc-item__title {
color: var(--brightBlue);
opacity: 1;
}
.toc-item__tag {
color: var(--brightBlue);
}
// 左侧激活指示线
&::before {
content: '' !important;
position: absolute;
left: -10px !important;
top: 4px;
bottom: 4px;
width: 2px;
background-color: var(--brightBlue);
border-radius: 1px;
}
}
}
// 子级列表
.toc-children {
list-style: none;
margin: 0;
padding: 0;
padding-left: 16px;
&--collapsed {
display: none;
}
}
// 目录头部
.toc-header {
display: flex;
align-items: center;
padding: 4px 0 8px 0;
margin-bottom: 4px;
border-bottom: 1px solid color-mix(in srgb, var(--foreground) 15%, transparent);
color: var(--foreground);
opacity: 0.6;
font-size: 0.75rem;
&__icon {
margin-right: 6px;
color: var(--yellow);
}
&__title {
text-transform: uppercase;
letter-spacing: 0.5px;
}
}
+1 -54
查看文件
@@ -24,6 +24,7 @@
} }
.post { .post {
position: relative;
width: 100%; width: 100%;
text-align: left; text-align: left;
margin: 20px auto; margin: 20px auto;
@@ -133,60 +134,6 @@
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 { &-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; line-height: 1.54;
+5 -2
查看文件
@@ -2,11 +2,14 @@
@tailwind utilities; @tailwind utilities;
@tailwind components; @tailwind components;
@import 'main';
@import 'logo';
@import 'buttons'; @import 'buttons';
@import 'header'; @import 'header';
@import 'logo'; @import 'header-dividing';
@import 'main';
@import 'post'; @import 'post';
@import 'post-line-num';
@import 'post-toc';
@import 'pagination'; @import 'pagination';
@import 'footer'; @import 'footer';
@import 'typed-text'; @import 'typed-text';
+4 -2
查看文件
@@ -5,7 +5,9 @@
<img class="icon" th:if="${not #strings.isEmpty(theme.config.basic.logo)}" th:src="${theme.config.basic.logo}" alt="Logo" /> <img class="icon" th:if="${not #strings.isEmpty(theme.config.basic.logo)}" th:src="${theme.config.basic.logo}" alt="Logo" />
<div class="text" th:if="${not #strings.isEmpty(theme.config.basic.title)}" th:text="${theme.config.basic.title}"></div> <div class="text" th:if="${not #strings.isEmpty(theme.config.basic.title)}" th:text="${theme.config.basic.title}"></div>
</a> </a>
<div class="dividing"></div> <div class="header__center" x-data="headerDividing()">
<div class="dividing" :class="{ 'sticky': isSticky }"></div>
</div>
<button <button
type="button" type="button"
class="button" class="button"
@@ -47,7 +49,7 @@
</div> </div>
<nav class="menu"> <nav class="menu">
<ul th:if="${menu != null} and ${not #lists.isEmpty(menu.menuItems)}" class="menu__inner"> <ul th:if="${menu != null} and ${not #lists.isEmpty(menu.menuItems)}" class="menu__inner">
<li th:each="menuItem : ${menu.menuItems}" x-data="menu()" @mouseenter="handleToggleMenu()" @mouseleave="handleToggleMenu()"> <li th:each="menuItem : ${menu.menuItems}" x-data="headerMenu()" @mouseenter="handleToggleMenu()" @mouseleave="handleToggleMenu()">
<a <a
class="text-gray-600 hover:text-blue-600" class="text-gray-600 hover:text-blue-600"
th:href="${menuItem.status.href}" th:href="${menuItem.status.href}"
+22 -1
查看文件
@@ -4,7 +4,27 @@
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" x-data="lineNumbers" x-init="init()"> <div class="post" x-data="postLineNum" x-init="init()">
<!-- 目录组件 -->
<div class="post-toc" x-data="postToc" x-init="init()">
<div class="toc-header">
<span class="toc-header__icon">&lt;&gt;</span>
<span class="toc-header__title">Outline</span>
</div>
<ul class="toc-tree">
<template x-for="(item, index) in items" :key="index">
<li class="toc-item"
:class="{'toc-item--active': activeId === item.id}"
:style="`padding-left: ${getIndent(item.level)}`"
@click="scrollTo(item.id)">
<span class="toc-item__toggle toc-item__toggle--empty"></span>
<span class="toc-item__icon">&lt;&gt;</span>
<span class="toc-item__tag" x-text="item.tagName"></span>
<span class="toc-item__title" x-text="item.title"></span>
</li>
</template>
</ul>
</div>
<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>
@@ -26,6 +46,7 @@
> >
</span> </span>
<div class="post-body"> <div class="post-body">
<!-- 行数组件 -->
<div class="post-line-gutter"></div> <div class="post-line-gutter"></div>
<div class="post-content"> <div class="post-content">
<div th:utext="${post.content.content}">Post Content</div> <div th:utext="${post.content.content}">Post Content</div>
+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.10 version: 1.2.1
require: ">=2.22.0" require: ">=2.22.0"
+1 -1
查看文件
@@ -16,5 +16,5 @@
"noImplicitReturns": true, "noImplicitReturns": true,
"skipLibCheck": true "skipLibCheck": true
}, },
"include": ["src", "./env.d.ts"] "include": ["types", "src", "./env.d.ts"]
} }