Source code
Revision control
Copy as Markdown
Other Tools
<!DOCTYPE HTML>
<html>
<head>
  <meta charset="utf-8">
  <meta name="viewport" content="width=device-width; initial-scale=1.0">
  <title>Complex touch-action test</title>
  <script type="application/javascript" src="apz_test_native_event_utils.js"></script>
  <script type="application/javascript" src="apz_test_utils.js"></script>
  <script src="/tests/SimpleTest/paint_listener.js"></script>
  <script type="application/javascript">
function checkScroll(target, x, y, desc) {
  is(target.scrollLeft, x, desc + " - x axis");
  is(target.scrollTop, y, desc + " - y axis");
}
async function resetConfiguration(config) {
  // Cycle through all the configuration_X elements, setting them to display:none
  // except for when X == config, in which case set it to display:block
  var i = 0;
  while (true) {
    i++;
    var element = document.getElementById("configuration_" + i);
    if (element == null) {
      if (i <= config) {
        ok(false, "The configuration requested was not encountered!");
      }
      break;
    }
    if (i == config) {
      element.style.display = "block";
    } else {
      element.style.display = "none";
    }
  }
  // Also reset the scroll position on the scrollframe
  var s = document.getElementById("scrollframe");
  s.scrollLeft = 0;
  s.scrollTop = 0;
  await promiseAllPaintsDone();
  await promiseApzFlushedRepaints();
}
async function test() {
  var scrollframe = document.getElementById("scrollframe");
  // Helper function for the tests below.
  // Touch-pan configuration |configuration| towards scroll offset (dx, dy) with
  // the pan touching down at (x, y). Check that the final scroll offset is
  // (ex, ey). |desc| is some description string.
  async function scrollAndCheck(configuration, x, y, dx, dy, ex, ey, desc) {
    // Start with a clean slate
    await resetConfiguration(configuration);
    // Reverse the touch delta in order to scroll in the desired direction
    dx = -dx;
    dy = -dy;
    // Do the pan
    let touchEndPromise = promiseTouchEnd(document.body);
    ok(await synthesizeNativeTouchDrag(scrollframe, x, y, dx, dy),
        "Synthesized drag of (" + dx + ", " + dy + ") on configuration " + configuration);
    await touchEndPromise;
    await promiseAllPaintsDone();
    await promiseOnlyApzControllerFlushed();
    // Check for expected scroll position
    checkScroll(scrollframe, ex, ey, "configuration " + configuration + " " + desc);
  }
  // Test configuration_1, which contains two sibling elements that are
  // overlapping. The touch-action from the second sibling (which is on top)
  // should be used for the overlapping area.
  await scrollAndCheck(1, 25, 75, 20, 0, 20, 0, "first element horizontal scroll");
  await scrollAndCheck(1, 25, 75, 0, 50, 0, 0, "first element vertical scroll");
  await scrollAndCheck(1, 75, 75, 50, 0, 0, 0, "overlap horizontal scroll");
  await scrollAndCheck(1, 75, 75, 0, 50, 0, 50, "overlap vertical scroll");
  await scrollAndCheck(1, 125, 75, 20, 0, 0, 0, "second element horizontal scroll");
  await scrollAndCheck(1, 125, 75, 0, 50, 0, 50, "second element vertical scroll");
  // Test configuration_2, which contains two overlapping elements with a
  // parent/child relationship. The parent has pan-x and the child has pan-y,
  // which means that panning on the parent should work horizontally only, and
  // on the child no panning should occur at all.
  await scrollAndCheck(2, 125, 125, 50, 50, 0, 0, "child scroll");
  await scrollAndCheck(2, 75, 75, 50, 50, 0, 0, "overlap scroll");
  await scrollAndCheck(2, 25, 75, 0, 50, 0, 0, "parent vertical scroll");
  await scrollAndCheck(2, 75, 25, 50, 0, 50, 0, "parent horizontal scroll");
  // Test configuration_3, which is the same as configuration_2, except the child
  // has a rotation transform applied. This forces the event regions on the two
  // elements to be built separately and then get merged.
  await scrollAndCheck(3, 125, 125, 50, 50, 0, 0, "child scroll");
  await scrollAndCheck(3, 75, 75, 50, 50, 0, 0, "overlap scroll");
  await scrollAndCheck(3, 25, 75, 0, 50, 0, 0, "parent vertical scroll");
  await scrollAndCheck(3, 75, 25, 50, 0, 50, 0, "parent horizontal scroll");
  // Test configuration_4 has two elements, one above the other, not overlapping,
  // and the second element is a child of the first. The parent has pan-x, the
  // child has pan-y, but that means panning horizontally on the parent should
  // work and panning in any direction on the child should not do anything.
  await scrollAndCheck(4, 75, 75, 50, 50, 50, 0, "parent diagonal scroll");
  await scrollAndCheck(4, 75, 150, 50, 50, 0, 0, "child diagonal scroll");
}
waitUntilApzStable()
.then(test)
.then(subtestDone, subtestFailed);
  </script>
</head>
<body>
 <div id="scrollframe" style="width: 300px; height: 300px; overflow:scroll">
  <div id="scrolled_content" style="width: 1000px; height: 1000px; background-color: green">
  </div>
  <div id="configuration_1" style="display:none; position: relative; top: -1000px">
   <div style="touch-action: pan-x; width: 100px; height: 100px; background-color: blue"></div>
   <div style="touch-action: pan-y; width: 100px; height: 100px; position: relative; top: -100px; left: 50px; background-color: yellow"></div>
  </div>
  <div id="configuration_2" style="display:none; position: relative; top: -1000px">
   <div style="touch-action: pan-x; width: 100px; height: 100px; background-color: blue">
    <div style="touch-action: pan-y; width: 100px; height: 100px; position: relative; top: 50px; left: 50px; background-color: yellow"></div>
   </div>
  </div>
  <div id="configuration_3" style="display:none; position: relative; top: -1000px">
   <div style="touch-action: pan-x; width: 100px; height: 100px; background-color: blue">
    <div style="touch-action: pan-y; width: 100px; height: 100px; position: relative; top: 50px; left: 50px; background-color: yellow; transform: rotate(90deg)"></div>
   </div>
  </div>
  <div id="configuration_4" style="display:none; position: relative; top: -1000px">
   <div style="touch-action: pan-x; width: 100px; height: 100px; background-color: blue">
    <div style="touch-action: pan-y; width: 100px; height: 100px; position: relative; top: 125px; background-color: yellow"></div>
   </div>
  </div>
 </div>
</body>
</html>