123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293 |
- /*
- * Copyright (C) 2005 Apple Computer, 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.
- * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
- * its contributors may be used to endorse or promote products derived
- * from this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "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 OR ITS 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.
- */
- #import <WebKit/WebAuthenticationPanel.h>
- #import "WebLocalizableStringsInternal.h"
- #import <Foundation/NSURLAuthenticationChallenge.h>
- #import <Foundation/NSURLProtectionSpace.h>
- #import <Foundation/NSURLCredential.h>
- #import <WebKit/WebKitNSStringExtras.h>
- #import <WebKit/WebNSURLExtras.h>
- #import <wtf/Assertions.h>
- #import <WebKit/WebNSControlExtras.h>
- #define WebAuthenticationPanelNibName @"WebAuthenticationPanel"
- @implementation WebAuthenticationPanel
- -(id)initWithCallback:(id)cb selector:(SEL)sel
- {
- self = [self init];
- if (self != nil) {
- callback = [cb retain];
- selector = sel;
- }
- return self;
- }
- - (void)dealloc
- {
- [panel release];
- [callback release];
-
- [super dealloc];
- }
- // IB actions
- - (IBAction)cancel:(id)sender
- {
- // This is required because the body of this method is going to
- // remove all of the panel's remaining refs, which can cause a
- // crash later when finishing button hit tracking. So we make
- // sure it lives on a bit longer.
- [[panel retain] autorelease];
-
- // This is required as a workaround for AppKit issue 4118422
- [[self retain] autorelease];
- [panel close];
- if (usingSheet) {
- [[NSApplication sharedApplication] endSheet:panel returnCode:1];
- } else {
- [[NSApplication sharedApplication] stopModalWithCode:1];
- }
- }
- - (IBAction)logIn:(id)sender
- {
- // This is required because the body of this method is going to
- // remove all of the panel's remaining refs, which can cause a
- // crash later when finishing button hit tracking. So we make
- // sure it lives on a bit longer.
- [[panel retain] autorelease];
- [panel close];
- if (usingSheet) {
- [[NSApplication sharedApplication] endSheet:panel returnCode:0];
- } else {
- [[NSApplication sharedApplication] stopModalWithCode:0];
- }
- }
- - (BOOL)loadNib
- {
- if (!nibLoaded) {
- #pragma clang diagnostic push
- #pragma clang diagnostic ignored "-Wdeprecated-declarations"
- if ([NSBundle loadNibNamed:WebAuthenticationPanelNibName owner:self]) {
- #pragma clang diagnostic pop
- nibLoaded = YES;
- [imageView setImage:[NSImage imageNamed:@"NSApplicationIcon"]];
- } else {
- LOG_ERROR("couldn't load nib named '%@'", WebAuthenticationPanelNibName);
- return FALSE;
- }
- }
- return TRUE;
- }
- // Methods related to displaying the panel
- -(void)setUpForChallenge:(NSURLAuthenticationChallenge *)chall
- {
- [self loadNib];
- NSURLProtectionSpace *space = [chall protectionSpace];
- NSString *host;
- if ([space port] == 0) {
- host = [[space host] _web_decodeHostName];
- } else {
- host = [NSString stringWithFormat:@"%@:%ld", [[space host] _web_decodeHostName], (long)[space port]];
- }
- NSString *realm = [space realm];
- if (!realm)
- realm = @"";
- NSString *message;
- // Consider the realm name to be "simple" if it does not contain any whitespace or newline characters.
- // If the realm name is determined to be complex, we will use a slightly different sheet layout, designed
- // to keep a malicious realm name from spoofing the wording in the sheet text.
- BOOL realmNameIsSimple = [realm rangeOfCharacterFromSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]].location == NSNotFound;
-
- if ([chall previousFailureCount] == 0) {
- if ([space isProxy]) {
- message = [NSString stringWithFormat:UI_STRING_INTERNAL("To view this page, you must log in to the %@ proxy server %@.",
- "prompt string in authentication panel"),
- [space proxyType], host];
- } else {
- if (realmNameIsSimple)
- message = [NSString stringWithFormat:UI_STRING_INTERNAL("To view this page, you must log in to area “%@” on %@.",
- "prompt string in authentication panel"), realm, host];
- else
- message = [NSString stringWithFormat:UI_STRING_INTERNAL("To view this page, you must log in to this area on %@:",
- "prompt string in authentication panel"), host];
- }
- } else {
- if ([space isProxy]) {
- message = [NSString stringWithFormat:UI_STRING_INTERNAL("The user name or password you entered for the %@ proxy server %@ was incorrect. Make sure you’re entering them correctly, and then try again.",
- "prompt string in authentication panel"),
- [space proxyType], host];
- } else {
- if (realmNameIsSimple)
- message = [NSString stringWithFormat:UI_STRING_INTERNAL("The user name or password you entered for area “%@” on %@ was incorrect. Make sure you’re entering them correctly, and then try again.",
- "prompt string in authentication panel"), realm, host];
- else
- message = [NSString stringWithFormat:UI_STRING_INTERNAL("The user name or password you entered for this area on %@ was incorrect. Make sure you’re entering them correctly, and then try again.",
- "prompt string in authentication panel"), host];
- }
- }
-
- if (![space isProxy] && !realmNameIsSimple) {
- [separateRealmLabel setHidden:NO];
- [separateRealmLabel setStringValue:realm];
- [separateRealmLabel setAutoresizingMask:NSViewMinYMargin];
- [separateRealmLabel sizeToFitAndAdjustWindowHeight];
- [separateRealmLabel setAutoresizingMask:NSViewMaxYMargin];
- } else {
- // In the proxy or "simple" realm name case, we need to hide the 'separateRealmLabel'
- // and move the rest of the contents up appropriately to fill the space.
- NSRect mainLabelFrame = [mainLabel frame];
- NSRect realmFrame = [separateRealmLabel frame];
- NSRect smallLabelFrame = [smallLabel frame];
- // Find the distance between the 'smallLabel' and the label above it, initially the 'separateRealmLabel'.
- // Then, find the current distance between 'smallLabel' and 'mainLabel'. The difference between
- // these two is how much shorter the panel needs to be after hiding the 'separateRealmLabel'.
- CGFloat smallLabelMargin = NSMinY(realmFrame) - NSMaxY(smallLabelFrame);
- CGFloat smallLabelToMainLabel = NSMinY(mainLabelFrame) - NSMaxY(smallLabelFrame);
- CGFloat deltaMargin = smallLabelToMainLabel - smallLabelMargin;
-
- [separateRealmLabel setHidden:YES];
- NSRect windowFrame = [panel frame];
- windowFrame.size.height -= deltaMargin;
- [panel setFrame:windowFrame display:NO];
- }
-
- [mainLabel setStringValue:message];
- [mainLabel sizeToFitAndAdjustWindowHeight];
- if ([space receivesCredentialSecurely] || [[space protocol] _webkit_isCaseInsensitiveEqualToString:@"https"]) {
- [smallLabel setStringValue:
- UI_STRING_INTERNAL("Your login information will be sent securely.",
- "message in authentication panel")];
- } else {
- // Use this scary-sounding phrase only when using basic auth with non-https servers. In this case the password
- // could be sniffed by intercepting the network traffic.
- [smallLabel setStringValue:
- UI_STRING_INTERNAL("Your password will be sent unencrypted.",
- "message in authentication panel")];
- }
- if ([[chall proposedCredential] user] != nil) {
- [username setStringValue:[[chall proposedCredential] user]];
- [panel setInitialFirstResponder:password];
- } else {
- [username setStringValue:@""];
- [password setStringValue:@""];
- [panel setInitialFirstResponder:username];
- }
- }
- - (void)runAsModalDialogWithChallenge:(NSURLAuthenticationChallenge *)chall
- {
- [self setUpForChallenge:chall];
- usingSheet = FALSE;
- [chall retain];
- NSURLCredential *credential = nil;
- if ([[NSApplication sharedApplication] runModalForWindow:panel] == 0) {
- credential = [[NSURLCredential alloc] initWithUser:[username stringValue] password:[password stringValue] persistence:([remember state] == NSOnState) ? NSURLCredentialPersistencePermanent : NSURLCredentialPersistenceForSession];
- }
- [callback performSelector:selector withObject:chall withObject:credential];
- [credential release];
- [chall release];
- }
- - (void)runAsSheetOnWindow:(NSWindow *)window withChallenge:(NSURLAuthenticationChallenge *)chall
- {
- ASSERT(!usingSheet);
- [self setUpForChallenge:chall];
- usingSheet = TRUE;
- challenge = [chall retain];
-
- [[NSApplication sharedApplication] beginSheet:panel modalForWindow:window modalDelegate:self didEndSelector:@selector(sheetDidEnd:returnCode:contextInfo:) contextInfo:NULL];
- }
- - (void)sheetDidEnd:(NSWindow *)sheet returnCode:(int)returnCode contextInfo:(void *)contextInfo
- {
- NSURLCredential *credential = nil;
- NSURLAuthenticationChallenge *chall;
- ASSERT(usingSheet);
- ASSERT(challenge != nil);
- if (returnCode == 0) {
- credential = [[NSURLCredential alloc] initWithUser:[username stringValue] password:[password stringValue] persistence:([remember state] == NSOnState) ? NSURLCredentialPersistencePermanent : NSURLCredentialPersistenceForSession];
- }
- // We take this tricky approach to nilling out and releasing the challenge
- // because the callback below might remove our last ref.
- chall = challenge;
- challenge = nil;
- [callback performSelector:selector withObject:chall withObject:credential];
- [credential release];
- [chall release];
- }
- @end
- @implementation WebNonBlockingPanel
- - (BOOL)_blocksActionWhenModal:(SEL)theAction
- {
- // This override of a private AppKit method allows the user to quit when a login dialog
- // is onscreen, which is nice in general but in particular prevents pathological cases
- // like 3744583 from requiring a Force Quit.
- //
- // It would be nice to allow closing the individual window as well as quitting the app when
- // a login sheet is up, but this _blocksActionWhenModal: mechanism doesn't support that.
- // This override matches those in NSOpenPanel and NSToolbarConfigPanel.
- if (theAction == @selector(terminate:)) {
- return NO;
- }
- return YES;
- }
- @end
|