// 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/gtk/bookmark_context_menu.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/browser.h" #include "chrome/browser/browser_list.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/browser/tab_contents/tab_contents.h" #include "chrome/common/pref_names.h" #include "chrome/common/pref_service.h" #include "grit/generated_resources.h" // TODO(port): Port these files. #if defined(OS_WIN) #include "views/window/window.h" #endif 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::NativeView 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::NativeView 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_) { #if defined(OS_WIN) || (defined(OS_LINUX) && !defined(TOOLKIT_VIEWS)) BookmarkManager::SelectInTree(profile_, node); #else NOTIMPLEMENTED() << "BookmarkManager not yet implemented"; #endif } } 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); }; #if defined(OS_WIN) || (defined(OS_LINUX) && !defined(TOOLKIT_VIEWS)) // 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); }; #endif // #if defined(OS_WIN) || (defined(OS_LINUX) && !defined(TOOLKIT_VIEWS)) } // namespace // BookmarkContextMenu ------------------------------------------- BookmarkContextMenu::BookmarkContextMenu( gfx::NativeView wnd, Profile* profile, Browser* browser, PageNavigator* navigator, const BookmarkNode* parent, const std::vector& selection, ConfigurationType configuration) : wnd_(wnd), profile_(profile), browser_(browser), navigator_(navigator), parent_(parent), selection_(selection), model_(profile->GetBookmarkModel()), configuration_(configuration) { DCHECK(profile_); DCHECK(model_->IsLoaded()); CreateMenuObject(); if (configuration != BOOKMARK_MANAGER_ORGANIZE_MENU) { if (selection.size() == 1 && selection[0]->is_url()) { AppendItem(IDS_BOOMARK_BAR_OPEN_ALL, IDS_BOOMARK_BAR_OPEN_IN_NEW_TAB); AppendItem(IDS_BOOMARK_BAR_OPEN_ALL_NEW_WINDOW, IDS_BOOMARK_BAR_OPEN_IN_NEW_WINDOW); AppendItem(IDS_BOOMARK_BAR_OPEN_ALL_INCOGNITO, IDS_BOOMARK_BAR_OPEN_INCOGNITO); } else { AppendItem(IDS_BOOMARK_BAR_OPEN_ALL, IDS_BOOMARK_BAR_OPEN_ALL); AppendItem(IDS_BOOMARK_BAR_OPEN_ALL_NEW_WINDOW, IDS_BOOMARK_BAR_OPEN_ALL_NEW_WINDOW); AppendItem(IDS_BOOMARK_BAR_OPEN_ALL_INCOGNITO, IDS_BOOMARK_BAR_OPEN_ALL_INCOGNITO); } AppendSeparator(); } if (selection.size() == 1 && selection[0]->is_folder()) { AppendItem(IDS_BOOKMARK_BAR_RENAME_FOLDER); } else { AppendItem(IDS_BOOKMARK_BAR_EDIT); } AppendItem(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) { AppendItem(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) { AppendSeparator(); AppendItem(IDS_CUT); AppendItem(IDS_COPY); AppendItem(IDS_PASTE); } if (configuration == BOOKMARK_MANAGER_ORGANIZE_MENU) { AppendSeparator(); AppendItem(IDS_BOOKMARK_MANAGER_SORT); } AppendSeparator(); AppendItem(IDS_BOOMARK_BAR_ADD_NEW_BOOKMARK); AppendItem(IDS_BOOMARK_BAR_NEW_FOLDER); if (configuration == BOOKMARK_BAR) { AppendSeparator(); AppendItem(IDS_BOOKMARK_MANAGER); AppendCheckboxItem(IDS_BOOMARK_BAR_ALWAYS_SHOW); } model_->AddObserver(this); } BookmarkContextMenu::~BookmarkContextMenu() { if (model_) model_->RemoveObserver(this); } void BookmarkContextMenu::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: { PageNavigator* navigator = browser_ ? browser_->GetSelectedTabContents() : navigator_; 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(wnd_, 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()) { #if defined(OS_WIN) || (defined(OS_LINUX) && !defined(TOOLKIT_VIEWS)) BookmarkEditor::Configuration editor_config; if (configuration_ == BOOKMARK_BAR) editor_config = BookmarkEditor::SHOW_TREE; else editor_config = BookmarkEditor::NO_TREE; BookmarkEditor::Show(wnd_, profile_, NULL, selection_[0], editor_config, NULL); #else NOTIMPLEMENTED(); #endif } else { EditFolderController::Show(profile_, wnd_, 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_); #if defined(OS_WIN) || (defined(OS_LINUX) && !defined(TOOLKIT_VIEWS)) 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(wnd_, profile_, GetParentForNewNodes(), NULL, editor_config, handler); #else NOTIMPLEMENTED(); #endif break; } case IDS_BOOMARK_BAR_NEW_FOLDER: { UserMetrics::RecordAction(L"BookmarkBar_ContextMenu_NewFolder", profile_); EditFolderController::Show(profile_, wnd_, 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; } #if defined(OS_WIN) || (defined(OS_LINUX) && !defined(TOOLKIT_VIEWS)) BookmarkManager::SelectInTree(profile_, selection_[0]); #else NOTIMPLEMENTED() << "Bookmark Manager not implemented"; #endif break; case IDS_BOOKMARK_MANAGER: UserMetrics::RecordAction(L"ShowBookmarkManager", profile_); #if defined(OS_WIN) || (defined(OS_LINUX) && !defined(TOOLKIT_VIEWS)) BookmarkManager::Show(profile_); #else NOTIMPLEMENTED() << "Bookmark Manager not implemented"; #endif 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 BookmarkContextMenu::IsItemChecked(int id) const { DCHECK(id == IDS_BOOMARK_BAR_ALWAYS_SHOW); return profile_->GetPrefs()->GetBoolean(prefs::kShowBookmarkBar); } bool BookmarkContextMenu::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 BookmarkContextMenu::BookmarkModelBeingDeleted(BookmarkModel* model) { ModelChanged(); } void BookmarkContextMenu::BookmarkNodeMoved(BookmarkModel* model, const BookmarkNode* old_parent, int old_index, const BookmarkNode* new_parent, int new_index) { ModelChanged(); } void BookmarkContextMenu::BookmarkNodeAdded(BookmarkModel* model, const BookmarkNode* parent, int index) { ModelChanged(); } void BookmarkContextMenu::BookmarkNodeRemoved(BookmarkModel* model, const BookmarkNode* parent, int index, const BookmarkNode* node) { ModelChanged(); } void BookmarkContextMenu::BookmarkNodeChanged(BookmarkModel* model, const BookmarkNode* node) { ModelChanged(); } void BookmarkContextMenu::BookmarkNodeChildrenReordered( BookmarkModel* model, const BookmarkNode* node) { ModelChanged(); } void BookmarkContextMenu::ModelChanged() { menu_->Cancel(); } BookmarkModel* BookmarkContextMenu::RemoveModelObserver() { BookmarkModel* model = model_; model_->RemoveObserver(this); model_ = NULL; return model; } bool BookmarkContextMenu::HasURLs() const { for (size_t i = 0; i < selection_.size(); ++i) { if (NodeHasURLs(selection_[i])) return true; } return false; } const BookmarkNode* BookmarkContextMenu::GetParentForNewNodes() const { return (selection_.size() == 1 && selection_[0]->is_folder()) ? selection_[0] : parent_; }