150 lines
6.9 KiB
TypeScript
150 lines
6.9 KiB
TypeScript
import type { Metadata } from 'next';
|
||
import Link from 'next/link';
|
||
import { notFound } from 'next/navigation';
|
||
import { cookies, headers } from 'next/headers';
|
||
import { getListingBySlug, DEFAULT_LOCALE, withResolvedListingImages } from '../../../lib/listings';
|
||
import { SAMPLE_LISTING_SLUGS } from '../../../lib/sampleListing';
|
||
import { resolveLocale, t as translate } from '../../../lib/i18n';
|
||
|
||
type ListingPageProps = {
|
||
params: { slug: string };
|
||
};
|
||
|
||
const amenityIcons: Record<string, string> = {
|
||
sauna: '🧖',
|
||
fireplace: '🔥',
|
||
wifi: '📶',
|
||
pets: '🐾',
|
||
lake: '🌊',
|
||
ac: '❄️',
|
||
ev: '⚡',
|
||
};
|
||
|
||
export async function generateMetadata({ params }: ListingPageProps): Promise<Metadata> {
|
||
const translation = await getListingBySlug({ slug: params.slug, locale: DEFAULT_LOCALE });
|
||
|
||
return {
|
||
title: translation ? `${translation.title} | Lomavuokraus.fi` : `${params.slug} | Lomavuokraus.fi`,
|
||
description: translation?.teaser ?? translation?.description?.slice(0, 140),
|
||
};
|
||
}
|
||
|
||
export default async function ListingPage({ params }: ListingPageProps) {
|
||
const cookieStore = cookies();
|
||
const locale = resolveLocale({ cookieLocale: cookieStore.get('locale')?.value, acceptLanguage: headers().get('accept-language') });
|
||
const t = (key: any, vars?: Record<string, string | number>) => translate(locale, key as any, vars);
|
||
|
||
const translationRaw = await getListingBySlug({ slug: params.slug, locale: locale ?? DEFAULT_LOCALE });
|
||
const translation = translationRaw ? withResolvedListingImages(translationRaw) : null;
|
||
|
||
if (!translation) {
|
||
notFound();
|
||
}
|
||
|
||
const { listing, title, description, teaser, locale: translationLocale } = translation;
|
||
const isSample = listing.isSample || listing.contactEmail === 'host@lomavuokraus.fi' || SAMPLE_LISTING_SLUGS.includes(params.slug);
|
||
const amenities = [
|
||
listing.hasSauna ? { icon: amenityIcons.sauna, label: t('amenitySauna') } : null,
|
||
listing.hasFireplace ? { icon: amenityIcons.fireplace, label: t('amenityFireplace') } : null,
|
||
listing.hasWifi ? { icon: amenityIcons.wifi, label: t('amenityWifi') } : null,
|
||
listing.petsAllowed ? { icon: amenityIcons.pets, label: t('amenityPets') } : null,
|
||
listing.byTheLake ? { icon: amenityIcons.lake, label: t('amenityLake') } : null,
|
||
listing.hasAirConditioning ? { icon: amenityIcons.ac, label: t('amenityAirConditioning') } : null,
|
||
listing.evCharging === 'FREE' ? { icon: amenityIcons.ev, label: t('amenityEvFree') } : null,
|
||
listing.evCharging === 'PAID' ? { icon: amenityIcons.ev, label: t('amenityEvPaid') } : null,
|
||
].filter(Boolean) as { icon: string; label: string }[];
|
||
const addressLine = `${listing.streetAddress ? `${listing.streetAddress}, ` : ''}${listing.city}, ${listing.region}, ${listing.country}`;
|
||
const capacityLine = `${t('capacityGuests', { count: listing.maxGuests })} · ${t('capacityBedrooms', { count: listing.bedrooms })} · ${t('capacityBeds', { count: listing.beds })} · ${t('capacityBathrooms', { count: listing.bathrooms })}`;
|
||
const contactLine = `${listing.contactName} · ${listing.contactEmail}${listing.contactPhone ? ` · ${listing.contactPhone}` : ''}`;
|
||
|
||
return (
|
||
<main className="listing-shell">
|
||
<div className="breadcrumb">
|
||
<Link href="/">{t('homeCrumb')}</Link> / <span>{params.slug}</span>
|
||
</div>
|
||
<div className="listing-layout">
|
||
<div className="panel listing-main">
|
||
{isSample ? (
|
||
<div className="badge warning" style={{ marginBottom: 10, display: 'inline-block' }}>
|
||
{t('sampleBadge')}
|
||
</div>
|
||
) : null}
|
||
<h1>{title}</h1>
|
||
<p style={{ marginTop: 8 }}>{teaser ?? description}</p>
|
||
{listing.addressNote ? (
|
||
<div style={{ marginTop: 4, color: '#cbd5e1' }}>
|
||
<em>{listing.addressNote}</em>
|
||
</div>
|
||
) : null}
|
||
{listing.externalUrl ? (
|
||
<div style={{ marginTop: 12 }}>
|
||
<a href={listing.externalUrl} target="_blank" rel="noreferrer" className="button secondary">
|
||
{t('listingMoreInfo')}
|
||
</a>
|
||
</div>
|
||
) : null}
|
||
{listing.images.length > 0 ? (
|
||
<div style={{ marginTop: 12, display: 'grid', gap: 12, gridTemplateColumns: 'repeat(auto-fit, minmax(240px, 1fr))' }}>
|
||
{listing.images
|
||
.filter((img) => Boolean(img.url))
|
||
.map((img) => (
|
||
<figure key={img.id} style={{ border: '1px solid rgba(148, 163, 184, 0.25)', borderRadius: 12, overflow: 'hidden', background: 'rgba(255,255,255,0.03)' }}>
|
||
<a href={img.url || ''} target="_blank" rel="noreferrer" style={{ display: 'block', cursor: 'zoom-in' }}>
|
||
<img src={img.url || ''} alt={img.altText ?? title} style={{ width: '100%', height: '200px', objectFit: 'cover' }} />
|
||
</a>
|
||
{img.altText ? (
|
||
<figcaption style={{ padding: '10px 12px', fontSize: 14, color: '#cbd5e1' }}>{img.altText}</figcaption>
|
||
) : null}
|
||
</figure>
|
||
))}
|
||
</div>
|
||
) : null}
|
||
<div style={{ marginTop: 16, fontSize: 14, color: '#666' }}>
|
||
{t('localeLabel')}: <code>{translationLocale}</code>
|
||
</div>
|
||
</div>
|
||
<aside className="panel listing-aside">
|
||
<div className="fact-row">
|
||
<span aria-hidden className="amenity-icon">📍</span>
|
||
<div>
|
||
<div style={{ color: '#cbd5e1', fontSize: 12 }}>{t('listingLocation')}</div>
|
||
<div>{addressLine}</div>
|
||
</div>
|
||
</div>
|
||
<div className="fact-row">
|
||
<span aria-hidden className="amenity-icon">👥</span>
|
||
<div>
|
||
<div style={{ color: '#cbd5e1', fontSize: 12 }}>{t('listingCapacity')}</div>
|
||
<div>{capacityLine}</div>
|
||
</div>
|
||
</div>
|
||
<div className="fact-row">
|
||
<span aria-hidden className="amenity-icon">✉️</span>
|
||
<div>
|
||
<div style={{ color: '#cbd5e1', fontSize: 12 }}>{t('listingContact')}</div>
|
||
<div>{contactLine}</div>
|
||
</div>
|
||
</div>
|
||
<div className="amenity-list">
|
||
<div style={{ color: '#cbd5e1', fontSize: 12 }}>{t('listingAmenities')}</div>
|
||
{amenities.length === 0 ? (
|
||
<div className="amenity-row" style={{ borderStyle: 'dashed' }}>
|
||
<span className="amenity-icon">…</span>
|
||
<span>{t('listingNoAmenities')}</span>
|
||
</div>
|
||
) : (
|
||
amenities.map((item) => (
|
||
<div key={item.label} className="amenity-row">
|
||
<span className="amenity-icon" aria-hidden>
|
||
{item.icon}
|
||
</span>
|
||
<span>{item.label}</span>
|
||
</div>
|
||
))
|
||
)}
|
||
</div>
|
||
</aside>
|
||
</div>
|
||
</main>
|
||
);
|
||
}
|