Chromium Code Reviews| Index: chrome/browser/ui/cocoa/passwords/credential_item_view.mm |
| diff --git a/chrome/browser/ui/cocoa/passwords/credential_item_view.mm b/chrome/browser/ui/cocoa/passwords/credential_item_view.mm |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..49801dd049860103f56699423a0fa8eb0d3f22a1 |
| --- /dev/null |
| +++ b/chrome/browser/ui/cocoa/passwords/credential_item_view.mm |
| @@ -0,0 +1,227 @@ |
| +// Copyright 2015 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/ui/cocoa/passwords/credential_item_view.h" |
| + |
| +#include <algorithm> |
| + |
| +#include "base/i18n/rtl.h" |
| +#include "base/mac/foundation_util.h" |
| +#include "base/strings/sys_string_conversions.h" |
| +#include "chrome/browser/ui/passwords/manage_passwords_bubble_model.h" |
| +#include "chrome/browser/ui/passwords/manage_passwords_view_utils.h" |
| +#include "grit/theme_resources.h" |
| +#include "ui/base/resource/resource_bundle.h" |
| +#include "ui/gfx/image/image_skia.h" |
| +#include "ui/gfx/image/image_skia_util_mac.h" |
| + |
| +namespace { |
| +const CGFloat kVerticalPaddingBetweenCredentials = 10.0f; |
| +const CGFloat kHorizontalPaddingBetweenAvatarAndLabels = 10.0f; |
| +const CGFloat kVerticalPaddingBetweenLabels = 2.0f; |
| +} // namespace |
| + |
| +// Custom button cell that draws nothing. |
| +@interface CredentialItemViewCell : NSButtonCell |
|
groby-ooo-7-16
2015/02/19 17:01:33
Are you sure you need an NSButton at all? :)
dconnelly
2015/02/20 15:36:06
Replaced all this handling and ListView stuff with
|
| +@end |
| + |
| +@implementation CredentialItemViewCell |
| +- (void)drawWithFrame:(CGRect)frame inView:(NSView*)view { |
| + // Do nothing. |
| +} |
| +@end |
| + |
| +@interface CredentialItemView() |
| +- (void)click:(id)sender; |
| +@end |
| + |
| +@implementation CredentialItemView |
| + |
| +- (id)initWithPasswordForm:(const autofill::PasswordForm&)passwordForm |
| + credentialType:(password_manager::CredentialType)credentialType |
| + delegate:(id<CredentialItemDelegate>)delegate { |
| + if ((self = [super init])) { |
| + passwordForm_ = passwordForm; |
| + credentialType_ = credentialType; |
| + delegate_ = delegate; |
| + |
| + // Forward clicks to the delegate. |
| + [self setTarget:self]; |
| + [self setAction:@selector(click:)]; |
| + |
| + // ----------------------------------------------- |
| + // | | John Q. Facebooker | |
| + // | icon | john@somewhere.com | |
| + // ----------------------------------------------- |
| + |
| + // Create the views. |
| + |
| + avatarView_.reset([[NSImageView alloc] initWithFrame:CGRectZero]); |
|
groby-ooo-7-16
2015/02/19 17:01:33
I vaguely recall the profile switcher does similar
dconnelly
2015/02/20 15:36:06
The profile switchers I see on trunk Chromium and
|
| + [avatarView_ setWantsLayer:YES]; |
| + [[avatarView_ layer] setCornerRadius:kAvatarImageSize / 2.0f]; |
| + [[avatarView_ layer] setMasksToBounds:YES]; |
| + [self addSubview:avatarView_]; |
| + |
| + if (!passwordForm_.display_name.empty()) { |
| + nameLabel_.reset([[NSTextField alloc] initWithFrame:NSZeroRect]); |
| + [self addSubview:nameLabel_]; |
| + [nameLabel_ setBezeled:NO]; |
| + [nameLabel_ setDrawsBackground:NO]; |
| + [nameLabel_ setEditable:NO]; |
| + [nameLabel_ setSelectable:NO]; |
| + [nameLabel_ |
| + setStringValue:base::SysUTF16ToNSString(passwordForm_.display_name)]; |
| + [nameLabel_ setAlignment:base::i18n::IsRTL() ? NSRightTextAlignment |
| + : NSLeftTextAlignment]; |
| + [nameLabel_ sizeToFit]; |
| + } |
| + |
| + usernameLabel_.reset([[NSTextField alloc] initWithFrame:NSZeroRect]); |
| + [self addSubview:usernameLabel_]; |
| + [usernameLabel_ setBezeled:NO]; |
| + [usernameLabel_ setDrawsBackground:NO]; |
| + [usernameLabel_ setEditable:NO]; |
| + [usernameLabel_ setSelectable:NO]; |
| + [usernameLabel_ |
| + setStringValue:base::SysUTF16ToNSString(passwordForm_.username_value)]; |
| + [usernameLabel_ setAlignment:base::i18n::IsRTL() ? NSRightTextAlignment |
| + : NSLeftTextAlignment]; |
| + [usernameLabel_ sizeToFit]; |
| + |
| + // Compute the heights and widths of everything, as the layout depends on |
| + // these measurements. |
| + const CGFloat labelsHeight = CGRectGetHeight([nameLabel_ frame]) + |
| + CGRectGetHeight([usernameLabel_ frame]) + |
| + kVerticalPaddingBetweenLabels; |
| + const CGFloat height = std::max(labelsHeight, CGFloat(kAvatarImageSize)); |
| + const CGFloat width = kAvatarImageSize + |
| + kHorizontalPaddingBetweenAvatarAndLabels + |
| + std::max(CGRectGetWidth([nameLabel_ frame]), |
| + CGRectGetWidth([usernameLabel_ frame])); |
| + self.frame = CGRectMake(0, 0, width, height); |
| + |
| + // Lay out the views (RTL reverses the order horizontally). |
| + |
| + const CGFloat avatarX = base::i18n::IsRTL() ? width - kAvatarImageSize : 0; |
| + const CGFloat avatarY = |
| + (kAvatarImageSize > height) ? 0 : (height - kAvatarImageSize) / 2.0f; |
| + [avatarView_ setFrame:CGRectMake(avatarX, avatarY, kAvatarImageSize, |
| + kAvatarImageSize)]; |
| + |
| + const CGFloat nameX = base::i18n::IsRTL() |
| + ? NSMinX([avatarView_ frame]) - |
| + kHorizontalPaddingBetweenAvatarAndLabels - |
| + CGRectGetWidth([nameLabel_ frame]) |
| + : NSMaxX([avatarView_ frame]) + |
| + kHorizontalPaddingBetweenAvatarAndLabels; |
| + const CGFloat nameLabelY = |
| + (labelsHeight > height) ? 0 : (height - labelsHeight) / 2.0f; |
| + CGRect nameFrame = [nameLabel_ frame]; |
| + nameFrame.origin = CGPointMake(nameX, nameLabelY); |
|
groby-ooo-7-16
2015/02/19 17:01:33
NSMakePoint
dconnelly
2015/02/20 15:36:06
Done.
|
| + [nameLabel_ setFrame:nameFrame]; |
| + |
| + const CGFloat usernameX = |
| + base::i18n::IsRTL() |
| + ? NSMinX([avatarView_ frame]) - |
| + kHorizontalPaddingBetweenAvatarAndLabels - |
| + CGRectGetWidth([usernameLabel_ frame]) |
|
groby-ooo-7-16
2015/02/19 17:01:33
NSWidth - you *do* love CG functions, don't you? ;
dconnelly
2015/02/20 15:36:06
Sorry, yeah, the iOS idioms are just stickier in m
|
| + : NSMaxX([avatarView_ frame]) + |
| + kHorizontalPaddingBetweenAvatarAndLabels; |
| + const CGFloat usernameLabelY = |
| + NSMaxY(nameFrame) + kVerticalPaddingBetweenLabels; |
| + CGRect usernameFrame = [usernameLabel_ frame]; |
| + usernameFrame.origin = CGPointMake(usernameX, usernameLabelY); |
| + [usernameLabel_ setFrame:usernameFrame]; |
| + |
| + // Use a default avatar and fetch the custom one, if it exists. |
| + [self updateAvatar:[[self class] defaultAvatar]]; |
| + if (passwordForm_.avatar_url.is_valid()) |
| + [delegate_ fetchAvatar:passwordForm_.avatar_url forView:self]; |
| + |
| + // When resizing, stick to the left (resp. right for RTL) edge. |
| + const NSUInteger autoresizingMask = |
| + (base::i18n::IsRTL() ? NSViewMinXMargin : NSViewMaxXMargin); |
| + [avatarView_ setAutoresizingMask:autoresizingMask]; |
| + [usernameLabel_ setAutoresizingMask:autoresizingMask]; |
| + [nameLabel_ setAutoresizingMask:autoresizingMask]; |
| + [self setAutoresizingMask:NSViewWidthSizable]; |
| + } |
| + |
| + return self; |
| +} |
| + |
| +- (void)updateAvatar:(NSImage*)avatar { |
| + [avatarView_ setImage:avatar]; |
| +} |
| + |
| ++ (NSImage*)defaultAvatar { |
| + return gfx::NSImageFromImageSkia(ScaleImageForAccountAvatar( |
| + *ResourceBundle::GetSharedInstance() |
| + .GetImageNamed(IDR_PROFILE_AVATAR_PLACEHOLDER_LARGE) |
| + .ToImageSkia())); |
| +} |
| + |
| ++ (Class)cellClass { |
| + return [CredentialItemViewCell class]; |
| +} |
| + |
| +- (void)click:(id)sender { |
| + DCHECK_EQ(sender, self); |
| + [delegate_ selectPasswordForm:passwordForm_ credentialType:credentialType_]; |
| +} |
| + |
| +- (NSTextField*)nameLabel { |
| + return nameLabel_.get(); |
| +} |
| + |
| +- (NSTextField*)usernameLabel { |
| + return usernameLabel_.get(); |
| +} |
| + |
| +- (NSImageView*)avatarView { |
| + return avatarView_.get(); |
| +} |
| + |
| +@end |
| + |
| +@interface CredentialItemListView() |
| +- (void)addCredentialItem:(CredentialItemView*)item; |
| +@end |
| + |
| +@implementation CredentialItemListView |
|
groby-ooo-7-16
2015/02/19 17:01:33
Question: Would this be better off as an NSTableVi
dconnelly
2015/02/20 15:36:06
You know, I've never written an NSTableView before
|
| + |
| +- (id)initWithItems:(NSArray*)items { |
| + if ((self = [super initWithFrame:NSZeroRect])) { |
| + for (CredentialItemView* item in items) |
| + [self addCredentialItem:item]; |
| + } |
| + return self; |
| +} |
| + |
| +- (void)addCredentialItem:(CredentialItemView*)item { |
| + [self addSubview:item]; |
| + |
| + // Stack the credentials on top of each other. |
|
groby-ooo-7-16
2015/02/19 17:01:33
Shorter (and hopefully clearer) version:
NSRect v
dconnelly
2015/02/20 15:36:06
TableView - removed all this
|
| + const CGFloat y = |
| + CGRectGetHeight([self frame]) + |
| + ([[self subviews] count] > 1 ? kVerticalPaddingBetweenCredentials : 0.0f); |
| + |
| + // Update the size of the list view to accomodate the new item. |
| + const CGFloat width = |
| + std::max(CGRectGetWidth([self frame]), CGRectGetWidth([item frame])); |
| + const CGFloat height = y + CGRectGetHeight([item frame]); |
| + [self setFrameSize:CGSizeMake(width, height)]; |
| + |
| + const CGFloat x = 0; |
|
groby-ooo-7-16
2015/02/19 17:01:33
You could replace the entire code block with
[item
dconnelly
2015/02/20 15:36:06
TableView - removed all this
|
| + NSRect itemFrame = [item frame]; |
| + itemFrame.size.width = width; |
| + itemFrame.origin = CGPointMake(x, y); |
| + [item setFrame:itemFrame]; |
| +} |
| + |
| +- (NSArray*)items { |
| + return [self subviews]; |
| +} |
| + |
| +@end |