Skip to content

Commit

Permalink
[NEW] Adds drag-and-drop on sidebar (RocketChat#539)
Browse files Browse the repository at this point in the history
* Adds drag-and-drop on sidebar

it allows you to reorder it and then saves it using localstorage

* Set active to the dragged server

when dropping the dragged server on the final position it should be set as the active one
  • Loading branch information
vcapretz authored and gdelavald committed Sep 16, 2017
1 parent 325d723 commit aa9cd3f
Showing 1 changed file with 86 additions and 9 deletions.
95 changes: 86 additions & 9 deletions src/scripts/sidebar.js
Original file line number Diff line number Diff line change
Expand Up @@ -9,13 +9,16 @@ class SideBar extends EventEmitter {
constructor () {
super();

this.hostCount = 0;
this.sortOrder = JSON.parse(localStorage.getItem(this.sortOrderKey)) || [];
localStorage.setItem(this.sortOrderKey, JSON.stringify(this.sortOrder));

this.listElement = document.getElementById('serverList');

servers.forEach((host) => {
this.add(host);
});
Object.values(servers.hosts)
.sort((a, b) => this.sortOrder.indexOf(a.url) - this.sortOrder.indexOf(b.url))
.forEach((host) => {
this.add(host);
});

servers.on('host-added', (hostUrl) => {
this.add(servers.get(hostUrl));
Expand Down Expand Up @@ -48,6 +51,10 @@ class SideBar extends EventEmitter {

}

get sortOrderKey () {
return 'rocket.chat.sortOrder';
}

add (host) {
let name = host.title.replace(/^https?:\/\/(?:www\.)?([^\/]+)(.*)/, '$1');
name = name.split('.');
Expand All @@ -69,14 +76,21 @@ class SideBar extends EventEmitter {
img.style.display = 'initial';
initials.style.display = 'none';
};
// img.src = `${host.url}/assets/favicon.svg?v=${Math.round(Math.random()*10000)}`;

let hostOrder = 0;
if (this.sortOrder.includes(host.url)) {
hostOrder = this.sortOrder.indexOf(host.url) + 1;
} else {
hostOrder = this.sortOrder.length + 1;
this.sortOrder.push(host.url);
}

const hotkey = document.createElement('div');
hotkey.classList.add('name');
if (process.platform === 'darwin') {
hotkey.innerHTML = '⌘' + (++this.hostCount);
hotkey.innerHTML = `⌘${hostOrder}`;
} else {
hotkey.innerHTML = '^' + (++this.hostCount);
hotkey.innerHTML = `^${hostOrder}`;
}

const item = document.createElement('li');
Expand All @@ -87,16 +101,68 @@ class SideBar extends EventEmitter {
item.appendChild(hotkey);

item.dataset.host = host.url;
item.dataset.sortOrder = hostOrder;
item.setAttribute('server', host.url);
item.classList.add('instance');

item.setAttribute('draggable', true);

item.ondragstart = (event) => {
window.dragged = event.target.nodeName !== 'LI' ? event.target.closest('li') : event.target;
event.dataTransfer.effectAllowed = 'move';
event.dataTransfer.dropEffect = 'move';
event.target.style.opacity = .5;
};

item.ondragover = (event) => {
event.preventDefault();
};

item.ondragenter = (event) => {
if (this.isBefore(window.dragged, event.target)) {
event.currentTarget.parentNode.insertBefore(window.dragged, event.currentTarget);
} else if (event.currentTarget !== event.currentTarget.parentNode.lastChild) {
event.currentTarget.parentNode.insertBefore(window.dragged, event.currentTarget.nextSibling);
} else {
event.currentTarget.parentNode.appendChild(window.dragged);
}
};

item.ondragend = (event) => {
event.target.style.opacity = '';
};

item.ondrop = (event) => {
event.preventDefault();

const newSortOrder = [];
Array.from(event.currentTarget.parentNode.children)
.map((sideBarElement) => {
const url = sideBarElement.dataset.host;
newSortOrder.push(url);
this.remove(url);

return sideBarElement;
})
.map((sideBarElement) => {
this.sortOrder = newSortOrder;
localStorage.setItem(this.sortOrderKey, JSON.stringify(this.sortOrder));

const url = sideBarElement.dataset.host;
const host = { url, title: sideBarElement.querySelector('div.tooltip').innerHTML };
this.add(host);
this.setImage(url);
});

this.setActive(window.dragged.dataset.host);
};

item.onclick = () => {
servers.setActive(host.url);
};

this.listElement.appendChild(item);

menus.addServer(host, this.hostCount);
menus.addServer(host, hostOrder);
}

setImage (hostUrl) {
Expand Down Expand Up @@ -220,6 +286,17 @@ class SideBar extends EventEmitter {
isHidden () {
return localStorage.getItem('sidebar-closed') === 'true';
}

isBefore (a, b) {
if (a.parentNode === b.parentNode) {
for (let cur = a; cur; cur = cur.previousSibling) {
if (cur === b) {
return true;
}
}
}
return false;
}
}

export default new SideBar();
Expand Down

0 comments on commit aa9cd3f

Please sign in to comment.