| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571 | 
							- 'use strict';
 
- /**
 
-  *
 
-  *
 
-  * @author Mai
 
-  * @date
 
-  * @version
 
-  */
 
- const itemsPre = 'id_';
 
- class baseTree {
 
-     /**
 
-      * 构造函数
 
-      */
 
-     constructor (ctx, setting) {
 
-         this.ctx = ctx;
 
-         // 无索引
 
-         this.datas = [];
 
-         // 以key为索引
 
-         this.items = {};
 
-         // 以排序为索引
 
-         this.nodes = [];
 
-         // 根节点
 
-         this.children = [];
 
-         // 树设置
 
-         this.setting = setting;
 
-     }
 
-     /**
 
-      * 根据id获取树结构节点数据
 
-      * @param {Number} id
 
-      * @returns {Object}
 
-      */
 
-     getItems (id) {
 
-         return this.items[itemsPre + id];
 
-     };
 
-     /**
 
-      * 查找node的parent
 
-      * @param {Object} node
 
-      * @returns {Object}
 
-      */
 
-     getParent (node) {
 
-         return this.getItems(node[this.setting.pid]);
 
-     };
 
-     /**
 
-      * 查询node的已下载子节点
 
-      * @param {Object} node
 
-      * @returns {Array}
 
-      */
 
-     getChildren (node) {
 
-         const setting = this.setting;
 
-         const pid = node ? node[setting.id] : setting.rootId;
 
-         const children = this.datas.filter(function (x) {
 
-             return x[setting.pid] === pid;
 
-         });
 
-         children.sort(function (a, b) {
 
-             return a.order - b.order;
 
-         });
 
-         return children;
 
-     };
 
-     /**
 
-      * 获取节点的 index
 
-      * @param node
 
-      * @returns {number}
 
-      */
 
-     getNodeSerialNo(node) {
 
-         return this.nodes.indexOf(node);
 
-     }
 
-     /**
 
-      * 树结构根据显示排序
 
-      */
 
-     sortTreeNode (isResort) {
 
-         const self = this;
 
-         const addSortNodes = function (nodes) {
 
-             if (!nodes) { return }
 
-             for (let i = 0; i < nodes.length; i++) {
 
-                 self.nodes.push(nodes[i]);
 
-                 nodes[i].index = self.nodes.length - 1;
 
-                 if (!isResort) {
 
-                     nodes[i].children = self.getChildren(nodes[i]);
 
-                 } else {
 
-                     nodes[i].children.sort(function (a, b) {
 
-                         return a.order - b.order;
 
-                     })
 
-                 }
 
-                 addSortNodes(nodes[i].children);
 
-             }
 
-         };
 
-         this.nodes = [];
 
-         if (!isResort) {
 
-             this.children = this.getChildren();
 
-         } else {
 
-             this.children.sort(function (a, b) {
 
-                 return a.order - b.order;
 
-             })
 
-         }
 
-         addSortNodes(this.children);
 
-     }
 
-     /**
 
-      * 加载数据(初始化), 并给数据添加部分树结构必须数据
 
-      * @param datas
 
-      */
 
-     loadDatas (datas) {
 
-         // 清空旧数据
 
-         this.items = {};
 
-         this.nodes = [];
 
-         this.datas = [];
 
-         this.children = [];
 
-         // 加载全部数据
 
-         datas.sort(function (a, b) {
 
-             return a.level - b.level;
 
-         });
 
-         for (const data of datas) {
 
-             const keyName = itemsPre + data[this.setting.id];
 
-             if (!this.items[keyName]) {
 
-                 const item = JSON.parse(JSON.stringify(data));
 
-                 item.children = [];
 
-                 item.expanded = true;
 
-                 item.visible = true;
 
-                 this.items[keyName] = item;
 
-                 this.datas.push(item);
 
-                 if (item[this.setting.pid] === this.setting.rootId) {
 
-                     this.children.push(item);
 
-                 } else {
 
-                     const parent = this.getParent(item);
 
-                     if (parent) {
 
-                         parent.children.push(item);
 
-                     }
 
-                 }
 
-             }
 
-         }
 
-         this.children.sort(function (a, b) {
 
-             return a.order - b.order;
 
-         });
 
-         this.sortTreeNode(true);
 
-     }
 
-     /**
 
-      * 递归方式 查询node的已下载的全部后代 (兼容full_path不存在的情况)
 
-      * @param node
 
-      * @returns {*}
 
-      * @private
 
-      */
 
-     _recursiveGetPosterity (node) {
 
-         let posterity = node.children;
 
-         for (const c of node.children) {
 
-             posterity = posterity.concat(this._recursiveGetPosterity(c));
 
-         }
 
-         return posterity;
 
-     };
 
-     /**
 
-      * 查询node的已下载的全部后代
 
-      * @param {Object} node
 
-      * @returns {Array}
 
-      */
 
-     getPosterity (node) {
 
-         const self = this;
 
-         let posterity;
 
-         if (node.full_path !== '') {
 
-             const reg = new RegExp('^' + node.full_path + '-');
 
-             posterity = this.datas.filter(function (x) {
 
-                 return reg.test(x.full_path);
 
-             });
 
-         } else {
 
-             posterity = this._recursiveGetPosterity(node);
 
-         }
 
-         posterity.sort(function (x, y) {
 
-             return self.getNodeSerialNo(x) - self.getNodeSerialNo(y);
 
-         });
 
-         return posterity;
 
-     };
 
-     /**
 
-      * 根据 字段名称 获取数据
 
-      * @param fields
 
-      * @returns {Array}
 
-      */
 
-     getDatas (fields) {
 
-         const datas = [];
 
-         for (const node of this.nodes) {
 
-             if (node.b_code && node.b_code !== '') node.chapter = this.ctx.helper.getChapterCode(node.b_code);
 
-             node.is_leaf = !node.children || node.children.length === 0;
 
-             const data = {};
 
-             for (const field of fields) {
 
-                 data[field] = node[field];
 
-             }
 
-             datas.push(data);
 
-         }
 
-         return datas;
 
-     }
 
-     /**
 
-      * 排除 某些字段 获取数据
 
-      * @param fields
 
-      * @returns {Array}
 
-      */
 
-     getDatasWithout (fields, filter) {
 
-         const datas = [];
 
-         for (const node of this.nodes) {
 
-             if (filter && filter(node)) {
 
-                 continue;
 
-             }
 
-             if (node.b_code && node.b_code !== '') node.chapter = this.ctx.helper.getChapterCode(node.b_code);
 
-             node.is_leaf = !node.children || node.children.length === 0;
 
-             const data = {};
 
-             for (const field in node) {
 
-                 if (fields.indexOf(field) === -1) {
 
-                     data[field] = node[field];
 
-                 }
 
-             }
 
-             datas.push(data);
 
-         }
 
-         return datas;
 
-     }
 
-     /**
 
-      * 获取默认数据 剔除一些树结构需要的缓存数据
 
-      * @returns {Array}
 
-      */
 
-     getDefaultDatas(filter) {
 
-         return this.getDatasWithout(['expanded', 'visible', 'children', 'index'], filter);
 
-     }
 
-     _mapTreeNode () {
 
-         let map = {}, maxLevel = 0;
 
-         for (const node of this.nodes) {
 
-             let levelArr = map[node.level];
 
-             if (!levelArr) {
 
-                 levelArr = [];
 
-                 map[node.level] = levelArr;
 
-             }
 
-             if (node.level > maxLevel) {
 
-                 maxLevel = node.level;
 
-             }
 
-             levelArr.push(node);
 
-         }
 
-         return [maxLevel, map];
 
-     }
 
-     _calculateNode (node, fun) {
 
-         const self = this;
 
-         if (node.children && node.children.length > 0) {
 
-             const gather = node.children.reduce(function (rst, x) {
 
-                 const result = {};
 
-                 for (const cf of self.setting.calcFields) {
 
-                     result[cf] = self.ctx.helper.add(rst[cf], x[cf]);
 
-                 }
 
-                 return result;
 
-             });
 
-             // 汇总子项
 
-             for (const cf of this.setting.calcFields) {
 
-                 if (gather[cf]) {
 
-                     node[cf] = gather[cf];
 
-                 } else {
 
-                     node[cf] = null;
 
-                 }
 
-             }
 
-         }
 
-         // 自身运算
 
-         if (fun) {
 
-             fun(node);
 
-         } else if (this.setting.calc) {
 
-             this.setting.calc(node);
 
-         }
 
-     }
 
-     calculateAll(fun) {
 
-         const [maxLevel, levelMap] = this._mapTreeNode();
 
-         for (let i = maxLevel; i >= 0; i--) {
 
-             const levelNodes = levelMap[i];
 
-             if (levelNodes && levelNodes.length > 0) {
 
-                 for (const node of levelNodes) {
 
-                     this._calculateNode(node, fun);
 
-                 }
 
-             }
 
-         }
 
-     }
 
- }
 
