2017-05-19 19:35:13 +00:00
|
|
|
// Copyright (c) 2017 GitHub, Inc.
|
|
|
|
// Use of this source code is governed by the MIT license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
|
|
#include "atom/renderer/atom_autofill_agent.h"
|
|
|
|
|
2017-05-19 19:52:18 +00:00
|
|
|
#include <vector>
|
|
|
|
|
2017-05-19 19:35:13 +00:00
|
|
|
#include "atom/common/api/api_messages.h"
|
|
|
|
#include "content/public/renderer/render_frame.h"
|
|
|
|
#include "content/public/renderer/render_view.h"
|
|
|
|
#include "third_party/WebKit/public/platform/WebKeyboardEvent.h"
|
|
|
|
#include "third_party/WebKit/public/platform/WebString.h"
|
|
|
|
#include "third_party/WebKit/public/web/WebDocument.h"
|
|
|
|
#include "third_party/WebKit/public/web/WebLocalFrame.h"
|
|
|
|
#include "third_party/WebKit/public/web/WebOptionElement.h"
|
|
|
|
#include "third_party/WebKit/public/web/WebUserGestureIndicator.h"
|
|
|
|
#include "ui/events/keycodes/keyboard_codes.h"
|
|
|
|
#include "ui/gfx/geometry/rect_f.h"
|
|
|
|
|
|
|
|
namespace atom {
|
2017-05-19 19:52:18 +00:00
|
|
|
|
2017-05-19 19:35:13 +00:00
|
|
|
namespace {
|
|
|
|
const size_t kMaxDataLength = 1024;
|
|
|
|
const size_t kMaxListSize = 512;
|
2017-05-19 19:52:18 +00:00
|
|
|
|
2017-05-19 19:35:13 +00:00
|
|
|
void GetDataListSuggestions(const blink::WebInputElement& element,
|
|
|
|
std::vector<base::string16>* values,
|
|
|
|
std::vector<base::string16>* labels) {
|
|
|
|
for (const auto& option : element.filteredDataListOptions()) {
|
|
|
|
values->push_back(option.value().utf16());
|
|
|
|
if (option.value() != option.label())
|
|
|
|
labels->push_back(option.label().utf16());
|
|
|
|
else
|
|
|
|
labels->push_back(base::string16());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void TrimStringVectorForIPC(std::vector<base::string16>* strings) {
|
|
|
|
// Limit the size of the vector.
|
|
|
|
if (strings->size() > kMaxListSize)
|
|
|
|
strings->resize(kMaxListSize);
|
|
|
|
|
|
|
|
// Limit the size of the strings in the vector.
|
|
|
|
for (size_t i = 0; i < strings->size(); ++i) {
|
|
|
|
if ((*strings)[i].length() > kMaxDataLength)
|
|
|
|
(*strings)[i].resize(kMaxDataLength);
|
|
|
|
}
|
|
|
|
}
|
2017-05-19 19:52:18 +00:00
|
|
|
} // namespace
|
2017-05-19 19:35:13 +00:00
|
|
|
|
|
|
|
AutofillAgent::AutofillAgent(
|
|
|
|
content::RenderFrame* frame)
|
|
|
|
: content::RenderFrameObserver(frame),
|
2017-05-20 02:21:11 +00:00
|
|
|
helper_(new Helper(this)),
|
|
|
|
focused_node_was_last_clicked_(false),
|
|
|
|
was_focused_before_now_(false),
|
2017-05-19 19:35:13 +00:00
|
|
|
weak_ptr_factory_(this) {
|
2017-05-26 01:38:27 +00:00
|
|
|
render_frame()->GetWebFrame()->setAutofillClient(this);
|
2017-05-20 02:21:11 +00:00
|
|
|
}
|
|
|
|
|
2017-05-19 19:35:13 +00:00
|
|
|
void AutofillAgent::OnDestruct() {
|
|
|
|
delete this;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutofillAgent::DidChangeScrollOffset() {
|
|
|
|
HidePopup();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutofillAgent::FocusedNodeChanged(const blink::WebNode&) {
|
2017-05-21 17:55:19 +00:00
|
|
|
focused_node_was_last_clicked_ = false;
|
2017-05-20 02:21:11 +00:00
|
|
|
was_focused_before_now_ = false;
|
2017-05-19 19:35:13 +00:00
|
|
|
HidePopup();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutofillAgent::textFieldDidEndEditing(
|
|
|
|
const blink::WebInputElement&) {
|
|
|
|
HidePopup();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutofillAgent::textFieldDidChange(
|
2017-05-19 19:52:18 +00:00
|
|
|
const blink::WebFormControlElement& element) {
|
2017-05-19 19:35:13 +00:00
|
|
|
if (!IsUserGesture() && !render_frame()->IsPasting())
|
|
|
|
return;
|
2017-05-19 19:52:18 +00:00
|
|
|
|
2017-05-19 19:35:13 +00:00
|
|
|
weak_ptr_factory_.InvalidateWeakPtrs();
|
|
|
|
base::ThreadTaskRunnerHandle::Get()->PostTask(
|
|
|
|
FROM_HERE, base::Bind(&AutofillAgent::textFieldDidChangeImpl,
|
|
|
|
weak_ptr_factory_.GetWeakPtr(), element));
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutofillAgent::textFieldDidChangeImpl(
|
2017-05-19 19:52:18 +00:00
|
|
|
const blink::WebFormControlElement& element) {
|
2017-05-19 19:35:13 +00:00
|
|
|
ShowSuggestionsOptions options;
|
|
|
|
options.requires_caret_at_end = true;
|
|
|
|
ShowSuggestions(element, options);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutofillAgent::textFieldDidReceiveKeyDown(
|
2017-05-19 19:52:18 +00:00
|
|
|
const blink::WebInputElement& element,
|
2017-05-19 19:35:13 +00:00
|
|
|
const blink::WebKeyboardEvent& event) {
|
|
|
|
if (event.windowsKeyCode == ui::VKEY_DOWN ||
|
|
|
|
event.windowsKeyCode == ui::VKEY_UP) {
|
|
|
|
ShowSuggestionsOptions options;
|
|
|
|
options.autofill_on_empty_values = true;
|
|
|
|
options.requires_caret_at_end = true;
|
|
|
|
ShowSuggestions(element, options);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutofillAgent::openTextDataListChooser(
|
|
|
|
const blink::WebInputElement& element) {
|
|
|
|
ShowSuggestionsOptions options;
|
|
|
|
options.autofill_on_empty_values = true;
|
|
|
|
ShowSuggestions(element, options);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutofillAgent::dataListOptionsChanged(
|
|
|
|
const blink::WebInputElement& element) {
|
|
|
|
if (!element.focused())
|
|
|
|
return;
|
2017-05-19 19:52:18 +00:00
|
|
|
|
2017-05-19 19:35:13 +00:00
|
|
|
ShowSuggestionsOptions options;
|
|
|
|
options.requires_caret_at_end = true;
|
|
|
|
ShowSuggestions(element, options);
|
|
|
|
}
|
|
|
|
|
|
|
|
AutofillAgent::ShowSuggestionsOptions::ShowSuggestionsOptions()
|
|
|
|
: autofill_on_empty_values(false),
|
|
|
|
requires_caret_at_end(false) {
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutofillAgent::ShowSuggestions(
|
|
|
|
const blink::WebFormControlElement& element,
|
|
|
|
const ShowSuggestionsOptions& options) {
|
|
|
|
if (!element.isEnabled() || element.isReadOnly())
|
|
|
|
return;
|
|
|
|
const blink::WebInputElement* input_element = toWebInputElement(&element);
|
|
|
|
if (input_element) {
|
|
|
|
if (!input_element->isTextField())
|
|
|
|
return;
|
|
|
|
}
|
2017-05-19 19:52:18 +00:00
|
|
|
|
2017-05-19 19:35:13 +00:00
|
|
|
blink::WebString value = element.editingValue();
|
|
|
|
if (value.length() > kMaxDataLength ||
|
|
|
|
(!options.autofill_on_empty_values && value.isEmpty()) ||
|
|
|
|
(options.requires_caret_at_end &&
|
|
|
|
(element.selectionStart() != element.selectionEnd() ||
|
|
|
|
element.selectionEnd() != static_cast<int>(value.length())))) {
|
|
|
|
HidePopup();
|
|
|
|
return;
|
|
|
|
}
|
2017-05-19 19:52:18 +00:00
|
|
|
|
2017-05-19 19:35:13 +00:00
|
|
|
std::vector<base::string16> data_list_values;
|
|
|
|
std::vector<base::string16> data_list_labels;
|
|
|
|
if (input_element) {
|
|
|
|
GetDataListSuggestions(
|
|
|
|
*input_element, &data_list_values, &data_list_labels);
|
|
|
|
TrimStringVectorForIPC(&data_list_values);
|
|
|
|
TrimStringVectorForIPC(&data_list_labels);
|
|
|
|
}
|
2017-05-19 19:52:18 +00:00
|
|
|
|
2017-05-19 19:35:13 +00:00
|
|
|
ShowPopup(element, data_list_values, data_list_labels);
|
|
|
|
}
|
|
|
|
|
2017-05-20 02:21:11 +00:00
|
|
|
AutofillAgent::Helper::Helper(AutofillAgent* agent)
|
2017-05-26 01:38:27 +00:00
|
|
|
: content::RenderViewObserver(agent->render_frame()->GetRenderView()),
|
2017-05-20 02:21:11 +00:00
|
|
|
agent_(agent) {
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutofillAgent::Helper::OnMouseDown(const blink::WebNode& node) {
|
|
|
|
agent_->focused_node_was_last_clicked_ = !node.isNull() && node.focused();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutofillAgent::Helper::FocusChangeComplete() {
|
|
|
|
agent_->DoFocusChangeComplete();
|
2017-05-19 19:35:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool AutofillAgent::OnMessageReceived(const IPC::Message& message) {
|
|
|
|
bool handled = true;
|
|
|
|
IPC_BEGIN_MESSAGE_MAP(AutofillAgent, message)
|
2017-05-26 01:38:27 +00:00
|
|
|
IPC_MESSAGE_HANDLER(AtomAutofillFrameMsg_AcceptSuggestion,
|
2017-05-19 19:35:13 +00:00
|
|
|
OnAcceptSuggestion)
|
|
|
|
IPC_MESSAGE_UNHANDLED(handled = false)
|
|
|
|
IPC_END_MESSAGE_MAP()
|
|
|
|
|
|
|
|
return handled;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AutofillAgent::IsUserGesture() const {
|
|
|
|
return blink::WebUserGestureIndicator::isProcessingUserGesture();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutofillAgent::HidePopup() {
|
2017-05-26 01:38:27 +00:00
|
|
|
Send(new AtomAutofillFrameHostMsg_HidePopup(render_frame()->GetRoutingID()));
|
2017-05-19 19:35:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void AutofillAgent::ShowPopup(
|
|
|
|
const blink::WebFormControlElement& element,
|
|
|
|
const std::vector<base::string16>& values,
|
|
|
|
const std::vector<base::string16>& labels) {
|
2017-05-19 19:52:18 +00:00
|
|
|
gfx::RectF bounds =
|
2017-05-26 01:38:27 +00:00
|
|
|
render_frame()->GetRenderView()->ElementBoundsInWindow(element);
|
|
|
|
Send(new AtomAutofillFrameHostMsg_ShowPopup(
|
|
|
|
render_frame()->GetRoutingID(), bounds, values, labels));
|
2017-05-19 19:35:13 +00:00
|
|
|
}
|
|
|
|
|
2017-05-20 02:21:11 +00:00
|
|
|
void AutofillAgent::OnAcceptSuggestion(base::string16 suggestion) {
|
2017-05-26 01:38:27 +00:00
|
|
|
auto element = render_frame()->GetWebFrame()->document().focusedElement();
|
2017-05-20 02:21:11 +00:00
|
|
|
if (element.isFormControlElement()) {
|
2017-05-21 17:55:19 +00:00
|
|
|
toWebInputElement(&element)->setSuggestedValue(
|
2017-05-20 02:21:11 +00:00
|
|
|
blink::WebString::fromUTF16(suggestion));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutofillAgent::DoFocusChangeComplete() {
|
2017-05-26 01:38:27 +00:00
|
|
|
auto element = render_frame()->GetWebFrame()->document().focusedElement();
|
2017-05-20 02:21:11 +00:00
|
|
|
if (element.isNull() || !element.isFormControlElement())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (focused_node_was_last_clicked_ && was_focused_before_now_) {
|
|
|
|
ShowSuggestionsOptions options;
|
|
|
|
options.autofill_on_empty_values = true;
|
2017-05-21 17:55:19 +00:00
|
|
|
auto input_element = toWebInputElement(&element);
|
|
|
|
if (input_element)
|
|
|
|
ShowSuggestions(*input_element, options);
|
2017-05-20 02:21:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
was_focused_before_now_ = true;
|
|
|
|
focused_node_was_last_clicked_ = false;
|
|
|
|
}
|
|
|
|
|
2017-05-19 19:52:18 +00:00
|
|
|
} // namespace atom
|