|
|
@@ -7,7 +7,7 @@ import { v4 as uuid } from 'uuid'
|
|
7
|
7
|
const CURRENT_VERSION = 'code_slate_0.0.5'
|
|
8
|
8
|
const DOCUMENT_ID = '0001'
|
|
9
|
9
|
|
|
10
|
|
-function storageId(label: string, fileId: string, id: string) {
|
|
|
10
|
+function storageId(fileId: string, label: string, id: string) {
|
|
11
|
11
|
return `${CURRENT_VERSION}_doc_${fileId}_${label}_${id}`
|
|
12
|
12
|
}
|
|
13
|
13
|
|
|
|
@@ -17,17 +17,22 @@ class Storage {
|
|
17
|
17
|
firstLoad(data: Data) {
|
|
18
|
18
|
const lastOpened = localStorage.getItem(`${CURRENT_VERSION}_lastOpened`)
|
|
19
|
19
|
this.loadDocumentFromLocalStorage(data, lastOpened || DOCUMENT_ID)
|
|
|
20
|
+ this.loadPage(data, data.currentPageId)
|
|
|
21
|
+ this.saveToLocalStorage(data, data.document.id)
|
|
|
22
|
+ localStorage.setItem(`${CURRENT_VERSION}_lastOpened`, data.document.id)
|
|
20
|
23
|
}
|
|
21
|
24
|
|
|
22
|
25
|
load(data: Data, restoredData: any) {
|
|
23
|
|
- // Empty shapes in state for each page
|
|
|
26
|
+ console.log('loading restored data', restoredData)
|
|
|
27
|
+
|
|
|
28
|
+ // Before loading the state, save the pages / page states
|
|
24
|
29
|
for (let key in restoredData.document.pages) {
|
|
25
|
|
- restoredData.document.pages[key].shapes = {}
|
|
|
30
|
+ this.savePage(restoredData, restoredData.document.id, key)
|
|
26
|
31
|
}
|
|
27
|
32
|
|
|
28
|
|
- // Empty page states for each page
|
|
29
|
|
- for (let key in restoredData.pageStates) {
|
|
30
|
|
- restoredData.document.pages[key].shapes = {}
|
|
|
33
|
+ // Empty shapes in state for each page
|
|
|
34
|
+ for (let key in restoredData.document.pages) {
|
|
|
35
|
+ // restoredData.document.pages[key].shapes = {}
|
|
31
|
36
|
}
|
|
32
|
37
|
|
|
33
|
38
|
data.document = {} as TLDocument
|
|
|
@@ -42,36 +47,6 @@ class Storage {
|
|
42
|
47
|
name: 'My Document',
|
|
43
|
48
|
...restoredData.document,
|
|
44
|
49
|
}
|
|
45
|
|
-
|
|
46
|
|
- localStorage.setItem(`${CURRENT_VERSION}_lastOpened`, data.document.id)
|
|
47
|
|
-
|
|
48
|
|
- console.log('loaded', data.document)
|
|
49
|
|
-
|
|
50
|
|
- // Let's also save the loaded document to local storage
|
|
51
|
|
- this.saveToLocalStorage(data, data.document.id)
|
|
52
|
|
-
|
|
53
|
|
- // Load current page
|
|
54
|
|
- this.loadPage(data, data.currentPageId)
|
|
55
|
|
- }
|
|
56
|
|
-
|
|
57
|
|
- async loadDocumentFromFilesystem() {
|
|
58
|
|
- const blob = await fa.fileOpen({
|
|
59
|
|
- description: 'tldraw files',
|
|
60
|
|
- })
|
|
61
|
|
-
|
|
62
|
|
- const text = await getTextFromBlob(blob)
|
|
63
|
|
-
|
|
64
|
|
- const restoredData = JSON.parse(text)
|
|
65
|
|
-
|
|
66
|
|
- if (restoredData === null) {
|
|
67
|
|
- console.warn('Could not load that data.')
|
|
68
|
|
- return
|
|
69
|
|
- }
|
|
70
|
|
-
|
|
71
|
|
- // Save blob for future saves
|
|
72
|
|
- this.previousSaveHandle = blob.handle
|
|
73
|
|
-
|
|
74
|
|
- state.send('LOADED_FROM_FILE', { restoredData })
|
|
75
|
50
|
}
|
|
76
|
51
|
|
|
77
|
52
|
loadDocumentFromLocalStorage(data: Data, fileId = DOCUMENT_ID) {
|
|
|
@@ -79,7 +54,9 @@ class Storage {
|
|
79
|
54
|
if (typeof localStorage === 'undefined') return
|
|
80
|
55
|
|
|
81
|
56
|
// Load data from local storage
|
|
82
|
|
- const savedData = localStorage.getItem(fileId)
|
|
|
57
|
+ const savedData = localStorage.getItem(
|
|
|
58
|
+ storageId(fileId, 'document', fileId)
|
|
|
59
|
+ )
|
|
83
|
60
|
|
|
84
|
61
|
if (savedData === null) {
|
|
85
|
62
|
// If we're going to use the default data, assign the
|
|
|
@@ -96,7 +73,8 @@ class Storage {
|
|
96
|
73
|
getDataToSave = (data: Data) => {
|
|
97
|
74
|
const dataToSave: any = { ...data }
|
|
98
|
75
|
|
|
99
|
|
- for (let pageId in data.document) {
|
|
|
76
|
+ for (let pageId in data.document.pages) {
|
|
|
77
|
+ // Page
|
|
100
|
78
|
const savedPage = localStorage.getItem(
|
|
101
|
79
|
storageId(data.document.id, 'page', pageId)
|
|
102
|
80
|
)
|
|
|
@@ -105,9 +83,9 @@ class Storage {
|
|
105
|
83
|
const restored: Page = JSON.parse(savedPage)
|
|
106
|
84
|
dataToSave.document.pages[pageId] = restored
|
|
107
|
85
|
}
|
|
108
|
|
- }
|
|
109
|
86
|
|
|
110
|
|
- dataToSave.pageStates = {}
|
|
|
87
|
+ dataToSave.pageStates = {}
|
|
|
88
|
+ }
|
|
111
|
89
|
|
|
112
|
90
|
return JSON.stringify(dataToSave, null, 2)
|
|
113
|
91
|
}
|
|
|
@@ -116,61 +94,66 @@ class Storage {
|
|
116
|
94
|
if (typeof window === 'undefined') return
|
|
117
|
95
|
if (typeof localStorage === 'undefined') return
|
|
118
|
96
|
|
|
119
|
|
- // Save current data to local storage
|
|
120
|
|
- localStorage.setItem(id, this.getDataToSave(data))
|
|
|
97
|
+ const dataToSave = this.getDataToSave(data)
|
|
121
|
98
|
|
|
122
|
|
- // Save current page too
|
|
123
|
|
- this.savePage(data, id, data.currentPageId)
|
|
|
99
|
+ // Save current data to local storage
|
|
|
100
|
+ localStorage.setItem(storageId(id, 'document', id), dataToSave)
|
|
124
|
101
|
}
|
|
125
|
102
|
|
|
126
|
|
- saveAsToFileSystem = (data: Data) => {
|
|
127
|
|
- // Create a new document id when saving to the file system
|
|
128
|
|
-
|
|
129
|
|
- this.saveToFileSystem(data, uuid())
|
|
|
103
|
+ loadDocumentFromJson(data: Data, restoredData: any) {
|
|
|
104
|
+ this.load(data, restoredData)
|
|
|
105
|
+ this.loadPage(data, data.currentPageId)
|
|
|
106
|
+ this.saveToLocalStorage(data, data.document.id)
|
|
|
107
|
+ localStorage.setItem(`${CURRENT_VERSION}_lastOpened`, data.document.id)
|
|
130
|
108
|
}
|
|
|
109
|
+ /* ---------------------- Pages --------------------- */
|
|
131
|
110
|
|
|
132
|
|
- saveToFileSystem = (data: Data, id?: string) => {
|
|
133
|
|
- // Save to local storage first
|
|
134
|
|
- this.saveToLocalStorage(data, id)
|
|
|
111
|
+ savePage(data: Data, fileId = data.document.id, pageId = data.currentPageId) {
|
|
|
112
|
+ if (typeof window === 'undefined') return
|
|
|
113
|
+ if (typeof localStorage === 'undefined') return
|
|
135
|
114
|
|
|
136
|
|
- const json = this.getDataToSave(data)
|
|
|
115
|
+ // Save page
|
|
|
116
|
+ const page = data.document.pages[pageId]
|
|
|
117
|
+ const json = JSON.stringify(page)
|
|
137
|
118
|
|
|
138
|
|
- const blob = new Blob([json], {
|
|
139
|
|
- type: 'application/vnd.tldraw+json',
|
|
140
|
|
- })
|
|
|
119
|
+ localStorage.setItem(storageId(fileId, 'page', pageId), json)
|
|
141
|
120
|
|
|
142
|
|
- fa.fileSave(
|
|
143
|
|
- blob,
|
|
144
|
|
- {
|
|
145
|
|
- fileName: `${
|
|
146
|
|
- id
|
|
147
|
|
- ? data.document.name
|
|
148
|
|
- : this.previousSaveHandle?.name || 'My Document'
|
|
149
|
|
- }.tldr`,
|
|
150
|
|
- description: 'tldraw file',
|
|
151
|
|
- extensions: ['.tldr'],
|
|
|
121
|
+ // Save page state
|
|
|
122
|
+
|
|
|
123
|
+ let currentPageState = {
|
|
|
124
|
+ camera: {
|
|
|
125
|
+ point: [0, 0],
|
|
|
126
|
+ zoom: 1,
|
|
152
|
127
|
},
|
|
153
|
|
- id ? undefined : this.previousSaveHandle,
|
|
154
|
|
- true
|
|
|
128
|
+ selectedIds: new Set([]),
|
|
|
129
|
+ ...data.pageStates[pageId],
|
|
|
130
|
+ }
|
|
|
131
|
+
|
|
|
132
|
+ const pageState = {
|
|
|
133
|
+ ...currentPageState,
|
|
|
134
|
+ selectedIds: setToArray(currentPageState.selectedIds),
|
|
|
135
|
+ }
|
|
|
136
|
+
|
|
|
137
|
+ localStorage.setItem(
|
|
|
138
|
+ storageId(fileId, 'pageState', pageId),
|
|
|
139
|
+ JSON.stringify(pageState)
|
|
155
|
140
|
)
|
|
156
|
|
- .then((handle) => {
|
|
157
|
|
- this.previousSaveHandle = handle
|
|
158
|
|
- state.send('SAVED_FILE_TO_FILE_SYSTEM')
|
|
159
|
|
- })
|
|
160
|
|
- .catch((e) => {
|
|
161
|
|
- state.send('CANCELLED_SAVE', { reason: e.message })
|
|
162
|
|
- })
|
|
163
|
141
|
}
|
|
164
|
142
|
|
|
165
|
|
- loadPageFromLocalStorage(fileId: string, pageId: string) {
|
|
166
|
|
- let restored: Page
|
|
|
143
|
+ loadPage(data: Data, pageId = data.currentPageId) {
|
|
|
144
|
+ if (typeof window === 'undefined') return
|
|
|
145
|
+ if (typeof localStorage === 'undefined') return
|
|
|
146
|
+
|
|
|
147
|
+ const fileId = data.document.id
|
|
|
148
|
+
|
|
|
149
|
+ // Page
|
|
167
|
150
|
|
|
168
|
151
|
const savedPage = localStorage.getItem(storageId(fileId, 'page', pageId))
|
|
169
|
152
|
|
|
170
|
153
|
if (savedPage !== null) {
|
|
171
|
|
- restored = JSON.parse(savedPage)
|
|
|
154
|
+ data.document.pages[pageId] = JSON.parse(savedPage)
|
|
172
|
155
|
} else {
|
|
173
|
|
- restored = {
|
|
|
156
|
+ data.document.pages[pageId] = {
|
|
174
|
157
|
id: pageId,
|
|
175
|
158
|
type: 'page',
|
|
176
|
159
|
childIndex: 0,
|
|
|
@@ -179,21 +162,18 @@ class Storage {
|
|
179
|
162
|
}
|
|
180
|
163
|
}
|
|
181
|
164
|
|
|
182
|
|
- return restored
|
|
183
|
|
- }
|
|
184
|
|
-
|
|
185
|
|
- loadPageStateFromLocalStorage(fileId: string, pageId: string) {
|
|
186
|
|
- let restored: PageState
|
|
|
165
|
+ // Page state
|
|
187
|
166
|
|
|
188
|
167
|
const savedPageState = localStorage.getItem(
|
|
189
|
168
|
storageId(fileId, 'pageState', pageId)
|
|
190
|
169
|
)
|
|
191
|
170
|
|
|
192
|
171
|
if (savedPageState !== null) {
|
|
193
|
|
- restored = JSON.parse(savedPageState)
|
|
|
172
|
+ const restored: PageState = JSON.parse(savedPageState)
|
|
194
|
173
|
restored.selectedIds = new Set(restored.selectedIds)
|
|
|
174
|
+ data.pageStates[pageId] = restored
|
|
195
|
175
|
} else {
|
|
196
|
|
- restored = {
|
|
|
176
|
+ data.pageStates[pageId] = {
|
|
197
|
177
|
camera: {
|
|
198
|
178
|
point: [0, 0],
|
|
199
|
179
|
zoom: 1,
|
|
|
@@ -202,64 +182,86 @@ class Storage {
|
|
202
|
182
|
}
|
|
203
|
183
|
}
|
|
204
|
184
|
|
|
205
|
|
- return restored
|
|
|
185
|
+ // Empty shapes in state for other pages
|
|
|
186
|
+
|
|
|
187
|
+ for (let key in data.document.pages) {
|
|
|
188
|
+ if (key === pageId) continue
|
|
|
189
|
+ data.document.pages[key].shapes = {}
|
|
|
190
|
+ }
|
|
|
191
|
+
|
|
|
192
|
+ // Update camera for the new page state
|
|
|
193
|
+ document.documentElement.style.setProperty(
|
|
|
194
|
+ '--camera-zoom',
|
|
|
195
|
+ data.pageStates[data.currentPageId].camera.zoom.toString()
|
|
|
196
|
+ )
|
|
206
|
197
|
}
|
|
207
|
198
|
|
|
208
|
|
- savePage(data: Data, fileId = data.document.id, pageId = data.currentPageId) {
|
|
209
|
|
- if (typeof window === 'undefined') return
|
|
210
|
|
- if (typeof localStorage === 'undefined') return
|
|
|
199
|
+ /* ------------------- File System ------------------ */
|
|
211
|
200
|
|
|
212
|
|
- // Save page
|
|
213
|
|
- const page = data.document.pages[pageId]
|
|
|
201
|
+ saveToFileSystem = (data: Data) => {
|
|
|
202
|
+ this.saveDataToFileSystem(data, data.document.id, false)
|
|
|
203
|
+ }
|
|
214
|
204
|
|
|
215
|
|
- localStorage.setItem(
|
|
216
|
|
- storageId(fileId, 'page', pageId),
|
|
217
|
|
- JSON.stringify(page)
|
|
218
|
|
- )
|
|
|
205
|
+ saveAsToFileSystem = (data: Data) => {
|
|
|
206
|
+ this.saveDataToFileSystem(data, uuid(), true)
|
|
|
207
|
+ }
|
|
219
|
208
|
|
|
220
|
|
- // Save page state
|
|
|
209
|
+ saveDataToFileSystem = (data: Data, id: string, saveAs: boolean) => {
|
|
|
210
|
+ const json = this.getDataToSave(data)
|
|
221
|
211
|
|
|
222
|
|
- let currentPageState = {
|
|
223
|
|
- camera: {
|
|
224
|
|
- point: [0, 0],
|
|
225
|
|
- zoom: 1,
|
|
226
|
|
- },
|
|
227
|
|
- selectedIds: new Set([]),
|
|
228
|
|
- ...data.pageStates[pageId],
|
|
229
|
|
- }
|
|
|
212
|
+ this.saveToLocalStorage(data, id)
|
|
230
|
213
|
|
|
231
|
|
- const pageState = {
|
|
232
|
|
- ...currentPageState,
|
|
233
|
|
- selectedIds: setToArray(currentPageState.selectedIds),
|
|
234
|
|
- }
|
|
|
214
|
+ const blob = new Blob([json], {
|
|
|
215
|
+ type: 'application/vnd.tldraw+json',
|
|
|
216
|
+ })
|
|
235
|
217
|
|
|
236
|
|
- localStorage.setItem(
|
|
237
|
|
- storageId(fileId, 'pageState', pageId),
|
|
238
|
|
- JSON.stringify(pageState)
|
|
|
218
|
+ fa.fileSave(
|
|
|
219
|
+ blob,
|
|
|
220
|
+ {
|
|
|
221
|
+ fileName: `${
|
|
|
222
|
+ saveAs
|
|
|
223
|
+ ? data.document.name
|
|
|
224
|
+ : this.previousSaveHandle?.name || 'My Document'
|
|
|
225
|
+ }.tldr`,
|
|
|
226
|
+ description: 'tldraw file',
|
|
|
227
|
+ extensions: ['.tldr'],
|
|
|
228
|
+ },
|
|
|
229
|
+ saveAs ? undefined : this.previousSaveHandle,
|
|
|
230
|
+ true
|
|
239
|
231
|
)
|
|
|
232
|
+ .then((handle) => {
|
|
|
233
|
+ this.previousSaveHandle = handle
|
|
|
234
|
+ state.send('SAVED_FILE_TO_FILE_SYSTEM')
|
|
|
235
|
+ })
|
|
|
236
|
+ .catch((e) => {
|
|
|
237
|
+ state.send('CANCELLED_SAVE', { reason: e.message })
|
|
|
238
|
+ })
|
|
240
|
239
|
}
|
|
241
|
240
|
|
|
242
|
|
- loadPage(data: Data, pageId = data.currentPageId) {
|
|
243
|
|
- if (typeof window === 'undefined') return
|
|
244
|
|
- if (typeof localStorage === 'undefined') return
|
|
|
241
|
+ loadDocumentFromFilesystem() {
|
|
|
242
|
+ fa.fileOpen({
|
|
|
243
|
+ description: 'tldraw files',
|
|
|
244
|
+ })
|
|
|
245
|
+ .then((blob) =>
|
|
|
246
|
+ getTextFromBlob(blob).then((text) => {
|
|
|
247
|
+ const restoredData = JSON.parse(text)
|
|
245
|
248
|
|
|
246
|
|
- const fileId = data.document.id
|
|
|
249
|
+ console.log('restoring data', restoredData)
|
|
247
|
250
|
|
|
248
|
|
- data.document.pages[pageId] = this.loadPageFromLocalStorage(fileId, pageId)
|
|
|
251
|
+ if (restoredData === null) {
|
|
|
252
|
+ console.warn('Could not load that data.')
|
|
|
253
|
+ return
|
|
|
254
|
+ }
|
|
249
|
255
|
|
|
250
|
|
- data.pageStates[pageId] = this.loadPageStateFromLocalStorage(fileId, pageId)
|
|
|
256
|
+ // Save blob for future saves
|
|
|
257
|
+ this.previousSaveHandle = blob.handle
|
|
251
|
258
|
|
|
252
|
|
- // Empty shapes in state for other pages
|
|
253
|
|
- for (let key in data.document.pages) {
|
|
254
|
|
- if (key === pageId) continue
|
|
255
|
|
- data.document.pages[key].shapes = {}
|
|
256
|
|
- }
|
|
257
|
|
-
|
|
258
|
|
- // Update camera for the new page state
|
|
259
|
|
- document.documentElement.style.setProperty(
|
|
260
|
|
- '--camera-zoom',
|
|
261
|
|
- data.pageStates[data.currentPageId].camera.zoom.toString()
|
|
262
|
|
- )
|
|
|
259
|
+ state.send('LOADED_FROM_FILE', { restoredData: { ...restoredData } })
|
|
|
260
|
+ })
|
|
|
261
|
+ )
|
|
|
262
|
+ .catch((e) => {
|
|
|
263
|
+ state.send('CANCELLED_SAVE', { reason: e.message })
|
|
|
264
|
+ })
|
|
263
|
265
|
}
|
|
264
|
266
|
}
|
|
265
|
267
|
|