| OLD | NEW |
| 1 // Copyright 2013 The Chromium Authors. All rights reserved. | 1 // Copyright 2013 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "ui/views/controls/styled_label.h" | 5 #include "ui/views/controls/styled_label.h" |
| 6 | 6 |
| 7 #include <limits> |
| 7 #include <vector> | 8 #include <vector> |
| 8 | 9 |
| 9 #include "base/strings/string_util.h" | 10 #include "base/strings/string_util.h" |
| 10 #include "ui/gfx/font_list.h" | 11 #include "ui/gfx/font_list.h" |
| 11 #include "ui/gfx/text_elider.h" | 12 #include "ui/gfx/text_elider.h" |
| 12 #include "ui/native_theme/native_theme.h" | 13 #include "ui/native_theme/native_theme.h" |
| 13 #include "ui/views/controls/label.h" | 14 #include "ui/views/controls/label.h" |
| 14 #include "ui/views/controls/link.h" | 15 #include "ui/views/controls/link.h" |
| 15 #include "ui/views/controls/styled_label_listener.h" | 16 #include "ui/views/controls/styled_label_listener.h" |
| 16 | 17 |
| (...skipping 75 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 92 | 93 |
| 93 // StyledLabel ---------------------------------------------------------------- | 94 // StyledLabel ---------------------------------------------------------------- |
| 94 | 95 |
| 95 // static | 96 // static |
| 96 const char StyledLabel::kViewClassName[] = "StyledLabel"; | 97 const char StyledLabel::kViewClassName[] = "StyledLabel"; |
| 97 | 98 |
| 98 StyledLabel::StyledLabel(const base::string16& text, | 99 StyledLabel::StyledLabel(const base::string16& text, |
| 99 StyledLabelListener* listener) | 100 StyledLabelListener* listener) |
| 100 : specified_line_height_(0), | 101 : specified_line_height_(0), |
| 101 listener_(listener), | 102 listener_(listener), |
| 103 width_at_last_size_calculation_(0), |
| 102 width_at_last_layout_(0), | 104 width_at_last_layout_(0), |
| 103 displayed_on_background_color_(SkColorSetRGB(0xFF, 0xFF, 0xFF)), | 105 displayed_on_background_color_(SkColorSetRGB(0xFF, 0xFF, 0xFF)), |
| 104 displayed_on_background_color_set_(false), | 106 displayed_on_background_color_set_(false), |
| 105 auto_color_readability_enabled_(true) { | 107 auto_color_readability_enabled_(true) { |
| 106 base::TrimWhitespace(text, base::TRIM_TRAILING, &text_); | 108 base::TrimWhitespace(text, base::TRIM_TRAILING, &text_); |
| 107 } | 109 } |
| 108 | 110 |
| 109 StyledLabel::~StyledLabel() {} | 111 StyledLabel::~StyledLabel() {} |
| 110 | 112 |
| 111 void StyledLabel::SetText(const base::string16& text) { | 113 void StyledLabel::SetText(const base::string16& text) { |
| (...skipping 40 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 152 displayed_on_background_color_ = color; | 154 displayed_on_background_color_ = color; |
| 153 displayed_on_background_color_set_ = true; | 155 displayed_on_background_color_set_ = true; |
| 154 | 156 |
| 155 for (int i = 0, count = child_count(); i < count; ++i) { | 157 for (int i = 0, count = child_count(); i < count; ++i) { |
| 156 DCHECK((child_at(i)->GetClassName() == Label::kViewClassName) || | 158 DCHECK((child_at(i)->GetClassName() == Label::kViewClassName) || |
| 157 (child_at(i)->GetClassName() == Link::kViewClassName)); | 159 (child_at(i)->GetClassName() == Link::kViewClassName)); |
| 158 static_cast<Label*>(child_at(i))->SetBackgroundColor(color); | 160 static_cast<Label*>(child_at(i))->SetBackgroundColor(color); |
| 159 } | 161 } |
| 160 } | 162 } |
| 161 | 163 |
| 164 void StyledLabel::SizeToFit(int max_width) { |
| 165 if (max_width == 0) |
| 166 max_width = std::numeric_limits<int>::max(); |
| 167 |
| 168 SetSize(CalculateAndDoLayout(max_width, true)); |
| 169 } |
| 170 |
| 162 const char* StyledLabel::GetClassName() const { | 171 const char* StyledLabel::GetClassName() const { |
| 163 return kViewClassName; | 172 return kViewClassName; |
| 164 } | 173 } |
| 165 | 174 |
| 166 gfx::Insets StyledLabel::GetInsets() const { | 175 gfx::Insets StyledLabel::GetInsets() const { |
| 167 gfx::Insets insets = View::GetInsets(); | 176 gfx::Insets insets = View::GetInsets(); |
| 168 | 177 |
| 169 // We need a focus border iff we contain a link that will have a focus border. | 178 // We need a focus border iff we contain a link that will have a focus border. |
| 170 // That in turn will be true only if the link is non-empty. | 179 // That in turn will be true only if the link is non-empty. |
| 171 for (StyleRanges::const_iterator i(style_ranges_.begin()); | 180 for (StyleRanges::const_iterator i(style_ranges_.begin()); |
| 172 i != style_ranges_.end(); ++i) { | 181 i != style_ranges_.end(); ++i) { |
| 173 if (i->style_info.is_link && !i->range.is_empty()) { | 182 if (i->style_info.is_link && !i->range.is_empty()) { |
| 174 const gfx::Insets focus_border_padding( | 183 const gfx::Insets focus_border_padding( |
| 175 Label::kFocusBorderPadding, Label::kFocusBorderPadding, | 184 Label::kFocusBorderPadding, Label::kFocusBorderPadding, |
| 176 Label::kFocusBorderPadding, Label::kFocusBorderPadding); | 185 Label::kFocusBorderPadding, Label::kFocusBorderPadding); |
| 177 insets += focus_border_padding; | 186 insets += focus_border_padding; |
| 178 break; | 187 break; |
| 179 } | 188 } |
| 180 } | 189 } |
| 181 | 190 |
| 182 return insets; | 191 return insets; |
| 183 } | 192 } |
| 184 | 193 |
| 194 gfx::Size StyledLabel::GetPreferredSize() const { |
| 195 return calculated_size_; |
| 196 } |
| 197 |
| 185 int StyledLabel::GetHeightForWidth(int w) const { | 198 int StyledLabel::GetHeightForWidth(int w) const { |
| 186 // TODO(erg): Munge the const-ness of the style label. CalculateAndDoLayout | 199 // TODO(erg): Munge the const-ness of the style label. CalculateAndDoLayout |
| 187 // doesn't actually make any changes to member variables when |dry_run| is | 200 // doesn't actually make any changes to member variables when |dry_run| is |
| 188 // set to true. In general, the mutating and non-mutating parts shouldn't | 201 // set to true. In general, the mutating and non-mutating parts shouldn't |
| 189 // be in the same codepath. | 202 // be in the same codepath. |
| 190 calculated_size_ = | 203 return const_cast<StyledLabel*>(this)->CalculateAndDoLayout(w, true).height(); |
| 191 const_cast<StyledLabel*>(this)->CalculateAndDoLayout(w, true); | |
| 192 return calculated_size_.height(); | |
| 193 } | 204 } |
| 194 | 205 |
| 195 void StyledLabel::Layout() { | 206 void StyledLabel::Layout() { |
| 196 calculated_size_ = CalculateAndDoLayout(GetLocalBounds().width(), false); | 207 CalculateAndDoLayout(GetLocalBounds().width(), false); |
| 197 width_at_last_layout_ = calculated_size_.width(); | |
| 198 } | 208 } |
| 199 | 209 |
| 200 void StyledLabel::PreferredSizeChanged() { | 210 void StyledLabel::PreferredSizeChanged() { |
| 201 calculated_size_ = gfx::Size(); | 211 calculated_size_ = gfx::Size(); |
| 212 width_at_last_size_calculation_ = 0; |
| 213 width_at_last_layout_ = 0; |
| 202 View::PreferredSizeChanged(); | 214 View::PreferredSizeChanged(); |
| 203 } | 215 } |
| 204 | 216 |
| 205 void StyledLabel::LinkClicked(Link* source, int event_flags) { | 217 void StyledLabel::LinkClicked(Link* source, int event_flags) { |
| 206 if (listener_) | 218 if (listener_) |
| 207 listener_->StyledLabelLinkClicked(link_targets_[source], event_flags); | 219 listener_->StyledLabelLinkClicked(link_targets_[source], event_flags); |
| 208 } | 220 } |
| 209 | 221 |
| 210 gfx::Size StyledLabel::CalculateAndDoLayout(int width, bool dry_run) { | 222 gfx::Size StyledLabel::CalculateAndDoLayout(int width, bool dry_run) { |
| 223 if (width == width_at_last_size_calculation_ && |
| 224 (dry_run || width == width_at_last_layout_)) |
| 225 return calculated_size_; |
| 226 |
| 227 width_at_last_size_calculation_ = width; |
| 228 if (!dry_run) |
| 229 width_at_last_layout_ = width; |
| 230 |
| 211 width -= GetInsets().width(); | 231 width -= GetInsets().width(); |
| 212 if (width == calculated_size_.width() && | |
| 213 (dry_run || width_at_last_layout_ == width)) | |
| 214 return calculated_size_; | |
| 215 | 232 |
| 216 if (!dry_run) { | 233 if (!dry_run) { |
| 217 RemoveAllChildViews(true); | 234 RemoveAllChildViews(true); |
| 218 link_targets_.clear(); | 235 link_targets_.clear(); |
| 219 } | 236 } |
| 220 | 237 |
| 221 if (width <= 0 || text_.empty()) | 238 if (width <= 0 || text_.empty()) |
| 222 return gfx::Size(); | 239 return gfx::Size(); |
| 223 | 240 |
| 224 const int line_height = specified_line_height_ > 0 ? specified_line_height_ | 241 const int line_height = specified_line_height_ > 0 ? specified_line_height_ |
| 225 : CalculateLineHeight(font_list_); | 242 : CalculateLineHeight(font_list_); |
| 226 // The index of the line we're on. | 243 // The index of the line we're on. |
| 227 int line = 0; | 244 int line = 0; |
| 228 // The x position (in pixels) of the line we're on, relative to content | 245 // The x position (in pixels) of the line we're on, relative to content |
| 229 // bounds. | 246 // bounds. |
| 230 int x = 0; | 247 int x = 0; |
| 248 // The width that was actually used. Guaranteed to be no larger than |width|. |
| 249 int used_width = 0; |
| 231 | 250 |
| 232 base::string16 remaining_string = text_; | 251 base::string16 remaining_string = text_; |
| 233 StyleRanges::const_iterator current_range = style_ranges_.begin(); | 252 StyleRanges::const_iterator current_range = style_ranges_.begin(); |
| 234 | 253 |
| 235 // Iterate over the text, creating a bunch of labels and links and laying them | 254 // Iterate over the text, creating a bunch of labels and links and laying them |
| 236 // out in the appropriate positions. | 255 // out in the appropriate positions. |
| 237 while (!remaining_string.empty()) { | 256 while (!remaining_string.empty()) { |
| 238 // Don't put whitespace at beginning of a line with an exception for the | 257 // Don't put whitespace at beginning of a line with an exception for the |
| 239 // first line (so the text's leading whitespace is respected). | 258 // first line (so the text's leading whitespace is respected). |
| 240 if (x == 0 && line > 0) { | 259 if (x == 0 && line > 0) { |
| (...skipping 85 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 326 const gfx::Size view_size = label->GetPreferredSize(); | 345 const gfx::Size view_size = label->GetPreferredSize(); |
| 327 if (!dry_run) { | 346 if (!dry_run) { |
| 328 label->SetBoundsRect(gfx::Rect( | 347 label->SetBoundsRect(gfx::Rect( |
| 329 gfx::Point(GetInsets().left() + x - focus_border_insets.left(), | 348 gfx::Point(GetInsets().left() + x - focus_border_insets.left(), |
| 330 GetInsets().top() + line * line_height - | 349 GetInsets().top() + line * line_height - |
| 331 focus_border_insets.top()), | 350 focus_border_insets.top()), |
| 332 view_size)); | 351 view_size)); |
| 333 AddChildView(label.release()); | 352 AddChildView(label.release()); |
| 334 } | 353 } |
| 335 x += view_size.width() - focus_border_insets.width(); | 354 x += view_size.width() - focus_border_insets.width(); |
| 355 used_width = std::max(used_width, x); |
| 336 | 356 |
| 337 remaining_string = remaining_string.substr(chunk.size()); | 357 remaining_string = remaining_string.substr(chunk.size()); |
| 338 } | 358 } |
| 339 | 359 |
| 360 DCHECK_LE(used_width, width); |
| 340 // The user-specified line height only applies to interline spacing, so the | 361 // The user-specified line height only applies to interline spacing, so the |
| 341 // final line's height is unaffected. | 362 // final line's height is unaffected. |
| 342 int total_height = line * line_height + | 363 int total_height = line * line_height + |
| 343 CalculateLineHeight(font_list_) + GetInsets().height(); | 364 CalculateLineHeight(font_list_) + GetInsets().height(); |
| 344 return gfx::Size(width, total_height); | 365 calculated_size_ = gfx::Size(used_width + GetInsets().width(), total_height); |
| 366 return calculated_size_; |
| 345 } | 367 } |
| 346 | 368 |
| 347 } // namespace views | 369 } // namespace views |
| OLD | NEW |