Compare commits

...

10 Commits

Author SHA1 Message Date
91df15309a Delete servers using console 2025-10-14 15:20:41 +02:00
b3636f2710 Support adding servers 2025-10-14 15:20:20 +02:00
6e410f04a1 Cleanup 2025-10-14 14:36:44 +02:00
b74a4d8462 delete instead of = null 2025-10-14 14:35:19 +02:00
5a1cd5634a Do not nuke magnet URL when torrent gets removed 2025-10-14 14:31:55 +02:00
07d2677af1 Store server type on save 2025-10-14 14:29:51 +02:00
6c35320164 Add server manager
Only removing is supported for now
2025-10-14 14:28:54 +02:00
7274285f95 Change scroll behavior
Only scroll the torrent list, not the whole document
2025-10-14 13:00:09 +02:00
1ed720d7d6 Re-enable auto-refresh
Also change its loglevel
2025-10-14 12:58:58 +02:00
8d434cc114 Re-enable magnet button 2025-10-14 12:57:30 +02:00
5 changed files with 189 additions and 35 deletions

View File

@@ -14,9 +14,33 @@ class trdom_servermanager extends EventTarget {
return this.#servers; return this.#servers;
} }
#registerServer = function(name, initdata) { #elements = {
menuentry: document.createElement('div'),
serverlist: {
shade: document.createElement('div'),
form: document.createElement('div'),
header: {
container: document.createElement('div'),
addbutton: document.createElement('div')
},
list: document.createElement('div')
}
};
get menuentry() {
return this.#elements.menuentry;
}
#registerServer(name, initdata) {
let server = new trserver(name, initdata, this.#logcb); let server = new trserver(name, initdata, this.#logcb);
this.#servers[name] = server; this.#servers[name] = server;
this.#elements.serverlist.list.appendChild(server.element);
let button = document.createElement('div');
server.addServerListButton('deleteserver', button);
button.setAttributeNS('trweb', 'servermanagerbutton', 'delete');
button.setAttribute('server', name);
button.appendChild(document.createTextNode('[Delete]'));
button.addEventListener('click', this);
//server.addEventListener('torrent-updated', this); //server.addEventListener('torrent-updated', this);
server.addEventListener('torrent-initialize', this); server.addEventListener('torrent-initialize', this);
let e = new CustomEvent('torrentserver-added', { let e = new CustomEvent('torrentserver-added', {
@@ -29,11 +53,21 @@ class trdom_servermanager extends EventTarget {
server.refreshTorrentList(); server.refreshTorrentList();
} }
addServer(name, url, user, pass) { addServer(name, url, user = null, pass = '') {
this.#registerServer(name, {'rpcurl': url, 'auth': `Basic ${btoa(`${user}:${pass}`)}`}); let initdata = {rpcurl: url};
if (user != null) {
initdata.auth = `Basic ${btoa(`${user}:${pass}`)}`
}
this.#registerServer(name, initdata);
this.#saveServers(); this.#saveServers();
} }
deleteServer(name) {
delete this.#servers[name];
this.saveServers();
window.location.reload();
}
#loadcb; #loadcb;
#loadServers() { #loadServers() {
let serverjson = this.#loadcb('servers'); let serverjson = this.#loadcb('servers');
@@ -71,11 +105,100 @@ class trdom_servermanager extends EventTarget {
this.#loadcb = loadcb; this.#loadcb = loadcb;
this.#savecb = savecb; this.#savecb = savecb;
this.#logcb = logcb this.#logcb = logcb
this.menuentry.appendChild(document.createTextNode('[Servers]'));
this.menuentry.setAttributeNS('trweb', 'servermanagerbutton', 'manager');
this.menuentry.addEventListener('click', this);
this.#elements.serverlist.shade.classList.add('trweb_formshade', 'd-flex', 'justify-content-center', 'align-items-center');
this.#elements.serverlist.shade.setAttributeNS('trweb', 'servermanagerbutton', 'hide');
this.#elements.serverlist.shade.addEventListener('click', this);
this.#elements.serverlist.shade.appendChild(this.#elements.serverlist.form);
this.#elements.serverlist.form.classList.add('trweb_form', 'd-flex', 'flex-column');
this.#elements.serverlist.form.setAttributeNS('trweb', 'servermanagerbutton', 'block');
this.#elements.serverlist.form.addEventListener('click', this);
this.#elements.serverlist.form.style.backgroundColor = 'cyan';
this.#elements.serverlist.form.appendChild(this.#elements.serverlist.header.container);
this.#elements.serverlist.form.appendChild(this.#elements.serverlist.list);
this.#elements.serverlist.header.container.classList.add('d-flex', 'flex-row');
this.#elements.serverlist.header.container.appendChild(this.#elements.serverlist.header.addbutton);
this.#elements.serverlist.header.addbutton.appendChild(document.createTextNode('[Add server]'));
this.#elements.serverlist.header.addbutton.setAttributeNS('trweb', 'servermanagerbutton', 'add');
this.#elements.serverlist.header.addbutton.addEventListener('click', this);
this.#elements.serverlist.list.classList.add('d-flex', 'flex-column');
} }
handleEvent(e) { handleEvent(e) {
this.#log(6, `Handling event of type ${e.type}`); this.#log(6, `Handling event of type ${e.type}`);
switch (e.type) { switch (e.type) {
case 'click':
if (e.currentTarget.hasAttributeNS('trweb', 'servermanagerbutton')) {
switch (e.currentTarget.getAttributeNS('trweb', 'servermanagerbutton')) {
case 'add':
e.stopPropagation();
let addName = window.prompt('Enter name for the new server');
if (addName != null) {
if (addName == '') {
window.alert('No name was provided, aborting');
}
else {
if (this.#servers[addName] != null) {
window.alert('Server with the provided name already exists, aborting');
}
else {
let addUrl = window.prompt('Enter server URL', 'http://localhost:9091/transmission/rpc');
if (addUrl != null) {
if (addUrl == '') {
window.alert('No URL provided, aborting');
}
else {
let addUser;
let addPass;
addUser = window.prompt('Enter username, leave empty of not required');
if (addUser == null) {
return;
}
if (addUser == '') {
addUser = null;
}
else {
addPass = window.prompt(`Enter password for user ${addUser}`);
if (addPass == null) {
return;
}
}
this.addServer(addName, addUrl, addUser, addPass);
}
}
}
}
}
break;
case 'block':
e.stopPropagation();
break;
case 'delete':
e.stopPropagation();
let deleteSrv = e.currentTarget.getAttribute('server');
if (window.confirm(`Are you sure you want to delete server ${deleteSrv}?`)) {
this.deleteServer(deleteSrv);
}
break;
case 'hide':
e.stopPropagation();
document.body.removeChild(this.#elements.serverlist.shade);
break;
case 'manager':
e.stopPropagation();
document.body.appendChild(this.#elements.serverlist.shade);
break;
}
}
break;
case 'torrent-initialize': case 'torrent-initialize':
let initControls = {}; let initControls = {};
let initHash = e.detail.torrentHash; let initHash = e.detail.torrentHash;

View File

@@ -119,21 +119,14 @@ class trdom_torrentcontrol extends EventTarget {
e.stopPropagation(); e.stopPropagation();
break; break;
case 'magnet': case 'magnet':
/*let magneturl; e.stopPropagation();
for (const [srv, data] of Object.entries(this.#getKnownTorrents()[this.#hash].servers)) { if (this.magnet != null) {
if (data.magnetLink != null) { this.#server.torrent_add_url(this.magnet, true, null);
magneturl = data.magnetLink;
break;
}
}
if (magneturl != null) {
this.#server.torrent_add_url(magneturl, false, null);
} }
else { else {
this.#log(3, `Couldn't find magnet link! ${JSON.stringify(this.#getKnownTorrents()[this.#hash].servers)}`); this.#log(3, 'Couldn\'t find magnet link!');
}*/ window.alert('Couldn\'t find magnet link!');
e.stopPropagation(); }
window.alert('not implemented');
break; break;
case 'download': case 'download':
e.stopPropagation(); e.stopPropagation();
@@ -144,8 +137,10 @@ class trdom_torrentcontrol extends EventTarget {
break; break;
case 'torrent-deleted': case 'torrent-deleted':
this.#log(6, 'we be delet'); this.#log(6, 'we be delet');
this.#data = null; if (this.exists) {
delete this.#data.name;
this.updateDOM(); this.updateDOM();
}
break; break;
case 'torrent-updated': case 'torrent-updated':
if (e.detail.torrentHash == this.#hash) { if (e.detail.torrentHash == this.#hash) {

View File

@@ -17,6 +17,30 @@ class trserver extends EventTarget {
#sessionHeader; #sessionHeader;
#requests = []; #requests = [];
#elements = {
serverlist: {
row: document.createElement('div'),
name: document.createElement('div'),
buttons: {}
}
};
get element() {
return this.#elements.serverlist.row;
}
addServerListButton(name, button) {
if (this.#elements.serverlist.buttons[name] != null) {
this.removeServerListButton(name);
}
this.#elements.serverlist.buttons[name] = button;
this.#elements.serverlist.row.appendChild(button);
}
removeServerListButton(name) {
if (this.#elements.serverlist.buttons[name] != null) {
this.#elements.serverlist.row.removeChild(this.#elements.serverlist.buttons[name]);
delete this.#elements.serverlist.buttons[name];
}
}
#torrentControls = {}; #torrentControls = {};
addControl(hash, control) { addControl(hash, control) {
this.#torrentControls[hash] = control; this.#torrentControls[hash] = control;
@@ -50,6 +74,12 @@ class trserver extends EventTarget {
this.#rpcurl = initdata.rpcurl; this.#rpcurl = initdata.rpcurl;
this.#authHeader = initdata.auth; this.#authHeader = initdata.auth;
this.#sessionHeader = null; this.#sessionHeader = null;
this.#elements.serverlist.row.classList.add('d-flex', 'flex-row');
this.#elements.serverlist.row.appendChild(this.#elements.serverlist.name);
this.#elements.serverlist.name.classList.add('flex-grow-1');
this.#elements.serverlist.name.appendChild(document.createTextNode(this.name));
} }
#rpccall_prepare(method, args) { #rpccall_prepare(method, args) {
@@ -243,8 +273,9 @@ class trserver extends EventTarget {
getInstanceData() { getInstanceData() {
return { return {
'rpcurl': this.#rpcurl, type: 'transmission',
'auth': this.#authHeader rpcurl: this.#rpcurl,
auth: this.#authHeader
}; };
} }

View File

@@ -44,6 +44,8 @@ class trweb {
this.#dom_servermanager.addEventListener('torrentserver-added', this.#dom_torrentmanagger); this.#dom_servermanager.addEventListener('torrentserver-added', this.#dom_torrentmanagger);
this.#dom_servermanager.addEventListener('torrent-created', this.#dom_torrentmanagger); this.#dom_servermanager.addEventListener('torrent-created', this.#dom_torrentmanagger);
this.#dom_menubar.addMenuNode('servermanager', this.#dom_servermanager.menuentry);
this.#guiFooter = document.createElement('div'); this.#guiFooter = document.createElement('div');
this.#guiFooter.classList.add('trweb_footer'); this.#guiFooter.classList.add('trweb_footer');
@@ -65,13 +67,13 @@ class trweb {
this.#logger('Activating TRWEB instance'); this.#logger('Activating TRWEB instance');
this.#container = document.getElementById(this.#container); this.#container = document.getElementById(this.#container);
this.#container.classList.add('trweb_container'); this.#container.classList.add('trweb_container', 'd-flex', 'flex-column', 'h-100');
this.#container.appendChild(this.#dom_menubar.element); this.#container.appendChild(this.#dom_menubar.element);
this.#container.appendChild(this.#dom_torrentmanagger.element); this.#container.appendChild(this.#dom_torrentmanagger.element);
this.#container.appendChild(this.#guiFooter); this.#container.appendChild(this.#guiFooter);
this.#dom_servermanager.loadServers(); this.#dom_servermanager.loadServers();
//this.setTimer(); this.setTimer();
break; break;
default: default:
this.#log(5, `Event type ${e.type} not supported`); this.#log(5, `Event type ${e.type} not supported`);
@@ -80,14 +82,14 @@ class trweb {
} }
refresh() { refresh() {
this.#log(5,'Refresh'); this.#log(6,'Refresh');
for (const [key, value] of Object.entries(this.#dom_servermanager.getServers())) { for (const [key, value] of Object.entries(this.#dom_servermanager.getServers())) {
value.refreshTorrentList(); value.refreshTorrentList();
} }
} }
#timercb = function() { #timercb = function() {
this.#log(5,'Timer tick'); this.#log(6,'Timer tick');
this.refresh(); this.refresh();
}.bind(this); }.bind(this);
@@ -111,17 +113,9 @@ class trweb {
this.#dom_servermanager.addServer(name, url, user, pass); this.#dom_servermanager.addServer(name, url, user, pass);
} }
/*removeServer(name) { removeServer(name) {
delete this.#dom_servermanager.getServers()[name]; this.#dom_servermanager.removeServer(name);
for (const [hash, entry] of Object.entries(this.#knownTorrents)) {
entry.element.removeChild(entry.element_server[name]);
delete entry.servers[name];
delete entry.element_server[name]
} }
this.#dom_servermanager.saveServers();
}*/
} }
var trinstance = new trweb('trcontainer', null, null, function(msg) {console.info(`TRWEB: ${msg}`)}); var trinstance = new trweb('trcontainer', null, null, function(msg) {console.info(`TRWEB: ${msg}`)});

View File

@@ -1,5 +1,7 @@
.trweb_torrentlistview { .trweb_torrentlistview {
background-color: cadetblue; background-color: cadetblue;
overflow-x: clip;
overflow-y: auto;
} }
.trweb_torrentlistentry:nth-child(3n-1) { .trweb_torrentlistentry:nth-child(3n-1) {
background-color: darkcyan; background-color: darkcyan;
@@ -93,3 +95,12 @@
display: block; display: block;
} }
.trweb_formshade {
background-color: rgba(0,0,0,0.75);
height: 100%;
position:absolute;
top: 0;
left: 0;
width: 100%;
z-index: 10;
}