mirror of
https://github.com/pacnpal/thrilltrack-explorer.git
synced 2025-12-23 17:31:14 -05:00
feat: Add collapsible filter sidebar
This commit is contained in:
@@ -6,9 +6,10 @@ import { Button } from '@/components/ui/button';
|
|||||||
import { Dialog, DialogContent } from '@/components/ui/dialog';
|
import { Dialog, DialogContent } from '@/components/ui/dialog';
|
||||||
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from '@/components/ui/select';
|
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from '@/components/ui/select';
|
||||||
import { Collapsible, CollapsibleContent } from '@/components/ui/collapsible';
|
import { Collapsible, CollapsibleContent } from '@/components/ui/collapsible';
|
||||||
import { Card, CardContent } from '@/components/ui/card';
|
import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card';
|
||||||
import { Badge } from '@/components/ui/badge';
|
import { Badge } from '@/components/ui/badge';
|
||||||
import { Search, SlidersHorizontal, Ruler, Plus, ChevronDown, Filter } from 'lucide-react';
|
import { Search, SlidersHorizontal, Ruler, Plus, ChevronDown, Filter, PanelLeftClose, PanelLeftOpen } from 'lucide-react';
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
import { DesignerFilters, DesignerFilterState, defaultDesignerFilters } from '@/components/designers/DesignerFilters';
|
import { DesignerFilters, DesignerFilterState, defaultDesignerFilters } from '@/components/designers/DesignerFilters';
|
||||||
import { Company } from '@/types/database';
|
import { Company } from '@/types/database';
|
||||||
import { supabase } from '@/integrations/supabase/client';
|
import { supabase } from '@/integrations/supabase/client';
|
||||||
@@ -33,6 +34,10 @@ export default function Designers() {
|
|||||||
const [filters, setFilters] = useState<DesignerFilterState>(defaultDesignerFilters);
|
const [filters, setFilters] = useState<DesignerFilterState>(defaultDesignerFilters);
|
||||||
const [showFilters, setShowFilters] = useState(false);
|
const [showFilters, setShowFilters] = useState(false);
|
||||||
const [isCreateModalOpen, setIsCreateModalOpen] = useState(false);
|
const [isCreateModalOpen, setIsCreateModalOpen] = useState(false);
|
||||||
|
const [sidebarCollapsed, setSidebarCollapsed] = useState(() => {
|
||||||
|
const saved = localStorage.getItem('designers-sidebar-collapsed');
|
||||||
|
return saved ? JSON.parse(saved) : false;
|
||||||
|
});
|
||||||
|
|
||||||
const handleCreateSubmit = async (data: any) => {
|
const handleCreateSubmit = async (data: any) => {
|
||||||
try {
|
try {
|
||||||
@@ -52,6 +57,10 @@ export default function Designers() {
|
|||||||
fetchCompanies();
|
fetchCompanies();
|
||||||
}, [sortBy]);
|
}, [sortBy]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
localStorage.setItem('designers-sidebar-collapsed', JSON.stringify(sidebarCollapsed));
|
||||||
|
}, [sidebarCollapsed]);
|
||||||
|
|
||||||
const fetchCompanies = async () => {
|
const fetchCompanies = async () => {
|
||||||
try {
|
try {
|
||||||
let query = supabase
|
let query = supabase
|
||||||
@@ -173,6 +182,15 @@ export default function Designers() {
|
|||||||
<span className="hidden sm:inline">Filters</span>
|
<span className="hidden sm:inline">Filters</span>
|
||||||
<ChevronDown className={`w-4 h-4 transition-transform ${showFilters ? 'rotate-180' : ''}`} />
|
<ChevronDown className={`w-4 h-4 transition-transform ${showFilters ? 'rotate-180' : ''}`} />
|
||||||
</Button>
|
</Button>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
onClick={() => setSidebarCollapsed(!sidebarCollapsed)}
|
||||||
|
className="gap-2 hidden lg:flex"
|
||||||
|
title={sidebarCollapsed ? "Show filters" : "Hide filters"}
|
||||||
|
>
|
||||||
|
<PanelLeftClose className={`w-4 h-4 transition-transform ${sidebarCollapsed ? 'rotate-180' : ''}`} />
|
||||||
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -193,13 +211,44 @@ export default function Designers() {
|
|||||||
{/* Main Content Area with Sidebar */}
|
{/* Main Content Area with Sidebar */}
|
||||||
<div className="flex flex-col lg:flex-row gap-6">
|
<div className="flex flex-col lg:flex-row gap-6">
|
||||||
{/* Desktop Filter Sidebar */}
|
{/* Desktop Filter Sidebar */}
|
||||||
<aside className="hidden lg:block lg:w-[340px] xl:w-[380px] 2xl:w-[420px] flex-shrink-0">
|
<aside
|
||||||
|
className={cn(
|
||||||
|
"hidden lg:block flex-shrink-0 transition-all duration-300",
|
||||||
|
sidebarCollapsed
|
||||||
|
? "lg:w-[60px]"
|
||||||
|
: "lg:w-[340px] xl:w-[380px] 2xl:w-[420px]"
|
||||||
|
)}
|
||||||
|
>
|
||||||
<div className="sticky top-24">
|
<div className="sticky top-24">
|
||||||
<Card>
|
{sidebarCollapsed ? (
|
||||||
<CardContent className="pt-6">
|
<Card className="p-2">
|
||||||
<DesignerFilters filters={filters} onFiltersChange={setFilters} designers={companies} />
|
<Button
|
||||||
</CardContent>
|
variant="ghost"
|
||||||
</Card>
|
className="w-full"
|
||||||
|
onClick={() => setSidebarCollapsed(false)}
|
||||||
|
title="Show filters"
|
||||||
|
>
|
||||||
|
<PanelLeftOpen className="w-5 h-5" />
|
||||||
|
</Button>
|
||||||
|
</Card>
|
||||||
|
) : (
|
||||||
|
<Card>
|
||||||
|
<CardHeader className="flex flex-row items-center justify-between pb-2">
|
||||||
|
<CardTitle className="text-base">Filters</CardTitle>
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => setSidebarCollapsed(true)}
|
||||||
|
title="Hide filters"
|
||||||
|
>
|
||||||
|
<PanelLeftClose className="w-4 h-4" />
|
||||||
|
</Button>
|
||||||
|
</CardHeader>
|
||||||
|
<CardContent>
|
||||||
|
<DesignerFilters filters={filters} onFiltersChange={setFilters} designers={companies} />
|
||||||
|
</CardContent>
|
||||||
|
</Card>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</aside>
|
</aside>
|
||||||
|
|
||||||
|
|||||||
@@ -6,9 +6,10 @@ import { Button } from '@/components/ui/button';
|
|||||||
import { Dialog, DialogContent } from '@/components/ui/dialog';
|
import { Dialog, DialogContent } from '@/components/ui/dialog';
|
||||||
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from '@/components/ui/select';
|
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from '@/components/ui/select';
|
||||||
import { Collapsible, CollapsibleContent } from '@/components/ui/collapsible';
|
import { Collapsible, CollapsibleContent } from '@/components/ui/collapsible';
|
||||||
import { Card, CardContent } from '@/components/ui/card';
|
import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card';
|
||||||
import { Badge } from '@/components/ui/badge';
|
import { Badge } from '@/components/ui/badge';
|
||||||
import { Search, SlidersHorizontal, Factory, Plus, ChevronDown, Filter } from 'lucide-react';
|
import { Search, SlidersHorizontal, Factory, Plus, ChevronDown, Filter, PanelLeftClose, PanelLeftOpen } from 'lucide-react';
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
import { ManufacturerFilters, ManufacturerFilterState, defaultManufacturerFilters } from '@/components/manufacturers/ManufacturerFilters';
|
import { ManufacturerFilters, ManufacturerFilterState, defaultManufacturerFilters } from '@/components/manufacturers/ManufacturerFilters';
|
||||||
import { Company } from '@/types/database';
|
import { Company } from '@/types/database';
|
||||||
import { supabase } from '@/integrations/supabase/client';
|
import { supabase } from '@/integrations/supabase/client';
|
||||||
@@ -33,12 +34,20 @@ export default function Manufacturers() {
|
|||||||
const [filters, setFilters] = useState<ManufacturerFilterState>(defaultManufacturerFilters);
|
const [filters, setFilters] = useState<ManufacturerFilterState>(defaultManufacturerFilters);
|
||||||
const [showFilters, setShowFilters] = useState(false);
|
const [showFilters, setShowFilters] = useState(false);
|
||||||
const [isCreateModalOpen, setIsCreateModalOpen] = useState(false);
|
const [isCreateModalOpen, setIsCreateModalOpen] = useState(false);
|
||||||
|
const [sidebarCollapsed, setSidebarCollapsed] = useState(() => {
|
||||||
|
const saved = localStorage.getItem('manufacturers-sidebar-collapsed');
|
||||||
|
return saved ? JSON.parse(saved) : false;
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
fetchCompanies();
|
fetchCompanies();
|
||||||
}, [sortBy]);
|
}, [sortBy]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
localStorage.setItem('manufacturers-sidebar-collapsed', JSON.stringify(sidebarCollapsed));
|
||||||
|
}, [sidebarCollapsed]);
|
||||||
|
|
||||||
const fetchCompanies = async () => {
|
const fetchCompanies = async () => {
|
||||||
try {
|
try {
|
||||||
let query = supabase
|
let query = supabase
|
||||||
@@ -186,6 +195,15 @@ export default function Manufacturers() {
|
|||||||
<span className="hidden sm:inline">Filters</span>
|
<span className="hidden sm:inline">Filters</span>
|
||||||
<ChevronDown className={`w-4 h-4 transition-transform ${showFilters ? 'rotate-180' : ''}`} />
|
<ChevronDown className={`w-4 h-4 transition-transform ${showFilters ? 'rotate-180' : ''}`} />
|
||||||
</Button>
|
</Button>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
onClick={() => setSidebarCollapsed(!sidebarCollapsed)}
|
||||||
|
className="gap-2 hidden lg:flex"
|
||||||
|
title={sidebarCollapsed ? "Show filters" : "Hide filters"}
|
||||||
|
>
|
||||||
|
<PanelLeftClose className={`w-4 h-4 transition-transform ${sidebarCollapsed ? 'rotate-180' : ''}`} />
|
||||||
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -206,13 +224,44 @@ export default function Manufacturers() {
|
|||||||
{/* Main Content Area with Sidebar */}
|
{/* Main Content Area with Sidebar */}
|
||||||
<div className="flex flex-col lg:flex-row gap-6">
|
<div className="flex flex-col lg:flex-row gap-6">
|
||||||
{/* Desktop Filter Sidebar */}
|
{/* Desktop Filter Sidebar */}
|
||||||
<aside className="hidden lg:block lg:w-[340px] xl:w-[380px] 2xl:w-[420px] flex-shrink-0">
|
<aside
|
||||||
|
className={cn(
|
||||||
|
"hidden lg:block flex-shrink-0 transition-all duration-300",
|
||||||
|
sidebarCollapsed
|
||||||
|
? "lg:w-[60px]"
|
||||||
|
: "lg:w-[340px] xl:w-[380px] 2xl:w-[420px]"
|
||||||
|
)}
|
||||||
|
>
|
||||||
<div className="sticky top-24">
|
<div className="sticky top-24">
|
||||||
<Card>
|
{sidebarCollapsed ? (
|
||||||
<CardContent className="pt-6">
|
<Card className="p-2">
|
||||||
<ManufacturerFilters filters={filters} onFiltersChange={setFilters} manufacturers={companies} />
|
<Button
|
||||||
</CardContent>
|
variant="ghost"
|
||||||
</Card>
|
className="w-full"
|
||||||
|
onClick={() => setSidebarCollapsed(false)}
|
||||||
|
title="Show filters"
|
||||||
|
>
|
||||||
|
<PanelLeftOpen className="w-5 h-5" />
|
||||||
|
</Button>
|
||||||
|
</Card>
|
||||||
|
) : (
|
||||||
|
<Card>
|
||||||
|
<CardHeader className="flex flex-row items-center justify-between pb-2">
|
||||||
|
<CardTitle className="text-base">Filters</CardTitle>
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => setSidebarCollapsed(true)}
|
||||||
|
title="Hide filters"
|
||||||
|
>
|
||||||
|
<PanelLeftClose className="w-4 h-4" />
|
||||||
|
</Button>
|
||||||
|
</CardHeader>
|
||||||
|
<CardContent>
|
||||||
|
<ManufacturerFilters filters={filters} onFiltersChange={setFilters} manufacturers={companies} />
|
||||||
|
</CardContent>
|
||||||
|
</Card>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</aside>
|
</aside>
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import React, { useState } from 'react';
|
import React, { useState, useEffect } from 'react';
|
||||||
import { useQuery } from '@tanstack/react-query';
|
import { useQuery } from '@tanstack/react-query';
|
||||||
import { useNavigate } from 'react-router-dom';
|
import { useNavigate } from 'react-router-dom';
|
||||||
import { Header } from '@/components/layout/Header';
|
import { Header } from '@/components/layout/Header';
|
||||||
@@ -8,8 +8,9 @@ import { Badge } from '@/components/ui/badge';
|
|||||||
import { Dialog, DialogContent } from '@/components/ui/dialog';
|
import { Dialog, DialogContent } from '@/components/ui/dialog';
|
||||||
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from '@/components/ui/select';
|
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from '@/components/ui/select';
|
||||||
import { Collapsible, CollapsibleContent } from '@/components/ui/collapsible';
|
import { Collapsible, CollapsibleContent } from '@/components/ui/collapsible';
|
||||||
import { Card, CardContent } from '@/components/ui/card';
|
import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card';
|
||||||
import { Search, Filter, Building, Plus, ChevronDown } from 'lucide-react';
|
import { Search, Filter, Building, Plus, ChevronDown, PanelLeftClose, PanelLeftOpen } from 'lucide-react';
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
import { supabase } from '@/integrations/supabase/client';
|
import { supabase } from '@/integrations/supabase/client';
|
||||||
import OperatorCard from '@/components/operators/OperatorCard';
|
import OperatorCard from '@/components/operators/OperatorCard';
|
||||||
import { OperatorForm } from '@/components/admin/OperatorForm';
|
import { OperatorForm } from '@/components/admin/OperatorForm';
|
||||||
@@ -32,6 +33,14 @@ const Operators = () => {
|
|||||||
const [filters, setFilters] = useState<OperatorFilterState>(defaultOperatorFilters);
|
const [filters, setFilters] = useState<OperatorFilterState>(defaultOperatorFilters);
|
||||||
const [showFilters, setShowFilters] = useState(false);
|
const [showFilters, setShowFilters] = useState(false);
|
||||||
const [isCreateModalOpen, setIsCreateModalOpen] = useState(false);
|
const [isCreateModalOpen, setIsCreateModalOpen] = useState(false);
|
||||||
|
const [sidebarCollapsed, setSidebarCollapsed] = useState(() => {
|
||||||
|
const saved = localStorage.getItem('operators-sidebar-collapsed');
|
||||||
|
return saved ? JSON.parse(saved) : false;
|
||||||
|
});
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
localStorage.setItem('operators-sidebar-collapsed', JSON.stringify(sidebarCollapsed));
|
||||||
|
}, [sidebarCollapsed]);
|
||||||
|
|
||||||
const { data: operators, isLoading } = useQuery({
|
const { data: operators, isLoading } = useQuery({
|
||||||
queryKey: ['operators'],
|
queryKey: ['operators'],
|
||||||
@@ -244,6 +253,15 @@ const Operators = () => {
|
|||||||
<span className="hidden sm:inline">Filters</span>
|
<span className="hidden sm:inline">Filters</span>
|
||||||
<ChevronDown className={`w-4 h-4 transition-transform ${showFilters ? 'rotate-180' : ''}`} />
|
<ChevronDown className={`w-4 h-4 transition-transform ${showFilters ? 'rotate-180' : ''}`} />
|
||||||
</Button>
|
</Button>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
onClick={() => setSidebarCollapsed(!sidebarCollapsed)}
|
||||||
|
className="gap-2 hidden lg:flex"
|
||||||
|
title={sidebarCollapsed ? "Show filters" : "Hide filters"}
|
||||||
|
>
|
||||||
|
<PanelLeftClose className={`w-4 h-4 transition-transform ${sidebarCollapsed ? 'rotate-180' : ''}`} />
|
||||||
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -264,13 +282,44 @@ const Operators = () => {
|
|||||||
{/* Main Content Area with Sidebar */}
|
{/* Main Content Area with Sidebar */}
|
||||||
<div className="flex flex-col lg:flex-row gap-6">
|
<div className="flex flex-col lg:flex-row gap-6">
|
||||||
{/* Desktop Filter Sidebar */}
|
{/* Desktop Filter Sidebar */}
|
||||||
<aside className="hidden lg:block lg:w-[340px] xl:w-[380px] 2xl:w-[420px] flex-shrink-0">
|
<aside
|
||||||
|
className={cn(
|
||||||
|
"hidden lg:block flex-shrink-0 transition-all duration-300",
|
||||||
|
sidebarCollapsed
|
||||||
|
? "lg:w-[60px]"
|
||||||
|
: "lg:w-[340px] xl:w-[380px] 2xl:w-[420px]"
|
||||||
|
)}
|
||||||
|
>
|
||||||
<div className="sticky top-24">
|
<div className="sticky top-24">
|
||||||
<Card>
|
{sidebarCollapsed ? (
|
||||||
<CardContent className="pt-6">
|
<Card className="p-2">
|
||||||
<OperatorFilters filters={filters} onFiltersChange={setFilters} operators={operators || []} />
|
<Button
|
||||||
</CardContent>
|
variant="ghost"
|
||||||
</Card>
|
className="w-full"
|
||||||
|
onClick={() => setSidebarCollapsed(false)}
|
||||||
|
title="Show filters"
|
||||||
|
>
|
||||||
|
<PanelLeftOpen className="w-5 h-5" />
|
||||||
|
</Button>
|
||||||
|
</Card>
|
||||||
|
) : (
|
||||||
|
<Card>
|
||||||
|
<CardHeader className="flex flex-row items-center justify-between pb-2">
|
||||||
|
<CardTitle className="text-base">Filters</CardTitle>
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => setSidebarCollapsed(true)}
|
||||||
|
title="Hide filters"
|
||||||
|
>
|
||||||
|
<PanelLeftClose className="w-4 h-4" />
|
||||||
|
</Button>
|
||||||
|
</CardHeader>
|
||||||
|
<CardContent>
|
||||||
|
<OperatorFilters filters={filters} onFiltersChange={setFilters} operators={operators || []} />
|
||||||
|
</CardContent>
|
||||||
|
</Card>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</aside>
|
</aside>
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
import { useState, useEffect, useMemo } from 'react';
|
import { useState, useEffect, useMemo } from 'react';
|
||||||
import { Header } from '@/components/layout/Header';
|
import { Header } from '@/components/layout/Header';
|
||||||
import { Button } from '@/components/ui/button';
|
import { Button } from '@/components/ui/button';
|
||||||
import { Card, CardContent } from '@/components/ui/card';
|
import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card';
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
import { Collapsible, CollapsibleContent } from '@/components/ui/collapsible';
|
import { Collapsible, CollapsibleContent } from '@/components/ui/collapsible';
|
||||||
import { Badge } from '@/components/ui/badge';
|
import { Badge } from '@/components/ui/badge';
|
||||||
import { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs';
|
import { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs';
|
||||||
@@ -17,7 +18,9 @@ import {
|
|||||||
Sliders,
|
Sliders,
|
||||||
X,
|
X,
|
||||||
FerrisWheel,
|
FerrisWheel,
|
||||||
Plus
|
Plus,
|
||||||
|
PanelLeftClose,
|
||||||
|
PanelLeftOpen
|
||||||
} from 'lucide-react';
|
} from 'lucide-react';
|
||||||
import { Park } from '@/types/database';
|
import { Park } from '@/types/database';
|
||||||
import { supabase } from '@/integrations/supabase/client';
|
import { supabase } from '@/integrations/supabase/client';
|
||||||
@@ -94,6 +97,10 @@ export default function Parks() {
|
|||||||
const [viewMode, setViewMode] = useState<'grid' | 'list'>('grid');
|
const [viewMode, setViewMode] = useState<'grid' | 'list'>('grid');
|
||||||
const [showFilters, setShowFilters] = useState(false);
|
const [showFilters, setShowFilters] = useState(false);
|
||||||
const [isAddParkModalOpen, setIsAddParkModalOpen] = useState(false);
|
const [isAddParkModalOpen, setIsAddParkModalOpen] = useState(false);
|
||||||
|
const [sidebarCollapsed, setSidebarCollapsed] = useState(() => {
|
||||||
|
const saved = localStorage.getItem('parks-sidebar-collapsed');
|
||||||
|
return saved ? JSON.parse(saved) : false;
|
||||||
|
});
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
const { toast } = useToast();
|
const { toast } = useToast();
|
||||||
const { user } = useAuth();
|
const { user } = useAuth();
|
||||||
@@ -104,6 +111,10 @@ export default function Parks() {
|
|||||||
fetchParks();
|
fetchParks();
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
localStorage.setItem('parks-sidebar-collapsed', JSON.stringify(sidebarCollapsed));
|
||||||
|
}, [sidebarCollapsed]);
|
||||||
|
|
||||||
const fetchParks = async () => {
|
const fetchParks = async () => {
|
||||||
try {
|
try {
|
||||||
setLoading(true);
|
setLoading(true);
|
||||||
@@ -423,6 +434,20 @@ export default function Parks() {
|
|||||||
)}
|
)}
|
||||||
</Button>
|
</Button>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
onClick={() => setSidebarCollapsed(!sidebarCollapsed)}
|
||||||
|
className="shrink-0 gap-2 hidden lg:flex"
|
||||||
|
title={sidebarCollapsed ? "Show filters" : "Hide filters"}
|
||||||
|
>
|
||||||
|
<PanelLeftClose className={`w-4 h-4 transition-transform ${sidebarCollapsed ? 'rotate-180' : ''}`} />
|
||||||
|
{!sidebarCollapsed && activeFilterCount > 0 && (
|
||||||
|
<Badge variant="secondary" className="ml-1">
|
||||||
|
{activeFilterCount}
|
||||||
|
</Badge>
|
||||||
|
)}
|
||||||
|
</Button>
|
||||||
|
|
||||||
<Tabs value={viewMode} onValueChange={(v) => setViewMode(v as 'grid' | 'list')} className="hidden md:inline-flex">
|
<Tabs value={viewMode} onValueChange={(v) => setViewMode(v as 'grid' | 'list')} className="hidden md:inline-flex">
|
||||||
<TabsList>
|
<TabsList>
|
||||||
<TabsTrigger value="grid">
|
<TabsTrigger value="grid">
|
||||||
@@ -457,17 +482,53 @@ export default function Parks() {
|
|||||||
{/* Main Content Area with Sidebar */}
|
{/* Main Content Area with Sidebar */}
|
||||||
<div className="flex flex-col lg:flex-row gap-6">
|
<div className="flex flex-col lg:flex-row gap-6">
|
||||||
{/* Desktop Filter Sidebar */}
|
{/* Desktop Filter Sidebar */}
|
||||||
<aside className="hidden lg:block lg:w-[340px] xl:w-[380px] 2xl:w-[420px] flex-shrink-0">
|
<aside
|
||||||
|
className={cn(
|
||||||
|
"hidden lg:block flex-shrink-0 transition-all duration-300",
|
||||||
|
sidebarCollapsed
|
||||||
|
? "lg:w-[60px]"
|
||||||
|
: "lg:w-[340px] xl:w-[380px] 2xl:w-[420px]"
|
||||||
|
)}
|
||||||
|
>
|
||||||
<div className="sticky top-24">
|
<div className="sticky top-24">
|
||||||
<Card>
|
{sidebarCollapsed ? (
|
||||||
<CardContent className="pt-6">
|
<Card className="p-2">
|
||||||
<ParkFilters
|
<Button
|
||||||
filters={filters}
|
variant="ghost"
|
||||||
onFiltersChange={setFilters}
|
className="w-full"
|
||||||
parks={parks}
|
onClick={() => setSidebarCollapsed(false)}
|
||||||
/>
|
title="Show filters"
|
||||||
</CardContent>
|
>
|
||||||
</Card>
|
<PanelLeftOpen className="w-5 h-5" />
|
||||||
|
</Button>
|
||||||
|
{activeFilterCount > 0 && (
|
||||||
|
<Badge variant="secondary" className="w-full mt-2 justify-center">
|
||||||
|
{activeFilterCount}
|
||||||
|
</Badge>
|
||||||
|
)}
|
||||||
|
</Card>
|
||||||
|
) : (
|
||||||
|
<Card>
|
||||||
|
<CardHeader className="flex flex-row items-center justify-between pb-2">
|
||||||
|
<CardTitle className="text-base">Filters</CardTitle>
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => setSidebarCollapsed(true)}
|
||||||
|
title="Hide filters"
|
||||||
|
>
|
||||||
|
<PanelLeftClose className="w-4 h-4" />
|
||||||
|
</Button>
|
||||||
|
</CardHeader>
|
||||||
|
<CardContent>
|
||||||
|
<ParkFilters
|
||||||
|
filters={filters}
|
||||||
|
onFiltersChange={setFilters}
|
||||||
|
parks={parks}
|
||||||
|
/>
|
||||||
|
</CardContent>
|
||||||
|
</Card>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</aside>
|
</aside>
|
||||||
|
|
||||||
|
|||||||
@@ -6,8 +6,9 @@ import { Badge } from '@/components/ui/badge';
|
|||||||
import { Dialog, DialogContent } from '@/components/ui/dialog';
|
import { Dialog, DialogContent } from '@/components/ui/dialog';
|
||||||
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from '@/components/ui/select';
|
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from '@/components/ui/select';
|
||||||
import { Collapsible, CollapsibleContent, CollapsibleTrigger } from '@/components/ui/collapsible';
|
import { Collapsible, CollapsibleContent, CollapsibleTrigger } from '@/components/ui/collapsible';
|
||||||
import { Card, CardContent } from '@/components/ui/card';
|
import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card';
|
||||||
import { Filter, SlidersHorizontal, FerrisWheel, Plus, ChevronDown } from 'lucide-react';
|
import { Filter, SlidersHorizontal, FerrisWheel, Plus, ChevronDown, PanelLeftClose, PanelLeftOpen } from 'lucide-react';
|
||||||
|
import { cn } from '@/lib/utils';
|
||||||
import { AutocompleteSearch } from '@/components/search/AutocompleteSearch';
|
import { AutocompleteSearch } from '@/components/search/AutocompleteSearch';
|
||||||
import { RideCard } from '@/components/rides/RideCard';
|
import { RideCard } from '@/components/rides/RideCard';
|
||||||
import { RideForm } from '@/components/admin/RideForm';
|
import { RideForm } from '@/components/admin/RideForm';
|
||||||
@@ -32,11 +33,19 @@ export default function Rides() {
|
|||||||
const [filters, setFilters] = useState<RideFilterState>(defaultRideFilters);
|
const [filters, setFilters] = useState<RideFilterState>(defaultRideFilters);
|
||||||
const [showFilters, setShowFilters] = useState(false);
|
const [showFilters, setShowFilters] = useState(false);
|
||||||
const [isCreateModalOpen, setIsCreateModalOpen] = useState(false);
|
const [isCreateModalOpen, setIsCreateModalOpen] = useState(false);
|
||||||
|
const [sidebarCollapsed, setSidebarCollapsed] = useState(() => {
|
||||||
|
const saved = localStorage.getItem('rides-sidebar-collapsed');
|
||||||
|
return saved ? JSON.parse(saved) : false;
|
||||||
|
});
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
fetchRides();
|
fetchRides();
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
localStorage.setItem('rides-sidebar-collapsed', JSON.stringify(sidebarCollapsed));
|
||||||
|
}, [sidebarCollapsed]);
|
||||||
|
|
||||||
const fetchRides = async () => {
|
const fetchRides = async () => {
|
||||||
try {
|
try {
|
||||||
const { data } = await supabase
|
const { data } = await supabase
|
||||||
@@ -193,6 +202,15 @@ export default function Rides() {
|
|||||||
<span className="hidden sm:inline">Filters</span>
|
<span className="hidden sm:inline">Filters</span>
|
||||||
<ChevronDown className={`w-4 h-4 transition-transform ${showFilters ? 'rotate-180' : ''}`} />
|
<ChevronDown className={`w-4 h-4 transition-transform ${showFilters ? 'rotate-180' : ''}`} />
|
||||||
</Button>
|
</Button>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
onClick={() => setSidebarCollapsed(!sidebarCollapsed)}
|
||||||
|
className="gap-2 hidden lg:flex"
|
||||||
|
title={sidebarCollapsed ? "Show filters" : "Hide filters"}
|
||||||
|
>
|
||||||
|
<PanelLeftClose className={`w-4 h-4 transition-transform ${sidebarCollapsed ? 'rotate-180' : ''}`} />
|
||||||
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -213,13 +231,44 @@ export default function Rides() {
|
|||||||
{/* Main Content Area with Sidebar */}
|
{/* Main Content Area with Sidebar */}
|
||||||
<div className="flex flex-col lg:flex-row gap-6">
|
<div className="flex flex-col lg:flex-row gap-6">
|
||||||
{/* Desktop Filter Sidebar */}
|
{/* Desktop Filter Sidebar */}
|
||||||
<aside className="hidden lg:block lg:w-[340px] xl:w-[380px] 2xl:w-[420px] flex-shrink-0">
|
<aside
|
||||||
|
className={cn(
|
||||||
|
"hidden lg:block flex-shrink-0 transition-all duration-300",
|
||||||
|
sidebarCollapsed
|
||||||
|
? "lg:w-[60px]"
|
||||||
|
: "lg:w-[340px] xl:w-[380px] 2xl:w-[420px]"
|
||||||
|
)}
|
||||||
|
>
|
||||||
<div className="sticky top-24">
|
<div className="sticky top-24">
|
||||||
<Card>
|
{sidebarCollapsed ? (
|
||||||
<CardContent className="pt-6">
|
<Card className="p-2">
|
||||||
<RideFilters filters={filters} onFiltersChange={setFilters} rides={rides} />
|
<Button
|
||||||
</CardContent>
|
variant="ghost"
|
||||||
</Card>
|
className="w-full"
|
||||||
|
onClick={() => setSidebarCollapsed(false)}
|
||||||
|
title="Show filters"
|
||||||
|
>
|
||||||
|
<PanelLeftOpen className="w-5 h-5" />
|
||||||
|
</Button>
|
||||||
|
</Card>
|
||||||
|
) : (
|
||||||
|
<Card>
|
||||||
|
<CardHeader className="flex flex-row items-center justify-between pb-2">
|
||||||
|
<CardTitle className="text-base">Filters</CardTitle>
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => setSidebarCollapsed(true)}
|
||||||
|
title="Hide filters"
|
||||||
|
>
|
||||||
|
<PanelLeftClose className="w-4 h-4" />
|
||||||
|
</Button>
|
||||||
|
</CardHeader>
|
||||||
|
<CardContent>
|
||||||
|
<RideFilters filters={filters} onFiltersChange={setFilters} rides={rides} />
|
||||||
|
</CardContent>
|
||||||
|
</Card>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</aside>
|
</aside>
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user