mirror of
https://github.com/servo/servo.git
synced 2025-06-06 16:45:39 +00:00
- Better detect situations where emoji is necessary by looking ahead one character while laying out. This allow processing Unicode presentation selectors. When detecting emoji, put emoji fonts at the front of fallback lists for all platforms. This enables monochrome emoji on Windows. Full-color emoji on Windows probably needs full support for processing the COLR table and drawing separate glyph color layers. - Improve the font fallback list on FreeType platforms. Ideally, Servo would be able to look through the entire font list to find the best font for a certain character, but until that time we can make sure the font list contains the "Noto Sans" fonts which cover most situations. Fixes #31664. Fixes #12944.
326 lines
9.9 KiB
Rust
326 lines
9.9 KiB
Rust
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */
|
|
|
|
use std::cell::{Cell, RefCell};
|
|
use std::collections::HashMap;
|
|
use std::fs::File;
|
|
use std::io::prelude::*;
|
|
use std::path::PathBuf;
|
|
use std::rc::Rc;
|
|
|
|
use app_units::Au;
|
|
use gfx::font::{
|
|
fallback_font_families, FontDescriptor, FontFamilyDescriptor, FontFamilyName, FontSearchScope,
|
|
};
|
|
use gfx::font_cache_thread::{CSSFontFaceDescriptors, FontIdentifier, FontSource};
|
|
use gfx::font_context::FontContext;
|
|
use gfx::font_store::FontTemplates;
|
|
use gfx::font_template::{FontTemplate, FontTemplateRef};
|
|
use gfx::text::FallbackFontSelectionOptions;
|
|
use ipc_channel::ipc;
|
|
use net_traits::ResourceThreads;
|
|
use servo_arc::Arc;
|
|
use servo_atoms::Atom;
|
|
use servo_url::ServoUrl;
|
|
use style::properties::longhands::font_variant_caps::computed_value::T as FontVariantCaps;
|
|
use style::properties::style_structs::Font as FontStyleStruct;
|
|
use style::stylesheets::Stylesheet;
|
|
use style::values::computed::font::{
|
|
FamilyName, FontFamily, FontFamilyList, FontFamilyNameSyntax, FontSize, FontStretch, FontStyle,
|
|
FontWeight, SingleFontFamily,
|
|
};
|
|
use style::values::computed::{FontLanguageOverride, XLang};
|
|
use style::values::generics::font::LineHeight;
|
|
use webrender_api::{FontInstanceFlags, FontInstanceKey, FontKey, IdNamespace};
|
|
|
|
#[derive(Clone)]
|
|
struct MockFontCacheThread {
|
|
families: RefCell<HashMap<String, FontTemplates>>,
|
|
find_font_count: Rc<Cell<isize>>,
|
|
}
|
|
|
|
impl MockFontCacheThread {
|
|
fn new() -> MockFontCacheThread {
|
|
let mut csstest_ascii = FontTemplates::default();
|
|
Self::add_face(&mut csstest_ascii, "csstest-ascii");
|
|
|
|
let mut csstest_basic = FontTemplates::default();
|
|
Self::add_face(&mut csstest_basic, "csstest-basic-regular");
|
|
|
|
let mut fallback = FontTemplates::default();
|
|
Self::add_face(&mut fallback, "csstest-basic-regular");
|
|
|
|
let mut families = HashMap::new();
|
|
families.insert("CSSTest ASCII".to_owned(), csstest_ascii);
|
|
families.insert("CSSTest Basic".to_owned(), csstest_basic);
|
|
families.insert(
|
|
fallback_font_families(FallbackFontSelectionOptions::default())[0].to_owned(),
|
|
fallback,
|
|
);
|
|
|
|
MockFontCacheThread {
|
|
families: RefCell::new(families),
|
|
find_font_count: Rc::new(Cell::new(0)),
|
|
}
|
|
}
|
|
|
|
fn identifier_for_font_name(name: &str) -> FontIdentifier {
|
|
FontIdentifier::Web(Self::url_for_font_name(name))
|
|
}
|
|
|
|
fn url_for_font_name(name: &str) -> ServoUrl {
|
|
let mut path: PathBuf = [env!("CARGO_MANIFEST_DIR"), "tests", "support", "CSSTest"]
|
|
.iter()
|
|
.collect();
|
|
path.push(format!("{}.ttf", name));
|
|
ServoUrl::from_file_path(path).unwrap()
|
|
}
|
|
|
|
fn add_face(family: &mut FontTemplates, name: &str) {
|
|
let mut path: PathBuf = [env!("CARGO_MANIFEST_DIR"), "tests", "support", "CSSTest"]
|
|
.iter()
|
|
.collect();
|
|
path.push(format!("{}.ttf", name));
|
|
|
|
let file = File::open(path).unwrap();
|
|
let data: Vec<u8> = file.bytes().map(|b| b.unwrap()).collect();
|
|
family.add_template(
|
|
FontTemplate::new_for_remote_web_font(
|
|
Self::url_for_font_name(name),
|
|
std::sync::Arc::new(data),
|
|
&CSSFontFaceDescriptors::new(name),
|
|
None,
|
|
)
|
|
.unwrap(),
|
|
);
|
|
}
|
|
}
|
|
|
|
impl FontSource for MockFontCacheThread {
|
|
fn find_matching_font_templates(
|
|
&self,
|
|
descriptor_to_match: Option<&FontDescriptor>,
|
|
font_family_name: &FontFamilyName,
|
|
) -> Vec<FontTemplateRef> {
|
|
self.find_font_count.set(self.find_font_count.get() + 1);
|
|
self.families
|
|
.borrow_mut()
|
|
.get_mut(font_family_name.name())
|
|
.map(|family| family.find_for_descriptor(descriptor_to_match))
|
|
.unwrap_or_default()
|
|
}
|
|
|
|
fn get_system_font_instance(
|
|
&self,
|
|
_: FontIdentifier,
|
|
_: Au,
|
|
_: FontInstanceFlags,
|
|
) -> FontInstanceKey {
|
|
FontInstanceKey(IdNamespace(0), 0)
|
|
}
|
|
|
|
fn get_web_font(&self, _: std::sync::Arc<Vec<u8>>, _: u32) -> webrender_api::FontKey {
|
|
FontKey(IdNamespace(0), 0)
|
|
}
|
|
|
|
fn get_web_font_instance(
|
|
&self,
|
|
_: webrender_api::FontKey,
|
|
_: f32,
|
|
_: FontInstanceFlags,
|
|
) -> FontInstanceKey {
|
|
FontInstanceKey(IdNamespace(0), 0)
|
|
}
|
|
}
|
|
|
|
fn style() -> FontStyleStruct {
|
|
let mut style = FontStyleStruct {
|
|
font_family: FontFamily::serif(),
|
|
font_style: FontStyle::NORMAL,
|
|
font_variant_caps: FontVariantCaps::Normal,
|
|
font_weight: FontWeight::normal(),
|
|
font_size: FontSize::medium(),
|
|
font_stretch: FontStretch::hundred(),
|
|
hash: 0,
|
|
font_language_override: FontLanguageOverride::normal(),
|
|
line_height: LineHeight::Normal,
|
|
_x_lang: XLang::get_initial_value(),
|
|
};
|
|
style.compute_font_hash();
|
|
style
|
|
}
|
|
|
|
fn font_family(names: Vec<&str>) -> FontFamily {
|
|
let names: Vec<SingleFontFamily> = names
|
|
.into_iter()
|
|
.map(|name| {
|
|
SingleFontFamily::FamilyName(FamilyName {
|
|
name: Atom::from(name),
|
|
syntax: FontFamilyNameSyntax::Quoted,
|
|
})
|
|
})
|
|
.collect();
|
|
|
|
FontFamily {
|
|
families: FontFamilyList {
|
|
list: names.into_boxed_slice(),
|
|
},
|
|
is_system_font: false,
|
|
is_initial: false,
|
|
}
|
|
}
|
|
|
|
fn mock_resource_threads() -> ResourceThreads {
|
|
let (core_sender, _) = ipc::channel().unwrap();
|
|
let (storage_sender, _) = ipc::channel().unwrap();
|
|
ResourceThreads::new(core_sender, storage_sender)
|
|
}
|
|
|
|
#[test]
|
|
fn test_font_group_is_cached_by_style() {
|
|
let source = MockFontCacheThread::new();
|
|
let context = FontContext::new(source, mock_resource_threads());
|
|
|
|
let style1 = style();
|
|
|
|
let mut style2 = style();
|
|
style2.set_font_style(FontStyle::ITALIC);
|
|
|
|
assert!(
|
|
std::ptr::eq(
|
|
&*context.font_group(Arc::new(style1.clone())).read(),
|
|
&*context.font_group(Arc::new(style1.clone())).read()
|
|
),
|
|
"the same font group should be returned for two styles with the same hash"
|
|
);
|
|
|
|
assert!(
|
|
!std::ptr::eq(
|
|
&*context.font_group(Arc::new(style1.clone())).read(),
|
|
&*context.font_group(Arc::new(style2.clone())).read()
|
|
),
|
|
"different font groups should be returned for two styles with different hashes"
|
|
)
|
|
}
|
|
|
|
#[test]
|
|
fn test_font_group_find_by_codepoint() {
|
|
let source = MockFontCacheThread::new();
|
|
let count = source.find_font_count.clone();
|
|
let mut context = FontContext::new(source, mock_resource_threads());
|
|
|
|
let mut style = style();
|
|
style.set_font_family(font_family(vec!["CSSTest ASCII", "CSSTest Basic"]));
|
|
|
|
let group = context.font_group(Arc::new(style));
|
|
|
|
let font = group
|
|
.write()
|
|
.find_by_codepoint(&mut context, 'a', None)
|
|
.unwrap();
|
|
assert_eq!(
|
|
font.identifier(),
|
|
MockFontCacheThread::identifier_for_font_name("csstest-ascii")
|
|
);
|
|
assert_eq!(
|
|
count.get(),
|
|
1,
|
|
"only the first font in the list should have been loaded"
|
|
);
|
|
|
|
let font = group
|
|
.write()
|
|
.find_by_codepoint(&mut context, 'a', None)
|
|
.unwrap();
|
|
assert_eq!(
|
|
font.identifier(),
|
|
MockFontCacheThread::identifier_for_font_name("csstest-ascii")
|
|
);
|
|
assert_eq!(
|
|
count.get(),
|
|
1,
|
|
"we shouldn't load the same font a second time"
|
|
);
|
|
|
|
let font = group
|
|
.write()
|
|
.find_by_codepoint(&mut context, 'á', None)
|
|
.unwrap();
|
|
assert_eq!(
|
|
font.identifier(),
|
|
MockFontCacheThread::identifier_for_font_name("csstest-basic-regular")
|
|
);
|
|
assert_eq!(count.get(), 2, "both fonts should now have been loaded");
|
|
}
|
|
|
|
#[test]
|
|
fn test_font_fallback() {
|
|
let source = MockFontCacheThread::new();
|
|
let mut context = FontContext::new(source, mock_resource_threads());
|
|
|
|
let mut style = style();
|
|
style.set_font_family(font_family(vec!["CSSTest ASCII"]));
|
|
|
|
let group = context.font_group(Arc::new(style));
|
|
|
|
let font = group
|
|
.write()
|
|
.find_by_codepoint(&mut context, 'a', None)
|
|
.unwrap();
|
|
assert_eq!(
|
|
font.identifier(),
|
|
MockFontCacheThread::identifier_for_font_name("csstest-ascii"),
|
|
"a family in the group should be used if there is a matching glyph"
|
|
);
|
|
|
|
let font = group
|
|
.write()
|
|
.find_by_codepoint(&mut context, 'á', None)
|
|
.unwrap();
|
|
assert_eq!(
|
|
font.identifier(),
|
|
MockFontCacheThread::identifier_for_font_name("csstest-basic-regular"),
|
|
"a fallback font should be used if there is no matching glyph in the group"
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn test_font_template_is_cached() {
|
|
let source = MockFontCacheThread::new();
|
|
let count = source.find_font_count.clone();
|
|
let context = FontContext::new(source, mock_resource_threads());
|
|
|
|
let mut font_descriptor = FontDescriptor {
|
|
weight: FontWeight::normal(),
|
|
stretch: FontStretch::hundred(),
|
|
style: FontStyle::normal(),
|
|
variant: FontVariantCaps::Normal,
|
|
pt_size: Au(10),
|
|
};
|
|
|
|
let family_descriptor =
|
|
FontFamilyDescriptor::new(FontFamilyName::from("CSSTest Basic"), FontSearchScope::Any);
|
|
|
|
let font_template = context.matching_templates(&font_descriptor, &family_descriptor)[0].clone();
|
|
|
|
let font1 = context
|
|
.font(font_template.clone(), &font_descriptor)
|
|
.unwrap();
|
|
|
|
font_descriptor.pt_size = Au(20);
|
|
let font2 = context
|
|
.font(font_template.clone(), &font_descriptor)
|
|
.unwrap();
|
|
|
|
assert_ne!(
|
|
font1.descriptor.pt_size, font2.descriptor.pt_size,
|
|
"the same font should not have been returned"
|
|
);
|
|
|
|
assert_eq!(
|
|
count.get(),
|
|
1,
|
|
"we should only have fetched the template data from the cache thread once"
|
|
);
|
|
}
|