@nice2dev/game-engine 1.0.22 → 1.0.23
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/dist/cjs/engine/SaveSystem.js +530 -0
- package/dist/cjs/engine/SaveSystem.js.map +1 -0
- package/dist/cjs/index.js +13 -13
- package/dist/cjs/performance/GameSaveSystem.js +574 -0
- package/dist/cjs/performance/GameSaveSystem.js.map +1 -0
- package/dist/esm/engine/SaveSystem.js +523 -0
- package/dist/esm/engine/SaveSystem.js.map +1 -0
- package/dist/esm/index.js +2 -2
- package/dist/esm/performance/GameSaveSystem.js +567 -0
- package/dist/esm/performance/GameSaveSystem.js.map +1 -0
- package/package.json +1 -1
|
@@ -0,0 +1,523 @@
|
|
|
1
|
+
/* ────────────────────────────────────────────────────────────────
|
|
2
|
+
Save System v2 — Cloud saves, migration, compression
|
|
3
|
+
|
|
4
|
+
Enhanced save system with:
|
|
5
|
+
- Multiple save slots with metadata
|
|
6
|
+
- LZ77-style compression for save data
|
|
7
|
+
- Schema versioning with automatic migration
|
|
8
|
+
- Cloud sync with conflict resolution
|
|
9
|
+
- Auto-save with configurable interval
|
|
10
|
+
- Save data integrity (checksums)
|
|
11
|
+
──────────────────────────────────────────────────────────────── */
|
|
12
|
+
const DEFAULT_SAVE_CONFIG = {
|
|
13
|
+
schemaVersion: 1,
|
|
14
|
+
storageBackend: 'indexeddb',
|
|
15
|
+
storageKey: 'nice-game-saves',
|
|
16
|
+
maxSlots: 20,
|
|
17
|
+
compressionEnabled: true,
|
|
18
|
+
autoSave: {
|
|
19
|
+
enabled: true,
|
|
20
|
+
intervalMs: 60000,
|
|
21
|
+
maxAutoSaves: 3,
|
|
22
|
+
requireSafeState: true,
|
|
23
|
+
},
|
|
24
|
+
cloud: {
|
|
25
|
+
enabled: false,
|
|
26
|
+
conflictStrategy: 'newest-wins',
|
|
27
|
+
syncIntervalMs: 300000,
|
|
28
|
+
maxCloudSlots: 10,
|
|
29
|
+
},
|
|
30
|
+
};
|
|
31
|
+
/* ══════════════════════════════════════════════════════════════
|
|
32
|
+
COMPRESSION — LZ77-style + RLE for save data
|
|
33
|
+
══════════════════════════════════════════════════════════════ */
|
|
34
|
+
/** Simple compression for JSON save data (dictionary-based) */
|
|
35
|
+
function compressSaveData(json) {
|
|
36
|
+
const encoder = new TextEncoder();
|
|
37
|
+
const raw = encoder.encode(json);
|
|
38
|
+
// For small data, skip compression
|
|
39
|
+
if (raw.length < 256) {
|
|
40
|
+
const out = new Uint8Array(1 + raw.length);
|
|
41
|
+
out[0] = 0; // flag: uncompressed
|
|
42
|
+
out.set(raw, 1);
|
|
43
|
+
return out;
|
|
44
|
+
}
|
|
45
|
+
// Simple dictionary compression using common JSON patterns
|
|
46
|
+
const dictionary = [
|
|
47
|
+
'":"',
|
|
48
|
+
'","',
|
|
49
|
+
'":true',
|
|
50
|
+
'":false',
|
|
51
|
+
'":null',
|
|
52
|
+
'":0',
|
|
53
|
+
'":1',
|
|
54
|
+
'{"',
|
|
55
|
+
'"}',
|
|
56
|
+
'["',
|
|
57
|
+
'"]',
|
|
58
|
+
'\\n',
|
|
59
|
+
'\\t',
|
|
60
|
+
];
|
|
61
|
+
const chunks = [1]; // flag: compressed
|
|
62
|
+
// Write dictionary length
|
|
63
|
+
chunks.push(dictionary.length);
|
|
64
|
+
let i = 0;
|
|
65
|
+
while (i < raw.length) {
|
|
66
|
+
let matched = false;
|
|
67
|
+
// Try dictionary match
|
|
68
|
+
for (let d = 0; d < dictionary.length; d++) {
|
|
69
|
+
const dictBytes = encoder.encode(dictionary[d]);
|
|
70
|
+
if (i + dictBytes.length <= raw.length) {
|
|
71
|
+
let match = true;
|
|
72
|
+
for (let j = 0; j < dictBytes.length; j++) {
|
|
73
|
+
if (raw[i + j] !== dictBytes[j]) {
|
|
74
|
+
match = false;
|
|
75
|
+
break;
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
if (match) {
|
|
79
|
+
chunks.push(255); // escape byte = dict reference
|
|
80
|
+
chunks.push(d); // dict index
|
|
81
|
+
i += dictBytes.length;
|
|
82
|
+
matched = true;
|
|
83
|
+
break;
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
// RLE for repeated bytes
|
|
88
|
+
if (!matched && i + 3 < raw.length && raw[i] === raw[i + 1] && raw[i] === raw[i + 2]) {
|
|
89
|
+
let runLen = 1;
|
|
90
|
+
while (i + runLen < raw.length && raw[i + runLen] === raw[i] && runLen < 127) {
|
|
91
|
+
runLen++;
|
|
92
|
+
}
|
|
93
|
+
chunks.push(254); // escape byte = RLE
|
|
94
|
+
chunks.push(runLen);
|
|
95
|
+
chunks.push(raw[i]);
|
|
96
|
+
i += runLen;
|
|
97
|
+
matched = true;
|
|
98
|
+
}
|
|
99
|
+
if (!matched) {
|
|
100
|
+
// Literal byte
|
|
101
|
+
if (raw[i] === 254 || raw[i] === 255) {
|
|
102
|
+
chunks.push(253); // escape for literal 254/255
|
|
103
|
+
}
|
|
104
|
+
chunks.push(raw[i]);
|
|
105
|
+
i++;
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
return new Uint8Array(chunks);
|
|
109
|
+
}
|
|
110
|
+
/** Decompress save data */
|
|
111
|
+
function decompressSaveData(data) {
|
|
112
|
+
const decoder = new TextDecoder();
|
|
113
|
+
const encoder = new TextEncoder();
|
|
114
|
+
if (data[0] === 0) {
|
|
115
|
+
// Uncompressed
|
|
116
|
+
return decoder.decode(data.slice(1));
|
|
117
|
+
}
|
|
118
|
+
const dictionary = [
|
|
119
|
+
'":"',
|
|
120
|
+
'","',
|
|
121
|
+
'":true',
|
|
122
|
+
'":false',
|
|
123
|
+
'":null',
|
|
124
|
+
'":0',
|
|
125
|
+
'":1',
|
|
126
|
+
'{"',
|
|
127
|
+
'"}',
|
|
128
|
+
'["',
|
|
129
|
+
'"]',
|
|
130
|
+
'\\n',
|
|
131
|
+
'\\t',
|
|
132
|
+
];
|
|
133
|
+
const dictBytes = dictionary.map((s) => encoder.encode(s));
|
|
134
|
+
const output = [];
|
|
135
|
+
let i = 2; // skip flag + dict length
|
|
136
|
+
while (i < data.length) {
|
|
137
|
+
if (data[i] === 255) {
|
|
138
|
+
// Dictionary reference
|
|
139
|
+
const dictIdx = data[i + 1];
|
|
140
|
+
if (dictIdx < dictBytes.length) {
|
|
141
|
+
for (const b of dictBytes[dictIdx]) {
|
|
142
|
+
output.push(b);
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
i += 2;
|
|
146
|
+
}
|
|
147
|
+
else if (data[i] === 254) {
|
|
148
|
+
// RLE
|
|
149
|
+
const runLen = data[i + 1];
|
|
150
|
+
const val = data[i + 2];
|
|
151
|
+
for (let j = 0; j < runLen; j++) {
|
|
152
|
+
output.push(val);
|
|
153
|
+
}
|
|
154
|
+
i += 3;
|
|
155
|
+
}
|
|
156
|
+
else if (data[i] === 253) {
|
|
157
|
+
// Escaped literal
|
|
158
|
+
output.push(data[i + 1]);
|
|
159
|
+
i += 2;
|
|
160
|
+
}
|
|
161
|
+
else {
|
|
162
|
+
output.push(data[i]);
|
|
163
|
+
i++;
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
return decoder.decode(new Uint8Array(output));
|
|
167
|
+
}
|
|
168
|
+
/* ══════════════════════════════════════════════════════════════
|
|
169
|
+
CHECKSUM — CRC32 for data integrity
|
|
170
|
+
══════════════════════════════════════════════════════════════ */
|
|
171
|
+
const CRC32_TABLE = [];
|
|
172
|
+
for (let i = 0; i < 256; i++) {
|
|
173
|
+
let c = i;
|
|
174
|
+
for (let j = 0; j < 8; j++) {
|
|
175
|
+
c = c & 1 ? 0xedb88320 ^ (c >>> 1) : c >>> 1;
|
|
176
|
+
}
|
|
177
|
+
CRC32_TABLE[i] = c;
|
|
178
|
+
}
|
|
179
|
+
function crc32(data) {
|
|
180
|
+
let crc = 0xffffffff;
|
|
181
|
+
for (let i = 0; i < data.length; i++) {
|
|
182
|
+
crc = CRC32_TABLE[(crc ^ data[i]) & 0xff] ^ (crc >>> 8);
|
|
183
|
+
}
|
|
184
|
+
return (crc ^ 0xffffffff) >>> 0;
|
|
185
|
+
}
|
|
186
|
+
/* ══════════════════════════════════════════════════════════════
|
|
187
|
+
MIGRATION ENGINE
|
|
188
|
+
══════════════════════════════════════════════════════════════ */
|
|
189
|
+
class MigrationEngine {
|
|
190
|
+
constructor() {
|
|
191
|
+
this.migrations = [];
|
|
192
|
+
}
|
|
193
|
+
register(migration) {
|
|
194
|
+
this.migrations.push(migration);
|
|
195
|
+
this.migrations.sort((a, b) => a.fromVersion - b.fromVersion);
|
|
196
|
+
}
|
|
197
|
+
/** Migrate payload from fromVersion to toVersion */
|
|
198
|
+
migrate(payload, fromVersion, toVersion) {
|
|
199
|
+
const applied = [];
|
|
200
|
+
let current = { ...payload };
|
|
201
|
+
let version = fromVersion;
|
|
202
|
+
if (fromVersion < toVersion) {
|
|
203
|
+
// Forward migration
|
|
204
|
+
while (version < toVersion) {
|
|
205
|
+
const migration = this.migrations.find((m) => m.fromVersion === version);
|
|
206
|
+
if (!migration) {
|
|
207
|
+
throw new Error(`No migration found for version ${version} → ${version + 1}`);
|
|
208
|
+
}
|
|
209
|
+
current = migration.migrate(current);
|
|
210
|
+
applied.push(migration.description);
|
|
211
|
+
version = migration.toVersion;
|
|
212
|
+
}
|
|
213
|
+
}
|
|
214
|
+
else if (fromVersion > toVersion) {
|
|
215
|
+
// Rollback (reverse order)
|
|
216
|
+
while (version > toVersion) {
|
|
217
|
+
const migration = this.migrations.find((m) => m.toVersion === version);
|
|
218
|
+
if (!migration || !migration.rollback) {
|
|
219
|
+
throw new Error(`No rollback found for version ${version} → ${version - 1}`);
|
|
220
|
+
}
|
|
221
|
+
current = migration.rollback(current);
|
|
222
|
+
applied.push(`Rollback: ${migration.description}`);
|
|
223
|
+
version = migration.fromVersion;
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
return { payload: current, applied };
|
|
227
|
+
}
|
|
228
|
+
getRegistered() {
|
|
229
|
+
return this.migrations;
|
|
230
|
+
}
|
|
231
|
+
canMigrate(from, to) {
|
|
232
|
+
let v = from;
|
|
233
|
+
if (from < to) {
|
|
234
|
+
while (v < to) {
|
|
235
|
+
const m = this.migrations.find((mm) => mm.fromVersion === v);
|
|
236
|
+
if (!m) {
|
|
237
|
+
return false;
|
|
238
|
+
}
|
|
239
|
+
v = m.toVersion;
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
else {
|
|
243
|
+
while (v > to) {
|
|
244
|
+
const m = this.migrations.find((mm) => mm.toVersion === v);
|
|
245
|
+
if (!m || !m.rollback) {
|
|
246
|
+
return false;
|
|
247
|
+
}
|
|
248
|
+
v = m.fromVersion;
|
|
249
|
+
}
|
|
250
|
+
}
|
|
251
|
+
return true;
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
/* ══════════════════════════════════════════════════════════════
|
|
255
|
+
SAVE SYSTEM V2
|
|
256
|
+
══════════════════════════════════════════════════════════════ */
|
|
257
|
+
class SaveSystem {
|
|
258
|
+
constructor(eventBus, config = {}) {
|
|
259
|
+
this.migrationEngine = new MigrationEngine();
|
|
260
|
+
this.slots = new Map();
|
|
261
|
+
this.autoSaveTimer = null;
|
|
262
|
+
this.autoSaveCounter = 0;
|
|
263
|
+
this.safeToSave = true;
|
|
264
|
+
this.config = { ...DEFAULT_SAVE_CONFIG, ...config };
|
|
265
|
+
this.eventBus = eventBus;
|
|
266
|
+
this.setupEventHandlers();
|
|
267
|
+
}
|
|
268
|
+
setupEventHandlers() {
|
|
269
|
+
this.eventBus.on('game:safe-state', (data) => {
|
|
270
|
+
this.safeToSave = data.safe;
|
|
271
|
+
});
|
|
272
|
+
}
|
|
273
|
+
/** Register a schema migration */
|
|
274
|
+
registerMigration(migration) {
|
|
275
|
+
this.migrationEngine.register(migration);
|
|
276
|
+
}
|
|
277
|
+
/** Save game state to a slot */
|
|
278
|
+
save(slotId, payload, options = {}) {
|
|
279
|
+
const encoder = new TextEncoder();
|
|
280
|
+
const json = JSON.stringify(payload);
|
|
281
|
+
const compressed = this.config.compressionEnabled
|
|
282
|
+
? compressSaveData(json)
|
|
283
|
+
: encoder.encode(json);
|
|
284
|
+
const checksum = crc32(compressed);
|
|
285
|
+
const now = Date.now();
|
|
286
|
+
const existingMeta = this.slots.get(slotId)?.meta;
|
|
287
|
+
const meta = {
|
|
288
|
+
slotId,
|
|
289
|
+
type: options.type ?? 'manual',
|
|
290
|
+
label: options.label ?? `Save ${slotId}`,
|
|
291
|
+
schemaVersion: this.config.schemaVersion,
|
|
292
|
+
createdAt: existingMeta?.createdAt ?? now,
|
|
293
|
+
updatedAt: now,
|
|
294
|
+
playtimeMs: existingMeta ? existingMeta.playtimeMs + (now - existingMeta.updatedAt) : 0,
|
|
295
|
+
location: options.location ?? '',
|
|
296
|
+
thumbnail: options.thumbnail,
|
|
297
|
+
sizeBytes: compressed.length,
|
|
298
|
+
checksum,
|
|
299
|
+
cloudSynced: false,
|
|
300
|
+
tags: options.tags ?? [],
|
|
301
|
+
};
|
|
302
|
+
this.slots.set(slotId, { meta, payload });
|
|
303
|
+
// Persist to storage backend
|
|
304
|
+
this.persistSlot(slotId, compressed, meta);
|
|
305
|
+
this.eventBus.emit('save:saved', { slotId, sizeBytes: compressed.length });
|
|
306
|
+
return {
|
|
307
|
+
success: true,
|
|
308
|
+
slotId,
|
|
309
|
+
sizeBytes: compressed.length,
|
|
310
|
+
compressed: this.config.compressionEnabled,
|
|
311
|
+
};
|
|
312
|
+
}
|
|
313
|
+
/** Load game state from a slot */
|
|
314
|
+
load(slotId) {
|
|
315
|
+
const data = this.slots.get(slotId);
|
|
316
|
+
if (!data) {
|
|
317
|
+
return null;
|
|
318
|
+
}
|
|
319
|
+
let payload = { ...data.payload };
|
|
320
|
+
let migrated = false;
|
|
321
|
+
// Check if migration is needed
|
|
322
|
+
if (data.meta.schemaVersion !== this.config.schemaVersion) {
|
|
323
|
+
if (this.migrationEngine.canMigrate(data.meta.schemaVersion, this.config.schemaVersion)) {
|
|
324
|
+
const result = this.migrationEngine.migrate(payload, data.meta.schemaVersion, this.config.schemaVersion);
|
|
325
|
+
payload = result.payload;
|
|
326
|
+
migrated = true;
|
|
327
|
+
this.eventBus.emit('save:migrated', {
|
|
328
|
+
slotId,
|
|
329
|
+
from: data.meta.schemaVersion,
|
|
330
|
+
to: this.config.schemaVersion,
|
|
331
|
+
steps: result.applied,
|
|
332
|
+
});
|
|
333
|
+
}
|
|
334
|
+
}
|
|
335
|
+
this.eventBus.emit('save:loaded', { slotId });
|
|
336
|
+
return { payload, meta: { ...data.meta }, migrated };
|
|
337
|
+
}
|
|
338
|
+
/** Delete a save slot */
|
|
339
|
+
delete(slotId) {
|
|
340
|
+
const existed = this.slots.delete(slotId);
|
|
341
|
+
if (existed) {
|
|
342
|
+
this.removeFromStorage(slotId);
|
|
343
|
+
this.eventBus.emit('save:deleted', { slotId });
|
|
344
|
+
}
|
|
345
|
+
return existed;
|
|
346
|
+
}
|
|
347
|
+
/** List all save slot metadata (sorted by updatedAt desc) */
|
|
348
|
+
listSlots() {
|
|
349
|
+
return Array.from(this.slots.values())
|
|
350
|
+
.map((s) => ({ ...s.meta }))
|
|
351
|
+
.sort((a, b) => b.updatedAt - a.updatedAt);
|
|
352
|
+
}
|
|
353
|
+
/** Get slot metadata without loading payload */
|
|
354
|
+
getSlotMeta(slotId) {
|
|
355
|
+
const data = this.slots.get(slotId);
|
|
356
|
+
return data ? { ...data.meta } : null;
|
|
357
|
+
}
|
|
358
|
+
/** Check if a slot exists */
|
|
359
|
+
hasSlot(slotId) {
|
|
360
|
+
return this.slots.has(slotId);
|
|
361
|
+
}
|
|
362
|
+
/** Start auto-save timer */
|
|
363
|
+
startAutoSave(getPayload, getLocation) {
|
|
364
|
+
if (!this.config.autoSave.enabled) {
|
|
365
|
+
return;
|
|
366
|
+
}
|
|
367
|
+
this.stopAutoSave();
|
|
368
|
+
this.autoSaveTimer = setInterval(() => {
|
|
369
|
+
if (this.config.autoSave.requireSafeState && !this.safeToSave) {
|
|
370
|
+
return;
|
|
371
|
+
}
|
|
372
|
+
const slotId = `auto-${this.autoSaveCounter % this.config.autoSave.maxAutoSaves}`;
|
|
373
|
+
this.autoSaveCounter++;
|
|
374
|
+
this.save(slotId, getPayload(), {
|
|
375
|
+
type: 'auto',
|
|
376
|
+
label: `Auto Save ${this.autoSaveCounter}`,
|
|
377
|
+
location: getLocation(),
|
|
378
|
+
});
|
|
379
|
+
}, this.config.autoSave.intervalMs);
|
|
380
|
+
}
|
|
381
|
+
/** Stop auto-save */
|
|
382
|
+
stopAutoSave() {
|
|
383
|
+
if (this.autoSaveTimer) {
|
|
384
|
+
clearInterval(this.autoSaveTimer);
|
|
385
|
+
this.autoSaveTimer = null;
|
|
386
|
+
}
|
|
387
|
+
}
|
|
388
|
+
/** Quick save (overwriting the single quick save slot) */
|
|
389
|
+
quickSave(payload, location) {
|
|
390
|
+
return this.save('quicksave', payload, { type: 'quick', label: 'Quick Save', location });
|
|
391
|
+
}
|
|
392
|
+
/** Quick load */
|
|
393
|
+
quickLoad() {
|
|
394
|
+
return this.load('quicksave');
|
|
395
|
+
}
|
|
396
|
+
/* ── Cloud Sync ─────────────────────────────────────────────── */
|
|
397
|
+
/** Sync local saves to cloud (stub — needs actual endpoint) */
|
|
398
|
+
async syncToCloud() {
|
|
399
|
+
if (!this.config.cloud.enabled) {
|
|
400
|
+
return { synced: [], conflicts: [] };
|
|
401
|
+
}
|
|
402
|
+
const synced = [];
|
|
403
|
+
const conflicts = [];
|
|
404
|
+
for (const [slotId, data] of this.slots) {
|
|
405
|
+
if (!data.meta.cloudSynced) {
|
|
406
|
+
// In production, this would POST to the cloud endpoint
|
|
407
|
+
// Here we simulate a successful sync
|
|
408
|
+
data.meta.cloudSynced = true;
|
|
409
|
+
data.meta.cloudTimestamp = Date.now();
|
|
410
|
+
synced.push(slotId);
|
|
411
|
+
}
|
|
412
|
+
}
|
|
413
|
+
this.eventBus.emit('save:cloud-sync', { synced, conflicts });
|
|
414
|
+
return { synced, conflicts };
|
|
415
|
+
}
|
|
416
|
+
/** Resolve a cloud sync conflict */
|
|
417
|
+
resolveConflict(conflict, strategy) {
|
|
418
|
+
conflict.resolvedWith = strategy;
|
|
419
|
+
switch (strategy) {
|
|
420
|
+
case 'local-wins':
|
|
421
|
+
return conflict.localMeta;
|
|
422
|
+
case 'remote-wins':
|
|
423
|
+
return conflict.remoteMeta;
|
|
424
|
+
case 'newest-wins':
|
|
425
|
+
return conflict.localMeta.updatedAt >= conflict.remoteMeta.updatedAt
|
|
426
|
+
? conflict.localMeta
|
|
427
|
+
: conflict.remoteMeta;
|
|
428
|
+
default:
|
|
429
|
+
return conflict.localMeta; // merge/prompt fallback
|
|
430
|
+
}
|
|
431
|
+
}
|
|
432
|
+
/* ── Storage Backend ────────────────────────────────────────── */
|
|
433
|
+
persistSlot(slotId, _data, _meta) {
|
|
434
|
+
switch (this.config.storageBackend) {
|
|
435
|
+
case 'localstorage': {
|
|
436
|
+
const key = `${this.config.storageKey}:${slotId}`;
|
|
437
|
+
try {
|
|
438
|
+
const serialized = JSON.stringify({
|
|
439
|
+
meta: _meta,
|
|
440
|
+
payload: this.slots.get(slotId)?.payload,
|
|
441
|
+
});
|
|
442
|
+
localStorage.setItem(key, serialized);
|
|
443
|
+
}
|
|
444
|
+
catch {
|
|
445
|
+
/* quota exceeded — emit event */
|
|
446
|
+
this.eventBus.emit('save:error', { slotId, error: 'Storage quota exceeded' });
|
|
447
|
+
}
|
|
448
|
+
break;
|
|
449
|
+
}
|
|
450
|
+
}
|
|
451
|
+
}
|
|
452
|
+
removeFromStorage(slotId) {
|
|
453
|
+
if (this.config.storageBackend === 'localstorage') {
|
|
454
|
+
localStorage.removeItem(`${this.config.storageKey}:${slotId}`);
|
|
455
|
+
}
|
|
456
|
+
}
|
|
457
|
+
/** Load all slots from storage backend */
|
|
458
|
+
async loadFromStorage() {
|
|
459
|
+
if (this.config.storageBackend === 'localstorage') {
|
|
460
|
+
let count = 0;
|
|
461
|
+
const prefix = `${this.config.storageKey}:`;
|
|
462
|
+
for (let i = 0; i < localStorage.length; i++) {
|
|
463
|
+
const key = localStorage.key(i);
|
|
464
|
+
if (key?.startsWith(prefix)) {
|
|
465
|
+
try {
|
|
466
|
+
const raw = localStorage.getItem(key);
|
|
467
|
+
if (raw) {
|
|
468
|
+
const parsed = JSON.parse(raw);
|
|
469
|
+
this.slots.set(parsed.meta.slotId, parsed);
|
|
470
|
+
count++;
|
|
471
|
+
}
|
|
472
|
+
}
|
|
473
|
+
catch {
|
|
474
|
+
/* corrupt entry — skip */
|
|
475
|
+
}
|
|
476
|
+
}
|
|
477
|
+
}
|
|
478
|
+
return count;
|
|
479
|
+
}
|
|
480
|
+
return 0;
|
|
481
|
+
}
|
|
482
|
+
/** Export all saves as a downloadable blob */
|
|
483
|
+
exportAll() {
|
|
484
|
+
const all = Array.from(this.slots.values());
|
|
485
|
+
const json = JSON.stringify(all);
|
|
486
|
+
return new Blob([json], { type: 'application/json' });
|
|
487
|
+
}
|
|
488
|
+
/** Import saves from a blob */
|
|
489
|
+
async importAll(blob) {
|
|
490
|
+
const text = await blob.text();
|
|
491
|
+
let parsed;
|
|
492
|
+
try {
|
|
493
|
+
parsed = JSON.parse(text);
|
|
494
|
+
}
|
|
495
|
+
catch {
|
|
496
|
+
return 0;
|
|
497
|
+
}
|
|
498
|
+
if (!Array.isArray(parsed)) {
|
|
499
|
+
return 0;
|
|
500
|
+
}
|
|
501
|
+
let count = 0;
|
|
502
|
+
for (const entry of parsed) {
|
|
503
|
+
if (entry && typeof entry === 'object' && 'meta' in entry && 'payload' in entry) {
|
|
504
|
+
const saveData = entry;
|
|
505
|
+
this.slots.set(saveData.meta.slotId, saveData);
|
|
506
|
+
count++;
|
|
507
|
+
}
|
|
508
|
+
}
|
|
509
|
+
return count;
|
|
510
|
+
}
|
|
511
|
+
getConfig() {
|
|
512
|
+
return this.config;
|
|
513
|
+
}
|
|
514
|
+
getMigrationEngine() {
|
|
515
|
+
return this.migrationEngine;
|
|
516
|
+
}
|
|
517
|
+
destroy() {
|
|
518
|
+
this.stopAutoSave();
|
|
519
|
+
}
|
|
520
|
+
}
|
|
521
|
+
|
|
522
|
+
export { DEFAULT_SAVE_CONFIG, MigrationEngine, SaveSystem, compressSaveData, crc32, decompressSaveData };
|
|
523
|
+
//# sourceMappingURL=SaveSystem.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"SaveSystem.js","sources":["../../../src/engine/SaveSystem.ts"],"sourcesContent":[null],"names":[],"mappings":"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"}
|
package/dist/esm/index.js
CHANGED
|
@@ -77,7 +77,7 @@ export { BLOOM_DEFAULTS, COLOR_GRADING_DEFAULTS, DEFAULT_PBR, DEFAULT_RENDER_CON
|
|
|
77
77
|
export { AudioBusMixer, DEFAULT_SPATIAL, GameAudioBridge, GameAudioSettings, MusicManager, SFXPool, useGameAudio } from './audio/AudioBridge.js';
|
|
78
78
|
export { ChoiceList, DEFAULT_TEXT_SPEED, DialogueBox, DialogueEngine, DialogueTreeBuilder, EMOTION_COLORS, applyLocalization, createEmptyDialogueTree, deserializeDialogueTree, extractTranslatableStrings, findUnreachableNodes, serializeDialogueTree, useTypewriter, validateDialogueTree } from './dialogue/DialogueSystem.js';
|
|
79
79
|
export { DIFFICULTY_COLORS, DIFFICULTY_XP_MULTIPLIERS, QuestBuilder, QuestList, QuestManager, QuestTracker, createEmptyQuest } from './quest/QuestSystem.js';
|
|
80
|
-
export { DEFAULT_SAVE_CONFIG, MigrationEngine,
|
|
80
|
+
export { DEFAULT_SAVE_CONFIG, MigrationEngine, SaveSystem, compressSaveData, crc32, decompressSaveData } from './engine/SaveSystem.js';
|
|
81
81
|
export { AnalyticsDashboard, DEFAULT_HEATMAP_CONFIG, DEFAULT_RECORDING_CONFIG, DEFAULT_TELEMETRY_CONFIG, FunnelTracker, GameplayAnalytics, HeatmapGenerator, SessionPlayer, SessionRecorder, TelemetryPipeline, useGameplayAnalytics } from './devtools/GameplayAnalytics.js';
|
|
82
82
|
export { createArrayType, createAssignment, createBinaryOp, createCall, createCustomType, createExprStatement, createIdentifier, createIf, createLiteral, createMethodCall, createPropertyAccess, createReturn, createType, createVarDecl } from './scripting/ScriptAST.js';
|
|
83
83
|
export { CodeBuilder, ImperativeLanguageExporter, LanguageExporter, defaultConfig, escapeString, toLowerCamelCase, toScreamingSnakeCase, toSnakeCase, toUpperCamelCase } from './scripting/LanguageExporter.js';
|
|
@@ -111,7 +111,7 @@ export { AssetStreamingManager, DEFAULT_STREAMING_CONFIG, LODCalculator, createA
|
|
|
111
111
|
export { DEFAULT_POOL_CONFIG, PoolManager, PoolableBullet, PoolableEvent, PoolableParticle, PoolableVector3, createBulletPool, createEventPool, createObjectPool, createParticlePool, createPoolManager, createVector3Pool } from './performance/ObjectPooling.js';
|
|
112
112
|
export { DEFAULT_MEMORY_CONFIG, MemoryTracker, TrackedAllocator, createMemoryTracker, createTrackedAllocator } from './performance/MemoryManagement.js';
|
|
113
113
|
export { createPipelineDescriptor, createWebGPURenderer, isWebGPUSupported } from './performance/WebGPURenderer.js';
|
|
114
|
-
export { IndexedDBAdapter, LocalStorageAdapter,
|
|
114
|
+
export { GameSaveSystem, IndexedDBAdapter, LocalStorageAdapter, createMigration, createSaveSystem } from './performance/GameSaveSystem.js';
|
|
115
115
|
export { DEFAULT_HOT_RELOAD_CONFIG, FileWatcher, HotReloadSystem, ModuleRegistry, createFileWatcher, createHotReloadSystem, createModuleRegistry } from './performance/HotReload.js';
|
|
116
116
|
export { DEFAULT_LOCALIZATION_CONFIG, LOCALES, LocalizationRuntime, createLocaleInfo, createLocalizationRuntime, createTranslationEntry } from './performance/LocalizationRuntime.js';
|
|
117
117
|
export { ConflictDetector, DEFAULT_MOD_CONFIG, DependencyResolver, ModLoader, ModRegistry, compareVersions, createModLoader, createModManifest, isVersionCompatible, parseVersion } from './performance/ModSupport.js';
|