Modern.js provides out-of-the-box data fetching capabilities, allowing developers to develop in an isomorphic way in both client-side and server-side code.
It should be noted that these APIs do not help applications initiate requests, but rather help developers better manage data and improve project performance.
Modern.js recommends using conventional routing for routing management. Through Modern.js's conventional (nested) routing, each routing component (layout.ts
or page.ts
) can have a same-named loader
file. The loader
file needs to export a function that will be executed before the component is rendered to provide data for the routing component.
Modern.js v1 supports fetching data via useLoader, which is no longer the recommended usage. We do not recommend mixing the two except during the migration process.
Routing components such as layout.ts
or page.ts
can define a same-named loader
file. The function exported by the loader
file provides the data required by the component, and then the data is obtained in the routing component through the useLoaderData
function, as shown in the following example:
.
└── routes
├── layout.tsx
└── user
├── layout.tsx
├── layout.loader.ts
├── page.tsx
└── page.loader.ts
Define the following code in the file:
import { useLoaderData } from '@modern-js/runtime/router';
import type { ProfileData } from './page.loader.ts';
export default function UserPage() {
const profileData = useLoaderData() as ProfileData;
return <div>{profileData}</div>;
}
export type ProfileData = {
/* some types */
};
export default async (): Promise<ProfileData> => {
const res = await fetch('https://api/user/profile');
return await res.json();
};
Here, routing components and loader
files share a type, so the import type
syntax should be used.
In the CSR environment, the loader
function is executed on the client and can use browser APIs (although it is not necessary and not recommended).
In the SSR environment, whether it is the first screen or client navigation, the loader
function will only be executed on the server, and any Node.js API can be called here. Also, any dependencies and code used here will not be included in the client's bundle.
In future versions, Modern.js may support running the loader
function on the server in the CSR environment to improve performance and security. Therefore, it is recommended to ensure that the loader
function is as pure as possible and only used for data fetching scenarios.
When navigating on the client based on Modern.js's conventional routing, all loader
functions will be executed in parallel (requested). That is, when accessing /user/profile
, the loader functions under /user
and /user/profile
will be executed in parallel (requested) to improve the performance of the client.
loader
FunctionThe loader
function has two input parameters:
params
When the route file is accessed through []
, it is used as dynamic routing, and the dynamic routing fragment is passed as a parameter to the loader
function:
// routes/user/[id]/page.loader.tsx
import { LoaderFunctionArgs } from '@modern-js/runtime/router';
export default async ({ params }: LoaderFunctionArgs) => {
const { id } = params;
const res = await fetch(`https://api/user/${id}`);
return res.json();
};
When accessing /user/123
, the parameter of the loader
function is { params: { id: '123' } }
.
request
request
is a Fetch Request instance.
A common usage scenario is to get query parameters through request
:
// routes/user/[id]/page.loader.ts
import { LoaderFunctionArgs } from '@modern-js/runtime/router';
export default async ({ request }: LoaderFunctionArgs) => {
const url = new URL(request.url);
const userId = url.searchParams.get('id');
return queryUser(userId);
};
The return value of the loader
function can be any serializable content or a Fetch Response instance:
const loader = async (): Promise<ProfileData> => {
return {
message: 'hello world',
};
};
export default loader;
By default, the Content-type
of the response returned by the loader
is application/json
, and the status
is 200. You can customize the Response
to set it:
const loader = async (): Promise<ProfileData> => {
const data = { message: 'hello world' };
return new Response(JSON.stringify(data), {
status: 200,
headers: {
'Content-Type': 'application/json; utf-8',
},
});
};
Modern.js provides a polyfill for the fetch
API to make requests. This API is consistent with the browser's fetch
API, but can also be used to make requests on the server. This means that whether it is CSR or SSR, a unified fetch
API can be used to get data:
function loader() {
const res = await fetch('https://api/user/profile');
}
In the loader
function, errors can be handled by throwing an error
or a response
. When an error is thrown in the loader
function, Modern.js will stop executing the code in the current loader
and switch the front-end UI to the defined ErrorBoundary
component:
// routes/user/profile/page.loader.tsx
export default async function loader() {
const res = await fetch('https://api/user/profile');
if (!res.ok) {
throw res;
}
return res.json();
}
// routes/user/profile/error.tsx
import { useRouteError } from '@modern-js/runtime/router';
const ErrorBoundary = () => {
const error = useRouteError() as Response;
return (
<div>
<h1>{error.status}</h1>
<h2>{error.statusText}</h2>
</div>
);
};
export default ErrorBoundary;
In many cases, child components need to access data in the parent component loader
. You can easily get the data from the parent component using useRouteLoaderData
:
// routes/user/profile/page.tsx
import { useRouteLoaderData } from '@modern-js/runtime/router';
export default function UserLayout() {
// Get the data returned by the loader in routes/user/layout.loader.ts
const data = useRouteLoaderData('user/layout');
return (
<div>
<h1>{data.name}</h1>
<h2>{data.age}</h2>
</div>
);
}
The useRouteLoaderData
function accepts a parameter routeId
. When using conventional routing, Modern.js will automatically generate the routeId
for you. The value of routeId
is the path of the corresponding component relative to src/routes
. For example, in the above example, if the child component wants to get the data returned by the loader in routes/user/layout.tsx
, the value of routeId
is user/layout
.
In a multi-entry (MPA) scenario, the value of routeId
needs to include the name of the corresponding entry. Unless specified, the entry name is generally the name of the entry directory. For example, in the following directory structure:
.
└── src
├── entry1
│ └── routes
│ └── layout.tsx
└── entry2
└── routes
└── layout.tsx
If you want to get the data returned by the loader
in entry1/routes/layout.tsx
, the value of routeId
is entry1_layout
.
This feature is currently experimental and the API may change in the future.
Currently only supports CSR, please look forward to Streaming SSR.
Create user/layout.loader.ts
and add the following code:
import { defer } from "@modern-js/runtime/router"
const loader = () =>
defer({
userInfo: new Promise((resolve) => {
setTimeout(() => {
resolve({
age: 1,
name: 'user layout'
})
}, 1000)
})
})
export default loader;
Add the following code in user/layout.tsx
:
import {
Await,
defer,
useLoaderData,
Outlet
} from '@modern-js/runtime/router';
export default function UserLayout() {
const { userInfo } = useLoaderData() as {userInfo: Promise<UserInfo>};
return (
<div>
<React.Suspense
fallback={<p>Loading...</p>}
>
<Await resolve={userInfo} children={userInfo => (
<div>
<span>{userInfo.name}</span>
<span>{userInfo.age}</span>
<Outlet>
</div>
)}>
</Await>
</React.Suspense>
</div>
);
}
loader
can only return serializable data. In the SSR environment, the return value of the loader
function will be serialized as a JSON string and then deserialized into an object on the client side. Therefore, the loader
function cannot return non-serializable data (such as functions).Currently, there is no such restriction under CSR, but we strongly recommend that you follow this restriction, and we may also add this restriction under CSR in the future.
// This won't work!
export default () => {
return {
user: {},
method: () => {},
};
};
loader
function for you, so you should not call the loader
function yourself:// This won't work!
export default async () => {
const res = fetch('https://api/user/profile');
return res.json();
};
import loader from './page.loader.ts';
export default function RouteComp() {
const data = loader();
}
import type
.// Not allowed
// routes/layout.tsx
import { useLoaderData } from '@modern-js/runtime/router';
import { ProfileData } from './page.loader.ts'; // should use "import type" instead
export const fetch = wrapFetch(fetch);
export default function UserPage() {
const profileData = useLoaderData() as ProfileData;
return <div>{profileData}</div>;
}
// routes/layout.loader.ts
import { fetch } from './layout.tsx'; // should not be imported from the routing component
export type ProfileData = {
/* some types */
};
export default async (): Promise<ProfileData> => {
const res = await fetch('https://api/user/profile');
return await res.json();
};
loader
function will be packaged into a unified bundle, so we do not recommend using __filename
and __dirname
in server-side code.loader
and BFF functionsIn CSR projects, loader
is executed on the client side, and the BFF function can be called directly in the loader
to make interface requests.
In SSR projects, each loader
is also a server-side interface. We recommend using the loader
instead of the BFF function with an http method of get
as the interface layer to avoid an extra layer of forwarding and execution.
useLoader
is a legacy API in Modern.js v1. This API is a React Hook designed specifically for SSR applications, allowing developers to fetch data in components in isomorphic development.
It is not necessary to use useLoader
to fetch data in CSR projects.
Here is the simplest example:
import { useLoader } from '@modern-js/runtime';
export default () => {
const { data } = useLoader(async () => {
console.log('fetch in useLoader');
// No real request is sent here, just a hard coding data is returned.
// In a real project, the data obtained from the remote end should be returned.
return {
name: 'Modern.js',
};
});
return <div>Hello, {data?.name}</div>;
};
After running the above code, when you access the page, you can see that logs are output to the terminal, but not printed in the browser console.
This is because Modern.js collects the data returned by useLoader
during server-side rendering and injects it into the corresponding HTML. If the SSR rendering is successful, you can see the following code snippet in the HTML:
<script>
window._SSR_DATA = {};
</script>
This global variable is used to record data, and during the browser-side rendering process, this data is used first. If the data does not exist, the useLoader
function will be executed again.
During the build phase, Modern.js will automatically generate a Loader ID for each useLoader
and inject it into the SSR and CSR JS Bundles to associate the Loader with the data.
Compared to getServerSideProps
in Next.js, which fetches data before rendering, using useLoader
allows you to get data required for local UI in the component without passing data through multiple layers. Similarly, you don't have to add redundant logic to the outermost data acquisition function because different routes require different data requests. Of course, useLoader
also has some issues, such as difficulties in server-side code tree shaking and the need for an additional pre-rendering step on the server.
In the new version of Modern.js, a new Loader solution has been designed. The new solution solves these problems and can be optimized for page performance in conjunction with conventional routing.
For detailed API information, see useLoader.