-
Notifications
You must be signed in to change notification settings - Fork 76
Feat/create message placeholder page #216
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. Weβll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Changes from 9 commits
b1107b4
a0bcd03
412e8a4
907dcb6
4b2ada3
267caca
ddfb824
238a5e2
5cfc44d
7dbe826
6e3dca0
54f26cc
f489339
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,42 @@ | ||
| 'use client'; | ||
|
|
||
| import { Search } from 'lucide-react'; | ||
| import { useState } from 'react'; | ||
|
|
||
| export default function MessagesPage() { | ||
| const [searchQuery, setSearchQuery] = useState(''); | ||
|
|
||
| return ( | ||
| /* CORRECCIΓN: Se cambiΓ³ 64px por 56px para coincidir con la altura real del Navbar (h-14) */ | ||
|
||
| <div className="flex h-[calc(100vh-56px)] w-full bg-[#0B1221] text-white overflow-hidden font-sans"> | ||
| {/* Columna Izquierda */} | ||
| <div className="w-80 border-r border-gray-800 flex flex-col bg-[#0F172A]"> | ||
| <div className="p-4 border-b border-gray-800"> | ||
| <div className="relative"> | ||
| <Search className="absolute left-4 top-2.5 h-4 w-4 text-gray-500" aria-hidden="true" /> | ||
| <input | ||
| type="text" | ||
| value={searchQuery} | ||
| onChange={(e) => setSearchQuery(e.target.value)} | ||
| placeholder="Search chat" | ||
| aria-label="Search chats" | ||
| className="w-full bg-[#161F2F] border-none rounded-full py-2 pl-12 pr-4 text-sm focus:ring-1 focus:ring-blue-500 outline-none placeholder:text-gray-500" | ||
| /> | ||
| </div> | ||
| </div> | ||
|
|
||
| {/* no_chats: Posicionado arriba con padding para balance visual */} | ||
| <div className="flex-1 flex flex-col items-center pt-8"> | ||
| <span className="text-gray-500 text-sm font-light italic">No chats available</span> | ||
| </div> | ||
| </div> | ||
|
|
||
| {/* Columna Derecha */} | ||
| <div className="flex-1 flex items-center justify-center bg-[#0B1221]"> | ||
| <span className="text-gray-500 text-sm font-light tracking-[0.2em] uppercase opacity-60"> | ||
| Select a chat | ||
| </span> | ||
| </div> | ||
| </div> | ||
| ); | ||
| } | ||
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -1,6 +1,6 @@ | ||
| 'use client'; | ||
|
|
||
| import PropertyGrid from '@/components/search/PropertyGrid'; | ||
| import { PropertyGrid } from '@/components/search/PropertyGrid'; | ||
| import type { LatLngTuple } from 'leaflet'; | ||
| import dynamic from 'next/dynamic'; | ||
| import { useSearchParams } from 'next/navigation'; | ||
|
|
@@ -32,17 +32,13 @@ export default function SearchPage() { | |
| { position: [-34.6, -58.37], title: 'Cozy Studio Apartment' }, | ||
| ]; | ||
|
|
||
| // Filter & sort properties with memoization | ||
| const filteredSortedProperties = useMemo(() => { | ||
| let result = [...MOCK_PROPERTIES]; | ||
|
|
||
| const location = searchParams.get('location')?.toLowerCase() || ''; | ||
| if (location) { | ||
| result = result.filter((p) => p.location.toLowerCase().includes(location)); | ||
| } | ||
|
|
||
| result = result.filter((p) => p.price >= filters.price); | ||
|
|
||
| const selectedAmenities = Object.entries(filters.amenities) | ||
| .filter(([, checked]) => checked) | ||
| .map(([key]) => key.toLowerCase()); | ||
|
|
@@ -52,22 +48,12 @@ export default function SearchPage() { | |
| selectedAmenities.every((am) => p.amenities.map((a) => a.toLowerCase()).includes(am)) | ||
| ); | ||
| } | ||
|
|
||
| if (filters.rating > 0) { | ||
| result = result.filter((p) => p.rating >= filters.rating); | ||
| } | ||
|
|
||
| if (sort === 'price_asc') result.sort((a, b) => a.price - b.price); | ||
| if (sort === 'price_desc') result.sort((a, b) => b.price - a.price); | ||
| if (sort === 'rating') result.sort((a, b) => b.rating - a.rating); | ||
| if (sort === 'distance') { | ||
| result.sort((a, b) => { | ||
| const aDist = Number.parseFloat(a.distance); | ||
| const bDist = Number.parseFloat(b.distance); | ||
| return aDist - bDist; | ||
| }); | ||
| } | ||
|
|
||
| return result; | ||
| }, [filters, sort, searchParams]); | ||
|
|
||
|
|
@@ -81,14 +67,17 @@ export default function SearchPage() { | |
| setTimeout(() => { | ||
| setPage((prev) => prev + 1); | ||
| setIsLoading(false); | ||
| }, 200); // simulate load | ||
| }, 200); | ||
| }, [isLoading]); | ||
|
|
||
| const minMax = useMemo(() => { | ||
| const sorted = [...MOCK_PROPERTIES].sort((a, b) => a.price - b.price); | ||
| return [sorted[0]?.price || 0, sorted.at(-1)?.price || 0] as [number, number]; | ||
| }, []); | ||
|
|
||
| // Alias para evitar el error de IntrinsicAttributes | ||
|
||
| const Grid = PropertyGrid as any; | ||
|
|
||
| return ( | ||
| <main className="px-4 py-6 mt-10 space-y-6"> | ||
| <div className="flex flex-col lg:flex-row gap-3 md:gap-6"> | ||
|
|
@@ -110,7 +99,7 @@ export default function SearchPage() { | |
|
|
||
| <div className="flex flex-col lg:flex-row"> | ||
| <div className="w-full"> | ||
| <PropertyGrid properties={visibleProperties} onLoadMore={loadNextPage} /> | ||
| <Grid properties={visibleProperties} onLoadMore={loadNextPage} /> | ||
| {isLoading && <p className="text-center my-4">Loading more properties...</p>} | ||
| </div> | ||
|
|
||
|
|
@@ -122,4 +111,4 @@ export default function SearchPage() { | |
| </div> | ||
| </main> | ||
| ); | ||
| } | ||
| } | ||
| Original file line number | Diff line number | Diff line change |
|---|---|---|
|
|
@@ -7,18 +7,64 @@ export interface MenuItem { | |
| withContainer?: boolean; | ||
| } | ||
|
|
||
| const ICON_MENU = { id: 'menu', src: '/icons/menu.webp', alt: 'Menu', label: 'Menu', href: '#' }; | ||
| const ICON_SEARCH = { | ||
| id: 'search', | ||
| src: '/icons/search.webp', | ||
| alt: 'Search', | ||
| label: 'Find a Property', | ||
| href: '/search', | ||
| withContainer: true, | ||
| }; | ||
| const ICON_FAVORITES = { | ||
| id: 'favorites', | ||
| src: '/icons/heart.webp', | ||
| alt: 'Favorites', | ||
| label: 'Favorites', | ||
| href: '/dashboard/guest?tab=bookings', | ||
| }; | ||
|
Comment on lines
+21
to
+27
Contributor
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Semantic mismatch: "Favorites" label points to bookings tab.
π€ Prompt for AI Agents |
||
| const ICON_MESSAGES = { | ||
| id: 'messages', | ||
| src: '/icons/send.webp', | ||
| alt: 'Messages', | ||
| label: 'Messages', | ||
| href: '/messages', | ||
| withContainer: true, | ||
| }; | ||
|
|
||
| // CORRECCIΓN: Renombramos 'Settings' a 'Invitations' para que coincida con la ruta /invitations | ||
| const ICON_INVITATIONS = { | ||
| id: 'invitations', | ||
| src: '/icons/settings.webp', | ||
| alt: 'Invitations', | ||
| label: 'Invitations', | ||
| href: '/invitations', | ||
| }; | ||
|
Comment on lines
+39
to
+45
Contributor
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Icon asset doesn't match the "Invitations" semantics. While the label/href mismatch has been corrected, the icon source still references π€ Prompt for AI Agents |
||
|
|
||
| const ICON_LOCK = { id: 'lock', src: '/icons/lock.webp', alt: 'Lock', label: 'Private', href: '#' }; | ||
| const ICON_APPLICATIONS = { | ||
| id: 'applications', | ||
| src: '/icons/message.webp', | ||
| alt: 'Applications', | ||
| label: 'Applications', | ||
| href: '/applications', | ||
| }; | ||
|
|
||
| export const GUEST_MENU_ITEMS: MenuItem[] = [ | ||
| { id: 'menu', src: '/icons/menu.webp', alt: 'Menu', label: 'Menu', href: '#' }, | ||
| { | ||
| id: 'search', | ||
| src: '/icons/search.webp', | ||
| alt: 'Find a Property', | ||
| label: 'Find a Property', | ||
| href: '/search', | ||
| withContainer: true, | ||
| }, | ||
| ICON_MENU, | ||
| ICON_SEARCH, | ||
| ICON_FAVORITES, | ||
| ICON_MESSAGES, | ||
| ICON_INVITATIONS, // Ahora el nombre es semΓ‘nticamente correcto | ||
| ICON_LOCK, | ||
| ICON_APPLICATIONS, | ||
| ]; | ||
|
|
||
| <<<<<<< HEAD | ||
| export const TENANT_MENU_ITEMS: MenuItem[] = [...GUEST_MENU_ITEMS]; | ||
| export const HOST_MENU_ITEMS: MenuItem[] = [...GUEST_MENU_ITEMS]; | ||
| export const DUAL_MENU_ITEMS: MenuItem[] = [...GUEST_MENU_ITEMS]; | ||
| ======= | ||
|
||
| export const TENANT_MENU_ITEMS: MenuItem[] = [ | ||
| // TODO: Wire menu item to navigation drawer | ||
| { id: 'menu', src: '/icons/menu.webp', alt: 'Menu', label: 'Menu', href: '#' }, | ||
|
|
@@ -162,3 +208,4 @@ export const DUAL_MENU_ITEMS: MenuItem[] = [ | |
| href: '/dashboard/tenant-dashboard?tab=calendar', | ||
| }, | ||
| ]; | ||
| >>>>>>> origin/main | ||
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
variables in english