chromium/third_party/blink/web_tests/fast/dom/shadow/shadow-boundary-crossing.html

<html>
<head>
<script>

var logDiv;

function log(msg, success)
{
    logDiv.appendChild(document.createElement('div')).textContent = msg + ': ' + (success ? 'PASS' : 'FAIL');
}

function moveOver(element)
{
    if (!window.eventSender)
        return;

    var x = element.offsetLeft + element.offsetWidth / 2;
    var y = element.offsetTop + element.offsetHeight / 2;
    eventSender.mouseMoveTo(x, y);
}

function moveOverLeftQuarterOf(element)
{
    if (!window.eventSender)
        return;

    var x = element.offsetLeft + element.offsetWidth / 4;
    var y = element.offsetTop + element.offsetHeight / 2;
    eventSender.mouseMoveTo(x, y);
}

function moveOverRightQuarterOf(element)
{
    if (!window.eventSender)
        return;

    var x = element.offsetLeft + element.offsetWidth * 3 / 4;
    var y = element.offsetTop + element.offsetHeight / 2;
    eventSender.mouseMoveTo(x, y);
}

function clickOn(element)
{
    moveOver(element);
    eventSender.mouseDown();
    eventSender.mouseUp();
}

function clickOnLeftQuarterOf(element)
{
    if (!window.eventSender)
        return;

    moveOverLeftQuarterOf(element);
    eventSender.mouseDown();
    eventSender.mouseUp();
}

function leapForward()
{
    if (!window.eventSender)
        return;

    eventSender.leapForward(1000);
}

