A lib for Angular that binds media-queries and @ngrx/store together. The idea behind the creation of the library cames from a talk of Michael Madsen at Rxjs Live called "Reactive Responsive Design" (Source).
Possible uses cases are :
- display / hide a component or enable a functionnality according to a specific media query
- compose selectors with media queries to make the UI more responsive (ie: if the screen width is lower than 400px AND the menu is open, then put the content on two columns)
By default, the following
Npm : npm i @yoozly/ngrx-mediaqueries
Yarn : yarn add @yoozly/ngrx-mediaqueries
In the root module of your application, add the reducer ngrxMediaQueriesReducer()
. In order to merge ngrxMediaQueriesReducer()
in the map of root reducers, use MERGE_REDUCERS injection token, like below :
import {
MERGE_REDUCERS,
ngrxMediaQueriesReducer
} from "@yoozly/ngrx-mediaqueries";
@NgModule({
StoreModule.forRoot(MERGE_REDUCERS, {
metaReducers: [someMetaReducers]
}),
providers: [
{
provide: MERGE_REDUCERS,
useFactory: (): ActionReducerMap<any> => ({
...reducers,
...ngrxMediaQueriesReducer()
})
}
],
})
export class RootModule {}
Use APP_INITIALIZER token to start the library automatically.
import { APP_INITIALIZER } from "@angular/core";
import { MediaQueriesService } from "@yoozly/ngrx-mediaqueries";
@NgModule({
providers: [
{
provide: APP_INITIALIZER,
useFactory: (init: MediaQueriesService) => _ => init,
deps: [MediaQueriesService],
multi: true
}
]
})
export class RootModule {}
Add NgrxMediaqueriesModule
in the array of imports.
import { NgrxMediaqueriesModule } from "@yoozly/ngrx-mediaqueries";
@NgModule({
imports: [NgrxMediaqueriesModule]
})
export class RootModule {}
By default, the following media queries are watched :
- xxsmall: "(min-width:375px)"
- xsmall: "(min-width:480px)"
- small: "(min-width:600px)"
- medium: "(min-width:768px)"
- large: "(min-width:1024px)"
- xlarge: "(min-width:1280px)"
- xxlarge: "(min-width:1440px)"
- portrait: "(orientation: portrait)"
- landscape: "(orientation: landscape)"
- speech: "speech"
- touchscreen: "(hover: none) and (pointer: coarse)"
- highres: "(-webkit-min-device-pixel-ratio: 2), (min-resolution: 192dpi)"
You can override that behavior and add your own set of media queries by using the NgrxMediaqueriesModule.forRoot({...})
and by passing a configuration object.
import { NgrxMediaqueriesModule } from "@yoozly/ngrx-mediaqueries";
@NgModule({
imports: [
NgrxMediaqueriesModule.forRoot({
small: "(min-width:480px)",
medium: "(min-width:720px)",
large: "(min-width:1024px)"
})
]
})
export class RootModule {}
Here is the full configuration :
import { APP_INITIALIZER } from "@angular/core";
import {
MERGE_REDUCERS,
ngrxMediaQueriesReducer,MediaQueriesService,NgrxMediaqueriesModule
} from "@yoozly/ngrx-mediaqueries";
@NgModule({
imports: [NgrxMediaqueriesModule],
StoreModule.forRoot(MERGE_REDUCERS, {
metaReducers: [someMetaReducers]
}),
providers: [
{
provide: MERGE_REDUCERS,
useFactory: (): ActionReducerMap<any> => ({
...reducers,
...ngrxMediaQueriesReducer()
})
},
{
provide: APP_INITIALIZER,
useFactory: (init: MediaQueriesService) => _ => init,
deps: [MediaQueriesService],
multi: true
}
],
A set of selectors is provided by default. In order to use it, use the NgrxMediaQueriesFacade
service, like that :
import { NgrxMediaQueriesFacade } from "ngrx-mediaqueries";
constructor(
private mqService: NgrxMediaQueriesFacade
) {}
By default, the following selectors are available :
- getMediaQueryStateXXSmall();
- getMediaQueryStateXSmall();
- getMediaQueryStateSmall();
- getMediaQueryStateMedium();
- getMediaQueryStateLarge();
- getMediaQueryStateXLarge();
- getMediaQueryStateXXLarge();
- getMediaQueryStatePortrait();
- getMediaQueryStateLandscape();
- getMediaQueryStateSpeech();
- getMediaQueryStateTouchscreen();
- getMediaQueryStateHighres();
If you have a custom configuration, there is a special selector that accepts the name of the media query as a param :
// in the root module
NgrxMediaqueriesModule.forRoot({
small: "(min-width:480px)",
medium: "(min-width:720px)",
large: "(min-width:1024px)"
});
// then in a component
this.mqService.getMediaQueryState("small");
this.mqService.getMediaQueryState("medium");
this.mqService.getMediaQueryState("large");