TODOs: - [x] Do restart when checkout / restore if there is a DB - [x] List all branches (branch id, name, date) - [x] Allow checking out versions with no DB - [x] safeguard to never delete main branches - [x] create app hook for neon template - [x] weird UX with connector on configure panel - [x] tiny neon logo in connector - [x] deploy to vercel - [x] build forgot password page - [x] what about email setup - [x] lots of imgix errors - [x] edit file - db snapshot - [x] DYAD_DISABLE_DB_PUSH - [ ] update portal doc - [x] switch preview branch to be read-only endpoint - [x] disable supabase sys prompt if neon is enabled - [ ] https://payloadcms.com/docs/upload/storage-adapters - [x] need to use main branch... Phase 2? - [x] generate DB migrations
90 lines
3.0 KiB
TypeScript
90 lines
3.0 KiB
TypeScript
import React from "react";
|
|
import { toast } from "sonner";
|
|
import { X, AlertTriangle } from "lucide-react";
|
|
import { Button } from "./ui/button";
|
|
|
|
interface InputRequestToastProps {
|
|
message: string;
|
|
toastId: string | number;
|
|
onResponse: (response: "y" | "n") => void;
|
|
}
|
|
|
|
export function InputRequestToast({
|
|
message,
|
|
toastId,
|
|
onResponse,
|
|
}: InputRequestToastProps) {
|
|
const handleClose = () => {
|
|
toast.dismiss(toastId);
|
|
};
|
|
|
|
const handleResponse = (response: "y" | "n") => {
|
|
onResponse(response);
|
|
toast.dismiss(toastId);
|
|
};
|
|
|
|
// Clean up the message by removing excessive newlines and whitespace
|
|
const cleanMessage = message
|
|
.split("\n")
|
|
.map((line) => line.trim())
|
|
.filter((line) => line.length > 0)
|
|
.join("\n");
|
|
|
|
return (
|
|
<div className="relative bg-amber-50/95 dark:bg-slate-800/95 backdrop-blur-sm border border-amber-200 dark:border-slate-600 rounded-xl shadow-lg min-w-[400px] max-w-[500px] overflow-hidden">
|
|
{/* Content */}
|
|
<div className="p-5">
|
|
<div className="flex items-start">
|
|
<div className="flex-1">
|
|
<div className="flex items-center mb-4">
|
|
<div className="flex-shrink-0">
|
|
<div className="w-6 h-6 bg-gradient-to-br from-amber-500 to-amber-600 dark:from-amber-400 dark:to-amber-500 rounded-full flex items-center justify-center shadow-sm">
|
|
<AlertTriangle className="w-3.5 h-3.5 text-white" />
|
|
</div>
|
|
</div>
|
|
<h3 className="ml-3 text-base font-semibold text-amber-900 dark:text-amber-100">
|
|
Input Required
|
|
</h3>
|
|
|
|
{/* Close button */}
|
|
<button
|
|
onClick={handleClose}
|
|
className="ml-auto flex-shrink-0 p-1.5 text-amber-500 dark:text-slate-400 hover:text-amber-700 dark:hover:text-slate-200 transition-colors duration-200 rounded-md hover:bg-amber-100/50 dark:hover:bg-slate-700/50"
|
|
aria-label="Close"
|
|
>
|
|
<X className="w-4 h-4" />
|
|
</button>
|
|
</div>
|
|
|
|
{/* Message */}
|
|
<div className="mb-5">
|
|
<p className="text-sm text-amber-900 dark:text-slate-200 whitespace-pre-wrap leading-relaxed">
|
|
{cleanMessage}
|
|
</p>
|
|
</div>
|
|
|
|
{/* Action buttons */}
|
|
<div className="flex items-center gap-3">
|
|
<Button
|
|
onClick={() => handleResponse("y")}
|
|
size="sm"
|
|
className="bg-primary text-white dark:bg-primary dark:text-black px-6"
|
|
>
|
|
Yes
|
|
</Button>
|
|
<Button
|
|
onClick={() => handleResponse("n")}
|
|
size="sm"
|
|
variant="outline"
|
|
className="border-amber-300 dark:border-slate-500 text-amber-800 dark:text-slate-300 hover:bg-amber-100 dark:hover:bg-slate-700 px-6"
|
|
>
|
|
No
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|