element树形图el-tree的懒加载和分页加载

假如我们的树形图含有大量的数据,一次全部加载出来会很慢,用户体验不好,所以我们要做数据的分级加载,也就是懒加载,但如果每级的数据量也很大,就需要每一级都做分页加载了。

懒加载

数据懒加载elementUI给出了方法,这里不再赘述,链接如下:https://element.eleme.cn/#/zh-CN/component/tree

分页加载+懒加载

如果我们既需要数据的懒加载,也需要数据的分页加载,那么我们便不能使用elementUI提供的懒加载方法了,因为我们没有办法在分页加载时拿到loadNode函数的所需要的参数noderesolve

先来说下我们的需求:

  • 数据分级加载
  • 每级数据分页加载
  • 点击树形图的每一项时,该项高亮,并派发事件
  • 分页加载的触发方式为在固定高度的区域上拉加载更多
  • 在别的模块显示树形图点击层级,eg:XXX总公司/北京分公司/研发部/前端
  • 默认展开树形图的第一项

看下我们的页面:

    <div class="tree" @scroll="rollingLoad">        //绑定树形图上拉加载更多数据
        <div class="tree-wrapper" v-loading="loading">        //加载数据时loading状态
            <el-tree
                class="el-tree"
                :data="departmentData"        //树形图的所有数据
                node-key="id"
                ref="department"
                highlight-current     
                lazy      //开启懒加载
                :props="departmentDefaultProps"        //配置选项
                :default-expanded-keys="defaultExpanded"       //默认选中
                @node-click="departmentNodeClick"        //节点被点击时的回调
                @node-expand="departmentNodeExpand"        //节点被展开时触发的事件
                @node-collapse="departmentNodeClose"        //  节点被关闭时触发的事件
                empty-text="暂无数据~">
                <span class="custom-tree-node" slot-scope="{ node, data }">
                    <span>
                        <i :class="data.icons" class="iconfont"></i>
                        <span
                            class="peopel-title"
                            :class="{ 'checked-color' : data.isChecked }">
                            {{ data.name }}
                        </span>
                        <span
                            class="peopel-dept-name"
                            :class="{ 'checked-color' : data.isChecked }">
                            ({{ data.user_count }})
                        </span>
                    </span>
                </span>
            </el-tree>
        </div>
    </div>
上拉加载更多的实现
/**
 * 滚动加载
 * @return {Boolean} 滚动加载开关
 */
scrollLoad(event) {
    let target = event.target;
    const offsetHeight = target.offsetHeight;
    const scrollTop = target.scrollTop;
    const scrollHeight = target.scrollHeight;
    if ((offsetHeight + scrollTop) - scrollHeight >= -100) {
        return true;
    } else {
        return false;
    }
},
/**
 * 判断当前tab是否需要滚动加载
 * @param {String} name 那个地方触发了滚动加载
 */
rollingLoad(event) {
    clearTimeout(this.scrollTimer)
    this.scrollTimer=setTimeout(()=>{
      if (this.scrollLoad(event) && this.scrollLoadingFlag) {  //scrollLoadingFlag为true说明可以进行下一次请求
          this.checkedData.page += 1;      //下一次请求时,pn++
          this.loadData(this.checkedData.id, this.checkedData, this.checkedData.page)        //请求函数
      }
  },300)
},
加载数据

首先分析下我们需要在加载数据的时候有哪些情况:

  • 第一次加载最外层数据,此时需要一个初始id,这个id我们是知道的,在我的项目里为-1,需要当前层级pn为0,还需要默认选中第一项,并展开(展开意味着加载它的下一级数据)
  • 当前层级数据的分页加载,此时需要当前数据的pn值和id,还需要当前层级的数据数组,以便在我们请求回来数据后进行拼装
  • 请求下一层数据,此时需要当前层级的id值,pn为0,还需要当前点击的item的值,以便在我们请求会回数据后,把数据设置为item的children属性,去渲染下一层数据

综上,我们的每一次请求都需要一个id值,一个pn值,最好还有一个当前点击的item。所以loadData()函数接收3个形参,即loadData(id,data.pn)

我们可以在element文档中发现,当节点被展开和点击时,我们都可以拿到当前节点的数据,当前节点的 Node 对象。


