feat(analytics): add blog engagement, ToC tracking, and 404 monitoring
Some checks failed
Build & Deploy / 🔍 Prepare (push) Successful in 6s
Build & Deploy / 🧪 QA (push) Failing after 1m13s
Build & Deploy / 🏗️ Build (push) Has been skipped
Build & Deploy / 🚀 Deploy (push) Has been skipped
Build & Deploy / 🧪 Smoke Test (push) Has been skipped
Build & Deploy / 🔔 Notify (push) Successful in 2s

- Added BlogEngagementTracker for reading time and completion tracking
- Added ToC click tracking in blog posts
- Added global 404 error monitoring in not-found.tsx
- Completed 'Total Transparency' suite
This commit is contained in:
2026-02-16 18:31:28 +01:00
parent 0e089f9471
commit c0f5799667
4 changed files with 88 additions and 11 deletions

View File

@@ -0,0 +1,53 @@
'use client';
import { useEffect } from 'react';
import { useAnalytics } from './useAnalytics';
import { AnalyticsEvents } from './analytics-events';
interface BlogEngagementTrackerProps {
title: string;
slug: string;
category?: string;
readingTime: number;
}
/**
* BlogEngagementTracker
* Tracks reading time and article completion.
*/
export default function BlogEngagementTracker({
title,
slug,
category,
readingTime,
}: BlogEngagementTrackerProps) {
const { trackEvent } = useAnalytics();
useEffect(() => {
// Article start
trackEvent(AnalyticsEvents.BLOG_POST_VIEW, {
title,
slug,
category,
estimated_reading_time: readingTime,
location: 'blog_post_pdp',
});
const startTime = Date.now();
return () => {
const dwellTime = Math.round((Date.now() - startTime) / 1000);
// We only consider it a "read" if they stay a reasonable amount of time
// or if they scroll (covered by ScrollDepthTracker)
trackEvent('blog_dwell_time', {
title,
slug,
seconds: dwellTime,
reading_time_completion: Math.min(100, Math.round((dwellTime / (readingTime * 60)) * 100)),
});
};
}, [title, slug, category, readingTime, trackEvent]);
return null;
}

View File

@@ -2,6 +2,8 @@
import React, { useEffect, useState } from 'react';
import { cn } from '@/components/ui/utils';
import { useAnalytics } from '../analytics/useAnalytics';
import { AnalyticsEvents } from '../analytics/analytics-events';
interface TocItem {
id: string;
@@ -16,11 +18,12 @@ interface TableOfContentsProps {
export default function TableOfContents({ headings, locale }: TableOfContentsProps) {
const [activeId, setActiveId] = useState<string>('');
const { trackEvent } = useAnalytics();
useEffect(() => {
const observerOptions = {
rootMargin: '-10% 0% -70% 0%',
threshold: 0
threshold: 0,
};
const observer = new IntersectionObserver((entries) => {
@@ -66,15 +69,20 @@ export default function TableOfContents({ headings, locale }: TableOfContentsPro
<a
href={`#${heading.id}`}
className={cn(
"text-sm md:text-base transition-all duration-300 hover:text-primary block leading-snug",
'text-sm md:text-base transition-all duration-300 hover:text-primary block leading-snug',
activeId === heading.id
? "text-primary font-bold translate-x-1"
: "text-text-secondary font-medium hover:translate-x-1"
? 'text-primary font-bold translate-x-1'
: 'text-text-secondary font-medium hover:translate-x-1',
)}
onClick={(e) => {
e.preventDefault();
const element = document.getElementById(heading.id);
if (element) {
trackEvent(AnalyticsEvents.TOC_CLICK, {
heading_id: heading.id,
heading_text: heading.text,
location: 'blog_sidebar',
});
const yOffset = -100;
const y = element.getBoundingClientRect().top + window.pageYOffset + yOffset;
window.scrollTo({ top: y, behavior: 'smooth' });