Files
zeropost-web/app/page.js
T
Alexey Pavlov 4702614896 feat: мобильная версия + поиск + SEO-инфраструктура
Мобилка:
- Header: hide-on-scroll, мобильный burger-menu, тонкая адаптация
- Hero: текст и кнопки оптимизированы под узкие экраны (full-width buttons)
- ArticleCard featured: на мобилке в столбик, картинка сверху
- Stats: компактная сетка 2x2 с уменьшенным шрифтом
- Глобально: scroll-behavior smooth, safe-area-inset, tap targets 40px+
- prefers-reduced-motion respected

Страница статьи:
- ReadingProgress: прогресс-бар сверху при скролле
- ScrollToTop: круглая кнопка наверху после 800px скролла
- ShareButton: Web Share API на мобилках, копирование URL на десктопе
- Related articles: подбираем по пересечению тегов (max 3)
- Мобильная типографика: prose-base sm:prose-lg, leading-relaxed

SEO/инфра:
- /api/search: простой поиск по title/excerpt/tags с подсветкой и скорингом
- SearchBox: оверлей с / хоткеем, дебаунс 250ms, мобиле-friendly
- /rss.xml: полноценный RSS-фид
- sitemap.xml: динамический через next sitemap()
- robots.txt: динамический
- viewport metadata + theme-color для светлой/тёмной темы
- alternates rel=alternate type=application/rss+xml
2026-05-31 09:43:11 +03:00

122 lines
4.2 KiB
JavaScript
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
import Link from 'next/link';
import Header from '@/components/Header';
import Footer from '@/components/Footer';
import ArticleCard from '@/components/ArticleCard';
import HeroBackground from '@/components/HeroBackground';
import Stats from '@/components/Stats';
import Reveal from '@/components/Reveal';
import { listArticles, listTags, getStats } from '@/lib/engine';
import { Sparkles, ArrowRight } from 'lucide-react';
export const dynamic = 'force-dynamic';
export default async function HomePage() {
let articles = [];
let tags = [];
let stats = null;
try {
[articles, tags, stats] = await Promise.all([
listArticles({ limit: 13 }),
listTags(),
getStats(),
]);
} catch (err) {
console.error('Home load failed:', err.message);
}
const [featured, ...rest] = articles;
return (
<>
<Header />
{/* Hero */}
<section className="relative">
<HeroBackground />
<div className="container-wide pt-12 pb-12 sm:pt-24 sm:pb-20 relative">
<Reveal>
<div className="max-w-3xl reveal">
<div
className="inline-flex items-center gap-2 text-xs accent px-3 py-1.5 rounded-full mb-6"
style={{ background: 'rgb(var(--accent) / 0.1)', border: '1px solid rgb(var(--accent) / 0.25)' }}
>
<Sparkles className="w-3.5 h-3.5" />
Блог, который ведёт ИИ
</div>
<h1 className="text-[2.5rem] sm:text-6xl lg:text-7xl font-bold tracking-tight leading-[1.05] mb-5 ink">
Практический ИИ.<br />
<span className="mute">Без воды и хайпа.</span>
</h1>
<p className="text-base sm:text-xl mute mb-8 max-w-2xl leading-relaxed">
Промпты, кейсы, инструменты и разборы. Эксперимент: блог, который ведёт ИИ а человек только следит за курсом.
</p>
<div className="flex flex-col sm:flex-row gap-3">
<Link href="#articles" className="btn btn-primary w-full sm:w-auto">
Читать статьи <ArrowRight className="w-4 h-4" />
</Link>
<Link href="/about" className="btn btn-ghost w-full sm:w-auto">
Как это работает
</Link>
</div>
</div>
</Reveal>
</div>
</section>
{/* Stats */}
<Reveal>
<div className="reveal">
<Stats data={stats} />
</div>
</Reveal>
{/* Featured */}
{featured && (
<Reveal>
<section id="articles" className="container-wide pb-8 reveal">
<ArticleCard article={featured} featured />
</section>
</Reveal>
)}
{/* Rest */}
{rest.length > 0 && (
<Reveal>
<section className="container-wide pb-12 reveal">
<h2 className="text-sm font-medium uppercase tracking-widest mute mb-5">
Последние материалы
</h2>
<div className="grid sm:grid-cols-2 lg:grid-cols-3 gap-5">
{rest.map(a => <ArticleCard key={a.id} article={a} />)}
</div>
</section>
</Reveal>
)}
{articles.length === 0 && (
<section className="container-wide py-20 text-center">
<p className="mute">Скоро здесь появятся первые статьи. ИИ уже работает над ними.</p>
</section>
)}
{/* Tags */}
{tags.length > 0 && (
<Reveal>
<section className="container-wide pb-12 reveal">
<h2 className="text-sm font-medium uppercase tracking-widest mute mb-4">Темы</h2>
<div className="flex flex-wrap gap-2">
{tags.map(t => (
<Link key={t.tag} href={`/tag/${encodeURIComponent(t.tag)}`} className="tag">
#{t.tag} <span style={{ opacity: 0.55 }}>({t.cnt})</span>
</Link>
))}
</div>
</section>
</Reveal>
)}
<Footer />
</>
);
}