layout: Add initial support for text-transform (#31396)

This adds basic support for `text-transform` in a way that is more
complete than legacy layout. There are still many missing elements of
proper `text-transform` support such as:

1. Support for `full-width` and `full-size-kana`
2. Support for grapheme based uppercasing, lowercasing, and
   capitalization. These are all done per-code point right now.
3. Support for the language-specific `SpecialCasing.txt` cases for case
   mapping such as the ones for Irish and Turkish.

Co-authored-by: Rakhi Sharma <atbrakhi@igalia.com>
This commit is contained in:
Martin Robinson 2024-02-22 15:15:59 +01:00 committed by GitHub
parent f60e5e767b
commit d8b326528b
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
105 changed files with 181 additions and 292 deletions

1
Cargo.lock generated
View file

@ -3130,6 +3130,7 @@ dependencies = [
"style", "style",
"style_traits", "style_traits",
"unicode-script", "unicode-script",
"unicode-segmentation",
"webrender_api", "webrender_api",
"xi-unicode", "xi-unicode",
] ]

View file

@ -107,6 +107,7 @@ tokio-rustls = "0.24"
tungstenite = "0.20" tungstenite = "0.20"
unicode-bidi = "0.3.15" unicode-bidi = "0.3.15"
unicode-script = "0.5" unicode-script = "0.5"
unicode-segmentation = "1.1.0"
url = "2.5" url = "2.5"
uuid = { version = "1.7.0", features = ["v4"] } uuid = { version = "1.7.0", features = ["v4"] }
webdriver = "0.49.0" webdriver = "0.49.0"

View file

@ -42,6 +42,7 @@ servo_url = { path = "../url" }
style = { path = "../style", features = ["servo"] } style = { path = "../style", features = ["servo"] }
style_traits = { workspace = true } style_traits = { workspace = true }
unicode-script = { workspace = true } unicode-script = { workspace = true }
unicode-segmentation = { workspace = true }
webrender_api = { workspace = true } webrender_api = { workspace = true }
xi-unicode = { workspace = true } xi-unicode = { workspace = true }

View file

