middleware.ts 1.3 KB

12345678910111213141516171819202122232425262728293031323334353637383940
  1. import { match } from '@formatjs/intl-localematcher'
  2. import Negotiator from 'negotiator'
  3. import { NextResponse } from 'next/server'
  4. import type { NextRequest } from 'next/server'
  5. import type { Locale } from './i18n'
  6. import { i18n } from './i18n'
  7. export const getLocale = (request: NextRequest): Locale => {
  8. // @ts-expect-error locales are readonly
  9. const locales: Locale[] = i18n.locales
  10. let languages: string[] | undefined
  11. // get locale from cookie
  12. const localeCookie = request.cookies.get('locale')
  13. languages = localeCookie?.value ? [localeCookie.value] : []
  14. if (!languages.length) {
  15. // Negotiator expects plain object so we need to transform headers
  16. const negotiatorHeaders: Record<string, string> = {}
  17. request.headers.forEach((value, key) => (negotiatorHeaders[key] = value))
  18. // Use negotiator and intl-localematcher to get best locale
  19. languages = new Negotiator({ headers: negotiatorHeaders }).languages()
  20. }
  21. // match locale
  22. const matchedLocale = match(languages, locales, i18n.defaultLocale) as Locale
  23. return matchedLocale
  24. }
  25. export const middleware = async (request: NextRequest) => {
  26. const pathname = request.nextUrl.pathname
  27. if (/\.(css|js(on)?|ico|svg|png)$/.test(pathname))
  28. return
  29. const locale = getLocale(request)
  30. const response = NextResponse.next()
  31. response.cookies.set('locale', locale)
  32. return response
  33. }