summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--chrome/browser/bookmarks/bookmark_model.cc15
-rw-r--r--chrome/browser/bookmarks/bookmark_model.h10
-rw-r--r--chrome/browser/bookmarks/bookmark_model_unittest.cc52
-rw-r--r--chrome/views/tree_node_model.h9
4 files changed, 70 insertions, 16 deletions
diff --git a/chrome/browser/bookmarks/bookmark_model.cc b/chrome/browser/bookmarks/bookmark_model.cc
index 12c8c97..1ec048a 100644
--- a/chrome/browser/bookmarks/bookmark_model.cc
+++ b/chrome/browser/bookmarks/bookmark_model.cc
@@ -8,6 +8,7 @@
#include "build/build_config.h"
#include "chrome/browser/bookmarks/bookmark_utils.h"
#include "chrome/browser/bookmarks/bookmark_storage.h"
+#include "chrome/browser/browser_process.h"
#include "chrome/browser/profile.h"
#include "chrome/common/l10n_util.h"
#include "chrome/common/notification_service.h"
@@ -298,6 +299,20 @@ BookmarkNode* BookmarkModel::AddURLWithCreationTime(
return AddNode(parent, index, new_node, was_bookmarked);
}
+void BookmarkModel::SortChildren(BookmarkNode* parent) {
+ if (!parent || !parent->is_folder() || parent == root_node() ||
+ parent->GetChildCount() <= 1) {
+ return;
+ }
+
+ l10n_util::SortStringsUsingMethod(g_browser_process->GetApplicationLocale(),
+ &(parent->children()),
+ &BookmarkNode::GetTitle);
+
+ FOR_EACH_OBSERVER(BookmarkModelObserver, observers_,
+ BookmarkNodeChildrenReordered(this, parent));
+}
+
void BookmarkModel::SetURLStarred(const GURL& url,
const std::wstring& title,
bool is_starred) {
diff --git a/chrome/browser/bookmarks/bookmark_model.h b/chrome/browser/bookmarks/bookmark_model.h
index 52cbe46..a886725 100644
--- a/chrome/browser/bookmarks/bookmark_model.h
+++ b/chrome/browser/bookmarks/bookmark_model.h
@@ -169,6 +169,12 @@ class BookmarkModelObserver {
// Invoked when a favicon has finished loading.
virtual void BookmarkNodeFavIconLoaded(BookmarkModel* model,
BookmarkNode* node) = 0;
+
+ // Invoked when the children (just direct children, not descendants) of
+ // |node| have been reordered in some way, such as sorted.
+ // TODO(sky): make this pure virtual when all observers have been updated.
+ virtual void BookmarkNodeChildrenReordered(BookmarkModel* model,
+ BookmarkNode* node) {}
};
// BookmarkModel --------------------------------------------------------------
@@ -270,6 +276,10 @@ class BookmarkModel : public NotificationObserver, public BookmarkService {
const GURL& url,
const base::Time& creation_time);
+ // Sorts the children of |parent|, notifying observers by way of the
+ // BookmarkNodeChildrenReordered method.
+ void SortChildren(BookmarkNode* parent);
+
// This is the convenience that makes sure the url is starred or not starred.
// If is_starred is false, all bookmarks for URL are removed. If is_starred is
// true and there are no bookmarks for url, a bookmark is created.
diff --git a/chrome/browser/bookmarks/bookmark_model_unittest.cc b/chrome/browser/bookmarks/bookmark_model_unittest.cc
index 0264c3a..e2fb371 100644
--- a/chrome/browser/bookmarks/bookmark_model_unittest.cc
+++ b/chrome/browser/bookmarks/bookmark_model_unittest.cc
@@ -90,6 +90,11 @@ class BookmarkModelTest : public testing::Test, public BookmarkModelObserver {
observer_details.Set(node, NULL, -1, -1);
}
+ virtual void BookmarkNodeChildrenReordered(BookmarkModel* model,
+ BookmarkNode* node) {
+ reordered_count_++;
+ }
+
virtual void BookmarkNodeFavIconLoaded(BookmarkModel* model,
BookmarkNode* node) {
// We never attempt to load favicons, so that this method never
@@ -97,17 +102,20 @@ class BookmarkModelTest : public testing::Test, public BookmarkModelObserver {
}
void ClearCounts() {
- moved_count = added_count = removed_count = changed_count = 0;
+ reordered_count_ = moved_count = added_count = removed_count =
+ changed_count = 0;
}
void AssertObserverCount(int added_count,
int moved_count,
int removed_count,
- int changed_count) {
+ int changed_count,
+ int reordered_count) {
ASSERT_EQ(added_count, this->added_count);
ASSERT_EQ(moved_count, this->moved_count);
ASSERT_EQ(removed_count, this->removed_count);
ASSERT_EQ(changed_count, this->changed_count);
+ ASSERT_EQ(reordered_count, reordered_count_);
}
void AssertNodesEqual(BookmarkNode* expected, BookmarkNode* actual) {
@@ -145,6 +153,8 @@ class BookmarkModelTest : public testing::Test, public BookmarkModelObserver {
int changed_count;
+ int reordered_count_;
+
ObserverDetails observer_details;
};
@@ -168,7 +178,7 @@ TEST_F(BookmarkModelTest, AddURL) {
const GURL url("http://foo.com");
BookmarkNode* new_node = model.AddURL(root, 0, title, url);
- AssertObserverCount(1, 0, 0, 0);
+ AssertObserverCount(1, 0, 0, 0, 0);
observer_details.AssertEquals(root, NULL, 0, -1);
ASSERT_EQ(1, root->GetChildCount());
@@ -186,7 +196,7 @@ TEST_F(BookmarkModelTest, AddGroup) {
const std::wstring title(L"foo");
BookmarkNode* new_node = model.AddGroup(root, 0, title);
- AssertObserverCount(1, 0, 0, 0);
+ AssertObserverCount(1, 0, 0, 0, 0);
observer_details.AssertEquals(root, NULL, 0, -1);
ASSERT_EQ(1, root->GetChildCount());
@@ -199,7 +209,7 @@ TEST_F(BookmarkModelTest, AddGroup) {
// Add another group, just to make sure group_ids are incremented correctly.
ClearCounts();
BookmarkNode* new_node2 = model.AddGroup(root, 0, title);
- AssertObserverCount(1, 0, 0, 0);
+ AssertObserverCount(1, 0, 0, 0, 0);
observer_details.AssertEquals(root, NULL, 0, -1);
}
@@ -212,7 +222,7 @@ TEST_F(BookmarkModelTest, RemoveURL) {
model.Remove(root, 0);
ASSERT_EQ(0, root->GetChildCount());
- AssertObserverCount(0, 0, 1, 0);
+ AssertObserverCount(0, 0, 1, 0, 0);
observer_details.AssertEquals(root, NULL, 0, -1);
// Make sure there is no mapping for the URL.
@@ -235,7 +245,7 @@ TEST_F(BookmarkModelTest, RemoveGroup) {
// Now remove the group.
model.Remove(root, 0);
ASSERT_EQ(0, root->GetChildCount());
- AssertObserverCount(0, 0, 1, 0);
+ AssertObserverCount(0, 0, 1, 0, 0);
observer_details.AssertEquals(root, NULL, 0, -1);
// Make sure there is no mapping for the URL.
@@ -252,7 +262,7 @@ TEST_F(BookmarkModelTest, SetTitle) {
title = L"foo2";
model.SetTitle(node, title);
- AssertObserverCount(0, 0, 0, 1);
+ AssertObserverCount(0, 0, 0, 1, 0);
observer_details.AssertEquals(node, NULL, -1, -1);
EXPECT_EQ(title, node->GetTitle());
}
@@ -267,7 +277,7 @@ TEST_F(BookmarkModelTest, Move) {
model.Move(node, group1, 0);
- AssertObserverCount(0, 1, 0, 0);
+ AssertObserverCount(0, 1, 0, 0, 0);
observer_details.AssertEquals(root, group1, 1, 0);
EXPECT_TRUE(group1 == node->GetParent());
EXPECT_EQ(1, root->GetChildCount());
@@ -278,7 +288,7 @@ TEST_F(BookmarkModelTest, Move) {
// And remove the group.
ClearCounts();
model.Remove(root, 0);
- AssertObserverCount(0, 0, 1, 0);
+ AssertObserverCount(0, 0, 1, 0, 0);
observer_details.AssertEquals(root, NULL, 0, -1);
EXPECT_TRUE(model.GetMostRecentlyAddedNodeForURL(url) == NULL);
EXPECT_EQ(0, root->GetChildCount());
@@ -827,3 +837,25 @@ TEST_F(BookmarkModelTestWithProfile2, RemoveNotification) {
// This triggers blocking on the BookmarkModel.
profile_->GetHistoryService(Profile::EXPLICIT_ACCESS)->DeleteURL(url);
}
+
+TEST_F(BookmarkModelTest, Sort) {
+ // Populate the bookmark bar node with nodes for 'B', 'a', 'd' and 'C'.
+ TestNode bbn;
+ PopulateNodeFromString(L"B a d C", &bbn);
+ BookmarkNode* parent = model.GetBookmarkBarNode();
+ PopulateBookmarkNode(&bbn, &model, parent);
+
+ ClearCounts();
+
+ // Sort the children of the bookmark bar node.
+ model.SortChildren(parent);
+
+ // Make sure we were notified.
+ AssertObserverCount(0, 0, 0, 0, 1);
+
+ // Make sure the order matches.
+ EXPECT_TRUE(parent->GetChild(0)->GetTitle() == L"a");
+ EXPECT_TRUE(parent->GetChild(1)->GetTitle() == L"B");
+ EXPECT_TRUE(parent->GetChild(2)->GetTitle() == L"C");
+ EXPECT_TRUE(parent->GetChild(3)->GetTitle() == L"d");
+}
diff --git a/chrome/views/tree_node_model.h b/chrome/views/tree_node_model.h
index 18b93f9..52126d6 100644
--- a/chrome/views/tree_node_model.h
+++ b/chrome/views/tree_node_model.h
@@ -92,11 +92,6 @@ class TreeNode : public TreeModelNode {
return node;
}
- // Returns the children.
- std::vector<NodeType*> GetChildren() {
- return children_->v;
- }
-
// Returns the number of children.
int GetChildCount() {
return static_cast<int>(children_->size());
@@ -146,6 +141,9 @@ class TreeNode : public TreeModelNode {
return parent_ ? parent_->HasAncestor(ancestor) : false;
}
+ protected:
+ std::vector<NodeType*>& children() { return children_.get(); }
+
private:
// Title displayed in the tree.
std::wstring title_;
@@ -274,4 +272,3 @@ class TreeNodeModel : public TreeModel {
} // namespace views
#endif // CHROME_VIEWS_TREE_NODE_MODEL_H__
-