netsurf/cocoa/DownloadWindowController.m

388 lines
12 KiB
Objective-C

/*
* Copyright 2011 Sven Weidauer <sven.weidauer@gmail.com>
*
* This file is part of NetSurf, http://www.netsurf-browser.org/
*
* NetSurf 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; version 2 of the License.
*
* NetSurf 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, see <http://www.gnu.org/licenses/>.
*/
#import "cocoa/DownloadWindowController.h"
#import "cocoa/gui.h"
#import "desktop/download.h"
#import "desktop/gui.h"
#import "utils/log.h"
@interface DownloadWindowController ()
@property (readwrite, retain, nonatomic) NSFileHandle *outputFile;
@property (readwrite, retain, nonatomic) NSMutableData *savedData;
@property (readwrite, copy, nonatomic) NSDate *startDate;
- (void)savePanelDidEnd:(NSSavePanel *)sheet returnCode:(int)returnCode contextInfo:(void *)contextInfo;
- (void)alertDidEnd:(NSAlert *)alert returnCode:(NSInteger)returnCode contextInfo:(void *)contextInfo;
- (void)askCancelDidEnd:(NSAlert *)alert returnCode:(NSInteger)returnCode contextInfo:(void *)contextInfo;
- (BOOL) receivedData: (NSData *)data;
- (void) showError: (NSString *)error;
- (void) downloadDone;
- (void) removeIfPossible;
@end
static void cocoa_unregister_download( DownloadWindowController *download );
static void cocoa_register_download( DownloadWindowController *download );
@implementation DownloadWindowController
- initWithContext: (struct download_context *)ctx;
{
if ((self = [super initWithWindowNibName: @"DownloadWindow"]) == nil) return nil;
context = ctx;
totalSize = download_context_get_total_length( context );
[self setURL: [NSURL URLWithString: [NSString stringWithUTF8String: nsurl_access(download_context_get_url( context ))]]];
[self setMIMEType: [NSString stringWithUTF8String: download_context_get_mime_type( context )]];
[self setStartDate: [NSDate date]];
return self;
}
- (void) dealloc;
{
download_context_destroy( context );
[self setURL: nil];
[self setMIMEType: nil];
[self setSaveURL: nil];
[self setOutputFile: nil];
[self setSavedData: nil];
[self setStartDate: nil];
[super dealloc];
}
- (void) abort;
{
download_context_abort( context );
[self removeIfPossible];
}
- (void) askForSave;
{
canClose = NO;
[[NSSavePanel savePanel] beginSheetForDirectory: nil
file: [NSString stringWithUTF8String: download_context_get_filename( context )]
modalForWindow: [self window]
modalDelegate: self
didEndSelector: @selector(savePanelDidEnd:returnCode:contextInfo:)
contextInfo: NULL];
}
- (void)savePanelDidEnd:(NSSavePanel *)sheet returnCode:(int)returnCode contextInfo:(void *)contextInfo;
{
canClose = YES;
if (returnCode == NSCancelButton) {
[self abort];
return;
}
NSURL *targetURL = [sheet URL];
NSString *path = [targetURL path];
[[NSFileManager defaultManager] createFileAtPath: path contents: nil attributes: nil];
FSRef ref;
if (CFURLGetFSRef( (CFURLRef)targetURL, &ref )) {
NSDictionary *attributes = [NSDictionary dictionaryWithObjectsAndKeys:
url, (NSString *)kLSQuarantineDataURLKey,
(NSString *)kLSQuarantineTypeWebDownload, (NSString *)kLSQuarantineTypeKey,
nil];
LSSetItemAttribute( &ref, kLSRolesAll, kLSItemQuarantineProperties, (CFDictionaryRef)attributes );
LOG(("Set quarantine attributes on file %s", [path UTF8String]));
}
[self setOutputFile: [NSFileHandle fileHandleForWritingAtPath: path]];
[self setSaveURL: targetURL];
NSWindow *win = [self window];
[win setRepresentedURL: targetURL];
[win setTitle: [self fileName]];
if (nil == outputFile) {
[self performSelector: @selector(showError:) withObject: @"Cannot create file" afterDelay: 0];
return;
}
if (nil != savedData) {
[outputFile writeData: savedData];
[self setSavedData: nil];
}
[self removeIfPossible];
}
- (BOOL) receivedData: (NSData *)data;
{
if (outputFile) {
[outputFile writeData: data];
} else {
if (nil == savedData) [self setSavedData: [NSMutableData data]];
[savedData appendData: data];
}
[self setReceivedSize: receivedSize + [data length]];
return YES;
}
- (void) showError: (NSString *)error;
{
canClose = NO;
NSAlert *alert = [NSAlert alertWithMessageText: NSLocalizedString( @"Error", @"show error" )
defaultButton: NSLocalizedString( @"OK", @"'OK' button" )
alternateButton: nil otherButton: nil
informativeTextWithFormat: @"%@", error];
[alert beginSheetModalForWindow: [self window] modalDelegate: self
didEndSelector: @selector(alertDidEnd:returnCode:contextInfo:)
contextInfo: NULL];
}
- (void)alertDidEnd:(NSAlert *)alert returnCode:(NSInteger)returnCode contextInfo:(void *)contextInfo;
{
[self abort];
}
- (void) removeIfPossible;
{
if (canClose && shouldClose) {
cocoa_unregister_download( self );
}
}
- (void) downloadDone;
{
shouldClose = YES;
[self removeIfPossible];
}
- (BOOL) windowShouldClose: (id)sender;
{
if ([[NSUserDefaults standardUserDefaults] boolForKey: kAlwaysCancelDownload]) return YES;
NSAlert *ask = [NSAlert alertWithMessageText: NSLocalizedString( @"Cancel download?", @"Download" )
defaultButton: NSLocalizedString( @"Yes", @"" )
alternateButton: NSLocalizedString( @"No", @"" )
otherButton: nil
informativeTextWithFormat: NSLocalizedString( @"Should the download of '%@' really be cancelled?", @"Download" ),
[self fileName]];
[ask setShowsSuppressionButton: YES];
[ask beginSheetModalForWindow: [self window] modalDelegate: self
didEndSelector: @selector(askCancelDidEnd:returnCode:contextInfo:) contextInfo: NULL];
return NO;
}
- (void) windowWillClose: (NSNotification *)notification;
{
[self abort];
}
- (void)askCancelDidEnd:(NSAlert *)alert returnCode:(NSInteger)returnCode contextInfo:(void *)contextInfo;
{
if (returnCode == NSOKButton) {
[[NSUserDefaults standardUserDefaults] setBool: [[alert suppressionButton] state] == NSOnState
forKey: kAlwaysCancelDownload];
[self close];
}
}
#pragma mark -
#pragma mark Properties
@synthesize URL = url;
@synthesize MIMEType = mimeType;
@synthesize totalSize;
@synthesize saveURL;
@synthesize outputFile;
@synthesize savedData;
@synthesize receivedSize;
@synthesize startDate;
+ (NSSet *) keyPathsForValuesAffectingStatusText;
{
return [NSSet setWithObjects: @"totalSize", @"receivedSize", nil];
}
#ifndef NSAppKitVersionNumber10_5
#define NSAppKitVersionNumber10_5 949
#endif
static NSString *cocoa_file_size_string( float size )
{
static unsigned factor = 0;
if (factor == 0) {
if (floor( NSAppKitVersionNumber ) > NSAppKitVersionNumber10_5) factor = 1000;
else factor = 1024;
}
if (size == 0) return @"nothing";
if (size <= 1.0) return @"1 byte";
if (size < factor - 1) return [NSString stringWithFormat:@"%1.0f bytes",size];
size /= factor;
if (size < factor - 1) return [NSString stringWithFormat:@"%1.1f KB", size];
size /= factor;
if (size < factor - 1) return [NSString stringWithFormat:@"%1.1f MB", size];
size /= factor;
if (size < factor - 1) return [NSString stringWithFormat:@"%1.1f GB", size];
size /= factor;
return [NSString stringWithFormat:@"%1.1f TB", size];
}
static NSString *cocoa_time_string( unsigned seconds )
{
if (seconds <= 10) return NSLocalizedString( @"less than 10 seconds", @"time remaining" );
if (seconds < 60) return [NSString stringWithFormat: NSLocalizedString( @"%u seconds", @"time remaining" ), seconds];
unsigned minutes = seconds / 60;
seconds = seconds % 60;
if (minutes < 60) return [NSString stringWithFormat: NSLocalizedString( @"%u:%02u minutes", @"time remaining: minutes, seconds" ) , minutes, seconds];
unsigned hours = minutes / 60;
minutes = minutes % 60;
return [NSString stringWithFormat: NSLocalizedString( @"%2:%02u hours", @"time remaining: hours, minutes" ), hours, minutes];
}
- (NSString *) statusText;
{
NSString *speedString = @"";
float speed = 0.0;
NSTimeInterval elapsedTime = [[NSDate date] timeIntervalSinceDate: startDate];
if (elapsedTime >= 0.1) {
speed = (float)receivedSize / elapsedTime;
speedString = [NSString stringWithFormat: @" (%@/s)", cocoa_file_size_string( speed )];
}
NSString *timeRemainingString = @"";
NSString *totalSizeString = @"";
if (totalSize != 0) {
if (speed > 0.0) {
float timeRemaining = (float)(totalSize - receivedSize) / speed;
timeRemainingString = [NSString stringWithFormat: @": %@", cocoa_time_string( timeRemaining )];
}
totalSizeString = [NSString stringWithFormat: NSLocalizedString( @" of %@", @"... of (total size)" ), cocoa_file_size_string( totalSize )];
}
return [NSString stringWithFormat: @"%@%@%@%@", cocoa_file_size_string( receivedSize ),
totalSizeString, speedString, timeRemainingString];
}
+ (NSSet *) keyPathsForValuesAffectingFileName;
{
return [NSSet setWithObject: @"saveURL"];
}
- (NSString *) fileName;
{
return [[saveURL path] lastPathComponent];
}
+ (NSSet *) keyPathsForValuesAffectingIcon;
{
return [NSSet setWithObjects: @"mimeType", @"URL", nil];
}
- (NSImage *) icon;
{
NSString *type = [(NSString *)UTTypeCreatePreferredIdentifierForTag( kUTTagClassMIMEType, (CFStringRef)mimeType, NULL ) autorelease];
if ([type hasPrefix: @"dyn."] || [type isEqualToString: (NSString *)kUTTypeData]) {
NSString *pathExt = [[url path] pathExtension];
type = [(NSString *)UTTypeCreatePreferredIdentifierForTag( kUTTagClassFilenameExtension, (CFStringRef)pathExt, NULL ) autorelease];
}
return [[NSWorkspace sharedWorkspace] iconForFileType: type];
}
#pragma mark -
#pragma mark NetSurf interface functions
static struct gui_download_window *gui_download_window_create(download_context *ctx,
struct gui_window *parent)
{
DownloadWindowController * const window = [[DownloadWindowController alloc] initWithContext: ctx];
cocoa_register_download( window );
[window askForSave];
[window release];
return (struct gui_download_window *)window;
}
static nserror gui_download_window_data(struct gui_download_window *dw,
const char *data, unsigned int size)
{
DownloadWindowController * const window = (DownloadWindowController *)dw;
return [window receivedData: [NSData dataWithBytes: data length: size]] ? NSERROR_OK : NSERROR_SAVE_FAILED;
}
static void gui_download_window_error(struct gui_download_window *dw,
const char *error_msg)
{
DownloadWindowController * const window = (DownloadWindowController *)dw;
[window showError: [NSString stringWithUTF8String: error_msg]];
}
static void gui_download_window_done(struct gui_download_window *dw)
{
DownloadWindowController * const window = (DownloadWindowController *)dw;
[window downloadDone];
}
@end
#pragma mark -
static NSMutableSet *cocoa_all_downloads = nil;
static void cocoa_register_download( DownloadWindowController *download )
{
if (cocoa_all_downloads == nil) {
cocoa_all_downloads = [[NSMutableSet alloc] init];
}
[cocoa_all_downloads addObject: download];
}
static void cocoa_unregister_download( DownloadWindowController *download )
{
[cocoa_all_downloads removeObject: download];
}
static struct gui_download_table download_table = {
.create = gui_download_window_create,
.data = gui_download_window_data,
.error = gui_download_window_error,
.done = gui_download_window_done,
};
struct gui_download_table *cocoa_download_table = &download_table;