Index: chrome/browser/prerender/prerender_handle.cc |
diff --git a/chrome/browser/prerender/prerender_handle.cc b/chrome/browser/prerender/prerender_handle.cc |
index 24de791cdb0a9427ffd5ec3343c9d3ac4bcdb62c..5ef9b4d8a269fb948824db4471f43f09a36ca084 100644 |
--- a/chrome/browser/prerender/prerender_handle.cc |
+++ b/chrome/browser/prerender/prerender_handle.cc |
@@ -6,66 +6,118 @@ |
#include <algorithm> |
+#include "base/logging.h" |
#include "chrome/browser/prerender/prerender_contents.h" |
namespace prerender { |
-PrerenderHandle::~PrerenderHandle() { |
- DCHECK(!IsValid()); |
- // This shouldn't occur, but we also shouldn't leak if it does. |
- if (IsValid()) |
- OnCancel(); |
+PrerenderHandle::Observer::Observer() { |
} |
-void PrerenderHandle::OnNavigateAway() { |
- DCHECK(CalledOnValidThread()); |
- if (!IsValid()) |
- return; |
- prerender_data_->OnNavigateAwayByHandle(); |
- prerender_data_.reset(); |
+PrerenderHandle::Observer::~Observer() { |
} |
-void PrerenderHandle::OnCancel() { |
- DCHECK(CalledOnValidThread()); |
- if (!IsValid()) |
- return; |
- prerender_data_->OnCancelByHandle(); |
- prerender_data_.reset(); |
+PrerenderHandle::~PrerenderHandle() { |
+ if (prerender_data_) { |
+ prerender_data_->contents()->RemoveObserver(this); |
+ } |
} |
-bool PrerenderHandle::IsValid() const { |
- return prerender_data_ != NULL; |
+void PrerenderHandle::SetObserver(Observer* observer) { |
+ DCHECK_NE(static_cast<Observer*>(NULL), observer); |
+ observer_ = observer; |
} |
-bool PrerenderHandle::IsPending() const { |
+void PrerenderHandle::OnNavigateAway() { |
DCHECK(CalledOnValidThread()); |
- return prerender_data_ && !prerender_data_->contents(); |
+ if (prerender_data_) |
+ prerender_data_->OnHandleNavigatedAway(this); |
+} |
+ |
+void PrerenderHandle::OnCancel() { |
+ DCHECK(CalledOnValidThread()); |
+ if (prerender_data_) |
+ prerender_data_->OnHandleCanceled(this); |
} |
bool PrerenderHandle::IsPrerendering() const { |
DCHECK(CalledOnValidThread()); |
- return prerender_data_ && prerender_data_->contents(); |
+ return prerender_data_ != NULL; |
} |
bool PrerenderHandle::IsFinishedLoading() const { |
DCHECK(CalledOnValidThread()); |
- if (!prerender_data_ || IsPending()) |
+ if (!prerender_data_) |
return false; |
return prerender_data_->contents()->has_finished_loading(); |
} |
PrerenderHandle::PrerenderHandle( |
PrerenderManager::PrerenderData* prerender_data) |
- : prerender_data_(prerender_data->AsWeakPtr()), |
+ : observer_(NULL), |
weak_ptr_factory_(this) { |
- prerender_data->OnNewHandle(); |
+ if (prerender_data) { |
+ prerender_data_ = prerender_data->AsWeakPtr(); |
+ prerender_data->OnHandleCreated(this); |
+ } |
+} |
+ |
+void PrerenderHandle::AdoptPrerenderDataFrom(PrerenderHandle* other_handle) { |
+ DCHECK_EQ(static_cast<PrerenderManager::PrerenderData*>(NULL), |
+ prerender_data_); |
+ if (other_handle->prerender_data_ && |
+ other_handle->prerender_data_->contents()) { |
+ other_handle->prerender_data_->contents()->RemoveObserver(other_handle); |
+ } |
+ |
+ prerender_data_ = other_handle->prerender_data_; |
+ other_handle->prerender_data_.reset(); |
+ |
+ if (prerender_data_) { |
+ DCHECK_NE(static_cast<PrerenderContents*>(NULL), |
+ prerender_data_->contents()); |
+ prerender_data_->contents()->AddObserver(this); |
+ // We are joining a prerender that has already started so we fire off an |
+ // extra start event at ourselves. |
+ OnPrerenderStart(prerender_data_->contents()); |
+ } |
+} |
+ |
+void PrerenderHandle::OnPrerenderStart(PrerenderContents* prerender_contents) { |
+ DCHECK(CalledOnValidThread()); |
+ DCHECK(prerender_data_); |
+ DCHECK_EQ(prerender_data_->contents(), prerender_contents); |
+ if (observer_) |
+ observer_->OnPrerenderStart(this); |
+} |
+ |
+void PrerenderHandle::OnPrerenderAddAlias(PrerenderContents* prerender_contents, |
+ const GURL& alias_url) { |
+ DCHECK(CalledOnValidThread()); |
+ DCHECK(prerender_data_); |
+ DCHECK_EQ(prerender_data_->contents(), prerender_contents); |
+ if (observer_) |
+ observer_->OnPrerenderAddAlias(this, alias_url); |
} |
-void PrerenderHandle::SwapPrerenderDataWith( |
- PrerenderHandle* other_prerender_handle) { |
+void PrerenderHandle::OnPrerenderStop(PrerenderContents* prerender_contents) { |
DCHECK(CalledOnValidThread()); |
- DCHECK(other_prerender_handle); |
- std::swap(prerender_data_, other_prerender_handle->prerender_data_); |
+ if (observer_) |
+ observer_->OnPrerenderStop(this); |
+} |
+ |
+void PrerenderHandle::OnPrerenderCreatedMatchCompleteReplacement( |
+ PrerenderContents* contents, PrerenderContents* replacement) { |
+ DCHECK(CalledOnValidThread()); |
+ |
+ // This should occur in the middle of the surgery on the PrerenderData, and |
+ // so we expect to not have our new contents in our PrerenderData yet. The |
+ // switch occurs in |
+ // PrerenderManager::PrerenderData::MakeIntoMatchCompleteReplacement, so |
+ // this method only needs to switch observing. |
+ |
+ contents->RemoveObserver(this); |
+ replacement->AddObserver(this); |
} |
} // namespace prerender |