• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2011 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "chrome/browser/autocomplete/autocomplete_edit_view_win.h"
6 
7 #include <algorithm>
8 #include <locale>
9 #include <string>
10 
11 #include <richedit.h>
12 #include <textserv.h>
13 
14 #include "app/win/iat_patch_function.h"
15 #include "base/auto_reset.h"
16 #include "base/basictypes.h"
17 #include "base/i18n/rtl.h"
18 #include "base/lazy_instance.h"
19 #include "base/memory/ref_counted.h"
20 #include "base/string_util.h"
21 #include "base/utf_string_conversions.h"
22 #include "chrome/app/chrome_command_ids.h"
23 #include "chrome/browser/autocomplete/autocomplete_accessibility.h"
24 #include "chrome/browser/autocomplete/autocomplete_match.h"
25 #include "chrome/browser/autocomplete/autocomplete_popup_model.h"
26 #include "chrome/browser/autocomplete/keyword_provider.h"
27 #include "chrome/browser/browser_process.h"
28 #include "chrome/browser/command_updater.h"
29 #include "chrome/browser/metrics/user_metrics.h"
30 #include "chrome/browser/net/url_fixer_upper.h"
31 #include "chrome/browser/profiles/profile.h"
32 #include "chrome/browser/search_engines/template_url.h"
33 #include "chrome/browser/search_engines/template_url_model.h"
34 #include "chrome/browser/ui/views/location_bar/location_bar_view.h"
35 #include "content/browser/tab_contents/tab_contents.h"
36 #include "content/common/notification_service.h"
37 #include "googleurl/src/url_util.h"
38 #include "grit/generated_resources.h"
39 #include "net/base/escape.h"
40 #include "skia/ext/skia_utils_win.h"
41 #include "ui/base/clipboard/clipboard.h"
42 #include "ui/base/clipboard/scoped_clipboard_writer.h"
43 #include "ui/base/dragdrop/drag_drop_types.h"
44 #include "ui/base/dragdrop/drag_source.h"
45 #include "ui/base/dragdrop/drop_target.h"
46 #include "ui/base/dragdrop/os_exchange_data.h"
47 #include "ui/base/dragdrop/os_exchange_data_provider_win.h"
48 #include "ui/base/keycodes/keyboard_codes.h"
49 #include "ui/base/l10n/l10n_util.h"
50 #include "ui/base/l10n/l10n_util_win.h"
51 #include "ui/gfx/canvas.h"
52 #include "ui/gfx/canvas_skia.h"
53 #include "views/controls/textfield/native_textfield_win.h"
54 #include "views/drag_utils.h"
55 #include "views/events/event_utils_win.h"
56 #include "views/focus/focus_util_win.h"
57 #include "views/widget/widget.h"
58 
59 #pragma comment(lib, "oleacc.lib")  // Needed for accessibility support.
60 #pragma comment(lib, "riched20.lib")  // Needed for the richedit control.
61 
62 ///////////////////////////////////////////////////////////////////////////////
63 // AutocompleteEditModel
64 
65 namespace {
66 
67 // A helper method for determining a valid DROPEFFECT given the allowed
68 // DROPEFFECTS.  We prefer copy over link.
CopyOrLinkDropEffect(DWORD effect)69 DWORD CopyOrLinkDropEffect(DWORD effect) {
70   if (effect & DROPEFFECT_COPY)
71     return DROPEFFECT_COPY;
72   if (effect & DROPEFFECT_LINK)
73     return DROPEFFECT_LINK;
74   return DROPEFFECT_NONE;
75 }
76 
77 // A helper method for determining a valid drag operation given the allowed
78 // operation.  We prefer copy over link.
CopyOrLinkDragOperation(int drag_operation)79 int CopyOrLinkDragOperation(int drag_operation) {
80   if (drag_operation & ui::DragDropTypes::DRAG_COPY)
81     return ui::DragDropTypes::DRAG_COPY;
82   if (drag_operation & ui::DragDropTypes::DRAG_LINK)
83     return ui::DragDropTypes::DRAG_LINK;
84   return ui::DragDropTypes::DRAG_NONE;
85 }
86 
87 // The AutocompleteEditState struct contains enough information about the
88 // AutocompleteEditModel and AutocompleteEditViewWin to save/restore a user's
89 // typing, caret position, etc. across tab changes.  We explicitly don't
90 // preserve things like whether the popup was open as this might be weird.
91 struct AutocompleteEditState {
AutocompleteEditState__anon5cfdf0bd0111::AutocompleteEditState92   AutocompleteEditState(const AutocompleteEditModel::State& model_state,
93                         const AutocompleteEditViewWin::State& view_state)
94       : model_state(model_state),
95         view_state(view_state) {
96   }
97 
98   const AutocompleteEditModel::State model_state;
99   const AutocompleteEditViewWin::State view_state;
100 };
101 
102 // Returns true if the current point is far enough from the origin that it
103 // would be considered a drag.
IsDrag(const POINT & origin,const POINT & current)104 bool IsDrag(const POINT& origin, const POINT& current) {
105   return views::View::ExceededDragThreshold(current.x - origin.x,
106                                             current.y - origin.y);
107 }
108 
109 }  // namespace
110 
111 // EditDropTarget is the IDropTarget implementation installed on
112 // AutocompleteEditViewWin. EditDropTarget prefers URL over plain text. A drop
113 // of a URL replaces all the text of the edit and navigates immediately to the
114 // URL. A drop of plain text from the same edit either copies or moves the
115 // selected text, and a drop of plain text from a source other than the edit
116 // does a paste and go.
117 class AutocompleteEditViewWin::EditDropTarget : public ui::DropTarget {
118  public:
119   explicit EditDropTarget(AutocompleteEditViewWin* edit);
120 
121  protected:
122   virtual DWORD OnDragEnter(IDataObject* data_object,
123                             DWORD key_state,
124                             POINT cursor_position,
125                             DWORD effect);
126   virtual DWORD OnDragOver(IDataObject* data_object,
127                            DWORD key_state,
128                            POINT cursor_position,
129                            DWORD effect);
130   virtual void OnDragLeave(IDataObject* data_object);
131   virtual DWORD OnDrop(IDataObject* data_object,
132                        DWORD key_state,
133                        POINT cursor_position,
134                        DWORD effect);
135 
136  private:
137   // If dragging a string, the drop highlight position of the edit is reset
138   // based on the mouse position.
139   void UpdateDropHighlightPosition(const POINT& cursor_screen_position);
140 
141   // Resets the visual drop indicates we install on the edit.
142   void ResetDropHighlights();
143 
144   // The edit we're the drop target for.
145   AutocompleteEditViewWin* edit_;
146 
147   // If true, the drag session contains a URL.
148   bool drag_has_url_;
149 
150   // If true, the drag session contains a string. If drag_has_url_ is true,
151   // this is false regardless of whether the clipboard has a string.
152   bool drag_has_string_;
153 
154   DISALLOW_COPY_AND_ASSIGN(EditDropTarget);
155 };
156 
EditDropTarget(AutocompleteEditViewWin * edit)157 AutocompleteEditViewWin::EditDropTarget::EditDropTarget(
158     AutocompleteEditViewWin* edit)
159     : ui::DropTarget(edit->m_hWnd),
160       edit_(edit),
161       drag_has_url_(false),
162       drag_has_string_(false) {
163 }
164 
OnDragEnter(IDataObject * data_object,DWORD key_state,POINT cursor_position,DWORD effect)165 DWORD AutocompleteEditViewWin::EditDropTarget::OnDragEnter(
166     IDataObject* data_object,
167     DWORD key_state,
168     POINT cursor_position,
169     DWORD effect) {
170   ui::OSExchangeData os_data(new ui::OSExchangeDataProviderWin(data_object));
171   drag_has_url_ = os_data.HasURL();
172   drag_has_string_ = !drag_has_url_ && os_data.HasString();
173   if (drag_has_url_) {
174     if (edit_->in_drag()) {
175       // The edit we're associated with originated the drag. No point in
176       // allowing the user to drop back on us.
177       drag_has_url_ = false;
178     }
179     // NOTE: it would be nice to visually show all the text is going to
180     // be replaced by selecting all, but this caused painting problems. In
181     // particular the flashing caret would appear outside the edit! For now
182     // we stick with no visual indicator other than that shown own the mouse
183     // cursor.
184   }
185   return OnDragOver(data_object, key_state, cursor_position, effect);
186 }
187 
OnDragOver(IDataObject * data_object,DWORD key_state,POINT cursor_position,DWORD effect)188 DWORD AutocompleteEditViewWin::EditDropTarget::OnDragOver(
189     IDataObject* data_object,
190     DWORD key_state,
191     POINT cursor_position,
192     DWORD effect) {
193   if (drag_has_url_)
194     return CopyOrLinkDropEffect(effect);
195 
196   if (drag_has_string_) {
197     UpdateDropHighlightPosition(cursor_position);
198     if (edit_->drop_highlight_position() == -1 && edit_->in_drag())
199       return DROPEFFECT_NONE;
200     if (edit_->in_drag()) {
201       // The edit we're associated with originated the drag.  Do the normal drag
202       // behavior.
203       DCHECK((effect & DROPEFFECT_COPY) && (effect & DROPEFFECT_MOVE));
204       return (key_state & MK_CONTROL) ? DROPEFFECT_COPY : DROPEFFECT_MOVE;
205     }
206     // Our edit didn't originate the drag, only allow link or copy.
207     return CopyOrLinkDropEffect(effect);
208   }
209 
210   return DROPEFFECT_NONE;
211 }
212 
OnDragLeave(IDataObject * data_object)213 void AutocompleteEditViewWin::EditDropTarget::OnDragLeave(
214     IDataObject* data_object) {
215   ResetDropHighlights();
216 }
217 
OnDrop(IDataObject * data_object,DWORD key_state,POINT cursor_position,DWORD effect)218 DWORD AutocompleteEditViewWin::EditDropTarget::OnDrop(
219     IDataObject* data_object,
220     DWORD key_state,
221     POINT cursor_position,
222     DWORD effect) {
223   effect = OnDragOver(data_object, key_state, cursor_position, effect);
224 
225   ui::OSExchangeData os_data(new ui::OSExchangeDataProviderWin(data_object));
226   views::DropTargetEvent event(os_data, cursor_position.x, cursor_position.y,
227       ui::DragDropTypes::DropEffectToDragOperation(effect));
228 
229   int drag_operation = edit_->OnPerformDropImpl(event, edit_->in_drag());
230 
231   if (!drag_has_url_)
232     ResetDropHighlights();
233 
234   return ui::DragDropTypes::DragOperationToDropEffect(drag_operation);
235 }
236 
UpdateDropHighlightPosition(const POINT & cursor_screen_position)237 void AutocompleteEditViewWin::EditDropTarget::UpdateDropHighlightPosition(
238     const POINT& cursor_screen_position) {
239   if (drag_has_string_) {
240     POINT client_position = cursor_screen_position;
241     ::ScreenToClient(edit_->m_hWnd, &client_position);
242     int drop_position = edit_->CharFromPos(client_position);
243     if (edit_->in_drag()) {
244       // Our edit originated the drag, don't allow a drop if over the selected
245       // region.
246       LONG sel_start, sel_end;
247       edit_->GetSel(sel_start, sel_end);
248       if ((sel_start != sel_end) && (drop_position >= sel_start) &&
249           (drop_position <= sel_end))
250         drop_position = -1;
251     } else {
252       // A drop from a source other than the edit replaces all the text, so
253       // we don't show the drop location. See comment in OnDragEnter as to why
254       // we don't try and select all here.
255       drop_position = -1;
256     }
257     edit_->SetDropHighlightPosition(drop_position);
258   }
259 }
260 
ResetDropHighlights()261 void AutocompleteEditViewWin::EditDropTarget::ResetDropHighlights() {
262   if (drag_has_string_)
263     edit_->SetDropHighlightPosition(-1);
264 }
265 
266 
267 ///////////////////////////////////////////////////////////////////////////////
268 // Helper classes
269 
ScopedFreeze(AutocompleteEditViewWin * edit,ITextDocument * text_object_model)270 AutocompleteEditViewWin::ScopedFreeze::ScopedFreeze(
271     AutocompleteEditViewWin* edit,
272     ITextDocument* text_object_model)
273     : edit_(edit),
274       text_object_model_(text_object_model) {
275   // Freeze the screen.
276   if (text_object_model_) {
277     long count;
278     text_object_model_->Freeze(&count);
279   }
280 }
281 
~ScopedFreeze()282 AutocompleteEditViewWin::ScopedFreeze::~ScopedFreeze() {
283   // Unfreeze the screen.
284   // NOTE: If this destructor is reached while the edit is being destroyed (for
285   // example, because we double-clicked the edit of a popup and caused it to
286   // transform to an unconstrained window), it will no longer have an HWND, and
287   // text_object_model_ may point to a destroyed object, so do nothing here.
288   if (edit_->IsWindow() && text_object_model_) {
289     long count;
290     text_object_model_->Unfreeze(&count);
291     if (count == 0) {
292       // We need to UpdateWindow() here in addition to InvalidateRect() because,
293       // as far as I can tell, the edit likes to synchronously erase its
294       // background when unfreezing, thus requiring us to synchronously redraw
295       // if we don't want flicker.
296       edit_->InvalidateRect(NULL, false);
297       edit_->UpdateWindow();
298     }
299   }
300 }
301 
ScopedSuspendUndo(ITextDocument * text_object_model)302 AutocompleteEditViewWin::ScopedSuspendUndo::ScopedSuspendUndo(
303     ITextDocument* text_object_model)
304     : text_object_model_(text_object_model) {
305   // Suspend Undo processing.
306   if (text_object_model_)
307     text_object_model_->Undo(tomSuspend, NULL);
308 }
309 
~ScopedSuspendUndo()310 AutocompleteEditViewWin::ScopedSuspendUndo::~ScopedSuspendUndo() {
311   // Resume Undo processing.
312   if (text_object_model_)
313     text_object_model_->Undo(tomResume, NULL);
314 }
315 
316 ///////////////////////////////////////////////////////////////////////////////
317 // AutocompleteEditViewWin
318 
319 namespace {
320 
321 // These are used to hook the CRichEditCtrl's calls to BeginPaint() and
322 // EndPaint() and provide a memory DC instead.  See OnPaint().
323 HWND edit_hwnd = NULL;
324 PAINTSTRUCT paint_struct;
325 
326 // Intercepted method for BeginPaint(). Must use __stdcall convention.
BeginPaintIntercept(HWND hWnd,LPPAINTSTRUCT lpPaint)327 HDC WINAPI BeginPaintIntercept(HWND hWnd, LPPAINTSTRUCT lpPaint) {
328   if (!edit_hwnd || (hWnd != edit_hwnd))
329     return ::BeginPaint(hWnd, lpPaint);
330 
331   *lpPaint = paint_struct;
332   return paint_struct.hdc;
333 }
334 
335 // Intercepted method for EndPaint(). Must use __stdcall convention.
EndPaintIntercept(HWND hWnd,const PAINTSTRUCT * lpPaint)336 BOOL WINAPI EndPaintIntercept(HWND hWnd, const PAINTSTRUCT* lpPaint) {
337   return (edit_hwnd && (hWnd == edit_hwnd)) || ::EndPaint(hWnd, lpPaint);
338 }
339 
340 // Returns a lazily initialized property bag accessor for saving our state in a
341 // TabContents.
GetStateAccessor()342 PropertyAccessor<AutocompleteEditState>* GetStateAccessor() {
343   static PropertyAccessor<AutocompleteEditState> state;
344   return &state;
345 }
346 
347 class PaintPatcher {
348  public:
349   PaintPatcher();
350   ~PaintPatcher();
351 
352   void RefPatch();
353   void DerefPatch();
354 
355  private:
356   size_t refcount_;
357   app::win::IATPatchFunction begin_paint_;
358   app::win::IATPatchFunction end_paint_;
359 
360   DISALLOW_COPY_AND_ASSIGN(PaintPatcher);
361 };
362 
PaintPatcher()363 PaintPatcher::PaintPatcher() : refcount_(0) {
364 }
365 
~PaintPatcher()366 PaintPatcher::~PaintPatcher() {
367   DCHECK_EQ(0U, refcount_);
368 }
369 
RefPatch()370 void PaintPatcher::RefPatch() {
371   if (refcount_ == 0) {
372     DCHECK(!begin_paint_.is_patched());
373     DCHECK(!end_paint_.is_patched());
374     begin_paint_.Patch(L"riched20.dll", "user32.dll", "BeginPaint",
375                        &BeginPaintIntercept);
376     end_paint_.Patch(L"riched20.dll", "user32.dll", "EndPaint",
377                      &EndPaintIntercept);
378   }
379   ++refcount_;
380 }
381 
DerefPatch()382 void PaintPatcher::DerefPatch() {
383   DCHECK(begin_paint_.is_patched());
384   DCHECK(end_paint_.is_patched());
385   --refcount_;
386   if (refcount_ == 0) {
387     begin_paint_.Unpatch();
388     end_paint_.Unpatch();
389   }
390 }
391 
392 base::LazyInstance<PaintPatcher> g_paint_patcher(base::LINKER_INITIALIZED);
393 
394 // twips are a unit of type measurement, and RichEdit controls use them
395 // to set offsets.
396 const int kTwipsPerInch = 1440;
397 
398 }  // namespace
399 
AutocompleteEditViewWin(const gfx::Font & font,AutocompleteEditController * controller,ToolbarModel * toolbar_model,LocationBarView * parent_view,HWND hwnd,Profile * profile,CommandUpdater * command_updater,bool popup_window_mode,const views::View * location_bar)400 AutocompleteEditViewWin::AutocompleteEditViewWin(
401     const gfx::Font& font,
402     AutocompleteEditController* controller,
403     ToolbarModel* toolbar_model,
404     LocationBarView* parent_view,
405     HWND hwnd,
406     Profile* profile,
407     CommandUpdater* command_updater,
408     bool popup_window_mode,
409     const views::View* location_bar)
410     : model_(new AutocompleteEditModel(this, controller, profile)),
411       popup_view_(new AutocompletePopupContentsView(font, this, model_.get(),
412                                                     profile, location_bar)),
413       controller_(controller),
414       parent_view_(parent_view),
415       toolbar_model_(toolbar_model),
416       command_updater_(command_updater),
417       popup_window_mode_(popup_window_mode),
418       force_hidden_(false),
419       tracking_click_(),
420       tracking_double_click_(false),
421       double_click_time_(0),
422       can_discard_mousemove_(false),
423       ignore_ime_messages_(false),
424       delete_at_end_pressed_(false),
425       font_(font),
426       possible_drag_(false),
427       in_drag_(false),
428       initiated_drag_(false),
429       drop_highlight_position_(-1),
430       background_color_(skia::SkColorToCOLORREF(LocationBarView::GetColor(
431           ToolbarModel::NONE, LocationBarView::BACKGROUND))),
432       security_level_(ToolbarModel::NONE),
433       text_object_model_(NULL) {
434   // Dummy call to a function exported by riched20.dll to ensure it sets up an
435   // import dependency on the dll.
436   CreateTextServices(NULL, NULL, NULL);
437 
438   saved_selection_for_focus_change_.cpMin = -1;
439 
440   g_paint_patcher.Pointer()->RefPatch();
441 
442   Create(hwnd, 0, 0, 0, l10n_util::GetExtendedStyles());
443   SetReadOnly(popup_window_mode_);
444   SetFont(font_.GetNativeFont());
445 
446   // NOTE: Do not use SetWordBreakProcEx() here, that is no longer supported as
447   // of Rich Edit 2.0 onward.
448   SendMessage(m_hWnd, EM_SETWORDBREAKPROC, 0,
449               reinterpret_cast<LPARAM>(&WordBreakProc));
450 
451   // Get the metrics for the font.
452   HDC dc = ::GetDC(NULL);
453   SelectObject(dc, font_.GetNativeFont());
454   TEXTMETRIC tm = {0};
455   GetTextMetrics(dc, &tm);
456   const float kXHeightRatio = 0.7f;  // The ratio of a font's x-height to its
457                                      // cap height.  Sadly, Windows doesn't
458                                      // provide a true value for a font's
459                                      // x-height in its text metrics, so we
460                                      // approximate.
461   font_x_height_ = static_cast<int>((static_cast<float>(font_.GetBaseline() -
462       tm.tmInternalLeading) * kXHeightRatio) + 0.5);
463   // The distance from the top of the field to the desired baseline of the
464   // rendered text.
465   const int kTextBaseline = popup_window_mode_ ? 15 : 18;
466   font_y_adjustment_ = kTextBaseline - font_.GetBaseline();
467 
468   // Get the number of twips per pixel, which we need below to offset our text
469   // by the desired number of pixels.
470   const long kTwipsPerPixel = kTwipsPerInch / GetDeviceCaps(dc, LOGPIXELSY);
471   ::ReleaseDC(NULL, dc);
472 
473   // Set the default character style -- adjust to our desired baseline.
474   CHARFORMAT cf = {0};
475   cf.dwMask = CFM_OFFSET;
476   cf.yOffset = -font_y_adjustment_ * kTwipsPerPixel;
477   SetDefaultCharFormat(cf);
478 
479   SetBackgroundColor(background_color_);
480 
481   // By default RichEdit has a drop target. Revoke it so that we can install our
482   // own. Revoke takes care of deleting the existing one.
483   RevokeDragDrop(m_hWnd);
484 
485   // Register our drop target. RichEdit appears to invoke RevokeDropTarget when
486   // done so that we don't have to explicitly.
487   if (!popup_window_mode_) {
488     scoped_refptr<EditDropTarget> drop_target = new EditDropTarget(this);
489     RegisterDragDrop(m_hWnd, drop_target.get());
490   }
491 }
492 
~AutocompleteEditViewWin()493 AutocompleteEditViewWin::~AutocompleteEditViewWin() {
494   NotificationService::current()->Notify(
495       NotificationType::AUTOCOMPLETE_EDIT_DESTROYED,
496       Source<AutocompleteEditViewWin>(this),
497       NotificationService::NoDetails());
498 
499   // Explicitly release the text object model now that we're done with it, and
500   // before we free the library. If the library gets unloaded before this
501   // released, it becomes garbage.
502   text_object_model_->Release();
503 
504   // We balance our reference count and unpatch when the last instance has
505   // been destroyed.  This prevents us from relying on the AtExit or static
506   // destructor sequence to do our unpatching, which is generally fragile.
507   g_paint_patcher.Pointer()->DerefPatch();
508 }
509 
parent_view() const510 views::View* AutocompleteEditViewWin::parent_view() const {
511   return parent_view_;
512 }
513 
WidthOfTextAfterCursor()514 int AutocompleteEditViewWin::WidthOfTextAfterCursor() {
515   CHARRANGE selection;
516   GetSelection(selection);
517   const int start = std::max(0, static_cast<int>(selection.cpMax - 1));
518   return WidthNeededToDisplay(GetText().substr(start));
519 }
520 
GetFont()521 gfx::Font AutocompleteEditViewWin::GetFont() {
522   return font_;
523 }
524 
SaveStateToTab(TabContents * tab)525 void AutocompleteEditViewWin::SaveStateToTab(TabContents* tab) {
526   DCHECK(tab);
527 
528   const AutocompleteEditModel::State model_state(
529       model_->GetStateForTabSwitch());
530 
531   CHARRANGE selection;
532   GetSelection(selection);
533   GetStateAccessor()->SetProperty(tab->property_bag(),
534       AutocompleteEditState(
535           model_state,
536           State(selection, saved_selection_for_focus_change_)));
537 }
538 
Update(const TabContents * tab_for_state_restoring)539 void AutocompleteEditViewWin::Update(
540     const TabContents* tab_for_state_restoring) {
541   const bool visibly_changed_permanent_text =
542       model_->UpdatePermanentText(toolbar_model_->GetText());
543 
544   const ToolbarModel::SecurityLevel security_level =
545       toolbar_model_->GetSecurityLevel();
546   const bool changed_security_level = (security_level != security_level_);
547 
548   // Bail early when no visible state will actually change (prevents an
549   // unnecessary ScopedFreeze, and thus UpdateWindow()).
550   if (!changed_security_level && !visibly_changed_permanent_text &&
551       !tab_for_state_restoring)
552     return;
553 
554   // Update our local state as desired.  We set security_level_ here so it will
555   // already be correct before we get to any RevertAll()s below and use it.
556   security_level_ = security_level;
557 
558   // When we're switching to a new tab, restore its state, if any.
559   ScopedFreeze freeze(this, GetTextObjectModel());
560   if (tab_for_state_restoring) {
561     // Make sure we reset our own state first.  The new tab may not have any
562     // saved state, or it may not have had input in progress, in which case we
563     // won't overwrite all our local state.
564     RevertAll();
565 
566     const AutocompleteEditState* state = GetStateAccessor()->GetProperty(
567         tab_for_state_restoring->property_bag());
568     if (state) {
569       model_->RestoreState(state->model_state);
570 
571       // Restore user's selection.  We do this after restoring the user_text
572       // above so we're selecting in the correct string.
573       SetSelectionRange(state->view_state.selection);
574       saved_selection_for_focus_change_ =
575           state->view_state.saved_selection_for_focus_change;
576     }
577   } else if (visibly_changed_permanent_text) {
578     // Not switching tabs, just updating the permanent text.  (In the case where
579     // we _were_ switching tabs, the RevertAll() above already drew the new
580     // permanent text.)
581 
582     // Tweak: if the edit was previously nonempty and had all the text selected,
583     // select all the new text.  This makes one particular case better: the
584     // user clicks in the box to change it right before the permanent URL is
585     // changed.  Since the new URL is still fully selected, the user's typing
586     // will replace the edit contents as they'd intended.
587     //
588     // NOTE: The selection can be longer than the text length if the edit is in
589     // in rich text mode and the user has selected the "phantom newline" at the
590     // end, so use ">=" instead of "==" to see if all the text is selected.  In
591     // theory we prevent this case from ever occurring, but this is still safe.
592     CHARRANGE sel;
593     GetSelection(sel);
594     const bool was_reversed = (sel.cpMin > sel.cpMax);
595     const bool was_sel_all = (sel.cpMin != sel.cpMax) &&
596       IsSelectAllForRange(sel);
597 
598     RevertAll();
599 
600     if (was_sel_all)
601       SelectAll(was_reversed);
602   } else if (changed_security_level) {
603     // Only the security style changed, nothing else.  Redraw our text using it.
604     EmphasizeURLComponents();
605   }
606 }
607 
OpenURL(const GURL & url,WindowOpenDisposition disposition,PageTransition::Type transition,const GURL & alternate_nav_url,size_t selected_line,const string16 & keyword)608 void AutocompleteEditViewWin::OpenURL(const GURL& url,
609                                       WindowOpenDisposition disposition,
610                                       PageTransition::Type transition,
611                                       const GURL& alternate_nav_url,
612                                       size_t selected_line,
613                                       const string16& keyword) {
614   if (!url.is_valid())
615     return;
616 
617   // When we navigate, we first revert to the unedited state, then if necessary
618   // synchronously change the permanent text to the new URL.  If we don't freeze
619   // here, the user could potentially see a flicker of the current URL before
620   // the new one reappears, which would look glitchy.
621   ScopedFreeze freeze(this, GetTextObjectModel());
622   model_->OpenURL(url, disposition, transition, alternate_nav_url,
623                   selected_line, keyword);
624 }
625 
GetText() const626 string16 AutocompleteEditViewWin::GetText() const {
627   const int len = GetTextLength() + 1;
628   string16 str;
629   GetWindowText(WriteInto(&str, len), len);
630   return str;
631 }
632 
IsEditingOrEmpty() const633 bool AutocompleteEditViewWin::IsEditingOrEmpty() const {
634   return model_->user_input_in_progress() || (GetTextLength() == 0);
635 }
636 
GetIcon() const637 int AutocompleteEditViewWin::GetIcon() const {
638   return IsEditingOrEmpty() ?
639       AutocompleteMatch::TypeToIcon(model_->CurrentTextType()) :
640       toolbar_model_->GetIcon();
641 }
642 
SetUserText(const string16 & text)643 void AutocompleteEditViewWin::SetUserText(const string16& text) {
644   SetUserText(text, text, true);
645 }
646 
SetUserText(const string16 & text,const string16 & display_text,bool update_popup)647 void AutocompleteEditViewWin::SetUserText(const string16& text,
648                                           const string16& display_text,
649                                           bool update_popup) {
650   ScopedFreeze freeze(this, GetTextObjectModel());
651   model_->SetUserText(text);
652   saved_selection_for_focus_change_.cpMin = -1;
653   SetWindowTextAndCaretPos(display_text, display_text.length());
654   if (update_popup)
655     UpdatePopup();
656   TextChanged();
657 }
658 
SetWindowTextAndCaretPos(const string16 & text,size_t caret_pos)659 void AutocompleteEditViewWin::SetWindowTextAndCaretPos(const string16& text,
660                                                        size_t caret_pos) {
661   SetWindowText(text.c_str());
662   PlaceCaretAt(caret_pos);
663 }
664 
SetForcedQuery()665 void AutocompleteEditViewWin::SetForcedQuery() {
666   const string16 current_text(GetText());
667   const size_t start = current_text.find_first_not_of(kWhitespaceWide);
668   if (start == string16::npos || (current_text[start] != '?'))
669     SetUserText(L"?");
670   else
671     SetSelection(current_text.length(), start + 1);
672 }
673 
IsSelectAll()674 bool AutocompleteEditViewWin::IsSelectAll() {
675   CHARRANGE selection;
676   GetSel(selection);
677   return IsSelectAllForRange(selection);
678 }
679 
DeleteAtEndPressed()680 bool AutocompleteEditViewWin::DeleteAtEndPressed() {
681   return delete_at_end_pressed_;
682 }
683 
GetSelectionBounds(string16::size_type * start,string16::size_type * end)684 void AutocompleteEditViewWin::GetSelectionBounds(string16::size_type* start,
685                                                  string16::size_type* end) {
686   CHARRANGE selection;
687   GetSel(selection);
688   *start = static_cast<size_t>(selection.cpMin);
689   *end = static_cast<size_t>(selection.cpMax);
690 }
691 
SelectAll(bool reversed)692 void AutocompleteEditViewWin::SelectAll(bool reversed) {
693   if (reversed)
694     SetSelection(GetTextLength(), 0);
695   else
696     SetSelection(0, GetTextLength());
697 }
698 
RevertAll()699 void AutocompleteEditViewWin::RevertAll() {
700   ScopedFreeze freeze(this, GetTextObjectModel());
701   ClosePopup();
702   model_->Revert();
703   saved_selection_for_focus_change_.cpMin = -1;
704   TextChanged();
705 }
706 
UpdatePopup()707 void AutocompleteEditViewWin::UpdatePopup() {
708   ScopedFreeze freeze(this, GetTextObjectModel());
709   model_->SetInputInProgress(true);
710 
711   if (!model_->has_focus()) {
712     // When we're in the midst of losing focus, don't rerun autocomplete.  This
713     // can happen when losing focus causes the IME to cancel/finalize a
714     // composition.  We still want to note that user input is in progress, we
715     // just don't want to do anything else.
716     //
717     // Note that in this case the ScopedFreeze above was unnecessary; however,
718     // we're inside the callstack of OnKillFocus(), which has already frozen the
719     // edit, so this will never result in an unnecessary UpdateWindow() call.
720     return;
721   }
722 
723   // Don't inline autocomplete when:
724   //   * The user is deleting text
725   //   * The caret/selection isn't at the end of the text
726   //   * The user has just pasted in something that replaced all the text
727   //   * The user is trying to compose something in an IME
728   CHARRANGE sel;
729   GetSel(sel);
730   model_->StartAutocomplete(sel.cpMax != sel.cpMin,
731                             (sel.cpMax < GetTextLength()) || IsImeComposing());
732 }
733 
ClosePopup()734 void AutocompleteEditViewWin::ClosePopup() {
735   model_->StopAutocomplete();
736 }
737 
SetFocus()738 void AutocompleteEditViewWin::SetFocus() {
739   ::SetFocus(m_hWnd);
740   parent_view_->GetWidget()->NotifyAccessibilityEvent(
741       parent_view_,
742       ui::AccessibilityTypes::EVENT_FOCUS,
743       false);
744 }
745 
GetIAccessible()746 IAccessible* AutocompleteEditViewWin::GetIAccessible() {
747   if (!autocomplete_accessibility_) {
748     CComObject<AutocompleteAccessibility>* accessibility = NULL;
749     if (!SUCCEEDED(CComObject<AutocompleteAccessibility>::CreateInstance(
750             &accessibility)) || !accessibility)
751       return NULL;
752 
753     // Wrap the created object in a smart pointer so it won't leak.
754     base::win::ScopedComPtr<IAccessible> accessibility_comptr(accessibility);
755     if (!SUCCEEDED(accessibility->Initialize(this)))
756       return NULL;
757 
758     // Copy to the class smart pointer, and notify that an instance of
759     // IAccessible was allocated for m_hWnd.
760     autocomplete_accessibility_ = accessibility_comptr;
761     NotifyWinEvent(EVENT_OBJECT_CREATE, m_hWnd, OBJID_CLIENT, CHILDID_SELF);
762   }
763   // Detach to leave ref counting to the caller.
764   return autocomplete_accessibility_.Detach();
765 }
766 
SetDropHighlightPosition(int position)767 void AutocompleteEditViewWin::SetDropHighlightPosition(int position) {
768   if (drop_highlight_position_ != position) {
769     RepaintDropHighlight(drop_highlight_position_);
770     drop_highlight_position_ = position;
771     RepaintDropHighlight(drop_highlight_position_);
772   }
773 }
774 
MoveSelectedText(int new_position)775 void AutocompleteEditViewWin::MoveSelectedText(int new_position) {
776   const string16 selected_text(GetSelectedText());
777   CHARRANGE sel;
778   GetSel(sel);
779   DCHECK((sel.cpMax != sel.cpMin) && (new_position >= 0) &&
780          (new_position <= GetTextLength()));
781 
782   ScopedFreeze freeze(this, GetTextObjectModel());
783   OnBeforePossibleChange();
784 
785   // Nuke the selected text.
786   ReplaceSel(L"", TRUE);
787 
788   // And insert it into the new location.
789   if (new_position >= sel.cpMin)
790     new_position -= (sel.cpMax - sel.cpMin);
791   PlaceCaretAt(new_position);
792   ReplaceSel(selected_text.c_str(), TRUE);
793 
794   OnAfterPossibleChange();
795 }
796 
InsertText(int position,const string16 & text)797 void AutocompleteEditViewWin::InsertText(int position,
798                                          const string16& text) {
799   DCHECK((position >= 0) && (position <= GetTextLength()));
800   ScopedFreeze freeze(this, GetTextObjectModel());
801   OnBeforePossibleChange();
802   SetSelection(position, position);
803   ReplaceSel(text.c_str());
804   OnAfterPossibleChange();
805 }
806 
OnTemporaryTextMaybeChanged(const string16 & display_text,bool save_original_selection)807 void AutocompleteEditViewWin::OnTemporaryTextMaybeChanged(
808     const string16& display_text,
809     bool save_original_selection) {
810   if (save_original_selection)
811     GetSelection(original_selection_);
812 
813   // Set new text and cursor position.  Sometimes this does extra work (e.g.
814   // when the new text and the old text are identical), but it's only called
815   // when the user manually changes the selected line in the popup, so that's
816   // not really a problem.  Also, even when the text hasn't changed we'd want to
817   // update the caret, because if the user had the cursor in the middle of the
818   // text and then arrowed to another entry with the same text, we'd still want
819   // to move the caret.
820   ScopedFreeze freeze(this, GetTextObjectModel());
821   SetWindowTextAndCaretPos(display_text, display_text.length());
822   TextChanged();
823 }
824 
OnInlineAutocompleteTextMaybeChanged(const string16 & display_text,size_t user_text_length)825 bool AutocompleteEditViewWin::OnInlineAutocompleteTextMaybeChanged(
826     const string16& display_text,
827     size_t user_text_length) {
828   // Update the text and selection.  Because this can be called repeatedly while
829   // typing, we've careful not to freeze the edit unless we really need to.
830   // Also, unlike in the temporary text case above, here we don't want to update
831   // the caret/selection unless we have to, since this might make the user's
832   // caret position change without warning during typing.
833   if (display_text == GetText())
834     return false;
835 
836   ScopedFreeze freeze(this, GetTextObjectModel());
837   SetWindowText(display_text.c_str());
838   // Set a reversed selection to keep the caret in the same position, which
839   // avoids scrolling the user's text.
840   SetSelection(static_cast<LONG>(display_text.length()),
841                static_cast<LONG>(user_text_length));
842   TextChanged();
843   return true;
844 }
845 
OnRevertTemporaryText()846 void AutocompleteEditViewWin::OnRevertTemporaryText() {
847   SetSelectionRange(original_selection_);
848   TextChanged();
849 }
850 
OnBeforePossibleChange()851 void AutocompleteEditViewWin::OnBeforePossibleChange() {
852   // Record our state.
853   text_before_change_ = GetText();
854   GetSelection(sel_before_change_);
855 }
856 
OnAfterPossibleChange()857 bool AutocompleteEditViewWin::OnAfterPossibleChange() {
858   return OnAfterPossibleChangeInternal(false);
859 }
860 
OnAfterPossibleChangeInternal(bool force_text_changed)861 bool AutocompleteEditViewWin::OnAfterPossibleChangeInternal(
862     bool force_text_changed) {
863   // Prevent the user from selecting the "phantom newline" at the end of the
864   // edit.  If they try, we just silently move the end of the selection back to
865   // the end of the real text.
866   CHARRANGE new_sel;
867   GetSelection(new_sel);
868   const int length = GetTextLength();
869   if ((new_sel.cpMin > length) || (new_sel.cpMax > length)) {
870     if (new_sel.cpMin > length)
871       new_sel.cpMin = length;
872     if (new_sel.cpMax > length)
873       new_sel.cpMax = length;
874     SetSelectionRange(new_sel);
875   }
876   const bool selection_differs =
877       ((new_sel.cpMin != new_sel.cpMax) ||
878        (sel_before_change_.cpMin != sel_before_change_.cpMax)) &&
879       ((new_sel.cpMin != sel_before_change_.cpMin) ||
880        (new_sel.cpMax != sel_before_change_.cpMax));
881 
882   // See if the text or selection have changed since OnBeforePossibleChange().
883   const string16 new_text(GetText());
884   const bool text_differs = (new_text != text_before_change_) ||
885       force_text_changed;
886 
887   // When the user has deleted text, we don't allow inline autocomplete.  Make
888   // sure to not flag cases like selecting part of the text and then pasting
889   // (or typing) the prefix of that selection.  (We detect these by making
890   // sure the caret, which should be after any insertion, hasn't moved
891   // forward of the old selection start.)
892   const bool just_deleted_text =
893       (text_before_change_.length() > new_text.length()) &&
894       (new_sel.cpMin <= std::min(sel_before_change_.cpMin,
895                                  sel_before_change_.cpMax));
896 
897   const bool something_changed = model_->OnAfterPossibleChange(
898       new_text, new_sel.cpMin, new_sel.cpMax, selection_differs,
899       text_differs, just_deleted_text, !IsImeComposing());
900 
901   if (selection_differs)
902     controller_->OnSelectionBoundsChanged();
903 
904   if (something_changed && text_differs)
905     TextChanged();
906 
907   if (text_differs) {
908     // Note that a TEXT_CHANGED event implies that the cursor/selection
909     // probably changed too, so we don't need to send both.
910     parent_view_->GetWidget()->NotifyAccessibilityEvent(
911         parent_view_, ui::AccessibilityTypes::EVENT_TEXT_CHANGED, true);
912   } else if (selection_differs) {
913     // Notify assistive technology that the cursor or selection changed.
914     parent_view_->GetWidget()->NotifyAccessibilityEvent(
915         parent_view_, ui::AccessibilityTypes::EVENT_SELECTION_CHANGED, true);
916   } else if (delete_at_end_pressed_) {
917     model_->OnChanged();
918   }
919 
920   return something_changed;
921 }
922 
GetNativeView() const923 gfx::NativeView AutocompleteEditViewWin::GetNativeView() const {
924   return m_hWnd;
925 }
926 
GetCommandUpdater()927 CommandUpdater* AutocompleteEditViewWin::GetCommandUpdater() {
928   return command_updater_;
929 }
930 
SetInstantSuggestion(const string16 & suggestion,bool animate_to_complete)931 void AutocompleteEditViewWin::SetInstantSuggestion(const string16& suggestion,
932                                                    bool animate_to_complete) {
933   parent_view_->SetInstantSuggestion(suggestion, animate_to_complete);
934 }
935 
TextWidth() const936 int AutocompleteEditViewWin::TextWidth() const {
937   return WidthNeededToDisplay(GetText());
938 }
939 
GetInstantSuggestion() const940 string16 AutocompleteEditViewWin::GetInstantSuggestion() const {
941   return parent_view_->GetInstantSuggestion();
942 }
943 
IsImeComposing() const944 bool AutocompleteEditViewWin::IsImeComposing() const {
945   bool ime_composing = false;
946   HIMC context = ImmGetContext(m_hWnd);
947   if (context) {
948     ime_composing = !!ImmGetCompositionString(context, GCS_COMPSTR, NULL, 0);
949     ImmReleaseContext(m_hWnd, context);
950   }
951   return ime_composing;
952 }
953 
AddToView(views::View * parent)954 views::View* AutocompleteEditViewWin::AddToView(views::View* parent) {
955   views::NativeViewHost* host = new views::NativeViewHost;
956   parent->AddChildView(host);
957   host->set_focus_view(parent);
958   host->Attach(GetNativeView());
959   return host;
960 }
961 
OnPerformDrop(const views::DropTargetEvent & event)962 int AutocompleteEditViewWin::OnPerformDrop(
963     const views::DropTargetEvent& event) {
964   return OnPerformDropImpl(event, false);
965 }
966 
OnPerformDropImpl(const views::DropTargetEvent & event,bool in_drag)967 int AutocompleteEditViewWin::OnPerformDropImpl(
968     const views::DropTargetEvent& event,
969     bool in_drag) {
970   const ui::OSExchangeData& data = event.data();
971 
972   if (data.HasURL()) {
973     GURL url;
974     string16 title;
975     if (data.GetURLAndTitle(&url, &title)) {
976       SetUserText(UTF8ToWide(url.spec()));
977       model()->AcceptInput(CURRENT_TAB, true);
978       return CopyOrLinkDragOperation(event.source_operations());
979     }
980   } else if (data.HasString()) {
981     int string_drop_position = drop_highlight_position();
982     string16 text;
983     if ((string_drop_position != -1 || !in_drag) && data.GetString(&text)) {
984       DCHECK(string_drop_position == -1 ||
985              ((string_drop_position >= 0) &&
986               (string_drop_position <= GetTextLength())));
987       if (in_drag) {
988         if (event.source_operations()== ui::DragDropTypes::DRAG_MOVE)
989           MoveSelectedText(string_drop_position);
990         else
991           InsertText(string_drop_position, text);
992       } else {
993         PasteAndGo(CollapseWhitespace(text, true));
994       }
995       return CopyOrLinkDragOperation(event.source_operations());
996     }
997   }
998 
999   return ui::DragDropTypes::DRAG_NONE;
1000 }
1001 
PasteAndGo(const string16 & text)1002 void AutocompleteEditViewWin::PasteAndGo(const string16& text) {
1003   if (CanPasteAndGo(text))
1004     model_->PasteAndGo();
1005 }
1006 
SkipDefaultKeyEventProcessing(const views::KeyEvent & event)1007 bool AutocompleteEditViewWin::SkipDefaultKeyEventProcessing(
1008     const views::KeyEvent& event) {
1009   ui::KeyboardCode key = event.key_code();
1010   // We don't process ALT + numpad digit as accelerators, they are used for
1011   // entering special characters.  We do translate alt-home.
1012   if (event.IsAltDown() && (key != ui::VKEY_HOME) &&
1013       views::NativeTextfieldWin::IsNumPadDigit(key,
1014                                                views::IsExtendedKey(event)))
1015     return true;
1016 
1017   // Skip accelerators for key combinations omnibox wants to crack. This list
1018   // should be synced with OnKeyDownOnlyWritable() (but for tab which is dealt
1019   // with above in LocationBarView::SkipDefaultKeyEventProcessing).
1020   //
1021   // We cannot return true for all keys because we still need to handle some
1022   // accelerators (e.g., F5 for reload the page should work even when the
1023   // Omnibox gets focused).
1024   switch (key) {
1025     case ui::VKEY_ESCAPE: {
1026       ScopedFreeze freeze(this, GetTextObjectModel());
1027       return model_->OnEscapeKeyPressed();
1028     }
1029 
1030     case ui::VKEY_RETURN:
1031       return true;
1032 
1033     case ui::VKEY_UP:
1034     case ui::VKEY_DOWN:
1035       return !event.IsAltDown();
1036 
1037     case ui::VKEY_DELETE:
1038     case ui::VKEY_INSERT:
1039       return !event.IsAltDown() && event.IsShiftDown() &&
1040           !event.IsControlDown();
1041 
1042     case ui::VKEY_X:
1043     case ui::VKEY_V:
1044       return !event.IsAltDown() && event.IsControlDown();
1045 
1046     case ui::VKEY_BACK:
1047     case ui::VKEY_OEM_PLUS:
1048       return true;
1049 
1050     default:
1051       return false;
1052   }
1053 }
1054 
HandleExternalMsg(UINT msg,UINT flags,const CPoint & screen_point)1055 void AutocompleteEditViewWin::HandleExternalMsg(UINT msg,
1056                                                 UINT flags,
1057                                                 const CPoint& screen_point) {
1058   if (msg == WM_CAPTURECHANGED) {
1059     SendMessage(msg, 0, NULL);
1060     return;
1061   }
1062 
1063   CPoint client_point(screen_point);
1064   ::MapWindowPoints(NULL, m_hWnd, &client_point, 1);
1065   SendMessage(msg, flags, MAKELPARAM(client_point.x, client_point.y));
1066 }
1067 
IsCommandIdChecked(int command_id) const1068 bool AutocompleteEditViewWin::IsCommandIdChecked(int command_id) const {
1069   return false;
1070 }
1071 
IsCommandIdEnabled(int command_id) const1072 bool AutocompleteEditViewWin::IsCommandIdEnabled(int command_id) const {
1073   switch (command_id) {
1074     case IDS_UNDO:         return !!CanUndo();
1075     case IDC_CUT:          return !!CanCut();
1076     case IDC_COPY:         return !!CanCopy();
1077     case IDC_PASTE:        return !!CanPaste();
1078     case IDS_PASTE_AND_GO: return CanPasteAndGo(GetClipboardText());
1079     case IDS_SELECT_ALL:   return !!CanSelectAll();
1080     case IDS_EDIT_SEARCH_ENGINES:
1081       return command_updater_->IsCommandEnabled(IDC_EDIT_SEARCH_ENGINES);
1082     default:
1083       NOTREACHED();
1084       return false;
1085   }
1086 }
1087 
GetAcceleratorForCommandId(int command_id,ui::Accelerator * accelerator)1088 bool AutocompleteEditViewWin::GetAcceleratorForCommandId(
1089     int command_id,
1090     ui::Accelerator* accelerator) {
1091   return parent_view_->GetWidget()->GetAccelerator(command_id, accelerator);
1092 }
1093 
IsItemForCommandIdDynamic(int command_id) const1094 bool AutocompleteEditViewWin::IsItemForCommandIdDynamic(int command_id) const {
1095   // No need to change the default IDS_PASTE_AND_GO label unless this is a
1096   // search.
1097   return command_id == IDS_PASTE_AND_GO;
1098 }
1099 
GetLabelForCommandId(int command_id) const1100 string16 AutocompleteEditViewWin::GetLabelForCommandId(
1101     int command_id) const {
1102   DCHECK_EQ(IDS_PASTE_AND_GO, command_id);
1103   return l10n_util::GetStringUTF16(model_->is_paste_and_search() ?
1104       IDS_PASTE_AND_SEARCH : IDS_PASTE_AND_GO);
1105 }
1106 
ExecuteCommand(int command_id)1107 void AutocompleteEditViewWin::ExecuteCommand(int command_id) {
1108   ScopedFreeze freeze(this, GetTextObjectModel());
1109   if (command_id == IDS_PASTE_AND_GO) {
1110     // This case is separate from the switch() below since we don't want to wrap
1111     // it in OnBefore/AfterPossibleChange() calls.
1112     model_->PasteAndGo();
1113     return;
1114   }
1115 
1116   OnBeforePossibleChange();
1117   switch (command_id) {
1118     case IDS_UNDO:
1119       Undo();
1120       break;
1121 
1122     case IDC_CUT:
1123       Cut();
1124       break;
1125 
1126     case IDC_COPY:
1127       Copy();
1128       break;
1129 
1130     case IDC_PASTE:
1131       Paste();
1132       break;
1133 
1134     case IDS_SELECT_ALL:
1135       SelectAll(false);
1136       break;
1137 
1138     case IDS_EDIT_SEARCH_ENGINES:
1139       command_updater_->ExecuteCommand(IDC_EDIT_SEARCH_ENGINES);
1140       break;
1141 
1142     default:
1143       NOTREACHED();
1144       break;
1145   }
1146   OnAfterPossibleChange();
1147 }
1148 
1149 // static
WordBreakProc(LPTSTR edit_text,int current_pos,int num_bytes,int action)1150 int CALLBACK AutocompleteEditViewWin::WordBreakProc(LPTSTR edit_text,
1151                                                     int current_pos,
1152                                                     int num_bytes,
1153                                                     int action) {
1154   // TODO(pkasting): http://b/1111308 We should let other people, like ICU and
1155   // GURL, do the work for us here instead of writing all this ourselves.
1156 
1157   // Sadly, even though the MSDN docs claim that the third parameter here is a
1158   // number of characters, they lie.  It's a number of bytes.
1159   const int length = num_bytes / sizeof(wchar_t);
1160 
1161   // With no clear guidance from the MSDN docs on how to handle "not found" in
1162   // the "find the nearest xxx..." cases below, I cap the return values at
1163   // [0, length].  Since one of these (0) is also a valid position, the return
1164   // values are thus ambiguous :(
1165   switch (action) {
1166     // Find nearest character before current position that begins a word.
1167     case WB_LEFT:
1168     case WB_MOVEWORDLEFT: {
1169       if (current_pos < 2) {
1170         // Either current_pos == 0, so we have a "not found" case and return 0,
1171         // or current_pos == 1, and the only character before this position is
1172         // at 0.
1173         return 0;
1174       }
1175 
1176       // Look for a delimiter before the previous character; the previous word
1177       // starts immediately after.  (If we looked for a delimiter before the
1178       // current character, we could stop on the immediate prior character,
1179       // which would mean we'd return current_pos -- which isn't "before the
1180       // current position".)
1181       const int prev_delim =
1182           WordBreakProc(edit_text, current_pos - 1, num_bytes, WB_LEFTBREAK);
1183 
1184       if ((prev_delim == 0) &&
1185           !WordBreakProc(edit_text, 0, num_bytes, WB_ISDELIMITER)) {
1186         // Got back 0, but position 0 isn't a delimiter.  This was a "not
1187         // found" 0, so return one of our own.
1188         return 0;
1189       }
1190 
1191       return prev_delim + 1;
1192     }
1193 
1194     // Find nearest character after current position that begins a word.
1195     case WB_RIGHT:
1196     case WB_MOVEWORDRIGHT: {
1197       if (WordBreakProc(edit_text, current_pos, num_bytes, WB_ISDELIMITER)) {
1198         // The current character is a delimiter, so the next character starts
1199         // a new word.  Done.
1200         return current_pos + 1;
1201       }
1202 
1203       // Look for a delimiter after the current character; the next word starts
1204       // immediately after.
1205       const int next_delim =
1206           WordBreakProc(edit_text, current_pos, num_bytes, WB_RIGHTBREAK);
1207       if (next_delim == length) {
1208         // Didn't find a delimiter.  Return length to signal "not found".
1209         return length;
1210       }
1211 
1212       return next_delim + 1;
1213     }
1214 
1215     // Determine if the current character delimits words.
1216     case WB_ISDELIMITER:
1217       return !!(WordBreakProc(edit_text, current_pos, num_bytes, WB_CLASSIFY) &
1218                 WBF_BREAKLINE);
1219 
1220     // Return the classification of the current character.
1221     case WB_CLASSIFY:
1222       if (IsWhitespace(edit_text[current_pos])) {
1223         // Whitespace normally breaks words, but the MSDN docs say that we must
1224         // not break on the CRs in a "CR, LF" or a "CR, CR, LF" sequence.  Just
1225         // check for an arbitrarily long sequence of CRs followed by LF and
1226         // report "not a delimiter" for the current CR in that case.
1227         while ((current_pos < (length - 1)) &&
1228                (edit_text[current_pos] == 0x13)) {
1229           if (edit_text[++current_pos] == 0x10)
1230             return WBF_ISWHITE;
1231         }
1232         return WBF_BREAKLINE | WBF_ISWHITE;
1233       }
1234 
1235       // Punctuation normally breaks words, but the first two characters in
1236       // "://" (end of scheme) should not be breaks, so that "http://" will be
1237       // treated as one word.
1238       if (ispunct(edit_text[current_pos], std::locale()) &&
1239           !SchemeEnd(edit_text, current_pos, length) &&
1240           !SchemeEnd(edit_text, current_pos - 1, length))
1241         return WBF_BREAKLINE;
1242 
1243       // Normal character, no flags.
1244       return 0;
1245 
1246     // Finds nearest delimiter before current position.
1247     case WB_LEFTBREAK:
1248       for (int i = current_pos - 1; i >= 0; --i) {
1249         if (WordBreakProc(edit_text, i, num_bytes, WB_ISDELIMITER))
1250           return i;
1251       }
1252       return 0;
1253 
1254     // Finds nearest delimiter after current position.
1255     case WB_RIGHTBREAK:
1256       for (int i = current_pos + 1; i < length; ++i) {
1257         if (WordBreakProc(edit_text, i, num_bytes, WB_ISDELIMITER))
1258           return i;
1259       }
1260       return length;
1261   }
1262 
1263   NOTREACHED();
1264   return 0;
1265 }
1266 
1267 // static
SchemeEnd(LPTSTR edit_text,int current_pos,int length)1268 bool AutocompleteEditViewWin::SchemeEnd(LPTSTR edit_text,
1269                                         int current_pos,
1270                                         int length) {
1271   return (current_pos >= 0) &&
1272          ((length - current_pos) > 2) &&
1273          (edit_text[current_pos] == ':') &&
1274          (edit_text[current_pos + 1] == '/') &&
1275          (edit_text[current_pos + 2] == '/');
1276 }
1277 
OnChar(TCHAR ch,UINT repeat_count,UINT flags)1278 void AutocompleteEditViewWin::OnChar(TCHAR ch, UINT repeat_count, UINT flags) {
1279   // Don't let alt-enter beep.  Not sure this is necessary, as the standard
1280   // alt-enter will hit DiscardWMSysChar() and get thrown away, and
1281   // ctrl-alt-enter doesn't seem to reach here for some reason?  At least not on
1282   // my system... still, this is harmless and maybe necessary in other locales.
1283   if (ch == VK_RETURN && (flags & KF_ALTDOWN))
1284     return;
1285 
1286   // Escape is processed in OnKeyDown.  Don't let any WM_CHAR messages propagate
1287   // as we don't want the RichEdit to do anything funky.
1288   if (ch == VK_ESCAPE && !(flags & KF_ALTDOWN))
1289     return;
1290 
1291   if (ch == VK_TAB) {
1292     // Don't add tabs to the input.
1293     return;
1294   }
1295 
1296   HandleKeystroke(GetCurrentMessage()->message, ch, repeat_count, flags);
1297 }
1298 
OnContextMenu(HWND window,const CPoint & point)1299 void AutocompleteEditViewWin::OnContextMenu(HWND window, const CPoint& point) {
1300   BuildContextMenu();
1301   if (point.x == -1 || point.y == -1) {
1302     POINT p;
1303     GetCaretPos(&p);
1304     MapWindowPoints(HWND_DESKTOP, &p, 1);
1305     context_menu_->RunContextMenuAt(gfx::Point(p));
1306   } else {
1307     context_menu_->RunContextMenuAt(gfx::Point(point));
1308   }
1309 }
1310 
OnCopy()1311 void AutocompleteEditViewWin::OnCopy() {
1312   string16 text(GetSelectedText());
1313   if (text.empty())
1314     return;
1315 
1316   CHARRANGE sel;
1317   GURL url;
1318   bool write_url = false;
1319   GetSel(sel);
1320   // GetSel() doesn't preserve selection direction, so sel.cpMin will always be
1321   // the smaller value.
1322   model_->AdjustTextForCopy(sel.cpMin, IsSelectAll(), &text, &url, &write_url);
1323   ui::ScopedClipboardWriter scw(g_browser_process->clipboard());
1324   scw.WriteText(text);
1325   if (write_url) {
1326     scw.WriteBookmark(text, url.spec());
1327     scw.WriteHyperlink(EscapeForHTML(text), url.spec());
1328   }
1329 }
1330 
OnCut()1331 void AutocompleteEditViewWin::OnCut() {
1332   OnCopy();
1333 
1334   // This replace selection will have no effect (even on the undo stack) if the
1335   // current selection is empty.
1336   ReplaceSel(L"", true);
1337 }
1338 
OnGetObject(UINT uMsg,WPARAM wparam,LPARAM lparam)1339 LRESULT AutocompleteEditViewWin::OnGetObject(UINT uMsg,
1340                                              WPARAM wparam,
1341                                              LPARAM lparam) {
1342   // Accessibility readers will send an OBJID_CLIENT message.
1343   if (lparam == OBJID_CLIENT) {
1344     // Re-attach for internal re-usage of accessibility pointer.
1345     autocomplete_accessibility_.Attach(GetIAccessible());
1346 
1347     if (autocomplete_accessibility_) {
1348       return LresultFromObject(IID_IAccessible, wparam,
1349                                autocomplete_accessibility_);
1350     }
1351   }
1352   return 0;
1353 }
1354 
OnImeComposition(UINT message,WPARAM wparam,LPARAM lparam)1355 LRESULT AutocompleteEditViewWin::OnImeComposition(UINT message,
1356                                                   WPARAM wparam,
1357                                                   LPARAM lparam) {
1358   if (ignore_ime_messages_) {
1359     // This message was sent while we're in the middle of meddling with the
1360     // underlying edit control.  If we handle it below, OnAfterPossibleChange()
1361     // can get bogus text for the edit, and rerun autocomplete, destructively
1362     // modifying the result set that we're in the midst of using.  For example,
1363     // if SetWindowTextAndCaretPos() was called due to the user clicking an
1364     // entry in the popup, we're in the middle of executing SetSelectedLine(),
1365     // and changing the results can cause checkfailures.
1366     return DefWindowProc(message, wparam, lparam);
1367   }
1368 
1369   ScopedFreeze freeze(this, GetTextObjectModel());
1370   OnBeforePossibleChange();
1371   LRESULT result = DefWindowProc(message, wparam, lparam);
1372   // Force an IME composition confirmation operation to trigger the text_changed
1373   // code in OnAfterPossibleChange(), even if identical contents are confirmed,
1374   // to make sure the model can update its internal states correctly.
1375   OnAfterPossibleChangeInternal((lparam & GCS_RESULTSTR) != 0);
1376   return result;
1377 }
1378 
OnKeyDown(TCHAR key,UINT repeat_count,UINT flags)1379 void AutocompleteEditViewWin::OnKeyDown(TCHAR key,
1380                                         UINT repeat_count,
1381                                         UINT flags) {
1382   delete_at_end_pressed_ = false;
1383 
1384   if (OnKeyDownAllModes(key, repeat_count, flags))
1385     return;
1386 
1387   // Make sure that we handle system key events like Alt-F4.
1388   if (popup_window_mode_) {
1389     DefWindowProc(GetCurrentMessage()->message, key, MAKELPARAM(repeat_count,
1390                                                                 flags));
1391     return;
1392   }
1393 
1394   if (OnKeyDownOnlyWritable(key, repeat_count, flags))
1395     return;
1396 
1397   // CRichEditCtrl changes its text on WM_KEYDOWN instead of WM_CHAR for many
1398   // different keys (backspace, ctrl-v, ...), so we call this in both cases.
1399   HandleKeystroke(GetCurrentMessage()->message, key, repeat_count, flags);
1400 }
1401 
OnKeyUp(TCHAR key,UINT repeat_count,UINT flags)1402 void AutocompleteEditViewWin::OnKeyUp(TCHAR key,
1403                                       UINT repeat_count,
1404                                       UINT flags) {
1405   if (key == VK_CONTROL)
1406     model_->OnControlKeyChanged(false);
1407 
1408   // On systems with RTL input languages, ctrl+shift toggles the reading order
1409   // (depending on which shift key is pressed). But by default the CRichEditCtrl
1410   // only changes the current reading order, and as soon as the user deletes all
1411   // the text, or we call SetWindowText(), it reverts to the "default" order.
1412   // To work around this, if the user hits ctrl+shift, we pass it to
1413   // DefWindowProc() while the edit is empty, which toggles the default reading
1414   // order; then we restore the user's input.
1415   if (!(flags & KF_ALTDOWN) &&
1416       (((key == VK_CONTROL) && (GetKeyState(VK_SHIFT) < 0)) ||
1417        ((key == VK_SHIFT) && (GetKeyState(VK_CONTROL) < 0)))) {
1418     ScopedFreeze freeze(this, GetTextObjectModel());
1419 
1420     string16 saved_text(GetText());
1421     CHARRANGE saved_sel;
1422     GetSelection(saved_sel);
1423 
1424     SetWindowText(L"");
1425 
1426     DefWindowProc(WM_KEYUP, key, MAKELPARAM(repeat_count, flags));
1427 
1428     SetWindowText(saved_text.c_str());
1429     SetSelectionRange(saved_sel);
1430     return;
1431   }
1432 
1433   SetMsgHandled(false);
1434 }
1435 
OnKillFocus(HWND focus_wnd)1436 void AutocompleteEditViewWin::OnKillFocus(HWND focus_wnd) {
1437   if (m_hWnd == focus_wnd) {
1438     // Focus isn't actually leaving.
1439     SetMsgHandled(false);
1440     return;
1441   }
1442 
1443   // This must be invoked before ClosePopup.
1444   model_->OnWillKillFocus(focus_wnd);
1445 
1446   // Close the popup.
1447   ClosePopup();
1448 
1449   // Save the user's existing selection to restore it later.
1450   GetSelection(saved_selection_for_focus_change_);
1451 
1452   // Tell the model to reset itself.
1453   model_->OnKillFocus();
1454 
1455   // Let the CRichEditCtrl do its default handling.  This will complete any
1456   // in-progress IME composition.  We must do this after setting has_focus_ to
1457   // false so that UpdatePopup() will know not to rerun autocomplete.
1458   ScopedFreeze freeze(this, GetTextObjectModel());
1459   DefWindowProc(WM_KILLFOCUS, reinterpret_cast<WPARAM>(focus_wnd), 0);
1460 
1461   // Cancel any user selection and scroll the text back to the beginning of the
1462   // URL.  We have to do this after calling DefWindowProc() because otherwise
1463   // an in-progress IME composition will be completed at the new caret position,
1464   // resulting in the string jumping unexpectedly to the front of the edit.
1465   PlaceCaretAt(0);
1466 }
1467 
OnLButtonDblClk(UINT keys,const CPoint & point)1468 void AutocompleteEditViewWin::OnLButtonDblClk(UINT keys, const CPoint& point) {
1469   // Save the double click info for later triple-click detection.
1470   tracking_double_click_ = true;
1471   double_click_point_ = point;
1472   double_click_time_ = GetCurrentMessage()->time;
1473   possible_drag_ = false;
1474 
1475   // Modifying the selection counts as accepting any inline autocompletion, so
1476   // track "changes" made by clicking the mouse button.
1477   ScopedFreeze freeze(this, GetTextObjectModel());
1478   OnBeforePossibleChange();
1479   DefWindowProc(WM_LBUTTONDBLCLK, keys,
1480                 MAKELPARAM(ClipXCoordToVisibleText(point.x, false), point.y));
1481   OnAfterPossibleChange();
1482 
1483   gaining_focus_.reset();  // See NOTE in OnMouseActivate().
1484 }
1485 
OnLButtonDown(UINT keys,const CPoint & point)1486 void AutocompleteEditViewWin::OnLButtonDown(UINT keys, const CPoint& point) {
1487   TrackMousePosition(kLeft, point);
1488   if (gaining_focus_.get()) {
1489     // When Chrome was already the activated app, we haven't reached
1490     // OnSetFocus() yet.  When we get there, don't restore the saved selection,
1491     // since it will just screw up the user's interaction with the edit.
1492     saved_selection_for_focus_change_.cpMin = -1;
1493 
1494     // Crazy hack: In this particular case, the CRichEditCtrl seems to have an
1495     // internal flag that discards the next WM_LBUTTONDOWN without processing
1496     // it, so that clicks on the edit when its owning app is not activated are
1497     // eaten rather than processed (despite whatever the return value of
1498     // DefWindowProc(WM_MOUSEACTIVATE, ...) may say).  This behavior is
1499     // confusing and we want the click to be treated normally.  So, to reset the
1500     // CRichEditCtrl's internal flag, we pass it an extra WM_LBUTTONDOWN here
1501     // (as well as a matching WM_LBUTTONUP, just in case we'd be confusing some
1502     // kind of state tracking otherwise).
1503     DefWindowProc(WM_LBUTTONDOWN, keys, MAKELPARAM(point.x, point.y));
1504     DefWindowProc(WM_LBUTTONUP, keys, MAKELPARAM(point.x, point.y));
1505   }
1506 
1507   // Check for triple click, then reset tracker.  Should be safe to subtract
1508   // double_click_time_ from the current message's time even if the timer has
1509   // wrapped in between.
1510   const bool is_triple_click = tracking_double_click_ &&
1511       views::NativeTextfieldWin::IsDoubleClick(double_click_point_, point,
1512           GetCurrentMessage()->time - double_click_time_);
1513   tracking_double_click_ = false;
1514 
1515   if (!gaining_focus_.get() && !is_triple_click)
1516     OnPossibleDrag(point);
1517 
1518 
1519   // Modifying the selection counts as accepting any inline autocompletion, so
1520   // track "changes" made by clicking the mouse button.
1521   ScopedFreeze freeze(this, GetTextObjectModel());
1522   OnBeforePossibleChange();
1523   DefWindowProc(WM_LBUTTONDOWN, keys,
1524                 MAKELPARAM(ClipXCoordToVisibleText(point.x, is_triple_click),
1525                            point.y));
1526   OnAfterPossibleChange();
1527 
1528   gaining_focus_.reset();
1529 }
1530 
OnLButtonUp(UINT keys,const CPoint & point)1531 void AutocompleteEditViewWin::OnLButtonUp(UINT keys, const CPoint& point) {
1532   // default processing should happen first so we can see the result of the
1533   // selection
1534   ScopedFreeze freeze(this, GetTextObjectModel());
1535   DefWindowProc(WM_LBUTTONUP, keys,
1536                 MAKELPARAM(ClipXCoordToVisibleText(point.x, false), point.y));
1537 
1538   SelectAllIfNecessary(kLeft, point);
1539 
1540   tracking_click_[kLeft] = false;
1541 
1542   possible_drag_ = false;
1543 }
1544 
OnMButtonDblClk(UINT,const CPoint &)1545 void AutocompleteEditViewWin::OnMButtonDblClk(UINT /*keys*/,
1546                                               const CPoint& /*point*/) {
1547   gaining_focus_.reset();  // See NOTE in OnMouseActivate().
1548 
1549   // By default, the edit responds to middle-clicks by capturing the mouse and
1550   // ignoring all subsequent events until it receives another click (of any of
1551   // the left, middle, or right buttons).  This bizarre behavior is not only
1552   // useless but can cause the UI to appear unresponsive if a user accidentally
1553   // middle-clicks the edit (instead of a tab above it), so we purposefully eat
1554   // this message (instead of calling SetMsgHandled(false)) to avoid triggering
1555   // this.
1556 }
1557 
OnMButtonDown(UINT,const CPoint &)1558 void AutocompleteEditViewWin::OnMButtonDown(UINT /*keys*/,
1559                                             const CPoint& /*point*/) {
1560   tracking_double_click_ = false;
1561 
1562   // See note in OnMButtonDblClk above.
1563 }
1564 
OnMButtonUp(UINT,const CPoint &)1565 void AutocompleteEditViewWin::OnMButtonUp(UINT /*keys*/,
1566                                           const CPoint& /*point*/) {
1567   possible_drag_ = false;
1568 
1569   // See note in OnMButtonDblClk above.
1570 }
1571 
OnMouseActivate(HWND window,UINT hit_test,UINT mouse_message)1572 LRESULT AutocompleteEditViewWin::OnMouseActivate(HWND window,
1573                                                  UINT hit_test,
1574                                                  UINT mouse_message) {
1575   // First, give other handlers a chance to handle the message to see if we are
1576   // actually going to activate and gain focus.
1577   LRESULT result = DefWindowProc(WM_MOUSEACTIVATE,
1578                                  reinterpret_cast<WPARAM>(window),
1579                                  MAKELPARAM(hit_test, mouse_message));
1580   // Check if we're getting focus from a click.  We have to do this here rather
1581   // than in OnXButtonDown() since in many scenarios OnSetFocus() will be
1582   // reached before OnXButtonDown(), preventing us from detecting this properly
1583   // there.  Also in those cases, we need to already know in OnSetFocus() that
1584   // we should not restore the saved selection.
1585   if (!model_->has_focus() &&
1586       ((mouse_message == WM_LBUTTONDOWN || mouse_message == WM_RBUTTONDOWN)) &&
1587       (result == MA_ACTIVATE)) {
1588     DCHECK(!gaining_focus_.get());
1589     gaining_focus_.reset(new ScopedFreeze(this, GetTextObjectModel()));
1590     // NOTE: Despite |mouse_message| being WM_XBUTTONDOWN here, we're not
1591     // guaranteed to call OnXButtonDown() later!  Specifically, if this is the
1592     // second click of a double click, we'll reach here but later call
1593     // OnXButtonDblClk().  Make sure |gaining_focus_| gets reset both places,
1594     // or we'll have visual glitchiness and then DCHECK failures.
1595 
1596     // Don't restore saved selection, it will just screw up our interaction
1597     // with this edit.
1598     saved_selection_for_focus_change_.cpMin = -1;
1599   }
1600   return result;
1601 }
1602 
OnMouseMove(UINT keys,const CPoint & point)1603 void AutocompleteEditViewWin::OnMouseMove(UINT keys, const CPoint& point) {
1604   if (possible_drag_) {
1605     StartDragIfNecessary(point);
1606     // Don't fall through to default mouse handling, otherwise a second
1607     // drag session may start.
1608     return;
1609   }
1610 
1611   if (tracking_click_[kLeft] && !IsDrag(click_point_[kLeft], point))
1612     return;
1613 
1614   tracking_click_[kLeft] = false;
1615 
1616   // Return quickly if this can't change the selection/cursor, so we don't
1617   // create a ScopedFreeze (and thus do an UpdateWindow()) on every
1618   // WM_MOUSEMOVE.
1619   if (!(keys & MK_LBUTTON)) {
1620     DefWindowProc(WM_MOUSEMOVE, keys, MAKELPARAM(point.x, point.y));
1621     return;
1622   }
1623 
1624   // Clamp the selection to the visible text so the user can't drag to select
1625   // the "phantom newline".  In theory we could achieve this by clipping the X
1626   // coordinate, but in practice the edit seems to behave nondeterministically
1627   // with similar sequences of clipped input coordinates fed to it.  Maybe it's
1628   // reading the mouse cursor position directly?
1629   //
1630   // This solution has a minor visual flaw, however: if there's a visible cursor
1631   // at the edge of the text (only true when there's no selection), dragging the
1632   // mouse around outside that edge repaints the cursor on every WM_MOUSEMOVE
1633   // instead of allowing it to blink normally.  To fix this, we special-case
1634   // this exact case and discard the WM_MOUSEMOVE messages instead of passing
1635   // them along.
1636   //
1637   // But even this solution has a flaw!  (Argh.)  In the case where the user has
1638   // a selection that starts at the edge of the edit, and proceeds to the middle
1639   // of the edit, and the user is dragging back past the start edge to remove
1640   // the selection, there's a redraw problem where the change between having the
1641   // last few bits of text still selected and having nothing selected can be
1642   // slow to repaint (which feels noticeably strange).  This occurs if you only
1643   // let the edit receive a single WM_MOUSEMOVE past the edge of the text.  I
1644   // think on each WM_MOUSEMOVE the edit is repainting its previous state, then
1645   // updating its internal variables to the new state but not repainting.  To
1646   // fix this, we allow one more WM_MOUSEMOVE through after the selection has
1647   // supposedly been shrunk to nothing; this makes the edit redraw the selection
1648   // quickly so it feels smooth.
1649   CHARRANGE selection;
1650   GetSel(selection);
1651   const bool possibly_can_discard_mousemove =
1652       (selection.cpMin == selection.cpMax) &&
1653       (((selection.cpMin == 0) &&
1654         (ClipXCoordToVisibleText(point.x, false) > point.x)) ||
1655        ((selection.cpMin == GetTextLength()) &&
1656         (ClipXCoordToVisibleText(point.x, false) < point.x)));
1657   if (!can_discard_mousemove_ || !possibly_can_discard_mousemove) {
1658     can_discard_mousemove_ = possibly_can_discard_mousemove;
1659     ScopedFreeze freeze(this, GetTextObjectModel());
1660     OnBeforePossibleChange();
1661     // Force the Y coordinate to the center of the clip rect.  The edit
1662     // behaves strangely when the cursor is dragged vertically: if the cursor
1663     // is in the middle of the text, drags inside the clip rect do nothing,
1664     // and drags outside the clip rect act as if the cursor jumped to the
1665     // left edge of the text.  When the cursor is at the right edge, drags of
1666     // just a few pixels vertically end up selecting the "phantom newline"...
1667     // sometimes.
1668     RECT r;
1669     GetRect(&r);
1670     DefWindowProc(WM_MOUSEMOVE, keys,
1671                   MAKELPARAM(point.x, (r.bottom - r.top) / 2));
1672     OnAfterPossibleChange();
1673   }
1674 }
1675 
OnPaint(HDC bogus_hdc)1676 void AutocompleteEditViewWin::OnPaint(HDC bogus_hdc) {
1677   // We need to paint over the top of the edit.  If we simply let the edit do
1678   // its default painting, then do ours into the window DC, the screen is
1679   // updated in between and we can get flicker.  To avoid this, we force the
1680   // edit to paint into a memory DC, which we also paint onto, then blit the
1681   // whole thing to the screen.
1682 
1683   // Don't paint if not necessary.
1684   CRect paint_clip_rect;
1685   if (!GetUpdateRect(&paint_clip_rect, true))
1686     return;
1687 
1688   // Begin painting, and create a memory DC for the edit to paint into.
1689   CPaintDC paint_dc(m_hWnd);
1690   CDC memory_dc(CreateCompatibleDC(paint_dc));
1691   CRect rect;
1692   GetClientRect(&rect);
1693   // NOTE: This next call uses |paint_dc| instead of |memory_dc| because
1694   // |memory_dc| contains a 1x1 monochrome bitmap by default, which will cause
1695   // |memory_bitmap| to be monochrome, which isn't what we want.
1696   CBitmap memory_bitmap(CreateCompatibleBitmap(paint_dc, rect.Width(),
1697                                                rect.Height()));
1698   HBITMAP old_bitmap = memory_dc.SelectBitmap(memory_bitmap);
1699 
1700   // Tell our intercept functions to supply our memory DC to the edit when it
1701   // tries to call BeginPaint().
1702   //
1703   // The sane way to do this would be to use WM_PRINTCLIENT to ask the edit to
1704   // paint into our desired DC.  Unfortunately, the Rich Edit 3.0 that ships
1705   // with Windows 2000/XP/Vista doesn't handle WM_PRINTCLIENT correctly; it
1706   // treats it just like WM_PAINT and calls BeginPaint(), ignoring our provided
1707   // DC.  The Rich Edit 6.0 that ships with Office 2007 handles this better, but
1708   // has other issues, and we can't redistribute that DLL anyway.  So instead,
1709   // we use this scary hack.
1710   //
1711   // NOTE: It's possible to get nested paint calls (!) (try setting the
1712   // permanent URL to something longer than the edit width, then selecting the
1713   // contents of the edit, typing a character, and hitting <esc>), so we can't
1714   // DCHECK(!edit_hwnd_) here.  Instead, just save off the old HWND, which most
1715   // of the time will be NULL.
1716   HWND old_edit_hwnd = edit_hwnd;
1717   edit_hwnd = m_hWnd;
1718   paint_struct = paint_dc.m_ps;
1719   paint_struct.hdc = memory_dc;
1720   DefWindowProc(WM_PAINT, reinterpret_cast<WPARAM>(bogus_hdc), 0);
1721 
1722   // Make the selection look better.
1723   EraseTopOfSelection(&memory_dc, rect, paint_clip_rect);
1724 
1725   // Draw a slash through the scheme if this is insecure.
1726   if (insecure_scheme_component_.is_nonempty())
1727     DrawSlashForInsecureScheme(memory_dc, rect, paint_clip_rect);
1728 
1729   // Draw the drop highlight.
1730   if (drop_highlight_position_ != -1)
1731     DrawDropHighlight(memory_dc, rect, paint_clip_rect);
1732 
1733   // Blit the memory DC to the actual paint DC and clean up.
1734   BitBlt(paint_dc, rect.left, rect.top, rect.Width(), rect.Height(), memory_dc,
1735          rect.left, rect.top, SRCCOPY);
1736   memory_dc.SelectBitmap(old_bitmap);
1737   edit_hwnd = old_edit_hwnd;
1738 }
1739 
OnPaste()1740 void AutocompleteEditViewWin::OnPaste() {
1741   // Replace the selection if we have something to paste.
1742   const string16 text(GetClipboardText());
1743   if (!text.empty()) {
1744     // Record this paste, so we can do different behavior.
1745     model_->on_paste();
1746     // Force a Paste operation to trigger the text_changed code in
1747     // OnAfterPossibleChange(), even if identical contents are pasted into the
1748     // text box.
1749     text_before_change_.clear();
1750     ReplaceSel(text.c_str(), true);
1751   }
1752 }
1753 
OnRButtonDblClk(UINT,const CPoint &)1754 void AutocompleteEditViewWin::OnRButtonDblClk(UINT /*keys*/,
1755                                               const CPoint& /*point*/) {
1756   gaining_focus_.reset();  // See NOTE in OnMouseActivate().
1757   SetMsgHandled(false);
1758 }
1759 
OnRButtonDown(UINT,const CPoint & point)1760 void AutocompleteEditViewWin::OnRButtonDown(UINT /*keys*/,
1761                                             const CPoint& point) {
1762   TrackMousePosition(kRight, point);
1763   tracking_double_click_ = false;
1764   possible_drag_ = false;
1765   gaining_focus_.reset();
1766   SetMsgHandled(false);
1767 }
1768 
OnRButtonUp(UINT,const CPoint & point)1769 void AutocompleteEditViewWin::OnRButtonUp(UINT /*keys*/, const CPoint& point) {
1770   SelectAllIfNecessary(kRight, point);
1771   tracking_click_[kRight] = false;
1772   SetMsgHandled(false);
1773 }
1774 
OnSetFocus(HWND focus_wnd)1775 void AutocompleteEditViewWin::OnSetFocus(HWND focus_wnd) {
1776   views::FocusManager* focus_manager = parent_view_->GetFocusManager();
1777   if (focus_manager) {
1778     // Notify the FocusManager that the focused view is now the location bar
1779     // (our parent view).
1780     focus_manager->SetFocusedView(parent_view_);
1781   } else {
1782     NOTREACHED();
1783   }
1784 
1785   model_->OnSetFocus(GetKeyState(VK_CONTROL) < 0);
1786 
1787   // Restore saved selection if available.
1788   if (saved_selection_for_focus_change_.cpMin != -1) {
1789     SetSelectionRange(saved_selection_for_focus_change_);
1790     saved_selection_for_focus_change_.cpMin = -1;
1791   }
1792 
1793   SetMsgHandled(false);
1794 }
1795 
OnSetText(const wchar_t * text)1796 LRESULT AutocompleteEditViewWin::OnSetText(const wchar_t* text) {
1797   // Ignore all IME messages while we process this WM_SETTEXT message.
1798   // When SetWindowText() is called while an IME is composing text, the IME
1799   // calls SendMessage() to send a WM_IME_COMPOSITION message. When we receive
1800   // this WM_IME_COMPOSITION message, we update the omnibox and may call
1801   // SetWindowText() again. To stop this recursive message-handler call, we
1802   // stop updating the omnibox while we process a WM_SETTEXT message.
1803   // We wouldn't need to do this update anyway, because either we're in the
1804   // middle of updating the omnibox already or the caller of SetWindowText()
1805   // is going to update the omnibox next.
1806   AutoReset<bool> auto_reset_ignore_ime_messages(&ignore_ime_messages_, true);
1807   return DefWindowProc(WM_SETTEXT, 0, reinterpret_cast<LPARAM>(text));
1808 }
1809 
OnSysChar(TCHAR ch,UINT repeat_count,UINT flags)1810 void AutocompleteEditViewWin::OnSysChar(TCHAR ch,
1811                                         UINT repeat_count,
1812                                         UINT flags) {
1813   // Nearly all alt-<xxx> combos result in beeping rather than doing something
1814   // useful, so we discard most.  Exceptions:
1815   //   * ctrl-alt-<xxx>, which is sometimes important, generates WM_CHAR instead
1816   //     of WM_SYSCHAR, so it doesn't need to be handled here.
1817   //   * alt-space gets translated by the default WM_SYSCHAR handler to a
1818   //     WM_SYSCOMMAND to open the application context menu, so we need to allow
1819   //     it through.
1820   if (ch == VK_SPACE)
1821     SetMsgHandled(false);
1822 }
1823 
OnWindowPosChanging(WINDOWPOS * window_pos)1824 void AutocompleteEditViewWin::OnWindowPosChanging(WINDOWPOS* window_pos) {
1825   if (force_hidden_)
1826     window_pos->flags &= ~SWP_SHOWWINDOW;
1827   SetMsgHandled(true);
1828 }
1829 
OnMouseWheel(UINT flags,short delta,CPoint point)1830 BOOL AutocompleteEditViewWin::OnMouseWheel(UINT flags,
1831                                            short delta,
1832                                            CPoint point) {
1833   // Forward the mouse-wheel message to the window under the mouse.
1834   if (!views::RerouteMouseWheel(m_hWnd, MAKEWPARAM(flags, delta),
1835                                 MAKELPARAM(point.x, point.y)))
1836     SetMsgHandled(false);
1837   return 0;
1838 }
1839 
HandleKeystroke(UINT message,TCHAR key,UINT repeat_count,UINT flags)1840 void AutocompleteEditViewWin::HandleKeystroke(UINT message,
1841                                               TCHAR key,
1842                                               UINT repeat_count,
1843                                               UINT flags) {
1844   ScopedFreeze freeze(this, GetTextObjectModel());
1845   OnBeforePossibleChange();
1846 
1847   if (key == ui::VKEY_HOME || key == ui::VKEY_END) {
1848     // DefWindowProc() might reset the keyboard layout when it receives a
1849     // keydown event for VKEY_HOME or VKEY_END. When the window was created
1850     // with WS_EX_LAYOUTRTL and the current keyboard layout is not a RTL one,
1851     // if the input text is pure LTR text, the layout changes to the first RTL
1852     // keyboard layout in keyboard layout queue; if the input text is
1853     // bidirectional text, the layout changes to the keyboard layout of the
1854     // first RTL character in input text. When the window was created without
1855     // WS_EX_LAYOUTRTL and the current keyboard layout is not a LTR one, if the
1856     // input text is pure RTL text, the layout changes to English; if the input
1857     // text is bidirectional text, the layout changes to the keyboard layout of
1858     // the first LTR character in input text. Such keyboard layout change
1859     // behavior is surprising and inconsistent with keyboard behavior
1860     // elsewhere, so reset the layout in this case.
1861     HKL layout = GetKeyboardLayout(0);
1862     DefWindowProc(message, key, MAKELPARAM(repeat_count, flags));
1863     ActivateKeyboardLayout(layout, KLF_REORDER);
1864   } else {
1865     DefWindowProc(message, key, MAKELPARAM(repeat_count, flags));
1866   }
1867 
1868   // CRichEditCtrl automatically turns on IMF_AUTOKEYBOARD when the user
1869   // inputs an RTL character, making it difficult for the user to control
1870   // what language is set as they type. Force this off to make the edit's
1871   // behavior more stable.
1872   const int lang_options = SendMessage(EM_GETLANGOPTIONS, 0, 0);
1873   if (lang_options & IMF_AUTOKEYBOARD)
1874     SendMessage(EM_SETLANGOPTIONS, 0, lang_options & ~IMF_AUTOKEYBOARD);
1875 
1876   OnAfterPossibleChange();
1877 }
1878 
OnKeyDownOnlyWritable(TCHAR key,UINT repeat_count,UINT flags)1879 bool AutocompleteEditViewWin::OnKeyDownOnlyWritable(TCHAR key,
1880                                                     UINT repeat_count,
1881                                                     UINT flags) {
1882   // NOTE: Annoyingly, ctrl-alt-<key> generates WM_KEYDOWN rather than
1883   // WM_SYSKEYDOWN, so we need to check (flags & KF_ALTDOWN) in various places
1884   // in this function even with a WM_SYSKEYDOWN handler.
1885 
1886   // If adding a new key that could possibly be an accelerator then you'll need
1887   // to update LocationBarView::SkipDefaultKeyEventProcessing() as well.
1888   int count = repeat_count;
1889   switch (key) {
1890     case VK_RIGHT:
1891       // TODO(sky): figure out RTL.
1892       if (base::i18n::IsRTL())
1893         return false;
1894       {
1895         CHARRANGE selection;
1896         GetSel(selection);
1897         return (selection.cpMin == selection.cpMax) &&
1898             (selection.cpMin == GetTextLength()) &&
1899             model_->CommitSuggestedText(true);
1900       }
1901 
1902     case VK_RETURN:
1903       model_->AcceptInput((flags & KF_ALTDOWN) ?
1904           NEW_FOREGROUND_TAB : CURRENT_TAB, false);
1905       return true;
1906 
1907     case VK_PRIOR:
1908     case VK_NEXT:
1909       count = model_->result().size();
1910       // FALL THROUGH
1911     case VK_UP:
1912     case VK_DOWN:
1913       // Ignore alt + numpad, but treat alt + (non-numpad) like (non-numpad).
1914       if ((flags & KF_ALTDOWN) && !(flags & KF_EXTENDED))
1915         return false;
1916 
1917       model_->OnUpOrDownKeyPressed(((key == VK_PRIOR) || (key == VK_UP)) ?
1918           -count : count);
1919       return true;
1920 
1921     // Hijacking Editing Commands
1922     //
1923     // We hijack the keyboard short-cuts for Cut, Copy, and Paste here so that
1924     // they go through our clipboard routines.  This allows us to be smarter
1925     // about how we interact with the clipboard and avoid bugs in the
1926     // CRichEditCtrl.  If we didn't hijack here, the edit control would handle
1927     // these internally with sending the WM_CUT, WM_COPY, or WM_PASTE messages.
1928     //
1929     // Cut:   Shift-Delete and Ctrl-x are treated as cut.  Ctrl-Shift-Delete and
1930     //        Ctrl-Shift-x are not treated as cut even though the underlying
1931     //        CRichTextEdit would treat them as such.  Also note that we bring
1932     //        up 'clear browsing data' on control-shift-delete.
1933     // Copy:  Ctrl-Insert and Ctrl-c is treated as copy.  Shift-Ctrl-c is not.
1934     //        (This is handled in OnKeyDownAllModes().)
1935     // Paste: Shift-Insert and Ctrl-v are treated as paste.  Ctrl-Shift-Insert
1936     //        and Ctrl-Shift-v are not.
1937     //
1938     // This behavior matches most, but not all Windows programs, and largely
1939     // conforms to what users expect.
1940 
1941     case VK_DELETE:
1942       if (flags & KF_ALTDOWN)
1943         return false;
1944       if (GetKeyState(VK_SHIFT) >= 0) {
1945         if (GetKeyState(VK_CONTROL) >= 0) {
1946           CHARRANGE selection;
1947           GetSel(selection);
1948           delete_at_end_pressed_ = ((selection.cpMin == selection.cpMax) &&
1949                                     (selection.cpMin == GetTextLength()));
1950         }
1951         return false;
1952       }
1953       if (GetKeyState(VK_CONTROL) >= 0) {
1954         // Cut text if possible.
1955         CHARRANGE selection;
1956         GetSel(selection);
1957         if (selection.cpMin != selection.cpMax) {
1958           ScopedFreeze freeze(this, GetTextObjectModel());
1959           OnBeforePossibleChange();
1960           Cut();
1961           OnAfterPossibleChange();
1962         } else {
1963           if (model_->popup_model()->IsOpen()) {
1964             // This is a bit overloaded, but we hijack Shift-Delete in this
1965             // case to delete the current item from the pop-up.  We prefer
1966             // cutting to this when possible since that's the behavior more
1967             // people expect from Shift-Delete, and it's more commonly useful.
1968             model_->popup_model()->TryDeletingCurrentItem();
1969           }
1970         }
1971       }
1972       return true;
1973 
1974     case 'X':
1975       if ((flags & KF_ALTDOWN) || (GetKeyState(VK_CONTROL) >= 0))
1976         return false;
1977       if (GetKeyState(VK_SHIFT) >= 0) {
1978         ScopedFreeze freeze(this, GetTextObjectModel());
1979         OnBeforePossibleChange();
1980         Cut();
1981         OnAfterPossibleChange();
1982       }
1983       return true;
1984 
1985     case VK_INSERT:
1986       // Ignore insert by itself, so we don't turn overtype mode on/off.
1987       if (!(flags & KF_ALTDOWN) && (GetKeyState(VK_SHIFT) >= 0) &&
1988           (GetKeyState(VK_CONTROL) >= 0))
1989         return true;
1990       // FALL THROUGH
1991     case 'V':
1992       if ((flags & KF_ALTDOWN) ||
1993           (GetKeyState((key == 'V') ? VK_CONTROL : VK_SHIFT) >= 0))
1994         return false;
1995       if (GetKeyState((key == 'V') ? VK_SHIFT : VK_CONTROL) >= 0) {
1996         ScopedFreeze freeze(this, GetTextObjectModel());
1997         OnBeforePossibleChange();
1998         Paste();
1999         OnAfterPossibleChange();
2000       }
2001       return true;
2002 
2003     case VK_BACK: {
2004       if ((flags & KF_ALTDOWN) || model_->is_keyword_hint() ||
2005           model_->keyword().empty())
2006         return false;
2007 
2008       {
2009         CHARRANGE selection;
2010         GetSel(selection);
2011         if ((selection.cpMin != selection.cpMax) || (selection.cpMin != 0))
2012           return false;
2013       }
2014 
2015       // We're showing a keyword and the user pressed backspace at the beginning
2016       // of the text. Delete the selected keyword.
2017       ScopedFreeze freeze(this, GetTextObjectModel());
2018       model_->ClearKeyword(GetText());
2019       return true;
2020     }
2021 
2022     case VK_TAB: {
2023       if (model_->is_keyword_hint()) {
2024         // Accept the keyword.
2025         ScopedFreeze freeze(this, GetTextObjectModel());
2026         model_->AcceptKeyword();
2027       } else if (!IsCaretAtEnd()) {
2028         ScopedFreeze freeze(this, GetTextObjectModel());
2029         OnBeforePossibleChange();
2030         PlaceCaretAt(GetTextLength());
2031         OnAfterPossibleChange();
2032       } else {
2033         model_->CommitSuggestedText(true);
2034       }
2035       return true;
2036     }
2037 
2038     case 0xbb:  // Ctrl-'='.  Triggers subscripting (even in plain text mode).
2039                 // We don't use VK_OEM_PLUS in case the macro isn't defined.
2040                 // (e.g., we don't have this symbol in embeded environment).
2041       return true;
2042 
2043     default:
2044       return false;
2045   }
2046 }
2047 
OnKeyDownAllModes(TCHAR key,UINT repeat_count,UINT flags)2048 bool AutocompleteEditViewWin::OnKeyDownAllModes(TCHAR key,
2049                                                 UINT repeat_count,
2050                                                 UINT flags) {
2051   // See KF_ALTDOWN comment atop OnKeyDownOnlyWritable().
2052 
2053   switch (key) {
2054     case VK_CONTROL:
2055       model_->OnControlKeyChanged(true);
2056       return false;
2057 
2058     case VK_INSERT:
2059     case 'C':
2060       // See more detailed comments in OnKeyDownOnlyWritable().
2061       if ((flags & KF_ALTDOWN) || (GetKeyState(VK_CONTROL) >= 0))
2062         return false;
2063       if (GetKeyState(VK_SHIFT) >= 0)
2064         Copy();
2065       return true;
2066 
2067     default:
2068       return false;
2069   }
2070 }
2071 
GetSelection(CHARRANGE & sel) const2072 void AutocompleteEditViewWin::GetSelection(CHARRANGE& sel) const {
2073   GetSel(sel);
2074 
2075   // See if we need to reverse the direction of the selection.
2076   ITextDocument* const text_object_model = GetTextObjectModel();
2077   if (!text_object_model)
2078     return;
2079   base::win::ScopedComPtr<ITextSelection> selection;
2080   const HRESULT hr = text_object_model->GetSelection(selection.Receive());
2081   DCHECK_EQ(S_OK, hr);
2082   long flags;
2083   selection->GetFlags(&flags);
2084   if (flags & tomSelStartActive)
2085     std::swap(sel.cpMin, sel.cpMax);
2086 }
2087 
GetSelectedText() const2088 string16 AutocompleteEditViewWin::GetSelectedText() const {
2089   // Figure out the length of the selection.
2090   CHARRANGE sel;
2091   GetSel(sel);
2092 
2093   // Grab the selected text.
2094   string16 str;
2095   GetSelText(WriteInto(&str, sel.cpMax - sel.cpMin + 1));
2096   return str;
2097 }
2098 
SetSelection(LONG start,LONG end)2099 void AutocompleteEditViewWin::SetSelection(LONG start, LONG end) {
2100   SetSel(start, end);
2101 
2102   if (start <= end)
2103     return;
2104 
2105   // We need to reverse the direction of the selection.
2106   ITextDocument* const text_object_model = GetTextObjectModel();
2107   if (!text_object_model)
2108     return;
2109   base::win::ScopedComPtr<ITextSelection> selection;
2110   const HRESULT hr = text_object_model->GetSelection(selection.Receive());
2111   DCHECK_EQ(S_OK, hr);
2112   selection->SetFlags(tomSelStartActive);
2113 }
2114 
PlaceCaretAt(string16::size_type pos)2115 void AutocompleteEditViewWin::PlaceCaretAt(string16::size_type pos) {
2116   SetSelection(static_cast<LONG>(pos), static_cast<LONG>(pos));
2117 }
2118 
IsSelectAllForRange(const CHARRANGE & sel) const2119 bool AutocompleteEditViewWin::IsSelectAllForRange(const CHARRANGE& sel) const {
2120   const int text_length = GetTextLength();
2121   return ((sel.cpMin == 0) && (sel.cpMax >= text_length)) ||
2122       ((sel.cpMax == 0) && (sel.cpMin >= text_length));
2123 }
2124 
ClipXCoordToVisibleText(LONG x,bool is_triple_click) const2125 LONG AutocompleteEditViewWin::ClipXCoordToVisibleText(
2126     LONG x, bool is_triple_click) const {
2127   // Clip the X coordinate to the left edge of the text. Careful:
2128   // PosFromChar(0) may return a negative X coordinate if the beginning of the
2129   // text has scrolled off the edit, so don't go past the clip rect's edge.
2130   PARAFORMAT2 pf2;
2131   GetParaFormat(pf2);
2132   // Calculation of the clipped coordinate is more complicated if the paragraph
2133   // layout is RTL layout, or if there is RTL characters inside the LTR layout
2134   // paragraph.
2135   const bool ltr_text_in_ltr_layout = !(pf2.wEffects & PFE_RTLPARA) &&
2136       !base::i18n::StringContainsStrongRTLChars(GetText());
2137   const int length = GetTextLength();
2138   RECT r;
2139   GetRect(&r);
2140   // The values returned by PosFromChar() seem to refer always
2141   // to the left edge of the character's bounding box.
2142   const LONG first_position_x = PosFromChar(0).x;
2143   LONG min_x = first_position_x;
2144   if (!ltr_text_in_ltr_layout) {
2145     for (int i = 1; i < length; ++i)
2146       min_x = std::min(min_x, PosFromChar(i).x);
2147   }
2148   const LONG left_bound = std::max(r.left, min_x);
2149   // PosFromChar(length) is a phantom character past the end of the text. It is
2150   // not necessarily a right bound; in RTL controls it may be a left bound. So
2151   // treat it as a right bound only if it is to the right of the first
2152   // character.
2153   LONG right_bound = r.right;
2154   LONG end_position_x = PosFromChar(length).x;
2155   if (end_position_x >= first_position_x) {
2156     right_bound = std::min(right_bound, end_position_x);  // LTR case.
2157   }
2158   // For trailing characters that are 2 pixels wide or less (like "l" in some
2159   // fonts), we have a problem:
2160   //   * Clicks on any pixel within the character will place the cursor before
2161   //     the character.
2162   //   * Clicks on the pixel just after the character will not allow triple-
2163   //     click to work properly (true for any last character width).
2164   // So, we move to the last pixel of the character when this is a
2165   // triple-click, and moving to one past the last pixel in all other
2166   // scenarios.  This way, all clicks that can move the cursor will place it at
2167   // the end of the text, but triple-click will still work.
2168   if (x < left_bound) {
2169     return (is_triple_click && ltr_text_in_ltr_layout) ? left_bound - 1 :
2170                                                          left_bound;
2171   }
2172   if ((length == 0) || (x < right_bound))
2173     return x;
2174   return is_triple_click ? (right_bound - 1) : right_bound;
2175 }
2176 
EmphasizeURLComponents()2177 void AutocompleteEditViewWin::EmphasizeURLComponents() {
2178   ITextDocument* const text_object_model = GetTextObjectModel();
2179   ScopedFreeze freeze(this, text_object_model);
2180   ScopedSuspendUndo suspend_undo(text_object_model);
2181 
2182   // Save the selection.
2183   CHARRANGE saved_sel;
2184   GetSelection(saved_sel);
2185 
2186   // See whether the contents are a URL with a non-empty host portion, which we
2187   // should emphasize.  To check for a URL, rather than using the type returned
2188   // by Parse(), ask the model, which will check the desired page transition for
2189   // this input.  This can tell us whether an UNKNOWN input string is going to
2190   // be treated as a search or a navigation, and is the same method the Paste
2191   // And Go system uses.
2192   url_parse::Component scheme, host;
2193   AutocompleteInput::ParseForEmphasizeComponents(
2194       GetText(), model_->GetDesiredTLD(), &scheme, &host);
2195   const bool emphasize = model_->CurrentTextIsURL() && (host.len > 0);
2196 
2197   // Set the baseline emphasis.
2198   CHARFORMAT cf = {0};
2199   cf.dwMask = CFM_COLOR;
2200   // If we're going to emphasize parts of the text, then the baseline state
2201   // should be "de-emphasized".  If not, then everything should be rendered in
2202   // the standard text color.
2203   cf.crTextColor = skia::SkColorToCOLORREF(LocationBarView::GetColor(
2204       security_level_,
2205       emphasize ? LocationBarView::DEEMPHASIZED_TEXT : LocationBarView::TEXT));
2206   // NOTE: Don't use SetDefaultCharFormat() instead of the below; that sets the
2207   // format that will get applied to text added in the future, not to text
2208   // already in the edit.
2209   SelectAll(false);
2210   SetSelectionCharFormat(cf);
2211 
2212   if (emphasize) {
2213     // We've found a host name, give it more emphasis.
2214     cf.crTextColor = skia::SkColorToCOLORREF(LocationBarView::GetColor(
2215         security_level_, LocationBarView::TEXT));
2216     SetSelection(host.begin, host.end());
2217     SetSelectionCharFormat(cf);
2218   }
2219 
2220   // Emphasize the scheme for security UI display purposes (if necessary).
2221   insecure_scheme_component_.reset();
2222   if (!model_->user_input_in_progress() && scheme.is_nonempty() &&
2223       (security_level_ != ToolbarModel::NONE)) {
2224     if (security_level_ == ToolbarModel::SECURITY_ERROR) {
2225       insecure_scheme_component_.begin = scheme.begin;
2226       insecure_scheme_component_.len = scheme.len;
2227     }
2228     cf.crTextColor = skia::SkColorToCOLORREF(LocationBarView::GetColor(
2229         security_level_, LocationBarView::SECURITY_TEXT));
2230     SetSelection(scheme.begin, scheme.end());
2231     SetSelectionCharFormat(cf);
2232   }
2233 
2234   // Restore the selection.
2235   SetSelectionRange(saved_sel);
2236 }
2237 
EraseTopOfSelection(CDC * dc,const CRect & client_rect,const CRect & paint_clip_rect)2238 void AutocompleteEditViewWin::EraseTopOfSelection(
2239     CDC* dc, const CRect& client_rect, const CRect& paint_clip_rect) {
2240   // Find the area we care about painting.   We could calculate the rect
2241   // containing just the selected portion, but there's no harm in simply erasing
2242   // the whole top of the client area, and at least once I saw us manage to
2243   // select the "phantom newline" briefly, which looks very weird if not clipped
2244   // off at the same height.
2245   CRect erase_rect(client_rect.left, client_rect.top, client_rect.right,
2246                    client_rect.top + font_y_adjustment_);
2247   erase_rect.IntersectRect(erase_rect, paint_clip_rect);
2248 
2249   // Erase to the background color.
2250   if (!erase_rect.IsRectNull())
2251     dc->FillSolidRect(&erase_rect, background_color_);
2252 }
2253 
DrawSlashForInsecureScheme(HDC hdc,const CRect & client_rect,const CRect & paint_clip_rect)2254 void AutocompleteEditViewWin::DrawSlashForInsecureScheme(
2255     HDC hdc,
2256     const CRect& client_rect,
2257     const CRect& paint_clip_rect) {
2258   DCHECK(insecure_scheme_component_.is_nonempty());
2259 
2260   // Calculate the rect, in window coordinates, containing the portion of the
2261   // scheme where we'll be drawing the slash.  Vertically, we draw across one
2262   // x-height of text, plus an additional 3 stroke diameters (the stroke width
2263   // plus a half-stroke width of space between the stroke and the text, both
2264   // above and below the text).
2265   const int font_top = client_rect.top + font_y_adjustment_;
2266   const SkScalar kStrokeWidthPixels = SkIntToScalar(2);
2267   const int kAdditionalSpaceOutsideFont =
2268       static_cast<int>(ceil(kStrokeWidthPixels * 1.5f));
2269   const CRect scheme_rect(PosFromChar(insecure_scheme_component_.begin).x,
2270                           font_top + font_.GetBaseline() - font_x_height_ -
2271                               kAdditionalSpaceOutsideFont,
2272                           PosFromChar(insecure_scheme_component_.end()).x,
2273                           font_top + font_.GetBaseline() +
2274                               kAdditionalSpaceOutsideFont);
2275 
2276   // Clip to the portion we care about and translate to canvas coordinates
2277   // (see the canvas creation below) for use later.
2278   CRect canvas_clip_rect, canvas_paint_clip_rect;
2279   canvas_clip_rect.IntersectRect(scheme_rect, client_rect);
2280   canvas_paint_clip_rect.IntersectRect(canvas_clip_rect, paint_clip_rect);
2281   if (canvas_paint_clip_rect.IsRectNull())
2282     return;  // We don't need to paint any of this region, so just bail early.
2283   canvas_clip_rect.OffsetRect(-scheme_rect.left, -scheme_rect.top);
2284   canvas_paint_clip_rect.OffsetRect(-scheme_rect.left, -scheme_rect.top);
2285 
2286   // Create a paint context for drawing the antialiased stroke.
2287   SkPaint paint;
2288   paint.setAntiAlias(true);
2289   paint.setStrokeWidth(kStrokeWidthPixels);
2290   paint.setStrokeCap(SkPaint::kRound_Cap);
2291 
2292   // Create a canvas as large as |scheme_rect| to do our drawing, and initialize
2293   // it to fully transparent so any antialiasing will look nice when painted
2294   // atop the edit.
2295   gfx::CanvasSkia canvas(scheme_rect.Width(), scheme_rect.Height(), false);
2296   canvas.getDevice()->accessBitmap(true).eraseARGB(0, 0, 0, 0);
2297 
2298   // Calculate the start and end of the stroke, which are just the lower left
2299   // and upper right corners of the canvas, inset by the radius of the endcap
2300   // so we don't clip the endcap off.
2301   const SkScalar kEndCapRadiusPixels = kStrokeWidthPixels / SkIntToScalar(2);
2302   const SkPoint start_point = {
2303       kEndCapRadiusPixels,
2304       SkIntToScalar(scheme_rect.Height()) - kEndCapRadiusPixels };
2305   const SkPoint end_point = {
2306       SkIntToScalar(scheme_rect.Width()) - kEndCapRadiusPixels,
2307       kEndCapRadiusPixels };
2308 
2309   // Calculate the selection rectangle in canvas coordinates, which we'll use
2310   // to clip the stroke so we can draw the unselected and selected portions.
2311   CHARRANGE sel;
2312   GetSel(sel);
2313   const SkRect selection_rect = {
2314       SkIntToScalar(PosFromChar(sel.cpMin).x - scheme_rect.left),
2315       SkIntToScalar(0),
2316       SkIntToScalar(PosFromChar(sel.cpMax).x - scheme_rect.left),
2317       SkIntToScalar(scheme_rect.Height()) };
2318 
2319   // Draw the unselected portion of the stroke.
2320   canvas.save();
2321   if (selection_rect.isEmpty() ||
2322       canvas.clipRect(selection_rect, SkRegion::kDifference_Op)) {
2323     paint.setColor(LocationBarView::GetColor(security_level_,
2324                                              LocationBarView::SECURITY_TEXT));
2325     canvas.drawLine(start_point.fX, start_point.fY,
2326                     end_point.fX, end_point.fY, paint);
2327   }
2328   canvas.restore();
2329 
2330   // Draw the selected portion of the stroke.
2331   if (!selection_rect.isEmpty() && canvas.clipRect(selection_rect)) {
2332     paint.setColor(LocationBarView::GetColor(security_level_,
2333                                              LocationBarView::SELECTED_TEXT));
2334     canvas.drawLine(start_point.fX, start_point.fY,
2335                     end_point.fX, end_point.fY, paint);
2336   }
2337 
2338   // Now copy what we drew to the target HDC.
2339   canvas.getTopPlatformDevice().drawToHDC(hdc,
2340       scheme_rect.left + canvas_paint_clip_rect.left - canvas_clip_rect.left,
2341       std::max(scheme_rect.top, client_rect.top) + canvas_paint_clip_rect.top -
2342           canvas_clip_rect.top, &canvas_paint_clip_rect);
2343 }
2344 
DrawDropHighlight(HDC hdc,const CRect & client_rect,const CRect & paint_clip_rect)2345 void AutocompleteEditViewWin::DrawDropHighlight(HDC hdc,
2346                                                 const CRect& client_rect,
2347                                                 const CRect& paint_clip_rect) {
2348   DCHECK_NE(-1, drop_highlight_position_);
2349 
2350   const int highlight_y = client_rect.top + font_y_adjustment_;
2351   const int highlight_x = PosFromChar(drop_highlight_position_).x - 1;
2352   const CRect highlight_rect(highlight_x,
2353                              highlight_y,
2354                              highlight_x + 1,
2355                              highlight_y + font_.GetHeight());
2356 
2357   // Clip the highlight to the region being painted.
2358   CRect clip_rect;
2359   clip_rect.IntersectRect(highlight_rect, paint_clip_rect);
2360   if (clip_rect.IsRectNull())
2361     return;
2362 
2363   HGDIOBJ last_pen = SelectObject(hdc, CreatePen(PS_SOLID, 1, RGB(0, 0, 0)));
2364   MoveToEx(hdc, clip_rect.left, clip_rect.top, NULL);
2365   LineTo(hdc, clip_rect.left, clip_rect.bottom);
2366   DeleteObject(SelectObject(hdc, last_pen));
2367 }
2368 
TextChanged()2369 void AutocompleteEditViewWin::TextChanged() {
2370   ScopedFreeze freeze(this, GetTextObjectModel());
2371   EmphasizeURLComponents();
2372   model_->OnChanged();
2373 }
2374 
GetClipboardText() const2375 string16 AutocompleteEditViewWin::GetClipboardText() const {
2376   // Try text format.
2377   ui::Clipboard* clipboard = g_browser_process->clipboard();
2378   if (clipboard->IsFormatAvailable(ui::Clipboard::GetPlainTextWFormatType(),
2379                                    ui::Clipboard::BUFFER_STANDARD)) {
2380     string16 text;
2381     clipboard->ReadText(ui::Clipboard::BUFFER_STANDARD, &text);
2382 
2383     // Note: Unlike in the find popup and textfield view, here we completely
2384     // remove whitespace strings containing newlines.  We assume users are
2385     // most likely pasting in URLs that may have been split into multiple
2386     // lines in terminals, email programs, etc., and so linebreaks indicate
2387     // completely bogus whitespace that would just cause the input to be
2388     // invalid.
2389     return CollapseWhitespace(text, true);
2390   }
2391 
2392   // Try bookmark format.
2393   //
2394   // It is tempting to try bookmark format first, but the URL we get out of a
2395   // bookmark has been cannonicalized via GURL.  This means if a user copies
2396   // and pastes from the URL bar to itself, the text will get fixed up and
2397   // cannonicalized, which is not what the user expects.  By pasting in this
2398   // order, we are sure to paste what the user copied.
2399   if (clipboard->IsFormatAvailable(ui::Clipboard::GetUrlWFormatType(),
2400                                    ui::Clipboard::BUFFER_STANDARD)) {
2401     std::string url_str;
2402     clipboard->ReadBookmark(NULL, &url_str);
2403     // pass resulting url string through GURL to normalize
2404     GURL url(url_str);
2405     if (url.is_valid())
2406       return UTF8ToWide(url.spec());
2407   }
2408 
2409   return string16();
2410 }
2411 
CanPasteAndGo(const string16 & text) const2412 bool AutocompleteEditViewWin::CanPasteAndGo(const string16& text) const {
2413   return !popup_window_mode_ && model_->CanPasteAndGo(text);
2414 }
2415 
GetTextObjectModel() const2416 ITextDocument* AutocompleteEditViewWin::GetTextObjectModel() const {
2417   if (!text_object_model_) {
2418     // This is lazily initialized, instead of being initialized in the
2419     // constructor, in order to avoid hurting startup performance.
2420     base::win::ScopedComPtr<IRichEditOle, NULL> ole_interface;
2421     ole_interface.Attach(GetOleInterface());
2422     if (ole_interface) {
2423       ole_interface.QueryInterface(
2424           __uuidof(ITextDocument),
2425           reinterpret_cast<void**>(&text_object_model_));
2426     }
2427   }
2428   return text_object_model_;
2429 }
2430 
StartDragIfNecessary(const CPoint & point)2431 void AutocompleteEditViewWin::StartDragIfNecessary(const CPoint& point) {
2432   if (initiated_drag_ || !IsDrag(click_point_[kLeft], point))
2433     return;
2434 
2435   ui::OSExchangeData data;
2436 
2437   DWORD supported_modes = DROPEFFECT_COPY;
2438 
2439   CHARRANGE sel;
2440   GetSelection(sel);
2441 
2442   // We're about to start a drag session, but the edit is expecting a mouse up
2443   // that it uses to reset internal state.  If we don't send a mouse up now,
2444   // when the mouse moves back into the edit the edit will reset the selection.
2445   // So, we send the event now which resets the selection.  We then restore the
2446   // selection and start the drag.  We always send lbuttonup as otherwise we
2447   // might trigger a context menu (right up).  This seems scary, but doesn't
2448   // seem to cause problems.
2449   {
2450     ScopedFreeze freeze(this, GetTextObjectModel());
2451     DefWindowProc(WM_LBUTTONUP, 0,
2452                   MAKELPARAM(click_point_[kLeft].x, click_point_[kLeft].y));
2453     SetSelectionRange(sel);
2454   }
2455 
2456   const string16 start_text(GetText());
2457   string16 text_to_write(GetSelectedText());
2458   GURL url;
2459   bool write_url;
2460   const bool is_all_selected = IsSelectAllForRange(sel);
2461 
2462   // |sel| was set by GetSelection(), which preserves selection direction, so
2463   // sel.cpMin may not be the smaller value.
2464   model()->AdjustTextForCopy(std::min(sel.cpMin, sel.cpMax), is_all_selected,
2465                              &text_to_write, &url, &write_url);
2466 
2467   if (write_url) {
2468     string16 title;
2469     SkBitmap favicon;
2470     if (is_all_selected)
2471       model_->GetDataForURLExport(&url, &title, &favicon);
2472     drag_utils::SetURLAndDragImage(url, title, favicon, &data);
2473     supported_modes |= DROPEFFECT_LINK;
2474     UserMetrics::RecordAction(UserMetricsAction("Omnibox_DragURL"),
2475                               model_->profile());
2476   } else {
2477     supported_modes |= DROPEFFECT_MOVE;
2478     UserMetrics::RecordAction(UserMetricsAction("Omnibox_DragString"),
2479                               model_->profile());
2480   }
2481 
2482   data.SetString(text_to_write);
2483 
2484   scoped_refptr<ui::DragSource> drag_source(new ui::DragSource);
2485   DWORD dropped_mode;
2486   AutoReset<bool> auto_reset_in_drag(&in_drag_, true);
2487   if (DoDragDrop(ui::OSExchangeDataProviderWin::GetIDataObject(data),
2488                  drag_source, supported_modes, &dropped_mode) ==
2489           DRAGDROP_S_DROP) {
2490     if ((dropped_mode == DROPEFFECT_MOVE) && (start_text == GetText())) {
2491       ScopedFreeze freeze(this, GetTextObjectModel());
2492       OnBeforePossibleChange();
2493       SetSelectionRange(sel);
2494       ReplaceSel(L"", true);
2495       OnAfterPossibleChange();
2496     }
2497     // else case, not a move or it was a move and the drop was on us.
2498     // If the drop was on us, EditDropTarget took care of the move so that
2499     // we don't have to delete the text.
2500     possible_drag_ = false;
2501   } else {
2502     // Drag was canceled or failed. The mouse may still be down and
2503     // over us, in which case we need possible_drag_ to remain true so
2504     // that we don't forward mouse move events to the edit which will
2505     // start another drag.
2506     //
2507     // NOTE: we didn't use mouse capture during the mouse down as DoDragDrop
2508     // does its own capture.
2509     CPoint cursor_location;
2510     GetCursorPos(&cursor_location);
2511 
2512     CRect client_rect;
2513     GetClientRect(&client_rect);
2514 
2515     CPoint client_origin_on_screen(client_rect.left, client_rect.top);
2516     ClientToScreen(&client_origin_on_screen);
2517     client_rect.MoveToXY(client_origin_on_screen.x,
2518                          client_origin_on_screen.y);
2519     possible_drag_ = (client_rect.PtInRect(cursor_location) &&
2520                       ((GetKeyState(VK_LBUTTON) != 0) ||
2521                        (GetKeyState(VK_MBUTTON) != 0) ||
2522                        (GetKeyState(VK_RBUTTON) != 0)));
2523   }
2524 
2525   initiated_drag_ = true;
2526   tracking_click_[kLeft] = false;
2527 }
2528 
OnPossibleDrag(const CPoint & point)2529 void AutocompleteEditViewWin::OnPossibleDrag(const CPoint& point) {
2530   if (possible_drag_)
2531     return;
2532 
2533   click_point_[kLeft] = point;
2534   initiated_drag_ = false;
2535 
2536   CHARRANGE selection;
2537   GetSel(selection);
2538   if (selection.cpMin != selection.cpMax) {
2539     const POINT min_sel_location(PosFromChar(selection.cpMin));
2540     const POINT max_sel_location(PosFromChar(selection.cpMax));
2541     // NOTE: we don't consider the y location here as we always pass a
2542     // y-coordinate in the middle to the default handler which always triggers
2543     // a drag regardless of the y-coordinate.
2544     possible_drag_ = (point.x >= min_sel_location.x) &&
2545                      (point.x < max_sel_location.x);
2546   }
2547 }
2548 
RepaintDropHighlight(int position)2549 void AutocompleteEditViewWin::RepaintDropHighlight(int position) {
2550   if ((position != -1) && (position <= GetTextLength())) {
2551     const POINT min_loc(PosFromChar(position));
2552     const RECT highlight_bounds = {min_loc.x - 1, font_y_adjustment_,
2553         min_loc.x + 2, font_.GetHeight() + font_y_adjustment_};
2554     InvalidateRect(&highlight_bounds, false);
2555   }
2556 }
2557 
BuildContextMenu()2558 void AutocompleteEditViewWin::BuildContextMenu() {
2559   if (context_menu_contents_.get())
2560     return;
2561 
2562   context_menu_contents_.reset(new ui::SimpleMenuModel(this));
2563   // Set up context menu.
2564   if (popup_window_mode_) {
2565     context_menu_contents_->AddItemWithStringId(IDC_COPY, IDS_COPY);
2566   } else {
2567     context_menu_contents_->AddItemWithStringId(IDS_UNDO, IDS_UNDO);
2568     context_menu_contents_->AddSeparator();
2569     context_menu_contents_->AddItemWithStringId(IDC_CUT, IDS_CUT);
2570     context_menu_contents_->AddItemWithStringId(IDC_COPY, IDS_COPY);
2571     context_menu_contents_->AddItemWithStringId(IDC_PASTE, IDS_PASTE);
2572     // GetContextualLabel() will override this next label with the
2573     // IDS_PASTE_AND_SEARCH label as needed.
2574     context_menu_contents_->AddItemWithStringId(IDS_PASTE_AND_GO,
2575                                                 IDS_PASTE_AND_GO);
2576     context_menu_contents_->AddSeparator();
2577     context_menu_contents_->AddItemWithStringId(IDS_SELECT_ALL, IDS_SELECT_ALL);
2578     context_menu_contents_->AddSeparator();
2579     context_menu_contents_->AddItemWithStringId(IDS_EDIT_SEARCH_ENGINES,
2580                                                 IDS_EDIT_SEARCH_ENGINES);
2581   }
2582   context_menu_.reset(new views::Menu2(context_menu_contents_.get()));
2583 }
2584 
SelectAllIfNecessary(MouseButton button,const CPoint & point)2585 void AutocompleteEditViewWin::SelectAllIfNecessary(MouseButton button,
2586                                                    const CPoint& point) {
2587   // When the user has clicked and released to give us focus, select all.
2588   if (tracking_click_[button] &&
2589       !IsDrag(click_point_[button], point)) {
2590     // Select all in the reverse direction so as not to scroll the caret
2591     // into view and shift the contents jarringly.
2592     SelectAll(true);
2593     possible_drag_ = false;
2594   }
2595 }
2596 
TrackMousePosition(MouseButton button,const CPoint & point)2597 void AutocompleteEditViewWin::TrackMousePosition(MouseButton button,
2598                                                  const CPoint& point) {
2599   if (gaining_focus_.get()) {
2600     // This click is giving us focus, so we need to track how much the mouse
2601     // moves to see if it's a drag or just a click. Clicks should select all
2602     // the text.
2603     tracking_click_[button] = true;
2604     click_point_[button] = point;
2605   }
2606 }
2607 
GetHorizontalMargin() const2608 int AutocompleteEditViewWin::GetHorizontalMargin() const {
2609   RECT rect;
2610   GetRect(&rect);
2611   RECT client_rect;
2612   GetClientRect(&client_rect);
2613   return (rect.left - client_rect.left) + (client_rect.right - rect.right);
2614 }
2615 
WidthNeededToDisplay(const string16 & text) const2616 int AutocompleteEditViewWin::WidthNeededToDisplay(
2617     const string16& text) const {
2618   // Use font_.GetStringWidth() instead of
2619   // PosFromChar(location_entry_->GetTextLength()) because PosFromChar() is
2620   // apparently buggy. In both LTR UI and RTL UI with left-to-right layout,
2621   // PosFromChar(i) might return 0 when i is greater than 1.
2622   return font_.GetStringWidth(text) + GetHorizontalMargin();
2623 }
2624 
IsCaretAtEnd() const2625 bool AutocompleteEditViewWin::IsCaretAtEnd() const {
2626   long length = GetTextLength();
2627   CHARRANGE sel;
2628   GetSelection(sel);
2629   return sel.cpMin == sel.cpMax && sel.cpMin == length;
2630 }
2631