tags:

views:

428

answers:

7

Hi folks, i am in the process of implementing JQuery, and taking out Prototype libraries in my codebase, and i am wondering if you could give me the best way to implement this functionality in jQuery. I am familiar with the jQuery ancestor>descendant sntax, but just want to check if an element is a descendant by true of false, like the code below: can someone give me the most efficient jQuery solution for this ? Thanks!

<div id="australopithecus">
  <div id="homo-herectus">
    <div id="homo-sapiens"></div>
  </div>
</div>

$('homo-sapiens').descendantOf('australopithecus');
// -> true

$('homo-herectus').descendantOf('homo-sapiens');
// -> false
A: 

You could attempt to .find() it in the Elements .children()

$("#lucy").find("#homo-erectus").length;

Or the opposite direction:

$("#homo-erectus").parents().find("#lucy").length;
Jonathan Sampson
I don't know the jQuery syntax, but it would be quicker to check the parents() of the descendant item, rather than all the children of the ancestor, right?
harpo
@harpo, Potentially. Depending on users particular DOM. Good advice though - definitely something to consider.
Jonathan Sampson
A: 

How about


$("#homo-herectus").parents().is("#australopithecus");
mgroves
A: 
function descendantOf(parentId, childId) {
   return ( $('#'+parentId+' > #'+childId).length === 1 );
}

That should work.

As was pointed out in the comment below, if you don't want it to be just direct descendants:

function descendantOf(parentId, childId) {
   return ( $('#'+childId, $('#'+parentId)).length === 1 );
}
Thomas
The use of the > character in the selector will limit the check to only first-generation decdendants. If you want to check to see if its any kind of descendant (as opposed to direct descendant) then simply remove the > character.
Ken Browning
+3  A: 

I would think you could take advantage of CSS style selection here, with returned length..

$('#australopithecus #homo-sapiens').length // Should be 1
$('#homo-sapiens #homo-herectus').length // Should be 0

Not exactly true/false, but checking 0/1 as a boolean should work. :)

Alternately, you could do something like $('#parent').find('#child') and check the length there.

Brian Arnold
thanks for all the good answers!! i didn't realize i had so many options :)
29er
A: 

You can use the is() function like so:

alert($('#homo-sapiens').is('#australopithecus *'));
// -> true

alert($('#homo-herectus').is('#homo-sapiens *'));
// -> false
John Kugelman
A: 

Supposing to rewrite your initial statement in:

$('#homo-sapiens').descendantOf('#australopithecus');

try to plugin:

(function($) {
    $.fn.descendantOf = function(parentId) {
        return this.closest(parentId).length != 0;
    }
})(jQuery)
tanathos
A: 
$.fn.descendantOf = function(element) {
    element = $(element)[0];
    var current = this;
    var body    = document.body;
    while (current && current != element && current != document.body) {
        current = $(current).parent()[0];
    }
    if (typeof(current) == "undefined" || typeof(current) == "null") {
        return false;
    } else if (current == element) {
        return true;
    } else if (current == document.body) {
        return false;
    }
}

Example:

<div id="foo">
    <div id="bar">
        <div id="baz"></div>
    </div>
</div>

And:

$('#foo').descendantOf('#bar');  // false
$('#foo').descendantOf('#foo');  // false
$('#foo').descendantOf(document.body);  // true
$('#bar').descendantOf('#foo');  // true
$('#baz').descendantOf('#foo');  // true
Hongli