2016-06-23 09:47:54 -07:00
|
|
|
/**
|
|
|
|
* @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
|
|
|
|
*/
|
|
|
|
|
2016-06-10 18:57:03 +03:00
|
|
|
import {LocationStrategy} from '@angular/common';
|
2016-06-30 14:01:26 -07:00
|
|
|
import {Directive, HostBinding, HostListener, Input, OnChanges, OnDestroy} from '@angular/core';
|
|
|
|
import {Subscription} from 'rxjs/Subscription';
|
2016-06-08 11:13:41 -07:00
|
|
|
|
2016-06-30 14:01:26 -07:00
|
|
|
import {NavigationEnd, Router} from '../router';
|
2016-05-26 16:51:56 -07:00
|
|
|
import {ActivatedRoute} from '../router_state';
|
2016-06-15 09:01:05 -07:00
|
|
|
import {UrlTree} from '../url_tree';
|
|
|
|
|
2016-06-30 14:01:26 -07:00
|
|
|
|
2016-05-24 14:33:34 -07:00
|
|
|
/**
|
2016-09-10 16:50:38 -07:00
|
|
|
* @whatItDoes Lets you link to specific parts of your app.
|
|
|
|
*
|
|
|
|
* @howToUse
|
2016-05-24 14:33:34 -07:00
|
|
|
*
|
|
|
|
* Consider the following route configuration:
|
|
|
|
|
|
|
|
* ```
|
2016-06-28 14:49:29 -07:00
|
|
|
* [{ path: 'user/:name', component: UserCmp }]
|
2016-05-24 14:33:34 -07:00
|
|
|
* ```
|
|
|
|
*
|
2016-09-10 16:50:38 -07:00
|
|
|
* When linking to this `user/:name` route, you can write:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* <a routerLink='/user/bob'>link to user component</a>
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* @description
|
|
|
|
*
|
2016-09-12 09:47:44 -07:00
|
|
|
* The RouterLink directives let you link to specific parts of your app.
|
2016-09-10 16:50:38 -07:00
|
|
|
*
|
|
|
|
* Whe the link is static, you can use the directive as follows:
|
2016-05-24 14:33:34 -07:00
|
|
|
*
|
|
|
|
* ```
|
2016-09-10 16:50:38 -07:00
|
|
|
* <a routerLink="/user/bob">link to user component</a>
|
2016-06-28 14:49:29 -07:00
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* If you use dynamic values to generate the link, you can pass an array of path
|
|
|
|
* segments, followed by the params for each segment.
|
|
|
|
*
|
|
|
|
* For instance `['/team', teamId, 'user', userName, {details: true}]`
|
|
|
|
* means that we want to generate a link to `/team/11/user/bob;details=true`.
|
2016-09-10 16:50:38 -07:00
|
|
|
*
|
2016-06-28 14:49:29 -07:00
|
|
|
* Multiple static segments can be merged into one (e.g., `['/team/11/user', userName, {details:
|
|
|
|
true}]`).
|
|
|
|
*
|
|
|
|
* The first segment name can be prepended with `/`, `./`, or `../`:
|
|
|
|
* * If the first segment begins with `/`, the router will look up the route from the root of the
|
|
|
|
app.
|
|
|
|
* * If the first segment begins with `./`, or doesn't begin with a slash, the router will
|
|
|
|
* instead look in the children of the current activated route.
|
|
|
|
* * And if the first segment begins with `../`, the router will go up one level.
|
|
|
|
*
|
|
|
|
* You can set query params and fragment as follows:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* <a [routerLink]="['/user/bob']" [queryParams]="{debug: true}" fragment="education">link to user
|
|
|
|
component</a>
|
2016-05-24 14:33:34 -07:00
|
|
|
* ```
|
2016-06-28 14:49:29 -07:00
|
|
|
* RouterLink will use these to generate this link: `/user/bob#education?debug=true`.
|
2016-05-24 14:33:34 -07:00
|
|
|
*
|
2016-07-20 14:30:04 -07:00
|
|
|
* You can also tell the directive to preserve the current query params and fragment:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* <a [routerLink]="['/user/bob']" preserveQueryParams preserveFragment>link to user
|
|
|
|
component</a>
|
|
|
|
* ```
|
|
|
|
*
|
2016-07-22 18:32:26 -07:00
|
|
|
* The router link directive always treats the provided input as a delta to the current url.
|
2016-07-22 13:25:48 -07:00
|
|
|
*
|
|
|
|
* For instance, if the current url is `/user/(box//aux:team)`.
|
|
|
|
*
|
|
|
|
* Then the following link `<a [routerLink]="['/user/jim']">Jim</a>` will generate the link
|
2016-09-10 16:50:38 -07:00
|
|
|
* `/user/(jim//aux:team)`.
|
|
|
|
*
|
|
|
|
* @selector ':not(a)[routerLink]'
|
|
|
|
* @ngModule RouterModule
|
|
|
|
*
|
|
|
|
* See {@link Router.createUrlTree} for more information.
|
2016-07-22 13:25:48 -07:00
|
|
|
*
|
2016-06-28 14:49:29 -07:00
|
|
|
* @stable
|
2016-05-24 14:33:34 -07:00
|
|
|
*/
|
2016-06-28 16:47:13 -07:00
|
|
|
@Directive({selector: ':not(a)[routerLink]'})
|
|
|
|
export class RouterLink {
|
|
|
|
@Input() queryParams: {[k: string]: any};
|
|
|
|
@Input() fragment: string;
|
2016-07-20 14:30:04 -07:00
|
|
|
@Input() preserveQueryParams: boolean;
|
|
|
|
@Input() preserveFragment: boolean;
|
2016-11-15 05:30:13 +03:00
|
|
|
@Input() skipLocationChange: boolean;
|
|
|
|
@Input() replaceUrl: boolean;
|
|
|
|
private commands: any[] = [];
|
2016-06-28 16:47:13 -07:00
|
|
|
|
|
|
|
constructor(
|
|
|
|
private router: Router, private route: ActivatedRoute,
|
|
|
|
private locationStrategy: LocationStrategy) {}
|
|
|
|
|
|
|
|
@Input()
|
|
|
|
set routerLink(data: any[]|string) {
|
|
|
|
if (Array.isArray(data)) {
|
2016-06-28 16:53:54 -07:00
|
|
|
this.commands = data;
|
2016-06-28 16:47:13 -07:00
|
|
|
} else {
|
|
|
|
this.commands = [data];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-01 14:35:41 -07:00
|
|
|
@HostListener('click', [])
|
|
|
|
onClick(): boolean {
|
2016-07-07 15:29:54 -07:00
|
|
|
this.router.navigateByUrl(this.urlTree);
|
2016-11-01 14:35:41 -07:00
|
|
|
return true;
|
2016-07-07 15:29:54 -07:00
|
|
|
}
|
2016-06-28 16:47:13 -07:00
|
|
|
|
2016-07-07 15:29:54 -07:00
|
|
|
get urlTree(): UrlTree {
|
2016-07-20 14:30:04 -07:00
|
|
|
return this.router.createUrlTree(this.commands, {
|
|
|
|
relativeTo: this.route,
|
|
|
|
queryParams: this.queryParams,
|
|
|
|
fragment: this.fragment,
|
|
|
|
preserveQueryParams: toBool(this.preserveQueryParams),
|
2016-11-15 05:30:13 +03:00
|
|
|
preserveFragment: toBool(this.preserveFragment),
|
|
|
|
skipLocationChange: toBool(this.skipLocationChange),
|
|
|
|
replaceUrl: toBool(this.replaceUrl),
|
2016-07-20 14:30:04 -07:00
|
|
|
});
|
2016-06-28 16:47:13 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-09-10 16:50:38 -07:00
|
|
|
* @whatItDoes Lets you link to specific parts of your app.
|
|
|
|
*
|
2016-06-28 16:47:13 -07:00
|
|
|
* See {@link RouterLink} for more information.
|
2016-09-10 16:50:38 -07:00
|
|
|
*
|
|
|
|
* @selector 'a[routerLink]'
|
|
|
|
* @ngModule RouterModule
|
|
|
|
*
|
2016-06-28 16:47:13 -07:00
|
|
|
* @stable
|
|
|
|
*/
|
|
|
|
@Directive({selector: 'a[routerLink]'})
|
2016-06-30 14:01:26 -07:00
|
|
|
export class RouterLinkWithHref implements OnChanges, OnDestroy {
|
2016-05-24 14:33:34 -07:00
|
|
|
@Input() target: string;
|
2016-06-08 11:13:41 -07:00
|
|
|
@Input() queryParams: {[k: string]: any};
|
2016-06-06 15:44:12 -07:00
|
|
|
@Input() fragment: string;
|
2016-07-20 14:30:04 -07:00
|
|
|
@Input() routerLinkOptions: {preserveQueryParams: boolean, preserveFragment: boolean};
|
|
|
|
@Input() preserveQueryParams: boolean;
|
|
|
|
@Input() preserveFragment: boolean;
|
2016-11-15 05:30:13 +03:00
|
|
|
@Input() skipLocationChange: boolean;
|
|
|
|
@Input() replaceUrl: boolean;
|
|
|
|
private commands: any[] = [];
|
2016-06-30 14:01:26 -07:00
|
|
|
private subscription: Subscription;
|
2016-05-24 14:33:34 -07:00
|
|
|
|
|
|
|
// the url displayed on the anchor element.
|
|
|
|
@HostBinding() href: string;
|
|
|
|
|
2016-06-15 15:45:42 -07:00
|
|
|
constructor(
|
|
|
|
private router: Router, private route: ActivatedRoute,
|
2016-06-30 14:01:26 -07:00
|
|
|
private locationStrategy: LocationStrategy) {
|
|
|
|
this.subscription = router.events.subscribe(s => {
|
|
|
|
if (s instanceof NavigationEnd) {
|
|
|
|
this.updateTargetUrlAndHref();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2016-05-24 14:33:34 -07:00
|
|
|
|
|
|
|
@Input()
|
2016-06-08 11:13:41 -07:00
|
|
|
set routerLink(data: any[]|string) {
|
2016-05-24 14:33:34 -07:00
|
|
|
if (Array.isArray(data)) {
|
2016-06-28 16:53:54 -07:00
|
|
|
this.commands = data;
|
2016-05-24 14:33:34 -07:00
|
|
|
} else {
|
2016-05-26 16:51:56 -07:00
|
|
|
this.commands = [data];
|
2016-05-24 14:33:34 -07:00
|
|
|
}
|
2016-06-06 15:44:12 -07:00
|
|
|
}
|
|
|
|
|
2016-06-08 11:13:41 -07:00
|
|
|
ngOnChanges(changes: {}): any { this.updateTargetUrlAndHref(); }
|
2016-06-30 14:01:26 -07:00
|
|
|
ngOnDestroy(): any { this.subscription.unsubscribe(); }
|
2016-06-06 15:44:12 -07:00
|
|
|
|
2016-06-15 15:45:42 -07:00
|
|
|
@HostListener('click', ['$event.button', '$event.ctrlKey', '$event.metaKey'])
|
2016-06-10 18:57:03 +03:00
|
|
|
onClick(button: number, ctrlKey: boolean, metaKey: boolean): boolean {
|
|
|
|
if (button !== 0 || ctrlKey || metaKey) {
|
|
|
|
return true;
|
2016-05-24 14:33:34 -07:00
|
|
|
}
|
2016-06-10 18:57:03 +03:00
|
|
|
|
|
|
|
if (typeof this.target === 'string' && this.target != '_self') {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.router.navigateByUrl(this.urlTree);
|
|
|
|
return false;
|
2016-05-24 14:33:34 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private updateTargetUrlAndHref(): void {
|
2016-08-25 07:27:22 -07:00
|
|
|
this.href = this.locationStrategy.prepareExternalUrl(this.router.serializeUrl(this.urlTree));
|
|
|
|
}
|
|
|
|
|
|
|
|
get urlTree(): UrlTree {
|
|
|
|
return this.router.createUrlTree(this.commands, {
|
2016-07-20 14:30:04 -07:00
|
|
|
relativeTo: this.route,
|
|
|
|
queryParams: this.queryParams,
|
|
|
|
fragment: this.fragment,
|
|
|
|
preserveQueryParams: toBool(this.preserveQueryParams),
|
2016-11-15 05:30:13 +03:00
|
|
|
preserveFragment: toBool(this.preserveFragment),
|
|
|
|
skipLocationChange: toBool(this.skipLocationChange),
|
|
|
|
replaceUrl: toBool(this.replaceUrl),
|
2016-07-20 14:30:04 -07:00
|
|
|
});
|
2016-05-24 14:33:34 -07:00
|
|
|
}
|
|
|
|
}
|
2016-07-20 14:30:04 -07:00
|
|
|
|
|
|
|
function toBool(s?: any): boolean {
|
|
|
|
if (s === '') return true;
|
|
|
|
return !!s;
|
2016-11-15 05:30:13 +03:00
|
|
|
}
|