Skip to content

Commit 4cf1e9a

Browse files
committed
feat: migrate to App Router architecture
1 parent ccb53bb commit 4cf1e9a

7 files changed

+158
-9
lines changed

next-env.d.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
/// <reference types="next" />
22
/// <reference types="next/image-types/global" />
3+
/// <reference types="next/navigation-types/compat/navigation" />
34

45
// NOTE: This file should not be edited
5-
// see https://nextjs.org/docs/pages/api-reference/config/typescript for more information.
6+
// see https://nextjs.org/docs/app/api-reference/config/typescript for more information.

next.config.js

+6-8
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,14 @@
11
/** @type {import('next').NextConfig} */
22
const nextConfig = {
3-
// Next.js 15 now uses app directory by default
4-
// We're keeping pages directory for backward compatibility
5-
// but can migrate to app directory later
3+
// Explicitly enable the App Router
4+
experimental: {
5+
appDir: true
6+
},
67
eslint: {
78
ignoreDuringBuilds: true,
89
},
9-
// Modern i18n config
10-
i18n: {
11-
locales: ['fr'],
12-
defaultLocale: 'fr',
13-
}
10+
// Replace legacy i18n config with newer approach
11+
// (i18n is now handled in app/[locale] structure or middleware)
1412
}
1513

1614
module.exports = nextConfig

src/app/fr/layout.tsx

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
import { getLoggedUserId } from '../../utils/getLoggedUserId'
2+
3+
// Default way to get a logged user
4+
export const loggedUserId = getLoggedUserId()
5+
6+
export default function FrLayout({
7+
children,
8+
}: {
9+
children: React.ReactNode
10+
}) {
11+
return (
12+
<div className="min-h-screen p-2 flex flex-col justify-center items-center">
13+
{children}
14+
</div>
15+
)
16+
}

src/app/fr/page.tsx

+57
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,57 @@
1+
import Image from 'next/image'
2+
import Logo from '../assets/lbc-logo.webp'
3+
4+
export default function Home() {
5+
const year = new Date().getFullYear()
6+
7+
return (
8+
<>
9+
<main className="p-4 md:p-16 flex-1 flex flex-col justify-center items-center">
10+
<Image src={Logo} alt="Leboncoin Frontend Team" width={400} height={125} priority />
11+
<h1 className="mt-8 mb-4 text-4xl font-bold text-center">
12+
Welcome !
13+
</h1>
14+
15+
<p className="text-xl leading-relaxed text-center">
16+
This test is based on a <a className="text-black underline hover:text-[--color-leboncoin-orange]" title="Next.js documentation" href="https://nextjs.org/docs/getting-started" target="_blank" rel="noopener noreferrer">Next.js</a> application.<br />
17+
Fork the repository and use the <code className="bg-gray-200 rounded px-1 py-0.5 text-base font-mono">main</code> branch as your starting point.
18+
<br /><br />
19+
20+
Get started by reading{' '}
21+
<code className="bg-gray-200 rounded px-1 py-0.5 text-base font-mono">README.md</code> and editing <code className="bg-gray-200 rounded px-1 py-0.5 text-base font-mono">src/app/page.tsx</code>
22+
<br />
23+
Once you are done, send the repository link to your HR contact.
24+
</p>
25+
26+
<div className="flex flex-wrap justify-center items-center max-w-4xl mt-4 w-full">
27+
<article className="m-2 flex-[45%] p-6 text-left border border-gray-200 rounded-lg transition-colors hover:border-leboncoin-orange hover:text-leboncoin-orange">
28+
<h2 className="mb-4 text-2xl font-bold">Design</h2>
29+
<p className="text-lg leading-relaxed">Feel free to create any design you want for this exercise. Let your creativity talks !</p>
30+
</article>
31+
32+
<article className="m-2 flex-[45%] p-6 text-left border border-gray-200 rounded-lg transition-colors hover:border-leboncoin-orange hover:text-leboncoin-orange">
33+
<h2 className="mb-4 text-2xl font-bold">Libraries</h2>
34+
<p className="text-lg leading-relaxed">Feel free to use any library you want. Only Next.js / React are required.</p>
35+
</article>
36+
37+
<article className="m-2 flex-[45%] p-6 text-left border border-gray-200 rounded-lg transition-colors hover:border-leboncoin-orange hover:text-leboncoin-orange">
38+
<h2 className="mb-4 text-2xl font-bold">API Server</h2>
39+
<p className="text-lg leading-relaxed">
40+
Start the API server on port <code className="bg-gray-200 rounded px-1 py-0.5 text-base font-mono">3005</code> by running<br /><code className="bg-gray-200 rounded px-1 py-0.5 text-base font-mono">npm run start-server</code>.<br/>
41+
Find the swagger definitions in <code className="bg-gray-200 rounded px-1 py-0.5 text-base font-mono">docs/api-swagger.yml</code> or <a className="text-black underline hover:text-[--color-leboncoin-orange]" title="API Swagger documentation" href="https://leboncoin.tech/frontend-technical-test/" target="_blank" rel="noopener noreferrer">the online documentation</a>.
42+
</p>
43+
</article>
44+
45+
<article className="m-2 flex-[45%] p-6 text-left border border-gray-200 rounded-lg transition-colors hover:border-leboncoin-orange hover:text-leboncoin-orange">
46+
<h2 className="mb-4 text-2xl font-bold">Timing</h2>
47+
<p className="text-lg leading-relaxed">We recommend 4 hours for this test. You are free to spend more (or less) time, let us know how much time did you spend.</p>
48+
</article>
49+
</div>
50+
</main>
51+
52+
<footer className="w-full h-[50px] border-t border-gray-200 flex justify-center items-center">
53+
&copy; leboncoin - {year}
54+
</footer>
55+
</>
56+
)
57+
}

