Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(61)

Unified Diff: ios/chrome/browser/payments/payment_request_selector_view_controller.mm

Issue 2805273002: [Payment Request] Selector view controller (Closed)
Patch Set: Created 3 years, 8 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: ios/chrome/browser/payments/payment_request_selector_view_controller.mm
diff --git a/ios/chrome/browser/payments/payment_request_selector_view_controller.mm b/ios/chrome/browser/payments/payment_request_selector_view_controller.mm
new file mode 100644
index 0000000000000000000000000000000000000000..912d5a523a89fb2a8be3ecf390b516242f6028e6
--- /dev/null
+++ b/ios/chrome/browser/payments/payment_request_selector_view_controller.mm
@@ -0,0 +1,242 @@
+// Copyright 2017 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 "ios/chrome/browser/payments/payment_request_selector_view_controller.h"
+
+#include "base/mac/foundation_util.h"
+#include "components/strings/grit/components_strings.h"
+#import "ios/chrome/browser/payments/cells/payments_text_item.h"
+#import "ios/chrome/browser/payments/payment_request_selector_view_controller_actions.h"
+#import "ios/chrome/browser/payments/payment_request_selector_view_controller_data_source.h"
+#import "ios/chrome/browser/ui/autofill/cells/status_item.h"
+#import "ios/chrome/browser/ui/collection_view/cells/MDCCollectionViewCell+Chrome.h"
+#import "ios/chrome/browser/ui/collection_view/cells/collection_view_item.h"
+#import "ios/chrome/browser/ui/collection_view/cells/collection_view_text_item.h"
+#import "ios/chrome/browser/ui/collection_view/collection_view_model.h"
+#import "ios/chrome/browser/ui/colors/MDCPalette+CrAdditions.h"
+#import "ios/chrome/browser/ui/icons/chrome_icon.h"
+#include "ios/chrome/browser/ui/uikit_ui_util.h"
+#include "ios/chrome/grit/ios_strings.h"
+#include "ios/chrome/grit/ios_theme_resources.h"
+#include "ui/base/l10n/l10n_util.h"
+
+#if !defined(__has_feature) || !__has_feature(objc_arc)
+#error "This file requires ARC support."
+#endif
+
+namespace {
+
+NSString* const kPaymentRequestSelectorCollectionViewAccessibilityID =
+ @"kPaymentRequestSelectorCollectionViewAccessibilityID";
+
+const CGFloat kSeparatorEdgeInset = 14;
+
+typedef NS_ENUM(NSInteger, SectionIdentifier) {
+ SectionIdentifierItems = kSectionIdentifierEnumZero,
+};
+
+typedef NS_ENUM(NSInteger, ItemType) {
+ ItemTypeHeader = kItemTypeHeaderItem, // This is a repeated item type.
gambard 2017/04/10 09:47:35 Is this item type the same as the one in the data
Moe 2017/04/10 17:53:49 Correct. Nothing should be added between the heade
+ ItemTypeSelectableItem,
+ ItemTypeSpinner,
+ ItemTypeAddItem,
+};
+
+} // namespace
+
+@interface PaymentRequestSelectorViewController ()<
+ PaymentRequestSelectorViewControllerActions>
+
+@end
+
+@implementation PaymentRequestSelectorViewController
+
+@synthesize delegate = _delegate;
+@synthesize dataSource = _dataSource;
+
+- (instancetype)init {
+ if ((self = [super initWithStyle:CollectionViewControllerStyleAppBar])) {
+ // Set up leading (return) button.
+ UIBarButtonItem* returnButton =
+ [ChromeIcon templateBarButtonItemWithImage:[ChromeIcon backIcon]
+ target:nil
+ action:@selector(onReturn)];
+ returnButton.accessibilityLabel = l10n_util::GetNSString(IDS_ACCNAME_BACK);
+ self.navigationItem.leftBarButtonItem = returnButton;
gambard 2017/04/10 09:47:34 I think the coordinator should set the navigation
Moe 2017/04/10 17:53:49 All the view controllers in ios/chrome/browser/pay
gambard 2017/04/18 08:06:37 Acknowledged.
+ }
+ return self;
+}
+
gambard 2017/04/10 09:47:34 Add #pragma mark - PaymentRequestSelectorViewContr
Moe 2017/04/10 17:53:49 Done.
+- (void)onReturn {
+ [_delegate paymentRequestSelectorViewControllerDidReturn:self];
gambard 2017/04/10 09:47:35 s/_delegate/self.delegate in the whole file (excep
Moe 2017/04/10 17:53:49 Done.
+}
+
+#pragma mark - CollectionViewController methods
+
+- (void)loadModel {
+ [super loadModel];
+ CollectionViewModel* model = self.collectionViewModel;
+
+ [model addSectionWithIdentifier:SectionIdentifierItems];
+
+ // If the view controller is in the pending state, only display a spinner and
+ // a message indicating the pending state.
+ if (_dataSource.state == PaymentRequestSelectorViewControllerStatePending) {
gambard 2017/04/10 09:47:35 s/_dataSource/self.dataSource in the whole file (e
Moe 2017/04/10 17:53:49 Done.
+ StatusItem* statusItem = [[StatusItem alloc] initWithType:ItemTypeSpinner];
+ statusItem.text = l10n_util::GetNSString(IDS_PAYMENTS_CHECKING_OPTION);
gambard 2017/04/10 09:47:35 Don't you need to set the status?
Moe 2017/04/10 17:53:49 Done.
+ [model addItem:statusItem toSectionWithIdentifier:SectionIdentifierItems];
+ return;
+ }
+
+ CollectionViewItem* headerItem = [_dataSource headerItem];
gambard 2017/04/10 09:47:35 I think it is better to pass the value to be displ
Moe 2017/04/10 17:53:49 I discussed this with lpromero@ previously and he
gambard 2017/04/18 08:06:37 I am writing the ContentSuggestions classes. My me
+ if (headerItem) {
+ [model addItem:headerItem toSectionWithIdentifier:SectionIdentifierItems];
gambard 2017/04/10 09:47:35 Use |setHeader:forSectionWithIdentifier:|?
Moe 2017/04/10 17:53:49 Unfortunately we don't have support for arbitrary
gambard 2017/04/18 08:06:37 Acknowledged.
+ }
+
+ NSArray<CollectionViewItem*>* selectableItems = [_dataSource selectableItems];
+ for (NSUInteger index = 0; index < selectableItems.count; ++index) {
+ CollectionViewItem* item = [selectableItems objectAtIndex:index];
gambard 2017/04/10 09:47:34 Nit: you can use: selectableItems[index]
Moe 2017/04/10 17:53:49 Done.
+ DCHECK(item);
gambard 2017/04/10 09:47:34 You cannot have nil items in an array. You should
Moe 2017/04/10 17:53:49 Done.
+ item.accessibilityTraits |= UIAccessibilityTraitButton;
+ item.accessoryType = (index == _dataSource.selectedItemIndex)
+ ? MDCCollectionViewCellAccessoryCheckmark
+ : MDCCollectionViewCellAccessoryNone;
+ [model addItem:item toSectionWithIdentifier:SectionIdentifierItems];
+ }
+
+ if ([[_dataSource addButtonTitle] length]) {
gambard 2017/04/10 09:47:35 Why are you returning items for all the others met
Moe 2017/04/10 17:53:49 Because the add button, if any, will be the same k
+ PaymentsTextItem* addButton =
+ [[PaymentsTextItem alloc] initWithType:ItemTypeAddItem];
+ addButton.text = [_dataSource addButtonTitle];
+ addButton.image = NativeImage(IDR_IOS_PAYMENTS_ADD);
+ addButton.accessibilityTraits |= UIAccessibilityTraitButton;
+ [model addItem:addButton toSectionWithIdentifier:SectionIdentifierItems];
+ }
+}
+
+- (void)viewDidLoad {
+ [super viewDidLoad];
+ self.collectionView.accessibilityIdentifier =
+ kPaymentRequestSelectorCollectionViewAccessibilityID;
+
+ // Customize collection view settings.
+ self.styler.cellStyle = MDCCollectionViewCellStyleCard;
+ self.styler.separatorInset =
+ UIEdgeInsetsMake(0, kSeparatorEdgeInset, 0, kSeparatorEdgeInset);
+}
+
+#pragma mark UICollectionViewDataSource
+
+- (UICollectionViewCell*)collectionView:(UICollectionView*)collectionView
gambard 2017/04/10 09:47:34 I don't think this is how the items should be used
Moe 2017/04/10 17:53:49 I agree. Done.
+ cellForItemAtIndexPath:(nonnull NSIndexPath*)indexPath {
+ CollectionViewModel* model = self.collectionViewModel;
+
+ UICollectionViewCell* cell =
+ [super collectionView:collectionView cellForItemAtIndexPath:indexPath];
+
+ CollectionViewItem* item = [model itemAtIndexPath:indexPath];
+ switch (item.type) {
+ case ItemTypeSelectableItem: {
lpromero 2017/04/11 15:59:06 Move this right above the default case to bundle t
+ break;
+ }
+ case ItemTypeHeader: {
+ if ([cell isKindOfClass:[PaymentsTextCell class]]) {
gambard 2017/04/10 09:47:35 I think this is a discrepancy between the idea you
Moe 2017/04/10 17:53:49 I see your point of view. Again this was another p
lpromero 2017/04/11 15:59:06 It's the idea of an agnostic view controller that
Moe 2017/04/14 06:05:49 Brought this back as we decided the mediators shou
+ PaymentsTextCell* headerCell =
+ base::mac::ObjCCastStrict<PaymentsTextCell>(cell);
+ headerCell.textLabel.textColor =
+ _dataSource.state == PaymentRequestSelectorViewControllerStateError
gambard 2017/04/10 09:47:35 Does the data source state is supposed to change o
Moe 2017/04/10 17:53:49 because the state changes always involve going to/
gambard 2017/04/18 08:06:37 Acknowledged.
+ ? [[MDCPalette cr_redPalette] tint600]
+ : [[MDCPalette greyPalette] tint600];
+ }
+ break;
+ }
+ case ItemTypeAddItem: {
+ PaymentsTextCell* addItemCell =
+ base::mac::ObjCCastStrict<PaymentsTextCell>(cell);
+ addItemCell.textLabel.textColor = [[MDCPalette greyPalette] tint900];
+ break;
+ }
+ default:
+ break;
+ }
+ return cell;
+}
+
+#pragma mark UICollectionViewDelegate
+
+- (void)collectionView:(UICollectionView*)collectionView
+ didSelectItemAtIndexPath:(NSIndexPath*)indexPath {
+ [super collectionView:collectionView didSelectItemAtIndexPath:indexPath];
+
+ CollectionViewModel* model = self.collectionViewModel;
+
+ CollectionViewItem* item = [model itemAtIndexPath:indexPath];
+ switch (item.type) {
+ case ItemTypeSelectableItem: {
+ // Update the currently selected cell, if any.
+ if (_dataSource.selectedItemIndex != NSUIntegerMax) {
+ CollectionViewItem* selectedItem =
+ [_dataSource selectableItemAtIndex:_dataSource.selectedItemIndex];
+ DCHECK(selectedItem);
gambard 2017/04/10 09:47:35 Same, I don't think you need this DCHECK.
Moe 2017/04/10 17:53:49 Done.
+ selectedItem.accessoryType = MDCCollectionViewCellAccessoryNone;
+ [self reconfigureCellsForItems:@[ selectedItem ]
+ inSectionWithIdentifier:SectionIdentifierItems];
+ }
+
+ // Update the data source with the selection.
+ NSUInteger index =
+ [self.collectionViewModel indexInItemTypeForIndexPath:indexPath];
+ DCHECK(index < [[_dataSource selectableItems] count]);
+ [_dataSource setSelectedItemIndex:index];
gambard 2017/04/10 09:47:34 I don't think the ViewController should set things
Moe 2017/04/10 17:53:49 Good point. Done.
+
+ // Update the newly selected cell.
+ item.accessoryType = MDCCollectionViewCellAccessoryCheckmark;
+ [self reconfigureCellsForItems:@[ item ]
+ inSectionWithIdentifier:SectionIdentifierItems];
+
+ // Notify the delegate of the selection.
+ [_delegate paymentRequestSelectorViewController:self
+ didSelectItemAtIndex:index];
+ break;
+ }
+ case ItemTypeAddItem: {
+ [_delegate paymentRequestSelectorViewControllerDidSelectAddItem:self];
+ break;
+ }
+ default:
+ break;
+ }
+}
+
+#pragma mark MDCCollectionViewStylingDelegate
+
+- (CGFloat)collectionView:(UICollectionView*)collectionView
+ cellHeightAtIndexPath:(NSIndexPath*)indexPath {
+ CollectionViewItem* item =
+ [self.collectionViewModel itemAtIndexPath:indexPath];
+ switch (item.type) {
gambard 2017/04/10 09:47:35 Nit: I would remove the the switch. Is there a par
Moe 2017/04/10 17:53:48 Done.
+ case ItemTypeSpinner:
+ case ItemTypeHeader:
+ case ItemTypeSelectableItem:
+ case ItemTypeAddItem:
+ return [MDCCollectionViewCell
+ cr_preferredHeightForWidth:CGRectGetWidth(collectionView.bounds)
gambard 2017/04/10 09:47:35 I don't know if this is relevant in your case, but
Moe 2017/04/10 17:53:49 Definitely relevant. Thank you.
+ forItem:item];
+ default:
+ NOTREACHED();
+ return MDCCellDefaultOneLineHeight;
+ }
+}
+
+- (BOOL)collectionView:(UICollectionView*)collectionView
+ hidesInkViewAtIndexPath:(NSIndexPath*)indexPath {
+ NSInteger type = [self.collectionViewModel itemTypeForIndexPath:indexPath];
+ if (type == ItemTypeHeader) {
gambard 2017/04/10 09:47:34 I would have say the ink does not appear on header
Moe 2017/04/10 17:53:49 please see above.
+ return YES;
+ } else {
+ return NO;
+ }
+}
+
+@end

Powered by Google App Engine
This is Rietveld 408576698