This repository has been archived by the owner on Jan 6, 2025. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 771
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
fix(matchMediaObservable): expose observable for rxjs operators
* rename token to 'ObservableMediaService' * rename MatchMediaObservableProvider to ObservableMediaServiceProvider * add method `.asObservable()` to MediaService fixes #125.
- Loading branch information
1 parent
0ca7d07
commit aab836d
Showing
13 changed files
with
223 additions
and
145 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,140 @@ | ||
/** | ||
* @license | ||
* Copyright Google Inc. All Rights Reserved. | ||
* | ||
* Use of this source code is governed by an MIT-style license that can be | ||
* found in the LICENSE file at https://angular.io/license | ||
*/ | ||
|
||
import {OpaqueToken} from '@angular/core'; // tslint:disable-line:no-unused-variable | ||
|
||
import {Subscription} from 'rxjs/Subscription'; | ||
import {Observable, Subscribable} from "rxjs/Observable"; | ||
import 'rxjs/add/operator/map'; | ||
import 'rxjs/add/operator/filter'; | ||
|
||
import {BreakPointRegistry} from './breakpoints/break-point-registry'; | ||
|
||
import {MediaChange} from './media-change'; | ||
import {MatchMedia} from './match-media'; | ||
import {mergeAlias} from './../utils/add-alias'; | ||
import {BreakPoint} from './breakpoints/break-point'; | ||
|
||
|
||
/** | ||
* Opaque Token unique to the flex-layout library. | ||
* Note: Developers must use this token when building their own custom | ||
* `ObservableMediaServiceProvider` provider. | ||
* | ||
* @see ./providers/match-media-observable-provider.ts | ||
*/ | ||
// tslint:disable-next-line:variable-name | ||
export const ObservableMediaService: OpaqueToken = new OpaqueToken('flex-layout-media-service'); | ||
|
||
|
||
/** | ||
* Factory returns a simple service instance that exposes a feature to subscribe to mediaQuery | ||
* changes and a validator to test if a mediaQuery (or alias) is currently active. | ||
* | ||
* !! This is not an actual Observable. It is a wrapper of an Observable used to publish additional | ||
* methods like `isActive(<alias>). To access the Observable and use RxJS operators, use | ||
* `.toObservable()` with syntax like media.toObservable().map(....). | ||
* | ||
* !! Only mediaChange activations (not de-activations) are announced by the ObservableMediaService | ||
* | ||
* This factory uses the BreakPoint Registry to inject alias information into the raw MediaChange | ||
* notification. For custom mediaQuery notifications, alias information will not be injected and | ||
* those fields will be ''. | ||
* | ||
* @return Object with two (2) methods: subscribe(observer) and isActive(alias|query) | ||
*/ | ||
export class MediaService implements Subscribable<MediaChange> { | ||
private observable$: Observable<MediaChange>; | ||
|
||
constructor(private mediaWatcher: MatchMedia, | ||
private breakpoints: BreakPointRegistry) { | ||
this._registerBreakPoints(); | ||
this.observable$ = this._buildObservable(); | ||
} | ||
|
||
/** | ||
* Test if specified query/alias is active. | ||
*/ | ||
isActive(alias): boolean { | ||
let query = this._toMediaQuery(alias); | ||
return this.mediaWatcher.isActive(query); | ||
}; | ||
|
||
/** | ||
* Proxy to the Observable subscribe method | ||
*/ | ||
subscribe(next?: (value: MediaChange) => void, | ||
error?: (error: any) => void, | ||
complete?: () => void): Subscription { | ||
return this.observable$.subscribe(next, error, complete); | ||
}; | ||
|
||
/** | ||
* Access to observable for use with operators like | ||
* .filter(), .map(), etc. | ||
*/ | ||
asObservable(): Observable<MediaChange> { | ||
return this.observable$; | ||
} | ||
|
||
// ************************************************ | ||
// Internal Methods | ||
// ************************************************ | ||
|
||
/** | ||
* Register all the mediaQueries registered in the BreakPointRegistry | ||
* This is needed so subscribers can be auto-notified of all standard, registered | ||
* mediaQuery activations | ||
*/ | ||
private _registerBreakPoints() { | ||
this.breakpoints.items.forEach((bp: BreakPoint) => { | ||
this.mediaWatcher.registerQuery(bp.mediaQuery); | ||
return bp; | ||
}); | ||
} | ||
|
||
/** | ||
* Prepare internal observable | ||
* NOTE: the raw MediaChange events [from MatchMedia] do not contain important alias information | ||
* these must be injected into the MediaChange | ||
*/ | ||
private _buildObservable() { | ||
return this.mediaWatcher.observe() | ||
.filter((change: MediaChange) => { | ||
// Only pass/announce activations (not de-activations) | ||
return change.matches === true; | ||
}) | ||
.map((change: MediaChange) => { | ||
// Inject associated (if any) alias information into the MediaChange event | ||
return mergeAlias(change, this._findByQuery(change.mediaQuery)); | ||
}); | ||
} | ||
|
||
/** | ||
* Breakpoint locator by alias | ||
*/ | ||
private _findByAlias(alias) { | ||
return this.breakpoints.findByAlias(alias); | ||
}; | ||
|
||
/** | ||
* Breakpoint locator by mediaQuery | ||
*/ | ||
private _findByQuery(query) { | ||
return this.breakpoints.findByQuery(query); | ||
}; | ||
|
||
/** | ||
* Find associated breakpoint (if any) | ||
*/ | ||
private _toMediaQuery(query) { | ||
let bp: BreakPoint = this._findByAlias(query) || this._findByQuery(query); | ||
return bp ? bp.mediaQuery : query; | ||
}; | ||
|
||
} |
Oops, something went wrong.