/* * Copyright (C) 2009 Apple Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #if !ENABLE_SPARKLE void initializeSparkle() { // No-op. } #else // ENABLE_SPARKLE #import <Cocoa/Cocoa.h> #import <Sparkle/SUUpdater.h> #import <objc/objc-runtime.h> #import "WebKitNightlyEnabler.h" // We need to tweak the wording of the prompt to make sense in the context of WebKit and Safari. static NSString* updatePermissionPromptDescription(id self, SEL _cmd) { return @"Should WebKit automatically check for updates? You can always check for updates manually from the Safari menu."; } static NSPanel *updateAlertPanel(id updateItem, id host) { NSString *hostName = objc_msgSend(host, @selector(name)); NSPanel *panel = NSGetInformationalAlertPanel([NSString stringWithFormat:@"Would you like to download and install %@ %@ now?", hostName, objc_msgSend(updateItem, @selector(displayVersionString))], [NSString stringWithFormat:@"You are currently running %@ %@.", hostName, objc_msgSend(host, @selector(displayVersion))], @"Install Update", @"Skip This Version", @"Remind Me Later"); NSArray *subviews = [[panel contentView] subviews]; NSEnumerator *e = [subviews objectEnumerator]; NSView *view; while ((view = [e nextObject])) { if (![view isKindOfClass:[NSButton class]]) continue; NSButton *button = (NSButton *)view; [button setAction:@selector(webKitHandleButtonPress:)]; if ([button tag] == NSAlertOtherReturn) [button setKeyEquivalent:@"\033"]; } [panel center]; return panel; } // Sparkle's udpate alert panel looks odd with the release notes hidden, so we // swap it out with a standard NSAlert-style panel instead. static id updateAlertInitForAlertPanel(id self, SEL _cmd, id updateItem, id host) { NSPanel *panel = updateAlertPanel(updateItem, host); [panel setDelegate:self]; self = [self initWithWindow:panel]; if (!self) return nil; [updateItem retain]; [host retain]; object_setInstanceVariable(self, "updateItem", (void*)updateItem); object_setInstanceVariable(self, "host", (void*)host); [self setShouldCascadeWindows:NO]; return self; } @implementation NSAlert (WebKitLauncherExtensions) - (void)webKitHandleButtonPress:(id)sender { // We rely on the fact that NSAlertOtherReturn == -1, NSAlertAlternateReturn == 0 and NSAlertDefaultReturn == 1 // to map the button tag to the corresponding selector SEL selectors[] = { @selector(remindMeLater:), @selector(skipThisVersion:), @selector(installUpdate:) }; SEL selector = selectors[[sender tag] + 1]; id delegate = [[sender window] delegate]; objc_msgSend(delegate, selector, sender); } @end #if __LP64__ #define setMethodImplementation method_setImplementation #else static void setMethodImplementation(Method m, IMP imp) { m->method_imp = imp; } #endif static NSString *userAgentStringForSparkle() { NSBundle *safariBundle = [NSBundle mainBundle]; NSString *safariVersion = [[safariBundle localizedInfoDictionary] valueForKey:@"CFBundleShortVersionString"]; NSString *safariBuild = [[[safariBundle infoDictionary] valueForKey:(NSString *)kCFBundleVersionKey] substringFromIndex:1]; NSString *webKitRevision = [[webKitLauncherBundle() infoDictionary] valueForKey:(NSString *)kCFBundleVersionKey]; NSString *applicationName = [NSString stringWithFormat:@"Version/%@ Safari/%@ WebKitRevision/%@", safariVersion, safariBuild, webKitRevision]; Class WebView = objc_lookUpClass("WebView"); return objc_msgSend(WebView, @selector(_standardUserAgentWithApplicationName:), applicationName); } void initializeSparkle() { // Override some Sparkle behaviour Method methodToPatch = class_getInstanceMethod(objc_getRequiredClass("SUUpdatePermissionPrompt"), @selector(promptDescription)); setMethodImplementation(methodToPatch, (IMP)updatePermissionPromptDescription); methodToPatch = class_getInstanceMethod(objc_getRequiredClass("SUUpdateAlert"), @selector(initWithAppcastItem:host:)); setMethodImplementation(methodToPatch, (IMP)updateAlertInitForAlertPanel); SUUpdater *updater = [SUUpdater updaterForBundle:webKitLauncherBundle()]; [updater setUserAgentString:userAgentStringForSparkle()]; // Find the first separator on the Safari menu… NSMenu *applicationSubmenu = [[[NSApp mainMenu] itemAtIndex:0] submenu]; int i = 0; for (; i < [applicationSubmenu numberOfItems]; i++) { if ([[applicationSubmenu itemAtIndex:i] isSeparatorItem]) break; } // … and insert a menu item that can be used to manually trigger update checks. NSMenuItem *updateMenuItem = [[NSMenuItem alloc] initWithTitle:@"Check for WebKit Updates…" action:@selector(checkForUpdates:) keyEquivalent:@""]; [updateMenuItem setTarget:updater]; [applicationSubmenu insertItem:updateMenuItem atIndex:i]; [updateMenuItem release]; } #endif // ENABLE_SPARKLE