<!DOCTYPE html> <link rel="help" href="https://drafts.css-houdini.org/css-properties-values-api-1/#dependency-cycles-via-relative-units" /> <meta name="assert" content="This test verifies that reference cycles via units are detected" /> <script src="/resources/testharness.js"></script> <script src="/resources/testharnessreport.js"></script> <script> function register_length(name, inherits=true) { CSS.registerProperty({ name: name, syntax: '<length>', initialValue: '0px', inherits: inherits }); } register_length('--font-size-em'); register_length('--font-size-rem'); register_length('--font-size-ex'); register_length('--font-size-ch'); register_length('--font-size-px'); register_length('--font-size-em-via-var'); register_length('--font-size-rem-via-var'); register_length('--font-size-ex-via-var'); register_length('--font-size-ch-via-var'); register_length('--font-size-em-inherited', true); register_length('--font-size-ex-inherited', true); register_length('--font-size-ch-inherited', true); </script> <style> :root { --unregistered-em: 10em; --unregistered-rem: 10rem; --unregistered-ex: 10ex; --unregistered-ch: 10ch; } :root, #target { --font-size-em: 2em; --font-size-rem: 2rem; --font-size-ex: 2ex; --font-size-ch: 2ch; --font-size-px: 42px; --font-size-em-via-var: var(--unregistered-em); --font-size-rem-via-var: var(--unregistered-rem); --font-size-ex-via-var: var(--unregistered-ex); --font-size-ch-via-var: var(--unregistered-ch); } #parent { --font-size-em-inherited: 4em; --font-size-ex-inherited: 4ex; --font-size-ch-inherited: 4ch; } #target { font-size: 11px; } </style> <div id=parent> <div id=target></div> </div> <div id=ref></div> <script> // Compute a dimension (e.g. 1em) given a certain fontSize. function compute_dimension(dimension, fontSize, element = ref) { try { element.attributeStyleMap.set('font-size', fontSize); element.attributeStyleMap.set('margin-bottom', dimension); return getComputedStyle(element).marginBottom; } finally { element.attributeStyleMap.clear(); } } function assert_property_equals(name, value, element = target) { var computedStyle = getComputedStyle(element); assert_equals(computedStyle.getPropertyValue(name), value); } let unsetFontSize = compute_dimension('1em', 'unset'); add_result_callback(function(){ target.attributeStyleMap.clear(); document.documentElement.attributeStyleMap.clear(); }); test(function() { target.style = 'font-size: var(--font-size-px);'; assert_property_equals('font-size', '42px'); assert_property_equals('--font-size-px', '42px'); }, 'Non-font-dependent variables can be used in font-size'); test(function() { target.style = 'font-size: var(--font-size-em);'; assert_property_equals('font-size', unsetFontSize); assert_property_equals('--font-size-em', compute_dimension('2em', 'unset')); }, 'Lengths with em units may not be referenced from font-size'); test(function() { target.style = 'font-size: var(--font-size-ex);'; assert_property_equals('font-size', unsetFontSize); assert_property_equals('--font-size-ex', compute_dimension('2ex', 'unset')); }, 'Lengths with ex units may not be referenced from font-size'); test(function() { target.style = 'font-size: var(--font-size-ch);'; assert_property_equals('font-size', unsetFontSize); assert_property_equals('--font-size-ch', compute_dimension('2ch', 'unset')); }, 'Lengths with ch units may not be referenced from font-size'); test(function() { target.style = 'font-size: var(--font-size-rem);'; let expected = compute_dimension('2rem', 'unset', document.documentElement); assert_property_equals('--font-size-rem', expected); assert_property_equals('font-size', expected); }, 'Lengths with rem units may be referenced from font-size on non-root element'); test(function() { let root = document.documentElement; let expected1rem = compute_dimension('1rem', 'unset', root); let expected2rem = compute_dimension('2rem', 'unset', root); root.style = 'font-size: var(--font-size-rem);'; assert_property_equals('font-size', expected1rem, root); assert_property_equals('--font-size-rem', expected2rem, root); }, 'Lengths with rem units may not be referenced from font-size on root element'); test(function() { target.style = 'font-size: var(--noexist, var(--font-size-em));'; assert_property_equals('font-size', unsetFontSize); }, 'Fallback may not use font-relative units'); test(function() { target.style = 'font-size: var(--font-size-em, 42px);'; assert_property_equals('font-size', '42px'); }, 'Fallback triggered when em unit cycle is detected'); test(function() { target.style = 'font-size: var(--font-size-ex, 42px);'; assert_property_equals('font-size', '42px'); }, 'Fallback triggered when ex unit cycle is detected'); test(function() { target.style = 'font-size: var(--font-size-ch, 42px);'; assert_property_equals('font-size', '42px'); }, 'Fallback triggered when ch unit cycle is detected'); test(function() { let root = document.documentElement; root.style = 'font-size: var(--font-size-rem, 42px);'; assert_property_equals('font-size', '42px', root); root.style = 'font-size: unset;'; }, 'Fallback triggered when rem unit cycle is detected on root element'); test(function() { target.style = 'font-size: var(--font-size-em-via-var);'; assert_property_equals('font-size', unsetFontSize); assert_property_equals('--font-size-em-via-var', compute_dimension('10em', 'unset')); }, 'Lengths with em units are detected via var references'); test(function() { target.style = 'font-size: var(--font-size-ex-via-var);'; assert_property_equals('font-size', unsetFontSize); assert_property_equals('--font-size-ex-via-var', compute_dimension('10ex', 'unset')); }, 'Lengths with ex units are detected via var references'); test(function() { target.style = 'font-size: var(--font-size-ch-via-var);'; assert_property_equals('font-size', unsetFontSize); assert_property_equals('--font-size-ch-via-var', compute_dimension('10ch', 'unset')); }, 'Lengths with ch units are detected via var references'); test(function() { let root = document.documentElement; let expected1rem = compute_dimension('1rem', 'unset', root); let expected10rem = compute_dimension('10rem', 'unset', root); root.style = 'font-size: var(--font-size-rem-via-var);'; assert_property_equals('font-size', expected1rem, root); assert_property_equals('--font-size-rem-via-var', expected10rem, root); }, 'Lengths with rem units are detected via var references'); test(function() { let expected4em = compute_dimension('4em', 'unset'); target.style = 'font-size: var(--font-size-em-inherited);'; assert_property_equals('font-size', expected4em); assert_property_equals('--font-size-em-inherited', expected4em); }, 'Inherited lengths with em units may be used'); test(function() { let expected4ex = compute_dimension('4ex', 'unset'); target.style = 'font-size: var(--font-size-ex-inherited);'; assert_property_equals('font-size', expected4ex); assert_property_equals('--font-size-ex-inherited', expected4ex); }, 'Inherited lengths with ex units may be used'); test(function() { let expected4ch = compute_dimension('4ch', 'unset'); target.style = 'font-size: var(--font-size-ch-inherited);'; assert_property_equals('font-size', expected4ch); assert_property_equals('--font-size-ch-inherited', expected4ch); }, 'Inherited lengths with ch units may be used'); </script>