Как я могу восстановить дерево DOM? - PullRequest
2 голосов
/ 28 декабря 2010

Итак, у меня есть ряд вложенных элементов ul как часть дерева, как показано ниже:

<ul>
<li>
    <ul>
        <li>1.1</li>
        <li>1.2</li>
    </ul>
    <ul>
        <li>2.1</li>
        <li>
            <ul>
                <li>2.2</li>
            </ul>
        </li>
    </ul>
    <ul>
        <li>3.1</li>
        <li>3.2</li>
    </ul>
</li>
</ul>

Скажем, когда 3.1 - выбранный узел, а когда пользователь нажимает предыдущий, выбранный узел должен быть равен 2.2.,Плохая новость заключается в том, что может быть любое количество уровней.Как я могу найти предыдущий узел (li) по отношению к текущему выбранному узлу, используя jquery?

Ответы [ 5 ]

5 голосов
/ 28 декабря 2010

В DOM определены классы обхода документов, которые позволяют последовательно обрабатывать содержимое дерева. Классы TreeWalker и NodeIterator для идеальных кандидатов на это.

Сначала мы создаем экземпляр TreeWalker, который может последовательно перебирать узлы <li>.

var walker = document.createTreeWalker(
    document.body, 
    NodeFilter.SHOW_ELEMENT,
    function(node) {
            var hasNoElements = node.getElementsByTagName('*').length == 0;
            return (node.nodeName == "LI" && hasNoElements) ? 
                        NodeFilter.FILTER_ACCEPT : NodeFilter.FILTER_SKIP;
    },
    false
);

Далее мы итерируем до текущего узла в этом TreeWalker. Допустим, у нас была ссылка на наш текущий узел в myNode. Мы пройдем по этому ходунку, пока не достигнем myNode или нулевого значения.

while(walker.nextNode() != myNode && walker.currentNode);

Достигнув нашего узла в этом TreeWalker, получить предыдущий узел - кусок пирога.

var previousNode = walker.previousNode();

Вы можете попробовать пример здесь .


Вот общая версия бродилки по дереву. Это крошечная оболочка для интерфейса TreeWalker.

function backwardsIterator(startingNode, nodeFilter) {
    var walker = document.createTreeWalker(
        document.body, 
        NodeFilter.SHOW_ELEMENT,
        function(node) {
            return nodeFilter(node) ? NodeFilter.FILTER_ACCEPT : NodeFilter.FILTER_SKIP
        },
        false
    );

    walker.currentNode = startingNode;

    return walker;
}

Требуется начальный узел и пользовательская функция фильтра для удаления нежелательных элементов. Вот пример использования запуска в данном узле и только включения листовых li элементов.

var startNode = document.getElementById("2.1.1");

// creates a backwards iterator with a given start node, and a function to filter out unwanted elements.
var iterator = backwardsIterator(startNode, function(node) {
    var hasNoChildElements = node.childElementCount == 0;
    var isListItemNode = node.nodeName == "LI";

    return isListItemNode && hasNoChildElements;
});

// Call previousNode() on the iterator to walk backwards by one node.
// Can keep calling previousNode() to keep iterating backwards until the beginning.
iterator.previousNode()

Обновлено с интерактивным примером . Нажмите на элемент списка, чтобы выделить предыдущий элемент списка.

2 голосов
/ 28 декабря 2010

Вот как это сделать как плагин jQuery (лицензируется по лицензии MIT). Загрузите его сразу после загрузки jQuery:

/*
 * jQuery Previous/Next by Tag Name Plugin.
 * Copyright (c) 2010 idealmachine
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in
 * all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 * THE SOFTWARE.
 *
 */

(function($) {
    $.fn.prevByTagName = function(containerSelector) {
        return this.map(function() {
            var container = containerSelector ? $(this).parents(containerSelector).last()[0] : document,
                others = container.getElementsByTagName(this.tagName),
                result = null;
            for(var i = 0; i < others.length; ++i) {
                if(others[i] === this) {
                    break;
                }
                result = others[i];
            }
            return result;
        });
    };
    $.fn.nextByTagName = function(containerSelector) {
        return this.map(function() {
            var container = containerSelector ? $(this).parents(containerSelector).last()[0] : document,
                others = container.getElementsByTagName(this.tagName),
                result = null;
            for(var i = others.length; i--;) {
                if(others[i] === this) {
                    break;
                }
                result = others[i];
            }
            return result;
        });
    };
})(jQuery);

// End of plugin

Чтобы найти последний предыдущий элемент li (самый внешний ul с идентификатором myUL), который не имеет ul потомков, вы можете использовать плагин для $(this), например:

var result = $(this);
while((result = result.prevByTagName('#myUL')).children('ul').length);

Вы можете попробовать это на jsFiddle .

2 голосов
/ 28 декабря 2010

Вы можете сделать это следующим образом в jQuery:

http://jsfiddle.net/haP5c/

$('li').click(function() {
    var $this = $(this);

    if ($this.children().length == 0) {
        var $lis = $('#myUL').find('li'),
            indCount = 0,
            prevLI = null;

        $lis.each(function(ind, el) {
            if (el == $this[0]) {
                indCount = ind;
                return false;
            }
        });

        for (indCount=indCount-1; indCount >= 0; indCount--) {
            if ($($lis[indCount]).children().size() == 0) {
                prevLI = $lis[indCount];
                break;
            }
        }

        if (prevLI) {
            alert($(prevLI).text());
        }
    }
});

По сути, если вы щелкнете по элементу, он будет искать предыдущий узел LI, который не 'У меня нет детей.Если это так, он считает его предыдущим в цепочке.Как вы можете видеть на jsfiddle, он отлично работает.

1 голос
/ 28 декабря 2010

Слегка "раздражен" тем, что у моего простого подхода были недостатки, вот одно решение с XPath:

$('li').click(function() {
    var lis = document.getElementsByTagName("li"), 
        prev = null,
        myCurLI = this;

    if ($(this).children().length == 0) {
        // collect all LI-elements that are leaf nodes
        var expr = "//li[count(child::*) = 0]", xp, cur;

        xp = document.evaluate(expr, document, null, XPathResult.ORDERED_NODE_ITERATOR_TYPE, null);

        while (cur = xp.iterateNext()) {
            if (cur == myCurLI) break;
            prev = cur;
        }

        if (prev) alert($(prev).text());
        else alert("No previous list element found");
    }
});

ПРИМЕЧАНИЕ: Я скопировал код обработки событий из ответа treeface, так как я не очень знаком с этой платформой.

1 голос
/ 28 декабря 2010

Довольно простой подход, который бы работал без какой-либо JavaScript-инфраструктуры:

var lis = document.getElementsByTagName("li"), prev;

for (var i = 0; i < lis.length; i++) {
    if (lis[i] == myCurLI) break;
    prev = lis[i];
}

Теперь prev содержит предыдущий LI.


РЕДАКТИРОВАТЬ: Пожалуйста, смотрите мой другой ответ для решения с использованием XPath, в котором нет недостатков, упомянутых treeface.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...