@willcgage/module-schematic 0.1.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.
package/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ MIT License
2
+
3
+ Copyright (c) 2026 Will Gage
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
package/README.md ADDED
@@ -0,0 +1,42 @@
1
+ # @willcgage/module-schematic
2
+
3
+ Shared **FreeMo module operations-schematic** (track-graph) — the single source of
4
+ truth for the structured schematic that the **Module Repository** authors and
5
+ **Free-Dispatcher** imports and renders as a straightened CTC dispatcher panel.
6
+
7
+ Topological and straightened-first: positions are 1-D inches along the module
8
+ (from endplate A), lanes are integer track indices (`0` = primary main).
9
+
10
+ ## What's here
11
+
12
+ - **Doc types** — `ModuleSchematicDoc` and its parts (`SchematicTrack`,
13
+ `SchematicTurnout`, `SchematicSignal`, `SchematicControlPoint`, …).
14
+ - **`asModuleSchematic(x)`** — lenient parser (docs arrive as jsonb / `unknown`).
15
+ - **`moduleFeatures(doc)`** — pure resolver → positioned drawables (fractions of
16
+ the module length) both renderers draw.
17
+ - **N-scale helpers** — `inchesToScaleFeet`, `scaleFeetToInches`, `N_SCALE_RATIO`
18
+ (1:160; 396in = one mile).
19
+ - **Editor state machine** — `emptyEditorState`, `stateToDoc`, `docToState`,
20
+ `buildPassingSiding`, `nextId` — what an authoring UI binds to.
21
+
22
+ Framework-agnostic and side-effect-free: consumable from Next.js (server +
23
+ client) and Electron. Ships ESM + CJS + type declarations.
24
+
25
+ ## Usage
26
+
27
+ ```ts
28
+ import { asModuleSchematic, moduleFeatures } from "@willcgage/module-schematic";
29
+
30
+ const doc = asModuleSchematic(row.schematic);
31
+ if (doc) {
32
+ const { extraTracks, turnouts, signals, doubleMain } = moduleFeatures(doc);
33
+ // …draw them
34
+ }
35
+ ```
36
+
37
+ The wire format is documented in the free-dispatcher repo:
38
+ `docs/module-schematic-format.md`.
39
+
40
+ ## License
41
+
42
+ MIT
package/dist/index.cjs ADDED
@@ -0,0 +1,299 @@
1
+ 'use strict';
2
+
3
+ // src/index.ts
4
+ var MAIN_TRACK_ID = "main";
5
+ var N_SCALE_RATIO = 160;
6
+ function inchesToScaleFeet(inches, ratio = N_SCALE_RATIO) {
7
+ return inches * ratio / 12;
8
+ }
9
+ function scaleFeetToInches(feet, ratio = N_SCALE_RATIO) {
10
+ return feet * 12 / ratio;
11
+ }
12
+ function asModuleSchematic(x) {
13
+ if (!x || typeof x !== "object") return null;
14
+ const d = x;
15
+ if (typeof d.version !== "number") return null;
16
+ if (!Array.isArray(d.endplates) || !Array.isArray(d.tracks)) return null;
17
+ return d;
18
+ }
19
+ function emptyEditorState(lengthInches) {
20
+ return {
21
+ lengthInches: lengthInches > 0 ? lengthInches : 24,
22
+ configA: "single",
23
+ configB: "single",
24
+ extraTracks: [],
25
+ turnouts: [],
26
+ controlPoints: []
27
+ };
28
+ }
29
+ function stateToDoc(state, recordNumber) {
30
+ return {
31
+ version: 1,
32
+ module: recordNumber,
33
+ lengthInches: state.lengthInches,
34
+ endplates: [
35
+ { id: "A", label: "West", tracks: [{ trackId: MAIN_TRACK_ID, lane: 0, config: state.configA }] },
36
+ { id: "B", label: "East", tracks: [{ trackId: MAIN_TRACK_ID, lane: 0, config: state.configB }] }
37
+ ],
38
+ tracks: [
39
+ { id: MAIN_TRACK_ID, role: "main", lane: 0, from: "A", to: "B" },
40
+ ...state.extraTracks.map((t) => ({
41
+ id: t.id,
42
+ role: t.role,
43
+ lane: t.lane,
44
+ fromPos: t.fromPos,
45
+ toPos: t.toPos,
46
+ moduleTrackId: t.moduleTrackId,
47
+ trackName: t.trackName || void 0,
48
+ capacityFeet: Math.round(inchesToScaleFeet(Math.abs(t.toPos - t.fromPos)))
49
+ }))
50
+ ],
51
+ turnouts: state.turnouts.map((t) => ({
52
+ id: t.id,
53
+ pos: t.pos,
54
+ onTrack: t.onTrack,
55
+ divergeTrack: t.divergeTrack,
56
+ kind: t.kind,
57
+ name: t.name || void 0
58
+ })),
59
+ controlPoints: state.controlPoints.map((c) => ({
60
+ id: c.id,
61
+ name: c.name,
62
+ turnouts: c.turnouts,
63
+ signals: c.signals.map((s) => ({
64
+ id: s.id,
65
+ pos: s.pos,
66
+ track: s.track,
67
+ facing: s.facing,
68
+ kind: "mast",
69
+ side: s.side
70
+ }))
71
+ }))
72
+ };
73
+ }
74
+ function docToState(doc, fallbackLength, moduleTracks = []) {
75
+ const base = emptyEditorState(fallbackLength);
76
+ const d = doc && typeof doc === "object" ? doc : null;
77
+ const hasDoc = !!d && typeof d.lengthInches === "number" && Array.isArray(d.tracks);
78
+ const len = fallbackLength > 0 ? fallbackLength : hasDoc ? d.lengthInches : 24;
79
+ const docLen = hasDoc && d.lengthInches > 0 ? d.lengthInches : len;
80
+ const scale = docLen > 0 ? len / docLen : 1;
81
+ const sc = (p) => Math.round(p * scale);
82
+ const nameOf = (id) => {
83
+ const mt = id != null ? moduleTracks.find((m) => m.id === id) : void 0;
84
+ return mt?.track_name ?? "";
85
+ };
86
+ const extraTracks = [];
87
+ const usedMt = /* @__PURE__ */ new Set();
88
+ if (hasDoc) {
89
+ for (const t of d.tracks) {
90
+ if (t.role === "main") continue;
91
+ const moduleTrackId = t.moduleTrackId ?? null;
92
+ if (moduleTrackId != null) usedMt.add(moduleTrackId);
93
+ extraTracks.push({
94
+ id: t.id,
95
+ role: t.role ?? "siding",
96
+ lane: t.lane ?? 1,
97
+ fromPos: sc(t.fromPos ?? 0),
98
+ toPos: t.toPos != null ? sc(t.toPos) : len,
99
+ moduleTrackId,
100
+ trackName: t.trackName ?? nameOf(moduleTrackId)
101
+ });
102
+ }
103
+ }
104
+ const unused = moduleTracks.filter((mt) => !usedMt.has(mt.id));
105
+ let ui = 0;
106
+ for (const et of extraTracks) {
107
+ if (et.moduleTrackId == null && ui < unused.length) {
108
+ const mt = unused[ui++];
109
+ et.moduleTrackId = mt.id;
110
+ if (!et.trackName) et.trackName = mt.track_name ?? "";
111
+ usedMt.add(mt.id);
112
+ }
113
+ }
114
+ let lane = Math.max(0, ...extraTracks.map((t) => t.lane));
115
+ for (const mt of moduleTracks) {
116
+ if (usedMt.has(mt.id)) continue;
117
+ lane += 1;
118
+ extraTracks.push({
119
+ id: `mt${mt.id}`,
120
+ role: "siding",
121
+ lane,
122
+ fromPos: Math.round(len * 0.2),
123
+ toPos: Math.round(len * 0.8),
124
+ moduleTrackId: mt.id,
125
+ trackName: mt.track_name ?? ""
126
+ });
127
+ }
128
+ if (!hasDoc) return { ...base, lengthInches: len, extraTracks };
129
+ const configOf = (id) => {
130
+ const ep = (d.endplates ?? []).find((e) => e.id === id);
131
+ return ep?.tracks?.[0]?.config === "double" ? "double" : "single";
132
+ };
133
+ return {
134
+ lengthInches: len,
135
+ configA: configOf("A"),
136
+ configB: configOf("B"),
137
+ extraTracks,
138
+ turnouts: (d.turnouts ?? []).map((t) => ({
139
+ id: t.id,
140
+ name: t.name ?? "",
141
+ pos: sc(t.pos),
142
+ onTrack: t.onTrack,
143
+ divergeTrack: t.divergeTrack,
144
+ kind: t.kind ?? "right"
145
+ })),
146
+ controlPoints: readControlPoints(d, sc)
147
+ };
148
+ }
149
+ function readControlPoints(d, sc = (p) => p) {
150
+ if (Array.isArray(d.controlPoints)) {
151
+ return d.controlPoints.map((c) => ({
152
+ id: c.id,
153
+ name: c.name ?? "",
154
+ turnouts: c.turnouts ?? [],
155
+ signals: (c.signals ?? []).map((s) => ({
156
+ id: s.id,
157
+ pos: sc(s.pos),
158
+ track: s.track ?? MAIN_TRACK_ID,
159
+ facing: s.facing ?? "AtoB",
160
+ side: s.side ?? "above"
161
+ }))
162
+ }));
163
+ }
164
+ const groups = /* @__PURE__ */ new Map();
165
+ let n = 0;
166
+ for (const s of d.signals ?? []) {
167
+ const key = s.turnout || `blk-${s.id}`;
168
+ let cp = groups.get(key);
169
+ if (!cp) {
170
+ cp = { id: `cp${++n}`, name: s.name ?? "", turnouts: s.turnout ? [s.turnout] : [], signals: [] };
171
+ groups.set(key, cp);
172
+ }
173
+ cp.signals.push({
174
+ id: s.id,
175
+ pos: sc(s.pos),
176
+ track: s.track ?? MAIN_TRACK_ID,
177
+ facing: s.facing ?? "AtoB",
178
+ side: s.side ?? "above"
179
+ });
180
+ }
181
+ return [...groups.values()];
182
+ }
183
+ function nextId(prefix, existing) {
184
+ let n = 1;
185
+ while (existing.includes(`${prefix}${n}`)) n += 1;
186
+ return `${prefix}${n}`;
187
+ }
188
+ function buildPassingSiding(state) {
189
+ const len = state.lengthInches > 0 ? state.lengthInches : 24;
190
+ const inset = Math.max(6, Math.round(len * 0.08));
191
+ const fromPos = inset;
192
+ const toPos = Math.max(fromPos + 1, len - inset);
193
+ const lane = Math.max(1, ...state.extraTracks.map((t) => t.lane + 1));
194
+ const trackIds = [MAIN_TRACK_ID, ...state.extraTracks.map((t) => t.id)];
195
+ const sidId = nextId("sid", trackIds);
196
+ const track = {
197
+ id: sidId,
198
+ role: "siding",
199
+ lane,
200
+ fromPos,
201
+ toPos,
202
+ moduleTrackId: null,
203
+ trackName: "Passing siding"
204
+ };
205
+ const swIds = state.turnouts.map((t) => t.id);
206
+ const swW = nextId("sw", swIds);
207
+ const swE = nextId("sw", [...swIds, swW]);
208
+ const turnouts = [
209
+ { id: swW, name: "West Siding", pos: fromPos, onTrack: MAIN_TRACK_ID, divergeTrack: sidId, kind: "right" },
210
+ { id: swE, name: "East Siding", pos: toPos, onTrack: MAIN_TRACK_ID, divergeTrack: sidId, kind: "left" }
211
+ ];
212
+ const cpIds = state.controlPoints.map((c) => c.id);
213
+ const cpW = nextId("cp", cpIds);
214
+ const cpE = nextId("cp", [...cpIds, cpW]);
215
+ const sig = (cpId, pos, facing) => ({
216
+ id: `${cpId}-${facing}`,
217
+ pos,
218
+ track: MAIN_TRACK_ID,
219
+ facing,
220
+ // opposite directions on opposite sides so they never overlap
221
+ side: facing === "AtoB" ? "above" : "below"
222
+ });
223
+ const controlPoints = [
224
+ { id: cpW, name: "West Siding", turnouts: [swW], signals: [sig(cpW, fromPos, "AtoB"), sig(cpW, fromPos, "BtoA")] },
225
+ { id: cpE, name: "East Siding", turnouts: [swE], signals: [sig(cpE, toPos, "AtoB"), sig(cpE, toPos, "BtoA")] }
226
+ ];
227
+ return { track, turnouts, controlPoints };
228
+ }
229
+ function moduleFeatures(doc) {
230
+ const len = doc.lengthInches && doc.lengthInches > 0 ? doc.lengthInches : Math.max(
231
+ 1,
232
+ ...doc.tracks.map((t) => Math.max(t.fromPos ?? 0, t.toPos ?? 0)),
233
+ ...(doc.turnouts ?? []).map((t) => t.pos)
234
+ );
235
+ const clampFrac = (p) => Math.min(1, Math.max(0, p / len));
236
+ const trackLane = /* @__PURE__ */ new Map();
237
+ for (const t of doc.tracks) trackLane.set(t.id, t.lane);
238
+ const endplatePos = /* @__PURE__ */ new Map();
239
+ doc.endplates.forEach((e, i) => endplatePos.set(e.id, i === 0 ? 0 : len));
240
+ const turnoutPos = /* @__PURE__ */ new Map();
241
+ for (const t of doc.turnouts ?? []) turnoutPos.set(t.id, t.pos);
242
+ const posOf = (nodeId) => {
243
+ if (nodeId == null) return null;
244
+ if (endplatePos.has(nodeId)) return endplatePos.get(nodeId);
245
+ if (turnoutPos.has(nodeId)) return turnoutPos.get(nodeId);
246
+ return null;
247
+ };
248
+ const doubleMain = doc.endplates.some(
249
+ (e) => e.tracks?.some((t) => t.config === "double")
250
+ );
251
+ const extraTracks = [];
252
+ for (const t of doc.tracks) {
253
+ if (t.role === "main") continue;
254
+ const from = t.fromPos ?? posOf(t.from);
255
+ const to = t.toPos ?? posOf(t.to);
256
+ if (from == null || to == null) continue;
257
+ extraTracks.push({
258
+ id: t.id,
259
+ role: t.role,
260
+ lane: t.lane,
261
+ fromFrac: clampFrac(Math.min(from, to)),
262
+ toFrac: clampFrac(Math.max(from, to)),
263
+ capacityFeet: t.capacityFeet ?? null
264
+ });
265
+ }
266
+ const turnouts = (doc.turnouts ?? []).map((t) => ({
267
+ id: t.id,
268
+ name: t.name ?? "",
269
+ posFrac: clampFrac(t.pos),
270
+ onLane: trackLane.get(t.onTrack) ?? 0,
271
+ divergeLane: trackLane.get(t.divergeTrack) ?? 1
272
+ }));
273
+ const drawSignal = (s, name) => ({
274
+ id: s.id,
275
+ name,
276
+ posFrac: clampFrac(s.pos),
277
+ lane: s.track ? trackLane.get(s.track) ?? 0 : 0,
278
+ facing: s.facing ?? "AtoB",
279
+ side: s.side === "below" ? "below" : "above"
280
+ });
281
+ const signals = Array.isArray(doc.controlPoints) ? doc.controlPoints.flatMap(
282
+ (c) => (c.signals ?? []).map((s) => drawSignal(s, c.name ?? ""))
283
+ ) : (doc.signals ?? []).map((s) => drawSignal(s, s.name ?? ""));
284
+ return { doubleMain, extraTracks, turnouts, signals };
285
+ }
286
+
287
+ exports.MAIN_TRACK_ID = MAIN_TRACK_ID;
288
+ exports.N_SCALE_RATIO = N_SCALE_RATIO;
289
+ exports.asModuleSchematic = asModuleSchematic;
290
+ exports.buildPassingSiding = buildPassingSiding;
291
+ exports.docToState = docToState;
292
+ exports.emptyEditorState = emptyEditorState;
293
+ exports.inchesToScaleFeet = inchesToScaleFeet;
294
+ exports.moduleFeatures = moduleFeatures;
295
+ exports.nextId = nextId;
296
+ exports.scaleFeetToInches = scaleFeetToInches;
297
+ exports.stateToDoc = stateToDoc;
298
+ //# sourceMappingURL=index.cjs.map
299
+ //# sourceMappingURL=index.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/index.ts"],"names":[],"mappings":"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","file":"index.cjs","sourcesContent":["/**\n * @willcgage/module-schematic — the shared module operations-schematic (track-graph)\n * that the Module Repository authors and Free-Dispatcher imports.\n *\n * Topological, straightened-first: positions are 1-D inches along the module\n * (from endplate A), lanes are integer track indices (0 = primary main). This is\n * the single source of truth for both apps — the doc types, the lenient parser\n * (docs arrive as jsonb / unknown), the pure feature resolver both renderers\n * draw, the N-scale helpers, and the editor <-> doc state machine an authoring\n * UI binds to. See docs/module-schematic-format.md in the free-dispatcher repo.\n *\n * Framework-agnostic and side-effect-free so it can be unit-tested and consumed\n * from Next.js (server + client) and Electron alike.\n */\n\nexport type TrackConfig = \"single\" | \"double\";\nexport type TrackRole = \"main\" | \"siding\" | \"spur\" | \"yard\" | \"crossover\";\nexport type TurnoutKind = \"left\" | \"right\" | \"wye\";\nexport type SignalFacing = \"AtoB\" | \"BtoA\";\nexport type SignalSide = \"above\" | \"below\";\n\nexport interface SchematicEndplateTrack {\n trackId: string;\n lane: number;\n config?: TrackConfig | null;\n}\nexport interface SchematicEndplate {\n id: string; // \"A\" (West) | \"B\" (East)\n label?: string | null;\n tracks?: SchematicEndplateTrack[];\n}\nexport interface SchematicTrack {\n id: string;\n role: TrackRole;\n lane: number;\n from?: string;\n to?: string;\n fromPos?: number | null;\n toPos?: number | null;\n capacityFeet?: number | null;\n industryRef?: number | null;\n /** The module_tracks row this track is (single source of truth); null = new. */\n moduleTrackId?: number | null;\n /** Owner's track name, mirrored to module_tracks.track_name. */\n trackName?: string;\n}\nexport interface SchematicTurnout {\n id: string;\n pos: number;\n onTrack: string;\n divergeTrack: string;\n kind?: TurnoutKind;\n name?: string | null;\n address?: string | null;\n}\nexport interface SchematicSignal {\n id: string;\n pos: number;\n /** Track the signal governs; absent = the primary main (lane 0). */\n track?: string;\n facing?: SignalFacing;\n kind?: \"mast\" | \"dwarf\";\n name?: string | null;\n aspects?: string[];\n /** Which side of the track the signal sits on (#122). */\n side?: SignalSide;\n /** Turnout this control point governs; absent = standalone block signal. */\n turnout?: string;\n}\nexport interface SchematicBlock {\n id: string;\n name: string;\n tracks?: string[];\n from: number;\n to: number;\n}\n/**\n * A control point is an interlocking: a named group of one or more signals and\n * zero or more turnouts. A passing siding has two (West/East); a lone block\n * signal is a control point with one signal and no turnouts.\n */\nexport interface SchematicControlPoint {\n id: string;\n name?: string | null;\n turnouts?: string[];\n signals?: SchematicSignal[];\n}\nexport interface ModuleSchematicDoc {\n version: number;\n module?: string;\n lengthInches?: number;\n endplates: SchematicEndplate[];\n tracks: SchematicTrack[];\n turnouts?: SchematicTurnout[];\n controlPoints?: SchematicControlPoint[];\n /** @deprecated pre-grouping flat signals; read for back-compat. */\n signals?: SchematicSignal[];\n}\n\nexport const MAIN_TRACK_ID = \"main\";\n\n// North American N scale (1:160): 396 real inches → 5280 scale feet = one mile.\nexport const N_SCALE_RATIO = 160;\n/** Real inches on the module → scale feet of prototype track represented. */\nexport function inchesToScaleFeet(inches: number, ratio = N_SCALE_RATIO): number {\n return (inches * ratio) / 12;\n}\n/** Scale feet of prototype track → real inches on the module. */\nexport function scaleFeetToInches(feet: number, ratio = N_SCALE_RATIO): number {\n return (feet * 12) / ratio;\n}\n\n/** Parse a jsonb value into a schematic doc, or null if it isn't one. */\nexport function asModuleSchematic(x: unknown): ModuleSchematicDoc | null {\n if (!x || typeof x !== \"object\") return null;\n const d = x as Record<string, unknown>;\n if (typeof d.version !== \"number\") return null;\n if (!Array.isArray(d.endplates) || !Array.isArray(d.tracks)) return null;\n return d as unknown as ModuleSchematicDoc;\n}\n\n// ---- Editor state (a flatter shape an authoring form binds to) -------------\n\nexport interface EditorTrack {\n id: string;\n role: TrackRole;\n lane: number;\n fromPos: number;\n toPos: number;\n /** module_tracks row id (single source of truth), or null for a new track. */\n moduleTrackId: number | null;\n /** Owner's track name → module_tracks.track_name. */\n trackName: string;\n}\n\n/** A module_tracks row as loaded for the editor. */\nexport interface ModuleTrackRow {\n id: number;\n track_name: string | null;\n capacity_scale_feet: number | null;\n}\nexport interface EditorTurnout {\n id: string;\n name: string;\n pos: number;\n onTrack: string;\n divergeTrack: string;\n kind: TurnoutKind;\n}\nexport interface EditorCpSignal {\n id: string;\n pos: number;\n track: string;\n facing: SignalFacing;\n side: SignalSide;\n}\nexport interface EditorControlPoint {\n id: string;\n name: string;\n turnouts: string[]; // turnout ids grouped under this control point\n signals: EditorCpSignal[];\n}\nexport interface EditorState {\n lengthInches: number;\n configA: TrackConfig;\n configB: TrackConfig;\n extraTracks: EditorTrack[]; // sidings/spurs/…; the main track is implicit\n turnouts: EditorTurnout[];\n controlPoints: EditorControlPoint[];\n}\n\n/** Build the empty editor state for a module of the given length. */\nexport function emptyEditorState(lengthInches: number): EditorState {\n return {\n lengthInches: lengthInches > 0 ? lengthInches : 24,\n configA: \"single\",\n configB: \"single\",\n extraTracks: [],\n turnouts: [],\n controlPoints: [],\n };\n}\n\n/** Assemble a spec-conformant doc from the editor state. */\nexport function stateToDoc(\n state: EditorState,\n recordNumber: string,\n): ModuleSchematicDoc {\n return {\n version: 1,\n module: recordNumber,\n lengthInches: state.lengthInches,\n endplates: [\n { id: \"A\", label: \"West\", tracks: [{ trackId: MAIN_TRACK_ID, lane: 0, config: state.configA }] },\n { id: \"B\", label: \"East\", tracks: [{ trackId: MAIN_TRACK_ID, lane: 0, config: state.configB }] },\n ],\n tracks: [\n { id: MAIN_TRACK_ID, role: \"main\", lane: 0, from: \"A\", to: \"B\" },\n ...state.extraTracks.map((t) => ({\n id: t.id,\n role: t.role,\n lane: t.lane,\n fromPos: t.fromPos,\n toPos: t.toPos,\n moduleTrackId: t.moduleTrackId,\n trackName: t.trackName || undefined,\n capacityFeet: Math.round(inchesToScaleFeet(Math.abs(t.toPos - t.fromPos))),\n })),\n ],\n turnouts: state.turnouts.map((t) => ({\n id: t.id,\n pos: t.pos,\n onTrack: t.onTrack,\n divergeTrack: t.divergeTrack,\n kind: t.kind,\n name: t.name || undefined,\n })),\n controlPoints: state.controlPoints.map((c) => ({\n id: c.id,\n name: c.name,\n turnouts: c.turnouts,\n signals: c.signals.map((s) => ({\n id: s.id,\n pos: s.pos,\n track: s.track,\n facing: s.facing,\n kind: \"mast\" as const,\n side: s.side,\n })),\n })),\n };\n}\n\n/**\n * Derive editor state from the doc and the module's Track section rows. Tracks\n * are the single source of truth for name/capacity (module_tracks), while the\n * schematic doc adds geometry (lane, positions). We merge: doc tracks first\n * (they carry geometry + their moduleTrackId link), then any module_tracks not\n * yet positioned in the schematic.\n */\nexport function docToState(\n doc: unknown,\n fallbackLength: number,\n moduleTracks: ModuleTrackRow[] = [],\n): EditorState {\n const base = emptyEditorState(fallbackLength);\n const d =\n doc && typeof doc === \"object\" ? (doc as ModuleSchematicDoc) : null;\n const hasDoc = !!d && typeof d.lengthInches === \"number\" && Array.isArray(d.tracks);\n // The module's length is authoritative (the mainline is the module). If the\n // saved doc used a different length, rescale its feature positions to fit so\n // the mainline always reads as the module's true length.\n const len = fallbackLength > 0 ? fallbackLength : hasDoc ? d!.lengthInches! : 24;\n const docLen = hasDoc && d!.lengthInches! > 0 ? d!.lengthInches! : len;\n const scale = docLen > 0 ? len / docLen : 1;\n const sc = (p: number) => Math.round(p * scale);\n\n const nameOf = (id: number | null | undefined): string => {\n const mt = id != null ? moduleTracks.find((m) => m.id === id) : undefined;\n return mt?.track_name ?? \"\";\n };\n\n const extraTracks: EditorTrack[] = [];\n const usedMt = new Set<number>();\n if (hasDoc) {\n for (const t of d!.tracks) {\n if (t.role === \"main\") continue;\n const moduleTrackId = t.moduleTrackId ?? null;\n if (moduleTrackId != null) usedMt.add(moduleTrackId);\n extraTracks.push({\n id: t.id,\n role: (t.role as TrackRole) ?? \"siding\",\n lane: t.lane ?? 1,\n fromPos: sc(t.fromPos ?? 0),\n toPos: t.toPos != null ? sc(t.toPos) : len,\n moduleTrackId,\n trackName: t.trackName ?? nameOf(moduleTrackId),\n });\n }\n }\n // Link pre-migration doc tracks (no moduleTrackId yet) to unused module_tracks\n // by order — keeping the doc track's id so turnout/signal references stay\n // valid. Only after that do leftover module_tracks become new tracks.\n const unused = moduleTracks.filter((mt) => !usedMt.has(mt.id));\n let ui = 0;\n for (const et of extraTracks) {\n if (et.moduleTrackId == null && ui < unused.length) {\n const mt = unused[ui++];\n et.moduleTrackId = mt.id;\n if (!et.trackName) et.trackName = mt.track_name ?? \"\";\n usedMt.add(mt.id);\n }\n }\n let lane = Math.max(0, ...extraTracks.map((t) => t.lane));\n for (const mt of moduleTracks) {\n if (usedMt.has(mt.id)) continue;\n lane += 1;\n extraTracks.push({\n id: `mt${mt.id}`,\n role: \"siding\",\n lane,\n fromPos: Math.round(len * 0.2),\n toPos: Math.round(len * 0.8),\n moduleTrackId: mt.id,\n trackName: mt.track_name ?? \"\",\n });\n }\n\n if (!hasDoc) return { ...base, lengthInches: len, extraTracks };\n\n const configOf = (id: string): TrackConfig => {\n const ep = (d!.endplates ?? []).find((e) => e.id === id);\n return ep?.tracks?.[0]?.config === \"double\" ? \"double\" : \"single\";\n };\n return {\n lengthInches: len,\n configA: configOf(\"A\"),\n configB: configOf(\"B\"),\n extraTracks,\n turnouts: (d!.turnouts ?? []).map((t) => ({\n id: t.id,\n name: t.name ?? \"\",\n pos: sc(t.pos),\n onTrack: t.onTrack,\n divergeTrack: t.divergeTrack,\n kind: (t.kind as TurnoutKind) ?? \"right\",\n })),\n controlPoints: readControlPoints(d!, sc),\n };\n}\n\n/** Control points from a doc, migrating pre-grouping flat signals into groups. */\nfunction readControlPoints(\n d: ModuleSchematicDoc,\n sc: (p: number) => number = (p) => p,\n): EditorControlPoint[] {\n if (Array.isArray(d.controlPoints)) {\n return d.controlPoints.map((c) => ({\n id: c.id,\n name: c.name ?? \"\",\n turnouts: c.turnouts ?? [],\n signals: (c.signals ?? []).map((s) => ({\n id: s.id,\n pos: sc(s.pos),\n track: s.track ?? MAIN_TRACK_ID,\n facing: (s.facing as SignalFacing) ?? \"AtoB\",\n side: (s.side as SignalSide) ?? \"above\",\n })),\n }));\n }\n // Back-compat: group old flat signals by their turnout (or standalone).\n const groups = new Map<string, EditorControlPoint>();\n let n = 0;\n for (const s of d.signals ?? []) {\n const key = s.turnout || `blk-${s.id}`;\n let cp = groups.get(key);\n if (!cp) {\n cp = { id: `cp${++n}`, name: s.name ?? \"\", turnouts: s.turnout ? [s.turnout] : [], signals: [] };\n groups.set(key, cp);\n }\n cp.signals.push({\n id: s.id,\n pos: sc(s.pos),\n track: s.track ?? MAIN_TRACK_ID,\n facing: (s.facing as SignalFacing) ?? \"AtoB\",\n side: (s.side as SignalSide) ?? \"above\",\n });\n }\n return [...groups.values()];\n}\n\n/** Find an unused `${prefix}${n}` id given the ones already present. */\nexport function nextId(prefix: string, existing: string[]): string {\n let n = 1;\n while (existing.includes(`${prefix}${n}`)) n += 1;\n return `${prefix}${n}`;\n}\n\n/**\n * Build a passing siding as one unit: the siding track, a switch at each end,\n * and control-point signals for both directions at each end (prototype Station\n * Entering Signal). Returns the new items to merge into the editor state.\n */\nexport function buildPassingSiding(state: EditorState): {\n track: EditorTrack;\n turnouts: EditorTurnout[];\n controlPoints: EditorControlPoint[];\n} {\n const len = state.lengthInches > 0 ? state.lengthInches : 24;\n const inset = Math.max(6, Math.round(len * 0.08));\n const fromPos = inset;\n const toPos = Math.max(fromPos + 1, len - inset);\n const lane = Math.max(1, ...state.extraTracks.map((t) => t.lane + 1));\n\n const trackIds = [MAIN_TRACK_ID, ...state.extraTracks.map((t) => t.id)];\n const sidId = nextId(\"sid\", trackIds);\n const track: EditorTrack = {\n id: sidId,\n role: \"siding\",\n lane,\n fromPos,\n toPos,\n moduleTrackId: null,\n trackName: \"Passing siding\",\n };\n\n const swIds = state.turnouts.map((t) => t.id);\n const swW = nextId(\"sw\", swIds);\n const swE = nextId(\"sw\", [...swIds, swW]);\n const turnouts: EditorTurnout[] = [\n { id: swW, name: \"West Siding\", pos: fromPos, onTrack: MAIN_TRACK_ID, divergeTrack: sidId, kind: \"right\" },\n { id: swE, name: \"East Siding\", pos: toPos, onTrack: MAIN_TRACK_ID, divergeTrack: sidId, kind: \"left\" },\n ];\n\n // One control point at each end, each grouping its switch and both-direction\n // signals on the main (prototype Station Entering Signal).\n const cpIds = state.controlPoints.map((c) => c.id);\n const cpW = nextId(\"cp\", cpIds);\n const cpE = nextId(\"cp\", [...cpIds, cpW]);\n const sig = (cpId: string, pos: number, facing: SignalFacing): EditorCpSignal => ({\n id: `${cpId}-${facing}`,\n pos,\n track: MAIN_TRACK_ID,\n facing,\n // opposite directions on opposite sides so they never overlap\n side: facing === \"AtoB\" ? \"above\" : \"below\",\n });\n const controlPoints: EditorControlPoint[] = [\n { id: cpW, name: \"West Siding\", turnouts: [swW], signals: [sig(cpW, fromPos, \"AtoB\"), sig(cpW, fromPos, \"BtoA\")] },\n { id: cpE, name: \"East Siding\", turnouts: [swE], signals: [sig(cpE, toPos, \"AtoB\"), sig(cpE, toPos, \"BtoA\")] },\n ];\n\n return { track, turnouts, controlPoints };\n}\n\n// ---- Pure feature resolver (both renderers draw these) --------------------\n\nexport interface DrawTrack {\n id: string;\n role: TrackRole;\n lane: number;\n fromFrac: number;\n toFrac: number;\n capacityFeet: number | null;\n}\nexport interface DrawTurnout {\n id: string;\n name: string;\n posFrac: number;\n onLane: number;\n divergeLane: number;\n}\nexport interface DrawSignal {\n id: string;\n name: string;\n posFrac: number;\n lane: number;\n facing: SignalFacing;\n side: SignalSide;\n}\nexport interface ModuleFeatures {\n /** Whether either endplate declares a double-track main. */\n doubleMain: boolean;\n /** Non-main tracks (sidings/spurs/yard/crossover). */\n extraTracks: DrawTrack[];\n turnouts: DrawTurnout[];\n signals: DrawSignal[];\n}\n\n/**\n * Resolve a schematic doc into positioned drawables. `pos` (inches) becomes a\n * fraction of the module length; endplate A = 0, B = length; turnouts sit at\n * their pos. Tracks may carry explicit fromPos/toPos (overriding node lookup).\n * To-scale: a feature renders at its true position, clamped only to the\n * module's own extent — so signals near an end read at their real spot, not\n * bunched at an inset (#122).\n */\nexport function moduleFeatures(doc: ModuleSchematicDoc): ModuleFeatures {\n const len =\n doc.lengthInches && doc.lengthInches > 0\n ? doc.lengthInches\n : Math.max(\n 1,\n ...doc.tracks.map((t) => Math.max(t.fromPos ?? 0, t.toPos ?? 0)),\n ...(doc.turnouts ?? []).map((t) => t.pos),\n );\n const clampFrac = (p: number) => Math.min(1, Math.max(0, p / len));\n\n const trackLane = new Map<string, number>();\n for (const t of doc.tracks) trackLane.set(t.id, t.lane);\n\n // Endplate positions: first endplate = West (0), the rest = East (len).\n const endplatePos = new Map<string, number>();\n doc.endplates.forEach((e, i) => endplatePos.set(e.id, i === 0 ? 0 : len));\n const turnoutPos = new Map<string, number>();\n for (const t of doc.turnouts ?? []) turnoutPos.set(t.id, t.pos);\n const posOf = (nodeId?: string): number | null => {\n if (nodeId == null) return null;\n if (endplatePos.has(nodeId)) return endplatePos.get(nodeId)!;\n if (turnoutPos.has(nodeId)) return turnoutPos.get(nodeId)!;\n return null;\n };\n\n const doubleMain = doc.endplates.some((e) =>\n e.tracks?.some((t) => t.config === \"double\"),\n );\n\n const extraTracks: DrawTrack[] = [];\n for (const t of doc.tracks) {\n if (t.role === \"main\") continue; // the spine draws mains\n const from = t.fromPos ?? posOf(t.from);\n const to = t.toPos ?? posOf(t.to);\n if (from == null || to == null) continue; // can't place it\n extraTracks.push({\n id: t.id,\n role: t.role,\n lane: t.lane,\n fromFrac: clampFrac(Math.min(from, to)),\n toFrac: clampFrac(Math.max(from, to)),\n capacityFeet: t.capacityFeet ?? null,\n });\n }\n\n const turnouts: DrawTurnout[] = (doc.turnouts ?? []).map((t) => ({\n id: t.id,\n name: t.name ?? \"\",\n posFrac: clampFrac(t.pos),\n onLane: trackLane.get(t.onTrack) ?? 0,\n divergeLane: trackLane.get(t.divergeTrack) ?? 1,\n }));\n\n const drawSignal = (s: SchematicSignal, name: string): DrawSignal => ({\n id: s.id,\n name,\n posFrac: clampFrac(s.pos),\n lane: s.track ? (trackLane.get(s.track) ?? 0) : 0,\n facing: (s.facing as SignalFacing) ?? \"AtoB\",\n side: s.side === \"below\" ? \"below\" : \"above\",\n });\n // Signals come from control-point groups; fall back to pre-grouping flat\n // signals for docs authored before the model changed.\n const signals: DrawSignal[] = Array.isArray(doc.controlPoints)\n ? doc.controlPoints.flatMap((c) =>\n (c.signals ?? []).map((s) => drawSignal(s, c.name ?? \"\")),\n )\n : (doc.signals ?? []).map((s) => drawSignal(s, s.name ?? \"\"));\n\n return { doubleMain, extraTracks, turnouts, signals };\n}\n"]}
@@ -0,0 +1,216 @@
1
+ /**
2
+ * @willcgage/module-schematic — the shared module operations-schematic (track-graph)
3
+ * that the Module Repository authors and Free-Dispatcher imports.
4
+ *
5
+ * Topological, straightened-first: positions are 1-D inches along the module
6
+ * (from endplate A), lanes are integer track indices (0 = primary main). This is
7
+ * the single source of truth for both apps — the doc types, the lenient parser
8
+ * (docs arrive as jsonb / unknown), the pure feature resolver both renderers
9
+ * draw, the N-scale helpers, and the editor <-> doc state machine an authoring
10
+ * UI binds to. See docs/module-schematic-format.md in the free-dispatcher repo.
11
+ *
12
+ * Framework-agnostic and side-effect-free so it can be unit-tested and consumed
13
+ * from Next.js (server + client) and Electron alike.
14
+ */
15
+ type TrackConfig = "single" | "double";
16
+ type TrackRole = "main" | "siding" | "spur" | "yard" | "crossover";
17
+ type TurnoutKind = "left" | "right" | "wye";
18
+ type SignalFacing = "AtoB" | "BtoA";
19
+ type SignalSide = "above" | "below";
20
+ interface SchematicEndplateTrack {
21
+ trackId: string;
22
+ lane: number;
23
+ config?: TrackConfig | null;
24
+ }
25
+ interface SchematicEndplate {
26
+ id: string;
27
+ label?: string | null;
28
+ tracks?: SchematicEndplateTrack[];
29
+ }
30
+ interface SchematicTrack {
31
+ id: string;
32
+ role: TrackRole;
33
+ lane: number;
34
+ from?: string;
35
+ to?: string;
36
+ fromPos?: number | null;
37
+ toPos?: number | null;
38
+ capacityFeet?: number | null;
39
+ industryRef?: number | null;
40
+ /** The module_tracks row this track is (single source of truth); null = new. */
41
+ moduleTrackId?: number | null;
42
+ /** Owner's track name, mirrored to module_tracks.track_name. */
43
+ trackName?: string;
44
+ }
45
+ interface SchematicTurnout {
46
+ id: string;
47
+ pos: number;
48
+ onTrack: string;
49
+ divergeTrack: string;
50
+ kind?: TurnoutKind;
51
+ name?: string | null;
52
+ address?: string | null;
53
+ }
54
+ interface SchematicSignal {
55
+ id: string;
56
+ pos: number;
57
+ /** Track the signal governs; absent = the primary main (lane 0). */
58
+ track?: string;
59
+ facing?: SignalFacing;
60
+ kind?: "mast" | "dwarf";
61
+ name?: string | null;
62
+ aspects?: string[];
63
+ /** Which side of the track the signal sits on (#122). */
64
+ side?: SignalSide;
65
+ /** Turnout this control point governs; absent = standalone block signal. */
66
+ turnout?: string;
67
+ }
68
+ interface SchematicBlock {
69
+ id: string;
70
+ name: string;
71
+ tracks?: string[];
72
+ from: number;
73
+ to: number;
74
+ }
75
+ /**
76
+ * A control point is an interlocking: a named group of one or more signals and
77
+ * zero or more turnouts. A passing siding has two (West/East); a lone block
78
+ * signal is a control point with one signal and no turnouts.
79
+ */
80
+ interface SchematicControlPoint {
81
+ id: string;
82
+ name?: string | null;
83
+ turnouts?: string[];
84
+ signals?: SchematicSignal[];
85
+ }
86
+ interface ModuleSchematicDoc {
87
+ version: number;
88
+ module?: string;
89
+ lengthInches?: number;
90
+ endplates: SchematicEndplate[];
91
+ tracks: SchematicTrack[];
92
+ turnouts?: SchematicTurnout[];
93
+ controlPoints?: SchematicControlPoint[];
94
+ /** @deprecated pre-grouping flat signals; read for back-compat. */
95
+ signals?: SchematicSignal[];
96
+ }
97
+ declare const MAIN_TRACK_ID = "main";
98
+ declare const N_SCALE_RATIO = 160;
99
+ /** Real inches on the module → scale feet of prototype track represented. */
100
+ declare function inchesToScaleFeet(inches: number, ratio?: number): number;
101
+ /** Scale feet of prototype track → real inches on the module. */
102
+ declare function scaleFeetToInches(feet: number, ratio?: number): number;
103
+ /** Parse a jsonb value into a schematic doc, or null if it isn't one. */
104
+ declare function asModuleSchematic(x: unknown): ModuleSchematicDoc | null;
105
+ interface EditorTrack {
106
+ id: string;
107
+ role: TrackRole;
108
+ lane: number;
109
+ fromPos: number;
110
+ toPos: number;
111
+ /** module_tracks row id (single source of truth), or null for a new track. */
112
+ moduleTrackId: number | null;
113
+ /** Owner's track name → module_tracks.track_name. */
114
+ trackName: string;
115
+ }
116
+ /** A module_tracks row as loaded for the editor. */
117
+ interface ModuleTrackRow {
118
+ id: number;
119
+ track_name: string | null;
120
+ capacity_scale_feet: number | null;
121
+ }
122
+ interface EditorTurnout {
123
+ id: string;
124
+ name: string;
125
+ pos: number;
126
+ onTrack: string;
127
+ divergeTrack: string;
128
+ kind: TurnoutKind;
129
+ }
130
+ interface EditorCpSignal {
131
+ id: string;
132
+ pos: number;
133
+ track: string;
134
+ facing: SignalFacing;
135
+ side: SignalSide;
136
+ }
137
+ interface EditorControlPoint {
138
+ id: string;
139
+ name: string;
140
+ turnouts: string[];
141
+ signals: EditorCpSignal[];
142
+ }
143
+ interface EditorState {
144
+ lengthInches: number;
145
+ configA: TrackConfig;
146
+ configB: TrackConfig;
147
+ extraTracks: EditorTrack[];
148
+ turnouts: EditorTurnout[];
149
+ controlPoints: EditorControlPoint[];
150
+ }
151
+ /** Build the empty editor state for a module of the given length. */
152
+ declare function emptyEditorState(lengthInches: number): EditorState;
153
+ /** Assemble a spec-conformant doc from the editor state. */
154
+ declare function stateToDoc(state: EditorState, recordNumber: string): ModuleSchematicDoc;
155
+ /**
156
+ * Derive editor state from the doc and the module's Track section rows. Tracks
157
+ * are the single source of truth for name/capacity (module_tracks), while the
158
+ * schematic doc adds geometry (lane, positions). We merge: doc tracks first
159
+ * (they carry geometry + their moduleTrackId link), then any module_tracks not
160
+ * yet positioned in the schematic.
161
+ */
162
+ declare function docToState(doc: unknown, fallbackLength: number, moduleTracks?: ModuleTrackRow[]): EditorState;
163
+ /** Find an unused `${prefix}${n}` id given the ones already present. */
164
+ declare function nextId(prefix: string, existing: string[]): string;
165
+ /**
166
+ * Build a passing siding as one unit: the siding track, a switch at each end,
167
+ * and control-point signals for both directions at each end (prototype Station
168
+ * Entering Signal). Returns the new items to merge into the editor state.
169
+ */
170
+ declare function buildPassingSiding(state: EditorState): {
171
+ track: EditorTrack;
172
+ turnouts: EditorTurnout[];
173
+ controlPoints: EditorControlPoint[];
174
+ };
175
+ interface DrawTrack {
176
+ id: string;
177
+ role: TrackRole;
178
+ lane: number;
179
+ fromFrac: number;
180
+ toFrac: number;
181
+ capacityFeet: number | null;
182
+ }
183
+ interface DrawTurnout {
184
+ id: string;
185
+ name: string;
186
+ posFrac: number;
187
+ onLane: number;
188
+ divergeLane: number;
189
+ }
190
+ interface DrawSignal {
191
+ id: string;
192
+ name: string;
193
+ posFrac: number;
194
+ lane: number;
195
+ facing: SignalFacing;
196
+ side: SignalSide;
197
+ }
198
+ interface ModuleFeatures {
199
+ /** Whether either endplate declares a double-track main. */
200
+ doubleMain: boolean;
201
+ /** Non-main tracks (sidings/spurs/yard/crossover). */
202
+ extraTracks: DrawTrack[];
203
+ turnouts: DrawTurnout[];
204
+ signals: DrawSignal[];
205
+ }
206
+ /**
207
+ * Resolve a schematic doc into positioned drawables. `pos` (inches) becomes a
208
+ * fraction of the module length; endplate A = 0, B = length; turnouts sit at
209
+ * their pos. Tracks may carry explicit fromPos/toPos (overriding node lookup).
210
+ * To-scale: a feature renders at its true position, clamped only to the
211
+ * module's own extent — so signals near an end read at their real spot, not
212
+ * bunched at an inset (#122).
213
+ */
214
+ declare function moduleFeatures(doc: ModuleSchematicDoc): ModuleFeatures;
215
+
216
+ export { type DrawSignal, type DrawTrack, type DrawTurnout, type EditorControlPoint, type EditorCpSignal, type EditorState, type EditorTrack, type EditorTurnout, MAIN_TRACK_ID, type ModuleFeatures, type ModuleSchematicDoc, type ModuleTrackRow, N_SCALE_RATIO, type SchematicBlock, type SchematicControlPoint, type SchematicEndplate, type SchematicEndplateTrack, type SchematicSignal, type SchematicTrack, type SchematicTurnout, type SignalFacing, type SignalSide, type TrackConfig, type TrackRole, type TurnoutKind, asModuleSchematic, buildPassingSiding, docToState, emptyEditorState, inchesToScaleFeet, moduleFeatures, nextId, scaleFeetToInches, stateToDoc };
@@ -0,0 +1,216 @@
1
+ /**
2
+ * @willcgage/module-schematic — the shared module operations-schematic (track-graph)
3
+ * that the Module Repository authors and Free-Dispatcher imports.
4
+ *
5
+ * Topological, straightened-first: positions are 1-D inches along the module
6
+ * (from endplate A), lanes are integer track indices (0 = primary main). This is
7
+ * the single source of truth for both apps — the doc types, the lenient parser
8
+ * (docs arrive as jsonb / unknown), the pure feature resolver both renderers
9
+ * draw, the N-scale helpers, and the editor <-> doc state machine an authoring
10
+ * UI binds to. See docs/module-schematic-format.md in the free-dispatcher repo.
11
+ *
12
+ * Framework-agnostic and side-effect-free so it can be unit-tested and consumed
13
+ * from Next.js (server + client) and Electron alike.
14
+ */
15
+ type TrackConfig = "single" | "double";
16
+ type TrackRole = "main" | "siding" | "spur" | "yard" | "crossover";
17
+ type TurnoutKind = "left" | "right" | "wye";
18
+ type SignalFacing = "AtoB" | "BtoA";
19
+ type SignalSide = "above" | "below";
20
+ interface SchematicEndplateTrack {
21
+ trackId: string;
22
+ lane: number;
23
+ config?: TrackConfig | null;
24
+ }
25
+ interface SchematicEndplate {
26
+ id: string;
27
+ label?: string | null;
28
+ tracks?: SchematicEndplateTrack[];
29
+ }
30
+ interface SchematicTrack {
31
+ id: string;
32
+ role: TrackRole;
33
+ lane: number;
34
+ from?: string;
35
+ to?: string;
36
+ fromPos?: number | null;
37
+ toPos?: number | null;
38
+ capacityFeet?: number | null;
39
+ industryRef?: number | null;
40
+ /** The module_tracks row this track is (single source of truth); null = new. */
41
+ moduleTrackId?: number | null;
42
+ /** Owner's track name, mirrored to module_tracks.track_name. */
43
+ trackName?: string;
44
+ }
45
+ interface SchematicTurnout {
46
+ id: string;
47
+ pos: number;
48
+ onTrack: string;
49
+ divergeTrack: string;
50
+ kind?: TurnoutKind;
51
+ name?: string | null;
52
+ address?: string | null;
53
+ }
54
+ interface SchematicSignal {
55
+ id: string;
56
+ pos: number;
57
+ /** Track the signal governs; absent = the primary main (lane 0). */
58
+ track?: string;
59
+ facing?: SignalFacing;
60
+ kind?: "mast" | "dwarf";
61
+ name?: string | null;
62
+ aspects?: string[];
63
+ /** Which side of the track the signal sits on (#122). */
64
+ side?: SignalSide;
65
+ /** Turnout this control point governs; absent = standalone block signal. */
66
+ turnout?: string;
67
+ }
68
+ interface SchematicBlock {
69
+ id: string;
70
+ name: string;
71
+ tracks?: string[];
72
+ from: number;
73
+ to: number;
74
+ }
75
+ /**
76
+ * A control point is an interlocking: a named group of one or more signals and
77
+ * zero or more turnouts. A passing siding has two (West/East); a lone block
78
+ * signal is a control point with one signal and no turnouts.
79
+ */
80
+ interface SchematicControlPoint {
81
+ id: string;
82
+ name?: string | null;
83
+ turnouts?: string[];
84
+ signals?: SchematicSignal[];
85
+ }
86
+ interface ModuleSchematicDoc {
87
+ version: number;
88
+ module?: string;
89
+ lengthInches?: number;
90
+ endplates: SchematicEndplate[];
91
+ tracks: SchematicTrack[];
92
+ turnouts?: SchematicTurnout[];
93
+ controlPoints?: SchematicControlPoint[];
94
+ /** @deprecated pre-grouping flat signals; read for back-compat. */
95
+ signals?: SchematicSignal[];
96
+ }
97
+ declare const MAIN_TRACK_ID = "main";
98
+ declare const N_SCALE_RATIO = 160;
99
+ /** Real inches on the module → scale feet of prototype track represented. */
100
+ declare function inchesToScaleFeet(inches: number, ratio?: number): number;
101
+ /** Scale feet of prototype track → real inches on the module. */
102
+ declare function scaleFeetToInches(feet: number, ratio?: number): number;
103
+ /** Parse a jsonb value into a schematic doc, or null if it isn't one. */
104
+ declare function asModuleSchematic(x: unknown): ModuleSchematicDoc | null;
105
+ interface EditorTrack {
106
+ id: string;
107
+ role: TrackRole;
108
+ lane: number;
109
+ fromPos: number;
110
+ toPos: number;
111
+ /** module_tracks row id (single source of truth), or null for a new track. */
112
+ moduleTrackId: number | null;
113
+ /** Owner's track name → module_tracks.track_name. */
114
+ trackName: string;
115
+ }
116
+ /** A module_tracks row as loaded for the editor. */
117
+ interface ModuleTrackRow {
118
+ id: number;
119
+ track_name: string | null;
120
+ capacity_scale_feet: number | null;
121
+ }
122
+ interface EditorTurnout {
123
+ id: string;
124
+ name: string;
125
+ pos: number;
126
+ onTrack: string;
127
+ divergeTrack: string;
128
+ kind: TurnoutKind;
129
+ }
130
+ interface EditorCpSignal {
131
+ id: string;
132
+ pos: number;
133
+ track: string;
134
+ facing: SignalFacing;
135
+ side: SignalSide;
136
+ }
137
+ interface EditorControlPoint {
138
+ id: string;
139
+ name: string;
140
+ turnouts: string[];
141
+ signals: EditorCpSignal[];
142
+ }
143
+ interface EditorState {
144
+ lengthInches: number;
145
+ configA: TrackConfig;
146
+ configB: TrackConfig;
147
+ extraTracks: EditorTrack[];
148
+ turnouts: EditorTurnout[];
149
+ controlPoints: EditorControlPoint[];
150
+ }
151
+ /** Build the empty editor state for a module of the given length. */
152
+ declare function emptyEditorState(lengthInches: number): EditorState;
153
+ /** Assemble a spec-conformant doc from the editor state. */
154
+ declare function stateToDoc(state: EditorState, recordNumber: string): ModuleSchematicDoc;
155
+ /**
156
+ * Derive editor state from the doc and the module's Track section rows. Tracks
157
+ * are the single source of truth for name/capacity (module_tracks), while the
158
+ * schematic doc adds geometry (lane, positions). We merge: doc tracks first
159
+ * (they carry geometry + their moduleTrackId link), then any module_tracks not
160
+ * yet positioned in the schematic.
161
+ */
162
+ declare function docToState(doc: unknown, fallbackLength: number, moduleTracks?: ModuleTrackRow[]): EditorState;
163
+ /** Find an unused `${prefix}${n}` id given the ones already present. */
164
+ declare function nextId(prefix: string, existing: string[]): string;
165
+ /**
166
+ * Build a passing siding as one unit: the siding track, a switch at each end,
167
+ * and control-point signals for both directions at each end (prototype Station
168
+ * Entering Signal). Returns the new items to merge into the editor state.
169
+ */
170
+ declare function buildPassingSiding(state: EditorState): {
171
+ track: EditorTrack;
172
+ turnouts: EditorTurnout[];
173
+ controlPoints: EditorControlPoint[];
174
+ };
175
+ interface DrawTrack {
176
+ id: string;
177
+ role: TrackRole;
178
+ lane: number;
179
+ fromFrac: number;
180
+ toFrac: number;
181
+ capacityFeet: number | null;
182
+ }
183
+ interface DrawTurnout {
184
+ id: string;
185
+ name: string;
186
+ posFrac: number;
187
+ onLane: number;
188
+ divergeLane: number;
189
+ }
190
+ interface DrawSignal {
191
+ id: string;
192
+ name: string;
193
+ posFrac: number;
194
+ lane: number;
195
+ facing: SignalFacing;
196
+ side: SignalSide;
197
+ }
198
+ interface ModuleFeatures {
199
+ /** Whether either endplate declares a double-track main. */
200
+ doubleMain: boolean;
201
+ /** Non-main tracks (sidings/spurs/yard/crossover). */
202
+ extraTracks: DrawTrack[];
203
+ turnouts: DrawTurnout[];
204
+ signals: DrawSignal[];
205
+ }
206
+ /**
207
+ * Resolve a schematic doc into positioned drawables. `pos` (inches) becomes a
208
+ * fraction of the module length; endplate A = 0, B = length; turnouts sit at
209
+ * their pos. Tracks may carry explicit fromPos/toPos (overriding node lookup).
210
+ * To-scale: a feature renders at its true position, clamped only to the
211
+ * module's own extent — so signals near an end read at their real spot, not
212
+ * bunched at an inset (#122).
213
+ */
214
+ declare function moduleFeatures(doc: ModuleSchematicDoc): ModuleFeatures;
215
+
216
+ export { type DrawSignal, type DrawTrack, type DrawTurnout, type EditorControlPoint, type EditorCpSignal, type EditorState, type EditorTrack, type EditorTurnout, MAIN_TRACK_ID, type ModuleFeatures, type ModuleSchematicDoc, type ModuleTrackRow, N_SCALE_RATIO, type SchematicBlock, type SchematicControlPoint, type SchematicEndplate, type SchematicEndplateTrack, type SchematicSignal, type SchematicTrack, type SchematicTurnout, type SignalFacing, type SignalSide, type TrackConfig, type TrackRole, type TurnoutKind, asModuleSchematic, buildPassingSiding, docToState, emptyEditorState, inchesToScaleFeet, moduleFeatures, nextId, scaleFeetToInches, stateToDoc };
package/dist/index.js ADDED
@@ -0,0 +1,287 @@
1
+ // src/index.ts
2
+ var MAIN_TRACK_ID = "main";
3
+ var N_SCALE_RATIO = 160;
4
+ function inchesToScaleFeet(inches, ratio = N_SCALE_RATIO) {
5
+ return inches * ratio / 12;
6
+ }
7
+ function scaleFeetToInches(feet, ratio = N_SCALE_RATIO) {
8
+ return feet * 12 / ratio;
9
+ }
10
+ function asModuleSchematic(x) {
11
+ if (!x || typeof x !== "object") return null;
12
+ const d = x;
13
+ if (typeof d.version !== "number") return null;
14
+ if (!Array.isArray(d.endplates) || !Array.isArray(d.tracks)) return null;
15
+ return d;
16
+ }
17
+ function emptyEditorState(lengthInches) {
18
+ return {
19
+ lengthInches: lengthInches > 0 ? lengthInches : 24,
20
+ configA: "single",
21
+ configB: "single",
22
+ extraTracks: [],
23
+ turnouts: [],
24
+ controlPoints: []
25
+ };
26
+ }
27
+ function stateToDoc(state, recordNumber) {
28
+ return {
29
+ version: 1,
30
+ module: recordNumber,
31
+ lengthInches: state.lengthInches,
32
+ endplates: [
33
+ { id: "A", label: "West", tracks: [{ trackId: MAIN_TRACK_ID, lane: 0, config: state.configA }] },
34
+ { id: "B", label: "East", tracks: [{ trackId: MAIN_TRACK_ID, lane: 0, config: state.configB }] }
35
+ ],
36
+ tracks: [
37
+ { id: MAIN_TRACK_ID, role: "main", lane: 0, from: "A", to: "B" },
38
+ ...state.extraTracks.map((t) => ({
39
+ id: t.id,
40
+ role: t.role,
41
+ lane: t.lane,
42
+ fromPos: t.fromPos,
43
+ toPos: t.toPos,
44
+ moduleTrackId: t.moduleTrackId,
45
+ trackName: t.trackName || void 0,
46
+ capacityFeet: Math.round(inchesToScaleFeet(Math.abs(t.toPos - t.fromPos)))
47
+ }))
48
+ ],
49
+ turnouts: state.turnouts.map((t) => ({
50
+ id: t.id,
51
+ pos: t.pos,
52
+ onTrack: t.onTrack,
53
+ divergeTrack: t.divergeTrack,
54
+ kind: t.kind,
55
+ name: t.name || void 0
56
+ })),
57
+ controlPoints: state.controlPoints.map((c) => ({
58
+ id: c.id,
59
+ name: c.name,
60
+ turnouts: c.turnouts,
61
+ signals: c.signals.map((s) => ({
62
+ id: s.id,
63
+ pos: s.pos,
64
+ track: s.track,
65
+ facing: s.facing,
66
+ kind: "mast",
67
+ side: s.side
68
+ }))
69
+ }))
70
+ };
71
+ }
72
+ function docToState(doc, fallbackLength, moduleTracks = []) {
73
+ const base = emptyEditorState(fallbackLength);
74
+ const d = doc && typeof doc === "object" ? doc : null;
75
+ const hasDoc = !!d && typeof d.lengthInches === "number" && Array.isArray(d.tracks);
76
+ const len = fallbackLength > 0 ? fallbackLength : hasDoc ? d.lengthInches : 24;
77
+ const docLen = hasDoc && d.lengthInches > 0 ? d.lengthInches : len;
78
+ const scale = docLen > 0 ? len / docLen : 1;
79
+ const sc = (p) => Math.round(p * scale);
80
+ const nameOf = (id) => {
81
+ const mt = id != null ? moduleTracks.find((m) => m.id === id) : void 0;
82
+ return mt?.track_name ?? "";
83
+ };
84
+ const extraTracks = [];
85
+ const usedMt = /* @__PURE__ */ new Set();
86
+ if (hasDoc) {
87
+ for (const t of d.tracks) {
88
+ if (t.role === "main") continue;
89
+ const moduleTrackId = t.moduleTrackId ?? null;
90
+ if (moduleTrackId != null) usedMt.add(moduleTrackId);
91
+ extraTracks.push({
92
+ id: t.id,
93
+ role: t.role ?? "siding",
94
+ lane: t.lane ?? 1,
95
+ fromPos: sc(t.fromPos ?? 0),
96
+ toPos: t.toPos != null ? sc(t.toPos) : len,
97
+ moduleTrackId,
98
+ trackName: t.trackName ?? nameOf(moduleTrackId)
99
+ });
100
+ }
101
+ }
102
+ const unused = moduleTracks.filter((mt) => !usedMt.has(mt.id));
103
+ let ui = 0;
104
+ for (const et of extraTracks) {
105
+ if (et.moduleTrackId == null && ui < unused.length) {
106
+ const mt = unused[ui++];
107
+ et.moduleTrackId = mt.id;
108
+ if (!et.trackName) et.trackName = mt.track_name ?? "";
109
+ usedMt.add(mt.id);
110
+ }
111
+ }
112
+ let lane = Math.max(0, ...extraTracks.map((t) => t.lane));
113
+ for (const mt of moduleTracks) {
114
+ if (usedMt.has(mt.id)) continue;
115
+ lane += 1;
116
+ extraTracks.push({
117
+ id: `mt${mt.id}`,
118
+ role: "siding",
119
+ lane,
120
+ fromPos: Math.round(len * 0.2),
121
+ toPos: Math.round(len * 0.8),
122
+ moduleTrackId: mt.id,
123
+ trackName: mt.track_name ?? ""
124
+ });
125
+ }
126
+ if (!hasDoc) return { ...base, lengthInches: len, extraTracks };
127
+ const configOf = (id) => {
128
+ const ep = (d.endplates ?? []).find((e) => e.id === id);
129
+ return ep?.tracks?.[0]?.config === "double" ? "double" : "single";
130
+ };
131
+ return {
132
+ lengthInches: len,
133
+ configA: configOf("A"),
134
+ configB: configOf("B"),
135
+ extraTracks,
136
+ turnouts: (d.turnouts ?? []).map((t) => ({
137
+ id: t.id,
138
+ name: t.name ?? "",
139
+ pos: sc(t.pos),
140
+ onTrack: t.onTrack,
141
+ divergeTrack: t.divergeTrack,
142
+ kind: t.kind ?? "right"
143
+ })),
144
+ controlPoints: readControlPoints(d, sc)
145
+ };
146
+ }
147
+ function readControlPoints(d, sc = (p) => p) {
148
+ if (Array.isArray(d.controlPoints)) {
149
+ return d.controlPoints.map((c) => ({
150
+ id: c.id,
151
+ name: c.name ?? "",
152
+ turnouts: c.turnouts ?? [],
153
+ signals: (c.signals ?? []).map((s) => ({
154
+ id: s.id,
155
+ pos: sc(s.pos),
156
+ track: s.track ?? MAIN_TRACK_ID,
157
+ facing: s.facing ?? "AtoB",
158
+ side: s.side ?? "above"
159
+ }))
160
+ }));
161
+ }
162
+ const groups = /* @__PURE__ */ new Map();
163
+ let n = 0;
164
+ for (const s of d.signals ?? []) {
165
+ const key = s.turnout || `blk-${s.id}`;
166
+ let cp = groups.get(key);
167
+ if (!cp) {
168
+ cp = { id: `cp${++n}`, name: s.name ?? "", turnouts: s.turnout ? [s.turnout] : [], signals: [] };
169
+ groups.set(key, cp);
170
+ }
171
+ cp.signals.push({
172
+ id: s.id,
173
+ pos: sc(s.pos),
174
+ track: s.track ?? MAIN_TRACK_ID,
175
+ facing: s.facing ?? "AtoB",
176
+ side: s.side ?? "above"
177
+ });
178
+ }
179
+ return [...groups.values()];
180
+ }
181
+ function nextId(prefix, existing) {
182
+ let n = 1;
183
+ while (existing.includes(`${prefix}${n}`)) n += 1;
184
+ return `${prefix}${n}`;
185
+ }
186
+ function buildPassingSiding(state) {
187
+ const len = state.lengthInches > 0 ? state.lengthInches : 24;
188
+ const inset = Math.max(6, Math.round(len * 0.08));
189
+ const fromPos = inset;
190
+ const toPos = Math.max(fromPos + 1, len - inset);
191
+ const lane = Math.max(1, ...state.extraTracks.map((t) => t.lane + 1));
192
+ const trackIds = [MAIN_TRACK_ID, ...state.extraTracks.map((t) => t.id)];
193
+ const sidId = nextId("sid", trackIds);
194
+ const track = {
195
+ id: sidId,
196
+ role: "siding",
197
+ lane,
198
+ fromPos,
199
+ toPos,
200
+ moduleTrackId: null,
201
+ trackName: "Passing siding"
202
+ };
203
+ const swIds = state.turnouts.map((t) => t.id);
204
+ const swW = nextId("sw", swIds);
205
+ const swE = nextId("sw", [...swIds, swW]);
206
+ const turnouts = [
207
+ { id: swW, name: "West Siding", pos: fromPos, onTrack: MAIN_TRACK_ID, divergeTrack: sidId, kind: "right" },
208
+ { id: swE, name: "East Siding", pos: toPos, onTrack: MAIN_TRACK_ID, divergeTrack: sidId, kind: "left" }
209
+ ];
210
+ const cpIds = state.controlPoints.map((c) => c.id);
211
+ const cpW = nextId("cp", cpIds);
212
+ const cpE = nextId("cp", [...cpIds, cpW]);
213
+ const sig = (cpId, pos, facing) => ({
214
+ id: `${cpId}-${facing}`,
215
+ pos,
216
+ track: MAIN_TRACK_ID,
217
+ facing,
218
+ // opposite directions on opposite sides so they never overlap
219
+ side: facing === "AtoB" ? "above" : "below"
220
+ });
221
+ const controlPoints = [
222
+ { id: cpW, name: "West Siding", turnouts: [swW], signals: [sig(cpW, fromPos, "AtoB"), sig(cpW, fromPos, "BtoA")] },
223
+ { id: cpE, name: "East Siding", turnouts: [swE], signals: [sig(cpE, toPos, "AtoB"), sig(cpE, toPos, "BtoA")] }
224
+ ];
225
+ return { track, turnouts, controlPoints };
226
+ }
227
+ function moduleFeatures(doc) {
228
+ const len = doc.lengthInches && doc.lengthInches > 0 ? doc.lengthInches : Math.max(
229
+ 1,
230
+ ...doc.tracks.map((t) => Math.max(t.fromPos ?? 0, t.toPos ?? 0)),
231
+ ...(doc.turnouts ?? []).map((t) => t.pos)
232
+ );
233
+ const clampFrac = (p) => Math.min(1, Math.max(0, p / len));
234
+ const trackLane = /* @__PURE__ */ new Map();
235
+ for (const t of doc.tracks) trackLane.set(t.id, t.lane);
236
+ const endplatePos = /* @__PURE__ */ new Map();
237
+ doc.endplates.forEach((e, i) => endplatePos.set(e.id, i === 0 ? 0 : len));
238
+ const turnoutPos = /* @__PURE__ */ new Map();
239
+ for (const t of doc.turnouts ?? []) turnoutPos.set(t.id, t.pos);
240
+ const posOf = (nodeId) => {
241
+ if (nodeId == null) return null;
242
+ if (endplatePos.has(nodeId)) return endplatePos.get(nodeId);
243
+ if (turnoutPos.has(nodeId)) return turnoutPos.get(nodeId);
244
+ return null;
245
+ };
246
+ const doubleMain = doc.endplates.some(
247
+ (e) => e.tracks?.some((t) => t.config === "double")
248
+ );
249
+ const extraTracks = [];
250
+ for (const t of doc.tracks) {
251
+ if (t.role === "main") continue;
252
+ const from = t.fromPos ?? posOf(t.from);
253
+ const to = t.toPos ?? posOf(t.to);
254
+ if (from == null || to == null) continue;
255
+ extraTracks.push({
256
+ id: t.id,
257
+ role: t.role,
258
+ lane: t.lane,
259
+ fromFrac: clampFrac(Math.min(from, to)),
260
+ toFrac: clampFrac(Math.max(from, to)),
261
+ capacityFeet: t.capacityFeet ?? null
262
+ });
263
+ }
264
+ const turnouts = (doc.turnouts ?? []).map((t) => ({
265
+ id: t.id,
266
+ name: t.name ?? "",
267
+ posFrac: clampFrac(t.pos),
268
+ onLane: trackLane.get(t.onTrack) ?? 0,
269
+ divergeLane: trackLane.get(t.divergeTrack) ?? 1
270
+ }));
271
+ const drawSignal = (s, name) => ({
272
+ id: s.id,
273
+ name,
274
+ posFrac: clampFrac(s.pos),
275
+ lane: s.track ? trackLane.get(s.track) ?? 0 : 0,
276
+ facing: s.facing ?? "AtoB",
277
+ side: s.side === "below" ? "below" : "above"
278
+ });
279
+ const signals = Array.isArray(doc.controlPoints) ? doc.controlPoints.flatMap(
280
+ (c) => (c.signals ?? []).map((s) => drawSignal(s, c.name ?? ""))
281
+ ) : (doc.signals ?? []).map((s) => drawSignal(s, s.name ?? ""));
282
+ return { doubleMain, extraTracks, turnouts, signals };
283
+ }
284
+
285
+ export { MAIN_TRACK_ID, N_SCALE_RATIO, asModuleSchematic, buildPassingSiding, docToState, emptyEditorState, inchesToScaleFeet, moduleFeatures, nextId, scaleFeetToInches, stateToDoc };
286
+ //# sourceMappingURL=index.js.map
287
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/index.ts"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["/**\n * @willcgage/module-schematic — the shared module operations-schematic (track-graph)\n * that the Module Repository authors and Free-Dispatcher imports.\n *\n * Topological, straightened-first: positions are 1-D inches along the module\n * (from endplate A), lanes are integer track indices (0 = primary main). This is\n * the single source of truth for both apps — the doc types, the lenient parser\n * (docs arrive as jsonb / unknown), the pure feature resolver both renderers\n * draw, the N-scale helpers, and the editor <-> doc state machine an authoring\n * UI binds to. See docs/module-schematic-format.md in the free-dispatcher repo.\n *\n * Framework-agnostic and side-effect-free so it can be unit-tested and consumed\n * from Next.js (server + client) and Electron alike.\n */\n\nexport type TrackConfig = \"single\" | \"double\";\nexport type TrackRole = \"main\" | \"siding\" | \"spur\" | \"yard\" | \"crossover\";\nexport type TurnoutKind = \"left\" | \"right\" | \"wye\";\nexport type SignalFacing = \"AtoB\" | \"BtoA\";\nexport type SignalSide = \"above\" | \"below\";\n\nexport interface SchematicEndplateTrack {\n trackId: string;\n lane: number;\n config?: TrackConfig | null;\n}\nexport interface SchematicEndplate {\n id: string; // \"A\" (West) | \"B\" (East)\n label?: string | null;\n tracks?: SchematicEndplateTrack[];\n}\nexport interface SchematicTrack {\n id: string;\n role: TrackRole;\n lane: number;\n from?: string;\n to?: string;\n fromPos?: number | null;\n toPos?: number | null;\n capacityFeet?: number | null;\n industryRef?: number | null;\n /** The module_tracks row this track is (single source of truth); null = new. */\n moduleTrackId?: number | null;\n /** Owner's track name, mirrored to module_tracks.track_name. */\n trackName?: string;\n}\nexport interface SchematicTurnout {\n id: string;\n pos: number;\n onTrack: string;\n divergeTrack: string;\n kind?: TurnoutKind;\n name?: string | null;\n address?: string | null;\n}\nexport interface SchematicSignal {\n id: string;\n pos: number;\n /** Track the signal governs; absent = the primary main (lane 0). */\n track?: string;\n facing?: SignalFacing;\n kind?: \"mast\" | \"dwarf\";\n name?: string | null;\n aspects?: string[];\n /** Which side of the track the signal sits on (#122). */\n side?: SignalSide;\n /** Turnout this control point governs; absent = standalone block signal. */\n turnout?: string;\n}\nexport interface SchematicBlock {\n id: string;\n name: string;\n tracks?: string[];\n from: number;\n to: number;\n}\n/**\n * A control point is an interlocking: a named group of one or more signals and\n * zero or more turnouts. A passing siding has two (West/East); a lone block\n * signal is a control point with one signal and no turnouts.\n */\nexport interface SchematicControlPoint {\n id: string;\n name?: string | null;\n turnouts?: string[];\n signals?: SchematicSignal[];\n}\nexport interface ModuleSchematicDoc {\n version: number;\n module?: string;\n lengthInches?: number;\n endplates: SchematicEndplate[];\n tracks: SchematicTrack[];\n turnouts?: SchematicTurnout[];\n controlPoints?: SchematicControlPoint[];\n /** @deprecated pre-grouping flat signals; read for back-compat. */\n signals?: SchematicSignal[];\n}\n\nexport const MAIN_TRACK_ID = \"main\";\n\n// North American N scale (1:160): 396 real inches → 5280 scale feet = one mile.\nexport const N_SCALE_RATIO = 160;\n/** Real inches on the module → scale feet of prototype track represented. */\nexport function inchesToScaleFeet(inches: number, ratio = N_SCALE_RATIO): number {\n return (inches * ratio) / 12;\n}\n/** Scale feet of prototype track → real inches on the module. */\nexport function scaleFeetToInches(feet: number, ratio = N_SCALE_RATIO): number {\n return (feet * 12) / ratio;\n}\n\n/** Parse a jsonb value into a schematic doc, or null if it isn't one. */\nexport function asModuleSchematic(x: unknown): ModuleSchematicDoc | null {\n if (!x || typeof x !== \"object\") return null;\n const d = x as Record<string, unknown>;\n if (typeof d.version !== \"number\") return null;\n if (!Array.isArray(d.endplates) || !Array.isArray(d.tracks)) return null;\n return d as unknown as ModuleSchematicDoc;\n}\n\n// ---- Editor state (a flatter shape an authoring form binds to) -------------\n\nexport interface EditorTrack {\n id: string;\n role: TrackRole;\n lane: number;\n fromPos: number;\n toPos: number;\n /** module_tracks row id (single source of truth), or null for a new track. */\n moduleTrackId: number | null;\n /** Owner's track name → module_tracks.track_name. */\n trackName: string;\n}\n\n/** A module_tracks row as loaded for the editor. */\nexport interface ModuleTrackRow {\n id: number;\n track_name: string | null;\n capacity_scale_feet: number | null;\n}\nexport interface EditorTurnout {\n id: string;\n name: string;\n pos: number;\n onTrack: string;\n divergeTrack: string;\n kind: TurnoutKind;\n}\nexport interface EditorCpSignal {\n id: string;\n pos: number;\n track: string;\n facing: SignalFacing;\n side: SignalSide;\n}\nexport interface EditorControlPoint {\n id: string;\n name: string;\n turnouts: string[]; // turnout ids grouped under this control point\n signals: EditorCpSignal[];\n}\nexport interface EditorState {\n lengthInches: number;\n configA: TrackConfig;\n configB: TrackConfig;\n extraTracks: EditorTrack[]; // sidings/spurs/…; the main track is implicit\n turnouts: EditorTurnout[];\n controlPoints: EditorControlPoint[];\n}\n\n/** Build the empty editor state for a module of the given length. */\nexport function emptyEditorState(lengthInches: number): EditorState {\n return {\n lengthInches: lengthInches > 0 ? lengthInches : 24,\n configA: \"single\",\n configB: \"single\",\n extraTracks: [],\n turnouts: [],\n controlPoints: [],\n };\n}\n\n/** Assemble a spec-conformant doc from the editor state. */\nexport function stateToDoc(\n state: EditorState,\n recordNumber: string,\n): ModuleSchematicDoc {\n return {\n version: 1,\n module: recordNumber,\n lengthInches: state.lengthInches,\n endplates: [\n { id: \"A\", label: \"West\", tracks: [{ trackId: MAIN_TRACK_ID, lane: 0, config: state.configA }] },\n { id: \"B\", label: \"East\", tracks: [{ trackId: MAIN_TRACK_ID, lane: 0, config: state.configB }] },\n ],\n tracks: [\n { id: MAIN_TRACK_ID, role: \"main\", lane: 0, from: \"A\", to: \"B\" },\n ...state.extraTracks.map((t) => ({\n id: t.id,\n role: t.role,\n lane: t.lane,\n fromPos: t.fromPos,\n toPos: t.toPos,\n moduleTrackId: t.moduleTrackId,\n trackName: t.trackName || undefined,\n capacityFeet: Math.round(inchesToScaleFeet(Math.abs(t.toPos - t.fromPos))),\n })),\n ],\n turnouts: state.turnouts.map((t) => ({\n id: t.id,\n pos: t.pos,\n onTrack: t.onTrack,\n divergeTrack: t.divergeTrack,\n kind: t.kind,\n name: t.name || undefined,\n })),\n controlPoints: state.controlPoints.map((c) => ({\n id: c.id,\n name: c.name,\n turnouts: c.turnouts,\n signals: c.signals.map((s) => ({\n id: s.id,\n pos: s.pos,\n track: s.track,\n facing: s.facing,\n kind: \"mast\" as const,\n side: s.side,\n })),\n })),\n };\n}\n\n/**\n * Derive editor state from the doc and the module's Track section rows. Tracks\n * are the single source of truth for name/capacity (module_tracks), while the\n * schematic doc adds geometry (lane, positions). We merge: doc tracks first\n * (they carry geometry + their moduleTrackId link), then any module_tracks not\n * yet positioned in the schematic.\n */\nexport function docToState(\n doc: unknown,\n fallbackLength: number,\n moduleTracks: ModuleTrackRow[] = [],\n): EditorState {\n const base = emptyEditorState(fallbackLength);\n const d =\n doc && typeof doc === \"object\" ? (doc as ModuleSchematicDoc) : null;\n const hasDoc = !!d && typeof d.lengthInches === \"number\" && Array.isArray(d.tracks);\n // The module's length is authoritative (the mainline is the module). If the\n // saved doc used a different length, rescale its feature positions to fit so\n // the mainline always reads as the module's true length.\n const len = fallbackLength > 0 ? fallbackLength : hasDoc ? d!.lengthInches! : 24;\n const docLen = hasDoc && d!.lengthInches! > 0 ? d!.lengthInches! : len;\n const scale = docLen > 0 ? len / docLen : 1;\n const sc = (p: number) => Math.round(p * scale);\n\n const nameOf = (id: number | null | undefined): string => {\n const mt = id != null ? moduleTracks.find((m) => m.id === id) : undefined;\n return mt?.track_name ?? \"\";\n };\n\n const extraTracks: EditorTrack[] = [];\n const usedMt = new Set<number>();\n if (hasDoc) {\n for (const t of d!.tracks) {\n if (t.role === \"main\") continue;\n const moduleTrackId = t.moduleTrackId ?? null;\n if (moduleTrackId != null) usedMt.add(moduleTrackId);\n extraTracks.push({\n id: t.id,\n role: (t.role as TrackRole) ?? \"siding\",\n lane: t.lane ?? 1,\n fromPos: sc(t.fromPos ?? 0),\n toPos: t.toPos != null ? sc(t.toPos) : len,\n moduleTrackId,\n trackName: t.trackName ?? nameOf(moduleTrackId),\n });\n }\n }\n // Link pre-migration doc tracks (no moduleTrackId yet) to unused module_tracks\n // by order — keeping the doc track's id so turnout/signal references stay\n // valid. Only after that do leftover module_tracks become new tracks.\n const unused = moduleTracks.filter((mt) => !usedMt.has(mt.id));\n let ui = 0;\n for (const et of extraTracks) {\n if (et.moduleTrackId == null && ui < unused.length) {\n const mt = unused[ui++];\n et.moduleTrackId = mt.id;\n if (!et.trackName) et.trackName = mt.track_name ?? \"\";\n usedMt.add(mt.id);\n }\n }\n let lane = Math.max(0, ...extraTracks.map((t) => t.lane));\n for (const mt of moduleTracks) {\n if (usedMt.has(mt.id)) continue;\n lane += 1;\n extraTracks.push({\n id: `mt${mt.id}`,\n role: \"siding\",\n lane,\n fromPos: Math.round(len * 0.2),\n toPos: Math.round(len * 0.8),\n moduleTrackId: mt.id,\n trackName: mt.track_name ?? \"\",\n });\n }\n\n if (!hasDoc) return { ...base, lengthInches: len, extraTracks };\n\n const configOf = (id: string): TrackConfig => {\n const ep = (d!.endplates ?? []).find((e) => e.id === id);\n return ep?.tracks?.[0]?.config === \"double\" ? \"double\" : \"single\";\n };\n return {\n lengthInches: len,\n configA: configOf(\"A\"),\n configB: configOf(\"B\"),\n extraTracks,\n turnouts: (d!.turnouts ?? []).map((t) => ({\n id: t.id,\n name: t.name ?? \"\",\n pos: sc(t.pos),\n onTrack: t.onTrack,\n divergeTrack: t.divergeTrack,\n kind: (t.kind as TurnoutKind) ?? \"right\",\n })),\n controlPoints: readControlPoints(d!, sc),\n };\n}\n\n/** Control points from a doc, migrating pre-grouping flat signals into groups. */\nfunction readControlPoints(\n d: ModuleSchematicDoc,\n sc: (p: number) => number = (p) => p,\n): EditorControlPoint[] {\n if (Array.isArray(d.controlPoints)) {\n return d.controlPoints.map((c) => ({\n id: c.id,\n name: c.name ?? \"\",\n turnouts: c.turnouts ?? [],\n signals: (c.signals ?? []).map((s) => ({\n id: s.id,\n pos: sc(s.pos),\n track: s.track ?? MAIN_TRACK_ID,\n facing: (s.facing as SignalFacing) ?? \"AtoB\",\n side: (s.side as SignalSide) ?? \"above\",\n })),\n }));\n }\n // Back-compat: group old flat signals by their turnout (or standalone).\n const groups = new Map<string, EditorControlPoint>();\n let n = 0;\n for (const s of d.signals ?? []) {\n const key = s.turnout || `blk-${s.id}`;\n let cp = groups.get(key);\n if (!cp) {\n cp = { id: `cp${++n}`, name: s.name ?? \"\", turnouts: s.turnout ? [s.turnout] : [], signals: [] };\n groups.set(key, cp);\n }\n cp.signals.push({\n id: s.id,\n pos: sc(s.pos),\n track: s.track ?? MAIN_TRACK_ID,\n facing: (s.facing as SignalFacing) ?? \"AtoB\",\n side: (s.side as SignalSide) ?? \"above\",\n });\n }\n return [...groups.values()];\n}\n\n/** Find an unused `${prefix}${n}` id given the ones already present. */\nexport function nextId(prefix: string, existing: string[]): string {\n let n = 1;\n while (existing.includes(`${prefix}${n}`)) n += 1;\n return `${prefix}${n}`;\n}\n\n/**\n * Build a passing siding as one unit: the siding track, a switch at each end,\n * and control-point signals for both directions at each end (prototype Station\n * Entering Signal). Returns the new items to merge into the editor state.\n */\nexport function buildPassingSiding(state: EditorState): {\n track: EditorTrack;\n turnouts: EditorTurnout[];\n controlPoints: EditorControlPoint[];\n} {\n const len = state.lengthInches > 0 ? state.lengthInches : 24;\n const inset = Math.max(6, Math.round(len * 0.08));\n const fromPos = inset;\n const toPos = Math.max(fromPos + 1, len - inset);\n const lane = Math.max(1, ...state.extraTracks.map((t) => t.lane + 1));\n\n const trackIds = [MAIN_TRACK_ID, ...state.extraTracks.map((t) => t.id)];\n const sidId = nextId(\"sid\", trackIds);\n const track: EditorTrack = {\n id: sidId,\n role: \"siding\",\n lane,\n fromPos,\n toPos,\n moduleTrackId: null,\n trackName: \"Passing siding\",\n };\n\n const swIds = state.turnouts.map((t) => t.id);\n const swW = nextId(\"sw\", swIds);\n const swE = nextId(\"sw\", [...swIds, swW]);\n const turnouts: EditorTurnout[] = [\n { id: swW, name: \"West Siding\", pos: fromPos, onTrack: MAIN_TRACK_ID, divergeTrack: sidId, kind: \"right\" },\n { id: swE, name: \"East Siding\", pos: toPos, onTrack: MAIN_TRACK_ID, divergeTrack: sidId, kind: \"left\" },\n ];\n\n // One control point at each end, each grouping its switch and both-direction\n // signals on the main (prototype Station Entering Signal).\n const cpIds = state.controlPoints.map((c) => c.id);\n const cpW = nextId(\"cp\", cpIds);\n const cpE = nextId(\"cp\", [...cpIds, cpW]);\n const sig = (cpId: string, pos: number, facing: SignalFacing): EditorCpSignal => ({\n id: `${cpId}-${facing}`,\n pos,\n track: MAIN_TRACK_ID,\n facing,\n // opposite directions on opposite sides so they never overlap\n side: facing === \"AtoB\" ? \"above\" : \"below\",\n });\n const controlPoints: EditorControlPoint[] = [\n { id: cpW, name: \"West Siding\", turnouts: [swW], signals: [sig(cpW, fromPos, \"AtoB\"), sig(cpW, fromPos, \"BtoA\")] },\n { id: cpE, name: \"East Siding\", turnouts: [swE], signals: [sig(cpE, toPos, \"AtoB\"), sig(cpE, toPos, \"BtoA\")] },\n ];\n\n return { track, turnouts, controlPoints };\n}\n\n// ---- Pure feature resolver (both renderers draw these) --------------------\n\nexport interface DrawTrack {\n id: string;\n role: TrackRole;\n lane: number;\n fromFrac: number;\n toFrac: number;\n capacityFeet: number | null;\n}\nexport interface DrawTurnout {\n id: string;\n name: string;\n posFrac: number;\n onLane: number;\n divergeLane: number;\n}\nexport interface DrawSignal {\n id: string;\n name: string;\n posFrac: number;\n lane: number;\n facing: SignalFacing;\n side: SignalSide;\n}\nexport interface ModuleFeatures {\n /** Whether either endplate declares a double-track main. */\n doubleMain: boolean;\n /** Non-main tracks (sidings/spurs/yard/crossover). */\n extraTracks: DrawTrack[];\n turnouts: DrawTurnout[];\n signals: DrawSignal[];\n}\n\n/**\n * Resolve a schematic doc into positioned drawables. `pos` (inches) becomes a\n * fraction of the module length; endplate A = 0, B = length; turnouts sit at\n * their pos. Tracks may carry explicit fromPos/toPos (overriding node lookup).\n * To-scale: a feature renders at its true position, clamped only to the\n * module's own extent — so signals near an end read at their real spot, not\n * bunched at an inset (#122).\n */\nexport function moduleFeatures(doc: ModuleSchematicDoc): ModuleFeatures {\n const len =\n doc.lengthInches && doc.lengthInches > 0\n ? doc.lengthInches\n : Math.max(\n 1,\n ...doc.tracks.map((t) => Math.max(t.fromPos ?? 0, t.toPos ?? 0)),\n ...(doc.turnouts ?? []).map((t) => t.pos),\n );\n const clampFrac = (p: number) => Math.min(1, Math.max(0, p / len));\n\n const trackLane = new Map<string, number>();\n for (const t of doc.tracks) trackLane.set(t.id, t.lane);\n\n // Endplate positions: first endplate = West (0), the rest = East (len).\n const endplatePos = new Map<string, number>();\n doc.endplates.forEach((e, i) => endplatePos.set(e.id, i === 0 ? 0 : len));\n const turnoutPos = new Map<string, number>();\n for (const t of doc.turnouts ?? []) turnoutPos.set(t.id, t.pos);\n const posOf = (nodeId?: string): number | null => {\n if (nodeId == null) return null;\n if (endplatePos.has(nodeId)) return endplatePos.get(nodeId)!;\n if (turnoutPos.has(nodeId)) return turnoutPos.get(nodeId)!;\n return null;\n };\n\n const doubleMain = doc.endplates.some((e) =>\n e.tracks?.some((t) => t.config === \"double\"),\n );\n\n const extraTracks: DrawTrack[] = [];\n for (const t of doc.tracks) {\n if (t.role === \"main\") continue; // the spine draws mains\n const from = t.fromPos ?? posOf(t.from);\n const to = t.toPos ?? posOf(t.to);\n if (from == null || to == null) continue; // can't place it\n extraTracks.push({\n id: t.id,\n role: t.role,\n lane: t.lane,\n fromFrac: clampFrac(Math.min(from, to)),\n toFrac: clampFrac(Math.max(from, to)),\n capacityFeet: t.capacityFeet ?? null,\n });\n }\n\n const turnouts: DrawTurnout[] = (doc.turnouts ?? []).map((t) => ({\n id: t.id,\n name: t.name ?? \"\",\n posFrac: clampFrac(t.pos),\n onLane: trackLane.get(t.onTrack) ?? 0,\n divergeLane: trackLane.get(t.divergeTrack) ?? 1,\n }));\n\n const drawSignal = (s: SchematicSignal, name: string): DrawSignal => ({\n id: s.id,\n name,\n posFrac: clampFrac(s.pos),\n lane: s.track ? (trackLane.get(s.track) ?? 0) : 0,\n facing: (s.facing as SignalFacing) ?? \"AtoB\",\n side: s.side === \"below\" ? \"below\" : \"above\",\n });\n // Signals come from control-point groups; fall back to pre-grouping flat\n // signals for docs authored before the model changed.\n const signals: DrawSignal[] = Array.isArray(doc.controlPoints)\n ? doc.controlPoints.flatMap((c) =>\n (c.signals ?? []).map((s) => drawSignal(s, c.name ?? \"\")),\n )\n : (doc.signals ?? []).map((s) => drawSignal(s, s.name ?? \"\"));\n\n return { doubleMain, extraTracks, turnouts, signals };\n}\n"]}
package/package.json ADDED
@@ -0,0 +1,52 @@
1
+ {
2
+ "name": "@willcgage/module-schematic",
3
+ "version": "0.1.0",
4
+ "description": "Shared FreeMo module operations-schematic (track-graph): doc types, parser, feature resolver, N-scale helpers, and the editor state machine. Authored by the Module Repository, imported by Free-Dispatcher.",
5
+ "keywords": [
6
+ "freemo",
7
+ "free-mo",
8
+ "model-railroad",
9
+ "dispatcher",
10
+ "ctc",
11
+ "schematic",
12
+ "track-graph"
13
+ ],
14
+ "license": "MIT",
15
+ "author": "Will Gage",
16
+ "repository": {
17
+ "type": "git",
18
+ "url": "git+https://github.com/willcgage/module-schematic.git"
19
+ },
20
+ "type": "module",
21
+ "main": "./dist/index.cjs",
22
+ "module": "./dist/index.js",
23
+ "types": "./dist/index.d.ts",
24
+ "exports": {
25
+ ".": {
26
+ "types": "./dist/index.d.ts",
27
+ "import": "./dist/index.js",
28
+ "require": "./dist/index.cjs"
29
+ }
30
+ },
31
+ "files": [
32
+ "dist"
33
+ ],
34
+ "sideEffects": false,
35
+ "scripts": {
36
+ "build": "tsup",
37
+ "typecheck": "tsc --noEmit",
38
+ "test": "vitest run",
39
+ "prepublishOnly": "npm run build"
40
+ },
41
+ "devDependencies": {
42
+ "tsup": "^8.3.5",
43
+ "typescript": "^5.7.2",
44
+ "vitest": "^2.1.8"
45
+ },
46
+ "publishConfig": {
47
+ "access": "public"
48
+ },
49
+ "engines": {
50
+ "node": ">=18"
51
+ }
52
+ }