image.png

所以我们可以使用如下方法去加载数据:

loadData(parent_id = -1, data = {children: []}, pn = 0){
    this.loading=true;    //设置loading状态
     //如果pn=0,data.children的长度大于0 ,说明该级数据已经加载过,应该是树形被合上又展开的操作
    //正常加载数据时,pn==0的同时data.children==0说明第一次加载该级数据
    //pn!=0,data.children.length>0,分页加载该级数据
    if (pn === 0 && data.children.length && data.children.length > 0) {
        return; 
    }
    this.scrollLoadingFlag=false;            //加载开关关闭
    let params = {
        parent_id: parent_id,
        pn: pn,
        pl: this.pl,
        type: this.type
    };
    this.$axios
        .get(
            `XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX`,
            { params: params }
        )
        .then(res => {
            this.scrollLoadingFlag=true;              //加载开关打开
            this.loading=false;                              
            let { list } = res.data||[];
            // 重置滚动加载事件开关
            if (list.length < this.pl) {
                this.scrollLoadingFlag = false;      //如果获取到的数据小于pl,数据加载完,关闭开关
            }
            //初始化右侧数据
            list.map((v, i) => {
                if (v.subordinate_count > 1) {        //,subordinate_count 大于1说明有下一级
                    v["isLeaf"] = false;
                    v["children"] = [];
                }
            })
            if(parent_id&&parent_id===-1){      //如果是第一级数据
                list.map(v=>{
                    v["icons"] = "el-icon-max-zuzhijiagou-";
                })
                list[0]["isChecked"] = true;        //默认选中第一级数据第一个
                this.defaultExpanded=[list[0].id];        //默认展开第一级数据第一个
                this.checkList.push(list[0]);        //缓存点击层级顺序
                this.checkedData=list[0];           //缓存点击层级顺序
                this.loadData(list[0].id, list[0], list[0].page, list[0]);        //加载默认展开的下一级数据
            }
            // 判断当前节点是要添加子节点还是滚动加载
            if (this.departmentData.length === 0 && pn === 0) {      //如果树形图数据长度为0并且pn=0,说明第一次加载数据,直接赋值
                this.departmentData = list;
            } else if (data.children && data.children.length === 0) {        //如果当前点击项children长度等于0,说明第一次加载下一级
                this.$set(data, "children", list)
            } else {
                data.children = data.children.concat(list)        //其他情况为分页加载,拼接数据
            }
        })
        .catch(err => {
            this.loading=false;
            this.scrollLoadingFlag=true;
            this.$message.closeAll();
            this.$message.error("请求超时");
        });
},
节点展开
/**
 * 节点被展开
 * @param {Object} e tree的node节点数据
 * @param {Object} node tree的node的props数据
 */
departmentNodeExpand(e, node) {
    console.log(e, '节点展开')
    this.checkedData = e;
    if (e.isLeaf && node.isLeaf||(e.children&&e.children.length>0)) return;
    this.loadData(e.id, e, e.page);
},
节点点击
/**
 * 节点被点击
 * @param {Object} e tree的node节点数据
 */
departmentNodeClick(e, node, a) {
    this.handleGray(this.checkList);        //将被点击列表取消高亮
    this.$store.dispatch("set_getRequestUser", Object.assign({}, {    //往外派发点击数据
        id:e.id,
        name:e.name
    }));
    this.checkedData = e;      //设置checkedData 
    this.parent_id = e.id;
    this.checkList = [];
    this.handleBlue(node);        //将选中项高亮
},
其他方法
handleBlue(node) {
    if (node.parent) {
        this.checkList.push(node.data);
        this.handleBlue(node.parent);
    } else {
        this.checkList.reverse();
        this.checkList.map(v => {
            v["isChecked"] = true;
        });
        this.$store.dispatch("set_getRequestUserAll",JSON.parse(JSON.stringify(this.checkList)));
        return;
    }
},
handleGray(arr){
    arr.map(v=>{
        v['isChecked']=false;
    })
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,377评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,390评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,967评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,344评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,441评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,492评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,497评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,274评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,732评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,008评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,184评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,837评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,520评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,156评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,407评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,056评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,074评论 2 352

推荐阅读更多精彩内容