Chromium Code Reviews| OLD | NEW |
|---|---|
| (Empty) | |
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | |
| 2 // Use of this source code is governed by a BSD-style license that can be | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 #include "ui/message_center/views/bounded_label.h" | |
| 6 | |
| 7 #include <limits> | |
| 8 | |
| 9 #include "base/string_util.h" | |
| 10 #include "base/utf_string_conversions.h" | |
| 11 #include "ui/base/text/text_elider.h" | |
| 12 #include "ui/gfx/canvas.h" | |
| 13 | |
| 14 namespace message_center { | |
| 15 | |
| 16 BoundedLabel::BoundedLabel(string16 text, size_t max_lines) | |
| 17 : views::Label(text, gfx::Font()) { | |
| 18 Init(max_lines); | |
| 19 } | |
| 20 | |
| 21 BoundedLabel::BoundedLabel(string16 text, gfx::Font font, size_t max_lines) | |
| 22 : views::Label(text, font) { | |
| 23 Init(max_lines); | |
| 24 } | |
| 25 | |
| 26 BoundedLabel::~BoundedLabel() { | |
| 27 } | |
| 28 | |
| 29 void BoundedLabel::SetMaxLines(size_t lines) { | |
| 30 is_preferred_lines_valid_ = false; | |
| 31 is_text_size_valid_ = false; | |
| 32 max_lines_ = lines; | |
| 33 } | |
| 34 | |
| 35 size_t BoundedLabel::GetMaxLines() { | |
| 36 return max_lines_; | |
| 37 } | |
| 38 | |
| 39 size_t BoundedLabel::GetPreferredLines() { | |
| 40 if (!is_preferred_lines_valid_) { | |
| 41 int wrap_width = width() - GetInsets().width(); | |
| 42 int unlimited_lines = std::numeric_limits<size_t>::max(); | |
| 43 preferred_lines_ = SplitLines(wrap_width, unlimited_lines).size(); | |
| 44 is_preferred_lines_valid_ = true; | |
| 45 } | |
| 46 return preferred_lines_; | |
| 47 } | |
| 48 | |
| 49 int BoundedLabel::GetHeightForWidth(int width) const { | |
| 50 width = std::max(0, width - GetInsets().width()); | |
| 51 int height = font().GetHeight(); | |
| 52 string16 text = JoinString(SplitLines(width, max_lines_), '\n'); | |
| 53 gfx::Canvas::SizeStringInt(text, font(), &width, &height, GetTextFlags()); | |
| 54 return height + GetInsets().height(); | |
| 55 } | |
| 56 | |
| 57 gfx::Size BoundedLabel::GetTextSize() const { | |
| 58 if (!is_text_size_valid_) { | |
| 59 text_size_.set_width(width() - GetInsets().width()); | |
| 60 text_size_.set_height(GetHeightForWidth(width()) - GetInsets().height()); | |
| 61 is_text_size_valid_ = true; | |
| 62 } | |
| 63 return text_size_; | |
| 64 } | |
| 65 | |
| 66 void BoundedLabel::OnBoundsChanged(const gfx::Rect& previous_bounds) { | |
| 67 is_preferred_lines_valid_ = false; | |
| 68 is_text_size_valid_ = false; | |
| 69 } | |
| 70 | |
| 71 void BoundedLabel::PaintText(gfx::Canvas* canvas, | |
| 72 const string16& paint_text, | |
| 73 const gfx::Rect& text_bounds, | |
| 74 int flags) { | |
| 75 gfx::Insets insets = GetInsets(); | |
| 76 gfx::Rect bounds(gfx::Point(insets.left(), insets.top()), GetTextSize()); | |
| 77 string16 text = JoinString(SplitLines(bounds.width(), max_lines_), '\n'); | |
| 78 views::Label::PaintText(canvas, text, bounds, GetTextFlags()); | |
| 79 } | |
| 80 | |
| 81 void BoundedLabel::Init(size_t max_lines) { | |
| 82 SetMultiLine(true); | |
| 83 SetAllowCharacterBreak(true); | |
| 84 SetElideBehavior(views::Label::ELIDE_AT_END); | |
| 85 max_lines_ = max_lines; | |
| 86 is_preferred_lines_valid_ = false; | |
| 87 is_text_size_valid_ = false; | |
| 88 } | |
| 89 | |
| 90 int BoundedLabel::GetTextFlags() const { | |
| 91 int flags = gfx::Canvas::MULTI_LINE | gfx::Canvas::CHARACTER_BREAK; | |
| 92 | |
| 93 // We can't use subpixel rendering if the background is non-opaque. | |
| 94 if (SkColorGetA(background_color()) != 0xFF) | |
| 95 flags |= gfx::Canvas::NO_SUBPIXEL_RENDERING; | |
| 96 | |
| 97 if (directionality_mode() == AUTO_DETECT_DIRECTIONALITY) { | |
| 98 base::i18n::TextDirection direction = | |
| 99 base::i18n::GetFirstStrongCharacterDirection(text()); | |
| 100 if (direction == base::i18n::RIGHT_TO_LEFT) | |
| 101 flags |= gfx::Canvas::FORCE_RTL_DIRECTIONALITY; | |
| 102 else | |
| 103 flags |= gfx::Canvas::FORCE_LTR_DIRECTIONALITY; | |
| 104 } | |
| 105 | |
| 106 switch (horizontal_alignment()) { | |
| 107 case gfx::ALIGN_LEFT: | |
| 108 flags |= gfx::Canvas::TEXT_ALIGN_LEFT; | |
| 109 break; | |
| 110 case gfx::ALIGN_CENTER: | |
| 111 flags |= gfx::Canvas::TEXT_ALIGN_CENTER; | |
| 112 break; | |
| 113 case gfx::ALIGN_RIGHT: | |
| 114 flags |= gfx::Canvas::TEXT_ALIGN_RIGHT; | |
| 115 break; | |
| 116 } | |
| 117 | |
| 118 return flags; | |
| 119 } | |
| 120 | |
| 121 std::vector<string16> BoundedLabel::SplitLines(int width, | |
| 122 size_t max_lines) const { | |
| 123 // Adjust max_lines so (max_lines + 1) * line_height <= INT_MAX, then use the | |
| 124 // adjusted max_lines to get a max_height of (max_lines + 1) * line_height. | |
| 125 size_t max_height = std::numeric_limits<int>::max(); | |
| 126 size_t line_height = std::max(font().GetHeight(), 2); // At least 2 pixels! | |
| 127 max_lines = std::min(max_lines, max_height / line_height - 1); | |
| 128 max_height = (max_lines + 1) * line_height; | |
| 129 | |
| 130 // Split. Do not use ui::WRAP_LONG_WORDS instead of ui::IGNORE_LONG_WORDS | |
| 131 // below as this may cause ui::ElideRectangleText() to go into an infinite | |
| 132 // loop for small width values. | |
| 133 std::vector<string16> lines; | |
| 134 ui::ElideRectangleText(text(), font(), width, max_height, | |
| 135 ui::IGNORE_LONG_WORDS, &lines); | |
| 136 | |
| 137 // Elide if necessary. | |
| 138 if (lines.size() > max_lines) { | |
| 139 string16 last = lines[max_lines - 1] + UTF8ToUTF16(ui::kEllipsis); | |
|
Jun Mukai
2013/03/14 09:26:47
what will happen if lines[max_lines - 1] is shorte
dharcourt
2013/03/14 20:21:57
If lines[max_lines - 1] + '...' exceeds the width,
| |
| 140 lines.resize(max_lines - 1); | |
| 141 lines.push_back((font().GetStringWidth(last) > width) ? | |
| 142 ui::ElideText(last, font(), width, ui::ELIDE_AT_END) : | |
| 143 last); | |
| 144 } | |
| 145 | |
| 146 return lines; | |
| 147 } | |
| 148 | |
| 149 } // namespace message_center | |
| OLD | NEW |