Skip to content

Commit b4de1a0

Browse files
committed
Merge remote-tracking branch 'origin/main' into canary
2 parents b44c8ef + 1f44136 commit b4de1a0

File tree

8 files changed

+93
-45
lines changed

8 files changed

+93
-45
lines changed

.github/ISSUE_TEMPLATE/02_feature_request.yml

+10-2
Original file line numberDiff line numberDiff line change
@@ -6,10 +6,18 @@ labels:
66
body:
77
- type: markdown
88
attributes:
9-
value: Thanks for taking the time to file a feature request! Please fill out this form completely.
9+
value: Thanks for taking the time to improve next-intl!
1010
- type: markdown
1111
attributes:
12-
value: Please keep in mind that if something doesn't work as expected, you might want to [create a bug report](https://github.com/amannn/next-intl/issues/new/choose) instead. Bug reports disguised as feature requests will be closed.
12+
value: |
13+
Please keep in mind:
14+
15+
- If you're looking for specific support with your project, you can consider [asking for community support](https://github.com/amannn/next-intl/issues/new/choose) instead.
16+
- If something doesn't work as expected, you might want to [create a bug report](https://github.com/amannn/next-intl/issues/new/choose) instead.
17+
- Please double check the [docs](https://next-intl.dev) to see if your feature request is already supported.
18+
- Please also check the [existing issues](https://github.com/amannn/next-intl/issues) to see if your feature request was already discussed.
19+
20+
If you have a feature request that has not been discussed yet, please fill out this form completely.
1321
- type: textarea
1422
attributes:
1523
label: Is your feature request related to a problem? Please describe.

.github/ISSUE_TEMPLATE/config.yml

+8-2
Original file line numberDiff line numberDiff line change
@@ -3,9 +3,15 @@ contact_links:
33
- name: View documentation
44
url: https://next-intl.dev
55
about: Check out the official docs for answers to common questions
6-
- name: Get community support
6+
- name: Get community support on the Next.js Discord
7+
url: https://discord.com/invite/nextjs
8+
about: Join the Next.js community on Discord to get real-time help and connect with other developers
9+
- name: Get community support on Stack Overflow
710
url: https://stackoverflow.com/questions/ask
8-
about: Ask questions and discuss with other community members
11+
about: Post your technical questions on Stack Overflow where experienced developers can help
12+
- name: Get community support on GitHub Discussions
13+
url: https://github.com/amannn/next-intl/discussions
14+
about: Start a discussion to share ideas and ask questions
915
- name: Get maintainer support
1016
url: https://github.com/sponsors/amannn?frequency=one-time
1117
about: Book a consulting session to get help with your specific project

docs/src/pages/blog/next-intl-4-0.mdx

+31-8
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,8 @@ Here's what's new in `next-intl@4.0`:
1818
3. [**Strictly-typed ICU arguments**](#strictly-typed-icu-arguments)
1919
4. [**GDPR compliance**](#gdpr-compliance)
2020
5. [**Modernized build output**](#modernized-build-output)
21-
6. [**Preparation for upcoming Next.js features**](#nextjs-future)
21+
6. [**Improved inheritance in `NextIntlClientProvider`**](#nextintlclientprovider-inheritance)
22+
7. [**Preparation for upcoming Next.js features**](#nextjs-future)
2223

2324
Please also have a look at the [other breaking changes](#other-breaking-changes) listed below before you upgrade.
2425

@@ -188,6 +189,30 @@ The build output of `next-intl` has been modernized and now leverages the follow
188189

189190
With these changes, the bundle size of `next-intl` has been reduced by ~7% ([all details](https://github.com/amannn/next-intl/pull/1470)).
190191

192+
## Improved inheritance of `NextIntlClientProvider` [#nextintlclientprovider-inheritance]
193+
194+
Previously, [`NextIntlClientProvider`](/docs/usage/configuration#nextintlclientprovider) would conservatively inherit only a subset from `i18n/request.ts`.
195+
196+
To improve the getting started experience, the provider now also inherits:
197+
198+
- `messages` ([PR #1682](https://github.com/amannn/next-intl/pull/1682))
199+
- `formats` ([PR #1191](https://github.com/amannn/next-intl/pull/1191))
200+
201+
Due to this, you can now remove these props from `NextIntlClientProvider` if you've previously passed them manually:
202+
203+
```diff
204+
<NextIntlClientProvider
205+
- messages={messages}
206+
- formats={formats}
207+
>
208+
{/* ... */}
209+
</NextIntlClientProvider>
210+
```
211+
212+
With this, `NextIntlClientProvider` now inherits all of your configuration, with the minor exception of [error handling functions](/docs/usage/configuration#error-handling). Since functions are not serializable, they cannot be passed across the server/client boundary. However, [an alternative](https://github.com/amannn/next-intl/issues/1285) for this is also on the horizon.
213+
214+
To make it easier to work with error handling functions on the client side, `NextIntlClientProvider` can now also be used in a nested fashion and will inherit the configuration from a parent provider ([PR #1413](https://github.com/amannn/next-intl/pull/1413)).
215+
191216
## Preparation for upcoming Next.js features [#nextjs-future]
192217

193218
To ensure that the sails of `next-intl` are set for a steady course in the upcoming future, I've investigated the implications of upcoming Next.js features like [`ppr`](https://nextjs.org/docs/app/api-reference/next-config-js/ppr), [`dynamicIO`](https://nextjs.org/docs/canary/app/api-reference/config/next-config-js/dynamicIO) and [`rootParams`](https://github.com/vercel/next.js/pull/72837) for `next-intl`.
@@ -205,13 +230,11 @@ I'm particularly excited about the announcement of `rootParams`, as it seems lik
205230
## Other breaking changes
206231

207232
1. Return type-safe messages from `useMessages` and `getMessages` (see [PR #1489](https://github.com/amannn/next-intl/pull/1489))
208-
2. Inherit context in case nested `NextIntlClientProvider` instances are present (see [PR #1413](https://github.com/amannn/next-intl/pull/1413))
209-
3. Automatically inherit formats when `NextIntlClientProvider` is rendered from a Server Component (see [PR #1191](https://github.com/amannn/next-intl/pull/1191))
210-
4. Require locale to be returned from `getRequestConfig` (see [PR #1486](https://github.com/amannn/next-intl/pull/1486))
211-
5. Disallow passing `null`, `undefined` or `boolean` as an ICU argument (see [PR #1561](https://github.com/amannn/next-intl/pull/1561))
212-
6. Bump minimum required TypeScript version to 5 for projects using TypeScript (see [PR #1481](https://github.com/amannn/next-intl/pull/1481))
213-
7. Remove deprecated APIs (see [PR #1479](https://github.com/amannn/next-intl/pull/1479))
214-
8. Remove deprecated APIs pt. 2 (see [PR #1482](https://github.com/amannn/next-intl/pull/1482))
233+
2. Require locale to be returned from `getRequestConfig` (see [PR #1486](https://github.com/amannn/next-intl/pull/1486))
234+
3. Disallow passing `null`, `undefined` or `boolean` as an ICU argument (see [PR #1561](https://github.com/amannn/next-intl/pull/1561))
235+
4. Bump minimum required TypeScript version to 5 for projects using TypeScript (see [PR #1481](https://github.com/amannn/next-intl/pull/1481))
236+
5. Remove deprecated APIs (see [PR #1479](https://github.com/amannn/next-intl/pull/1479))
237+
6. Remove deprecated APIs pt. 2 (see [PR #1482](https://github.com/amannn/next-intl/pull/1482))
215238

216239
## Upgrade now
217240

docs/src/pages/docs/environments/actions-metadata-route-handlers.mdx

+26-25
Original file line numberDiff line numberDiff line change
@@ -80,22 +80,18 @@ async function loginAction(data: FormData) {
8080
const t = await getTranslations('LoginForm');
8181
const values = Object.fromEntries(data);
8282

83-
const result = await loginFormSchema
84-
.refine(async (credentials) => loginUser(credentials), {
85-
message: t('invalidCredentials')
86-
})
87-
.safeParseAsync(values, {
88-
errorMap(issue, ctx) {
89-
const path = issue.path.join('.');
90-
91-
const message = {
92-
email: t('invalidEmail')
93-
password: t('invalidPassword')
94-
}[path];
95-
96-
return {message: message || ctx.defaultError};
97-
}
98-
});
83+
const result = loginFormSchema.safeParse(values, {
84+
errorMap(issue, ctx) {
85+
const path = issue.path.join('.');
86+
87+
const message = {
88+
email: t('invalidEmail')
89+
password: t('invalidPassword')
90+
}[path];
91+
92+
return {message: message || ctx.defaultError};
93+
}
94+
});
9995

10096
// ...
10197
}
@@ -173,7 +169,7 @@ If you're using a sitemap to inform search engines about all pages of your site,
173169

174170
Note that by default, `next-intl` returns [the `link` response header](/docs/routing#alternate-links) to instruct search engines that a page is available in multiple languages. While this sufficiently links localized pages for search engines, you may choose to provide this information in a sitemap in case you have more specific requirements.
175171

176-
Next.js supports providing alternate URLs per language via the [`alternates` entry](https://nextjs.org/docs/app/api-reference/file-conventions/metadata/sitemap#generate-a-localized-sitemap) as of version 14.2. You can use your default locale for the main URL and provide alternate URLs based on all locales that your app supports. Keep in mind that also the default locale should be included in the `alternates` object.
172+
Next.js supports providing alternate URLs per language via the [`alternates` entry](https://nextjs.org/docs/app/api-reference/file-conventions/metadata/sitemap#generate-a-localized-sitemap). You can construct a list of entries for each pathname and locale as follows:
177173

178174
```tsx filename="app/sitemap.ts" {4-5,8-9}
179175
import {MetadataRoute} from 'next';
@@ -184,20 +180,20 @@ const host = 'https://acme.com';
184180

185181
export default function sitemap(): MetadataRoute.Sitemap {
186182
// Adapt this as necessary
187-
return [getEntry('/'), getEntry('/users')];
183+
return [...getEntries('/'), ...getEntries('/users')];
188184
}
189185

190186
type Href = Parameters<typeof getPathname>[0]['href'];
191187

192-
function getEntry(href: Href) {
193-
return {
194-
url: getUrl(href, routing.defaultLocale),
188+
function getEntries(href: Href) {
189+
return routing.locales.map((locale) => ({
190+
url: getUrl(href, locale),
195191
alternates: {
196192
languages: Object.fromEntries(
197-
routing.locales.map((locale) => [locale, getUrl(href, locale)])
193+
routing.locales.map((cur) => [cur, getUrl(href, cur)])
198194
)
199195
}
200-
};
196+
}));
201197
}
202198

203199
function getUrl(href: Href, locale: (typeof routing.locales)[number]) {
@@ -210,15 +206,20 @@ Depending on if you're using the [`pathnames`](/docs/routing#pathnames) setting,
210206

211207
```tsx
212208
// 1. A final string (when not using `pathnames`)
213-
getEntry('/users/1');
209+
getEntries('/users/1');
214210

215211
// 2. An object (when using `pathnames`)
216-
getEntry({
212+
getEntries({
217213
pathname: '/users/[id]',
218214
params: {id: '1'}
219215
});
220216
```
221217

218+
**Keep in mind**:
219+
220+
- Each pathname should have a separate entry for every locale that your app supports.
221+
- Also the locale of a given pathname should be included in the `alternates` object.
222+
222223
([working implementation](https://github.com/amannn/next-intl/blob/main/examples/example-app-router/src/app/sitemap.ts))
223224

224225
### Route Handlers

docs/src/pages/docs/usage/configuration.mdx

+1-1
Original file line numberDiff line numberDiff line change
@@ -190,7 +190,7 @@ export default async function RootLayout({children}) {
190190
messages={messages}
191191
>
192192
{children}
193-
</NextIntlClientProvider>
193+
</IntlProvider>
194194
</body>
195195
</html>
196196
);

examples/example-app-router/src/app/sitemap.ts

+6-6
Original file line numberDiff line numberDiff line change
@@ -3,20 +3,20 @@ import {host} from '@/config';
33
import {Locale, getPathname, routing} from '@/i18n/routing';
44

55
export default function sitemap(): MetadataRoute.Sitemap {
6-
return [getEntry('/'), getEntry('/pathnames')];
6+
return [...getEntries('/'), ...getEntries('/pathnames')];
77
}
88

99
type Href = Parameters<typeof getPathname>[0]['href'];
1010

11-
function getEntry(href: Href) {
12-
return {
13-
url: getUrl(href, routing.defaultLocale),
11+
function getEntries(href: Href) {
12+
return routing.locales.map((locale) => ({
13+
url: getUrl(href, locale),
1414
alternates: {
1515
languages: Object.fromEntries(
16-
routing.locales.map((locale) => [locale, getUrl(href, locale)])
16+
routing.locales.map((cur) => [cur, getUrl(href, cur)])
1717
)
1818
}
19-
};
19+
}));
2020
}
2121

2222
function getUrl(href: Href, locale: Locale) {

examples/example-app-router/tests/main.spec.ts

+11-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import {test as it, expect} from '@playwright/test';
1+
import {expect, test as it} from '@playwright/test';
22

33
it('handles i18n routing', async ({page}) => {
44
await page.goto('/');
@@ -111,10 +111,20 @@ it('serves a sitemap.xml', async ({page}) => {
111111
<xhtml:link rel="alternate" hreflang="de" href="http://localhost:3000/de" />
112112
</url>
113113
<url>
114+
<loc>http://localhost:3000/de</loc>
115+
<xhtml:link rel="alternate" hreflang="en" href="http://localhost:3000/en" />
116+
<xhtml:link rel="alternate" hreflang="de" href="http://localhost:3000/de" />
117+
</url>
118+
<url>
114119
<loc>http://localhost:3000/en/pathnames</loc>
115120
<xhtml:link rel="alternate" hreflang="en" href="http://localhost:3000/en/pathnames" />
116121
<xhtml:link rel="alternate" hreflang="de" href="http://localhost:3000/de/pfadnamen" />
117122
</url>
123+
<url>
124+
<loc>http://localhost:3000/de/pfadnamen</loc>
125+
<xhtml:link rel="alternate" hreflang="en" href="http://localhost:3000/en/pathnames" />
126+
<xhtml:link rel="alternate" hreflang="de" href="http://localhost:3000/de/pfadnamen" />
127+
</url>
118128
</urlset>
119129
`
120130
);

media/assets.sketch

32.9 KB
Binary file not shown.

0 commit comments

Comments
 (0)