chromium/content/child/dwrite_font_proxy/font_fallback_win_unittest.cc

// Copyright 2016 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "content/child/dwrite_font_proxy/font_fallback_win.h"

#include <dwrite.h>
#include <shlobj.h>
#include <wrl.h>

#include <memory>
#include <vector>

#include "base/memory/ref_counted.h"
#include "base/test/task_environment.h"
#include "content/child/dwrite_font_proxy/dwrite_font_proxy_win.h"
#include "content/test/dwrite_font_fake_sender_win.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/gfx/win/text_analysis_source.h"

namespace mswr = Microsoft::WRL;

namespace content {

namespace {

class FontFallbackUnitTest : public testing::Test {
 public:
  FontFallbackUnitTest() {
    DWriteCreateFactory(DWRITE_FACTORY_TYPE_SHARED, __uuidof(IDWriteFactory),
                        &factory_);

    factory_->CreateNumberSubstitution(DWRITE_NUMBER_SUBSTITUTION_METHOD_NONE,
                                       L"en-us", true /* ignoreUserOverride */,
                                       &number_substitution_);

    std::vector<wchar_t> font_path;
    font_path.resize(MAX_PATH);
    SHGetSpecialFolderPath(nullptr /* hwndOwner - reserved */, font_path.data(),
                           CSIDL_FONTS, FALSE /* fCreate*/);
    base::FilePath segoe_path = base::FilePath(std::wstring(font_path.data()))
                                    .Append(L"\\seguisym.ttf");

    fake_collection_ = std::make_unique<FakeFontCollection>();
    fake_collection_->AddFont(u"Segoe UI Symbol")
        .AddFamilyName(u"en-us", u"Segoe UI Symbol")
        .AddFilePath(segoe_path);

    DWriteFontCollectionProxy::Create(&collection_, factory_.Get(),
                                      fake_collection_->CreateRemote());
  }

