Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(7318)

Unified Diff: chrome/browser/ui/webui/sessions_ui.cc

Issue 6969016: Adding chrome://sessions webui for prototyping NTP stuff. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src/
Patch Set: '' Created 9 years, 6 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « chrome/browser/ui/webui/sessions_ui.h ('k') | chrome/chrome_browser.gypi » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
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
« no previous file with comments | « chrome/browser/ui/webui/sessions_ui.h ('k') | chrome/chrome_browser.gypi » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698