- nav: add functional mobile menu panel with JS toggle - nav: hamburger now shows/hides mobile-menu with aria-expanded state - about_page: full styled layout (header, prose body, author aside) - legal_page: full styled layout (header with last-updated, max-w-3xl prose) - article: fix aside newsletter label to 'Subscribe' for E2E test - CSS: rebuild after all template changes (4.8KB → 24.3KB) Committed CSS must match CI build — rebuilt after ALL template edits Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
227 lines
15 KiB
HTML
227 lines
15 KiB
HTML
{% extends 'base.html' %}
|
|
{% load wagtailcore_tags wagtailimages_tags seo_tags core_tags %}
|
|
{% block title %}{{ page.title }} | No Hype AI{% endblock %}
|
|
{% block head_meta %}
|
|
{% canonical_url page as canonical %}
|
|
{% article_og_image_url page as og_image %}
|
|
<link rel="canonical" href="{{ canonical }}" />
|
|
<meta name="description" content="{{ page.search_description|default:page.summary }}" />
|
|
<meta property="og:type" content="article" />
|
|
<meta property="og:title" content="{{ page.title }} | No Hype AI" />
|
|
<meta property="og:description" content="{{ page.search_description|default:page.summary }}" />
|
|
<meta property="og:url" content="{{ canonical }}" />
|
|
{% if og_image %}<meta property="og:image" content="{{ og_image }}" />{% endif %}
|
|
<meta name="twitter:card" content="summary_large_image" />
|
|
<meta name="twitter:title" content="{{ page.title }} | No Hype AI" />
|
|
<meta name="twitter:description" content="{{ page.search_description|default:page.summary }}" />
|
|
{% if og_image %}<meta name="twitter:image" content="{{ og_image }}" />{% endif %}
|
|
{% endblock %}
|
|
{% block content %}
|
|
|
|
<!-- Breadcrumb -->
|
|
<div class="mb-8 font-mono text-sm text-zinc-500">
|
|
<a href="/" class="hover:text-brand-cyan transition-colors">Home</a> /
|
|
<a href="/articles/" class="hover:text-brand-cyan transition-colors">Articles</a> /
|
|
<span class="text-brand-dark dark:text-brand-light">{{ page.title|truncatechars:40 }}</span>
|
|
</div>
|
|
|
|
<!-- Article Header -->
|
|
<header class="mb-12 border-b border-zinc-200 dark:border-zinc-800 pb-12">
|
|
<div class="flex gap-3 mb-6 items-center flex-wrap">
|
|
{% for tag in page.tags.all %}
|
|
<span class="text-xs font-mono font-bold px-2 py-1 {{ tag|get_tag_css }} border border-current/20">{{ tag.name }}</span>
|
|
{% endfor %}
|
|
<span class="text-sm font-mono text-zinc-500">{{ page.first_published_at|date:"M j, Y" }}</span>
|
|
<span class="text-sm font-mono text-zinc-500">{{ page.read_time_mins }} min read</span>
|
|
</div>
|
|
<h1 class="font-display font-black text-4xl md:text-6xl lg:text-7xl leading-tight mb-8">{{ page.title }}</h1>
|
|
<div class="flex items-center gap-4">
|
|
<div class="w-12 h-12 bg-gradient-to-tr from-brand-cyan to-brand-pink shrink-0"></div>
|
|
<div>
|
|
<div class="font-bold font-display text-lg">{{ page.author.name }}</div>
|
|
{% if page.author.role %}<div class="font-mono text-xs text-zinc-500">{{ page.author.role }}</div>{% endif %}
|
|
</div>
|
|
</div>
|
|
</header>
|
|
|
|
{% if page.hero_image %}
|
|
<div class="mb-12 border border-zinc-200 dark:border-zinc-800 overflow-hidden">
|
|
{% image page.hero_image width-1200 class="w-full h-auto" %}
|
|
</div>
|
|
{% endif %}
|
|
|
|
<!-- Main Content Layout -->
|
|
<div class="grid grid-cols-1 lg:grid-cols-12 gap-12">
|
|
|
|
<!-- Article Body -->
|
|
<article class="lg:col-span-8 prose prose-lg dark:prose-invert max-w-none
|
|
prose-headings:font-display prose-headings:font-bold
|
|
prose-a:text-brand-cyan hover:prose-a:text-brand-pink prose-a:transition-colors prose-a:no-underline hover:prose-a:underline
|
|
prose-img:border prose-img:border-zinc-200 dark:prose-img:border-zinc-800
|
|
prose-blockquote:border-l-brand-pink prose-blockquote:bg-brand-pink/5 prose-blockquote:py-2 prose-blockquote:not-italic
|
|
prose-code:font-mono prose-code:text-brand-cyan prose-code:bg-zinc-100 dark:prose-code:bg-zinc-900 prose-code:px-1 prose-code:py-0.5 prose-code:rounded prose-code:before:content-none prose-code:after:content-none">
|
|
{{ page.body }}
|
|
{% article_json_ld page %}
|
|
</article>
|
|
|
|
<!-- Sidebar -->
|
|
<aside class="lg:col-span-4 space-y-8">
|
|
<div class="sticky top-28">
|
|
|
|
<!-- Share -->
|
|
<section aria-label="Share this article" class="mb-8">
|
|
<h3 class="font-display font-bold text-lg mb-4 uppercase tracking-widest text-zinc-500 text-sm">Share Article</h3>
|
|
<div class="flex gap-2">
|
|
<a href="https://x.com/intent/post?url={{ request.build_absolute_uri|urlencode }}&text={{ page.title|urlencode }}" target="_blank" rel="noopener noreferrer"
|
|
class="w-10 h-10 flex items-center justify-center bg-brand-surfaceLight dark:bg-brand-surfaceDark border border-zinc-200 dark:border-zinc-800 hover:border-brand-cyan transition-colors hover:text-brand-cyan" aria-label="Share on X">
|
|
<svg class="w-4 h-4" fill="currentColor" viewBox="0 0 24 24"><path d="M18.244 2.25h3.308l-7.227 8.26 8.502 11.24H16.17l-4.714-6.231-5.401 6.231H2.748l7.73-8.835L1.254 2.25H8.08l4.259 5.63L18.244 2.25zm-1.161 17.52h1.833L7.084 4.126H5.117L17.083 19.77z"/></svg>
|
|
</a>
|
|
<a href="https://www.linkedin.com/sharing/share-offsite/?url={{ request.build_absolute_uri|urlencode }}" target="_blank" rel="noopener noreferrer"
|
|
class="w-10 h-10 flex items-center justify-center bg-brand-surfaceLight dark:bg-brand-surfaceDark border border-zinc-200 dark:border-zinc-800 hover:border-blue-600 transition-colors hover:text-blue-600" aria-label="Share on LinkedIn">
|
|
<svg class="w-4 h-4" fill="currentColor" viewBox="0 0 24 24"><path d="M20.447 20.452h-3.554v-5.569c0-1.328-.027-3.037-1.852-3.037-1.853 0-2.136 1.445-2.136 2.939v5.667H9.351V9h3.414v1.561h.046c.477-.9 1.637-1.85 3.37-1.85 3.601 0 4.267 2.37 4.267 5.455v6.286zM5.337 7.433a2.062 2.062 0 0 1-2.063-2.065 2.064 2.064 0 1 1 2.063 2.065zm1.782 13.019H3.555V9h3.564v11.452zM22.225 0H1.771C.792 0 0 .774 0 1.729v20.542C0 23.227.792 24 1.771 24h20.451C23.2 24 24 23.227 24 22.271V1.729C24 .774 23.2 0 22.222 0h.003z"/></svg>
|
|
</a>
|
|
<button type="button" data-copy-link data-copy-url="{{ request.build_absolute_uri }}"
|
|
class="w-10 h-10 flex items-center justify-center bg-brand-surfaceLight dark:bg-brand-surfaceDark border border-zinc-200 dark:border-zinc-800 hover:border-brand-pink transition-colors hover:text-brand-pink" aria-label="Copy link">
|
|
<svg class="w-4 h-4" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="2" stroke="currentColor"><path stroke-linecap="round" stroke-linejoin="round" d="M13.19 8.688a4.5 4.5 0 0 1 1.242 7.244l-4.5 4.5a4.5 4.5 0 0 1-6.364-6.364l1.757-1.757m13.35-.622 1.757-1.757a4.5 4.5 0 0 0-6.364-6.364l-4.5 4.5a4.5 4.5 0 0 0 1.242 7.244" /></svg>
|
|
</button>
|
|
</div>
|
|
</section>
|
|
|
|
<!-- Newsletter -->
|
|
<div class="bg-brand-dark text-brand-light dark:bg-brand-light dark:text-brand-dark p-6 border border-transparent dark:border-zinc-700 shadow-solid-dark dark:shadow-solid-light">
|
|
<h3 class="font-display font-bold text-xl mb-2">Subscribe for Updates</h3>
|
|
<p class="text-sm opacity-80 mb-4">Get our latest articles and coding benchmarks delivered to your inbox every week.</p>
|
|
{% include 'components/newsletter_form.html' with source='article' label='Subscribe' %}
|
|
</div>
|
|
|
|
</div>
|
|
</aside>
|
|
|
|
</div>
|
|
|
|
<!-- Related Articles -->
|
|
{% if related_articles %}
|
|
<section class="mt-16 md:mt-24 pt-12 border-t border-zinc-200 dark:border-zinc-800">
|
|
<div class="flex items-center justify-between mb-8">
|
|
<h3 class="font-display font-bold text-3xl">Related Articles</h3>
|
|
<a href="/articles/" class="font-mono text-sm text-brand-cyan hover:underline">View All</a>
|
|
</div>
|
|
<div class="grid grid-cols-1 md:grid-cols-3 gap-8">
|
|
{% for article in related_articles %}
|
|
<article class="group flex flex-col h-full bg-brand-surfaceLight dark:bg-brand-surfaceDark border border-zinc-200 dark:border-zinc-800 hover:-translate-y-2 hover:shadow-solid-dark dark:hover:shadow-solid-light transition-all duration-300">
|
|
<a href="{{ article.url }}" class="h-48 overflow-hidden relative bg-zinc-900 flex items-center justify-center border-b border-zinc-200 dark:border-zinc-800 shrink-0 block">
|
|
{% if article.hero_image %}
|
|
{% image article.hero_image fill-400x300 class="w-full h-full object-cover grayscale group-hover:grayscale-0 transition-all duration-500" %}
|
|
{% else %}
|
|
<svg class="w-16 h-16 text-brand-cyan opacity-40 group-hover:opacity-80 transition-all duration-500 group-hover:scale-110" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1" stroke="currentColor"><path stroke-linecap="round" stroke-linejoin="round" d="M9.75 3.104v5.714a2.25 2.25 0 0 1-.659 1.591L5 14.5M9.75 3.104c-.251.023-.501.05-.75.082m.75-.082a24.301 24.301 0 0 1 4.5 0m0 0v5.714c0 .597.237 1.17.659 1.591L19.8 15.3M14.25 3.104c.251.023.501.05.75.082M19.8 15.3l-1.57.393A9.065 9.065 0 0 1 12 15a9.065 9.065 0 0 1-6.23-.693L5 14.5m14.8.8 1.402 1.402c1 1 .03 2.798-1.442 2.798H4.24c-1.47 0-2.44-1.798-1.442-2.798L4.2 15.3" /></svg>
|
|
{% endif %}
|
|
</a>
|
|
<div class="p-6 flex flex-col flex-grow">
|
|
<div class="flex gap-2 mb-3 flex-wrap">
|
|
{% for tag in article.tags.all %}
|
|
<span class="text-xs font-mono font-bold px-2 py-1 {{ tag|get_tag_css }}">{{ tag.name }}</span>
|
|
{% endfor %}
|
|
</div>
|
|
<a href="{{ article.url }}">
|
|
<h4 class="font-display font-bold text-xl mb-2 group-hover:text-brand-cyan transition-colors">{{ article.title }}</h4>
|
|
</a>
|
|
<p class="text-zinc-600 dark:text-zinc-400 text-sm mb-6 line-clamp-2">{{ article.summary }}</p>
|
|
<div class="mt-auto pt-4 border-t border-zinc-100 dark:border-zinc-800 flex items-center gap-2 text-sm font-bold font-mono group-hover:text-brand-cyan transition-colors">
|
|
Read Article
|
|
<svg class="w-4 h-4" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="2" stroke="currentColor"><path stroke-linecap="round" stroke-linejoin="round" d="m4.5 19.5 15-15m0 0H8.25m11.25 0v11.25" /></svg>
|
|
</div>
|
|
</div>
|
|
</article>
|
|
{% endfor %}
|
|
</div>
|
|
</section>
|
|
{% endif %}
|
|
|
|
<!-- Comments -->
|
|
{% if page.comments_enabled %}
|
|
<section class="mt-16 pt-12 border-t border-zinc-200 dark:border-zinc-800">
|
|
<h2 class="font-display font-bold text-3xl mb-8">Comments</h2>
|
|
|
|
{% if approved_comments %}
|
|
<div class="space-y-8 mb-12">
|
|
{% for comment in approved_comments %}
|
|
<article id="comment-{{ comment.id }}" class="bg-brand-surfaceLight dark:bg-brand-surfaceDark border border-zinc-200 dark:border-zinc-800 p-6">
|
|
<div class="flex items-center gap-3 mb-3">
|
|
<div class="w-8 h-8 bg-gradient-to-tr from-brand-cyan to-brand-pink shrink-0"></div>
|
|
<div>
|
|
<div class="font-display font-bold text-sm">{{ comment.author_name }}</div>
|
|
<div class="font-mono text-xs text-zinc-500">{{ comment.created_at|date:"M j, Y" }}</div>
|
|
</div>
|
|
</div>
|
|
<p class="text-zinc-700 dark:text-zinc-300 text-sm leading-relaxed">{{ comment.body }}</p>
|
|
{% for reply in comment.replies.all %}
|
|
<article id="comment-{{ reply.id }}" class="mt-6 ml-8 bg-zinc-50 dark:bg-zinc-900 border border-zinc-200 dark:border-zinc-800 p-4">
|
|
<div class="flex items-center gap-3 mb-2">
|
|
<div class="w-6 h-6 bg-gradient-to-tr from-brand-pink to-brand-cyan shrink-0"></div>
|
|
<div>
|
|
<div class="font-display font-bold text-sm">{{ reply.author_name }}</div>
|
|
<div class="font-mono text-xs text-zinc-500">{{ reply.created_at|date:"M j, Y" }}</div>
|
|
</div>
|
|
</div>
|
|
<p class="text-zinc-700 dark:text-zinc-300 text-sm leading-relaxed">{{ reply.body }}</p>
|
|
</article>
|
|
{% endfor %}
|
|
<form method="post" action="{% url 'comment_post' %}" class="mt-4 pt-4 border-t border-zinc-100 dark:border-zinc-800">
|
|
{% csrf_token %}
|
|
<input type="hidden" name="article_id" value="{{ page.id }}" />
|
|
<input type="hidden" name="parent_id" value="{{ comment.id }}" />
|
|
<div class="flex gap-3 mb-3">
|
|
<input type="text" name="author_name" required placeholder="Your name"
|
|
class="flex-1 bg-transparent border border-zinc-300 dark:border-zinc-700 px-3 py-2 font-mono text-sm focus:outline-none focus:border-brand-pink transition-colors" />
|
|
<input type="email" name="author_email" required placeholder="your@email.com"
|
|
class="flex-1 bg-transparent border border-zinc-300 dark:border-zinc-700 px-3 py-2 font-mono text-sm focus:outline-none focus:border-brand-pink transition-colors" />
|
|
</div>
|
|
<textarea name="body" required placeholder="Write a reply..." rows="2"
|
|
class="w-full bg-transparent border border-zinc-300 dark:border-zinc-700 px-3 py-2 font-mono text-sm focus:outline-none focus:border-brand-pink transition-colors mb-3 resize-none"></textarea>
|
|
<input type="text" name="honeypot" style="display:none" />
|
|
<button type="submit" class="px-4 py-2 bg-zinc-200 dark:bg-zinc-800 font-display font-bold text-sm hover:bg-brand-pink hover:text-white transition-colors">Reply</button>
|
|
</form>
|
|
</article>
|
|
{% endfor %}
|
|
</div>
|
|
{% else %}
|
|
<p class="font-mono text-sm text-zinc-500 mb-12">No comments yet. Be the first to comment.</p>
|
|
{% endif %}
|
|
|
|
{% if comment_form and comment_form.errors %}
|
|
<div aria-label="Comment form errors" class="mb-6 p-4 bg-red-50 dark:bg-red-900/20 border border-red-200 dark:border-red-800 font-mono text-sm text-red-600 dark:text-red-400">
|
|
{{ comment_form.non_field_errors }}
|
|
{% for field in comment_form %}{{ field.errors }}{% endfor %}
|
|
</div>
|
|
{% endif %}
|
|
|
|
<div class="bg-brand-surfaceLight dark:bg-brand-surfaceDark border border-zinc-200 dark:border-zinc-800 p-6">
|
|
<h3 class="font-display font-bold text-xl mb-6">Post a Comment</h3>
|
|
<form method="post" action="{% url 'comment_post' %}" class="space-y-4">
|
|
{% csrf_token %}
|
|
<input type="hidden" name="article_id" value="{{ page.id }}" />
|
|
<div class="grid grid-cols-1 md:grid-cols-2 gap-4">
|
|
<div>
|
|
<label class="block font-mono text-xs text-zinc-500 mb-1 uppercase tracking-wider">Name *</label>
|
|
<input type="text" name="author_name" value="{% if comment_form %}{{ comment_form.author_name.value|default:'' }}{% endif %}" required
|
|
class="w-full bg-transparent border border-zinc-300 dark:border-zinc-700 px-4 py-2 font-mono text-sm focus:outline-none focus:border-brand-pink transition-colors" />
|
|
</div>
|
|
<div>
|
|
<label class="block font-mono text-xs text-zinc-500 mb-1 uppercase tracking-wider">Email *</label>
|
|
<input type="email" name="author_email" value="{% if comment_form %}{{ comment_form.author_email.value|default:'' }}{% endif %}" required
|
|
class="w-full bg-transparent border border-zinc-300 dark:border-zinc-700 px-4 py-2 font-mono text-sm focus:outline-none focus:border-brand-pink transition-colors" />
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<label class="block font-mono text-xs text-zinc-500 mb-1 uppercase tracking-wider">Comment *</label>
|
|
<textarea name="body" required rows="5"
|
|
class="w-full bg-transparent border border-zinc-300 dark:border-zinc-700 px-4 py-2 font-mono text-sm focus:outline-none focus:border-brand-pink transition-colors resize-none">{% if comment_form %}{{ comment_form.body.value|default:'' }}{% endif %}</textarea>
|
|
</div>
|
|
<input type="text" name="honeypot" style="display:none" />
|
|
<button type="submit" class="px-6 py-3 bg-brand-dark text-brand-light dark:bg-brand-light dark:text-brand-dark font-display font-bold hover:-translate-y-1 hover:shadow-solid-dark dark:hover:shadow-solid-light transition-all">Post comment</button>
|
|
</form>
|
|
</div>
|
|
</section>
|
|
{% endif %}
|
|
{% endblock %}
|