@spartacus/storefront
Version:
Spartacus Storefront is a package that you can include in your application, which allows you to add default storefront features.
124 lines • 16 kB
JavaScript
import { Component, Input } from '@angular/core';
import * as i0 from "@angular/core";
import * as i1 from "@angular/router";
import * as i2 from "./generic-link-component.service";
import * as i3 from "@angular/common";
/**
* This component navigates using [routerLink] attribute when input 'url' is a relative url. Otherwise (when it's absolute), [href] is used.
*/
export class GenericLinkComponent {
constructor(router, service) {
this.router = router;
this.service = service;
/**
* Pattern matching string starting with `http://` or `https://`.
*/
this.PROTOCOL_REGEX = /^https?:\/\//i;
/**
* Pattern matching string starting with `mailto:`.
*/
this.MAILTO_PROTOCOL_REGEX = /^mailto:/i;
/**
* Pattern matching string starting with `tel:`.
*/
this.TEL_PROTOCOL_REGEX = /^tel:/i;
/**
* Used to split url into 2 parts:
* 1. the path
* 2. query params + hash fragment
*/
this.URL_SPLIT = /(^[^#?]*)(.*)/;
/**
* Parsed parts of the @Input `url`, when it's a local URL.
* It should not be used when the `url` is external.
* @see `url`
*/
this.routeParts = {};
}
isExternalUrl() {
var _a;
return (((_a = this.service) === null || _a === void 0 ? void 0 : _a.isExternalUrl(this.url)) ||
(typeof this.url === 'string' &&
(this.PROTOCOL_REGEX.test(this.url) ||
this.MAILTO_PROTOCOL_REGEX.test(this.url) ||
this.TEL_PROTOCOL_REGEX.test(this.url))));
}
get rel() {
return this.target === '_blank' ? 'noopener' : null;
}
ngOnChanges(changes) {
if (changes['url']) {
this.setUrlParts(changes['url'].currentValue);
}
}
/**
* The part with the path of the local url.
*/
get routerUrl() {
return this.routeParts.path;
}
/**
* The part with the query params of the local url.
*/
get queryParams() {
return this.routeParts.queryParams;
}
/**
* The part with the hash fragment of the local url.
*/
get fragment() {
return this.routeParts.fragment;
}
/**
* Parses the given url and sets the property `urlParts` accordingly.
*/
setUrlParts(url) {
if (typeof url === 'string') {
url = this.getAbsoluteUrl(url); // string links in CMS sometimes don't have the leading slash, so fix it here
this.routeParts = this.splitUrl(url);
}
else {
this.routeParts = { path: url };
}
}
/**
* Parses the given string into 3 parts:
* - string path (wrapped in an array to be compatible with Angular syntax for the `routerLink`)
* - query params (as an object)
* - hash fragment (string)
*/
splitUrl(url = '') {
const { queryParams, fragment } = this.router.parseUrl(url);
const [, path] = url.match(this.URL_SPLIT);
// wrap path in an array, to have the Angular-like path format
return { path: [path], queryParams, fragment };
}
/**
* Prepends a leading slash to the given URL string, in case it doesn't have it.
*/
getAbsoluteUrl(url) {
return url.startsWith('/') ? url : '/' + url;
}
}
GenericLinkComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: GenericLinkComponent, deps: [{ token: i1.Router }, { token: i2.GenericLinkComponentService }], target: i0.ɵɵFactoryTarget.Component });
GenericLinkComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.0.5", type: GenericLinkComponent, selector: "cx-generic-link", inputs: { url: "url", target: "target", id: "id", class: "class", style: "style", title: "title" }, usesOnChanges: true, ngImport: i0, template: "<!-- https://github.com/angular/angular/issues/24567 -->\n\n<ng-container *ngIf=\"isExternalUrl(); else isLocalUrl\">\n <a\n [href]=\"url\"\n [attr.target]=\"target\"\n [attr.rel]=\"rel\"\n [attr.id]=\"id\"\n [attr.class]=\"class\"\n [attr.style]=\"style\"\n [attr.title]=\"title\"\n >\n <ng-container *ngTemplateOutlet=\"content\"></ng-container>\n </a>\n</ng-container>\n\n<ng-template #isLocalUrl>\n <a\n [routerLink]=\"routerUrl\"\n [queryParams]=\"queryParams\"\n [fragment]=\"fragment\"\n [target]=\"target\"\n [attr.id]=\"id\"\n [attr.class]=\"class\"\n [attr.style]=\"style\"\n [attr.title]=\"title\"\n >\n <ng-container *ngTemplateOutlet=\"content\"></ng-container>\n </a>\n</ng-template>\n\n<ng-template #content>\n <ng-content></ng-content>\n</ng-template>\n", directives: [{ type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i1.RouterLinkWithHref, selector: "a[routerLink],area[routerLink]", inputs: ["routerLink", "target", "queryParams", "fragment", "queryParamsHandling", "preserveFragment", "skipLocationChange", "replaceUrl", "state", "relativeTo"] }] });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: GenericLinkComponent, decorators: [{
type: Component,
args: [{
selector: 'cx-generic-link',
templateUrl: './generic-link.component.html',
}]
}], ctorParameters: function () { return [{ type: i1.Router }, { type: i2.GenericLinkComponentService }]; }, propDecorators: { url: [{
type: Input
}], target: [{
type: Input
}], id: [{
type: Input
}], class: [{
type: Input
}], style: [{
type: Input
}], title: [{
type: Input
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"generic-link.component.js","sourceRoot":"","sources":["../../../../../../projects/storefrontlib/shared/components/generic-link/generic-link.component.ts","../../../../../../projects/storefrontlib/shared/components/generic-link/generic-link.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAA4B,MAAM,eAAe,CAAC;;;;;AAgB3E;;GAEG;AAKH,MAAM,OAAO,oBAAoB;IA2B/B,YACY,MAAc,EACd,OAAqC;QADrC,WAAM,GAAN,MAAM,CAAQ;QACd,YAAO,GAAP,OAAO,CAA8B;QA5BjD;;WAEG;QACc,mBAAc,GAAW,eAAe,CAAC;QAE1D;;WAEG;QACO,0BAAqB,GAAW,WAAW,CAAC;QAEtD;;WAEG;QACO,uBAAkB,GAAW,QAAQ,CAAC;QAkBhD;;;;WAIG;QACc,cAAS,GAAG,eAAe,CAAC;QAE7C;;;;WAIG;QACK,eAAU,GAAe,EAAE,CAAC;IAdjC,CAAC;IAuBJ,aAAa;;QACX,OAAO,CACL,CAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,aAAa,CAAC,IAAI,CAAC,GAAG,CAAC;YACrC,CAAC,OAAO,IAAI,CAAC,GAAG,KAAK,QAAQ;gBAC3B,CAAC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC;oBACjC,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC;oBACzC,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,CAC7C,CAAC;IACJ,CAAC;IAED,IAAI,GAAG;QACL,OAAO,IAAI,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC;IACtD,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,KAAK,CAAC,EAAE;YAClB,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,YAAY,CAAC,CAAC;SAC/C;IACH,CAAC;IAED;;OAEG;IACH,IAAI,SAAS;QACX,OAAO,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC;IAC9B,CAAC;IAED;;OAEG;IACH,IAAI,WAAW;QACb,OAAO,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC;IACrC,CAAC;IAED;;OAEG;IACH,IAAI,QAAQ;QACV,OAAO,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC;IAClC,CAAC;IAED;;OAEG;IACK,WAAW,CAAC,GAAmB;QACrC,IAAI,OAAO,GAAG,KAAK,QAAQ,EAAE;YAC3B,GAAG,GAAG,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC,CAAC,6EAA6E;YAC7G,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAa,CAAC,CAAC;SAChD;aAAM;YACL,IAAI,CAAC,UAAU,GAAG,EAAE,IAAI,EAAE,GAAG,EAAE,CAAC;SACjC;IACH,CAAC;IAED;;;;;OAKG;IACK,QAAQ,CAAC,MAAc,EAAE;QAC/B,MAAM,EAAE,WAAW,EAAE,QAAQ,EAAE,GAAG,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;QAC5D,MAAM,CAAC,EAAE,IAAI,CAAC,GAAG,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAE3C,8DAA8D;QAC9D,OAAO,EAAE,IAAI,EAAE,CAAC,IAAI,CAAC,EAAE,WAAW,EAAE,QAAQ,EAAE,CAAC;IACjD,CAAC;IAED;;OAEG;IACK,cAAc,CAAC,GAAW;QAChC,OAAO,GAAG,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,GAAG,CAAC;IAC/C,CAAC;;iHA7HU,oBAAoB;qGAApB,oBAAoB,gLCvBjC,6zBAkCA;2FDXa,oBAAoB;kBAJhC,SAAS;mBAAC;oBACT,QAAQ,EAAE,iBAAiB;oBAC3B,WAAW,EAAE,+BAA+B;iBAC7C;uIA+CU,GAAG;sBAAX,KAAK;gBACG,MAAM;sBAAd,KAAK;gBACG,EAAE;sBAAV,KAAK;gBACG,KAAK;sBAAb,KAAK;gBACG,KAAK;sBAAb,KAAK;gBACG,KAAK;sBAAb,KAAK","sourcesContent":["import { Component, Input, OnChanges, SimpleChanges } from '@angular/core';\nimport { Params, Router } from '@angular/router';\nimport { GenericLinkComponentService } from './generic-link-component.service';\n\n// private\ninterface RouteParts {\n  /** Path in the Angular-like array format */\n  path?: string[];\n\n  /** Query params */\n  queryParams?: Params;\n\n  /** Hash fragment */\n  fragment?: string;\n}\n\n/**\n * This component navigates using [routerLink] attribute when input 'url' is a relative url. Otherwise (when it's absolute), [href] is used.\n */\n@Component({\n  selector: 'cx-generic-link',\n  templateUrl: './generic-link.component.html',\n})\nexport class GenericLinkComponent implements OnChanges {\n  /**\n   * Pattern matching string starting with `http://` or `https://`.\n   */\n  private readonly PROTOCOL_REGEX: RegExp = /^https?:\\/\\//i;\n\n  /**\n   * Pattern matching string starting with `mailto:`.\n   */\n  protected MAILTO_PROTOCOL_REGEX: RegExp = /^mailto:/i;\n\n  /**\n   * Pattern matching string starting with `tel:`.\n   */\n  protected TEL_PROTOCOL_REGEX: RegExp = /^tel:/i;\n\n  /**\n   * @deprecated since version 5.0\n   * Use the following constructor instead:\n   * ```\n   * constructor(\n   *   protected router: Router,\n   *   protected service?: GenericLinkComponentService\n   * )\n   * ```\n   */\n  constructor(router: Router);\n  constructor(\n    protected router: Router,\n    protected service?: GenericLinkComponentService\n  ) {}\n\n  /**\n   * Used to split url into 2 parts:\n   * 1. the path\n   * 2. query params + hash fragment\n   */\n  private readonly URL_SPLIT = /(^[^#?]*)(.*)/;\n\n  /**\n   * Parsed parts of the @Input `url`, when it's a local URL.\n   * It should not be used when the `url` is external.\n   * @see `url`\n   */\n  private routeParts: RouteParts = {};\n\n  @Input() url: string | any[];\n  @Input() target: string | null;\n  @Input() id: string;\n  @Input() class: string;\n  @Input() style: string;\n  @Input() title: string;\n\n  isExternalUrl(): boolean {\n    return (\n      this.service?.isExternalUrl(this.url) ||\n      (typeof this.url === 'string' &&\n        (this.PROTOCOL_REGEX.test(this.url) ||\n          this.MAILTO_PROTOCOL_REGEX.test(this.url) ||\n          this.TEL_PROTOCOL_REGEX.test(this.url)))\n    );\n  }\n\n  get rel() {\n    return this.target === '_blank' ? 'noopener' : null;\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes['url']) {\n      this.setUrlParts(changes['url'].currentValue);\n    }\n  }\n\n  /**\n   * The part with the path of the local url.\n   */\n  get routerUrl(): any[] {\n    return this.routeParts.path;\n  }\n\n  /**\n   * The part with the query params of the local url.\n   */\n  get queryParams(): Params {\n    return this.routeParts.queryParams;\n  }\n\n  /**\n   * The part with the hash fragment of the local url.\n   */\n  get fragment(): string {\n    return this.routeParts.fragment;\n  }\n\n  /**\n   * Parses the given url and sets the property `urlParts` accordingly.\n   */\n  private setUrlParts(url: string | any[]) {\n    if (typeof url === 'string') {\n      url = this.getAbsoluteUrl(url); // string links in CMS sometimes don't have the leading slash, so fix it here\n      this.routeParts = this.splitUrl(url as string);\n    } else {\n      this.routeParts = { path: url };\n    }\n  }\n\n  /**\n   * Parses the given string into 3 parts:\n   * - string path (wrapped in an array to be compatible with Angular syntax for the `routerLink`)\n   * - query params (as an object)\n   * - hash fragment (string)\n   */\n  private splitUrl(url: string = ''): RouteParts {\n    const { queryParams, fragment } = this.router.parseUrl(url);\n    const [, path] = url.match(this.URL_SPLIT);\n\n    // wrap path in an array, to have the Angular-like path format\n    return { path: [path], queryParams, fragment };\n  }\n\n  /**\n   * Prepends a leading slash to the given URL string, in case it doesn't have it.\n   */\n  private getAbsoluteUrl(url: string): string {\n    return url.startsWith('/') ? url : '/' + url;\n  }\n}\n","<!-- https://github.com/angular/angular/issues/24567 -->\n\n<ng-container *ngIf=\"isExternalUrl(); else isLocalUrl\">\n  <a\n    [href]=\"url\"\n    [attr.target]=\"target\"\n    [attr.rel]=\"rel\"\n    [attr.id]=\"id\"\n    [attr.class]=\"class\"\n    [attr.style]=\"style\"\n    [attr.title]=\"title\"\n  >\n    <ng-container *ngTemplateOutlet=\"content\"></ng-container>\n  </a>\n</ng-container>\n\n<ng-template #isLocalUrl>\n  <a\n    [routerLink]=\"routerUrl\"\n    [queryParams]=\"queryParams\"\n    [fragment]=\"fragment\"\n    [target]=\"target\"\n    [attr.id]=\"id\"\n    [attr.class]=\"class\"\n    [attr.style]=\"style\"\n    [attr.title]=\"title\"\n  >\n    <ng-container *ngTemplateOutlet=\"content\"></ng-container>\n  </a>\n</ng-template>\n\n<ng-template #content>\n  <ng-content></ng-content>\n</ng-template>\n"]}