// Public pages: Tours listing, Tour detail, Vehicles, Blog, Page, Contact // -------- TOURS LISTING -------- const PublicTours = ({ state, navigate }) => { const [category, setCategory] = useState('all'); const [sort, setSort] = useState('featured'); const [priceRange, setPriceRange] = useState('any'); const [durationFilter, setDurationFilter] = useState('any'); const [destFilter, setDestFilter] = useState('all'); const [page, setPage] = useState(1); const perPage = 8; const destinations = ['all', ...new Set(state.packages.filter(p => p.status === 'published').map(p => p.destination))]; const list = state.packages .filter(p => p.status === 'published') .filter(p => category === 'all' || p.category === category) .filter(p => destFilter === 'all' || p.destination === destFilter) .filter(p => { if (priceRange === 'any') return true; if (priceRange === '<200') return p.priceFrom < 200; if (priceRange === '200-500') return p.priceFrom >= 200 && p.priceFrom <= 500; if (priceRange === '500-1500') return p.priceFrom > 500 && p.priceFrom <= 1500; return p.priceFrom > 1500; }) .filter(p => { if (durationFilter === 'any') return true; if (durationFilter === 'half') return p.durationDays <= 0.5; if (durationFilter === 'day') return p.durationDays === 1; if (durationFilter === 'multi') return p.durationDays >= 2 && p.durationDays <= 3; return p.durationDays > 3; }); const sorted = [...list].sort((a, b) => { if (sort === 'price-asc') return a.priceFrom - b.priceFrom; if (sort === 'price-desc') return b.priceFrom - a.priceFrom; if (sort === 'rating') return b.rating - a.rating; return (b.featured ? 1 : 0) - (a.featured ? 1 : 0); }); const totalPages = Math.ceil(sorted.length / perPage) || 1; const paged = sorted.slice((page - 1) * perPage, page * perPage); return (
{state.packages.filter(p => p.status === 'published').length} packages · inbound Sarawak + outbound from Kuching.
{pkg.short}
{pkg.description}
{/* Itinerary */}Self-drive economy to executive MPV to 4WD for the interior. All vehicles insured, serviced monthly.
{v.description}
{hero.excerpt}
{b.excerpt}
Fill this in and we'll send it straight to our inbox + your WhatsApp. A real human replies within 4 working hours.
A confirmation is on its way to {form.email}. Our team usually replies within 4 working hours.