425 lines
14 KiB
Plaintext
425 lines
14 KiB
Plaintext
// This file Copyright © 2010-2022 Transmission authors and contributors.
|
|
// It may be used under the MIT (SPDX: MIT) license.
|
|
// License text can be found in the licenses/ folder.
|
|
|
|
#include <libtransmission/transmission.h>
|
|
#include <libtransmission/utils.h> //tr_getRatio()
|
|
|
|
#import "InfoActivityViewController.h"
|
|
#import "NSStringAdditions.h"
|
|
#import "PiecesView.h"
|
|
#import "Torrent.h"
|
|
|
|
#define PIECES_CONTROL_PROGRESS 0
|
|
#define PIECES_CONTROL_AVAILABLE 1
|
|
|
|
#define STACKVIEW_INSET 12.0
|
|
#define STACKVIEW_HORIZONTAL_SPACING 20.0
|
|
#define STACKVIEW_VERTICAL_SPACING 8.0
|
|
|
|
@interface InfoActivityViewController ()
|
|
|
|
@property(nonatomic, copy) NSArray<Torrent*>* fTorrents;
|
|
|
|
@property(nonatomic) BOOL fSet;
|
|
|
|
@property(nonatomic) IBOutlet NSTextField* fDateAddedField;
|
|
@property(nonatomic) IBOutlet NSTextField* fDateCompletedField;
|
|
@property(nonatomic) IBOutlet NSTextField* fDateActivityField;
|
|
@property(nonatomic) IBOutlet NSTextField* fStateField;
|
|
@property(nonatomic) IBOutlet NSTextField* fProgressField;
|
|
@property(nonatomic) IBOutlet NSTextField* fHaveField;
|
|
@property(nonatomic) IBOutlet NSTextField* fDownloadedTotalField;
|
|
@property(nonatomic) IBOutlet NSTextField* fUploadedTotalField;
|
|
@property(nonatomic) IBOutlet NSTextField* fFailedHashField;
|
|
@property(nonatomic) IBOutlet NSTextField* fRatioField;
|
|
@property(nonatomic) IBOutlet NSTextField* fDownloadTimeField;
|
|
@property(nonatomic) IBOutlet NSTextField* fSeedTimeField;
|
|
@property(nonatomic) IBOutlet NSTextView* fErrorMessageView;
|
|
|
|
@property(nonatomic) IBOutlet PiecesView* fPiecesView;
|
|
@property(nonatomic) IBOutlet NSSegmentedControl* fPiecesControl;
|
|
|
|
//remove when we switch to auto layout
|
|
@property(nonatomic) IBOutlet NSTextField* fTransferSectionLabel;
|
|
@property(nonatomic) IBOutlet NSTextField* fDatesSectionLabel;
|
|
@property(nonatomic) IBOutlet NSTextField* fTimeSectionLabel;
|
|
@property(nonatomic) IBOutlet NSTextField* fStateLabel;
|
|
@property(nonatomic) IBOutlet NSTextField* fProgressLabel;
|
|
@property(nonatomic) IBOutlet NSTextField* fHaveLabel;
|
|
@property(nonatomic) IBOutlet NSTextField* fDownloadedLabel;
|
|
@property(nonatomic) IBOutlet NSTextField* fUploadedLabel;
|
|
@property(nonatomic) IBOutlet NSTextField* fFailedDLLabel;
|
|
@property(nonatomic) IBOutlet NSTextField* fRatioLabel;
|
|
@property(nonatomic) IBOutlet NSTextField* fErrorLabel;
|
|
@property(nonatomic) IBOutlet NSTextField* fDateAddedLabel;
|
|
@property(nonatomic) IBOutlet NSTextField* fDateCompletedLabel;
|
|
@property(nonatomic) IBOutlet NSTextField* fDateActivityLabel;
|
|
@property(nonatomic) IBOutlet NSTextField* fDownloadTimeLabel;
|
|
@property(nonatomic) IBOutlet NSTextField* fSeedTimeLabel;
|
|
@property(nonatomic) IBOutlet NSScrollView* fErrorScrollView;
|
|
|
|
@property(nonatomic) IBOutlet NSStackView* fActivityStackView;
|
|
@property(nonatomic) IBOutlet NSView* fDatesView;
|
|
@property(nonatomic, readonly) CGFloat currentHeight;
|
|
@property(nonatomic, readonly) CGFloat horizLayoutHeight;
|
|
@property(nonatomic, readonly) CGFloat horizLayoutWidth;
|
|
@property(nonatomic, readonly) CGFloat vertLayoutHeight;
|
|
|
|
- (void)setupInfo;
|
|
|
|
@end
|
|
|
|
@implementation InfoActivityViewController
|
|
|
|
- (instancetype)init
|
|
{
|
|
if ((self = [super initWithNibName:@"InfoActivityView" bundle:nil]))
|
|
{
|
|
self.title = NSLocalizedString(@"Activity", "Inspector view -> title");
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
- (void)awakeFromNib
|
|
{
|
|
[self.fTransferSectionLabel sizeToFit];
|
|
[self.fDatesSectionLabel sizeToFit];
|
|
[self.fTimeSectionLabel sizeToFit];
|
|
|
|
NSArray* labels = @[
|
|
self.fStateLabel,
|
|
self.fProgressLabel,
|
|
self.fHaveLabel,
|
|
self.fDownloadedLabel,
|
|
self.fUploadedLabel,
|
|
self.fFailedDLLabel,
|
|
self.fRatioLabel,
|
|
self.fErrorLabel,
|
|
self.fDateAddedLabel,
|
|
self.fDateCompletedLabel,
|
|
self.fDateActivityLabel,
|
|
self.fDownloadTimeLabel,
|
|
self.fSeedTimeLabel
|
|
];
|
|
|
|
CGFloat oldMaxWidth = 0.0, originX = 0.0, newMaxWidth = 0.0;
|
|
for (NSTextField* label in labels)
|
|
{
|
|
NSRect const oldFrame = label.frame;
|
|
if (oldFrame.size.width > oldMaxWidth)
|
|
{
|
|
oldMaxWidth = oldFrame.size.width;
|
|
originX = oldFrame.origin.x;
|
|
}
|
|
|
|
[label sizeToFit];
|
|
CGFloat const newWidth = label.bounds.size.width;
|
|
if (newWidth > newMaxWidth)
|
|
{
|
|
newMaxWidth = newWidth;
|
|
}
|
|
}
|
|
|
|
for (NSTextField* label in labels)
|
|
{
|
|
NSRect frame = label.frame;
|
|
frame.origin.x = originX + (newMaxWidth - frame.size.width);
|
|
label.frame = frame;
|
|
}
|
|
|
|
NSArray* fields = @[
|
|
self.fDateAddedField,
|
|
self.fDateCompletedField,
|
|
self.fDateActivityField,
|
|
self.fStateField,
|
|
self.fProgressField,
|
|
self.fHaveField,
|
|
self.fDownloadedTotalField,
|
|
self.fUploadedTotalField,
|
|
self.fFailedHashField,
|
|
self.fRatioField,
|
|
self.fDownloadTimeField,
|
|
self.fSeedTimeField,
|
|
self.fErrorScrollView
|
|
];
|
|
|
|
CGFloat const widthIncrease = newMaxWidth - oldMaxWidth;
|
|
for (NSView* field in fields)
|
|
{
|
|
NSRect frame = field.frame;
|
|
frame.origin.x += widthIncrease;
|
|
frame.size.width -= widthIncrease;
|
|
field.frame = frame;
|
|
}
|
|
|
|
//set the click action of the pieces view
|
|
#warning after 2.8 just hook this up in the xib
|
|
self.fPiecesView.action = @selector(updatePiecesView:);
|
|
self.fPiecesView.target = self;
|
|
}
|
|
|
|
- (void)dealloc
|
|
{
|
|
[NSNotificationCenter.defaultCenter removeObserver:self];
|
|
}
|
|
|
|
- (CGFloat)currentHeight
|
|
{
|
|
return NSHeight(self.view.frame);
|
|
}
|
|
|
|
- (CGFloat)horizLayoutHeight
|
|
{
|
|
return NSHeight(self.fTransferView.frame) + 2 * STACKVIEW_INSET;
|
|
}
|
|
|
|
- (CGFloat)horizLayoutWidth
|
|
{
|
|
return NSWidth(self.fTransferView.frame) + NSWidth(self.fDatesView.frame) + (2 * STACKVIEW_INSET) + STACKVIEW_HORIZONTAL_SPACING;
|
|
}
|
|
|
|
- (CGFloat)vertLayoutHeight
|
|
{
|
|
return NSHeight(self.fTransferView.frame) + NSHeight(self.fDatesView.frame) + (2 * STACKVIEW_INSET) + STACKVIEW_VERTICAL_SPACING;
|
|
}
|
|
|
|
- (CGFloat)changeInWindowHeight
|
|
{
|
|
CGFloat difference = 0;
|
|
|
|
if (NSWidth(self.view.window.frame) >= self.horizLayoutWidth + 1)
|
|
{
|
|
self.fActivityStackView.orientation = NSUserInterfaceLayoutOrientationHorizontal;
|
|
|
|
//add some padding between views in horizontal layout
|
|
self.fActivityStackView.spacing = STACKVIEW_HORIZONTAL_SPACING;
|
|
|
|
difference = NSHeight(self.view.frame) - self.horizLayoutHeight;
|
|
}
|
|
else
|
|
{
|
|
self.fActivityStackView.orientation = NSUserInterfaceLayoutOrientationVertical;
|
|
self.fActivityStackView.spacing = STACKVIEW_VERTICAL_SPACING;
|
|
|
|
difference = NSHeight(self.view.frame) - self.vertLayoutHeight;
|
|
}
|
|
|
|
return difference;
|
|
}
|
|
|
|
- (NSRect)viewRect
|
|
{
|
|
CGFloat difference = self.changeInWindowHeight;
|
|
|
|
NSRect windowRect = self.view.window.frame, viewRect = self.view.frame;
|
|
if (difference != 0)
|
|
{
|
|
viewRect.size.height -= difference;
|
|
viewRect.size.width = NSWidth(windowRect);
|
|
}
|
|
|
|
return viewRect;
|
|
}
|
|
|
|
- (void)updateWindowLayout
|
|
{
|
|
CGFloat difference = self.changeInWindowHeight;
|
|
|
|
if (difference != 0)
|
|
{
|
|
NSRect windowRect = self.view.window.frame;
|
|
windowRect.origin.y += difference;
|
|
windowRect.size.height -= difference;
|
|
|
|
self.view.window.minSize = NSMakeSize(self.view.window.minSize.width, NSHeight(windowRect));
|
|
self.view.window.maxSize = NSMakeSize(FLT_MAX, NSHeight(windowRect));
|
|
|
|
self.view.frame = [self viewRect];
|
|
[self.view.window setFrame:windowRect display:YES animate:YES];
|
|
}
|
|
}
|
|
|
|
- (void)setInfoForTorrents:(NSArray<Torrent*>*)torrents
|
|
{
|
|
//don't check if it's the same in case the metadata changed
|
|
self.fTorrents = torrents;
|
|
|
|
self.fSet = NO;
|
|
}
|
|
|
|
- (void)updateInfo
|
|
{
|
|
if (!self.fSet)
|
|
{
|
|
[self setupInfo];
|
|
}
|
|
|
|
NSInteger const numberSelected = self.fTorrents.count;
|
|
if (numberSelected == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
uint64_t have = 0, haveVerified = 0, downloadedTotal = 0, uploadedTotal = 0, failedHash = 0;
|
|
NSDate* lastActivity = nil;
|
|
for (Torrent* torrent in self.fTorrents)
|
|
{
|
|
have += torrent.haveTotal;
|
|
haveVerified += torrent.haveVerified;
|
|
downloadedTotal += torrent.downloadedTotal;
|
|
uploadedTotal += torrent.uploadedTotal;
|
|
failedHash += torrent.failedHash;
|
|
|
|
NSDate* nextLastActivity;
|
|
if ((nextLastActivity = torrent.dateActivity))
|
|
{
|
|
lastActivity = lastActivity ? [lastActivity laterDate:nextLastActivity] : nextLastActivity;
|
|
}
|
|
}
|
|
|
|
if (have == 0)
|
|
{
|
|
self.fHaveField.stringValue = [NSString stringForFileSize:0];
|
|
}
|
|
else
|
|
{
|
|
NSString* verifiedString = [NSString stringWithFormat:NSLocalizedString(@"%@ verified", "Inspector -> Activity tab -> have"),
|
|
[NSString stringForFileSize:haveVerified]];
|
|
if (have == haveVerified)
|
|
{
|
|
self.fHaveField.stringValue = verifiedString;
|
|
}
|
|
else
|
|
{
|
|
self.fHaveField.stringValue = [NSString stringWithFormat:@"%@ (%@)", [NSString stringForFileSize:have], verifiedString];
|
|
}
|
|
}
|
|
|
|
self.fDownloadedTotalField.stringValue = [NSString stringForFileSize:downloadedTotal];
|
|
self.fUploadedTotalField.stringValue = [NSString stringForFileSize:uploadedTotal];
|
|
self.fFailedHashField.stringValue = [NSString stringForFileSize:failedHash];
|
|
|
|
self.fDateActivityField.objectValue = lastActivity;
|
|
|
|
if (numberSelected == 1)
|
|
{
|
|
Torrent* torrent = self.fTorrents[0];
|
|
|
|
self.fStateField.stringValue = torrent.stateString;
|
|
|
|
NSString* progressString = [NSString percentString:torrent.progress longDecimals:YES];
|
|
if (torrent.folder)
|
|
{
|
|
NSString* progressSelectedString = [NSString
|
|
stringWithFormat:NSLocalizedString(@"%@ selected", "Inspector -> Activity tab -> progress"),
|
|
[NSString percentString:torrent.progressDone longDecimals:YES]];
|
|
progressString = [progressString stringByAppendingFormat:@" (%@)", progressSelectedString];
|
|
}
|
|
self.fProgressField.stringValue = progressString;
|
|
|
|
self.fRatioField.stringValue = [NSString stringForRatio:torrent.ratio];
|
|
|
|
NSString* errorMessage = torrent.errorMessage;
|
|
if (![errorMessage isEqualToString:self.fErrorMessageView.string])
|
|
self.fErrorMessageView.string = errorMessage;
|
|
|
|
self.fDateCompletedField.objectValue = torrent.dateCompleted;
|
|
|
|
//uses a relative date, so can't be set once
|
|
self.fDateAddedField.objectValue = torrent.dateAdded;
|
|
|
|
static NSDateComponentsFormatter* timeFormatter;
|
|
static dispatch_once_t onceToken;
|
|
dispatch_once(&onceToken, ^{
|
|
timeFormatter = [NSDateComponentsFormatter new];
|
|
timeFormatter.unitsStyle = NSDateComponentsFormatterUnitsStyleShort;
|
|
timeFormatter.allowedUnits = NSCalendarUnitDay | NSCalendarUnitHour | NSCalendarUnitMinute | NSCalendarUnitSecond;
|
|
timeFormatter.zeroFormattingBehavior = NSDateComponentsFormatterZeroFormattingBehaviorDropLeading;
|
|
});
|
|
|
|
self.fDownloadTimeField.stringValue = [timeFormatter stringFromTimeInterval:torrent.secondsDownloading];
|
|
self.fSeedTimeField.stringValue = [timeFormatter stringFromTimeInterval:torrent.secondsSeeding];
|
|
|
|
[self.fPiecesView updateView];
|
|
}
|
|
else if (numberSelected > 1)
|
|
{
|
|
self.fRatioField.stringValue = [NSString stringForRatio:tr_getRatio(uploadedTotal, downloadedTotal)];
|
|
}
|
|
}
|
|
|
|
- (void)setPiecesView:(id)sender
|
|
{
|
|
BOOL const availability = [sender selectedSegment] == PIECES_CONTROL_AVAILABLE;
|
|
[NSUserDefaults.standardUserDefaults setBool:availability forKey:@"PiecesViewShowAvailability"];
|
|
[self updatePiecesView:nil];
|
|
}
|
|
|
|
- (void)updatePiecesView:(id)sender
|
|
{
|
|
BOOL const piecesAvailableSegment = [NSUserDefaults.standardUserDefaults boolForKey:@"PiecesViewShowAvailability"];
|
|
|
|
[self.fPiecesControl setSelected:piecesAvailableSegment forSegment:PIECES_CONTROL_AVAILABLE];
|
|
[self.fPiecesControl setSelected:!piecesAvailableSegment forSegment:PIECES_CONTROL_PROGRESS];
|
|
|
|
[self.fPiecesView updateView];
|
|
}
|
|
|
|
- (void)clearView
|
|
{
|
|
[self.fPiecesView clearView];
|
|
}
|
|
|
|
#pragma mark - Private
|
|
|
|
- (void)setupInfo
|
|
{
|
|
NSUInteger const count = self.fTorrents.count;
|
|
if (count != 1)
|
|
{
|
|
if (count == 0)
|
|
{
|
|
self.fHaveField.stringValue = @"";
|
|
self.fDownloadedTotalField.stringValue = @"";
|
|
self.fUploadedTotalField.stringValue = @"";
|
|
self.fFailedHashField.stringValue = @"";
|
|
self.fDateActivityField.objectValue = @""; //using [field setStringValue: @""] causes "December 31, 1969 7:00 PM" to be displayed, at least on 10.7.3
|
|
self.fRatioField.stringValue = @"";
|
|
}
|
|
|
|
self.fStateField.stringValue = @"";
|
|
self.fProgressField.stringValue = @"";
|
|
|
|
self.fErrorMessageView.string = @"";
|
|
|
|
//using [field setStringValue: @""] causes "December 31, 1969 7:00 PM" to be displayed, at least on 10.7.3
|
|
self.fDateAddedField.objectValue = @"";
|
|
self.fDateCompletedField.objectValue = @"";
|
|
|
|
self.fDownloadTimeField.stringValue = @"";
|
|
self.fSeedTimeField.stringValue = @"";
|
|
|
|
[self.fPiecesControl setSelected:NO forSegment:PIECES_CONTROL_AVAILABLE];
|
|
[self.fPiecesControl setSelected:NO forSegment:PIECES_CONTROL_PROGRESS];
|
|
self.fPiecesControl.enabled = NO;
|
|
self.fPiecesView.torrent = nil;
|
|
}
|
|
else
|
|
{
|
|
Torrent* torrent = self.fTorrents[0];
|
|
|
|
BOOL const piecesAvailableSegment = [NSUserDefaults.standardUserDefaults boolForKey:@"PiecesViewShowAvailability"];
|
|
[self.fPiecesControl setSelected:piecesAvailableSegment forSegment:PIECES_CONTROL_AVAILABLE];
|
|
[self.fPiecesControl setSelected:!piecesAvailableSegment forSegment:PIECES_CONTROL_PROGRESS];
|
|
self.fPiecesControl.enabled = YES;
|
|
|
|
self.fPiecesView.torrent = torrent;
|
|
}
|
|
|
|
self.fSet = YES;
|
|
}
|
|
|
|
@end
|