Loading spinner
Dan Edwards Developer icon

Dan Edwards developer

21 March 2025Live siteGitHub

ClassicReader.org

ClassicReader.org

ClassicReader.org is a 1,450-page statically generated site for reading classic literature in the browser.

Motivation

There are plenty of websites where you can read classic books for free. However, I haven’t found any that offer a pleasant reading experience.

The internet is an incredible and openly documented technology, but people often use it for aggressive marketing, user manipulation, and privacy invasion.

I wanted to make a genuinely lovely project with a wholesome purpose. As anyone can edit the books on GitHub, it has a community aspect, too.

Also, I find static site generation inherently satisfying. So, I wanted to make a site with at least 1,000 pages as an arbitrary personal goal, all while demonstrating the power of Next.js to create a high-performance search engine-optimised site.

Technology

  • Next.js, app router
    • 1,450 statically generated pages using generateStaticParams
    • Here’s a fantastic introduction to Next.js if you’re unfamiliar with this incredible framework
app/writers/[writer]/novels/[novel]/[chapter]/page.tsx
TypeScript
1// This function creates all 1,450 URL combinations so each page can be generated statically at build time
2						
3export async function generateStaticParams(): StaticParams {
4	const novels = await getAllNovels()
5
6	return novels.flatMap((novelData) =>
7		novelData.chapters.map((_chapter, index) => ({
8			writer: novelData.writerSlug,
9			novel: novelData.titleSlug,
10			chapter: generateChapterParam({
11				novelData,
12				oneIndexedChapterNumber: index + 1,
13			}),
14		})),
15	)
16}
  • Tailwind CSS
  • Simple Analytics
    • Privacy-first data collection that doesn’t use cookies
    • Provides valuable data insights without invading anyone’s privacy
    • Doesn’t require a cookie consent banner
  • epub Node package
    • I can add a new novel in about 90 seconds
    • I simply add the title and writer information to my as const enums, run my parsing script, then delete any ‘chapters’ that are just copyright information or dedications
  • Deployed with Vercel

Features

  • Excellent reading experience
  • Responsive design
ClassicReader.org has a responsive layout
ClassicReader.org is optimised for search engines
  • Perfect Lighthouse scores
    • Exceptional largest contentful paint of 0.6 seconds
    • No cumulative layout shift
ClassicReader.org has perfect scores on Google Lighthouse

Challenges

Bundle size & deployment

I initially created a local storage system so that users could track their progress - it would mark a chapter as read after pressing the next chapter button. The novel overview page allowed users to manage the overall progress.

It was quite an elegant solution; however, it made the statically generated site huge - around 5GB - and would have required a custom hosting solution, as the three platforms I have used most - Fly.io, Vercel and Hostinger - don’t allow sites of this size without custom support.

Solution

I decided to scrap the book progress system, as content library size and SEO are more important for this project.

Also, I’m not even sure that anyone would have used the feature, and it’s easy enough to bookmark the URL and keep replacing it as you read.

Navigation design

Initially, I wanted to lay the page like a traditional book, with the title and chapter in all caps at the top.

A classic example of a printed novel layout, with the chapter name and novel title in all caps

However, I recently read three books - Universal Principles of UX, Letting Go of the Words, and Building a Story Brand - that all stressed the importance of straightforward navigation, albeit from different angles.

100 Universal UX Principals also emphasises that many users never see the homepage, so it's vital that every page on the site explains the information architecture clearly.

Solution

Instead of my print-inspired menu idea, I went for breadcrumbs, which act as a “You are here” marker and navigation system while remaining minimal, elegant and unobtrusive.

ClassicReader.org has a clear but unobtrusive navigation using breadcrumbs.

Reception

The site has been a success, getting surprising traffic already - currently over 500 page views per day after going live only a week ago.

It’s hard to say what’s driving the traffic. However, it’s probably a combination of all these:

  • Content that people love
  • Optimised metadata
  • Descriptive URLs
  • Great UX

Almost everyone I’ve shown the site has said they love the font; however, it’s just the system font. I suspect that people are actually enjoying these instead:

  • Generous line height
  • Restricted print width

These design aspects significantly enhance readability while operating on a less conscious level, allowing the more recognisable element - the font - to receive all the credit.

Plans for the future

  • Add more literature
  • Add support for poems and plays
  • Get to 10,000 pages by the end of 2025

Conclusion

Developing ClassicReader has been immensely satisfying and demonstrates the power of static generation, even though it hardly scratches the surface of what Next.js can do.

I've achieved everything I set out to do:

  • Enabled a pleasant browser-based reading experience
  • Achieved perfect Lighthouse scores
  • Created 1,000+ static pages

I’m delighted with how this project has turned out and look forward to nurturing and expanding it.