// 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 <string>
#include <vector>
#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<ui::InfolistEntry>& 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_use_focusless(true);
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<views::BubbleBorder>(new InfolistBorder()));
SizeToContents();
}
void InfolistWindow::Relayout(const std::vector<ui::InfolistEntry>& 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