ReactJs:嵌套元素中的转换

ReactJs: transitions in nested elements

我是 ReactJs 的新手,开发了多嵌套文件夹浏览 - 我的意思是这里 - https://jsfiddle.net/69z2wepo/7804/ .

var ReactCSSTransitionGroup = React.addons.CSSTransitionGroup;
      var List = React.createClass({
         getInitialState: function() {
            return {data:this.props.data};
        },
        onClickHandler: function (nodeName, i, e) {
            e.stopPropagation();
            var newState = this.state.data[nodeName][i].isExpanded?false:true;
            this.state.data[nodeName][i].isExpanded = newState;
            this.setState(this.state);
        },
        render: function() {

            var getList = function(nodes) {

                var listElements = nodes.map(function(node) {
                    return this.state.data[node].map(function (nodeData,i) {
                        var subList = null;
                        if(nodeData.isExpanded) {
                            subList = getList(nodeData.nodes);
                        }
                        return (<li  onClick={this.onClickHandler.bind(this, node, i)}>
                                {nodeData.label}{subList}
                            </li>);
                    }.bind(this));
                }.bind(this));

                return <ul>{listElements}</ul>;

            }.bind(this);


            return  getList(['root'], 0);
        }
    });



    var data = {
        root:[{
            label:'1',
            nodes: ['a','b']
        },{
            label:'2',
            nodes: ['d','e']
        }],
        a:[{
            label:'label a',
            nodes: ['c']
        }],
        b:[{
            label: 'label b',
            nodes:[]
        }],
        c:[{
            label: 'label c',
            nodes: []
        }],
        d:[{
            label: 'label d',
            nodes: []
        }],
        e:[{
            label: 'label e',
            nodes: []
        }]
    }
    React.render(<List data={data} />, document.getElementById('container'));

一切正常,隐藏和显示嵌套节点。现在我想将所有内容都包装在 CSSTransitionGroup 中,以便在隐藏和显示时进行转换。我在这里阅读了所有相关内容 - https://facebook.github.io/react/docs/animation.html 我真的很困惑我应该如何在这个多嵌套节点场景中实现它。

有人能指出我正确的方向吗?

谢谢。 托马斯

所以在经历了 x 痛苦但快乐的时光后,我找到了解决方案。 此处更新 - https://jsfiddle.net/69z2wepo/7856/

为什么会痛?好吧,我花了很长时间才弄清楚 ReactCSSTransitionGroup 需要始终在 DOM 中,即使我没有添加 nested 。

我还需要以与嵌套相同的方式嵌套 ReactCSSTransitionGroup,ReactCSSTransitionGroup 创建特殊元素,子元素将被动画化。

而 ReactCSSTransitionGroup 只关心顶级子项的动画。所以如果你 ONLY 改变状态(removing/adding)嵌套 div 但离开它的父级 是 ReactCSSTransitionGroup 的子级 它会不是动画。

代码:

    var ReactCSSTransitionGroup = React.addons.CSSTransitionGroup;
    var List = React.createClass({
        getInitialState: function() {
        return {data:this.props.data};
    },
    onClickHandler: function (nodeName, i, e) {
        e.stopPropagation();
        var newState = this.state.data[nodeName][i].isExpanded?false:true;
        this.state.data[nodeName][i].isExpanded = newState;
        this.setState(this.state);
    },
    render: function() {

        var getList = function(nodes) {

            var listElements = nodes.map(function(node) {
                return this.state.data[node].map(function (nodeData,i) {
                    var subList = '';
                    if(nodeData.isExpanded) {
                        subList = getList(nodeData.nodes);
                    }
                    return (<li  onClick={this.onClickHandler.bind(this, node, i)}>
                            {nodeData.label}
                            <ReactCSSTransitionGroup transitionName="example">
                                {subList}
                            </ReactCSSTransitionGroup>
                        </li>);
                }.bind(this));
            }.bind(this));

            return <ul key={nodes.join('_')}>{listElements}</ul>;

        }.bind(this);


        return  getList(['root'], 0);
    }
});



var data = {
    root:[{
        label:'1',
        nodes: ['a','b']
    },{
        label:'2',
        nodes: ['d','e']
    }],
    a:[{
        label:'label a',
        nodes: ['c']
    }],
    b:[{
        label: 'label b',
        nodes:[]
    }],
    c:[{
        label: 'label c',
        nodes: []
    }],
    d:[{
        label: 'label d',
        nodes: []
    }],
    e:[{
        label: 'label e',
        nodes: []
    }]
}
React.render(<List data={data} />, document.getElementById('container'));