// 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 #include //tr_getRatio() #import "InfoActivityViewController.h" #import "NSApplicationAdditions.h" #import "NSStringAdditions.h" #import "PiecesView.h" #import "Torrent.h" #define PIECES_CONTROL_PROGRESS 0 #define PIECES_CONTROL_AVAILABLE 1 @interface InfoActivityViewController () @property(nonatomic, copy) NSArray* 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; - (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]; } - (void)setInfoForTorrents:(NSArray*)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