/* * Bittorrent Client using Qt and libtorrent. * Copyright (C) 2009 Christophe Dumez * * 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 2 * of the License, 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 this program with the OpenSSL project's "OpenSSL" library (or with * modified versions of it that use the same license as the "OpenSSL" library), * and distribute the linked executables. 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), 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. */ "use strict"; window.qBittorrent ??= {}; window.qBittorrent.PropTrackers ??= (() => { const exports = () => { return { updateData: updateData }; }; let current_hash = ""; const torrentTrackersTable = new window.qBittorrent.DynamicTable.TorrentTrackersTable(); let loadTrackersDataTimer = -1; const loadTrackersData = function() { if ($("propTrackers").hasClass("invisible") || $("propertiesPanel_collapseToggle").hasClass("panel-expand")) { // Tab changed, don't do anything return; } const new_hash = torrentsTable.getCurrentTorrentID(); if (new_hash === "") { torrentTrackersTable.clear(); clearTimeout(loadTrackersDataTimer); loadTrackersDataTimer = loadTrackersData.delay(10000); return; } if (new_hash !== current_hash) { torrentTrackersTable.clear(); current_hash = new_hash; } const url = new URI("api/v2/torrents/trackers?hash=" + current_hash); new Request.JSON({ url: url, method: "get", noCache: true, onComplete: function() { clearTimeout(loadTrackersDataTimer); loadTrackersDataTimer = loadTrackersData.delay(10000); }, onSuccess: function(trackers) { const selectedTrackers = torrentTrackersTable.selectedRowsIds(); torrentTrackersTable.clear(); if (trackers) { trackers.each((tracker) => { let status; switch (tracker.status) { case 0: status = "Disabled"; break; case 1: status = "Not contacted yet"; break; case 2: status = "Working"; break; case 3: status = "Updating..."; break; case 4: status = "Not working"; break; } const row = { rowId: tracker.url, tier: (tracker.tier >= 0) ? tracker.tier : "", url: tracker.url, status: status, peers: (tracker.num_peers >= 0) ? tracker.num_peers : "N/A", seeds: (tracker.num_seeds >= 0) ? tracker.num_seeds : "N/A", leeches: (tracker.num_leeches >= 0) ? tracker.num_leeches : "N/A", downloaded: (tracker.num_downloaded >= 0) ? tracker.num_downloaded : "N/A", message: tracker.msg }; torrentTrackersTable.updateRowData(row); }); torrentTrackersTable.updateTable(false); if (selectedTrackers.length > 0) torrentTrackersTable.reselectRows(selectedTrackers); } } }).send(); }; const updateData = function() { clearTimeout(loadTrackersDataTimer); loadTrackersDataTimer = -1; loadTrackersData(); }; const torrentTrackersContextMenu = new window.qBittorrent.ContextMenu.ContextMenu({ targets: "#torrentTrackersTableDiv", menu: "torrentTrackersMenu", actions: { AddTracker: function(element, ref) { addTrackerFN(); }, EditTracker: function(element, ref) { // only allow editing of one row element.firstChild.click(); editTrackerFN(element); }, RemoveTracker: function(element, ref) { removeTrackerFN(element); } }, offsets: { x: -15, y: 2 }, onShow: function() { const selectedTrackers = torrentTrackersTable.selectedRowsIds(); const containsStaticTracker = selectedTrackers.some((tracker) => { return (tracker.indexOf("** [") === 0); }); if (containsStaticTracker || (selectedTrackers.length === 0)) { this.hideItem("EditTracker"); this.hideItem("RemoveTracker"); this.hideItem("CopyTrackerUrl"); } else { this.showItem("EditTracker"); this.showItem("RemoveTracker"); this.showItem("CopyTrackerUrl"); } } }); const addTrackerFN = function() { if (current_hash.length === 0) return; new MochaUI.Window({ id: "trackersPage", icon: "images/qbittorrent-tray.svg", title: "Add trackers", loadMethod: "iframe", contentURL: "addtrackers.html?hash=" + current_hash, scrollbars: true, resizable: false, maximizable: false, closable: true, paddingVertical: 0, paddingHorizontal: 0, width: 500, height: 250, onCloseComplete: function() { updateData(); } }); }; const editTrackerFN = function(element) { if (current_hash.length === 0) return; const trackerUrl = encodeURIComponent(element.childNodes[1].textContent); new MochaUI.Window({ id: "trackersPage", icon: "images/qbittorrent-tray.svg", title: "Tracker editing", loadMethod: "iframe", contentURL: "edittracker.html?hash=" + current_hash + "&url=" + trackerUrl, scrollbars: true, resizable: false, maximizable: false, closable: true, paddingVertical: 0, paddingHorizontal: 0, width: 500, height: 150, onCloseComplete: function() { updateData(); } }); }; const removeTrackerFN = function(element) { if (current_hash.length === 0) return; const selectedTrackers = torrentTrackersTable.selectedRowsIds(); new Request({ url: "api/v2/torrents/removeTrackers", method: "post", data: { hash: current_hash, urls: selectedTrackers.map(encodeURIComponent).join("|") }, onSuccess: function() { updateData(); } }).send(); }; new ClipboardJS("#CopyTrackerUrl", { text: function(trigger) { return torrentTrackersTable.selectedRowsIds().join("\n"); } }); torrentTrackersTable.setup("torrentTrackersTableDiv", "torrentTrackersTableFixedHeaderDiv", torrentTrackersContextMenu); return exports(); })(); Object.freeze(window.qBittorrent.PropTrackers);