- class billsTree extends baseTree {
 
-     /**
 
-      * 检查节点是否是最底层项目节
 
-      * @param node
 
-      * @returns {boolean}
 
-      */
 
-     isLeafXmj(node) {
 
-         if (node.b_code && node.b_code !== '') {
 
-             return false;
 
-         }
 
-         for (const child of node.children) {
 
-             if (!child.b_code || child.b_code === '') {
 
-                 return false;
 
-             }
 
-         }
 
-         return true;
 
-     }
 
-     /**
 
-      * 查询最底层项目节(本身或父项)
 
-      * @param {Object} node - 查询节点
 
-      * @returns {Object}
 
-      */
 
-     getLeafXmjParent(node) {
 
-         let parent = node;
 
-         while (parent) {
 
-             if (this.isLeafXmj(parent)) {
 
-                 return parent;
 
-             } else {
 
-                 parent = this.getParent(parent);
 
-             }
 
-         }
 
-         return null;
 
-     }
 
- }
 
- class filterTree extends baseTree {
 
-     addData(data, fields) {
 
-         const item = {};
 
-         for (const prop in data) {
 
-             if (fields.indexOf(prop) >= 0) {
 
-                 item[prop] = data[prop];
 
-             }
 
-         }
 
-         const keyName = itemsPre + item[this.setting.id];
 
-         if (!this.items[keyName]) {
 
-             item.children = [];
 
-             item.is_leaf = true;
 
-             item.expanded = true;
 
-             item.visible = true;
 
-             this.items[keyName] = item;
 
-             this.datas.push(item);
 
-             if (item[this.setting.pid] === this.setting.rootId) {
 
-                 this.children.push(item);
 
-             } else {
 
-                 const parent = this.getParent(item);
 
-                 if (parent) {
 
-                     parent.is_leaf = false;
 
-                     parent.children.push(item);
 
-                 }
 
-             }
 
-         } else {
 
-             return this.items[keyName];
 
-         }
 
-         return item;
 
-     }
 
- }
 
