On the front page of a site I am building, several <div>
s use the CSS :hover
pseudo-class to add a border when the mouse is over them. One of the <div>
s contains a <form>
which, using jQuery, will keep the border if an input within it has focus. This works perfectly except that IE6 does not support :hover
on any elements other than <a>
s. So, for this browser only we are using jQuery to mimic CSS :hover
using the $(#element).hover()
method. The only problem is, now that jQuery handles both the form focus()
and hover()
, when an input has focus then the user moves the mouse in and out, the border goes away.
I was thinking we could use some kind of conditional to stop this behavior. For instance, if we tested on mouse out if any of the inputs had focus, we could stop the border from going away. AFAIK, there is no :focus
selector in jQuery, so I'm not sure how to make this happen. Any ideas?
-
2can you try to boil that down to a few sentences of what is required, and what i happening?mkoryak– mkoryak2009年06月08日 21:28:28 +00:00Commented Jun 8, 2009 at 21:28
-
I think you need to look into capturing the focus and blur events (docs.jquery.com/Events/focus and docs.jquery.com/Events/blur)Wolfr– Wolfr2009年06月08日 21:34:54 +00:00Commented Jun 8, 2009 at 21:34
-
I posted an answer on the related question "Is there a ‘has focus’ in JavaScript (or jQuery)?". I've enhanced [cletus approach cited by gnarf](#2684561).luistamawong– luistamawong2010年08月13日 15:36:55 +00:00Commented Aug 13, 2010 at 15:36
15 Answers 15
jQuery 1.6+
jQuery added a :focus
selector so we no longer need to add it ourselves. Just use $("..").is(":focus")
jQuery 1.5 and below
Edit: As times change, we find better methods for testing focus, the new favorite is this gist from Ben Alman:
jQuery.expr[':'].focus = function( elem ) {
return elem === document.activeElement && ( elem.type || elem.href );
};
Quoted from Mathias Bynens here:
Note that the
(elem.type || elem.href)
test was added to filter out false positives like body. This way, we make sure to filter out all elements except form controls and hyperlinks.
You're defining a new selector. See Plugins/Authoring. Then you can do:
if ($("...").is(":focus")) {
...
}
or:
$("input:focus").doStuff();
Any jQuery
If you just want to figure out which element has focus, you can use
$(document.activeElement)
If you aren't sure if the version will be 1.6 or lower, you can add the :focus
selector if it is missing:
(function ( $ ) {
var filters = $.expr[":"];
if ( !filters.focus ) {
filters.focus = function( elem ) {
return elem === document.activeElement && ( elem.type || elem.href );
};
}
})( jQuery );
-
4This can cause false positives. See stackoverflow.com/questions/967096/… for more information (thanks to Ben Alman and Diego Perini).Mathias Bynens– Mathias Bynens2011年03月22日 13:01:47 +00:00Commented Mar 22, 2011 at 13:01
-
@Mathias Bynens - Thanks for the update (you're more than welcome to edit this community wiki answer btw!)gnarf– gnarf2011年03月22日 21:31:41 +00:00Commented Mar 22, 2011 at 21:31
-
What about when you need it to work with both 1.5- and 1.6? I don't want to override jQuery's own focus selector. Something like
if (!jQuery.expr[':'].focus) { /* gist from Ben Alman */ }
?Betamos– Betamos2011年06月03日 19:11:52 +00:00Commented Jun 3, 2011 at 19:11 -
@betamos - Exactly right... I'll add something into the answer to reflect that.gnarf– gnarf2011年06月03日 20:51:45 +00:00Commented Jun 3, 2011 at 20:51
-
2@Alex - Please provide a reduced test case on jsFiddle showing the problem, because as far as I can tell, there is no reason this should not work on "dynamic" elements. I call shenanigans...gnarf– gnarf2012年06月14日 22:14:01 +00:00Commented Jun 14, 2012 at 22:14
CSS:
.focus {
border-color:red;
}
JQuery:
$(document).ready(function() {
$('input').blur(function() {
$('input').removeClass("focus");
})
.focus(function() {
$(this).addClass("focus")
});
});
Here’s a more robust answer than the currently accepted one:
jQuery.expr[':'].focus = function(elem) {
return elem === document.activeElement && (elem.type || elem.href);
};
Note that the (elem.type || elem.href)
test was added to filter out false positives like body
. This way, we make sure to filter out all elements except form controls and hyperlinks.
April 2015 Update
Since this question has been around a while, and some new conventions have come into play, I feel that I should mention the .live
method has been depreciated.
In its place, the .on
method has now been introduced.
Their documentation is quite useful in explaining how it works;
The .on() method attaches event handlers to the currently selected set of elements in the jQuery object. As of jQuery 1.7, the .on() method provides all functionality required for attaching event handlers. For help in converting from older jQuery event methods, see .bind(), .delegate(), and .live().
So, in order for you to target the 'input focused' event, you can use this in a script. Something like:
$('input').on("focus", function(){
//do some stuff
});
This is quite robust and even allows you to use the TAB key as well.
-
1very good solution, thank you. Use
on("focusout",
for the opposite functionBarbz_YHOOL– Barbz_YHOOL2020年08月19日 23:30:17 +00:00Commented Aug 19, 2020 at 23:30
I'm not entirely sure what you're after but this sounds like it can be achieved by storing the state of the input elements (or the div?) as a variable:
$('div').each(function(){
var childInputHasFocus = false;
$(this).hover(function(){
if (childInputHasFocus) {
// do something
} else { }
}, function() {
if (childInputHasFocus) {
// do something
} else { }
});
$('input', this)
.focus(function(){
childInputHasFocus = true;
})
.blur(function(){
childInputHasFocus = false;
});
});
-
This is what I wound up doing, but I used an arbitrary CSS class rather than a global javascript variable.bloudermilk– bloudermilk2009年06月08日 21:42:53 +00:00Commented Jun 8, 2009 at 21:42
-
I used a variation of this. No new jQuery plugin needed. That makes me a happy camper!Dennis Day– Dennis Day2010年11月08日 21:06:13 +00:00Commented Nov 8, 2010 at 21:06
An alternative to using classes to mark the state of an element is the internal data store functionality.
P.S.: You are able to store booleans and whatever you desire using the data()
function. It's not just about strings :)
$("...").mouseover(function ()
{
// store state on element
}).mouseout(function ()
{
// remove stored state on element
});
And then it's just a matter of accessing the state of elements.
if anyone cares there is a much better way to capture focus now, $(foo).focus(...)
Have you thought about using mouseOver and mouseOut to simulate this. Also look into mouseEnter and mouseLeave
-
That is essentially what I am doing with jQuery's hover. You supply two functions, one for mouse in and one for mouse out.bloudermilk– bloudermilk2009年06月08日 21:30:40 +00:00Commented Jun 8, 2009 at 21:30
Keep track of both states (hovered, focused) as true/false flags, and whenever one changes, run a function that removes border if both are false, otherwise shows border.
So: onfocus sets focused = true, onblur sets focused = false. onmouseover sets hovered = true, onmouseout sets hovered = false. After each of these events run a function that adds/removes border.
As far as I know, you can't ask the browser if any input on the screen has focus, you have to set up some sort of focus tracking.
I usually have a variable called "noFocus" and set it to true. Then I add a focus event to all inputs that makes noFocus false. Then I add a blur event to all inputs that set noFocus back to true.
I have a MooTools class that handles this quite easily, I'm sure you could create a jquery plugin to do the same.
Once that's created, you could do check noFocus before doing any border swapping.
There is no :focus, but there is :selected http://docs.jquery.com/Selectors/selected
but if you want to change how things look based on what is selected you should probably be working with the blur events.
There is a plugin to check if an element is focused: http://plugins.jquery.com/project/focused
$('input').each(function(){
if ($(this) == $.focused()) {
$(this).addClass('focused');
}
})
-
2why use a plugin when you can just use core jquery?Marcy Sutton– Marcy Sutton2011年12月16日 18:40:00 +00:00Commented Dec 16, 2011 at 18:40
-
1You are right. But I didn't know a solution for this problem at 2 years ago.Murat Çorlu– Murat Çorlu2011年12月17日 22:04:28 +00:00Commented Dec 17, 2011 at 22:04
I had a .live("focus") event set to select() (highlight) the contents of a text input so that the user wouldn't have to select it before typing a new value.
$(formObj).select();
Because of quirks between different browsers, the select would sometimes be superseded by the click that caused it, and it would deselect the contents right after in favor of placing the cursor within the text field (worked mostly ok in FF but failed in IE)
I thought I could solve this by putting a slight delay on the select...
setTimeout(function(){$(formObj).select();},200);
This worked fine and the select would persist, but a funny problem arose.. If you tabbed from one field to the next, the focus would switch to the next field before the select took place. Since select steals focus, the focus would then go back and trigger a new "focus" event. This ended up in a cascade of input selects dancing all over the screen.
A workable solution would be to check that the field still has focus before executing the select(), but as mentioned, there's no simple way to check... I ended up just dispensing with the whole auto highlight, rather than turning what should be a single jQuery select() call into a huge function laden with subroutines...
What I wound up doing is creating an arbitrary class called .elementhasfocus which is added and removed within the jQuery focus() function. When the hover() function runs on mouse out, it checks for .elementhasfocus:
if(!$("#quotebox").is(".boxhasfocus")) $(this).removeClass("box_border");
So if it doesn't have that class (read: no elements within the div have focus) the border is removed. Otherwise, nothing happens.
Simple
<input type="text" />
<script>
$("input").focusin(function() {
alert("I am in Focus");
});
</script>
-
That does not tell you which element is currently in focus. It is called when a new element gets the focus. So if an element already has the focus and you want to know whether 'this' is the one, this code is useless.Alexis Wilke– Alexis Wilke2014年02月01日 00:46:04 +00:00Commented Feb 1, 2014 at 0:46
Explore related questions
See similar questions with these tags.