553 lines
18 KiB
JavaScript
553 lines
18 KiB
JavaScript
/*
|
|
Script: Deluge.Add.js
|
|
Contains the Add Torrent window.
|
|
|
|
Copyright:
|
|
(C) Damien Churchill 2009 <damoxc@gmail.com>
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 3, or (at your option)
|
|
any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, write to:
|
|
The Free Software Foundation, Inc.,
|
|
51 Franklin Street, Fifth Floor
|
|
Boston, MA 02110-1301, USA.
|
|
|
|
In addition, as a special exception, the copyright holders give
|
|
permission to link the code of portions of this program with the OpenSSL
|
|
library.
|
|
You must obey the GNU General Public License in all respects for all of
|
|
the code used other than OpenSSL. If you modify file(s) with this
|
|
exception, you may extend this exception to your version of the file(s),
|
|
but you are not obligated to do so. If you do not wish to do so, delete
|
|
this exception statement from your version. If you delete this exception
|
|
statement from all source files in the program, then also delete it here.
|
|
|
|
*/
|
|
|
|
Ext.namespace('Ext.deluge.add');
|
|
Ext.deluge.add.OptionsPanel = Ext.extend(Ext.TabPanel, {
|
|
|
|
torrents: {},
|
|
|
|
constructor: function(config) {
|
|
config = Ext.apply({
|
|
region: 'south',
|
|
margins: '5 5 5 5',
|
|
activeTab: 0,
|
|
height: 220
|
|
}, config);
|
|
Ext.deluge.add.OptionsPanel.superclass.constructor.call(this, config);
|
|
},
|
|
|
|
initComponent: function() {
|
|
Ext.deluge.add.OptionsPanel.superclass.initComponent.call(this);
|
|
this.files = this.add(new Ext.tree.ColumnTree({
|
|
layout: 'fit',
|
|
title: _('Files'),
|
|
rootVisible: false,
|
|
autoScroll: true,
|
|
height: 170,
|
|
border: false,
|
|
animate: false,
|
|
|
|
columns: [{
|
|
header: _('Filename'),
|
|
width: 275,
|
|
dataIndex: 'filename'
|
|
},{
|
|
header: _('Size'),
|
|
width: 80,
|
|
dataIndex: 'size'
|
|
}],
|
|
|
|
root: new Ext.tree.AsyncTreeNode({
|
|
text: 'Files'
|
|
})
|
|
}));
|
|
new Ext.tree.TreeSorter(this.files, {
|
|
folderSort: true
|
|
});
|
|
|
|
this.optionsManager = new Deluge.MultiOptionsManager();
|
|
|
|
this.form = this.add({
|
|
xtype: 'form',
|
|
labelWidth: 1,
|
|
title: _('Options'),
|
|
bodyStyle: 'padding: 5px;',
|
|
border: false,
|
|
height: 170,
|
|
disabled: true
|
|
});
|
|
|
|
var fieldset = this.form.add({
|
|
xtype: 'fieldset',
|
|
title: _('Download Location'),
|
|
border: false,
|
|
autoHeight: true,
|
|
defaultType: 'textfield',
|
|
labelWidth: 1,
|
|
fieldLabel: ''
|
|
});
|
|
this.optionsManager.bind('download_location', fieldset.add({
|
|
fieldLabel: '',
|
|
name: 'download_location',
|
|
width: 400,
|
|
labelSeparator: ''
|
|
}));
|
|
|
|
var panel = this.form.add({
|
|
border: false,
|
|
layout: 'column',
|
|
defaultType: 'fieldset'
|
|
});
|
|
fieldset = panel.add({
|
|
title: _('Allocation'),
|
|
border: false,
|
|
autoHeight: true,
|
|
defaultType: 'radio',
|
|
width: 100
|
|
});
|
|
|
|
this.optionsManager.bind('compact_allocation', fieldset.add({
|
|
xtype: 'radiogroup',
|
|
columns: 1,
|
|
vertical: true,
|
|
labelSeparator: '',
|
|
items: [{
|
|
name: 'compact_allocation',
|
|
value: false,
|
|
inputValue: false,
|
|
boxLabel: _('Full'),
|
|
fieldLabel: '',
|
|
labelSeparator: ''
|
|
}, {
|
|
name: 'compact_allocation',
|
|
value: true,
|
|
inputValue: true,
|
|
boxLabel: _('Compact'),
|
|
fieldLabel: '',
|
|
labelSeparator: '',
|
|
}]
|
|
}));
|
|
|
|
fieldset = panel.add({
|
|
title: _('Bandwidth'),
|
|
border: false,
|
|
autoHeight: true,
|
|
labelWidth: 100,
|
|
width: 200,
|
|
defaultType: 'uxspinner'
|
|
});
|
|
this.optionsManager.bind('max_download_speed', fieldset.add({
|
|
fieldLabel: _('Max Down Speed'),
|
|
/*labelStyle: 'margin-left: 10px',*/
|
|
name: 'max_download_speed',
|
|
width: 60
|
|
}));
|
|
this.optionsManager.bind('max_upload_speed', fieldset.add({
|
|
fieldLabel: _('Max Up Speed'),
|
|
/*labelStyle: 'margin-left: 10px',*/
|
|
name: 'max_upload_speed',
|
|
width: 60
|
|
}));
|
|
this.optionsManager.bind('max_connections', fieldset.add({
|
|
fieldLabel: _('Max Connections'),
|
|
/*labelStyle: 'margin-left: 10px',*/
|
|
name: 'max_connections',
|
|
width: 60
|
|
}));
|
|
this.optionsManager.bind('max_upload_slots', fieldset.add({
|
|
fieldLabel: _('Max Upload Slots'),
|
|
/*labelStyle: 'margin-left: 10px',*/
|
|
name: 'max_upload_slots',
|
|
width: 60
|
|
}));
|
|
|
|
fieldset = panel.add({
|
|
title: _('General'),
|
|
border: false,
|
|
autoHeight: true,
|
|
defaultType: 'checkbox'
|
|
});
|
|
this.optionsManager.bind('add_paused', fieldset.add({
|
|
name: 'add_paused',
|
|
boxLabel: _('Add In Paused State'),
|
|
fieldLabel: '',
|
|
labelSeparator: '',
|
|
}));
|
|
this.optionsManager.bind('prioritize_first_last_pieces', fieldset.add({
|
|
name: 'prioritize_first_last_pieces',
|
|
boxLabel: _('Prioritize First/Last Pieces'),
|
|
fieldLabel: '',
|
|
labelSeparator: '',
|
|
}));
|
|
|
|
this.form.on('render', this.onFormRender, this);
|
|
},
|
|
|
|
onFormRender: function(form) {
|
|
form.layout = new Ext.layout.FormLayout();
|
|
form.layout.setContainer(form);
|
|
form.doLayout();
|
|
|
|
this.optionsManager.changeId(null);
|
|
},
|
|
|
|
addTorrent: function(torrent) {
|
|
this.torrents[torrent['info_hash']] = torrent;
|
|
var fileIndexes = {};
|
|
this.walkFileTree(torrent['files_tree'], function(filename, type, entry, parent) {
|
|
if (type != 'file') return;
|
|
fileIndexes[entry[0]] = entry[2];
|
|
}, this);
|
|
|
|
var priorities = [];
|
|
Ext.each(Ext.keys(fileIndexes), function(index) {
|
|
priorities[index] = fileIndexes[index];
|
|
});
|
|
|
|
var oldId = this.optionsManager.changeId(torrent['info_hash'], false);
|
|
this.optionsManager.set('file_priorities', priorities);
|
|
this.optionsManager.changeId(oldId, false);
|
|
},
|
|
|
|
clear: function() {
|
|
this.clearFiles();
|
|
},
|
|
|
|
clearFiles: function() {
|
|
var root = this.files.getRootNode();
|
|
if (!root.hasChildNodes()) return;
|
|
root.cascade(function(node) {
|
|
if (!node.parentNode || !node.getOwnerTree()) return;
|
|
node.remove();
|
|
});
|
|
},
|
|
|
|
getDefaults: function() {
|
|
var keys = ['add_paused','compact_allocation','download_location',
|
|
'max_connections_per_torrent','max_download_speed_per_torrent',
|
|
'max_upload_slots_per_torrent','max_upload_speed_per_torrent',
|
|
'prioritize_first_last_pieces'];
|
|
|
|
Deluge.Client.core.get_config_values(keys, {
|
|
success: function(config) {
|
|
var options = {
|
|
'file_priorities': [],
|
|
'add_paused': config.add_paused,
|
|
'compact_allocation': config.compact_allocation,
|
|
'download_location': config.download_location,
|
|
'max_connections': config.max_connections_per_torrent,
|
|
'max_download_speed': config.max_download_speed_per_torrent,
|
|
'max_upload_slots': config.max_upload_slots_per_torrent,
|
|
'max_upload_speed': config.max_upload_speed_per_torrent,
|
|
'prioritize_first_last_pieces': config.prioritize_first_last_pieces
|
|
}
|
|
this.optionsManager.options = options;
|
|
},
|
|
scope: this
|
|
});
|
|
},
|
|
|
|
getFilename: function(torrentId) {
|
|
return this.torrents[torrentId]['filename'];
|
|
},
|
|
|
|
getOptions: function(torrentId) {
|
|
var oldId = this.optionsManager.changeId(torrentId, false);
|
|
var options = this.optionsManager.get();
|
|
this.optionsManager.changeId(oldTorrentId, false);
|
|
Ext.each(options['file_priorities'], function(priority, index) {
|
|
options['file_priorities'][index] = (priority) ? 1 : 0;
|
|
});
|
|
return options;
|
|
},
|
|
|
|
setTorrent: function(torrentId) {
|
|
if (!torrentId) return;
|
|
|
|
this.torrentId = torrentId;
|
|
this.optionsManager.changeId(torrentId);
|
|
|
|
this.clearFiles();
|
|
var root = this.files.getRootNode();
|
|
var priorities = this.optionsManager.get('file_priorities');
|
|
|
|
this.walkFileTree(this.torrents[torrentId]['files_tree'], function(filename, type, entry, parent) {
|
|
if (type == 'dir') {
|
|
var folder = new Ext.tree.TreeNode({
|
|
text: filename,
|
|
checked: true
|
|
});
|
|
folder.on('checkchange', this.onFolderCheck, this);
|
|
parent.appendChild(folder);
|
|
return folder;
|
|
} else {
|
|
var node = new Ext.tree.TreeNode({
|
|
filename: filename,
|
|
fileindex: entry[0],
|
|
text: filename, // this needs to be here for sorting reasons
|
|
size: fsize(entry[1]),
|
|
leaf: true,
|
|
checked: priorities[entry[0]],
|
|
iconCls: 'x-deluge-file',
|
|
uiProvider: Ext.tree.ColumnNodeUI
|
|
});
|
|
node.on('checkchange', this.onNodeCheck, this);
|
|
parent.appendChild(node);
|
|
}
|
|
}, this, root);
|
|
root.firstChild.expand();
|
|
},
|
|
|
|
walkFileTree: function(files, callback, scope, parent) {
|
|
for (var filename in files) {
|
|
var entry = files[filename];
|
|
var type = (Ext.type(entry) == 'object') ? 'dir' : 'file';
|
|
|
|
if (scope) {
|
|
var ret = callback.apply(scope, [filename, type, entry, parent]);
|
|
} else {
|
|
var ret = callback(filename, type, entry, parent);
|
|
}
|
|
|
|
if (type == 'dir') this.walkFileTree(entry, callback, scope, ret);
|
|
}
|
|
},
|
|
|
|
onFolderCheck: function(node, checked) {
|
|
var priorities = this.optionsManager.get('file_priorities');
|
|
node.cascade(function(child) {
|
|
if (!child.ui.checkbox) {
|
|
child.attributes.checked = checked;
|
|
} else {
|
|
child.ui.checkbox.checked = checked;
|
|
}
|
|
priorities[child.attributes.fileindex] = checked;
|
|
}, this);
|
|
this.optionsManager.update('file_priorities', priorities);
|
|
},
|
|
|
|
onNodeCheck: function(node, checked) {
|
|
var priorities = this.optionsManager.get('file_priorities');
|
|
priorities[node.attributes.fileindex] = checked;
|
|
this.optionsManager.update('file_priorities', priorities);
|
|
}
|
|
});
|
|
|
|
Ext.deluge.add.Window = Ext.extend(Ext.Window, {
|
|
initComponent: function() {
|
|
Ext.deluge.add.Window.superclass.initComponent.call(this);
|
|
this.addEvents(
|
|
'beforeadd',
|
|
'add'
|
|
);
|
|
},
|
|
|
|
createTorrentId: function() {
|
|
return new Date().getTime();
|
|
}
|
|
});
|
|
|
|
Ext.deluge.add.AddWindow = Ext.extend(Ext.deluge.add.Window, {
|
|
|
|
constructor: function(config) {
|
|
config = Ext.apply({
|
|
title: _('Add Torrents'),
|
|
layout: 'border',
|
|
width: 470,
|
|
height: 450,
|
|
bodyStyle: 'padding: 10px 5px;',
|
|
buttonAlign: 'right',
|
|
closeAction: 'hide',
|
|
closable: true,
|
|
plain: true,
|
|
iconCls: 'x-deluge-add-window-icon'
|
|
}, config);
|
|
Ext.deluge.add.AddWindow.superclass.constructor.call(this, config);
|
|
},
|
|
|
|
initComponent: function() {
|
|
Ext.deluge.add.AddWindow.superclass.initComponent.call(this);
|
|
|
|
this.addButton(_('Cancel'), this.onCancel, this);
|
|
this.addButton(_('Add'), this.onAdd, this);
|
|
|
|
function torrentRenderer(value, p, r) {
|
|
if (r.data['info_hash']) {
|
|
return String.format('<div class="x-deluge-add-torrent-name">{0}</div>', value);
|
|
} else {
|
|
return String.format('<div class="x-deluge-add-torrent-name-loading">{0}</div>', value);
|
|
}
|
|
}
|
|
|
|
this.grid = this.add({
|
|
xtype: 'grid',
|
|
region: 'center',
|
|
store: new Ext.data.SimpleStore({
|
|
fields: [
|
|
{name: 'info_hash', mapping: 1},
|
|
{name: 'text', mapping: 2}
|
|
],
|
|
id: 0
|
|
}),
|
|
columns: [{
|
|
id: 'torrent',
|
|
width: 150,
|
|
sortable: true,
|
|
renderer: torrentRenderer,
|
|
dataIndex: 'text'
|
|
}],
|
|
stripeRows: true,
|
|
selModel: new Ext.grid.RowSelectionModel({
|
|
singleSelect: true,
|
|
listeners: {
|
|
'rowselect': {
|
|
fn: this.onSelect,
|
|
scope: this
|
|
}
|
|
}
|
|
}),
|
|
hideHeaders: true,
|
|
autoExpandColumn: 'torrent',
|
|
deferredRender: false,
|
|
autoScroll: true,
|
|
margins: '5 5 5 5',
|
|
bbar: new Ext.Toolbar({
|
|
items: [{
|
|
id: 'file',
|
|
cls: 'x-btn-text-icon',
|
|
iconCls: 'x-deluge-add-file',
|
|
text: _('File'),
|
|
handler: this.onFile,
|
|
scope: this
|
|
}, {
|
|
id: 'url',
|
|
cls: 'x-btn-text-icon',
|
|
text: _('Url'),
|
|
icon: '/icons/add_url.png',
|
|
handler: this.onUrl,
|
|
scope: this
|
|
}, {
|
|
id: 'infohash',
|
|
cls: 'x-btn-text-icon',
|
|
text: _('Infohash'),
|
|
icon: '/icons/add_magnet.png',
|
|
disabled: true
|
|
}, '->', {
|
|
id: 'remove',
|
|
cls: 'x-btn-text-icon',
|
|
text: _('Remove'),
|
|
icon: '/icons/remove.png',
|
|
handler: this.onRemove,
|
|
scope: this
|
|
}]
|
|
})
|
|
});
|
|
|
|
this.optionsPanel = this.add(new Ext.deluge.add.OptionsPanel());
|
|
this.on('show', this.onShow, this);
|
|
},
|
|
|
|
clear: function() {
|
|
this.grid.getStore().removeAll();
|
|
this.optionsPanel.clear();
|
|
},
|
|
|
|
onAdd: function() {
|
|
var torrents = [];
|
|
this.grid.getStore().each(function(r) {
|
|
var id = r.get('info_hash');
|
|
torrents.push({
|
|
path: this.optionsPanel.getFilename(id),
|
|
options: this.optionsPanel.getOptions(id)
|
|
});
|
|
}, this);
|
|
|
|
Deluge.Client.web.add_torrents(torrents, {
|
|
success: function(result) {
|
|
}
|
|
})
|
|
this.clear();
|
|
this.hide();
|
|
},
|
|
|
|
onCancel: function() {
|
|
this.clear();
|
|
this.hide();
|
|
},
|
|
|
|
onFile: function() {
|
|
this.file.show();
|
|
},
|
|
|
|
onRemove: function() {
|
|
var selection = this.grid.getSelectionModel();
|
|
if (!selection.hasSelection()) return;
|
|
var torrent = selection.getSelected();
|
|
this.grid.getStore().remove(torrent);
|
|
this.optionsPanel.clear();
|
|
|
|
if (this.torrents && this.torrents[torrent.id]) delete this.torrents[torrent.id];
|
|
},
|
|
|
|
onSelect: function(selModel, rowIndex, record) {
|
|
this.optionsPanel.setTorrent(record.get('info_hash'));
|
|
},
|
|
|
|
onShow: function() {
|
|
if (!this.url) {
|
|
this.url = new Ext.deluge.add.UrlWindow();
|
|
this.url.on('beforeadd', this.onTorrentBeforeAdd, this);
|
|
this.url.on('add', this.onTorrentAdd, this);
|
|
}
|
|
|
|
if (!this.file) {
|
|
this.file = new Ext.deluge.add.FileWindow();
|
|
this.file.on('beforeadd', this.onTorrentBeforeAdd, this);
|
|
this.file.on('add', this.onTorrentAdd, this);
|
|
}
|
|
|
|
this.optionsPanel.getDefaults();
|
|
},
|
|
|
|
onTorrentBeforeAdd: function(torrentId, text) {
|
|
var store = this.grid.getStore();
|
|
store.loadData([[torrentId, null, text]], true);
|
|
},
|
|
|
|
onTorrentAdd: function(torrentId, info) {
|
|
if (!info) {
|
|
Ext.MessageBox.show({
|
|
title: _('Error'),
|
|
msg: _('Not a valid torrent'),
|
|
buttons: Ext.MessageBox.OK,
|
|
modal: false,
|
|
icon: Ext.MessageBox.ERROR,
|
|
iconCls: 'x-deluge-icon-error'
|
|
});
|
|
return;
|
|
}
|
|
|
|
var r = this.grid.getStore().getById(torrentId);
|
|
r.set('info_hash', info['info_hash']);
|
|
r.set('text', info['name']);
|
|
this.grid.getStore().commitChanges();
|
|
this.optionsPanel.addTorrent(info);
|
|
},
|
|
|
|
onUrl: function(button, event) {
|
|
this.url.show();
|
|
}
|
|
});
|
|
Deluge.Add = new Ext.deluge.add.AddWindow();
|