 f595443a22
			
		
	
	
	
	
	f595443a22* refactor: avoid repeating the return type from the declaration; use a braced initializer list instead [modernize-return-braced-init-list] * refactor: avoid repeating the return type from the declaration; use a braced initializer list instead [modernize-return-braced-init-list] NB: using the braced-initializer list uncovered an error here: the float returned by std::floor() can't be implicitly cast to an int. This is solved by using base::ClampFloor<int>() instead. std::floor()
		
			
				
	
	
		
			314 lines
		
	
	
	
		
			11 KiB
			
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			314 lines
		
	
	
	
		
			11 KiB
			
		
	
	
	
		
			C++
		
	
	
	
	
	
| // 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.
 | |
| 
 | |
| #include <algorithm>
 | |
| #include <memory>
 | |
| #include <vector>
 | |
| 
 | |
| #include "base/feature_list.h"
 | |
| #include "base/i18n/rtl.h"
 | |
| #include "components/autofill/core/common/autofill_features.h"
 | |
| #include "content/public/browser/render_frame_host.h"
 | |
| #include "electron/buildflags/buildflags.h"
 | |
| #include "mojo/public/cpp/bindings/associated_remote.h"
 | |
| #include "shell/browser/osr/osr_render_widget_host_view.h"
 | |
| #include "shell/browser/osr/osr_view_proxy.h"
 | |
| #include "shell/browser/ui/autofill_popup.h"
 | |
| #include "shell/browser/ui/views/autofill_popup_view.h"
 | |
| #include "shell/common/api/api.mojom.h"
 | |
| #include "third_party/blink/public/common/associated_interfaces/associated_interface_provider.h"
 | |
| #include "ui/color/color_id.h"
 | |
| #include "ui/color/color_provider.h"
 | |
| #include "ui/display/screen.h"
 | |
| #include "ui/gfx/geometry/point.h"
 | |
| #include "ui/gfx/geometry/rect.h"
 | |
| #include "ui/gfx/geometry/rect_conversions.h"
 | |
| #include "ui/gfx/geometry/vector2d.h"
 | |
| #include "ui/gfx/text_utils.h"
 | |
| 
 | |
