mirror of
https://github.com/elisiariocouto/leggen.git
synced 2025-12-13 12:32:18 +00:00
161 lines
4.8 KiB
TypeScript
161 lines
4.8 KiB
TypeScript
import { useEffect, useState } from "react";
|
|
import { X, Download, RotateCcw } from "lucide-react";
|
|
|
|
interface BeforeInstallPromptEvent extends Event {
|
|
prompt(): Promise<void>;
|
|
userChoice: Promise<{ outcome: "accepted" | "dismissed" }>;
|
|
}
|
|
|
|
interface PWAPromptProps {
|
|
onInstall?: () => void;
|
|
}
|
|
|
|
export function PWAInstallPrompt({ onInstall }: PWAPromptProps) {
|
|
const [deferredPrompt, setDeferredPrompt] =
|
|
useState<BeforeInstallPromptEvent | null>(null);
|
|
const [showPrompt, setShowPrompt] = useState(false);
|
|
|
|
useEffect(() => {
|
|
const handler = (e: Event) => {
|
|
// Prevent the mini-infobar from appearing on mobile
|
|
e.preventDefault();
|
|
setDeferredPrompt(e as BeforeInstallPromptEvent);
|
|
setShowPrompt(true);
|
|
};
|
|
|
|
window.addEventListener("beforeinstallprompt", handler);
|
|
|
|
return () => window.removeEventListener("beforeinstallprompt", handler);
|
|
}, []);
|
|
|
|
const handleInstall = async () => {
|
|
if (!deferredPrompt) return;
|
|
|
|
try {
|
|
await deferredPrompt.prompt();
|
|
const { outcome } = await deferredPrompt.userChoice;
|
|
|
|
if (outcome === "accepted") {
|
|
onInstall?.();
|
|
}
|
|
|
|
setDeferredPrompt(null);
|
|
setShowPrompt(false);
|
|
} catch (error) {
|
|
console.error("Error installing PWA:", error);
|
|
}
|
|
};
|
|
|
|
const handleDismiss = () => {
|
|
setShowPrompt(false);
|
|
setDeferredPrompt(null);
|
|
};
|
|
|
|
if (!showPrompt || !deferredPrompt) return null;
|
|
|
|
return (
|
|
<div className="fixed bottom-4 left-4 right-4 md:left-auto md:right-4 md:max-w-sm bg-white dark:bg-gray-800 border border-gray-200 dark:border-gray-700 rounded-lg shadow-lg p-4 z-50">
|
|
<div className="flex items-start gap-3">
|
|
<div className="flex-shrink-0">
|
|
<Download className="h-5 w-5 text-blue-600 dark:text-blue-400" />
|
|
</div>
|
|
<div className="flex-1 min-w-0">
|
|
<p className="text-sm font-medium text-gray-900 dark:text-gray-100">
|
|
Install Leggen
|
|
</p>
|
|
<p className="text-sm text-gray-500 dark:text-gray-400">
|
|
Add to your home screen for quick access
|
|
</p>
|
|
</div>
|
|
<button
|
|
onClick={handleDismiss}
|
|
className="flex-shrink-0 text-gray-400 hover:text-gray-500 dark:hover:text-gray-300"
|
|
>
|
|
<X className="h-4 w-4" />
|
|
</button>
|
|
</div>
|
|
<div className="mt-3 flex gap-2">
|
|
<button
|
|
onClick={handleInstall}
|
|
className="flex-1 bg-blue-600 text-white text-sm font-medium px-3 py-2 rounded-md hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-blue-500"
|
|
>
|
|
Install
|
|
</button>
|
|
<button
|
|
onClick={handleDismiss}
|
|
className="px-3 py-2 text-sm font-medium text-gray-700 dark:text-gray-300 hover:text-gray-900 dark:hover:text-gray-100"
|
|
>
|
|
Not now
|
|
</button>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
interface PWAUpdatePromptProps {
|
|
updateAvailable: boolean;
|
|
onUpdate: () => void;
|
|
}
|
|
|
|
export function PWAUpdatePrompt({
|
|
updateAvailable,
|
|
onUpdate,
|
|
}: PWAUpdatePromptProps) {
|
|
const [showPrompt, setShowPrompt] = useState(false);
|
|
|
|
useEffect(() => {
|
|
if (updateAvailable) {
|
|
setShowPrompt(true);
|
|
}
|
|
}, [updateAvailable]);
|
|
|
|
const handleUpdate = () => {
|
|
onUpdate();
|
|
setShowPrompt(false);
|
|
};
|
|
|
|
const handleDismiss = () => {
|
|
setShowPrompt(false);
|
|
};
|
|
|
|
if (!showPrompt || !updateAvailable) return null;
|
|
|
|
return (
|
|
<div className="fixed top-4 left-4 right-4 md:left-auto md:right-4 md:max-w-sm bg-white dark:bg-gray-800 border border-gray-200 dark:border-gray-700 rounded-lg shadow-lg p-4 z-50">
|
|
<div className="flex items-start gap-3">
|
|
<div className="flex-shrink-0">
|
|
<RotateCcw className="h-5 w-5 text-green-600 dark:text-green-400" />
|
|
</div>
|
|
<div className="flex-1 min-w-0">
|
|
<p className="text-sm font-medium text-gray-900 dark:text-gray-100">
|
|
Update Available
|
|
</p>
|
|
<p className="text-sm text-gray-500 dark:text-gray-400">
|
|
A new version of Leggen is ready to install
|
|
</p>
|
|
</div>
|
|
<button
|
|
onClick={handleDismiss}
|
|
className="flex-shrink-0 text-gray-400 hover:text-gray-500 dark:hover:text-gray-300"
|
|
>
|
|
<X className="h-4 w-4" />
|
|
</button>
|
|
</div>
|
|
<div className="mt-3 flex gap-2">
|
|
<button
|
|
onClick={handleUpdate}
|
|
className="flex-1 bg-green-600 text-white text-sm font-medium px-3 py-2 rounded-md hover:bg-green-700 focus:outline-none focus:ring-2 focus:ring-green-500"
|
|
>
|
|
Update Now
|
|
</button>
|
|
<button
|
|
onClick={handleDismiss}
|
|
className="px-3 py-2 text-sm font-medium text-gray-700 dark:text-gray-300 hover:text-gray-900 dark:hover:text-gray-100"
|
|
>
|
|
Later
|
|
</button>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|