2023-02-11 20:49:42 +00:00
// This file Copyright © 2005-2023 Transmission authors and contributors.
2022-01-20 18:27:56 +00:00
// It may be used under the MIT (SPDX: MIT) license.
// License text can be found in the licenses/ folder.
2007-09-16 01:02:06 +00:00
2018-09-30 10:37:30 +00:00
#include <libtransmission/transmission.h>
#include <libtransmission/utils.h>
2007-09-16 01:02:06 +00:00
#import "NSStringAdditions.h"
2022-12-23 02:07:45 +00:00
#import "NSDataAdditions.h"
2010-04-23 16:59:14 +00:00
2010-07-10 02:31:05 +00:00
@interface NSString (Private)
2021-08-15 09:41:48 +00:00
+ (NSString*)stringForSpeed:(CGFloat)speed kb:(NSString*)kb mb:(NSString*)mb gb:(NSString*)gb;
2023-03-10 07:05:21 +00:00
+ (NSString*)stringForSpeedCompact:(CGFloat)speed kb:(NSString*)kb mb:(NSString*)mb gb:(NSString*)gb;
2010-07-10 02:31:05 +00:00
@end
2007-09-16 01:02:06 +00:00
@implementation NSString (NSStringAdditions)
2021-08-15 09:41:48 +00:00
+ (NSString*)ellipsis
2007-09-16 01:02:06 +00:00
{
2017-07-08 08:06:32 +00:00
return @"\xE2\x80\xA6";
2007-09-16 01:02:06 +00:00
}
2021-08-15 09:41:48 +00:00
- (NSString*)stringByAppendingEllipsis
2007-09-16 01:02:06 +00:00
{
2021-08-15 09:41:48 +00:00
return [self stringByAppendingString:NSString.ellipsis];
2007-09-16 01:02:06 +00:00
}
2022-05-14 19:00:55 +00:00
// Maximum supported localization is 9.22 EB, which is the maximum supported filesystem size by macOS, 8 EiB.
// https://developer.apple.com/library/archive/documentation/FileManagement/Conceptual/APFS_Guide/VolumeFormatComparison/VolumeFormatComparison.html
2021-08-15 09:41:48 +00:00
+ (NSString*)stringForFileSize:(uint64_t)size
2007-09-16 01:02:06 +00:00
{
2021-08-15 09:41:48 +00:00
return [NSByteCountFormatter stringFromByteCount:size countStyle:NSByteCountFormatterCountStyleFile];
2011-01-08 05:11:28 +00:00
}
2022-05-14 19:00:55 +00:00
// Maximum supported localization is 9.22 EB, which is the maximum supported filesystem size by macOS, 8 EiB.
// https://developer.apple.com/library/archive/documentation/FileManagement/Conceptual/APFS_Guide/VolumeFormatComparison/VolumeFormatComparison.html
2021-08-15 09:41:48 +00:00
+ (NSString*)stringForFilePartialSize:(uint64_t)partialSize fullSize:(uint64_t)fullSize
2011-01-08 05:11:28 +00:00
{
2021-08-15 09:41:48 +00:00
NSByteCountFormatter* fileSizeFormatter = [[NSByteCountFormatter alloc] init];
2017-01-24 17:53:16 +00:00
2022-11-14 18:30:03 +00:00
NSString* fullSizeString = [fileSizeFormatter stringFromByteCount:fullSize];
2017-01-24 17:53:16 +00:00
2022-04-21 14:28:38 +00:00
//figure out the magnitude of the two, since we can't rely on comparing the units because of localization and pluralization issues (for example, "1 byte of 2 bytes")
2017-02-05 19:29:20 +00:00
BOOL partialUnitsSame;
if (partialSize == 0)
2021-08-15 09:41:48 +00:00
{
2017-02-05 19:29:20 +00:00
partialUnitsSame = YES; //we want to just show "0" when we have no partial data, so always set to the same units
2021-08-15 09:41:48 +00:00
}
2012-07-25 12:46:49 +00:00
else
{
2022-12-21 20:21:16 +00:00
auto const magnitudePartial = static_cast<unsigned int>(log(partialSize) / log(1000));
2021-08-15 09:41:48 +00:00
// we have to catch 0 with a special case, so might as well avoid the math for all of magnitude 0
2022-12-21 20:21:16 +00:00
auto const magnitudeFull = static_cast<unsigned int>(fullSize < 1000 ? 0 : log(fullSize) / log(1000));
2017-02-05 19:29:20 +00:00
partialUnitsSame = magnitudePartial == magnitudeFull;
2012-07-25 12:46:49 +00:00
}
2017-01-24 17:53:16 +00:00
2021-08-07 07:27:56 +00:00
fileSizeFormatter.includesUnit = !partialUnitsSame;
2022-11-14 18:30:03 +00:00
NSString* partialSizeString = [fileSizeFormatter stringFromByteCount:partialSize];
2017-02-05 19:29:20 +00:00
2022-11-14 18:30:03 +00:00
return [NSString stringWithFormat:NSLocalizedString(@"%@ of %@", "file size string"), partialSizeString, fullSizeString];
2007-09-16 01:02:06 +00:00
}
2021-08-15 09:41:48 +00:00
+ (NSString*)stringForSpeed:(CGFloat)speed
2007-09-16 01:02:06 +00:00
{
2021-08-15 09:41:48 +00:00
return [self stringForSpeed:speed kb:NSLocalizedString(@"KB/s", "Transfer speed (kilobytes per second)")
mb:NSLocalizedString(@"MB/s", "Transfer speed (megabytes per second)")
gb:NSLocalizedString(@"GB/s", "Transfer speed (gigabytes per second)")];
2007-09-16 01:02:06 +00:00
}
2021-08-15 09:41:48 +00:00
+ (NSString*)stringForSpeedAbbrev:(CGFloat)speed
2007-09-16 01:02:06 +00:00
{
2021-08-15 09:41:48 +00:00
return [self stringForSpeed:speed kb:@"K" mb:@"M" gb:@"G"];
2007-09-16 01:02:06 +00:00
}
2023-03-10 07:05:21 +00:00
+ (NSString*)stringForSpeedAbbrevCompact:(CGFloat)speed
{
return [self stringForSpeedCompact:speed kb:@"K" mb:@"M" gb:@"G"];
}
2021-08-15 09:41:48 +00:00
+ (NSString*)stringForRatio:(CGFloat)ratio
2009-03-05 04:31:30 +00:00
{
2009-03-05 04:59:24 +00:00
//N/A is different than libtransmission's
2022-01-17 23:16:37 +00:00
2022-12-21 20:21:16 +00:00
if (static_cast<int>(ratio) == TR_RATIO_NA)
2021-08-15 09:41:48 +00:00
{
2007-09-16 01:02:06 +00:00
return NSLocalizedString(@"N/A", "No Ratio");
2021-08-15 09:41:48 +00:00
}
2022-01-17 23:16:37 +00:00
2022-12-21 20:21:16 +00:00
if (static_cast<int>(ratio) == TR_RATIO_INF)
2021-08-15 09:41:48 +00:00
{
2017-07-08 08:06:32 +00:00
return @"\xE2\x88\x9E";
2021-08-15 09:41:48 +00:00
}
2022-01-17 23:16:37 +00:00
if (ratio < 10.0)
2009-11-17 01:48:00 +00:00
{
2022-01-17 23:16:37 +00:00
return [NSString localizedStringWithFormat:@"%.2f", tr_truncd(ratio, 2)];
2009-11-17 01:48:00 +00:00
}
2022-01-17 23:16:37 +00:00
if (ratio < 100.0)
{
return [NSString localizedStringWithFormat:@"%.1f", tr_truncd(ratio, 1)];
}
return [NSString localizedStringWithFormat:@"%.0f", tr_truncd(ratio, 0)];
2007-09-16 01:02:06 +00:00
}
2021-08-15 09:41:48 +00:00
+ (NSString*)percentString:(CGFloat)progress longDecimals:(BOOL)longDecimals
2010-06-24 00:00:43 +00:00
{
2023-06-22 04:21:24 +00:00
static NSNumberFormatter* longFormatter;
static NSNumberFormatter* shortFormatter;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
longFormatter = [[NSNumberFormatter alloc] init];
longFormatter.numberStyle = NSNumberFormatterPercentStyle;
longFormatter.maximumFractionDigits = 2;
shortFormatter = [[NSNumberFormatter alloc] init];
shortFormatter.numberStyle = NSNumberFormatterPercentStyle;
shortFormatter.maximumFractionDigits = 1;
});
2010-06-24 00:00:43 +00:00
if (progress >= 1.0)
2021-08-15 09:41:48 +00:00
{
2023-06-22 04:21:24 +00:00
return [shortFormatter stringFromNumber:@(1)];
2021-08-15 09:41:48 +00:00
}
2010-06-24 00:00:43 +00:00
else if (longDecimals)
2021-08-15 09:41:48 +00:00
{
2023-06-22 04:21:24 +00:00
return [longFormatter stringFromNumber:@(MIN(progress, 0.9999))];
2021-08-15 09:41:48 +00:00
}
2010-06-24 00:00:43 +00:00
else
2021-08-15 09:41:48 +00:00
{
2023-06-22 04:21:24 +00:00
return [shortFormatter stringFromNumber:@(MIN(progress, 0.999))];
2021-08-15 09:41:48 +00:00
}
2010-06-24 00:00:43 +00:00
}
2021-08-15 09:41:48 +00:00
- (NSComparisonResult)compareNumeric:(NSString*)string
2008-06-27 05:40:22 +00:00
{
2021-08-15 09:41:48 +00:00
NSStringCompareOptions const comparisonOptions = NSNumericSearch | NSForcedOrderingSearch;
return [self compare:string options:comparisonOptions range:NSMakeRange(0, self.length) locale:NSLocale.currentLocale];
2007-09-16 01:02:06 +00:00
}
2022-05-14 19:00:55 +00:00
- (NSArray<NSString*>*)nonEmptyComponentsSeparatedByCharactersInSet:(NSCharacterSet*)separators
2011-09-19 00:48:30 +00:00
{
2022-05-14 19:00:55 +00:00
NSMutableArray<NSString*>* components = [NSMutableArray array];
for (NSString* evaluatedObject in [self componentsSeparatedByCharactersInSet:separators])
2011-09-19 01:37:43 +00:00
{
2022-05-14 19:00:55 +00:00
if (evaluatedObject.length > 0)
2011-09-19 01:37:43 +00:00
{
2022-05-14 19:00:55 +00:00
[components addObject:evaluatedObject];
2011-09-19 01:37:43 +00:00
}
2022-05-14 19:00:55 +00:00
}
2011-09-19 00:48:30 +00:00
return components;
}
2022-12-23 02:07:45 +00:00
+ (NSString*)convertedStringFromCString:(nonnull char const*)bytes
{
// UTF-8 encoding
NSString* fullPath = @(bytes);
if (fullPath)
{
return fullPath;
}
// autodetection of the encoding (#3434)
NSData* data = [NSData dataWithBytes:(void const*)bytes length:sizeof(unsigned char) * strlen(bytes)];
[NSString stringEncodingForData:data encodingOptions:nil convertedString:&fullPath usedLossyConversion:nil];
if (fullPath)
{
return fullPath;
}
// hexa encoding
return data.hexString;
}
2007-09-16 01:02:06 +00:00
@end
2010-07-10 02:31:05 +00:00
@implementation NSString (Private)
2021-08-15 09:41:48 +00:00
+ (NSString*)stringForSpeed:(CGFloat)speed kb:(NSString*)kb mb:(NSString*)mb gb:(NSString*)gb
2010-07-10 02:31:05 +00:00
{
2023-03-01 17:33:17 +00:00
if (speed < 999.95) // 0.0 KB/s to 999.9 KB/s
2021-08-15 09:41:48 +00:00
{
return [NSString localizedStringWithFormat:@"%.1f %@", speed, kb];
}
2017-01-24 17:53:16 +00:00
2011-10-06 00:30:40 +00:00
speed /= 1000.0;
2017-01-24 17:53:16 +00:00
2023-03-01 17:33:17 +00:00
if (speed < 99.995) // 1.00 MB/s to 99.99 MB/s
2021-08-15 09:41:48 +00:00
{
return [NSString localizedStringWithFormat:@"%.2f %@", speed, mb];
}
2023-03-01 17:33:17 +00:00
else if (speed < 999.95) // 100.0 MB/s to 999.9 MB/s
2021-08-15 09:41:48 +00:00
{
return [NSString localizedStringWithFormat:@"%.1f %@", speed, mb];
}
2023-03-08 03:16:06 +00:00
speed /= 1000.0;
if (speed < 99.995) // 1.00 GB/s to 99.99 GB/s
{
return [NSString localizedStringWithFormat:@"%.2f %@", speed, gb];
}
2023-03-13 19:43:39 +00:00
// 100.0 GB/s and above
return [NSString localizedStringWithFormat:@"%.1f %@", speed, gb];
2010-07-10 02:31:05 +00:00
}
2023-03-10 07:05:21 +00:00
+ (NSString*)stringForSpeedCompact:(CGFloat)speed kb:(NSString*)kb mb:(NSString*)mb gb:(NSString*)gb
{
if (speed < 99.95) // 0.0 KB/s to 99.9 KB/s
{
return [NSString localizedStringWithFormat:@"%.1f %@", speed, kb];
}
2023-03-13 19:43:39 +00:00
if (speed < 999.5) // 100 KB/s to 999 KB/s
2023-03-10 07:05:21 +00:00
{
return [NSString localizedStringWithFormat:@"%.0f %@", speed, kb];
}
speed /= 1000.0;
2023-03-13 19:43:39 +00:00
if (speed < 9.995) // 1.00 MB/s to 9.99 MB/s
2023-03-10 07:05:21 +00:00
{
return [NSString localizedStringWithFormat:@"%.2f %@", speed, mb];
}
if (speed < 99.95) // 10.0 MB/s to 99.9 MB/s
{
return [NSString localizedStringWithFormat:@"%.1f %@", speed, mb];
}
2023-03-13 19:43:39 +00:00
if (speed < 999.5) // 100 MB/s to 999 MB/s
2023-03-10 07:05:21 +00:00
{
return [NSString localizedStringWithFormat:@"%.0f %@", speed, mb];
}
speed /= 1000.0;
2023-03-13 19:43:39 +00:00
if (speed < 9.995) // 1.00 GB/s to 9.99 GB/s
2023-03-10 07:05:21 +00:00
{
return [NSString localizedStringWithFormat:@"%.2f %@", speed, gb];
}
if (speed < 99.95) // 10.0 GB/s to 99.9 GB/s
{
return [NSString localizedStringWithFormat:@"%.1f %@", speed, gb];
}
2023-03-13 19:43:39 +00:00
// 100 GB/s and above
return [NSString localizedStringWithFormat:@"%.0f %@", speed, gb];
2023-03-10 07:05:21 +00:00
}
2010-07-10 02:31:05 +00:00
@end