OLD | NEW |
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "chrome/renderer/searchbox_extension.h" | 5 #include "chrome/renderer/searchbox_extension.h" |
6 | 6 |
| 7 #include <ctype.h> |
7 #include <string> | 8 #include <string> |
8 #include <vector> | 9 #include <vector> |
9 | 10 |
10 #include "base/basictypes.h" | 11 #include "base/basictypes.h" |
11 #include "base/command_line.h" | 12 #include "base/string_number_conversions.h" |
| 13 #include "base/string_piece.h" |
12 #include "base/string_split.h" | 14 #include "base/string_split.h" |
13 #include "base/stringprintf.h" | 15 #include "base/stringprintf.h" |
| 16 #include "chrome/common/chrome_switches.h" |
14 #include "chrome/renderer/searchbox.h" | 17 #include "chrome/renderer/searchbox.h" |
15 #include "content/public/renderer/render_view.h" | 18 #include "content/public/renderer/render_view.h" |
16 #include "grit/renderer_resources.h" | 19 #include "grit/renderer_resources.h" |
17 #include "third_party/WebKit/Source/WebKit/chromium/public/WebFrame.h" | 20 #include "third_party/WebKit/Source/WebKit/chromium/public/WebFrame.h" |
18 #include "third_party/WebKit/Source/WebKit/chromium/public/WebScriptSource.h" | 21 #include "third_party/WebKit/Source/WebKit/chromium/public/WebScriptSource.h" |
| 22 #include "third_party/WebKit/Source/WebKit/chromium/public/WebView.h" |
19 #include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebString.h" | 23 #include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebString.h" |
| 24 #include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebURLReques
t.h" |
20 #include "ui/base/resource/resource_bundle.h" | 25 #include "ui/base/resource/resource_bundle.h" |
21 #include "v8/include/v8.h" | 26 #include "v8/include/v8.h" |
22 | 27 |
23 using WebKit::WebFrame; | 28 using WebKit::WebFrame; |
24 using WebKit::WebScriptSource; | 29 using WebKit::WebScriptSource; |
25 using WebKit::WebString; | 30 using WebKit::WebString; |
26 using WebKit::WebView; | 31 using WebKit::WebView; |
27 | 32 |
| 33 namespace { |
| 34 |
| 35 // Splits the string in |number| into two pieces, a leading number token (saved |
| 36 // in |number|) and the rest (saved in |suffix|). Either piece may become empty, |
| 37 // depending on whether the input had no digits or only digits. Neither argument |
| 38 // may be NULL. |
| 39 void SplitLeadingNumberToken(std::string* number, std::string* suffix) { |
| 40 size_t i = 0; |
| 41 while (i < number->size() && isdigit((*number)[i])) |
| 42 ++i; |
| 43 suffix->assign(*number, i, number->size() - i); |
| 44 number->resize(i); |
| 45 } |
| 46 |
| 47 } // namespace |
| 48 |
28 namespace extensions_v8 { | 49 namespace extensions_v8 { |
29 | 50 |
30 static const char kSearchBoxExtensionName[] = "v8/SearchBox"; | 51 static const char kSearchBoxExtensionName[] = "v8/SearchBox"; |
31 | 52 |
32 static const char kDispatchChangeEventScript[] = | 53 static const char kDispatchChangeEventScript[] = |
33 "if (window.chrome &&" | 54 "if (window.chrome &&" |
34 " window.chrome.searchBox &&" | 55 " window.chrome.searchBox &&" |
35 " window.chrome.searchBox.onchange &&" | 56 " window.chrome.searchBox.onchange &&" |
36 " typeof window.chrome.searchBox.onchange == 'function') {" | 57 " typeof window.chrome.searchBox.onchange == 'function') {" |
37 " window.chrome.searchBox.onchange();" | 58 " window.chrome.searchBox.onchange();" |
(...skipping 20 matching lines...) Expand all Loading... |
58 | 79 |
59 static const char kDispatchResizeEventScript[] = | 80 static const char kDispatchResizeEventScript[] = |
60 "if (window.chrome &&" | 81 "if (window.chrome &&" |
61 " window.chrome.searchBox &&" | 82 " window.chrome.searchBox &&" |
62 " window.chrome.searchBox.onresize &&" | 83 " window.chrome.searchBox.onresize &&" |
63 " typeof window.chrome.searchBox.onresize == 'function') {" | 84 " typeof window.chrome.searchBox.onresize == 'function') {" |
64 " window.chrome.searchBox.onresize();" | 85 " window.chrome.searchBox.onresize();" |
65 " true;" | 86 " true;" |
66 "}"; | 87 "}"; |
67 | 88 |
68 // Deprecated API support. | 89 // Extended API. |
69 // TODO(tonyg): Remove these when they are no longer used. | |
70 // ---------------------------------------------------------------------------- | |
71 // Script sent as the user is typing and the provider supports instant. | |
72 // Params: | |
73 // . the text the user typed. | |
74 // '46' forces the server to give us verbatim results. | |
75 static const char kUserInputScript[] = | |
76 "if (window.chrome.userInput)" | |
77 " window.chrome.userInput(" | |
78 " window.chrome.searchBox.value," | |
79 " window.chrome.searchBox.verbatim ? 46 : 0," | |
80 " window.chrome.searchBox.selectionStart);"; | |
81 | 90 |
82 // Script sent when the page is committed and the provider supports instant. | 91 static const char kDispatchNativeSuggestionsEventScript[] = |
83 // Params: | 92 "if (window.chrome &&" |
84 // . the text the user typed. | 93 " window.chrome.searchBox &&" |
85 // . boolean indicating if the user pressed enter to accept the text. | 94 " window.chrome.searchBox.onnativesuggestions &&" |
86 static const char kUserDoneScript[] = | 95 " typeof window.chrome.searchBox.onnativesuggestions == 'function') {" |
87 "if (window.chrome.userWantsQuery)" | 96 " window.chrome.searchBox.onnativesuggestions();" |
88 " window.chrome.userWantsQuery(" | 97 " true;" |
89 " window.chrome.searchBox.value," | 98 "}"; |
90 " window.chrome.searchBox.verbatim);"; | |
91 | 99 |
92 // Script sent when the bounds of the omnibox changes and the provider supports | 100 static const char kDispatchKeyPressEventScript[] = |
93 // instant. The params are the bounds relative to the origin of the preview | 101 "if (window.chrome &&" |
94 // (x, y, width, height). | 102 " window.chrome.searchBox &&" |
95 static const char kSetOmniboxBoundsScript[] = | 103 " window.chrome.searchBox.onkeypress &&" |
96 "if (window.chrome.setDropdownDimensions)" | 104 " typeof window.chrome.searchBox.onkeypress == 'function') {" |
97 " window.chrome.setDropdownDimensions(" | 105 " window.chrome.searchBox.onkeypress(" |
98 " window.chrome.searchBox.x," | 106 " {keyCode:window.chrome.searchBox.keyCode});" |
99 " window.chrome.searchBox.y," | 107 " true;" |
100 " window.chrome.searchBox.width," | 108 "}"; |
101 " window.chrome.searchBox.height);"; | 109 |
| 110 static const char kDispatchFocusEventScript[] = |
| 111 "if (window.chrome &&" |
| 112 " window.chrome.searchBox &&" |
| 113 " window.chrome.searchBox.onfocus &&" |
| 114 " typeof window.chrome.searchBox.onfocus == 'function') {" |
| 115 " window.chrome.searchBox.onfocus();" |
| 116 " true;" |
| 117 "}"; |
| 118 |
| 119 static const char kDispatchBlurEventScript[] = |
| 120 "if (window.chrome &&" |
| 121 " window.chrome.searchBox &&" |
| 122 " window.chrome.searchBox.onblur &&" |
| 123 " typeof window.chrome.searchBox.onblur == 'function') {" |
| 124 " window.chrome.searchBox.onblur();" |
| 125 " true;" |
| 126 "}"; |
102 | 127 |
103 // We first send this script down to determine if the page supports instant. | 128 // We first send this script down to determine if the page supports instant. |
104 static const char kSupportsInstantScript[] = | 129 static const char kSupportsInstantScript[] = |
105 "if (window.chrome.sv) true; else false;"; | 130 "if (window.chrome &&" |
106 | 131 " window.chrome.searchBox &&" |
107 // The google.y.first array is a list of functions which are to be executed | 132 " window.chrome.searchBox.onsubmit &&" |
108 // after the external JavaScript used by Google web search loads. The deprecated | 133 " typeof window.chrome.searchBox.onsubmit == 'function') {" |
109 // API requires setDropdownDimensions and userInput to be invoked after | 134 " true;" |
110 // the external JavaScript loads. So if they are not already registered, we add | 135 "} else {" |
111 // them to the array of functions the page will execute after load. This tight | 136 " false;" |
112 // coupling discourages proliferation of the deprecated API. | 137 "}"; |
113 static const char kInitScript[] = | |
114 "(function() {" | |
115 "var initScript = function(){%s%s};" | |
116 "if (window.chrome.setDropdownDimensions)" | |
117 " initScript();" | |
118 "else if (window.google && window.google.y)" | |
119 " window.google.y.first.push(initScript);" | |
120 "})();"; | |
121 // ---------------------------------------------------------------------------- | |
122 | 138 |
123 class SearchBoxExtensionWrapper : public v8::Extension { | 139 class SearchBoxExtensionWrapper : public v8::Extension { |
124 public: | 140 public: |
125 explicit SearchBoxExtensionWrapper(const base::StringPiece& code); | 141 explicit SearchBoxExtensionWrapper(const base::StringPiece& code); |
126 | 142 |
127 // Allows v8's javascript code to call the native functions defined | 143 // Allows v8's javascript code to call the native functions defined |
128 // in this class for window.chrome. | 144 // in this class for window.chrome. |
129 virtual v8::Handle<v8::FunctionTemplate> GetNativeFunction( | 145 virtual v8::Handle<v8::FunctionTemplate> GetNativeFunction( |
130 v8::Handle<v8::String> name); | 146 v8::Handle<v8::String> name); |
131 | 147 |
(...skipping 21 matching lines...) Expand all Loading... |
153 // Gets the y coordinate (relative to |window|) of the right edge of the | 169 // Gets the y coordinate (relative to |window|) of the right edge of the |
154 // region of the search box that overlaps the window. | 170 // region of the search box that overlaps the window. |
155 static v8::Handle<v8::Value> GetY(const v8::Arguments& args); | 171 static v8::Handle<v8::Value> GetY(const v8::Arguments& args); |
156 | 172 |
157 // Gets the width of the region of the search box that overlaps the window. | 173 // Gets the width of the region of the search box that overlaps the window. |
158 static v8::Handle<v8::Value> GetWidth(const v8::Arguments& args); | 174 static v8::Handle<v8::Value> GetWidth(const v8::Arguments& args); |
159 | 175 |
160 // Gets the height of the region of the search box that overlaps the window. | 176 // Gets the height of the region of the search box that overlaps the window. |
161 static v8::Handle<v8::Value> GetHeight(const v8::Arguments& args); | 177 static v8::Handle<v8::Value> GetHeight(const v8::Arguments& args); |
162 | 178 |
| 179 // Gets the native suggestions from search box. |
| 180 static v8::Handle<v8::Value> GetNativeSuggestions(const v8::Arguments& args); |
| 181 |
| 182 // Gets the last key code entered in search box. |
| 183 static v8::Handle<v8::Value> GetKeyCode(const v8::Arguments& args); |
| 184 |
| 185 // Gets some info about the last search session committed in the search box. |
| 186 static v8::Handle<v8::Value> GetSessionContext(const v8::Arguments& args); |
| 187 |
| 188 // Gets whether the search box currently has keyboard focus or not. |
| 189 static v8::Handle<v8::Value> GetIsFocused(const v8::Arguments& args); |
| 190 |
163 // Sets ordered suggestions. Valid for current |value|. | 191 // Sets ordered suggestions. Valid for current |value|. |
164 static v8::Handle<v8::Value> SetSuggestions(const v8::Arguments& args); | 192 static v8::Handle<v8::Value> SetSuggestions(const v8::Arguments& args); |
165 | 193 |
| 194 // Sets the text to be autocompleted into the search box. |
| 195 static v8::Handle<v8::Value> SetAutocompleteText(const v8::Arguments& args); |
| 196 |
| 197 // Like |SetAutocompleteText| but uses a restricted ID to identify the text. |
| 198 static v8::Handle<v8::Value> SetRestrictedAutocompleteText( |
| 199 const v8::Arguments& args); |
| 200 |
| 201 // Sets the search box text, completely replacing what the user typed. |
| 202 static v8::Handle<v8::Value> SetValue(const v8::Arguments& args); |
| 203 |
| 204 // Like |SetValue| but uses a restricted ID to identify the text. |
| 205 static v8::Handle<v8::Value> SetRestrictedValue(const v8::Arguments& args); |
| 206 |
| 207 // Resize the preview to the given height. |
| 208 static v8::Handle<v8::Value> SetNonNativeDropdownHeight( |
| 209 const v8::Arguments& args); |
| 210 |
| 211 // Navigate the window to the URL identified by a restricted ID. |
| 212 static v8::Handle<v8::Value> NavigateContentWindow(const v8::Arguments& args); |
| 213 |
| 214 |
166 private: | 215 private: |
167 DISALLOW_COPY_AND_ASSIGN(SearchBoxExtensionWrapper); | 216 DISALLOW_COPY_AND_ASSIGN(SearchBoxExtensionWrapper); |
168 }; | 217 }; |
169 | 218 |
170 SearchBoxExtensionWrapper::SearchBoxExtensionWrapper( | 219 SearchBoxExtensionWrapper::SearchBoxExtensionWrapper( |
171 const base::StringPiece& code) | 220 const base::StringPiece& code) |
172 : v8::Extension(kSearchBoxExtensionName, code.data(), 0, 0, code.size()) {} | 221 : v8::Extension(kSearchBoxExtensionName, code.data(), 0, 0, code.size()) {} |
173 | 222 |
174 v8::Handle<v8::FunctionTemplate> SearchBoxExtensionWrapper::GetNativeFunction( | 223 v8::Handle<v8::FunctionTemplate> SearchBoxExtensionWrapper::GetNativeFunction( |
175 v8::Handle<v8::String> name) { | 224 v8::Handle<v8::String> name) { |
176 if (name->Equals(v8::String::New("GetValue"))) { | 225 if (name->Equals(v8::String::New("GetValue"))) { |
177 return v8::FunctionTemplate::New(GetValue); | 226 return v8::FunctionTemplate::New(GetValue); |
178 } else if (name->Equals(v8::String::New("GetVerbatim"))) { | 227 } else if (name->Equals(v8::String::New("GetVerbatim"))) { |
179 return v8::FunctionTemplate::New(GetVerbatim); | 228 return v8::FunctionTemplate::New(GetVerbatim); |
180 } else if (name->Equals(v8::String::New("GetSelectionStart"))) { | 229 } else if (name->Equals(v8::String::New("GetSelectionStart"))) { |
181 return v8::FunctionTemplate::New(GetSelectionStart); | 230 return v8::FunctionTemplate::New(GetSelectionStart); |
182 } else if (name->Equals(v8::String::New("GetSelectionEnd"))) { | 231 } else if (name->Equals(v8::String::New("GetSelectionEnd"))) { |
183 return v8::FunctionTemplate::New(GetSelectionEnd); | 232 return v8::FunctionTemplate::New(GetSelectionEnd); |
184 } else if (name->Equals(v8::String::New("GetX"))) { | 233 } else if (name->Equals(v8::String::New("GetX"))) { |
185 return v8::FunctionTemplate::New(GetX); | 234 return v8::FunctionTemplate::New(GetX); |
186 } else if (name->Equals(v8::String::New("GetY"))) { | 235 } else if (name->Equals(v8::String::New("GetY"))) { |
187 return v8::FunctionTemplate::New(GetY); | 236 return v8::FunctionTemplate::New(GetY); |
188 } else if (name->Equals(v8::String::New("GetWidth"))) { | 237 } else if (name->Equals(v8::String::New("GetWidth"))) { |
189 return v8::FunctionTemplate::New(GetWidth); | 238 return v8::FunctionTemplate::New(GetWidth); |
190 } else if (name->Equals(v8::String::New("GetHeight"))) { | 239 } else if (name->Equals(v8::String::New("GetHeight"))) { |
191 return v8::FunctionTemplate::New(GetHeight); | 240 return v8::FunctionTemplate::New(GetHeight); |
| 241 } else if (name->Equals(v8::String::New("GetNativeSuggestions"))) { |
| 242 return v8::FunctionTemplate::New(GetNativeSuggestions); |
| 243 } else if (name->Equals(v8::String::New("GetKeyCode"))) { |
| 244 return v8::FunctionTemplate::New(GetKeyCode); |
| 245 } else if (name->Equals(v8::String::New("GetSessionContext"))) { |
| 246 return v8::FunctionTemplate::New(GetSessionContext); |
| 247 } else if (name->Equals(v8::String::New("GetIsFocused"))) { |
| 248 return v8::FunctionTemplate::New(GetIsFocused); |
192 } else if (name->Equals(v8::String::New("SetSuggestions"))) { | 249 } else if (name->Equals(v8::String::New("SetSuggestions"))) { |
193 return v8::FunctionTemplate::New(SetSuggestions); | 250 return v8::FunctionTemplate::New(SetSuggestions); |
| 251 } else if (name->Equals(v8::String::New("SetAutocompleteText"))) { |
| 252 return v8::FunctionTemplate::New(SetAutocompleteText); |
| 253 } else if (name->Equals(v8::String::New("SetRestrictedAutocompleteText"))) { |
| 254 return v8::FunctionTemplate::New(SetRestrictedAutocompleteText); |
| 255 } else if (name->Equals(v8::String::New("SetValue"))) { |
| 256 return v8::FunctionTemplate::New(SetValue); |
| 257 } else if (name->Equals(v8::String::New("SetRestrictedValue"))) { |
| 258 return v8::FunctionTemplate::New(SetRestrictedValue); |
| 259 } else if (name->Equals(v8::String::New("SetNonNativeDropdownHeight"))) { |
| 260 return v8::FunctionTemplate::New(SetNonNativeDropdownHeight); |
| 261 } else if (name->Equals(v8::String::New("NavigateContentWindow"))) { |
| 262 return v8::FunctionTemplate::New(NavigateContentWindow); |
194 } | 263 } |
195 return v8::Handle<v8::FunctionTemplate>(); | 264 return v8::Handle<v8::FunctionTemplate>(); |
196 } | 265 } |
197 | 266 |
198 // static | 267 // static |
199 content::RenderView* SearchBoxExtensionWrapper::GetRenderView() { | 268 content::RenderView* SearchBoxExtensionWrapper::GetRenderView() { |
200 WebFrame* webframe = WebFrame::frameForEnteredContext(); | 269 WebFrame* webframe = WebFrame::frameForEnteredContext(); |
201 DCHECK(webframe) << "There should be an active frame since we just got " | 270 DCHECK(webframe) << "There should be an active frame since we just got " |
202 "a native function called."; | 271 "a native function called."; |
203 if (!webframe) return NULL; | 272 if (!webframe) return NULL; |
204 | 273 |
205 WebView* webview = webframe->view(); | 274 WebView* webview = webframe->view(); |
206 if (!webview) return NULL; // can happen during closing | 275 if (!webview) return NULL; // can happen during closing |
207 | 276 |
208 return content::RenderView::FromWebView(webview); | 277 return content::RenderView::FromWebView(webview); |
209 } | 278 } |
210 | 279 |
211 // static | 280 // static |
212 v8::Handle<v8::Value> SearchBoxExtensionWrapper::GetValue( | 281 v8::Handle<v8::Value> SearchBoxExtensionWrapper::GetValue( |
213 const v8::Arguments& args) { | 282 const v8::Arguments& args) { |
214 content::RenderView* render_view = GetRenderView(); | 283 content::RenderView* render_view = GetRenderView(); |
215 if (!render_view) return v8::Undefined(); | 284 if (!render_view) return v8::Undefined(); |
216 return v8::String::New( | 285 return v8::String::New( |
217 reinterpret_cast<const uint16_t*>( | 286 reinterpret_cast<const uint16_t*>( |
218 SearchBox::Get(render_view)->value().c_str()), | 287 SearchBox::Get(render_view)->value().c_str()), |
219 SearchBox::Get(render_view)->value().length()); | 288 SearchBox::Get(render_view)->value().length()); |
220 } | 289 } |
221 | 290 |
222 // static | 291 // static |
223 v8::Handle<v8::Value> SearchBoxExtensionWrapper::GetVerbatim( | 292 v8::Handle<v8::Value> SearchBoxExtensionWrapper::GetVerbatim( |
224 const v8::Arguments& args) { | 293 const v8::Arguments& args) { |
225 content::RenderView* render_view = GetRenderView(); | 294 content::RenderView* render_view = GetRenderView(); |
226 if (!render_view) return v8::Undefined(); | 295 if (!render_view) return v8::Undefined(); |
227 return v8::Boolean::New(SearchBox::Get(render_view)->verbatim()); | 296 return v8::Boolean::New(SearchBox::Get(render_view)->verbatim()); |
228 } | 297 } |
229 | 298 |
(...skipping 38 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
268 } | 337 } |
269 | 338 |
270 // static | 339 // static |
271 v8::Handle<v8::Value> SearchBoxExtensionWrapper::GetHeight( | 340 v8::Handle<v8::Value> SearchBoxExtensionWrapper::GetHeight( |
272 const v8::Arguments& args) { | 341 const v8::Arguments& args) { |
273 content::RenderView* render_view = GetRenderView(); | 342 content::RenderView* render_view = GetRenderView(); |
274 if (!render_view) return v8::Undefined(); | 343 if (!render_view) return v8::Undefined(); |
275 return v8::Int32::New(SearchBox::Get(render_view)->GetRect().height()); | 344 return v8::Int32::New(SearchBox::Get(render_view)->GetRect().height()); |
276 } | 345 } |
277 | 346 |
278 // Accepts a single argument in form: | 347 // static |
279 // { | 348 v8::Handle<v8::Value> SearchBoxExtensionWrapper::GetNativeSuggestions( |
280 // suggestions: [ | 349 const v8::Arguments& args) { |
281 // { | 350 content::RenderView* render_view = GetRenderView(); |
282 // value: "..." | 351 if (!render_view) return v8::Undefined(); |
283 // } | 352 const std::vector<InstantNativeSuggestionsParts>& suggestions = |
284 // ] | 353 SearchBox::Get(render_view)->native_suggestions(); |
285 // } | 354 const int rid_base = SearchBox::Get(render_view)->rid_base(); |
| 355 v8::Handle<v8::Array> suggestions_array(v8::Array::New(suggestions.size())); |
| 356 for (size_t i = 0; i < suggestions.size(); ++i) { |
| 357 v8::Handle<v8::Object> suggestion(v8::Object::New()); |
| 358 suggestion->Set(v8::String::New("provider"), |
| 359 v8::String::New(suggestions[i].provider.c_str())); |
| 360 suggestion->Set(v8::String::New("contents"), |
| 361 v8::String::New(suggestions[i].contents.c_str())); |
| 362 suggestion->Set(v8::String::New("destination_url"), |
| 363 v8::String::New(suggestions[i].destination_url.spec().c_str())); |
| 364 suggestion->Set(v8::String::New("rid"), v8::Uint32::New(rid_base + i)); |
| 365 |
| 366 v8::Handle<v8::Object> ranking_data(v8::Object::New()); |
| 367 ranking_data->Set(v8::String::New("relevance"), |
| 368 v8::Int32::New(suggestions[i].relevance)); |
| 369 suggestion->Set(v8::String::New("rankingData"), ranking_data); |
| 370 |
| 371 suggestions_array->Set(i, suggestion); |
| 372 } |
| 373 |
| 374 return suggestions_array; |
| 375 } |
| 376 |
| 377 // static |
| 378 v8::Handle<v8::Value> SearchBoxExtensionWrapper::GetKeyCode( |
| 379 const v8::Arguments& args) { |
| 380 content::RenderView* render_view = GetRenderView(); |
| 381 if (!render_view) return v8::Undefined(); |
| 382 return v8::Int32::New(SearchBox::Get(render_view)->key_code()); |
| 383 } |
| 384 |
| 385 // static |
| 386 v8::Handle<v8::Value> SearchBoxExtensionWrapper::GetSessionContext( |
| 387 const v8::Arguments& args) { |
| 388 content::RenderView* render_view = GetRenderView(); |
| 389 if (!render_view) return v8::Undefined(); |
| 390 v8::Handle<v8::Object> session_context(v8::Object::New()); |
| 391 session_context->Set(v8::String::New("lastSearchQuery"), |
| 392 v8::String::New(SearchBox::Get(render_view)->last_query().c_str())); |
| 393 // NOTE: SECURITY RISK -- Providing the current url in this way is insecure. |
| 394 session_context->Set(v8::String::New("currentUrl"), |
| 395 v8::String::New(SearchBox::Get(render_view)->current_url().c_str())); |
| 396 return session_context; |
| 397 } |
| 398 |
| 399 // static |
| 400 v8::Handle<v8::Value> SearchBoxExtensionWrapper::GetIsFocused( |
| 401 const v8::Arguments& args) { |
| 402 content::RenderView* render_view = GetRenderView(); |
| 403 if (!render_view) return v8::Undefined(); |
| 404 return v8::Boolean::New(SearchBox::Get(render_view)->is_focused()); |
| 405 } |
| 406 |
286 // static | 407 // static |
287 v8::Handle<v8::Value> SearchBoxExtensionWrapper::SetSuggestions( | 408 v8::Handle<v8::Value> SearchBoxExtensionWrapper::SetSuggestions( |
288 const v8::Arguments& args) { | 409 const v8::Arguments& args) { |
289 std::vector<std::string> suggestions; | 410 std::vector<InstantSuggestion> suggestions; |
290 InstantCompleteBehavior behavior = INSTANT_COMPLETE_NOW; | 411 |
291 | 412 if (args.Length() && args[0]->IsObject()) { |
292 if (args.Length() && args[0]->IsArray()) { | 413 v8::Local<v8::Object> suggestion_json(args[0]->ToObject()); |
293 // For backwards compatibility, also accept an array of strings. | 414 |
294 // TODO(tonyg): Remove this when it is confirmed to be unused. | 415 InstantCompleteBehavior behavior = INSTANT_COMPLETE_NOW; |
295 v8::Local<v8::Array> suggestions_array = | 416 InstantSuggestionType type = INSTANT_SUGGESTION_SEARCH; |
296 v8::Local<v8::Array>::Cast(args[0]); | 417 v8::Local<v8::Value> complete_value( |
297 uint32_t length = suggestions_array->Length(); | 418 suggestion_json->Get(v8::String::New("complete_behavior"))); |
298 for (uint32_t i = 0; i < length; i++) { | 419 if (complete_value->IsString()) { |
299 std::string suggestion = *v8::String::Utf8Value( | 420 if (complete_value->Equals(v8::String::New("now"))) { |
300 suggestions_array->Get(v8::Integer::New(i))->ToString()); | 421 behavior = INSTANT_COMPLETE_NOW; |
301 if (!suggestion.length()) continue; | 422 } else if (complete_value->Equals(v8::String::New("never"))) { |
302 suggestions.push_back(suggestion); | 423 behavior = INSTANT_COMPLETE_NEVER; |
303 } | 424 } else if (complete_value->Equals(v8::String::New("delayed"))) { |
304 } else if (args.Length() && args[0]->IsObject()) { | 425 behavior = INSTANT_COMPLETE_DELAYED; |
305 // Standard version, object argument. | 426 } else if (complete_value->Equals(v8::String::New("replace"))) { |
306 v8::Local<v8::Object> suggestion_json = | 427 behavior = INSTANT_COMPLETE_REPLACE; |
307 v8::Local<v8::Object>::Cast(args[0]); | 428 } else { |
308 v8::Local<v8::Value> suggestions_field = | 429 VLOG(1) << "Unsupported complete behavior '" |
309 suggestion_json->Get(v8::String::New("suggestions")); | 430 << *v8::String::Utf8Value(complete_value) << "'"; |
310 | 431 } |
| 432 } |
| 433 |
| 434 v8::Local<v8::Value> suggestions_field( |
| 435 suggestion_json->Get(v8::String::New("suggestions"))); |
311 if (suggestions_field->IsArray()) { | 436 if (suggestions_field->IsArray()) { |
312 v8::Local<v8::Array> suggestions_array = | 437 v8::Local<v8::Array> suggestions_array( |
313 suggestions_field.As<v8::Array>(); | 438 suggestions_field.As<v8::Array>()); |
314 | 439 size_t length = suggestions_array->Length(); |
315 uint32_t length = suggestions_array->Length(); | 440 for (size_t i = 0; i < length; i++) { |
316 for (uint32_t i = 0; i < length; i++) { | 441 v8::Local<v8::Value> suggestion_value(suggestions_array->Get(i)); |
317 v8::Local<v8::Value> suggestion_value = | |
318 suggestions_array->Get(v8::Integer::New(i)); | |
319 if (!suggestion_value->IsObject()) continue; | 442 if (!suggestion_value->IsObject()) continue; |
320 | 443 |
321 v8::Local<v8::Object> suggestion_object = | 444 v8::Local<v8::Object> suggestion_object(suggestion_value->ToObject()); |
322 suggestion_value.As<v8::Object>(); | 445 v8::Local<v8::Value> suggestion_object_value( |
323 v8::Local<v8::Value> suggestion_object_value = | 446 suggestion_object->Get(v8::String::New("value"))); |
324 suggestion_object->Get(v8::String::New("value")); | |
325 if (!suggestion_object_value->IsString()) continue; | 447 if (!suggestion_object_value->IsString()) continue; |
326 | 448 |
327 std::string suggestion = *v8::String::Utf8Value( | 449 std::string text = *v8::String::Utf8Value(suggestion_object_value); |
328 suggestion_object_value->ToString()); | 450 |
329 if (!suggestion.length()) continue; | 451 suggestions.push_back(InstantSuggestion(text, behavior, type)); |
330 suggestions.push_back(suggestion); | |
331 } | |
332 } | |
333 if (suggestion_json->Has(v8::String::New("complete_behavior"))) { | |
334 v8::Local<v8::Value> complete_value = | |
335 suggestion_json->Get(v8::String::New("complete_behavior")); | |
336 if (complete_value->IsString()) { | |
337 if (complete_value->Equals(v8::String::New("never"))) | |
338 behavior = INSTANT_COMPLETE_NEVER; | |
339 else if (complete_value->Equals(v8::String::New("delayed"))) | |
340 behavior = INSTANT_COMPLETE_DELAYED; | |
341 } | 452 } |
342 } | 453 } |
343 } | 454 } |
344 | 455 |
345 if (content::RenderView* render_view = GetRenderView()) | 456 if (content::RenderView* render_view = GetRenderView()) |
346 SearchBox::Get(render_view)->SetSuggestions(suggestions, behavior); | 457 SearchBox::Get(render_view)->SetSuggestions(suggestions); |
347 return v8::Undefined(); | 458 return v8::Undefined(); |
348 } | 459 } |
349 | 460 |
350 // static | 461 // static |
351 void Dispatch(WebFrame* frame, | 462 v8::Handle<v8::Value> SearchBoxExtensionWrapper::SetAutocompleteText( |
352 WebString event_dispatch_script, | 463 const v8::Arguments& args) { |
353 WebString no_event_handler_script) { | 464 if (1 <= args.Length() && args.Length() <= 2 && args[0]->IsString()) { |
| 465 std::string text = *v8::String::Utf8Value(args[0]); |
| 466 InstantCompleteBehavior behavior = INSTANT_COMPLETE_NOW; |
| 467 InstantSuggestionType type = INSTANT_SUGGESTION_URL; |
| 468 |
| 469 if (args.Length() >= 2 && args[1]->Uint32Value() == 2) { |
| 470 behavior = INSTANT_COMPLETE_NEVER; |
| 471 // TODO(sreeram): The page should really set the type explicitly. |
| 472 type = INSTANT_SUGGESTION_SEARCH; |
| 473 } |
| 474 |
| 475 if (content::RenderView* render_view = GetRenderView()) { |
| 476 std::vector<InstantSuggestion> suggestions; |
| 477 suggestions.push_back(InstantSuggestion(text, behavior, type)); |
| 478 SearchBox::Get(render_view)->SetSuggestions(suggestions); |
| 479 } |
| 480 } |
| 481 return v8::Undefined(); |
| 482 } |
| 483 |
| 484 // static |
| 485 v8::Handle<v8::Value> SearchBoxExtensionWrapper::SetRestrictedAutocompleteText( |
| 486 const v8::Arguments& args) { |
| 487 content::RenderView* render_view = GetRenderView(); |
| 488 if (1 <= args.Length() && args.Length() <= 2 && args[0]->IsNumber() && |
| 489 render_view) { |
| 490 const int rid = args[0]->Uint32Value(); |
| 491 const int rid_base = SearchBox::Get(render_view)->rid_base(); |
| 492 // Note that stale rids, less than the current rid_base, will wrap. |
| 493 const size_t index = rid - rid_base; |
| 494 const std::vector<InstantNativeSuggestionsParts>& suggestions = |
| 495 SearchBox::Get(render_view)->native_suggestions(); |
| 496 if (index < suggestions.size()) { |
| 497 std::string text = suggestions[index].destination_url.spec(); |
| 498 InstantCompleteBehavior behavior = INSTANT_COMPLETE_NOW; |
| 499 InstantSuggestionType type = INSTANT_SUGGESTION_URL; |
| 500 |
| 501 if (args.Length() >= 2 && args[1]->Uint32Value() == 2) |
| 502 behavior = INSTANT_COMPLETE_NEVER; |
| 503 |
| 504 if (suggestions[index].is_search) { |
| 505 text = suggestions[index].contents; |
| 506 type = INSTANT_SUGGESTION_SEARCH; |
| 507 } |
| 508 |
| 509 std::vector<InstantSuggestion> suggestions; |
| 510 suggestions.push_back(InstantSuggestion(text, behavior, type)); |
| 511 SearchBox::Get(render_view)->SetSuggestions(suggestions); |
| 512 } else { |
| 513 VLOG(1) << "Invalid rid " << rid << "; " |
| 514 << "rid_base is " << rid_base << "."; |
| 515 } |
| 516 } |
| 517 return v8::Undefined(); |
| 518 } |
| 519 |
| 520 // static |
| 521 v8::Handle<v8::Value> SearchBoxExtensionWrapper::SetValue( |
| 522 const v8::Arguments& args) { |
| 523 // TODO(sreeram): Make the second argument (type) mandatory. |
| 524 if (1 <= args.Length() && args.Length() <= 2 && args[0]->IsString()) { |
| 525 std::string text = *v8::String::Utf8Value(args[0]); |
| 526 InstantCompleteBehavior behavior = INSTANT_COMPLETE_REPLACE; |
| 527 InstantSuggestionType type = INSTANT_SUGGESTION_SEARCH; |
| 528 |
| 529 if (args.Length() >= 2 && args[1]->Uint32Value() == 1) |
| 530 type = INSTANT_SUGGESTION_URL; |
| 531 |
| 532 if (content::RenderView* render_view = GetRenderView()) { |
| 533 std::vector<InstantSuggestion> suggestions; |
| 534 suggestions.push_back(InstantSuggestion(text, behavior, type)); |
| 535 SearchBox::Get(render_view)->SetSuggestions(suggestions); |
| 536 } |
| 537 } |
| 538 return v8::Undefined(); |
| 539 } |
| 540 |
| 541 // static |
| 542 v8::Handle<v8::Value> SearchBoxExtensionWrapper::SetRestrictedValue( |
| 543 const v8::Arguments& args) { |
| 544 content::RenderView* render_view = GetRenderView(); |
| 545 if (1 <= args.Length() && args.Length() <= 2 && args[0]->IsNumber() && |
| 546 render_view) { |
| 547 const int rid = args[0]->Uint32Value(); |
| 548 const int rid_base = SearchBox::Get(render_view)->rid_base(); |
| 549 // Note that stale rids, less than the current rid_base, will wrap. |
| 550 const size_t index = rid - rid_base; |
| 551 const std::vector<InstantNativeSuggestionsParts>& suggestions = |
| 552 SearchBox::Get(render_view)->native_suggestions(); |
| 553 if (index < suggestions.size()) { |
| 554 std::string text = suggestions[index].destination_url.spec(); |
| 555 InstantCompleteBehavior behavior = INSTANT_COMPLETE_REPLACE; |
| 556 InstantSuggestionType type = INSTANT_SUGGESTION_URL; |
| 557 |
| 558 if ((args.Length() >= 2 && args[1]->Uint32Value() == 0) || |
| 559 (args.Length() < 2 && suggestions[index].is_search)) { |
| 560 text = suggestions[index].contents; |
| 561 type = INSTANT_SUGGESTION_SEARCH; |
| 562 } |
| 563 |
| 564 std::vector<InstantSuggestion> suggestions; |
| 565 suggestions.push_back(InstantSuggestion(text, behavior, type)); |
| 566 SearchBox::Get(render_view)->SetSuggestions(suggestions); |
| 567 } else { |
| 568 VLOG(1) << "Invalid rid " << rid << "; " |
| 569 << "rid_base is " << rid_base << "."; |
| 570 } |
| 571 } |
| 572 return v8::Undefined(); |
| 573 } |
| 574 |
| 575 // static |
| 576 v8::Handle<v8::Value> SearchBoxExtensionWrapper::SetNonNativeDropdownHeight( |
| 577 const v8::Arguments& args) { |
| 578 if (args.Length() == 1) { |
| 579 int height = 0; |
| 580 InstantSizeUnits units = INSTANT_SIZE_PIXELS; |
| 581 if (args[0]->IsInt32()) { |
| 582 height = args[0]->Int32Value(); |
| 583 } else if (args[0]->IsString()) { |
| 584 std::string height_str = *v8::String::Utf8Value(args[0]); |
| 585 std::string units_str; |
| 586 SplitLeadingNumberToken(&height_str, &units_str); |
| 587 if (!base::StringToInt(height_str, &height)) |
| 588 return v8::Undefined(); |
| 589 if (units_str == "%") { |
| 590 units = INSTANT_SIZE_PERCENT; |
| 591 } else if (!units_str.empty() && units_str != "px") { |
| 592 return v8::Undefined(); |
| 593 } |
| 594 } else { |
| 595 return v8::Undefined(); |
| 596 } |
| 597 content::RenderView* render_view = GetRenderView(); |
| 598 if (render_view && height >= 0) |
| 599 SearchBox::Get(render_view)->SetInstantPreviewHeight(height, units); |
| 600 } |
| 601 return v8::Undefined(); |
| 602 } |
| 603 |
| 604 // static |
| 605 v8::Handle<v8::Value> SearchBoxExtensionWrapper::NavigateContentWindow( |
| 606 const v8::Arguments& args) { |
| 607 content::RenderView* render_view = GetRenderView(); |
| 608 if (args.Length() == 1 && args[0]->IsNumber() && render_view) { |
| 609 const int rid = args[0]->Uint32Value(); |
| 610 const int rid_base = SearchBox::Get(render_view)->rid_base(); |
| 611 // Note that stale rids, less than the current rid_base, will wrap. |
| 612 const size_t index = rid - rid_base; |
| 613 const std::vector<InstantNativeSuggestionsParts>& suggestions = |
| 614 SearchBox::Get(render_view)->native_suggestions(); |
| 615 if (index < suggestions.size()) { |
| 616 // Navigate directly to the selected URL. |
| 617 // TODO(sreeram): SECURITY RISK! Disconnect the InstantLoader and call |
| 618 // SearchBox::Reset to not leak sensitive data to the new page. |
| 619 WebKit::WebURLRequest request(suggestions[index].destination_url); |
| 620 render_view->GetWebView()->mainFrame()->loadRequest(request); |
| 621 } else { |
| 622 VLOG(1) << "Invalid rid " << rid << "; " |
| 623 << "rid_base is " << rid_base << "."; |
| 624 } |
| 625 } |
| 626 return v8::Undefined(); |
| 627 } |
| 628 |
| 629 // static |
| 630 void Dispatch(WebFrame* frame, WebString event_dispatch_script) { |
354 DCHECK(frame) << "Dispatch requires frame"; | 631 DCHECK(frame) << "Dispatch requires frame"; |
355 if (!frame) | 632 if (!frame) return; |
356 return; | 633 frame->executeScript(WebScriptSource(event_dispatch_script)); |
357 | |
358 v8::Handle<v8::Value> result = frame->executeScriptAndReturnValue( | |
359 WebScriptSource(event_dispatch_script)); | |
360 if (result.IsEmpty() || result->IsUndefined() || result->IsNull() || | |
361 result->IsFalse()) { | |
362 frame->executeScript(WebScriptSource(no_event_handler_script)); | |
363 } | |
364 } | 634 } |
365 | 635 |
366 // static | 636 // static |
367 void SearchBoxExtension::DispatchChange(WebFrame* frame) { | 637 void SearchBoxExtension::DispatchChange(WebFrame* frame) { |
368 Dispatch(frame, kDispatchChangeEventScript, kUserInputScript); | 638 Dispatch(frame, kDispatchChangeEventScript); |
369 } | 639 } |
370 | 640 |
371 // static | 641 // static |
372 void SearchBoxExtension::DispatchSubmit(WebFrame* frame) { | 642 void SearchBoxExtension::DispatchSubmit(WebFrame* frame) { |
373 Dispatch(frame, kDispatchSubmitEventScript, kUserDoneScript); | 643 Dispatch(frame, kDispatchSubmitEventScript); |
374 } | 644 } |
375 | 645 |
376 // static | 646 // static |
377 void SearchBoxExtension::DispatchCancel(WebFrame* frame) { | 647 void SearchBoxExtension::DispatchCancel(WebFrame* frame) { |
378 Dispatch(frame, kDispatchCancelEventScript, kUserDoneScript); | 648 Dispatch(frame, kDispatchCancelEventScript); |
379 } | 649 } |
380 | 650 |
381 // static | 651 // static |
382 void SearchBoxExtension::DispatchResize(WebFrame* frame) { | 652 void SearchBoxExtension::DispatchResize(WebFrame* frame) { |
383 Dispatch(frame, kDispatchResizeEventScript, kSetOmniboxBoundsScript); | 653 Dispatch(frame, kDispatchResizeEventScript); |
384 } | 654 } |
385 | 655 |
386 // static | 656 // static |
| 657 void SearchBoxExtension::DispatchNativeSuggestions(WebFrame* frame) { |
| 658 Dispatch(frame, kDispatchNativeSuggestionsEventScript); |
| 659 } |
| 660 |
| 661 // static |
| 662 void SearchBoxExtension::DispatchKeyPress(WebFrame* frame) { |
| 663 Dispatch(frame, kDispatchKeyPressEventScript); |
| 664 } |
| 665 |
| 666 // static |
| 667 void SearchBoxExtension::DispatchFocus(WebFrame* frame) { |
| 668 Dispatch(frame, kDispatchFocusEventScript); |
| 669 } |
| 670 |
| 671 // static |
| 672 void SearchBoxExtension::DispatchBlur(WebFrame* frame) { |
| 673 Dispatch(frame, kDispatchBlurEventScript); |
| 674 } |
| 675 |
| 676 // static |
387 bool SearchBoxExtension::PageSupportsInstant(WebFrame* frame) { | 677 bool SearchBoxExtension::PageSupportsInstant(WebFrame* frame) { |
388 DCHECK(frame) << "PageSupportsInstant requires frame"; | 678 DCHECK(frame) << "PageSupportsInstant requires frame"; |
389 if (!frame) return false; | 679 if (!frame) return false; |
390 | 680 |
391 v8::Handle<v8::Value> v = frame->executeScriptAndReturnValue( | 681 v8::Handle<v8::Value> v = frame->executeScriptAndReturnValue( |
392 WebScriptSource(kSupportsInstantScript)); | 682 WebScriptSource(kSupportsInstantScript)); |
393 bool supports_deprecated_api = !v.IsEmpty() && v->BooleanValue(); | 683 if (!v.IsEmpty() && v->BooleanValue()) { |
394 // TODO(tonyg): Add way of detecting instant support to SearchBox API. | 684 // Send a resize message to tell the page that Chrome is actively using |
395 bool supports_searchbox_api = supports_deprecated_api; | 685 // the searchbox API with it. The page uses the message to transition from |
396 | 686 // "homepage" mode to "search" mode. |
397 // The deprecated API needs to notify the page of events it may have missed. | 687 DispatchResize(frame); |
398 // This isn't necessary in the SearchBox API, since the page can query the | 688 return true; |
399 // API at any time. | |
400 CR_DEFINE_STATIC_LOCAL(std::string, init_script, | |
401 (StringPrintf(kInitScript, kSetOmniboxBoundsScript, kUserInputScript))); | |
402 if (supports_deprecated_api) { | |
403 frame->executeScript(WebScriptSource(WebString::fromUTF8(init_script))); | |
404 } | 689 } |
405 | 690 return false; |
406 return supports_searchbox_api || supports_deprecated_api; | |
407 } | 691 } |
408 | 692 |
409 // static | 693 // static |
410 v8::Extension* SearchBoxExtension::Get() { | 694 v8::Extension* SearchBoxExtension::Get() { |
411 const base::StringPiece code = | 695 const base::StringPiece code = |
412 ResourceBundle::GetSharedInstance().GetRawDataResource( | 696 ResourceBundle::GetSharedInstance().GetRawDataResource( |
413 IDR_SEARCHBOX_API, ui::SCALE_FACTOR_NONE); | 697 IDR_SEARCHBOX_API, ui::SCALE_FACTOR_NONE); |
414 return new SearchBoxExtensionWrapper(code); | 698 return new SearchBoxExtensionWrapper(code); |
415 } | 699 } |
416 | 700 |
417 } // namespace extensions_v8 | 701 } // namespace extensions_v8 |
OLD | NEW |