@ -684,7 +684,7 @@ impl<'a> BuilderForBoxFragment<'a> {
); );
if let Some(layer) = if let Some(layer) =
background::layout_layer(self, &painter, builder, index, intrinsic) background::layout_layer(self, painter, builder, index, intrinsic)
{ {
let image_rendering = image_rendering(style.clone_image_rendering()); let image_rendering = image_rendering(style.clone_image_rendering());
if layer.repeat { if layer.repeat {

View file

@ -1578,6 +1578,9 @@ impl InlineFormattingContext {
// > (It is invisible, but retains its soft wrap opportunity, if any.) // > (It is invisible, but retains its soft wrap opportunity, if any.)
let mut last_inline_box_ended_with_white_space = false; let mut last_inline_box_ended_with_white_space = false;
// For the purposes of `text-transform: capitalize` the start of the IFC is a word boundary.
let mut on_word_boundary = true;
crate::context::with_thread_local_font_context(layout_context, |font_context| { crate::context::with_thread_local_font_context(layout_context, |font_context| {
let mut linebreaker = None; let mut linebreaker = None;
self.foreach(|iter_item| match iter_item { self.foreach(|iter_item| match iter_item {
@ -1589,6 +1592,7 @@ impl InlineFormattingContext {
&mut linebreaker, &mut linebreaker,
&mut ifc_fonts, &mut ifc_fonts,
&mut last_inline_box_ended_with_white_space, &mut last_inline_box_ended_with_white_space,
&mut on_word_boundary,
); );
}, },
InlineFormattingContextIterItem::Item(InlineLevelBox::InlineBox(inline_box)) => { InlineFormattingContextIterItem::Item(InlineLevelBox::InlineBox(inline_box)) => {
@ -1601,6 +1605,7 @@ impl InlineFormattingContext {
}, },
InlineFormattingContextIterItem::Item(InlineLevelBox::Atomic(_)) => { InlineFormattingContextIterItem::Item(InlineLevelBox::Atomic(_)) => {
last_inline_box_ended_with_white_space = false; last_inline_box_ended_with_white_space = false;
on_word_boundary = true;
}, },
_ => {}, _ => {},
}); });

View file

@ -2,8 +2,8 @@
* License, v. 2.0. If a copy of the MPL was not distributed with this * 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/. */ * file, You can obtain one at https://mozilla.org/MPL/2.0/. */
use std::char::{ToLowercase, ToUppercase};
use std::mem; use std::mem;
use std::str::Chars;
use app_units::Au; use app_units::Au;
use gfx::font::{FontRef, ShapingFlags, ShapingOptions}; use gfx::font::{FontRef, ShapingFlags, ShapingOptions};
@ -19,7 +19,10 @@ use style::computed_values::text_rendering::T as TextRendering;
use style::computed_values::white_space::T as WhiteSpace; use style::computed_values::white_space::T as WhiteSpace;
use style::computed_values::word_break::T as WordBreak; use style::computed_values::word_break::T as WordBreak;
use style::properties::ComputedValues; use style::properties::ComputedValues;
use style::values::specified::text::TextTransformCase;
use style::values::specified::TextTransform;
use unicode_script::Script; use unicode_script::Script;
use unicode_segmentation::UnicodeSegmentation;
use xi_unicode::{linebreak_property, LineBreakLeafIter}; use xi_unicode::{linebreak_property, LineBreakLeafIter};
use super::inline::{FontKeyAndMetrics, InlineFormattingContextState}; use super::inline::{FontKeyAndMetrics, InlineFormattingContextState};
@ -209,11 +212,13 @@ impl TextRun {
linebreaker: &mut Option<LineBreakLeafIter>, linebreaker: &mut Option<LineBreakLeafIter>,
font_cache: &mut Vec<FontKeyAndMetrics>, font_cache: &mut Vec<FontKeyAndMetrics>,
last_inline_box_ended_with_white_space: &mut bool, last_inline_box_ended_with_white_space: &mut bool,
on_word_boundary: &mut bool,
) { ) {
let segment_results = self.segment_text( let segment_results = self.segment_text(
font_context, font_context,
font_cache, font_cache,
last_inline_box_ended_with_white_space, last_inline_box_ended_with_white_space,
on_word_boundary,
); );
let inherited_text_style = self.parent_style.get_inherited_text().clone(); let inherited_text_style = self.parent_style.get_inherited_text().clone();
let letter_spacing = if inherited_text_style.letter_spacing.0.px() != 0. { let letter_spacing = if inherited_text_style.letter_spacing.0.px() != 0. {
@ -278,25 +283,49 @@ impl TextRun {
font_context: &mut FontContext<FontCacheThread>, font_context: &mut FontContext<FontCacheThread>,
font_cache: &mut Vec<FontKeyAndMetrics>, font_cache: &mut Vec<FontKeyAndMetrics>,
last_inline_box_ended_with_white_space: &mut bool, last_inline_box_ended_with_white_space: &mut bool,
on_word_boundary: &mut bool,
) -> Vec<(TextRunSegment, FontRef)> { ) -> Vec<(TextRunSegment, FontRef)> {
let font_group = font_context.font_group(self.parent_style.clone_font()); let font_group = font_context.font_group(self.parent_style.clone_font());
let mut current: Option<(TextRunSegment, FontRef)> = None; let mut current: Option<(TextRunSegment, FontRef)> = None;
let mut results = Vec::new(); let mut results = Vec::new();
let text = std::mem::replace(&mut self.text, String::new()); // TODO: Eventually the text should come directly from the Cow strings of the DOM nodes.
let text = std::mem::take(&mut self.text);
let collapsed = WhitespaceCollapse::new( let collapsed = WhitespaceCollapse::new(
text.as_str(), text.as_str().chars(),
self.parent_style.clone_white_space(), self.parent_style.clone_white_space(),
*last_inline_box_ended_with_white_space, *last_inline_box_ended_with_white_space,
); );
let text_transform = self.parent_style.clone_text_transform();
let collected_text: String;
let char_iterator: Box<dyn Iterator<Item = char>> =
if text_transform.case_ == TextTransformCase::Capitalize {
// `TextTransformation` doesn't support capitalization, so we must capitalize the whole
// string at once and make a copy. Here `on_word_boundary` indicates whether or not the
// inline formatting context as a whole is on a word boundary. This is different from
// `last_inline_box_ended_with_white_space` because the word boundaries are between
// atomic inlines and at the start of the IFC.
let collapsed_string: String = collapsed.collect();
collected_text = capitalize_string(&collapsed_string, *on_word_boundary);
Box::new(collected_text.chars())
} else if !text_transform.is_none() {
// If `text-transform` is active, wrap the `WhitespaceCollapse` iterator in
// a `TextTransformation` iterator.
Box::new(TextTransformation::new(collapsed, text_transform))
} else {
Box::new(collapsed)
};
let mut next_byte_index = 0; let mut next_byte_index = 0;
let text = collapsed let text = char_iterator
.map(|character| { .map(|character| {
let current_byte_index = next_byte_index; let current_byte_index = next_byte_index;
next_byte_index += character.len_utf8(); next_byte_index += character.len_utf8();
*last_inline_box_ended_with_white_space = character.is_whitespace(); *last_inline_box_ended_with_white_space = character.is_whitespace();
*on_word_boundary = *last_inline_box_ended_with_white_space;
let prevents_soft_wrap_opportunity = let prevents_soft_wrap_opportunity =
char_prevents_soft_wrap_opportunity_when_before_or_after_atomic(character); char_prevents_soft_wrap_opportunity_when_before_or_after_atomic(character);
if current_byte_index == 0 && prevents_soft_wrap_opportunity { if current_byte_index == 0 && prevents_soft_wrap_opportunity {
@ -331,7 +360,7 @@ impl TextRun {
// segment in the middle of the run (ie the start should be 0). // segment in the middle of the run (ie the start should be 0).
let start_byte_index = match current { let start_byte_index = match current {
Some(_) => ByteIndex(current_byte_index as isize), Some(_) => ByteIndex(current_byte_index as isize),
None => ByteIndex(0 as isize), None => ByteIndex(0_isize),
}; };
let new = ( let new = (
TextRunSegment::new(font_index, script, start_byte_index), TextRunSegment::new(font_index, script, start_byte_index),
@ -491,8 +520,8 @@ fn preserve_segment_break() -> bool {
true true
} }
pub struct WhitespaceCollapse<'a> { pub struct WhitespaceCollapse<InputIterator> {
char_iterator: Chars<'a>, char_iterator: InputIterator,
white_space: WhiteSpace, white_space: WhiteSpace,
/// Whether or not we should collapse white space completely at the start of the string. /// Whether or not we should collapse white space completely at the start of the string.
@ -519,10 +548,14 @@ pub struct WhitespaceCollapse<'a> {
character_pending_to_return: Option<char>, character_pending_to_return: Option<char>,
} }
impl<'a> WhitespaceCollapse<'a> { impl<InputIterator> WhitespaceCollapse<InputIterator> {
pub fn new(input: &'a str, white_space: WhiteSpace, trim_beginning_white_space: bool) -> Self { pub fn new(
char_iterator: InputIterator,
white_space: WhiteSpace,
trim_beginning_white_space: bool,
) -> Self {
Self { Self {
char_iterator: input.chars(), char_iterator,
white_space, white_space,
remove_collapsible_white_space_at_start: trim_beginning_white_space, remove_collapsible_white_space_at_start: trim_beginning_white_space,
inside_white_space: false, inside_white_space: false,
@ -545,7 +578,10 @@ impl<'a> WhitespaceCollapse<'a> {
} }
} }
impl<'a> Iterator for WhitespaceCollapse<'a> { impl<InputIterator> Iterator for WhitespaceCollapse<InputIterator>
where
InputIterator: Iterator<Item = char>,
{
type Item = char; type Item = char;
fn next(&mut self) -> Option<Self::Item> { fn next(&mut self) -> Option<Self::Item> {
@ -645,3 +681,115 @@ impl<'a> Iterator for WhitespaceCollapse<'a> {
self.char_iterator.count() self.char_iterator.count()
} }
} }
enum PendingCaseConversionResult {
Uppercase(ToUppercase),
Lowercase(ToLowercase),
}
impl PendingCaseConversionResult {
fn next(&mut self) -> Option<char> {
match self {
PendingCaseConversionResult::Uppercase(to_uppercase) => to_uppercase.next(),
PendingCaseConversionResult::Lowercase(to_lowercase) => to_lowercase.next(),
}
}
}
/// This is an interator that consumes a char iterator and produces character transformed
/// by the given CSS `text-transform` value. It currently does not support
/// `text-transform: capitalize` because Unicode segmentation libraries do not support
/// streaming input one character at a time.
pub struct TextTransformation<InputIterator> {
/// The input character iterator.
char_iterator: InputIterator,
/// The `text-transform` value to use.
text_transform: TextTransform,
/// If an uppercasing or lowercasing produces more than one character, this
/// caches them so that they can be returned in subsequent iterator calls.
pending_case_conversion_result: Option<PendingCaseConversionResult>,
}
impl<'a, InputIterator> TextTransformation<InputIterator> {
pub fn new(char_iterator: InputIterator, text_transform: TextTransform) -> Self {
Self {
char_iterator,
text_transform,
pending_case_conversion_result: None,
}
}
}
impl<InputIterator> Iterator for TextTransformation<InputIterator>
where
InputIterator: Iterator<Item = char>,
{
type Item = char;
fn next(&mut self) -> Option<Self::Item> {
if let Some(character) = self
.pending_case_conversion_result
.as_mut()
.and_then(|result| result.next())
{
return Some(character);
}
self.pending_case_conversion_result = None;
for character in self.char_iterator.by_ref() {
match self.text_transform.case_ {
TextTransformCase::None => return Some(character),
TextTransformCase::Uppercase => {
let mut pending_result =
PendingCaseConversionResult::Uppercase(character.to_uppercase());
if let Some(character) = pending_result.next() {
self.pending_case_conversion_result = Some(pending_result);
return Some(character);
}
},
TextTransformCase::Lowercase => {
let mut pending_result =
PendingCaseConversionResult::Lowercase(character.to_lowercase());
if let Some(character) = pending_result.next() {
self.pending_case_conversion_result = Some(pending_result);
return Some(character);
}
},
// `text-transform: capitalize` currently cannot work on a per-character basis,
// so must be handled outside of this iterator.
// TODO: Add support for `full-width` and `full-size-kana`.
_ => return Some(character),
}
}
None
}
}
/// Given a string and whether the start of the string represents a word boundary, create a copy of
/// the string with letters after word boundaries capitalized.
fn capitalize_string(string: &str, allow_word_at_start: bool) -> String {
let mut output_string = String::new();
output_string.reserve(string.len());
let mut bounds = string.unicode_word_indices().peekable();
let mut byte_index = 0;
for character in string.chars() {
let current_byte_index = byte_index;
byte_index += character.len_utf8();
if let Some((next_index, _)) = bounds.peek() {
if *next_index == current_byte_index {
bounds.next();
if current_byte_index != 0 || allow_word_at_start {
output_string.extend(character.to_uppercase());
continue;
}
}
}
output_string.push(character);
}
output_string
}

View file

@ -8,8 +8,8 @@ mod text {
#[test] #[test]
fn test_collapse_whitespace() { fn test_collapse_whitespace() {
let collapse = |input, white_space, trim_beginning_white_space| { let collapse = |input: &str, white_space, trim_beginning_white_space| {
WhitespaceCollapse::new(input, white_space, trim_beginning_white_space) WhitespaceCollapse::new(input.chars(), white_space, trim_beginning_white_space)
.collect::<String>() .collect::<String>()
}; };

View file

@ -107,7 +107,7 @@ tempfile = "3"
tendril = { version = "0.4.1", features = ["encoding_rs"] } tendril = { version = "0.4.1", features = ["encoding_rs"] }
time = { workspace = true } time = { workspace = true }
unicode-bidi = { workspace = true } unicode-bidi = { workspace = true }
unicode-segmentation = "1.1.0" unicode-segmentation = { workspace = true }
url = { workspace = true } url = { workspace = true }
utf-8 = "0.7" utf-8 = "0.7"
uuid = { workspace = true, features = ["serde"] } uuid = { workspace = true, features = ["serde"] }

View file

@ -33,7 +33,6 @@ ${helpers.predefined_type(
"TextTransform", "TextTransform",
"computed::TextTransform::none()", "computed::TextTransform::none()",
engines="gecko servo", engines="gecko servo",
servo_pref="layout.legacy_layout",
animation_value_type="discrete", animation_value_type="discrete",
spec="https://drafts.csswg.org/css-text/#propdef-text-transform", spec="https://drafts.csswg.org/css-text/#propdef-text-transform",
servo_restyle_damage="rebuild_and_reflow", servo_restyle_damage="rebuild_and_reflow",

View file

@ -1,2 +0,0 @@
[c545-txttrans-000.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-001.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-002.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-003.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-005.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-applies-to-001.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-applies-to-002.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-applies-to-003.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-applies-to-005.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-applies-to-006.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-applies-to-007.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-applies-to-008.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-applies-to-009.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-applies-to-010.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-applies-to-011.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-applies-to-014.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-applies-to-015.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-bicameral-001.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-bicameral-002.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-bicameral-008.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-bicameral-009.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-bicameral-010.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-bicameral-012.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-bicameral-013.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-bicameral-015.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-cap-001.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-cap-002.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-lowercase-001.xht]
expected: FAIL

View file

@ -0,0 +1,2 @@
[text-transform-unicase-001.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-uppercase-001.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-uppercase-002.xht]
expected: FAIL

View file

@ -41,12 +41,6 @@
[Property overflow-wrap inherits] [Property overflow-wrap inherits]
expected: FAIL expected: FAIL
[Property text-transform has initial value none]
expected: FAIL
[Property text-transform inherits]
expected: FAIL
[Property word-break has initial value normal] [Property word-break has initial value normal]
expected: FAIL expected: FAIL

View file

@ -1,30 +1,3 @@
[text-transform-computed.html] [text-transform-computed.html]
[Property text-transform value 'none']
expected: FAIL
[Property text-transform value 'capitalize']
expected: FAIL
[Property text-transform value 'uppercase']
expected: FAIL
[Property text-transform value 'lowercase']
expected: FAIL
[Property text-transform value 'full-width']
expected: FAIL
[Property text-transform value 'full-size-kana']
expected: FAIL
[Property text-transform value 'capitalize full-width']
expected: FAIL
[Property text-transform value 'full-width full-size-kana']
expected: FAIL
[Property text-transform value 'uppercase full-width full-size-kana']
expected: FAIL
[Property text-transform value 'math-auto'] [Property text-transform value 'math-auto']
expected: FAIL expected: FAIL

View file

@ -1,57 +1,3 @@
[text-transform-valid.html] [text-transform-valid.html]
[e.style['text-transform'\] = "none" should set the property value]
expected: FAIL
[e.style['text-transform'\] = "capitalize" should set the property value]
expected: FAIL
[e.style['text-transform'\] = "uppercase" should set the property value]
expected: FAIL
[e.style['text-transform'\] = "lowercase" should set the property value]
expected: FAIL
[e.style['text-transform'\] = "full-width" should set the property value]
expected: FAIL
[e.style['text-transform'\] = "full-size-kana" should set the property value]
expected: FAIL
[e.style['text-transform'\] = "capitalize full-width" should set the property value]
expected: FAIL
[e.style['text-transform'\] = "uppercase full-size-kana" should set the property value]
expected: FAIL
[e.style['text-transform'\] = "full-width full-size-kana" should set the property value]
expected: FAIL
[e.style['text-transform'\] = "full-width lowercase" should set the property value]
expected: FAIL
[e.style['text-transform'\] = "full-size-kana capitalize" should set the property value]
expected: FAIL
[e.style['text-transform'\] = "full-size-kana full-width" should set the property value]
expected: FAIL
[e.style['text-transform'\] = "capitalize full-width full-size-kana" should set the property value]
expected: FAIL
[e.style['text-transform'\] = "full-width full-size-kana uppercase" should set the property value]
expected: FAIL
[e.style['text-transform'\] = "full-size-kana lowercase full-width" should set the property value]
expected: FAIL
[e.style['text-transform'\] = "lowercase full-size-kana full-width" should set the property value]
expected: FAIL
[e.style['text-transform'\] = "full-width uppercase full-size-kana" should set the property value]
expected: FAIL
[e.style['text-transform'\] = "full-size-kana full-width capitalize" should set the property value]
expected: FAIL
[e.style['text-transform'\] = "math-auto" should set the property value] [e.style['text-transform'\] = "math-auto" should set the property value]
expected: FAIL expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-capitalize-001.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-capitalize-003.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-capitalize-005.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-capitalize-009.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-capitalize-010.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-capitalize-014.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-capitalize-018.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-capitalize-020.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-capitalize-022.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-capitalize-024.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-capitalize-028.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-capitalize-030.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-capitalize-031.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-capitalize-032.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-capitalize-034.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-capitalize-035.html]
expected: FAIL

View file

@ -0,0 +1,2 @@
[text-transform-full-size-kana-001.html]
expected: FAIL

View file

@ -0,0 +1,2 @@
[text-transform-full-size-kana-002.html]
expected: FAIL

View file

@ -0,0 +1,2 @@
[text-transform-full-size-kana-003.html]
expected: FAIL

View file

@ -0,0 +1,2 @@
[text-transform-full-size-kana-004.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-lowercase-101.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-lowercase-102.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-tailoring-004.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-uppercase-101.xht]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-001.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-002.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-003.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-004.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-005.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-006.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-007.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-008.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-009.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-010.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-014.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-015.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-017.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-018.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-019.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-020.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-021.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-022.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-023.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-024.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-025.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-028.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-029.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-030.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-031.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-032.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-033.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-034.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-035.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-101.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-102.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-103.html]
expected: FAIL

View file

@ -1,2 +0,0 @@
[text-transform-upperlower-104.html]
expected: FAIL

Some files were not shown because too many files have changed in this diff Show more