mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-01-08 04:16:16 +00:00
775 lines
31 KiB
XML
775 lines
31 KiB
XML
<?xml version="1.0"?>
|
|
|
|
<!-- This Source Code Form is subject to the terms of the Mozilla Public
|
|
- License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
- file, You can obtain one at http://mozilla.org/MPL/2.0/. -->
|
|
|
|
<?xml-stylesheet href="chrome://global/skin" type="text/css"?>
|
|
|
|
<window id="NativeMenuWindow"
|
|
xmlns="http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul"
|
|
xmlns:html="http://www.w3.org/1999/xhtml"
|
|
width="600"
|
|
height="600"
|
|
title="Native Mouse Event Test"
|
|
orient="vertical">
|
|
<script type="application/javascript"
|
|
src="chrome://mochikit/content/tests/SimpleTest/SimpleTest.js" />
|
|
|
|
<box height="200" id="box"/>
|
|
<menupopup id="popup" width="250" height="50"/>
|
|
<panel id="panel" width="250" height="50" noautohide="true"/>
|
|
|
|
<script type="application/javascript"><![CDATA[
|
|
|
|
function ok(condition, message) {
|
|
window.opener.wrappedJSObject.SimpleTest.ok(condition, message);
|
|
}
|
|
|
|
function is(a, b, message) {
|
|
window.opener.wrappedJSObject.SimpleTest.is(a, b, message);
|
|
}
|
|
|
|
function todo(condition, message) {
|
|
window.opener.wrappedJSObject.SimpleTest.todo(condition, message);
|
|
}
|
|
|
|
function todo_is(a, b, message) {
|
|
window.opener.wrappedJSObject.SimpleTest.todo_is(a, b, message);
|
|
}
|
|
|
|
function onTestsFinished() {
|
|
clearTimeout(gAfterLoopExecution);
|
|
observe(window, eventMonitor, false);
|
|
observe(gRightWindow, eventMonitor, false);
|
|
observe(gPopup, eventMonitor, false);
|
|
gRightWindow.close();
|
|
var openerSimpleTest = window.opener.wrappedJSObject.SimpleTest;
|
|
window.close();
|
|
openerSimpleTest.finish();
|
|
}
|
|
|
|
const XUL_NS = "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul";
|
|
const xulWin = 'data:application/vnd.mozilla.xul+xml,<?xml version="1.0"?><?xml-stylesheet href="chrome://global/skin" type="text/css"?><window xmlns="' + XUL_NS + '"/>';
|
|
|
|
const NSLeftMouseDown = 1,
|
|
NSLeftMouseUp = 2,
|
|
NSRightMouseDown = 3,
|
|
NSRightMouseUp = 4,
|
|
NSMouseMoved = 5,
|
|
NSLeftMouseDragged = 6,
|
|
NSRightMouseDragged = 7,
|
|
NSMouseEntered = 8,
|
|
NSMouseExited = 9,
|
|
NSKeyDown = 10,
|
|
NSKeyUp = 11,
|
|
NSFlagsChanged = 12,
|
|
NSAppKitDefined = 13,
|
|
NSSystemDefined = 14,
|
|
NSApplicationDefined = 15,
|
|
NSPeriodic = 16,
|
|
NSCursorUpdate = 17,
|
|
NSScrollWheel = 22,
|
|
NSTabletPoint = 23,
|
|
NSTabletProximity = 24,
|
|
NSOtherMouseDown = 25,
|
|
NSOtherMouseUp = 26,
|
|
NSOtherMouseDragged = 27,
|
|
NSEventTypeGesture = 29,
|
|
NSEventTypeMagnify = 30,
|
|
NSEventTypeSwipe = 31,
|
|
NSEventTypeRotate = 18,
|
|
NSEventTypeBeginGesture = 19,
|
|
NSEventTypeEndGesture = 20;
|
|
|
|
const NSAlphaShiftKeyMask = 1 << 16,
|
|
NSShiftKeyMask = 1 << 17,
|
|
NSControlKeyMask = 1 << 18,
|
|
NSAlternateKeyMask = 1 << 19,
|
|
NSCommandKeyMask = 1 << 20,
|
|
NSNumericPadKeyMask = 1 << 21,
|
|
NSHelpKeyMask = 1 << 22,
|
|
NSFunctionKeyMask = 1 << 23;
|
|
|
|
const gDebug = false;
|
|
|
|
function printDebug(msg) { if (gDebug) dump(msg); }
|
|
|
|
var gExpectedEvents = [];
|
|
var gRightWindow = null, gPopup = null;
|
|
var gCurrentMouseX = 0, gCurrentMouseY = 0;
|
|
var gAfterLoopExecution = 0;
|
|
|
|
function testMouse(x, y, msg, elem, win, exp, flags, callback) {
|
|
clearExpectedEvents();
|
|
var syntheticEvent = null;
|
|
exp.forEach(function (expEv) {
|
|
expEv.screenX = x;
|
|
expEv.screenY = y;
|
|
if (expEv.synthetic) {
|
|
is(syntheticEvent, null,
|
|
"Can't handle two synthetic events in a single testMouse call");
|
|
syntheticEvent = expEv;
|
|
}
|
|
gExpectedEvents.push(expEv);
|
|
});
|
|
printDebug("sending event: " + x + ", " + y + " (" + msg + ")\n");
|
|
gCurrentMouseX = x;
|
|
gCurrentMouseY = y;
|
|
netscape.security.PrivilegeManager.enablePrivilege('UniversalXPConnect');
|
|
var utils = win.QueryInterface(Components.interfaces.nsIInterfaceRequestor).
|
|
getInterface(Components.interfaces.nsIDOMWindowUtils);
|
|
var callbackFunc = function() {
|
|
clearExpectedEvents();
|
|
callback();
|
|
}
|
|
if (syntheticEvent) {
|
|
// Set up this listener before we sendNativeMouseEvent, just
|
|
// in case that synchronously calls us.
|
|
eventListenOnce(syntheticEvent.target, syntheticEvent.type,
|
|
// Trigger callbackFunc async, so we're not assuming
|
|
// anything about how our listener gets ordered with
|
|
// others.
|
|
function () { SimpleTest.executeSoon(callbackFunc) });
|
|
}
|
|
utils.sendNativeMouseEvent(x, y, msg, flags || 0, elem);
|
|
if (!syntheticEvent) {
|
|
gAfterLoopExecution = setTimeout(callbackFunc, 0);
|
|
}
|
|
}
|
|
|
|
function eventListenOnce(elem, name, callback) {
|
|
elem.addEventListener(name, function(e) {
|
|
elem.removeEventListener(name, arguments.callee, false);
|
|
callback(e);
|
|
}, false);
|
|
}
|
|
|
|
function focusAndThen(win, callback) {
|
|
eventListenOnce(win, "focus", callback);
|
|
printDebug("focusing a window\n");
|
|
win.focus();
|
|
}
|
|
|
|
function eventToString(e) {
|
|
return JSON.stringify({
|
|
type: e.type, target: e.target.nodeName, screenX: e.screenX, screenY: e.screenY
|
|
});
|
|
}
|
|
|
|
function clearExpectedEvents() {
|
|
while (gExpectedEvents.length > 0) {
|
|
var expectedEvent = gExpectedEvents.shift();
|
|
var errFun = expectedEvent.shouldFireButDoesnt ? todo : ok;
|
|
errFun(false, "Didn't receive expected event: " + eventToString(expectedEvent));
|
|
}
|
|
}
|
|
|
|
var gEventNum = 0;
|
|
|
|
function eventMonitor(e) {
|
|
printDebug("got event: " + eventToString(e) + "\n");
|
|
processEvent(e);
|
|
}
|
|
|
|
function processEvent(e) {
|
|
if (e.screenX != gCurrentMouseX || e.screenY != gCurrentMouseY) {
|
|
todo(false, "Oh no! Received a stray event from a confused tracking area. Aborting test.");
|
|
onTestsFinished();
|
|
return;
|
|
}
|
|
var expectedEvent = gExpectedEvents.shift();
|
|
if (!expectedEvent) {
|
|
ok(false, "received event I didn't expect: " + eventToString(e));
|
|
return;
|
|
}
|
|
if (e.type != expectedEvent.type) {
|
|
// Didn't get expectedEvent.
|
|
var errFun = expectedEvent.shouldFireButDoesnt ? todo : ok;
|
|
errFun(false, "Didn't receive expected event: " + eventToString(expectedEvent));
|
|
return processEvent(e);
|
|
}
|
|
gEventNum++;
|
|
is(e.screenX, expectedEvent.screenX, gEventNum + " | wrong X coord for event " + eventToString(e));
|
|
is(e.screenY, expectedEvent.screenY, gEventNum + " | wrong Y coord for event " + eventToString(e));
|
|
is(e.target, expectedEvent.target, gEventNum + " | wrong target for event " + eventToString(e));
|
|
if (expectedEvent.firesButShouldnt) {
|
|
todo(false, gEventNum + " | Got an event that should not have fired: " + eventToString(e));
|
|
}
|
|
}
|
|
|
|
function observe(elem, fun, add) {
|
|
var addOrRemove = add ? "addEventListener" : "removeEventListener";
|
|
elem[addOrRemove]("mousemove", fun, false);
|
|
elem[addOrRemove]("mouseover", fun, false);
|
|
elem[addOrRemove]("mouseout", fun, false);
|
|
elem[addOrRemove]("mousedown", fun, false);
|
|
elem[addOrRemove]("mouseup", fun, false);
|
|
elem[addOrRemove]("click", fun, false);
|
|
}
|
|
|
|
function start() {
|
|
window.resizeTo(200, 200);
|
|
window.moveTo(50, 50);
|
|
gRightWindow = open(xulWin, '', 'chrome,screenX=300,screenY=50,width=200,height=200');
|
|
eventListenOnce(gRightWindow, "focus", function () {
|
|
focusAndThen(window, runTests);
|
|
});
|
|
gPopup = document.getElementById("popup");
|
|
}
|
|
|
|
function runTests() {
|
|
observe(window, eventMonitor, true);
|
|
observe(gRightWindow, eventMonitor, true);
|
|
var left = window, right = gRightWindow;
|
|
var leftElem = document.getElementById("box");
|
|
var rightElem = gRightWindow.document.documentElement;
|
|
var panel = document.getElementById("panel");
|
|
var tooltip = (function createTooltipInRightWindow() {
|
|
var _tooltip = right.document.createElementNS(XUL_NS, "tooltip");
|
|
_tooltip.setAttribute("id", "tip");
|
|
_tooltip.setAttribute("width", "80");
|
|
_tooltip.setAttribute("height", "20");
|
|
right.document.documentElement.appendChild(_tooltip);
|
|
return _tooltip;
|
|
})();
|
|
var tests = [
|
|
|
|
// Part 1: Disallow click-through
|
|
|
|
function blockClickThrough(callback) {
|
|
document.documentElement.setAttribute("clickthrough", "never");
|
|
gRightWindow.document.documentElement.setAttribute("clickthrough", "never");
|
|
callback();
|
|
},
|
|
// Enter the left window, which is focused.
|
|
[150, 150, NSMouseMoved, null, left, [
|
|
{ type: "mouseover", target: leftElem },
|
|
{ type: "mousemove", target: leftElem }
|
|
]],
|
|
// Test that moving inside the window fires mousemove events.
|
|
[170, 150, NSMouseMoved, null, left, [
|
|
{ type: "mousemove", target: leftElem },
|
|
]],
|
|
// Leaving the window should fire a mouseout event...
|
|
[170, 20, NSMouseMoved, null, left, [
|
|
{ type: "mouseout", target: leftElem },
|
|
]],
|
|
// ... and entering a mouseover event.
|
|
[170, 120, NSMouseMoved, null, left, [
|
|
{ type: "mouseover", target: leftElem },
|
|
{ type: "mousemove", target: leftElem },
|
|
]],
|
|
// Move over the right window, which is inactive.
|
|
// Inactive windows shouldn't respond to mousemove events when clickthrough="never",
|
|
// so we should only get a mouseout event, no mouseover event.
|
|
[400, 150, NSMouseMoved, null, right, [
|
|
{ type: "mouseout", target: leftElem },
|
|
]],
|
|
// Left-clicking while holding Cmd and middle clicking should work even
|
|
// on inactive windows, but without making them active.
|
|
[400, 150, NSLeftMouseDown, null, right, [
|
|
{ type: "mousedown", target: rightElem },
|
|
], NSCommandKeyMask],
|
|
[400, 150, NSLeftMouseUp, null, right, [
|
|
{ type: "mouseup", target: rightElem },
|
|
{ type: "click", target: rightElem },
|
|
], NSCommandKeyMask],
|
|
[400, 150, NSOtherMouseDown, null, right, [
|
|
{ type: "mousedown", target: rightElem },
|
|
]],
|
|
[400, 150, NSOtherMouseUp, null, right, [
|
|
{ type: "mouseup", target: rightElem },
|
|
{ type: "click", target: rightElem },
|
|
]],
|
|
// Clicking an inactive window should make it active and fire a mouseover
|
|
// event.
|
|
[400, 150, NSLeftMouseDown, null, right, [
|
|
{ type: "mouseover", target: rightElem, synthetic: true },
|
|
]],
|
|
[400, 150, NSLeftMouseUp, null, right, [
|
|
]],
|
|
// Now it's focused, so we should get a mousedown event when clicking.
|
|
[400, 150, NSLeftMouseDown, null, right, [
|
|
{ type: "mousedown", target: rightElem },
|
|
]],
|
|
// Let's drag to the right without letting the button go.
|
|
[410, 150, NSLeftMouseDragged, null, right, [
|
|
{ type: "mousemove", target: rightElem },
|
|
]],
|
|
// Let go of the mouse.
|
|
[410, 150, NSLeftMouseUp, null, right, [
|
|
{ type: "mouseup", target: rightElem },
|
|
{ type: "click", target: rightElem },
|
|
]],
|
|
// Move the mouse back over the left window, which is inactive.
|
|
[150, 170, NSMouseMoved, null, left, [
|
|
{ type: "mouseout", target: rightElem },
|
|
]],
|
|
// Now we're being sneaky. The left window is inactive, but *right*-clicks to it
|
|
// should still get through. Test that.
|
|
// Ideally we'd be bracketing that event with over and out events, too, but it
|
|
// probably doesn't matter too much.
|
|
[150, 170, NSRightMouseDown, null, left, [
|
|
{ type: "mouseover", target: leftElem, shouldFireButDoesnt: true },
|
|
{ type: "mousedown", target: leftElem },
|
|
{ type: "mouseout", target: leftElem, shouldFireButDoesnt: true },
|
|
]],
|
|
// Let go of the mouse.
|
|
[150, 170, NSRightMouseUp, null, left, [
|
|
{ type: "mouseover", target: leftElem, shouldFireButDoesnt: true },
|
|
{ type: "mouseup", target: leftElem },
|
|
{ type: "click", target: leftElem },
|
|
{ type: "mouseout", target: leftElem, shouldFireButDoesnt: true },
|
|
]],
|
|
// Right clicking hasn't focused it, so the window is still inactive.
|
|
// Let's focus it; this time without the mouse, for variaton's sake.
|
|
// Still, mouseout and mouseover events should fire.
|
|
function raiseLeftWindow(callback) {
|
|
clearExpectedEvents();
|
|
gExpectedEvents.push({ screenX: 150, screenY: 170, type: "mouseover", target: leftElem });
|
|
// We have to be a bit careful here. The synthetic mouse event may
|
|
// not fire for a bit after we focus the left window.
|
|
eventListenOnce(leftElem, "mouseover", function() {
|
|
// Trigger callback async, so we're not assuming
|
|
// anything about how our listener gets ordered with others.
|
|
SimpleTest.executeSoon(callback);
|
|
});
|
|
printDebug("focusing left window");
|
|
left.focus();
|
|
},
|
|
// It's active, so it should respond to mousemove events now.
|
|
[150, 170, NSMouseMoved, null, left, [
|
|
{ type: "mousemove", target: leftElem },
|
|
]],
|
|
// This was boring... let's introduce a popup. It will overlap both the left
|
|
// and the right window.
|
|
function openPopupInLeftWindow(callback) {
|
|
eventListenOnce(gPopup, "popupshown", callback);
|
|
gPopup.openPopupAtScreen(150, 50, true);
|
|
},
|
|
// Move the mouse over the popup.
|
|
[200, 80, NSMouseMoved, gPopup, left, [
|
|
{ type: "mouseout", target: leftElem },
|
|
{ type: "mouseover", target: gPopup },
|
|
{ type: "mousemove", target: gPopup },
|
|
]],
|
|
// Move the mouse back over the left window outside the popup.
|
|
[160, 170, NSMouseMoved, null, left, [
|
|
{ type: "mouseout", target: gPopup },
|
|
{ type: "mouseover", target: leftElem },
|
|
{ type: "mousemove", target: leftElem },
|
|
]],
|
|
// Back over the popup...
|
|
[190, 80, NSMouseMoved, gPopup, left, [
|
|
{ type: "mouseout", target: leftElem },
|
|
{ type: "mouseover", target: gPopup },
|
|
{ type: "mousemove", target: gPopup },
|
|
]],
|
|
// ...and over into the right window.
|
|
// It's inactive, so it shouldn't get mouseover events yet.
|
|
[400, 170, NSMouseMoved, null, right, [
|
|
{ type: "mouseout", target: gPopup },
|
|
]],
|
|
// Again, no mouse events please, even though a popup is open. (bug 425556)
|
|
[400, 180, NSMouseMoved, null, right, [
|
|
]],
|
|
// Activate the right window with a click.
|
|
// This will close the popup and make the mouse enter the right window.
|
|
[400, 180, NSLeftMouseDown, null, right, [
|
|
{ type: "mouseover", target: rightElem, synthetic: true },
|
|
]],
|
|
[400, 180, NSLeftMouseUp, null, right, [
|
|
]],
|
|
function verifyPopupClosed2(callback) {
|
|
is(gPopup.popupBoxObject.popupState, "closed", "popup should have closed when clicking");
|
|
callback();
|
|
},
|
|
// Now the right window is active; click it again, just for fun.
|
|
[400, 180, NSLeftMouseDown, null, right, [
|
|
{ type: "mousedown", target: rightElem },
|
|
]],
|
|
[400, 180, NSLeftMouseUp, null, right, [
|
|
{ type: "mouseup", target: rightElem },
|
|
{ type: "click", target: rightElem },
|
|
]],
|
|
|
|
// Time for our next trick: a tooltip!
|
|
// Install the tooltip, but don't show it yet.
|
|
function setTooltip(callback) {
|
|
rightElem.setAttribute("tooltip", "tip");
|
|
gExpectedEvents.push({ screenX: 410, screenY: 180, type: "mousemove", target: rightElem });
|
|
eventListenOnce(rightElem, "popupshown", callback);
|
|
gCurrentMouseX = 410;
|
|
gCurrentMouseY = 180;
|
|
var utils = right.QueryInterface(Components.interfaces.nsIInterfaceRequestor).
|
|
getInterface(Components.interfaces.nsIDOMWindowUtils);
|
|
utils.sendNativeMouseEvent(410, 180, NSMouseMoved, 0, null);
|
|
},
|
|
// Now the tooltip is visible.
|
|
// Move the mouse a little to the right.
|
|
[411, 180, NSMouseMoved, null, right, [
|
|
{ type: "mousemove", target: rightElem },
|
|
]],
|
|
// Move another pixel.
|
|
[412, 180, NSMouseMoved, null, right, [
|
|
{ type: "mousemove", target: rightElem },
|
|
]],
|
|
// Move up and click to make the tooltip go away.
|
|
[412, 80, NSMouseMoved, null, right, [
|
|
{ type: "mousemove", target: rightElem },
|
|
]],
|
|
[412, 80, NSLeftMouseDown, null, right, [
|
|
{ type: "mousedown", target: rightElem },
|
|
]],
|
|
[412, 80, NSLeftMouseUp, null, right, [
|
|
{ type: "mouseup", target: rightElem },
|
|
{ type: "click", target: rightElem },
|
|
]],
|
|
// OK, next round. Open a panel in the left window, which is inactive.
|
|
function openPanel(callback) {
|
|
eventListenOnce(panel, "popupshown", callback);
|
|
panel.openPopupAtScreen(150, 150, false);
|
|
},
|
|
// The panel is parented, so it will be z-ordered over its parent but
|
|
// under the active window.
|
|
// Now we move the mouse over the part where the panel rect intersects the
|
|
// right window's rect. Since the panel is under the window, all the events
|
|
// should target the right window.
|
|
[390, 170, NSMouseMoved, null, right, [
|
|
{ type: "mousemove", target: rightElem },
|
|
]],
|
|
[390, 171, NSMouseMoved, null, right, [
|
|
{ type: "mousemove", target: rightElem },
|
|
]],
|
|
[391, 171, NSMouseMoved, null, right, [
|
|
{ type: "mousemove", target: rightElem },
|
|
]],
|
|
// Now move off the right window, so that the mouse is directly over the
|
|
// panel.
|
|
[260, 170, NSMouseMoved, panel, left, [
|
|
{ type: "mouseout", target: rightElem },
|
|
]],
|
|
[260, 171, NSMouseMoved, panel, left, [
|
|
]],
|
|
[261, 171, NSMouseMoved, panel, left, [
|
|
]],
|
|
// Let's be evil and click it.
|
|
[261, 171, NSLeftMouseDown, panel, left, [
|
|
]],
|
|
[261, 171, NSLeftMouseUp, panel, left, [
|
|
]],
|
|
// This didn't focus the window, unfortunately, so let's do it ourselves.
|
|
function raiseLeftWindowTakeTwo(callback) {
|
|
focusAndThen(left, callback);
|
|
},
|
|
// Now mouse events should get through to the panel (which is now over the
|
|
// right window).
|
|
[387, 170, NSMouseMoved, panel, left, [
|
|
{ type: "mouseover", target: panel },
|
|
{ type: "mousemove", target: panel },
|
|
]],
|
|
[387, 171, NSMouseMoved, panel, left, [
|
|
{ type: "mousemove", target: panel },
|
|
]],
|
|
[388, 171, NSMouseMoved, panel, left, [
|
|
{ type: "mousemove", target: panel },
|
|
]],
|
|
// Click the panel.
|
|
[388, 171, NSLeftMouseDown, panel, left, [
|
|
{ type: "mousedown", target: panel }
|
|
]],
|
|
[388, 171, NSLeftMouseUp, panel, left, [
|
|
{ type: "mouseup", target: panel },
|
|
{ type: "click", target: panel },
|
|
]],
|
|
|
|
// Last test for this part: Hit testing in the Canyon of Nowhere -
|
|
// the pixel row directly south of the panel, over the left window.
|
|
// Before bug 515003 we wrongly thought the mouse wasn't over any window.
|
|
[173, 200, NSMouseMoved, null, left, [
|
|
{ type: "mouseout", target: panel },
|
|
{ type: "mouseover", target: leftElem },
|
|
{ type: "mousemove", target: leftElem },
|
|
]],
|
|
[173, 201, NSMouseMoved, null, left, [
|
|
{ type: "mousemove", target: leftElem },
|
|
]],
|
|
|
|
// Part 2: Allow click-through
|
|
|
|
function hideThatPanel(callback) {
|
|
eventListenOnce(panel, "popuphidden", callback);
|
|
panel.hidePopup();
|
|
},
|
|
function unblockClickThrough(callback) {
|
|
document.documentElement.removeAttribute("clickthrough");
|
|
gRightWindow.document.documentElement.removeAttribute("clickthrough");
|
|
callback();
|
|
},
|
|
// Enter the left window, which is focused.
|
|
[150, 150, NSMouseMoved, null, left, [
|
|
{ type: "mousemove", target: leftElem }
|
|
]],
|
|
// Test that moving inside the window fires mousemove events.
|
|
[170, 150, NSMouseMoved, null, left, [
|
|
{ type: "mousemove", target: leftElem },
|
|
]],
|
|
// Leaving the window should fire a mouseout event...
|
|
[170, 20, NSMouseMoved, null, left, [
|
|
{ type: "mouseout", target: leftElem },
|
|
]],
|
|
// ... and entering a mouseover event.
|
|
[170, 120, NSMouseMoved, null, left, [
|
|
{ type: "mouseover", target: leftElem },
|
|
{ type: "mousemove", target: leftElem },
|
|
]],
|
|
// Move over the right window, which is inactive but still accepts
|
|
// mouse events.
|
|
[400, 150, NSMouseMoved, null, right, [
|
|
{ type: "mouseout", target: leftElem },
|
|
{ type: "mouseover", target: rightElem },
|
|
{ type: "mousemove", target: rightElem },
|
|
]],
|
|
// Left-clicking while holding Cmd and middle clicking should work
|
|
// on inactive windows, but without making them active.
|
|
[400, 150, NSLeftMouseDown, null, right, [
|
|
{ type: "mousedown", target: rightElem },
|
|
], NSCommandKeyMask],
|
|
[400, 150, NSLeftMouseUp, null, right, [
|
|
{ type: "mouseup", target: rightElem },
|
|
{ type: "click", target: rightElem },
|
|
], NSCommandKeyMask],
|
|
[400, 150, NSOtherMouseDown, null, right, [
|
|
{ type: "mousedown", target: rightElem },
|
|
]],
|
|
[400, 150, NSOtherMouseUp, null, right, [
|
|
{ type: "mouseup", target: rightElem },
|
|
{ type: "click", target: rightElem },
|
|
]],
|
|
// Clicking an inactive window should make it active
|
|
[400, 150, NSLeftMouseDown, null, right, [
|
|
{ type: "mousedown", target: rightElem },
|
|
]],
|
|
[400, 150, NSLeftMouseUp, null, right, [
|
|
{ type: "mouseup", target: rightElem },
|
|
{ type: "click", target: rightElem },
|
|
]],
|
|
// Now it's focused.
|
|
[401, 150, NSLeftMouseDown, null, right, [
|
|
{ type: "mousedown", target: rightElem },
|
|
]],
|
|
// Let's drag to the right without letting the button go.
|
|
[410, 150, NSLeftMouseDragged, null, right, [
|
|
{ type: "mousemove", target: rightElem },
|
|
]],
|
|
// Let go of the mouse.
|
|
[410, 150, NSLeftMouseUp, null, right, [
|
|
{ type: "mouseup", target: rightElem },
|
|
{ type: "click", target: rightElem },
|
|
]],
|
|
// Move the mouse back over the left window, which is inactive.
|
|
[150, 170, NSMouseMoved, null, left, [
|
|
{ type: "mouseout", target: rightElem },
|
|
{ type: "mouseover", target: leftElem },
|
|
{ type: "mousemove", target: leftElem },
|
|
]],
|
|
// Right-click it.
|
|
[150, 170, NSRightMouseDown, null, left, [
|
|
{ type: "mousedown", target: leftElem },
|
|
]],
|
|
// Let go of the mouse.
|
|
[150, 170, NSRightMouseUp, null, left, [
|
|
{ type: "mouseup", target: leftElem },
|
|
{ type: "click", target: leftElem },
|
|
]],
|
|
// Right clicking hasn't focused it, so the window is still inactive.
|
|
// Let's focus it; this time without the mouse, for variaton's sake.
|
|
function raiseLeftWindow(callback) {
|
|
clearExpectedEvents();
|
|
focusAndThen(left, function () { SimpleTest.executeSoon(callback); });
|
|
},
|
|
// It's active and should still respond to mousemove events.
|
|
[150, 170, NSMouseMoved, null, left, [
|
|
{ type: "mousemove", target: leftElem },
|
|
]],
|
|
|
|
// This was boring... let's introduce a popup. It will overlap both the left
|
|
// and the right window.
|
|
function openPopupInLeftWindow(callback) {
|
|
eventListenOnce(gPopup, "popupshown", callback);
|
|
gPopup.openPopupAtScreen(150, 50, true);
|
|
},
|
|
// Move the mouse over the popup.
|
|
[200, 80, NSMouseMoved, gPopup, left, [
|
|
{ type: "mouseout", target: leftElem },
|
|
{ type: "mouseover", target: gPopup },
|
|
{ type: "mousemove", target: gPopup },
|
|
]],
|
|
// Move the mouse back over the left window outside the popup.
|
|
[160, 170, NSMouseMoved, null, left, [
|
|
{ type: "mouseout", target: gPopup },
|
|
{ type: "mouseover", target: leftElem },
|
|
{ type: "mousemove", target: leftElem },
|
|
]],
|
|
// Back over the popup...
|
|
[190, 80, NSMouseMoved, gPopup, left, [
|
|
{ type: "mouseout", target: leftElem },
|
|
{ type: "mouseover", target: gPopup },
|
|
{ type: "mousemove", target: gPopup },
|
|
]],
|
|
// ...and over into the right window.
|
|
[400, 170, NSMouseMoved, null, right, [
|
|
{ type: "mouseout", target: gPopup },
|
|
{ type: "mouseover", target: rightElem },
|
|
{ type: "mousemove", target: rightElem },
|
|
]],
|
|
[400, 180, NSMouseMoved, null, right, [
|
|
{ type: "mousemove", target: rightElem },
|
|
]],
|
|
// Activate the right window with a click.
|
|
[400, 180, NSLeftMouseDown, null, right, [
|
|
{ type: "mousedown", target: rightElem },
|
|
]],
|
|
[400, 180, NSLeftMouseUp, null, right, [
|
|
{ type: "mouseup", target: rightElem },
|
|
{ type: "click", target: rightElem },
|
|
]],
|
|
function verifyPopupClosed2(callback) {
|
|
is(gPopup.popupBoxObject.popupState, "closed", "popup should have closed when clicking");
|
|
callback();
|
|
},
|
|
// Now the right window is active; click it again, just for fun.
|
|
[400, 180, NSLeftMouseDown, null, right, [
|
|
{ type: "mousedown", target: rightElem },
|
|
]],
|
|
[400, 180, NSLeftMouseUp, null, right, [
|
|
{ type: "mouseup", target: rightElem },
|
|
{ type: "click", target: rightElem },
|
|
]],
|
|
|
|
// Time for our next trick: a tooltip!
|
|
// Install the tooltip, but don't show it yet.
|
|
function setTooltip2(callback) {
|
|
rightElem.setAttribute("tooltip", "tip");
|
|
gExpectedEvents.push({ screenX: 410, screenY: 180, type: "mousemove", target: rightElem });
|
|
eventListenOnce(rightElem, "popupshown", callback);
|
|
gCurrentMouseX = 410;
|
|
gCurrentMouseY = 180;
|
|
var utils = right.QueryInterface(Components.interfaces.nsIInterfaceRequestor).
|
|
getInterface(Components.interfaces.nsIDOMWindowUtils);
|
|
utils.sendNativeMouseEvent(410, 180, NSMouseMoved, 0, null);
|
|
},
|
|
// Now the tooltip is visible.
|
|
// Move the mouse a little to the right.
|
|
[411, 180, NSMouseMoved, null, right, [
|
|
{ type: "mousemove", target: rightElem },
|
|
]],
|
|
// Move another pixel.
|
|
[412, 180, NSMouseMoved, null, right, [
|
|
{ type: "mousemove", target: rightElem },
|
|
]],
|
|
// Move up and click to make the tooltip go away.
|
|
[412, 80, NSMouseMoved, null, right, [
|
|
{ type: "mousemove", target: rightElem },
|
|
]],
|
|
[412, 80, NSLeftMouseDown, null, right, [
|
|
{ type: "mousedown", target: rightElem },
|
|
]],
|
|
[412, 80, NSLeftMouseUp, null, right, [
|
|
{ type: "mouseup", target: rightElem },
|
|
{ type: "click", target: rightElem },
|
|
]],
|
|
// OK, next round. Open a panel in the left window, which is inactive.
|
|
function openPanel2(callback) {
|
|
eventListenOnce(panel, "popupshown", callback);
|
|
panel.openPopupAtScreen(150, 150, false);
|
|
},
|
|
// The panel is parented, so it will be z-ordered over its parent but
|
|
// under the active window.
|
|
// Now we move the mouse over the part where the panel rect intersects the
|
|
// right window's rect. Since the panel is under the window, all the events
|
|
// should target the right window.
|
|
[390, 170, NSMouseMoved, null, right, [
|
|
{ type: "mousemove", target: rightElem },
|
|
]],
|
|
[390, 171, NSMouseMoved, null, right, [
|
|
{ type: "mousemove", target: rightElem },
|
|
]],
|
|
[391, 171, NSMouseMoved, null, right, [
|
|
{ type: "mousemove", target: rightElem },
|
|
]],
|
|
// Now move off the right window, so that the mouse is directly over the
|
|
// panel.
|
|
[260, 170, NSMouseMoved, panel, left, [
|
|
{ type: "mouseout", target: rightElem },
|
|
{ type: "mouseover", target: panel },
|
|
{ type: "mousemove", target: panel },
|
|
]],
|
|
[260, 171, NSMouseMoved, panel, left, [
|
|
{ type: "mousemove", target: panel },
|
|
]],
|
|
[261, 171, NSMouseMoved, panel, left, [
|
|
{ type: "mousemove", target: panel },
|
|
]],
|
|
// Let's be evil and click it.
|
|
[261, 171, NSLeftMouseDown, panel, left, [
|
|
{ type: "mousedown", target: panel },
|
|
]],
|
|
[261, 171, NSLeftMouseUp, panel, left, [
|
|
{ type: "mouseup", target: panel },
|
|
{ type: "click", target: panel },
|
|
]],
|
|
// This didn't focus the window, unfortunately, so let's do it ourselves.
|
|
function raiseLeftWindowTakeTwo(callback) {
|
|
focusAndThen(left, callback);
|
|
},
|
|
[387, 170, NSMouseMoved, panel, left, [
|
|
{ type: "mousemove", target: panel },
|
|
]],
|
|
[387, 171, NSMouseMoved, panel, left, [
|
|
{ type: "mousemove", target: panel },
|
|
]],
|
|
[388, 171, NSMouseMoved, panel, left, [
|
|
{ type: "mousemove", target: panel },
|
|
]],
|
|
// Click the panel.
|
|
[388, 171, NSLeftMouseDown, panel, left, [
|
|
{ type: "mousedown", target: panel }
|
|
]],
|
|
[388, 171, NSLeftMouseUp, panel, left, [
|
|
{ type: "mouseup", target: panel },
|
|
{ type: "click", target: panel },
|
|
]],
|
|
|
|
// Last test for today: Hit testing in the Canyon of Nowhere -
|
|
// the pixel row directly south of the panel, over the left window.
|
|
// Before bug 515003 we wrongly thought the mouse wasn't over any window.
|
|
[173, 200, NSMouseMoved, null, left, [
|
|
{ type: "mouseout", target: panel },
|
|
{ type: "mouseover", target: leftElem },
|
|
{ type: "mousemove", target: leftElem },
|
|
]],
|
|
[173, 201, NSMouseMoved, null, left, [
|
|
{ type: "mousemove", target: leftElem },
|
|
]],
|
|
];
|
|
function runNextTest() {
|
|
if (!tests.length)
|
|
return onTestsFinished();
|
|
|
|
var test = tests.shift();
|
|
if (typeof test == "function")
|
|
return test(runNextTest);
|
|
|
|
var [x, y, msg, elem, win, exp, flags] = test;
|
|
testMouse(x, y, msg, elem, win, exp, flags, runNextTest);
|
|
}
|
|
runNextTest();
|
|
}
|
|
|
|
SimpleTest.waitForFocus(start);
|
|
|
|
]]></script>
|
|
</window>
|