mirror of
https://github.com/BetterSEQTA/BetterSEQTA-Plus.git
synced 2026-06-06 03:34:40 +00:00
f2fa9c39a9
Theme updates
288 lines
12 KiB
Svelte
288 lines
12 KiB
Svelte
<script lang="ts">
|
|
import type { CustomTheme, ThemeList } from '@/types/CustomThemes'
|
|
import { onDestroy, onMount } from 'svelte'
|
|
import browser from 'webextension-polyfill'
|
|
import { OpenThemeCreator } from '@/plugins/built-in/themes/ThemeCreator'
|
|
import { OpenStorePage } from '@/seqta/ui/renderStore'
|
|
import { themeUpdates } from '@/interface/hooks/ThemeUpdates'
|
|
import { closeExtensionPopup } from '@/seqta/utils/Closers/closeExtensionPopup'
|
|
import { ThemeManager } from '@/plugins/built-in/themes/theme-manager'
|
|
import { cloudAuth } from '@/seqta/utils/CloudAuth'
|
|
import SignInToFavoriteModal from '@/interface/components/SignInToFavoriteModal.svelte'
|
|
|
|
const themeManager = ThemeManager.getInstance();
|
|
|
|
let themes = $state<ThemeList | null>(null);
|
|
let { isEditMode } = $props<{ isEditMode: boolean }>();
|
|
let isDragging = $state(false);
|
|
let tempTheme = $state(null);
|
|
let favoriteStatus = $state<Record<string, boolean>>({});
|
|
let cloudLoggedIn = $state(cloudAuth.state.isLoggedIn);
|
|
let prevLoggedIn = $state(false);
|
|
let showSignInModal = $state(false);
|
|
|
|
cloudAuth.subscribe((s) => {
|
|
const now = s.isLoggedIn;
|
|
if (now && !prevLoggedIn && themes) void fetchThemes();
|
|
prevLoggedIn = now;
|
|
cloudLoggedIn = now;
|
|
});
|
|
|
|
const handleThemeClick = async (theme: CustomTheme, e: MouseEvent) => {
|
|
if (isEditMode) return;
|
|
if (theme.id === themes?.selectedTheme) {
|
|
themeManager.setTransitionPoint(e.clientX, e.clientY);
|
|
await themeManager.disableTheme();
|
|
themes.selectedTheme = '';
|
|
} else {
|
|
themeManager.setTransitionPoint(e.clientX, e.clientY);
|
|
await themeManager.setTheme(theme.id);
|
|
if (!themes) return;
|
|
themes.selectedTheme = theme.id;
|
|
}
|
|
}
|
|
|
|
const handleThemeDelete = async (themeId: string) => {
|
|
try {
|
|
await themeManager.deleteTheme(themeId);
|
|
if (!themes) return;
|
|
|
|
themes.themes = themes.themes.filter(theme => theme.id !== themeId);
|
|
if (themeId === themes.selectedTheme) {
|
|
themes.selectedTheme = '';
|
|
await themeManager.disableTheme();
|
|
}
|
|
} catch (error) {
|
|
console.error('Error deleting theme:', error);
|
|
}
|
|
}
|
|
|
|
const handleShareTheme = async (theme: CustomTheme) => {
|
|
try {
|
|
await themeManager.shareTheme(theme.id);
|
|
} catch (error) {
|
|
console.error('Error sharing theme:', error);
|
|
}
|
|
}
|
|
|
|
const handleDragOver = (e: DragEvent) => {
|
|
e.preventDefault();
|
|
isDragging = true;
|
|
}
|
|
|
|
const handleDragLeave = () => {
|
|
isDragging = false;
|
|
}
|
|
|
|
const handleDrop = async (e: DragEvent) => {
|
|
e.preventDefault();
|
|
isDragging = false;
|
|
const file = e.dataTransfer?.files[0];
|
|
if (!file) return;
|
|
|
|
const reader = new FileReader();
|
|
reader.onload = async (event: ProgressEvent<FileReader>) => {
|
|
try {
|
|
const result = JSON.parse(event.target?.result as string);
|
|
tempTheme = result;
|
|
await themeManager.installTheme(result, { fromStore: false });
|
|
await fetchThemes();
|
|
} catch (error) {
|
|
console.error('Error parsing file:', error);
|
|
alert('Error parsing file. Please upload a valid JSON theme file.');
|
|
}
|
|
tempTheme = null;
|
|
};
|
|
reader.readAsText(file);
|
|
}
|
|
|
|
const fetchThemes = async () => {
|
|
themes = {
|
|
themes: await themeManager.getAvailableThemes(),
|
|
selectedTheme: themeManager.getSelectedThemeId() || '',
|
|
}
|
|
if (themes && cloudLoggedIn) {
|
|
const token = await cloudAuth.getStoredToken();
|
|
if (token) {
|
|
const status: Record<string, boolean> = {};
|
|
await Promise.all(
|
|
themes.themes.map(async (t) => {
|
|
try {
|
|
const res = (await browser.runtime.sendMessage({
|
|
type: 'fetchThemeDetails',
|
|
themeId: t.id,
|
|
token,
|
|
})) as { success?: boolean; data?: { theme?: { is_favorited?: boolean } } };
|
|
if (res?.success && res?.data?.theme) {
|
|
status[t.id] = !!res.data.theme.is_favorited;
|
|
}
|
|
} catch {
|
|
// Theme may not exist on store (e.g. locally created)
|
|
}
|
|
})
|
|
);
|
|
favoriteStatus = status;
|
|
}
|
|
} else {
|
|
favoriteStatus = {};
|
|
}
|
|
}
|
|
|
|
const handleToggleFavorite = async (theme: CustomTheme, e: MouseEvent) => {
|
|
e.stopPropagation();
|
|
if (!cloudLoggedIn) {
|
|
showSignInModal = true;
|
|
return;
|
|
}
|
|
const token = await cloudAuth.getStoredToken();
|
|
if (!token) return;
|
|
const isFavorite = !favoriteStatus[theme.id];
|
|
const result = (await browser.runtime.sendMessage({
|
|
type: 'cloudFavorite',
|
|
themeId: theme.id,
|
|
token,
|
|
action: isFavorite ? 'favorite' : 'unfavorite',
|
|
})) as { success?: boolean };
|
|
if (result?.success) {
|
|
favoriteStatus = { ...favoriteStatus, [theme.id]: isFavorite };
|
|
}
|
|
}
|
|
|
|
onMount(async () => {
|
|
await fetchThemes();
|
|
themeUpdates.addListener(fetchThemes);
|
|
})
|
|
|
|
onDestroy(() => {
|
|
themeUpdates.removeListener(fetchThemes);
|
|
})
|
|
</script>
|
|
|
|
<div
|
|
class="pt-5 mb-1 w-full"
|
|
role="list"
|
|
tabindex="-1"
|
|
ondragover={handleDragOver}
|
|
ondragleave={handleDragLeave}
|
|
ondrop={handleDrop}
|
|
>
|
|
<div class="{isDragging ? 'opacity-100' : 'opacity-0'} transition pointer-events-none absolute w-full p-2 z-50">
|
|
<div class="sticky top-5 w-full h-64 bg-white rounded-xl shadow-xl dark:bg-zinc-900 dark:text-white outline-dashed outline-4 outline-zinc-200 dark:outline-zinc-700">
|
|
<div class="flex justify-center items-center h-full">
|
|
<div class="flex flex-col justify-center items-center">
|
|
<svg height="48" width="48" viewBox="0 0 48 48" xmlns="http://www.w3.org/2000/svg">
|
|
<g fill="currentColor">
|
|
<path d="M44,31a1,1,0,0,0-1,1v8a3,3,0,0,1-3,3H8a3,3,0,0,1-3-3V32a1,1,0,0,0-2,0v8a5.006,5.006,0,0,0,5,5H40a5.006,5.006,0,0,0,5-5V32A1,1,0,0,0,44,31Z" fill="currentColor"/>
|
|
<path d="M23.2,33.6a1,1,0,0,0,1.6,0l9-12A1,1,0,0,0,33,20H26V5a2,2,0,0,0-4,0V20H15a1,1,0,0,0-.8,1.6Z" fill="currentColor"/>
|
|
</g>
|
|
</svg>
|
|
<span class="text-lg">Import Theme</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<h2 class="pb-2 text-lg font-bold">Themes</h2>
|
|
<div class="flex flex-col gap-2 px-2">
|
|
{#if themes}
|
|
{#each themes.themes as theme (theme.id)}
|
|
<button
|
|
class="relative group w-full aspect-theme flex justify-center items-center rounded-xl transition ring dark:ring-white ring-zinc-300 {theme.id === themes.selectedTheme ? 'dark:ring-2 ring-4' : 'ring-0'}"
|
|
onclick={(e) => handleThemeClick(theme, e)}
|
|
>
|
|
{#if isEditMode}
|
|
<div
|
|
class="flex absolute top-2 right-2 z-20 place-items-center p-2 w-6 h-6 text-white bg-red-600 rounded-full opacity-100"
|
|
onclick={(event) => { event.stopPropagation(); handleThemeDelete(theme.id) }}
|
|
onkeydown={(event) => { if (event.key === 'Enter' || event.key === ' ') handleThemeDelete(theme.id) }}
|
|
role="button"
|
|
tabindex="-1"
|
|
>
|
|
<div class="w-4 h-0.5 bg-white"></div>
|
|
</div>
|
|
{/if}
|
|
|
|
{#if !isEditMode}
|
|
<div
|
|
class="flex absolute right-24 top-1/4 z-20 place-items-center p-2 w-8 h-8 text-center rounded-full opacity-0 transition-all -translate-y-1/2 group-hover:opacity-100 group-hover:top-1/2 {(favoriteStatus[theme.id] ?? false) ? 'text-red-400' : 'text-white/80'} bg-black/50"
|
|
onclick={(event) => handleToggleFavorite(theme, event)}
|
|
onkeydown={(event) => { if (event.key === 'Enter' || event.key === ' ') handleToggleFavorite(theme, event as any) }}
|
|
role="button"
|
|
tabindex="-1"
|
|
title={cloudLoggedIn ? ((favoriteStatus[theme.id] ?? false) ? 'Remove from favorites' : 'Add to favorites') : 'Sign in to favorite themes'}
|
|
>
|
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill={(favoriteStatus[theme.id] ?? false) ? 'currentColor' : 'none'} stroke="currentColor" stroke-width="2" class="w-5 h-5">
|
|
<path stroke-linecap="round" stroke-linejoin="round" d="M4.318 6.318a4.5 4.5 0 000 6.364L12 20.364l7.682-7.682a4.5 4.5 0 00-6.364-6.364L12 7.636l-1.318-1.318a4.5 4.5 0 00-6.364 0z" />
|
|
</svg>
|
|
</div>
|
|
<div
|
|
class="absolute z-20 flex w-8 h-8 p-2 text-white transition-all rounded-full delay-[20ms] opacity-0 top-1/4 right-2 bg-black/50 place-items-center group-hover:opacity-100 group-hover:top-1/2 -translate-y-1/2"
|
|
onclick={(event) => { event.stopPropagation(); OpenThemeCreator(theme.id); closeExtensionPopup() }}
|
|
onkeydown={(event) => { if (event.key === 'Enter' || event.key === ' ') OpenThemeCreator(theme.id); closeExtensionPopup() }}
|
|
role="button"
|
|
tabindex="-1"
|
|
>
|
|
<span class="text-lg font-IconFamily"></span>
|
|
</div>
|
|
|
|
<div
|
|
class="flex absolute right-12 top-1/4 z-20 place-items-center p-2 w-8 h-8 text-center rounded-full opacity-0 transition-all -translate-y-1/2 text-white/80 bg-black/50 group-hover:opacity-100 group-hover:top-1/2"
|
|
onclick={(event) => { event.stopPropagation(); handleShareTheme(theme) }}
|
|
onkeydown={(event) => { if (event.key === 'Enter' || event.key === ' ') handleShareTheme(theme) }}
|
|
role="button"
|
|
tabindex="-1"
|
|
>
|
|
<span class="text-lg font-IconFamily"></span>
|
|
</div>
|
|
{/if}
|
|
|
|
<div class="relative top-0 z-10 flex justify-center w-full h-full overflow-hidden transition dark:text-white rounded-xl group place-items-center bg-zinc-100 dark:bg-zinc-900 { isEditMode ? 'animate-shake brightness-90' : ''}">
|
|
{#if theme.coverImage}
|
|
<img
|
|
src={typeof theme.coverImage === 'string' ? theme.coverImage : URL.createObjectURL(theme.coverImage)}
|
|
alt={theme.name}
|
|
class="object-cover absolute inset-0 z-0 w-full h-full pointer-events-none"
|
|
/>
|
|
{/if}
|
|
{#if !theme.hideThemeName}
|
|
<div class="z-10 {theme.coverImage ? 'text-white' : ''}">{theme.name}</div>
|
|
{/if}
|
|
</div>
|
|
</button>
|
|
{/each}
|
|
{/if}
|
|
|
|
{#if tempTheme}
|
|
<div class="flex justify-center place-items-center w-full bg-gray-200 rounded-xl animate-pulse dark:bg-zinc-700/50 aspect-theme">
|
|
<svg class="w-5 h-5 text-white animate-spin" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24">
|
|
<circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle>
|
|
<path class="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path>
|
|
</svg>
|
|
</div>
|
|
{/if}
|
|
|
|
{#if themes && themes.themes.length > 0}
|
|
<div id="divider" class="w-full h-[1px] my-2 bg-zinc-100 dark:bg-zinc-600"></div>
|
|
{/if}
|
|
|
|
<button
|
|
onclick={() => OpenStorePage()}
|
|
class="flex justify-center items-center w-full rounded-xl transition aspect-theme bg-zinc-100 dark:bg-zinc-900 dark:text-white"
|
|
>
|
|
<span class="text-xl font-IconFamily"></span>
|
|
<span class="ml-2">Theme Store</span>
|
|
</button>
|
|
|
|
<button
|
|
onclick={() => { OpenThemeCreator(); closeExtensionPopup() }}
|
|
class="flex justify-center items-center w-full rounded-xl transition aspect-theme bg-zinc-100 dark:bg-zinc-900 dark:text-white"
|
|
>
|
|
<span class="text-xl font-IconFamily"></span>
|
|
<span class="ml-2">Create your own</span>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
{#if showSignInModal}
|
|
<SignInToFavoriteModal onClose={() => (showSignInModal = false)} />
|
|
{/if}
|