@@ -36,27 +36,12 @@ import {pathJoin} from '../../utils/vfs';
36
36
37
37
const tapper = doubleTap ( ) ;
38
38
39
- //
40
- // FIXME: Excessive render on drop events
41
- //
42
-
43
- // TODO: Needs real values
44
- const ICON_WIDTH = 5.0 ; // ems
45
- const ICON_HEIGHT = 6.5 ; // ems
46
- const ICON_MARGIN = 0.5 ; // ems
47
-
48
39
const validVfsDrop = data => data && data . path ;
49
40
const validInternalDrop = data => data && data . internal ;
50
41
51
- // TODO: Use internal storage
52
- const loadIconPositions = ( ) => JSON . parse (
53
- localStorage . getItem ( '___osjs_iconview_positions' ) || '[]'
54
- ) ;
55
-
56
- // TODO: Use internal storage
57
- const saveIconPositions = positions =>
58
- localStorage . setItem ( '___osjs_iconview_positions' , JSON . stringify ( positions || [ ] ) ) ;
59
-
42
+ /**
43
+ * Drop handler
44
+ */
60
45
const onDropAction = actions => ( ev , data , files , shortcut = true ) => {
61
46
if ( validVfsDrop ( data ) ) {
62
47
actions . addEntry ( { entry : data , shortcut, ev} ) ;
@@ -67,62 +52,34 @@ const onDropAction = actions => (ev, data, files, shortcut = true) => {
67
52
}
68
53
} ;
69
54
55
+ /**
56
+ * Checks event is on a root element
57
+ */
70
58
const isRootElement = ev =>
71
59
ev . target && ev . target . classList . contains ( 'osjs-desktop-iconview__wrapper' ) ;
72
60
73
- const calculateGridSizes = el => {
74
- const { offsetWidth, offsetHeight} = el ;
75
- // TODO: Might cause reflow, do cache here
76
- const sizeX = emToPx ( ICON_WIDTH ) + ( emToPx ( ICON_MARGIN ) * 2 ) ;
77
- const sizeY = emToPx ( ICON_HEIGHT ) + ( emToPx ( ICON_MARGIN ) * 2 ) ;
78
- const cols = Math . floor ( offsetWidth / sizeX ) ;
79
- const rows = Math . floor ( offsetHeight / sizeY ) ;
80
- return [ rows , cols , sizeX , sizeY ] ;
81
- } ;
82
-
83
- const calculateIconPositions = ( entries , positions , cols ) => {
84
- const savedPositions = entries . map ( entry => {
85
- const key = entry . shortcut === false ? entry . filename : entry . shortcut ;
86
- const found = positions . findIndex ( s => s . key === key ) ;
87
- return found === - 1 ? undefined : positions [ found ] . position ;
88
- } ) ;
89
-
90
- return entries . map ( ( entry , index ) => {
91
- const x = index % cols ;
92
- const y = Math . floor ( index / cols ) ;
93
- const _position = savedPositions [ index ] || [ x , y ] ;
94
-
95
- return Object . assign ( entry , { _position} ) ;
96
- } ) ;
97
- } ;
98
-
99
- const isIconPositionBusy = ( ev , { entries, grid : { sizeX, sizeY} } ) => {
100
- const col = Math . floor ( ev . clientX / sizeX ) ;
101
- const row = Math . floor ( ev . clientY / sizeY ) ;
102
-
103
- return entries . findIndex ( e => {
104
- return e . _position [ 0 ] === col &&
105
- e . _position [ 1 ] === row ;
106
- } ) !== - 1 ;
107
- } ;
108
-
109
- const createIconStyle = ( entry , index , { grid : { enabled, sizeX, sizeY} } ) => {
110
- const [ left , top ] = entry . _position || [ 0 , 0 ] ;
61
+ /**
62
+ * Creates UI icon styles
63
+ */
64
+ const createIconStyle = ( entry , grid , enabled ) => {
65
+ const [ left , top ] = grid . getPosition ( entry . _position || [ 0 , 0 ] ) ;
111
66
112
67
return enabled ? {
113
68
position : 'absolute' ,
114
- top : String ( top * sizeY ) + 'px' ,
115
- left : String ( left * sizeX ) + 'px'
69
+ top : String ( top ) + 'px' ,
70
+ left : String ( left ) + 'px'
116
71
} : { } ;
117
72
} ;
118
73
119
- const createGhostStyle = ( { ghost, grid : { enabled, sizeX, sizeY} } ) => {
74
+ /**
75
+ * Creates UI drop ghost
76
+ */
77
+ const createGhostStyle = ( grid , ghost , enabled ) => {
120
78
const style = { } ;
121
79
if ( ghost instanceof Event ) {
122
- const col = Math . floor ( ghost . clientX / sizeX ) ;
123
- const row = Math . floor ( ghost . clientY / sizeY ) ;
124
- style . top = String ( row * sizeY ) + 'px' ;
125
- style . left = String ( col * sizeX ) + 'px' ;
80
+ const [ col , row ] = grid . getEventPosition ( ghost ) ;
81
+ style . top = row + 'px' ;
82
+ style . left = col + 'px' ;
126
83
}
127
84
128
85
return Object . assign ( {
@@ -131,7 +88,10 @@ const createGhostStyle = ({ghost, grid: {enabled, sizeX, sizeY}}) => {
131
88
} , style ) ;
132
89
} ;
133
90
134
- const view = ( fileIcon , themeIcon , droppable ) => ( state , actions ) =>
91
+ /**
92
+ * Creates UI view
93
+ */
94
+ const view = ( fileIcon , themeIcon , grid ) => ( state , actions ) =>
135
95
h ( 'div' , {
136
96
class : 'osjs-desktop-iconview__wrapper' ,
137
97
oncontextmenu : ev => {
@@ -164,7 +124,7 @@ const view = (fileIcon, themeIcon, droppable) => (state, actions) =>
164
124
} , [
165
125
...state . entries . map ( ( entry , index ) => {
166
126
return h ( 'div' , {
167
- style : createIconStyle ( entry , index , state ) ,
127
+ style : createIconStyle ( entry , grid , state . grid ) ,
168
128
class : 'osjs-desktop-iconview__entry' + (
169
129
state . selected === index
170
130
? ' osjs-desktop-iconview__entry--selected'
@@ -205,10 +165,115 @@ const view = (fileIcon, themeIcon, droppable) => (state, actions) =>
205
165
} ) ,
206
166
h ( 'div' , {
207
167
class : 'osjs-desktop-iconview__entry osjs-desktop-iconview__entry--ghost' ,
208
- style : createGhostStyle ( state )
168
+ style : createGhostStyle ( grid , state . ghost , state . grid )
209
169
} )
210
170
] ) ;
211
171
172
+ /**
173
+ * Handles grid
174
+ * FIXME: Excessive render on drop events
175
+ */
176
+ const createGrid = ( root ) => {
177
+
178
+ // TODO: Needs real values
179
+ const ICON_WIDTH = 5.0 ; // ems
180
+ const ICON_HEIGHT = 6.5 ; // ems
181
+ const ICON_MARGIN = 0.5 ; // ems
182
+
183
+ /* eslint-disable no-unused-vars */
184
+ let rows = 0 ;
185
+ let cols = 0 ;
186
+ let sizeX = 0 ;
187
+ let sizeY = 0 ;
188
+ let positions = [ ] ;
189
+
190
+ const resize = ( ) => {
191
+ const { offsetWidth, offsetHeight} = root ;
192
+ sizeX = emToPx ( ICON_WIDTH ) + ( emToPx ( ICON_MARGIN ) * 2 ) ;
193
+ sizeY = emToPx ( ICON_HEIGHT ) + ( emToPx ( ICON_MARGIN ) * 2 ) ;
194
+ cols = Math . floor ( offsetWidth / sizeX ) ;
195
+ rows = Math . floor ( offsetHeight / sizeY ) ;
196
+ } ;
197
+
198
+ const load = ( ) => {
199
+ // TODO: Use internal storage
200
+ positions = JSON . parse (
201
+ localStorage . getItem (
202
+ '___osjs_iconview_positions'
203
+ ) || '[]'
204
+ ) ;
205
+ } ;
206
+
207
+ const save = ( ) => {
208
+ // TODO: Use internal storage
209
+ return localStorage . setItem (
210
+ '___osjs_iconview_positions' ,
211
+ JSON . stringify ( positions || [ ] )
212
+ ) ;
213
+ } ;
214
+
215
+ const getOffset = ev => ( [
216
+ Math . floor ( ev . clientX / sizeX ) ,
217
+ Math . floor ( ev . clientY / sizeY )
218
+ ] ) ;
219
+
220
+ const getPosition = ( [ left , top ] ) => ( [
221
+ left * sizeX ,
222
+ top * sizeY
223
+ ] ) ;
224
+
225
+ const getEventPosition = ev => {
226
+ const [ col , row ] = getOffset ( ev ) ;
227
+ return [ col * sizeX , row * sizeY ] ;
228
+ } ;
229
+
230
+ const isBusy = ( ev , entries ) => {
231
+ const [ col , row ] = getOffset ( ev ) ;
232
+
233
+ return entries . findIndex ( e => {
234
+ return e . _position [ 0 ] === col &&
235
+ e . _position [ 1 ] === row ;
236
+ } ) !== - 1 ;
237
+ } ;
238
+
239
+ const calculate = entries => {
240
+ const savedPositions = entries . map ( entry => {
241
+ const key = entry . shortcut === false ? entry . filename : entry . shortcut ;
242
+ const found = positions . findIndex ( s => s . key === key ) ;
243
+ return found === - 1 ? undefined : positions [ found ] . position ;
244
+ } ) ;
245
+
246
+ return entries . map ( ( entry , index ) => {
247
+ const x = index % cols ;
248
+ const y = Math . floor ( index / cols ) ;
249
+ const _position = savedPositions [ index ] || [ x , y ] ;
250
+
251
+ return Object . assign ( entry , { _position} ) ;
252
+ } ) ;
253
+ } ;
254
+
255
+ const move = ( ev , key ) => {
256
+ const [ col , row ] = getOffset ( ev ) ;
257
+ const found = positions . findIndex ( s => s . key === key ) ;
258
+
259
+ const position = [ col , row ] ;
260
+ const value = { key, position} ;
261
+
262
+ if ( found !== - 1 ) {
263
+ positions [ found ] = value ;
264
+ } else {
265
+ positions . push ( value ) ;
266
+ }
267
+
268
+ return save ( ) ;
269
+ } ;
270
+
271
+ return { resize, load, save, calculate, move, isBusy, getPosition, getEventPosition} ;
272
+ } ;
273
+
274
+ /**
275
+ * Handles shortcuts
276
+ */
212
277
const createShortcuts = ( root , readfile , writefile ) => {
213
278
const read = ( ) => {
214
279
const filename = pathJoin ( root , '.shortcuts.json' ) ;
@@ -240,6 +305,9 @@ const createShortcuts = (root, readfile, writefile) => {
240
305
return { read, add, remove} ;
241
306
} ;
242
307
308
+ /**
309
+ * Wrapper for handling reading the desktop folder
310
+ */
243
311
const readDesktopFolder = ( root , readdir , shortcuts ) => {
244
312
const read = ( ) => readdir ( root , {
245
313
showHiddenFiles : false
@@ -336,25 +404,19 @@ export class DesktopIconView extends EventEmitter {
336
404
const error = err => console . error ( err ) ;
337
405
const shortcuts = createShortcuts ( root , readfile , writefile ) ;
338
406
const read = readDesktopFolder ( root , readdir , shortcuts ) ;
407
+ const grid = createGrid ( this . $root ) ;
339
408
340
- const [ rows , cols , sizeX , sizeY ] = calculateGridSizes ( this . $root ) ;
409
+ grid . load ( ) ;
341
410
342
411
this . iconview = app ( {
343
412
selected : - 1 ,
344
413
entries : [ ] ,
345
- positions : loadIconPositions ( ) ,
346
414
ghost : false ,
347
- grid : {
348
- enabled : settings . grid ,
349
- rows,
350
- cols,
351
- sizeX,
352
- sizeY
353
- }
415
+ grid : settings . grid
354
416
} , {
355
- setEntries : entries => state => {
356
- return { entries : calculateIconPositions ( entries , state . positions , state . grid . cols ) } ;
357
- } ,
417
+ setEntries : entries => state => ( {
418
+ entries : grid . calculate ( entries )
419
+ } ) ,
358
420
359
421
openDropContextMenu : ( { ev, data, files} ) => {
360
422
this . createDropContextMenu ( ev , data , files ) ;
@@ -429,27 +491,11 @@ export class DesktopIconView extends EventEmitter {
429
491
} ,
430
492
431
493
moveEntry : ( { entry, ev} ) => ( state ) => {
432
- if ( ! isIconPositionBusy ( ev , state ) ) {
433
- const positions = state . positions ;
494
+ if ( ! grid . isBusy ( ev , state . entries ) ) {
434
495
const key = entry . shortcut === false ? entry . filename : entry . shortcut ;
435
- const found = positions . findIndex ( s => s . key === key ) ;
436
- const col = Math . floor ( ev . clientX / sizeX ) ;
437
- const row = Math . floor ( ev . clientY / sizeY ) ;
438
- const position = [ col , row ] ;
439
- const value = { key, position} ;
440
-
441
- if ( found !== - 1 ) {
442
- positions [ found ] = value ;
443
- } else {
444
- positions . push ( value ) ;
445
- }
446
-
447
- saveIconPositions ( positions ) ;
496
+ grid . move ( ev , key ) ;
448
497
449
- return {
450
- positions,
451
- entries : calculateIconPositions ( state . entries , positions , state . cols )
452
- } ;
498
+ return { entries : grid . calculate ( state . entries ) } ;
453
499
}
454
500
return { } ;
455
501
} ,
@@ -460,9 +506,8 @@ export class DesktopIconView extends EventEmitter {
460
506
. then ( entries => actions . setEntries ( entries ) ) ;
461
507
} ,
462
508
463
- resize : ( ) => ( { grid : { enabled} } ) => {
464
- const [ rows , cols , sizeX , sizeY ] = calculateGridSizes ( this . $root ) ;
465
- return { grid : { enabled, rows, cols, sizeX, sizeY} } ;
509
+ resize : ( ) => {
510
+ grid . resize ( ) ;
466
511
} ,
467
512
468
513
toggleGrid : enabled => ( { grid} ) => {
@@ -472,7 +517,7 @@ export class DesktopIconView extends EventEmitter {
472
517
setGhost : ev => {
473
518
return { ghost : ev } ;
474
519
}
475
- } , view ( fileIcon , themeIcon , droppable ) , this . $root ) ;
520
+ } , view ( fileIcon , themeIcon , grid ) , this . $root ) ;
476
521
477
522
this . iconview . reload ( ) ;
478
523
}
0 commit comments