@nlabs/lex 1.55.1 → 1.56.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,257 @@
1
+ function _define_property(obj, key, value) {
2
+ if (key in obj) {
3
+ Object.defineProperty(obj, key, {
4
+ value: value,
5
+ enumerable: true,
6
+ configurable: true,
7
+ writable: true
8
+ });
9
+ } else {
10
+ obj[key] = value;
11
+ }
12
+ return obj;
13
+ }
14
+ /**
15
+ * Copyright (c) 2018-Present, Nitrogen Labs, Inc.
16
+ * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.
17
+ */ import { readFileSync } from 'fs';
18
+ import { sync as globSync } from 'glob';
19
+ import { basename as pathBasename, dirname as pathDirname, extname as pathExtname, resolve as pathResolve } from 'path';
20
+ import { optimize } from 'svgo';
21
+ import webpack from 'webpack';
22
+ const DEFAULT_FILENAME = 'icons/icons.svg';
23
+ const PLUGIN_NAME = 'LexSvgSpritemapPlugin';
24
+ const ROOT_EXCLUDED_ATTRIBUTES = new Set([
25
+ 'height',
26
+ 'id',
27
+ 'version',
28
+ 'viewBox',
29
+ 'width',
30
+ 'xmlns'
31
+ ]);
32
+ const escapeAttribute = (value)=>{
33
+ return value.replaceAll('&', '&amp;').replaceAll('"', '&quot;').replaceAll('<', '&lt;').replaceAll('>', '&gt;');
34
+ };
35
+ const normalizeAssetFilename = (filename)=>{
36
+ return filename.replace(/^[./]+/, '');
37
+ };
38
+ const parseAttributes = (input)=>{
39
+ const attributes = {};
40
+ const attributeRegex = /([:@\w-]+)\s*=\s*("([^"]*)"|'([^']*)')/g;
41
+ let match = attributeRegex.exec(input);
42
+ while(match){
43
+ attributes[match[1]] = match[3] ?? match[4] ?? '';
44
+ match = attributeRegex.exec(input);
45
+ }
46
+ return attributes;
47
+ };
48
+ const parseSvg = (source)=>{
49
+ const svgMatch = source.replace(/<\?xml[\s\S]*?\?>/ig, '').replace(/<!doctype[\s\S]*?>/ig, '').match(/<svg\b([^>]*)>([\s\S]*?)<\/svg>/i);
50
+ if (!svgMatch) {
51
+ return null;
52
+ }
53
+ return {
54
+ attributes: parseAttributes(svgMatch[1] || ''),
55
+ content: svgMatch[2]?.trim() || ''
56
+ };
57
+ };
58
+ const buildAttributeString = (attributes)=>{
59
+ return Object.entries(attributes).map(([name, value])=>`${name}="${escapeAttribute(value)}"`).join(' ');
60
+ };
61
+ const sanitizeIdentifier = (value)=>{
62
+ const sanitizedValue = value.trim().replaceAll(/\s+/g, '-').replaceAll(/[^A-Za-z0-9_-]/g, '-').replaceAll(/-{2,}/g, '-').replaceAll(/^[-_]+|[-_]+$/g, '');
63
+ const safeValue = sanitizedValue || 'icon';
64
+ return /^[A-Za-z_]/.test(safeValue) ? safeValue : `icon-${safeValue}`;
65
+ };
66
+ const getPrefix = (filePath, prefix)=>{
67
+ if (prefix === false) {
68
+ return '';
69
+ }
70
+ if (typeof prefix === 'function') {
71
+ return prefix(filePath);
72
+ }
73
+ return prefix || '';
74
+ };
75
+ const optimizeSvg = (content, enabled)=>{
76
+ if (!enabled) {
77
+ return content;
78
+ }
79
+ return optimize(content, {
80
+ multipass: true,
81
+ plugins: [
82
+ {
83
+ name: 'preset-default',
84
+ params: {
85
+ overrides: {
86
+ cleanupIds: false
87
+ }
88
+ }
89
+ }
90
+ ]
91
+ }).data;
92
+ };
93
+ const createSpriteSymbol = (filePath, usedIdentifiers, prefix, warnings)=>{
94
+ const source = readFileSync(filePath, 'utf8');
95
+ const parsedSvg = parseSvg(source);
96
+ if (!parsedSvg) {
97
+ warnings.push(new webpack.WebpackError(`Invalid SVG icon: ${filePath}`));
98
+ return null;
99
+ }
100
+ if (!parsedSvg.content) {
101
+ warnings.push(new webpack.WebpackError(`SVG icon is empty: ${filePath}`));
102
+ return null;
103
+ }
104
+ const rawIdentifier = `${getPrefix(filePath, prefix)}${pathBasename(filePath, pathExtname(filePath))}`;
105
+ const baseIdentifier = sanitizeIdentifier(rawIdentifier);
106
+ let identifier = baseIdentifier;
107
+ let duplicateIndex = 1;
108
+ while(usedIdentifiers.has(identifier)){
109
+ duplicateIndex += 1;
110
+ identifier = `${baseIdentifier}-${duplicateIndex}`;
111
+ }
112
+ if (identifier !== baseIdentifier) {
113
+ warnings.push(new webpack.WebpackError(`Duplicate SVG icon id "${baseIdentifier}" detected, using "${identifier}" for ${filePath}`));
114
+ }
115
+ usedIdentifiers.add(identifier);
116
+ const namespaceAttributes = Object.entries(parsedSvg.attributes).reduce((result, [name, value])=>{
117
+ if (name.toLowerCase().startsWith('xmlns:')) {
118
+ result[name] = value;
119
+ }
120
+ return result;
121
+ }, {});
122
+ const symbolAttributes = Object.entries(parsedSvg.attributes).reduce((result, [name, value])=>{
123
+ if (ROOT_EXCLUDED_ATTRIBUTES.has(name)) {
124
+ return result;
125
+ }
126
+ if (name.toLowerCase().startsWith('xmlns:')) {
127
+ return result;
128
+ }
129
+ result[name] = value;
130
+ return result;
131
+ }, {
132
+ id: identifier
133
+ });
134
+ if (parsedSvg.attributes.viewBox) {
135
+ symbolAttributes.viewBox = parsedSvg.attributes.viewBox;
136
+ } else {
137
+ const width = Number.parseFloat(parsedSvg.attributes.width || '');
138
+ const height = Number.parseFloat(parsedSvg.attributes.height || '');
139
+ if (Number.isFinite(width) && Number.isFinite(height)) {
140
+ symbolAttributes.viewBox = `0 0 ${width} ${height}`;
141
+ } else {
142
+ warnings.push(new webpack.WebpackError(`SVG icon is missing a viewBox and readable width/height: ${filePath}`));
143
+ return null;
144
+ }
145
+ }
146
+ return {
147
+ filePath,
148
+ namespaceAttributes,
149
+ symbol: `<symbol ${buildAttributeString(symbolAttributes)}>${parsedSvg.content}</symbol>`
150
+ };
151
+ };
152
+ const collectFiles = (patterns, allowDuplicates)=>{
153
+ const files = patterns.flatMap((pattern)=>{
154
+ return globSync(pattern, {
155
+ absolute: true,
156
+ dot: false,
157
+ nodir: true
158
+ });
159
+ }).map((filePath)=>pathResolve(filePath));
160
+ if (allowDuplicates) {
161
+ return files;
162
+ }
163
+ return [
164
+ ...new Set(files)
165
+ ];
166
+ };
167
+ export class LexSvgSpritemapPlugin {
168
+ buildSpritemap() {
169
+ const warnings = [];
170
+ const filePaths = collectFiles(this.patterns, this.options.allowDuplicates);
171
+ if (filePaths.length === 0) {
172
+ return {
173
+ content: null,
174
+ filePaths,
175
+ warnings
176
+ };
177
+ }
178
+ const namespaceAttributes = {};
179
+ const usedIdentifiers = new Set();
180
+ const symbols = filePaths.map((filePath)=>{
181
+ return createSpriteSymbol(filePath, usedIdentifiers, this.options.prefix, warnings);
182
+ }).filter((entry)=>entry !== null);
183
+ if (symbols.length === 0) {
184
+ return {
185
+ content: null,
186
+ filePaths,
187
+ warnings
188
+ };
189
+ }
190
+ for (const symbol of symbols){
191
+ for (const [name, value] of Object.entries(symbol.namespaceAttributes)){
192
+ namespaceAttributes[name] = value;
193
+ }
194
+ }
195
+ const rootAttributes = buildAttributeString({
196
+ xmlns: 'http://www.w3.org/2000/svg',
197
+ ...namespaceAttributes
198
+ });
199
+ const spriteContent = `<svg ${rootAttributes}>${symbols.map((symbol)=>symbol.symbol).join('')}</svg>`;
200
+ return {
201
+ content: optimizeSvg(spriteContent, this.options.optimize),
202
+ filePaths,
203
+ warnings
204
+ };
205
+ }
206
+ apply(compiler) {
207
+ compiler.hooks.thisCompilation.tap(PLUGIN_NAME, (compilation)=>{
208
+ const directories = new Set();
209
+ const spritemap = this.buildSpritemap();
210
+ for (const filePath of spritemap.filePaths){
211
+ compilation.fileDependencies.add(filePath);
212
+ directories.add(pathDirname(filePath));
213
+ }
214
+ for (const pattern of this.patterns){
215
+ const baseDirectory = pathResolve(pattern.replace(/\*.*$/, ''));
216
+ directories.add(baseDirectory);
217
+ }
218
+ for (const directory of directories){
219
+ compilation.contextDependencies.add(directory);
220
+ }
221
+ if (spritemap.warnings.length > 0) {
222
+ compilation.warnings.push(...spritemap.warnings);
223
+ }
224
+ compilation.hooks.processAssets.tap({
225
+ name: PLUGIN_NAME,
226
+ stage: webpack.Compilation.PROCESS_ASSETS_STAGE_ADDITIONAL
227
+ }, ()=>{
228
+ if (!spritemap.content) {
229
+ return;
230
+ }
231
+ const assetName = this.options.filename;
232
+ const source = new webpack.sources.RawSource(spritemap.content);
233
+ if (compilation.getAsset(assetName)) {
234
+ compilation.updateAsset(assetName, source);
235
+ return;
236
+ }
237
+ compilation.emitAsset(assetName, source);
238
+ });
239
+ });
240
+ }
241
+ constructor(patterns, options = {}){
242
+ _define_property(this, "patterns", void 0);
243
+ _define_property(this, "options", void 0);
244
+ this.patterns = Array.isArray(patterns) ? patterns : [
245
+ patterns
246
+ ];
247
+ this.options = {
248
+ allowDuplicates: options.allowDuplicates ?? false,
249
+ filename: normalizeAssetFilename(options.filename || DEFAULT_FILENAME),
250
+ optimize: options.optimize ?? true,
251
+ prefix: options.prefix ?? false
252
+ };
253
+ }
254
+ }
255
+ export default LexSvgSpritemapPlugin;
256
+
257
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/utils/webpack/LexSvgSpritemapPlugin.ts"],"sourcesContent":["/**\n * Copyright (c) 2018-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {readFileSync} from 'fs';\nimport {sync as globSync} from 'glob';\nimport {basename as pathBasename, dirname as pathDirname, extname as pathExtname, resolve as pathResolve} from 'path';\nimport {optimize} from 'svgo';\nimport webpack from 'webpack';\n\ntype IdentifierPrefix = false | string | ((filePath: string) => string);\n\nexport interface LexSvgSpritemapPluginOptions {\n  readonly allowDuplicates?: boolean;\n  readonly filename?: string;\n  readonly optimize?: boolean;\n  readonly prefix?: IdentifierPrefix;\n}\n\ninterface ParsedSvg {\n  readonly attributes: Record<string, string>;\n  readonly content: string;\n}\n\ninterface SpriteSymbol {\n  readonly filePath: string;\n  readonly namespaceAttributes: Record<string, string>;\n  readonly symbol: string;\n}\n\nconst DEFAULT_FILENAME = 'icons/icons.svg';\nconst PLUGIN_NAME = 'LexSvgSpritemapPlugin';\nconst ROOT_EXCLUDED_ATTRIBUTES = new Set([\n  'height',\n  'id',\n  'version',\n  'viewBox',\n  'width',\n  'xmlns'\n]);\n\nconst escapeAttribute = (value: string): string => {\n  return value\n    .replaceAll('&', '&amp;')\n    .replaceAll('\"', '&quot;')\n    .replaceAll('<', '&lt;')\n    .replaceAll('>', '&gt;');\n};\n\nconst normalizeAssetFilename = (filename: string): string => {\n  return filename.replace(/^[./]+/, '');\n};\n\nconst parseAttributes = (input: string): Record<string, string> => {\n  const attributes: Record<string, string> = {};\n  const attributeRegex = /([:@\\w-]+)\\s*=\\s*(\"([^\"]*)\"|'([^']*)')/g;\n\n  let match: RegExpExecArray | null = attributeRegex.exec(input);\n  while(match) {\n    attributes[match[1]] = match[3] ?? match[4] ?? '';\n    match = attributeRegex.exec(input);\n  }\n\n  return attributes;\n};\n\nconst parseSvg = (source: string): ParsedSvg | null => {\n  const svgMatch = source\n    .replace(/<\\?xml[\\s\\S]*?\\?>/ig, '')\n    .replace(/<!doctype[\\s\\S]*?>/ig, '')\n    .match(/<svg\\b([^>]*)>([\\s\\S]*?)<\\/svg>/i);\n\n  if(!svgMatch) {\n    return null;\n  }\n\n  return {\n    attributes: parseAttributes(svgMatch[1] || ''),\n    content: svgMatch[2]?.trim() || ''\n  };\n};\n\nconst buildAttributeString = (attributes: Record<string, string>): string => {\n  return Object.entries(attributes)\n    .map(([name, value]) => `${name}=\"${escapeAttribute(value)}\"`)\n    .join(' ');\n};\n\nconst sanitizeIdentifier = (value: string): string => {\n  const sanitizedValue = value\n    .trim()\n    .replaceAll(/\\s+/g, '-')\n    .replaceAll(/[^A-Za-z0-9_-]/g, '-')\n    .replaceAll(/-{2,}/g, '-')\n    .replaceAll(/^[-_]+|[-_]+$/g, '');\n\n  const safeValue = sanitizedValue || 'icon';\n\n  return /^[A-Za-z_]/.test(safeValue) ? safeValue : `icon-${safeValue}`;\n};\n\nconst getPrefix = (filePath: string, prefix: IdentifierPrefix): string => {\n  if(prefix === false) {\n    return '';\n  }\n\n  if(typeof prefix === 'function') {\n    return prefix(filePath);\n  }\n\n  return prefix || '';\n};\n\nconst optimizeSvg = (content: string, enabled: boolean): string => {\n  if(!enabled) {\n    return content;\n  }\n\n  return optimize(content, {\n    multipass: true,\n    plugins: [{\n      name: 'preset-default',\n      params: {\n        overrides: {\n          cleanupIds: false\n        }\n      }\n    }]\n  }).data;\n};\n\nconst createSpriteSymbol = (\n  filePath: string,\n  usedIdentifiers: Set<string>,\n  prefix: IdentifierPrefix,\n  warnings: webpack.WebpackError[]\n): SpriteSymbol | null => {\n  const source = readFileSync(filePath, 'utf8');\n  const parsedSvg = parseSvg(source);\n\n  if(!parsedSvg) {\n    warnings.push(new webpack.WebpackError(`Invalid SVG icon: ${filePath}`));\n    return null;\n  }\n\n  if(!parsedSvg.content) {\n    warnings.push(new webpack.WebpackError(`SVG icon is empty: ${filePath}`));\n    return null;\n  }\n\n  const rawIdentifier = `${getPrefix(filePath, prefix)}${pathBasename(filePath, pathExtname(filePath))}`;\n  const baseIdentifier = sanitizeIdentifier(rawIdentifier);\n\n  let identifier = baseIdentifier;\n  let duplicateIndex = 1;\n\n  while(usedIdentifiers.has(identifier)) {\n    duplicateIndex += 1;\n    identifier = `${baseIdentifier}-${duplicateIndex}`;\n  }\n\n  if(identifier !== baseIdentifier) {\n    warnings.push(new webpack.WebpackError(`Duplicate SVG icon id \"${baseIdentifier}\" detected, using \"${identifier}\" for ${filePath}`));\n  }\n\n  usedIdentifiers.add(identifier);\n\n  const namespaceAttributes = Object.entries(parsedSvg.attributes).reduce<Record<string, string>>((result, [name, value]) => {\n    if(name.toLowerCase().startsWith('xmlns:')) {\n      result[name] = value;\n    }\n\n    return result;\n  }, {});\n\n  const symbolAttributes = Object.entries(parsedSvg.attributes).reduce<Record<string, string>>((result, [name, value]) => {\n    if(ROOT_EXCLUDED_ATTRIBUTES.has(name)) {\n      return result;\n    }\n\n    if(name.toLowerCase().startsWith('xmlns:')) {\n      return result;\n    }\n\n    result[name] = value;\n\n    return result;\n  }, {\n    id: identifier\n  });\n\n  if(parsedSvg.attributes.viewBox) {\n    symbolAttributes.viewBox = parsedSvg.attributes.viewBox;\n  } else {\n    const width = Number.parseFloat(parsedSvg.attributes.width || '');\n    const height = Number.parseFloat(parsedSvg.attributes.height || '');\n\n    if(Number.isFinite(width) && Number.isFinite(height)) {\n      symbolAttributes.viewBox = `0 0 ${width} ${height}`;\n    } else {\n      warnings.push(new webpack.WebpackError(`SVG icon is missing a viewBox and readable width/height: ${filePath}`));\n      return null;\n    }\n  }\n\n  return {\n    filePath,\n    namespaceAttributes,\n    symbol: `<symbol ${buildAttributeString(symbolAttributes)}>${parsedSvg.content}</symbol>`\n  };\n};\n\nconst collectFiles = (patterns: readonly string[], allowDuplicates: boolean): string[] => {\n  const files = patterns.flatMap((pattern) => {\n    return globSync(pattern, {\n      absolute: true,\n      dot: false,\n      nodir: true\n    }) as string[];\n  }).map((filePath) => pathResolve(filePath));\n\n  if(allowDuplicates) {\n    return files;\n  }\n\n  return [...new Set(files)];\n};\n\nexport class LexSvgSpritemapPlugin {\n  readonly patterns: string[];\n  readonly options: Required<LexSvgSpritemapPluginOptions>;\n\n  constructor(patterns: string | string[], options: LexSvgSpritemapPluginOptions = {}) {\n    this.patterns = Array.isArray(patterns) ? patterns : [patterns];\n    this.options = {\n      allowDuplicates: options.allowDuplicates ?? false,\n      filename: normalizeAssetFilename(options.filename || DEFAULT_FILENAME),\n      optimize: options.optimize ?? true,\n      prefix: options.prefix ?? false\n    };\n  }\n\n  buildSpritemap(): {content: string | null; filePaths: string[]; warnings: webpack.WebpackError[]} {\n    const warnings: webpack.WebpackError[] = [];\n    const filePaths = collectFiles(this.patterns, this.options.allowDuplicates);\n\n    if(filePaths.length === 0) {\n      return {\n        content: null,\n        filePaths,\n        warnings\n      };\n    }\n\n    const namespaceAttributes: Record<string, string> = {};\n    const usedIdentifiers = new Set<string>();\n    const symbols = filePaths.map((filePath) => {\n      return createSpriteSymbol(filePath, usedIdentifiers, this.options.prefix, warnings);\n    }).filter((entry): entry is SpriteSymbol => entry !== null);\n\n    if(symbols.length === 0) {\n      return {\n        content: null,\n        filePaths,\n        warnings\n      };\n    }\n\n    for(const symbol of symbols) {\n      for(const [name, value] of Object.entries(symbol.namespaceAttributes)) {\n        namespaceAttributes[name] = value;\n      }\n    }\n\n    const rootAttributes = buildAttributeString({\n      xmlns: 'http://www.w3.org/2000/svg',\n      ...namespaceAttributes\n    });\n    const spriteContent = `<svg ${rootAttributes}>${symbols.map((symbol) => symbol.symbol).join('')}</svg>`;\n\n    return {\n      content: optimizeSvg(spriteContent, this.options.optimize),\n      filePaths,\n      warnings\n    };\n  }\n\n  apply(compiler: webpack.Compiler): void {\n    compiler.hooks.thisCompilation.tap(PLUGIN_NAME, (compilation) => {\n      const directories = new Set<string>();\n      const spritemap = this.buildSpritemap();\n\n      for(const filePath of spritemap.filePaths) {\n        compilation.fileDependencies.add(filePath);\n        directories.add(pathDirname(filePath));\n      }\n\n      for(const pattern of this.patterns) {\n        const baseDirectory = pathResolve(pattern.replace(/\\*.*$/, ''));\n        directories.add(baseDirectory);\n      }\n\n      for(const directory of directories) {\n        compilation.contextDependencies.add(directory);\n      }\n\n      if(spritemap.warnings.length > 0) {\n        compilation.warnings.push(...spritemap.warnings);\n      }\n\n      compilation.hooks.processAssets.tap({\n        name: PLUGIN_NAME,\n        stage: webpack.Compilation.PROCESS_ASSETS_STAGE_ADDITIONAL\n      }, () => {\n        if(!spritemap.content) {\n          return;\n        }\n\n        const assetName = this.options.filename;\n        const source = new webpack.sources.RawSource(spritemap.content);\n\n        if(compilation.getAsset(assetName)) {\n          compilation.updateAsset(assetName, source);\n          return;\n        }\n\n        compilation.emitAsset(assetName, source);\n      });\n    });\n  }\n}\n\nexport default LexSvgSpritemapPlugin;\n"],"names":["readFileSync","sync","globSync","basename","pathBasename","dirname","pathDirname","extname","pathExtname","resolve","pathResolve","optimize","webpack","DEFAULT_FILENAME","PLUGIN_NAME","ROOT_EXCLUDED_ATTRIBUTES","Set","escapeAttribute","value","replaceAll","normalizeAssetFilename","filename","replace","parseAttributes","input","attributes","attributeRegex","match","exec","parseSvg","source","svgMatch","content","trim","buildAttributeString","Object","entries","map","name","join","sanitizeIdentifier","sanitizedValue","safeValue","test","getPrefix","filePath","prefix","optimizeSvg","enabled","multipass","plugins","params","overrides","cleanupIds","data","createSpriteSymbol","usedIdentifiers","warnings","parsedSvg","push","WebpackError","rawIdentifier","baseIdentifier","identifier","duplicateIndex","has","add","namespaceAttributes","reduce","result","toLowerCase","startsWith","symbolAttributes","id","viewBox","width","Number","parseFloat","height","isFinite","symbol","collectFiles","patterns","allowDuplicates","files","flatMap","pattern","absolute","dot","nodir","LexSvgSpritemapPlugin","buildSpritemap","filePaths","options","length","symbols","filter","entry","rootAttributes","xmlns","spriteContent","apply","compiler","hooks","thisCompilation","tap","compilation","directories","spritemap","fileDependencies","baseDirectory","directory","contextDependencies","processAssets","stage","Compilation","PROCESS_ASSETS_STAGE_ADDITIONAL","assetName","sources","RawSource","getAsset","updateAsset","emitAsset","Array","isArray"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@nlabs/lex",
3
- "version": "1.55.1",
3
+ "version": "1.56.0",
4
4
  "description": "Lex",
5
5
  "license": "MIT",
6
6
  "type": "module",
@@ -83,66 +83,66 @@
83
83
  "dependencies": {
84
84
  "@mdx-js/loader": "^3.1.1",
85
85
  "@nlabs/webpack-plugin-static-site": "*",
86
- "@storybook/addon-docs": "^10.2.12",
87
- "@storybook/addon-links": "^10.2.12",
86
+ "@storybook/addon-docs": "^10.3.4",
87
+ "@storybook/addon-links": "^10.3.4",
88
88
  "@storybook/addon-postcss": "^2.0.0",
89
- "@storybook/addon-styling-webpack": "^3.0.0",
90
- "@storybook/addon-themes": "^10.2.12",
91
- "@storybook/cli": "^10.2.12",
92
- "@storybook/react": "^10.2.12",
93
- "@storybook/react-webpack5": "^10.2.12",
94
- "@swc/cli": "^0.8.0",
95
- "@swc/core": "^1.15.13",
89
+ "@storybook/addon-styling-webpack": "^3.0.2",
90
+ "@storybook/addon-themes": "^10.3.4",
91
+ "@storybook/cli": "^10.3.4",
92
+ "@storybook/react": "^10.3.4",
93
+ "@storybook/react-webpack5": "^10.3.4",
94
+ "@swc/cli": "^0.8.1",
95
+ "@swc/core": "^1.15.24",
96
96
  "@tailwindcss/forms": "^0.5.11",
97
97
  "@tailwindcss/nesting": "^0.0.0-insiders.565cd3e",
98
- "@tailwindcss/postcss": "4.2.1",
98
+ "@tailwindcss/postcss": "4.2.2",
99
99
  "@testing-library/jest-dom": "^6.9.1",
100
100
  "@testing-library/react": "^16.3.2",
101
- "@vitest/coverage-v8": "^4.0.18",
101
+ "@vitest/coverage-v8": "^4.1.2",
102
102
  "assert": "^2.1.0",
103
103
  "autoprefixer": "^10.4.27",
104
104
  "boxen": "8.0.1",
105
105
  "buffer": "^6.0.3",
106
- "caniuse-lite": "1.0.30001774",
106
+ "caniuse-lite": "1.0.30001786",
107
107
  "chalk": "^5.6.2",
108
108
  "commander": "^14.0.3",
109
109
  "compare-versions": "^6.1.1",
110
- "compression-webpack-plugin": "^11.1.0",
111
- "copy-webpack-plugin": "^13.0.1",
112
- "core-js": "^3.48.0",
110
+ "compression-webpack-plugin": "^12.0.0",
111
+ "copy-webpack-plugin": "^14.0.0",
112
+ "core-js": "^3.49.0",
113
113
  "crypto-js": "^4.2.0",
114
114
  "css-loader": "^7.1.4",
115
- "css-minimizer-webpack-plugin": "^7.0.4",
116
- "cssnano": "^7.1.2",
117
- "dotenv-webpack": "^8.1.1",
115
+ "css-minimizer-webpack-plugin": "^8.0.0",
116
+ "cssnano": "^7.1.4",
117
+ "dotenv-webpack": "^9.0.0",
118
118
  "download-npm-package": "^3.1.12",
119
- "eslint": "^10.0.2",
120
- "eslint-config-styleguidejs": "^5.0.2",
119
+ "eslint": "^10.2.0",
120
+ "eslint-config-styleguidejs": "^5.0.4",
121
121
  "execa": "9.6.1",
122
- "expect": "^30.2.0",
122
+ "expect": "^30.3.0",
123
123
  "express": "^5.2.1",
124
124
  "favicons-webpack-plugin": "^6.0.1",
125
125
  "find-file-up": "^2.0.1",
126
126
  "glob": "^13.0.6",
127
127
  "global": "^4.4.0",
128
- "graphql": "^16.13.0",
128
+ "graphql": "^16.13.2",
129
129
  "graphql-tag": "^2.12.6",
130
130
  "html-loader": "^5.1.0",
131
131
  "html-webpack-plugin": "^5.6.6",
132
132
  "https-browserify": "^1.0.0",
133
133
  "image-webpack-loader": "^8.1.0",
134
- "jsdom": "^28.1.0",
134
+ "jsdom": "^29.0.1",
135
135
  "latest-version": "9.0.0",
136
- "lodash": "^4.17.23",
136
+ "lodash": "^4.18.1",
137
137
  "luxon": "^3.7.2",
138
138
  "math-expression-evaluator": "^2.0.7",
139
- "mini-css-extract-plugin": "^2.10.0",
140
- "npm-check-updates": "^19.5.0",
141
- "openai": "^6.25.0",
139
+ "mini-css-extract-plugin": "^2.10.2",
140
+ "npm-check-updates": "^20.0.0",
141
+ "openai": "^6.33.0",
142
142
  "ora": "9.3.0",
143
143
  "os-browserify": "^0.3.0",
144
144
  "path-browserify": "^1.0.1",
145
- "postcss": "^8.5.6",
145
+ "postcss": "^8.5.8",
146
146
  "postcss-browser-reporter": "^0.7.0",
147
147
  "postcss-cli": "^11.0.1",
148
148
  "postcss-custom-properties": "^15.0.1",
@@ -155,7 +155,7 @@
155
155
  "postcss-percentage": "^0.0.0",
156
156
  "postcss-preset-env": "^11.2.0",
157
157
  "postcss-simple-vars": "^7.0.1",
158
- "postcss-svgo": "7.1.0",
158
+ "postcss-svgo": "7.1.1",
159
159
  "postcss-url": "10.1.3",
160
160
  "postcss-value-parser": "^4.2.0",
161
161
  "process": "^0.11.10",
@@ -167,45 +167,43 @@
167
167
  "rimraf": "^6.1.3",
168
168
  "semver": "^7.7.4",
169
169
  "source-map-loader": "^5.0.0",
170
- "storybook": "^10.2.12",
170
+ "storybook": "^10.3.4",
171
171
  "stream-browserify": "^3.0.0",
172
172
  "stream-http": "^3.2.0",
173
173
  "style-loader": "^4.0.0",
174
- "svg-spritemap-webpack-plugin": "^5.1.0",
175
- "svgo": "4.0.0",
174
+ "svgo": "4.0.1",
176
175
  "swc-loader": "^0.2.7",
177
- "tailwindcss": "^4.2.1",
176
+ "tailwindcss": "^4.2.2",
178
177
  "ts-node": "^10.9.2",
179
178
  "tsconfig-paths-webpack-plugin": "^4.2.0",
180
- "typescript": "5.9.3",
179
+ "typescript": "^6.0.2",
181
180
  "util": "^0.12.5",
182
- "vitest": "^4.0.18",
181
+ "vitest": "^4.1.2",
183
182
  "vm-browserify": "^1.1.2",
184
- "webpack": "5.105.2",
185
- "webpack-bundle-analyzer": "^5.2.0",
186
- "webpack-cli": "^6.0.1",
183
+ "webpack": "5.105.4",
184
+ "webpack-bundle-analyzer": "^5.3.0",
185
+ "webpack-cli": "^7.0.2",
187
186
  "webpack-merge": "^6.0.1",
188
187
  "webpack-plugin-serve": "^1.6.0",
189
- "ws": "^8.19.0"
188
+ "ws": "^8.20.0"
190
189
  },
191
190
  "devDependencies": {
192
191
  "@types/express": "^5.0.6",
193
192
  "@types/luxon": "^3.7.1",
194
- "@types/node": "^25.3.1",
193
+ "@types/node": "^25.5.2",
195
194
  "@types/ora": "^3.2.0",
196
195
  "@types/react": "^19.2.14",
197
196
  "@types/webpack": "^5.28.5"
198
197
  },
199
198
  "overrides": {
200
199
  "cross-spawn": "^7.0.6",
201
- "fast-xml-parser": ">=5.4.1",
202
200
  "got": "^14.6.6",
203
- "http-cache-semantics": "^4.2.0",
204
201
  "http-proxy-middleware": "^3.0.5",
205
202
  "minimatch": ">=10.2.4",
206
- "postcss": "^8.5.6",
203
+ "postcss": "^8.5.8",
207
204
  "semver-regex": "^4.0.5",
208
- "tmp": "^0.2.5"
205
+ "tmp": "^0.2.5",
206
+ "file-type": "^21.3.1"
209
207
  },
210
208
  "gitHead": "0f5fb22fc0f0ab1abab5adf62afed24c7c87e4a8"
211
209
  }
package/tsconfig.json CHANGED
@@ -21,6 +21,7 @@
21
21
  "pretty": true,
22
22
  "resolveJsonModule": true,
23
23
  "skipLibCheck": true,
24
+ "useUnknownInCatchVariables": false,
24
25
  "target": "ESNext",
25
26
  "types": [
26
27
  "vitest/globals",
package/webpack.config.js CHANGED
@@ -25,7 +25,6 @@ import postcssImport from 'postcss-import';
25
25
  import postcssNesting from 'postcss-nesting';
26
26
  import postcssPresetEnv from 'postcss-preset-env';
27
27
  import postcssUrl from 'postcss-url';
28
- import SVGSpriteMapPlugin from 'svg-spritemap-webpack-plugin';
29
28
  import TsconfigPathsPlugin from 'tsconfig-paths-webpack-plugin';
30
29
  import {URL} from 'url';
31
30
  import {default as webpack} from 'webpack';
@@ -35,6 +34,7 @@ import {WebpackPluginServe} from 'webpack-plugin-serve';
35
34
 
36
35
  import {LexConfig} from './lib/LexConfig.js';
37
36
  import {relativeFilePath, relativeNodePath} from './lib/utils/file.js';
37
+ import LexSvgSpritemapPlugin from './lib/utils/webpack/LexSvgSpritemapPlugin.js';
38
38
  import postcssFor from './lib/utils/postcss/postcss-for.js';
39
39
  import postcssPercentage from './lib/utils/postcss/postcss-percentage.js';
40
40
  import tail from 'lodash/tail.js';
@@ -175,17 +175,10 @@ const svgPaths = `${sourceFullPath}/icons/**/**.svg`;
175
175
 
176
176
  if(globSync(svgPaths, globOptions).length) {
177
177
  plugins.push(
178
- new SVGSpriteMapPlugin(svgPaths, {
179
- input: {
180
- allowDuplicates: false
181
- },
182
- output: {
183
- chunk: {keep: true},
184
- filename: './icons/icons.svg'
185
- },
186
- sprite: {
187
- prefix: false
188
- }
178
+ new LexSvgSpritemapPlugin(svgPaths, {
179
+ allowDuplicates: false,
180
+ filename: './icons/icons.svg',
181
+ prefix: false
189
182
  })
190
183
  );
191
184
  }
@@ -952,4 +945,4 @@ export default (webpackEnv, webpackOptions) => {
952
945
  }
953
946
 
954
947
  return mergedConfig;
955
- };
948
+ };