feat: implement osu! installation validation

This commit is contained in:
2025-06-30 23:07:45 +02:00
parent c43fd4395d
commit 1834d8dfb3
12 changed files with 476 additions and 44 deletions

View File

@@ -1,8 +1,78 @@
<script lang="ts">
import { LoaderCircle } from "lucide-svelte";
import Logo from '$assets/logo.png';
import { animate } from 'animejs';
import { onMount } from 'svelte';
let ezppLogo: HTMLImageElement;
let spinnerCircle: SVGCircleElement;
//TODO: use this to check for updates upon launch
const doBPMAnimation = () => {
setInterval(async () => {
animate(ezppLogo, {
scale: 1.1,
duration: 900,
ease: (t: number) => Math.pow(2, -5 * t) * Math.sin((t - 0.075) * 20.94) + 1 - 0.0005 * t,
onComplete: () => {},
});
await new Promise((resolve) => setTimeout(resolve, 200));
animate(ezppLogo, {
scale: 1,
duration: 900,
ease: (t: number) => (t - 1) ** 7 + 1,
onComplete: () => {},
});
}, 450);
};
// Animate logo on mount: pop-in, then pulse
onMount(() => {
// Logo pop-in and pulse
animate(ezppLogo, {
opacity: [0, 1],
scale: [0, 1],
duration: 900,
ease: (t: number) => Math.pow(2, -5 * t) * Math.sin((t - 0.075) * 20.94) + 1 - 0.0005 * t,
onComplete: doBPMAnimation,
});
// Spinner animation (seamless, starts at 12 o'clock)
if (spinnerCircle) {
animate(spinnerCircle, {
strokeDashoffset: [0, -565],
duration: 1800,
easing: 'linear',
loop: true,
});
}
});
</script>
<div class="flex flex-col items-center justify-center mt-[50px] h-[calc(100vh-50px)] w-full">
<LoaderCircle class="animate-spin" size={80} />
<span></span>
</div>
<div class="relative w-80 h-80 flex items-center justify-center">
<svg
class="absolute top-0 left-0 w-full h-full animate-spin"
style="animation-duration: 5s;"
viewBox="0 0 208 208"
>
<circle
cx="104"
cy="104"
r="90"
fill="none"
stroke="#ff0098"
stroke-width="8"
stroke-linecap="round"
stroke-dasharray="180 385"
stroke-dashoffset="0"
bind:this={spinnerCircle}
/>
</svg>
<img
src={Logo}
alt="EZPPLauncher Logo"
class="w-52 h-52 mb-2 relative z-10"
bind:this={ezppLogo}
/>
</div>
</div>

28
src/pages/Login.svelte Normal file
View File

@@ -0,0 +1,28 @@
<script lang="ts">
import { ezppfarm } from '@/api/ezpp';
let username = $state('');
let password = $state('');
let errorMessage = $state('');
let isLoading = $state(false);
const performLogin = async () => {
isLoading = true;
errorMessage = '';
try {
const loginResult = await ezppfarm.login(username, password);
if (loginResult) {
// Handle successful login, e.g., redirect to the main app or store the token
console.log('Login successful:', loginResult);
} else {
errorMessage = 'Invalid username or password.';
}
} catch {
errorMessage = 'Login failed. Please check your credentials.';
} finally {
isLoading = false;
}
};
</script>

View File

