Routing
Edit this pageRouting serves as a key component to web applications. Within SolidStart, there are two types:
- UI routes — define the user interface in your app
- API routes — define the serverless functions in your app
To read more about API routes, see the API Routes section.
Creating new routes
SolidStart uses file based routing which is a way of defining your routes by creating files and folders in your project. This includes your pages and API routes.
SolidStart traverses your routes
directory, collects all of the routes, and then makes them accessible using the <FileRoutes />
.
This component will only include your UI routes, and not your API routes.
Rather than manually defining each Route
inside a Router
component, <FileRoutes />
will generate the routes for you based on the file system.
Because <FileRoutes />
returns a routing config object, you can use it with the router of your choice.
In this example we use solid-router
:
The <Router />
component expects a root
prop which functions as the root layout of your entire app.
You will want to make sure props.children
is wrapped in <Suspense />
because each component will be lazy loaded automatically for you. Without this you may see some unexpected hydration errors.
<FileRoutes />
will generate a route for each file in the routes
directory and its subdirectories. For a route to be rendered as a page, it must default export a component.
This component represents the content that will be rendered when users visit the page:
This means that all you have to do is create a file in your routes
folder and SolidStart takes care of everything else needed to make that route available to visit in your application!
File based routing
Each file in the routes
directory is treated as a route.
To create a new route or page in your application, simply create a new file in the routes
directory.
The file name will be the URL path for the route:
example.com/blog
➜/routes/blog.tsx
example.com/contact
➜/routes/contact.tsx
example.com/directions
➜/routes/directions.tsx
Nested routes
If you need nested routes, you can create a directory with the name of the preceding route segment, and create new files in that directory:
example.com/blog/article-1
➜/routes/blog/article-1.tsx
example.com/work/job-1
➜/routes/work/job-1.tsx
When a file is named index
, it will be rendered when there are no additional URL route segments being requested for a matching directory:
example.com
➜/routes/index.tsx
example.com/socials
➜/routes/socials/index.tsx
Nested layouts
If you want to create nested layouts you can create a file with the same name as a route folder.
In this case the blog.tsx
file will act as a layout for the articles in the blog
folder. You can reference the child content
by using props.children
in the layout.
Note: Creating a blog/index.tsx
or blog/(blogIndex).tsx
is not the same as it would only be used for the index route.
Renaming Index
By default, the component that is rendered for a route comes from the default export of the index.tsx
file in each folder.
However, this can make it difficult to find the correct index.tsx
file when searching, since there will be multiple files with that name.
To avoid this, you can rename the index.tsx
file to the name of the folder it is in, enclosed in parenthesis.
This way, it will be treated as the default export for that route:
Escaping nested routes
When you have a path that is nested but wish for it to have a separate Layout, you can escape the nested route by applying a name between ( )
.
This will allow you to create a new route that is not nested under the previous route:
Additionally, you can incorporate nested layouts of their own:
Dynamic routes
Dynamic routes are routes that can match any value for one segment of the route.
When your URL path contains a dynamic segment, square brackets ([]
) are used to define the dynamic segment:
example.com/users/:id
➜/routes/users/[id].tsx
example.com/users/:id/:name
➜/routes/users/[id]/[name].tsx
example.com/*missing
➜/routes/[...missing].tsx
This allows you to create a single route that can match any value for that segment of the URL path.
For example, /users/1
and /users/2
are both valid routes and rather than defining separate routes for each user, you can use a dynamic route to match any value for the id
segment.
For example, using solid-router
, you could use the useParams
primitive to match the dynamic segment:
Optional parameter
If you have optional parameters in your route, you can use the double square brackets ([[id]]
) to define the dynamic segment.
This will match a route with or without a parameter.
In this case, some pages that could be matched include:
/users
/users/1
/users/abc
Catch-all routes
Catch-all routes are a special type of dynamic route that can match any number of segments.
They are defined using square brackets with ...
before the label for the route (e.g. [...post]
).
A catch-all route will have one parameter which is a forward-slash delimited string of all the URL segments after the last valid segment.
For example, with the route [...post]
and a URL path of /post/foo
the params
object returned from the useParams
primitive will have a post
property with the value of post/foo
.
For a URL path of /post/foo/baz
it will be post/foo/baz
.
Route groups
Using route groups, you can organize your routes in a way that makes sense for your application, without affecting the URL structure. Since file-based routing is based on the file system, it can be difficult to organize your routes in a way that makes sense for your application.
In SolidStart, route groups are defined by using parenthesis (()
) surrounding the folder name:
Additional route config
SolidStart offers a way to add additional route configuration outside of the file system.
Since SolidStart supports the use of other routers, you can use the route
export provided by <FileRoutes />
to define the route configuration for the router of your choice.