Chore: Completed Splash-Screen & Iniital Tauri Wire-Up

This commit is contained in:
Youwes09
2026-06-02 08:27:37 -05:00
parent c5243ba30c
commit 18027baee1
45 changed files with 2981 additions and 2013 deletions
+176 -79
View File
@@ -1,62 +1,79 @@
<script lang="ts">
import { onMount } from 'svelte'
import { detectOs } from '$lib/components/chrome/titlebarOs'
import type { OsKind } from '$lib/components/chrome/titlebarOs'
import { getCurrentWindow } from '@tauri-apps/api/window'
import { platform } from '@tauri-apps/plugin-os'
import { invoke } from '@tauri-apps/api/core'
import { settingsState, updateSettings } from '$lib/state/settings.svelte'
let { onClose }: { onClose: () => void } = $props()
const { }: {} = $props()
const isTauri = typeof window !== 'undefined' && '__TAURI_INTERNALS__' in window
const win = getCurrentWindow()
const os = platform()
const isMac = os === 'macos'
const isWindows = os === 'windows'
let os: OsKind = $state('unknown')
let isFullscreen = $state(false)
let isFullscreen = $state(false)
let closeDialogOpen = $state(false)
let closeRemember = $state(false)
onMount(async () => {
if (!isTauri) return
const { getCurrentWindow } = await import('@tauri-apps/api/window')
const win = getCurrentWindow()
os = await detectOs()
isFullscreen = await win.isFullscreen()
const unlisten = await win.onResized(async () => {
const unlistenResize = await win.onResized(async () => {
isFullscreen = await win.isFullscreen()
})
return unlisten
const unlistenClose = await win.listen('tauri://close-requested', handleCloseRequested)
return () => {
unlistenResize()
unlistenClose()
}
})
const isMac = $derived(os === 'macos')
const isWindows = $derived(os === 'windows')
async function minimize() {
const { getCurrentWindow } = await import('@tauri-apps/api/window')
getCurrentWindow().minimize()
async function doQuit() {
if (settingsState.settings.autoStartServer) {
await Promise.race([
invoke('kill_server').catch(() => {}),
new Promise(res => setTimeout(res, 2000)),
])
}
await invoke('exit_app')
}
async function toggleMaximize() {
const { getCurrentWindow } = await import('@tauri-apps/api/window')
getCurrentWindow().toggleMaximize()
async function doHide() {
await win.hide()
}
async function exitFullscreen() {
const { getCurrentWindow } = await import('@tauri-apps/api/window')
getCurrentWindow().setFullscreen(false)
async function handleCloseRequested() {
const action = settingsState.settings.closeAction ?? 'ask'
if (action === 'tray') { await doHide(); return }
if (action === 'quit') { await doQuit(); return }
closeDialogOpen = true
}
async function confirmClose(choice: 'tray' | 'quit') {
closeDialogOpen = false
if (closeRemember) updateSettings({ closeAction: choice })
closeRemember = false
if (choice === 'tray') await doHide()
else await doQuit()
}
</script>
{#if !isFullscreen}
<div class="bar" data-tauri-drag-region>
{#if isMac}<div class="mac-spacer" data-tauri-drag-region></div>{/if}
{#if isMac}<div class="mac-spacer"></div>{/if}
<span class="title" data-tauri-drag-region>Moku</span>
{#if !isMac}
<div class="controls">
<button onclick={minimize} title="Minimize" aria-label="Minimize">
<svg width="10" height="1" viewBox="0 0 10 1"><line x1="0" y1="0.5" x2="10" y2="0.5" stroke="currentColor" stroke-width="1.5"/></svg>
<button onclick={() => win.minimize()} title="Minimize" aria-label="Minimize">
<svg width="10" height="1" viewBox="0 0 10 1"><line x1="0" y1="0.5" x2="10" y2="0.5" stroke="currentColor" stroke-width="1.5" /></svg>
</button>
<button onclick={toggleMaximize} title="Maximize" aria-label="Maximize">
<svg width="9" height="9" viewBox="0 0 9 9"><rect x="0.75" y="0.75" width="7.5" height="7.5" rx="1" fill="none" stroke="currentColor" stroke-width="1.5"/></svg>
<button onclick={() => win.toggleMaximize()} title="Maximize" aria-label="Maximize">
<svg width="9" height="9" viewBox="0 0 9 9"><rect x="0.75" y="0.75" width="7.5" height="7.5" rx="1" fill="none" stroke="currentColor" stroke-width="1.5" /></svg>
</button>
<button class="close" onclick={onClose} title="Close" aria-label="Close">
<button class="close" onclick={handleCloseRequested} title="Close" aria-label="Close">
<svg width="10" height="10" viewBox="0 0 10 10">
<line x1="1" y1="1" x2="9" y2="9" stroke="currentColor" stroke-width="1.5" stroke-linecap="round"/>
<line x1="9" y1="1" x2="1" y2="9" stroke="currentColor" stroke-width="1.5" stroke-linecap="round"/>
<line x1="1" y1="1" x2="9" y2="9" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" />
<line x1="9" y1="1" x2="1" y2="9" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" />
</svg>
</button>
</div>
@@ -64,7 +81,7 @@
</div>
{:else if isWindows}
<div class="fullscreen-controls">
<button onclick={exitFullscreen} title="Exit Fullscreen" aria-label="Exit Fullscreen">
<button onclick={() => win.setFullscreen(false)} title="Exit Fullscreen" aria-label="Exit Fullscreen">
<svg width="10" height="10" viewBox="0 0 10 10">
<polyline points="1,4 1,1 4,1" fill="none" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/>
<polyline points="6,1 9,1 9,4" fill="none" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/>
@@ -72,63 +89,143 @@
<polyline points="4,9 1,9 1,6" fill="none" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/>
</svg>
</button>
<button class="close" onclick={onClose} title="Close" aria-label="Close">
<button class="close" onclick={handleCloseRequested} title="Close" aria-label="Close">
<svg width="10" height="10" viewBox="0 0 10 10">
<line x1="1" y1="1" x2="9" y2="9" stroke="currentColor" stroke-width="1.5" stroke-linecap="round"/>
<line x1="9" y1="1" x2="1" y2="9" stroke="currentColor" stroke-width="1.5" stroke-linecap="round"/>
<line x1="1" y1="1" x2="9" y2="9" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" />
<line x1="9" y1="1" x2="1" y2="9" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" />
</svg>
</button>
</div>
{/if}
{#if closeDialogOpen}
<div class="close-backdrop" role="presentation" onclick={() => { closeDialogOpen = false; closeRemember = false; }}>
<div class="close-dialog" role="dialog" aria-modal="true" onclick={(e) => e.stopPropagation()}>
<div class="close-header">
<p class="close-title">Close Moku?</p>
<p class="close-sub">Choose how the app should exit.</p>
</div>
<div class="close-actions">
<button class="close-btn" onclick={() => confirmClose('tray')}>
<span class="close-btn-label">Minimize to Tray</span>
<span class="close-btn-desc">Keep running in the background</span>
</button>
<button class="close-btn close-btn-danger" onclick={() => confirmClose('quit')}>
<span class="close-btn-label">Quit</span>
<span class="close-btn-desc">Stop Moku entirely</span>
</button>
</div>
<button class="close-remember" onclick={() => closeRemember = !closeRemember}>
<span class="close-remember-toggle" class:on={closeRemember}><span class="close-remember-thumb"></span></span>
<span class="close-remember-label">Remember my choice</span>
</button>
</div>
</div>
{/if}
<style>
.bar {
display: flex;
align-items: center;
justify-content: space-between;
height: var(--titlebar-height);
padding: 0 6px 0 var(--sp-4);
background: transparent;
flex-shrink: 0;
user-select: none;
}
.bar { display: flex; align-items: center; justify-content: space-between; height: var(--titlebar-height); padding: 0 6px 0 var(--sp-4); background: transparent; flex-shrink: 0; user-select: none; -webkit-app-region: drag; }
.mac-spacer { width: 70px; flex-shrink: 0; -webkit-app-region: drag; }
.title { font-family: var(--font-ui); font-size: var(--text-2xs); color: var(--text-faint); letter-spacing: var(--tracking-wider); text-transform: uppercase; opacity: 0.5; -webkit-app-region: drag; }
.controls { display: flex; align-items: center; gap: 2px; -webkit-app-region: no-drag; }
.mac-spacer { width: 70px; flex-shrink: 0; }
.title {
font-family: var(--font-ui);
font-size: var(--text-2xs);
color: var(--text-faint);
letter-spacing: var(--tracking-wider);
text-transform: uppercase;
opacity: 0.5;
}
.controls { display: flex; align-items: center; gap: 2px; }
button {
display: flex;
align-items: center;
justify-content: center;
width: 28px;
height: 28px;
border-radius: var(--radius-sm);
.controls button,
.fullscreen-controls button {
display: flex; align-items: center; justify-content: center;
width: 28px; height: 28px; border-radius: var(--radius-sm);
color: var(--text-faint);
transition: color var(--t-base), background var(--t-base);
-webkit-app-region: no-drag;
}
button:hover { color: var(--text-muted); background: rgba(255,255,255,0.06); }
.close:hover { color: #fff; background: #c0392b; }
.controls button:hover,
.fullscreen-controls button:hover { color: var(--text-muted); background: rgba(255,255,255,0.06); }
.controls .close:hover,
.fullscreen-controls .close:hover { color: #fff; background: #c0392b; }
.fullscreen-controls {
position: fixed;
top: 0;
right: 0;
z-index: 9999;
display: flex;
align-items: center;
gap: 2px;
padding: 4px;
opacity: 0;
transition: opacity var(--t-base);
}
.fullscreen-controls { position: fixed; top: 0; right: 0; z-index: 9999; display: flex; align-items: center; gap: 2px; padding: 4px; opacity: 0; transition: opacity 0.2s ease; -webkit-app-region: no-drag; }
.fullscreen-controls:hover { opacity: 1; }
.close-backdrop {
position: fixed; inset: 0;
background: rgba(0,0,0,0.5);
backdrop-filter: blur(8px);
-webkit-backdrop-filter: blur(8px);
display: flex; align-items: center; justify-content: center;
z-index: var(--z-modal);
animation: cdFade 0.18s ease both;
}
@keyframes cdFade { from { opacity: 0 } to { opacity: 1 } }
.close-dialog {
font-family: var(--font-ui);
background: var(--bg-surface);
border: 1px solid var(--border-base);
border-radius: var(--radius-2xl);
padding: var(--sp-5);
display: flex; flex-direction: column; gap: var(--sp-3);
width: 300px;
box-shadow:
0 0 0 1px rgba(255,255,255,0.04) inset,
0 24px 64px rgba(0,0,0,0.7),
0 8px 24px rgba(0,0,0,0.4);
animation: cdPop 0.22s cubic-bezier(0.16,1,0.3,1) both;
}
@keyframes cdPop { from { opacity: 0; transform: scale(0.96) translateY(6px) } to { opacity: 1; transform: none } }
.close-header { display: flex; flex-direction: column; gap: 3px; }
.close-title { font-size: var(--text-base); font-weight: var(--weight-medium); color: var(--text-primary); letter-spacing: var(--tracking-tight); margin: 0; }
.close-sub { font-size: var(--text-xs); color: var(--text-faint); letter-spacing: var(--tracking-wide); margin: 0; }
.close-actions { display: flex; flex-direction: column; gap: var(--sp-1); }
.close-btn {
display: flex; flex-direction: column; align-items: flex-start; gap: 3px;
width: 100%; padding: 10px var(--sp-3);
border-radius: var(--radius-lg);
border: 1px solid var(--border-dim);
background: var(--bg-raised);
cursor: pointer; text-align: left;
font-family: var(--font-ui);
transition: background var(--t-base), border-color var(--t-base), transform 80ms ease;
}
.close-btn:hover { background: var(--bg-overlay); border-color: var(--border-strong); }
.close-btn:active { transform: scale(0.985); }
.close-btn-danger { border-color: color-mix(in srgb, var(--color-error) 28%, transparent); }
.close-btn-danger:hover { background: var(--color-error-bg); border-color: color-mix(in srgb, var(--color-error) 55%, transparent); }
.close-btn-danger .close-btn-label { color: var(--color-error); }
.close-btn-danger .close-btn-desc { color: color-mix(in srgb, var(--color-error) 50%, var(--text-faint)); }
.close-btn-label { font-size: var(--text-sm); color: var(--text-secondary); font-weight: var(--weight-medium); line-height: 1.2; }
.close-btn-desc { font-size: var(--text-2xs); color: var(--text-faint); letter-spacing: var(--tracking-wide); line-height: 1.2; }
.close-remember {
display: flex; align-items: center; gap: var(--sp-2);
width: 100%; padding: var(--sp-3) 0 0;
border-top: 1px solid var(--border-dim);
background: none; border-left: none; border-right: none; border-bottom: none;
cursor: pointer; user-select: none;
font-family: var(--font-ui);
}
.close-remember:hover .close-remember-label { color: var(--text-muted); }
.close-remember-toggle {
position: relative; flex-shrink: 0;
width: 28px; height: 16px;
border-radius: var(--radius-full);
border: 1px solid var(--border-strong);
background: var(--bg-overlay);
transition: background var(--t-base), border-color var(--t-base);
}
.close-remember-toggle.on { background: var(--accent); border-color: var(--accent); }
.close-remember-thumb {
position: absolute; top: 1px; left: 1px;
width: 12px; height: 12px; border-radius: 50%;
background: var(--text-faint);
transition: transform var(--t-base), background var(--t-base);
}
.close-remember-toggle.on .close-remember-thumb { transform: translateX(12px); background: #fff; }
.close-remember-label { font-size: var(--text-xs); color: var(--text-faint); letter-spacing: var(--tracking-wide); transition: color var(--t-base); }
</style>