blob: 6193db4fcaf81c7bc197dbc1edb5ea9805e615fa [file] [log] [blame]
// 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);
}