2014-10-25 14:53:29 +04:00
/*
2018-06-06 16:48:17 +03:00
* Bittorrent Client using Qt and libtorrent .
* Copyright ( C ) 2006 Christophe Dumez < chris @ qbittorrent . org >
2014-10-25 14:53:29 +04:00
*
* 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 .
*/
# include "transferlistdelegate.h"
# include <QApplication>
2019-05-16 08:41:29 +03:00
# include <QDateTime>
2018-06-06 16:48:17 +03:00
# include <QModelIndex>
2014-10-25 14:53:29 +04:00
# include <QPainter>
2018-06-06 16:48:17 +03:00
# include <QStyleOptionViewItem>
2014-10-25 14:53:29 +04:00
2019-01-02 10:01:32 +03:00
# if defined(Q_OS_WIN) || defined(Q_OS_MACOS)
2014-10-25 14:53:29 +04:00
# include <QProxyStyle>
# endif
2018-06-06 16:48:17 +03:00
# include "base/bittorrent/torrenthandle.h"
# include "base/preferences.h"
# include "base/types.h"
# include "base/unicodestrings.h"
# include "base/utils/misc.h"
# include "base/utils/string.h"
2018-06-14 12:54:23 +03:00
# include "transferlistmodel.h"
2018-06-06 16:48:17 +03:00
2016-01-24 12:15:39 +03:00
TransferListDelegate : : TransferListDelegate ( QObject * parent )
: QItemDelegate ( parent )
{
}
2014-10-25 14:53:29 +04:00
2018-06-06 16:48:17 +03:00
void TransferListDelegate : : paint ( QPainter * painter , const QStyleOptionViewItem & option , const QModelIndex & index ) const
2016-01-24 12:15:39 +03:00
{
2016-01-25 09:09:17 +03:00
painter - > save ( ) ;
2016-02-01 15:19:28 +03:00
bool isHideState = true ;
if ( Preferences : : instance ( ) - > getHideZeroComboValues ( ) = = 1 ) { // paused torrents only
2018-06-14 12:54:23 +03:00
QModelIndex stateIndex = index . sibling ( index . row ( ) , TransferListModel : : TR_STATUS ) ;
2017-02-18 03:38:40 +03:00
if ( stateIndex . data ( ) . value < BitTorrent : : TorrentState > ( ) ! = BitTorrent : : TorrentState : : PausedDownloading )
2016-02-01 15:19:28 +03:00
isHideState = false ;
}
const bool hideValues = Preferences : : instance ( ) - > getHideZeroValues ( ) & isHideState ;
2016-01-25 09:09:17 +03:00
2016-12-06 05:41:53 +03:00
QStyleOptionViewItem opt = QItemDelegate : : setOptions ( index , option ) ;
2016-01-25 09:09:17 +03:00
QItemDelegate : : drawBackground ( painter , opt , index ) ;
2016-01-24 12:15:39 +03:00
switch ( index . column ( ) ) {
2018-06-14 12:54:23 +03:00
case TransferListModel : : TR_AMOUNT_DOWNLOADED :
case TransferListModel : : TR_AMOUNT_UPLOADED :
case TransferListModel : : TR_AMOUNT_DOWNLOADED_SESSION :
case TransferListModel : : TR_AMOUNT_UPLOADED_SESSION :
case TransferListModel : : TR_AMOUNT_LEFT :
case TransferListModel : : TR_COMPLETED :
case TransferListModel : : TR_SIZE :
case TransferListModel : : TR_TOTAL_SIZE : {
2018-06-06 16:48:17 +03:00
qlonglong size = index . data ( ) . toLongLong ( ) ;
if ( hideValues & & ! size )
break ;
opt . displayAlignment = ( Qt : : AlignRight | Qt : : AlignVCenter ) ;
QItemDelegate : : drawDisplay ( painter , opt , option . rect , Utils : : Misc : : friendlyUnit ( size ) ) ;
}
2016-01-24 11:01:33 +03:00
break ;
2018-06-14 12:54:23 +03:00
case TransferListModel : : TR_ETA : {
2016-01-24 12:15:39 +03:00
opt . displayAlignment = Qt : : AlignRight | Qt : : AlignVCenter ;
QItemDelegate : : drawDisplay ( painter , opt , option . rect , Utils : : Misc : : userFriendlyDuration ( index . data ( ) . toLongLong ( ) ) ) ;
2015-06-30 11:03:46 +03:00
break ;
2016-01-24 12:15:39 +03:00
}
2018-06-14 12:54:23 +03:00
case TransferListModel : : TR_SEEDS :
case TransferListModel : : TR_PEERS : {
2018-06-06 16:48:17 +03:00
qlonglong value = index . data ( ) . toLongLong ( ) ;
qlonglong total = index . data ( Qt : : UserRole ) . toLongLong ( ) ;
if ( hideValues & & ( ! value & & ! total ) )
break ;
2018-07-21 08:28:13 +03:00
QString display = QString : : number ( value ) + " ( " + QString : : number ( total ) + ' ) ' ;
2018-06-06 16:48:17 +03:00
opt . displayAlignment = Qt : : AlignRight | Qt : : AlignVCenter ;
QItemDelegate : : drawDisplay ( painter , opt , opt . rect , display ) ;
}
2015-02-05 02:54:51 +03:00
break ;
2018-06-14 12:54:23 +03:00
case TransferListModel : : TR_STATUS : {
2018-06-06 16:48:17 +03:00
const auto state = index . data ( ) . value < BitTorrent : : TorrentState > ( ) ;
QString display = getStatusString ( state ) ;
QItemDelegate : : drawDisplay ( painter , opt , opt . rect , display ) ;
}
2015-03-28 22:09:13 +03:00
break ;
2018-06-14 12:54:23 +03:00
case TransferListModel : : TR_UPSPEED :
case TransferListModel : : TR_DLSPEED : {
2018-09-07 14:12:38 +03:00
const int speed = index . data ( ) . toInt ( ) ;
2018-06-06 16:48:17 +03:00
if ( hideValues & & ! speed )
break ;
opt . displayAlignment = Qt : : AlignRight | Qt : : AlignVCenter ;
QItemDelegate : : drawDisplay ( painter , opt , opt . rect , Utils : : Misc : : friendlyUnit ( speed , true ) ) ;
}
2015-02-05 02:54:51 +03:00
break ;
2018-06-14 12:54:23 +03:00
case TransferListModel : : TR_UPLIMIT :
case TransferListModel : : TR_DLLIMIT : {
2018-06-06 16:48:17 +03:00
const qlonglong limit = index . data ( ) . toLongLong ( ) ;
if ( hideValues & & ! limit )
break ;
opt . displayAlignment = Qt : : AlignRight | Qt : : AlignVCenter ;
QItemDelegate : : drawDisplay ( painter , opt , opt . rect , limit > 0 ? Utils : : Misc : : friendlyUnit ( limit , true ) : QString : : fromUtf8 ( C_INFINITY ) ) ;
}
2016-01-24 11:01:33 +03:00
break ;
2018-06-14 12:54:23 +03:00
case TransferListModel : : TR_TIME_ELAPSED : {
2018-06-06 16:48:17 +03:00
const int elapsedTime = index . data ( ) . toInt ( ) ;
const int seedingTime = index . data ( Qt : : UserRole ) . toInt ( ) ;
const QString txt = ( seedingTime > 0 )
? tr ( " %1 (seeded for %2) " , " e.g. 4m39s (seeded for 3m10s) " )
. arg ( Utils : : Misc : : userFriendlyDuration ( elapsedTime )
, Utils : : Misc : : userFriendlyDuration ( seedingTime ) )
: Utils : : Misc : : userFriendlyDuration ( elapsedTime ) ;
QItemDelegate : : drawDisplay ( painter , opt , opt . rect , txt ) ;
}
2016-01-24 11:01:33 +03:00
break ;
2018-06-14 12:54:23 +03:00
case TransferListModel : : TR_ADD_DATE :
case TransferListModel : : TR_SEED_DATE :
2016-01-24 12:15:39 +03:00
QItemDelegate : : drawDisplay ( painter , opt , opt . rect , index . data ( ) . toDateTime ( ) . toLocalTime ( ) . toString ( Qt : : DefaultLocaleShortDate ) ) ;
break ;
2018-06-14 12:54:23 +03:00
case TransferListModel : : TR_RATIO_LIMIT :
case TransferListModel : : TR_RATIO : {
2018-06-06 16:48:17 +03:00
const qreal ratio = index . data ( ) . toDouble ( ) ;
if ( hideValues & & ( ratio < = 0 ) )
break ;
QString str = ( ( ratio = = - 1 ) | | ( ratio > BitTorrent : : TorrentHandle : : MAX_RATIO ) ) ? QString : : fromUtf8 ( C_INFINITY ) : Utils : : String : : fromDouble ( ratio , 2 ) ;
opt . displayAlignment = Qt : : AlignRight | Qt : : AlignVCenter ;
QItemDelegate : : drawDisplay ( painter , opt , opt . rect , str ) ;
}
2016-01-24 12:15:39 +03:00
break ;
2018-06-14 12:54:23 +03:00
case TransferListModel : : TR_PRIORITY : {
2018-06-06 16:48:17 +03:00
const int priority = index . data ( ) . toInt ( ) ;
opt . displayAlignment = Qt : : AlignRight | Qt : : AlignVCenter ;
if ( priority > 0 ) {
QItemDelegate : : paint ( painter , opt , index ) ;
}
else {
QItemDelegate : : drawDisplay ( painter , opt , opt . rect , " * " ) ;
}
2016-01-24 12:15:39 +03:00
}
break ;
2018-06-14 12:54:23 +03:00
case TransferListModel : : TR_PROGRESS : {
2018-06-06 16:48:17 +03:00
QStyleOptionProgressBar newopt ;
qreal progress = index . data ( ) . toDouble ( ) * 100. ;
newopt . rect = opt . rect ;
2018-07-21 08:28:13 +03:00
newopt . text = ( ( progress = = 100.0 ) ? QString ( " 100% " ) : Utils : : String : : fromDouble ( progress , 1 ) + ' % ' ) ;
2018-06-06 16:48:17 +03:00
newopt . progress = static_cast < int > ( progress ) ;
newopt . maximum = 100 ;
newopt . minimum = 0 ;
newopt . state | = QStyle : : State_Enabled ;
newopt . textVisible = true ;
2019-01-02 10:01:32 +03:00
# if !defined(Q_OS_WIN) && !defined(Q_OS_MACOS)
2018-06-06 16:48:17 +03:00
QApplication : : style ( ) - > drawControl ( QStyle : : CE_ProgressBar , & newopt , painter ) ;
2014-10-25 14:53:29 +04:00
# else
2018-06-06 16:48:17 +03:00
// XXX: To avoid having the progress text on the right of the bar
QProxyStyle st ( " fusion " ) ;
st . drawControl ( QStyle : : CE_ProgressBar , & newopt , painter , 0 ) ;
2014-10-25 14:53:29 +04:00
# endif
2018-06-06 16:48:17 +03:00
}
2016-01-24 11:01:33 +03:00
break ;
2018-06-14 12:54:23 +03:00
case TransferListModel : : TR_LAST_ACTIVITY : {
2018-06-06 16:48:17 +03:00
qlonglong elapsed = index . data ( ) . toLongLong ( ) ;
if ( hideValues & & ( ( elapsed < 0 ) | | ( elapsed > = MAX_ETA ) ) )
break ;
2016-01-24 11:01:33 +03:00
2018-06-06 16:48:17 +03:00
// Show '< 1m ago' when elapsed time is 0
if ( elapsed = = 0 )
elapsed = 1 ;
2017-09-22 07:42:24 +03:00
2018-06-06 16:48:17 +03:00
QString elapsedString = ( elapsed > = 0 )
? tr ( " %1 ago " , " e.g.: 1h 20m ago " ) . arg ( Utils : : Misc : : userFriendlyDuration ( elapsed ) )
: Utils : : Misc : : userFriendlyDuration ( elapsed ) ;
2017-09-22 07:42:24 +03:00
2018-06-06 16:48:17 +03:00
opt . displayAlignment = Qt : : AlignRight | Qt : : AlignVCenter ;
QItemDelegate : : drawDisplay ( painter , opt , option . rect , elapsedString ) ;
}
2016-01-24 12:15:39 +03:00
break ;
default :
QItemDelegate : : paint ( painter , option , index ) ;
}
painter - > restore ( ) ;
2014-10-25 14:53:29 +04:00
}
2018-06-06 16:48:17 +03:00
QWidget * TransferListDelegate : : createEditor ( QWidget * , const QStyleOptionViewItem & , const QModelIndex & ) const
2016-01-24 12:15:39 +03:00
{
// No editor here
2018-06-06 16:48:17 +03:00
return nullptr ;
2014-10-25 14:53:29 +04:00
}
2018-06-06 16:48:17 +03:00
QSize TransferListDelegate : : sizeHint ( const QStyleOptionViewItem & option , const QModelIndex & index ) const
2016-01-24 12:15:39 +03:00
{
2016-04-04 12:58:21 +03:00
// Reimplementing sizeHint() because the 'name' column contains text+icon.
// When that WHOLE column goes out of view(eg user scrolls horizontally)
// the rows shrink if the text's height is smaller than the icon's height.
// This happens because icon from the 'name' column is no longer drawn.
static int nameColHeight = - 1 ;
if ( nameColHeight = = - 1 ) {
2018-06-14 12:54:23 +03:00
QModelIndex nameColumn = index . sibling ( index . row ( ) , TransferListModel : : TR_NAME ) ;
2016-04-04 12:58:21 +03:00
nameColHeight = QItemDelegate : : sizeHint ( option , nameColumn ) . height ( ) ;
2016-01-24 12:15:39 +03:00
}
2014-10-25 14:53:29 +04:00
2016-01-25 09:09:17 +03:00
QSize size = QItemDelegate : : sizeHint ( option , index ) ;
2016-04-04 12:58:21 +03:00
size . setHeight ( std : : max ( nameColHeight , size . height ( ) ) ) ;
2016-01-24 12:15:39 +03:00
return size ;
2014-10-25 14:53:29 +04:00
}
2016-01-25 08:54:13 +03:00
2017-02-18 03:38:40 +03:00
QString TransferListDelegate : : getStatusString ( const BitTorrent : : TorrentState state ) const
2016-01-25 08:54:13 +03:00
{
QString str ;
switch ( state ) {
case BitTorrent : : TorrentState : : Downloading :
str = tr ( " Downloading " ) ;
break ;
case BitTorrent : : TorrentState : : StalledDownloading :
str = tr ( " Stalled " , " Torrent is waiting for download to begin " ) ;
break ;
case BitTorrent : : TorrentState : : DownloadingMetadata :
str = tr ( " Downloading metadata " , " used when loading a magnet link " ) ;
break ;
case BitTorrent : : TorrentState : : ForcedDownloading :
str = tr ( " [F] Downloading " , " used when the torrent is forced started. You probably shouldn't translate the F. " ) ;
break ;
case BitTorrent : : TorrentState : : Allocating :
str = tr ( " Allocating " , " qBittorrent is allocating the files on disk " ) ;
break ;
case BitTorrent : : TorrentState : : Uploading :
case BitTorrent : : TorrentState : : StalledUploading :
str = tr ( " Seeding " , " Torrent is complete and in upload-only mode " ) ;
break ;
case BitTorrent : : TorrentState : : ForcedUploading :
str = tr ( " [F] Seeding " , " used when the torrent is forced started. You probably shouldn't translate the F. " ) ;
break ;
case BitTorrent : : TorrentState : : QueuedDownloading :
case BitTorrent : : TorrentState : : QueuedUploading :
str = tr ( " Queued " , " i.e. torrent is queued " ) ;
break ;
case BitTorrent : : TorrentState : : CheckingDownloading :
case BitTorrent : : TorrentState : : CheckingUploading :
str = tr ( " Checking " , " Torrent local data is being checked " ) ;
break ;
case BitTorrent : : TorrentState : : CheckingResumeData :
str = tr ( " Checking resume data " , " used when loading the torrents from disk after qbt is launched. It checks the correctness of the .fastresume file. Normally it is completed in a fraction of a second, unless loading many many torrents. " ) ;
break ;
case BitTorrent : : TorrentState : : PausedDownloading :
str = tr ( " Paused " ) ;
break ;
case BitTorrent : : TorrentState : : PausedUploading :
str = tr ( " Completed " ) ;
break ;
2018-04-09 18:19:33 +03:00
case BitTorrent : : TorrentState : : Moving :
str = tr ( " Moving " , " Torrent local data are being moved/relocated " ) ;
break ;
2016-01-25 08:54:13 +03:00
case BitTorrent : : TorrentState : : MissingFiles :
str = tr ( " Missing Files " ) ;
break ;
case BitTorrent : : TorrentState : : Error :
str = tr ( " Errored " , " torrent status, the torrent has an error " ) ;
break ;
default :
str = " " ;
}
return str ;
}