// Copyright (c) 2009 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "chrome/browser/bookmarks/bookmark_context_menu_controller.h" #include "app/l10n_util.h" #include "base/compiler_specific.h" #include "chrome/browser/bookmarks/bookmark_editor.h" #include "chrome/browser/bookmarks/bookmark_manager.h" #include "chrome/browser/bookmarks/bookmark_model.h" #include "chrome/browser/bookmarks/bookmark_utils.h" #include "chrome/browser/input_window_dialog.h" #include "chrome/browser/metrics/user_metrics.h" #include "chrome/browser/profile.h" #include "chrome/browser/tab_contents/page_navigator.h" #include "chrome/common/pref_names.h" #include "chrome/common/pref_service.h" #include "grit/generated_resources.h" namespace { // Returns true if the specified node is of type URL, or has a descendant // of type URL. bool NodeHasURLs(const BookmarkNode* node) { if (node->is_url()) return true; for (int i = 0; i < node->GetChildCount(); ++i) { if (NodeHasURLs(node->GetChild(i))) return true; } return false; } // EditFolderController ------------------------------------------------------- // EditFolderController manages the editing and/or creation of a folder. If the // user presses ok, the name change is committed to the model. // // EditFolderController deletes itself when the window is closed. class EditFolderController : public InputWindowDialog::Delegate, public BookmarkModelObserver { public: virtual ~EditFolderController() { if (model_) model_->RemoveObserver(this); } static void Show(Profile* profile, gfx::NativeWindow wnd, const BookmarkNode* node, bool is_new, bool show_in_manager) { // EditFolderController deletes itself when done. EditFolderController* controller = new EditFolderController(profile, wnd, node, is_new, show_in_manager); controller->Show(); } private: EditFolderController(Profile* profile, gfx::NativeWindow wnd, const BookmarkNode* node, bool is_new, bool show_in_manager) : profile_(profile), model_(profile->GetBookmarkModel()), node_(node), is_new_(is_new), show_in_manager_(show_in_manager) { DCHECK(is_new_ || node); std::wstring title = is_new_ ? l10n_util::GetString(IDS_BOOMARK_FOLDER_EDITOR_WINDOW_TITLE_NEW) : l10n_util::GetString(IDS_BOOMARK_FOLDER_EDITOR_WINDOW_TITLE); std::wstring label = l10n_util::GetString(IDS_BOOMARK_BAR_EDIT_FOLDER_LABEL); std::wstring contents = is_new_ ? l10n_util::GetString(IDS_BOOMARK_EDITOR_NEW_FOLDER_NAME) : node_->GetTitle(); dialog_ = InputWindowDialog::Create(wnd, title, label, contents, this); model_->AddObserver(this); } void Show() { dialog_->Show(); } // InputWindowDialog::Delegate methods. virtual bool IsValid(const std::wstring& text) { return !text.empty(); } virtual void InputAccepted(const std::wstring& text) { if (is_new_) { ALLOW_UNUSED const BookmarkNode* node = model_->AddGroup(node_, node_->GetChildCount(), text); if (show_in_manager_) { BookmarkManager::SelectInTree(profile_, node); } } else { model_->SetTitle(node_, text); } } virtual void InputCanceled() { } // BookmarkModelObserver methods, all invoke ModelChanged and close the // dialog. virtual void Loaded(BookmarkModel* model) {} virtual void BookmarkModelBeingDeleted(BookmarkModel* model) { model_->RemoveObserver(this); model_ = NULL; ModelChanged(); } virtual void BookmarkNodeMoved(BookmarkModel* model, const BookmarkNode* old_parent, int old_index, const BookmarkNode* new_parent, int new_index) { ModelChanged(); } virtual void BookmarkNodeAdded(BookmarkModel* model, const BookmarkNode* parent, int index) { ModelChanged(); } virtual void BookmarkNodeRemoved(BookmarkModel* model, const BookmarkNode* parent, int index, const BookmarkNode* node) { ModelChanged(); } virtual void BookmarkNodeChanged(BookmarkModel* model, const BookmarkNode* node) { ModelChanged(); } virtual void BookmarkNodeFavIconLoaded(BookmarkModel* model, const BookmarkNode* node) {} virtual void BookmarkNodeChildrenReordered(BookmarkModel* model, const BookmarkNode* node) { ModelChanged(); } void ModelChanged() { dialog_->Close(); } Profile* profile_; BookmarkModel* model_; // If is_new is true, this is the parent to create the new node under. // Otherwise this is the node to change the title of. const BookmarkNode* node_; bool is_new_; // If is_new_ is true and a new node is created, it is selected in the // bookmark manager. bool show_in_manager_; InputWindowDialog* dialog_; DISALLOW_COPY_AND_ASSIGN(EditFolderController); }; // SelectOnCreationHandler ---------------------------------------------------- // Used when adding a new bookmark. If a new bookmark is created it is selected // in the bookmark manager. class SelectOnCreationHandler : public BookmarkEditor::Handler { public: explicit SelectOnCreationHandler(Profile* profile) : profile_(profile) { } virtual void NodeCreated(const BookmarkNode* new_node) { BookmarkManager::SelectInTree(profile_, new_node); } private: Profile* profile_; DISALLOW_COPY_AND_ASSIGN(SelectOnCreationHandler); }; } // namespace BookmarkContextMenuController::BookmarkContextMenuController( gfx::NativeWindow parent_window, BookmarkContextMenuControllerDelegate* delegate, Profile* profile, PageNavigator* navigator, const BookmarkNode* parent, const std::vector& selection, ConfigurationType configuration) : parent_window_(parent_window), delegate_(delegate), profile_(profile), navigator_(navigator), parent_(parent), selection_(selection), configuration_(configuration), model_(profile->GetBookmarkModel()) { DCHECK(profile_); DCHECK(model_->IsLoaded()); model_->AddObserver(this); } BookmarkContextMenuController::~BookmarkContextMenuController() { if (model_) model_->RemoveObserver(this); } void BookmarkContextMenuController::BuildMenu() { if (configuration_ != BOOKMARK_MANAGER_ORGANIZE_MENU) { if (selection_.size() == 1 && selection_[0]->is_url()) { delegate_->AddItemWithStringId(IDS_BOOMARK_BAR_OPEN_ALL, IDS_BOOMARK_BAR_OPEN_IN_NEW_TAB); delegate_->AddItemWithStringId(IDS_BOOMARK_BAR_OPEN_ALL_NEW_WINDOW, IDS_BOOMARK_BAR_OPEN_IN_NEW_WINDOW); delegate_->AddItemWithStringId(IDS_BOOMARK_BAR_OPEN_ALL_INCOGNITO, IDS_BOOMARK_BAR_OPEN_INCOGNITO); } else { delegate_->AddItem(IDS_BOOMARK_BAR_OPEN_ALL); delegate_->AddItem(IDS_BOOMARK_BAR_OPEN_ALL_NEW_WINDOW); delegate_->AddItem(IDS_BOOMARK_BAR_OPEN_ALL_INCOGNITO); } delegate_->AddSeparator(); } if (selection_.size() == 1 && selection_[0]->is_folder()) { delegate_->AddItem(IDS_BOOKMARK_BAR_RENAME_FOLDER); } else { delegate_->AddItem(IDS_BOOKMARK_BAR_EDIT); } delegate_->AddItem(IDS_BOOKMARK_BAR_REMOVE); if (configuration_ == BOOKMARK_MANAGER_TABLE || configuration_ == BOOKMARK_MANAGER_TABLE_OTHER || configuration_ == BOOKMARK_MANAGER_ORGANIZE_MENU || configuration_ == BOOKMARK_MANAGER_ORGANIZE_MENU_OTHER) { delegate_->AddItem(IDS_BOOKMARK_MANAGER_SHOW_IN_FOLDER); } if (configuration_ == BOOKMARK_MANAGER_TABLE || configuration_ == BOOKMARK_MANAGER_TABLE_OTHER || configuration_ == BOOKMARK_MANAGER_TREE || configuration_ == BOOKMARK_MANAGER_ORGANIZE_MENU || configuration_ == BOOKMARK_MANAGER_ORGANIZE_MENU_OTHER) { delegate_->AddSeparator(); delegate_->AddItem(IDS_CUT); delegate_->AddItem(IDS_COPY); delegate_->AddItem(IDS_PASTE); } if (configuration_ == BOOKMARK_MANAGER_ORGANIZE_MENU) { delegate_->AddSeparator(); delegate_->AddItem(IDS_BOOKMARK_MANAGER_SORT); } delegate_->AddSeparator(); delegate_->AddItem(IDS_BOOMARK_BAR_ADD_NEW_BOOKMARK); delegate_->AddItem(IDS_BOOMARK_BAR_NEW_FOLDER); if (configuration_ == BOOKMARK_BAR) { delegate_->AddSeparator(); delegate_->AddItem(IDS_BOOKMARK_MANAGER); delegate_->AddCheckboxItem(IDS_BOOMARK_BAR_ALWAYS_SHOW); } } void BookmarkContextMenuController::ExecuteCommand(int id) { switch (id) { case IDS_BOOMARK_BAR_OPEN_ALL: case IDS_BOOMARK_BAR_OPEN_ALL_INCOGNITO: case IDS_BOOMARK_BAR_OPEN_ALL_NEW_WINDOW: { WindowOpenDisposition initial_disposition; if (id == IDS_BOOMARK_BAR_OPEN_ALL) { initial_disposition = NEW_FOREGROUND_TAB; UserMetrics::RecordAction(L"BookmarkBar_ContextMenu_OpenAll", profile_); } else if (id == IDS_BOOMARK_BAR_OPEN_ALL_NEW_WINDOW) { initial_disposition = NEW_WINDOW; UserMetrics::RecordAction( L"BookmarkBar_ContextMenu_OpenAllInNewWindow", profile_); } else { initial_disposition = OFF_THE_RECORD; UserMetrics::RecordAction( L"BookmarkBar_ContextMenu_OpenAllIncognito", profile_); } bookmark_utils::OpenAll(parent_window_, profile_, navigator_, selection_, initial_disposition); break; } case IDS_BOOKMARK_BAR_RENAME_FOLDER: case IDS_BOOKMARK_BAR_EDIT: UserMetrics::RecordAction(L"BookmarkBar_ContextMenu_Edit", profile_); if (selection_.size() != 1) { NOTREACHED(); return; } if (selection_[0]->is_url()) { BookmarkEditor::Configuration editor_config; if (configuration_ == BOOKMARK_BAR) editor_config = BookmarkEditor::SHOW_TREE; else editor_config = BookmarkEditor::NO_TREE; BookmarkEditor::Show(parent_window_, profile_, parent_, BookmarkEditor::EditDetails(selection_[0]), editor_config, NULL); } else { EditFolderController::Show(profile_, parent_window_, selection_[0], false, false); } break; case IDS_BOOKMARK_BAR_REMOVE: { UserMetrics::RecordAction(L"BookmarkBar_ContextMenu_Remove", profile_); BookmarkModel* model = RemoveModelObserver(); for (size_t i = 0; i < selection_.size(); ++i) { model->Remove(selection_[i]->GetParent(), selection_[i]->GetParent()->IndexOfChild(selection_[i])); } selection_.clear(); break; } case IDS_BOOMARK_BAR_ADD_NEW_BOOKMARK: { UserMetrics::RecordAction(L"BookmarkBar_ContextMenu_Add", profile_); BookmarkEditor::Configuration editor_config; BookmarkEditor::Handler* handler = NULL; if (configuration_ == BOOKMARK_BAR) { editor_config = BookmarkEditor::SHOW_TREE; } else { editor_config = BookmarkEditor::NO_TREE; // This is owned by the BookmarkEditorView. handler = new SelectOnCreationHandler(profile_); } BookmarkEditor::Show(parent_window_, profile_, GetParentForNewNodes(), BookmarkEditor::EditDetails(), editor_config, handler); break; } case IDS_BOOMARK_BAR_NEW_FOLDER: { UserMetrics::RecordAction(L"BookmarkBar_ContextMenu_NewFolder", profile_); EditFolderController::Show(profile_, parent_window_, GetParentForNewNodes(), true, configuration_ != BOOKMARK_BAR); break; } case IDS_BOOMARK_BAR_ALWAYS_SHOW: bookmark_utils::ToggleWhenVisible(profile_); break; case IDS_BOOKMARK_MANAGER_SHOW_IN_FOLDER: UserMetrics::RecordAction(L"BookmarkBar_ContextMenu_ShowInFolder", profile_); if (selection_.size() != 1) { NOTREACHED(); return; } BookmarkManager::SelectInTree(profile_, selection_[0]); break; case IDS_BOOKMARK_MANAGER: UserMetrics::RecordAction(L"ShowBookmarkManager", profile_); BookmarkManager::Show(profile_); break; case IDS_BOOKMARK_MANAGER_SORT: UserMetrics::RecordAction(L"BookmarkManager_Sort", profile_); model_->SortChildren(parent_); break; case IDS_COPY: case IDS_CUT: bookmark_utils::CopyToClipboard(profile_->GetBookmarkModel(), selection_, id == IDS_CUT); break; case IDS_PASTE: { // Always paste to parent. if (!parent_) return; int index = (selection_.size() == 1) ? parent_->IndexOfChild(selection_[0]) : -1; if (index != -1) index++; bookmark_utils::PasteFromClipboard(profile_->GetBookmarkModel(), parent_, index); break; } default: NOTREACHED(); } } bool BookmarkContextMenuController::IsItemChecked(int id) const { DCHECK(id == IDS_BOOMARK_BAR_ALWAYS_SHOW); return profile_->GetPrefs()->GetBoolean(prefs::kShowBookmarkBar); } bool BookmarkContextMenuController::IsCommandEnabled(int id) const { bool is_root_node = (selection_.size() == 1 && selection_[0]->GetParent() == model_->root_node()); switch (id) { case IDS_BOOMARK_BAR_OPEN_INCOGNITO: return !profile_->IsOffTheRecord(); case IDS_BOOMARK_BAR_OPEN_ALL_INCOGNITO: return HasURLs() && !profile_->IsOffTheRecord(); case IDS_BOOMARK_BAR_OPEN_ALL: case IDS_BOOMARK_BAR_OPEN_ALL_NEW_WINDOW: return HasURLs(); case IDS_BOOKMARK_BAR_RENAME_FOLDER: case IDS_BOOKMARK_BAR_EDIT: return selection_.size() == 1 && !is_root_node; case IDS_BOOKMARK_BAR_REMOVE: return !selection_.empty() && !is_root_node; case IDS_BOOKMARK_MANAGER_SHOW_IN_FOLDER: return (configuration_ == BOOKMARK_MANAGER_TABLE_OTHER || configuration_ == BOOKMARK_MANAGER_ORGANIZE_MENU_OTHER) && selection_.size() == 1; case IDS_BOOKMARK_MANAGER_SORT: return parent_ && parent_ != model_->root_node(); case IDS_BOOMARK_BAR_NEW_FOLDER: case IDS_BOOMARK_BAR_ADD_NEW_BOOKMARK: return GetParentForNewNodes() != NULL; case IDS_COPY: case IDS_CUT: return selection_.size() > 0 && !is_root_node; case IDS_PASTE: // Always paste to parent. return bookmark_utils::CanPasteFromClipboard(parent_); } return true; } void BookmarkContextMenuController::BookmarkModelBeingDeleted( BookmarkModel* model) { ModelChanged(); } void BookmarkContextMenuController::BookmarkNodeMoved( BookmarkModel* model, const BookmarkNode* old_parent, int old_index, const BookmarkNode* new_parent, int new_index) { ModelChanged(); } void BookmarkContextMenuController::BookmarkNodeAdded( BookmarkModel* model, const BookmarkNode* parent, int index) { ModelChanged(); } void BookmarkContextMenuController::BookmarkNodeRemoved( BookmarkModel* model, const BookmarkNode* parent, int index, const BookmarkNode* node) { ModelChanged(); } void BookmarkContextMenuController::BookmarkNodeChanged( BookmarkModel* model, const BookmarkNode* node) { ModelChanged(); } void BookmarkContextMenuController::BookmarkNodeChildrenReordered( BookmarkModel* model, const BookmarkNode* node) { ModelChanged(); } void BookmarkContextMenuController::ModelChanged() { delegate_->CloseMenu(); } BookmarkModel* BookmarkContextMenuController::RemoveModelObserver() { BookmarkModel* model = model_; model_->RemoveObserver(this); model_ = NULL; return model; } bool BookmarkContextMenuController::HasURLs() const { for (size_t i = 0; i < selection_.size(); ++i) { if (NodeHasURLs(selection_[i])) return true; } return false; } const BookmarkNode* BookmarkContextMenuController::GetParentForNewNodes() const { return (selection_.size() == 1 && selection_[0]->is_folder()) ? selection_[0] : parent_; }