In my last two tutorials, I explained how to use event delegation for showing and hiding tooltips. In a comment on the first one, Jan Aagaard asked how we might go about enhancing the script by adding a small delay before showing a tooltip. The answer lies with two JavaScript functions, setTimeout()
and clearTimeout()
.
Setting Up
Picking up where we left off last time, I'm going to create the tooltip div, declare a couple variables, and then bind 'mouseover,' 'mousemove,' and 'mouseout' all at once. In addition to $livetip
and tipTitle
, I'm declaring showTip
(and leaving it undefined for the moment) and delay
. The showTip
variable will be used as a reference for the setTimeout
function, and delay
will be used to set the time after which the function within setTimeout
's argument will execute (in this case, 300ms).
Of course, you don't have to use a variable for the delay, but I like to have that sort of setting up top where I can find it easily. Also, if I ever decide to convert this script into a plugin, having the delay stored in a variable will make it simpler to convert it to an option.
Setting the Timeout
Inside the .bind()
, and after the business of determining whether the event is being triggered by a link or one of its descendants, the setTimeout
is called on line 12:
- var tipTitle = '',
- showTip,
- delay = 300;
- var $link = $(event.target).closest('a');
- var link = $link[0];
- if (event.type == 'mouseover') {
- showTip = window.setTimeout(function() {
- tipTitle = link.title;
- link.title = '';
- $liveTip
- top: event.pageY + 12,
- left: event.pageX + 12
- });
- }, delay);
- }
- // ... code continues
- });
setTimeout
is actually a method of the window
object; as such, the this
keyword inside its function argument will refer to window
. It's not of much concern here because, with event delegation, we're relying on event.target
rather than this
, but I thought it might be worth mentioning anyway.
Note that within the setTimeout
function, I'm storing a little data (line 13) for the link to indicate that the tooltip is active on it. This tipActive
information then determines what to do on mouseout.
Clearing the Timeout
On mouseout, if the link has the tipActive
data, that data is removed (line 29, below), the tooltip is hidden (line 30), and the link's original title attribute value is restored if tipTitle has a value (line 31). If that tipActive
data is not associated with the link, then the timeout that was set on mouseover is reset with clearTimeout
(line 33).
- var tipTitle = '',
- showTip,
- delay = 300;
- var $link = $(event.target).closest('a');
- var link = $link[0];
- if (event.type == 'mouseover') {
- showTip = window.setTimeout(function() {
- tipTitle = link.title;
- link.title = '';
- $liveTip
- top: event.pageY + 12,
- left: event.pageX + 12
- });
- }, delay);
- }
- if (event.type == 'mouseout') {
- link.title = tipTitle || link.title;
- } else {
- window.clearTimeout(showTip);
- }
- }
- top: event.pageY + 12,
- left: event.pageX + 12
- });
- }
- });
The mousemove part hasn't changed from what it was in the previous post, except that it first checks if the 'tipActive' data is assigned to the link. I figured that setting the top
and left
properties of the tooltip is probably more work than doing a little check on the link's data, so why bother setting them if I don't have to?
So, that's it. Just add a simple setTimeout
delay and conditionally clear it. If you want to set a delay on mouseout, you can do that the same way.
Check out the demo or download the zip (or neither, if you're stubborn).