A quick introduction to the folders and files in this repo.
Source Organization: src/
To keep things modular, the resources are divided into folders namely /screens
, /pages
, /components
, /actions
.
-
/screens
: Contains Next.ts pages along with their necessary styles and any extra files.- While it's possible to create Next.ts pages in
/src/pages
directly, several problems occur. Pages are able to import CSS modules like any other component, but CSS cannot be placed in the/src/pages
directory. - To work, CSS files must be placed in another directory away from the page, leading to messy code. Similarly, any other files or utils must be placed in a separate directory.
- To fix this, screens are placed in a separate directory from pages, and each in their own directory. This makes it very simple to keep the logic, styles, and utilities for each page in their separate space.
- Each screen behaves as a page, so
getInitialProps
and other Next.ts helpers can be used.
- While it's possible to create Next.ts pages in
-
/pages
: Used for creating file-system routing to screens and creating API routes.- The
/src/pages
directory acts as file-system routing for our screens and for API routes. - For regular pages,
import
andexport default
the screen directly. - API routes are placed in the
/src/pages/api
directory.- To simplify API routes and promote code reuse, server-side actions are used from the
/server/actions
directory. - Every API route must return a HTTP status code and body matching the template:
for successful requests, and:
res.status(201).json({ success: true, payload: ..., })
for errors. This makes processing the results much easier.res.status(500).json({ success: false, message: "...", })
- The request body can be accessed with
req.body
and cookies withreq.cookies
.
- To simplify API routes and promote code reuse, server-side actions are used from the
- /pages/api: Contains API routes
- The
-
/components
: Contains reusable React components.- Each reusable component is placed in this directory with a similar structure to screens.
- Create a directory for each component, and include any necessary styles and utils for this component only in the same directory.
- Each directory must include a
index.tsx
file that imports and export defaults the component. This makes it easier to import the component from/componentDir
instead of/componentDir/component
. - Any sub-components that are used by, and only by, this component, should be placed within their own sub-directory within the component's directory.
-
/actions
: Contains functions for calling API routes.- Each file in this directory contains multiple functions for interacting with a group of API routes.
- Each function should return a fetch request matching the template:
fetch(urls.baseUrl + urls.api.example, { method: "get", mode: "same-origin", credentials: "include", }) .then((response) => response.json()) .then((json) => { if (json == null) { throw new Error("Could not connect to API!"); } else if (!json.success) { throw new Error(json.message); } return json.payload; });
Server Organization: server/
The server directory includes the backend actions used in API routes separated by their type.
-
/mongodb
: Contains backend Mongoose models and actions for interacting with MongoDB.- Mongoose models should be placed within the
server/mongodb/models
directory.- The export for each model should follow the template:
export default mongoose.models.User ?? mongoose.model("User", UserSchema);
- The export for each model should follow the template:
- MongoDB (using Mongoose) actions should be placed within the
server/mongodb/actions
directory.- Each file should use the same name as the model, and include all related actions.
- Each file needs to import
import mongoDB from "../index";
, and each function needs to includeawait mongoDB();
(once per function) before any interactions are made with the database.
- Mongoose models should be placed within the
Utils Organization: utils/
The utils directory includes any utilities needed for the frontend and backend.
-
urls.ts
: Exports an object containing the urls for each page and API route.- After creating a page or API route, the path needs to be added to
urls.ts
immediately. - This makes it easy use urls in the project, because the urls object only needs to be imported, and then urls can be changed at a later date without needing to search the code to replace urls as strings.
- Plain strings should NEVER be used to reference pages/API routes, ALWAYS import the urls object.
- If a dynamic route is needed format it as
pageKey: "/somePage/[aKey]"
(with the corresponding page route being/pages/somePage/[aKey].tsx
), then use theNavLink
component to navigate to this page:<NavLink href={pages.pageKey} hrefParts={{ aKey: 123 }}>Link</NavLink>
.
- After creating a page or API route, the path needs to be added to
Public Organization: public/
The public directory hosts any included files on the website.
/public
: Files placed in this directory can be accessed atbaseUrl/file
. Be VERY careful to not include a file with the same name as a page!/public/static
: Files placed in this directory can be accessed atbaseUrl/static/file
.
[...nextauth].ts
: Is a catch-all API route to handle all authentication requests to Auth0.pages/_app.tsx
: Includes a provider to pass session data to all pages, screens, and components.actions/User.ts
: Includes helper functions for login and logout using url redirects.- All pages, screens, and components can access the session using
useSession
, as shown inHomePage.tsx
.