239 lines
8.4 KiB
TypeScript
239 lines
8.4 KiB
TypeScript
import type { Metadata } from 'next';
|
|
import Link from 'next/link';
|
|
import { notFound } from 'next/navigation';
|
|
import { Section } from '@/components/Section';
|
|
import { buildFallbackProperty, getPublishedPropertyBySlug } from '@/lib/properties';
|
|
import { site } from '@/lib/site';
|
|
|
|
export const dynamic = 'force-dynamic';
|
|
|
|
type PropertyPageProps = {
|
|
params: Promise<{
|
|
slug: string;
|
|
}>;
|
|
};
|
|
|
|
function formatPounds(cents: number) {
|
|
return new Intl.NumberFormat('en-GB', {
|
|
style: 'currency',
|
|
currency: 'GBP',
|
|
maximumFractionDigits: 0,
|
|
}).format(cents / 100);
|
|
}
|
|
|
|
function formatDate(date: Date) {
|
|
return date.toISOString().slice(0, 10);
|
|
}
|
|
|
|
function formatReason(reason: string) {
|
|
return reason
|
|
.toLowerCase()
|
|
.split('_')
|
|
.map((chunk) => chunk.charAt(0).toUpperCase() + chunk.slice(1))
|
|
.join(' ');
|
|
}
|
|
|
|
function fallbackMetadata(slug: string) {
|
|
const seeded = buildFallbackProperty(slug);
|
|
if (!seeded) {
|
|
return {
|
|
title: site.name,
|
|
description: site.description,
|
|
};
|
|
}
|
|
|
|
return {
|
|
title: `${seeded.title} | ${site.name}`,
|
|
description: seeded.summary,
|
|
};
|
|
}
|
|
|
|
export async function generateMetadata({ params }: PropertyPageProps): Promise<Metadata> {
|
|
const { slug } = await params;
|
|
|
|
try {
|
|
const property = await getPublishedPropertyBySlug(slug);
|
|
if (!property) {
|
|
return fallbackMetadata(slug);
|
|
}
|
|
|
|
return {
|
|
title: `${property.title} | ${site.name}`,
|
|
description: property.summary,
|
|
};
|
|
} catch {
|
|
return fallbackMetadata(slug);
|
|
}
|
|
}
|
|
|
|
export default async function PropertyDetailPage({ params }: PropertyPageProps) {
|
|
const { slug } = await params;
|
|
|
|
const property = (await getPublishedPropertyBySlug(slug)) ?? buildFallbackProperty(slug);
|
|
if (!property) {
|
|
notFound();
|
|
}
|
|
|
|
const primaryImage = property.images[0] ?? null;
|
|
const standardRate = property.pricingRules.find((rule) => !rule.validFrom && !rule.validTo) ?? property.pricingRules[0];
|
|
const seasonalRates = property.pricingRules.filter((rule) => rule.validFrom && rule.validTo);
|
|
|
|
return (
|
|
<>
|
|
<section className="page-hero property-hero">
|
|
<div>
|
|
<p className="brand-kicker">Property detail</p>
|
|
<h2>{property.title}</h2>
|
|
<p>{property.summary}</p>
|
|
<div className="hero-actions">
|
|
<Link className="btn btn-primary" href={`/bookings/new?propertySlug=${property.slug}`}>
|
|
Check availability
|
|
</Link>
|
|
<Link className="btn btn-outline-dark" href={`/contact?property=${encodeURIComponent(property.title)}`}>
|
|
Ask a question first
|
|
</Link>
|
|
</div>
|
|
</div>
|
|
|
|
<article className="property-callout">
|
|
<p className="footer-label">Booking context</p>
|
|
<strong>{standardRate ? `From ${formatPounds(standardRate.basePriceCents)} per night` : 'Price on request'}</strong>
|
|
<p>{property.locationText}</p>
|
|
<dl className="property-metrics">
|
|
<div>
|
|
<dt>Sleeps</dt>
|
|
<dd>{property.sleeps}</dd>
|
|
</div>
|
|
<div>
|
|
<dt>Bedrooms</dt>
|
|
<dd>{property.bedrooms}</dd>
|
|
</div>
|
|
<div>
|
|
<dt>Bathrooms</dt>
|
|
<dd>{property.bathrooms}</dd>
|
|
</div>
|
|
</dl>
|
|
</article>
|
|
</section>
|
|
|
|
<div className="page-layout property-layout">
|
|
<Section
|
|
eyebrow="Gallery and overview"
|
|
title="Everything a guest needs before starting the booking flow"
|
|
description="This page is intentionally practical first: imagery, essentials, stay rules, and a clear route into booking or enquiry."
|
|
>
|
|
<div className="property-gallery" aria-label={`${property.title} gallery`}>
|
|
{property.images.map((image) => (
|
|
<figure key={image.id} className={`property-gallery-card ${image.primaryImage ? 'is-primary' : ''}`}>
|
|
<img src={image.url} alt={image.altText} />
|
|
</figure>
|
|
))}
|
|
</div>
|
|
|
|
<div className="content-stack">
|
|
<article className="content-card">
|
|
<h3>About this stay</h3>
|
|
<p>{property.longDescription}</p>
|
|
</article>
|
|
|
|
<article className="content-card">
|
|
<h3>Amenities</h3>
|
|
<ul className="tag-list">
|
|
{property.amenities.map((item) => (
|
|
<li key={item.amenityId}>{item.amenity.name}</li>
|
|
))}
|
|
</ul>
|
|
</article>
|
|
|
|
<article className="content-card">
|
|
<h3>Stay policies and practical details</h3>
|
|
<div className="policy-grid">
|
|
<div className="metric">
|
|
<strong>{property.minStayNights}</strong>
|
|
<span>Minimum nights</span>
|
|
</div>
|
|
<div className="metric">
|
|
<strong>{property.checkInTime ?? '16:00'}</strong>
|
|
<span>Check-in</span>
|
|
</div>
|
|
<div className="metric">
|
|
<strong>{property.checkOutTime ?? '10:00'}</strong>
|
|
<span>Check-out</span>
|
|
</div>
|
|
<div className="metric">
|
|
<strong>{property.petsAllowed ? 'Allowed' : 'Not allowed'}</strong>
|
|
<span>Pets</span>
|
|
</div>
|
|
</div>
|
|
</article>
|
|
</div>
|
|
</Section>
|
|
|
|
<aside className="content-sidebar">
|
|
<article className="content-card">
|
|
<p className="footer-label">Primary image</p>
|
|
<h3>{primaryImage ? primaryImage.altText : 'Property overview'}</h3>
|
|
<p className="mb-0">
|
|
{primaryImage ? 'The first gallery image sets the main expectation for the stay.' : property.summary}
|
|
</p>
|
|
</article>
|
|
|
|
<article className="content-card">
|
|
<h3>Availability and pricing context</h3>
|
|
<ul className="admin-bullet-list">
|
|
<li>{standardRate ? `Base nightly rate: ${formatPounds(standardRate.basePriceCents)}` : 'Rate available on request'}</li>
|
|
<li>{standardRate?.weekendPriceCents ? `Weekend rate: ${formatPounds(standardRate.weekendPriceCents)}` : 'Weekend pricing follows the base rate'}</li>
|
|
<li>{standardRate?.guestDeltaCents ? `Additional guest supplement: ${formatPounds(standardRate.guestDeltaCents)} per night` : 'No guest supplement applies'}</li>
|
|
</ul>
|
|
{seasonalRates.length ? (
|
|
<div className="availability-list">
|
|
{seasonalRates.map((rule) => (
|
|
<div key={rule.id} className="availability-item">
|
|
<strong>{rule.label || 'Seasonal rate'}</strong>
|
|
<span>
|
|
{formatDate(rule.validFrom as Date)} to {formatDate(rule.validTo as Date)} · {formatPounds(rule.basePriceCents)}
|
|
</span>
|
|
</div>
|
|
))}
|
|
</div>
|
|
) : null}
|
|
</article>
|
|
|
|
<article className="content-card">
|
|
<h3>Known unavailable dates</h3>
|
|
<div className="availability-list">
|
|
{property.availability.length ? (
|
|
property.availability.map((block) => (
|
|
<div key={block.id} className="availability-item">
|
|
<strong>{formatReason(block.reason)}</strong>
|
|
<span>
|
|
{formatDate(block.startDate)} to {formatDate(block.endDate)}
|
|
</span>
|
|
</div>
|
|
))
|
|
) : (
|
|
<p className="mb-0">No blocked dates are currently published for this property.</p>
|
|
)}
|
|
</div>
|
|
</article>
|
|
|
|
<article className="content-card">
|
|
<h3>Guest feedback</h3>
|
|
<div className="content-stack">
|
|
{property.testimonials.map((testimonial) => (
|
|
<blockquote key={testimonial.id} className="testimonial-card">
|
|
<p>{testimonial.content}</p>
|
|
<footer>
|
|
<strong>{testimonial.authorName}</strong>
|
|
<span>{testimonial.rating ? `${testimonial.rating}/5 rating` : 'Published guest feedback'}</span>
|
|
</footer>
|
|
</blockquote>
|
|
))}
|
|
</div>
|
|
</article>
|
|
</aside>
|
|
</div>
|
|
</>
|
|
);
|
|
}
|