Files
sapiens-web/client/src/pages/Landing.tsx
kimjaehyeon0101 5a6d98208d Update home and landing pages to show consistent UI regardless of login status
Refactor routing and page components to ensure the main content is displayed identically for both logged-in and logged-out users, removing conditional rendering based on authentication status in the main router and consolidating content display logic into a new `MainContent` component.

Replit-Commit-Author: Agent
Replit-Commit-Session-Id: 069d4324-6c40-4355-955e-c714a50de1ea
Replit-Commit-Checkpoint-Type: intermediate_checkpoint
Replit-Commit-Screenshot-Url: https://storage.googleapis.com/screenshot-production-us-central1/3df548ff-50ae-432f-9be4-25d34eccc983/069d4324-6c40-4355-955e-c714a50de1ea/InMLMqG
2025-09-29 16:40:42 +00:00

256 lines
8.8 KiB
TypeScript

import { useState } from "react";
import { Button } from "@/components/ui/button";
import { Input } from "@/components/ui/input";
import { Dialog, DialogContent, DialogHeader, DialogTitle } from "@/components/ui/dialog";
import { useToast } from "@/hooks/use-toast";
import { apiRequest, queryClient } from "@/lib/queryClient";
import { useMutation } from "@tanstack/react-query";
import { Search, Settings } from "lucide-react";
import MainContent from "@/components/MainContent";
export default function Landing() {
const [showLoginModal, setShowLoginModal] = useState(false);
const [showRequestModal, setShowRequestModal] = useState(false);
const [loginForm, setLoginForm] = useState({ username: "", password: "" });
const [requestForm, setRequestForm] = useState({
name: "",
category: "people",
description: ""
});
const { toast } = useToast();
const loginMutation = useMutation({
mutationFn: async (credentials: { username: string; password: string }) => {
return await apiRequest("POST", "/api/login", credentials);
},
onSuccess: () => {
toast({
title: "Success",
description: "Successfully logged in!",
});
setShowLoginModal(false);
setLoginForm({ username: "", password: "" });
// Refresh user data
queryClient.invalidateQueries({ queryKey: ["/api/auth/user"] });
// Redirect to home
window.location.href = "/";
},
onError: () => {
toast({
title: "Error",
description: "Invalid credentials. Please try again.",
variant: "destructive"
});
}
});
const requestMutation = useMutation({
mutationFn: async (data: typeof requestForm) => {
await apiRequest("POST", "/api/media-outlet-requests", data);
},
onSuccess: () => {
toast({
title: "Success",
description: "Your media outlet request has been submitted for review.",
});
setShowRequestModal(false);
setRequestForm({ name: "", category: "people", description: "" });
},
onError: () => {
toast({
title: "Error",
description: "Failed to submit request. Please try again.",
variant: "destructive"
});
}
});
const handleLogin = () => {
setShowLoginModal(true);
};
const handleLoginSubmit = (e: React.FormEvent) => {
e.preventDefault();
loginMutation.mutate(loginForm);
};
const handleRequestSubmit = (e: React.FormEvent) => {
e.preventDefault();
requestMutation.mutate(requestForm);
};
return (
<div className="min-h-screen bg-gray-50">
{/* Header */}
<header className="bg-white border-b border-gray-200 sticky top-0 z-50">
<div className="max-w-7xl mx-auto px-6 py-4">
<div className="flex items-center justify-between">
<div className="flex items-center space-x-6">
<img
src="/attached_assets/logo_black_1759162717640.png"
alt="SAPIENS"
className="h-6 w-auto"
data-testid="logo-sapiens"
/>
<span className="text-sm text-gray-600 font-medium">Nostra</span>
</div>
<div className="flex items-center space-x-4">
<div className="relative">
<Search className="absolute left-3 top-1/2 transform -translate-y-1/2 h-4 w-4 text-gray-400" />
<Input
type="text"
placeholder="Search the website"
className="w-80 pl-10 bg-gray-50 border-gray-200"
data-testid="input-search"
/>
</div>
<Button
variant="ghost"
size="sm"
onClick={handleLogin}
data-testid="button-login"
>
</Button>
<Button
variant="ghost"
size="sm"
data-testid="button-settings"
>
<Settings className="h-4 w-4" />
</Button>
</div>
</div>
</div>
</header>
{/* Main Content */}
<MainContent />
{/* Login Modal */}
<Dialog open={showLoginModal} onOpenChange={setShowLoginModal}>
<DialogContent className="sm:max-w-md" data-testid="modal-login">
<DialogHeader>
<DialogTitle>Login to SAPIENS</DialogTitle>
</DialogHeader>
<form onSubmit={handleLoginSubmit} className="space-y-4">
<div>
<label className="block text-sm font-medium mb-2">Username</label>
<Input
value={loginForm.username}
onChange={(e) => setLoginForm(prev => ({ ...prev, username: e.target.value }))}
placeholder="Enter username"
required
data-testid="input-username"
/>
</div>
<div>
<label className="block text-sm font-medium mb-2">Password</label>
<Input
type="password"
value={loginForm.password}
onChange={(e) => setLoginForm(prev => ({ ...prev, password: e.target.value }))}
placeholder="Enter password"
required
data-testid="input-password"
/>
</div>
<div className="text-xs text-muted-foreground">
Use: admin/1234 or superadmin/1234
</div>
<div className="flex space-x-2">
<Button
type="submit"
disabled={loginMutation.isPending}
className="flex-1"
data-testid="button-submit-login"
>
{loginMutation.isPending ? "Logging in..." : "Login"}
</Button>
<Button
type="button"
variant="outline"
onClick={() => setShowLoginModal(false)}
data-testid="button-cancel-login"
>
Cancel
</Button>
</div>
</form>
</DialogContent>
</Dialog>
{/* Request Modal */}
<Dialog open={showRequestModal} onOpenChange={setShowRequestModal}>
<DialogContent data-testid="modal-request">
<DialogHeader>
<DialogTitle>Request New Media Outlet</DialogTitle>
</DialogHeader>
<form onSubmit={handleRequestSubmit} className="space-y-4">
<div>
<label className="block text-sm font-medium mb-2">Name</label>
<Input
value={requestForm.name}
onChange={(e) => setRequestForm(prev => ({ ...prev, name: e.target.value }))}
placeholder="Enter outlet name"
required
data-testid="input-outlet-name"
/>
</div>
<div>
<label className="block text-sm font-medium mb-2">Category</label>
<select
value={requestForm.category}
onChange={(e) => setRequestForm(prev => ({ ...prev, category: e.target.value }))}
className="w-full px-3 py-2 border border-border rounded-lg focus:outline-none focus:ring-2 focus:ring-ring"
data-testid="select-category"
>
<option value="people">People</option>
<option value="topics">Topics</option>
<option value="companies">Companies</option>
</select>
</div>
<div>
<label className="block text-sm font-medium mb-2">Description</label>
<textarea
value={requestForm.description}
onChange={(e) => setRequestForm(prev => ({ ...prev, description: e.target.value }))}
placeholder="Describe why this outlet should be added"
className="w-full px-3 py-2 border border-border rounded-lg focus:outline-none focus:ring-2 focus:ring-ring"
rows={3}
data-testid="textarea-description"
/>
</div>
<div className="flex space-x-2">
<Button
type="submit"
disabled={requestMutation.isPending}
data-testid="button-submit-request"
>
{requestMutation.isPending ? "Submitting..." : "Submit Request"}
</Button>
<Button
type="button"
variant="outline"
onClick={() => setShowRequestModal(false)}
data-testid="button-cancel-request"
>
Cancel
</Button>
</div>
</form>
</DialogContent>
</Dialog>
</div>
);
}