  base::test::TaskEnvironment task_environment;
  std::unique_ptr<FakeFontCollection> fake_collection_;
  mswr::ComPtr<IDWriteFactory> factory_;
  mswr::ComPtr<DWriteFontCollectionProxy> collection_;
  mswr::ComPtr<IDWriteNumberSubstitution> number_substitution_;
};

TEST_F(FontFallbackUnitTest, MapCharacters) {
  mswr::ComPtr<FontFallback> fallback;
  FontFallback::Create(&fallback, collection_.Get());

  mswr::ComPtr<IDWriteFont> font;
  UINT32 mapped_length = 0;
  float scale = 0.0;

  mswr::ComPtr<IDWriteTextAnalysisSource> text;
  gfx::win::TextAnalysisSource::Create(&text, L"hello", L"en-us",
                                       number_substitution_.Get(),
                                       DWRITE_READING_DIRECTION_LEFT_TO_RIGHT);
  fallback->MapCharacters(text.Get(), 0, 5, nullptr, nullptr,
                          DWRITE_FONT_WEIGHT_NORMAL, DWRITE_FONT_STYLE_NORMAL,
                          DWRITE_FONT_STRETCH_NORMAL, &mapped_length, &font,
                          &scale);

  EXPECT_EQ(1u, mapped_length);  // The fake sender only maps one character
  EXPECT_NE(nullptr, font.Get());
}

TEST_F(FontFallbackUnitTest, DuplicateCallsShouldNotRepeatIPC) {
  mswr::ComPtr<FontFallback> fallback;
  FontFallback::Create(&fallback, collection_.Get());

  mswr::ComPtr<IDWriteFont> font;
  UINT32 mapped_length = 0;
  float scale = 0.0;

  mswr::ComPtr<IDWriteTextAnalysisSource> text;
  gfx::win::TextAnalysisSource::Create(&text, L"hello", L"en-us",
                                       number_substitution_.Get(),
                                       DWRITE_READING_DIRECTION_LEFT_TO_RIGHT);
  fallback->MapCharacters(text.Get(), 0, 5, nullptr, nullptr,
                          DWRITE_FONT_WEIGHT_NORMAL, DWRITE_FONT_STYLE_NORMAL,
                          DWRITE_FONT_STRETCH_NORMAL, &mapped_length, &font,
                          &scale);
  mapped_length = 0;
  fallback->MapCharacters(text.Get(), 0, 5, nullptr, nullptr,
                          DWRITE_FONT_WEIGHT_NORMAL, DWRITE_FONT_STYLE_NORMAL,
                          DWRITE_FONT_STRETCH_NORMAL, &mapped_length, &font,
                          &scale);

  EXPECT_EQ(3u, fake_collection_->MessageCount());

  EXPECT_EQ(FakeFontCollection::MessageType::kMapCharacters,
            fake_collection_->GetMessageType(0));
  EXPECT_NE(FakeFontCollection::MessageType::kMapCharacters,
            fake_collection_->GetMessageType(1));
  EXPECT_NE(FakeFontCollection::MessageType::kMapCharacters,
            fake_collection_->GetMessageType(2));
  EXPECT_EQ(5u, mapped_length);
}

TEST_F(FontFallbackUnitTest, DifferentFamilyShouldNotReuseCache) {
  mswr::ComPtr<FontFallback> fallback;
  FontFallback::Create(&fallback, collection_.Get());

  mswr::ComPtr<IDWriteFont> font;
  UINT32 mapped_length = 0;
  float scale = 0.0;

  mswr::ComPtr<IDWriteTextAnalysisSource> text;
  gfx::win::TextAnalysisSource::Create(&text, L"hello", L"en-us",
                                       number_substitution_.Get(),
                                       DWRITE_READING_DIRECTION_LEFT_TO_RIGHT);
  fallback->MapCharacters(text.Get(), 0, 5, nullptr, L"font1",
                          DWRITE_FONT_WEIGHT_NORMAL, DWRITE_FONT_STYLE_NORMAL,
                          DWRITE_FONT_STRETCH_NORMAL, &mapped_length, &font,
                          &scale);
  fallback->MapCharacters(text.Get(), 0, 5, nullptr, L"font2",
                          DWRITE_FONT_WEIGHT_NORMAL, DWRITE_FONT_STYLE_NORMAL,
                          DWRITE_FONT_STRETCH_NORMAL, &mapped_length, &font,
                          &scale);

  EXPECT_EQ(4u, fake_collection_->MessageCount());
  EXPECT_EQ(FakeFontCollection::MessageType::kMapCharacters,
            fake_collection_->GetMessageType(0));
  EXPECT_NE(FakeFontCollection::MessageType::kMapCharacters,
            fake_collection_->GetMessageType(1));
  EXPECT_NE(FakeFontCollection::MessageType::kMapCharacters,
            fake_collection_->GetMessageType(2));
  EXPECT_EQ(FakeFontCollection::MessageType::kMapCharacters,
            fake_collection_->GetMessageType(3));
}

TEST_F(FontFallbackUnitTest, CacheMissShouldRepeatIPC) {
  mswr::ComPtr<FontFallback> fallback;
  FontFallback::Create(&fallback, collection_.Get());

  mswr::ComPtr<IDWriteFont> font;
  UINT32 mapped_length = 0;
  float scale = 0.0;

  mswr::ComPtr<IDWriteTextAnalysisSource> text;
  gfx::win::TextAnalysisSource::Create(&text, L"hello", L"en-us",
                                       number_substitution_.Get(),
                                       DWRITE_READING_DIRECTION_LEFT_TO_RIGHT);
  mswr::ComPtr<IDWriteTextAnalysisSource> unmappable_text;
  gfx::win::TextAnalysisSource::Create(&unmappable_text, L"\uffff", L"en-us",
                                       number_substitution_.Get(),
                                       DWRITE_READING_DIRECTION_LEFT_TO_RIGHT);
  fallback->MapCharacters(text.Get(), 0, 5, nullptr, nullptr,
                          DWRITE_FONT_WEIGHT_NORMAL, DWRITE_FONT_STYLE_NORMAL,
                          DWRITE_FONT_STRETCH_NORMAL, &mapped_length, &font,
                          &scale);
  fallback->MapCharacters(unmappable_text.Get(), 0, 1, nullptr, nullptr,
                          DWRITE_FONT_WEIGHT_NORMAL, DWRITE_FONT_STYLE_NORMAL,
                          DWRITE_FONT_STRETCH_NORMAL, &mapped_length, &font,
                          &scale);

  EXPECT_EQ(4u, fake_collection_->MessageCount());
  EXPECT_EQ(FakeFontCollection::MessageType::kMapCharacters,
            fake_collection_->GetMessageType(0));
  EXPECT_NE(FakeFontCollection::MessageType::kMapCharacters,
            fake_collection_->GetMessageType(1));
  EXPECT_NE(FakeFontCollection::MessageType::kMapCharacters,
            fake_collection_->GetMessageType(2));
  EXPECT_EQ(FakeFontCollection::MessageType::kMapCharacters,
            fake_collection_->GetMessageType(3));
}

TEST_F(FontFallbackUnitTest, SurrogatePairCacheHit) {
  mswr::ComPtr<FontFallback> fallback;
  FontFallback::Create(&fallback, collection_.Get());

  mswr::ComPtr<IDWriteFont> font;
  UINT32 mapped_length = 0;
  float scale = 0.0;

  mswr::ComPtr<IDWriteTextAnalysisSource> text;
  gfx::win::TextAnalysisSource::Create(&text, L"hello", L"en-us",
                                       number_substitution_.Get(),
                                       DWRITE_READING_DIRECTION_LEFT_TO_RIGHT);
  mswr::ComPtr<IDWriteTextAnalysisSource> surrogate_pair_text;
  gfx::win::TextAnalysisSource::Create(&surrogate_pair_text, L"\U0001d300",
                                       L"en-us", number_substitution_.Get(),
                                       DWRITE_READING_DIRECTION_LEFT_TO_RIGHT);
  fallback->MapCharacters(text.Get(), 0, 5, nullptr, nullptr,
                          DWRITE_FONT_WEIGHT_NORMAL, DWRITE_FONT_STYLE_NORMAL,
                          DWRITE_FONT_STRETCH_NORMAL, &mapped_length, &font,
                          &scale);
  fallback->MapCharacters(surrogate_pair_text.Get(), 0, 2, nullptr, nullptr,
                          DWRITE_FONT_WEIGHT_NORMAL, DWRITE_FONT_STYLE_NORMAL,
                          DWRITE_FONT_STRETCH_NORMAL, &mapped_length, &font,
                          &scale);

  EXPECT_EQ(3u, fake_collection_->MessageCount());
  EXPECT_EQ(FakeFontCollection::MessageType::kMapCharacters,
            fake_collection_->GetMessageType(0));
  EXPECT_NE(FakeFontCollection::MessageType::kMapCharacters,
            fake_collection_->GetMessageType(1));
  EXPECT_NE(FakeFontCollection::MessageType::kMapCharacters,
            fake_collection_->GetMessageType(2));
  EXPECT_EQ(2u, mapped_length);
}

}  // namespace
}  // namespace content