@tscircuit/eval 0.0.491 → 0.0.493
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,292 @@
|
|
|
1
|
+
// node_modules/@tscircuit/parts-engine/lib/footprint-translators/get-footprinter-string-from-kicad.ts
|
|
2
|
+
var getFootprinterStringFromKicad = (kicadFootprint) => {
|
|
3
|
+
let match = kicadFootprint.match(/:[RC]_(\d{4})_/);
|
|
4
|
+
if (match) return match[1];
|
|
5
|
+
match = kicadFootprint.match(
|
|
6
|
+
/:(SOIC-\d+|SOT-\d+|SOD-\d+|SSOP-\d+|TSSOP-\d+|QFP-\d+|QFN-\d+)/
|
|
7
|
+
);
|
|
8
|
+
if (match) return match[1];
|
|
9
|
+
return void 0;
|
|
10
|
+
};
|
|
11
|
+
|
|
12
|
+
// node_modules/@tscircuit/parts-engine/lib/footprint-translators/get-jlc-package-from-footprinter-string.ts
|
|
13
|
+
var getJlcPackageFromFootprinterString = (footprinterString) => {
|
|
14
|
+
if (footprinterString.includes("cap")) {
|
|
15
|
+
return footprinterString.replace(/cap/g, "");
|
|
16
|
+
}
|
|
17
|
+
return footprinterString;
|
|
18
|
+
};
|
|
19
|
+
|
|
20
|
+
// node_modules/@tscircuit/parts-engine/lib/footprint-translators/index.ts
|
|
21
|
+
var getJlcpcbPackageName = (footprint) => {
|
|
22
|
+
if (!footprint) return void 0;
|
|
23
|
+
if (footprint.startsWith("kicad:")) {
|
|
24
|
+
const footprinterString = getFootprinterStringFromKicad(footprint);
|
|
25
|
+
if (footprinterString) {
|
|
26
|
+
return getJlcPackageFromFootprinterString(footprinterString);
|
|
27
|
+
}
|
|
28
|
+
return footprint;
|
|
29
|
+
}
|
|
30
|
+
return getJlcPackageFromFootprinterString(footprint);
|
|
31
|
+
};
|
|
32
|
+
|
|
33
|
+
// node_modules/@tscircuit/parts-engine/lib/jlc-parts-engine.ts
|
|
34
|
+
var cache = /* @__PURE__ */ new Map();
|
|
35
|
+
var getJlcPartsCached = async (name, params) => {
|
|
36
|
+
const paramString = new URLSearchParams({
|
|
37
|
+
...params,
|
|
38
|
+
json: "true"
|
|
39
|
+
}).toString();
|
|
40
|
+
if (cache.has(paramString)) {
|
|
41
|
+
return cache.get(paramString);
|
|
42
|
+
}
|
|
43
|
+
const response = await fetch(
|
|
44
|
+
`https://jlcsearch.tscircuit.com/${name}/list?${paramString}`
|
|
45
|
+
);
|
|
46
|
+
const responseJson = await response.json();
|
|
47
|
+
cache.set(paramString, responseJson);
|
|
48
|
+
return responseJson;
|
|
49
|
+
};
|
|
50
|
+
var withBasicPartPreference = (parts) => {
|
|
51
|
+
if (!parts) return [];
|
|
52
|
+
return [...parts].sort(
|
|
53
|
+
(a, b) => Number(b.is_basic ?? false) - Number(a.is_basic ?? false)
|
|
54
|
+
);
|
|
55
|
+
};
|
|
56
|
+
var jlcPartsEngine = {
|
|
57
|
+
findPart: async ({
|
|
58
|
+
sourceComponent,
|
|
59
|
+
footprinterString
|
|
60
|
+
}) => {
|
|
61
|
+
const jlcpcbPackage = getJlcpcbPackageName(footprinterString);
|
|
62
|
+
if (sourceComponent.type === "source_component" && sourceComponent.ftype === "simple_resistor") {
|
|
63
|
+
const { resistors } = await getJlcPartsCached("resistors", {
|
|
64
|
+
resistance: sourceComponent.resistance,
|
|
65
|
+
package: jlcpcbPackage
|
|
66
|
+
});
|
|
67
|
+
return {
|
|
68
|
+
jlcpcb: withBasicPartPreference(resistors).map((r) => `C${r.lcsc}`).slice(0, 3)
|
|
69
|
+
};
|
|
70
|
+
} else if (sourceComponent.type === "source_component" && sourceComponent.ftype === "simple_capacitor") {
|
|
71
|
+
const { capacitors } = await getJlcPartsCached("capacitors", {
|
|
72
|
+
capacitance: sourceComponent.capacitance,
|
|
73
|
+
package: jlcpcbPackage
|
|
74
|
+
});
|
|
75
|
+
return {
|
|
76
|
+
jlcpcb: withBasicPartPreference(capacitors).map((c) => `C${c.lcsc}`).slice(0, 3)
|
|
77
|
+
};
|
|
78
|
+
} else if (sourceComponent.type === "source_component" && sourceComponent.ftype === "simple_pin_header") {
|
|
79
|
+
let pitch;
|
|
80
|
+
if (footprinterString?.includes("_p")) {
|
|
81
|
+
pitch = Number(footprinterString.split("_p")[1]);
|
|
82
|
+
}
|
|
83
|
+
const { headers } = await getJlcPartsCached(
|
|
84
|
+
"headers",
|
|
85
|
+
pitch ? {
|
|
86
|
+
pitch,
|
|
87
|
+
num_pins: sourceComponent.pin_count,
|
|
88
|
+
gender: sourceComponent.gender
|
|
89
|
+
} : {
|
|
90
|
+
num_pins: sourceComponent.pin_count,
|
|
91
|
+
gender: sourceComponent.gender
|
|
92
|
+
}
|
|
93
|
+
);
|
|
94
|
+
return {
|
|
95
|
+
jlcpcb: withBasicPartPreference(headers).map((h) => `C${h.lcsc}`).slice(0, 3)
|
|
96
|
+
};
|
|
97
|
+
} else if (sourceComponent.type === "source_component" && sourceComponent.ftype === "simple_potentiometer") {
|
|
98
|
+
const { potentiometers } = await getJlcPartsCached("potentiometers", {
|
|
99
|
+
resistance: sourceComponent.max_resistance,
|
|
100
|
+
package: jlcpcbPackage
|
|
101
|
+
});
|
|
102
|
+
return {
|
|
103
|
+
jlcpcb: withBasicPartPreference(potentiometers).map((p) => `C${p.lcsc}`).slice(0, 3)
|
|
104
|
+
};
|
|
105
|
+
} else if (sourceComponent.type === "source_component" && sourceComponent.ftype === "simple_diode") {
|
|
106
|
+
const { diodes } = await getJlcPartsCached("diodes", {
|
|
107
|
+
package: jlcpcbPackage
|
|
108
|
+
});
|
|
109
|
+
return {
|
|
110
|
+
jlcpcb: withBasicPartPreference(diodes).map((d) => `C${d.lcsc}`).slice(0, 3)
|
|
111
|
+
};
|
|
112
|
+
} else if (sourceComponent.type === "source_component" && sourceComponent.ftype === "simple_chip") {
|
|
113
|
+
const { chips } = await getJlcPartsCached("chips", {
|
|
114
|
+
package: jlcpcbPackage
|
|
115
|
+
});
|
|
116
|
+
return {
|
|
117
|
+
jlcpcb: withBasicPartPreference(chips).map((c) => `C${c.lcsc}`).slice(0, 3)
|
|
118
|
+
};
|
|
119
|
+
} else if (sourceComponent.type === "source_component" && sourceComponent.ftype === "simple_transistor") {
|
|
120
|
+
const { transistors } = await getJlcPartsCached("transistors", {
|
|
121
|
+
package: jlcpcbPackage,
|
|
122
|
+
transistor_type: sourceComponent.transistor_type
|
|
123
|
+
});
|
|
124
|
+
return {
|
|
125
|
+
jlcpcb: withBasicPartPreference(transistors).map((t) => `C${t.lcsc}`).slice(0, 3)
|
|
126
|
+
};
|
|
127
|
+
} else if (sourceComponent.type === "source_component" && sourceComponent.ftype === "simple_power_source") {
|
|
128
|
+
const { power_sources } = await getJlcPartsCached("power_sources", {
|
|
129
|
+
voltage: sourceComponent.voltage,
|
|
130
|
+
package: jlcpcbPackage
|
|
131
|
+
});
|
|
132
|
+
return {
|
|
133
|
+
jlcpcb: withBasicPartPreference(power_sources).map((p) => `C${p.lcsc}`).slice(0, 3)
|
|
134
|
+
};
|
|
135
|
+
} else if (sourceComponent.type === "source_component" && sourceComponent.ftype === "simple_inductor") {
|
|
136
|
+
const { inductors } = await getJlcPartsCached("inductors", {
|
|
137
|
+
inductance: sourceComponent.inductance,
|
|
138
|
+
package: jlcpcbPackage
|
|
139
|
+
});
|
|
140
|
+
return {
|
|
141
|
+
jlcpcb: withBasicPartPreference(inductors).map((i) => `C${i.lcsc}`).slice(0, 3)
|
|
142
|
+
};
|
|
143
|
+
} else if (sourceComponent.type === "source_component" && sourceComponent.ftype === "simple_crystal") {
|
|
144
|
+
const { crystals } = await getJlcPartsCached("crystals", {
|
|
145
|
+
frequency: sourceComponent.frequency,
|
|
146
|
+
load_capacitance: sourceComponent.load_capacitance,
|
|
147
|
+
package: jlcpcbPackage
|
|
148
|
+
});
|
|
149
|
+
return {
|
|
150
|
+
jlcpcb: withBasicPartPreference(crystals).map((c) => `C${c.lcsc}`).slice(0, 3)
|
|
151
|
+
};
|
|
152
|
+
} else if (sourceComponent.type === "source_component" && sourceComponent.ftype === "simple_mosfet") {
|
|
153
|
+
const { mosfets } = await getJlcPartsCached("mosfets", {
|
|
154
|
+
package: jlcpcbPackage,
|
|
155
|
+
mosfet_mode: sourceComponent.mosfet_mode,
|
|
156
|
+
channel_type: sourceComponent.channel_type
|
|
157
|
+
});
|
|
158
|
+
return {
|
|
159
|
+
jlcpcb: withBasicPartPreference(mosfets).map((m) => `C${m.lcsc}`).slice(0, 3)
|
|
160
|
+
};
|
|
161
|
+
} else if (sourceComponent.type === "source_component" && sourceComponent.ftype === "simple_resonator") {
|
|
162
|
+
const { resonators } = await getJlcPartsCached("resonators", {
|
|
163
|
+
frequency: sourceComponent.frequency,
|
|
164
|
+
package: jlcpcbPackage
|
|
165
|
+
});
|
|
166
|
+
return {
|
|
167
|
+
jlcpcb: withBasicPartPreference(resonators).map((r) => `C${r.lcsc}`).slice(0, 3)
|
|
168
|
+
};
|
|
169
|
+
} else if (sourceComponent.type === "source_component" && sourceComponent.ftype === "simple_switch") {
|
|
170
|
+
const { switches } = await getJlcPartsCached("switches", {
|
|
171
|
+
switch_type: sourceComponent.type,
|
|
172
|
+
package: jlcpcbPackage
|
|
173
|
+
});
|
|
174
|
+
return {
|
|
175
|
+
jlcpcb: withBasicPartPreference(switches).map((s) => `C${s.lcsc}`).slice(0, 3)
|
|
176
|
+
};
|
|
177
|
+
} else if (sourceComponent.type === "source_component" && sourceComponent.ftype === "simple_led") {
|
|
178
|
+
const { leds } = await getJlcPartsCached("leds", {
|
|
179
|
+
package: jlcpcbPackage
|
|
180
|
+
});
|
|
181
|
+
return {
|
|
182
|
+
jlcpcb: withBasicPartPreference(leds).map((l) => `C${l.lcsc}`).slice(0, 3)
|
|
183
|
+
};
|
|
184
|
+
} else if (sourceComponent.type === "source_component" && sourceComponent.ftype === "simple_fuse") {
|
|
185
|
+
const { fuses } = await getJlcPartsCached("fuses", {
|
|
186
|
+
package: jlcpcbPackage
|
|
187
|
+
});
|
|
188
|
+
return {
|
|
189
|
+
jlcpcb: withBasicPartPreference(fuses).map((l) => `C${l.lcsc}`).slice(0, 3)
|
|
190
|
+
};
|
|
191
|
+
}
|
|
192
|
+
return {};
|
|
193
|
+
}
|
|
194
|
+
};
|
|
195
|
+
|
|
196
|
+
// lib/getPlatformConfig.ts
|
|
197
|
+
import { parseKicadModToCircuitJson } from "kicad-component-converter";
|
|
198
|
+
|
|
199
|
+
// lib/utils/dynamically-load-dependency-with-cdn-backup.ts
|
|
200
|
+
var dynamicallyLoadDependencyWithCdnBackup = async (packageName) => {
|
|
201
|
+
try {
|
|
202
|
+
const module = await import(packageName);
|
|
203
|
+
return module.default;
|
|
204
|
+
} catch (e) {
|
|
205
|
+
console.log(`Failed to load ${packageName} locally, trying CDN fallback...`);
|
|
206
|
+
try {
|
|
207
|
+
const res = await fetch(
|
|
208
|
+
`https://cdn.jsdelivr.net/npm/${packageName}/+esm`
|
|
209
|
+
);
|
|
210
|
+
if (!res.ok) {
|
|
211
|
+
throw new Error(
|
|
212
|
+
`Failed to fetch ${packageName} from CDN: ${res.statusText}`
|
|
213
|
+
);
|
|
214
|
+
}
|
|
215
|
+
const code = await res.text();
|
|
216
|
+
const blob = new Blob([code], { type: "application/javascript" });
|
|
217
|
+
const url = URL.createObjectURL(blob);
|
|
218
|
+
try {
|
|
219
|
+
const { default: loadedModule } = await import(url);
|
|
220
|
+
return loadedModule;
|
|
221
|
+
} finally {
|
|
222
|
+
URL.revokeObjectURL(url);
|
|
223
|
+
}
|
|
224
|
+
} catch (cdnError) {
|
|
225
|
+
console.error(`CDN fallback for ${packageName} also failed:`, cdnError);
|
|
226
|
+
throw cdnError;
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
};
|
|
230
|
+
|
|
231
|
+
// lib/getPlatformConfig.ts
|
|
232
|
+
var KICAD_FOOTPRINT_CACHE_URL = "https://kicad-mod-cache.tscircuit.com";
|
|
233
|
+
var ngspiceEngineCache = null;
|
|
234
|
+
var getPlatformConfig = () => ({
|
|
235
|
+
partsEngine: jlcPartsEngine,
|
|
236
|
+
spiceEngineMap: {
|
|
237
|
+
ngspice: {
|
|
238
|
+
simulate: async (spice) => {
|
|
239
|
+
if (!ngspiceEngineCache) {
|
|
240
|
+
const createNgspiceSpiceEngine = await dynamicallyLoadDependencyWithCdnBackup(
|
|
241
|
+
"@tscircuit/ngspice-spice-engine"
|
|
242
|
+
).catch((error) => {
|
|
243
|
+
throw new Error(
|
|
244
|
+
"Could not load ngspice engine from local node_modules or CDN fallback.",
|
|
245
|
+
{ cause: error }
|
|
246
|
+
);
|
|
247
|
+
});
|
|
248
|
+
if (createNgspiceSpiceEngine) {
|
|
249
|
+
ngspiceEngineCache = await createNgspiceSpiceEngine();
|
|
250
|
+
}
|
|
251
|
+
}
|
|
252
|
+
if (!ngspiceEngineCache) {
|
|
253
|
+
throw new Error(
|
|
254
|
+
"Could not load ngspice engine from local node_modules or CDN fallback."
|
|
255
|
+
);
|
|
256
|
+
}
|
|
257
|
+
return ngspiceEngineCache.simulate(spice);
|
|
258
|
+
}
|
|
259
|
+
}
|
|
260
|
+
},
|
|
261
|
+
footprintLibraryMap: {
|
|
262
|
+
kicad: async (footprintName) => {
|
|
263
|
+
const baseUrl = `${KICAD_FOOTPRINT_CACHE_URL}/${footprintName}`;
|
|
264
|
+
const circuitJsonUrl = `${baseUrl}.circuit.json`;
|
|
265
|
+
const res = await fetch(circuitJsonUrl);
|
|
266
|
+
const raw = await res.json();
|
|
267
|
+
const filtered = Array.isArray(raw) ? raw.filter(
|
|
268
|
+
(el) => el?.type === "pcb_silkscreen_text" ? el?.text === "REF**" : true
|
|
269
|
+
) : raw;
|
|
270
|
+
const wrlUrl = `${baseUrl}.wrl`;
|
|
271
|
+
return {
|
|
272
|
+
footprintCircuitJson: filtered,
|
|
273
|
+
cadModel: { wrlUrl, modelUnitToMmScale: 2.54 }
|
|
274
|
+
};
|
|
275
|
+
}
|
|
276
|
+
},
|
|
277
|
+
footprintFileParserMap: {
|
|
278
|
+
kicad_mod: {
|
|
279
|
+
loadFromUrl: async (url) => {
|
|
280
|
+
const kicadContent = await fetch(url).then((res) => res.text());
|
|
281
|
+
const kicadJson = await parseKicadModToCircuitJson(kicadContent);
|
|
282
|
+
return {
|
|
283
|
+
footprintCircuitJson: Array.isArray(kicadJson) ? kicadJson : [kicadJson]
|
|
284
|
+
};
|
|
285
|
+
}
|
|
286
|
+
}
|
|
287
|
+
}
|
|
288
|
+
});
|
|
289
|
+
export {
|
|
290
|
+
getPlatformConfig
|
|
291
|
+
};
|
|
292
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../node_modules/@tscircuit/parts-engine/lib/footprint-translators/get-footprinter-string-from-kicad.ts", "../../node_modules/@tscircuit/parts-engine/lib/footprint-translators/get-jlc-package-from-footprinter-string.ts", "../../node_modules/@tscircuit/parts-engine/lib/footprint-translators/index.ts", "../../node_modules/@tscircuit/parts-engine/lib/jlc-parts-engine.ts", "../../lib/getPlatformConfig.ts", "../../lib/utils/dynamically-load-dependency-with-cdn-backup.ts"],
  "sourcesContent": ["/**\n * Transforms a KiCad footprint string into a generic \"footprinter string\".\n * For now, this is a simplified conversion to a standard package name.\n * e.g. \"kicad:Resistor_SMD:R_0603_1608Metric\" -> \"0603\"\n */\nexport const getFootprinterStringFromKicad = (\n  kicadFootprint: string,\n): string | undefined => {\n  // kicad:Resistor_SMD:R_0603_1608Metric -> 0603\n  let match = kicadFootprint.match(/:[RC]_(\\d{4})_/)\n  if (match) return match[1]\n\n  // kicad:Package_SO:SOIC-8_3.9x4.9mm_P1.27mm -> SOIC-8\n  // kicad:Package_TO_SOT_SMD:SOT-23 -> SOT-23\n  match = kicadFootprint.match(\n    /:(SOIC-\\d+|SOT-\\d+|SOD-\\d+|SSOP-\\d+|TSSOP-\\d+|QFP-\\d+|QFN-\\d+)/,\n  )\n  if (match) return match[1]\n\n  return undefined\n}\n", "/**\n * Transforms a generic \"footprinter string\" into a JLC-compatible package name.\n * e.g. \"cap0603\" -> \"0603\"\n */\nexport const getJlcPackageFromFootprinterString = (\n  footprinterString: string,\n): string => {\n  if (footprinterString.includes(\"cap\")) {\n    return footprinterString.replace(/cap/g, \"\")\n  }\n  return footprinterString\n}\n", "import { getFootprinterStringFromKicad } from \"./get-footprinter-string-from-kicad\"\nimport { getJlcPackageFromFootprinterString } from \"./get-jlc-package-from-footprinter-string\"\n\n/**\n * Get a JLC-compatible package name from a footprint string, which could be\n * a KiCad footprint or a generic \"footprinter string\".\n */\nexport const getJlcpcbPackageName = (\n  footprint: string | undefined,\n): string | undefined => {\n  if (!footprint) return undefined\n\n  if (footprint.startsWith(\"kicad:\")) {\n    const footprinterString = getFootprinterStringFromKicad(footprint)\n    if (footprinterString) {\n      return getJlcPackageFromFootprinterString(footprinterString)\n    }\n\n    // Fallback for un-matched KiCad strings\n    return footprint\n  }\n\n  // Not a KiCad string, assume it's a footprinter string\n  return getJlcPackageFromFootprinterString(footprint)\n}\n", "import type { PartsEngine, SupplierPartNumbers } from \"@tscircuit/props\"\nimport { getJlcpcbPackageName } from \"./footprint-translators/index\"\n\nexport const cache = new Map<string, any>()\n\nconst getJlcPartsCached = async (name: any, params: any) => {\n  const paramString = new URLSearchParams({\n    ...params,\n    json: \"true\",\n  }).toString()\n  if (cache.has(paramString)) {\n    return cache.get(paramString)\n  }\n  const response = await fetch(\n    `https://jlcsearch.tscircuit.com/${name}/list?${paramString}`,\n  )\n  const responseJson = await response.json()\n  cache.set(paramString, responseJson)\n  return responseJson\n}\n\nconst withBasicPartPreference = (parts: any[] | undefined) => {\n  if (!parts) return []\n  return [...parts].sort(\n    (a, b) => Number(b.is_basic ?? false) - Number(a.is_basic ?? false),\n  )\n}\n\nexport const jlcPartsEngine: PartsEngine = {\n  findPart: async ({\n    sourceComponent,\n    footprinterString,\n  }): Promise<SupplierPartNumbers> => {\n    const jlcpcbPackage = getJlcpcbPackageName(footprinterString)\n\n    if (\n      sourceComponent.type === \"source_component\" &&\n      sourceComponent.ftype === \"simple_resistor\"\n    ) {\n      const { resistors } = await getJlcPartsCached(\"resistors\", {\n        resistance: sourceComponent.resistance,\n        package: jlcpcbPackage,\n      })\n\n      return {\n        jlcpcb: withBasicPartPreference(resistors)\n          .map((r: any) => `C${r.lcsc}`)\n          .slice(0, 3),\n      }\n    } else if (\n      sourceComponent.type === \"source_component\" &&\n      sourceComponent.ftype === \"simple_capacitor\"\n    ) {\n      const { capacitors } = await getJlcPartsCached(\"capacitors\", {\n        capacitance: sourceComponent.capacitance,\n        package: jlcpcbPackage,\n      })\n\n      return {\n        jlcpcb: withBasicPartPreference(capacitors)\n          .map((c: any) => `C${c.lcsc}`)\n          .slice(0, 3),\n      }\n    } else if (\n      sourceComponent.type === \"source_component\" &&\n      sourceComponent.ftype === \"simple_pin_header\"\n    ) {\n      let pitch: number | undefined\n      if (footprinterString?.includes(\"_p\")) {\n        pitch = Number(footprinterString.split(\"_p\")[1])\n      }\n      const { headers } = await getJlcPartsCached(\n        \"headers\",\n        pitch\n          ? {\n              pitch: pitch,\n              num_pins: sourceComponent.pin_count,\n              gender: sourceComponent.gender,\n            }\n          : {\n              num_pins: sourceComponent.pin_count,\n              gender: sourceComponent.gender,\n            },\n      )\n      return {\n        jlcpcb: withBasicPartPreference(headers)\n          .map((h: any) => `C${h.lcsc}`)\n          .slice(0, 3),\n      }\n    } else if (\n      sourceComponent.type === \"source_component\" &&\n      sourceComponent.ftype === \"simple_potentiometer\"\n    ) {\n      const { potentiometers } = await getJlcPartsCached(\"potentiometers\", {\n        resistance: sourceComponent.max_resistance,\n        package: jlcpcbPackage,\n      })\n      return {\n        jlcpcb: withBasicPartPreference(potentiometers)\n          .map((p: any) => `C${p.lcsc}`)\n          .slice(0, 3),\n      }\n    } else if (\n      sourceComponent.type === \"source_component\" &&\n      sourceComponent.ftype === \"simple_diode\"\n    ) {\n      const { diodes } = await getJlcPartsCached(\"diodes\", {\n        package: jlcpcbPackage,\n      })\n      return {\n        jlcpcb: withBasicPartPreference(diodes)\n          .map((d: any) => `C${d.lcsc}`)\n          .slice(0, 3),\n      }\n    } else if (\n      sourceComponent.type === \"source_component\" &&\n      sourceComponent.ftype === \"simple_chip\"\n    ) {\n      const { chips } = await getJlcPartsCached(\"chips\", {\n        package: jlcpcbPackage,\n      })\n      return {\n        jlcpcb: withBasicPartPreference(chips)\n          .map((c: any) => `C${c.lcsc}`)\n          .slice(0, 3),\n      }\n    } else if (\n      sourceComponent.type === \"source_component\" &&\n      sourceComponent.ftype === \"simple_transistor\"\n    ) {\n      const { transistors } = await getJlcPartsCached(\"transistors\", {\n        package: jlcpcbPackage,\n        transistor_type: sourceComponent.transistor_type,\n      })\n      return {\n        jlcpcb: withBasicPartPreference(transistors)\n          .map((t: any) => `C${t.lcsc}`)\n          .slice(0, 3),\n      }\n    } else if (\n      sourceComponent.type === \"source_component\" &&\n      sourceComponent.ftype === \"simple_power_source\"\n    ) {\n      const { power_sources } = await getJlcPartsCached(\"power_sources\", {\n        voltage: sourceComponent.voltage,\n        package: jlcpcbPackage,\n      })\n      return {\n        jlcpcb: withBasicPartPreference(power_sources)\n          .map((p: any) => `C${p.lcsc}`)\n          .slice(0, 3),\n      }\n    } else if (\n      sourceComponent.type === \"source_component\" &&\n      sourceComponent.ftype === \"simple_inductor\"\n    ) {\n      const { inductors } = await getJlcPartsCached(\"inductors\", {\n        inductance: sourceComponent.inductance,\n        package: jlcpcbPackage,\n      })\n      return {\n        jlcpcb: withBasicPartPreference(inductors)\n          .map((i: any) => `C${i.lcsc}`)\n          .slice(0, 3),\n      }\n    } else if (\n      sourceComponent.type === \"source_component\" &&\n      sourceComponent.ftype === \"simple_crystal\"\n    ) {\n      const { crystals } = await getJlcPartsCached(\"crystals\", {\n        frequency: sourceComponent.frequency,\n        load_capacitance: sourceComponent.load_capacitance,\n        package: jlcpcbPackage,\n      })\n      return {\n        jlcpcb: withBasicPartPreference(crystals)\n          .map((c: any) => `C${c.lcsc}`)\n          .slice(0, 3),\n      }\n    } else if (\n      sourceComponent.type === \"source_component\" &&\n      sourceComponent.ftype === \"simple_mosfet\"\n    ) {\n      const { mosfets } = await getJlcPartsCached(\"mosfets\", {\n        package: jlcpcbPackage,\n        mosfet_mode: sourceComponent.mosfet_mode,\n        channel_type: sourceComponent.channel_type,\n      })\n      return {\n        jlcpcb: withBasicPartPreference(mosfets)\n          .map((m: any) => `C${m.lcsc}`)\n          .slice(0, 3),\n      }\n    } else if (\n      sourceComponent.type === \"source_component\" &&\n      sourceComponent.ftype === \"simple_resonator\"\n    ) {\n      const { resonators } = await getJlcPartsCached(\"resonators\", {\n        frequency: sourceComponent.frequency,\n        package: jlcpcbPackage,\n      })\n      return {\n        jlcpcb: withBasicPartPreference(resonators)\n          .map((r: any) => `C${r.lcsc}`)\n          .slice(0, 3),\n      }\n    } else if (\n      sourceComponent.type === \"source_component\" &&\n      sourceComponent.ftype === \"simple_switch\"\n    ) {\n      const { switches } = await getJlcPartsCached(\"switches\", {\n        switch_type: sourceComponent.type,\n        package: jlcpcbPackage,\n      })\n      return {\n        jlcpcb: withBasicPartPreference(switches)\n          .map((s: any) => `C${s.lcsc}`)\n          .slice(0, 3),\n      }\n    } else if (\n      sourceComponent.type === \"source_component\" &&\n      sourceComponent.ftype === \"simple_led\"\n    ) {\n      const { leds } = await getJlcPartsCached(\"leds\", {\n        package: jlcpcbPackage,\n      })\n      return {\n        jlcpcb: withBasicPartPreference(leds)\n          .map((l: any) => `C${l.lcsc}`)\n          .slice(0, 3),\n      }\n    } else if (\n      sourceComponent.type === \"source_component\" &&\n      sourceComponent.ftype === \"simple_fuse\"\n    ) {\n      const { fuses } = await getJlcPartsCached(\"fuses\", {\n        package: jlcpcbPackage,\n      })\n      return {\n        jlcpcb: withBasicPartPreference(fuses)\n          .map((l: any) => `C${l.lcsc}`)\n          .slice(0, 3),\n      }\n    }\n    return {}\n  },\n}\n", "import type { PlatformConfig, SpiceEngine } from \"@tscircuit/props\"\nimport { jlcPartsEngine } from \"@tscircuit/parts-engine\"\nimport { parseKicadModToCircuitJson } from \"kicad-component-converter\"\nimport { dynamicallyLoadDependencyWithCdnBackup } from \"./utils/dynamically-load-dependency-with-cdn-backup\"\nconst KICAD_FOOTPRINT_CACHE_URL = \"https://kicad-mod-cache.tscircuit.com\"\n\nlet ngspiceEngineCache: SpiceEngine | null = null\n\nexport const getPlatformConfig = (): PlatformConfig => ({\n  partsEngine: jlcPartsEngine,\n  spiceEngineMap: {\n    ngspice: {\n      simulate: async (spice: string) => {\n        if (!ngspiceEngineCache) {\n          const createNgspiceSpiceEngine =\n            await dynamicallyLoadDependencyWithCdnBackup(\n              \"@tscircuit/ngspice-spice-engine\",\n            ).catch((error) => {\n              throw new Error(\n                \"Could not load ngspice engine from local node_modules or CDN fallback.\",\n                { cause: error },\n              )\n            })\n\n          if (createNgspiceSpiceEngine) {\n            ngspiceEngineCache = await createNgspiceSpiceEngine()\n          }\n        }\n\n        if (!ngspiceEngineCache) {\n          throw new Error(\n            \"Could not load ngspice engine from local node_modules or CDN fallback.\",\n          )\n        }\n\n        return ngspiceEngineCache.simulate(spice)\n      },\n    },\n  },\n  footprintLibraryMap: {\n    kicad: async (footprintName: string) => {\n      const baseUrl = `${KICAD_FOOTPRINT_CACHE_URL}/${footprintName}`\n      const circuitJsonUrl = `${baseUrl}.circuit.json`\n      const res = await fetch(circuitJsonUrl)\n      const raw = await res.json()\n      // Filter pcb_silkscreen_text to only keep entries with text === \"REF**\"\n      // Apply filtering only to elements coming from the kicad_mod_server response\n      const filtered = Array.isArray(raw)\n        ? raw.filter((el) =>\n            el?.type === \"pcb_silkscreen_text\" ? el?.text === \"REF**\" : true,\n          )\n        : raw\n      const wrlUrl = `${baseUrl}.wrl`\n      return {\n        footprintCircuitJson: filtered,\n        cadModel: { wrlUrl, modelUnitToMmScale: 2.54 },\n      }\n    },\n  },\n  footprintFileParserMap: {\n    kicad_mod: {\n      loadFromUrl: async (url: string) => {\n        const kicadContent = await fetch(url).then((res) => res.text())\n        const kicadJson = await parseKicadModToCircuitJson(kicadContent)\n        return {\n          footprintCircuitJson: Array.isArray(kicadJson)\n            ? kicadJson\n            : [kicadJson],\n        }\n      },\n    },\n  },\n})\n", "export const dynamicallyLoadDependencyWithCdnBackup = async (\n  packageName: string,\n): Promise<any> => {\n  try {\n    // First, try to import using Node.js resolution\n    const module = await import(packageName)\n    return module.default\n  } catch (e) {\n    console.log(`Failed to load ${packageName} locally, trying CDN fallback...`)\n    // Fallback to JsDelivr CDN for browser environments\n    try {\n      const res = await fetch(\n        `https://cdn.jsdelivr.net/npm/${packageName}/+esm`,\n      )\n      if (!res.ok) {\n        throw new Error(\n          `Failed to fetch ${packageName} from CDN: ${res.statusText}`,\n        )\n      }\n      const code = await res.text()\n      const blob = new Blob([code], { type: \"application/javascript\" })\n      const url = URL.createObjectURL(blob)\n      try {\n        const { default: loadedModule } = await import(url)\n        return loadedModule\n      } finally {\n        URL.revokeObjectURL(url)\n      }\n    } catch (cdnError) {\n      console.error(`CDN fallback for ${packageName} also failed:`, cdnError)\n      throw cdnError\n    }\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

|