summaryrefslogtreecommitdiffstats
path: root/chrome/browser/bookmark_bar_model.cc
diff options
context:
space:
mode:
Diffstat (limited to 'chrome/browser/bookmark_bar_model.cc')
-rw-r--r--chrome/browser/bookmark_bar_model.cc716
1 files changed, 716 insertions, 0 deletions
diff --git a/chrome/browser/bookmark_bar_model.cc b/chrome/browser/bookmark_bar_model.cc
new file mode 100644
index 0000000..661f7c5
--- /dev/null
+++ b/chrome/browser/bookmark_bar_model.cc
@@ -0,0 +1,716 @@
+// Copyright 2008, Google Inc.
+// All rights reserved.
+//
+// Redistribution and use in source and binary forms, with or without
+// modification, are permitted provided that the following conditions are
+// met:
+//
+// * Redistributions of source code must retain the above copyright
+// notice, this list of conditions and the following disclaimer.
+// * Redistributions in binary form must reproduce the above
+// copyright notice, this list of conditions and the following disclaimer
+// in the documentation and/or other materials provided with the
+// distribution.
+// * Neither the name of Google Inc. nor the names of its
+// contributors may be used to endorse or promote products derived from
+// this software without specific prior written permission.
+//
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+#include "chrome/browser/bookmark_bar_model.h"
+
+#include "base/gfx/png_decoder.h"
+#include "chrome/browser/profile.h"
+#include "generated_resources.h"
+
+// BookmarkBarNode ------------------------------------------------------------
+
+const SkBitmap& BookmarkBarNode::GetFavIcon() {
+ if (!loaded_favicon_) {
+ loaded_favicon_ = true;
+ model_->LoadFavIcon(this);
+ }
+ return favicon_;
+}
+
+BookmarkBarNode::BookmarkBarNode(BookmarkBarModel* model)
+ : model_(model),
+ group_id_(0),
+ star_id_(0),
+ loaded_favicon_(false),
+ favicon_load_handle_(0),
+ type_(history::StarredEntry::BOOKMARK_BAR),
+ date_added_(Time::Now()) {
+ DCHECK(model_);
+}
+
+void BookmarkBarNode::Reset(const history::StarredEntry& entry) {
+ // We should either have no id, or the id of the new entry should match
+ // this.
+ DCHECK(!star_id_ || star_id_ == entry.id);
+ star_id_ = entry.id;
+ group_id_ = entry.group_id;
+ url_ = entry.url;
+ favicon_ = SkBitmap();
+ loaded_favicon_ = false;
+ favicon_load_handle_ = 0;
+ type_ = entry.type;
+ date_added_ = entry.date_added;
+ date_group_modified_ = entry.date_group_modified;
+ SetTitle(entry.title);
+}
+
+void BookmarkBarNode::SetURL(const GURL& url) {
+ DCHECK(favicon_load_handle_ == 0);
+ loaded_favicon_ = false;
+ favicon_load_handle_ = 0;
+ url_ = url;
+ favicon_ .reset();
+}
+
+history::StarredEntry BookmarkBarNode::GetEntry() {
+ history::StarredEntry entry;
+ entry.id = GetStarID();
+ entry.group_id = group_id_;
+ entry.url = GetURL();
+ entry.title = GetTitle();
+ entry.type = type_;
+ entry.date_added = date_added_;
+ entry.date_group_modified = date_group_modified_;
+ // Only set the parent and visual order if we have a valid parent (the root
+ // node is not in the db and has a group_id of 0).
+ if (GetParent() && GetParent()->group_id_) {
+ entry.visual_order = GetParent()->IndexOfChild(this);
+ entry.parent_group_id = GetParent()->GetGroupID();
+ }
+ return entry;
+}
+
+// BookmarkBarModel -----------------------------------------------------------
+
+BookmarkBarModel::BookmarkBarModel(Profile* profile)
+ : profile_(profile),
+ loaded_(false),
+#pragma warning(suppress: 4355) // Okay to pass "this" here.
+ root_(this),
+ // See declaration for description.
+ next_group_id_(HistoryService::kBookmarkBarID + 1),
+ bookmark_bar_node_(NULL),
+ other_node_(NULL) {
+ if (!profile) {
+ // Profile is NULL during testing.
+ CreateBookmarkBarNode();
+ CreateOtherBookmarksNode();
+ AddRootChildren(NULL);
+ loaded_ = true;
+ return;
+ }
+
+ // Notifications we want.
+ NotificationService::current()->AddObserver(
+ this, NOTIFY_STARRED_FAVICON_CHANGED, Source<Profile>(profile_));
+
+ // Request the entries from the database.
+ HistoryService* history_service =
+ profile_->GetHistoryService(Profile::EXPLICIT_ACCESS);
+ if (!history_service)
+ return;
+
+ // Request the entries on the bookmark bar.
+ history_service->GetAllStarredEntries(&load_consumer_,
+ NewCallback(this, &BookmarkBarModel::OnGotStarredEntries));
+}
+
+BookmarkBarModel::~BookmarkBarModel() {
+ if (profile_) {
+ NotificationService::current()->RemoveObserver(
+ this, NOTIFY_STARRED_FAVICON_CHANGED, Source<Profile>(profile_));
+ }
+}
+
+BookmarkBarNode* BookmarkBarModel::GetParentForNewNodes() {
+ std::vector<BookmarkBarNode*> nodes;
+
+ GetMostRecentlyModifiedGroupNodes(&root_, 1, &nodes);
+ return nodes.empty() ? bookmark_bar_node_ : nodes[0];
+}
+
+std::vector<BookmarkBarNode*> BookmarkBarModel::GetMostRecentlyModifiedGroups(
+ size_t max_count) {
+ std::vector<BookmarkBarNode*> nodes;
+ GetMostRecentlyModifiedGroupNodes(&root_, max_count, &nodes);
+
+ if (nodes.size() < max_count) {
+ // Add the bookmark bar and other nodes if there is space.
+ if (find(nodes.begin(), nodes.end(), bookmark_bar_node_) == nodes.end())
+ nodes.push_back(bookmark_bar_node_);
+
+ if (nodes.size() < max_count &&
+ find(nodes.begin(), nodes.end(), other_node_) == nodes.end()) {
+ nodes.push_back(other_node_);
+ }
+ }
+ return nodes;
+}
+
+void BookmarkBarModel::Remove(BookmarkBarNode* parent, int index) {
+#ifndef NDEBUG
+ CheckIndex(parent, index, false);
+#endif
+ if (parent != &root_) {
+ RemoveAndDeleteNode(parent->GetChild(index));
+ } else {
+ NOTREACHED(); // Can't remove from the root.
+ }
+}
+
+void BookmarkBarModel::RemoveFromBookmarkBar(BookmarkBarNode* node) {
+ if (!node->HasAncestor(bookmark_bar_node_))
+ return;
+
+ if (node != &root_ && node != bookmark_bar_node_ && node != other_node_) {
+ Move(node, other_node_, other_node_->GetChildCount());
+ } else {
+ NOTREACHED(); // Can't move the root, bookmark bar or other nodes.
+ }
+}
+
+void BookmarkBarModel::Move(BookmarkBarNode* node,
+ BookmarkBarNode* new_parent,
+ int index) {
+ DCHECK(node && new_parent);
+ DCHECK(node->GetParent());
+#ifndef NDEBUG
+ CheckIndex(new_parent, index, true);
+#endif
+
+ if (new_parent == &root_ || node == &root_ || node == bookmark_bar_node_ ||
+ node == other_node_) {
+ NOTREACHED();
+ return;
+ }
+
+ if (new_parent->HasAncestor(node)) {
+ // Can't make an ancestor of the node be a child of the node.
+ NOTREACHED();
+ return;
+ }
+
+ SetDateGroupModified(new_parent, Time::Now());
+
+ BookmarkBarNode* old_parent = node->GetParent();
+ int old_index = old_parent->IndexOfChild(node);
+
+ if (old_parent == new_parent &&
+ (index == old_index || index == old_index + 1)) {
+ // Node is already in this position, nothing to do.
+ return;
+ }
+
+ if (old_parent == new_parent && index > old_index)
+ index--;
+ new_parent->Add(index, node);
+
+ HistoryService* history = !profile_ ? NULL :
+ profile_->GetHistoryService(Profile::EXPLICIT_ACCESS);
+ if (history)
+ history->UpdateStarredEntry(node->GetEntry());
+ FOR_EACH_OBSERVER(BookmarkBarModelObserver, observers_,
+ BookmarkNodeMoved(this, old_parent, old_index,
+ new_parent, index));
+}
+
+void BookmarkBarModel::SetTitle(BookmarkBarNode* node,
+ const std::wstring& title) {
+ DCHECK(node);
+ if (node->GetTitle() == title)
+ return;
+ node->SetTitle(title);
+ HistoryService* history = !profile_ ? NULL :
+ profile_->GetHistoryService(Profile::EXPLICIT_ACCESS);
+ if (history)
+ history->UpdateStarredEntry(node->GetEntry());
+ FOR_EACH_OBSERVER(BookmarkBarModelObserver, observers_,
+ BookmarkNodeChanged(this, node));
+}
+
+BookmarkBarNode* BookmarkBarModel::GetNodeByURL(const GURL& url) {
+ BookmarkBarNode tmp_node(this);
+ tmp_node.url_ = url;
+ NodesOrderedByURLSet::iterator i = nodes_ordered_by_url_set_.find(&tmp_node);
+ return (i != nodes_ordered_by_url_set_.end()) ? *i : NULL;
+}
+
+BookmarkBarNode* BookmarkBarModel::GetNodeByGroupID(
+ history::UIStarID group_id) {
+ // TODO(sky): TreeNode needs a method that visits all nodes using a predicate.
+ return GetNodeByGroupID(&root_, group_id);
+}
+
+BookmarkBarNode* BookmarkBarModel::AddGroup(
+ BookmarkBarNode* parent,
+ int index,
+ const std::wstring& title) {
+ DCHECK(IsLoaded());
+#ifndef NDEBUG
+ CheckIndex(parent, index, true);
+#endif
+ if (parent == &root_) {
+ // Can't add to the root.
+ NOTREACHED();
+ return NULL;
+ }
+
+ BookmarkBarNode* new_node = new BookmarkBarNode(this);
+ new_node->group_id_ = next_group_id_++;
+ new_node->SetTitle(title);
+ new_node->type_ = history::StarredEntry::USER_GROUP;
+
+ return AddNode(parent, index, new_node);
+}
+
+BookmarkBarNode* BookmarkBarModel::AddURL(BookmarkBarNode* parent,
+ int index,
+ const std::wstring& title,
+ const GURL& url) {
+ return AddURLWithCreationTime(parent, index, title, url, Time::Now());
+}
+
+BookmarkBarNode* BookmarkBarModel::AddURLWithCreationTime(
+ BookmarkBarNode* parent,
+ int index,
+ const std::wstring& title,
+ const GURL& url,
+ const Time& creation_time) {
+ DCHECK(IsLoaded() && url.is_valid() && parent);
+ if (parent == &root_) {
+ // Can't add to the root.
+ NOTREACHED();
+ return NULL;
+ }
+#ifndef NDEBUG
+ CheckIndex(parent, index, true);
+#endif
+
+ BookmarkBarNode* existing_node = GetNodeByURL(url);
+ if (existing_node) {
+ Move(existing_node, parent, index);
+ SetTitle(existing_node, title);
+ return existing_node;
+ }
+
+ SetDateGroupModified(parent, creation_time);
+
+ BookmarkBarNode* new_node = new BookmarkBarNode(this);
+ new_node->SetTitle(title);
+ new_node->SetURL(url);
+ new_node->date_added_ = creation_time;
+ new_node->type_ = history::StarredEntry::URL;
+
+ nodes_ordered_by_url_set_.insert(new_node);
+
+ return AddNode(parent, index, new_node);
+}
+
+void BookmarkBarModel::SetURLStarred(const GURL& url,
+ const std::wstring& title,
+ bool is_starred) {
+ BookmarkBarNode* node = GetNodeByURL(url);
+ if (is_starred && !node) {
+ // Add the url.
+ BookmarkBarNode* parent = GetParentForNewNodes();
+ AddURL(parent, parent->GetChildCount(), title, url);
+ } else if (!is_starred && node) {
+ Remove(node->GetParent(), node->GetParent()->IndexOfChild(node));
+ }
+}
+
+void BookmarkBarModel::ResetDateGroupModified(BookmarkBarNode* node) {
+ SetDateGroupModified(node, Time());
+}
+
+void BookmarkBarModel::FavIconLoaded(BookmarkBarNode* node) {
+ // Send out notification to the observer.
+ FOR_EACH_OBSERVER(BookmarkBarModelObserver, observers_,
+ BookmarkNodeFavIconLoaded(this, node));
+}
+
+void BookmarkBarModel::RemoveNode(BookmarkBarNode* node) {
+ DCHECK(node && node != bookmark_bar_node_ && node != other_node_);
+ if (node->GetType() == history::StarredEntry::URL) {
+ NodesOrderedByURLSet::iterator i = nodes_ordered_by_url_set_.find(node);
+ DCHECK(i != nodes_ordered_by_url_set_.end());
+ nodes_ordered_by_url_set_.erase(i);
+ }
+
+ NodeToHandleMap::iterator i = node_to_handle_map_.find(node);
+ if (i != node_to_handle_map_.end()) {
+ HistoryService* history = !profile_ ? NULL :
+ profile_->GetHistoryService(Profile::EXPLICIT_ACCESS);
+ if (history)
+ request_consumer_.SetClientData(history, i->second, NULL);
+ node_to_handle_map_.erase(i);
+ }
+
+ CancelPendingFavIconLoadRequests(node);
+
+ // Recurse through children.
+ for (int i = node->GetChildCount() - 1; i >= 0; --i)
+ RemoveNode(node->GetChild(i));
+}
+
+void BookmarkBarModel::OnGotStarredEntries(
+ HistoryService::Handle,
+ std::vector<history::StarredEntry>* entries) {
+ if (loaded_) {
+ NOTREACHED();
+ return;
+ }
+
+ DCHECK(entries);
+ // Create tree nodes for each of the elements.
+ PopulateNodes(entries);
+
+ // Yes, we've finished loading.
+ loaded_ = true;
+
+ FOR_EACH_OBSERVER(BookmarkBarModelObserver, observers_, Loaded(this));
+
+ NotificationService::current()->Notify(
+ NOTIFY_BOOKMARK_MODEL_LOADED,
+ Source<Profile>(profile_),
+ NotificationService::NoDetails());
+}
+
+void BookmarkBarModel::PopulateNodes(
+ std::vector<history::StarredEntry>* entries) {
+ std::map<history::UIStarID,history::StarID> group_id_to_id_map;
+ IDToNodeMap id_to_node_map;
+
+ // Iterate through the entries building a mapping between group_id and id as
+ // well as creating the bookmark bar node and other node.
+ for (std::vector<history::StarredEntry>::const_iterator i = entries->begin();
+ i != entries->end(); ++i) {
+ if (i->type == history::StarredEntry::URL)
+ continue;
+
+ if (i->type == history::StarredEntry::OTHER)
+ other_node_ = CreateRootNodeFromStarredEntry(*i);
+ else if (i->type == history::StarredEntry::BOOKMARK_BAR)
+ bookmark_bar_node_ = CreateRootNodeFromStarredEntry(*i);
+
+ group_id_to_id_map[i->group_id] = i->id;
+ }
+
+ // Add the bookmark bar and other nodes to the root node.
+ AddRootChildren(&id_to_node_map);
+
+ // If the db was corrupt and we didn't get the bookmark bar/other nodes,
+ // AddRootChildren will create them. Update the map to make sure it includes
+ // these nodes.
+ group_id_to_id_map[other_node_->group_id_] = other_node_->star_id_;
+ group_id_to_id_map[bookmark_bar_node_->group_id_] =
+ bookmark_bar_node_->star_id_;
+
+ // Iterate through the entries again creating the nodes.
+ for (std::vector<history::StarredEntry>::iterator i = entries->begin();
+ i != entries->end(); ++i) {
+ if (!i->parent_group_id) {
+ DCHECK(i->type == history::StarredEntry::BOOKMARK_BAR ||
+ i->type == history::StarredEntry::OTHER);
+ // Ignore entries not parented to the bookmark bar.
+ continue;
+ }
+
+ BookmarkBarNode* node = id_to_node_map[i->id];
+ if (!node) {
+ // Creating a node results in creating the parent. As such, it is
+ // possible for the node representing a group to have been created before
+ // encountering the details.
+
+ // The created nodes are owned by the root node.
+ node = new BookmarkBarNode(this);
+ id_to_node_map[i->id] = node;
+ }
+ node->Reset(*i);
+
+ DCHECK(group_id_to_id_map.find(i->parent_group_id) !=
+ group_id_to_id_map.end());
+ history::StarID parent_id = group_id_to_id_map[i->parent_group_id];
+ BookmarkBarNode* parent = id_to_node_map[parent_id];
+ if (!parent) {
+ // Haven't encountered the parent yet, create it now.
+ parent = new BookmarkBarNode(this);
+ id_to_node_map[parent_id] = parent;
+ }
+
+ if (i->type == history::StarredEntry::URL)
+ nodes_ordered_by_url_set_.insert(node);
+ else
+ next_group_id_ = std::max(next_group_id_, i->group_id + 1);
+
+ // Add the node to its parent. entries is ordered by parent then
+ // visual order so that we know we maintain visual order by always adding
+ // to the end.
+ parent->Add(parent->GetChildCount(), node);
+ }
+}
+
+void BookmarkBarModel::OnCreatedEntry(HistoryService::Handle handle,
+ history::StarID id) {
+ BookmarkBarNode* node = request_consumer_.GetClientData(
+ profile_->GetHistoryService(Profile::EXPLICIT_ACCESS), handle);
+ // Node is NULL if the node was removed by the user before the request
+ // was processed.
+ if (node) {
+ DCHECK(!node->star_id_);
+ node->star_id_ = id;
+ DCHECK(node_to_handle_map_.find(node) != node_to_handle_map_.end());
+ node_to_handle_map_.erase(node_to_handle_map_.find(node));
+ }
+}
+
+void BookmarkBarModel::RemoveAndDeleteNode(BookmarkBarNode* delete_me) {
+ scoped_ptr<BookmarkBarNode> node(delete_me);
+
+ BookmarkBarNode* parent = node->GetParent();
+ DCHECK(parent);
+ int index = parent->IndexOfChild(node.get());
+ parent->Remove(index);
+ RemoveNode(node.get());
+
+ HistoryService* history = profile_ ?
+ profile_->GetHistoryService(Profile::EXPLICIT_ACCESS) : NULL;
+ if (history) {
+ if (node->GetType() == history::StarredEntry::URL) {
+ history->DeleteStarredURL(node->GetURL());
+ } else {
+ history->DeleteStarredGroup(node->GetGroupID());
+ }
+ }
+ FOR_EACH_OBSERVER(BookmarkBarModelObserver, observers_,
+ BookmarkNodeRemoved(this, parent, index));
+}
+
+BookmarkBarNode* BookmarkBarModel::AddNode(BookmarkBarNode* parent,
+ int index,
+ BookmarkBarNode* node) {
+ parent->Add(index, node);
+
+ // NOTE: As history calls us back when we invoke CreateStarredEntry, we have
+ // to be sure to invoke it after we've updated the nodes appropriately.
+ HistoryService* history = !profile_ ? NULL :
+ profile_->GetHistoryService(Profile::EXPLICIT_ACCESS);
+ if (history) {
+ HistoryService::Handle handle =
+ history->CreateStarredEntry(node->GetEntry(), &request_consumer_,
+ NewCallback(this, &BookmarkBarModel::OnCreatedEntry));
+ request_consumer_.SetClientData(history, handle, node);
+ node_to_handle_map_[node] = handle;
+ }
+ FOR_EACH_OBSERVER(BookmarkBarModelObserver, observers_,
+ BookmarkNodeAdded(this, parent, index));
+ return node;
+}
+
+BookmarkBarNode* BookmarkBarModel::GetNodeByGroupID(
+ BookmarkBarNode* node,
+ history::UIStarID group_id) {
+ if (node->GetType() == history::StarredEntry::USER_GROUP &&
+ node->GetGroupID() == group_id) {
+ return node;
+ }
+ for (int i = 0; i < node->GetChildCount(); ++i) {
+ BookmarkBarNode* result = GetNodeByGroupID(node->GetChild(i), group_id);
+ if (result)
+ return result;
+ }
+ return NULL;
+}
+
+void BookmarkBarModel::SetDateGroupModified(BookmarkBarNode* parent,
+ const Time time) {
+ DCHECK(parent);
+ parent->date_group_modified_ = time;
+ HistoryService* history = profile_ ?
+ profile_->GetHistoryService(Profile::EXPLICIT_ACCESS) : NULL;
+ if (history)
+ history->UpdateStarredEntry(parent->GetEntry());
+}
+
+void BookmarkBarModel::CreateBookmarkBarNode() {
+ history::StarredEntry entry;
+ entry.id = HistoryService::kBookmarkBarID;
+ entry.group_id = HistoryService::kBookmarkBarID;
+ entry.type = history::StarredEntry::BOOKMARK_BAR;
+ bookmark_bar_node_ = CreateRootNodeFromStarredEntry(entry);
+}
+
+void BookmarkBarModel::CreateOtherBookmarksNode() {
+ history::StarredEntry entry;
+ entry.id = HistoryService::kBookmarkBarID;
+ for (NodesOrderedByURLSet::iterator i = nodes_ordered_by_url_set_.begin();
+ i != nodes_ordered_by_url_set_.end(); ++i) {
+ entry.id = std::max(entry.id, (*i)->GetStarID());
+ }
+ entry.id++;
+ entry.group_id = next_group_id_++;
+ entry.type = history::StarredEntry::OTHER;
+ other_node_ = CreateRootNodeFromStarredEntry(entry);
+}
+
+BookmarkBarNode* BookmarkBarModel::CreateRootNodeFromStarredEntry(
+ const history::StarredEntry& entry) {
+ DCHECK(entry.type == history::StarredEntry::BOOKMARK_BAR ||
+ entry.type == history::StarredEntry::OTHER);
+ BookmarkBarNode* node = new BookmarkBarNode(this);
+ node->Reset(entry);
+ if (entry.type == history::StarredEntry::BOOKMARK_BAR)
+ node->SetTitle(l10n_util::GetString(IDS_BOOMARK_BAR_FOLDER_NAME));
+ else
+ node->SetTitle(l10n_util::GetString(IDS_BOOMARK_BAR_OTHER_FOLDER_NAME));
+ next_group_id_ = std::max(next_group_id_, entry.group_id + 1);
+ return node;
+}
+
+void BookmarkBarModel::AddRootChildren(IDToNodeMap* id_to_node_map) {
+ // When invoked we should have created the bookmark bar and other nodes. If
+ // not, it indicates the db couldn't be loaded correctly or is corrupt.
+ // Force creation so that bookmark bar model is still usable.
+ if (!bookmark_bar_node_) {
+ LOG(WARNING) << "No bookmark bar entry in the database. This indicates "
+ "bookmarks database couldn't be loaded or is corrupt.";
+ CreateBookmarkBarNode();
+ }
+ if (!other_node_) {
+ LOG(WARNING) << "No other folders bookmark bar entry in the database. This "
+ "indicates bookmarks database couldn't be loaded or is "
+ "corrupt.";
+ CreateOtherBookmarksNode();
+ }
+
+ if (id_to_node_map) {
+ (*id_to_node_map)[other_node_->GetStarID()] = other_node_;
+ (*id_to_node_map)[bookmark_bar_node_->GetStarID()] = bookmark_bar_node_;
+ }
+
+ // WARNING: order is important here, various places assume bookmark bar then
+ // other node.
+ root_.Add(0, bookmark_bar_node_);
+ root_.Add(1, other_node_);
+}
+
+void BookmarkBarModel::OnFavIconDataAvailable(
+ HistoryService::Handle handle,
+ bool know_favicon,
+ scoped_refptr<RefCountedBytes> data,
+ bool expired,
+ GURL icon_url) {
+ SkBitmap fav_icon;
+ if (know_favicon && data.get() &&
+ PNGDecoder::Decode(&data->data, &fav_icon)) {
+ BookmarkBarNode* node =
+ load_consumer_.GetClientData(
+ profile_->GetHistoryService(Profile::EXPLICIT_ACCESS), handle);
+ DCHECK(node);
+ node->favicon_ = fav_icon;
+ node->favicon_load_handle_ = 0;
+ FavIconLoaded(node);
+ }
+}
+
+void BookmarkBarModel::LoadFavIcon(BookmarkBarNode* node) {
+ if (node->GetType() != history::StarredEntry::URL)
+ return;
+
+ DCHECK(node->GetURL().is_valid());
+ HistoryService* history_service =
+ profile_->GetHistoryService(Profile::EXPLICIT_ACCESS);
+ if (!history_service)
+ return;
+
+ HistoryService::Handle handle = history_service->GetFavIconForURL(
+ node->GetURL(), &load_consumer_,
+ NewCallback(this, &BookmarkBarModel::OnFavIconDataAvailable));
+ load_consumer_.SetClientData(history_service, handle, node);
+}
+
+void BookmarkBarModel::CancelPendingFavIconLoadRequests(BookmarkBarNode* node) {
+ if (node->favicon_load_handle_) {
+ HistoryService* history =
+ profile_->GetHistoryService(Profile::EXPLICIT_ACCESS);
+ if (history)
+ history->CancelRequest(node->favicon_load_handle_);
+ node->favicon_load_handle_ = 0;
+ }
+}
+
+// static
+bool BookmarkBarModel::MoreRecentlyModified(BookmarkBarNode* n1,
+ BookmarkBarNode* n2) {
+ return n1->date_group_modified_ > n2->date_group_modified_;
+}
+
+void BookmarkBarModel::GetMostRecentlyModifiedGroupNodes(
+ BookmarkBarNode* parent,
+ size_t count,
+ std::vector<BookmarkBarNode*>* nodes) {
+ if (parent->group_id_ && parent->date_group_modified_ > Time()) {
+ if (count == 0) {
+ nodes->push_back(parent);
+ } else {
+ std::vector<BookmarkBarNode*>::iterator i =
+ std::upper_bound(nodes->begin(), nodes->end(), parent,
+ &MoreRecentlyModified);
+ if (nodes->size() < count || i != nodes->end()) {
+ nodes->insert(i, parent);
+ while (nodes->size() > count)
+ nodes->pop_back();
+ }
+ }
+ } // else case, the root node, which we don't care about or imported nodes
+ // (which have a time of 0).
+ for (int i = 0; i < parent->GetChildCount(); ++i) {
+ BookmarkBarNode* child = parent->GetChild(i);
+ if (child->GetType() != history::StarredEntry::URL) {
+ GetMostRecentlyModifiedGroupNodes(child, count, nodes);
+ }
+ }
+}
+
+void BookmarkBarModel::Observe(NotificationType type,
+ const NotificationSource& source,
+ const NotificationDetails& details) {
+ switch (type) {
+ case NOTIFY_STARRED_FAVICON_CHANGED: {
+ // Prevent the observers from getting confused for multiple favicon loads.
+ Details<history::FavIconChangeDetails> favicon_details(details);
+ for (std::set<GURL>::const_iterator i = favicon_details->urls.begin();
+ i != favicon_details->urls.end(); ++i) {
+ BookmarkBarNode* node = GetNodeByURL(*i);
+ if (node) {
+ // Got an updated favicon, for a URL, do a new request.
+ node->InvalidateFavicon();
+ CancelPendingFavIconLoadRequests(node);
+ FOR_EACH_OBSERVER(BookmarkBarModelObserver, observers_,
+ BookmarkNodeChanged(this, node));
+ }
+ }
+ break;
+ }
+
+ default:
+ NOTREACHED();
+ break;
+ }
+}