624 lines
21 KiB
Objective-C
624 lines
21 KiB
Objective-C
/******************************************************************************
|
|
* $Id$
|
|
*
|
|
* Copyright (c) 2006-2007 Transmission authors and contributors
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
|
* copy of this software and associated documentation files (the "Software"),
|
|
* to deal in the Software without restriction, including without limitation
|
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
* and/or sell copies of the Software, and to permit persons to whom the
|
|
* Software is furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
* DEALINGS IN THE SOFTWARE.
|
|
*****************************************************************************/
|
|
|
|
#import "TorrentCell.h"
|
|
#import "TorrentTableView.h"
|
|
#import "StringAdditions.h"
|
|
#import "CTGradientAdditions.h"
|
|
|
|
#define BAR_HEIGHT 12.0
|
|
|
|
#define PADDING_HORIZONAL 2.0
|
|
#define IMAGE_SIZE_REG 32.0
|
|
#define IMAGE_SIZE_MIN 16.0
|
|
#define PADDING_BETWEEN_IMAGE_AND_TITLE 4.0
|
|
#define PADDING_ABOVE_TITLE 2.0
|
|
#define PADDING_ABOVE_MIN_STATUS 4.0
|
|
#define PADDING_BETWEEN_TITLE_AND_MIN_STATUS 2.0
|
|
#define PADDING_BETWEEN_TITLE_AND_PROGRESS 2.0
|
|
#define PADDING_LESS_BETWEEN_TITLE_AND_BAR 1.0
|
|
#define PADDING_BETWEEN_PROGRESS_AND_BAR 2.0
|
|
#define PADDING_BETWEEN_BAR_AND_STATUS 2.0
|
|
|
|
#define MAX_PIECES 324
|
|
#define BLANK_PIECE -99
|
|
|
|
@interface TorrentCell (Private)
|
|
|
|
- (NSImage *) simpleBar: (NSSize) barSize;
|
|
- (NSImage *) advancedBar: (NSSize) barSize;
|
|
- (NSImage *) advancedBarSimple;
|
|
|
|
- (NSRect) rectForTitleBasedOnMinimalStatusRect: (NSRect) statusRect inBounds: (NSRect) bounds;
|
|
- (NSRect) rectForProgressBasedOnTitleRect: (NSRect) titleRect inBounds: (NSRect) bounds;
|
|
- (NSRect) rectForBarBasedOnAboveRect: (NSRect) aboveRect inBounds: (NSRect) bounds;
|
|
- (NSRect) rectForStatusBasedOnProgressRect: (NSRect) progressRect inBounds: (NSRect) bounds;
|
|
|
|
- (NSAttributedString *) attributedTitleWithColor: (NSColor *) color;
|
|
- (NSAttributedString *) attributedProgressWithColor: (NSColor *) color;
|
|
- (NSAttributedString *) attributedStatusWithColor: (NSColor *) color;
|
|
- (NSAttributedString *) attributedMinimalStatusWithColor: (NSColor *) color;
|
|
|
|
@end
|
|
|
|
@implementation TorrentCell
|
|
|
|
// Used to optimize drawing. They contain packed RGBA pixels for every color needed.
|
|
#define BE OSSwapBigToHostConstInt32
|
|
|
|
static uint32_t kRed = BE(0xFF6450FF), //255, 100, 80
|
|
kBlue = BE(0x50A0FFFF), //80, 160, 255
|
|
kBlue2 = BE(0x1E46B4FF), //30, 70, 180
|
|
kGray = BE(0x969696FF), //150, 150, 150
|
|
kGreen1 = BE(0x99FFCCFF), //153, 255, 204
|
|
kGreen2 = BE(0x66FF99FF), //102, 255, 153
|
|
kGreen3 = BE(0x00FF66FF), //0, 255, 102
|
|
kWhite = BE(0xFFFFFFFF); //255, 255, 255
|
|
|
|
//only called one, so don't worry about release
|
|
- (id) init
|
|
{
|
|
if ((self = [super init]))
|
|
{
|
|
fDefaults = [NSUserDefaults standardUserDefaults];
|
|
|
|
NSMutableParagraphStyle * paragraphStyle = [[NSParagraphStyle defaultParagraphStyle] mutableCopy];
|
|
[paragraphStyle setLineBreakMode: NSLineBreakByTruncatingTail];
|
|
|
|
fTitleAttributes = [[NSMutableDictionary alloc] initWithObjectsAndKeys:
|
|
[NSFont messageFontOfSize: 12.0], NSFontAttributeName,
|
|
paragraphStyle, NSParagraphStyleAttributeName, nil];
|
|
fStatusAttributes = [[NSMutableDictionary alloc] initWithObjectsAndKeys:
|
|
[NSFont messageFontOfSize: 9.0], NSFontAttributeName,
|
|
paragraphStyle, NSParagraphStyleAttributeName, nil];
|
|
[paragraphStyle release];
|
|
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (id) copyWithZone: (NSZone *) zone
|
|
{
|
|
TorrentCell * copy = [super copyWithZone: zone];
|
|
|
|
copy->fBitmap = nil;
|
|
copy->fPieces = NULL;
|
|
}
|
|
|
|
- (void) dealloc
|
|
{
|
|
[fBitmap release];
|
|
if (fPieces)
|
|
free(fPieces);
|
|
|
|
[super dealloc];
|
|
}
|
|
|
|
- (NSRect) iconRectForBounds: (NSRect) bounds
|
|
{
|
|
NSRect result = bounds;
|
|
|
|
result.origin.x += PADDING_HORIZONAL;
|
|
|
|
float imageSize;
|
|
if ([fDefaults boolForKey: @"SmallView"])
|
|
{
|
|
imageSize = IMAGE_SIZE_MIN;
|
|
result.origin.y += (result.size.height - imageSize) * 0.5;
|
|
}
|
|
else
|
|
{
|
|
imageSize = IMAGE_SIZE_REG;
|
|
result.origin.y += (result.size.height - (imageSize + ACTION_BUTTON_HEIGHT)) * 0.5;
|
|
}
|
|
|
|
result.size = NSMakeSize(imageSize, imageSize);
|
|
|
|
return result;
|
|
}
|
|
|
|
- (NSRect) titleRectForBounds: (NSRect) bounds
|
|
{
|
|
return [self rectForTitleBasedOnMinimalStatusRect: [self minimalStatusRectForBounds: bounds] inBounds: bounds];
|
|
}
|
|
|
|
- (NSRect) minimalStatusRectForBounds: (NSRect) bounds
|
|
{
|
|
if (![fDefaults boolForKey: @"SmallView"])
|
|
return NSZeroRect;
|
|
|
|
NSAttributedString * status = [self attributedMinimalStatusWithColor: nil];
|
|
|
|
NSRect result = bounds;
|
|
result.size = [status size];
|
|
|
|
result.origin.x += bounds.size.width - result.size.width - PADDING_HORIZONAL;
|
|
result.origin.y += PADDING_ABOVE_MIN_STATUS;
|
|
|
|
return result;
|
|
}
|
|
|
|
- (NSRect) progressRectForBounds: (NSRect) bounds
|
|
{
|
|
return [self rectForProgressBasedOnTitleRect: [self titleRectForBounds: bounds] inBounds: bounds];
|
|
}
|
|
|
|
- (NSRect) barRectForBounds: (NSRect) bounds
|
|
{
|
|
NSRect aboveRect = [fDefaults boolForKey: @"SmallView"] ? [self titleRectForBounds: bounds]
|
|
: [self progressRectForBounds: bounds];
|
|
return [self rectForBarBasedOnAboveRect: aboveRect inBounds: bounds];
|
|
}
|
|
|
|
- (NSRect) statusRectForBounds: (NSRect) bounds
|
|
{
|
|
return [self rectForStatusBasedOnProgressRect: [self progressRectForBounds: bounds] inBounds: bounds];
|
|
}
|
|
|
|
- (void) drawWithFrame: (NSRect) cellFrame inView: (NSView *) controlView
|
|
{
|
|
Torrent * torrent = [self representedObject];
|
|
|
|
BOOL minimal = [fDefaults boolForKey: @"SmallView"];
|
|
|
|
//error image
|
|
BOOL error = [torrent isError];
|
|
if (error && !fErrorImage)
|
|
{
|
|
fErrorImage = [[NSImage imageNamed: @"Error.png"] copy];
|
|
[fErrorImage setFlipped: YES];
|
|
}
|
|
|
|
//icon
|
|
NSImage * icon = minimal ? (error ? fErrorImage : [torrent iconSmall]) : [torrent iconFlipped];
|
|
NSRect iconRect = [self iconRectForBounds: cellFrame];
|
|
[icon drawInRect: iconRect fromRect: NSZeroRect operation: NSCompositeSourceOver fraction: 1.0];
|
|
|
|
if (error && !minimal)
|
|
{
|
|
NSRect errorRect = NSMakeRect(NSMaxX(iconRect) - IMAGE_SIZE_MIN, NSMaxY(iconRect) - IMAGE_SIZE_MIN,
|
|
IMAGE_SIZE_MIN, IMAGE_SIZE_MIN);
|
|
[fErrorImage drawInRect: errorRect fromRect: NSZeroRect operation: NSCompositeSourceOver fraction: 1.0];
|
|
}
|
|
|
|
//text color
|
|
BOOL highlighted = [self isHighlighted] && [[self highlightColorWithFrame: cellFrame inView: controlView]
|
|
isEqual: [NSColor alternateSelectedControlColor]];
|
|
NSColor * titleColor, * statusColor;
|
|
if ([self isHighlighted]
|
|
&& [[self highlightColorWithFrame: cellFrame inView: controlView] isEqual: [NSColor alternateSelectedControlColor]])
|
|
{
|
|
titleColor = [NSColor whiteColor];
|
|
statusColor = [NSColor whiteColor];
|
|
}
|
|
else
|
|
{
|
|
titleColor = [NSColor controlTextColor];
|
|
statusColor = [NSColor darkGrayColor];
|
|
}
|
|
|
|
//minimal status
|
|
NSRect minimalStatusRect = [self minimalStatusRectForBounds: cellFrame];
|
|
if (minimal)
|
|
[[self attributedMinimalStatusWithColor: statusColor] drawInRect: minimalStatusRect];
|
|
|
|
//title
|
|
NSRect titleRect = [self rectForTitleBasedOnMinimalStatusRect: minimalStatusRect inBounds: cellFrame];
|
|
[[self attributedTitleWithColor: titleColor] drawInRect: titleRect];
|
|
|
|
//progress
|
|
NSRect progressRect;
|
|
if (!minimal)
|
|
{
|
|
progressRect = [self rectForProgressBasedOnTitleRect: titleRect inBounds: cellFrame];
|
|
[[self attributedProgressWithColor: statusColor] drawInRect: progressRect];
|
|
}
|
|
|
|
//bar
|
|
NSRect barRect = [self rectForBarBasedOnAboveRect: minimal ? titleRect : progressRect inBounds: cellFrame];
|
|
NSImage * bar = [fDefaults boolForKey: @"UseAdvancedBar"] ? [self advancedBar: barRect.size] : [self simpleBar: barRect.size];
|
|
[bar drawInRect: barRect fromRect: NSZeroRect operation: NSCompositeSourceOver fraction: 1.0];
|
|
|
|
//status
|
|
if (!minimal)
|
|
{
|
|
NSRect statusRect = [self rectForStatusBasedOnProgressRect: progressRect inBounds: cellFrame];
|
|
[[self attributedStatusWithColor: statusColor] drawInRect: statusRect];
|
|
}
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation TorrentCell (Private)
|
|
|
|
- (NSImage *) simpleBar: (NSSize) barSize
|
|
{
|
|
Torrent * torrent = [self representedObject];
|
|
|
|
NSImage * bar = [[NSImage alloc] initWithSize: barSize];
|
|
[bar lockFocus];
|
|
|
|
NSRect barBounds = NSMakeRect(0, 0, barSize.width, barSize.height);
|
|
|
|
float progress = [torrent progress];
|
|
|
|
float left = INVALID;
|
|
if (progress < 1.0)
|
|
{
|
|
if (!fWhiteGradient)
|
|
fWhiteGradient = [[CTGradient progressWhiteGradient] retain];
|
|
[fWhiteGradient fillRect: barBounds angle: -90];
|
|
|
|
left = [torrent progressLeft];
|
|
float include = progress + left;
|
|
if (include < 1.0)
|
|
{
|
|
NSRect blankBounds = barBounds;
|
|
blankBounds.origin.x += barBounds.size.width * include;
|
|
blankBounds.size.width *= (1.0 - include);
|
|
|
|
if (!fLightGrayGradient)
|
|
fLightGrayGradient = [[CTGradient progressLightGrayGradient] retain];
|
|
[fLightGrayGradient fillRect: blankBounds angle: -90];
|
|
}
|
|
}
|
|
|
|
NSRect completeBounds = barBounds;
|
|
completeBounds.size.width *= progress;
|
|
|
|
if ([torrent isActive])
|
|
{
|
|
if ([torrent isChecking])
|
|
{
|
|
if (!fYellowGradient)
|
|
fYellowGradient = [[CTGradient progressYellowGradient] retain];
|
|
[fYellowGradient fillRect: completeBounds angle: -90];
|
|
}
|
|
else if ([torrent isSeeding])
|
|
{
|
|
NSRect ratioBounds = completeBounds;
|
|
ratioBounds.size.width *= [torrent progressStopRatio];
|
|
|
|
if (ratioBounds.size.width < completeBounds.size.width)
|
|
{
|
|
if (!fLightGreenGradient)
|
|
fLightGreenGradient = [[CTGradient progressLightGreenGradient] retain];
|
|
[fLightGreenGradient fillRect: completeBounds angle: -90];
|
|
}
|
|
|
|
if (!fGreenGradient)
|
|
fGreenGradient = [[CTGradient progressGreenGradient] retain];
|
|
[fGreenGradient fillRect: ratioBounds angle: -90];
|
|
}
|
|
else
|
|
{
|
|
if (!fBlueGradient)
|
|
fBlueGradient = [[CTGradient progressBlueGradient] retain];
|
|
[fBlueGradient fillRect: completeBounds angle: -90];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ([torrent waitingToStart])
|
|
{
|
|
if (left == INVALID)
|
|
left = [torrent progressLeft];
|
|
|
|
if (left <= 0.0)
|
|
{
|
|
if (!fDarkGreenGradient)
|
|
fDarkGreenGradient = [[CTGradient progressDarkGreenGradient] retain];
|
|
[fDarkGreenGradient fillRect: completeBounds angle: -90];
|
|
}
|
|
else
|
|
{
|
|
if (!fDarkBlueGradient)
|
|
fDarkBlueGradient = [[CTGradient progressDarkBlueGradient] retain];
|
|
[fDarkBlueGradient fillRect: completeBounds angle: -90];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (!fGrayGradient)
|
|
fGrayGradient = [[CTGradient progressGrayGradient] retain];
|
|
[fGrayGradient fillRect: completeBounds angle: -90];
|
|
}
|
|
}
|
|
|
|
[[NSColor colorWithDeviceWhite: 0.0 alpha: 0.2] set];
|
|
[NSBezierPath strokeRect: NSInsetRect(barBounds, 0.5, 0.5)];
|
|
|
|
[bar unlockFocus];
|
|
|
|
return [bar autorelease];
|
|
}
|
|
|
|
- (NSImage *) advancedBar: (NSSize) barSize
|
|
{
|
|
NSImage * image = [self advancedBarSimple];
|
|
|
|
[image setScalesWhenResized: YES];
|
|
[image setSize: barSize];
|
|
|
|
[image lockFocus];
|
|
|
|
NSRect barBounds = NSMakeRect(0, 0, barSize.width, barSize.height);
|
|
|
|
if (!fTransparentGradient)
|
|
fTransparentGradient = [[CTGradient progressTransparentGradient] retain];
|
|
[fTransparentGradient fillRect: barBounds angle: 90];
|
|
|
|
[[NSColor colorWithDeviceWhite: 0.0 alpha: 0.2] set];
|
|
[NSBezierPath strokeRect: NSInsetRect(barBounds, 0.5, 0.5)];
|
|
|
|
[image unlockFocus];
|
|
return image;
|
|
}
|
|
|
|
- (NSImage *) advancedBarSimple
|
|
{
|
|
if (!fBitmap)
|
|
fBitmap = [[NSBitmapImageRep alloc] initWithBitmapDataPlanes: nil
|
|
pixelsWide: MAX_PIECES pixelsHigh: BAR_HEIGHT bitsPerSample: 8 samplesPerPixel: 4 hasAlpha: YES
|
|
isPlanar: NO colorSpaceName: NSCalibratedRGBColorSpace bytesPerRow: 0 bitsPerPixel: 0];
|
|
|
|
uint32_t * p;
|
|
uint8_t * bitmapData = [fBitmap bitmapData];
|
|
int bytesPerRow = [fBitmap bytesPerRow];
|
|
|
|
if (!fPieces)
|
|
{
|
|
fPieces = malloc(MAX_PIECES);
|
|
int i;
|
|
for (i = 0; i < MAX_PIECES; i++)
|
|
fPieces[i] = BLANK_PIECE;
|
|
}
|
|
|
|
Torrent * torrent = [self representedObject];
|
|
int pieceCount = [torrent pieceCount];
|
|
int8_t * piecesAvailablity = malloc(pieceCount);
|
|
[torrent getAvailability: piecesAvailablity size: pieceCount];
|
|
|
|
//lines 2 to 14: blue, green, or gray depending on piece availability
|
|
int i, h, index = 0;
|
|
float increment = (float)pieceCount / MAX_PIECES, indexValue = 0;
|
|
uint32_t color;
|
|
BOOL change;
|
|
for (i = 0; i < MAX_PIECES; i++)
|
|
{
|
|
change = NO;
|
|
if (piecesAvailablity[index] < 0)
|
|
{
|
|
if (fPieces[i] != -1)
|
|
{
|
|
color = kBlue;
|
|
fPieces[i] = -1;
|
|
change = YES;
|
|
}
|
|
}
|
|
else if (piecesAvailablity[index] == 0)
|
|
{
|
|
if (fPieces[i] != 0)
|
|
{
|
|
color = kGray;
|
|
fPieces[i] = 0;
|
|
change = YES;
|
|
}
|
|
}
|
|
else if (piecesAvailablity[index] <= 4)
|
|
{
|
|
if (fPieces[i] != 1)
|
|
{
|
|
color = kGreen1;
|
|
fPieces[i] = 1;
|
|
change = YES;
|
|
}
|
|
}
|
|
else if (piecesAvailablity[index] <= 8)
|
|
{
|
|
if (fPieces[i] != 2)
|
|
{
|
|
color = kGreen2;
|
|
fPieces[i] = 2;
|
|
change = YES;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (fPieces[i] != 3)
|
|
{
|
|
color = kGreen3;
|
|
fPieces[i] = 3;
|
|
change = YES;
|
|
}
|
|
}
|
|
|
|
if (change)
|
|
{
|
|
//point to pixel (i, 2) and draw "vertically"
|
|
p = (uint32_t *)(bitmapData + 2 * bytesPerRow) + i;
|
|
for (h = 2; h < BAR_HEIGHT; h++)
|
|
{
|
|
p[0] = color;
|
|
p = (uint32_t *)((uint8_t *)p + bytesPerRow);
|
|
}
|
|
}
|
|
|
|
indexValue += increment;
|
|
index = (int)indexValue;
|
|
}
|
|
|
|
//determine percentage finished and available
|
|
int have = rintf((float)MAX_PIECES * [torrent progress]), avail;
|
|
if (![torrent isActive] || [torrent progress] >= 1.0 || [torrent totalPeersConnected] <= 0)
|
|
avail = 0;
|
|
else
|
|
{
|
|
float * piecesFinished = malloc(pieceCount * sizeof(float));
|
|
[torrent getAmountFinished: piecesFinished size: pieceCount];
|
|
|
|
float available = 0;
|
|
for (i = 0; i < pieceCount; i++)
|
|
if (piecesAvailablity[i] > 0)
|
|
available += 1.0 - piecesFinished[i];
|
|
|
|
avail = rintf(MAX_PIECES * available / (float)pieceCount);
|
|
if (have + avail > MAX_PIECES) //case if both end in .5 and all pieces are available
|
|
avail--;
|
|
|
|
free(piecesFinished);
|
|
}
|
|
|
|
free(piecesAvailablity);
|
|
|
|
//first two lines: dark blue to show progression, green to show available
|
|
p = (uint32_t *)bitmapData;
|
|
for (i = 0; i < have; i++)
|
|
{
|
|
p[i] = kBlue2;
|
|
p[i + bytesPerRow / 4] = kBlue2;
|
|
}
|
|
for (; i < avail + have; i++)
|
|
{
|
|
p[i] = kGreen3;
|
|
p[i + bytesPerRow / 4] = kGreen3;
|
|
}
|
|
for (; i < MAX_PIECES; i++)
|
|
{
|
|
p[i] = kWhite;
|
|
p[i + bytesPerRow / 4] = kWhite;
|
|
}
|
|
|
|
//actually draw image
|
|
NSImage * bar = [[NSImage alloc] initWithSize: [fBitmap size]];
|
|
[bar setFlipped: YES];
|
|
[bar addRepresentation: fBitmap];
|
|
|
|
return [bar autorelease];
|
|
}
|
|
|
|
- (NSRect) rectForTitleBasedOnMinimalStatusRect: (NSRect) statusRect inBounds: (NSRect) bounds
|
|
{
|
|
BOOL minimal = [fDefaults boolForKey: @"SmallView"];
|
|
|
|
NSRect result = bounds;
|
|
|
|
result.origin.y += PADDING_ABOVE_TITLE;
|
|
result.origin.x += PADDING_HORIZONAL + (minimal ? IMAGE_SIZE_MIN : IMAGE_SIZE_REG) + PADDING_BETWEEN_IMAGE_AND_TITLE;
|
|
|
|
result.size = [[self attributedTitleWithColor: nil] size];
|
|
result.size.width = MIN(result.size.width, NSMaxX(bounds) - result.origin.x - PADDING_HORIZONAL
|
|
- (minimal ? PADDING_BETWEEN_TITLE_AND_MIN_STATUS + statusRect.size.width : 0));
|
|
|
|
return result;
|
|
}
|
|
|
|
- (NSRect) rectForProgressBasedOnTitleRect: (NSRect) titleRect inBounds: (NSRect) bounds
|
|
{
|
|
if ([fDefaults boolForKey: @"SmallView"])
|
|
return NSZeroRect;
|
|
|
|
NSSize progressSize = [[self attributedProgressWithColor: nil] size];
|
|
|
|
NSRect result = titleRect;
|
|
|
|
result.size.width = MIN(progressSize.width, NSMaxX(bounds) - result.origin.x - PADDING_HORIZONAL);
|
|
|
|
result.origin.y += titleRect.size.height + PADDING_BETWEEN_TITLE_AND_PROGRESS;
|
|
result.size.height = progressSize.height;
|
|
|
|
return result;
|
|
}
|
|
|
|
- (NSRect) rectForBarBasedOnAboveRect: (NSRect) aboveRect inBounds: (NSRect) bounds
|
|
{
|
|
NSRect result;
|
|
result.size.height = BAR_HEIGHT;
|
|
result.origin.x = bounds.origin.x +
|
|
PADDING_HORIZONAL + ([fDefaults boolForKey: @"SmallView"] ? IMAGE_SIZE_MIN : IMAGE_SIZE_REG)
|
|
+ PADDING_BETWEEN_IMAGE_AND_TITLE - PADDING_LESS_BETWEEN_TITLE_AND_BAR;
|
|
result.origin.y = NSMaxY(aboveRect) + PADDING_BETWEEN_PROGRESS_AND_BAR;
|
|
result.size.width = NSMaxX(bounds) - result.origin.x - PADDING_HORIZONAL - BUTTONS_TOTAL_WIDTH;
|
|
|
|
return result;
|
|
}
|
|
|
|
- (NSRect) rectForStatusBasedOnProgressRect: (NSRect) progressRect inBounds: (NSRect) bounds
|
|
{
|
|
if ([fDefaults boolForKey: @"SmallView"])
|
|
return NSZeroRect;
|
|
|
|
NSSize statusSize = [[self attributedStatusWithColor: nil] size];
|
|
|
|
NSRect result = progressRect;
|
|
result.size.width = MIN(statusSize.width, NSMaxX(bounds) - result.origin.x - PADDING_HORIZONAL);
|
|
|
|
result.origin.y += statusSize.height + PADDING_BETWEEN_PROGRESS_AND_BAR + BAR_HEIGHT + PADDING_BETWEEN_BAR_AND_STATUS;
|
|
result.size.height = statusSize.height;
|
|
|
|
return result;
|
|
}
|
|
|
|
- (NSAttributedString *) attributedTitleWithColor: (NSColor *) color
|
|
{
|
|
if (color)
|
|
[fTitleAttributes setObject: color forKey: NSForegroundColorAttributeName];
|
|
|
|
NSString * title = [[self representedObject] name];
|
|
return [[[NSAttributedString alloc] initWithString: title attributes: fTitleAttributes] autorelease];
|
|
}
|
|
|
|
- (NSAttributedString *) attributedProgressWithColor: (NSColor *) color
|
|
{
|
|
if (color)
|
|
[fStatusAttributes setObject: color forKey: NSForegroundColorAttributeName];
|
|
|
|
NSString * progress = [[self representedObject] progressString];
|
|
return [[[NSAttributedString alloc] initWithString: progress attributes: fStatusAttributes] autorelease];
|
|
}
|
|
|
|
- (NSAttributedString *) attributedStatusWithColor: (NSColor *) color
|
|
{
|
|
if (color)
|
|
[fStatusAttributes setObject: color forKey: NSForegroundColorAttributeName];
|
|
|
|
NSString * status = [[self representedObject] statusString];
|
|
return [[[NSAttributedString alloc] initWithString: status attributes: fStatusAttributes] autorelease];
|
|
}
|
|
|
|
#warning combine?
|
|
- (NSAttributedString *) attributedMinimalStatusWithColor: (NSColor *) color
|
|
{
|
|
if (color)
|
|
[fStatusAttributes setObject: color forKey: NSForegroundColorAttributeName];
|
|
|
|
Torrent * torrent = [self representedObject];
|
|
NSString * status = ![fDefaults boolForKey: @"SmallStatusRegular"] && [torrent isActive]
|
|
? [torrent remainingTimeString] : [torrent shortStatusString];
|
|
|
|
return [[[NSAttributedString alloc] initWithString: status attributes: fStatusAttributes] autorelease];
|
|
}
|
|
|
|
@end
|