blob: 2fb66bab887204a5271ff4d591aeb13cb5d3153e [file] [log] [blame]
// Copyright (c) 2015 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 "content/child/dwrite_font_proxy/dwrite_font_proxy_win.h"
#include <dwrite.h>
#include <shlobj.h>
#include <wrl.h>
#include <memory>
#include "base/memory/ref_counted.h"
#include "content/common/dwrite_font_proxy_messages.h"
#include "content/common/view_messages.h"
#include "content/test/dwrite_font_fake_sender_win.h"
#include "ipc/ipc_message_macros.h"
#include "ipc/ipc_sender.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace mswr = Microsoft::WRL;
namespace content {
namespace {
class DWriteFontProxyUnitTest : public testing::Test {
public:
DWriteFontProxyUnitTest() {
fake_collection_ = new FakeFontCollection();
SetupFonts(fake_collection_.get());
mswr::MakeAndInitialize<DWriteFontCollectionProxy>(
&collection_, factory.Get(), fake_collection_->GetTrackingSender());
}
~DWriteFontProxyUnitTest() override {
if (collection_)
collection_->Unregister();
}
static void SetupFonts(FakeFontCollection* fonts) {
fonts->AddFont(L"Aardvark")
.AddFamilyName(L"en-us", L"Aardvark")
.AddFamilyName(L"de-de", L"Erdferkel")
.AddFilePath(L"X:\\Nonexistent\\Folder\\Aardvark.ttf");
FakeFont& arial =
fonts->AddFont(L"Arial").AddFamilyName(L"en-us", L"Arial");
for (auto& path : arial_font_files)
arial.AddFilePath(path);
fonts->AddFont(L"Times New Roman")
.AddFamilyName(L"en-us", L"Times New Roman")
.AddFilePath(L"X:\\Nonexistent\\Folder\\Times.ttf");
}
static void SetUpTestCase() {
DWriteCreateFactory(DWRITE_FACTORY_TYPE_SHARED, __uuidof(IDWriteFactory),
&factory);
std::vector<base::char16> font_path;
font_path.resize(MAX_PATH);
SHGetSpecialFolderPath(nullptr /* hwndOwner - reserved */, font_path.data(),
CSIDL_FONTS, FALSE /* fCreate*/);
base::string16 arial;
arial.append(font_path.data()).append(L"\\arial.ttf");
base::string16 arialbd;
arialbd.append(font_path.data()).append(L"\\arialbd.ttf");
arial_font_files.push_back(arial);
arial_font_files.push_back(arialbd);
}
protected:
scoped_refptr<FakeFontCollection> fake_collection_;
mswr::ComPtr<DWriteFontCollectionProxy> collection_;
static std::vector<base::string16> arial_font_files;
static mswr::ComPtr<IDWriteFactory> factory;
};
std::vector<base::string16> DWriteFontProxyUnitTest::arial_font_files;
mswr::ComPtr<IDWriteFactory> DWriteFontProxyUnitTest::factory;
TEST_F(DWriteFontProxyUnitTest, GetFontFamilyCount) {
UINT32 family_count = collection_->GetFontFamilyCount();
EXPECT_EQ(3u, family_count);
ASSERT_EQ(1u, fake_collection_->MessageCount());
EXPECT_EQ(DWriteFontProxyMsg_GetFamilyCount::ID,
fake_collection_->GetMessage(0)->type());
// Calling again should not cause another message to be sent.
family_count = collection_->GetFontFamilyCount();
EXPECT_EQ(3u, family_count);
ASSERT_EQ(1u, fake_collection_->MessageCount());
}
TEST_F(DWriteFontProxyUnitTest, FindFamilyNameShouldFindFamily) {
HRESULT hr;
UINT32 index = UINT_MAX;
BOOL exists = FALSE;
hr = collection_->FindFamilyName(L"Arial", &index, &exists);
EXPECT_EQ(S_OK, hr);
EXPECT_EQ(1u, index);
EXPECT_TRUE(exists);
ASSERT_EQ(2u, fake_collection_->MessageCount());
EXPECT_EQ(DWriteFontProxyMsg_FindFamily::ID,
fake_collection_->GetMessage(0)->type());
EXPECT_EQ(DWriteFontProxyMsg_GetFamilyCount::ID,
fake_collection_->GetMessage(1)->type());
}
TEST_F(DWriteFontProxyUnitTest, FindFamilyNameShouldReturnUINTMAXWhenNotFound) {
HRESULT hr;
UINT32 index = UINT_MAX;
BOOL exists = FALSE;
hr = collection_->FindFamilyName(L"Not a font", &index, &exists);
EXPECT_EQ(S_OK, hr);
EXPECT_EQ(UINT32_MAX, index);
EXPECT_FALSE(exists);
ASSERT_EQ(1u, fake_collection_->MessageCount());
EXPECT_EQ(DWriteFontProxyMsg_FindFamily::ID,
fake_collection_->GetMessage(0)->type());
}
TEST_F(DWriteFontProxyUnitTest, FindFamilyNameShouldNotSendDuplicateIPC) {
HRESULT hr;
UINT32 index = UINT_MAX;
BOOL exists = FALSE;
hr = collection_->FindFamilyName(L"Arial", &index, &exists);
ASSERT_EQ(S_OK, hr);
ASSERT_EQ(2u, fake_collection_->MessageCount());
hr = collection_->FindFamilyName(L"Arial", &index, &exists);
EXPECT_EQ(S_OK, hr);
EXPECT_EQ(2u, fake_collection_->MessageCount());
}
TEST_F(DWriteFontProxyUnitTest, GetFontFamilyShouldCreateFamily) {
HRESULT hr;
UINT32 index = UINT_MAX;
BOOL exists = FALSE;
hr = collection_->FindFamilyName(L"Arial", &index, &exists);
ASSERT_EQ(2u, fake_collection_->MessageCount());
mswr::ComPtr<IDWriteFontFamily> family;
hr = collection_->GetFontFamily(2, &family);
EXPECT_EQ(S_OK, hr);
EXPECT_EQ(2u, fake_collection_->MessageCount());
EXPECT_NE(nullptr, family.Get());
}
void CheckLocale(const base::string16& locale_name,
const base::string16& expected_value,
IDWriteLocalizedStrings* strings) {
UINT32 locale_index = 0;
BOOL locale_exists = FALSE;
strings->FindLocaleName(locale_name.data(), &locale_index, &locale_exists);
EXPECT_TRUE(locale_exists);
UINT32 name_length = 0;
strings->GetLocaleNameLength(locale_index, &name_length);
EXPECT_EQ(locale_name.size(), name_length);
base::string16 actual_name;
name_length++;
actual_name.resize(name_length);
strings->GetLocaleName(locale_index, const_cast<WCHAR*>(actual_name.data()),
name_length);
EXPECT_STREQ(locale_name.c_str(), actual_name.c_str());
UINT32 string_length = 0;
strings->GetStringLength(locale_index, &string_length);
EXPECT_EQ(expected_value.size(), string_length);
base::string16 actual_value;
string_length++;
actual_value.resize(string_length);
strings->GetString(locale_index, const_cast<WCHAR*>(actual_value.data()),
string_length);
EXPECT_STREQ(expected_value.c_str(), actual_value.c_str());
}
TEST_F(DWriteFontProxyUnitTest, GetFamilyNames) {
HRESULT hr;
UINT32 index = UINT_MAX;
BOOL exists = FALSE;
hr = collection_->FindFamilyName(L"Aardvark", &index, &exists);
ASSERT_EQ(2u, fake_collection_->MessageCount());
mswr::ComPtr<IDWriteFontFamily> family;
hr = collection_->GetFontFamily(index, &family);
EXPECT_EQ(S_OK, hr);
EXPECT_EQ(2u, fake_collection_->MessageCount());
mswr::ComPtr<IDWriteLocalizedStrings> names;
hr = family->GetFamilyNames(&names);
EXPECT_EQ(S_OK, hr);
EXPECT_EQ(3u, fake_collection_->MessageCount());
EXPECT_EQ(DWriteFontProxyMsg_GetFamilyNames::ID,
fake_collection_->GetMessage(2)->type());
EXPECT_EQ(2u, names->GetCount());
UINT32 locale_index = 0;
BOOL locale_exists = FALSE;
hr = names->FindLocaleName(L"fr-fr", &locale_index, &locale_exists);
EXPECT_EQ(S_OK, hr);
EXPECT_FALSE(locale_exists);
CheckLocale(L"en-us", L"Aardvark", names.Get());
CheckLocale(L"de-de", L"Erdferkel", names.Get());
base::string16 unused;
unused.resize(25);
hr = names->GetLocaleName(15234, const_cast<WCHAR*>(unused.data()),
unused.size() - 1);
EXPECT_FALSE(SUCCEEDED(hr));
}
TEST_F(DWriteFontProxyUnitTest, GetFontCollection) {
HRESULT hr;
UINT32 index = UINT_MAX;
BOOL exists = FALSE;
hr = collection_->FindFamilyName(L"Arial", &index, &exists);
ASSERT_EQ(2u, fake_collection_->MessageCount());
mswr::ComPtr<IDWriteFontFamily> family;
hr = collection_->GetFontFamily(2, &family);
EXPECT_EQ(S_OK, hr);
EXPECT_EQ(2u, fake_collection_->MessageCount());
mswr::ComPtr<IDWriteFontCollection> returned_collection;
hr = family->GetFontCollection(&returned_collection);
EXPECT_EQ(S_OK, hr);
EXPECT_EQ(2u, fake_collection_->MessageCount());
EXPECT_EQ(collection_.Get(), returned_collection.Get());
}
TEST_F(DWriteFontProxyUnitTest, GetFamilyNamesShouldNotIPCAfterLoadingFamily) {
HRESULT hr;
UINT32 index = UINT_MAX;
BOOL exists = FALSE;
collection_->FindFamilyName(L"Arial", &index, &exists);
mswr::ComPtr<IDWriteFontFamily> family;
collection_->GetFontFamily(index, &family);
family->GetFontCount();
EXPECT_EQ(3u, fake_collection_->MessageCount());
mswr::ComPtr<IDWriteLocalizedStrings> names;
hr = family->GetFamilyNames(&names);
EXPECT_EQ(S_OK, hr);
EXPECT_EQ(3u, fake_collection_->MessageCount());
}
TEST_F(DWriteFontProxyUnitTest,
GetFontFamilyShouldNotCreateFamilyWhenIndexIsInvalid) {
HRESULT hr;
UINT32 index = UINT_MAX;
BOOL exists = FALSE;
hr = collection_->FindFamilyName(L"Arial", &index, &exists);
ASSERT_EQ(2u, fake_collection_->MessageCount());
mswr::ComPtr<IDWriteFontFamily> family;
hr = collection_->GetFontFamily(1654, &family);
EXPECT_FALSE(SUCCEEDED(hr));
EXPECT_EQ(2u, fake_collection_->MessageCount());
}
TEST_F(DWriteFontProxyUnitTest, LoadingFontFamily) {
HRESULT hr;
UINT32 index = UINT_MAX;
BOOL exists = FALSE;
collection_->FindFamilyName(L"Arial", &index, &exists);
mswr::ComPtr<IDWriteFontFamily> family;
collection_->GetFontFamily(index, &family);
ASSERT_EQ(2u, fake_collection_->MessageCount());
UINT32 font_count = family->GetFontCount();
EXPECT_LT(0u, font_count);
EXPECT_EQ(3u, fake_collection_->MessageCount());
EXPECT_EQ(DWriteFontProxyMsg_GetFontFiles::ID,
fake_collection_->GetMessage(2)->type());
mswr::ComPtr<IDWriteFont> font;
hr = family->GetFirstMatchingFont(DWRITE_FONT_WEIGHT_NORMAL,
DWRITE_FONT_STRETCH_NORMAL,
DWRITE_FONT_STYLE_NORMAL, &font);
EXPECT_EQ(S_OK, hr);
EXPECT_EQ(3u, fake_collection_->MessageCount());
mswr::ComPtr<IDWriteFont> font2;
hr = family->GetFont(0, &font2);
EXPECT_EQ(S_OK, hr);
EXPECT_EQ(3u, fake_collection_->MessageCount());
mswr::ComPtr<IDWriteFontList> matching_fonts;
hr = family->GetMatchingFonts(DWRITE_FONT_WEIGHT_NORMAL,
DWRITE_FONT_STRETCH_NORMAL,
DWRITE_FONT_STYLE_NORMAL, &matching_fonts);
EXPECT_EQ(S_OK, hr);
EXPECT_EQ(3u, fake_collection_->MessageCount());
EXPECT_NE(nullptr, matching_fonts.Get());
}
TEST_F(DWriteFontProxyUnitTest, GetFontFromFontFaceShouldFindFont) {
HRESULT hr;
UINT32 index = UINT_MAX;
BOOL exists = FALSE;
collection_->FindFamilyName(L"Arial", &index, &exists);
mswr::ComPtr<IDWriteFontFamily> family;
collection_->GetFontFamily(index, &family);
mswr::ComPtr<IDWriteFont> font;
family->GetFirstMatchingFont(DWRITE_FONT_WEIGHT_NORMAL,
DWRITE_FONT_STRETCH_NORMAL,
DWRITE_FONT_STYLE_NORMAL, &font);
mswr::ComPtr<IDWriteFontFace> font_face;
hr = font->CreateFontFace(&font_face);
ASSERT_TRUE(SUCCEEDED(hr));
ASSERT_EQ(3u, fake_collection_->MessageCount());
mswr::ComPtr<IDWriteFont> found_font;
collection_->GetFontFromFontFace(font_face.Get(), &found_font);
EXPECT_NE(nullptr, found_font.Get());
EXPECT_EQ(3u, fake_collection_->MessageCount());
}
TEST_F(DWriteFontProxyUnitTest, TestCustomFontFiles) {
scoped_refptr<FakeFontCollection> fonts = new FakeFontCollection();
FakeFont& arial = fonts->AddFont(L"Arial").AddFamilyName(L"en-us", L"Arial");
for (auto& path : arial_font_files) {
base::File file(base::FilePath(path), base::File::FLAG_OPEN |
base::File::FLAG_READ |
base::File::FLAG_EXCLUSIVE_WRITE);
arial.AddFileHandle(IPC::TakePlatformFileForTransit(std::move(file)));
}
mswr::ComPtr<DWriteFontCollectionProxy> collection;
mswr::MakeAndInitialize<DWriteFontCollectionProxy>(
&collection, factory.Get(), fonts->GetTrackingSender());
// Check that we can get the font family and match a font.
UINT32 index = UINT_MAX;
BOOL exists = FALSE;
collection->FindFamilyName(L"Arial", &index, &exists);
mswr::ComPtr<IDWriteFontFamily> family;
collection->GetFontFamily(index, &family);
mswr::ComPtr<IDWriteFont> font;
HRESULT hr = family->GetFirstMatchingFont(DWRITE_FONT_WEIGHT_NORMAL,
DWRITE_FONT_STRETCH_NORMAL,
DWRITE_FONT_STYLE_NORMAL, &font);
EXPECT_TRUE(SUCCEEDED(hr));
EXPECT_NE(nullptr, font.Get());
mswr::ComPtr<IDWriteFontFace> font_face;
hr = font->CreateFontFace(&font_face);
EXPECT_TRUE(SUCCEEDED(hr));
}
} // namespace
} // namespace content