1
0
Fork 0
mirror of https://github.com/transmission/transmission synced 2025-01-02 13:05:08 +00:00
transmission/web/src/open-dialog.js
2023-02-11 14:49:42 -06:00

198 lines
5.7 KiB
JavaScript

/* @license This file Copyright © 2020-2023 Mnemosyne LLC.
It may be used under GPLv2 (SPDX: GPL-2.0-only), GPLv3 (SPDX: GPL-3.0-only),
or any future license endorsed by Mnemosyne LLC.
License text can be found in the licenses/ folder. */
import { AlertDialog } from './alert-dialog.js';
import { Formatter } from './formatter.js';
import { createDialogContainer, makeUUID } from './utils.js';
export class OpenDialog extends EventTarget {
constructor(controller, remote, url = '') {
super();
this.controller = controller;
this.remote = remote;
this.elements = this._create(url);
this.elements.dismiss.addEventListener('click', () => this._onDismiss());
this.elements.confirm.addEventListener('click', () => this._onConfirm());
this._updateFreeSpaceInAddDialog();
document.body.append(this.elements.root);
this.elements.url_input.focus();
}
close() {
if (!this.closed) {
clearInterval(this.interval);
this.elements.root.remove();
this.dispatchEvent(new Event('close'));
for (const key of Object.keys(this)) {
delete this[key];
}
this.closed = true;
}
}
_onDismiss() {
this.close();
}
_updateFreeSpaceInAddDialog() {
const path = this.elements.folder_input.value;
this.remote.getFreeSpace(path, (dir, bytes) => {
if (!this.closed) {
const string = bytes > 0 ? `${Formatter.size(bytes)} Free` : '';
this.elements.freespace.textContent = string;
}
});
}
_onConfirm() {
const { controller, elements, remote } = this;
const { file_input, folder_input, start_input, url_input } = elements;
const paused = !start_input.checked;
const destination = folder_input.value.trim();
for (const file of file_input.files) {
const reader = new FileReader();
reader.addEventListener('load', (e) => {
const contents = e.target.result;
const key = 'base64,';
const index = contents.indexOf(key);
if (index === -1) {
return;
}
const o = {
arguments: {
'download-dir': destination,
metainfo: contents.slice(Math.max(0, index + key.length)),
paused,
},
method: 'torrent-add',
};
remote.sendRequest(o, (response) => {
if (response.result !== 'success') {
alert(`Error adding "${file.name}": ${response.result}`);
controller.setCurrentPopup(
new AlertDialog({
heading: `Error adding "${file.name}"`,
message: response.result,
})
);
}
});
});
reader.readAsDataURL(file);
}
let url = url_input.value.trim();
if (url.length > 0) {
if (/^[\da-f]{40}$/i.test(url)) {
url = `magnet:?xt=urn:btih:${url}`;
}
const o = {
arguments: {
'download-dir': destination,
filename: url,
paused,
},
method: 'torrent-add',
};
remote.sendRequest(o, (payload) => {
if (payload.result !== 'success') {
controller.setCurrentPopup(
new AlertDialog({
heading: `Error adding "${url}"`,
message: payload.result,
})
);
}
});
}
this._onDismiss();
}
_create(url) {
const elements = createDialogContainer();
const { confirm, root, heading, workarea } = elements;
root.classList.add('open-torrent');
heading.textContent = 'Add Torrents';
confirm.textContent = 'Add';
let input_id = makeUUID();
let label = document.createElement('label');
label.setAttribute('for', input_id);
label.textContent = 'Please select torrent files to add:';
workarea.append(label);
let input = document.createElement('input');
input.type = 'file';
input.name = 'torrent-files[]';
input.id = input_id;
input.multiple = 'multiple';
workarea.append(input);
elements.file_input = input;
input_id = makeUUID();
label = document.createElement('label');
label.setAttribute('for', input_id);
label.textContent = 'Or enter a URL:';
workarea.append(label);
input = document.createElement('input');
input.type = 'url';
input.id = input_id;
input.value = url;
workarea.append(input);
elements.url_input = input;
input.addEventListener('keyup', ({ key }) => {
if (key === 'Enter') {
confirm.click();
}
});
input_id = makeUUID();
label = document.createElement('label');
label.id = 'add-dialog-folder-label';
label.for = input_id;
label.textContent = 'Destination folder: ';
workarea.append(label);
const freespace = document.createElement('span');
freespace.id = 'free-space-text';
label.append(freespace);
workarea.append(label);
elements.freespace = freespace;
input = document.createElement('input');
input.type = 'text';
input.id = 'add-dialog-folder-input';
input.addEventListener('change', () => this._updateFreeSpaceInAddDialog());
input.value = this.controller.session_properties['download-dir'];
workarea.append(input);
elements.folder_input = input;
const checkarea = document.createElement('div');
workarea.append(checkarea);
const check = document.createElement('input');
check.type = 'checkbox';
check.id = 'auto-start-check';
check.checked = this.controller.shouldAddedTorrentsStart();
checkarea.append(check);
elements.start_input = check;
label = document.createElement('label');
label.id = 'auto-start-label';
label.setAttribute('for', check.id);
label.textContent = 'Start when added';
checkarea.append(label);
return elements;
}
}