渲染单击的节点子节点,并隐藏其他节点的子节点

时间:2018-12-17 11:35:05

标签: reactjs mobx

当我单击节点时,我会发送一个API请求,该请求将带给我所有需要渲染的节点子节点。

我已经实现了此功能,但是存在性能错误。单击节点时,需要渲染其所有子级,并隐藏所有先前单击的节点子级。现在,所有单击的子节点都保留在DOM中。

这是Node组件:

@withRouter
@inject('dataStore')
@observer
class Node extends Component {
    constructor() {
        super()
        this.state = {
            myChildren: [],
            myPath: 'root',
            isChildrenShow: false
        };
    }

    componentDidMount() {
        this.mounted = true
    }

    componentWillUnmount(){
        this.mounted = false
    }

    handleOnNodeClick = (e, node) => {

        e.stopPropagation()

        const {dataStore, history} = this.props

        if(node.type === 1){
            dataStore.setMainPicture(node.url)
            history.push('/picture')
        }

        dataStore.getNodeChildren(this.props.myPath)
            .then((res) => {
                if(this.mounted){
                    this.setState({myChildren: res ? res.data.data.children : []})
                }
            })

    }


    render() {
        const {dataStore, node, history} = this.props
        return (
            <ul>
                <li onClick={(e) => this.handleOnNodeClick(e, node)}
                    className={this.props.node.type === 0 ? 'folder-node' : 'picture-node'}>
                    <h4 className="pointer" onClick={this.handleOnClick}>{node.label}</h4>

                    {
                        this.state.myChildren && this.state.myChildren.map(child => (
                            <div key={child.label}>
                                <Node
                                    myPath={this.props.myPath + '/' + child.label}
                                    node={child}
                                    dataStore={dataStore}
                                    history={history}
                                />
                            </div>
                        ))
                    }

                </li>
            </ul>
        );
    }
}

@withRouter
@inject('dataStore')
@observer
class RootNode extends Component {
    state = {
        myChildren: [],
    };

    handleOnClick = () => {
        const {dataStore} = this.props
        dataStore.getNodeChildren('root')
            .then((res) => this.setState({myChildren: res ? res.data.data.children : []}))
    }

    render() {
        const {dataStore, history} = this.props
        return (
            <div>
                <div className="root-folder" onClick={this.handleOnClick}/>

                {
                    this.state.myChildren && this.state.myChildren.map(child => (
                        <div key={child.label}>
                            <Node
                                myPath={'root/' + child.label}
                                node={child}
                                dataStore={dataStore}
                                history={history}
                            />
                        </div>
                    ))
                }
            </div>

        )
    }
}


export default RootNode;

这是商店:

import {observable, action, runInAction} from "mobx";
import axios from 'axios'
import Data from "../models/Data";

export default class DataStore {

    @observable url = 'http://dvns.me/yaniv/clientest/public/explorePictures'
    @observable selectedNode = null
    @observable myChildren = []
    @observable loadChildrenError = null
    @observable pictures = []
    @observable mainPicture = null


    _config = {
        headers: {
            'X-TOKEN': '2d4e69f4823176197ccf41caa5ee6456',
        }
    }

    @action
    setMainPicture = (pic) =>{
        this.mainPicture = pic
    }

    @action
    async getNodeChildren(link) {
        console.log('here, sending ajax to: ', link)
        try {
            const response = await axios(`${this.url}?path=${link}`, this._config)
            runInAction(() => this.selectedNode = Data.reconstituteFrom(response.data.data))
            runInAction(() => this.getPicturesFromNode(this.selectedNode))
            return response
        }
        catch (e) {
            console.log('Error! ', e.message)
            runInAction(() => this.loadChildrenError = e.message);
        }
        finally {
            // console.log('Selected Node From store ', this.selectedNode)
            console.log('Pics From store ', this.pictures)
    }
    }

    @action.bound
    getPicturesFromNode = (node) => {
        node.children.filter(child => {
            if(child.type === 1){
                this.pictures.push(child.url)
            }
        })

    }
}

项目存储库:https://github.com/RomanDrevo/devsense

0 个答案:

没有答案