2007-04-17 04:45:49 +00:00
|
|
|
<!DOCTYPE HTML>
|
|
|
|
<html>
|
|
|
|
<!--
|
|
|
|
-->
|
|
|
|
<head>
|
|
|
|
<title>Test for parsing, storage, and serialization of CSS values</title>
|
|
|
|
<script type="text/javascript" src="/MochiKit/MochiKit.js"></script>
|
|
|
|
<script type="text/javascript" src="/tests/SimpleTest/SimpleTest.js"></script>
|
|
|
|
<script type="text/javascript" src="property_database.js"></script>
|
|
|
|
<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
|
|
|
|
</head>
|
|
|
|
<body>
|
|
|
|
<p id="display"></p>
|
|
|
|
<div id="content" style="display: none">
|
|
|
|
|
|
|
|
<div id="testnode"></div>
|
|
|
|
|
|
|
|
</div>
|
|
|
|
<pre id="test">
|
|
|
|
<script class="testbody" type="text/javascript">
|
|
|
|
|
|
|
|
/** Test for parsing, storage, and serialization of CSS values **/
|
|
|
|
|
2007-04-17 18:57:35 +00:00
|
|
|
/*
|
|
|
|
* The idempotence tests here deserve a little bit of explanation. What
|
|
|
|
* we're testing here are the following operations:
|
|
|
|
* parse: string -> CSS rule
|
|
|
|
* serialize: CSS rule -> string (normalization 1)
|
|
|
|
* (this actually has two variants that go through partly different
|
|
|
|
* codepaths, which we exercise with getPropertyValue and cssText)
|
|
|
|
* compute: CSS rule -> computed style
|
|
|
|
* cserialize: computed style -> string (normalization 2)
|
|
|
|
*
|
|
|
|
* Both serialize and cserialize do some normalization, so we can't test
|
|
|
|
* for pure round-tripping, and we also can't compare their output since
|
|
|
|
* they could normalize differently. (We might at some point in the
|
|
|
|
* future want to guarantee that any output of cserialize is
|
|
|
|
* untouched by going through parse+serialize, though.)
|
|
|
|
*
|
|
|
|
* So we test idempotence of parse + serialize by running the whole
|
|
|
|
* operation twice. Likewise for parse + compute + cserialize.
|
|
|
|
*
|
2007-05-05 17:06:49 +00:00
|
|
|
* Slightly more interestingly, we test that serialize + parse is the
|
|
|
|
* identity transform by comparing the output of parse + compute +
|
|
|
|
* cserialize to the output of parse + serialize + parse + compute +
|
|
|
|
* cserialize.
|
2007-04-17 18:57:35 +00:00
|
|
|
*/
|
|
|
|
|
2007-04-17 07:41:44 +00:00
|
|
|
var gShorthandsWithoutCondensingSerialize = {
|
|
|
|
"-moz-border-radius": true,
|
|
|
|
"-moz-outline-radius": true,
|
|
|
|
"background": true, // really there, but not complete
|
|
|
|
"cue": true,
|
|
|
|
"font": true,
|
|
|
|
"list-style": true,
|
|
|
|
"outline": true,
|
|
|
|
"pause": true,
|
|
|
|
};
|
|
|
|
|
|
|
|
var gNoComputedValue = {
|
|
|
|
"background-position": true,
|
|
|
|
"content": true,
|
|
|
|
};
|
|
|
|
|
2007-04-17 17:06:23 +00:00
|
|
|
var gNotAccepted = {
|
|
|
|
"-moz-column-width": [ "50%" ],
|
|
|
|
"-moz-user-select": [ "auto" ],
|
|
|
|
"background-color": [ "rgb(255.0,0.387,3489)" ],
|
|
|
|
"list-style": [ "none disc outside" ],
|
|
|
|
};
|
|
|
|
|
|
|
|
var gSpecialFont = [
|
|
|
|
"caption", "icon", "menu", "message-box", "small-caption", "status-bar"
|
|
|
|
];
|
|
|
|
|
|
|
|
var gBadCompute = {
|
|
|
|
// output wrapped around to positive, in exponential notation
|
|
|
|
"-moz-box-ordinal-group": [ "-1", "-1000" ],
|
|
|
|
};
|
|
|
|
|
|
|
|
var gShortenableValues = {
|
|
|
|
"border-color": [ "currentColor currentColor currentcolor CURRENTcolor" ],
|
|
|
|
"border-style": [ "none none none none", "groove none none none", "none none double none" ],
|
|
|
|
};
|
|
|
|
|
|
|
|
function xfail_accepted(property, value)
|
|
|
|
{
|
|
|
|
if (property in gNotAccepted &&
|
|
|
|
gNotAccepted[property].indexOf(value) != -1)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (property == "font" && gSpecialFont.indexOf(value) != -1)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
function xfail_accepted_split(property, subprop, value)
|
|
|
|
{
|
|
|
|
if (property in gNotAccepted &&
|
|
|
|
gNotAccepted[property].indexOf(value) != -1)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (property == "font" && subprop != "font-family" &&
|
|
|
|
gSpecialFont.indexOf(value) != -1)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2007-04-17 07:41:44 +00:00
|
|
|
function xfail_ser_val(property, value)
|
|
|
|
{
|
2007-04-17 17:06:23 +00:00
|
|
|
if (property != "font" && xfail_accepted(property, value))
|
|
|
|
// We already failed the first test, which will make us always pass this
|
|
|
|
// one.
|
|
|
|
return false;
|
|
|
|
|
2007-04-17 07:41:44 +00:00
|
|
|
if (property in gShorthandsWithoutCondensingSerialize)
|
|
|
|
return true;
|
|
|
|
|
2007-04-17 17:06:23 +00:00
|
|
|
// We output unneeded -moz-use-text-color only in the value getter and
|
|
|
|
// not the serialization.
|
|
|
|
if (property.match(/^border(|-bottom|-left|-right|-top)$/) &&
|
|
|
|
!value.match(/(green|currentcolor)/i))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// We condense multiple values in the serialization, but not in the
|
|
|
|
// value getter.
|
|
|
|
if (property.match(/^(border-(color|style|width)|margin|padding)$/) &&
|
|
|
|
value.split(" ").length != 4)
|
|
|
|
return true;
|
|
|
|
if (property in gShortenableValues &&
|
|
|
|
gShortenableValues[property].indexOf(value) != -1)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
function xfail_idparseser(property, value)
|
|
|
|
{
|
|
|
|
if (property != "font" && xfail_accepted(property, value))
|
|
|
|
// We already failed the first test, which will make us always pass this
|
|
|
|
// one.
|
|
|
|
return false;
|
|
|
|
|
2007-04-17 07:41:44 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
function xfail_idserparse_compute(property, value)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2007-04-17 18:01:17 +00:00
|
|
|
function xfail_idsersplitparse_compute(property, subprop, value, step1subcomp)
|
2007-04-17 17:06:23 +00:00
|
|
|
{
|
2007-04-17 18:01:17 +00:00
|
|
|
// These failures depend on what the system fonts actually are,
|
|
|
|
// since they show up in computed style but not serialization!
|
2007-04-17 17:06:23 +00:00
|
|
|
if (property == "font" &&
|
2007-04-17 18:01:17 +00:00
|
|
|
gSpecialFont.indexOf(value) != -1 &&
|
|
|
|
gCSSProperties[subprop].initial_values.indexOf(step1subcomp) == -1)
|
2007-04-17 17:06:23 +00:00
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
function xfail_idparsesplitser(property, value)
|
2007-04-17 07:41:44 +00:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
function xfail_compute(property, value)
|
|
|
|
{
|
2007-04-17 17:06:23 +00:00
|
|
|
if (property in gBadCompute &&
|
|
|
|
gBadCompute[property].indexOf(value) != -1)
|
|
|
|
return true;
|
|
|
|
|
2007-04-17 07:41:44 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
function xfail_split_compute(property, value)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2007-04-17 04:45:49 +00:00
|
|
|
var gElement = document.getElementById("testnode");
|
|
|
|
var gDeclaration = gElement.style;
|
|
|
|
var gComputedStyle = window.getComputedStyle(gElement, "");
|
|
|
|
|
|
|
|
function test_property(property)
|
|
|
|
{
|
|
|
|
var info = gCSSProperties[property];
|
|
|
|
|
|
|
|
var test_computed = !("backend_only" in info);
|
|
|
|
|
|
|
|
function test_value(value) {
|
|
|
|
gDeclaration.setProperty(property, value, "");
|
|
|
|
|
2007-04-17 07:41:44 +00:00
|
|
|
var idx, func;
|
|
|
|
|
2007-04-17 04:45:49 +00:00
|
|
|
var step1val = gDeclaration.getPropertyValue(property);
|
|
|
|
var step1vals = [];
|
|
|
|
var step1ser = gDeclaration.cssText;
|
|
|
|
if ("subproperties" in info)
|
|
|
|
for (idx in info.subproperties)
|
|
|
|
step1vals.push(gDeclaration.getPropertyValue(info.subproperties[idx]));
|
|
|
|
var step1comp;
|
|
|
|
var step1comps = [];
|
|
|
|
if (test_computed && info.type != CSS_TYPE_TRUE_SHORTHAND)
|
|
|
|
step1comp = gComputedStyle.getPropertyValue(property);
|
|
|
|
if (test_computed && "subproperties" in info)
|
|
|
|
for (idx in info.subproperties)
|
|
|
|
step1comps.push(gComputedStyle.getPropertyValue(info.subproperties[idx]));
|
|
|
|
|
2007-04-17 17:06:23 +00:00
|
|
|
func = xfail_accepted(property, value) ? todo_isnot : isnot;
|
|
|
|
func(step1val, "", "setting '" + value + "' on '" + property);
|
2007-04-17 04:45:49 +00:00
|
|
|
if ("subproperties" in info)
|
2007-04-17 17:06:23 +00:00
|
|
|
for (idx in info.subproperties) {
|
|
|
|
var subprop = info.subproperties[idx];
|
|
|
|
func = xfail_accepted_split(property, subprop, value)
|
|
|
|
? todo_isnot : isnot;
|
|
|
|
func(gDeclaration.getPropertyValue(subprop), "",
|
|
|
|
"setting '" + value + "' on '" + property);
|
|
|
|
}
|
2007-04-17 04:45:49 +00:00
|
|
|
|
2007-04-17 07:41:44 +00:00
|
|
|
// We don't care particularly about the whitespace or the placement of
|
|
|
|
// semicolons, but for simplicity we'll test the current behavior.
|
|
|
|
func = xfail_ser_val(property, value) ? todo_is : is;
|
|
|
|
var expected_serialization = "";
|
|
|
|
if (step1val != "")
|
|
|
|
expected_serialization = property + ": " + step1val + ";";
|
|
|
|
func(step1ser, expected_serialization,
|
|
|
|
"serialization should match property value");
|
|
|
|
|
2007-04-17 04:45:49 +00:00
|
|
|
gDeclaration.removeProperty(property);
|
|
|
|
gDeclaration.setProperty(property, step1val, "");
|
|
|
|
|
2007-04-17 17:06:23 +00:00
|
|
|
func = xfail_idparseser(property, value) ? todo_is : is;
|
|
|
|
func(gDeclaration.getPropertyValue(property), step1val,
|
|
|
|
"parse+serialize should be idempotent for '" +
|
|
|
|
property + ": " + value + "'");
|
2007-04-17 04:45:49 +00:00
|
|
|
if (test_computed && info.type != CSS_TYPE_TRUE_SHORTHAND) {
|
2007-04-17 07:41:44 +00:00
|
|
|
func = xfail_idserparse_compute(property, value) ? todo_is : is;
|
|
|
|
func(gComputedStyle.getPropertyValue(property), step1comp,
|
2007-05-05 17:06:49 +00:00
|
|
|
"serialize+parse should be identity transform for '" +
|
2007-04-17 07:41:44 +00:00
|
|
|
property + ": " + value + "'");
|
2007-04-17 04:45:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if ("subproperties" in info) {
|
|
|
|
gDeclaration.removeProperty(property);
|
|
|
|
for (idx in info.subproperties) {
|
2007-04-17 07:41:44 +00:00
|
|
|
var subprop = info.subproperties[idx];
|
|
|
|
gDeclaration.setProperty(subprop, step1vals[idx], "");
|
2007-04-17 18:01:17 +00:00
|
|
|
if (test_computed && !("backend_only" in gCSSProperties[subprop])) {
|
2007-04-17 17:06:23 +00:00
|
|
|
func =
|
2007-04-17 18:01:17 +00:00
|
|
|
xfail_idsersplitparse_compute(property, subprop, value, step1comps[idx])
|
2007-04-17 17:06:23 +00:00
|
|
|
? todo_is : is;
|
2007-04-17 07:41:44 +00:00
|
|
|
func(gComputedStyle.getPropertyValue(subprop), step1comps[idx],
|
2007-05-05 17:06:49 +00:00
|
|
|
"serialize(" + subprop + ")+parse should be the identity " +
|
|
|
|
"transform for '" + property + ": " + value + "'");
|
2007-04-17 07:41:44 +00:00
|
|
|
}
|
2007-04-17 04:45:49 +00:00
|
|
|
}
|
2007-04-17 17:06:23 +00:00
|
|
|
func = xfail_idparsesplitser(property, value) ? todo_is : is;
|
|
|
|
func(gDeclaration.getPropertyValue(property), step1val,
|
|
|
|
"parse+split+serialize should be idempotent for '" +
|
|
|
|
property + ": " + value + "'");
|
2007-04-17 04:45:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (test_computed && info.type != CSS_TYPE_TRUE_SHORTHAND) {
|
|
|
|
gDeclaration.removeProperty(property);
|
|
|
|
gDeclaration.setProperty(property, step1comp, "");
|
2007-04-17 07:41:44 +00:00
|
|
|
func = xfail_compute(property, value) ? todo_is : is;
|
|
|
|
func(gComputedStyle.getPropertyValue(property), step1comp,
|
|
|
|
"parse+compute+serialize should be idempotent for '" +
|
|
|
|
property + ": " + value + "'");
|
2007-04-17 04:45:49 +00:00
|
|
|
}
|
|
|
|
if (test_computed && "subproperties" in info) {
|
|
|
|
gDeclaration.removeProperty(property);
|
|
|
|
for (idx in info.subproperties) {
|
2007-04-17 07:41:44 +00:00
|
|
|
var subprop = info.subproperties[idx];
|
2007-04-17 18:01:17 +00:00
|
|
|
if ("backend_only" in gCSSProperties[subprop])
|
|
|
|
continue;
|
2007-04-17 07:41:44 +00:00
|
|
|
gDeclaration.setProperty(subprop, step1comps[idx], "");
|
|
|
|
func = xfail_split_compute(property, value) ? todo_is : is;
|
|
|
|
func(gComputedStyle.getPropertyValue(subprop), step1comps[idx],
|
|
|
|
"parse+compute+serialize(" + subprop + ") should be idempotent for '" +
|
|
|
|
property + ": " + value + "'");
|
2007-04-17 04:45:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
gDeclaration.removeProperty(property);
|
|
|
|
}
|
|
|
|
|
|
|
|
var idx;
|
|
|
|
for (idx in info.initial_values)
|
|
|
|
test_value(info.initial_values[idx]);
|
|
|
|
for (idx in info.other_values)
|
|
|
|
test_value(info.other_values[idx]);
|
|
|
|
}
|
|
|
|
|
2007-04-17 07:41:44 +00:00
|
|
|
// To avoid triggering the slow script dialog, we have to test one
|
|
|
|
// property at a time.
|
|
|
|
SimpleTest.waitForExplicitFinish();
|
|
|
|
var props = [];
|
2007-04-17 04:45:49 +00:00
|
|
|
for (var prop in gCSSProperties)
|
2007-04-17 07:41:44 +00:00
|
|
|
props.push(prop);
|
|
|
|
props = props.reverse();
|
|
|
|
function do_one(l) {
|
|
|
|
if (l.length == 0) {
|
|
|
|
// SimpleTest.finish() is really slow, so we have to disable the
|
|
|
|
// slow script dialog for this part
|
|
|
|
netscape.security.PrivilegeManager.enablePrivilege("UniversalXPConnect");
|
|
|
|
var prefService = Components.classes["@mozilla.org/preferences-service;1"].
|
|
|
|
getService(Components.interfaces.nsIPrefService);
|
|
|
|
var domBranch = prefService.getBranch("dom.");
|
|
|
|
var oldVal = domBranch.getIntPref("max_script_run_time");
|
|
|
|
domBranch.setIntPref("max_script_run_time", 0);
|
|
|
|
|
|
|
|
SimpleTest.finish();
|
|
|
|
|
|
|
|
domBranch.setIntPref("max_script_run_time", oldVal);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
test_property(l.pop());
|
|
|
|
setTimeout(do_one, 0, l);
|
|
|
|
}
|
|
|
|
setTimeout(do_one, 0, props);
|
2007-04-17 04:45:49 +00:00
|
|
|
|
|
|
|
</script>
|
|
|
|
</pre>
|
|
|
|
</body>
|
|
|
|
</html>
|