2024-11-18 01:09:28 +08:00
|
|
|
import { ThemeConfig, ThemeTemplate } from "contracts/themeContract";
|
|
|
|
import { ApiService } from "./apiService";
|
2024-11-14 01:44:26 +08:00
|
|
|
|
|
|
|
export class ThemeService {
|
2024-11-18 01:09:28 +08:00
|
|
|
private static instance: ThemeService;
|
|
|
|
private currentTheme?: ThemeConfig;
|
|
|
|
private api: ApiService;
|
2024-11-14 01:44:26 +08:00
|
|
|
|
2024-11-18 01:09:28 +08:00
|
|
|
private constructor(api: ApiService) {
|
|
|
|
this.api = api;
|
|
|
|
}
|
2024-11-14 01:44:26 +08:00
|
|
|
|
2024-11-18 01:09:28 +08:00
|
|
|
public static getInstance(api?: ApiService): ThemeService {
|
|
|
|
if (!ThemeService.instance && api) {
|
|
|
|
ThemeService.instance = new ThemeService(api);
|
2024-11-14 01:44:26 +08:00
|
|
|
}
|
2024-11-18 01:09:28 +08:00
|
|
|
return ThemeService.instance;
|
|
|
|
}
|
2024-11-14 01:44:26 +08:00
|
|
|
|
2024-11-18 01:09:28 +08:00
|
|
|
public async initialize(): Promise<void> {
|
|
|
|
try {
|
|
|
|
const themeConfig = await this.api.request<ThemeConfig>(
|
2024-11-22 13:13:04 +08:00
|
|
|
"/theme/current",
|
|
|
|
{ method: "GET" },
|
2024-11-18 01:09:28 +08:00
|
|
|
);
|
|
|
|
await this.loadTheme(themeConfig);
|
|
|
|
} catch (error) {
|
2024-11-22 13:13:04 +08:00
|
|
|
console.error("Failed to initialize theme:", error);
|
2024-11-18 01:09:28 +08:00
|
|
|
throw error;
|
2024-11-14 01:44:26 +08:00
|
|
|
}
|
2024-11-18 01:09:28 +08:00
|
|
|
}
|
2024-11-14 01:44:26 +08:00
|
|
|
|
2024-11-18 01:09:28 +08:00
|
|
|
private async loadTheme(config: ThemeConfig): Promise<void> {
|
|
|
|
try {
|
|
|
|
this.currentTheme = config;
|
|
|
|
await this.loadTemplates();
|
|
|
|
} catch (error) {
|
2024-11-22 13:13:04 +08:00
|
|
|
console.error("Failed to load theme:", error);
|
2024-11-18 01:09:28 +08:00
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
2024-11-14 01:44:26 +08:00
|
|
|
|
2024-11-18 01:09:28 +08:00
|
|
|
private async loadTemplates(): Promise<void> {
|
|
|
|
if (!this.currentTheme) {
|
2024-11-22 13:13:04 +08:00
|
|
|
throw new Error("No theme configuration loaded");
|
2024-11-18 01:09:28 +08:00
|
|
|
}
|
2024-11-14 01:44:26 +08:00
|
|
|
|
2024-11-18 01:09:28 +08:00
|
|
|
const loadTemplate = async (template: ThemeTemplate) => {
|
|
|
|
try {
|
|
|
|
const response = await fetch(template.path);
|
|
|
|
const templateContent = await response.text();
|
|
|
|
this.templates.set(template.name, templateContent);
|
|
|
|
} catch (error) {
|
|
|
|
console.error(`Failed to load template ${template.name}:`, error);
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-11-22 13:13:04 +08:00
|
|
|
const loadPromises = Array.from(this.currentTheme.templates.values()).map(
|
|
|
|
(template) => loadTemplate(template),
|
|
|
|
);
|
|
|
|
|
2024-11-18 01:09:28 +08:00
|
|
|
await Promise.all(loadPromises);
|
|
|
|
}
|
|
|
|
|
|
|
|
public getThemeConfig(): ThemeConfig | undefined {
|
|
|
|
return this.currentTheme;
|
|
|
|
}
|
|
|
|
|
|
|
|
public getTemplate(templateName: string): string {
|
|
|
|
const template = this.templates.get(templateName);
|
|
|
|
if (!template) {
|
|
|
|
throw new Error(`Template ${templateName} not found`);
|
2024-11-14 01:44:26 +08:00
|
|
|
}
|
2024-11-18 01:09:28 +08:00
|
|
|
return template;
|
|
|
|
}
|
2024-11-14 01:44:26 +08:00
|
|
|
|
2024-11-18 01:09:28 +08:00
|
|
|
public getTemplateByRoute(route: string): string {
|
|
|
|
if (!this.currentTheme) {
|
2024-11-22 13:13:04 +08:00
|
|
|
throw new Error("No theme configuration loaded");
|
2024-11-14 01:44:26 +08:00
|
|
|
}
|
|
|
|
|
2024-11-18 01:09:28 +08:00
|
|
|
let templateName: string | undefined;
|
|
|
|
|
2024-11-22 13:13:04 +08:00
|
|
|
if (route === "/") {
|
2024-11-18 01:09:28 +08:00
|
|
|
templateName = this.currentTheme.routes.index;
|
2024-11-22 13:13:04 +08:00
|
|
|
} else if (route.startsWith("/post/")) {
|
2024-11-18 01:09:28 +08:00
|
|
|
templateName = this.currentTheme.routes.post;
|
2024-11-22 13:13:04 +08:00
|
|
|
} else if (route.startsWith("/tag/")) {
|
2024-11-18 01:09:28 +08:00
|
|
|
templateName = this.currentTheme.routes.tag;
|
2024-11-22 13:13:04 +08:00
|
|
|
} else if (route.startsWith("/category/")) {
|
2024-11-18 01:09:28 +08:00
|
|
|
templateName = this.currentTheme.routes.category;
|
|
|
|
} else {
|
|
|
|
templateName = this.currentTheme.routes.page.get(route);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!templateName) {
|
|
|
|
templateName = this.currentTheme.routes.error;
|
2024-11-14 01:44:26 +08:00
|
|
|
}
|
|
|
|
|
2024-11-18 01:09:28 +08:00
|
|
|
return this.getTemplate(templateName);
|
|
|
|
}
|
2024-11-14 01:44:26 +08:00
|
|
|
|
2024-11-18 01:09:28 +08:00
|
|
|
public async updateThemeConfig(config: Partial<ThemeConfig>): Promise<void> {
|
|
|
|
try {
|
|
|
|
const updatedConfig = await this.api.request<ThemeConfig>(
|
2024-11-22 13:13:04 +08:00
|
|
|
"/theme/config",
|
2024-11-18 01:09:28 +08:00
|
|
|
{
|
2024-11-22 13:13:04 +08:00
|
|
|
method: "PUT",
|
2024-11-18 01:09:28 +08:00
|
|
|
headers: {
|
2024-11-22 13:13:04 +08:00
|
|
|
"Content-Type": "application/json",
|
2024-11-18 01:09:28 +08:00
|
|
|
},
|
|
|
|
body: JSON.stringify(config),
|
2024-11-22 13:13:04 +08:00
|
|
|
},
|
2024-11-18 01:09:28 +08:00
|
|
|
);
|
|
|
|
|
|
|
|
await this.loadTheme(updatedConfig);
|
|
|
|
} catch (error) {
|
2024-11-22 13:13:04 +08:00
|
|
|
console.error("Failed to update theme configuration:", error);
|
2024-11-18 01:09:28 +08:00
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
2024-11-22 13:13:04 +08:00
|
|
|
}
|