// Copyright 2014 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 "ash/ime/infolist_window.h" #include #include #include "ash/ime/candidate_window_constants.h" #include "base/logging.h" #include "grit/ash_strings.h" #include "ui/base/l10n/l10n_util.h" #include "ui/gfx/color_utils.h" #include "ui/native_theme/native_theme.h" #include "ui/views/background.h" #include "ui/views/border.h" #include "ui/views/bubble/bubble_border.h" #include "ui/views/bubble/bubble_frame_view.h" #include "ui/views/controls/label.h" #include "ui/views/layout/box_layout.h" #include "ui/views/widget/widget.h" #include "ui/wm/core/window_animations.h" namespace ash { namespace ime { namespace { // The width of an info-list. const int kInfolistEntryWidth = 200; // The milliseconds of the delay to show the infolist window. const int kInfolistShowDelayMilliSeconds = 500; // The milliseconds of the delay to hide the infolist window. const int kInfolistHideDelayMilliSeconds = 500; /////////////////////////////////////////////////////////////////////////////// // InfolistBorder // The BubbleBorder subclass to draw the border and determine its position. class InfolistBorder : public views::BubbleBorder { public: InfolistBorder(); virtual ~InfolistBorder(); // views::BubbleBorder implementation. virtual gfx::Rect GetBounds(const gfx::Rect& anchor_rect, const gfx::Size& contents_size) const OVERRIDE; virtual gfx::Insets GetInsets() const OVERRIDE; private: DISALLOW_COPY_AND_ASSIGN(InfolistBorder); }; InfolistBorder::InfolistBorder() : views::BubbleBorder(views::BubbleBorder::LEFT_CENTER, views::BubbleBorder::NO_SHADOW, SK_ColorTRANSPARENT) { set_paint_arrow(views::BubbleBorder::PAINT_NONE); } InfolistBorder::~InfolistBorder() {} gfx::Rect InfolistBorder::GetBounds(const gfx::Rect& anchor_rect, const gfx::Size& contents_size) const { gfx::Rect bounds(contents_size); bounds.set_x(is_arrow_on_left(arrow()) ? anchor_rect.right() : anchor_rect.x() - contents_size.width()); // InfolistBorder modifies the vertical position based on the arrow offset // although it doesn't draw the arrow. The arrow offset is the half of // |contents_size| by default but can be modified through the off-screen logic // in BubbleFrameView. bounds.set_y(anchor_rect.y() + contents_size.height() / 2 - GetArrowOffset(contents_size)); return bounds; } gfx::Insets InfolistBorder::GetInsets() const { // This has to be specified and return empty insets to place the infolist // window without the gap. return gfx::Insets(); } } // namespace // InfolistRow renderes a row of a infolist. class InfolistEntryView : public views::View { public: InfolistEntryView(const ui::InfolistEntry& entry, const gfx::FontList& title_font_list, const gfx::FontList& description_font_list); virtual ~InfolistEntryView(); void SetEntry(const ui::InfolistEntry& entry); private: // views::View implementation. virtual gfx::Size GetPreferredSize() const OVERRIDE; void UpdateBackground(); ui::InfolistEntry entry_; // The title label. Owned by views hierarchy. views::Label* title_label_; // The description label. Owned by views hierarchy. views::Label* description_label_; DISALLOW_COPY_AND_ASSIGN(InfolistEntryView); }; InfolistEntryView::InfolistEntryView(const ui::InfolistEntry& entry, const gfx::FontList& title_font_list, const gfx::FontList& description_font_list) : entry_(entry) { SetLayoutManager(new views::BoxLayout(views::BoxLayout::kVertical, 0, 0, 0)); title_label_ = new views::Label(entry.title, title_font_list); title_label_->SetPosition(gfx::Point(0, 0)); title_label_->SetHorizontalAlignment(gfx::ALIGN_LEFT); title_label_->SetBorder(views::Border::CreateEmptyBorder(4, 7, 2, 4)); description_label_ = new views::Label(entry.body, description_font_list); description_label_->SetPosition(gfx::Point(0, 0)); description_label_->SetHorizontalAlignment(gfx::ALIGN_LEFT); description_label_->SetMultiLine(true); description_label_->SizeToFit(kInfolistEntryWidth); description_label_->SetBorder(views::Border::CreateEmptyBorder(2, 17, 4, 4)); AddChildView(title_label_); AddChildView(description_label_); UpdateBackground(); } InfolistEntryView::~InfolistEntryView() {} void InfolistEntryView::SetEntry(const ui::InfolistEntry& entry) { if (entry_ == entry) return; entry_ = entry; title_label_->SetText(entry_.title); description_label_->SetText(entry_.body); UpdateBackground(); } gfx::Size InfolistEntryView::GetPreferredSize() const { return gfx::Size(kInfolistEntryWidth, GetHeightForWidth(kInfolistEntryWidth)); } void InfolistEntryView::UpdateBackground() { if (entry_.highlighted) { set_background( views::Background::CreateSolidBackground(GetNativeTheme()->GetSystemColor( ui::NativeTheme::kColorId_TextfieldSelectionBackgroundFocused))); SetBorder(views::Border::CreateSolidBorder( 1, GetNativeTheme()->GetSystemColor( ui::NativeTheme::kColorId_FocusedBorderColor))); } else { set_background(NULL); SetBorder(views::Border::CreateEmptyBorder(1, 1, 1, 1)); } SchedulePaint(); } /////////////////////////////////////////////////////////////////////////////// // InfolistWindow InfolistWindow::InfolistWindow(views::View* candidate_window, const std::vector& entries) : views::BubbleDelegateView(candidate_window, views::BubbleBorder::NONE), title_font_list_(gfx::Font(kJapaneseFontName, kFontSizeDelta + 15)), description_font_list_(gfx::Font(kJapaneseFontName, kFontSizeDelta + 11)) { set_can_activate(false); set_accept_events(false); set_margins(gfx::Insets()); set_background( views::Background::CreateSolidBackground(GetNativeTheme()->GetSystemColor( ui::NativeTheme::kColorId_WindowBackground))); SetBorder(views::Border::CreateSolidBorder( 1, GetNativeTheme()->GetSystemColor( ui::NativeTheme::kColorId_MenuBorderColor))); SetLayoutManager(new views::BoxLayout(views::BoxLayout::kVertical, 0, 0, 0)); views::Label* caption_label = new views::Label( l10n_util::GetStringUTF16(IDS_ASH_IME_INFOLIST_WINDOW_TITLE)); caption_label->SetHorizontalAlignment(gfx::ALIGN_LEFT); caption_label->SetEnabledColor(GetNativeTheme()->GetSystemColor( ui::NativeTheme::kColorId_LabelEnabledColor)); caption_label->SetBorder(views::Border::CreateEmptyBorder(2, 2, 2, 2)); caption_label->set_background(views::Background::CreateSolidBackground( color_utils::AlphaBlend(SK_ColorBLACK, GetNativeTheme()->GetSystemColor( ui::NativeTheme::kColorId_WindowBackground), 0x10))); AddChildView(caption_label); for (size_t i = 0; i < entries.size(); ++i) { entry_views_.push_back(new InfolistEntryView( entries[i], title_font_list_, description_font_list_)); AddChildView(entry_views_.back()); } } InfolistWindow::~InfolistWindow() { } void InfolistWindow::InitWidget() { views::Widget* widget = views::BubbleDelegateView::CreateBubble(this); wm::SetWindowVisibilityAnimationType( widget->GetNativeView(), wm::WINDOW_VISIBILITY_ANIMATION_TYPE_FADE); // BubbleFrameView will be initialized through CreateBubble. GetBubbleFrameView()->SetBubbleBorder( scoped_ptr(new InfolistBorder())); SizeToContents(); } void InfolistWindow::Relayout(const std::vector& entries) { size_t i = 0; for (; i < entries.size(); ++i) { if (i < entry_views_.size()) { entry_views_[i]->SetEntry(entries[i]); } else { InfolistEntryView* new_entry = new InfolistEntryView( entries[i], title_font_list_, description_font_list_); AddChildView(new_entry); entry_views_.push_back(new_entry); } } if (i < entry_views_.size()) { for (; i < entry_views_.size(); ++i) delete entry_views_[i]; entry_views_.resize(entries.size()); } Layout(); GetBubbleFrameView()->bubble_border()->set_arrow_offset(0); SizeToContents(); } void InfolistWindow::ShowWithDelay() { show_hide_timer_.Start( FROM_HERE, base::TimeDelta::FromMilliseconds(kInfolistShowDelayMilliSeconds), GetWidget(), &views::Widget::Show); } void InfolistWindow::HideWithDelay() { show_hide_timer_.Start( FROM_HERE, base::TimeDelta::FromMilliseconds(kInfolistHideDelayMilliSeconds), GetWidget(), &views::Widget::Close); } void InfolistWindow::ShowImmediately() { show_hide_timer_.Stop(); GetWidget()->Show(); } void InfolistWindow::HideImmediately() { show_hide_timer_.Stop(); GetWidget()->Close(); } void InfolistWindow::WindowClosing() { show_hide_timer_.Stop(); } } // namespace ime } // namespace ash