OLD | NEW |
(Empty) | |
| 1 // Copyright (c) 2009 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_mac.h" |
| 6 |
| 7 #include "base/sys_string_conversions.h" |
| 8 #include "chrome/browser/autocomplete/autocomplete_edit.h" |
| 9 #include "chrome/browser/autocomplete/autocomplete_popup_view_mac.h" |
| 10 |
| 11 // Thin Obj-C bridge class that is the delegate of the omnibox field. |
| 12 // It intercepts various control delegate methods and vectors them to |
| 13 // the edit view. |
| 14 |
| 15 @interface AutocompleteFieldDelegate : NSObject { |
| 16 @private |
| 17 AutocompleteEditViewMac* edit_view_; // weak, owns us. |
| 18 } |
| 19 - initWithEditView:(AutocompleteEditViewMac*)view; |
| 20 @end |
| 21 |
| 22 AutocompleteEditViewMac::AutocompleteEditViewMac( |
| 23 AutocompleteEditController* controller, |
| 24 ToolbarModel* toolbar_model, |
| 25 Profile* profile, |
| 26 CommandUpdater* command_updater) |
| 27 : model_(new AutocompleteEditModel(this, controller, profile)), |
| 28 popup_view_(new AutocompletePopupViewMac(this, model_.get(), profile)), |
| 29 controller_(controller), |
| 30 toolbar_model_(toolbar_model), |
| 31 command_updater_(command_updater), |
| 32 field_(nil), |
| 33 edit_helper_([[AutocompleteFieldDelegate alloc] initWithEditView:this]) { |
| 34 DCHECK(controller); |
| 35 DCHECK(toolbar_model); |
| 36 DCHECK(profile); |
| 37 DCHECK(command_updater); |
| 38 } |
| 39 |
| 40 AutocompleteEditViewMac::~AutocompleteEditViewMac() { |
| 41 // TODO(shess): Having to be aware of destructor ordering in this |
| 42 // way seems brittle. There must be a better way. |
| 43 |
| 44 // Destroy popup view before this object in case it tries to call us |
| 45 // back in the destructor. Likewise for destroying the model before |
| 46 // this object. |
| 47 popup_view_.reset(); |
| 48 model_.reset(); |
| 49 |
| 50 // Disconnect field_ from edit_helper_ so that we don't get calls |
| 51 // after destruction. |
| 52 [field_ setDelegate:nil]; |
| 53 } |
| 54 |
| 55 // TODO(shess): This is the minimal change which seems to unblock |
| 56 // getting the minimal Omnibox code checked in without making the |
| 57 // world worse. Browser::TabSelectedAt() calls this when the tab |
| 58 // changes, but that is only wired up for Windows. I do not yet |
| 59 // understand that code well enough to go for it. Once wired up, then |
| 60 // code can be removed at: |
| 61 // [TabContentsController defocusLocationBar] |
| 62 // [TabStripController selectTabWithContents:...] |
| 63 void AutocompleteEditViewMac::SaveStateToTab(TabContents* tab) { |
| 64 // TODO(shess): Actually save the state to the tab area. |
| 65 |
| 66 // Drop the popup before we change to another tab. |
| 67 ClosePopup(); |
| 68 } |
| 69 |
| 70 void AutocompleteEditViewMac::OpenURL(const GURL& url, |
| 71 WindowOpenDisposition disposition, |
| 72 PageTransition::Type transition, |
| 73 const GURL& alternate_nav_url, |
| 74 size_t selected_line, |
| 75 const std::wstring& keyword) { |
| 76 // TODO(shess): Why is the caller passing an invalid url in the |
| 77 // first place? Make sure that case isn't being dropped on the |
| 78 // floor. |
| 79 if (!url.is_valid()) { |
| 80 return; |
| 81 } |
| 82 |
| 83 model_->SendOpenNotification(selected_line, keyword); |
| 84 |
| 85 if (disposition != NEW_BACKGROUND_TAB) |
| 86 RevertAll(); // Revert the box to its unedited state. |
| 87 controller_->OnAutocompleteAccept(url, disposition, transition, |
| 88 alternate_nav_url); |
| 89 } |
| 90 |
| 91 std::wstring AutocompleteEditViewMac::GetText() const { |
| 92 return base::SysNSStringToWide([field_ stringValue]); |
| 93 } |
| 94 |
| 95 void AutocompleteEditViewMac::SetWindowTextAndCaretPos(const std::wstring& text, |
| 96 size_t caret_pos) { |
| 97 UpdateAndStyleText(text, text.size()); |
| 98 } |
| 99 |
| 100 void AutocompleteEditViewMac::SelectAll(bool reversed) { |
| 101 // TODO(shess): Figure out what reversed implies. The gtk version |
| 102 // has it imply inverting the selection front to back, but I don't |
| 103 // even know if that makes sense for Mac. |
| 104 UpdateAndStyleText(GetText(), 0); |
| 105 } |
| 106 |
| 107 void AutocompleteEditViewMac::RevertAll() { |
| 108 ClosePopup(); |
| 109 model_->Revert(); |
| 110 |
| 111 std::wstring tt = GetText(); |
| 112 UpdateAndStyleText(tt, tt.size()); |
| 113 controller_->OnChanged(); |
| 114 } |
| 115 |
| 116 void AutocompleteEditViewMac::UpdatePopup() { |
| 117 model_->SetInputInProgress(true); |
| 118 if (!model_->has_focus()) |
| 119 return; |
| 120 |
| 121 // TODO(shess): |
| 122 // Shouldn't inline autocomplete when the caret/selection isn't at |
| 123 // the end of the text. |
| 124 // |
| 125 // One option would seem to be to check for a non-nil field |
| 126 // editor, and check it's selected range against its length. |
| 127 model_->StartAutocomplete(false); |
| 128 } |
| 129 |
| 130 void AutocompleteEditViewMac::ClosePopup() { |
| 131 popup_view_->StopAutocomplete(); |
| 132 } |
| 133 |
| 134 void AutocompleteEditViewMac::UpdateAndStyleText( |
| 135 const std::wstring& display_text, size_t user_text_length) { |
| 136 NSString* ss = base::SysWideToNSString(display_text); |
| 137 NSMutableAttributedString* as = |
| 138 [[[NSMutableAttributedString alloc] initWithString:ss] autorelease]; |
| 139 |
| 140 url_parse::Parsed parts; |
| 141 AutocompleteInput::Parse(display_text, model_->GetDesiredTLD(), |
| 142 &parts, NULL); |
| 143 bool emphasize = model_->CurrentTextIsURL() && (parts.host.len > 0); |
| 144 if (emphasize) { |
| 145 // TODO(shess): Pull color out as a constant. |
| 146 [as addAttribute:NSForegroundColorAttributeName |
| 147 value:[NSColor greenColor] |
| 148 range:NSMakeRange((NSInteger)parts.host.begin, |
| 149 (NSInteger)parts.host.end())]; |
| 150 } |
| 151 |
| 152 // TODO(shess): GTK has this as a member var, figure out why. |
| 153 ToolbarModel::SecurityLevel scheme_security_level = |
| 154 toolbar_model_->GetSchemeSecurityLevel(); |
| 155 |
| 156 // Emphasize the scheme for security UI display purposes (if necessary). |
| 157 if (!model_->user_input_in_progress() && parts.scheme.is_nonempty() && |
| 158 (scheme_security_level != ToolbarModel::NORMAL)) { |
| 159 // TODO(shess): Pull colors out as constants. |
| 160 NSColor* color; |
| 161 if (scheme_security_level == ToolbarModel::SECURE) { |
| 162 color = [NSColor blueColor]; |
| 163 } else { |
| 164 color = [NSColor blackColor]; |
| 165 } |
| 166 [as addAttribute:NSForegroundColorAttributeName value:color |
| 167 range:NSMakeRange((NSInteger)parts.scheme.begin, |
| 168 (NSInteger)parts.scheme.end())]; |
| 169 } |
| 170 |
| 171 // TODO(shess): Check that this updates the model's sense of focus |
| 172 // correctly. |
| 173 [field_ setObjectValue:as]; |
| 174 if (![field_ currentEditor]) { |
| 175 [field_ becomeFirstResponder]; |
| 176 DCHECK_EQ(field_, [[field_ window] firstResponder]); |
| 177 } |
| 178 |
| 179 NSRange selected_range = NSMakeRange(user_text_length, [as length]); |
| 180 // TODO(shess): What if it didn't get first responder, and there is |
| 181 // no field editor? This will do nothing. Well, at least it won't |
| 182 // crash. Think of something more productive to do, or prove that |
| 183 // it cannot occur and DCHECK appropriately. |
| 184 [[field_ currentEditor] setSelectedRange:selected_range]; |
| 185 } |
| 186 |
| 187 void AutocompleteEditViewMac::OnTemporaryTextMaybeChanged( |
| 188 const std::wstring& display_text, bool save_original_selection) { |
| 189 // TODO(shess): I believe this is for when the user arrows around |
| 190 // the popup, will be restored if they hit escape. Figure out if |
| 191 // that is for certain it. |
| 192 if (save_original_selection) { |
| 193 saved_temporary_text_ = GetText(); |
| 194 } |
| 195 |
| 196 UpdateAndStyleText(display_text, display_text.size()); |
| 197 } |
| 198 |
| 199 bool AutocompleteEditViewMac::OnInlineAutocompleteTextMaybeChanged( |
| 200 const std::wstring& display_text, size_t user_text_length) { |
| 201 // TODO(shess): Make sure that this actually works. The round trip |
| 202 // to native form and back may mean that it's the same but not the |
| 203 // same. |
| 204 if (display_text == GetText()) { |
| 205 return false; |
| 206 } |
| 207 |
| 208 UpdateAndStyleText(display_text, user_text_length); |
| 209 return true; |
| 210 } |
| 211 |
| 212 void AutocompleteEditViewMac::OnRevertTemporaryText() { |
| 213 UpdateAndStyleText(saved_temporary_text_, saved_temporary_text_.size()); |
| 214 saved_temporary_text_.clear(); |
| 215 } |
| 216 |
| 217 void AutocompleteEditViewMac::OnUpOrDownKeyPressed(int dir) { |
| 218 model_->OnUpOrDownKeyPressed(dir); |
| 219 } |
| 220 void AutocompleteEditViewMac::OnEscapeKeyPressed() { |
| 221 model_->OnEscapeKeyPressed(); |
| 222 } |
| 223 void AutocompleteEditViewMac::OnSetFocus(bool f) { |
| 224 model_->OnSetFocus(f); |
| 225 } |
| 226 void AutocompleteEditViewMac::OnKillFocus() { |
| 227 model_->OnKillFocus(); |
| 228 } |
| 229 void AutocompleteEditViewMac::AcceptInput( |
| 230 WindowOpenDisposition disposition, bool for_drop) { |
| 231 model_->AcceptInput(disposition, for_drop); |
| 232 } |
| 233 void AutocompleteEditViewMac::OnAfterPossibleChange( |
| 234 const std::wstring& new_text, |
| 235 bool selection_differs, |
| 236 bool text_differs, |
| 237 bool just_deleted_text, |
| 238 bool at_end_of_edit) { |
| 239 model_->OnAfterPossibleChange(new_text, selection_differs, text_differs, |
| 240 just_deleted_text, at_end_of_edit); |
| 241 } |
| 242 void AutocompleteEditViewMac::SetField(NSTextField* field) { |
| 243 field_ = field; |
| 244 [field_ setDelegate:edit_helper_]; |
| 245 |
| 246 // The popup code needs the field for sizing and placement. |
| 247 popup_view_->SetField(field_); |
| 248 } |
| 249 |
| 250 void AutocompleteEditViewMac::FocusLocation() { |
| 251 [[field_ window] makeFirstResponder:field_]; |
| 252 } |
| 253 |
| 254 @implementation AutocompleteFieldDelegate |
| 255 |
| 256 - initWithEditView:(AutocompleteEditViewMac*)view { |
| 257 self = [super init]; |
| 258 if (self) { |
| 259 edit_view_ = view; |
| 260 } |
| 261 return self; |
| 262 } |
| 263 |
| 264 - (BOOL)control:(NSControl*)control |
| 265 textView:(NSTextView*)textView doCommandBySelector:(SEL)cmd { |
| 266 if (cmd == @selector(moveDown:)) { |
| 267 edit_view_->OnUpOrDownKeyPressed(1); |
| 268 return YES; |
| 269 } |
| 270 |
| 271 if (cmd == @selector(moveUp:)) { |
| 272 edit_view_->OnUpOrDownKeyPressed(-1); |
| 273 return YES; |
| 274 } |
| 275 |
| 276 if (cmd == @selector(cancelOperation:)) { |
| 277 edit_view_->OnEscapeKeyPressed(); |
| 278 return YES; |
| 279 } |
| 280 |
| 281 if (cmd == @selector(insertNewline:)) { |
| 282 edit_view_->AcceptInput(CURRENT_TAB, false); |
| 283 return YES; |
| 284 } |
| 285 |
| 286 return NO; |
| 287 } |
| 288 |
| 289 - (void)controlTextDidBeginEditing:(NSNotification*)aNotification { |
| 290 edit_view_->OnSetFocus(false); |
| 291 } |
| 292 |
| 293 - (void)controlTextDidChange:(NSNotification*)aNotification { |
| 294 // TODO(shess): Make this more efficient? Or not. For now, just |
| 295 // pass in the current text, indicating that the text and |
| 296 // selection differ, ignoring deletions, and assuming that we're |
| 297 // at the end of the text. |
| 298 edit_view_->OnAfterPossibleChange(edit_view_->GetText(), |
| 299 true, true, false, true); |
| 300 } |
| 301 |
| 302 - (void)controlTextDidEndEditing:(NSNotification*)aNotification { |
| 303 edit_view_->OnKillFocus(); |
| 304 |
| 305 // TODO(shess): Figure out where the selection belongs. On GTK, |
| 306 // it's set to the start of the text. |
| 307 } |
| 308 |
| 309 @end |
OLD | NEW |