var tests = {
    mutationEventPropagation: function()
    {
        var textarea = document.body.appendChild(document.createElement('textarea'));
        var mutationEventFired;
        textarea.addEventListener('DOMSubtreeModified', function(e)
        {
            mutationEventFired = true;
        }, false);
        textarea.value = 'test';
        // Trigger style recalc and sadly, the actual mutation of the textarea shadow DOM.
        textarea.offsetHeight;
        log('Mutation events should not propagate out of the shadow DOM', !mutationEventFired);
        textarea.parentNode.removeChild(textarea);
    },
    selectstartEventPropagation: function()
    {
        var textInput = document.body.appendChild(document.createElement('input'));
        var selectstartEventFired = false;
        document.selectstart = function()
        {
            selectstartEventFired = true;
        }
        clickOn(textInput);
        log('The selectstart event should not propagate out of the shadow DOM', !selectstartEventFired);
        textInput.parentNode.removeChild(textInput);
        document.selectstart = null;
    },
    mouseOverAndOutPropagation: function()
    {
        var count = 0;
        var fileInput = document.body.appendChild(document.createElement('input'));
        fileInput.setAttribute('type', 'file');
        var countEventDispatch = function()
        {
            count++;
        }
        moveOverLeftQuarterOf(fileInput);

        document.body.addEventListener('mouseover', countEventDispatch, false);
        document.body.addEventListener('mouseout', countEventDispatch, false);

        moveOverRightQuarterOf(fileInput);

        log("The mouseover/mouseout event between two elements inside the same shadow subtree should not propagate out of the shadow DOM", count == 0);

        document.body.removeEventListener('mouseover', countEventDispatch, false);
        document.body.removeEventListener('mouseout', countEventDispatch, false);
        fileInput.parentNode.removeChild(fileInput);
    },
    relatedTargetAsHost: function()
    {
        var count = 0;
        var relatedTarget = document.createElement('div');
        relatedTarget.style.cssText = 'width: 50px; height: 50px; padding-left: 50px;';
        document.body.appendChild(relatedTarget);
        var target = document.createElement('div');
        target.style.cssText = 'width: 50px; height: 50px';
        relatedTarget.attachShadow({mode: 'open'}).appendChild(target);
        moveOverLeftQuarterOf(relatedTarget);
        var countEventDispatch = function()
        {
            count++;
        }
        relatedTarget.addEventListener('mouseover', countEventDispatch, false)
        moveOverRightQuarterOf(relatedTarget);

        log("The mouseover event in a shadow subtree, where related target is the tree host should not escape out of shadow DOM", count == 0);

        relatedTarget.removeEventListener('mouseover', countEventDispatch, false);
        document.body.removeChild(relatedTarget);
    },
    targetAsHost: function()
    {
        var count = 0;
        var target = document.createElement('div');
        target.style.cssText = 'width: 50px; height: 50px; padding-left: 50px;';
        document.body.appendChild(target);
        var relatedTarget = document.createElement('div');
        relatedTarget.style.cssText = 'width: 50px; height: 50px';
        target.attachShadow({mode: 'open'}).appendChild(relatedTarget);
        moveOverRightQuarterOf(target);
        var countEventDispatch = function(evt)
        {
            count++;
        }
        target.addEventListener('mouseover', countEventDispatch, false)
        moveOverLeftQuarterOf(target);

        log("Events with relatedTarget should not escape out of shadow subtree when its host is the target", count == 0);

        target.removeEventListener('mouseout', countEventDispatch, false);
        document.body.removeChild(target);
    },
    mouseOverOnHost: function()
    {
        var count = 0;
        var input = document.body.appendChild(document.createElement('input'));
        var countEventDispatch = function()
        {
            count++;
        }

        moveOver(document.body);
        input.addEventListener('mouseover', countEventDispatch, false);
        moveOver(input);

        log("The mouseover/mouseout event on a shadow subtree host should propagate out of the shadow DOM", count == 1);

        document.body.removeEventListener('mouseover', countEventDispatch, false);
        input.parentNode.removeChild(input);
    },
    labelSyntheticClick: function()
    {
        var count = 0;
        var label = document.body.appendChild(document.createElement('label'));
        var searchInput = label.appendChild(document.createElement('input'));
        searchInput.setAttribute('type', 'search');
        searchInput.setAttribute('id', 'baz');
        label.setAttribute('for', 'baz');
        searchInput.addEventListener('click', function(e)
        {
            count++;
        }, false);
        clickOn(searchInput);
        log("Label should look beyond shadow boundary to detect if it encloses its associated element", count == 1);
        label.parentNode.removeChild(label);
    },
    /* This subtest started crashing after r89007:
     *   https://bugs.webkit.org/show_bug.cgi?id=62788
     * I'm disabling this test for now while I ask for help understanding the problem.
    defaultEventRetargeting: function()
    {
        var count = 0;
        var fileInput = document.body.appendChild(document.createElement('input'));
        fileInput.setAttribute('type', 'file');
        var counter = function()
        {
            count++;
        }
        document.body.addEventListener('DOMActivate', counter, false);
        clickOnLeftQuarterOf(fileInput);
        log("Events for default event handler should not be retargeted", count == 1);
        document.body.removeEventListener('DOMActivate', counter, false);
        fileInput.parentNode.removeChild(fileInput);
    },
    */
    relatedTargetRetargeting: function()
    {
        var count = 0;
        var textInput = document.body.appendChild(document.createElement('input'));
        var counter = function(evt)
        {
            if (evt.relatedTarget && !evt.relatedTarget.parentNode)
                count++;
        }
        moveOver(textInput);
        document.body.addEventListener("mouseover", counter, false);
        moveOver(document.body);
        document.body.removeEventListener("mouseover", counter, false);
        log("Event's relatedTarget should be retargeted", count == 0);
        textInput.parentNode.removeChild(textInput);
    },
    eventInProgress: function()
    {
        var textInput = document.body.appendChild(document.createElement('input'));
        textInput.addEventListener('click', function(e)
        {
            log('Other events should be retargeted', e.target == textInput);
        }, false);
        clickOn(textInput);
        textInput.parentNode.removeChild(textInput);
    },
    finalEventObject: function()
    {
        var textInput = document.body.appendChild(document.createElement('input'));
        var storedEvent;
        textInput.addEventListener('click', function(e)
        {
            storedEvent = e;
        }, false);
        clickOn(textInput);
        log('After event dispatch, the event object should not reveal shadow DOM', storedEvent && storedEvent.target == textInput);
        textInput.parentNode.removeChild(textInput);
    },
    focusEventPropagation: function()
    {
        var searchInput = document.body.appendChild(document.createElement('input'));
        searchInput.setAttribute('type', 'search');
        var count = 0;
        searchInput.addEventListener('focus', function(evt)
        {
            count++;
        });
        clickOn(searchInput);
        leapForward();
        clickOn(searchInput);
        log('Focusing same shadow DOM element repeatedly should not trigger multiple focus/blur events', count == 1);
        searchInput.parentNode.removeChild(searchInput);
    }
};

function runTest()
{
    if (window.testRunner)
        testRunner.dumpAsText();

    logDiv = document.getElementById('log');
    for(var testName in tests) {
        tests[testName]();
    }
}

</script>
</head>
<body onload="runTest()">
    <p>Tests to ensure that shadow DOM boundary is not crossed during event propagation. Can only run within DRT.
    <p>See <a href="https://bugs.webkit.org/show_bug.cgi?id=46015">bug 46015</a> for details.
    <div id="log"></div>
</body>
</html>