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