src/app/layout.tsx

+24
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
import '../styles/globals.css'
2+
import type { Metadata } from 'next'
3+
4+
export const metadata: Metadata = {
5+
title: {
6+
default: 'Frontend Technical test - Leboncoin',
7+
template: '%s | Leboncoin',
8+
},
9+
description: 'Frontend exercise for developpers who want to join us on leboncoin.fr',
10+
}
11+
12+
export default function RootLayout({
13+
children,
14+
}: {
15+
children: React.ReactNode
16+
}) {
17+
return (
18+
<html lang="fr">
19+
<body>
20+
{children}
21+
</body>
22+
</html>
23+
)
24+
}

src/app/not-found.tsx

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
import Link from 'next/link'
2+
3+
export default function NotFound() {
4+
return (
5+
<div className="min-h-screen flex flex-col items-center justify-center">
6+
<h2 className="text-4xl font-bold mb-4">Page not found</h2>
7+
<p className="text-lg mb-6">The page you are looking for does not exist.</p>
8+
<Link href="/fr" className="px-6 py-3 bg-[--color-leboncoin-orange] text-white rounded hover:opacity-90">
9+
Return to home page
10+
</Link>
11+
</div>
12+
)
13+
}

src/middleware.ts

+40
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
import { NextResponse } from 'next/server'
2+
import type { NextRequest } from 'next/server'
3+
4+
// This middleware handles internationalization for the App Router
5+
// It sets the default locale to 'fr'
6+
export function middleware(request: NextRequest) {
7+
// Get the pathname from the request
8+
const pathname = request.nextUrl.pathname
9+
10+
// Set locale to 'fr' (the default and only supported locale)
11+
const locale = 'fr'
12+
13+
// Skip internationalization middleware for API routes and static assets
14+
if (
15+
pathname.startsWith('/api') ||
16+
pathname.startsWith('/_next') ||
17+
pathname.includes('/.')
18+
) {
19+
return NextResponse.next()
20+
}
21+
22+
// The locale is already in the pathname, skip
23+
if (pathname.startsWith(`/${locale}`)) {
24+
return NextResponse.next()
25+
}
26+
27+
// Redirect to the path with the locale prefix (e.g. /fr/about)
28+
// In this case, since 'fr' is the only locale, we always redirect to /fr/...
29+
return NextResponse.redirect(
30+
new URL(`/${locale}${pathname}`, request.url)
31+
)
32+
}
33+
34+
// Configure the middleware to run on specific paths
35+
export const config = {
36+
// Match all routes except for api routes, static files, etc.
37+
matcher: [
38+
'/((?!api|_next/static|_next/image|favicon.ico|.*\\..*|_vercel).*)',
39+
],
40+
}

0 commit comments

Comments
 (0)