148 lines
4.2 KiB
TypeScript
Raw Normal View History

/**
* @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 {LocationStrategy} from '@angular/common';
import {Directive, HostBinding, HostListener, Input, OnChanges} from '@angular/core';
2016-06-08 11:13:41 -07:00
2016-05-24 14:33:34 -07:00
import {Router} from '../router';
import {ActivatedRoute} from '../router_state';
import {UrlTree} from '../url_tree';
2016-05-24 14:33:34 -07:00
/**
* The RouterLink directive lets you link to specific parts of your app.
*
* 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
* ```
*
* When linking to this `User` route, you can write:
*
* ```
2016-06-28 14:49:29 -07:00
* <a [routerLink]="/user/bob">link to user component</a>
* ```
*
* 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`.
* 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-06-28 14:49:29 -07:00
* @stable
2016-05-24 14:33:34 -07:00
*/
@Directive({selector: ':not(a)[routerLink]'})
export class RouterLink {
private commands: any[] = [];
@Input() queryParams: {[k: string]: any};
@Input() fragment: string;
urlTree: UrlTree;
constructor(
private router: Router, private route: ActivatedRoute,
private locationStrategy: LocationStrategy) {}
@Input()
set routerLink(data: any[]|string) {
if (Array.isArray(data)) {
this.commands = data;
} else {
this.commands = [data];
}
}
@HostListener('click', ['$event.button', '$event.ctrlKey', '$event.metaKey'])
onClick(button: number, ctrlKey: boolean, metaKey: boolean): boolean {
if (button !== 0 || ctrlKey || metaKey) {
return true;
}
this.router.navigate(
this.commands,
{relativeTo: this.route, queryParams: this.queryParams, fragment: this.fragment});
return false;
}
}
/**
* See {@link RouterLink} for more information.
* @stable
*/
@Directive({selector: 'a[routerLink]'})
export class RouterLinkWithHref implements OnChanges {
2016-05-24 14:33:34 -07:00
@Input() target: string;
private commands: any[] = [];
2016-06-08 11:13:41 -07:00
@Input() queryParams: {[k: string]: any};
@Input() fragment: string;
2016-05-24 14:33:34 -07:00
// the url displayed on the anchor element.
@HostBinding() href: string;
urlTree: UrlTree;
/**
* @internal
*/
constructor(
private router: Router, private route: ActivatedRoute,
private locationStrategy: LocationStrategy) {}
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)) {
this.commands = data;
2016-05-24 14:33:34 -07:00
} else {
this.commands = [data];
2016-05-24 14:33:34 -07:00
}
}
2016-06-08 11:13:41 -07:00
ngOnChanges(changes: {}): any { this.updateTargetUrlAndHref(); }
@HostListener('click', ['$event.button', '$event.ctrlKey', '$event.metaKey'])
onClick(button: number, ctrlKey: boolean, metaKey: boolean): boolean {
if (button !== 0 || ctrlKey || metaKey) {
return true;
2016-05-24 14:33:34 -07: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 {
this.urlTree = this.router.createUrlTree(
2016-06-08 11:13:41 -07:00
this.commands,
{relativeTo: this.route, queryParams: this.queryParams, fragment: this.fragment});
if (this.urlTree) {
this.href = this.locationStrategy.prepareExternalUrl(this.router.serializeUrl(this.urlTree));
2016-05-24 14:33:34 -07:00
}
}
}