Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(3956)

Unified Diff: chrome/browser/ui/views/website_settings/chooser_bubble_ui.cc

Issue 1408193003: Add chrome side webusb permission UI code (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: removed mac sources from chrome_browser_ui.gypi Created 5 years, 2 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: chrome/browser/ui/views/website_settings/chooser_bubble_ui.cc
diff --git a/chrome/browser/ui/views/website_settings/chooser_bubble_ui.cc b/chrome/browser/ui/views/website_settings/chooser_bubble_ui.cc
new file mode 100644
index 0000000000000000000000000000000000000000..25f604287a13b31ddfb3da506ce54ff9bd5c32f8
--- /dev/null
+++ b/chrome/browser/ui/views/website_settings/chooser_bubble_ui.cc
@@ -0,0 +1,346 @@
+// Copyright 2015 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/ui/views/website_settings/chooser_bubble_ui.h"
+
+#include <string>
+
+#include "base/prefs/pref_service.h"
+#include "base/strings/string16.h"
+#include "base/strings/utf_string_conversions.h"
+#include "chrome/browser/profiles/profile.h"
+#include "chrome/browser/ui/browser.h"
+#include "chrome/browser/ui/views/exclusive_access_bubble_views.h"
+#include "chrome/browser/ui/views/frame/browser_view.h"
+#include "chrome/browser/ui/views/frame/top_container_view.h"
+#include "chrome/browser/ui/views/location_bar/location_bar_view.h"
+#include "chrome/browser/ui/views/location_bar/location_icon_view.h"
+#include "chrome/browser/usb/web_usb_permission_bubble_request.h"
+#include "chrome/common/pref_names.h"
+#include "chrome/grit/generated_resources.h"
+#include "ui/accessibility/ax_view_state.h"
+#include "ui/base/l10n/l10n_util.h"
+#include "ui/base/resource/resource_bundle.h"
+#include "ui/gfx/paint_vector_icon.h"
+#include "ui/gfx/text_constants.h"
+#include "ui/gfx/vector_icons_public.h"
+#include "ui/views/bubble/bubble_delegate.h"
+#include "ui/views/bubble/bubble_frame_view.h"
+#include "ui/views/controls/button/label_button.h"
+#include "ui/views/controls/button/label_button_border.h"
+#include "ui/views/controls/label.h"
+#include "ui/views/controls/table/table_view.h"
+#include "ui/views/controls/table/table_view_observer.h"
+#include "ui/views/layout/box_layout.h"
+#include "ui/views/layout/grid_layout.h"
+
+namespace {
+
+// chooser permission bubble width
+const int kChooserPermissionBubbleWidth = 300;
+
+// chooser permission bubble height
+const int kChooserPermissionBubbleHeight = 200;
+
+// text label height when no devices found
+const int kLabelHeight = 30;
+
+// Spacing constant for outer margin. This is added to the
+// bubble margin itself to equalize the margins at 13px.
+const int kBubbleOuterMargin = 5;
+
+// Spacing between major items should be 9px.
+const int kItemMajorSpacing = 9;
+
+// Button border size, draws inside the spacing distance.
+const int kButtonBorderSize = 2;
+
+} // namespace
+
+///////////////////////////////////////////////////////////////////////////////
+// View implementation for the chooser bubble.
+class ChooserBubbleUiDelegate : public views::BubbleDelegateView,
+ public views::ButtonListener,
+ public views::TableViewObserver {
+ public:
+ ChooserBubbleUiDelegate(views::View* anchor_view,
+ views::BubbleBorder::Arrow anchor_arrow,
+ ChooserBubbleUi* owner,
+ const WebUsbPermissionBubbleRequest* request);
+ ~ChooserBubbleUiDelegate() override;
+
+ void Close();
+
+ // BubbleDelegateView:
+ bool ShouldShowCloseButton() const override;
+ bool ShouldShowWindowTitle() const override;
+ base::string16 GetWindowTitle() const override;
+ void OnWidgetDestroying(views::Widget* widget) override;
+
+ // ButtonListener:
+ void ButtonPressed(views::Button* button, const ui::Event& event) override;
+
+ // views::TableViewObserver:
+ void OnSelectionChanged() override;
+
+ // Updates the anchor's arrow and view. Also repositions the bubble so it's
+ // displayed in the correct location.
+ void UpdateAnchor(views::View* anchor_view,
+ views::BubbleBorder::Arrow anchor_arrow);
+
+ private:
+ ChooserBubbleUi* owner_;
+ views::Button* connect_;
+ views::Button* cancel_;
+ views::TableView* table_view_;
+
+ DISALLOW_COPY_AND_ASSIGN(ChooserBubbleUiDelegate);
+};
+
+ui::TableColumn ChooserTableColumn(int id, const std::string& title) {
+ ui::TableColumn column;
+ column.id = id;
+ column.title = base::ASCIIToUTF16(title.c_str());
+ return column;
+}
+
+class ChooserTableModel : public ui::TableModel {
+ public:
+ explicit ChooserTableModel(const std::vector<std::string>& device_names)
+ : observer_(nullptr) {
+ device_names_ = device_names;
+ row_count_ = static_cast<int>(device_names_.size());
+ }
+
+ // ui::TableModel:
+ int RowCount() override { return row_count_; }
+
+ base::string16 GetText(int row, int column_id) override {
+ if (row >= 0 && row < row_count_) {
+ return base::ASCIIToUTF16(device_names_[row]);
+ } else {
+ return base::string16();
+ }
+ }
+
+ void SetObserver(ui::TableModelObserver* observer) override {
+ observer_ = observer;
+ }
+
+ private:
+ ui::TableModelObserver* observer_;
+ std::vector<std::string> device_names_;
+ int row_count_;
+};
+
+ChooserBubbleUiDelegate::ChooserBubbleUiDelegate(
+ views::View* anchor_view,
+ views::BubbleBorder::Arrow anchor_arrow,
+ ChooserBubbleUi* owner,
+ const WebUsbPermissionBubbleRequest* request)
+ : views::BubbleDelegateView(anchor_view, anchor_arrow),
+ owner_(owner),
+ connect_(nullptr),
+ cancel_(nullptr) {
+ views::GridLayout* layout = new views::GridLayout(this);
+ SetLayoutManager(layout);
+
+ views::ColumnSet* column_set = layout->AddColumnSet(0);
+ column_set->AddColumn(views::GridLayout::FILL, views::GridLayout::FILL, 1,
+ views::GridLayout::USE_PREF, 0, 0);
+
+ layout->StartRow(1, 0);
+
+ const std::vector<std::string>& device_names = request->device_names();
+ if (!device_names.empty()) {
+ // create a table view and list the device names into it.
+ std::vector<ui::TableColumn> table_columns;
+ table_columns.push_back(ChooserTableColumn(
+ 0, "" /* empty string makes the column title invisible */));
+ table_view_ = new views::TableView(new ChooserTableModel(device_names),
+ table_columns, views::TEXT_ONLY, true);
+ table_view_->SetObserver(this);
+ layout->AddView(table_view_->CreateParentIfNecessary(), 1, 1,
+ views::GridLayout::FILL, views::GridLayout::FILL,
+ kChooserPermissionBubbleWidth,
+ kChooserPermissionBubbleHeight);
+ } else {
+ // show a text label saying no devices found.
+ views::Label* label = new views::Label(l10n_util::GetStringUTF16(
+ IDS_WEBUSB_PERMISSIONS_BUBBLE_NO_DEVICES_FOUND_PROMPT));
+ label->SetHorizontalAlignment(gfx::ALIGN_CENTER);
+ layout->AddView(label, 1, 1, views::GridLayout::FILL,
+ views::GridLayout::FILL, kChooserPermissionBubbleWidth,
+ kLabelHeight);
+ }
+
+ layout->AddPaddingRow(0, kItemMajorSpacing);
+
+ views::View* button_row = new views::View();
+ views::GridLayout* button_layout = new views::GridLayout(button_row);
+ views::ColumnSet* button_columns = button_layout->AddColumnSet(0);
+ button_row->SetLayoutManager(button_layout);
+ layout->StartRow(1, 0);
+ layout->AddView(button_row);
+
+ // lay out the Connect/Cancel buttons.
+ button_columns->AddColumn(views::GridLayout::TRAILING,
+ views::GridLayout::FILL, 100,
+ views::GridLayout::USE_PREF, 0, 0);
+ button_columns->AddPaddingColumn(0,
+ kItemMajorSpacing - (2 * kButtonBorderSize));
+ button_columns->AddColumn(views::GridLayout::TRAILING,
+ views::GridLayout::FILL, 0,
+ views::GridLayout::USE_PREF, 0, 0);
+ button_layout->StartRow(0, 0);
+
+ base::string16 connect_text = base::ASCIIToUTF16("Connect");
+ views::LabelButton* connect_button =
+ new views::LabelButton(this, connect_text);
+ connect_button->SetStyle(views::Button::STYLE_BUTTON);
+ // disable the connect button at the beginning since no device selected yet.
+ connect_button->SetEnabled(false);
+ button_layout->AddView(connect_button);
+ connect_ = connect_button;
+
+ base::string16 cancel_text = base::ASCIIToUTF16("Cancel");
+ views::LabelButton* cancel_button = new views::LabelButton(this, cancel_text);
+ cancel_button->SetStyle(views::Button::STYLE_BUTTON);
+ button_layout->AddView(cancel_button);
+ cancel_ = cancel_button;
+
+ button_layout->AddPaddingRow(0, kBubbleOuterMargin);
+}
+
+ChooserBubbleUiDelegate::~ChooserBubbleUiDelegate() {
+ RemoveAllChildViews(true);
+ if (owner_)
+ owner_->Close();
+}
+
+void ChooserBubbleUiDelegate::Close() {
+ owner_ = nullptr;
+ GetWidget()->Close();
+}
+
+bool ChooserBubbleUiDelegate::ShouldShowCloseButton() const {
+ return true;
+}
+
+bool ChooserBubbleUiDelegate::ShouldShowWindowTitle() const {
+ return true;
+}
+
+base::string16 ChooserBubbleUiDelegate::GetWindowTitle() const {
+ return l10n_util::GetStringUTF16(IDS_WEBUSB_PERMISSIONS_BUBBLE_PROMPT);
+}
+
+void ChooserBubbleUiDelegate::OnWidgetDestroying(views::Widget* widget) {
+ views::BubbleDelegateView::OnWidgetDestroying(widget);
+ if (owner_) {
+ owner_->Close();
+ owner_ = nullptr;
+ }
+}
+
+void ChooserBubbleUiDelegate::ButtonPressed(views::Button* button,
+ const ui::Event& event) {
+ if (!owner_)
+ return;
+
+ if (button == connect_)
+ owner_->Connect(table_view_->selection_model().active());
+ else if (button == cancel_)
+ owner_->Cancel();
+}
+
+void ChooserBubbleUiDelegate::OnSelectionChanged() {
+ // enable the connect button since user has selected an item.
+ connect_->SetEnabled(true);
+}
+
+void ChooserBubbleUiDelegate::UpdateAnchor(
+ views::View* anchor_view,
+ views::BubbleBorder::Arrow anchor_arrow) {
+ if (GetAnchorView() == anchor_view && arrow() == anchor_arrow)
+ return;
+
+ set_arrow(anchor_arrow);
+
+ // Update the border in the bubble: will either add or remove the arrow.
+ views::BubbleFrameView* frame =
+ views::BubbleDelegateView::GetBubbleFrameView();
+ views::BubbleBorder::Arrow adjusted_arrow = anchor_arrow;
+ if (base::i18n::IsRTL())
+ adjusted_arrow = views::BubbleBorder::horizontal_mirror(adjusted_arrow);
+ frame->SetBubbleBorder(scoped_ptr<views::BubbleBorder>(
+ new views::BubbleBorder(adjusted_arrow, shadow(), color())));
+
+ // Reposition the bubble based on the updated arrow and view.
+ SetAnchorView(anchor_view);
+}
+
+//////////////////////////////////////////////////////////////////////////////
+// ChooserBubbleUi
+
+ChooserBubbleUi::ChooserBubbleUi(Browser* browser,
+ WebUsbPermissionBubbleRequest* request)
+ : browser_(browser), request_(request), bubble_delegate_(nullptr) {
+ DCHECK(browser);
+}
+
+ChooserBubbleUi::~ChooserBubbleUi() {}
+
+void ChooserBubbleUi::Show(BubbleReference bubble_reference) {
+ if (bubble_delegate_)
+ bubble_delegate_->Close();
+ bubble_delegate_ = new ChooserBubbleUiDelegate(
+ GetAnchorView(), GetAnchorArrow(), this, request_);
+
+ // Set |parent_window| because some valid anchors can become hidden.
+ views::Widget* widget = views::Widget::GetWidgetForNativeWindow(
+ browser_->window()->GetNativeWindow());
+ bubble_delegate_->set_parent_window(widget->GetNativeView());
+
+ views::BubbleDelegateView::CreateBubble(bubble_delegate_)->Show();
+}
+
+void ChooserBubbleUi::Close() {
+ if (bubble_delegate_) {
+ bubble_delegate_->Close();
+ bubble_delegate_ = nullptr;
+ }
+}
+
+void ChooserBubbleUi::UpdateAnchorPosition() {
+ bubble_delegate_->UpdateAnchor(GetAnchorView(), GetAnchorArrow());
+}
+
+void ChooserBubbleUi::Connect(int index) {
+ request_->Connect(index);
+ Close();
+}
+
+void ChooserBubbleUi::Cancel() {
+ request_->Cancel();
+ Close();
+}
+
+views::View* ChooserBubbleUi::GetAnchorView() {
+ BrowserView* browser_view = BrowserView::GetBrowserViewForBrowser(browser_);
+
+ if (browser_->SupportsWindowFeature(Browser::FEATURE_LOCATIONBAR))
+ return browser_view->GetLocationBarView()->location_icon_view();
+
+ if (browser_view->IsFullscreenBubbleVisible())
+ return browser_view->exclusive_access_bubble()->GetView();
+
+ return browser_view->top_container();
+}
+
+views::BubbleBorder::Arrow ChooserBubbleUi::GetAnchorArrow() {
+ if (browser_->SupportsWindowFeature(Browser::FEATURE_LOCATIONBAR))
+ return views::BubbleBorder::TOP_LEFT;
+ return views::BubbleBorder::NONE;
+}

Powered by Google App Engine
This is Rietveld 408576698