// Copyright (c) 2014 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #include "atom/browser/ui/views/submenu_button.h" #include #include #include "base/strings/string_util.h" #include "base/strings/utf_string_conversions.h" #include "ui/gfx/canvas.h" #include "ui/gfx/color_utils.h" #include "ui/gfx/text_utils.h" #include "ui/views/animation/flood_fill_ink_drop_ripple.h" #include "ui/views/animation/ink_drop_host_view.h" #include "ui/views/animation/ink_drop_impl.h" #include "ui/views/controls/button/label_button_border.h" namespace atom { SubmenuButton::SubmenuButton(const base::string16& title, views::MenuButtonListener* menu_button_listener, const SkColor& background_color) : views::MenuButton(gfx::RemoveAcceleratorChar(title, '&', NULL, NULL), menu_button_listener, false), background_color_(background_color) { #if defined(OS_LINUX) // Dont' use native style border. SetBorder(CreateDefaultBorder()); #endif if (GetUnderlinePosition(title, &accelerator_, &underline_start_, &underline_end_)) gfx::Canvas::SizeStringInt(GetText(), gfx::FontList(), &text_width_, &text_height_, 0, 0); SetInkDropMode(InkDropMode::ON); set_ink_drop_base_color( color_utils::BlendTowardOppositeLuma(background_color_, 0x61)); } SubmenuButton::~SubmenuButton() {} std::unique_ptr SubmenuButton::CreateInkDropRipple() const { std::unique_ptr ripple( new views::FloodFillInkDropRipple( size(), GetInkDropCenterBasedOnLastEvent(), GetInkDropBaseColor(), ink_drop_visible_opacity())); return ripple; } std::unique_ptr SubmenuButton::CreateInkDrop() { std::unique_ptr ink_drop = views::Button::CreateDefaultInkDropImpl(); ink_drop->SetShowHighlightOnHover(false); return std::move(ink_drop); } void SubmenuButton::SetAcceleratorVisibility(bool visible) { if (visible == show_underline_) return; show_underline_ = visible; SchedulePaint(); } void SubmenuButton::SetUnderlineColor(SkColor color) { underline_color_ = color; } void SubmenuButton::PaintButtonContents(gfx::Canvas* canvas) { views::MenuButton::PaintButtonContents(canvas); if (show_underline_ && (underline_start_ != underline_end_)) { int padding = (width() - text_width_) / 2; int underline_height = (height() + text_height_) / 2 - 2; canvas->DrawLine(gfx::Point(underline_start_ + padding, underline_height), gfx::Point(underline_end_ + padding, underline_height), underline_color_); } } bool SubmenuButton::GetUnderlinePosition(const base::string16& text, base::char16* accelerator, int* start, int* end) const { int pos, span; base::string16 trimmed = gfx::RemoveAcceleratorChar(text, '&', &pos, &span); if (pos > -1 && span != 0) { *accelerator = base::ToUpperASCII(trimmed[pos]); GetCharacterPosition(trimmed, pos, start); GetCharacterPosition(trimmed, pos + span, end); return true; } return false; } void SubmenuButton::GetCharacterPosition(const base::string16& text, int index, int* pos) const { int height = 0; gfx::Canvas::SizeStringInt(text.substr(0, index), gfx::FontList(), pos, &height, 0, 0); } } // namespace atom