OLD | NEW |
(Empty) | |
| 1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. |
| 4 |
| 5 #include "chrome/browser/tabs/pinned_tab_codec.h" |
| 6 |
| 7 #include "base/values.h" |
| 8 #include "chrome/browser/browser.h" |
| 9 #include "chrome/browser/browser_list.h" |
| 10 #include "chrome/browser/pref_service.h" |
| 11 #include "chrome/browser/profile.h" |
| 12 #include "chrome/browser/tab_contents/tab_contents.h" |
| 13 #include "chrome/browser/tabs/tab_strip_model.h" |
| 14 #include "chrome/common/extensions/extension.h" |
| 15 #include "chrome/common/page_transition_types.h" |
| 16 #include "chrome/common/pref_names.h" |
| 17 #include "ipc/ipc_message.h" |
| 18 |
| 19 typedef BrowserInit::LaunchWithProfile::Tab Tab; |
| 20 |
| 21 // Key used in dictionaries for the app id. |
| 22 static const wchar_t kAppID[] = L"app_id"; |
| 23 |
| 24 // Key used in dictionaries for the url. |
| 25 static const wchar_t kURL[] = L"url"; |
| 26 |
| 27 // Returns true if |browser| has any pinned tabs. |
| 28 static bool HasPinnedTabs(Browser* browser) { |
| 29 TabStripModel* tab_model = browser->tabstrip_model(); |
| 30 for (int i = 0; i < tab_model->count(); ++i) { |
| 31 if (tab_model->IsTabPinned(i)) |
| 32 return true; |
| 33 } |
| 34 return false; |
| 35 } |
| 36 |
| 37 // Adds a DictionaryValue to |values| representing the pinned tab at the |
| 38 // specified index. |
| 39 static void EncodePinnedTab(TabStripModel* model, |
| 40 int index, |
| 41 ListValue* values) { |
| 42 scoped_ptr<DictionaryValue> value(new DictionaryValue()); |
| 43 |
| 44 TabContents* tab_contents = model->GetTabContentsAt(index); |
| 45 if (model->IsAppTab(index)) { |
| 46 Extension* extension = tab_contents->app_extension(); |
| 47 DCHECK(extension); |
| 48 value->SetString(kAppID, extension->id()); |
| 49 // For apps we use the launch url. We do this for two reasons: |
| 50 // . the user is effectively restarting the app, so that returning them to |
| 51 // the app's launch page seems closest to what they expect. |
| 52 // . we do the same when restoring a phantom tab. |
| 53 value->SetString(kURL, extension->app_launch_url().spec()); |
| 54 values->Append(value.release()); |
| 55 } else { |
| 56 NavigationEntry* entry = tab_contents->controller().GetActiveEntry(); |
| 57 if (!entry && tab_contents->controller().entry_count()) |
| 58 entry = tab_contents->controller().GetEntryAtIndex(0); |
| 59 if (entry) { |
| 60 value->SetString(kURL, entry->url().spec()); |
| 61 values->Append(value.release()); |
| 62 } |
| 63 } |
| 64 } |
| 65 |
| 66 // Invokes EncodePinnedTab for each pinned tab in browser. |
| 67 static void EncodePinnedTabs(Browser* browser, ListValue* values) { |
| 68 TabStripModel* tab_model = browser->tabstrip_model(); |
| 69 for (int i = 0; i < tab_model->count() && tab_model->IsTabPinned(i); ++i) |
| 70 EncodePinnedTab(tab_model, i, values); |
| 71 } |
| 72 |
| 73 // Decodes the previously written values in |value| to |tab|, returning true |
| 74 // on success. |
| 75 static bool DecodeTab(const DictionaryValue& value, Tab* tab) { |
| 76 tab->is_app = false; |
| 77 |
| 78 std::string url_string; |
| 79 if (!value.GetString(kURL, &url_string)) |
| 80 return false; |
| 81 tab->url = GURL(url_string); |
| 82 |
| 83 if (value.GetString(kAppID, &(tab->app_id))) |
| 84 tab->is_app = true; |
| 85 |
| 86 return true; |
| 87 } |
| 88 |
| 89 // static |
| 90 void PinnedTabCodec::RegisterUserPrefs(PrefService* prefs) { |
| 91 prefs->RegisterListPref(prefs::kPinnedTabs); |
| 92 } |
| 93 |
| 94 // static |
| 95 void PinnedTabCodec::WritePinnedTabs(Profile* profile) { |
| 96 PrefService* prefs = profile->GetPrefs(); |
| 97 if (!prefs) |
| 98 return; |
| 99 |
| 100 ListValue values; |
| 101 for (BrowserList::const_iterator i = BrowserList::begin(); |
| 102 i != BrowserList::end(); ++i) { |
| 103 Browser* browser = *i; |
| 104 if (browser->type() == Browser::TYPE_NORMAL && |
| 105 browser->profile() == profile && HasPinnedTabs(browser)) { |
| 106 EncodePinnedTabs(browser, &values); |
| 107 } |
| 108 } |
| 109 prefs->Set(prefs::kPinnedTabs, values); |
| 110 prefs->ScheduleSavePersistentPrefs(); |
| 111 } |
| 112 |
| 113 // static |
| 114 std::vector<Tab> PinnedTabCodec::ReadPinnedTabs(Profile* profile) { |
| 115 std::vector<Tab> results; |
| 116 |
| 117 PrefService* prefs = profile->GetPrefs(); |
| 118 if (!prefs) |
| 119 return results; |
| 120 |
| 121 const ListValue* pref_value = prefs->GetList(prefs::kPinnedTabs); |
| 122 if (!pref_value) |
| 123 return results; |
| 124 |
| 125 for (size_t i = 0, max = pref_value->GetSize(); i < max; ++i) { |
| 126 DictionaryValue* values = NULL; |
| 127 if (pref_value->GetDictionary(i, &values)) { |
| 128 Tab tab; |
| 129 if (DecodeTab(*values, &tab)) |
| 130 results.push_back(tab); |
| 131 } |
| 132 } |
| 133 return results; |
| 134 } |
OLD | NEW |