123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281 |
- import { DOCUMENT } from '@angular/common';
- import { HttpClient } from '@angular/common/http';
- import { Inject, Injectable, InjectionToken, Optional, SecurityContext } from '@angular/core';
- import { of, Observable, Subject } from 'rxjs';
- import { catchError, filter, finalize, map, share, take, tap } from 'rxjs/operators';
- import { cloneSVG, getIconDefinitionFromAbbr, getNameAndNamespace, getSecondaryColor, hasNamespace, isIconDefinition, replaceFillColor, warn, withSuffix, withSuffixAndColor } from '../utils';
- import { DynamicLoadingTimeoutError, HttpModuleNotImport, IconNotFoundError, NameSpaceIsNotSpecifyError, SVGTagNotFoundError, UrlNotSafeError } from './icon.error';
- import * as i0 from "@angular/core";
- import * as i1 from "@angular/common/http";
- import * as i2 from "@angular/platform-browser";
- const JSONP_HANDLER_NAME = '__ant_icon_load';
- export const ANT_ICONS = new InjectionToken('ant_icons');
- class IconService {
- set twoToneColor({ primaryColor, secondaryColor }) {
- this._twoToneColorPalette.primaryColor = primaryColor;
- this._twoToneColorPalette.secondaryColor =
- secondaryColor || getSecondaryColor(primaryColor);
- }
- get twoToneColor() {
- // Make a copy to avoid unexpected changes.
- return { ...this._twoToneColorPalette };
- }
- /**
- * Disable dynamic loading (support static loading only).
- */
- get _disableDynamicLoading() {
- return false;
- }
- constructor(_rendererFactory, _handler, _document, sanitizer, _antIcons) {
- this._rendererFactory = _rendererFactory;
- this._handler = _handler;
- this._document = _document;
- this.sanitizer = sanitizer;
- this._antIcons = _antIcons;
- this.defaultTheme = 'outline';
- /**
- * All icon definitions would be registered here.
- */
- this._svgDefinitions = new Map();
- /**
- * Cache all rendered icons. Icons are identified by name, theme,
- * and for twotone icons, primary color and secondary color.
- */
- this._svgRenderedDefinitions = new Map();
- this._inProgressFetches = new Map();
- /**
- * Url prefix for fetching inline SVG by dynamic importing.
- */
- this._assetsUrlRoot = '';
- this._twoToneColorPalette = {
- primaryColor: '#333333',
- secondaryColor: '#E6E6E6'
- };
- /** A flag indicates whether jsonp loading is enabled. */
- this._enableJsonpLoading = false;
- this._jsonpIconLoad$ = new Subject();
- this._renderer = this._rendererFactory.createRenderer(null, null);
- if (this._handler) {
- this._http = new HttpClient(this._handler);
- }
- if (this._antIcons) {
- this.addIcon(...this._antIcons);
- }
- }
- /**
- * Call this method to switch to jsonp like loading.
- */
- useJsonpLoading() {
- if (!this._enableJsonpLoading) {
- this._enableJsonpLoading = true;
- window[JSONP_HANDLER_NAME] = (icon) => {
- this._jsonpIconLoad$.next(icon);
- };
- }
- else {
- warn('You are already using jsonp loading.');
- }
- }
- /**
- * Change the prefix of the inline svg resources, so they could be deployed elsewhere, like CDN.
- * @param prefix
- */
- changeAssetsSource(prefix) {
- this._assetsUrlRoot = prefix.endsWith('/') ? prefix : prefix + '/';
- }
- /**
- * Add icons provided by ant design.
- * @param icons
- */
- addIcon(...icons) {
- icons.forEach(icon => {
- this._svgDefinitions.set(withSuffix(icon.name, icon.theme), icon);
- });
- }
- /**
- * Register an icon. Namespace is required.
- * @param type
- * @param literal
- */
- addIconLiteral(type, literal) {
- const [_, namespace] = getNameAndNamespace(type);
- if (!namespace) {
- throw NameSpaceIsNotSpecifyError();
- }
- this.addIcon({ name: type, icon: literal });
- }
- /**
- * Remove all cache.
- */
- clear() {
- this._svgDefinitions.clear();
- this._svgRenderedDefinitions.clear();
- }
- /**
- * Get a rendered `SVGElement`.
- * @param icon
- * @param twoToneColor
- */
- getRenderedContent(icon, twoToneColor) {
- // If `icon` is a `IconDefinition`, go to the next step. If not, try to fetch it from cache.
- const definition = isIconDefinition(icon)
- ? icon
- : this._svgDefinitions.get(icon) || null;
- if (!definition && this._disableDynamicLoading) {
- throw IconNotFoundError(icon);
- }
- // If `icon` is a `IconDefinition` of successfully fetch, wrap it in an `Observable`.
- // Otherwise try to fetch it from remote.
- const $iconDefinition = definition
- ? of(definition)
- : this._loadIconDynamically(icon);
- // If finally get an `IconDefinition`, render and return it. Otherwise throw an error.
- return $iconDefinition.pipe(map(i => {
- if (!i) {
- throw IconNotFoundError(icon);
- }
- return this._loadSVGFromCacheOrCreateNew(i, twoToneColor);
- }));
- }
- getCachedIcons() {
- return this._svgDefinitions;
- }
- /**
- * Get raw svg and assemble a `IconDefinition` object.
- * @param type
- */
- _loadIconDynamically(type) {
- // If developer doesn't provide HTTP module nor enable jsonp loading, just throw an error.
- if (!this._http && !this._enableJsonpLoading) {
- return of(HttpModuleNotImport());
- }
- // If multi directive ask for the same icon at the same time,
- // request should only be fired once.
- let inProgress = this._inProgressFetches.get(type);
- if (!inProgress) {
- const [name, namespace] = getNameAndNamespace(type);
- // If the string has a namespace within, create a simple `IconDefinition`.
- const icon = namespace
- ? { name: type, icon: '' }
- : getIconDefinitionFromAbbr(name);
- const suffix = this._enableJsonpLoading ? '.js' : '.svg';
- const url = (namespace
- ? `${this._assetsUrlRoot}assets/${namespace}/${name}`
- : `${this._assetsUrlRoot}assets/${icon.theme}/${icon.name}`) + suffix;
- const safeUrl = this.sanitizer.sanitize(SecurityContext.URL, url);
- if (!safeUrl) {
- throw UrlNotSafeError(url);
- }
- const source = !this._enableJsonpLoading
- ? this._http
- .get(safeUrl, { responseType: 'text' })
- .pipe(map(literal => ({ ...icon, icon: literal })))
- : this._loadIconDynamicallyWithJsonp(icon, safeUrl);
- inProgress = source.pipe(tap(definition => this.addIcon(definition)), finalize(() => this._inProgressFetches.delete(type)), catchError(() => of(null)), share());
- this._inProgressFetches.set(type, inProgress);
- }
- return inProgress;
- }
- _loadIconDynamicallyWithJsonp(icon, url) {
- return new Observable(subscriber => {
- const loader = this._document.createElement('script');
- const timer = setTimeout(() => {
- clean();
- subscriber.error(DynamicLoadingTimeoutError());
- }, 6000);
- loader.src = url;
- function clean() {
- loader.parentNode.removeChild(loader);
- clearTimeout(timer);
- }
- this._document.body.appendChild(loader);
- this._jsonpIconLoad$
- .pipe(filter(i => i.name === icon.name && i.theme === icon.theme), take(1))
- .subscribe(i => {
- subscriber.next(i);
- clean();
- });
- });
- }
- /**
- * Render a new `SVGElement` for a given `IconDefinition`, or make a copy from cache.
- * @param icon
- * @param twoToneColor
- */
- _loadSVGFromCacheOrCreateNew(icon, twoToneColor) {
- let svg;
- const pri = twoToneColor || this._twoToneColorPalette.primaryColor;
- const sec = getSecondaryColor(pri) || this._twoToneColorPalette.secondaryColor;
- const key = icon.theme === 'twotone'
- ? withSuffixAndColor(icon.name, icon.theme, pri, sec)
- : icon.theme === undefined
- ? icon.name
- : withSuffix(icon.name, icon.theme);
- // Try to make a copy from cache.
- const cached = this._svgRenderedDefinitions.get(key);
- if (cached) {
- svg = cached.icon;
- }
- else {
- svg = this._setSVGAttribute(this._colorizeSVGIcon(
- // Icons provided by ant design should be refined to remove preset colors.
- this._createSVGElementFromString(hasNamespace(icon.name) ? icon.icon : replaceFillColor(icon.icon)), icon.theme === 'twotone', pri, sec));
- // Cache it.
- this._svgRenderedDefinitions.set(key, {
- ...icon,
- icon: svg
- });
- }
- return cloneSVG(svg);
- }
- _createSVGElementFromString(str) {
- const div = this._document.createElement('div');
- div.innerHTML = str;
- const svg = div.querySelector('svg');
- if (!svg) {
- throw SVGTagNotFoundError;
- }
- return svg;
- }
- _setSVGAttribute(svg) {
- this._renderer.setAttribute(svg, 'width', '1em');
- this._renderer.setAttribute(svg, 'height', '1em');
- return svg;
- }
- _colorizeSVGIcon(svg, twotone, pri, sec) {
- if (twotone) {
- const children = svg.childNodes;
- const length = children.length;
- for (let i = 0; i < length; i++) {
- const child = children[i];
- if (child.getAttribute('fill') === 'secondaryColor') {
- this._renderer.setAttribute(child, 'fill', sec);
- }
- else {
- this._renderer.setAttribute(child, 'fill', pri);
- }
- }
- }
- this._renderer.setAttribute(svg, 'fill', 'currentColor');
- return svg;
- }
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.2", ngImport: i0, type: IconService, deps: [{ token: i0.RendererFactory2 }, { token: i1.HttpBackend, optional: true }, { token: DOCUMENT, optional: true }, { token: i2.DomSanitizer }, { token: ANT_ICONS, optional: true }], target: i0.ɵɵFactoryTarget.Injectable }); }
- static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.0.2", ngImport: i0, type: IconService }); }
- }
- export { IconService };
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.2", ngImport: i0, type: IconService, decorators: [{
- type: Injectable
- }], ctorParameters: function () { return [{ type: i0.RendererFactory2 }, { type: i1.HttpBackend, decorators: [{
- type: Optional
- }] }, { type: undefined, decorators: [{
- type: Optional
- }, {
- type: Inject,
- args: [DOCUMENT]
- }] }, { type: i2.DomSanitizer }, { type: undefined, decorators: [{
- type: Optional
- }, {
- type: Inject,
- args: [ANT_ICONS]
- }] }]; } });
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"icon.service.js","sourceRoot":"","sources":["../../../../src/component/icon.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAC3C,OAAO,EAAe,UAAU,EAAE,MAAM,sBAAsB,CAAC;AAC/D,OAAO,EAAE,MAAM,EAAE,UAAU,EAAE,cAAc,EAAE,QAAQ,EAA+B,eAAe,EAAE,MAAM,eAAe,CAAC;AAE3H,OAAO,EAAE,EAAE,EAAE,UAAU,EAAE,OAAO,EAAE,MAAM,MAAM,CAAC;AAC/C,OAAO,EACL,UAAU,EACV,MAAM,EACN,QAAQ,EACR,GAAG,EACH,KAAK,EACL,IAAI,EACJ,GAAG,EACJ,MAAM,gBAAgB,CAAC;AAQxB,OAAO,EACL,QAAQ,EACR,yBAAyB,EACzB,mBAAmB,EACnB,iBAAiB,EACjB,YAAY,EACZ,gBAAgB,EAChB,gBAAgB,EAChB,IAAI,EACJ,UAAU,EACV,kBAAkB,EACnB,MAAM,UAAU,CAAC;AAClB,OAAO,EACL,0BAA0B,EAC1B,mBAAmB,EACnB,iBAAiB,EACjB,0BAA0B,EAC1B,mBAAmB,EACnB,eAAe,EAChB,MAAM,cAAc,CAAC;;;;AAEtB,MAAM,kBAAkB,GAAG,iBAAiB,CAAC;AAE7C,MAAM,CAAC,MAAM,SAAS,GAAG,IAAI,cAAc,CAAmB,WAAW,CAAC,CAAC;AAE3E,MACa,WAAW;IAGtB,IAAI,YAAY,CAAC,EACf,YAAY,EACZ,cAAc,EACY;QAC1B,IAAI,CAAC,oBAAoB,CAAC,YAAY,GAAG,YAAY,CAAC;QACtD,IAAI,CAAC,oBAAoB,CAAC,cAAc;YACtC,cAAc,IAAI,iBAAiB,CAAC,YAAY,CAAC,CAAC;IACtD,CAAC;IAED,IAAI,YAAY;QACd,2CAA2C;QAC3C,OAAO,EAAE,GAAG,IAAI,CAAC,oBAAoB,EAAyB,CAAC;IACjE,CAAC;IAKD;;OAEG;IACH,IAAc,sBAAsB;QAClC,OAAO,KAAK,CAAC;IACf,CAAC;IAgCD,YACY,gBAAkC,EACtB,QAAqB,EACH,SAAc,EAC5C,SAAuB,EAEQ,SAA2B;QAL1D,qBAAgB,GAAhB,gBAAgB,CAAkB;QACtB,aAAQ,GAAR,QAAQ,CAAa;QACH,cAAS,GAAT,SAAS,CAAK;QAC5C,cAAS,GAAT,SAAS,CAAc;QAEQ,cAAS,GAAT,SAAS,CAAkB;QA9DtE,iBAAY,GAAc,SAAS,CAAC;QA0BpC;;WAEG;QACgB,oBAAe,GAAG,IAAI,GAAG,EAA0B,CAAC;QAEvE;;;WAGG;QACgB,4BAAuB,GAAG,IAAI,GAAG,EAAgC,CAAC;QAE3E,uBAAkB,GAAG,IAAI,GAAG,EAGnC,CAAC;QAEJ;;WAEG;QACO,mBAAc,GAAG,EAAE,CAAC;QAEpB,yBAAoB,GAAwB;YACpD,YAAY,EAAE,SAAS;YACvB,cAAc,EAAE,SAAS;SAC1B,CAAC;QAEF,yDAAyD;QACjD,wBAAmB,GAAG,KAAK,CAAC;QACnB,oBAAe,GAAG,IAAI,OAAO,EAAkB,CAAC;QAU/D,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,gBAAgB,CAAC,cAAc,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;QAElE,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjB,IAAI,CAAC,KAAK,GAAG,IAAI,UAAU,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;SAC5C;QAED,IAAI,IAAI,CAAC,SAAS,EAAE;YAClB,IAAI,CAAC,OAAO,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC;SACjC;IACH,CAAC;IAED;;OAEG;IACH,eAAe;QACb,IAAI,CAAC,IAAI,CAAC,mBAAmB,EAAE;YAC7B,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC;YAEhC,MAAM,CAAC,kBAAkB,CAAC,GAAG,CAAC,IAAoB,EAAE,EAAE;gBACpD,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAClC,CAAC,CAAC;SACH;aAAM;YACL,IAAI,CAAC,sCAAsC,CAAC,CAAC;SAC9C;IACH,CAAC;IAED;;;OAGG;IACH,kBAAkB,CAAC,MAAc;QAC/B,IAAI,CAAC,cAAc,GAAG,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,GAAG,GAAG,CAAC;IACrE,CAAC;IAED;;;OAGG;IACH,OAAO,CAAC,GAAG,KAAuB;QAChC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;YACnB,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,CAAC,EAAE,IAAI,CAAC,CAAC;QACpE,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;OAIG;IACH,cAAc,CAAC,IAAY,EAAE,OAAe;QAC1C,MAAM,CAAC,CAAC,EAAE,SAAS,CAAC,GAAG,mBAAmB,CAAC,IAAI,CAAC,CAAC;QACjD,IAAI,CAAC,SAAS,EAAE;YACd,MAAM,0BAA0B,EAAE,CAAC;SACpC;QACD,IAAI,CAAC,OAAO,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,OAAO,EAAE,CAAC,CAAC;IAC9C,CAAC;IAED;;OAEG;IACH,KAAK;QACH,IAAI,CAAC,eAAe,CAAC,KAAK,EAAE,CAAC;QAC7B,IAAI,CAAC,uBAAuB,CAAC,KAAK,EAAE,CAAC;IACvC,CAAC;IAED;;;;OAIG;IACH,kBAAkB,CAChB,IAA6B,EAC7B,YAAqB;QAErB,4FAA4F;QAC5F,MAAM,UAAU,GAA0B,gBAAgB,CAAC,IAAI,CAAC;YAC9D,CAAC,CAAE,IAAuB;YAC1B,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC;QAE3C,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,sBAAsB,EAAE;YAC9C,MAAM,iBAAiB,CAAC,IAAc,CAAC,CAAC;SACzC;QAED,qFAAqF;QACrF,yCAAyC;QACzC,MAAM,eAAe,GAAG,UAAU;YAChC,CAAC,CAAC,EAAE,CAAC,UAAU,CAAC;YAChB,CAAC,CAAC,IAAI,CAAC,oBAAoB,CAAC,IAAc,CAAC,CAAC;QAE9C,sFAAsF;QACtF,OAAO,eAAe,CAAC,IAAI,CACzB,GAAG,CAAC,CAAC,CAAC,EAAE;YACN,IAAI,CAAC,CAAC,EAAE;gBACN,MAAM,iBAAiB,CAAC,IAAc,CAAC,CAAC;aACzC;YACD,OAAO,IAAI,CAAC,4BAA4B,CAAC,CAAC,EAAE,YAAY,CAAC,CAAC;QAC5D,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAED,cAAc;QACZ,OAAO,IAAI,CAAC,eAAe,CAAC;IAC9B,CAAC;IAED;;;OAGG;IACO,oBAAoB,CAC5B,IAAY;QAEZ,0FAA0F;QAC1F,IAAI,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,mBAAmB,EAAE;YAC5C,OAAO,EAAE,CAAC,mBAAmB,EAAE,CAAC,CAAC;SAClC;QAED,6DAA6D;QAC7D,qCAAqC;QACrC,IAAI,UAAU,GAAG,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAEnD,IAAI,CAAC,UAAU,EAAE;YACf,MAAM,CAAC,IAAI,EAAE,SAAS,CAAC,GAAG,mBAAmB,CAAC,IAAI,CAAC,CAAC;YAEpD,0EAA0E;YAC1E,MAAM,IAAI,GAAmB,SAAS;gBACpC,CAAC,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,EAAE,EAAE;gBAC1B,CAAC,CAAC,yBAAyB,CAAC,IAAI,CAAC,CAAC;YAEpC,MAAM,MAAM,GAAG,IAAI,CAAC,mBAAmB,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC;YACzD,MAAM,GAAG,GACP,CAAC,SAAS;gBACR,CAAC,CAAC,GAAG,IAAI,CAAC,cAAc,UAAU,SAAS,IAAI,IAAI,EAAE;gBACrD,CAAC,CAAC,GAAG,IAAI,CAAC,cAAc,UAAU,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,IAAI,EAAE,CAAC,GAAG,MAAM,CAAC;YAE1E,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,eAAe,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;YAElE,IAAI,CAAC,OAAO,EAAE;gBACZ,MAAM,eAAe,CAAC,GAAG,CAAC,CAAC;aAC5B;YAED,MAAM,MAAM,GAAG,CAAC,IAAI,CAAC,mBAAmB;gBACtC,CAAC,CAAC,IAAI,CAAC,KAAK;qBACP,GAAG,CAAC,OAAO,EAAE,EAAE,YAAY,EAAE,MAAM,EAAE,CAAC;qBACtC,IAAI,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,EAAE,GAAG,IAAI,EAAE,IAAI,EAAE,OAAO,EAAE,CAAC,CAAC,CAAC;gBACvD,CAAC,CAAC,IAAI,CAAC,6BAA6B,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;YAEtD,UAAU,GAAG,MAAM,CAAC,IAAI,CACtB,GAAG,CAAC,UAAU,CAAC,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC,EAC3C,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,EACpD,UAAU,CAAC,GAAG,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,EAC1B,KAAK,EAAE,CACR,CAAC;YAEF,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,IAAI,EAAE,UAAU,CAAC,CAAC;SAC/C;QAED,OAAO,UAAU,CAAC;IACpB,CAAC;IAES,6BAA6B,CAAC,IAAoB,EAAE,GAAW;QACvE,OAAO,IAAI,UAAU,CAAiB,UAAU,CAAC,EAAE;YACjD,MAAM,MAAM,GAAG,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;YACtD,MAAM,KAAK,GAAG,UAAU,CAAC,GAAG,EAAE;gBAC5B,KAAK,EAAE,CAAC;gBACR,UAAU,CAAC,KAAK,CAAC,0BAA0B,EAAE,CAAC,CAAC;YACjD,CAAC,EAAE,IAAI,CAAC,CAAC;YAET,MAAM,CAAC,GAAG,GAAG,GAAG,CAAC;YAEjB,SAAS,KAAK;gBACZ,MAAM,CAAC,UAAU,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;gBACtC,YAAY,CAAC,KAAK,CAAC,CAAC;YACtB,CAAC;YAED,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;YACxC,IAAI,CAAC,eAAe;iBACf,IAAI,CACD,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,IAAI,CAAC,IAAI,IAAI,CAAC,CAAC,KAAK,KAAK,IAAI,CAAC,KAAK,CAAC,EAC3D,IAAI,CAAC,CAAC,CAAC,CACV;iBACA,SAAS,CAAC,CAAC,CAAC,EAAE;gBACb,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;gBACnB,KAAK,EAAE,CAAC;YACV,CAAC,CAAC,CAAC;QACT,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;OAIG;IACO,4BAA4B,CACpC,IAAoB,EACpB,YAAqB;QAErB,IAAI,GAAe,CAAC;QAEpB,MAAM,GAAG,GAAG,YAAY,IAAI,IAAI,CAAC,oBAAoB,CAAC,YAAY,CAAC;QACnE,MAAM,GAAG,GACP,iBAAiB,CAAC,GAAG,CAAC,IAAI,IAAI,CAAC,oBAAoB,CAAC,cAAc,CAAC;QACrE,MAAM,GAAG,GACP,IAAI,CAAC,KAAK,KAAK,SAAS;YACtB,CAAC,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,EAAE,GAAG,EAAE,GAAG,CAAC;YACrD,CAAC,CAAC,IAAI,CAAC,KAAK,KAAK,SAAS;gBAC1B,CAAC,CAAC,IAAI,CAAC,IAAI;gBACX,CAAC,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;QAExC,iCAAiC;QACjC,MAAM,MAAM,GAAG,IAAI,CAAC,uBAAuB,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QAErD,IAAI,MAAM,EAAE;YACV,GAAG,GAAG,MAAM,CAAC,IAAI,CAAC;SACnB;aAAM;YACL,GAAG,GAAG,IAAI,CAAC,gBAAgB,CACzB,IAAI,CAAC,gBAAgB;YACnB,0EAA0E;YAC1E,IAAI,CAAC,2BAA2B,CAC9B,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,CAClE,EACD,IAAI,CAAC,KAAK,KAAK,SAAS,EACxB,GAAG,EACH,GAAG,CACJ,CACF,CAAC;YACF,YAAY;YACZ,IAAI,CAAC,uBAAuB,CAAC,GAAG,CAAC,GAAG,EAAE;gBACpC,GAAG,IAAI;gBACP,IAAI,EAAE,GAAG;aACc,CAAC,CAAC;SAC5B;QAED,OAAO,QAAQ,CAAC,GAAG,CAAC,CAAC;IACvB,CAAC;IAES,2BAA2B,CAAC,GAAW;QAC/C,MAAM,GAAG,GAAG,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;QAChD,GAAG,CAAC,SAAS,GAAG,GAAG,CAAC;QACpB,MAAM,GAAG,GAAe,GAAG,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;QACjD,IAAI,CAAC,GAAG,EAAE;YACR,MAAM,mBAAmB,CAAC;SAC3B;QACD,OAAO,GAAG,CAAC;IACb,CAAC;IAES,gBAAgB,CAAC,GAAe;QACxC,IAAI,CAAC,SAAS,CAAC,YAAY,CAAC,GAAG,EAAE,OAAO,EAAE,KAAK,CAAC,CAAC;QACjD,IAAI,CAAC,SAAS,CAAC,YAAY,CAAC,GAAG,EAAE,QAAQ,EAAE,KAAK,CAAC,CAAC;QAClD,OAAO,GAAG,CAAC;IACb,CAAC;IAES,gBAAgB,CACxB,GAAe,EACf,OAAgB,EAChB,GAAW,EACX,GAAW;QAEX,IAAI,OAAO,EAAE;YACX,MAAM,QAAQ,GAAG,GAAG,CAAC,UAAU,CAAC;YAChC,MAAM,MAAM,GAAG,QAAQ,CAAC,MAAM,CAAC;YAC/B,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,MAAM,EAAE,CAAC,EAAE,EAAE;gBAC/B,MAAM,KAAK,GAAgB,QAAQ,CAAC,CAAC,CAAgB,CAAC;gBACtD,IAAI,KAAK,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,gBAAgB,EAAE;oBACnD,IAAI,CAAC,SAAS,CAAC,YAAY,CAAC,KAAK,EAAE,MAAM,EAAE,GAAG,CAAC,CAAC;iBACjD;qBAAM;oBACL,IAAI,CAAC,SAAS,CAAC,YAAY,CAAC,KAAK,EAAE,MAAM,EAAE,GAAG,CAAC,CAAC;iBACjD;aACF;SACF;QACD,IAAI,CAAC,SAAS,CAAC,YAAY,CAAC,GAAG,EAAE,MAAM,EAAE,cAAc,CAAC,CAAC;QACzD,OAAO,GAAG,CAAC;IACb,CAAC;8GAhVU,WAAW,6FA4DA,QAAQ,yDAGR,SAAS;kHA/DpB,WAAW;;SAAX,WAAW;2FAAX,WAAW;kBADvB,UAAU;;0BA4DN,QAAQ;;0BACR,QAAQ;;0BAAI,MAAM;2BAAC,QAAQ;;0BAG3B,QAAQ;;0BAAI,MAAM;2BAAC,SAAS","sourcesContent":["import { DOCUMENT } from '@angular/common';\nimport { HttpBackend, HttpClient } from '@angular/common/http';\nimport { Inject, Injectable, InjectionToken, Optional, Renderer2, RendererFactory2, SecurityContext } from '@angular/core';\nimport { DomSanitizer } from '@angular/platform-browser';\nimport { of, Observable, Subject } from 'rxjs';\nimport {\n  catchError,\n  filter,\n  finalize,\n  map,\n  share,\n  take,\n  tap\n} from 'rxjs/operators';\nimport {\n  CachedIconDefinition,\n  IconDefinition,\n  ThemeType,\n  TwoToneColorPalette,\n  TwoToneColorPaletteSetter\n} from '../types';\nimport {\n  cloneSVG,\n  getIconDefinitionFromAbbr,\n  getNameAndNamespace,\n  getSecondaryColor,\n  hasNamespace,\n  isIconDefinition,\n  replaceFillColor,\n  warn,\n  withSuffix,\n  withSuffixAndColor\n} from '../utils';\nimport {\n  DynamicLoadingTimeoutError,\n  HttpModuleNotImport,\n  IconNotFoundError,\n  NameSpaceIsNotSpecifyError,\n  SVGTagNotFoundError,\n  UrlNotSafeError\n} from './icon.error';\n\nconst JSONP_HANDLER_NAME = '__ant_icon_load';\n\nexport const ANT_ICONS = new InjectionToken<IconDefinition[]>('ant_icons');\n\n@Injectable()\nexport class IconService {\n  defaultTheme: ThemeType = 'outline';\n\n  set twoToneColor({\n    primaryColor,\n    secondaryColor\n  }: TwoToneColorPaletteSetter) {\n    this._twoToneColorPalette.primaryColor = primaryColor;\n    this._twoToneColorPalette.secondaryColor =\n      secondaryColor || getSecondaryColor(primaryColor);\n  }\n\n  get twoToneColor(): TwoToneColorPaletteSetter {\n    // Make a copy to avoid unexpected changes.\n    return { ...this._twoToneColorPalette } as TwoToneColorPalette;\n  }\n\n  protected _renderer: Renderer2;\n  protected _http: HttpClient;\n\n  /**\n   * Disable dynamic loading (support static loading only).\n   */\n  protected get _disableDynamicLoading(): boolean {\n    return false;\n  }\n\n  /**\n   * All icon definitions would be registered here.\n   */\n  protected readonly _svgDefinitions = new Map<string, IconDefinition>();\n\n  /**\n   * Cache all rendered icons. Icons are identified by name, theme,\n   * and for twotone icons, primary color and secondary color.\n   */\n  protected readonly _svgRenderedDefinitions = new Map<string, CachedIconDefinition>();\n\n  protected _inProgressFetches = new Map<\n    string,\n    Observable<IconDefinition | null>\n  >();\n\n  /**\n   * Url prefix for fetching inline SVG by dynamic importing.\n   */\n  protected _assetsUrlRoot = '';\n\n  protected _twoToneColorPalette: TwoToneColorPalette = {\n    primaryColor: '#333333',\n    secondaryColor: '#E6E6E6'\n  };\n\n  /** A flag indicates whether jsonp loading is enabled. */\n  private _enableJsonpLoading = false;\n  private readonly _jsonpIconLoad$ = new Subject<IconDefinition>();\n\n  constructor(\n    protected _rendererFactory: RendererFactory2,\n    @Optional() protected _handler: HttpBackend,\n    @Optional() @Inject(DOCUMENT) protected _document: any,\n    protected sanitizer: DomSanitizer,\n\n    @Optional() @Inject(ANT_ICONS) protected _antIcons: IconDefinition[]\n  ) {\n    this._renderer = this._rendererFactory.createRenderer(null, null);\n\n    if (this._handler) {\n      this._http = new HttpClient(this._handler);\n    }\n\n    if (this._antIcons) {\n      this.addIcon(...this._antIcons);\n    }\n  }\n\n  /**\n   * Call this method to switch to jsonp like loading.\n   */\n  useJsonpLoading(): void {\n    if (!this._enableJsonpLoading) {\n      this._enableJsonpLoading = true;\n\n      window[JSONP_HANDLER_NAME] = (icon: IconDefinition) => {\n        this._jsonpIconLoad$.next(icon);\n      };\n    } else {\n      warn('You are already using jsonp loading.');\n    }\n  }\n\n  /**\n   * Change the prefix of the inline svg resources, so they could be deployed elsewhere, like CDN.\n   * @param prefix\n   */\n  changeAssetsSource(prefix: string): void {\n    this._assetsUrlRoot = prefix.endsWith('/') ? prefix : prefix + '/';\n  }\n\n  /**\n   * Add icons provided by ant design.\n   * @param icons\n   */\n  addIcon(...icons: IconDefinition[]): void {\n    icons.forEach(icon => {\n      this._svgDefinitions.set(withSuffix(icon.name, icon.theme), icon);\n    });\n  }\n\n  /**\n   * Register an icon. Namespace is required.\n   * @param type\n   * @param literal\n   */\n  addIconLiteral(type: string, literal: string): void {\n    const [_, namespace] = getNameAndNamespace(type);\n    if (!namespace) {\n      throw NameSpaceIsNotSpecifyError();\n    }\n    this.addIcon({ name: type, icon: literal });\n  }\n\n  /**\n   * Remove all cache.\n   */\n  clear(): void {\n    this._svgDefinitions.clear();\n    this._svgRenderedDefinitions.clear();\n  }\n\n  /**\n   * Get a rendered `SVGElement`.\n   * @param icon\n   * @param twoToneColor\n   */\n  getRenderedContent(\n    icon: IconDefinition | string,\n    twoToneColor?: string\n  ): Observable<SVGElement> {\n    // If `icon` is a `IconDefinition`, go to the next step. If not, try to fetch it from cache.\n    const definition: IconDefinition | null = isIconDefinition(icon)\n      ? (icon as IconDefinition)\n      : this._svgDefinitions.get(icon) || null;\n    \n    if (!definition && this._disableDynamicLoading) {\n      throw IconNotFoundError(icon as string);\n    }\n\n    // If `icon` is a `IconDefinition` of successfully fetch, wrap it in an `Observable`.\n    // Otherwise try to fetch it from remote.\n    const $iconDefinition = definition\n      ? of(definition)\n      : this._loadIconDynamically(icon as string);\n\n    // If finally get an `IconDefinition`, render and return it. Otherwise throw an error.\n    return $iconDefinition.pipe(\n      map(i => {\n        if (!i) {\n          throw IconNotFoundError(icon as string);\n        }\n        return this._loadSVGFromCacheOrCreateNew(i, twoToneColor);\n      })\n    );\n  }\n\n  getCachedIcons(): Map<string, IconDefinition> {\n    return this._svgDefinitions;\n  }\n\n  /**\n   * Get raw svg and assemble a `IconDefinition` object.\n   * @param type\n   */\n  protected _loadIconDynamically(\n    type: string\n  ): Observable<IconDefinition | null> {\n    // If developer doesn't provide HTTP module nor enable jsonp loading, just throw an error.\n    if (!this._http && !this._enableJsonpLoading) {\n      return of(HttpModuleNotImport());\n    }\n\n    // If multi directive ask for the same icon at the same time,\n    // request should only be fired once.\n    let inProgress = this._inProgressFetches.get(type);\n\n    if (!inProgress) {\n      const [name, namespace] = getNameAndNamespace(type);\n\n      // If the string has a namespace within, create a simple `IconDefinition`.\n      const icon: IconDefinition = namespace\n        ? { name: type, icon: '' }\n        : getIconDefinitionFromAbbr(name);\n\n      const suffix = this._enableJsonpLoading ? '.js' : '.svg';\n      const url =\n        (namespace\n          ? `${this._assetsUrlRoot}assets/${namespace}/${name}`\n          : `${this._assetsUrlRoot}assets/${icon.theme}/${icon.name}`) + suffix;\n\n      const safeUrl = this.sanitizer.sanitize(SecurityContext.URL, url);\n\n      if (!safeUrl) {\n        throw UrlNotSafeError(url);\n      }\n\n      const source = !this._enableJsonpLoading\n        ? this._http\n            .get(safeUrl, { responseType: 'text' })\n            .pipe(map(literal => ({ ...icon, icon: literal })))\n        : this._loadIconDynamicallyWithJsonp(icon, safeUrl);\n\n      inProgress = source.pipe(\n        tap(definition => this.addIcon(definition)),\n        finalize(() => this._inProgressFetches.delete(type)),\n        catchError(() => of(null)),\n        share()\n      );\n\n      this._inProgressFetches.set(type, inProgress);\n    }\n\n    return inProgress;\n  }\n\n  protected _loadIconDynamicallyWithJsonp(icon: IconDefinition, url: string): Observable<IconDefinition> {\n    return new Observable<IconDefinition>(subscriber => {\n      const loader = this._document.createElement('script');\n      const timer = setTimeout(() => {\n        clean();\n        subscriber.error(DynamicLoadingTimeoutError());\n      }, 6000);\n\n      loader.src = url;\n\n      function clean(): void {\n        loader.parentNode.removeChild(loader);\n        clearTimeout(timer);\n      }\n\n      this._document.body.appendChild(loader);\n      this._jsonpIconLoad$\n          .pipe(\n              filter(i => i.name === icon.name && i.theme === icon.theme),\n              take(1)\n          )\n          .subscribe(i => {\n            subscriber.next(i);\n            clean();\n          });\n    });\n  }\n\n  /**\n   * Render a new `SVGElement` for a given `IconDefinition`, or make a copy from cache.\n   * @param icon\n   * @param twoToneColor\n   */\n  protected _loadSVGFromCacheOrCreateNew(\n    icon: IconDefinition,\n    twoToneColor?: string\n  ): SVGElement {\n    let svg: SVGElement;\n\n    const pri = twoToneColor || this._twoToneColorPalette.primaryColor;\n    const sec =\n      getSecondaryColor(pri) || this._twoToneColorPalette.secondaryColor;\n    const key =\n      icon.theme === 'twotone'\n        ? withSuffixAndColor(icon.name, icon.theme, pri, sec)\n        : icon.theme === undefined\n        ? icon.name\n        : withSuffix(icon.name, icon.theme);\n\n    // Try to make a copy from cache.\n    const cached = this._svgRenderedDefinitions.get(key);\n\n    if (cached) {\n      svg = cached.icon;\n    } else {\n      svg = this._setSVGAttribute(\n        this._colorizeSVGIcon(\n          // Icons provided by ant design should be refined to remove preset colors.\n          this._createSVGElementFromString(\n            hasNamespace(icon.name) ? icon.icon : replaceFillColor(icon.icon)\n          ),\n          icon.theme === 'twotone',\n          pri,\n          sec\n        )\n      );\n      // Cache it.\n      this._svgRenderedDefinitions.set(key, {\n        ...icon,\n        icon: svg\n      } as CachedIconDefinition);\n    }\n\n    return cloneSVG(svg);\n  }\n\n  protected _createSVGElementFromString(str: string): SVGElement {\n    const div = this._document.createElement('div');\n    div.innerHTML = str;\n    const svg: SVGElement = div.querySelector('svg');\n    if (!svg) {\n      throw SVGTagNotFoundError;\n    }\n    return svg;\n  }\n\n  protected _setSVGAttribute(svg: SVGElement): SVGElement {\n    this._renderer.setAttribute(svg, 'width', '1em');\n    this._renderer.setAttribute(svg, 'height', '1em');\n    return svg;\n  }\n\n  protected _colorizeSVGIcon(\n    svg: SVGElement,\n    twotone: boolean,\n    pri: string,\n    sec: string\n  ): SVGElement {\n    if (twotone) {\n      const children = svg.childNodes;\n      const length = children.length;\n      for (let i = 0; i < length; i++) {\n        const child: HTMLElement = children[i] as HTMLElement;\n        if (child.getAttribute('fill') === 'secondaryColor') {\n          this._renderer.setAttribute(child, 'fill', sec);\n        } else {\n          this._renderer.setAttribute(child, 'fill', pri);\n        }\n      }\n    }\n    this._renderer.setAttribute(svg, 'fill', 'currentColor');\n    return svg;\n  }\n}\n"]}
|