| Index: chrome/browser/ui/webui/sessions_ui.cc
|
| ===================================================================
|
| --- chrome/browser/ui/webui/sessions_ui.cc (revision 0)
|
| +++ chrome/browser/ui/webui/sessions_ui.cc (revision 0)
|
| @@ -0,0 +1,297 @@
|
| +// Copyright (c) 2011 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/sessions_ui.h"
|
| +
|
| +#include "chrome/browser/profiles/profile.h"
|
| +#include "chrome/browser/sync/engine/syncapi.h"
|
| +#include "chrome/browser/sync/glue/session_model_associator.h"
|
| +#include "chrome/browser/sync/profile_sync_service.h"
|
| +#include "chrome/browser/ui/webui/chrome_url_data_manager.h"
|
| +#include "chrome/browser/ui/webui/ntp/new_tab_ui.h"
|
| +#include "chrome/common/chrome_version_info.h"
|
| +#include "chrome/common/jstemplate_builder.h"
|
| +#include "chrome/common/url_constants.h"
|
| +#include "content/browser/webui/web_ui.h"
|
| +#include "grit/browser_resources.h"
|
| +#include "grit/chromium_strings.h"
|
| +#include "grit/generated_resources.h"
|
| +#include "grit/theme_resources.h"
|
| +#include "ui/base/l10n/l10n_util.h"
|
| +#include "ui/base/resource/resource_bundle.h"
|
| +
|
| +
|
| +namespace {
|
| +
|
| +///////////////////////////////////////////////////////////////////////////////
|
| +//
|
| +// SessionsUIHTMLSource
|
| +//
|
| +///////////////////////////////////////////////////////////////////////////////
|
| +
|
| +class SessionsUIHTMLSource : public ChromeURLDataManager::DataSource {
|
| + public:
|
| + SessionsUIHTMLSource()
|
| + : DataSource(chrome::kChromeUISessionsHost, MessageLoop::current()) {}
|
| +
|
| + // Called when the network layer has requested a resource underneath
|
| + // the path we registered.
|
| + virtual void StartDataRequest(const std::string& path,
|
| + bool is_incognito,
|
| + int request_id);
|
| + virtual std::string GetMimeType(const std::string&) const {
|
| + return "text/html";
|
| + }
|
| +
|
| + private:
|
| + ~SessionsUIHTMLSource() {}
|
| +
|
| + DISALLOW_COPY_AND_ASSIGN(SessionsUIHTMLSource);
|
| +};
|
| +
|
| +void SessionsUIHTMLSource::StartDataRequest(const std::string& path,
|
| + bool is_incognito,
|
| + int request_id) {
|
| + DictionaryValue localized_strings;
|
| + localized_strings.SetString("sessionsTitle",
|
| + l10n_util::GetStringUTF16(IDS_SESSIONS_TITLE));
|
| + localized_strings.SetString("sessionsCountFormat",
|
| + l10n_util::GetStringUTF16(IDS_SESSIONS_SESSION_COUNT_BANNER_FORMAT));
|
| + localized_strings.SetString("noSessionsMessage",
|
| + l10n_util::GetStringUTF16(IDS_SESSIONS_NO_SESSIONS_MESSAGE));
|
| +
|
| + localized_strings.SetString("magicCountFormat",
|
| + l10n_util::GetStringUTF16(IDS_SESSIONS_MAGIC_LIST_BANNER_FORMAT));
|
| + localized_strings.SetString("noMagicMessage",
|
| + l10n_util::GetStringUTF16(IDS_SESSIONS_NO_MAGIC_MESSAGE));
|
| +
|
| + ChromeURLDataManager::DataSource::SetFontAndTextDirection(&localized_strings);
|
| +
|
| + static const base::StringPiece sessions_html(
|
| + ResourceBundle::GetSharedInstance().
|
| + GetRawDataResource(IDR_SESSIONS_HTML));
|
| + std::string full_html =
|
| + jstemplate_builder::GetI18nTemplateHtml(sessions_html,
|
| + &localized_strings);
|
| + jstemplate_builder::AppendJsTemplateSourceHtml(&full_html);
|
| +
|
| + scoped_refptr<RefCountedBytes> html_bytes(new RefCountedBytes);
|
| + html_bytes->data.resize(full_html.size());
|
| + std::copy(full_html.begin(), full_html.end(), html_bytes->data.begin());
|
| +
|
| + SendResponse(request_id, html_bytes);
|
| +}
|
| +
|
| +////////////////////////////////////////////////////////////////////////////////
|
| +//
|
| +// SessionsDOMHandler
|
| +//
|
| +////////////////////////////////////////////////////////////////////////////////
|
| +
|
| +// The handler for Javascript messages for the chrome://sessions/ page.
|
| +class SessionsDOMHandler : public WebUIMessageHandler {
|
| + public:
|
| + SessionsDOMHandler();
|
| + virtual ~SessionsDOMHandler();
|
| +
|
| + // WebUIMessageHandler implementation.
|
| + virtual WebUIMessageHandler* Attach(WebUI* web_ui);
|
| + virtual void RegisterMessages();
|
| +
|
| + private:
|
| + // Asynchronously fetches the session list. Called from JS.
|
| + void HandleRequestSessions(const ListValue* args);
|
| +
|
| + // Sends the recent session list to JS.
|
| + void UpdateUI();
|
| +
|
| + // Returns the model associated for getting the list of sessions from sync.
|
| + browser_sync::SessionModelAssociator* GetModelAssociator();
|
| +
|
| + // Appends each entry in |tabs| to |tab_list| as a DictonaryValue.
|
| + void GetTabList(const std::vector<SessionTab*>& tabs, ListValue* tab_list);
|
| +
|
| + // Appends each entry in |windows| to |window_list| as a DictonaryValue.
|
| + void GetWindowList(const std::vector<SessionWindow*>& windows,
|
| + ListValue* window_list);
|
| +
|
| + // Appends each entry in |sessions| to |session_list| as a DictonaryValue.
|
| + void GetSessionList(const std::vector<const ForeignSession*>& sessions,
|
| + ListValue* session_list);
|
| +
|
| + // Traverses all tabs in |sessions| and adds them to |all_tabs|.
|
| + void GetAllTabs(const std::vector<const ForeignSession*>& sessions,
|
| + std::vector<SessionTab*>* all_tabs);
|
| +
|
| + // Creates a "magic" list of tabs from all the sessions.
|
| + void CreateMagicTabList(const std::vector<const ForeignSession*>& sessions,
|
| + ListValue* tab_list);
|
| +
|
| + DISALLOW_COPY_AND_ASSIGN(SessionsDOMHandler);
|
| +};
|
| +
|
| +SessionsDOMHandler::SessionsDOMHandler() {
|
| +}
|
| +
|
| +SessionsDOMHandler::~SessionsDOMHandler() {
|
| +}
|
| +
|
| +WebUIMessageHandler* SessionsDOMHandler::Attach(WebUI* web_ui) {
|
| + return WebUIMessageHandler::Attach(web_ui);
|
| +}
|
| +
|
| +void SessionsDOMHandler::RegisterMessages() {
|
| + web_ui_->RegisterMessageCallback("requestSessionList",
|
| + NewCallback(this, &SessionsDOMHandler::HandleRequestSessions));
|
| +}
|
| +
|
| +void SessionsDOMHandler::HandleRequestSessions(const ListValue* args) {
|
| + UpdateUI();
|
| +}
|
| +
|
| +browser_sync::SessionModelAssociator* SessionsDOMHandler::GetModelAssociator() {
|
| + // We only want to get the model associator if there is one, and it is done
|
| + // syncing sessions.
|
| + Profile* profile = web_ui_->GetProfile();
|
| + if (!profile->HasProfileSyncService())
|
| + return NULL;
|
| + ProfileSyncService* service = profile->GetProfileSyncService();
|
| + if (!service->ShouldPushChanges())
|
| + return NULL;
|
| + return service->GetSessionModelAssociator();
|
| +}
|
| +
|
| +void SessionsDOMHandler::GetTabList(
|
| + const std::vector<SessionTab*>& tabs, ListValue* tab_list) {
|
| + for (std::vector<SessionTab*>::const_iterator it = tabs.begin();
|
| + it != tabs.end(); ++it) {
|
| + const SessionTab* tab = *it;
|
| + const int nav_index = tab->current_navigation_index;
|
| + // Range-check |nav_index| as it has been observed to be invalid sometimes.
|
| + if (nav_index < 0 ||
|
| + static_cast<size_t>(nav_index) >= tab->navigations.size()) {
|
| + continue;
|
| + }
|
| + const TabNavigation& nav = tab->navigations[nav_index];
|
| + scoped_ptr<DictionaryValue> tab_data(new DictionaryValue());
|
| + tab_data->SetInteger("id", tab->tab_id.id());
|
| + tab_data->SetDouble("timestamp",
|
| + static_cast<double> (tab->timestamp.ToInternalValue()));
|
| + if (nav.title().empty())
|
| + tab_data->SetString("title", nav.virtual_url().spec());
|
| + else
|
| + tab_data->SetString("title", nav.title());
|
| + tab_data->SetString("url", nav.virtual_url().spec());
|
| + tab_list->Append(tab_data.release());
|
| + }
|
| +}
|
| +
|
| +void SessionsDOMHandler::GetWindowList(
|
| + const std::vector<SessionWindow*>& windows, ListValue* window_list) {
|
| + for (std::vector<SessionWindow*>::const_iterator it =
|
| + windows.begin(); it != windows.end(); ++it) {
|
| + const SessionWindow* window = *it;
|
| + scoped_ptr<DictionaryValue> window_data(new DictionaryValue());
|
| + window_data->SetInteger("id", window->window_id.id());
|
| + window_data->SetDouble("timestamp",
|
| + static_cast<double> (window->timestamp.ToInternalValue()));
|
| + scoped_ptr<ListValue> tab_list(new ListValue());
|
| + GetTabList(window->tabs, tab_list.get());
|
| + window_data->Set("tabs", tab_list.release());
|
| + window_list->Append(window_data.release());
|
| + }
|
| +}
|
| +
|
| +void SessionsDOMHandler::GetSessionList(
|
| + const std::vector<const ForeignSession*>& sessions,
|
| + ListValue* session_list) {
|
| + for (std::vector<const ForeignSession*>::const_iterator it =
|
| + sessions.begin(); it != sessions.end(); ++it) {
|
| + const ForeignSession* session = *it;
|
| + scoped_ptr<DictionaryValue> session_data(new DictionaryValue());
|
| + session_data->SetString("tag", session->foreign_session_tag);
|
| + scoped_ptr<ListValue> window_list(new ListValue());
|
| + GetWindowList(session->windows, window_list.get());
|
| + session_data->Set("windows", window_list.release());
|
| + session_list->Append(session_data.release());
|
| + }
|
| +}
|
| +
|
| +void SessionsDOMHandler::GetAllTabs(
|
| + const std::vector<const ForeignSession*>& sessions,
|
| + std::vector<SessionTab*>* all_tabs) {
|
| + for (size_t i = 0; i < sessions.size(); i++) {
|
| + const std::vector<SessionWindow*>& windows = sessions[i]->windows;
|
| + for (size_t j = 0; j < windows.size(); j++) {
|
| + const std::vector<SessionTab*>& tabs = windows[j]->tabs;
|
| + all_tabs->insert(all_tabs->end(), tabs.begin(), tabs.end());
|
| + }
|
| + }
|
| +}
|
| +
|
| +// Comparator function for sort() in CreateMagicTabList() below.
|
| +bool CompareTabsByTimestamp(SessionTab* lhs, SessionTab* rhs) {
|
| + return lhs->timestamp.ToInternalValue() > rhs->timestamp.ToInternalValue();
|
| +}
|
| +
|
| +void SessionsDOMHandler::CreateMagicTabList(
|
| + const std::vector<const ForeignSession*>& sessions,
|
| + ListValue* tab_list) {
|
| + std::vector<SessionTab*> all_tabs;
|
| + GetAllTabs(sessions, &all_tabs);
|
| +
|
| + // Sort the list by timestamp - newest first.
|
| + std::sort(all_tabs.begin(), all_tabs.end(), CompareTabsByTimestamp);
|
| +
|
| + // Truncate the list if necessary.
|
| + const size_t kMagicTabListMaxSize = 10;
|
| + if (all_tabs.size() > kMagicTabListMaxSize)
|
| + all_tabs.resize(kMagicTabListMaxSize);
|
| +
|
| + GetTabList(all_tabs, tab_list);
|
| +}
|
| +
|
| +void SessionsDOMHandler::UpdateUI() {
|
| + ListValue session_list;
|
| + ListValue magic_list;
|
| +
|
| + browser_sync::SessionModelAssociator* associator = GetModelAssociator();
|
| + // Make sure the associator has been created.
|
| + if (associator) {
|
| + std::vector<const ForeignSession*> sessions;
|
| + if (associator->GetAllForeignSessions(&sessions)) {
|
| + GetSessionList(sessions, &session_list);
|
| + CreateMagicTabList(sessions, &magic_list);
|
| + }
|
| + }
|
| +
|
| + // Send the results to JavaScript, even if the lists are empty, so that the
|
| + // UI can show a message that there is nothing.
|
| + web_ui_->CallJavascriptFunction("updateSessionList",
|
| + session_list,
|
| + magic_list);
|
| +}
|
| +
|
| +} // namespace
|
| +
|
| +///////////////////////////////////////////////////////////////////////////////
|
| +//
|
| +// SessionsUI
|
| +//
|
| +///////////////////////////////////////////////////////////////////////////////
|
| +
|
| +SessionsUI::SessionsUI(TabContents* contents) : ChromeWebUI(contents) {
|
| + AddMessageHandler((new SessionsDOMHandler())->Attach(this));
|
| +
|
| + SessionsUIHTMLSource* html_source = new SessionsUIHTMLSource();
|
| +
|
| + // Set up the chrome://sessions/ source.
|
| + contents->profile()->GetChromeURLDataManager()->AddDataSource(html_source);
|
| +}
|
| +
|
| +// static
|
| +RefCountedMemory* SessionsUI::GetFaviconResourceBytes() {
|
| + return ResourceBundle::GetSharedInstance().
|
| + LoadDataResourceBytes(IDR_HISTORY_FAVICON);
|
| +}
|
|
|
| Property changes on: chrome/browser/ui/webui/sessions_ui.cc
|
| ___________________________________________________________________
|
| Added: svn:eol-style
|
| + LF
|
|
|
|
|