- class filterGatherTree extends baseTree {
 
-     clearDatas() {
 
-         this.items = {};
 
-         this.nodes = [];
 
-         this.datas = [];
 
-         this.children = [];
 
-     }
 
-     get newId() {
 
-         if (!this._maxId) {
 
-             this._maxId = 0;
 
-         }
 
-         this._maxId++;
 
-         return this._maxId;
 
-     }
 
-     addNode(data, parent) {
 
-         data[this.setting.pid] = parent ? parent[this.setting.id] : this.setting.rootId;
 
-         let item = this.ctx.helper._.find(this.items, data);
 
-         if (item) return item;
 
-         item = data;
 
-         item.drawing_code = [];
 
-         item.memo = [];
 
-         item.postil = [];
 
-         item[this.setting.id] = this.newId;
 
-         const keyName = itemsPre + item[this.setting.id];
 
-         item.children = [];
 
-         item.is_leaf = true;
 
-         item.expanded = true;
 
-         item.visible = true;
 
-         this.items[keyName] = item;
 
-         this.datas.push(item);
 
-         if (parent) {
 
-             item[this.setting.fullPath] = parent[this.setting.fullPath] + '-' + item[this.setting.id];
 
-             item[this.setting.level] = parent[this.setting.level] + 1;
 
-             item[this.setting.order] = parent.children.length + 1;
 
-             parent.is_leaf = false;
 
-             parent.children.push(item);
 
-         } else {
 
-             item[this.setting.fullPath] = '' + item[this.setting.id];
 
-             item[this.setting.level] = 1;
 
-             item[this.setting.order] = this.children.length + 1;
 
-             this.children.push(item);
 
-         }
 
-         return item;
 
-     }
 
-     generateSortNodes() {
 
-         const self = this;
 
-         const addSortNode = function (node) {
 
-             self.nodes.push(node);
 
-             for (const c of node.children) {
 
-                 addSortNode(c);
 
-             }
 
-         };
 
-         this.nodes = [];
 
-         for (const n of this.children) {
 
-             addSortNode(n);
 
-         }
 
-     }
 
-     sortTreeNodeCustom(fun) {
 
-         const sortNodes = function (nodes) {
 
-             nodes.sort(fun);
 
-             for (const [i, node] of nodes.entries()) {
 
-                 node.order = i + 1;
 
-             }
 
-             for (const node of nodes) {
 
-                 if (node.children && node.children.length > 1) {
 
-                     sortNodes(node.children);
 
-                 }
 
-             }
 
-         };
 
-         this.nodes = [];
 
-         this.children = this.getChildren(null);
 
-         sortNodes(this.children);
 
-         this.generateSortNodes();
 
-     }
 
- }
 
