UNPKG

@ngaox/icons

Version:

Inline SVG icons from (src | files | your favorite icons library | ...)!

111 lines 13.5 kB
import { DOCUMENT } from '@angular/common'; import { HttpClient } from '@angular/common/http'; import { Inject, Injectable, Optional } from '@angular/core'; import { of, map, shareReplay } from 'rxjs'; import { NGAOX_FALLBACK } from './models'; import { NgaoxGlobalIcons } from './icons.module'; import * as i0 from "@angular/core"; import * as i1 from "@angular/common/http"; const defaultFallbackIcon = ` <svg viewBox="0 0 612 612" xmlns="http://www.w3.org/2000/svg" fill="currentColor"> <path d="M306 50C164.6 50 50 164.6 50 306C50 447.4 164.6 562 306 562C447.4 562 562 447.4 562 306C562 164.6 447.4 50 306 50ZM282 202C282 188.8 292.8 178 306 178C319.2 178 330 188.75 330 202V330C330 343.25 319.25 354 306 354C292.75 354 282 343.3 282 330V202ZM306 450C288.64 450 274.56 435.92 274.56 418.56C274.56 401.2 288.63 387.12 306 387.12C323.37 387.12 337.44 401.2 337.44 418.56C337.4 435.9 323.4 450 306 450Z"/> </svg> `; export class IconsService { constructor(http, document, fallbackIcon = defaultFallbackIcon, icons) { this.http = http; this.document = document; this.fallbackIcon = fallbackIcon; this.icons = new Map(); this.lazyIcons = new Map(); icons ??= []; this.addAll(icons); } textToSvgElement(svg) { const div = this.document.createElement('div'); div.innerHTML = svg; const svgEl = div.querySelector('svg'); if (svgEl) { svgEl.setAttribute('height', '100%'); svgEl.setAttribute('width', '100%'); } return svgEl; } /** * Get the fallback icon specified in the import of the `IconsModule.forRoot` * Used in the `ngaox-icon` component when the icon is not found * @see {@link https://ngaox-lab.web.app/docs/icons#fallback-icon} */ getFallbackIcon() { return this.fallbackIcon; } /** * get an already registered/added icon * * @see {@link IconsService.add} */ get(name) { if (this.icons.has(name)) { return of(this.icons.get(name)); } else if (this.lazyIcons.has(name)) { return this.lazyIcons.get(name) ?? of(undefined); } return of(undefined); } /** * Add an SVG icon to the NgaoxIcons registry * * @param name the name of the Icon in the registry (used in `ngaox-icon` component and {@link IconsService.get}) * @param value the SVG content or {@link ILazyIcon} for lazy loaded icons * @param override (default to true) whether or not replacing existing `svg` if `name` already exists * */ add(name, value) { if (typeof value === 'string') { this.icons.set(name, this.textToSvgElement(value)); } else { this.lazyIcons.set(name, this.http.get(value.url, { responseType: 'text' }).pipe(map(svg => this.textToSvgElement(svg)), shareReplay(1))); } return this.get(name); } addAll(icons) { icons.forEach(icon => { this.add(icon.name, icon.data); }); } /** * remove a registered icon */ remove(name) { if (this.icons.has(name)) { this.icons.delete(name); } if (this.lazyIcons.has(name)) { this.lazyIcons.delete(name); } } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.3", ngImport: i0, type: IconsService, deps: [{ token: i1.HttpClient }, { token: DOCUMENT }, { token: NGAOX_FALLBACK, optional: true }, { token: NgaoxGlobalIcons, optional: true }], target: i0.ɵɵFactoryTarget.Injectable }); } static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.0.3", ngImport: i0, type: IconsService, providedIn: 'root' }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.3", ngImport: i0, type: IconsService, decorators: [{ type: Injectable, args: [{ providedIn: 'root' }] }], ctorParameters: () => [{ type: i1.HttpClient }, { type: Document, decorators: [{ type: Inject, args: [DOCUMENT] }] }, { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [NGAOX_FALLBACK] }] }, { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [NgaoxGlobalIcons] }] }] }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"icons.service.js","sourceRoot":"","sources":["../../../../../packages/icons/src/lib/icons.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { DOCUMENT } from '@angular/common';\r\nimport { HttpClient } from '@angular/common/http';\r\nimport { Inject, Injectable, Optional } from '@angular/core';\r\nimport { Observable, of, map, shareReplay } from 'rxjs';\r\nimport { ILazyIcon, INgaoxIcon, NGAOX_FALLBACK } from './models';\r\nimport { NgaoxGlobalIcons } from './icons.module';\r\n\r\nconst defaultFallbackIcon = `\r\n  <svg viewBox=\"0 0 612 612\" xmlns=\"http://www.w3.org/2000/svg\" fill=\"currentColor\">\r\n    <path d=\"M306 50C164.6 50 50 164.6 50 306C50 447.4 164.6 562 306 562C447.4 562 562 447.4 562 306C562 164.6 447.4 50 306 50ZM282 202C282 188.8 292.8 178 306 178C319.2 178 330 188.75 330 202V330C330 343.25 319.25 354 306 354C292.75 354 282 343.3 282 330V202ZM306 450C288.64 450 274.56 435.92 274.56 418.56C274.56 401.2 288.63 387.12 306 387.12C323.37 387.12 337.44 401.2 337.44 418.56C337.4 435.9 323.4 450 306 450Z\"/>\r\n  </svg>\r\n`;\r\n\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class IconsService {\r\n  private icons = new Map<string, SVGElement>();\r\n  private lazyIcons = new Map<string, Observable<SVGElement>>();\r\n\r\n  constructor(\r\n    private http: HttpClient,\r\n    @Inject(DOCUMENT) private document: Document,\r\n    @Optional()\r\n    @Inject(NGAOX_FALLBACK)\r\n    private fallbackIcon: string = defaultFallbackIcon,\r\n    @Optional() @Inject(NgaoxGlobalIcons) icons?: INgaoxIcon[]\r\n  ) {\r\n    icons ??= [];\r\n    this.addAll(icons);\r\n  }\r\n\r\n  private textToSvgElement(svg: string): SVGElement {\r\n    const div = this.document.createElement('div');\r\n    div.innerHTML = svg;\r\n    const svgEl = div.querySelector('svg') as SVGElement;\r\n    if (svgEl) {\r\n      svgEl.setAttribute('height', '100%');\r\n      svgEl.setAttribute('width', '100%');\r\n    }\r\n    return svgEl;\r\n  }\r\n\r\n  /**\r\n   * Get the fallback icon specified in the import of the `IconsModule.forRoot`\r\n   * Used in the `ngaox-icon` component when the icon is not found\r\n   * @see {@link https://ngaox-lab.web.app/docs/icons#fallback-icon}\r\n   */\r\n  getFallbackIcon() {\r\n    return this.fallbackIcon;\r\n  }\r\n\r\n  /**\r\n   * get an already registered/added icon\r\n   *\r\n   * @see  {@link IconsService.add}\r\n   */\r\n  get(name: string): Observable<SVGElement | undefined> {\r\n    if (this.icons.has(name)) {\r\n      return of(this.icons.get(name));\r\n    } else if (this.lazyIcons.has(name)) {\r\n      return this.lazyIcons.get(name) ?? of(undefined);\r\n    }\r\n    return of(undefined);\r\n  }\r\n\r\n  /**\r\n   * Add an SVG icon to the NgaoxIcons registry\r\n   *\r\n   * @param name the name of the Icon in the registry (used in `ngaox-icon` component and {@link IconsService.get})\r\n   * @param value the SVG content or {@link ILazyIcon} for lazy loaded icons\r\n   * @param override (default to true) whether or not replacing existing `svg` if `name` already exists\r\n   *\r\n   */\r\n  add(\r\n    name: string,\r\n    value: string | ILazyIcon\r\n  ): Observable<SVGElement | undefined> {\r\n    if (typeof value === 'string') {\r\n      this.icons.set(name, this.textToSvgElement(value));\r\n    } else {\r\n      this.lazyIcons.set(\r\n        name,\r\n        this.http.get(value.url, { responseType: 'text' }).pipe(\r\n          map(svg => this.textToSvgElement(svg)),\r\n          shareReplay(1)\r\n        )\r\n      );\r\n    }\r\n    return this.get(name);\r\n  }\r\n\r\n  addAll(icons: INgaoxIcon[]) {\r\n    icons.forEach(icon => {\r\n      this.add(icon.name, icon.data);\r\n    });\r\n  }\r\n\r\n  /**\r\n   * remove a registered icon\r\n   */\r\n  remove(name: string) {\r\n    if (this.icons.has(name)) {\r\n      this.icons.delete(name);\r\n    }\r\n    if (this.lazyIcons.has(name)) {\r\n      this.lazyIcons.delete(name);\r\n    }\r\n  }\r\n}\r\n"]}