|  | // Copyright 2014 The Chromium Authors | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "ui/gfx/font_list_impl.h" | 
|  |  | 
|  | #include <stddef.h> | 
|  |  | 
|  | #include <algorithm> | 
|  |  | 
|  | #include "base/check_op.h" | 
|  | #include "base/strings/string_number_conversions.h" | 
|  | #include "base/strings/string_util.h" | 
|  | #include "ui/gfx/font_list.h" | 
|  |  | 
|  | namespace gfx { | 
|  | namespace { | 
|  |  | 
|  | // Returns a font description from |families|, |style|, and |size_pixels|. | 
|  | std::string BuildDescription(const std::vector<std::string>& families, | 
|  | int style, | 
|  | int size_pixels, | 
|  | Font::Weight weight) { | 
|  | std::string description = base::JoinString(families, ","); | 
|  | description += ","; | 
|  |  | 
|  | if (style & Font::ITALIC) | 
|  | description += "Italic "; | 
|  | switch (weight) { | 
|  | case Font::Weight::THIN: | 
|  | description += "Thin "; | 
|  | break; | 
|  | case Font::Weight::EXTRA_LIGHT: | 
|  | description += "Ultra-Light "; | 
|  | break; | 
|  | case Font::Weight::LIGHT: | 
|  | description += "Light "; | 
|  | break; | 
|  | case Font::Weight::MEDIUM: | 
|  | description += "Medium "; | 
|  | break; | 
|  | case Font::Weight::SEMIBOLD: | 
|  | description += "Semi-Bold "; | 
|  | break; | 
|  | case Font::Weight::BOLD: | 
|  | description += "Bold "; | 
|  | break; | 
|  | case Font::Weight::EXTRA_BOLD: | 
|  | description += "Ultra-Bold "; | 
|  | break; | 
|  | case Font::Weight::BLACK: | 
|  | description += "Heavy "; | 
|  | break; | 
|  | case Font::Weight::NORMAL: | 
|  | case Font::Weight::INVALID: | 
|  | break; | 
|  | } | 
|  |  | 
|  | description += base::NumberToString(size_pixels); | 
|  | description += "px"; | 
|  |  | 
|  | return description; | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | FontListImpl::FontListImpl(const std::string& font_description_string) | 
|  | : font_description_string_(font_description_string), | 
|  | common_height_(-1), | 
|  | common_baseline_(-1), | 
|  | font_style_(-1), | 
|  | font_size_(-1), | 
|  | font_weight_(Font::Weight::INVALID) { | 
|  | DCHECK(!font_description_string.empty()); | 
|  | // DCHECK description string ends with "px" for size in pixel. | 
|  | DCHECK(base::EndsWith(font_description_string, "px", | 
|  | base::CompareCase::SENSITIVE)); | 
|  | } | 
|  |  | 
|  | FontListImpl::FontListImpl(const std::vector<std::string>& font_names, | 
|  | int font_style, | 
|  | int font_size, | 
|  | Font::Weight font_weight) | 
|  | : font_description_string_( | 
|  | BuildDescription(font_names, font_style, font_size, font_weight)), | 
|  | common_height_(-1), | 
|  | common_baseline_(-1), | 
|  | font_style_(font_style), | 
|  | font_size_(font_size), | 
|  | font_weight_(font_weight) { | 
|  | DCHECK(!font_names.empty()); | 
|  | DCHECK(!font_names[0].empty()); | 
|  | } | 
|  |  | 
|  | FontListImpl::FontListImpl(const std::vector<Font>& fonts) | 
|  | : fonts_(fonts), | 
|  | common_height_(-1), | 
|  | common_baseline_(-1), | 
|  | font_style_(-1), | 
|  | font_size_(-1), | 
|  | font_weight_(Font::Weight::INVALID) { | 
|  | DCHECK(!fonts.empty()); | 
|  | font_style_ = fonts[0].GetStyle(); | 
|  | font_size_ = fonts[0].GetFontSize(); | 
|  | font_weight_ = fonts[0].GetWeight(); | 
|  | #if DCHECK_IS_ON() | 
|  | for (size_t i = 1; i < fonts.size(); ++i) { | 
|  | DCHECK_EQ(fonts[i].GetStyle(), font_style_); | 
|  | DCHECK_EQ(fonts[i].GetFontSize(), font_size_); | 
|  | } | 
|  | #endif | 
|  | } | 
|  |  | 
|  | FontListImpl::FontListImpl(const Font& font) | 
|  | : common_height_(-1), | 
|  | common_baseline_(-1), | 
|  | font_style_(-1), | 
|  | font_size_(-1), | 
|  | font_weight_(Font::Weight::INVALID) { | 
|  | fonts_.push_back(font); | 
|  | } | 
|  |  | 
|  | FontListImpl* FontListImpl::Derive(int size_delta, | 
|  | int font_style, | 
|  | Font::Weight weight) const { | 
|  | // If there is a font vector, derive from that. | 
|  | if (!fonts_.empty()) { | 
|  | std::vector<Font> fonts = fonts_; | 
|  | for (size_t i = 0; i < fonts.size(); ++i) | 
|  | fonts[i] = fonts[i].Derive(size_delta, font_style, weight); | 
|  | return new FontListImpl(fonts); | 
|  | } | 
|  |  | 
|  | // Otherwise, parse the font description string to derive from it. | 
|  | std::vector<std::string> font_names; | 
|  | int old_size; | 
|  | int old_style; | 
|  | Font::Weight old_weight; | 
|  | CHECK(FontList::ParseDescription(font_description_string_, &font_names, | 
|  | &old_style, &old_size, &old_weight)); | 
|  | const int size = std::max(1, old_size + size_delta); | 
|  | return new FontListImpl(font_names, font_style, size, weight); | 
|  | } | 
|  |  | 
|  | int FontListImpl::GetHeight() const { | 
|  | if (common_height_ == -1) | 
|  | CacheCommonFontHeightAndBaseline(); | 
|  | return common_height_; | 
|  | } | 
|  |  | 
|  | int FontListImpl::GetBaseline() const { | 
|  | if (common_baseline_ == -1) | 
|  | CacheCommonFontHeightAndBaseline(); | 
|  | return common_baseline_; | 
|  | } | 
|  |  | 
|  | int FontListImpl::GetCapHeight() const { | 
|  | // Assume the primary font is used to render Latin characters. | 
|  | return GetPrimaryFont().GetCapHeight(); | 
|  | } | 
|  |  | 
|  | int FontListImpl::GetExpectedTextWidth(int length) const { | 
|  | // Rely on the primary font metrics for the time being. | 
|  | return GetPrimaryFont().GetExpectedTextWidth(length); | 
|  | } | 
|  |  | 
|  | int FontListImpl::GetFontStyle() const { | 
|  | if (font_style_ == -1) | 
|  | CacheFontStyleAndSize(); | 
|  | return font_style_; | 
|  | } | 
|  |  | 
|  | int FontListImpl::GetFontSize() const { | 
|  | if (font_size_ == -1) | 
|  | CacheFontStyleAndSize(); | 
|  | return font_size_; | 
|  | } | 
|  |  | 
|  | Font::Weight FontListImpl::GetFontWeight() const { | 
|  | if (font_weight_ == Font::Weight::INVALID) | 
|  | CacheFontStyleAndSize(); | 
|  | return font_weight_; | 
|  | } | 
|  |  | 
|  | const std::vector<Font>& FontListImpl::GetFonts() const { | 
|  | if (fonts_.empty()) { | 
|  | DCHECK(!font_description_string_.empty()); | 
|  |  | 
|  | std::vector<std::string> font_names; | 
|  | // It's possible that Font::UNDERLINE is specified and it's already | 
|  | // stored in |font_style_| but |font_description_string_| doesn't have the | 
|  | // underline info.  So we should respect |font_style_| as long as it's | 
|  | // valid. | 
|  | int style = 0; | 
|  | CHECK(FontList::ParseDescription(font_description_string_, &font_names, | 
|  | &style, &font_size_, &font_weight_)); | 
|  | if (font_style_ == -1) | 
|  | font_style_ = style; | 
|  | for (size_t i = 0; i < font_names.size(); ++i) { | 
|  | DCHECK(!font_names[i].empty()); | 
|  |  | 
|  | Font font(font_names[i], font_size_); | 
|  | if (font_style_ == Font::NORMAL && font_weight_ == Font::Weight::NORMAL) | 
|  | fonts_.push_back(font); | 
|  | else | 
|  | fonts_.push_back(font.Derive(0, font_style_, font_weight_)); | 
|  | } | 
|  | } | 
|  | return fonts_; | 
|  | } | 
|  |  | 
|  | const Font& FontListImpl::GetPrimaryFont() const { | 
|  | return GetFonts()[0]; | 
|  | } | 
|  |  | 
|  | FontListImpl::~FontListImpl() {} | 
|  |  | 
|  | void FontListImpl::CacheCommonFontHeightAndBaseline() const { | 
|  | int ascent = 0; | 
|  | int descent = 0; | 
|  | const std::vector<Font>& fonts = GetFonts(); | 
|  | for (auto i = fonts.begin(); i != fonts.end(); ++i) { | 
|  | ascent = std::max(ascent, i->GetBaseline()); | 
|  | descent = std::max(descent, i->GetHeight() - i->GetBaseline()); | 
|  | } | 
|  | common_height_ = ascent + descent; | 
|  | common_baseline_ = ascent; | 
|  | } | 
|  |  | 
|  | void FontListImpl::CacheFontStyleAndSize() const { | 
|  | if (!fonts_.empty()) { | 
|  | font_style_ = fonts_[0].GetStyle(); | 
|  | font_size_ = fonts_[0].GetFontSize(); | 
|  | font_weight_ = fonts_[0].GetWeight(); | 
|  | } else { | 
|  | std::vector<std::string> font_names; | 
|  | CHECK(FontList::ParseDescription(font_description_string_, &font_names, | 
|  | &font_style_, &font_size_, &font_weight_)); | 
|  | } | 
|  | } | 
|  |  | 
|  | }  // namespace gfx |