123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629 |
- 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("");
-
- 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, 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 () {
-
-
- const BORDER_WIDTH = 9;
- const SPACING = 2 * BORDER_WIDTH - 7;
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- 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;
- }
-
-
- 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 };
- }
-
-
-
- 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);
-
-
-
-
- for (let i = -size; i < size; i += 7) {
-
-
-
- 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, true);
- });
- it("works with horizontal scrolling with RTL-documents", function () {
- const pages = makePages([
- [
- [-10, 50],
- [-20, 20],
- [-30, 10],
- ],
- ]);
- scrollOverDocument(pages, true, 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(),
- });
- });
-
-
- describe("backtrackBeforeAllVisibleElements", function () {
-
- const tallPage = [10, 50];
- const shortPage = [10, 10];
-
-
- const top1 =
- 20 +
- SPACING +
- 40;
-
-
-
- const top2 =
- 20 +
- SPACING +
- 10;
-
-
- 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]],
- ]);
-
- 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],
- ],
- ]);
-
- 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],
- ],
- ]);
-
- 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],
- ],
- ]);
-
- const bsResult = 4;
- expect(
- backtrackBeforeAllVisibleElements(bsResult, pages, top2)
- ).toEqual(4);
- });
- });
- });
- });
|