// Copyright (c) 2010 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 "chrome/browser/dom_ui/foreign_session_handler.h" #include "base/scoped_vector.h" #include "base/utf_string_conversions.h" #include "chrome/browser/browser.h" #include "chrome/browser/dom_ui/value_helper.h" #include "chrome/browser/sessions/session_restore.h" #include "chrome/browser/sessions/tab_restore_service.h" #include "chrome/browser/sync/engine/syncapi.h" #include "chrome/browser/sync/profile_sync_service.h" #include "chrome/browser/tab_contents/tab_contents.h" #include "chrome/common/notification_details.h" #include "chrome/common/notification_service.h" namespace browser_sync { static const int kMaxSessionsToShow = 10; ForeignSessionHandler::ForeignSessionHandler() { Init(); } void ForeignSessionHandler::RegisterMessages() { dom_ui_->RegisterMessageCallback("getForeignSessions", NewCallback(this, &ForeignSessionHandler::HandleGetForeignSessions)); dom_ui_->RegisterMessageCallback("reopenForeignSession", NewCallback(this, &ForeignSessionHandler::HandleReopenForeignSession)); } void ForeignSessionHandler::Init() { registrar_.Add(this, NotificationType::SYNC_CONFIGURE_DONE, NotificationService::AllSources()); registrar_.Add(this, NotificationType::FOREIGN_SESSION_UPDATED, NotificationService::AllSources()); registrar_.Add(this, NotificationType::FOREIGN_SESSION_DISABLED, NotificationService::AllSources()); } void ForeignSessionHandler::Observe(NotificationType type, const NotificationSource& source, const NotificationDetails& details) { ListValue list_value; switch (type.value) { case NotificationType::SYNC_CONFIGURE_DONE: case NotificationType::FOREIGN_SESSION_UPDATED: HandleGetForeignSessions(&list_value); break; case NotificationType::FOREIGN_SESSION_DISABLED: // Calling foreignSessions with empty list will automatically hide // foreign session section. dom_ui_->CallJavascriptFunction(L"foreignSessions", list_value); break; default: NOTREACHED(); } } SessionModelAssociator* ForeignSessionHandler::GetModelAssociator() { ProfileSyncService* service = dom_ui_->GetProfile()->GetProfileSyncService(); if (service == NULL) return NULL; // We only want to set the model associator if there is one, and it is done // syncing sessions. SessionModelAssociator* model_associator = service-> GetSessionModelAssociator(); if (model_associator == NULL || !service->ShouldPushChanges()) { return NULL; } return dom_ui_->GetProfile()->GetProfileSyncService()-> GetSessionModelAssociator(); } void ForeignSessionHandler::HandleGetForeignSessions(const ListValue* args) { SessionModelAssociator* associator = GetModelAssociator(); if (associator) GetForeignSessions(associator); } void ForeignSessionHandler::HandleReopenForeignSession( const ListValue* args) { // Extract the machine tag and use it to obtain the id for the node we are // looking for. Send it along with a valid associator to OpenForeignSessions. std::string session_string_value = WideToUTF8(ExtractStringValue(args)); SessionModelAssociator* associator = GetModelAssociator(); if (associator && !session_string_value.empty()) { int64 id = associator->GetSyncIdFromChromeId(session_string_value); OpenForeignSession(associator, id); } } void ForeignSessionHandler::OpenForeignSession( SessionModelAssociator* associator, int64 id) { // Obtain the session windows for the foreign session. // We don't have a ForeignSessionHandler in off the record mode, so we // expect the ProfileSyncService to exist. sync_api::ReadTransaction trans(dom_ui_->GetProfile()-> GetProfileSyncService()->backend()->GetUserShareHandle()); ScopedVector session; associator->AppendForeignSessionWithID(id, &session.get(), &trans); DCHECK(session.size() == 1); std::vector windows = (*session.begin())->windows; SessionRestore::RestoreForeignSessionWindows(dom_ui_->GetProfile(), &windows); } void ForeignSessionHandler::GetForeignSessions( SessionModelAssociator* associator) { ScopedVector clients; if (!associator->GetSessionData(&clients.get())) { LOG(ERROR) << "ForeignSessionHandler failed to get session data from" "SessionModelAssociator."; return; } int added_count = 0; ListValue client_list; for (std::vector::const_iterator i = clients->begin(); i != clients->end() && added_count < kMaxSessionsToShow; ++i) { ForeignSession* foreign_session = *i; std::vector entries; dom_ui_->GetProfile()->GetTabRestoreService()->CreateEntriesFromWindows( &foreign_session->windows, &entries); scoped_ptr window_list(new ListValue()); for (std::vector::const_iterator it = entries.begin(); it != entries.end(); ++it) { TabRestoreService::Entry* entry = *it; scoped_ptr window_data(new DictionaryValue()); if (entry->type == TabRestoreService::WINDOW && ValueHelper::WindowToValue( *static_cast(entry), window_data.get())) { // The javascript checks if the session id is a valid session id, // when rendering session information to the new tab page, and it // sends the sessionTag back when we need to restore a session. // TODO(zea): sessionTag is per client, it might be better per window. window_data->SetString("sessionTag", foreign_session->foreign_session_tag); window_data->SetInteger("sessionId", entry->id); // Give ownership to |list_value|. window_list->Append(window_data.release()); } } added_count++; // Give ownership to |client_list| client_list.Append(window_list.release()); } dom_ui_->CallJavascriptFunction(L"foreignSessions", client_list); } } // namespace browser_sync