lucide-angular
Version:
A Lucide icon library package for Angular applications.
176 lines • 23.2 kB
JavaScript
import { ChangeDetectorRef, Component, ElementRef, Inject, Input, Renderer2, } from '@angular/core';
import defaultAttributes from '../icons/constants/default-attributes';
import { LUCIDE_ICONS } from './lucide-icon.provider';
import { LucideIconConfig } from './lucide-icon.config';
import * as i0 from "@angular/core";
import * as i1 from "./lucide-icon.config";
export function formatFixed(number, decimals = 3) {
return parseFloat(number.toFixed(decimals)).toString(10);
}
export class LucideAngularComponent {
constructor(elem, renderer, changeDetector, iconProviders, iconConfig) {
this.elem = elem;
this.renderer = renderer;
this.changeDetector = changeDetector;
this.iconProviders = iconProviders;
this.iconConfig = iconConfig;
this.absoluteStrokeWidth = false;
this.defaultSize = defaultAttributes.height;
}
get size() {
return this._size ?? this.iconConfig.size;
}
set size(value) {
if (value) {
this._size = this.parseNumber(value);
}
else {
delete this._size;
}
}
get strokeWidth() {
return this._strokeWidth ?? this.iconConfig.strokeWidth;
}
set strokeWidth(value) {
if (value) {
this._strokeWidth = this.parseNumber(value);
}
else {
delete this._strokeWidth;
}
}
ngOnChanges(changes) {
if (changes.name ||
changes.img ||
changes.color ||
changes.size ||
changes.absoluteStrokeWidth ||
changes.strokeWidth ||
changes.class) {
this.color = this.color ?? this.iconConfig.color;
this.size = this.parseNumber(this.size ?? this.iconConfig.size);
this.strokeWidth = this.parseNumber(this.strokeWidth ?? this.iconConfig.strokeWidth);
this.absoluteStrokeWidth = this.absoluteStrokeWidth ?? this.iconConfig.absoluteStrokeWidth;
const nameOrIcon = this.img ?? this.name;
if (typeof nameOrIcon === 'string') {
const icoOfName = this.getIcon(this.toPascalCase(nameOrIcon));
if (icoOfName) {
this.replaceElement(icoOfName);
}
else {
throw new Error(`The "${nameOrIcon}" icon has not been provided by any available icon providers.`);
}
}
else if (Array.isArray(nameOrIcon)) {
this.replaceElement(nameOrIcon);
}
else {
throw new Error(`No icon name or image has been provided.`);
}
}
this.changeDetector.markForCheck();
}
replaceElement(img) {
const attributes = {
...defaultAttributes,
width: this.size,
height: this.size,
stroke: this.color ?? this.iconConfig.color,
'stroke-width': this.absoluteStrokeWidth
? formatFixed(this.strokeWidth / (this.size / this.defaultSize))
: this.strokeWidth.toString(10),
};
const icoElement = this.createElement(['svg', attributes, img]);
icoElement.classList.add('lucide');
if (typeof this.name === 'string') {
icoElement.classList.add(`lucide-${this.name.replace('_', '-')}`);
}
if (this.class) {
icoElement.classList.add(...this.class
.split(/ /)
.map((a) => a.trim())
.filter((a) => a.length > 0));
}
const childElements = this.elem.nativeElement.childNodes;
for (const child of childElements) {
this.renderer.removeChild(this.elem.nativeElement, child);
}
this.renderer.appendChild(this.elem.nativeElement, icoElement);
}
toPascalCase(str) {
return str.replace(/(\w)([a-z0-9]*)(_|-|\s*)/g, (g0, g1, g2) => g1.toUpperCase() + g2.toLowerCase());
}
parseNumber(value) {
if (typeof value === 'string') {
const parsedValue = parseInt(value, 10);
if (isNaN(parsedValue)) {
throw new Error(`${value} is not numeric.`);
}
return parsedValue;
}
return value;
}
getIcon(name) {
for (const iconProvider of Array.isArray(this.iconProviders)
? this.iconProviders
: [this.iconProviders]) {
if (iconProvider.hasIcon(name)) {
return iconProvider.getIcon(name);
}
}
return null;
}
createElement([tag, attrs, children = []]) {
const element = this.renderer.createElement(tag, 'http://www.w3.org/2000/svg');
Object.keys(attrs).forEach((name) => {
const attrValue = typeof attrs[name] === 'string' ? attrs[name] : attrs[name].toString(10);
this.renderer.setAttribute(element, name, attrValue);
});
if (children.length) {
children.forEach((child) => {
const childElement = this.createElement(child);
this.renderer.appendChild(element, childElement);
});
}
return element;
}
}
LucideAngularComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: LucideAngularComponent, deps: [{ token: ElementRef }, { token: Renderer2 }, { token: ChangeDetectorRef }, { token: LUCIDE_ICONS }, { token: LucideIconConfig }], target: i0.ɵɵFactoryTarget.Component });
LucideAngularComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.12", type: LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: { class: "class", name: "name", img: "img", color: "color", absoluteStrokeWidth: "absoluteStrokeWidth", size: "size", strokeWidth: "strokeWidth" }, usesOnChanges: true, ngImport: i0, template: '<ng-content></ng-content>', isInline: true });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: LucideAngularComponent, decorators: [{
type: Component,
args: [{
selector: 'lucide-angular, lucide-icon, i-lucide, span-lucide',
template: '<ng-content></ng-content>',
}]
}], ctorParameters: function () { return [{ type: i0.ElementRef, decorators: [{
type: Inject,
args: [ElementRef]
}] }, { type: i0.Renderer2, decorators: [{
type: Inject,
args: [Renderer2]
}] }, { type: i0.ChangeDetectorRef, decorators: [{
type: Inject,
args: [ChangeDetectorRef]
}] }, { type: undefined, decorators: [{
type: Inject,
args: [LUCIDE_ICONS]
}] }, { type: i1.LucideIconConfig, decorators: [{
type: Inject,
args: [LucideIconConfig]
}] }]; }, propDecorators: { class: [{
type: Input
}], name: [{
type: Input
}], img: [{
type: Input
}], color: [{
type: Input
}], absoluteStrokeWidth: [{
type: Input
}], size: [{
type: Input
}], strokeWidth: [{
type: Input
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lucide-angular.component.js","sourceRoot":"","sources":["../../../src/lib/lucide-angular.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  Inject,\n  Input,\n  OnChanges,\n  Renderer2,\n  SimpleChange,\n} from '@angular/core';\nimport { LucideIconData } from '../icons/types';\nimport defaultAttributes from '../icons/constants/default-attributes';\nimport { LUCIDE_ICONS, LucideIconProviderInterface } from './lucide-icon.provider';\nimport { LucideIconConfig } from './lucide-icon.config';\n\ninterface TypedChange<T> extends SimpleChange {\n  previousValue: T;\n  currentValue: T;\n}\n\ntype SvgAttributes = { [key: string]: string | number };\n\ntype LucideAngularComponentChanges = {\n  name?: TypedChange<string | LucideIconData>;\n  img?: TypedChange<LucideIconData | undefined>;\n  color?: TypedChange<string>;\n  size?: TypedChange<number>;\n  strokeWidth?: TypedChange<number>;\n  absoluteStrokeWidth?: TypedChange<boolean>;\n  class: TypedChange<string>;\n};\n\nexport function formatFixed(number: number, decimals = 3): string {\n  return parseFloat(number.toFixed(decimals)).toString(10);\n}\n\n@Component({\n  selector: 'lucide-angular, lucide-icon, i-lucide, span-lucide',\n  template: '<ng-content></ng-content>',\n})\nexport class LucideAngularComponent implements OnChanges {\n  @Input() class?: string;\n  @Input() name?: string | LucideIconData;\n  @Input() img?: LucideIconData;\n  @Input() color?: string;\n  @Input() absoluteStrokeWidth = false;\n  defaultSize: number;\n\n  constructor(\n    @Inject(ElementRef) private elem: ElementRef,\n    @Inject(Renderer2) private renderer: Renderer2,\n    @Inject(ChangeDetectorRef) private changeDetector: ChangeDetectorRef,\n    @Inject(LUCIDE_ICONS) private iconProviders: LucideIconProviderInterface[],\n    @Inject(LucideIconConfig) private iconConfig: LucideIconConfig,\n  ) {\n    this.defaultSize = defaultAttributes.height;\n  }\n\n  _size?: number;\n\n  get size(): number {\n    return this._size ?? this.iconConfig.size;\n  }\n\n  @Input() set size(value: string | number | undefined) {\n    if (value) {\n      this._size = this.parseNumber(value);\n    } else {\n      delete this._size;\n    }\n  }\n\n  _strokeWidth?: number;\n\n  get strokeWidth(): number {\n    return this._strokeWidth ?? this.iconConfig.strokeWidth;\n  }\n\n  @Input() set strokeWidth(value: string | number | undefined) {\n    if (value) {\n      this._strokeWidth = this.parseNumber(value);\n    } else {\n      delete this._strokeWidth;\n    }\n  }\n\n  ngOnChanges(changes: LucideAngularComponentChanges): void {\n    if (\n      changes.name ||\n      changes.img ||\n      changes.color ||\n      changes.size ||\n      changes.absoluteStrokeWidth ||\n      changes.strokeWidth ||\n      changes.class\n    ) {\n      this.color = this.color ?? this.iconConfig.color;\n      this.size = this.parseNumber(this.size ?? this.iconConfig.size);\n      this.strokeWidth = this.parseNumber(this.strokeWidth ?? this.iconConfig.strokeWidth);\n      this.absoluteStrokeWidth = this.absoluteStrokeWidth ?? this.iconConfig.absoluteStrokeWidth;\n      const nameOrIcon = this.img ?? this.name;\n      if (typeof nameOrIcon === 'string') {\n        const icoOfName = this.getIcon(this.toPascalCase(nameOrIcon));\n        if (icoOfName) {\n          this.replaceElement(icoOfName);\n        } else {\n          throw new Error(\n            `The \"${nameOrIcon}\" icon has not been provided by any available icon providers.`,\n          );\n        }\n      } else if (Array.isArray(nameOrIcon)) {\n        this.replaceElement(nameOrIcon);\n      } else {\n        throw new Error(`No icon name or image has been provided.`);\n      }\n    }\n\n    this.changeDetector.markForCheck();\n  }\n\n  replaceElement(img: LucideIconData): void {\n    const attributes = {\n      ...defaultAttributes,\n      width: this.size,\n      height: this.size,\n      stroke: this.color ?? this.iconConfig.color,\n      'stroke-width': this.absoluteStrokeWidth\n        ? formatFixed(this.strokeWidth / (this.size / this.defaultSize))\n        : this.strokeWidth.toString(10),\n    };\n    const icoElement = this.createElement(['svg', attributes, img]);\n    icoElement.classList.add('lucide');\n    if (typeof this.name === 'string') {\n      icoElement.classList.add(`lucide-${this.name.replace('_', '-')}`);\n    }\n    if (this.class) {\n      icoElement.classList.add(\n        ...this.class\n          .split(/ /)\n          .map((a) => a.trim())\n          .filter((a) => a.length > 0),\n      );\n    }\n    const childElements = this.elem.nativeElement.childNodes;\n    for (const child of childElements) {\n      this.renderer.removeChild(this.elem.nativeElement, child);\n    }\n    this.renderer.appendChild(this.elem.nativeElement, icoElement);\n  }\n\n  toPascalCase(str: string): string {\n    return str.replace(\n      /(\\w)([a-z0-9]*)(_|-|\\s*)/g,\n      (g0, g1, g2) => g1.toUpperCase() + g2.toLowerCase(),\n    );\n  }\n\n  private parseNumber(value: string | number): number {\n    if (typeof value === 'string') {\n      const parsedValue = parseInt(value, 10);\n      if (isNaN(parsedValue)) {\n        throw new Error(`${value} is not numeric.`);\n      }\n      return parsedValue;\n    }\n    return value;\n  }\n\n  private getIcon(name: string): LucideIconData | null {\n    for (const iconProvider of Array.isArray(this.iconProviders)\n      ? this.iconProviders\n      : [this.iconProviders]) {\n      if (iconProvider.hasIcon(name)) {\n        return iconProvider.getIcon(name);\n      }\n    }\n    return null;\n  }\n\n  private createElement([tag, attrs, children = []]: readonly [\n    string,\n    SvgAttributes,\n    LucideIconData?,\n  ]) {\n    const element = this.renderer.createElement(tag, 'http://www.w3.org/2000/svg');\n\n    Object.keys(attrs).forEach((name) => {\n      const attrValue: string =\n        typeof attrs[name] === 'string' ? (attrs[name] as string) : attrs[name].toString(10);\n      this.renderer.setAttribute(element, name, attrValue);\n    });\n\n    if (children.length) {\n      children.forEach((child) => {\n        const childElement = this.createElement(child);\n        this.renderer.appendChild(element, childElement);\n      });\n    }\n\n    return element;\n  }\n}\n"]}