| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629 | /* Copyright 2017 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 {  backtrackBeforeAllVisibleElements,  binarySearchFirstItem,  getPageSizeInches,  getVisibleElements,  isPortraitOrientation,  isValidRotation,  parseQueryString,  removeNullCharacters,} from "../../web/ui_utils.js";describe("ui_utils", function () {  describe("binary search", function () {    function isTrue(boolean) {      return boolean;    }    function isGreater3(number) {      return number > 3;    }    it("empty array", function () {      expect(binarySearchFirstItem([], isTrue)).toEqual(0);    });    it("single boolean entry", function () {      expect(binarySearchFirstItem([false], isTrue)).toEqual(1);      expect(binarySearchFirstItem([true], isTrue)).toEqual(0);    });    it("three boolean entries", function () {      expect(binarySearchFirstItem([true, true, true], isTrue)).toEqual(0);      expect(binarySearchFirstItem([false, true, true], isTrue)).toEqual(1);      expect(binarySearchFirstItem([false, false, true], isTrue)).toEqual(2);      expect(binarySearchFirstItem([false, false, false], isTrue)).toEqual(3);    });    it("three numeric entries", function () {      expect(binarySearchFirstItem([0, 1, 2], isGreater3)).toEqual(3);      expect(binarySearchFirstItem([2, 3, 4], isGreater3)).toEqual(2);      expect(binarySearchFirstItem([4, 5, 6], isGreater3)).toEqual(0);    });    it("three numeric entries and a start index", function () {      expect(binarySearchFirstItem([0, 1, 2, 3, 4], isGreater3, 2)).toEqual(4);      expect(binarySearchFirstItem([2, 3, 4], isGreater3, 2)).toEqual(2);      expect(binarySearchFirstItem([4, 5, 6], isGreater3, 1)).toEqual(1);    });  });  describe("isValidRotation", function () {    it("should reject non-integer angles", function () {      expect(isValidRotation()).toEqual(false);      expect(isValidRotation(null)).toEqual(false);      expect(isValidRotation(NaN)).toEqual(false);      expect(isValidRotation([90])).toEqual(false);      expect(isValidRotation("90")).toEqual(false);      expect(isValidRotation(90.5)).toEqual(false);    });    it("should reject non-multiple of 90 degree angles", function () {      expect(isValidRotation(45)).toEqual(false);      expect(isValidRotation(-123)).toEqual(false);    });    it("should accept valid angles", function () {      expect(isValidRotation(0)).toEqual(true);      expect(isValidRotation(90)).toEqual(true);      expect(isValidRotation(-270)).toEqual(true);      expect(isValidRotation(540)).toEqual(true);    });  });  describe("isPortraitOrientation", function () {    it("should be portrait orientation", function () {      expect(        isPortraitOrientation({          width: 200,          height: 400,        })      ).toEqual(true);      expect(        isPortraitOrientation({          width: 500,          height: 500,        })      ).toEqual(true);    });    it("should be landscape orientation", function () {      expect(        isPortraitOrientation({          width: 600,          height: 300,        })      ).toEqual(false);    });  });  describe("parseQueryString", function () {    it("should parse one key/value pair", function () {      const parameters = parseQueryString("key1=value1");      expect(parameters.size).toEqual(1);      expect(parameters.get("key1")).toEqual("value1");    });    it("should parse multiple key/value pairs", function () {      const parameters = parseQueryString(        "key1=value1&key2=value2&key3=value3"      );      expect(parameters.size).toEqual(3);      expect(parameters.get("key1")).toEqual("value1");      expect(parameters.get("key2")).toEqual("value2");      expect(parameters.get("key3")).toEqual("value3");    });    it("should parse keys without values", function () {      const parameters = parseQueryString("key1");      expect(parameters.size).toEqual(1);      expect(parameters.get("key1")).toEqual("");    });    it("should decode encoded key/value pairs", function () {      const parameters = parseQueryString("k%C3%ABy1=valu%C3%AB1");      expect(parameters.size).toEqual(1);      expect(parameters.get("këy1")).toEqual("valuë1");    });    it("should convert keys to lowercase", function () {      const parameters = parseQueryString("Key1=Value1&KEY2=Value2");      expect(parameters.size).toEqual(2);      expect(parameters.get("key1")).toEqual("Value1");      expect(parameters.get("key2")).toEqual("Value2");    });  });  describe("removeNullCharacters", function () {    it("should not modify string without null characters", function () {      const str = "string without null chars";      expect(removeNullCharacters(str)).toEqual("string without null chars");    });    it("should modify string with null characters", function () {      const str = "string\x00With\x00Null\x00Chars";      expect(removeNullCharacters(str)).toEqual("stringWithNullChars");    });    it("should modify string with non-displayable characters", function () {      const str = Array.from(Array(32).keys())        .map(x => String.fromCharCode(x) + "a")        .join("");      // \x00 is replaced by an empty string.      const expected =        "a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a a";      expect(removeNullCharacters(str, /* replaceInvisible */ true)).toEqual(        expected      );    });  });  describe("getPageSizeInches", function () {    it("gets page size (in inches)", function () {      const page = {        view: [0, 0, 595.28, 841.89],        userUnit: 1.0,        rotate: 0,      };      const { width, height } = getPageSizeInches(page);      expect(+width.toPrecision(3)).toEqual(8.27);      expect(+height.toPrecision(4)).toEqual(11.69);    });    it("gets page size (in inches), for non-default /Rotate entry", function () {      const pdfPage1 = { view: [0, 0, 612, 792], userUnit: 1, rotate: 0 };      const { width: width1, height: height1 } = getPageSizeInches(pdfPage1);      expect(width1).toEqual(8.5);      expect(height1).toEqual(11);      const pdfPage2 = { view: [0, 0, 612, 792], userUnit: 1, rotate: 90 };      const { width: width2, height: height2 } = getPageSizeInches(pdfPage2);      expect(width2).toEqual(11);      expect(height2).toEqual(8.5);    });  });  describe("getVisibleElements", function () {    // These values are based on margin/border values in the CSS, but there    // isn't any real need for them to be; they just need to take *some* value.    const BORDER_WIDTH = 9;    const SPACING = 2 * BORDER_WIDTH - 7;    // This is a helper function for assembling an array of view stubs from an    // array of arrays of [width, height] pairs, which represents wrapped lines    // of pages. It uses the above constants to add realistic spacing between    // the pages and the lines.    //    // If you're reading a test that calls makePages, you should think of the    // inputs to makePages as boxes with no borders, being laid out in a    // container that has no margins, so that the top of the tallest page in    // the first row will be at y = 0, and the left of the first page in    // the first row will be at x = 0. The spacing between pages in a row, and    // the spacing between rows, is SPACING. If you wanted to construct an    // actual HTML document with the same layout, you should give each page    // element a margin-right and margin-bottom of SPACING, and add no other    // margins, borders, or padding.    //    // If you're reading makePages itself, you'll see a somewhat more    // complicated picture because this suite of tests is exercising    // getVisibleElements' ability to account for the borders that real page    // elements have. makePages tests this by subtracting a BORDER_WIDTH from    // offsetLeft/Top and adding it to clientLeft/Top. So the element stubs that    // getVisibleElements sees may, for example, actually have an offsetTop of    // -9. If everything is working correctly, this detail won't leak out into    // the tests themselves, and so the tests shouldn't use the value of    // BORDER_WIDTH at all.    function makePages(lines) {      const result = [];      let lineTop = 0,        id = 0;      for (const line of lines) {        const lineHeight = line.reduce(function (maxHeight, pair) {          return Math.max(maxHeight, pair[1]);        }, 0);        let offsetLeft = -BORDER_WIDTH;        for (const [clientWidth, clientHeight] of line) {          const offsetTop =            lineTop + (lineHeight - clientHeight) / 2 - BORDER_WIDTH;          const div = {            offsetLeft,            offsetTop,            clientWidth,            clientHeight,            clientLeft: BORDER_WIDTH,            clientTop: BORDER_WIDTH,          };          result.push({ id, div });          ++id;          offsetLeft += clientWidth + SPACING;        }        lineTop += lineHeight + SPACING;      }      return result;    }    // This is a reimplementation of getVisibleElements without the    // optimizations.    function slowGetVisibleElements(scroll, pages) {      const views = [],        ids = new Set();      const { scrollLeft, scrollTop } = scroll;      const scrollRight = scrollLeft + scroll.clientWidth;      const scrollBottom = scrollTop + scroll.clientHeight;      for (const view of pages) {        const { div } = view;        const viewLeft = div.offsetLeft + div.clientLeft;        const viewRight = viewLeft + div.clientWidth;        const viewTop = div.offsetTop + div.clientTop;        const viewBottom = viewTop + div.clientHeight;        if (          viewLeft < scrollRight &&          viewRight > scrollLeft &&          viewTop < scrollBottom &&          viewBottom > scrollTop        ) {          const hiddenHeight =            Math.max(0, scrollTop - viewTop) +            Math.max(0, viewBottom - scrollBottom);          const hiddenWidth =            Math.max(0, scrollLeft - viewLeft) +            Math.max(0, viewRight - scrollRight);          const fractionHeight =            (div.clientHeight - hiddenHeight) / div.clientHeight;          const fractionWidth =            (div.clientWidth - hiddenWidth) / div.clientWidth;          const percent = (fractionHeight * fractionWidth * 100) | 0;          views.push({            id: view.id,            x: viewLeft,            y: viewTop,            view,            percent,            widthPercent: (fractionWidth * 100) | 0,          });          ids.add(view.id);        }      }      return { first: views[0], last: views.at(-1), views, ids };    }    // This function takes a fixed layout of pages and compares the system under    // test to the slower implementation above, for a range of scroll viewport    // sizes and positions.    function scrollOverDocument(pages, horizontal = false, rtl = false) {      const size = pages.reduce(function (max, { div }) {        return Math.max(          max,          horizontal            ? Math.abs(div.offsetLeft + div.clientLeft + div.clientWidth)            : div.offsetTop + div.clientTop + div.clientHeight        );      }, 0);      // The numbers (7 and 5) are mostly arbitrary, not magic: increase them to      // make scrollOverDocument tests faster, decrease them to make the tests      // more scrupulous, and keep them coprime to reduce the chance of missing      // weird edge case bugs.      for (let i = -size; i < size; i += 7) {        // The screen height (or width) here (j - i) doubles on each inner loop        // iteration; again, this is just to test an interesting range of cases        // without slowing the tests down to check every possible case.        for (let j = i + 5; j < size; j += j - i) {          const scrollEl = horizontal            ? {                scrollTop: 0,                scrollLeft: i,                clientHeight: 10000,                clientWidth: j - i,              }            : {                scrollTop: i,                scrollLeft: 0,                clientHeight: j - i,                clientWidth: 10000,              };          expect(            getVisibleElements({              scrollEl,              views: pages,              sortByVisibility: false,              horizontal,              rtl,            })          ).toEqual(slowGetVisibleElements(scrollEl, pages));        }      }    }    it("with pages of varying height", function () {      const pages = makePages([        [          [50, 20],          [20, 50],        ],        [          [30, 12],          [12, 30],        ],        [          [20, 50],          [50, 20],        ],        [          [50, 20],          [20, 50],        ],      ]);      scrollOverDocument(pages);    });    it("widescreen challenge", function () {      const pages = makePages([        [          [10, 50],          [10, 60],          [10, 70],          [10, 80],          [10, 90],        ],        [          [10, 90],          [10, 80],          [10, 70],          [10, 60],          [10, 50],        ],        [          [10, 50],          [10, 60],          [10, 70],          [10, 80],          [10, 90],        ],      ]);      scrollOverDocument(pages);    });    it("works with horizontal scrolling", function () {      const pages = makePages([        [          [10, 50],          [20, 20],          [30, 10],        ],      ]);      scrollOverDocument(pages, /* horizontal = */ true);    });    it("works with horizontal scrolling with RTL-documents", function () {      const pages = makePages([        [          [-10, 50],          [-20, 20],          [-30, 10],        ],      ]);      scrollOverDocument(pages, /* horizontal = */ true, /* rtl = */ true);    });    it("handles `sortByVisibility` correctly", function () {      const scrollEl = {        scrollTop: 75,        scrollLeft: 0,        clientHeight: 750,        clientWidth: 1500,      };      const views = makePages([[[100, 150]], [[100, 150]], [[100, 150]]]);      const visible = getVisibleElements({ scrollEl, views });      const visibleSorted = getVisibleElements({        scrollEl,        views,        sortByVisibility: true,      });      const viewsOrder = [],        viewsSortedOrder = [];      for (const view of visible.views) {        viewsOrder.push(view.id);      }      for (const view of visibleSorted.views) {        viewsSortedOrder.push(view.id);      }      expect(viewsOrder).toEqual([0, 1, 2]);      expect(viewsSortedOrder).toEqual([1, 2, 0]);    });    it("handles views being empty", function () {      const scrollEl = {        scrollTop: 10,        scrollLeft: 0,        clientHeight: 750,        clientWidth: 1500,      };      const views = [];      expect(getVisibleElements({ scrollEl, views })).toEqual({        first: undefined,        last: undefined,        views: [],        ids: new Set(),      });    });    it("handles all views being hidden (without errors)", function () {      const scrollEl = {        scrollTop: 100000,        scrollLeft: 0,        clientHeight: 750,        clientWidth: 1500,      };      const views = makePages([[[100, 150]], [[100, 150]], [[100, 150]]]);      expect(getVisibleElements({ scrollEl, views })).toEqual({        first: undefined,        last: undefined,        views: [],        ids: new Set(),      });    });    // This sub-suite is for a notionally internal helper function for    // getVisibleElements.    describe("backtrackBeforeAllVisibleElements", function () {      // Layout elements common to all tests      const tallPage = [10, 50];      const shortPage = [10, 10];      // A scroll position that ensures that only the tall pages in the second      // row are visible      const top1 =        20 +        SPACING + // height of the first row        40; // a value between 30 (so the short pages on the second row are      // hidden) and 50 (so the tall pages are visible)      // A scroll position that ensures that all of the pages in the second row      // are visible, but the tall ones are a tiny bit cut off      const top2 =        20 +        SPACING + // height of the first row        10; // a value greater than 0 but less than 30      // These tests refer to cases enumerated in the comments of      // backtrackBeforeAllVisibleElements.      it("handles case 1", function () {        const pages = makePages([          [            [10, 20],            [10, 20],            [10, 20],            [10, 20],          ],          [tallPage, shortPage, tallPage, shortPage],          [            [10, 50],            [10, 50],            [10, 50],            [10, 50],          ],          [            [10, 20],            [10, 20],            [10, 20],            [10, 20],          ],          [[10, 20]],        ]);        // binary search would land on the second row, first page        const bsResult = 4;        expect(          backtrackBeforeAllVisibleElements(bsResult, pages, top1)        ).toEqual(4);      });      it("handles case 2", function () {        const pages = makePages([          [            [10, 20],            [10, 20],            [10, 20],            [10, 20],          ],          [tallPage, shortPage, tallPage, tallPage],          [            [10, 50],            [10, 50],            [10, 50],            [10, 50],          ],          [            [10, 20],            [10, 20],            [10, 20],            [10, 20],          ],        ]);        // binary search would land on the second row, third page        const bsResult = 6;        expect(          backtrackBeforeAllVisibleElements(bsResult, pages, top1)        ).toEqual(4);      });      it("handles case 3", function () {        const pages = makePages([          [            [10, 20],            [10, 20],            [10, 20],            [10, 20],          ],          [tallPage, shortPage, tallPage, shortPage],          [            [10, 50],            [10, 50],            [10, 50],            [10, 50],          ],          [            [10, 20],            [10, 20],            [10, 20],            [10, 20],          ],        ]);        // binary search would land on the third row, first page        const bsResult = 8;        expect(          backtrackBeforeAllVisibleElements(bsResult, pages, top1)        ).toEqual(4);      });      it("handles case 4", function () {        const pages = makePages([          [            [10, 20],            [10, 20],            [10, 20],            [10, 20],          ],          [tallPage, shortPage, tallPage, shortPage],          [            [10, 50],            [10, 50],            [10, 50],            [10, 50],          ],          [            [10, 20],            [10, 20],            [10, 20],            [10, 20],          ],        ]);        // binary search would land on the second row, first page        const bsResult = 4;        expect(          backtrackBeforeAllVisibleElements(bsResult, pages, top2)        ).toEqual(4);      });    });  });});
 |