 Chromium Code Reviews
 Chromium Code Reviews Issue 1099403005:
  [AiS] changing mac omnibox suggestions form NSMatrix to NSTableView  (Closed) 
  Base URL: https://chromium.googlesource.com/chromium/src.git@master
    
  
    Issue 1099403005:
  [AiS] changing mac omnibox suggestions form NSMatrix to NSTableView  (Closed) 
  Base URL: https://chromium.googlesource.com/chromium/src.git@master| Index: chrome/browser/ui/cocoa/omnibox/omnibox_popup_matrix.mm | 
| diff --git a/chrome/browser/ui/cocoa/omnibox/omnibox_popup_matrix.mm b/chrome/browser/ui/cocoa/omnibox/omnibox_popup_matrix.mm | 
| index 3548b9f9881a2f951b9912894c79b92dfc3d4e94..3b0dcc51d2569ebeaab27cbae5b6634899547c96 100644 | 
| --- a/chrome/browser/ui/cocoa/omnibox/omnibox_popup_matrix.mm | 
| +++ b/chrome/browser/ui/cocoa/omnibox/omnibox_popup_matrix.mm | 
| @@ -5,7 +5,10 @@ | 
| #import "chrome/browser/ui/cocoa/omnibox/omnibox_popup_matrix.h" | 
| #include "base/logging.h" | 
| +#include "base/mac/foundation_util.h" | 
| #import "chrome/browser/ui/cocoa/omnibox/omnibox_popup_cell.h" | 
| +#include "chrome/browser/ui/cocoa/omnibox/omnibox_popup_view_mac.h" | 
| +#include "components/omnibox/autocomplete_result.h" | 
| namespace { | 
| @@ -14,48 +17,136 @@ const NSInteger kMiddleButtonNumber = 2; | 
| } // namespace | 
| -@interface OmniboxPopupMatrix() | 
| +@implementation OmniboxPopupTableController | 
| + | 
| +- (instancetype)init { | 
| + if ((self = [super init])) { | 
| + hoveredIndex_ = -1; | 
| + } | 
| + return self; | 
| +} | 
| + | 
| +- (instancetype)initWithMatchResults:(const AutocompleteResult&)result | 
| + popupView:(const OmniboxPopupViewMac&)popupView | 
| + answerImage:(NSImage*)answerImage { | 
| + if ((self = [super init])) { | 
| + hoveredIndex_ = -1; | 
| + array_.reset([[NSMutableArray alloc] init]); | 
| + CGFloat max_match_contents_width = 0.0f; | 
| + CGFloat contents_offset = -1.0f; | 
| + for (const AutocompleteMatch& match : result) { | 
| + base::scoped_nsobject<OmniboxPopupCellData> cellData( | 
| + [[OmniboxPopupCellData alloc] | 
| + initWithMatch:match | 
| + image:popupView.ImageForMatch(match)]); | 
| + if (match.answer && answerImage) | 
| 
groby-ooo-7-16
2015/06/11 01:22:18
Again, this modifies something that's technically
 
dschuyler
2015/06/11 22:34:22
Done.
 | 
| + [cellData setAnswerImage:answerImage]; | 
| + [array_ addObject:cellData]; | 
| + if (match.type == AutocompleteMatchType::SEARCH_SUGGEST_TAIL) { | 
| + max_match_contents_width = std::max(max_match_contents_width, | 
| + [cellData getMatchContentsWidth]); | 
| + if (contents_offset < 0.0f) | 
| + contents_offset = [OmniboxPopupCell computeContentsOffset:match]; | 
| + [cellData setContentsOffset:contents_offset]; | 
| + } | 
| + } | 
| + | 
| + for (const OmniboxPopupCellData* cellData : array_.get()) { | 
| + [cellData setMaxMatchContentsWidth:max_match_contents_width]; | 
| 
groby-ooo-7-16
2015/06/11 01:22:18
Grumblemumble immutable grumblemumble.
(We can ge
 
dschuyler
2015/06/11 22:34:23
Done.
 | 
| + } | 
| + } | 
| + return self; | 
| +} | 
| + | 
| +- (instancetype)initWithArray:(NSMutableArray*)array { | 
| + if ((self = [super init])) { | 
| + hoveredIndex_ = -1; | 
| + array_.reset([array retain]); | 
| 
groby-ooo-7-16
2015/06/11 01:22:17
Usually, if you init with a mutable array, you cop
 
Scott Hess - ex-Googler
2015/06/11 21:52:02
I think -copy returns a reference, so you just do:
 
dschuyler
2015/06/11 22:34:22
I'm having trouble following this, is this a misty
 
dschuyler
2015/06/11 22:34:23
Done.
 
Scott Hess - ex-Googler
2015/06/11 22:36:19
Yeah, I think there was a missing n't or im in the
 
dschuyler
2015/06/12 00:43:26
I think we still want the data member to be a muta
 
Scott Hess - ex-Googler
2015/06/12 20:21:32
I think maybe you're wavering back and forth based
 
dschuyler
2015/06/12 21:44:30
Thanks.  I'm hoping I'm closer now.  The array is
 
groby-ooo-7-16
2015/06/12 22:24:30
Yes, please. I now my future self, and she'll make
 | 
| + } | 
| + return self; | 
| +} | 
| + | 
| +- (NSInteger)numberOfRowsInTableView:(NSTableView*)tableView { | 
| + return [array_ count]; | 
| +} | 
| + | 
| +- (id)tableView:(NSTableView*)tableView | 
| + objectValueForTableColumn:(NSTableColumn*)tableColumn | 
| + row:(NSInteger)rowIndex { | 
| + return nil; | 
| 
groby-ooo-7-16
2015/06/11 01:22:18
As discussed in earlier CLs: This should return th
 
dschuyler
2015/06/11 22:34:23
Done.
 | 
| +} | 
| + | 
| +- (void)tableView:(NSTableView*)aTableView | 
| + setObjectValue:(id)anObject | 
| + forTableColumn:(NSTableColumn*)aTableColumn | 
| + row:(NSInteger)rowIndex { | 
| +} | 
| 
groby-ooo-7-16
2015/06/11 01:22:17
NOTREACHED, please, since we'll never modify value
 
dschuyler
2015/06/11 22:34:22
Done.
 | 
| + | 
| +- (void)tableView:(NSTableView*)tableView | 
| + willDisplayCell:(id)cell | 
| + forTableColumn:(NSTableColumn*)tableColumn | 
| + row:(NSInteger)rowIndex { | 
| + OmniboxPopupCell* popupCell = | 
| + base::mac::ObjCCastStrict<OmniboxPopupCell>(cell); | 
| + [popupCell setObjectValue:[array_ objectAtIndex:rowIndex]]; | 
| 
groby-ooo-7-16
2015/06/11 01:22:17
[popupCell setObjectValue:[array_ objectAtIndex:ro
 
dschuyler
2015/06/11 22:34:22
Done.
 | 
| + [popupCell | 
| + setState:([tableView selectedRow] == rowIndex) ? NSOnState : NSOffState]; | 
| + [popupCell highlight:(hoveredIndex_ == rowIndex) | 
| + withFrame:[tableView bounds] | 
| + inView:tableView]; | 
| +} | 
| + | 
| +- (NSInteger)highlightedRow { | 
| + return hoveredIndex_; | 
| +} | 
| + | 
| +- (void)highlightRowAt:(NSInteger)rowIndex { | 
| 
groby-ooo-7-16
2015/06/11 01:22:18
- (void)setHighlightedRow:(NSInteger)rowIndex
set
 
dschuyler
2015/06/11 22:34:23
Done.
 | 
| + hoveredIndex_ = rowIndex; | 
| +} | 
| + | 
| +- (CGFloat)tableView:(NSTableView*)tableView heightOfRow:(NSInteger)row { | 
| + return [[array_ objectAtIndex:row] rowHeight]; | 
| +} | 
| + | 
| +@end | 
| + | 
| +@interface OmniboxPopupMatrix () | 
| - (void)resetTrackingArea; | 
| -- (void)highlightRowAt:(NSInteger)rowIndex; | 
| - (void)highlightRowUnder:(NSEvent*)theEvent; | 
| - (BOOL)selectCellForEvent:(NSEvent*)theEvent; | 
| @end | 
| @implementation OmniboxPopupMatrix | 
| -- (id)initWithObserver:(OmniboxPopupMatrixObserver*)observer { | 
| +- (instancetype)initWithObserver:(OmniboxPopupMatrixObserver*)observer { | 
| if ((self = [super initWithFrame:NSZeroRect])) { | 
| observer_ = observer; | 
| - [self setCellClass:[OmniboxPopupCell class]]; | 
| + | 
| + base::scoped_nsobject<NSTableColumn> column( | 
| + [[NSTableColumn alloc] initWithIdentifier:@"MainCell"]); | 
| 
groby-ooo-7-16
2015/06/11 01:22:17
Well, it'd be the "MainColumn" if that's the ID we
 
dschuyler
2015/06/11 22:34:22
Done.
 | 
| + [column setDataCell:[[OmniboxPopupCell alloc] init]]; | 
| + [self addTableColumn:column]; | 
| // Cells pack with no spacing. | 
| [self setIntercellSpacing:NSMakeSize(0.0, 0.0)]; | 
| - [self setDrawsBackground:YES]; | 
| + [self setSelectionHighlightStyle:NSTableViewSelectionHighlightStyleNone]; | 
| [self setBackgroundColor:[NSColor controlBackgroundColor]]; | 
| - [self renewRows:0 columns:1]; | 
| [self setAllowsEmptySelection:YES]; | 
| - [self setMode:NSRadioModeMatrix]; | 
| - [self deselectAllCells]; | 
| + [self deselectAll:self]; | 
| [self resetTrackingArea]; | 
| } | 
| return self; | 
| } | 
| -- (void)setObserver:(OmniboxPopupMatrixObserver*)observer { | 
| - observer_ = observer; | 
| +- (OmniboxPopupTableController*)controller { | 
| + return base::mac::ObjCCastStrict<OmniboxPopupTableController>( | 
| + [self delegate]); | 
| } | 
| -- (NSInteger)highlightedRow { | 
| - NSArray* cells = [self cells]; | 
| - const NSUInteger count = [cells count]; | 
| - for(NSUInteger i = 0; i < count; ++i) { | 
| - if ([[cells objectAtIndex:i] isHighlighted]) { | 
| - return i; | 
| - } | 
| - } | 
| - return -1; | 
| +- (void)setObserver:(OmniboxPopupMatrixObserver*)observer { | 
| + observer_ = observer; | 
| } | 
| - (void)updateTrackingAreas { | 
| @@ -73,7 +164,7 @@ const NSInteger kMiddleButtonNumber = 2; | 
| } | 
| - (void)mouseExited:(NSEvent*)theEvent { | 
| - [self highlightRowAt:-1]; | 
| + [[self controller] highlightRowAt:-1]; | 
| } | 
| // The tracking area events aren't forwarded during a drag, so handle | 
| @@ -103,7 +194,7 @@ const NSInteger kMiddleButtonNumber = 2; | 
| // make sure the user is getting the right feedback. | 
| [self highlightRowUnder:theEvent]; | 
| - const NSInteger highlightedRow = [self highlightedRow]; | 
| + const NSInteger highlightedRow = [[self controller] highlightedRow]; | 
| if (highlightedRow != -1) { | 
| DCHECK(observer_); | 
| observer_->OnMatrixRowMiddleClicked(self, highlightedRow); | 
| @@ -117,7 +208,7 @@ const NSInteger kMiddleButtonNumber = 2; | 
| NSCell* selectedCell = [self selectedCell]; | 
| // Clear any existing highlight. | 
| - [self highlightRowAt:-1]; | 
| + [[self controller] highlightRowAt:-1]; | 
| do { | 
| if (![self selectCellForEvent:theEvent]) { | 
| @@ -145,6 +236,22 @@ const NSInteger kMiddleButtonNumber = 2; | 
| } | 
| } | 
| +- (void)selectRowIndex:(NSInteger)rowIndex { | 
| + NSIndexSet* indexSet = [NSIndexSet indexSetWithIndex:rowIndex]; | 
| + [self selectRowIndexes:indexSet byExtendingSelection:NO]; | 
| +} | 
| + | 
| +- (NSInteger)highlightedRow { | 
| + return [[self controller] highlightedRow]; | 
| +} | 
| + | 
| +- (void)setController:(OmniboxPopupTableController*)controller { | 
| + matrixController_.reset([controller retain]); | 
| + [self setDelegate:controller]; | 
| + [self setDataSource:controller]; | 
| + [self reloadData]; | 
| +} | 
| + | 
| - (void)resetTrackingArea { | 
| if (trackingArea_.get()) | 
| [self removeTrackingArea:trackingArea_.get()]; | 
| @@ -160,33 +267,23 @@ const NSInteger kMiddleButtonNumber = 2; | 
| [self addTrackingArea:trackingArea_.get()]; | 
| } | 
| -- (void)highlightRowAt:(NSInteger)rowIndex { | 
| - // highlightCell will be nil if rowIndex is out of range, so no cell will be | 
| - // highlighted. | 
| - NSCell* highlightCell = [self cellAtRow:rowIndex column:0]; | 
| - | 
| - for (NSCell* cell in [self cells]) { | 
| - [cell setHighlighted:(cell == highlightCell)]; | 
| - } | 
| -} | 
| - | 
| - (void)highlightRowUnder:(NSEvent*)theEvent { | 
| NSPoint point = [self convertPoint:[theEvent locationInWindow] fromView:nil]; | 
| - NSInteger row, column; | 
| - if ([self getRow:&row column:&column forPoint:point]) { | 
| - [self highlightRowAt:row]; | 
| - } else { | 
| - [self highlightRowAt:-1]; | 
| + NSInteger oldRow = [[self controller] highlightedRow]; | 
| + NSInteger newRow = [self rowAtPoint:point]; | 
| + if (oldRow != newRow) { | 
| + [[self controller] highlightRowAt:newRow]; | 
| + [self setNeedsDisplayInRect:[self rectOfRow:oldRow]]; | 
| + [self setNeedsDisplayInRect:[self rectOfRow:newRow]]; | 
| } | 
| } | 
| -// Select cell under |theEvent|, returning YES if a selection is made. | 
| - (BOOL)selectCellForEvent:(NSEvent*)theEvent { | 
| NSPoint point = [self convertPoint:[theEvent locationInWindow] fromView:nil]; | 
| - NSInteger row, column; | 
| - if ([self getRow:&row column:&column forPoint:point]) { | 
| - DCHECK_EQ(column, 0); | 
| + NSInteger row = [self rowAtPoint:point]; | 
| + [self selectRowIndex:row]; | 
| + if (row != -1) { | 
| DCHECK(observer_); | 
| observer_->OnMatrixRowSelected(self, row); | 
| return YES; |