1 // Copyright 2014 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.
5 #include "components/bookmarks/test/bookmark_test_helpers.h"
7 #include "base/basictypes.h"
8 #include "base/callback.h"
9 #include "base/compiler_specific.h"
10 #include "base/logging.h"
11 #include "base/run_loop.h"
12 #include "base/strings/utf_string_conversions.h"
13 #include "components/bookmarks/browser/base_bookmark_model_observer.h"
14 #include "components/bookmarks/browser/bookmark_model.h"
19 // BookmarkLoadObserver is used when blocking until the BookmarkModel finishes
20 // loading. As soon as the BookmarkModel finishes loading the message loop is
22 class BookmarkLoadObserver
: public BaseBookmarkModelObserver
{
24 explicit BookmarkLoadObserver(const base::Closure
& quit_task
);
25 virtual ~BookmarkLoadObserver();
28 // BaseBookmarkModelObserver:
29 virtual void BookmarkModelChanged() OVERRIDE
;
30 virtual void BookmarkModelLoaded(BookmarkModel
* model
,
31 bool ids_reassigned
) OVERRIDE
;
33 base::Closure quit_task_
;
35 DISALLOW_COPY_AND_ASSIGN(BookmarkLoadObserver
);
38 BookmarkLoadObserver::BookmarkLoadObserver(const base::Closure
& quit_task
)
39 : quit_task_(quit_task
) {}
41 BookmarkLoadObserver::~BookmarkLoadObserver() {}
43 void BookmarkLoadObserver::BookmarkModelChanged() {}
45 void BookmarkLoadObserver::BookmarkModelLoaded(BookmarkModel
* model
,
46 bool ids_reassigned
) {
50 // Helper function which does the actual work of creating the nodes for
51 // a particular level in the hierarchy.
52 std::string::size_type
AddNodesFromString(BookmarkModel
* model
,
53 const BookmarkNode
* node
,
54 const std::string
& model_string
,
55 std::string::size_type start_pos
) {
57 int index
= node
->child_count();
58 static const std::string
folder_tell(":[");
59 std::string::size_type end_pos
= model_string
.find(' ', start_pos
);
60 while (end_pos
!= std::string::npos
) {
61 std::string::size_type part_length
= end_pos
- start_pos
;
62 std::string node_name
= model_string
.substr(start_pos
, part_length
);
63 // Are we at the end of a folder group?
64 if (node_name
!= "]") {
65 // No, is it a folder?
68 tell
= node_name
.substr(part_length
- 2, 2);
69 if (tell
== folder_tell
) {
70 node_name
= node_name
.substr(0, part_length
- 2);
71 const BookmarkNode
* new_node
=
72 model
->AddFolder(node
, index
, base::UTF8ToUTF16(node_name
));
73 end_pos
= AddNodesFromString(model
, new_node
, model_string
,
76 std::string
url_string("http://");
77 url_string
+= std::string(node_name
.begin(), node_name
.end());
80 node
, index
, base::UTF8ToUTF16(node_name
), GURL(url_string
));
85 end_pos
= model_string
.find(' ', start_pos
);
98 void WaitForBookmarkModelToLoad(BookmarkModel
* model
) {
101 base::RunLoop run_loop
;
102 base::MessageLoop::ScopedNestableTaskAllower
allow(
103 base::MessageLoop::current());
105 BookmarkLoadObserver
observer(run_loop
.QuitClosure());
106 model
->AddObserver(&observer
);
108 model
->RemoveObserver(&observer
);
109 DCHECK(model
->loaded());
112 std::string
ModelStringFromNode(const BookmarkNode
* node
) {
113 // Since the children of the node are not available as a vector,
114 // we'll just have to do it the hard way.
115 int child_count
= node
->child_count();
116 std::string child_string
;
117 for (int i
= 0; i
< child_count
; ++i
) {
118 const BookmarkNode
* child
= node
->GetChild(i
);
119 if (child
->is_folder()) {
120 child_string
+= base::UTF16ToUTF8(child
->GetTitle()) + ":[ " +
121 ModelStringFromNode(child
) + "] ";
123 child_string
+= base::UTF16ToUTF8(child
->GetTitle()) + " ";
129 void AddNodesFromModelString(BookmarkModel
* model
,
130 const BookmarkNode
* node
,
131 const std::string
& model_string
) {
133 std::string::size_type start_pos
= 0;
134 std::string::size_type end_pos
=
135 AddNodesFromString(model
, node
, model_string
, start_pos
);
136 DCHECK(end_pos
== std::string::npos
);