【JavaScript】bootstrap-table-treegrid 异步加载实现

长孙沈义
2023-12-01

插件修改自 jquery.treegrid.extension.js,新增按需加载子节点的功能(即 lazyload: true 模式),需配合 TreeGrid plugin for jQuery 使用

改造思路如下:
1 > 首次只加载根节点
2 > 在节点上绑定点击事件,查询后台数据,动态加载子节点(插入到点击行的后面紧邻的位置,同样也绑定点击事件)
3 > 重新渲染节点图标,并缓存展开节点至cookie(treegrid save state 功能,依赖 jquery.cookie.js)

; (function($) {
    "use strict";

    $.fn.bootstrapTreeTable = function(options, param) {
        // 如果是调用方法
        if (typeof options == 'string') {
            return $.fn.bootstrapTreeTable.methods[options](this, param);
        }

        // 如果是初始化组件
        options = $.extend({}, $.fn.bootstrapTreeTable.defaults, options || {});
        // 是否有radio或checkbox
        var hasSelectItem = false;
        var target = $(this);

        // 在外层包装一下div,样式用的bootstrap-table的
        var _main_div = target.parent('.fixed-table-container');
        if (_main_div.length == 0) {
            _main_div = $("<div class='fixed-table-container'></div>");
            target.before(_main_div);
            _main_div.append(target);
        }
        target.addClass("table table-hover treegrid-table");
        if (options.bordered) {
            target.addClass('table-bordered');
        }
        if (options.striped) {
            target.addClass('table-striped');
        }
        // 工具条在外层包装一下div,样式用的bootstrap-table的
        if (options.toolbar) {
            var _tool_div = $("<div class='fixed-table-toolbar'></div>");
            var _tool_left_div = $("<div class='bs-bars pull-left'></div>");
            var _tool_bar = $(options.toolbar);
            if (_tool_bar.length == 0) {
                _tool_div.css('display', 'none');
            } else {
                _tool_left_div.append(_tool_bar);
            }
            _tool_div.append(_tool_left_div);
            _main_div.before(_tool_div);
        }

        // 得到根节点
        target.getRootNodes = function(data) {
            // 指定Root节点值
            var _root = options.rootCodeValue ? options.rootCodeValue: null;
            var result = [];
            $.each(data, function(index, item) {
                // 这里兼容几种常见Root节点写法
                // 默认的几种判断
                var _defaultRootFlag = item[options.parentCode] == '0' 
                                    || item[options.parentCode] == 0 
                                    || item[options.parentCode] == null 
                                    || item[options.parentCode] == '';
                if (!item[options.parentCode] || (_root ? (item[options.parentCode] == options.rootCodeValue) : _defaultRootFlag)) {
                    result.push(item);
                }
                // 添加一个默认属性,用来判断当前节点有没有被显示
                item.isShow = false;
            });
            return result;
        };

        var j = 0;
        // 获取子节点, 并且设置子节点
        target.getChildNodes = function(data, parentNode, parentIndex, tbody, isRecursive) {
            $.each(data, function(i, item) {
                if (item[options.parentCode] == parentNode[options.code]) {
                    var tr = $('<tr></tr>');
                    // var nowParentIndex = (parentIndex + (j++) + 1);
                    var nowParentIndex = item[options.code];
                    tr.addClass('treegrid-' + nowParentIndex);
                    tr.addClass('treegrid-parent-' + parentIndex);
                    target.renderRow(tr, item, nowParentIndex);
                    item.isShow = true;
                    tbody.append(tr);
                    if (typeof isRecursive == 'boolean' && isRecursive) {
                        target.getChildNodes(data, item, nowParentIndex, tbody, isRecursive);
                    }
                }
            });
        };

        // 绘制行
        target.renderRow = function(tr, item, parentIndex) {
            if (options.lazyload) {
                tr.attr('data-id', parentIndex);
            }
            $.each(options.columns, function(index, column) {
                // 判断有没有选择列
                if (index == 0 && column.field == 'selectItem') {
                    hasSelectItem = true;
                    var td = $('<td style="text-align:center;width:36px"></td>');
                    if (column.radio) {
                        var _ipt = $('<input name="select_item" type="radio" value="' + item[options.id] + '"></input>');
                        td.append(_ipt);
                    }
                    if (column.checkbox) {
                        var _ipt = $('<input name="select_item" type="checkbox" value="' + item[options.id] + '"></input>');
                        td.append(_ipt);
                    }
                    tr.append(td);
                } else {
                    var td = $('<td style="' + ((column.width) ? ('width:' + column.width) : '') + '"></td>');
                    // 增加formatter渲染
                    if (column.formatter) {
                        td.html(column.formatter.call(this, item, index));
                    } else {
                        td.text(item[column.field]);
                    }
                    tr.append(td);
                }
            });
        };

        // 加载数据
        target.load = function(params) {
            // 加载数据前先清空
            target.html("");
            // 构造表头
            var thr = $('<tr></tr>');
            $.each(options.columns, function(i, item) {
                var th = null;
                // 判断有没有选择列
                if (i == 0 && item.field == 'selectItem') {
                    hasSelectItem = true;
                    th = $('<th style="text-align:' + item.valign + ';width:36px"></th>');
                } else {
                    th = $('<th style="text-align:' + item.valign + ';padding:10px;' + ((item.width) ? ('width:' + item.width) : '') + '"></th>');
                }
                th.text(item.title);
                thr.append(th);
            });
            var thead = $('<thead class="treegrid-thead"></thead>');
            thead.append(thr);
            target.append(thead);
            // 构造表体
            var tbody = $('<tbody class="treegrid-tbody"></tbody>');
            target.append(tbody);
            // 添加加载loading
            var _loading = '<tr><td colspan="' + options.columns.length + '"><div style="display: block;text-align: center;">正在努力地加载数据中,请稍候……</div></td></tr>';
            tbody.html(_loading);
            // 默认高度
            if (options.height) {
                tbody.css("height", options.height);
            }
            $.ajax({
                type: options.type,
                url: options.url,
                data: params ? params: options.ajaxParams,
                dataType: "JSON",
                success: function(data, textStatus, jqXHR) {
                    // 加载完数据先清空
                    tbody.html("");
                    if (!data || data.length <= 0) {
                        var _empty = '<tr><td colspan="' + options.columns.length + '"><div style="display: block;text-align: center;">没有记录</div></td></tr>';
                        tbody.html(_empty);
                        return;
                    }
                    var rootNode = target.getRootNodes(data);
                    $.each(rootNode, function(i, item) {
                        var tr = $('<tr></tr>');
                        // tr.addClass('treegrid-' + (j + "_" + i));
                        tr.addClass('treegrid-' + item[options.code]);
                        // target.renderRow(tr, item, (j + "_" + i));
                        target.renderRow(tr, item, item[options.code]);
                        item.isShow = true;
                        tbody.append(tr);
                        if (!options.lazyload) {
                            // target.getChildNodes(data, item, (j + "_" + i), tbody, true);
                            target.getChildNodes(data, item, item[options.code], tbody, true);
                        }
                    });
                    // 下边的操作主要是为了查询时让一些没有根节点的节点显示
                    $.each(data, function(i, item) {
                        if (!item.isShow) {
                            var tr = $('<tr></tr>');
                            // tr.addClass('treegrid-' + (j + "_" + i));
                            tr.addClass('treegrid-' + item[options.code]);
                            target.renderRow(tr, item);
                            tbody.append(tr);
                        }
                    });
                    
                    target.append(tbody);
                    // 初始化treegrid
                    target.treegrid({
                        // 如果有radio或checkbox默认第二列层级显示,当前是在用户未设置的提前下
                        'treeColumn': options.expandColumn ? options.expandColumn: (hasSelectItem ? 1 : 0),
                        'expanderExpandedClass': options.expanderExpandedClass,
                        'expanderCollapsedClass': options.expanderCollapsedClass,
                        'initialState': options.expandAll ? 'expanded': 'collapsed',
                        'saveState': options.lazyload
                    });
                    if (!options.expandAll && !options.lazyload) {
                        target.treegrid('collapseAll');
                    }

                    target.find('tbody').on('click', 'tr', function(e) {
                        var e = e || window.event;
                        var _self = $(this);
                        // 行点击选中
                        if (hasSelectItem) {
                            var _ipt = _self.find("input[name='select_item']");
                            if (_ipt.attr("type") == "radio") {
                                _ipt.prop('checked', true);
                                target.find('tr.treegrid-selected').removeClass("treegrid-selected");
                                _self.addClass("treegrid-selected");
                            } else {
                                if (_ipt.prop('checked')) {
                                    _ipt.prop('checked', false);
                                    _self.removeClass("treegrid-selected");
                                } else {
                                    _ipt.prop('checked', true);
                                    _self.addClass("treegrid-selected");
                                }
                            }
                        }
                        // 子节点加载
                        if (options.lazyload) {
                            if (_self.attr('data-loaded') != 'true') {
                                if ($(e.target).hasClass('treegrid-expander')) {
                                    var _params = {};
                                    // _params[options.parentCode] = _self.attr('class').split(" ")[0].split("-")[1];
                                    _params[options.parentCode] = _self.attr('data-id');
                                    target.loadChilds(_params);
                                }
                            }
                        }
                    });

                    if (options.lazyload) {
                        target.repainExpends();
                    }
                },
                error: function(xhr, status, error) {
                    var _errorMsg = '<tr><td colspan="' + options.columns.length + '"><div style="display: block;text-align: center;">' + xhr.responseText + '</div></td></tr>';
                    tbody.html(_errorMsg);
                    debugger;
                },
            });
        }

        // 重新绘制节点图标
        target.repainExpends = function() {
            var trExpends = target.find('tbody').find('tr');
            $.each(trExpends,
            function(index, item) {
                var _item = $(item);
                if (_item.attr('data-isleaf') != 'true') {
                    if (_item.attr('data-loaded') != 'true') {
                        _item.find("span.treegrid-expander").addClass(options.expanderCollapsedClass);
                    }
                } else {
                    _item.find("span.treegrid-expander").removeClass(options.expanderCollapsedClass);
                }
            });
        }

        // 动态添加子节点
        target.renderChildRows = function(data, parentNode, parentIndex, tbody) {
            for (var x = data.length - 1; x >= 0; x--) {
                var item = data[x];
                if (item[options.parentCode] == parentNode[options.code]) {
                    var tr = $('<tr></tr>');
                    // var nowParentIndex = (parentIndex + x);
                    var nowParentIndex = item[options.code];
                    tr.addClass('treegrid-' + nowParentIndex);
                    tr.addClass('treegrid-parent-' + parentIndex);
                    target.renderRow(tr, item, nowParentIndex);
                    item.isShow = true;
                    tbody.after(tr);
                }
            }
        };

        // 加载子节点数据
        target.loadChilds = function(params) {
            $.ajax({
                type: options.type,
                url: options.url,
                data: params ? params: options.ajaxParams,
                dataType: "JSON",
                success: function(data, textStatus, jqXHR) {
                    var _tr = target.find(".treegrid-" + params[options.parentCode]);
                    _tr.attr('data-loaded', 'true');
                    if (data == null || data.length == 0) {
                        _tr.attr('data-isleaf', 'true');
                    } else {
                        var _pid = {};
                        _pid[options.code] = params[options.parentCode];
                        target.renderChildRows(data, _pid, params[options.parentCode], _tr);
                        target.treegrid({
                            // 如果有radio或checkbox默认第二列层级显示,当前是在用户未设置的提前下
                            'treeColumn': options.expandColumn ? options.expandColumn: (hasSelectItem ? 1 : 0),
                            'expanderExpandedClass': options.expanderExpandedClass,
                            'expanderCollapsedClass': options.expanderCollapsedClass,
                            'initialState': options.expandAll ? 'expanded': 'collapsed',
                            'saveState': options.lazyload
                        });
                        // 展开节点
                        _tr.treegrid('expand');
                    }
                    target.repainExpends();
                },
                error: function(xhr, status, error) {
                    var _errorMsg = '<tr><td colspan="' + options.columns.length + '"><div style="display: block;text-align: center;">' + xhr.responseText + '</div></td></tr>';
                    tbody.html(_errorMsg);
                    debugger;
                }
            });
        }

        if (options.url) {
            target.load();
        } else {
            // 也可以通过defaults里面的data属性通过传递一个数据集合进来对组件进行初始化....有兴趣可以自己实现,思路和上述类似
        }

        return target;
    };

    // 组件方法封装........
    $.fn.bootstrapTreeTable.methods = {
        // 返回选中记录的id(返回的id由配置中的id属性指定)
        // 为了兼容bootstrap-table的写法,统一返回数组,这里只返回了指定的id
        getSelections: function(target, data) {
            // 所有被选中的记录input
            var _ipt = target.find("tbody").find("tr").find("input[name='select_item']:checked");
            var chk_value = [];
            // 如果是radio
            if (_ipt.attr("type") == "radio") {
                chk_value.push({
                    id: _ipt.val()
                });
            } else {
                _ipt.each(function(_i, _item) {
                    chk_value.push({
                        id: $(_item).val()
                    });
                });
            }
            return chk_value;
        },
        // 刷新记录
        refresh: function(target, params) {
            if (params) {
                target.load(params);
            } else {
                target.load();
            }
        },
        // 重置表格视图
        resetView: function(target, params) {
            if (params) {
                target.find("tbody").css("height", params);
            } else {
                target.find("tbody").css("height", options.height);
            }
        }
        // 组件的其他方法也可以进行类似封装........
    };

    $.fn.bootstrapTreeTable.defaults = {
        id: 'menuId', // 选取记录返回的值
        code: 'menuId', // 用于设置父子关系
        parentCode: 'parentId', // 用于设置父子关系
        rootCodeValue: null, //设置根节点code值----可指定根节点,默认为null,"",0,"0"
        data: [], // 构造table的数据集合
        type: "GET", // 请求数据的ajax类型
        url: null, // 请求数据的ajax的url
        ajaxParams: {}, // 请求数据的ajax的data属性
        expandColumn: null, // 在哪一列上面显示展开按钮
        expandAll: false, // 是否全部展开
        striped: false, // 是否各行渐变色
        bordered: false, //是否显示边框
        lazyload: false, //是否延迟加载
        columns: [],
        toolbar: null, //顶部工具条
        height: 0,
        expanderExpandedClass: 'glyphicon glyphicon-chevron-down', // 展开的按钮的图标
        expanderCollapsedClass: 'glyphicon glyphicon-chevron-right' // 缩起的按钮的图标
    };
})(jQuery);
; (function($) {
    "use strict";

    var TreeTable = function(tableId, url, columns) {
        this.btInstance = null; // jquery和bootstrapTreeTable绑定的对象
        this.bstableId = tableId;
        this.url = url;
        this.method = "GET";
        this.columns = columns;
        this.data = {}; // ajax的参数
        this.expandColumn = null; // 展开显示的列 
        this.id = 'menuId'; // 选取记录返回的值
        this.code = 'menuId'; // 用于设置父子关系
        this.parentCode = 'parentId'; // 用于设置父子关系
        this.expandAll = false; // 是否默认全部展开
        this.lazyload = false; // 是否延迟加载
        this.striped = true; // 是否各行渐变色
        this.bordered = true; // 是否显示边框
        this.toolbarId = null;
        this.height = 440;
    };

    TreeTable.prototype = {
        /**
         * 初始化bootstrap table
         */
        init: function() {
            var tableId = this.bstableId;
            this.btInstance = $('#' + tableId).bootstrapTreeTable({
                id: this.id, // 选取记录返回的值
                code: this.code, // 用于设置父子关系
                parentCode: this.parentCode, // 用于设置父子关系
                rootCodeValue: this.rootCodeValue, //设置根节点code值----可指定根节点,默认为null,"",0,"0"
                type: this.method, //请求数据的ajax类型
                url: this.url, //请求数据的ajax的url
                ajaxParams: this.data, //请求数据的ajax的data属性
                expandColumn: this.expandColumn, //在哪一列上面显示展开按钮,从0开始
                striped: this.striped, //是否各行渐变色
                bordered: this.bordered, //是否显示边框
                expandAll: this.expandAll, //是否全部展开
                lazyload: this.lazyload, //是否延迟加载
                columns: this.columns, //列数组
                toolbar: this.toolbarId ? ('#' + this.toolbarId) : null, //顶部工具条
                height: this.height
            });
            return this;
        },
        /**
         * 设置顶部工具条
         */
        setToolbarId: function(toolbarId) {
            this.toolbar = toolbarId;
        },
        /**
         * 设置在哪一列上面显示展开按钮,从0开始
         */
        setExpandColumn: function(expandColumn) {
            this.expandColumn = expandColumn;
        },
        /**
         * 设置记录返回的id值
         */
        setIdField: function(id) {
            this.id = id;
        },
        /**
         * 设置记录分级的字段
         */
        setCodeField: function(code) {
            this.code = code;
        },
        /**
         * 设置记录分级的父级字段
         */
        setParentCodeField: function(parentCode) {
            this.parentCode = parentCode;
        },
        /**
         * 设置根节点code值----可指定根节点,默认为null,"",0,"0"
         */
        setRootCodeValue: function(rootCodeValue) {
            this.rootCodeValue = rootCodeValue;
        },
        /**
         * 设置是否默认全部展开
         */
        setExpandAll: function(expandAll) {
            this.expandAll = expandAll;
        },
        /**
         * 设置是否延迟加载
         */
        setLazyload: function(lazyload) {
            this.lazyload = lazyload;
        },
        /**
         * 设置表格高度
         */
        setHeight: function(height) {
            this.height = height;
        },
        /**
         * 设置ajax post请求时候附带的参数
         */
        set: function(key, value) {
            if (typeof key == "object") {
                for (var i in key) {
                    if (typeof i == "function") {
                        continue;
                    }
                    this.data[i] = key[i];
                }
            } else {
                this.data[key] = (typeof value == "undefined") ? $("#" + key).val() : value;
            }
            return this;
        },
        /**
         * 设置ajax get请求时候附带的参数
         */
        setData: function(data) {
            this.data = data;
            return this;
        },
        /**
         * 清空ajax post请求参数
         */
        clear: function() {
            this.data = {};
            return this;
        },
        /**
         * 刷新表格
         */
        refresh: function(params) {
            if (typeof params != "undefined") {
                this.btInstance.bootstrapTreeTable('refresh', params.query); // 为了兼容bootstrap-table的写法
            } else {
                this.btInstance.bootstrapTreeTable('refresh');
            }
        },
        /**
         * 重置表格视图
         */
        resetView: function(params) {
            if (typeof params != "undefined") {
                this.btInstance.bootstrapTreeTable('resetView', params.height); // 为了兼容bootstrap-table的写法
            }
        }
    };

    window.TreeTable = TreeTable;

})(jQuery);
 类似资料: