Скрыть детей и братьев и сестер для D3 Node Tree с поиском - PullRequest
0 голосов
/ 13 июня 2019

Итак, я пытался создать дерево узлов D3.js, которое похоже на "Поиск CollapsibleTree" Патрика Брокмана * .Моя цель - когда пользователь ищет термин в раскрывающемся поиске ввода, вместо того, чтобы расширять дерево выделенным путем выбранного термина, моя цель состоит в том, чтобы выделить термин и его родителей.В основном, показать прямую линию.После этого, если вы выберете узел на прямой линии, дерево раскроется, а путь будет по-прежнему выделен (когда дерево расширяется и вы щелкаете по красному узлу, выделение удаляется).В любом случае, существует проблема, когда на определенных узлах в этом процессе небольшое количество узлов выбрасывается из окна (значение Y исчисляется тысячами).Вот ссылка на мой Bl.ocks, «Дерево узлов организации с поиском и скрытием дочернего элемента / дочернего элемента»

Чтобы воссоздать проблему, выполните следующие действия (я добавил console.logs вкод, показывающий, что некоторые из них не совпадают): 1. Загрузите страницу 2. Нажмите «Analytics» (console.log) 3. Найдите и выберите «AggregateExpression» 4. Нажмите «query» 5. Нажмите на«Аналитика» (console.log)

Если вы сравниваете console.logs с того момента, когда нажали «Analytics», массив будет другим.Один - 3, а один - 6. Почти все работает, за исключением того, что вы просто щелкаете по дереву и выполняете поиск, есть несколько узлов, которые просто вылетают со страницы.

Вот пример JavaScript.Это все еще нуждается в рефакторинге, но опять же, он функционирует по большей части:

        //===============================================
        function select2DataCollectName(d) {
            if (d.children)
                d.children.forEach(select2DataCollectName);
            else if (d._children)
                d._children.forEach(select2DataCollectName);
            select2Data.push(d.name);
        }

        //===============================================
        function searchTree(d) {
            if (d.children)
                d.children.forEach(searchTree);
            else if (d._children)
                d._children.forEach(searchTree);
            var searchFieldValue = eval(searchField);
            if (searchFieldValue && searchFieldValue.match(searchText)) {
                // Walk parent chain
                var ancestors = [];
                var parent = d;
                while (typeof (parent) !== "undefined") {
                    ancestors.push(parent);
                    //console.log(parent);
                    parent.class = "found";
                    parent = parent.parent;
                }
                //console.log(ancestors);
            }
        }

        //===============================================
        function clearAll(d) {
            d.class = "";
            if (d.children)
                d.children.forEach(clearAll);
            else if (d._children)
                d._children.forEach(clearAll);
        }

        //===============================================
        function collapse(d) {
            if (d.children) {
                d._children = d.children;
                d._children.forEach(collapse);
                d.children = null;
            }
        }

        function expandEverything(d) {
            var children = (d.children) ? d.children : d._children;
            if (d._children) {
                d.children = d._children;
                d._children = null;
            }
            if (children)
                children.forEach(expandEverything);
        }

        function expandUp() {
            expandEverything(root);
            update(root);
        }

        function collapseDown() {
            root.children.forEach(collapse);
            collapse(root);
            update(root);
        }

        //===============================================
        function collapseAllNotFound2(d) {
            if (d.children) {
                if (d.class !== "found") {
                    d._children = d.children;
                    d._children.forEach(collapseAllNotFound);
                    d.children = null;
                } else
                    d.children.forEach(collapseAllNotFound);
            }
        }

        //===============================================
        var childrenToRemove = [];
        function collapseAllNotFound(d) {
            if (d.children) {

                if (d.class !== "found") {
                    d._children = d.children;
                    d._children.forEach(collapseAllNotFound);
                    d.children = null;

                    childrenToRemove.push(d);
                } else {
                    d.children.forEach(collapseAllNotFound);
                }
            }
            else {
                if (d.class !== "found" && d.parent.class === "found") {
                    childrenToRemove.push(d);
                }
            }
        }

        function resetTree(d) {
            if (d.children) {
                if (d._children) {
                    d._children = d._children.concat(d.children);
                }
                else {
                    d._children = d.children
                }
                d.children = undefined;
                d._children.forEach(resetTree);
            }
            else if (d._children) {
                d._children.forEach(resetTree);
            }
        }

        //===============================================
        function expandAll(d) {
            if (d._children) {
                d.children = d._children;
                d.children.forEach(expandAll);
                d._children = null;
            } else if (d.children)
                d.children.forEach(expandAll);
        }

        //Toggle children on click.
        function toggle(d) {
            var circleTest = $(childrenToRemove).hasClass("found");
            childrenToRemove = [];
            if (d.class === "found") {
                clearAll(root);
                expandAll(root);
                update(root);
                searchField = "d.name";
                searchText = $(".select2-chosen").text();
                searchTree(root);
                root.children.forEach(collapseAllNotFound2);
                update(root);
                childrenToRemove.class = "";
                d.class = "";
                parent.class = "";
                d.parent.class = "";
                parent.parent.class = "";
                d.children.class = "";
                d3.selectAll("circle").attr("class", "");
            } else if (d.class !== "found") {
                console.log(d);
                console.log(d._children);
                d.children = null;
                if (d.children) {
                    d._children = d.children;
                    d.children = null;
                } else {
                    d.children = d._children;
                    d._children = null;
                }
                clearAll(root);
                update(d);
                $("#searchName").select2("val", "");
            }
        }

        // Focus on the section above
        //=============================================================================================================================================
        //===============================================
        $("#searchName").on("select2-selecting", function (e) {
            clearAll(root);
            expandAll(root);
            update(root);
            searchField = "d.name";
            searchText = e.object.text;
            searchTree(root);
            root.children.forEach(collapseAllNotFound);

            for (var index in childrenToRemove) {
                var d = childrenToRemove[index];
                if (d.parent.children) {
                    d.parent.children.splice(d.parent.children.indexOf(d), 1);
                }
                if (d.parent._children) {
                    d.parent._children.push(d);
                }
                else {
                    d.parent._children = [d];
                }
            }
            childrenToRemove = [];
            update(root);
            resetTree(root);
        });

        //===============================================
        var margin = { top: 20, right: 120, bottom: 20, left: 120 },
            width = 960 - margin.right - margin.left,
            height = 800 - margin.top - margin.bottom;

        var i = 0,
            duration = 750,
            root;

        var tree = d3.layout.tree()
            .size([height, width]);

        var diagonal = d3.svg.diagonal()
            .projection(function (d) { return [d.y, d.x]; });

        var svg = d3.select("body").append("svg")
            .attr("id", "test-id")
            .attr("width", width + margin.right + margin.left)
            .attr("height", height + margin.top + margin.bottom)
            .append("g")
            .attr("transform", "translate(" + margin.left + "," + margin.top + ")");

        d3.json("flare-long.json", function (error, flare) {
            root = flare;
            root.x0 = height / 2;
            root.y0 = 0;

            select2Data = [];
            select2DataCollectName(root);
            select2DataObject = [];
            select2Data.sort(function (a, b) {
                if (a > b) return 1; // sort
                if (a < b) return -1;
                return 0;
            })
                .filter(function (item, i, ar) {
                    return ar.indexOf(item) === i;
                }) // remove duplicate items
                .filter(function (item, i, ar) {
                    select2DataObject.push({
                        "id": i,
                        "text": item
                    });
                });
            $("#searchName").select2({
                data: select2DataObject,
                containerCssClass: "search"
            });

            function collapse(d) {
                if (d.children) {
                    d._children = d.children;
                    d._children.forEach(collapse);
                    d.children = null;
                }
            }

            root.children.forEach(collapse);
            root.children.forEach(function (d) { d.hidden = false; });
            root.hidden = false;
            update(root);
        });

        d3.select(self.frameElement).style("height", "800px");

        function update(source) {

            // Compute the new tree layout.
            var nodes = tree.nodes(root).filter(function (d) { return !d.hidden; }).reverse(),
                links = tree.links(nodes);

            // Normalize for fixed-depth.
            nodes.forEach(function (d) { d.y = d.depth * 180; });

            // Update the nodes…
            var node = svg.selectAll("g.node")
                .data(nodes, function (d) { return d.id || (d.id = ++i); });

            // Enter any new nodes at the parent's previous position.
            var nodeEnter = node.enter().append("g")
                .attr("class", "node")
                .attr("transform", function (d) { return "translate(" + source.y0 + "," + source.x0 + ")"; })
                .on("click", toggle);

            nodeEnter.append("circle")
                .attr("r", 1e-6)
                .style("fill", function (d) { return d._children ? "lightsteelblue" : "#fff"; });

            // Possible use in the future.  Needs adjustable width based on text.
            // nodeEnter.append("rect")
            //     .attr("rx", 6)
            //     .attr("ry", 6)
            //     .attr("x", function (d) { return d.children || d._children ? -140 : -10; })
            //     .attr('y', -10)
            //     .attr("width", 150)
            //     .attr("height", 20)
            //     .style("fill", "#fff")
            //     .style("opacity", 0.2)
            //     .style("border-radius", "5px");


            nodeEnter.append("text")
                .attr("x", function (d) { return d.children || d._children ? -10 : 10; })
                .attr("dy", ".35em")
                .attr("text-anchor", function (d) { return d.children || d._children ? "end" : "start"; })
                .text(function (d) { return d.name; })
                .style("fill-opacity", 1e-6);

            // Transition nodes to their new position.
            var nodeUpdate = node.transition()
                .duration(duration)
                .attr("transform", function (d) { return "translate(" + d.y + "," + d.x + ")"; });

            nodeUpdate.select("circle")
                .attr("r", 4.5)
                .style("fill", function (d) {
                    if (d.class === "found") {
                        return "#ff4136"; //red
                    } else if (d._children) {
                        return "lightsteelblue";
                    } else {
                        return "#fff";
                    }
                })
                .style("stroke", function (d) {
                    if (d.class === "found") {
                        return "#ff4136"; //red
                    }
                });

            nodeUpdate.select("text")
                .style("fill-opacity", 1);

            // Transition exiting nodes to the parent's new position.
            var nodeExit = node.exit().transition()
                .duration(duration)
                .attr("transform", function (d) { return "translate(" + source.y + "," + source.x + ")"; })
                .remove();

            nodeExit.select("circle")
                .attr("r", 1e-6);

            nodeExit.select("text")
                .style("fill-opacity", 1e-6);

            // Update the links…
            var link = svg.selectAll("path.link")
                .data(links, function (d) { return d.target.id; });

            // Enter any new links at the parent's previous position.
            link.enter().insert("path", "g")
                .attr("class", "link")
                .attr("d", function (d) {
                    var o = { x: source.x0, y: source.y0 };
                    return diagonal({ source: o, target: o });
                });

            // Transition links to their new position.
            link.transition()
                .duration(duration)
                .attr("d", diagonal)
                .style("stroke", function (d) {
                    if (d.target.class === "found") {
                        return "#ff4136";
                    }
                });

            // Transition exiting nodes to the parent's new position.
            link.exit().transition()
                .duration(duration)
                .attr("d", function (d) {
                    var o = { x: source.x, y: source.y };
                    return diagonal({ source: o, target: o });
                })
                .remove();

            // Stash the old positions for transition.
            nodes.forEach(function (d) {
                d.x0 = d.x;
                d.y0 = d.y;
            });
        }

Пожалуйста, дайте мне знать, если у вас есть какие-либо предложения или идеи.Заранее спасибо.

...