1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #import "chrome/browser/ui/cocoa/certificate_viewer_mac.h"
7 #include <Security/Security.h>
8 #include <SecurityInterface/SFCertificatePanel.h>
11 #include "base/mac/foundation_util.h"
12 #include "base/mac/scoped_cftyperef.h"
13 #include "chrome/browser/certificate_viewer.h"
14 #import "chrome/browser/ui/cocoa/constrained_window/constrained_window_mac.h"
15 #import "chrome/browser/ui/cocoa/constrained_window/constrained_window_sheet.h"
16 #import "chrome/browser/ui/cocoa/constrained_window/constrained_window_sheet_controller.h"
17 #include "net/cert/x509_certificate.h"
18 #include "net/cert/x509_util_mac.h"
19 #import "ui/base/cocoa/window_size_constants.h"
21 class SSLCertificateViewerCocoaBridge;
23 @interface SFCertificatePanel (SystemPrivate)
24 // A system-private interface that dismisses a panel whose sheet was started by
25 // -beginSheetForWindow:
31 // as though the user clicked the button identified by returnCode. Verified
33 - (void)_dismissWithCode:(NSInteger)code;
36 @interface SSLCertificateViewerCocoa ()
37 - (void)onConstrainedWindowClosed;
40 class SSLCertificateViewerCocoaBridge : public ConstrainedWindowMacDelegate {
42 explicit SSLCertificateViewerCocoaBridge(SSLCertificateViewerCocoa *
44 : controller_(controller) {
47 virtual ~SSLCertificateViewerCocoaBridge() {}
49 // ConstrainedWindowMacDelegate implementation:
50 virtual void OnConstrainedWindowClosed(
51 ConstrainedWindowMac * window) OVERRIDE {
52 // |onConstrainedWindowClosed| will delete the sheet which might be still
53 // in use higher up the call stack. Wait for the next cycle of the event
54 // loop to call this function.
55 [controller_ performSelector:@selector(onConstrainedWindowClosed)
61 SSLCertificateViewerCocoa* controller_; // weak
63 DISALLOW_COPY_AND_ASSIGN(SSLCertificateViewerCocoaBridge);
66 void ShowCertificateViewer(content::WebContents* web_contents,
67 gfx::NativeWindow parent,
68 net::X509Certificate* cert) {
69 // SSLCertificateViewerCocoa will manage its own lifetime and will release
70 // itself when the dialog is closed.
71 // See -[SSLCertificateViewerCocoa onConstrainedWindowClosed].
72 SSLCertificateViewerCocoa* viewer =
73 [[SSLCertificateViewerCocoa alloc] initWithCertificate:cert];
74 [viewer displayForWebContents:web_contents];
77 @implementation SSLCertificateViewerCocoa
79 - (id)initWithCertificate:(net::X509Certificate*)certificate {
80 if ((self = [super init])) {
81 base::ScopedCFTypeRef<CFArrayRef> cert_chain(
82 certificate->CreateOSCertChainForCert());
83 NSArray* certificates = base::mac::CFToNSCast(cert_chain.get());
84 certificates_.reset([certificates retain]);
89 - (void)sheetDidEnd:(NSWindow*)parent
90 returnCode:(NSInteger)returnCode
91 context:(void*)context {
93 constrainedWindow_->CloseWebContentsModalDialog();
96 - (void)displayForWebContents:(content::WebContents*)webContents {
97 // Explicitly disable revocation checking, regardless of user preferences
98 // or system settings. The behaviour of SFCertificatePanel is to call
99 // SecTrustEvaluate on the certificate(s) supplied, effectively
100 // duplicating the behaviour of net::X509Certificate::Verify(). However,
101 // this call stalls the UI if revocation checking is enabled in the
102 // Keychain preferences or if the cert may be an EV cert. By disabling
103 // revocation checking, the stall is limited to the time taken for path
104 // building and verification, which should be minimized due to the path
105 // being provided in |certificates|. This does not affect normal
106 // revocation checking from happening, which is controlled by
107 // net::X509Certificate::Verify() and user preferences, but will prevent
108 // the certificate viewer UI from displaying which certificate is revoked.
109 // This is acceptable, as certificate revocation will still be shown in
110 // the page info bubble if a certificate in the chain is actually revoked.
111 base::ScopedCFTypeRef<CFMutableArrayRef> policies(
112 CFArrayCreateMutable(kCFAllocatorDefault, 0, &kCFTypeArrayCallBacks));
113 if (!policies.get()) {
117 // Add a basic X.509 policy, in order to match the behaviour of
118 // SFCertificatePanel when no policies are specified.
119 SecPolicyRef basic_policy = NULL;
120 OSStatus status = net::x509_util::CreateBasicX509Policy(&basic_policy);
121 if (status != noErr) {
125 CFArrayAppendValue(policies, basic_policy);
126 CFRelease(basic_policy);
128 status = net::x509_util::CreateRevocationPolicies(false, false, policies);
129 if (status != noErr) {
134 panel_.reset([[SFCertificatePanel alloc] init]);
135 [panel_ setPolicies:(id) policies.get()];
137 constrainedWindow_.reset(
138 new ConstrainedWindowMac(observer_.get(), webContents, self));
141 - (NSWindow*)overlayWindow {
142 return overlayWindow_;
145 - (void)showSheetForWindow:(NSWindow*)window {
146 overlayWindow_.reset([window retain]);
147 [panel_ beginSheetForWindow:window
149 didEndSelector:@selector(sheetDidEnd:
153 certificates:certificates_
157 - (void)closeSheetWithAnimation:(BOOL)withAnimation {
159 overlayWindow_.reset();
160 // Closing the sheet using -[NSApp endSheet:] doesn't work so use the private
162 [panel_ _dismissWithCode:NSFileHandlingPanelCancelButton];
166 NSWindow* sheetWindow = [overlayWindow_ attachedSheet];
167 [sheetWindow setAlphaValue:0.0];
169 oldResizesSubviews_ = [[sheetWindow contentView] autoresizesSubviews];
170 [[sheetWindow contentView] setAutoresizesSubviews:NO];
172 oldSheetFrame_ = [sheetWindow frame];
173 NSRect overlayFrame = [overlayWindow_ frame];
174 oldSheetFrame_.origin.x -= NSMinX(overlayFrame);
175 oldSheetFrame_.origin.y -= NSMinY(overlayFrame);
176 [sheetWindow setFrame:ui::kWindowSizeDeterminedLater display:NO];
179 - (void)unhideSheet {
180 NSWindow* sheetWindow = [overlayWindow_ attachedSheet];
181 NSRect overlayFrame = [overlayWindow_ frame];
182 oldSheetFrame_.origin.x += NSMinX(overlayFrame);
183 oldSheetFrame_.origin.y += NSMinY(overlayFrame);
184 [sheetWindow setFrame:oldSheetFrame_ display:NO];
185 [[sheetWindow contentView] setAutoresizesSubviews:oldResizesSubviews_];
186 [[overlayWindow_ attachedSheet] setAlphaValue:1.0];
193 - (void)makeSheetKeyAndOrderFront {
194 [[overlayWindow_ attachedSheet] makeKeyAndOrderFront:nil];
197 - (void)updateSheetPosition {
201 - (void)onConstrainedWindowClosed {
203 constrainedWindow_.reset();