Skip to content

Commit 746d935

Browse files
committed
Cleaned up grid handling in desktop iconview (#51)
1 parent 9359b49 commit 746d935

File tree

1 file changed

+146
-101
lines changed

1 file changed

+146
-101
lines changed

src/adapters/ui/iconview.js

+146-101
Original file line numberDiff line numberDiff line change
@@ -36,27 +36,12 @@ import {pathJoin} from '../../utils/vfs';
3636

3737
const tapper = doubleTap();
3838

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-
4839
const validVfsDrop = data => data && data.path;
4940
const validInternalDrop = data => data && data.internal;
5041

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+
*/
6045
const onDropAction = actions => (ev, data, files, shortcut = true) => {
6146
if (validVfsDrop(data)) {
6247
actions.addEntry({entry: data, shortcut, ev});
@@ -67,62 +52,34 @@ const onDropAction = actions => (ev, data, files, shortcut = true) => {
6752
}
6853
};
6954

55+
/**
56+
* Checks event is on a root element
57+
*/
7058
const isRootElement = ev =>
7159
ev.target && ev.target.classList.contains('osjs-desktop-iconview__wrapper');
7260

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]);
11166

11267
return enabled ? {
11368
position: 'absolute',
114-
top: String(top * sizeY) + 'px',
115-
left: String(left * sizeX) + 'px'
69+
top: String(top) + 'px',
70+
left: String(left) + 'px'
11671
} : {};
11772
};
11873

119-
const createGhostStyle = ({ghost, grid: {enabled, sizeX, sizeY}}) => {
74+
/**
75+
* Creates UI drop ghost
76+
*/
77+
const createGhostStyle = (grid, ghost, enabled) => {
12078
const style = {};
12179
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';
12683
}
12784

12885
return Object.assign({
@@ -131,7 +88,10 @@ const createGhostStyle = ({ghost, grid: {enabled, sizeX, sizeY}}) => {
13188
}, style);
13289
};
13390

134-
const view = (fileIcon, themeIcon, droppable) => (state, actions) =>
91+
/**
92+
* Creates UI view
93+
*/
94+
const view = (fileIcon, themeIcon, grid) => (state, actions) =>
13595
h('div', {
13696
class: 'osjs-desktop-iconview__wrapper',
13797
oncontextmenu: ev => {
@@ -164,7 +124,7 @@ const view = (fileIcon, themeIcon, droppable) => (state, actions) =>
164124
}, [
165125
...state.entries.map((entry, index) => {
166126
return h('div', {
167-
style: createIconStyle(entry, index, state),
127+
style: createIconStyle(entry, grid, state.grid),
168128
class: 'osjs-desktop-iconview__entry' + (
169129
state.selected === index
170130
? ' osjs-desktop-iconview__entry--selected'
@@ -205,10 +165,115 @@ const view = (fileIcon, themeIcon, droppable) => (state, actions) =>
205165
}),
206166
h('div', {
207167
class: 'osjs-desktop-iconview__entry osjs-desktop-iconview__entry--ghost',
208-
style: createGhostStyle(state)
168+
style: createGhostStyle(grid, state.ghost, state.grid)
209169
})
210170
]);
211171

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+
*/
212277
const createShortcuts = (root, readfile, writefile) => {
213278
const read = () => {
214279
const filename = pathJoin(root, '.shortcuts.json');
@@ -240,6 +305,9 @@ const createShortcuts = (root, readfile, writefile) => {
240305
return {read, add, remove};
241306
};
242307

308+
/**
309+
* Wrapper for handling reading the desktop folder
310+
*/
243311
const readDesktopFolder = (root, readdir, shortcuts) => {
244312
const read = () => readdir(root, {
245313
showHiddenFiles: false
@@ -336,25 +404,19 @@ export class DesktopIconView extends EventEmitter {
336404
const error = err => console.error(err);
337405
const shortcuts = createShortcuts(root, readfile, writefile);
338406
const read = readDesktopFolder(root, readdir, shortcuts);
407+
const grid = createGrid(this.$root);
339408

340-
const [rows, cols, sizeX, sizeY] = calculateGridSizes(this.$root);
409+
grid.load();
341410

342411
this.iconview = app({
343412
selected: -1,
344413
entries: [],
345-
positions: loadIconPositions(),
346414
ghost: false,
347-
grid: {
348-
enabled: settings.grid,
349-
rows,
350-
cols,
351-
sizeX,
352-
sizeY
353-
}
415+
grid: settings.grid
354416
}, {
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+
}),
358420

359421
openDropContextMenu: ({ev, data, files}) => {
360422
this.createDropContextMenu(ev, data, files);
@@ -429,27 +491,11 @@ export class DesktopIconView extends EventEmitter {
429491
},
430492

431493
moveEntry: ({entry, ev}) => (state) => {
432-
if (!isIconPositionBusy(ev, state)) {
433-
const positions = state.positions;
494+
if (!grid.isBusy(ev, state.entries)) {
434495
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);
448497

449-
return {
450-
positions,
451-
entries: calculateIconPositions(state.entries, positions, state.cols)
452-
};
498+
return {entries: grid.calculate(state.entries)};
453499
}
454500
return {};
455501
},
@@ -460,9 +506,8 @@ export class DesktopIconView extends EventEmitter {
460506
.then(entries => actions.setEntries(entries));
461507
},
462508

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();
466511
},
467512

468513
toggleGrid: enabled => ({grid}) => {
@@ -472,7 +517,7 @@ export class DesktopIconView extends EventEmitter {
472517
setGhost: ev => {
473518
return {ghost: ev};
474519
}
475-
}, view(fileIcon, themeIcon, droppable), this.$root);
520+
}, view(fileIcon, themeIcon, grid), this.$root);
476521

477522
this.iconview.reload();
478523
}

0 commit comments

Comments
 (0)