1 // Copyright 2013 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 "chrome/browser/undo/bookmark_undo_service.h"
7 #include "chrome/browser/bookmarks/bookmark_model.h"
8 #include "chrome/browser/bookmarks/bookmark_model_factory.h"
9 #include "chrome/browser/bookmarks/bookmark_node_data.h"
10 #include "chrome/browser/bookmarks/bookmark_utils.h"
11 #include "chrome/browser/profiles/profile.h"
12 #include "chrome/browser/undo/bookmark_renumber_observer.h"
13 #include "chrome/browser/undo/bookmark_undo_service_factory.h"
14 #include "chrome/browser/undo/undo_manager_utils.h"
15 #include "chrome/browser/undo/undo_operation.h"
19 // BookmarkUndoOperation ------------------------------------------------------
21 // Base class for all bookmark related UndoOperations that facilitates access to
22 // the BookmarkUndoService.
23 class BookmarkUndoOperation : public UndoOperation,
24 public BookmarkRenumberObserver {
26 explicit BookmarkUndoOperation(Profile* profile);
27 virtual ~BookmarkUndoOperation() {}
29 BookmarkModel* GetBookmarkModel() const;
30 BookmarkRenumberObserver* GetUndoRenumberObserver() const;
36 BookmarkUndoOperation::BookmarkUndoOperation(Profile* profile)
40 BookmarkModel* BookmarkUndoOperation::GetBookmarkModel() const {
41 return BookmarkModelFactory::GetForProfile(profile_);
44 BookmarkRenumberObserver* BookmarkUndoOperation::GetUndoRenumberObserver()
46 return BookmarkUndoServiceFactory::GetForProfile(profile_);
49 // BookmarkAddOperation -------------------------------------------------------
51 // Handles the undo of the insertion of a bookmark or folder.
52 class BookmarkAddOperation : public BookmarkUndoOperation {
54 BookmarkAddOperation(Profile* profile, const BookmarkNode* parent, int index);
55 virtual ~BookmarkAddOperation() {}
58 virtual void Undo() OVERRIDE;
60 // BookmarkRenumberObserver:
61 virtual void OnBookmarkRenumbered(int64 old_id, int64 new_id) OVERRIDE;
67 DISALLOW_COPY_AND_ASSIGN(BookmarkAddOperation);
70 BookmarkAddOperation::BookmarkAddOperation(Profile* profile,
71 const BookmarkNode* parent,
73 : BookmarkUndoOperation(profile),
74 parent_id_(parent->id()),
78 void BookmarkAddOperation::Undo() {
79 BookmarkModel* model = GetBookmarkModel();
80 const BookmarkNode* parent = model->GetNodeByID(parent_id_);
83 model->Remove(parent, index_);
86 void BookmarkAddOperation::OnBookmarkRenumbered(int64 old_id, int64 new_id) {
87 if (parent_id_ == old_id)
91 // BookmarkRemoveOperation ----------------------------------------------------
93 // Handles the undo of the deletion of a bookmark node. For a bookmark folder,
94 // the information for all descendant bookmark nodes is maintained.
96 // The BookmarkModel allows only single bookmark node to be removed.
97 class BookmarkRemoveOperation : public BookmarkUndoOperation {
99 BookmarkRemoveOperation(Profile* profile,
100 const BookmarkNode* parent,
102 const BookmarkNode* node);
103 virtual ~BookmarkRemoveOperation() {}
106 virtual void Undo() OVERRIDE;
108 // BookmarkRenumberObserver:
109 virtual void OnBookmarkRenumbered(int64 old_id, int64 new_id) OVERRIDE;
112 void UpdateBookmarkIds(const BookmarkNodeData::Element& element,
113 const BookmarkNode* parent,
114 int index_added_at) const;
117 const int old_index_;
118 BookmarkNodeData removed_node_;
120 DISALLOW_COPY_AND_ASSIGN(BookmarkRemoveOperation);
123 BookmarkRemoveOperation::BookmarkRemoveOperation(Profile* profile,
124 const BookmarkNode* parent,
126 const BookmarkNode* node)
127 : BookmarkUndoOperation(profile),
128 parent_id_(parent->id()),
129 old_index_(old_index),
130 removed_node_(node) {
133 void BookmarkRemoveOperation::Undo() {
134 DCHECK(removed_node_.is_valid());
135 BookmarkModel* model = GetBookmarkModel();
136 const BookmarkNode* parent = model->GetNodeByID(parent_id_);
139 bookmark_utils::CloneBookmarkNode(model, removed_node_.elements, parent,
141 UpdateBookmarkIds(removed_node_.elements[0], parent, old_index_);
144 void BookmarkRemoveOperation::UpdateBookmarkIds(
145 const BookmarkNodeData::Element& element,
146 const BookmarkNode* parent,
147 int index_added_at) const {
148 const BookmarkNode* node = parent->GetChild(index_added_at);
149 if (element.id() != node->id())
150 GetUndoRenumberObserver()->OnBookmarkRenumbered(element.id(), node->id());
151 if (!element.is_url) {
152 for (int i = 0; i < static_cast<int>(element.children.size()); ++i)
153 UpdateBookmarkIds(element.children[i], node, 0);
157 void BookmarkRemoveOperation::OnBookmarkRenumbered(int64 old_id, int64 new_id) {
158 if (parent_id_ == old_id)
162 // BookmarkEditOperation ------------------------------------------------------
164 // Handles the undo of the modification of a bookmark node.
165 class BookmarkEditOperation : public BookmarkUndoOperation {
167 BookmarkEditOperation(Profile* profile,
168 const BookmarkNode* node);
169 virtual ~BookmarkEditOperation() {}
172 virtual void Undo() OVERRIDE;
174 // BookmarkRenumberObserver:
175 virtual void OnBookmarkRenumbered(int64 old_id, int64 new_id) OVERRIDE;
179 BookmarkNodeData original_bookmark_;
181 DISALLOW_COPY_AND_ASSIGN(BookmarkEditOperation);
184 BookmarkEditOperation::BookmarkEditOperation(Profile* profile,
185 const BookmarkNode* node)
186 : BookmarkUndoOperation(profile),
187 node_id_(node->id()),
188 original_bookmark_(node) {
191 void BookmarkEditOperation::Undo() {
192 DCHECK(original_bookmark_.is_valid());
193 BookmarkModel* model = GetBookmarkModel();
194 const BookmarkNode* node = model->GetNodeByID(node_id_);
197 model->SetTitle(node, original_bookmark_.elements[0].title);
198 if (original_bookmark_.elements[0].is_url)
199 model->SetURL(node, original_bookmark_.elements[0].url);
202 void BookmarkEditOperation::OnBookmarkRenumbered(int64 old_id, int64 new_id) {
203 if (node_id_ == old_id)
207 // BookmarkMoveOperation ------------------------------------------------------
209 // Handles the undo of a bookmark being moved to a new location.
210 class BookmarkMoveOperation : public BookmarkUndoOperation {
212 BookmarkMoveOperation(Profile* profile,
213 const BookmarkNode* old_parent,
215 const BookmarkNode* new_parent,
217 virtual ~BookmarkMoveOperation() {}
220 virtual void Undo() OVERRIDE;
222 // BookmarkRenumberObserver:
223 virtual void OnBookmarkRenumbered(int64 old_id, int64 new_id) OVERRIDE;
226 int64 old_parent_id_;
227 int64 new_parent_id_;
231 DISALLOW_COPY_AND_ASSIGN(BookmarkMoveOperation);
234 BookmarkMoveOperation::BookmarkMoveOperation(Profile* profile,
235 const BookmarkNode* old_parent,
237 const BookmarkNode* new_parent,
239 : BookmarkUndoOperation(profile),
240 old_parent_id_(old_parent->id()),
241 new_parent_id_(new_parent->id()),
242 old_index_(old_index),
243 new_index_(new_index) {
246 void BookmarkMoveOperation::Undo() {
247 BookmarkModel* model = GetBookmarkModel();
248 const BookmarkNode* old_parent = model->GetNodeByID(old_parent_id_);
249 const BookmarkNode* new_parent = model->GetNodeByID(new_parent_id_);
253 const BookmarkNode* node = new_parent->GetChild(new_index_);
254 int destination_index = old_index_;
256 // If the bookmark was moved up within the same parent then the destination
257 // index needs to be incremented since the old index did not account for the
259 if (old_parent == new_parent && new_index_ < old_index_)
262 model->Move(node, old_parent, destination_index);
265 void BookmarkMoveOperation::OnBookmarkRenumbered(int64 old_id, int64 new_id) {
266 if (old_parent_id_ == old_id)
267 old_parent_id_ = new_id;
268 if (new_parent_id_ == old_id)
269 new_parent_id_ = new_id;
272 // BookmarkReorderOperation ---------------------------------------------------
274 // Handle the undo of reordering of bookmarks that can happen as a result of
275 // sorting a bookmark folder by name or the undo of that operation. The change
276 // of order is not recursive so only the order of the immediate children of the
277 // folder need to be restored.
278 class BookmarkReorderOperation : public BookmarkUndoOperation {
280 BookmarkReorderOperation(Profile* profile,
281 const BookmarkNode* parent);
282 virtual ~BookmarkReorderOperation();
285 virtual void Undo() OVERRIDE;
287 // BookmarkRenumberObserver:
288 virtual void OnBookmarkRenumbered(int64 old_id, int64 new_id) OVERRIDE;
292 std::vector<int64> ordered_bookmarks_;
294 DISALLOW_COPY_AND_ASSIGN(BookmarkReorderOperation);
297 BookmarkReorderOperation::BookmarkReorderOperation(Profile* profile,
298 const BookmarkNode* parent)
299 : BookmarkUndoOperation(profile),
300 parent_id_(parent->id()) {
301 ordered_bookmarks_.resize(parent->child_count());
302 for (int i = 0; i < parent->child_count(); ++i)
303 ordered_bookmarks_[i] = parent->GetChild(i)->id();
306 BookmarkReorderOperation::~BookmarkReorderOperation() {
309 void BookmarkReorderOperation::Undo() {
310 BookmarkModel* model = GetBookmarkModel();
311 const BookmarkNode* parent = model->GetNodeByID(parent_id_);
314 std::vector<const BookmarkNode*> ordered_nodes;
315 for (size_t i = 0; i < ordered_bookmarks_.size(); ++i)
316 ordered_nodes.push_back(model->GetNodeByID(ordered_bookmarks_[i]));
318 model->ReorderChildren(parent, ordered_nodes);
321 void BookmarkReorderOperation::OnBookmarkRenumbered(int64 old_id,
323 if (parent_id_ == old_id)
325 for (size_t i = 0; i < ordered_bookmarks_.size(); ++i) {
326 if (ordered_bookmarks_[i] == old_id)
327 ordered_bookmarks_[i] = new_id;
333 // BookmarkUndoService --------------------------------------------------------
335 BookmarkUndoService::BookmarkUndoService(Profile* profile) : profile_(profile) {
336 BookmarkModelFactory::GetForProfile(profile_)->AddObserver(this);
339 BookmarkUndoService::~BookmarkUndoService() {
340 BookmarkModelFactory::GetForProfile(profile_)->RemoveObserver(this);
343 void BookmarkUndoService::OnBookmarkRenumbered(int64 old_id, int64 new_id) {
344 std::vector<UndoOperation*> all_operations =
345 undo_manager()->GetAllUndoOperations();
346 for (std::vector<UndoOperation*>::iterator it = all_operations.begin();
347 it != all_operations.end(); ++it) {
348 static_cast<BookmarkUndoOperation*>(*it)->OnBookmarkRenumbered(old_id,
353 void BookmarkUndoService::Loaded(BookmarkModel* model, bool ids_reassigned) {
354 undo_manager_.RemoveAllOperations();
357 void BookmarkUndoService::BookmarkModelBeingDeleted(BookmarkModel* model) {
358 undo_manager_.RemoveAllOperations();
361 void BookmarkUndoService::BookmarkNodeMoved(BookmarkModel* model,
362 const BookmarkNode* old_parent,
364 const BookmarkNode* new_parent,
366 scoped_ptr<UndoOperation> op(new BookmarkMoveOperation(profile_,
371 undo_manager()->AddUndoOperation(op.Pass());
374 void BookmarkUndoService::BookmarkNodeAdded(BookmarkModel* model,
375 const BookmarkNode* parent,
377 scoped_ptr<UndoOperation> op(new BookmarkAddOperation(profile_,
380 undo_manager()->AddUndoOperation(op.Pass());
383 void BookmarkUndoService::OnWillRemoveBookmarks(BookmarkModel* model,
384 const BookmarkNode* parent,
386 const BookmarkNode* node) {
387 scoped_ptr<UndoOperation> op(new BookmarkRemoveOperation(profile_,
391 undo_manager()->AddUndoOperation(op.Pass());
394 void BookmarkUndoService::OnWillRemoveAllBookmarks(BookmarkModel* model) {
395 ScopedGroupingAction merge_removes(undo_manager());
396 for (int i = 0; i < model->root_node()->child_count(); ++i) {
397 const BookmarkNode* permanent_node = model->root_node()->GetChild(i);
398 for (int j = permanent_node->child_count() - 1; j >= 0; --j) {
399 scoped_ptr<UndoOperation> op(new BookmarkRemoveOperation(profile_,
400 permanent_node, j, permanent_node->GetChild(j)));
401 undo_manager()->AddUndoOperation(op.Pass());
406 void BookmarkUndoService::OnWillChangeBookmarkNode(BookmarkModel* model,
407 const BookmarkNode* node) {
408 scoped_ptr<UndoOperation> op(new BookmarkEditOperation(profile_, node));
409 undo_manager()->AddUndoOperation(op.Pass());
412 void BookmarkUndoService::OnWillReorderBookmarkNode(BookmarkModel* model,
413 const BookmarkNode* node) {
414 scoped_ptr<UndoOperation> op(new BookmarkReorderOperation(profile_, node));
415 undo_manager()->AddUndoOperation(op.Pass());