@@ -1,25 +1,73 @@
<script lang="ts">
import Logo from '$assets/logo.png';
import * as Avatar from '@/components/ui/avatar';
import Badge from '@/components/ui/badge/badge.svelte';
import Button from '@/components/ui/button/button.svelte';
import * as Select from '@/components/ui/select';
import { beatmap_sets, online_friends, server_connection_fails, server_ping } from '@/global';
import { WebviewWindow } from '@tauri-apps/api/webviewWindow';
import { LoaderCircle, Logs, Music2, Play, Users, Wifi, Gamepad2, WifiOff } from 'lucide-svelte';
import { Circle } from 'radix-icons-svelte';
import NumberFlow from '@number-flow/svelte';
import * as AlertDialog from '@/components/ui/alert-dialog';
import Progress from '@/components/ui/progress/progress.svelte';
import { numberHumanReadable } from '@/utils';
import Input from '@/components/ui/input/input.svelte';
import { animate } from 'animejs';
import { onMount } from 'svelte';
import { fade } from 'svelte/transition';
import { invoke } from '@tauri-apps/api/core';
import { Check, CheckCircle, CircleOff } from 'lucide-svelte';
import { open } from '@tauri-apps/plugin-dialog';
let selectedStep = $state(1);
const steps = [
'Welcome',
'Login/Register',
'Locate your osu! Installation',
'Select your default mode',
];
const steps = ['Welcome', 'Locate your osu! Installation', 'Appearance Settings'];
let osuInstallationPath = $state('');
let manualSelect = $state(false);
let manualSelectValid = $state(false);
let autoDetectedOsuPath = $state(false);
let ezppLogo: HTMLImageElement | undefined = $state(undefined);
const logo_mouseenter = () => {
if (ezppLogo) {
animate(ezppLogo, {
duration: 700,
scale: 1.2,
ease: (t: number) => Math.pow(2, -5 * t) * Math.sin((t - 0.075) * 20.94) + 1 - 0.0005 * t,
});
}
};
const logo_mouseleave = () => {
if (ezppLogo) {
animate(ezppLogo, {
duration: 700,
scale: 1,
ease: (t: number) => (t - 1) ** 7 + 1,
});
}
};
const browse_osu_installation = async () => {
const selectedPath = await open({
directory: true,
multiple: false,
title: 'Select osu! Installation Folder',
});
if (typeof selectedPath === 'string') {
const validFolder: boolean = await invoke('valid_osu_folder', { folder: selectedPath });
manualSelect = true;
if (!validFolder) {
manualSelectValid = false;
osuInstallationPath = '';
return;
}
osuInstallationPath = selectedPath;
autoDetectedOsuPath = false;
manualSelectValid = true;
}
};
onMount(async () => {
const osuInstallPath: string | null = await invoke('find_osu_installation');
console.log('osu install path: ' + osuInstallPath);
if (osuInstallPath) {
osuInstallationPath = osuInstallPath;
autoDetectedOsuPath = true;
}
});
</script>
<div class="grid grid-cols-[0.41fr_1fr] mt-[50px] h-[calc(100vh-50px)]">
@@ -28,27 +76,109 @@
<div
class="flex flex-row items-center gap-2 border {selectedStep === i + 1
? 'border-primary-800/30 bg-primary-900/30'
: 'border-theme-800 bg-theme-900'} rounded-lg p-2 transition-all"
: selectedStep > i
? 'border-green-800/30 bg-green-900/30'
: 'border-theme-800 bg-theme-900'} rounded-lg p-2 transition-all"
>
<div
class="flex flex-col items-center justify-center h-8 w-8 border-[2px] border-theme-600 rounded-full"
class="flex flex-col items-center justify-center h-8 w-8 border-[2px] {selectedStep > i + 1 ? "border-green-600" : "border-theme-600"} rounded-full"
>
<span class="text-lg font-semibold text-theme-100">{i + 1}</span>
{#if selectedStep > i + 1}
<Check class="mt-0.5 text-green-400" />
{:else}
<span class="text-lg font-semibold text-theme-100">{i + 1}</span>
{/if}
</div>
<span
class="{selectedStep === i + 1 ? 'text-white' : 'text-muted-foreground'} transition-all"
>{step}</span
class="{selectedStep === i + 1
? 'text-white'
: selectedStep > i
? 'text-green-500'
: "'text-muted-foreground'"} transition-all text-sm font-bold">{step}</span
>
</div>
{/each}
</div>
<div class="flex flex-col gap-6 w-full h-full bg-theme-900/40 p-6">
<div class="my-auto flex flex-col items-center justify-center">
{#if selectedStep === 1}
<h1 class="text-3xl">Welcome to EZPPLauncher!</h1>
{/if}
</div>
{#if selectedStep === 1}
<div
class="my-auto h-full w-full bg-theme-800/15 rounded-lg border border-900/60 p-6 flex flex-col items-center justify-center"
in:fade={{ duration: 100 }}
>
<img
src={Logo}
alt="EZPPLauncher Logo"
class="w-52 h-52 mb-2"
bind:this={ezppLogo}
onmouseenter={logo_mouseenter}
onmouseleave={logo_mouseleave}
/>
<h1 class="text-3xl font-semibold">Welcome to EZPPLauncher!</h1>
<p class="text-muted-foreground mt-2">
This setup wizard will guide you through the initial setup of EZPPLauncher.
</p>
<div class="bg-red-800/20 border border-red-900/20 text-red-500 p-4 rounded-lg mt-4">
Please make sure you have osu! installed on your system before proceeding.
</div>
</div>
{:else if selectedStep === 2}
<div
class="my-auto h-full w-full bg-theme-800/15 rounded-lg border border-900/60 p-6 flex flex-col items-center justify-center"
in:fade={{ duration: 100 }}
>
<h1 class="text-3xl font-semibold">Locate your osu! Installation</h1>
<p class="text-muted-foreground mt-2">
Please select the folder where your osu! installation is located.
</p>
<div class="flex flex-row w-full">
<Input
class="mt-4 w-full bg-theme-950 border-theme-800 border-r-0 rounded-r-none"
type="text"
placeholder="Path to osu! installation"
value={osuInstallationPath}
/>
<Button
class="mt-4 bg-theme-950 border-theme-800 rounded-l-none"
variant="outline"
onclick={browse_osu_installation}>Browse</Button
>
</div>
{#if !manualSelect}
{#if autoDetectedOsuPath}
<div
class="flex flex-row gap-3 bg-green-800/20 border border-green-900/20 text-green-500 p-4 rounded-lg mt-4"
>
<CheckCircle />
<span>Auto-detected osu! installation path! Please check if its correct!</span>
</div>
{:else}
<div
class="flex flex-row gap-3 bg-red-800/20 border border-red-900/20 text-red-500 p-4 rounded-lg mt-4"
>
<CircleOff />
<span>Could not auto-detect osu! installation path. Please select it manually.</span>
</div>
{/if}
{:else if manualSelectValid}
<div
class="flex flex-row gap-3 bg-green-800/20 border border-green-900/20 text-green-500 p-4 rounded-lg mt-4"
>
<CheckCircle />
<span>Selected osu! installation path is valid!</span>
</div>
{:else}
<div
class="flex flex-row gap-3 bg-red-800/20 border border-red-900/20 text-red-500 p-4 rounded-lg mt-4"
>
<CircleOff />
<span
>Selected osu! installation path is invalid! Please select a valid osu! installation.</span
>
</div>
{/if}
</div>
{/if}
<div class="mt-auto flex flex-row items-center justify-between">
<Button
class="bg-theme-950 hover:bg-theme-800"
@@ -58,7 +188,8 @@
>
<Button
onclick={() => (selectedStep = Math.min(selectedStep + 1, steps.length))}
disabled={selectedStep >= steps.length}>Next</Button
disabled={selectedStep >= steps.length ||
(selectedStep === 2 && osuInstallationPath.length <= 0)}>Next</Button
>
</div>
</div>