Bug 706743 tooltip listener should ignore mousemove and mouseout events during drag r=enndeakin

This commit is contained in:
Masayuki Nakano 2011-12-16 15:17:48 +09:00
Родитель a17c3dda89
Коммит 2872d3cc6b
3 изменённых файлов: 183 добавлений и 6 удалений

Просмотреть файл

@ -48,6 +48,8 @@
#include "nsIPopupBoxObject.h"
#include "nsMenuPopupFrame.h"
#include "nsIServiceManager.h"
#include "nsIDragService.h"
#include "nsIDragSession.h"
#ifdef MOZ_XUL
#include "nsITreeView.h"
#endif
@ -267,14 +269,38 @@ nsXULTooltipListener::HandleEvent(nsIDOMEvent* aEvent)
type.EqualsLiteral("keydown") ||
type.EqualsLiteral("mousedown") ||
type.EqualsLiteral("mouseup") ||
type.EqualsLiteral("dragstart"))
type.EqualsLiteral("dragstart")) {
HideTooltip();
else if (type.EqualsLiteral("mousemove"))
MouseMove(aEvent);
else if (type.EqualsLiteral("mouseout"))
MouseOut(aEvent);
else if (type.EqualsLiteral("popuphiding"))
return NS_OK;
}
if (type.EqualsLiteral("popuphiding")) {
DestroyTooltip();
return NS_OK;
}
// Note that mousemove, mouseover and mouseout might be
// fired even during dragging due to widget's bug.
nsCOMPtr<nsIDragService> dragService =
do_GetService("@mozilla.org/widget/dragservice;1");
NS_ENSURE_TRUE(dragService, NS_OK);
nsCOMPtr<nsIDragSession> dragSession;
dragService->GetCurrentSession(getter_AddRefs(dragSession));
if (dragSession) {
return NS_OK;
}
// Not dragging.
if (type.EqualsLiteral("mousemove")) {
MouseMove(aEvent);
return NS_OK;
}
if (type.EqualsLiteral("mouseout")) {
MouseOut(aEvent);
return NS_OK;
}
return NS_OK;
}

Просмотреть файл

@ -67,6 +67,7 @@ libs:: $(_CHROME_FILES)
ifneq (mobile,$(MOZ_BUILD_APP))
_BROWSER_FILES = \
browser_bug703210.js \
browser_bug706743.js \
$(NULL)
libs:: $(_BROWSER_FILES)

Просмотреть файл

@ -0,0 +1,150 @@
function test() {
waitForExplicitFinish();
gBrowser.selectedTab = gBrowser.addTab();
let target;
let doc;
let win;
let callbackOnPopupShown;
let callbackOnPopupHidden;
let dragService = Components.classes["@mozilla.org/widget/dragservice;1"].
getService(Components.interfaces.nsIDragService);
let onPopupHidden = function (aEvent)
{
if (aEvent.originalTarget.localName != "tooltip") {
return;
}
if (callbackOnPopupHidden) {
setTimeout(callbackOnPopupHidden, 0);
}
}
let onPopupShown = function (aEvent)
{
if (aEvent.originalTarget.localName != "tooltip") {
return;
}
if (callbackOnPopupShown) {
setTimeout(callbackOnPopupShown, 0);
}
}
let finishTest = function ()
{
document.removeEventListener("popupshown", onPopupShown, true);
document.removeEventListener("popuphidden", onPopupHidden, true);
gBrowser.removeCurrentTab();
finish();
}
let testHideTooltipByMouseDown = function ()
{
callbackOnPopupShown = function () {
callbackOnPopupShown = null;
ok(true, "tooltip is shown before testing mousedown");
// hide tooltip by mousemove to outside.
callbackOnPopupHidden = function () {
callbackOnPopupHidden = null;
ok(true, "The tooltip is hidden by mousedown");
EventUtils.synthesizeMouse(target, 5, 15, { type: "mouseup" }, win);
EventUtils.synthesizeMouse(target, -5, 15, { type: "mousemove" }, win);
setTimeout(finishTest, 0);
}
EventUtils.synthesizeMouse(target, 5, 15, { type: "mousedown" }, win);
}
EventUtils.synthesizeMouse(target, 5, 15, { type: "mousemove" }, win);
}
let testShowTooltipAgain = function ()
{
// If tooltip listener used a flag for managing D&D state, we would need
// to test if the tooltip is shown after drag.
callbackOnPopupShown = function () {
callbackOnPopupShown = null;
ok(true, "tooltip is shown after drag");
// hide tooltip by mousemove to outside.
callbackOnPopupHidden = function () {
callbackOnPopupHidden = null;
ok(true, "The tooltip is hidden again");
setTimeout(testHideTooltipByMouseDown, 0);
}
EventUtils.synthesizeMouse(target, -5, 15, { type: "mousemove" }, win);
}
EventUtils.synthesizeMouse(target, 5, 15, { type: "mousemove" }, win);
}
let testDuringDnD = function ()
{
// mousemove into the target and start drag by emulation via nsIDragService.
// Note that on some platforms, we cannot actually start the drag by
// synthesized events. E.g., Windows waits an actual mousemove event after
// dragstart.
callbackOnPopupShown = function () {
callbackOnPopupShown = null;
ok(false, "tooltip is shown during drag");
}
dragService.startDragSession();
// Emulate a buggy mousemove event. widget might dispatch mousemove event
// during drag.
EventUtils.synthesizeMouse(target, 5, 15, { type: "mousemove" }, win);
setTimeout(function () {
callbackOnPopupShown = null;
ok(true, "tooltip isn't shown during drag");
dragService.endDragSession(true);
EventUtils.synthesizeMouse(target, -5, -5, { type: "mousemove" }, win);
setTimeout(testShowTooltipAgain, 0);
}, 600);
}
let onLoad = function (aEvent)
{
doc = gBrowser.contentDocument;
win = gBrowser.contentWindow;
target = doc.getElementById("target");
document.addEventListener("popupshown", onPopupShown, true);
document.addEventListener("popuphidden", onPopupHidden, true);
EventUtils.synthesizeMouse(target, -5, -5, { type: "mousemove" }, win);
// show tooltip by mousemove into target.
callbackOnPopupShown = function ()
{
callbackOnPopupShown = null;
ok(true, "The tooltip is shown");
// hide tooltip by mousemove to outside.
callbackOnPopupHidden = function () {
callbackOnPopupHidden = null;
ok(true, "The tooltip is hidden");
setTimeout(testDuringDnD, 0);
}
EventUtils.synthesizeMouse(target, -5, 15, { type: "mousemove" }, win);
}
EventUtils.synthesizeMouse(target, 5, 15, { type: "mousemove" }, win);
}
gBrowser.selectedBrowser.addEventListener("load",
function () {
gBrowser.selectedBrowser.
removeEventListener("load", arguments.callee, true);
setTimeout(onLoad, 0);
}, true);
content.location = "data:text/html,<html><head></head><body>" +
"<a id=\"target\" href=\"about:blank\" title=\"This is tooltip text\" " +
"style=\"display:block;height:20px;margin:10px;\" " +
"onclick=\"return false;\">here is an anchor element</a></body></html>";
}