| namespace electron {
 | |
| 
 | |
| namespace {
 | |
| 
 | |
| void CalculatePopupXAndWidthHorizontallyCentered(
 | |
|     int popup_preferred_width,
 | |
|     const gfx::Rect& content_area_bounds,
 | |
|     const gfx::Rect& element_bounds,
 | |
|     bool is_rtl,
 | |
|     gfx::Rect* popup_bounds) {
 | |
|   // The preferred horizontal starting point for the pop-up is at the horizontal
 | |
|   // center of the field.
 | |
|   int preferred_starting_point =
 | |
|       std::clamp(element_bounds.x() + (element_bounds.size().width() / 2),
 | |
|                  content_area_bounds.x(), content_area_bounds.right());
 | |
| 
 | |
|   // The space available to the left and to the right.
 | |
|   int space_to_right = content_area_bounds.right() - preferred_starting_point;
 | |
|   int space_to_left = preferred_starting_point - content_area_bounds.x();
 | |
| 
 | |
|   // Calculate the pop-up width. This is either the preferred pop-up width, or
 | |
|   // alternatively the maximum space available if there is not sufficient space
 | |
|   // for the preferred width.
 | |
|   int popup_width =
 | |
|       std::min(popup_preferred_width, space_to_left + space_to_right);
 | |
| 
 | |
|   // Calculates the space that is available to grow into the preferred
 | |
|   // direction. In RTL, this is the space to the right side of the content
 | |
|   // area, in LTR this is the space to the left side of the content area.
 | |
|   int space_to_grow_in_preferred_direction =
 | |
|       is_rtl ? space_to_left : space_to_right;
 | |
| 
 | |
|   // Calculate how much the pop-up needs to grow into the non-preferred
 | |
|   // direction.
 | |
|   int amount_to_grow_in_unpreferred_direction =
 | |
|       std::max(0, popup_width - space_to_grow_in_preferred_direction);
 | |
| 
 | |
|   popup_bounds->set_width(popup_width);
 | |
|   if (is_rtl) {
 | |
|     // Note, in RTL the |pop_up_width| must be subtracted to achieve
 | |
|     // right-alignment of the pop-up with the element.
 | |
|     popup_bounds->set_x(preferred_starting_point - popup_width +
 | |
|                         amount_to_grow_in_unpreferred_direction);
 | |
|   } else {
 | |
|     popup_bounds->set_x(preferred_starting_point -
 | |
|                         amount_to_grow_in_unpreferred_direction);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void CalculatePopupXAndWidth(int popup_preferred_width,
 | |
|                              const gfx::Rect& content_area_bounds,
 | |
|                              const gfx::Rect& element_bounds,
 | |
|                              bool is_rtl,
 | |
|                              gfx::Rect* popup_bounds) {
 | |
|   int right_growth_start = std::clamp(
 | |
|       element_bounds.x(), content_area_bounds.x(), content_area_bounds.right());
 | |
|   int left_growth_end =
 | |
|       std::clamp(element_bounds.right(), content_area_bounds.x(),
 | |
|                  content_area_bounds.right());
 | |
| 
 | |
|   int right_available = content_area_bounds.right() - right_growth_start;
 | |
|   int left_available = left_growth_end - content_area_bounds.x();
 | |
| 
 | |
|   int popup_width = std::min(popup_preferred_width,
 | |
|                              std::max(left_available, right_available));
 | |
| 
 | |
|   // 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.
 | |
|   bool grow_left = false;
 | |
|   if (is_rtl) {
 | |
|     grow_left =
 | |
|         left_available >= popup_width || left_available >= right_available;
 | |
|   } else {
 | |
|     grow_left =
 | |
|         right_available < popup_width && right_available < left_available;
 | |
|   }
 | |
| 
 | |
|   popup_bounds->set_width(popup_width);
 | |
|   popup_bounds->set_x(grow_left ? left_growth_end - popup_width
 | |
|                                 : right_growth_start);
 | |
| }
 | |
| 
 | |
| void CalculatePopupYAndHeight(int popup_preferred_height,
 | |
|                               const gfx::Rect& content_area_bounds,
 | |
|                               const gfx::Rect& element_bounds,
 | |
|                               gfx::Rect* popup_bounds) {
 | |
|   int top_growth_end = std::clamp(element_bounds.y(), content_area_bounds.y(),
 | |
|                                   content_area_bounds.bottom());
 | |
|   int bottom_growth_start =
 | |
|       std::clamp(element_bounds.bottom(), content_area_bounds.y(),
 | |
|                  content_area_bounds.bottom());
 | |
| 
 | |
|   int top_available = top_growth_end - content_area_bounds.y();
 | |
|   int bottom_available = content_area_bounds.bottom() - bottom_growth_start;
 | |
| 
 | |
|   popup_bounds->set_height(popup_preferred_height);
 | |
|   popup_bounds->set_y(top_growth_end);
 | |
| 
 | |
|   if (bottom_available >= popup_preferred_height ||
 | |
|       bottom_available >= top_available) {
 | |
|     popup_bounds->AdjustToFit(
 | |
|         gfx::Rect(popup_bounds->x(), element_bounds.bottom(),
 | |
|                   popup_bounds->width(), bottom_available));
 | |
|   } else {
 | |
|     popup_bounds->AdjustToFit(gfx::Rect(popup_bounds->x(),
 | |
|                                         content_area_bounds.y(),
 | |
|                                         popup_bounds->width(), top_available));
 | |
|   }
 | |
| }
 | |
| 
 | |
| gfx::Rect CalculatePopupBounds(const gfx::Size& desired_size,
 | |
|                                const gfx::Rect& content_area_bounds,
 | |
|                                const gfx::Rect& element_bounds,
 | |
|                                bool is_rtl,
 | |
|                                bool horizontally_centered) {
 | |
|   gfx::Rect popup_bounds;
 | |
| 
 | |
|   if (horizontally_centered) {
 | |
|     CalculatePopupXAndWidthHorizontallyCentered(
 | |
|         desired_size.width(), content_area_bounds, element_bounds, is_rtl,
 | |
|         &popup_bounds);
 | |
|   } else {
 | |
|     CalculatePopupXAndWidth(desired_size.width(), content_area_bounds,
 | |
|                             element_bounds, is_rtl, &popup_bounds);
 | |
|   }
 | |
|   CalculatePopupYAndHeight(desired_size.height(), content_area_bounds,
 | |
|                            element_bounds, &popup_bounds);
 | |
| 
 | |
|   return popup_bounds;
 | |
| }
 | |
| 
 | |
| }  // namespace
 | |
| 
 | |
| AutofillPopup::AutofillPopup() = default;
 | |
| 
 | |
| AutofillPopup::~AutofillPopup() {
 | |
|   Hide();
 | |
| }
 | |
| 
 | |
| void AutofillPopup::CreateView(content::RenderFrameHost* frame_host,
 | |
|                                content::RenderFrameHost* embedder_frame_host,
 | |
|                                bool offscreen,
 | |
|                                views::View* parent,
 | |
|                                const gfx::RectF& r) {
 | |
|   Hide();
 | |
| 
 | |
|   frame_host_ = frame_host;
 | |
|   element_bounds_ = gfx::ToEnclosedRect(r);
 | |
| 
 | |
|   gfx::Vector2d height_offset(0, element_bounds_.height());
 | |
|   gfx::Point menu_position(element_bounds_.origin() + height_offset);
 | |
|   views::View::ConvertPointToScreen(parent, &menu_position);
 | |
|   popup_bounds_ = gfx::Rect(menu_position, element_bounds_.size());
 | |
| 
 | |
|   parent_ = parent;
 | |
|   parent_->AddObserver(this);
 | |
| 
 | |
|   view_ = new AutofillPopupView(this, parent->GetWidget());
 | |
| 
 | |
|   if (offscreen) {
 | |
|     auto* rwhv = embedder_frame_host ? embedder_frame_host->GetView()
 | |
|                                      : frame_host->GetView();
 | |
|     auto* osr_rwhv = static_cast<OffScreenRenderWidgetHostView*>(rwhv);
 | |
|     view_->view_proxy_ = std::make_unique<OffscreenViewProxy>(view_);
 | |
|     osr_rwhv->AddViewProxy(view_->view_proxy_.get());
 | |
|   }
 | |
| 
 | |
|   // Do this after OSR setup, we check for view_proxy_ when showing
 | |
|   view_->Show();
 | |
| }
 | |
| 
 | |
| void AutofillPopup::Hide() {
 | |
|   if (parent_) {
 | |
|     parent_->RemoveObserver(this);
 | |
|     parent_ = nullptr;
 | |
|   }
 | |
|   if (view_) {
 | |
|     view_->Hide();
 | |
|     view_ = nullptr;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void AutofillPopup::SetItems(const std::vector<std::u16string>& values,
 | |
|                              const std::vector<std::u16string>& labels) {
 | |
|   DCHECK(view_);
 | |
|   values_ = values;
 | |
|   labels_ = labels;
 | |
|   UpdatePopupBounds();
 | |
|   view_->OnSuggestionsChanged();
 | |
|   if (view_)  // could be hidden after the change
 | |
|     view_->DoUpdateBoundsAndRedrawPopup();
 | |
| }
 | |
| 
 | |
| void AutofillPopup::AcceptSuggestion(int index) {
 | |
|   mojo::AssociatedRemote<mojom::ElectronAutofillAgent> autofill_agent;
 | |
|   frame_host_->GetRemoteAssociatedInterfaces()->GetInterface(&autofill_agent);
 | |
|   autofill_agent->AcceptDataListSuggestion(value_at(index));
 | |
| }
 | |
| 
 | |
| void AutofillPopup::UpdatePopupBounds() {
 | |
|   DCHECK(parent_);
 | |
|   gfx::Point origin(element_bounds_.origin());
 | |
|   views::View::ConvertPointToScreen(parent_, &origin);
 | |
| 
 | |
|   gfx::Rect bounds(origin, element_bounds_.size());
 | |
|   gfx::Size preferred_size =
 | |
|       gfx::Size(GetDesiredPopupWidth(), GetDesiredPopupHeight());
 | |
| 
 | |
|   popup_bounds_ =
 | |
|       CalculatePopupBounds(preferred_size, parent_->GetBoundsInScreen(), bounds,
 | |
|                            base::i18n::IsRTL(), false);
 | |
| }
 | |
| 
 | |
| gfx::Rect AutofillPopup::popup_bounds_in_view() {
 | |
|   gfx::Point origin(popup_bounds_.origin());
 | |
|   views::View::ConvertPointFromScreen(parent_, &origin);
 | |
| 
 | |
|   return {origin, popup_bounds_.size()};
 | |
| }
 | |
| 
 | |
| void AutofillPopup::OnViewBoundsChanged(views::View* view) {
 | |
|   UpdatePopupBounds();
 | |
|   view_->DoUpdateBoundsAndRedrawPopup();
 | |
| }
 | |
| 
 | |
| void AutofillPopup::OnViewIsDeleting(views::View* view) {
 | |
|   Hide();
 | |
| }
 | |
| 
 | |
| int AutofillPopup::GetDesiredPopupHeight() {
 | |
|   return 2 * kPopupBorderThickness + values_.size() * kRowHeight;
 | |
| }
 | |
| 
 | |
| int AutofillPopup::GetDesiredPopupWidth() {
 | |
|   int popup_width = element_bounds_.width();
 | |
| 
 | |
|   for (size_t i = 0; i < values_.size(); ++i) {
 | |
|     int row_size = kEndPadding + 2 * kPopupBorderThickness +
 | |
|                    gfx::GetStringWidth(value_at(i), GetValueFontListForRow(i)) +
 | |
|                    gfx::GetStringWidth(label_at(i), GetLabelFontListForRow(i));
 | |
|     if (!label_at(i).empty())
 | |
|       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 {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::ColorId AutofillPopup::GetBackgroundColorIDForRow(int index) const {
 | |
|   return (view_ && index == view_->GetSelectedLine())
 | |
|              ? ui::kColorResultsTableHoveredBackground
 | |
|              : ui::kColorResultsTableNormalBackground;
 | |
| }
 | |
| 
 | |
| int AutofillPopup::LineFromY(int y) const {
 | |
|   int current_height = kPopupBorderThickness;
 | |
| 
 | |
|   for (size_t i = 0; i < values_.size(); ++i) {
 | |
|     current_height += kRowHeight;
 | |
| 
 | |
|     if (y <= current_height)
 | |
|       return i;
 | |
|   }
 | |
| 
 | |
|   return values_.size() - 1;
 | |
| }
 | |
| 
 | |
| }  // namespace electron
 |