| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  | // Copyright (c) 2012 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.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-19 21:52:18 +02:00
										 |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <utility>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-26 11:13:05 +02:00
										 |  |  | #if defined(ENABLE_OSR)
 | 
					
						
							| 
									
										
										
										
											2017-05-21 19:55:19 +02:00
										 |  |  | #include "atom/browser/osr/osr_render_widget_host_view.h"
 | 
					
						
							|  |  |  | #include "atom/browser/osr/osr_view_proxy.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-26 11:13:05 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  | #include "atom/browser/ui/autofill_popup.h"
 | 
					
						
							|  |  |  | #include "atom/common/api/api_messages.h"
 | 
					
						
							|  |  |  | #include "ui/display/display.h"
 | 
					
						
							|  |  |  | #include "ui/display/screen.h"
 | 
					
						
							|  |  |  | #include "ui/gfx/geometry/point.h"
 | 
					
						
							|  |  |  | #include "ui/gfx/geometry/rect.h"
 | 
					
						
							|  |  |  | #include "ui/gfx/geometry/vector2d.h"
 | 
					
						
							| 
									
										
										
										
											2017-05-19 21:52:18 +02:00
										 |  |  | #include "ui/gfx/text_utils.h"
 | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace atom { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::pair<int, int> CalculatePopupXAndWidth( | 
					
						
							|  |  |  |     const display::Display& left_display, | 
					
						
							|  |  |  |     const display::Display& right_display, | 
					
						
							|  |  |  |     int popup_required_width, | 
					
						
							|  |  |  |     const gfx::Rect& element_bounds, | 
					
						
							|  |  |  |     bool is_rtl) { | 
					
						
							|  |  |  |   int leftmost_display_x = left_display.bounds().x(); | 
					
						
							|  |  |  |   int rightmost_display_x = | 
					
						
							|  |  |  |       right_display.GetSizeInPixel().width() + right_display.bounds().x(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Calculate the start coordinates for the popup if it is growing right or
 | 
					
						
							|  |  |  |   // the end position if it is growing to the left, capped to screen space.
 | 
					
						
							|  |  |  |   int right_growth_start = std::max( | 
					
						
							|  |  |  |       leftmost_display_x, std::min(rightmost_display_x, element_bounds.x())); | 
					
						
							|  |  |  |   int left_growth_end = | 
					
						
							|  |  |  |       std::max(leftmost_display_x, | 
					
						
							|  |  |  |                std::min(rightmost_display_x, element_bounds.right())); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int right_available = rightmost_display_x - right_growth_start; | 
					
						
							|  |  |  |   int left_available = left_growth_end - leftmost_display_x; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int popup_width = | 
					
						
							|  |  |  |       std::min(popup_required_width, std::max(right_available, left_available)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   std::pair<int, int> grow_right(right_growth_start, popup_width); | 
					
						
							|  |  |  |   std::pair<int, int> grow_left(left_growth_end - popup_width, popup_width); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Prefer to grow towards the end (right for LTR, left for RTL). But if there
 | 
					
						
							|  |  |  |   // is not enough space available in the desired direction and more space in
 | 
					
						
							|  |  |  |   // the other direction, reverse it.
 | 
					
						
							|  |  |  |   if (is_rtl) { | 
					
						
							|  |  |  |     return left_available >= popup_width || left_available >= right_available | 
					
						
							|  |  |  |                ? grow_left | 
					
						
							|  |  |  |                : grow_right; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return right_available >= popup_width || right_available >= left_available | 
					
						
							|  |  |  |              ? grow_right | 
					
						
							|  |  |  |              : grow_left; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::pair<int, int> CalculatePopupYAndHeight( | 
					
						
							|  |  |  |     const display::Display& top_display, | 
					
						
							|  |  |  |     const display::Display& bottom_display, | 
					
						
							|  |  |  |     int popup_required_height, | 
					
						
							|  |  |  |     const gfx::Rect& element_bounds) { | 
					
						
							|  |  |  |   int topmost_display_y = top_display.bounds().y(); | 
					
						
							|  |  |  |   int bottommost_display_y = | 
					
						
							|  |  |  |       bottom_display.GetSizeInPixel().height() + bottom_display.bounds().y(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Calculate the start coordinates for the popup if it is growing down or
 | 
					
						
							|  |  |  |   // the end position if it is growing up, capped to screen space.
 | 
					
						
							|  |  |  |   int top_growth_end = std::max( | 
					
						
							|  |  |  |       topmost_display_y, std::min(bottommost_display_y, element_bounds.y())); | 
					
						
							|  |  |  |   int bottom_growth_start = | 
					
						
							|  |  |  |       std::max(topmost_display_y, | 
					
						
							|  |  |  |                std::min(bottommost_display_y, element_bounds.bottom())); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int top_available = bottom_growth_start - topmost_display_y; | 
					
						
							|  |  |  |   int bottom_available = bottommost_display_y - top_growth_end; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // TODO(csharp): Restrict the popup height to what is available.
 | 
					
						
							|  |  |  |   if (bottom_available >= popup_required_height || | 
					
						
							|  |  |  |       bottom_available >= top_available) { | 
					
						
							|  |  |  |     // The popup can appear below the field.
 | 
					
						
							|  |  |  |     return std::make_pair(bottom_growth_start, popup_required_height); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     // The popup must appear above the field.
 | 
					
						
							|  |  |  |     return std::make_pair(top_growth_end - popup_required_height, | 
					
						
							|  |  |  |                           popup_required_height); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | display::Display GetDisplayNearestPoint( | 
					
						
							|  |  |  |     const gfx::Point& point, | 
					
						
							|  |  |  |     gfx::NativeView container_view) { | 
					
						
							|  |  |  |   return display::Screen::GetScreen()->GetDisplayNearestPoint(point); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AutofillPopup::AutofillPopup(gfx::NativeView container_view) | 
					
						
							| 
									
										
										
										
											2017-05-21 19:55:19 +02:00
										 |  |  |     : container_view_(container_view), view_(nullptr) { | 
					
						
							| 
									
										
										
										
											2017-05-19 21:52:18 +02:00
										 |  |  |   bold_font_list_ = | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  |     gfx::FontList().DeriveWithWeight(gfx::Font::Weight::BOLD); | 
					
						
							| 
									
										
										
										
											2017-05-19 21:52:18 +02:00
										 |  |  |   smaller_font_list_ = | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  |     gfx::FontList().DeriveWithSizeDelta(kSmallerFontSizeDelta); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AutofillPopup::~AutofillPopup() { | 
					
						
							|  |  |  |   Hide(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AutofillPopup::CreateView( | 
					
						
							| 
									
										
										
										
											2017-05-26 03:38:27 +02:00
										 |  |  |     content::RenderFrameHost* frame_host, | 
					
						
							| 
									
										
										
										
											2017-05-21 19:55:19 +02:00
										 |  |  |     bool offscreen, | 
					
						
							| 
									
										
										
										
											2017-05-19 21:52:18 +02:00
										 |  |  |     views::Widget* parent_widget, | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  |     const gfx::RectF& r) { | 
					
						
							| 
									
										
										
										
											2017-05-26 03:38:27 +02:00
										 |  |  |   frame_host_ = frame_host; | 
					
						
							| 
									
										
										
										
											2017-05-19 21:52:18 +02:00
										 |  |  |   gfx::Rect lb(std::floor(r.x()), std::floor(r.y() + r.height()), | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  |                std::floor(r.width()), std::floor(r.height())); | 
					
						
							|  |  |  |   gfx::Point menu_position(lb.origin()); | 
					
						
							| 
									
										
										
										
											2017-05-21 19:55:19 +02:00
										 |  |  |   popup_bounds_in_view_ = lb; | 
					
						
							| 
									
										
										
										
											2017-05-19 21:52:18 +02:00
										 |  |  |   views::View::ConvertPointToScreen( | 
					
						
							|  |  |  |     parent_widget->GetContentsView(), &menu_position); | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  |   popup_bounds_ = gfx::Rect(menu_position, lb.size()); | 
					
						
							|  |  |  |   element_bounds_ = popup_bounds_; | 
					
						
							| 
									
										
										
										
											2017-05-19 21:52:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-21 19:55:19 +02:00
										 |  |  |   Hide(); | 
					
						
							|  |  |  |   view_ = new AutofillPopupView(this, parent_widget); | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  |   view_->Show(); | 
					
						
							| 
									
										
										
										
											2017-05-19 21:52:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-26 11:13:05 +02:00
										 |  |  | #if defined(ENABLE_OSR)
 | 
					
						
							| 
									
										
										
										
											2017-05-21 19:55:19 +02:00
										 |  |  |   if (offscreen) { | 
					
						
							|  |  |  |     auto* osr_rwhv = static_cast<OffScreenRenderWidgetHostView*>( | 
					
						
							| 
									
										
										
										
											2017-05-26 03:38:27 +02:00
										 |  |  |         frame_host_->GetView()); | 
					
						
							| 
									
										
										
										
											2017-05-21 19:55:19 +02:00
										 |  |  |     view_->view_proxy_.reset(new OffscreenViewProxy(view_)); | 
					
						
							|  |  |  |     osr_rwhv->AddViewProxy(view_->view_proxy_.get()); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-06-26 11:13:05 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AutofillPopup::Hide() { | 
					
						
							| 
									
										
										
										
											2017-05-21 19:55:19 +02:00
										 |  |  |   if (view_) { | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  |     view_->Hide(); | 
					
						
							| 
									
										
										
										
											2017-05-21 19:55:19 +02:00
										 |  |  |     view_ = nullptr; | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AutofillPopup::SetItems(const std::vector<base::string16>& values, | 
					
						
							|  |  |  |                             const std::vector<base::string16>& labels) { | 
					
						
							|  |  |  |   values_ = values; | 
					
						
							|  |  |  |   labels_ = labels; | 
					
						
							| 
									
										
										
										
											2017-05-21 19:55:19 +02:00
										 |  |  |   if (view_) { | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  |     view_->OnSuggestionsChanged(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AutofillPopup::AcceptSuggestion(int index) { | 
					
						
							| 
									
										
										
										
											2017-05-26 03:38:27 +02:00
										 |  |  |   frame_host_->Send(new AtomAutofillFrameMsg_AcceptSuggestion( | 
					
						
							|  |  |  |     frame_host_->GetRoutingID(), GetValueAt(index))); | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-03 00:25:59 +01:00
										 |  |  | void AutofillPopup::UpdatePopupBounds(int height_compensation) { | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  |   int desired_width = GetDesiredPopupWidth(); | 
					
						
							|  |  |  |   int desired_height = GetDesiredPopupHeight(); | 
					
						
							|  |  |  |   bool is_rtl = false; | 
					
						
							| 
									
										
										
										
											2017-05-19 21:52:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-03 00:42:30 +01:00
										 |  |  |   gfx::Point origin(element_bounds_.origin().x(), | 
					
						
							| 
									
										
										
										
											2017-11-13 16:13:54 +09:00
										 |  |  |                     element_bounds_.origin().y() - height_compensation); | 
					
						
							| 
									
										
										
										
											2017-11-03 00:25:59 +01:00
										 |  |  |   gfx::Rect bounds(origin, element_bounds_.size()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-13 16:13:54 +09:00
										 |  |  |   gfx::Point top_left_corner_of_popup = | 
					
						
							|  |  |  |       origin + gfx::Vector2d(bounds.width() - desired_width, -desired_height); | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // This is the bottom right point of the popup if the popup is below the
 | 
					
						
							|  |  |  |   // element and grows to the right (since the is the lowest and furthest right
 | 
					
						
							|  |  |  |   // the popup could go).
 | 
					
						
							| 
									
										
										
										
											2017-11-13 16:13:54 +09:00
										 |  |  |   gfx::Point bottom_right_corner_of_popup = | 
					
						
							|  |  |  |       origin + gfx::Vector2d(desired_width, bounds.height() + desired_height); | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   display::Display top_left_display = | 
					
						
							| 
									
										
										
										
											2017-11-13 16:13:54 +09:00
										 |  |  |       GetDisplayNearestPoint(top_left_corner_of_popup, container_view_); | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  |   display::Display bottom_right_display = | 
					
						
							| 
									
										
										
										
											2017-11-13 16:13:54 +09:00
										 |  |  |       GetDisplayNearestPoint(bottom_right_corner_of_popup, container_view_); | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   std::pair<int, int> popup_x_and_width = | 
					
						
							| 
									
										
										
										
											2017-11-13 16:13:54 +09:00
										 |  |  |       CalculatePopupXAndWidth(top_left_display, bottom_right_display, | 
					
						
							|  |  |  |                               desired_width, bounds, is_rtl); | 
					
						
							|  |  |  |   std::pair<int, int> popup_y_and_height = | 
					
						
							|  |  |  |       CalculatePopupYAndHeight(top_left_display, bottom_right_display, | 
					
						
							|  |  |  |                                desired_height, bounds); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   popup_bounds_ = gfx::Rect( | 
					
						
							|  |  |  |       popup_x_and_width.first, popup_y_and_height.first, | 
					
						
							|  |  |  |       popup_x_and_width.second, popup_y_and_height.second); | 
					
						
							|  |  |  |   popup_bounds_in_view_ = gfx::Rect( | 
					
						
							|  |  |  |       popup_bounds_in_view_.origin(), | 
					
						
							|  |  |  |       gfx::Size(popup_x_and_width.second, popup_y_and_height.second)); | 
					
						
							| 
									
										
										
										
											2017-11-03 00:25:59 +01:00
										 |  |  |   if (view_) | 
					
						
							|  |  |  |     view_->DoUpdateBoundsAndRedrawPopup(); | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int AutofillPopup::GetDesiredPopupHeight() { | 
					
						
							|  |  |  |   return 2 * kPopupBorderThickness + values_.size() * kRowHeight; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int AutofillPopup::GetDesiredPopupWidth() { | 
					
						
							|  |  |  |   int popup_width = element_bounds_.width(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-23 11:41:59 +02:00
										 |  |  |   for (size_t i = 0; i < values_.size(); ++i) { | 
					
						
							| 
									
										
										
										
											2017-05-19 21:52:18 +02:00
										 |  |  |     int row_size = kEndPadding + 2 * kPopupBorderThickness + | 
					
						
							|  |  |  |       gfx::GetStringWidth(GetValueAt(i), GetValueFontListForRow(i)) + | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  |       gfx::GetStringWidth(GetLabelAt(i), GetLabelFontListForRow(i)); | 
					
						
							| 
									
										
										
										
											2017-05-19 21:52:18 +02:00
										 |  |  |     if (GetLabelAt(i).length() > 0) | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  |       row_size += kNamePadding + kEndPadding; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     popup_width = std::max(popup_width, row_size); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return popup_width; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | gfx::Rect AutofillPopup::GetRowBounds(int index) { | 
					
						
							|  |  |  |   int top = kPopupBorderThickness + index * kRowHeight; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return gfx::Rect(kPopupBorderThickness, top, | 
					
						
							|  |  |  |                    popup_bounds_.width() - 2 * kPopupBorderThickness, | 
					
						
							|  |  |  |                    kRowHeight); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const gfx::FontList& AutofillPopup::GetValueFontListForRow(int index) const { | 
					
						
							|  |  |  |   return bold_font_list_; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const gfx::FontList& AutofillPopup::GetLabelFontListForRow(int index) const { | 
					
						
							|  |  |  |   return smaller_font_list_; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ui::NativeTheme::ColorId AutofillPopup::GetBackgroundColorIDForRow( | 
					
						
							|  |  |  |     int index) const { | 
					
						
							| 
									
										
										
										
											2017-05-21 19:55:19 +02:00
										 |  |  |   return (view_ && index == view_->GetSelectedLine()) | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  |       ? ui::NativeTheme::kColorId_ResultsTableHoveredBackground | 
					
						
							|  |  |  |       : ui::NativeTheme::kColorId_ResultsTableNormalBackground; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int AutofillPopup::GetLineCount() { | 
					
						
							|  |  |  |   return values_.size(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | base::string16 AutofillPopup::GetValueAt(int i) { | 
					
						
							|  |  |  |   return values_.at(i); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | base::string16 AutofillPopup::GetLabelAt(int i) { | 
					
						
							|  |  |  |   return labels_.at(i); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int AutofillPopup::LineFromY(int y) const { | 
					
						
							|  |  |  |   int current_height = kPopupBorderThickness; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-23 11:41:59 +02:00
										 |  |  |   for (size_t i = 0; i < values_.size(); ++i) { | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  |     current_height += kRowHeight; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (y <= current_height) | 
					
						
							|  |  |  |       return i; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-05-19 21:52:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-19 21:35:13 +02:00
										 |  |  |   return values_.size() - 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-19 21:52:18 +02:00
										 |  |  | }  // namespace atom
 |