356 lines
10 KiB
JavaScript
356 lines
10 KiB
JavaScript
// Copyright 2010 The Closure Library Authors. All Rights Reserved.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS-IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
goog.provide('goog.ui.HoverCardTest');
|
|
goog.setTestOnly('goog.ui.HoverCardTest');
|
|
|
|
goog.require('goog.dom');
|
|
goog.require('goog.events');
|
|
goog.require('goog.math.Coordinate');
|
|
goog.require('goog.style');
|
|
goog.require('goog.testing.MockClock');
|
|
goog.require('goog.testing.events');
|
|
goog.require('goog.testing.events.Event');
|
|
goog.require('goog.testing.jsunit');
|
|
goog.require('goog.ui.HoverCard');
|
|
|
|
var timer = new goog.testing.MockClock();
|
|
var card;
|
|
|
|
// Variables for mocks
|
|
var triggeredElement;
|
|
var cancelledElement;
|
|
var showDelay;
|
|
var shownCard;
|
|
var hideDelay;
|
|
|
|
// spans
|
|
var john;
|
|
var jane;
|
|
var james;
|
|
var bill;
|
|
var child;
|
|
|
|
// Inactive
|
|
var elsewhere;
|
|
var offAnchor;
|
|
|
|
function setUpPage() {
|
|
john = goog.dom.getElement('john');
|
|
jane = goog.dom.getElement('jane');
|
|
james = goog.dom.getElement('james');
|
|
bill = goog.dom.getElement('bill');
|
|
child = goog.dom.getElement('child');
|
|
}
|
|
|
|
function setUp() {
|
|
timer.install();
|
|
triggeredElement = null;
|
|
cancelledElement = null;
|
|
showDelay = null;
|
|
shownCard = null;
|
|
hideDelay = null;
|
|
elsewhere = goog.dom.getElement('notpopup');
|
|
offAnchor = new goog.math.Coordinate(1, 1);
|
|
}
|
|
|
|
function initCard(opt_isAnchor, opt_checkChildren, opt_maxSearchSteps) {
|
|
var isAnchor = opt_isAnchor || {SPAN: 'email'};
|
|
card = new goog.ui.HoverCard(isAnchor, opt_checkChildren);
|
|
card.setText('Test hovercard');
|
|
|
|
if (opt_maxSearchSteps != null) {
|
|
card.setMaxSearchSteps(opt_maxSearchSteps);
|
|
}
|
|
|
|
goog.events.listen(card, goog.ui.HoverCard.EventType.TRIGGER, onTrigger);
|
|
goog.events.listen(card, goog.ui.HoverCard.EventType.CANCEL_TRIGGER,
|
|
onCancel);
|
|
goog.events.listen(card, goog.ui.HoverCard.EventType.BEFORE_SHOW,
|
|
onBeforeShow);
|
|
|
|
// This gets around the problem where AdvancedToolTip thinks it's
|
|
// receiving a ghost event because cursor position hasn't moved off of
|
|
// (0, 0).
|
|
card.cursorPosition = new goog.math.Coordinate(1, 1);
|
|
}
|
|
|
|
// Event handlers
|
|
function onTrigger(event) {
|
|
triggeredElement = event.anchor;
|
|
if (showDelay) {
|
|
card.setShowDelayMs(showDelay);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
function onCancel(event) {
|
|
cancelledElement = event.anchor;
|
|
}
|
|
|
|
function onBeforeShow() {
|
|
shownCard = card.getAnchorElement();
|
|
if (hideDelay) {
|
|
card.setHideDelayMs(hideDelay);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
function tearDown() {
|
|
card.dispose();
|
|
timer.uninstall();
|
|
}
|
|
|
|
|
|
/**
|
|
* Verify that hovercard displays and goes away under normal circumstances.
|
|
*/
|
|
function testTrigger() {
|
|
initCard();
|
|
|
|
// Mouse over correct element fires trigger
|
|
showDelay = 500;
|
|
goog.testing.events.fireMouseOverEvent(john, elsewhere);
|
|
assertEquals('Hovercard should have triggered', john,
|
|
triggeredElement);
|
|
|
|
// Show card after delay
|
|
timer.tick(showDelay - 1);
|
|
assertNull('Card should not have shown', shownCard);
|
|
assertFalse(card.isVisible());
|
|
hideDelay = 5000;
|
|
timer.tick(1);
|
|
assertEquals('Card should have shown', john, shownCard);
|
|
assertTrue(card.isVisible());
|
|
|
|
// Mouse out leads to hide delay
|
|
goog.testing.events.fireMouseOutEvent(john, elsewhere);
|
|
goog.testing.events.fireMouseMoveEvent(document, offAnchor);
|
|
timer.tick(hideDelay - 1);
|
|
assertTrue('Card should still be visible', card.isVisible());
|
|
timer.tick(10);
|
|
assertFalse('Card should be hidden', card.isVisible());
|
|
}
|
|
|
|
|
|
/**
|
|
* Verify that CANCEL_TRIGGER event occurs when mouse goes out of
|
|
* triggering element before hovercard is shown.
|
|
*/
|
|
function testOnCancel() {
|
|
initCard();
|
|
|
|
showDelay = 500;
|
|
goog.testing.events.fireMouseOverEvent(john, elsewhere);
|
|
timer.tick(showDelay - 1);
|
|
goog.testing.events.fireMouseOutEvent(john, elsewhere);
|
|
goog.testing.events.fireMouseMoveEvent(document, offAnchor);
|
|
timer.tick(10);
|
|
assertFalse('Card should be hidden', card.isVisible());
|
|
assertEquals('Should have cancelled trigger', john, cancelledElement);
|
|
}
|
|
|
|
|
|
/**
|
|
* Verify that mousing over non-triggering elements don't interfere.
|
|
*/
|
|
function testMouseOverNonTrigger() {
|
|
initCard();
|
|
|
|
// Mouse over correct element fires trigger
|
|
showDelay = 500;
|
|
goog.testing.events.fireMouseOverEvent(john, elsewhere);
|
|
timer.tick(showDelay);
|
|
|
|
// Mouse over and out other element does nothing
|
|
triggeredElement = null;
|
|
goog.testing.events.fireMouseOverEvent(jane, elsewhere);
|
|
timer.tick(showDelay + 1);
|
|
assertNull(triggeredElement);
|
|
}
|
|
|
|
|
|
/**
|
|
* Verify that a mouse over event with no target will not break
|
|
* hover card.
|
|
*/
|
|
function testMouseOverNoTarget() {
|
|
initCard();
|
|
card.handleTriggerMouseOver_(new goog.testing.events.Event());
|
|
}
|
|
|
|
|
|
/**
|
|
* Verify that mousing over a second trigger before the first one shows
|
|
* will correctly cancel the first and show the second.
|
|
*/
|
|
function testMultipleTriggers() {
|
|
initCard();
|
|
|
|
// Test second trigger when first one still pending
|
|
showDelay = 500;
|
|
hideDelay = 1000;
|
|
goog.testing.events.fireMouseOverEvent(john, elsewhere);
|
|
timer.tick(250);
|
|
goog.testing.events.fireMouseOutEvent(john, james);
|
|
goog.testing.events.fireMouseOverEvent(james, john);
|
|
// First trigger should cancel because it isn't showing yet
|
|
assertEquals('Should cancel first trigger', john, cancelledElement);
|
|
timer.tick(300);
|
|
assertFalse(card.isVisible());
|
|
timer.tick(250);
|
|
assertEquals('Should show second card', james, shownCard);
|
|
assertTrue(card.isVisible());
|
|
|
|
goog.testing.events.fireMouseOutEvent(james, john);
|
|
goog.testing.events.fireMouseOverEvent(john, james);
|
|
assertEquals('Should still show second card', james,
|
|
card.getAnchorElement());
|
|
assertTrue(card.isVisible());
|
|
|
|
shownCard = null;
|
|
timer.tick(501);
|
|
assertEquals('Should show first card again', john, shownCard);
|
|
assertTrue(card.isVisible());
|
|
|
|
// Test that cancelling while another is showing gives correct cancel
|
|
// information
|
|
cancelledElement = null;
|
|
goog.testing.events.fireMouseOutEvent(john, james);
|
|
goog.testing.events.fireMouseOverEvent(james, john);
|
|
goog.testing.events.fireMouseOutEvent(james, elsewhere);
|
|
assertEquals('Should cancel second card', james, cancelledElement);
|
|
}
|
|
|
|
|
|
/**
|
|
* Verify manual triggering.
|
|
*/
|
|
function testManualTrigger() {
|
|
initCard();
|
|
|
|
// Doesn't normally trigger for div tag
|
|
showDelay = 500;
|
|
goog.testing.events.fireMouseOverEvent(bill, elsewhere);
|
|
timer.tick(showDelay);
|
|
assertFalse(card.isVisible());
|
|
|
|
// Manually trigger element
|
|
card.triggerForElement(bill);
|
|
hideDelay = 600;
|
|
timer.tick(showDelay);
|
|
assertTrue(card.isVisible());
|
|
goog.testing.events.fireMouseOutEvent(bill, elsewhere);
|
|
goog.testing.events.fireMouseMoveEvent(document, offAnchor);
|
|
timer.tick(hideDelay);
|
|
assertFalse(card.isVisible());
|
|
}
|
|
|
|
|
|
/**
|
|
* Verify creating with isAnchor function.
|
|
*/
|
|
function testIsAnchor() {
|
|
// Initialize card so only bill triggers it.
|
|
initCard(function(element) {
|
|
return element == bill;
|
|
});
|
|
|
|
showDelay = 500;
|
|
goog.testing.events.fireMouseOverEvent(bill, elsewhere);
|
|
timer.tick(showDelay);
|
|
assertTrue('Should trigger card', card.isVisible());
|
|
|
|
hideDelay = 300;
|
|
goog.testing.events.fireMouseOutEvent(bill, elsewhere);
|
|
goog.testing.events.fireMouseMoveEvent(document, offAnchor);
|
|
timer.tick(hideDelay);
|
|
assertFalse(card.isVisible());
|
|
|
|
goog.testing.events.fireMouseOverEvent(john, elsewhere);
|
|
timer.tick(showDelay);
|
|
assertFalse('Should not trigger card', card.isVisible());
|
|
}
|
|
|
|
|
|
/**
|
|
* Verify mouse over child of anchor triggers hovercard.
|
|
*/
|
|
function testAnchorWithChildren() {
|
|
initCard();
|
|
|
|
showDelay = 500;
|
|
goog.testing.events.fireMouseOverEvent(james, elsewhere);
|
|
timer.tick(250);
|
|
|
|
// Moving from an anchor to a child of that anchor shouldn't cancel
|
|
// or retrigger.
|
|
var childBounds = goog.style.getBounds(child);
|
|
var inChild = new goog.math.Coordinate(childBounds.left + 1,
|
|
childBounds.top + 1);
|
|
goog.testing.events.fireMouseOutEvent(james, child);
|
|
goog.testing.events.fireMouseMoveEvent(child, inChild);
|
|
assertNull("Shouldn't cancel trigger", cancelledElement);
|
|
triggeredElement = null;
|
|
goog.testing.events.fireMouseOverEvent(child, james);
|
|
assertNull("Shouldn't retrigger card", triggeredElement);
|
|
timer.tick(250);
|
|
assertTrue('Card should show with original delay', card.isVisible());
|
|
|
|
hideDelay = 300;
|
|
goog.testing.events.fireMouseOutEvent(child, elsewhere);
|
|
goog.testing.events.fireMouseMoveEvent(child, offAnchor);
|
|
timer.tick(hideDelay);
|
|
assertFalse(card.isVisible());
|
|
|
|
goog.testing.events.fireMouseOverEvent(child, elsewhere);
|
|
timer.tick(showDelay);
|
|
assertTrue('Mouse over child should trigger card', card.isVisible());
|
|
}
|
|
|
|
function testNoTriggerWithMaxSearchSteps() {
|
|
initCard(undefined, true, 0);
|
|
|
|
showDelay = 500;
|
|
goog.testing.events.fireMouseOverEvent(child, elsewhere);
|
|
timer.tick(showDelay);
|
|
assertFalse('Should not trigger card', card.isVisible());
|
|
}
|
|
|
|
function testTriggerWithMaxSearchSteps() {
|
|
initCard(undefined, true, 2);
|
|
|
|
showDelay = 500;
|
|
goog.testing.events.fireMouseOverEvent(child, elsewhere);
|
|
timer.tick(showDelay);
|
|
assertTrue('Should trigger card', card.isVisible());
|
|
}
|
|
|
|
function testPositionAfterSecondTriggerWithMaxSearchSteps() {
|
|
initCard(undefined, true, 2);
|
|
|
|
showDelay = 500;
|
|
goog.testing.events.fireMouseOverEvent(john, elsewhere);
|
|
timer.tick(showDelay);
|
|
assertTrue('Should trigger card', card.isVisible());
|
|
assertEquals('Card cursor x coordinate should be 1',
|
|
card.position_.coordinate.x, 1);
|
|
card.cursorPosition = new goog.math.Coordinate(2, 2);
|
|
goog.testing.events.fireMouseOverEvent(child, elsewhere);
|
|
timer.tick(showDelay);
|
|
assertTrue('Should trigger card', card.isVisible());
|
|
assertEquals('Card cursor x coordinate should be 2',
|
|
card.position_.coordinate.x, 2);
|
|
}
|