-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmiddleware.ts
44 lines (38 loc) · 1.49 KB
/
middleware.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
import { match } from '@formatjs/intl-localematcher'
import Negotiator from 'negotiator'
import { NextResponse } from 'next/server'
import type { NextRequest } from 'next/server'
import type { Locale } from './i18n'
import { i18n } from './i18n'
export const getLocale = (request: NextRequest): Locale => {
// @ts-expect-error locales are readonly
const locales: Locale[] = i18n.locales
let languages: string[] | undefined
// get locale from cookie
const localeCookie = request.cookies.get('locale')
languages = localeCookie?.value ? [localeCookie.value] : []
if (!languages.length) {
// Negotiator expects plain object so we need to transform headers
const negotiatorHeaders: Record<string, string> = {}
request.headers.forEach((value, key) => (negotiatorHeaders[key] = value))
// Use negotiator and intl-localematcher to get best locale
languages = new Negotiator({ headers: negotiatorHeaders }).languages()
}
// match locale
let matchedLocale: Locale = i18n.defaultLocale
try {
// If languages is ['*'], Error would happen in match function.
matchedLocale = match(languages, locales, i18n.defaultLocale) as Locale
}
catch (e) {}
return matchedLocale
}
export const middleware = async (request: NextRequest) => {
const pathname = request.nextUrl.pathname
if (/\.(css|js(on)?|ico|svg|png)$/.test(pathname))
return
const locale = getLocale(request)
const response = NextResponse.next()
response.cookies.set('locale', locale)
return response
}