1 // Copyright (c) 2012 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/ui/tabs/pinned_tab_codec.h" 6 7 #include "base/prefs/pref_service.h" 8 #include "base/prefs/scoped_user_pref_update.h" 9 #include "base/values.h" 10 #include "chrome/browser/extensions/tab_helper.h" 11 #include "chrome/browser/profiles/profile.h" 12 #include "chrome/browser/ui/browser.h" 13 #include "chrome/browser/ui/browser_iterator.h" 14 #include "chrome/browser/ui/browser_list.h" 15 #include "chrome/browser/ui/tabs/tab_strip_model.h" 16 #include "chrome/common/extensions/manifest_handlers/app_launch_info.h" 17 #include "chrome/common/pref_names.h" 18 #include "components/pref_registry/pref_registry_syncable.h" 19 #include "content/public/browser/navigation_entry.h" 20 #include "content/public/browser/web_contents.h" 21 #include "extensions/common/extension.h" 22 23 using content::NavigationEntry; 24 25 // Key used in dictionaries for the app id. 26 static const char kAppID[] = "app_id"; 27 28 // Key used in dictionaries for the url. 29 static const char kURL[] = "url"; 30 31 // Returns true if |browser| has any pinned tabs. 32 static bool HasPinnedTabs(Browser* browser) { 33 TabStripModel* tab_model = browser->tab_strip_model(); 34 for (int i = 0; i < tab_model->count(); ++i) { 35 if (tab_model->IsTabPinned(i)) 36 return true; 37 } 38 return false; 39 } 40 41 // Adds a DictionaryValue to |values| representing |tab|. 42 static void EncodeTab(const StartupTab& tab, base::ListValue* values) { 43 scoped_ptr<base::DictionaryValue> value(new base::DictionaryValue); 44 value->SetString(kURL, tab.url.spec()); 45 if (tab.is_app) 46 value->SetString(kAppID, tab.app_id); 47 values->Append(value.release()); 48 } 49 50 // Adds a base::DictionaryValue to |values| representing the pinned tab at the 51 // specified index. 52 static void EncodePinnedTab(TabStripModel* model, 53 int index, 54 base::ListValue* values) { 55 scoped_ptr<base::DictionaryValue> value(new base::DictionaryValue()); 56 57 content::WebContents* web_contents = model->GetWebContentsAt(index); 58 if (model->IsAppTab(index)) { 59 const extensions::Extension* extension = 60 extensions::TabHelper::FromWebContents(web_contents)->extension_app(); 61 DCHECK(extension); 62 value->SetString(kAppID, extension->id()); 63 // For apps we use the launch url. We do this because the user is 64 // effectively restarting the app, so returning them to the app's launch 65 // page seems closest to what they expect. 66 value->SetString( 67 kURL, extensions::AppLaunchInfo::GetFullLaunchURL(extension).spec()); 68 values->Append(value.release()); 69 } else { 70 NavigationEntry* entry = web_contents->GetController().GetActiveEntry(); 71 if (!entry && web_contents->GetController().GetEntryCount()) 72 entry = web_contents->GetController().GetEntryAtIndex(0); 73 if (entry) { 74 value->SetString(kURL, entry->GetURL().spec()); 75 values->Append(value.release()); 76 } 77 } 78 } 79 80 // Invokes EncodePinnedTab for each pinned tab in browser. 81 static void EncodePinnedTabs(Browser* browser, base::ListValue* values) { 82 TabStripModel* tab_model = browser->tab_strip_model(); 83 for (int i = 0; i < tab_model->count() && tab_model->IsTabPinned(i); ++i) 84 EncodePinnedTab(tab_model, i, values); 85 } 86 87 // Decodes the previously written values in |value| to |tab|, returning true 88 // on success. 89 static bool DecodeTab(const base::DictionaryValue& value, StartupTab* tab) { 90 tab->is_app = false; 91 92 std::string url_string; 93 if (!value.GetString(kURL, &url_string)) 94 return false; 95 tab->url = GURL(url_string); 96 97 if (value.GetString(kAppID, &(tab->app_id))) 98 tab->is_app = true; 99 100 return true; 101 } 102 103 // static 104 void PinnedTabCodec::RegisterProfilePrefs( 105 user_prefs::PrefRegistrySyncable* registry) { 106 registry->RegisterListPref(prefs::kPinnedTabs, 107 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); 108 } 109 110 // static 111 void PinnedTabCodec::WritePinnedTabs(Profile* profile) { 112 PrefService* prefs = profile->GetPrefs(); 113 if (!prefs) 114 return; 115 116 base::ListValue values; 117 for (chrome::BrowserIterator it; !it.done(); it.Next()) { 118 Browser* browser = *it; 119 if (browser->is_type_tabbed() && 120 browser->profile() == profile && HasPinnedTabs(browser)) { 121 EncodePinnedTabs(browser, &values); 122 } 123 } 124 prefs->Set(prefs::kPinnedTabs, values); 125 } 126 127 // static 128 void PinnedTabCodec::WritePinnedTabs(Profile* profile, 129 const StartupTabs& tabs) { 130 PrefService* prefs = profile->GetPrefs(); 131 if (!prefs) 132 return; 133 134 ListPrefUpdate update(prefs, prefs::kPinnedTabs); 135 base::ListValue* values = update.Get(); 136 values->Clear(); 137 for (StartupTabs::const_iterator i = tabs.begin(); i != tabs.end(); ++i) 138 EncodeTab(*i, values); 139 } 140 141 // static 142 StartupTabs PinnedTabCodec::ReadPinnedTabs(Profile* profile) { 143 PrefService* prefs = profile->GetPrefs(); 144 if (!prefs) 145 return StartupTabs(); 146 return ReadPinnedTabs(prefs->GetList(prefs::kPinnedTabs)); 147 } 148 149 // static 150 StartupTabs PinnedTabCodec::ReadPinnedTabs(const base::Value* value) { 151 StartupTabs results; 152 153 const base::ListValue* tabs_list = NULL; 154 if (!value->GetAsList(&tabs_list)) 155 return results; 156 157 for (size_t i = 0, max = tabs_list->GetSize(); i < max; ++i) { 158 const base::DictionaryValue* tab_values = NULL; 159 if (tabs_list->GetDictionary(i, &tab_values)) { 160 StartupTab tab; 161 if (DecodeTab(*tab_values, &tab)) 162 results.push_back(tab); 163 } 164 } 165 return results; 166 } 167