blob: b4c766955c2f0c6ebda9cf56a3eac0b898235871 [file] [log] [blame]
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "android_webview/browser/find_helper.h"
#include "android_webview/browser/scoped_allow_wait_for_legacy_web_view_api.h"
#include "base/message_loop.h"
#include "content/public/browser/render_view_host.h"
#include "content/public/browser/web_contents.h"
#include "content/public/common/stop_find_action.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/WebFindOptions.h"
using content::WebContents;
using WebKit::WebFindOptions;
namespace android_webview {
FindHelper::FindHelper(WebContents* web_contents)
: WebContentsObserver(web_contents),
listener_(NULL),
async_find_started_(false),
sync_find_started_(false),
find_request_id_counter_(0),
current_request_id_(0),
last_match_count_(-1),
last_active_ordinal_(-1),
weak_factory_(this) {
}
FindHelper::~FindHelper() {
}
void FindHelper::SetListener(Listener* listener) {
listener_ = listener;
}
int FindHelper::FindAllSync(const string16& search_string) {
sync_find_started_ = true;
async_find_started_ = false;
WebFindOptions options;
options.forward = true;
options.matchCase = false;
options.findNext = false;
int match_count = 0;
int active_ordinal = 0;
StartNewRequest(search_string);
// Any ongoing asynchronous requests will be stopped in the renderer when
// calling SynchronousFind. Using the asynchronous StopFinding message could
// lead to deadblocks as the message could arrive in the middle of the
// synchronous operation and cancel the reply back.
ScopedAllowWaitForLegacyWebViewApi wait;
web_contents()->GetRenderViewHost()->SynchronousFind(current_request_id_,
search_string,
options,
&match_count,
&active_ordinal);
// Post the task to ourselves to prevent trigerring the notification before
// we actually return from the request.
MessageLoop::current()->PostTask(
FROM_HERE,
base::Bind(&FindHelper::NotifyResults, weak_factory_.GetWeakPtr(),
active_ordinal, match_count, true));
return match_count;
}
void FindHelper::FindAllAsync(const string16& search_string) {
// Stop any ongoing asynchronous request.
web_contents()->GetRenderViewHost()->StopFinding(
content::STOP_FIND_ACTION_KEEP_SELECTION);
sync_find_started_ = false;
async_find_started_ = true;
WebFindOptions options;
options.forward = true;
options.matchCase = false;
options.findNext = false;
StartNewRequest(search_string);
web_contents()->GetRenderViewHost()->Find(current_request_id_,
search_string, options);
}
void FindHelper::HandleFindReply(int request_id,
int match_count,
int active_ordinal,
bool finished) {
if ((!async_find_started_ && !sync_find_started_) ||
request_id != current_request_id_) {
return;
}
NotifyResults(active_ordinal, match_count, finished);
}
void FindHelper::FindNext(bool forward) {
if (!sync_find_started_ && !async_find_started_)
return;
WebFindOptions options;
options.forward = forward;
options.matchCase = false;
options.findNext = true;
web_contents()->GetRenderViewHost()->Find(current_request_id_,
last_search_string_,
options);
}
void FindHelper::ClearMatches() {
web_contents()->GetRenderViewHost()->StopFinding(
content::STOP_FIND_ACTION_CLEAR_SELECTION);
sync_find_started_ = false;
async_find_started_ = false;
last_search_string_.clear();
last_match_count_ = -1;
last_active_ordinal_ = -1;
}
void FindHelper::StartNewRequest(const string16& search_string) {
current_request_id_ = find_request_id_counter_++;
last_search_string_ = search_string;
last_match_count_ = -1;
last_active_ordinal_ = -1;
}
void FindHelper::NotifyResults(int active_ordinal,
int match_count,
bool finished) {
// Match count or ordinal values set to -1 refer to the received replies.
if (match_count == -1)
match_count = last_match_count_;
else
last_match_count_ = match_count;
if (active_ordinal == -1)
active_ordinal = last_active_ordinal_;
else
last_active_ordinal_ = active_ordinal;
// Skip the update if we don't still have a valid ordinal.
// The next update, final or not, should have this information.
if (!finished && active_ordinal == -1)
return;
// Safeguard in case of errors to prevent reporting -1 to the API listeners.
if (match_count == -1) {
NOTREACHED();
match_count = 0;
}
if (active_ordinal == -1) {
NOTREACHED();
active_ordinal = 0;
}
// WebView.FindListener active match ordinals are 0-based while WebKit sends
// 1-based ordinals. Still we can receive 0 ordinal in case of no results.
active_ordinal = std::max(active_ordinal - 1, 0);
if (listener_)
listener_->OnFindResultReceived(active_ordinal, match_count, finished);
}
} // namespace android_webview