Hasan Zohdy
Posted on September 3, 2022
Introduction
This package is meant to make your code elegant, easy to maintain and decouple your project into multiple apps and modules.
Imagine you have a project that contains an admin dashboard and the website front office, Mongez React Router (MRR) will allow multiple teams/members to work separately in each app and both are in one project without any conflict, additionally you can share common styles/assets as well.
Highlighted Features
- ✅ Declaring routes in a more readable and clean way.
- ✅ Common Base Layouts for multiple routes so there won't be rerendering for partials such as header and footer.
- ✅ Easy Middleware definitions
- ✅ Lazy loading for entire apps and module to reduce production bundle size.
- ✅ Grouping routes with common features such as setting base path, common middleware between routes.
- ✅ Language switching without reloading the page
- ✅ Optionally refresh the same page when navigating to it again.
- ✅ Many helpers to navigate between routes using functions.
- ✅ Working with functions for navigation instead of hooks.
- ✅ Optionally redirect to Not found page route or just render a component instead.
- ✅ Defines a preloader (Progress Bar for instance) until the app/module bundle is loaded.
- ❌ No ugly writing for routes in components.
Before going on
This documentation will illustrate the package features, however, it is recommended to use it with Mongez React for better project organization.
React Router DOM is used under the hood.
Installation
yarn add @mongez/react-router
Or
npm i @mongez/react-router
Usage
In your src/index
file import the package and clear the ReactDOM.render
section.
// src/index.ts
import router from "@mongez/react-router";
// remove the following code from the file
import ReactDOM from "react-dom";
import App from "./App";
ReactDOM.render(
<React.StrictMode>
<App />
</React.StrictMode>,
document.getElementById("root")
);
Now let's add a route for our home page
// src/index.ts
import router from "@mongez/react-router";
import HomePage from "./Home";
router.add("/", HomePage);
// Start scanning for all of registered routes
router.scan();
We imported our HomePage
component which is a normal react component, then we used the router.add
method to define our first route which defines our home page route.
Next we called router.scan()
to start scanning all registered routes in the router to call the proper route.
Please keep in mind to set
router.scan
after declaring all of your routes first.
Lazy loading Apps
One of the most powered features in this tool si to lazy load your app especially if you've multiple apps in one project, so let's see how this can be done.
For example, none of the admin
files will be loaded unless the user navigates to the /admin
route in the browser, as well as any module inside the admin such as administrators
module.
Creating Apps
Any react project, can contain one or more apps in one project, for example a front-office
for the main website and admin
dashboard to manage the website.
So let's create these two apps in our src/apps
directory so it would be like this:
|--- src
|--- apps
|-- front-office
|-- admin
|--- index.ts
Apps Path Alias
To make lazy loading apps work, we need to define an absolute path for our apps, using tsconfig.json file or by using link-module-alias which is more recommended.
If you're using link-module-alias
then open package.json
file and add to it the following code:
"scripts": {
"postinstall": "link-module-alias"
},
"_moduleAliases": {
"apps": "src/apps"
}
Then run yarn postinstall
or npm run postinstall
.
Please keep in mind that when you upgrade any package don't forget to run this command again.
If you're installing new package this command will run automatically so you won't need to run it.
If theapps
is not defined in the path alias the lazy app loading won't work and would trigger an error.
App Modules Declaration
Each application must have at least two files to work properly.
- A
front-office-modules.json
to define the app structure, modules and its internal routes. - A
front-office-provider.ts
file as an entry point to the application, as it will be called at the very beginning before calling any internal module inside the app.
The modules.json file name must be in this sequence
appName-modules.json
as we named our admin modules file.
The app provider must be named asappName-provider.ts
orappName-provider.js
if you're using Javascript.
|--- src
|--- apps
|-- front-office
|-- front-office-modules.json
|-- front-office-provider.ts
|-- admin
|--- index.ts
Now open front-office-modules.json
file and put the following code inside it.
{
"name": "front-office",
"path": "/",
"modules": [
{
"entry": ["/"],
"module": "home"
}
]
}
Let's go in depth with each key in this file.
-
name
: App name, it should be the same as the directory name insrc/apps
otherwise it won't work. -
path
: App base path, for front-office it will be/
for something like admin it could be/admin
or whatever you desire. -
modules
: Defines list of modules that will be included in this app. -
modules.entry
: is an array that contains the starting segments to make the module be loaded, we'll get into it next. -
modules.module
: defines the module name, it must match the module directory name.
Module Structure
Each app consists of list of modules, each module must have at least a provider
file inside it.
The provider.ts|.js
file will be called directly once browser hits the entry path of the module which will be illustrated later, in this file our module settings should be imported inside it such as its routes.
Module Entry Concept
Let's take an example to make it clear.
Let's say we're working on an online store project, which will contain a customer account dashboard, this dashboard may have the following routes:
/account/dashboard
/account/edit-profile
/account/order-history
/account/order-history/:id
All of the previous routes are part of the account
module in our project, also they all starts with /account
segment.
In that case our project structure wil look like this:
|--- src
|--- apps
|-- front-office
|-- account
|-- components
|-- DashboardPage.tsx
|-- EditProfilePage.tsx
|-- OrderHistoryList.tsx
|-- SingleOrderHistory.tsx
|-- front-office-modules.json
|-- front-office-provider.ts
|-- admin
|--- index.ts
In that sense, they all share one module which is account
and they all start with /account
, to declare this in our front-office-modules.json
we will only add the starting segment of all of these routes which is /account
.
What happens here is Mongez React Router loads the
account
module when it sees the route starts with/account
.
{
"name": "front-office",
"path": "/",
"modules": [
{
"entry": ["/account"],
"module": "account"
},
{
"entry": ["/"],
"module": "home"
}
]
}
One more thing to mention is that we can also load the module with different routes such as if /login
is part of our account
module, then we can add it in our entry
section.
{
"name": "front-office",
"path": "/",
"modules": [
{
"entry": ["/account", "/login"],
"module": "account"
},
{
"entry": ["/"],
"module": "home"
}
]
}
The entry
key accepts only the first segment of the route, so don't define inside it the entire route.
✅
{
"name": "front-office",
"path": "/",
"modules": [
{
"entry": ["/account", "/login"],
"module": "account"
},
{
"entry": ["/"],
"module": "home"
}
]
}
❌
{
"name": "front-office",
"path": "/",
"modules": [
{
"entry": ["/account/dashboard", "/account/edit-profile", "/login"],
"module": "account"
},
{
"entry": ["/"],
"module": "home"
}
]
}
Defining apps list
Now let's create a src/shared/apps-list.ts
file to set our apps list.
For better organization, we'll create a
shared
directory inside oursrc
so we can set our shared configurations between multiple apps.
|--- src
|--- apps
|-- front-office
|-- front-office-modules.json
|-- front-office-provider.ts
|-- admin
|--- shared
|-- apps-list.ts
|--- index.ts
// src/shared/apps-list.ts
import { setApps } from "@mongez/react-router";
import frontOfficeApp from "apps/front-office/front-office-modules.json";
setApps([frontOfficeApp]);
We used the
apps/
alias directly as we already using path alias.
Now let's head back to our index file and import our apps-list.ts
file.
// src/index.ts
import "./shared/apps-list";
import router from "@mongez/react-router";
router.scan();
From now on, you can lazy load any new app or any new module by creating its directory and its provider.
Defining Module routes
Now we imported our apps list and everything works fine so far except that no routes will be loaded!
The reason behind this that we didn't declare any routes as we only informed MRR
to load the module provider, so now we need to define our routes for our module.
In src/apps/front-office/home
we should have two files: provider.ts
and routes.ts
.
// src/apps/front-office/home/provider.ts
import "./routes";
We just imported our routes.ts
file, now let's add our routes there.
// src/apps/front-office/home/routes.ts
import router from "@mongez/router";
import HomePage from "./components/HomePage";
router.add("/", HomePage);
Now we'are ready to go as we're done with our setup.
Route Middleware
Some routes require a step head before navigating to its component. for example the visitor can not access his/her account dashboard unless he/she is logged in, in such a scenario we can use a middleware.
In src/apps/front-office/account/routes.ts
file we can define our routes as follows:
// src/apps/front-office/account/routes.ts
import router from "@mongez/react-router";
import AccountDashboardPage from "./components/DashboardPage";
import EditProfilePage from "./components/EditProfilePage";
import OrderHistoryPage from "./components/OrderHistoryPage";
import SingleOrderHistoryPage from "./components/SingleOrderHistoryPage";
import Guardian from "./middleware/Guardian";
router.add("/account", AccountDashboardPage, [Guardian]);
router.add("/account/edit-profile", EditProfilePage), [Guardian];
router.add("/account/order-history", OrderHistoryPage, [Guardian]);
router.add("/account/order-history/:id", SingleOrderHistoryPage, [Guardian]);
Here we defined our routes, with a new argument in router.add
which is an array of middleware that will be declared before navigating to our pages.
Now let's see our new Guardian middleware file.
// src/apps/front-office/account/middleware/Guardian.tsx
import user from "somewhere-in-the-app";
import React from "react";
import { Redirect } from "@mongez/react-router";
export default function Guardian() {
if (user.isNotLoggedIn()) {
return <Redirect to="/login" />;
}
return null;
}
Here we defined a component that allows us to check if user is not logged in, then we'll redirect the user to the login route by using Redirect
component from MRR
.
Now whenever a user hits any of the account routes, the Guardian
component will be called first, if the user is not logged in then the redirect component will be called instead of the page component.
If the middleware returned a value, then it will be displayed instead of the page component.
So the middleware can look like:
// src/apps/front-office/account/middleware/Guardian.tsx
import user from "somewhere-in-the-app";
import React from "react";
export default function Guardian() {
if (user.isNotLoggedIn()) {
return <h1>You do not have access to this page, please login first.</h1>;
}
return null;
}
Grouped Routes
As we can use router.add
method do define a route, we can define one or more routes with common settings such as a prefix or a middleware.
In our previous middleware example, we can see that all routes starts with /account
and they all have the same middleware, we can group these routes in one method using router.group
method.
// src/apps/front-office/account/routes.ts
import router from "@mongez/react-router";
import AccountDashboardPage from "./components/DashboardPage";
import EditProfilePage from "./components/EditProfilePage";
import OrderHistoryPage from "./components/OrderHistoryPage";
import SingleOrderHistoryPage from "./components/SingleOrderHistoryPage";
import Guardian from "./middleware/Guardian";
router.group({
path: "/account",
middleware: [Guardian],
routes: [
{
path: "/",
component: AccountDashboardPage,
},
{
path: "/edit-profile",
component: EditProfilePage,
},
{
path: "/order-history",
component: OrderHistoryPage,
},
{
path: "/order-history/:id",
component: SingleOrderHistoryPage,
},
],
});
Our code now is more compact and cleaner, also you can pass an additional middleware to any route object if you want to add more middleware to certain routes.
The prefix in the group method will be glued with all routes in the routes array, so the AccountDashboardPage
route will be /account/
but the last /
will be trimmed by MRR
.
You can set the path of the
AccountDashboardPage
to be empty string '' it works as well.
Page Base Layout
Most of the apps has same layout structure such as a header and a footer among it the content of the page.
This can be done easily with MRR
by using router.partOf
method.
// src/apps/front-office/components/BaseLayout.tsx
import React from "react";
import Header from "./Header";
import Footer from "./Footer";
export default function BaseLayout({ children }) {
return (
<>
<Header />
<main>{children}</main>
<Footer />
</>
);
}
Now let's add our New Base layout to our HomePage.
// src/apps/front-office/home/routes.ts
import router from "@mongez/router";
import HomePage from "./components/HomePage";
import BaseLayout from "apps/front-office/components/BaseLayout";
router.partOf(BaseLayout, [
{
path: "/",
component: HomePage,
},
]);
Now our HomePage
component doesn't need to call the header or the footer of the page, it is now part of the BaseLayout
.
This can be useful with router.group
as well, we can set a common layout between list of pages.
Let's head back to our account module.
// src/apps/front-office/account/routes.ts
import router from "@mongez/react-router";
import AccountDashboardPage from "./components/DashboardPage";
import EditProfilePage from "./components/EditProfilePage";
import OrderHistoryPage from "./components/OrderHistoryPage";
import SingleOrderHistoryPage from "./components/SingleOrderHistoryPage";
import Guardian from "./middleware/Guardian";
import BaseLayout from "apps/front-office/components/BaseLayout";
router.group({
path: "/account",
layout: BaseLayout,
middleware: [Guardian],
routes: [
{
path: "/",
component: AccountDashboardPage,
},
{
path: "/edit-profile",
component: EditProfilePage,
},
{
path: "/order-history",
component: OrderHistoryPage,
},
{
path: "/order-history/:id",
component: SingleOrderHistoryPage,
},
],
});
Now we added a new property in the group object called layout
which defines the layout that will render all of the routes pages.
Extending Base Layout
Let's take another scenario that the account pages have a common sidebar between all of its page, we can make a newer layout that can hold the header, footer and the account sidebar.
// src/apps/front-office/account/components/AccountLayout.tsx
import React from "react";
import Header from "apps/front-office/components/Header";
import Header from "apps/front-office/components/Footer";
import AccountSidebar from "./AccountSidebar";
export default function AccountLayout({ children }) {
return (
<>
<Header />
<main>
<AccountSidebar />
<div>{children}</div>
</main>
<Footer />
</>
);
}
Now let's our account routes again to use our new layout instead of the base layout.
// src/apps/front-office/account/routes.ts
import router from "@mongez/react-router";
import AccountDashboardPage from "./components/DashboardPage";
import EditProfilePage from "./components/EditProfilePage";
import OrderHistoryPage from "./components/OrderHistoryPage";
import SingleOrderHistoryPage from "./components/SingleOrderHistoryPage";
import Guardian from "./middleware/Guardian";
import AccountLayout from "./components/AccountLayout";
router.group({
path: "/account",
layout: AccountLayout,
middleware: [Guardian],
routes: [
{
path: "/",
component: AccountDashboardPage,
},
{
path: "/edit-profile",
component: EditProfilePage,
},
{
path: "/order-history",
component: OrderHistoryPage,
},
{
path: "/order-history/:id",
component: SingleOrderHistoryPage,
},
],
});
Now we're almost done, one more thing to do is to extend our base layout as we injected the header and the footer again in the AccountLayout
component, let's fix this.
// src/apps/front-office/account/components/AccountLayout.tsx
import React from "react";
import BaseLayout from "apps/front-office/components/BaseLayout";
import AccountSidebar from "./AccountSidebar";
export default function AccountLayout({ children }) {
return (
<BaseLayout>
<AccountSidebar />
<div>{children}</div>
</BaseLayout>
);
}
Now our code is very neat and can be maintained easily.
Routes Relativity
All routes defined inside the routes.ts
files in the lazy load mode, are prefixed with the app path, so if we've a route in the admin such as /admin/login
, the defined route in src/apps/admin/administrators/routes.ts
will be /login
without adding /admin
at the beginning.
// src/apps/admin/administrators/routes.ts
import LoginPage from "./components";
import router from "@mongez/react-router";
// here we'll define the route as /login not /admin/login
router.add("/login", LoginPage);
Route Path Structure
Based on the app configurations, we've X route structures.
-
/
: The app root -
/en
: The app root appended with locale code -
/en/contact-us
: The contact us route prefixed with locale code. -
/admin
: Admin app home path. -
/en/admin
: Admin app home path withen
locale code. -
/en/admin/customers
: Admin app customers page withen
locale code. -
admin/customers
: Admin app customers page with default locale code.
So our full route structure will be something like:
/localeCode(optional)/app-path/route
When you define your route don't add the app path or locale code, for example:
✅
router.add('/contact-us', ContactUs)
❌
router.add('/admin/contact-us', ContactUs)
router.add('/en/contact-us', ContactUs)
router.add('/en/admin/contact-us', ContactUs)
Router Configurations
MRR
doesn't require any configurations to be set, but its recommended to define some configurations such as the locale code used in the project.
In src/shared
directory let's create a new file config.ts
// src/shared/config.ts
import { setRouterConfigurations } from "@mongez/react-router";
setRouterConfigurations({
// if your app is multilingual then define all locale codes in the app
localeCodes: ["en", "ar"],
// if the production build will be in a directory and not the root, then define the directory path in basePath
basePath: "/",
});
Now import the file in the index file.
// src/index.ts
// its important to import the config file before any route functions.
import "./shared/config";
import "./shared/apps-list";
import router from "@mongez/react-router";
router.scan();
If you're using Mongez React, it can be part of the entire application configurations.
Here is the full list of available configurations
/**
* Router configuration options list
*/
type RouterConfigurations = {
/**
* Default locale code
*/
defaultLocaleCode?: string;
/**
* Locale codes list
*/
localeCodes?: string[];
/**
* Router preloader that will be displayed until the module is loaded
*
* @default React.Fragment
*/
preloader?: React.ComponentType<any>;
/**
* If set to true, the current layout will not be unmounted and the preloader (if set) will be displayed before it
* Please note the of the base layout and the preloader will have position `relative`
* This feature is still experimental and can be changed in future versions
* @experimental
* @default false
*/
preloadOverlay?: boolean;
/**
* App base path in production
*
* @default: /
*/
basePath?: string;
/**
* Determine whether to re-render the page
* When navigating to any page, even same current page
*
* Please note that can not be changed during the application is running
* as its value is cached at the application bootstrap
*
* @default: true
*/
forceRefresh?: boolean;
/**
* Scroll to top of the page when rendering new page
*
* @default true
*/
scrollTop?: boolean;
/**
* Top Root component that will wrap the entire application regardless the lazy module
*/
rootComponent?: React.ComponentType<any>;
/**
* NotFound Options
*/
notFound?: {
/**
* Not found mode
* The redirect mode will redirect the client to the path
*
* Please note that can not be changed during the application is running
* as its value is cached at the application bootstrap
*
* @default: render
*/
mode?: "redirect" | "render";
/**
* The route that will be redirected when the page is not found
* Works only when the mode is set to redirect
*
* @default: /404
*/
route?: string;
/**
* The component that will be rendered when the page is not found
* Works only when the mode is set to render
*
* @default: React.Fragment
*/
component?: React.ComponentType<any>;
};
};
Root Component
The root component will wrap the entire application regardless wether current route is being lazy loaded or not.
Root Component does not receive any props at all.
Link Navigation
Using Link
component in react will provide some interesting features to make the link more readable and easier to use.
// src/apps/front-office/home/components/HomePage.tsx
import React from "react";
import { Link } from "@mongez/react";
export default function HomePage() {
return (
<div>
<Link to="/account">Go To Account Page</Link>
</div>
);
}
Simply put, a simple navigation to route by using to
or href
prop.
To navigate to route in a new tab:
<Link to="/account" newTab>
Go To Account Page In New Tab
</Link>
// outputs: <a href="/account" target="_blank" rel="noopener noreferrer">
To navigate to route in another locale code:
To navigate to route in another app:
<Link to="/customers/100" app="admin">
Go To Customer page in admin app.
</Link>
// outputs: /admin/customers/100
<Link to="/account" localeCode="ar">
Go To Account Page With Arabic Locale Code
</Link>
// outputs: /ar/account
Navigate to another app with a locale code
<Link to="/account" app="admin" localeCode="ar">
Go To Account Page In Admin App With Arabic Locale Code
</Link>
// outputs: /ar/admin/account
The app prop accepts the app name not the app path
To navigate to a url, just set the url :p.
<Link to="https://google.com">Go To Google</Link>
Make the link as email:
<Link mailTo="hassanzohdy@gmail.com">Email As Link</Link>
// outputs: <a href="mailto:hassanzohdy@gmail.com" .. />
or using MailLink
component directly
import { MailLink } from "@mongez/react-router";
<MailLink to="hassanzohdy@gmail.com">Email As Link</MailLink>;
// outputs: <a href="mailto:hassanzohdy@gmail.com" .. />
Make the link as a telephone number:
<Link tel="+201002221122">Phone Number As Link</Link>
// outputs: <a href="tel:+201002221122" .. />
or using MailLink
component directly
import { TelLink } from "@mongez/react-router";
<TelLink to="+201002221122">Phone Number As Link</TelLink>;
// outputs: <a href="tel:+201002221122" .. />
Of course you can send any other html props such as className
, id
and so on.
Use External Link
import { ExternalLink } from "@mongez/react-router";
<ExternalLink to="https://google.com">Google</ExternalLink>;
// outputs: <a href="https://google.com">Google</a>
Open it in a new tab
import { ExternalLink } from "@mongez/react-router";
<ExternalLink newTab to="https://google.com">
Google
</ExternalLink>;
// outputs: <a target="_blank" rel="noopener noreferrer" href="https://google.com">Google</a>
Redirect Component
This component usually used with middleware as mentioned earlier in the Middleware section.
// src/apps/admin/account/middleware/Guardian.tsx
import user from "somewhere-in-the-app";
import React from "react";
import { Redirect } from "@mongez/react-router";
export default function Guardian() {
if (user.isNotLoggedIn()) {
return <Redirect to="/login" localeCode="fr" app="admin" />;
}
return null;
}
The previous guardian will navigate to /admin/fr/login
as the admin
app path is /admin
and the locale code is appended after it then finally the route itself.
Routes Structure
There are 4 types of routes, let's take an example of /login
route for more illustrations.
- A route in the base app: final route:
/login
, which is constructed aspage-route
. - A route in the base app with locale code: final route:
/login
, which is constructed as/locale-code/page-route
. - A route in another app, such as admin:
/admin/login
, which is constructed as/app-path/page-route
. - A route in another app with a locale code, such as admin:
/admin/ar/login
, which is constructed as/app-path/locale-code/page-route
.
Receiving router params
Let's take a complex route and see how we can take its values.
Our route path will be: /admin/en/customers/101
and will be rendered in CustomerPage
Component.
This route path is defined as /customers/:id
.
// src/apps/admin/customers/routes.ts
import router from "@mongez/react-router";
import CustomerPage from "./components/CustomerPage";
router.add("/customers/:id", CustomerPage);
Now let's head to our CustomerPage
component.
// src/apps/admin/customers/components/CustomerPage.tsx
import React from "react";
export default function CustomerPage({ params }) {
const { localeCode, id } = params;
console.log(localeCode); // en
console.log(id); // 101
return <div>// component content</div>;
}
The /admin
segment will be ignored, we can only get localeCode
which is defined by MRR
internally and our defined segment /:id
is transformed into id
from the params object.
Route Wild Card
Let's take another scenario where we have dynamic routes such as:
/categories/electronics/smart-phones/tablets
The route is defining the category tree, as we will access the tablets category page.
In that sense, the route can be something else like /categories/electronics/smart-phones
where we'll go to Smart Phones category.
To get the dynamic route we can use wildcards.
// src/apps/front-office/categories/routes.ts
import router from "@mongez/react-router";
import CategoryPage from "./components/CategoryPage";
router.add("/categories/:slug(.+)", CategoryPage);
In our CategoryPage
component
// src/apps/front-office/categories/components/CategoryPage.tsx
import React from "react";
export default function CategoryPage({ params }) {
const { slug } = params;
console.log(slug); // /electronics/smart-phones
return <div>// component content</div>;
}
You can also use the dynamicSegment
helper for more readability.
// src/apps/front-office/categories/routes.ts
import router, { dynamicSegment } from "@mongez/react-router";
import CategoryPage from "./components/CategoryPage";
router.add("/categories/" + dynamicSegment("slug"), CategoryPage);
Will achieve the same result.
More Restrict Route Segments
Sometimes, segments such as :id
is usually integers only, so we can define a route that accepts only integer values in the route so we don't have to make another validation step on the given id, we can use integerSegment
helper.
// src/apps/admin/customers/routes.ts
import router, { integerSegment } from "@mongez/react-router";
import CustomerPage from "./components/CustomerPage";
router.add("/customers/" + integerSegment("id"), CustomerPage);
Now if the user hits /customers/some-text
he/she will be redirected automatically to not found page, a /customers/101
route will be valid though.
Another helper function floatSegment
can be used for float values.
Switching to another locale code
We can switch to another locale code by using switchLang
function
import { switchLang } from '@mongez/react-router';
import React from 'react'
export default function Header() {
const onClick = e => {
switchLang('ar'); // refreshes the page and changes the locale code
};
return (
<div>
<button onClick={changeLocaleCode}>Switch To Arabic<button>
</div>
)
}
Navigating to route
The navigateTo
function is one of the most powered functions that allows you to navigate to another page.
import { navigateTo } from "@mongez/react-router";
import React from "react";
export default function CreateAccountPage() {
const createAccount = (e) => {
axios.post("/register", { email, password }).then((response) => {
navigateTo("/home");
});
};
return (
<div>
<form onSubmit={createAccount}>
...
<button>Create a new account</button>
</form>
</div>
);
}
Navigate to with locale code
navigateTo("/login", "en"); // /en/login
Navigate to route with locale code and app.
navigateTo("/login", "en", "admin"); // /admin/en/login
Please note that the third argument accepts the app name not the app path, if you'd like to use the app path, just add it to the first argument.
If the project has multiple locale codes, then any navigation usingnavigateTo
function will prepend the locale code, for examplenavigateTo('/login')
and default locale code isen
, this will navigate to/en/login
Navigate Back
import { navigateBack } from "@mongez/react-router";
import React from "react";
export default function LoginPage() {
const login = (e) => {
axios.post("/login", { email, password }).then((response) => {
navigateBack();
});
};
return (
<div>
<form onSubmit={login}>
...
<button>Login</button>
</form>
</div>
);
}
Usually used with the login page as we can return the user back once he/she logged in successfully.
Page Refresh
We can refresh the page so it will re-render again using refresh
function, this won't cause full page reload but it will acts as a route navigation.
import { refresh } from '@mongez/react-router';
import React from 'react'
export default function Header() {
const refreshPage = () => {
refresh();
}
return (
<div>
<button onClick={refreshPage}>Refresh The Page<button>
</div>
)
}
Get current page route
To get current page route, we can use currentRoute
function.
// src/apps/front-office/front-office-provider.ts
import { currentRoute } from "@mongez/react-router";
// detect current route
console.log(currentRoute()); // will be something like /login or /account
Get full url of the page
To get full page path, we can use url
function.
// src/apps/front-office/front-office-provider.ts
import { url } from "@mongez/react-router";
// detect current route
console.log(url()); // will be something like https://sitename.com/online-store/account
Route Concatenation
Sometimes you may need to concat multiple routes in one route, for example adding a route from a variable with another variable to generate a brand new route, luckily you can use concatRoute
helper function to do it for you.
import { concatRoute } from "@mongez/react-router";
const localeCode = "ar";
const route = "login";
const appPath = "/admin";
const fullRoutePath = concatRoute(appPath, localeCode, route); // /admin/ar/login
Each argument can start with/without a trailing slash, also any doubled slashes will be converted into one slash and if there is any ending slashed will be trimmed as well.
import { concatRoute } from "@mongez/react-router";
const localeCode = "ar";
const route = "//login//";
console.log(localeCode, route); // /ar/login
Updating query string
We can also update query string in the url with/without navigating to the route with the new query string by using updateQueryString
helper function, this can be useful for cases such as filtering as we can only update the route without a full new re-render to the page.
import { updateQueryString } from "@mongez/react-router";
export default function FilterData() {
const filter = (e) => {
// Just dummy data for demo only
const filterData = {
name: "",
email: "",
age: 0,
published: true,
};
axios.get("/filter", filterData).then((response) => {
updateQueryString(filterData);
});
};
return (
<div>
<form onSubmit={filter}>
...
<button>Filter</button>
</form>
</div>
);
}
If we would like to navigate to the same page with the updated query string, we can set the second argument to true.
import { updateQueryString } from "@mongez/react-router";
export default function FilterData() {
const filter = (e) => {
// Just dummy data for demo only
const filterData = {
name: "",
email: "",
age: 0,
published: true,
};
axios.get("/filter", filterData).then((response) => {
updateQueryString(filterData, true); // update and navigate
});
};
return (
<div>
<form onSubmit={filter}>
...
<button>Filter</button>
</form>
</div>
);
}
Get current hash value in url
If the url has a #hash
value we can get it using hash
helper function.
import { hash } from "@mongez/react-router";
// if the url is something like https://site-name.com/online-store/products/10#comments
console.log(hash()); // comments
To get the value with the hash, pass an argument with true value to the function.
import { hash } from "@mongez/react-router";
// if the url is something like https://site-name.com/online-store/products/10#comments
console.log(hash(true)); // #comments
Get query string
To get a value from query string, we can use the queryString
helper function, this function returns three internal methods.
To get a key value from query string, we can use get
method:
import { queryString } from "@mongez/react-router";
// if the url is something like https://site-name.com/online-store/products?sortBy=price&categoryId=10
console.log(queryString().get("sortBy")); // price
console.log(queryString().get("categoryId")); // 10
Please note that if you're going to get multiple values from query string, initiate the queryString()
in a variable then use the variable instead.
import { queryString } from "@mongez/react-router";
// if the url is something like https://site-name.com/online-store/products?sortBy=price&categoryId=10
const params = queryString();
console.log(params.get("sortBy")); // price
console.log(params.get("categoryId")); // 10
You can also get a default value if the query param does not exist in the url.
import { queryString } from "@mongez/react-router";
// if the url is something like https://site-name.com/online-store/products?sortBy=price&categoryId=10
const params = queryString();
console.log(params.get("sortBy")); // price
console.log(params.get("sortDirection", "desc")); // desc
To get all query string params in object, we can use the all
method.
import { queryString } from "@mongez/react-router";
// if the url is something like https://site-name.com/online-store/products?sortBy=price&categoryId=10
console.log(queryString().all()); // {sortBy: price, categoryId: 10}
To get it as a string use toString
method.
import { queryString } from "@mongez/react-router";
// if the url is something like https://site-name.com/online-store/products?sortBy=price&categoryId=10
console.log(queryString().toString()); // sortBy=price&categoryId=10
Each time you call
queryString()
it starts collecting values from the query params value, so be aware to not cache its value if the query string params will be changed you need to call the method again.
✅
// src/front-office/products/components/ProductsListPage.tsx
import { queryString } from "@mongez/react-router";
// if the url is something like https://site-name.com/online-store/products?sortBy=price&categoryId=10
console.log(queryString().toString()); // sortBy=price&categoryId=10
import React from "react";
export default function ProductsListPage() {
const params = queryString();
console.log(params.get("sortBy")); // price
// Or you can cache its value when navigating to the products list each time
const params = React.useMemo(() => queryString(), []);
return <div>//</div>;
}
❌
// src/front-office/products/components/ProductsListPage.tsx
import { queryString } from "@mongez/react-router";
// if the url is something like https://site-name.com/online-store/products?sortBy=price&categoryId=10
console.log(queryString().toString()); // sortBy=price&categoryId=10
import React from "react";
const params = queryString();
export default function ProductsListPage() {
console.log(params.get("sortBy")); // empty string
return <div>//</div>;
}
Get project base url
To get the base url, import baseUrl
helper function, this will get the domain path suffixed with basePath
that is defined in the router configurations section.
import { baseUrl } from "@mongez/react-router";
console.log(baseUrl()); // something like https://sitename.com/online-store where /online-store is the basePath of the project.
Get current page direction
To get current page direction use currentDirection
helper.
currentDirection(): string
import { currentDirection } from "@mongez/react-router";
console.log(currentDirection()); // ltr for example
Please note that this utility depends on document.documentElement
's dir
property, if not set, then ltr
will be returned as default.
Direction Is
Check if current direction matches the given direction, using directionIs
utility.
directionIs(direction: 'ltr' | 'rtl'): boolean
import { directionIs } from "@mongez/react-router";
console.log(directionIs("ltr")); // true
console.log(directionIs("rtl")); // false
Get Previous Route
To get previous route use previousRoute
function.
import { previousRoute, navigateTo } from "@mongez/react-router";
navigateTo("/login");
console.log(previousRoute()); // /
navigateTo("/");
console.log(previousRoute()); // /login
Router Events
You may listen to any router change based on the navigation link change.
import { routerEvents } from "@mongez/react-router";
routerEvents.onChange(() => {
// route changed
});
Conclusion
The main purpose of the package is to write a good structure routing system from basic to complex scenarios, i've used it in dozens of projects so far and it works like a charm.
For more documentation View the Github Repository
Feel free to send me your feedback.
Posted on September 3, 2022
Join Our Newsletter. No Spam, Only the good stuff.
Sign up to receive the latest update from our blog.