- class gatherTree extends baseTree {
 
-     constructor(ctx, setting) {
 
-         super(ctx, setting);
 
-         this._newId = 1;
 
-     }
 
-     get newId() {
 
-         return this._newId++;
 
-     }
 
-     loadGatherNode(node, parent, loadFun) {
 
-         const siblings = parent ? parent.children : this.children;
 
-         let cur = siblings.find(function (x) {
 
-             return node.b_code
 
-                 ? x.b_code === node.b_code && x.name === node.name && x.unit === node.unit && x.unit_price === node.unit_price
 
-                 : x.code === node.code && x.name === node.name;
 
-         });
 
-         if (!cur) {
 
-             const id = this.newId;
 
-             cur = {
 
-                 id: id,
 
-                 pid: parent ? parent.id : this.setting.rootId,
 
-                 full_path: parent ? parent.full_path + '-' + id : '' + id,
 
-                 level: parent ? parent.level + 1 : 1,
 
-                 order: siblings.length + 1,
 
-                 children: [],
 
-                 code: node.code, b_code: node.b_code, name: node.name,
 
-                 unit: node.unit, unit_price: node.unit_price,
 
-             };
 
-             siblings.push(cur);
 
-             this.datas.push(cur);
 
-         }
 
-         loadFun(cur, node);
 
-         for (const c of node.children) {
 
-             this.loadGatherNode(c, cur, loadFun);
 
-         }
 
-     }
 
-     generateSortNodes() {
 
-         const self = this;
 
-         const addSortNode = function (node) {
 
-             self.nodes.push(node);
 
-             for (const c of node.children) {
 
-                 addSortNode(c);
 
-             }
 
-         };
 
-         this.nodes = [];
 
-         for (const n of this.children) {
 
-             addSortNode(n);
 
-         }
 
-     }
 
-     loadGatherTree(sourceTree,  loadFun) {
 
-         for (const c of sourceTree.children) {
 
-             this.loadGatherNode(c, null, loadFun);
 
-         }
 
-         // todo load Pos Data;
 
-     }
 
-     calculateSum() {
 
-         if (this.setting.calcSum) {
 
-             for (const d of this.datas) {
 
-                 this.setting.calcSum(d, this.count);
 
-             }
 
-         }
 
-     }
 
- }
 
- class pos {
 
-     /**
 
-      * 构造函数
 
-      * @param {id|Number, masterId|Number} setting
 
-      */
 
-     constructor (setting) {
 
-         // 无索引
 
-         this.datas = [];
 
-         // 以key为索引
 
-         this.items = {};
 
-         // 以分类id为索引的有序
 
-         this.ledgerPos = {};
 
-         // pos设置
 
-         this.setting = setting;
 
-     }
 
-     /**
 
-      * 加载部位明细数据
 
-      * @param datas
 
-      */
 
-     loadDatas(datas) {
 
-         this.datas = datas;
 
-         this.items = {};
 
-         this.ledgerPos = {};
 
-         for (const data of this.datas) {
 
-             const key = itemsPre + data[this.setting.id];
 
-             this.items[key] = data;
 
-             const masterKey = itemsPre + data[this.setting.ledgerId];
 
-             if (!this.ledgerPos[masterKey]) {
 
-                 this.ledgerPos[masterKey] = [];
 
-             }
 
-             this.ledgerPos[masterKey].push(data);
 
-         }
 
-         for (const prop in this.ledgerPos) {
 
-             this.resortLedgerPos(this.ledgerPos[prop]);
 
-         }
 
-     }
 
-     getLedgerPos(mid) {
 
-         return this.ledgerPos[itemsPre + mid];
 
-     }
 
-     resortLedgerPos(ledgerPos) {
 
-         if (ledgerPos instanceof Array) {
 
-             ledgerPos.sort(function (a, b) {
 
-                 return a.porder - b.porder;
 
-             })
 
-         }
 
-     }
 
-     /**
 
-      * 计算全部
 
-      */
 
-     calculateAll(fun) {
 
-         const calcFun = fun ? fun : this.setting.calc;
 
-         if (!calcFun) return;
 
-         for (const pos of this.datas) {
 
-             calcFun(pos);
 
-         }
 
-     }
 
-     getDatas () {
 
-         return this.datas;
 
-     }
 
- }
 
- module.exports = {
 
-     billsTree,
 
-     pos,
 
-     filterTree,
 
-     filterGatherTree,
 
-     gatherTree,
 
- };
 
 
  |