| 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);  });});
 |