123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649 |
- /* Copyright 2020 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.
- */
- import { isNodeJS } from "../../src/shared/is_node.js";
- import { XFAFactory } from "../../src/core/xfa/factory.js";
- describe("XFAFactory", function () {
- function searchHtmlNode(root, name, value, byAttributes = false, nth = [0]) {
- if (
- (!byAttributes && root[name] === value) ||
- (byAttributes && root.attributes && root.attributes[name] === value)
- ) {
- if (nth[0]-- === 0) {
- return root;
- }
- }
- if (!root.children) {
- return null;
- }
- for (const child of root.children) {
- const node = searchHtmlNode(child, name, value, byAttributes, nth);
- if (node) {
- return node;
- }
- }
- return null;
- }
- describe("toHTML", function () {
- it("should convert some basic properties to CSS", async () => {
- const xml = `
- <?xml version="1.0"?>
- <xdp:xdp xmlns:xdp="http://ns.adobe.com/xdp/">
- <template xmlns="http://www.xfa.org/schema/xfa-template/3.3">
- <subform name="root" mergeMode="matchTemplate">
- <pageSet>
- <pageArea>
- <contentArea x="123pt" w="456pt" h="789pt"/>
- <medium stock="default" short="456pt" long="789pt"/>
- <draw y="1pt" w="11pt" h="22pt" rotate="90" x="2pt">
- <assist><toolTip>A tooltip !!</toolTip></assist>
- <font size="7pt" typeface="FooBar" baselineShift="2pt">
- <fill>
- <color value="12,23,34"/>
- <solid/>
- </fill>
- </font>
- <value/>
- <margin topInset="1pt" bottomInset="2pt" leftInset="3pt" rightInset="4pt"/>
- <para spaceAbove="1pt" spaceBelow="2pt" textIndent="3pt" marginLeft="4pt" marginRight="5pt"/>
- </draw>
- </pageArea>
- </pageSet>
- <subform name="second">
- <breakBefore targetType="pageArea" startNew="1"/>
- <subform>
- <draw w="1pt" h="1pt"><value><text>foo</text></value></draw>
- </subform>
- </subform>
- <subform name="third">
- <breakBefore targetType="pageArea" startNew="1"/>
- <subform>
- <draw w="1pt" h="1pt"><value><text>bar</text></value></draw>
- </subform>
- </subform>
- </subform>
- </template>
- <xfa:datasets xmlns:xfa="http://www.xfa.org/schema/xfa-data/1.0/">
- <xfa:data>
- </xfa:data>
- </xfa:datasets>
- </xdp:xdp>
- `;
- const factory = new XFAFactory({ "xdp:xdp": xml });
- factory.setFonts([]);
- expect(await factory.getNumPages()).toEqual(2);
- const pages = await factory.getPages();
- const page1 = pages.children[0];
- expect(page1.attributes.style).toEqual({
- height: "789px",
- width: "456px",
- });
- expect(page1.children.length).toEqual(2);
- const container = page1.children[1];
- expect(container.attributes.class).toEqual(["xfaContentarea"]);
- expect(container.attributes.style).toEqual({
- height: "789px",
- width: "456px",
- left: "123px",
- top: "0px",
- });
- const wrapper = page1.children[0];
- const draw = wrapper.children[0];
- expect(wrapper.attributes.class).toEqual(["xfaWrapper"]);
- expect(wrapper.attributes.style).toEqual({
- alignSelf: "start",
- height: "22px",
- left: "2px",
- position: "absolute",
- top: "1px",
- transform: "rotate(-90deg)",
- transformOrigin: "top left",
- width: "11px",
- });
- expect(draw.attributes.class).toEqual([
- "xfaDraw",
- "xfaFont",
- "xfaWrapped",
- ]);
- expect(draw.attributes.title).toEqual("A tooltip !!");
- expect(draw.attributes.style).toEqual({
- color: "#0c1722",
- fontFamily: '"FooBar"',
- fontKerning: "none",
- letterSpacing: "0px",
- fontStyle: "normal",
- fontWeight: "normal",
- fontSize: "6.93px",
- padding: "1px 4px 2px 3px",
- verticalAlign: "2px",
- });
- // draw element must be on each page.
- expect(draw.attributes.style).toEqual(
- pages.children[1].children[0].children[0].attributes.style
- );
- });
- it("should have an alt attribute from toolTip", async () => {
- if (isNodeJS) {
- pending("Image is not supported in Node.js.");
- }
- const xml = `
- <?xml version="1.0"?>
- <xdp:xdp xmlns:xdp="http://ns.adobe.com/xdp/">
- <template xmlns="http://www.xfa.org/schema/xfa-template/3.3">
- <subform name="root" mergeMode="matchTemplate">
- <pageSet>
- <pageArea>
- <contentArea x="0pt" w="456pt" h="789pt"/>
- <draw name="BA-Logo" y="5.928mm" x="128.388mm" w="71.237mm" h="9.528mm">
- <value>
- <image contentType="image/png">iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAQAAAC1HAwCAAAAC0lEQVQYV2NgYAAAAAMAAWgmWQ0AAAAASUVORK5CYII=</image>
- </value>
- <assist><toolTip>alt text</toolTip></assist>
- </draw>
- </pageArea>
- </pageSet>
- </subform>
- </template>
- <xfa:datasets xmlns:xfa="http://www.xfa.org/schema/xfa-data/1.0/">
- <xfa:data>
- </xfa:data>
- </xfa:datasets>
- </xdp:xdp>
- `;
- const factory = new XFAFactory({ "xdp:xdp": xml });
- expect(await factory.getNumPages()).toEqual(1);
- const pages = await factory.getPages();
- const field = searchHtmlNode(pages, "name", "img");
- expect(field.attributes.alt).toEqual("alt text");
- });
- it("should have a aria heading role and level", async () => {
- const xml = `
- <?xml version="1.0"?>
- <xdp:xdp xmlns:xdp="http://ns.adobe.com/xdp/">
- <template xmlns="http://www.xfa.org/schema/xfa-template/3.3">
- <subform name="root" mergeMode="matchTemplate">
- <pageSet>
- <pageArea>
- <contentArea x="0pt" w="456pt" h="789pt"/>
- <medium stock="default" short="456pt" long="789pt"/>
- <draw name="BA-Logo" y="5.928mm" x="128.388mm" w="71.237mm" h="9.528mm">
- <value><text>foo</text></value>
- <assist role="H2"></assist>
- </draw>
- </pageArea>
- </pageSet>
- </subform>
- </template>
- <xfa:datasets xmlns:xfa="http://www.xfa.org/schema/xfa-data/1.0/">
- <xfa:data>
- </xfa:data>
- </xfa:datasets>
- </xdp:xdp>
- `;
- const factory = new XFAFactory({ "xdp:xdp": xml });
- expect(await factory.getNumPages()).toEqual(1);
- const pages = await factory.getPages();
- const page1 = pages.children[0];
- const wrapper = page1.children[0];
- const draw = wrapper.children[0];
- expect(draw.attributes.role).toEqual("heading");
- expect(draw.attributes["aria-level"]).toEqual("2");
- });
- it("should have aria table role", async () => {
- const xml = `
- <?xml version="1.0"?>
- <xdp:xdp xmlns:xdp="http://ns.adobe.com/xdp/">
- <template xmlns="http://www.xfa.org/schema/xfa-template/3.3">
- <subform name="root" mergeMode="matchTemplate">
- <pageSet>
- <pageArea>
- <contentArea x="0pt" w="456pt" h="789pt"/>
- <medium stock="default" short="456pt" long="789pt"/>
- <font size="7pt" typeface="FooBar" baselineShift="2pt">
- </font>
- </pageArea>
- </pageSet>
- <subform name="table" mergeMode="matchTemplate" layout="table">
- <subform layout="row" name="row1">
- <assist role="TH"></assist>
- <draw name="header1" y="5.928mm" x="128.388mm" w="71.237mm" h="9.528mm">
- <value><text>Header Col 1</text></value>
- </draw>
- <draw name="header2" y="5.928mm" x="128.388mm" w="71.237mm" h="9.528mm">
- <value><text>Header Col 2</text></value>
- </draw>
- </subform>
- <subform layout="row" name="row2">
- <draw name="cell1" y="5.928mm" x="128.388mm" w="71.237mm" h="9.528mm">
- <value><text>Cell 1</text></value>
- </draw>
- <draw name="cell2" y="5.928mm" x="128.388mm" w="71.237mm" h="9.528mm">
- <value><text>Cell 2</text></value>
- </draw>
- </subform>
- </subform>
- </subform>
- </template>
- <xfa:datasets xmlns:xfa="http://www.xfa.org/schema/xfa-data/1.0/">
- <xfa:data>
- </xfa:data>
- </xfa:datasets>
- </xdp:xdp>
- `;
- const factory = new XFAFactory({ "xdp:xdp": xml });
- factory.setFonts([]);
- expect(await factory.getNumPages()).toEqual(1);
- const pages = await factory.getPages();
- const table = searchHtmlNode(
- pages,
- "xfaName",
- "table",
- /* byAttributes */ true
- );
- expect(table.attributes.role).toEqual("table");
- const headerRow = searchHtmlNode(
- pages,
- "xfaName",
- "row1",
- /* byAttributes */ true
- );
- expect(headerRow.attributes.role).toEqual("row");
- const headerCell = searchHtmlNode(
- pages,
- "xfaName",
- "header2",
- /* byAttributes */ true
- );
- expect(headerCell.attributes.role).toEqual("columnheader");
- const row = searchHtmlNode(
- pages,
- "xfaName",
- "row2",
- /* byAttributes */ true
- );
- expect(row.attributes.role).toEqual("row");
- const cell = searchHtmlNode(
- pages,
- "xfaName",
- "cell2",
- /* byAttributes */ true
- );
- expect(cell.attributes.role).toEqual("cell");
- });
- it("should have a maxLength property", async () => {
- const xml = `
- <?xml version="1.0"?>
- <xdp:xdp xmlns:xdp="http://ns.adobe.com/xdp/">
- <template xmlns="http://www.xfa.org/schema/xfa-template/3.3">
- <subform name="root" mergeMode="matchTemplate">
- <pageSet>
- <pageArea>
- <contentArea x="0pt" w="456pt" h="789pt"/>
- <medium stock="default" short="456pt" long="789pt"/>
- <field y="1pt" w="11pt" h="22pt" x="2pt">
- <ui>
- <textEdit multiLine="0"/>
- </ui>
- <value>
- <text maxChars="123"/>
- </value>
- </field>
- </pageArea>
- </pageSet>
- <subform name="first">
- <draw w="1pt" h="1pt"><value><text>foo</text></value></draw>
- </subform>
- </subform>
- </template>
- <xfa:datasets xmlns:xfa="http://www.xfa.org/schema/xfa-data/1.0/">
- <xfa:data>
- </xfa:data>
- </xfa:datasets>
- </xdp:xdp>
- `;
- const factory = new XFAFactory({ "xdp:xdp": xml });
- expect(await factory.getNumPages()).toEqual(1);
- const pages = await factory.getPages();
- const field = searchHtmlNode(pages, "name", "input");
- expect(field.attributes.maxLength).toEqual(123);
- });
- it("should have an aria-label property from speak", async () => {
- const xml = `
- <?xml version="1.0"?>
- <xdp:xdp xmlns:xdp="http://ns.adobe.com/xdp/">
- <template xmlns="http://www.xfa.org/schema/xfa-template/3.3">
- <subform name="root" mergeMode="matchTemplate">
- <pageSet>
- <pageArea>
- <contentArea x="0pt" w="456pt" h="789pt"/>
- <medium stock="default" short="456pt" long="789pt"/>
- <field y="1pt" w="11pt" h="22pt" x="2pt">
- <assist><speak>Screen Reader</speak></assist>
- <ui>
- <textEdit multiLine="0"/>
- </ui>
- <value>
- <text maxChars="123"/>
- </value>
- </field>
- </pageArea>
- </pageSet>
- <subform name="first">
- <draw w="1pt" h="1pt"><value><text>foo</text></value></draw>
- </subform>
- </subform>
- </template>
- <xfa:datasets xmlns:xfa="http://www.xfa.org/schema/xfa-data/1.0/">
- <xfa:data>
- </xfa:data>
- </xfa:datasets>
- </xdp:xdp>
- `;
- const factory = new XFAFactory({ "xdp:xdp": xml });
- expect(await factory.getNumPages()).toEqual(1);
- const pages = await factory.getPages();
- const field = searchHtmlNode(pages, "name", "input");
- expect(field.attributes["aria-label"]).toEqual("Screen Reader");
- });
- it("should have an aria-label property from toolTip", async () => {
- const xml = `
- <?xml version="1.0"?>
- <xdp:xdp xmlns:xdp="http://ns.adobe.com/xdp/">
- <template xmlns="http://www.xfa.org/schema/xfa-template/3.3">
- <subform name="root" mergeMode="matchTemplate">
- <pageSet>
- <pageArea>
- <contentArea x="0pt" w="456pt" h="789pt"/>
- <medium stock="default" short="456pt" long="789pt"/>
- <field y="1pt" w="11pt" h="22pt" x="2pt">
- <assist><toolTip>Screen Reader</toolTip></assist>
- <ui>
- <textEdit multiLine="0"/>
- </ui>
- <value>
- <text maxChars="123"/>
- </value>
- </field>
- </pageArea>
- </pageSet>
- <subform name="first">
- <draw w="1pt" h="1pt"><value><text>foo</text></value></draw>
- </subform>
- </subform>
- </template>
- <xfa:datasets xmlns:xfa="http://www.xfa.org/schema/xfa-data/1.0/">
- <xfa:data>
- </xfa:data>
- </xfa:datasets>
- </xdp:xdp>
- `;
- const factory = new XFAFactory({ "xdp:xdp": xml });
- expect(await factory.getNumPages()).toEqual(1);
- const pages = await factory.getPages();
- const field = searchHtmlNode(pages, "name", "input");
- expect(field.attributes["aria-label"]).toEqual("Screen Reader");
- });
- it("should have an input or textarea", async () => {
- const xml = `
- <?xml version="1.0"?>
- <xdp:xdp xmlns:xdp="http://ns.adobe.com/xdp/">
- <template xmlns="http://www.xfa.org/schema/xfa-template/3.3">
- <subform name="root" mergeMode="matchTemplate">
- <pageSet>
- <pageArea>
- <contentArea x="123pt" w="456pt" h="789pt"/>
- <medium stock="default" short="456pt" long="789pt"/>
- <field y="1pt" w="11pt" h="22pt" x="2pt">
- <ui>
- <textEdit/>
- </ui>
- </field>
- <field y="1pt" w="11pt" h="22pt" x="2pt">
- <ui>
- <textEdit multiLine="1"/>
- </ui>
- </field>
- </pageArea>
- </pageSet>
- <subform name="first">
- <draw w="1pt" h="1pt"><value><text>foo</text></value></draw>
- </subform>
- </subform>
- </template>
- <xfa:datasets xmlns:xfa="http://www.xfa.org/schema/xfa-data/1.0/">
- <xfa:data>
- </xfa:data>
- </xfa:datasets>
- </xdp:xdp>
- `;
- const factory = new XFAFactory({ "xdp:xdp": xml });
- expect(await factory.getNumPages()).toEqual(1);
- const pages = await factory.getPages();
- const field1 = searchHtmlNode(pages, "name", "input");
- expect(field1).not.toEqual(null);
- const field2 = searchHtmlNode(pages, "name", "textarea");
- expect(field2).not.toEqual(null);
- });
- });
- it("should have an input or textarea", async () => {
- const xml = `
- <?xml version="1.0"?>
- <xdp:xdp xmlns:xdp="http://ns.adobe.com/xdp/">
- <template xmlns="http://www.xfa.org/schema/xfa-template/3.3">
- <subform name="root" mergeMode="matchTemplate">
- <pageSet>
- <pageArea>
- <contentArea x="123pt" w="456pt" h="789pt"/>
- <medium stock="default" short="456pt" long="789pt"/>
- <field y="1pt" w="11pt" h="22pt" x="2pt">
- <ui>
- <textEdit multiLine="1"/>
- </ui>
- </field>
- </pageArea>
- </pageSet>
- <subform name="first">
- <field y="1pt" w="11pt" h="22pt" x="2pt" name="hello">
- <ui>
- <textEdit/>
- </ui>
- <value>
- <integer/>
- </value>
- </field>
- </subform>
- </subform>
- </template>
- <xfa:datasets xmlns:xfa="http://www.xfa.org/schema/xfa-data/1.0/">
- <xfa:data>
- <toto>
- <first>
- <hello>123
- </hello>
- </first>
- </toto>
- </xfa:data>
- </xfa:datasets>
- </xdp:xdp>
- `;
- const factory = new XFAFactory({ "xdp:xdp": xml });
- expect(await factory.getNumPages()).toEqual(1);
- const pages = await factory.getPages();
- const field1 = searchHtmlNode(pages, "name", "input");
- expect(field1).not.toEqual(null);
- expect(field1.attributes.value).toEqual("123");
- });
- it("should parse URLs correctly", async () => {
- function getXml(href) {
- return `
- <?xml version="1.0"?>
- <xdp:xdp xmlns:xdp="http://ns.adobe.com/xdp/">
- <template xmlns="http://www.xfa.org/schema/xfa-template/3.3">
- <subform name="root" mergeMode="matchTemplate">
- <pageSet>
- <pageArea>
- <contentArea x="0pt" w="456pt" h="789pt"/>
- <medium stock="default" short="456pt" long="789pt"/>
- <draw name="url" y="5.928mm" x="128.388mm" w="71.237mm" h="9.528mm">
- <value>
- <exData contentType="text/html">
- <body xmlns="http://www.w3.org/1999/xhtml">
- <a href="${href}">${href}</a>
- </body>
- </exData>
- </value>
- </draw>
- </pageArea>
- </pageSet>
- </subform>
- </template>
- <xfa:datasets xmlns:xfa="http://www.xfa.org/schema/xfa-data/1.0/">
- <xfa:data>
- </xfa:data>
- </xfa:datasets>
- </xdp:xdp>
- `;
- }
- let factory, pages, a;
- // A valid, and complete, URL.
- factory = new XFAFactory({ "xdp:xdp": getXml("https://www.example.com/") });
- expect(await factory.getNumPages()).toEqual(1);
- pages = await factory.getPages();
- a = searchHtmlNode(pages, "name", "a");
- expect(a.value).toEqual("https://www.example.com/");
- expect(a.attributes.href).toEqual("https://www.example.com/");
- // A valid, but incomplete, URL.
- factory = new XFAFactory({ "xdp:xdp": getXml("www.example.com/") });
- expect(await factory.getNumPages()).toEqual(1);
- pages = await factory.getPages();
- a = searchHtmlNode(pages, "name", "a");
- expect(a.value).toEqual("www.example.com/");
- expect(a.attributes.href).toEqual("http://www.example.com/");
- // A valid email-address.
- factory = new XFAFactory({ "xdp:xdp": getXml("mailto:test@example.com") });
- expect(await factory.getNumPages()).toEqual(1);
- pages = await factory.getPages();
- a = searchHtmlNode(pages, "name", "a");
- expect(a.value).toEqual("mailto:test@example.com");
- expect(a.attributes.href).toEqual("mailto:test@example.com");
- // Not a valid URL.
- factory = new XFAFactory({ "xdp:xdp": getXml("qwerty/") });
- expect(await factory.getNumPages()).toEqual(1);
- pages = await factory.getPages();
- a = searchHtmlNode(pages, "name", "a");
- expect(a.value).toEqual("qwerty/");
- expect(a.attributes.href).toEqual("");
- });
- it("should replace button with an URL by a link", async () => {
- const xml = `
- <?xml version="1.0"?>
- <xdp:xdp xmlns:xdp="http://ns.adobe.com/xdp/">
- <template xmlns="http://www.xfa.org/schema/xfa-template/3.3">
- <subform name="root" mergeMode="matchTemplate">
- <pageSet>
- <pageArea>
- <contentArea x="123pt" w="456pt" h="789pt"/>
- <medium stock="default" short="456pt" long="789pt"/>
- </pageArea>
- </pageSet>
- <subform name="first">
- <field y="1pt" w="11pt" h="22pt" x="2pt">
- <ui>
- <button/>
- </ui>
- <event activity="click" name="event__click">
- <script contentType="application/x-javascript">
- app.launchURL("https://github.com/mozilla/pdf.js", true);
- </script>
- </event>
- </field>
- <field y="1pt" w="11pt" h="22pt" x="2pt">
- <ui>
- <button/>
- </ui>
- <event activity="click" name="event__click">
- <script contentType="application/x-javascript">
- xfa.host.gotoURL("https://github.com/allizom/pdf.js");
- </script>
- </event>
- </field>
- </subform>
- </subform>
- </template>
- <xfa:datasets xmlns:xfa="http://www.xfa.org/schema/xfa-data/1.0/">
- <xfa:data>
- </xfa:data>
- </xfa:datasets>
- </xdp:xdp>
- `;
- const factory = new XFAFactory({ "xdp:xdp": xml });
- expect(await factory.getNumPages()).toEqual(1);
- const pages = await factory.getPages();
- let a = searchHtmlNode(pages, "name", "a");
- expect(a.attributes.href).toEqual("https://github.com/mozilla/pdf.js");
- expect(a.attributes.newWindow).toEqual(true);
- a = searchHtmlNode(pages, "name", "a", false, [1]);
- expect(a.attributes.href).toEqual("https://github.com/allizom/pdf.js");
- expect(a.attributes.newWindow).toEqual(false);
- });
- });
|