// 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 "chrome/browser/ui/webui/ntp/foreign_session_handler.h" #include #include #include #include "base/bind.h" #include "base/bind_helpers.h" #include "base/i18n/time_formatting.h" #include "base/memory/scoped_vector.h" #include "base/prefs/pref_service.h" #include "base/strings/string_number_conversions.h" #include "base/utf_string_conversions.h" #include "base/values.h" #include "chrome/browser/prefs/pref_registry_syncable.h" #include "chrome/browser/prefs/scoped_user_pref_update.h" #include "chrome/browser/profiles/profile.h" #include "chrome/browser/sessions/session_restore.h" #include "chrome/browser/sync/profile_sync_service.h" #include "chrome/browser/sync/profile_sync_service_factory.h" #include "chrome/browser/ui/host_desktop.h" #include "chrome/browser/ui/webui/ntp/new_tab_ui.h" #include "chrome/browser/ui/webui/session_favicon_source.h" #include "chrome/common/chrome_notification_types.h" #include "chrome/common/pref_names.h" #include "chrome/common/time_format.h" #include "chrome/common/url_constants.h" #include "content/public/browser/notification_service.h" #include "content/public/browser/notification_source.h" #include "content/public/browser/url_data_source.h" #include "content/public/browser/web_contents.h" #include "content/public/browser/web_contents_view.h" #include "content/public/browser/web_ui.h" #include "grit/generated_resources.h" #include "ui/base/l10n/l10n_util.h" #include "ui/webui/web_ui_util.h" namespace browser_sync { // Maximum number of sessions we're going to display on the NTP static const size_t kMaxSessionsToShow = 10; namespace { // Comparator function for use with std::sort that will sort sessions by // descending modified_time (i.e., most recent first). bool SortSessionsByRecency(const SyncedSession* s1, const SyncedSession* s2) { return s1->modified_time > s2->modified_time; } } // namepace ForeignSessionHandler::ForeignSessionHandler() { } // static void ForeignSessionHandler::RegisterUserPrefs(PrefRegistrySyncable* registry) { registry->RegisterDictionaryPref(prefs::kNtpCollapsedForeignSessions, PrefRegistrySyncable::UNSYNCABLE_PREF); } // static void ForeignSessionHandler::OpenForeignSessionTab( content::WebUI* web_ui, const std::string& session_string_value, SessionID::id_type window_num, SessionID::id_type tab_id, const WindowOpenDisposition& disposition) { SessionModelAssociator* associator = GetModelAssociator(web_ui); if (!associator) return; // We don't actually care about |window_num|, this is just a sanity check. DCHECK_LT(kInvalidId, window_num); const SessionTab* tab; if (!associator->GetForeignTab(session_string_value, tab_id, &tab)) { LOG(ERROR) << "Failed to load foreign tab."; return; } if (tab->navigations.empty()) { LOG(ERROR) << "Foreign tab no longer has valid navigations."; return; } SessionRestore::RestoreForeignSessionTab( web_ui->GetWebContents(), *tab, disposition); } // static void ForeignSessionHandler::OpenForeignSessionWindows( content::WebUI* web_ui, const std::string& session_string_value, SessionID::id_type window_num) { SessionModelAssociator* associator = GetModelAssociator(web_ui); if (!associator) return; std::vector windows; // Note: we don't own the ForeignSessions themselves. if (!associator->GetForeignSession(session_string_value, &windows)) { LOG(ERROR) << "ForeignSessionHandler failed to get session data from" "SessionModelAssociator."; return; } std::vector::const_iterator iter_begin = windows.begin() + (window_num == kInvalidId ? 0 : window_num); std::vector::const_iterator iter_end = window_num == kInvalidId ? std::vector::const_iterator(windows.end()) : iter_begin + 1; chrome::HostDesktopType host_desktop_type = chrome::GetHostDesktopTypeForNativeView( web_ui->GetWebContents()->GetView()->GetNativeView()); SessionRestore::RestoreForeignSessionWindows( Profile::FromWebUI(web_ui), host_desktop_type, iter_begin, iter_end); } // static bool ForeignSessionHandler::SessionTabToValue( const SessionTab& tab, DictionaryValue* dictionary) { if (tab.navigations.empty()) return false; int selected_index = std::min(tab.current_navigation_index, static_cast(tab.navigations.size() - 1)); const TabNavigation& current_navigation = tab.navigations.at(selected_index); GURL tab_url = current_navigation.virtual_url(); if (tab_url == GURL(chrome::kChromeUINewTabURL)) return false; NewTabUI::SetUrlTitleAndDirection(dictionary, current_navigation.title(), tab_url); dictionary->SetString("type", "tab"); dictionary->SetDouble("timestamp", static_cast(tab.timestamp.ToInternalValue())); // TODO(jeremycho): This should probably be renamed to tabId to avoid // confusion with the ID corresponding to a session. Investigate all the // places (C++ and JS) where this is being used. (http://crbug.com/154865). dictionary->SetInteger("sessionId", tab.tab_id.id()); return true; } // static SessionModelAssociator* ForeignSessionHandler::GetModelAssociator( content::WebUI* web_ui) { Profile* profile = Profile::FromWebUI(web_ui); ProfileSyncService* service = ProfileSyncServiceFactory::GetInstance()->GetForProfile(profile); // Only return the associator if it exists and it is done syncing sessions. if (service && service->ShouldPushChanges()) return service->GetSessionModelAssociator(); return NULL; } void ForeignSessionHandler::RegisterMessages() { Init(); web_ui()->RegisterMessageCallback("deleteForeignSession", base::Bind(&ForeignSessionHandler::HandleDeleteForeignSession, base::Unretained(this))); web_ui()->RegisterMessageCallback("getForeignSessions", base::Bind(&ForeignSessionHandler::HandleGetForeignSessions, base::Unretained(this))); web_ui()->RegisterMessageCallback("openForeignSession", base::Bind(&ForeignSessionHandler::HandleOpenForeignSession, base::Unretained(this))); web_ui()->RegisterMessageCallback("setForeignSessionCollapsed", base::Bind(&ForeignSessionHandler::HandleSetForeignSessionCollapsed, base::Unretained(this))); } void ForeignSessionHandler::Init() { Profile* profile = Profile::FromWebUI(web_ui()); ProfileSyncService* service = ProfileSyncServiceFactory::GetInstance()->GetForProfile(profile); registrar_.Add(this, chrome::NOTIFICATION_SYNC_CONFIGURE_DONE, content::Source(service)); registrar_.Add(this, chrome::NOTIFICATION_FOREIGN_SESSION_UPDATED, content::Source(profile)); registrar_.Add(this, chrome::NOTIFICATION_FOREIGN_SESSION_DISABLED, content::Source(profile)); // Add the data source for synced favicons. content::URLDataSource::Add(profile, new SessionFaviconSource(profile)); } void ForeignSessionHandler::Observe( int type, const content::NotificationSource& source, const content::NotificationDetails& details) { ListValue list_value; switch (type) { case chrome::NOTIFICATION_FOREIGN_SESSION_DISABLED: // Tab sync is disabled, so clean up data about collapsed sessions. Profile::FromWebUI(web_ui())->GetPrefs()->ClearPref( prefs::kNtpCollapsedForeignSessions); // Fall through. case chrome::NOTIFICATION_SYNC_CONFIGURE_DONE: case chrome::NOTIFICATION_FOREIGN_SESSION_UPDATED: HandleGetForeignSessions(&list_value); break; default: NOTREACHED(); } } bool ForeignSessionHandler::IsTabSyncEnabled() { Profile* profile = Profile::FromWebUI(web_ui()); ProfileSyncService* service = ProfileSyncServiceFactory::GetInstance()->GetForProfile(profile); return service && service->GetPreferredDataTypes().Has(syncer::SESSIONS); } string16 ForeignSessionHandler::FormatSessionTime(const base::Time& time) { // Return a time like "1 hour ago", "2 days ago", etc. base::Time now = base::Time::Now(); // TimeElapsed does not support negative TimeDelta values, so then we use 0. return TimeFormat::TimeElapsed(now < time ? base::TimeDelta() : now - time); } void ForeignSessionHandler::HandleGetForeignSessions(const ListValue* args) { SessionModelAssociator* associator = GetModelAssociator(web_ui()); std::vector sessions; ListValue session_list; if (associator && associator->GetAllForeignSessions(&sessions)) { // Sort sessions from most recent to least recent. std::sort(sessions.begin(), sessions.end(), SortSessionsByRecency); // Use a pref to keep track of sessions that were collapsed by the user. // To prevent the pref from accumulating stale sessions, clear it each time // and only add back sessions that are still current. DictionaryPrefUpdate pref_update(Profile::FromWebUI(web_ui())->GetPrefs(), prefs::kNtpCollapsedForeignSessions); DictionaryValue* current_collapsed_sessions = pref_update.Get(); scoped_ptr collapsed_sessions( current_collapsed_sessions->DeepCopy()); current_collapsed_sessions->Clear(); // Note: we don't own the SyncedSessions themselves. for (size_t i = 0; i < sessions.size() && i < kMaxSessionsToShow; ++i) { const SyncedSession* session = sessions[i]; const std::string& session_tag = session->session_tag; scoped_ptr session_data(new DictionaryValue()); session_data->SetString("tag", session_tag); session_data->SetString("name", session->session_name); session_data->SetString("deviceType", session->DeviceTypeAsString()); session_data->SetString("modifiedTime", FormatSessionTime(session->modified_time)); bool is_collapsed = collapsed_sessions->HasKey(session_tag); session_data->SetBoolean("collapsed", is_collapsed); if (is_collapsed) current_collapsed_sessions->SetBoolean(session_tag, true); scoped_ptr window_list(new ListValue()); for (SyncedSession::SyncedWindowMap::const_iterator it = session->windows.begin(); it != session->windows.end(); ++it) { SessionWindow* window = it->second; scoped_ptr window_data(new DictionaryValue()); if (SessionWindowToValue(*window, window_data.get())) window_list->Append(window_data.release()); } session_data->Set("windows", window_list.release()); session_list.Append(session_data.release()); } } base::FundamentalValue tab_sync_enabled(IsTabSyncEnabled()); web_ui()->CallJavascriptFunction("ntp.setForeignSessions", session_list, tab_sync_enabled); } void ForeignSessionHandler::HandleOpenForeignSession(const ListValue* args) { size_t num_args = args->GetSize(); // Expect either 1 or 8 args. For restoring an entire session, only // one argument is required -- the session tag. To restore a tab, // the additional args required are the window id, the tab id, // and 4 properties of the event object (button, altKey, ctrlKey, // metaKey, shiftKey) for determining how to open the tab. if (num_args != 8U && num_args != 1U) { LOG(ERROR) << "openForeignSession called with " << args->GetSize() << " arguments."; return; } // Extract the session tag (always provided). std::string session_string_value; if (!args->GetString(0, &session_string_value)) { LOG(ERROR) << "Failed to extract session tag."; return; } // Extract window number. std::string window_num_str; int window_num = kInvalidId; if (num_args >= 2 && (!args->GetString(1, &window_num_str) || !base::StringToInt(window_num_str, &window_num))) { LOG(ERROR) << "Failed to extract window number."; return; } // Extract tab id. std::string tab_id_str; SessionID::id_type tab_id = kInvalidId; if (num_args >= 3 && (!args->GetString(2, &tab_id_str) || !base::StringToInt(tab_id_str, &tab_id))) { LOG(ERROR) << "Failed to extract tab SessionID."; return; } if (tab_id != kInvalidId) { WindowOpenDisposition disposition = webui::GetDispositionFromClick(args, 3); OpenForeignSessionTab( web_ui(), session_string_value, window_num, tab_id, disposition); } else { OpenForeignSessionWindows(web_ui(), session_string_value, window_num); } } void ForeignSessionHandler::HandleDeleteForeignSession(const ListValue* args) { if (args->GetSize() != 1U) { LOG(ERROR) << "Wrong number of args to deleteForeignSession"; return; } // Get the session tag argument (required). std::string session_tag; if (!args->GetString(0, &session_tag)) { LOG(ERROR) << "Unable to extract session tag"; return; } SessionModelAssociator* associator = GetModelAssociator(web_ui()); if (associator) associator->DeleteForeignSession(session_tag); } void ForeignSessionHandler::HandleSetForeignSessionCollapsed( const ListValue* args) { if (args->GetSize() != 2U) { LOG(ERROR) << "Wrong number of args to setForeignSessionCollapsed"; return; } // Get the session tag argument (required). std::string session_tag; if (!args->GetString(0, &session_tag)) { LOG(ERROR) << "Unable to extract session tag"; return; } bool is_collapsed; if (!args->GetBoolean(1, &is_collapsed)) { LOG(ERROR) << "Unable to extract boolean argument"; return; } // Store session tags for collapsed sessions in a preference so that the // collapsed state persists. PrefService* prefs = Profile::FromWebUI(web_ui())->GetPrefs(); DictionaryPrefUpdate update(prefs, prefs::kNtpCollapsedForeignSessions); if (is_collapsed) update.Get()->SetBoolean(session_tag, true); else update.Get()->Remove(session_tag, NULL); } bool ForeignSessionHandler::SessionWindowToValue( const SessionWindow& window, DictionaryValue* dictionary) { if (window.tabs.empty()) { NOTREACHED(); return false; } scoped_ptr tab_values(new ListValue()); // Calculate the last |modification_time| for all entries within a window. base::Time modification_time = window.timestamp; for (size_t i = 0; i < window.tabs.size(); ++i) { scoped_ptr tab_value(new DictionaryValue()); if (SessionTabToValue(*window.tabs[i], tab_value.get())) { modification_time = std::max(modification_time, window.tabs[i]->timestamp); tab_values->Append(tab_value.release()); } } if (tab_values->GetSize() == 0) return false; dictionary->SetString("type", "window"); dictionary->SetDouble("timestamp", modification_time.ToInternalValue()); const base::TimeDelta last_synced = base::Time::Now() - modification_time; // If clock skew leads to a future time, or we last synced less than a minute // ago, output "Just now". dictionary->SetString("userVisibleTimestamp", last_synced < base::TimeDelta::FromMinutes(1) ? l10n_util::GetStringUTF16(IDS_SYNC_TIME_JUST_NOW) : TimeFormat::TimeElapsed(last_synced)); dictionary->SetInteger("sessionId", window.window_id.id()); dictionary->Set("tabs", tab_values.release()); return true; } } // namespace browser_sync