// Copyright (c) 2011 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. #ifndef CHROME_BROWSER_AUTOCOMPLETE_AUTOCOMPLETE_POPUP_VIEW_MAC_H_ #define CHROME_BROWSER_AUTOCOMPLETE_AUTOCOMPLETE_POPUP_VIEW_MAC_H_ #pragma once #import <Cocoa/Cocoa.h> #include <string> #include "base/basictypes.h" #include "base/memory/scoped_nsobject.h" #include "base/memory/scoped_ptr.h" #include "chrome/browser/autocomplete/autocomplete.h" #include "chrome/browser/autocomplete/autocomplete_match.h" #include "chrome/browser/autocomplete/autocomplete_popup_view.h" #import "chrome/browser/ui/cocoa/location_bar/instant_opt_in_controller.h" #include "ui/gfx/font.h" #include "webkit/glue/window_open_disposition.h" class AutocompleteEditModel; class AutocompleteEditViewMac; @class AutocompleteMatrix; class AutocompletePopupModel; @class InstantOptInController; @class NSImage; class Profile; // Implements AutocompletePopupView using a raw NSWindow containing an // NSTableView. // // TODO(rohitrao): This class is set up in a way that makes testing hard. // Refactor and write unittests. http://crbug.com/9977 class AutocompletePopupViewMac : public AutocompletePopupView, public InstantOptInControllerDelegate { public: AutocompletePopupViewMac(AutocompleteEditViewMac* edit_view, AutocompleteEditModel* edit_model, Profile* profile, NSTextField* field); virtual ~AutocompletePopupViewMac(); // Implement the InstantOptInControllerDelegate interface. virtual void UserPressedOptIn(bool opt_in); // Implement the AutocompletePopupView interface. virtual bool IsOpen() const; virtual void InvalidateLine(size_t line) { // TODO(shess): Verify that there is no need to implement this. // This is currently used in two places in the model: // // When setting the selected line, the selected line is // invalidated, then the selected line is changed, then the new // selected line is invalidated, then PaintUpdatesNow() is called. // For us PaintUpdatesNow() should be sufficient. // // Same thing happens when changing the hovered line, except with // no call to PaintUpdatesNow(). Since this code does not // currently support special display of the hovered line, there's // nothing to do here. // // deanm indicates that this is an anti-flicker optimization, // which we probably cannot utilize (and may not need) so long as // we're using NSTableView to implement the popup contents. We // may need to move away from NSTableView to implement hover, // though. } virtual void UpdatePopupAppearance(); virtual gfx::Rect GetTargetBounds(); // Set |line| to be selected. void SetSelectedLine(size_t line); // This is only called by model in SetSelectedLine() after updating // everything. Popup should already be visible. virtual void PaintUpdatesNow(); virtual void OnDragCanceled() {} // Opens the URL corresponding to the given |row|. If |force_background| is // true, forces the URL to open in a background tab. Otherwise, determines // the proper window open disposition from the modifier flags on |[NSApp // currentEvent]|. void OpenURLForRow(int row, bool force_background); // Return the text to show for the match, based on the match's // contents and description. Result will be in |font|, with the // boldfaced version used for matches. static NSAttributedString* MatchText(const AutocompleteMatch& match, gfx::Font& font, float cellWidth); // Helper for MatchText() to allow sharing code between the contents // and description cases. Returns NSMutableAttributedString as a // convenience for MatchText(). static NSMutableAttributedString* DecorateMatchedString( const string16 &matchString, const AutocompleteMatch::ACMatchClassifications &classifications, NSColor* textColor, NSColor* dimTextColor, gfx::Font& font); // Helper for MatchText() to elide a marked-up string using // gfx::ElideText() as a model. Modifies |aString| in place. // TODO(shess): Consider breaking AutocompleteButtonCell out of this // code, and modifying it to have something like -setMatch:, so that // these convolutions to expose internals for testing can be // cleaner. static NSMutableAttributedString* ElideString( NSMutableAttributedString* aString, const string16 originalString, const gfx::Font& font, const float cellWidth); private: // Returns the AutocompleteMatrix for this popup view. AutocompleteMatrix* GetAutocompleteMatrix(); // Create the popup_ instance if needed. void CreatePopupIfNeeded(); // Calculate the appropriate position for the popup based on the // field's screen position and the given target for the matrix // height, and makes the popup visible. Animates to the new frame // if the popup shrinks, snaps to the new frame if the popup grows, // allows existing animations to continue if the size doesn't // change. void PositionPopup(const CGFloat matrixHeight); // Returns the NSImage that should be used as an icon for the given match. NSImage* ImageForMatch(const AutocompleteMatch& match); // Returns whether or not to show the instant opt-in prompt. bool ShouldShowInstantOptIn(); scoped_ptr<AutocompletePopupModel> model_; AutocompleteEditViewMac* edit_view_; NSTextField* field_; // owned by tab controller // Child window containing a matrix which implements the popup. scoped_nsobject<NSWindow> popup_; scoped_nsobject<InstantOptInController> opt_in_controller_; NSRect targetPopupFrame_; DISALLOW_COPY_AND_ASSIGN(AutocompletePopupViewMac); }; #endif // CHROME_BROWSER_AUTOCOMPLETE_AUTOCOMPLETE_POPUP_VIEW_MAC_H_