2021-03-19 18:11:40 +09:00
|
|
|
/* Copyright 2021 Mozilla Foundation
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2021-03-31 22:42:21 +09:00
|
|
|
import { $getParent, $toStyle, XFAObject } from "./xfa_object.js";
|
2021-03-25 21:02:39 +09:00
|
|
|
import { warn } from "../../shared/util.js";
|
2021-03-19 18:11:40 +09:00
|
|
|
|
|
|
|
function measureToString(m) {
|
2021-03-31 22:42:21 +09:00
|
|
|
if (typeof m === "string") {
|
|
|
|
return "0px";
|
|
|
|
}
|
2021-03-25 21:02:39 +09:00
|
|
|
return Number.isInteger(m) ? `${m}px` : `${m.toFixed(2)}px`;
|
2021-03-19 18:11:40 +09:00
|
|
|
}
|
|
|
|
|
2021-03-25 21:02:39 +09:00
|
|
|
const converters = {
|
|
|
|
anchorType(node, style) {
|
|
|
|
if (!("transform" in style)) {
|
|
|
|
style.transform = "";
|
2021-03-19 18:11:40 +09:00
|
|
|
}
|
2021-03-25 21:02:39 +09:00
|
|
|
switch (node.anchorType) {
|
|
|
|
case "bottomCenter":
|
|
|
|
style.transform += "translate(-50%, -100%)";
|
|
|
|
break;
|
|
|
|
case "bottomLeft":
|
|
|
|
style.transform += "translate(0,-100%)";
|
|
|
|
break;
|
|
|
|
case "bottomRight":
|
|
|
|
style.transform += "translate(-100%,-100%)";
|
|
|
|
break;
|
|
|
|
case "middleCenter":
|
|
|
|
style.transform += "translate(-50%,-50%)";
|
|
|
|
break;
|
|
|
|
case "middleLeft":
|
|
|
|
style.transform += "translate(0,-50%)";
|
|
|
|
break;
|
|
|
|
case "middleRight":
|
|
|
|
style.transform += "translate(-100%,-50%)";
|
|
|
|
break;
|
|
|
|
case "topCenter":
|
|
|
|
style.transform += "translate(-50%,0)";
|
|
|
|
break;
|
|
|
|
case "topRight":
|
|
|
|
style.transform += "translate(-100%,0)";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
dimensions(node, style) {
|
|
|
|
if (node.w) {
|
|
|
|
style.width = measureToString(node.w);
|
|
|
|
} else {
|
2021-03-31 22:42:21 +09:00
|
|
|
style.width = "auto";
|
|
|
|
if (node.maxW > 0) {
|
2021-03-25 21:02:39 +09:00
|
|
|
style.maxWidth = measureToString(node.maxW);
|
|
|
|
}
|
2021-03-31 22:42:21 +09:00
|
|
|
style.minWidth = measureToString(node.minW);
|
2021-03-19 18:11:40 +09:00
|
|
|
}
|
|
|
|
|
2021-03-25 21:02:39 +09:00
|
|
|
if (node.h) {
|
|
|
|
style.height = measureToString(node.h);
|
|
|
|
} else {
|
2021-03-31 22:42:21 +09:00
|
|
|
style.height = "auto";
|
|
|
|
if (node.maxH > 0) {
|
2021-03-25 21:02:39 +09:00
|
|
|
style.maxHeight = measureToString(node.maxH);
|
|
|
|
}
|
2021-03-31 22:42:21 +09:00
|
|
|
style.minHeight = measureToString(node.minH);
|
2021-03-19 18:11:40 +09:00
|
|
|
}
|
2021-03-25 21:02:39 +09:00
|
|
|
},
|
|
|
|
position(node, style) {
|
2021-03-31 22:42:21 +09:00
|
|
|
const parent = node[$getParent]();
|
|
|
|
if (parent && parent.layout && parent.layout !== "position") {
|
|
|
|
// IRL, we've some x/y in tb layout.
|
|
|
|
// Specs say x/y is only used in positioned layout.
|
|
|
|
return;
|
2021-03-19 18:11:40 +09:00
|
|
|
}
|
2021-03-31 22:42:21 +09:00
|
|
|
|
|
|
|
style.position = "absolute";
|
|
|
|
style.left = measureToString(node.x);
|
|
|
|
style.top = measureToString(node.y);
|
2021-03-25 21:02:39 +09:00
|
|
|
},
|
|
|
|
rotate(node, style) {
|
|
|
|
if (node.rotate) {
|
|
|
|
if (!("transform" in style)) {
|
|
|
|
style.transform = "";
|
|
|
|
}
|
|
|
|
style.transform += `rotate(-${node.rotate}deg)`;
|
|
|
|
style.transformOrigin = "top left";
|
|
|
|
}
|
|
|
|
},
|
2021-03-31 22:42:21 +09:00
|
|
|
presence(node, style) {
|
|
|
|
switch (node.presence) {
|
|
|
|
case "invisible":
|
|
|
|
style.visibility = "hidden";
|
|
|
|
break;
|
|
|
|
case "hidden":
|
|
|
|
case "inactive":
|
|
|
|
style.display = "none";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
},
|
2021-03-25 21:02:39 +09:00
|
|
|
};
|
2021-03-19 18:11:40 +09:00
|
|
|
|
2021-03-31 22:42:21 +09:00
|
|
|
function layoutClass(node) {
|
|
|
|
switch (node.layout) {
|
|
|
|
case "position":
|
|
|
|
return "xfaPosition";
|
|
|
|
case "lr-tb":
|
|
|
|
return "xfaLrTb";
|
|
|
|
case "rl-row":
|
|
|
|
return "xfaRlRow";
|
|
|
|
case "rl-tb":
|
|
|
|
return "xfaRlTb";
|
|
|
|
case "row":
|
|
|
|
return "xfaRow";
|
|
|
|
case "table":
|
|
|
|
return "xfaTable";
|
|
|
|
case "tb":
|
|
|
|
return "xfaTb";
|
|
|
|
default:
|
|
|
|
return "xfaPosition";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-25 21:02:39 +09:00
|
|
|
function toStyle(node, ...names) {
|
|
|
|
const style = Object.create(null);
|
|
|
|
for (const name of names) {
|
|
|
|
const value = node[name];
|
|
|
|
if (value === null) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (value instanceof XFAObject) {
|
|
|
|
const newStyle = value[$toStyle]();
|
|
|
|
if (newStyle) {
|
|
|
|
Object.assign(style, newStyle);
|
|
|
|
} else {
|
|
|
|
warn(`(DEBUG) - XFA - style for ${name} not implemented yet`);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
2021-03-19 18:11:40 +09:00
|
|
|
|
2021-03-25 21:02:39 +09:00
|
|
|
if (converters.hasOwnProperty(name)) {
|
|
|
|
converters[name](node, style);
|
|
|
|
}
|
2021-03-19 18:11:40 +09:00
|
|
|
}
|
2021-03-25 21:02:39 +09:00
|
|
|
return style;
|
2021-03-19 18:11:40 +09:00
|
|
|
}
|
|
|
|
|
2021-03-31 22:42:21 +09:00
|
|
|
export { layoutClass, measureToString, toStyle };
|