Index: chrome/browser/cocoa/external_protocol_dialog.mm |
=================================================================== |
--- chrome/browser/cocoa/external_protocol_dialog.mm (revision 0) |
+++ chrome/browser/cocoa/external_protocol_dialog.mm (revision 0) |
@@ -0,0 +1,148 @@ |
+// Copyright (c) 2009 The Chromium Authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+#import "chrome/browser/cocoa/external_protocol_dialog.h" |
+ |
+#include "app/l10n_util_mac.h" |
+#include "base/message_loop.h" |
+#include "base/sys_string_conversions.h" |
+#include "chrome/browser/external_protocol_handler.h" |
+#include "grit/chromium_strings.h" |
+#include "grit/generated_resources.h" |
+ |
+/////////////////////////////////////////////////////////////////////////////// |
+// ExternalProtocolHandler |
+ |
+// static |
+void ExternalProtocolHandler::RunExternalProtocolDialog( |
+ const GURL& url, int render_process_host_id, int routing_id) { |
+ [[ExternalProtocolDialogController alloc] initWithGURL:&url]; |
+} |
+ |
+/////////////////////////////////////////////////////////////////////////////// |
+// ExternalProtocolDialogController |
+ |
+@interface ExternalProtocolDialogController(Private) |
+- (void)alertEnded:(NSAlert *)alert |
+ returnCode:(int)returnCode |
+ contextInfo:(void*)contextInfo; |
+- (string16)appNameForProtocol; |
+@end |
+ |
+@implementation ExternalProtocolDialogController |
+- (id)initWithGURL:(const GURL*)url { |
+ DCHECK_EQ(MessageLoop::TYPE_UI, MessageLoop::current()->type()); |
+ |
+ url_ = *url; |
+ creation_time_ = base::Time::Now(); |
+ |
+ string16 appName = [self appNameForProtocol]; |
+ if (appName.length() == 0) { |
+ // No registered apps for this protocol; give up and go home. |
+ [self autorelease]; |
+ return nil; |
+ } |
+ |
+ alert_ = [[NSAlert alloc] init]; |
+ |
+ [alert_ setMessageText: |
+ l10n_util::GetNSStringWithFixup(IDS_EXTERNAL_PROTOCOL_TITLE)]; |
+ |
+ NSButton* allowButton = [alert_ addButtonWithTitle: |
+ l10n_util::GetNSStringWithFixup(IDS_EXTERNAL_PROTOCOL_OK_BUTTON_TEXT)]; |
+ [allowButton setKeyEquivalent:@""]; // disallow as default |
+ [alert_ addButtonWithTitle: |
+ l10n_util::GetNSStringWithFixup(IDS_CANCEL)]; |
+ |
+ NSString* urlString = l10n_util::GetNSStringFWithFixup( |
+ IDS_EXTERNAL_PROTOCOL_INFORMATION, |
+ ASCIIToUTF16(url_.scheme() + ":"), |
+ ASCIIToUTF16(url_.possibly_invalid_spec())); |
+ NSString* appString = l10n_util::GetNSStringFWithFixup( |
+ IDS_EXTERNAL_PROTOCOL_APPLICATION_TO_LAUNCH, |
+ appName); |
+ NSString* warningString = |
+ l10n_util::GetNSStringWithFixup(IDS_EXTERNAL_PROTOCOL_WARNING); |
+ NSString* informativeText = |
+ [NSString stringWithFormat:@"%@\n\n%@\n\n%@", |
+ urlString, |
+ appString, |
+ warningString]; |
TVL
2009/10/06 14:41:02
drive by- maybe a comment as to why this is always
Avi (use Gerrit)
2009/10/06 14:48:43
Yes, the other platforms do the same thing, but wh
|
+ |
+ [alert_ setInformativeText:informativeText]; |
+ |
+ [alert_ setShowsSuppressionButton:YES]; |
+ [[alert_ suppressionButton] setTitle: |
+ l10n_util::GetNSStringWithFixup(IDS_EXTERNAL_PROTOCOL_CHECKBOX_TEXT)]; |
+ |
+ [alert_ beginSheetModalForWindow:nil // nil here makes it app-modal |
+ modalDelegate:self |
+ didEndSelector:@selector(alertEnded:returnCode:contextInfo:) |
+ contextInfo:nil]; |
+ |
+ return self; |
+} |
+ |
+- (void)dealloc { |
+ [alert_ release]; |
+ |
+ [super dealloc]; |
+} |
+ |
+- (void)alertEnded:(NSAlert *)alert |
+ returnCode:(int)returnCode |
+ contextInfo:(void*)contextInfo { |
+ ExternalProtocolHandler::BlockState blockState = |
+ ExternalProtocolHandler::UNKNOWN; |
+ switch (returnCode) { |
+ case NSAlertFirstButtonReturn: |
+ blockState = ExternalProtocolHandler::DONT_BLOCK; |
+ break; |
+ case NSAlertSecondButtonReturn: |
+ blockState = ExternalProtocolHandler::BLOCK; |
+ break; |
+ default: |
+ NOTREACHED(); |
+ } |
+ |
+ // Set the "don't warn me again" info if the protocol was blocked ("cancel" |
+ // always means "make this dialog go away with no permanent effect" no matter |
+ // what). |
+ if ([[alert_ suppressionButton] state] == NSOnState && |
+ blockState == ExternalProtocolHandler::BLOCK) { |
+ ExternalProtocolHandler::SetBlockState(UTF8ToWide(url_.scheme()), |
+ blockState); |
+ } |
+ |
+ if (blockState == ExternalProtocolHandler::DONT_BLOCK) { |
+ UMA_HISTOGRAM_LONG_TIMES("clickjacking.launch_url", |
+ base::Time::Now() - creation_time_); |
+ |
+ ExternalProtocolHandler::LaunchUrlWithoutSecurityCheck(url_); |
+ } |
+ |
+ [self autorelease]; |
+} |
+ |
+- (string16)appNameForProtocol { |
+ NSURL* url = [NSURL URLWithString: |
+ base::SysUTF8ToNSString(url_.possibly_invalid_spec())]; |
+ CFURLRef openingApp = NULL; |
+ OSStatus status = LSGetApplicationForURL((CFURLRef)url, |
+ kLSRolesAll, |
+ NULL, |
+ &openingApp); |
+ if (status != noErr) { |
+ // likely kLSApplicationNotFoundErr |
+ return string16(); |
+ } |
+ NSString* appPath = [(NSURL*)openingApp path]; |
+ CFRelease(openingApp); // NOT A BUG; LSGetApplicationForURL retains for us |
+ NSString* appDisplayName = |
+ [[NSFileManager defaultManager] displayNameAtPath:appPath]; |
+ |
+ return base::SysNSStringToUTF16(appDisplayName); |
+} |
+ |
+@end |
Property changes on: chrome/browser/cocoa/external_protocol_dialog.mm |
___________________________________________________________________ |
Name: svn:eol-style |
+ LF |