Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/tarsisexistence/routeshub
ðĶA route management library + pattern for Angular Router
https://github.com/tarsisexistence/routeshub
angular hub javascript management routes routing rxjs state store typescript
Last synced: 4 days ago
JSON representation
ðĶA route management library + pattern for Angular Router
- Host: GitHub
- URL: https://github.com/tarsisexistence/routeshub
- Owner: tarsisexistence
- License: mit
- Archived: true
- Created: 2018-11-07T20:04:59.000Z (about 6 years ago)
- Default Branch: master
- Last Pushed: 2023-12-21T21:22:00.000Z (about 1 year ago)
- Last Synced: 2024-10-02T18:09:31.441Z (4 months ago)
- Topics: angular, hub, javascript, management, routes, routing, rxjs, state, store, typescript
- Language: TypeScript
- Homepage: https://routeshub.gitbook.io
- Size: 5.33 MB
- Stars: 47
- Watchers: 3
- Forks: 5
- Open Issues: 12
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- Contributing: CONTRIBUTING.md
- License: LICENSE
- Code of conduct: CODE_OF_CONDUCT.md
Awesome Lists containing this project
README
# Routeshub
A **route manager** and pattern for **[Angular]**
* **Manager.** Introduces a new route management experience that leads to better application control.
* **Navigation.** It provides declarative navigation experience.
* **Fast.** In addition to speeding up development, it works as fast as it does without it.
* **Pluggable.** Engineered as a plug-in and designed to be added at any time during the development process.
* **Pattern.** It provides a unified approach managing the routing of the entire application.
* **Small.** ~3.5 kB (minified + gzipped). It uses [Angular] and [rxjs] as peerDependencies.In short, this is an add-on to the **@angular/router**, which provides state-based routing for medium to large-size applications.
Read more about Routeshub on the [docs site](https://routeshub.gitbook.io)
## Install
To get started, you need to install the package from npm
```sh
npm install routeshub
```## Usage
#### Declare a simple **[module]**.routes.ts file
```typescript
...export const routes: Routes = [
{
path: '',
pathMatch: 'full',
component: AppComponent
},
{
path: 'about',
loadChildren: () => import('example-app/app/views/about/about.module').then(m => m.AboutModule)
},
{
path: '**',
redirectTo: ''
}
];
```#### Getting interface based on our routes and unique unit key (as symbol) in **[module]**.notes.ts
```typescript
import { Note } from 'routeshub';export interface AppNotes {
root: Note; // === ''
about: Note; // === 'about'
wildcard: Note; // === '**'
}export const APP_NOTES_KEY = Symbol();
```#### Creating a hub **[module]**.hub.ts file
```typescript
import { NgModule } from '@angular/core';
import { RouterModule } from '@angular/router';
import { NavigationModule, connectFeatures, createRoot } from 'routeshub';
import { routes } from './app.routes';
import { APP_NOTES_KEY, AppNotes } from './app.notes';
import { aboutUnit } from '../views/about/hub/about.hub';/**
* Creates stateful named App routes
*/
createRoot(routes, { key: APP_NOTES_KEY });/**
* connects features which have attached relating to the parent module
*
* about module has its routes which we want to connect
*/
connectFeatures(APP_NOTES_KEY, {
about: aboutUnit
});/**
* Routing module contains its configuration
* and providers (resolvers, guard, interceptors, etc.)
*
* Exports RouterModule
* and NavigationModule for navLink directives
*/
@NgModule({
imports: [RouterModule.forRoot(routes)],
exports: [RouterModule, NavigationModule]
})
export class AppHub {}
```#### Component
```typescript
...import { getUnit, Unit, Secluded } from 'routeshub';
import { AppNotes, APP_NOTES_KEY } from '../hub/app.notes';@Component({
selector: 'app-header',
template: `
Home
Location
`
})
export class HeaderComponent {
// getting unit by key
@Secluded(APP_NOTES_KEY)
public app: Unit;
//or
// getting the unit by name
@Secluded('app')
public app: Unit;
//or
// getting unit from getUnit function
public app = getUnit(APP_NOTES_KEY);
}
```[Angular]: https://github.com/storeon/angular
[rxjs]: https://github.com/ReactiveX/rxjsYou can find a more complex example in this [repo](https://github.com/retarsis/routeshub/tree/master/example-app) here or on [GitBook](https://routeshub.gitbook.io/docs/example).
# Hub
Routeshub offers an approach (pattern) to structure routing in the application. You can explore it in this [repo](https://github.com/retarsis/routeshub/tree/master/example-app).**Guideline**:
- Each module in the application has a hub directory
- hub directory contains three files that configure the routing of the current module.
1. hub - sets routing configuration; exports RoutingModule and NavigationModule; connects feature units.
2. notes - it is a place for interfaces and the unique key of the unit.
3. routes - simple routes file as is without any changes.![diagram](http://i.piccy.info/i9/baf248d94b1260e13d5bff6acb6e6727/1569138770/128577/1338898/2019_09_22_10_51_00.jpg)
# Concepts
## Unit
`Unit` is a modular entity which contains stateful module routes.![unit diagram](http://i.piccy.info/i9/395309194dfc663ad7c07dad3c482d15/1569140574/118208/1338898/unit.jpg)
There are two ways to create the `unit`:
- **createRoot**
- **createFeature**Each creator takes the `routes: Routes` and an object of options
- key - unit identifier and accepts string or symbol
- routeName - accepts an object with optional custom names for a wildcard ('**') and root ('') paths
- nearby - accepts lazy units (**connectors**), which are outputs of **feature creator**. A nearby option should use when one or more connected features are eager modules with their own routes files.**Root** creator invokes only once to initialize the hub in the application. `createRoot` takes initial `appNotes`.
**Usage example**:
```typescript
createRoot(routes, {
key: APP_NOTES_KEY,
routeName: { root: 'home', wildcard: 'notFound' },
nearby: { map: mapUnit }
});
```In turn, the **feature** creator is responsible for creating lazy units (**connectors**), which should connect to the parent unit.
```typescript
export const routes: Routes = [
{ path: '', component: AboutComponent }
];/**
* when module has only one root ('') path
* you can declare short type
*/
export type AboutNotes = Root;const ABOUT_NOTES_KEY = Symbol();
export const aboutConnector: Connector = createFeature(aboutNote, { key: ABOUT_NOTES_KEY });
```
## Note
You need to understand how routeshub transforms routes into the keys. All notes related things routeshub handles internally.The `Note` is input to reproduce the unit. Each `Note` represents one route.
**In short, it assigns the names to the 'spots'**
The example below shows capabilities and illustrates how this works. Unnecessary route information shortened.
```typescript
export const routes: Routes = [
{
path: '', // name => 'root' by default (customizable)
children: [
{ path: '' }, // name => 'root' by default (customizable)
{ path: 'about' } // name => about
]
},
{ path: ':first_name'}, // name => firstName
{ path: 'person/:person-age' }, // name => personAge
{ path: '**'} // name => 'wildcard' by default (customizable). In example we'll rename it to 'notFound'
];// Root interface helps to short frequently repeated actions
export interface AppChildNotes extends Root {
about: Note;
}/**
* it is equivalent of the previous interface
*
export interface AppChildNotes {
root: Note;
about: Note;
}
*//**
* btw, this shorthand provides an opportunity to describe root children interface through generics
*/
export interface AppNotes extends Root {
firstName: Note;
personAge: Note;
notFound: Note;
}/**
* it is equivalent of the previous interface
*
export interface AppNotes {
root: Note
firstName: Note;
personAge: Note;
wildcard: Note;
}
*/export const appUnit = createRoot(routes, { routeName: { wildcard: 'notFound' }});
```
## Get Unit
Essentially, you need the unit to pass it into a directive/decorator for navigation purposes.
There are several ways to get a unit:- **@United decorator**. Apply this decorator on the component's prop. You should pass the key or unit name.
```typescript
@Component({
...
})
export class HeaderComponent {
// getting unit by key
@Seclude(APP_NOTES_KEY)
private app: Unit;// getting unit by unit name
@Seclude('about')
private about: Unit;
}
```- **get unit** - This is a function that works as a decorator. Primarily, it created as an alternative to @Secluded decorator.
```typescript
@Component({
...
})
export class HeaderComponent {
// getting unit by key
private app = getUnit(APP_NOTES_KEY);// getting unit by name
private about = getUnit('about');
}
```- **getRegisteredUnits** - This is a function that returns all declared units in the project.
```typescript
@Component({
...
})
export class HeaderComponent {
public hub: Units = getRegisteredUnits();
}
```
## Navigation
**Routeshub** provides directives and functions to make your experience with navigation better.Before you start, don't forget to import **NavigationModule**.
It's a good practice importing NavigationModule into the **[module]**.hub.ts file.### navLink
```html
Profile
Map
```
### forwardParams
A function that inserts parameters into the route's state and outputs a ready-made dynamic state.
```typescript
...
@Component({
...
})
export class ExampleComponent {
...
constructor(private router: Router) {}
public navigateToProfile(): void {
const toProfile = forwardParams(this.userUnit.profileId.state, { profileId: 77 })
this.router.navigate(toProfile);
}
}
```
## Changelog
Stay tuned with [changelog](https://github.com/retarsis/routeshub/blob/master/CHANGELOG.md).
## License
This project is [MIT](https://choosealicense.com/licenses/mit/) licensed.
## Contributing
Contributions are welcome. For major changes, please open an issue first to discuss what you would like to change.
If you made a PR, make sure to update tests as appropriate and keep the examples consistent.
## Contributors âĻ
Thanks go to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)):
Max Tarsis
ðĪ ðŧ ð ð ð
Lars Gyrup Brink Nielsen
ð ðĪ
Denis Makarov
ðŧ ðĪ
This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome!