diff options
author | sky@chromium.org <sky@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2009-09-14 19:26:49 +0000 |
---|---|---|
committer | sky@chromium.org <sky@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2009-09-14 19:26:49 +0000 |
commit | 5e70f14644629b7f9fe9126553758aed25e17128 (patch) | |
tree | 65bd07608366d1f3004535e2ca91a9089327ed92 /app/gfx | |
parent | 1bf956b6f611bd3e116489bfdef7ac9048db0308 (diff) | |
download | chromium_src-5e70f14644629b7f9fe9126553758aed25e17128.zip chromium_src-5e70f14644629b7f9fe9126553758aed25e17128.tar.gz chromium_src-5e70f14644629b7f9fe9126553758aed25e17128.tar.bz2 |
Reverts linux font change again. Slow down isn't as bad as before. But
still not ideal.
BUG=none
TEST=none
TBR=agl
Review URL: http://codereview.chromium.org/194105
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@26141 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'app/gfx')
-rw-r--r-- | app/gfx/canvas_linux.cc | 4 | ||||
-rw-r--r-- | app/gfx/font.h | 111 | ||||
-rw-r--r-- | app/gfx/font_gtk.cc | 248 | ||||
-rw-r--r-- | app/gfx/font_skia.cc | 160 | ||||
-rw-r--r-- | app/gfx/font_unittest.cc | 7 |
5 files changed, 269 insertions, 261 deletions
diff --git a/app/gfx/canvas_linux.cc b/app/gfx/canvas_linux.cc index 1614f44..7ed1904 100644 --- a/app/gfx/canvas_linux.cc +++ b/app/gfx/canvas_linux.cc @@ -130,7 +130,9 @@ static void SetupPangoLayout(PangoLayout* layout, PANGO_WRAP_WORD_CHAR : PANGO_WRAP_WORD); } - pango_layout_set_font_description(layout, font.nativeFont()); + PangoFontDescription* desc = gfx::Font::PangoFontFromGfxFont(font); + pango_layout_set_font_description(layout, desc); + pango_font_description_free(desc); } // static diff --git a/app/gfx/font.h b/app/gfx/font.h index 60e94d2..e53bd79 100644 --- a/app/gfx/font.h +++ b/app/gfx/font.h @@ -9,12 +9,12 @@ #include <string> -#if defined(OS_LINUX) -#include <list> -#endif - #if defined(OS_WIN) typedef struct HFONT__* HFONT; +#elif defined(OS_LINUX) +#include "third_party/skia/include/core/SkRefCnt.h" +class SkPaint; +class SkTypeface; #endif #if defined(OS_WIN) @@ -28,7 +28,8 @@ class NSFont; typedef NSFont* NativeFont; #elif defined(OS_LINUX) typedef struct _PangoFontDescription PangoFontDescription; -typedef PangoFontDescription* NativeFont; +class SkTypeface; +typedef SkTypeface* NativeFont; #else // null port. #error No known OS defined #endif @@ -101,10 +102,6 @@ class Font { // Font Size. int FontSize(); - // Returns a handle to the native font. - // NOTE: on linux this returns the PangoFontDescription* being held by this - // object. You should not modify or free it. If you need to use it, make a - // copy of it by way of pango_font_description_copy(nativeFont()). NativeFont nativeFont() const; // Creates a font with the default name and style. @@ -129,6 +126,16 @@ class Font { } #elif defined(OS_LINUX) static Font CreateFont(PangoFontDescription* desc); + // We need a copy constructor and assignment operator to deal with + // the Skia reference counting. + Font(const Font& other); + Font& operator=(const Font& other); + // Setup a Skia context to use the current typeface + void PaintSetup(SkPaint* paint) const; + + // Converts |gfx_font| to a new pango font. Free the returned font with + // pango_font_description_free(). + static PangoFontDescription* PangoFontFromGfxFont(const gfx::Font& gfx_font); #endif private: @@ -189,74 +196,32 @@ class Font { // Indirect reference to the HFontRef, which references the underlying HFONT. scoped_refptr<HFontRef> font_ref_; #elif defined(OS_LINUX) - // Used internally on Linux to cache information about the font. Each - // Font maintains a reference to a PangoFontRef. Coping a Font ups the - // refcount of the corresponding PangoFontRef. - // - // As obtaining metrics (height, ascent, average char width) is expensive, - // the metrics are only obtained as needed. Additionally PangoFontRef - // maintains a static cache of PangoFontRefs. When a PangoFontRef is needed - // the cache is checked first. - class PangoFontRef : public base::RefCounted<PangoFontRef> { - public: - ~PangoFontRef(); - - // Creates or returns a cached PangoFontRef of the given family, size and - // style. - static PangoFontRef* Create(PangoFontDescription* pfd, - const std::wstring& family, - int size, - int style); - - PangoFontDescription* pfd() const { return pfd_; } - const std::wstring& family() const { return family_; } - int size() const { return size_; } - int style() const { return style_; } - int GetHeight() const; - int GetAscent() const; - int GetAveCharWidth() const; - - private: - typedef std::list<PangoFontRef*> Cache; - - PangoFontRef(PangoFontDescription* pfd, - const std::wstring& family, - int size, - int style); - - void CalculateMetricsIfNecessary() const; - - // Returns the cache. - static Cache* GetCache(); - - // Adds |ref| to the cache, removing an existing PangoFontRef if there are - // too many. - static void AddToCache(PangoFontRef* ref); - - PangoFontDescription* pfd_; - const std::wstring family_; - const int size_; - const int style_; - - // Metrics related members. As these are expensive to calculate they are - // calculated the first time requested. - mutable int height_; - mutable int ascent_; - mutable int ave_char_width_; - - // Have the metrics related members been determined yet (height_, ascent_ - // ave_char_width_)? - mutable bool calculated_metrics_; - - DISALLOW_COPY_AND_ASSIGN(PangoFontRef); - }; - - explicit Font(PangoFontDescription* pfd); + explicit Font(SkTypeface* typeface, const std::wstring& name, + int size, int style); + // Calculate and cache the font metrics. + void calculateMetrics(); + // Make |this| a copy of |other|. + void CopyFont(const Font& other); // The default font, used for the default constructor. static Font* default_font_; - scoped_refptr<PangoFontRef> font_ref_; + // These two both point to the same SkTypeface. We use the SkAutoUnref to + // handle the reference counting, but without @typeface_ we would have to + // cast the SkRefCnt from @typeface_helper_ every time. + scoped_ptr<SkAutoUnref> typeface_helper_; + SkTypeface *typeface_; + + // Additional information about the face + // Skia actually expects a family name and not a font name. + std::wstring font_family_; + int font_size_; + int style_; + + // Cached metrics, generated at construction + int height_; + int ascent_; + int avg_width_; #elif defined(OS_MACOSX) explicit Font(const std::wstring& font_name, int font_size, int style); diff --git a/app/gfx/font_gtk.cc b/app/gfx/font_gtk.cc index 897e8f0..244810a 100644 --- a/app/gfx/font_gtk.cc +++ b/app/gfx/font_gtk.cc @@ -7,87 +7,56 @@ #include <fontconfig/fontconfig.h> #include <gtk/gtk.h> -#include "app/gfx/canvas.h" #include "base/string_util.h" namespace gfx { -namespace { - -// Returns a PangoContext that is used to get metrics. The returned context -// should never be freed. -PangoContext* get_context() { - static PangoContext* context = NULL; - if (!context) { - context = gdk_pango_context_get_for_screen(gdk_screen_get_default()); - pango_context_set_language(context, gtk_get_default_language()); - } - return context; -} +Font* Font::default_font_ = NULL; +// Find the best match font for |family_name| in the same way as Skia +// to make sure CreateFont() successfully creates a default font. In +// Skia, it only checks the best match font. If it failed to find +// one, SkTypeface will be NULL for that font family. It eventually +// causes a segfault. For example, family_name = "Sans" and system +// may have various fonts. The first font family in FcPattern will be +// "DejaVu Sans" but a font family returned by FcFontMatch will be "VL +// PGothic". In this case, SkTypeface for "Sans" returns NULL even if +// the system has a font for "Sans" font family. See FontMatch() in +// skia/ports/SkFontHost_fontconfig.cpp for more detail. +static std::wstring FindBestMatchFontFamilyName(const char* family_name) { + FcPattern* pattern = FcPatternCreate(); + FcValue fcvalue; + fcvalue.type = FcTypeString; + char* family_name_copy = strdup(family_name); + fcvalue.u.s = reinterpret_cast<FcChar8*>(family_name_copy); + FcPatternAdd(pattern, FC_FAMILY, fcvalue, 0); + FcConfigSubstitute(0, pattern, FcMatchPattern); + FcDefaultSubstitute(pattern); + FcResult result; + FcPattern* match = FcFontMatch(0, pattern, &result); + DCHECK(match) << "Could not find font: " << family_name; + FcChar8* match_family; + FcPatternGetString(match, FC_FAMILY, 0, &match_family); + + std::wstring font_family = UTF8ToWide( + reinterpret_cast<char*>(match_family)); + FcPatternDestroy(match); + FcPatternDestroy(pattern); + free(family_name_copy); + return font_family; } -Font* Font::default_font_ = NULL; - // static Font Font::CreateFont(PangoFontDescription* desc) { - return Font(desc); -} + gint size = pango_font_description_get_size(desc); + const char* family_name = pango_font_description_get_family(desc); -Font Font::DeriveFont(int size_delta, int style) const { - // If the delta is negative, if must not push the size below 1 - if (size_delta < 0) - DCHECK_LT(-size_delta, font_ref_->size()); - - PangoFontDescription* pfd = pango_font_description_copy(nativeFont()); - pango_font_description_set_size( - pfd, (font_ref_->size() + size_delta) * PANGO_SCALE); - pango_font_description_set_style( - pfd, style & ITALIC ? PANGO_STYLE_ITALIC : PANGO_STYLE_NORMAL); - pango_font_description_set_weight( - pfd, style & BOLD ? PANGO_WEIGHT_BOLD : PANGO_WEIGHT_NORMAL); - Font font(pfd); - pango_font_description_free(pfd); - return font; -} - -int Font::height() const { - return font_ref_->GetHeight(); -} + // Find best match font for |family_name| to make sure we can get + // a SkTypeface for the default font. + // TODO(agl): remove this. + std::wstring font_family = FindBestMatchFontFamilyName(family_name); -int Font::baseline() const { - return font_ref_->GetAscent(); -} - -int Font::ave_char_width() const { - return font_ref_->GetAveCharWidth(); -} - -int Font::GetStringWidth(const std::wstring& text) const { - int width = 0, height = 0; - - Canvas::SizeStringInt(text, *this, &width, &height, 0); - return width; -} - -int Font::GetExpectedTextWidth(int length) const { - return length * font_ref_->GetAveCharWidth(); -} - -int Font::style() const { - return font_ref_->style(); -} - -std::wstring Font::FontName() { - return font_ref_->family(); -} - -int Font::FontSize() { - return font_ref_->size(); -} - -PangoFontDescription* Font::nativeFont() const { - return font_ref_->pfd(); + return Font(CreateFont(font_family, size / PANGO_SCALE)); } // Get the default gtk system font (name and size). @@ -106,130 +75,41 @@ Font::Font() { PangoFontDescription* desc = pango_font_description_from_string(font_name); - default_font_ = new Font(desc); + default_font_ = new Font(CreateFont(desc)); pango_font_description_free(desc); g_free(font_name); + + DCHECK(default_font_); } - *this = *default_font_; + CopyFont(*default_font_); } // static -Font Font::CreateFont(const std::wstring& font_family, int font_size) { - DCHECK_GT(font_size, 0); - +PangoFontDescription* Font::PangoFontFromGfxFont( + const gfx::Font& gfx_font) { + gfx::Font font = gfx_font; // Copy so we can call non-const methods. PangoFontDescription* pfd = pango_font_description_new(); - pango_font_description_set_family(pfd, WideToUTF8(font_family).c_str()); - pango_font_description_set_size(pfd, font_size * PANGO_SCALE); - pango_font_description_set_style(pfd, PANGO_STYLE_NORMAL); - pango_font_description_set_weight(pfd, PANGO_WEIGHT_NORMAL); - Font font(pfd); - pango_font_description_free(pfd); - return font; -} - -Font::Font(PangoFontDescription* desc) { - PangoContext* context = get_context(); - pango_context_set_font_description(context, desc); - int size = pango_font_description_get_size(desc) / PANGO_SCALE; - int style = 0; - if (pango_font_description_get_weight(desc) >= PANGO_WEIGHT_BOLD) - style |= BOLD; - if (pango_font_description_get_style(desc) == PANGO_STYLE_ITALIC) - style |= ITALIC; - std::wstring name(UTF8ToWide(pango_font_description_get_family(desc))); - font_ref_ = PangoFontRef::Create(desc, name, size, style); -} - -// PangoFontRef --------------------------------------------------------------- - -// Maximum number of PangoFontRefs cached. -static const size_t kMaxCacheSize = 20; - -// static -Font::PangoFontRef* Font::PangoFontRef::Create(PangoFontDescription* pfd, - const std::wstring& family, - int size, - int style) { - Cache* cache = GetCache(); - for (Cache::iterator i = cache->begin(); i != cache->end(); ++i) { - PangoFontRef* ref = *i; - if (ref->size() == size && ref->style() == style && - ref->family() == family) - return ref; + pango_font_description_set_family(pfd, WideToUTF8(font.FontName()).c_str()); + pango_font_description_set_size(pfd, font.FontSize() * PANGO_SCALE); + + switch (font.style()) { + case gfx::Font::NORMAL: + // Nothing to do, should already be PANGO_STYLE_NORMAL. + break; + case gfx::Font::BOLD: + pango_font_description_set_weight(pfd, PANGO_WEIGHT_BOLD); + break; + case gfx::Font::ITALIC: + pango_font_description_set_style(pfd, PANGO_STYLE_ITALIC); + break; + case gfx::Font::UNDERLINED: + // TODO(deanm): How to do underlined? Where do we use it? Probably have + // to paint it ourselves, see pango_font_metrics_get_underline_position. + break; } - PangoFontRef* ref = new PangoFontRef(pfd, family, size, style); - AddToCache(ref); - return ref; -} - -Font::PangoFontRef::~PangoFontRef() { - pango_font_description_free(pfd_); -} - -int Font::PangoFontRef::GetHeight() const { - CalculateMetricsIfNecessary(); - return height_; -} - -int Font::PangoFontRef::GetAscent() const { - CalculateMetricsIfNecessary(); - return ascent_; -} - -int Font::PangoFontRef::GetAveCharWidth() const { - CalculateMetricsIfNecessary(); - return ave_char_width_; -} - -Font::PangoFontRef::PangoFontRef(PangoFontDescription* pfd, - const std::wstring& family, - int size, - int style) - : pfd_(pango_font_description_copy(pfd)), - family_(family), - size_(size), - style_(style), - height_(0), - ascent_(0), - ave_char_width_(0), - calculated_metrics_(false) { -} - -void Font::PangoFontRef::CalculateMetricsIfNecessary() const { - if (calculated_metrics_) - return; - - calculated_metrics_ = true; - PangoContext* context = get_context(); - pango_context_set_font_description(context, pfd_); - PangoFontMetrics* metrics = pango_context_get_metrics(context, pfd_, NULL); - ascent_ = pango_font_metrics_get_ascent(metrics) / PANGO_SCALE; - height_ = ascent_ + pango_font_metrics_get_descent(metrics) / PANGO_SCALE; - ave_char_width_ = pango_font_metrics_get_approximate_char_width(metrics) / - PANGO_SCALE; - pango_font_metrics_unref(metrics); -} - -// static -Font::PangoFontRef::Cache* Font::PangoFontRef::GetCache() { - static Cache* cache = NULL; - if (!cache) - cache = new Cache(); - return cache; -} - -// static -void Font::PangoFontRef::AddToCache(PangoFontRef* ref) { - ref->AddRef(); - Cache* cache = GetCache(); - cache->push_back(ref); - while (cache->size() >= kMaxCacheSize) { - PangoFontRef* ref = cache->front(); - cache->pop_front(); - ref->Release(); - } + return pfd; } } // namespace gfx diff --git a/app/gfx/font_skia.cc b/app/gfx/font_skia.cc new file mode 100644 index 0000000..4308bf6 --- /dev/null +++ b/app/gfx/font_skia.cc @@ -0,0 +1,160 @@ +// Copyright (c) 2006-2008 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 "app/gfx/font.h" + +#include "app/gfx/canvas.h" + +#include "base/logging.h" +#include "base/sys_string_conversions.h" + +#include "third_party/skia/include/core/SkTypeface.h" +#include "third_party/skia/include/core/SkPaint.h" + +namespace gfx { + +Font::Font(const Font& other) { + CopyFont(other); +} + +Font& Font::operator=(const Font& other) { + CopyFont(other); + return *this; +} + +Font::Font(SkTypeface* tf, const std::wstring& font_family, int font_size, + int style) + : typeface_helper_(new SkAutoUnref(tf)), + typeface_(tf), + font_family_(font_family), + font_size_(font_size), + style_(style) { + tf->ref(); + calculateMetrics(); +} + +void Font::calculateMetrics() { + SkPaint paint; + SkPaint::FontMetrics metrics; + + PaintSetup(&paint); + paint.getFontMetrics(&metrics); + + ascent_ = SkScalarRound(-metrics.fAscent); + height_ = SkScalarRound(-metrics.fAscent + metrics.fDescent + + metrics.fLeading); + + if (metrics.fAvgCharWidth) { + avg_width_ = SkScalarRound(metrics.fAvgCharWidth); + } else { + static const char x_char = 'x'; + paint.setTextEncoding(SkPaint::kUTF8_TextEncoding); + SkScalar width = paint.measureText(&x_char, 1); + + avg_width_ = static_cast<int>(ceilf(SkScalarToFloat(width))); + } +} + +void Font::CopyFont(const Font& other) { + typeface_helper_.reset(new SkAutoUnref(other.typeface_)); + typeface_ = other.typeface_; + typeface_->ref(); + font_family_ = other.font_family_; + font_size_ = other.font_size_; + style_ = other.style_; + height_ = other.height_; + ascent_ = other.ascent_; + avg_width_ = other.avg_width_; +} + +int Font::height() const { + return height_; +} + +int Font::baseline() const { + return ascent_; +} + +int Font::ave_char_width() const { + return avg_width_; +} + +Font Font::CreateFont(const std::wstring& font_family, int font_size) { + DCHECK_GT(font_size, 0); + + SkTypeface* tf = SkTypeface::CreateFromName( + base::SysWideToUTF8(font_family).c_str(), SkTypeface::kNormal); + // Temporary CHECK for tracking down + // http://code.google.com/p/chromium/issues/detail?id=12530 + CHECK(tf) << "Could not find font: " << base::SysWideToUTF8(font_family); + SkAutoUnref tf_helper(tf); + + return Font(tf, font_family, font_size, NORMAL); +} + +Font Font::DeriveFont(int size_delta, int style) const { + // If the delta is negative, if must not push the size below 1 + if (size_delta < 0) { + DCHECK_LT(-size_delta, font_size_); + } + + if (style == style_) { + // Fast path, we just use the same typeface at a different size + return Font(typeface_, font_family_, font_size_ + size_delta, style_); + } + + // If the style has changed we may need to load a new face + int skstyle = SkTypeface::kNormal; + if (BOLD & style) + skstyle |= SkTypeface::kBold; + if (ITALIC & style) + skstyle |= SkTypeface::kItalic; + + SkTypeface* tf = SkTypeface::CreateFromName( + base::SysWideToUTF8(font_family_).c_str(), + static_cast<SkTypeface::Style>(skstyle)); + SkAutoUnref tf_helper(tf); + + return Font(tf, font_family_, font_size_ + size_delta, skstyle); +} + +void Font::PaintSetup(SkPaint* paint) const { + paint->setAntiAlias(false); + paint->setSubpixelText(false); + paint->setTextSize(SkFloatToScalar(font_size_)); + paint->setTypeface(typeface_); + paint->setFakeBoldText((BOLD & style_) && !typeface_->isBold()); + paint->setTextSkewX((ITALIC & style_) && !typeface_->isItalic() ? + -SK_Scalar1/4 : 0); +} + +int Font::GetStringWidth(const std::wstring& text) const { + int width = 0, height = 0; + + Canvas::SizeStringInt(text, *this, &width, &height, 0); + return width; +} + +int Font::GetExpectedTextWidth(int length) const { + return length * avg_width_; +} + + +int Font::style() const { + return style_; +} + +std::wstring Font::FontName() { + return font_family_; +} + +int Font::FontSize() { + return font_size_; +} + +NativeFont Font::nativeFont() const { + return typeface_; +} + +} // namespace gfx diff --git a/app/gfx/font_unittest.cc b/app/gfx/font_unittest.cc index d0f4fff..dc81ca8 100644 --- a/app/gfx/font_unittest.cc +++ b/app/gfx/font_unittest.cc @@ -10,7 +10,8 @@ namespace { using gfx::Font; -typedef testing::Test FontTest; +class FontTest : public testing::Test { +}; TEST_F(FontTest, LoadArial) { Font cf(Font::CreateFont(L"Arial", 16)); @@ -30,13 +31,13 @@ TEST_F(FontTest, LoadArialBold) { TEST_F(FontTest, Ascent) { Font cf(Font::CreateFont(L"Arial", 16)); ASSERT_GT(cf.baseline(), 2); - ASSERT_LT(cf.baseline(), 22); + ASSERT_LT(cf.baseline(), 20); } TEST_F(FontTest, Height) { Font cf(Font::CreateFont(L"Arial", 16)); ASSERT_GT(cf.baseline(), 2); - ASSERT_LT(cf.baseline(), 22); + ASSERT_LT(cf.baseline(), 20); } TEST_F(FontTest, AvgWidths) { |