diff options
Diffstat (limited to 'chrome/browser/views/bookmark_editor_view.cc')
-rw-r--r-- | chrome/browser/views/bookmark_editor_view.cc | 561 |
1 files changed, 561 insertions, 0 deletions
diff --git a/chrome/browser/views/bookmark_editor_view.cc b/chrome/browser/views/bookmark_editor_view.cc new file mode 100644 index 0000000..e7a834a --- /dev/null +++ b/chrome/browser/views/bookmark_editor_view.cc @@ -0,0 +1,561 @@ +// Copyright 2008, Google Inc. +// All rights reserved. +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +#include "chrome/browser/views/bookmark_editor_view.h" + +#include "base/basictypes.h" +#include "base/logging.h" +#include "base/string_util.h" +#include "chrome/app/locales/locale_settings.h" +#include "chrome/browser/history/history.h" +#include "chrome/browser/profile.h" +#include "chrome/browser/standard_layout.h" +#include "chrome/browser/url_fixer_upper.h" +#include "chrome/common/l10n_util.h" +#include "chrome/views/background.h" +#include "chrome/views/focus_manager.h" +#include "chrome/views/grid_layout.h" +#include "chrome/views/label.h" +#include "chrome/views/window.h" +#include "generated_resources.h" +#include "googleurl/src/gurl.h" + +using ChromeViews::ColumnSet; +using ChromeViews::GridLayout; +using ChromeViews::Label; +using ChromeViews::NativeButton; +using ChromeViews::TextField; +using ChromeViews::TreeNode; + +// Background color of text field when URL is invalid. +static const SkColor kErrorColor = SkColorSetRGB(0xFF, 0xBC, 0xBC); + +// Preferred width of the tree. +static const int kTreeWidth = 300; + +// ID for various children. +static const int kNewGroupButtonID = 1002; + +// static +void BookmarkEditorView::Show(HWND parent_hwnd, + Profile* profile, + const GURL& url, + const std::wstring& title) { + DCHECK(profile); + BookmarkEditorView* editor = new BookmarkEditorView(profile, url, title); + editor->Show(parent_hwnd); +} + +BookmarkEditorView::BookmarkEditorView(Profile* profile, + const GURL& url, + const std::wstring& title) + : profile_(profile), +#pragma warning(suppress: 4355) // Okay to pass "this" here. + new_group_button_( + l10n_util::GetString(IDS_BOOMARK_EDITOR_NEW_FOLDER_BUTTON)), + dialog_(NULL), + url_(url), + title_(title), + running_menu_for_root_(false) { + DCHECK(profile); + Init(); +} + +BookmarkEditorView::~BookmarkEditorView() { + bb_model_->RemoveObserver(this); +} + +bool BookmarkEditorView::IsDialogButtonEnabled(DialogButton button) const { + if (button == DIALOGBUTTON_OK) { + const GURL url(GetInputURL()); + return bb_model_->IsLoaded() && url.is_valid(); + } + return true; +} +bool BookmarkEditorView::IsModal() const { + return true; +} + +std::wstring BookmarkEditorView::GetWindowTitle() const { + return l10n_util::GetString(IDS_BOOMARK_EDITOR_TITLE); +} + +bool BookmarkEditorView::Accept() { + if (!IsDialogButtonEnabled(DIALOGBUTTON_OK)) { + // The url is invalid, focus the url field. + url_tf_.SelectAll(); + url_tf_.RequestFocus(); + return false; + } + // Otherwise save changes and close the dialog box. + ApplyEdits(); + return true; +} + +bool BookmarkEditorView::AreAcceleratorsEnabled(DialogButton button) { + return !tree_view_.GetEditingNode(); +} + +void BookmarkEditorView::Layout() { + // Let the grid layout manager lay out most of the dialog... + GetLayoutManager()->Layout(this); + + // Manually lay out the New Folder button in the same row as the OK/Cancel + // buttons... + CRect parent_bounds; + GetParent()->GetLocalBounds(&parent_bounds, false); + CSize prefsize; + new_group_button_.GetPreferredSize(&prefsize); + int button_y = parent_bounds.bottom - prefsize.cy - kButtonVEdgeMargin; + new_group_button_.SetBounds(kPanelHorizMargin, button_y, prefsize.cx, + prefsize.cy); +} + +void BookmarkEditorView::GetPreferredSize(CSize *out) { + DCHECK(out); + *out = ChromeViews::Window::GetLocalizedContentsSize( + IDS_EDITBOOKMARK_DIALOG_WIDTH_CHARS, + IDS_EDITBOOKMARK_DIALOG_HEIGHT_LINES).ToSIZE(); +} + +void BookmarkEditorView::DidChangeBounds(const CRect& previous, + const CRect& current) { + Layout(); +} + +void BookmarkEditorView::ViewHierarchyChanged(bool is_add, + ChromeViews::View* parent, + ChromeViews::View* child) { + if (child == this) { + // Add and remove the New Folder button from the ClientView's hierarchy. + if (is_add) { + parent->AddChildView(&new_group_button_); + } else { + parent->RemoveChildView(&new_group_button_); + } + } +} + +void BookmarkEditorView::OnTreeViewSelectionChanged( + ChromeViews::TreeView* tree_view) { +} + +bool BookmarkEditorView::CanEdit(ChromeViews::TreeView* tree_view, + ChromeViews::TreeModelNode* node) { + // Only allow editting of children of the bookmark bar node and other node. + BookmarkNode* bb_node = tree_model_->AsNode(node); + return (bb_node->GetParent() && bb_node->GetParent()->GetParent()); +} + +void BookmarkEditorView::ContentsChanged(TextField* sender, + const std::wstring& new_contents) { + UserInputChanged(); +} + +void BookmarkEditorView::ButtonPressed(NativeButton* sender) { + DCHECK(sender); + switch (sender->GetID()) { + case kNewGroupButtonID: + NewGroup(); + break; + + default: + NOTREACHED(); + } +} + +void BookmarkEditorView::ExecuteCommand(int id) { + DCHECK(tree_view_.GetSelectedNode()); + if (id == IDS_EDIT) { + tree_view_.StartEditing(tree_view_.GetSelectedNode()); + } else { + DCHECK(id == IDS_BOOMARK_EDITOR_NEW_FOLDER_MENU_ITEM); + NewGroup(); + } +} + +bool BookmarkEditorView::IsCommandEnabled(int id) const { + return (id != IDS_EDIT || !running_menu_for_root_); +} + +void BookmarkEditorView::Show(HWND parent_hwnd) { + dialog_ = ChromeViews::Window::CreateChromeWindow(parent_hwnd, gfx::Rect(), + this, this); + UserInputChanged(); + if (bb_model_->IsLoaded()) + ExpandAndSelect(); + dialog_->Show(); + // Select all the text in the name textfield. + title_tf_.SelectAll(); + // Give focus to the name textfield. + title_tf_.RequestFocus(); +} + +void BookmarkEditorView::Close() { + DCHECK(dialog_); + dialog_->Close(); +} + +void BookmarkEditorView::ShowContextMenu(View* source, + int x, + int y, + bool is_mouse_gesture) { + DCHECK(source == &tree_view_); + if (!tree_view_.GetSelectedNode()) + return; + running_menu_for_root_ = + (tree_model_->GetParent(tree_view_.GetSelectedNode()) == + tree_model_->GetRoot()); + context_menu_.reset(new Menu(this, Menu::TOPLEFT, + GetViewContainer()->GetHWND())); + context_menu_->AppendMenuItemWithLabel(IDS_EDIT, + l10n_util::GetString(IDS_EDIT)); + context_menu_->AppendMenuItemWithLabel( + IDS_BOOMARK_EDITOR_NEW_FOLDER_MENU_ITEM, + l10n_util::GetString(IDS_BOOMARK_EDITOR_NEW_FOLDER_MENU_ITEM)); + context_menu_->RunMenuAt(x, y); +} + +void BookmarkEditorView::Init() { + tree_view_.SetContextMenuController(this); + bb_model_ = profile_->GetBookmarkBarModel(); + DCHECK(bb_model_); + bb_model_->AddObserver(this); + + tree_view_.SetRootShown(false); + // Tell View not to delete all Views declared by value. + tree_view_.SetParentOwned(false); + new_group_button_.SetParentOwned(false); + url_tf_.SetParentOwned(false); + title_tf_.SetParentOwned(false); + + new_group_button_.SetEnabled(false); + new_group_button_.SetListener(this); + new_group_button_.SetID(kNewGroupButtonID); + + title_tf_.SetText(title_); + title_tf_.SetController(this); + + url_tf_.SetText(UTF8ToWide(url_.spec())); + url_tf_.SetController(this); + + // Yummy layout code. + const int labels_column_set_id = 0; + const int single_column_view_set_id = 1; + const int buttons_column_set_id = 2; + GridLayout* layout = CreatePanelGridLayout(this); + SetLayoutManager(layout); + ColumnSet* column_set = layout->AddColumnSet(labels_column_set_id); + column_set->AddColumn(GridLayout::LEADING, GridLayout::CENTER, 0, + GridLayout::USE_PREF, 0, 0); + column_set->AddPaddingColumn(0, kRelatedControlHorizontalSpacing); + column_set->AddColumn(GridLayout::FILL, GridLayout::CENTER, 1, + GridLayout::USE_PREF, 0, 0); + + column_set = layout->AddColumnSet(single_column_view_set_id); + column_set->AddColumn(GridLayout::FILL, GridLayout::FILL, 1, + GridLayout::FIXED, kTreeWidth, 0); + + column_set = layout->AddColumnSet(buttons_column_set_id); + column_set->AddColumn(GridLayout::FILL, GridLayout::LEADING, 0, + GridLayout::USE_PREF, 0, 0); + column_set->AddPaddingColumn(1, kRelatedControlHorizontalSpacing); + column_set->AddColumn(GridLayout::FILL, GridLayout::LEADING, 0, + GridLayout::USE_PREF, 0, 0); + column_set->AddPaddingColumn(0, kRelatedControlHorizontalSpacing); + column_set->AddColumn(GridLayout::FILL, GridLayout::LEADING, 0, + GridLayout::USE_PREF, 0, 0); + column_set->LinkColumnSizes(0, 2, 4, -1); + + layout->StartRow(0, labels_column_set_id); + layout->AddView( + new Label(l10n_util::GetString(IDS_BOOMARK_EDITOR_NAME_LABEL))); + layout->AddView(&title_tf_); + + layout->AddPaddingRow(0, kRelatedControlVerticalSpacing); + + layout->StartRow(0, labels_column_set_id); + layout->AddView( + new Label(l10n_util::GetString(IDS_BOOMARK_EDITOR_URL_LABEL))); + layout->AddView(&url_tf_); + + layout->AddPaddingRow(0, kRelatedControlVerticalSpacing); + + layout->StartRow(1, single_column_view_set_id); + layout->AddView(&tree_view_); + + layout->AddPaddingRow(0, kRelatedControlVerticalSpacing); + + if (bb_model_->IsLoaded()) + Loaded(bb_model_); +} + +void BookmarkEditorView::Loaded(BookmarkBarModel* model) { + Reset(true); +} + +void BookmarkEditorView::BookmarkNodeMoved(BookmarkBarModel* model, + BookmarkBarNode* old_parent, + int old_index, + BookmarkBarNode* new_parent, + int new_index) { + Reset(false); +} + +void BookmarkEditorView::BookmarkNodeAdded(BookmarkBarModel* model, + BookmarkBarNode* parent, + int index) { + Reset(false); +} + +void BookmarkEditorView::BookmarkNodeRemoved(BookmarkBarModel* model, + BookmarkBarNode* parent, + int index) { + Reset(false); +} + +void BookmarkEditorView::Reset(bool first_time) { + BookmarkBarNode* node_editing = bb_model_->GetNodeByURL(url_); + + // If the title is empty we need to fetch it from the node. + if (first_time && title_.empty()) { + if (node_editing) { + title_ = node_editing->GetTitle(); + title_tf_.SetText(title_); + } + } + + // Do this first, otherwise when we invoke SetModel with the real one + // tree_view will try to invoke something on the model we just deleted. + tree_view_.SetModel(NULL); + + BookmarkNode* root_node = CreateRootNode(); + tree_model_.reset(new BookmarkTreeModel(root_node)); + + tree_view_.SetModel(tree_model_.get()); + tree_view_.SetController(this); + + new_group_button_.SetEnabled(true); + + context_menu_.reset(); + + if (GetParent()) { + ExpandAndSelect(); + + if(!first_time) + UserInputChanged(); + } else if (GetParent()) { + tree_view_.ExpandAll(); + } +} +GURL BookmarkEditorView::GetInputURL() const { + std::wstring input = URLFixerUpper::FixupURL(url_tf_.GetText(), L""); + return GURL(input); +} + +std::wstring BookmarkEditorView::GetInputTitle() const { + return title_tf_.GetText(); +} + +void BookmarkEditorView::UserInputChanged() { + const GURL url(GetInputURL()); + if (!url.is_valid()) + url_tf_.SetBackgroundColor(kErrorColor); + else + url_tf_.SetDefaultBackgroundColor(); + dialog_->UpdateDialogButtons(); +} + +void BookmarkEditorView::NewGroup() { + // Create a new entry parented to the selected item, or the bookmark + // bar if nothing is selected. + BookmarkNode* parent = tree_model_->AsNode(tree_view_.GetSelectedNode()); + if (!parent) { + NOTREACHED(); + return; + } + + BookmarkNode* new_node = new BookmarkNode(); + new_node->SetTitle(l10n_util::GetString(IDS_BOOMARK_EDITOR_NEW_FOLDER_NAME)); + new_node->value = 0; + // new_node is now owned by parent. + tree_model_->Add(parent, parent->GetChildCount(), new_node); + // Edit the new node. + tree_view_.StartEditing(new_node); +} + +void BookmarkEditorView::ExpandAndSelect() { + tree_view_.ExpandAll(); + + BookmarkBarNode* to_select = bb_model_->GetNodeByURL(url_); + history::UIStarID group_id_to_select = + to_select ? to_select->GetParent()->GetGroupID() : + bb_model_->GetParentForNewNodes()->GetGroupID(); + + DCHECK(group_id_to_select); // GetMostRecentParent should never return NULL. + BookmarkNode* b_node = + FindNodeWithID(tree_model_->GetRoot(), group_id_to_select); + if (!b_node) + b_node = tree_model_->GetRoot()->GetChild(0); // Bookmark bar node. + + tree_view_.SetSelectedNode(b_node); +} + +BookmarkEditorView::BookmarkNode* BookmarkEditorView::CreateRootNode() { + BookmarkNode* root_node = new BookmarkNode(std::wstring(), 0); + BookmarkBarNode* bb_root_node = bb_model_->root_node(); + CreateNodes(bb_root_node, root_node); + DCHECK(root_node->GetChildCount() == 2); + DCHECK(bb_root_node->GetChild(0)->GetType() == + history::StarredEntry::BOOKMARK_BAR); + DCHECK(bb_root_node->GetChild(1)->GetType() == history::StarredEntry::OTHER); + return root_node; +} + +void BookmarkEditorView::CreateNodes(BookmarkBarNode* bb_node, + BookmarkEditorView::BookmarkNode* b_node) { + for (int i = 0; i < bb_node->GetChildCount(); ++i) { + BookmarkBarNode* child_bb_node = bb_node->GetChild(i); + if (child_bb_node->GetType() != history::StarredEntry::URL) { + BookmarkNode* new_b_node = new BookmarkNode(child_bb_node->GetTitle(), + child_bb_node->GetGroupID()); + b_node->Add(b_node->GetChildCount(), new_b_node); + CreateNodes(child_bb_node, new_b_node); + } + } +} + +BookmarkEditorView::BookmarkNode* BookmarkEditorView::FindNodeWithID( + BookmarkEditorView::BookmarkNode* node, + history::UIStarID id) { + if (node->value == id) + return node; + for (int i = 0; i < node->GetChildCount(); ++i) { + BookmarkNode* result = FindNodeWithID(node->GetChild(i), id); + if (result) + return result; + } + return NULL; +} + +void BookmarkEditorView::ApplyEdits() { + DCHECK(bb_model_->IsLoaded()); + + if (!tree_view_.GetSelectedNode()) { + NOTREACHED(); + return; + } + + // We're going to apply edits to the bookmark bar model, which will call us + // back. Normally when a structural edit occurs we reset the tree model. + // We don't want to do that here, so we remove ourselves as an observer. + bb_model_->RemoveObserver(this); + + GURL new_url(GetInputURL()); + std::wstring new_title(GetInputTitle()); + + BookmarkBarNode* old_node = bb_model_->GetNodeByURL(url_); + BookmarkBarNode* old_parent = old_node ? old_node->GetParent() : NULL; + const int old_index = old_parent ? old_parent->IndexOfChild(old_node) : -1; + + if (url_ != new_url) { + // The URL has changed, unstar the old url. + bb_model_->SetURLStarred(url_, std::wstring(), false); + } + + // Create the new groups and update the titles. + BookmarkBarNode* new_parent = NULL; + ApplyNameChangesAndCreateNewGroups( + bb_model_->root_node(), tree_model_->GetRoot(), + tree_model_->AsNode(tree_view_.GetSelectedNode()), &new_parent); + + if (!new_parent) { + // Bookmarks must be parented. + NOTREACHED(); + return; + } + + BookmarkBarNode* current_node = bb_model_->GetNodeByURL(new_url); + + if (current_node) { + // There's already a node with the URL. + bb_model_->SetTitle(current_node, new_title); + if (new_parent == old_parent) { + // Parent hasn't changed. + bb_model_->Move(current_node, new_parent, old_index); + } else { + // Parent changed, move to end of new parent. + bb_model_->Move(current_node, new_parent, new_parent->GetChildCount()); + } + } else { + // Adding a new URL. + if (new_parent == old_parent) { + // Parent hasn't changed. Place newly created bookmark at the same + // location as last bookmark. + bb_model_->AddURL(new_parent, old_index, new_title, new_url); + } else { + // Parent changed, put bookmark at end of new parent. + bb_model_->AddURL(new_parent, new_parent->GetChildCount(), new_title, + new_url); + } + } +} + +void BookmarkEditorView::ApplyNameChangesAndCreateNewGroups( + BookmarkBarNode* bb_node, + BookmarkEditorView::BookmarkNode* b_node, + BookmarkEditorView::BookmarkNode* parent_b_node, + BookmarkBarNode** parent_bb_node) { + if (parent_b_node == b_node) + *parent_bb_node = bb_node; + for (int i = 0; i < b_node->GetChildCount(); ++i) { + BookmarkNode* child_b_node = b_node->GetChild(i); + BookmarkBarNode* child_bb_node = NULL; + if (child_b_node->value == 0) { + // New group. + child_bb_node = bb_model_->AddGroup(bb_node, + bb_node->GetChildCount(), child_b_node->GetTitle()); + } else { + // Existing node, reset the title (BBModel ignores changes if the title + // is the same). + for (int j = 0; j < bb_node->GetChildCount(); ++j) { + BookmarkBarNode* node = bb_node->GetChild(j); + if (node->GetType() != history::StarredEntry::URL && + node->GetGroupID() == child_b_node->value) { + child_bb_node = node; + break; + } + } + DCHECK(child_bb_node); + bb_model_->SetTitle(child_bb_node, child_b_node->GetTitle()); + } + ApplyNameChangesAndCreateNewGroups(child_bb_node, child_b_node, + parent_b_node, parent_bb_node); + } +} |