-
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathStorage.ts
executable file
·336 lines (298 loc) · 12.2 KB
/
Storage.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
import { type Sort } from './Sort'
import { type Filter, type FilterCondition } from './Filter'
/**
* @group Storage Service
* @description Whether grid state is stored in {@link https://developer.mozilla.org/en-US/docs/Web/API/Window/localStorage | localStorage}
* or {@link https://developer.mozilla.org/en-US/docs/Web/API/Window/sessionStorage | sessionStorage}.
* @see {@link DataGridVueGrid.localStorageType}
* @see {@link DataGridVueGrid.storageKey}
*/
export enum LocalStorageType {
sessionStorage = 0,
localStorage = 1,
}
/**
* @group Storage Service
* @description The data that is saved as part of the grid state.
*/
export interface GridState {
/** The current page size */
pageSize: number
/** The field names of the hidden columns */
hiddenFields: string[]
/** The current sort definition */
sort: Sort[]
/** The current filter conditions */
filters: FilterCondition[]
/** The current external filter if applied */
externalFilter: Filter | undefined
/** The field name of the columns in the order they are currently displayed */
columnOrder: string[]
}
/**
* @group Storage Service
* @description Interface to implement to define a storage service to save and load grid state.
*/
export interface StorageService {
/**
* @description Retrieves the saved grid state. This is called once when the data grid component is mounted.
* @returns A Promise that returns the saved grid state or undefined if no grid state is saved.
*/
getGridStateAsync(): Promise<GridState | undefined>
/**
* @description Saves a new version of the grid state. This called every time data in the grid state changes.
* @param gridState The grid state to save
* @returns A Promise that returns when the grid state has been saved.
*/
setGridStateAsync(gridState: GridState): Promise<void>
}
/**
* @group Storage Service
* @description A stub {@link StorageService} that will just return empty resolved promises.
*/
export const StubStorageService = {
getGridStateAsync(): Promise<GridState | undefined> {
return Promise.resolve(undefined)
},
setGridStateAsync(gridState: GridState): Promise<void> {
return Promise.resolve()
},
} as StorageService
/**
* @group Storage Service
* @description Storage service that saves grid state in {@link https://developer.mozilla.org/en-US/docs/Web/API/Window/sessionStorage | sessionStorage}.
*/
export class SessionStorageService implements StorageService {
key: string
constructor(key: string) {
this.key = key
}
getGridStateAsync(): Promise<GridState | undefined> {
const rawValue = sessionStorage.getItem(this.key)
let result = undefined as GridState | undefined
if (rawValue) {
result = JSON.parse(rawValue) as GridState
}
return Promise.resolve(result)
}
setGridStateAsync(gridState: GridState): Promise<void> {
sessionStorage.setItem(this.key, JSON.stringify(gridState))
return Promise.resolve()
}
}
/**
* @group Storage Service
* @description Storage service that saves grid state in {@link https://developer.mozilla.org/en-US/docs/Web/API/Window/localStorage | localStorage}.
*/
export class LocalStorageService implements StorageService {
key: string
constructor(key: string) {
this.key = key
}
getGridStateAsync(): Promise<GridState | undefined> {
const rawValue = localStorage.getItem(this.key)
let result = undefined as GridState | undefined
if (rawValue) {
result = JSON.parse(rawValue) as GridState
}
return Promise.resolve(result)
}
setGridStateAsync(gridState: GridState): Promise<void> {
localStorage.setItem(this.key, JSON.stringify(gridState))
return Promise.resolve()
}
}
/**
* @group Storage Service
* @description Request data interface sent by the {@link ServerSideStorageService} to get the current grid state.
* @typeParam TUserId The type of the user identifier.
* @typeParam TGridId The type of the grid identifier.
*/
export interface GetGridStateRequest<TUserId, TGridId> {
/** The unique identifier for the current user. */
userId: TUserId
/** The unique identifier for the specific data grid instance. */
gridId: TGridId
}
/**
* @group Storage Service
* @description Request data interface sent by the {@link ServerSideStorageService} to save the current grid state.
* @typeParam TUserId The type of the user identifier.
* @typeParam TGridId The type of the grid identifier.
*/
export interface SetGridStateRequest<TUserId, TGridId> {
/** The unique identifier for the current user. */
userId: TUserId
/** The unique identifier for the specific data grid instance. */
gridId: TGridId
/** The current grid state to save. */
gridState: GridState
}
/**
* @group Storage Service
* @description Callback type to change the {@link https://developer.mozilla.org/docs/Web/API/Request | Request}
* object before it is sent to the server from the built-in server side storage service. This is useful
* when you need to map the {@link GetGridStateRequest} to a different data contract or alter the HTTP verb/headers.
* @typeParam TUserId The type of the user identifier.
* @typeParam TGridId The type of the grid identifier.
*/
export type BeforeGetRequestHandler<TUserId, TGridId> = (request: Request, body: GetGridStateRequest<TUserId, TGridId>) => Promise<Request>
/**
* @group Storage Service
* @description Callback type to change the {@link https://developer.mozilla.org/docs/Web/API/Response | Response}
* object before it is handled by the data grid from the built-in server side storage service.
* This is useful when you need to map the servers response data back to {@link GridState}.
*/
export type GetResponseHandler = (response: Response) => Promise<GridState>
/**
* @group Storage Service
* @description Callback type to change the {@link https://developer.mozilla.org/docs/Web/API/Request | Request}
* object before it is sent to the server from the built-in server side storage service. This is useful
* when you need to map the {@link SetGridStateRequest} to a different data contract or alter the HTTP verb/headers.
* @typeParam TUserId The type of the user identifier.
* @typeParam TGridId The type of the grid identifier.
*/
export type BeforeSetRequestHandler<TUserId, TGridId> = (request: Request, body: SetGridStateRequest<TUserId, TGridId>) => Promise<Request>
/**
* @group Storage Service
* @description Callback type to change the {@link https://developer.mozilla.org/docs/Web/API/Response | Response}
* object before it is handled by the data grid from the built-in server side data service.
*/
export type SetResponseHandler = (response: Response) => Promise<boolean>
/**
* @group Storage Service
* @description Options to configure the built-in server-side storage service.
* The server-side storage service will only attempt to deserialize the response body for `getGridState`
* if the HTTP status code is `200 OK` and the `Content-Type` response header is `application/json`.
* @typeParam TUserId The type of the user identifier.
* @typeParam TGridId The type of the grid identifier.
* @see {@link ServerSideStorageService}
*/
export interface ServerSideStorageServiceOptions<TUserId, TGridId> {
/**
* The unique identifier for the current user that will be sent to the server with the get and set requests.
*/
userId: TUserId
/**
* The unique identifier for the specific data grid instance that will be sent to the server with the get and set requests.
*/
gridId: TGridId
/**
* @description The full HTTP/HTTPS url to send the POST request to retrieve grid state.
* Use {@link beforeGetRequest} callback to alter the HTTP verb or headers.
*/
getPostRoute?: string | URL
/**
* Optional callback to change the {@link https://developer.mozilla.org/docs/Web/API/Request | Request}
* object before it is sent to the server from the built-in server side storage service. This is useful
* when you need to map the {@link GetGridStateRequest} to a different data contract or alter the HTTP verb/headers.
*/
beforeGetRequest?: BeforeGetRequestHandler<TUserId, TGridId>
/**
* Optional callback to change the {@link https://developer.mozilla.org/docs/Web/API/Response | Response}
* object before it is handled by the data grid. This is useful when you need to map the servers response
* data back to {@link GridState}.
*/
getResponseHandler?: GetResponseHandler
/**
* @description The full HTTP/HTTPS url to send the POST request to save grid state.
* Use {@link beforeSetRequest} callback to alter the HTTP verb or headers.
*/
setPostRoute?: string | URL
/**
* Optional callback to change the {@link https://developer.mozilla.org/docs/Web/API/Request | Request}
* object before it is sent to the server from the built-in server side storage service. This is useful
* when you need to map the {@link SetGridStateRequest} to a different data contract or alter the HTTP verb/headers.
*/
beforeSetRequest?: BeforeGetRequestHandler<TUserId, TGridId>
/**
* Optional callback type to change the {@link https://developer.mozilla.org/docs/Web/API/Response | Response}
* object before it is handled by the data grid from the built-in server side data service.
*/
setResponseHandler?: GetResponseHandler
}
function getDefaultRequestOptions(): RequestInit {
return {
method: 'POST',
headers: new Headers({
Accept: 'application/json',
'Content-Type': 'application/json',
}),
mode: 'cors',
cache: 'no-cache',
}
}
/**
* @group Storage Service
* @description The server-side {@link StorageService} used when {@link DataGridVueGrid.serverSideStorageOptions} is specified.
* This storage service will only attempt to deserialize the response body for `getGridState`
* if the HTTP status code is `200 OK` and the `Content-Type` response header is `application/json`.
* @typeParam TUserId The type of the user identifier.
* @typeParam TGridId The type of the grid identifier.
*/
export class ServerSideStorageService<TUserId, TGridId> implements StorageService {
options: ServerSideStorageServiceOptions<TUserId, TGridId>
constructor(options: ServerSideStorageServiceOptions<TUserId, TGridId>) {
this.options = options
}
async getGridStateAsync(): Promise<GridState | undefined> {
const body = {
userId: this.options.userId,
gridId: this.options.gridId,
} as GetGridStateRequest<TUserId, TGridId>
let request = new Request(this.options.getPostRoute ?? '', {
...getDefaultRequestOptions(),
body: JSON.stringify(body),
})
if (this.options.beforeGetRequest) {
request = await this.options.beforeGetRequest(request, body)
}
if (!request.url) {
console.error(
'A request url for the get grid state request has to be set either by supplying it directly or setting it as part of beforeRequest handler.',
)
return undefined
}
var response = await fetch(request)
if (this.options.getResponseHandler) {
return await this.options.getResponseHandler(response)
}
if (response.status >= 400) {
console.error('Failed to retrieve grid state', await response.text(), response)
return undefined
}
if (response.status == 200 && response.headers.get('Content-Type')?.includes('application/json')) {
return response.json()
}
return undefined
}
async setGridStateAsync(gridState: GridState): Promise<void> {
const body = {
userId: this.options.userId,
gridId: this.options.gridId,
gridState,
} as SetGridStateRequest<TUserId, TGridId>
let request = new Request(this.options.setPostRoute ?? '', {
...getDefaultRequestOptions(),
body: JSON.stringify(body),
})
if (this.options.beforeSetRequest) {
request = await this.options.beforeSetRequest(request, body)
}
if (!request.url) {
console.error(
'A request url for the set grid state request has to be set either by supplying it directly or setting it as part of beforeRequest handler.',
)
return undefined
}
var response = await fetch(request)
if (this.options.setResponseHandler) {
await this.options.setResponseHandler(response)
return
}
if (response.status >= 400) {
console.error('Failed to set grid state', await response.text(), response)
return undefined
}
}
}