| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498 | 
							- import {
 
- 	markNodeData,
 
- 	objectAssign,
 
- 	arrayFindIndex,
 
- 	getChildState,
 
- 	reInitChecked,
 
- 	getPropertyFromData,
 
- 	isNull,
 
- 	NODE_KEY
 
- } from '../tool/util';
 
- const getStore = function(store) {
 
- 	let thisStore = store;
 
- 	
 
- 	return function() {
 
- 		return thisStore;
 
- 	}
 
- }
 
- let nodeIdSeed = 0;
 
- export default class Node {
 
- 	constructor(options) {
 
- 		this.time = new Date().getTime();
 
- 		this.id = nodeIdSeed++;
 
- 		this.text = null;
 
- 		this.checked = false;
 
- 		this.indeterminate = false;
 
- 		this.data = null;
 
- 		this.expanded = false;
 
- 		this.parentId = null;
 
- 		this.visible = true;
 
- 		this.isCurrent = false;
 
- 		for (let name in options) {
 
- 			if (options.hasOwnProperty(name)) {
 
- 				if (name === 'store') {
 
- 					this.store = getStore(options[name]);
 
- 				} else {
 
- 					this[name] = options[name];
 
- 				}
 
- 			}
 
- 		}
 
- 		// internal
 
- 		this.level = 0;
 
- 		this.loaded = false;
 
- 		this.childNodesId = [];
 
- 		this.loading = false;
 
- 		this.label = getPropertyFromData(this, 'label');
 
- 		this.key = this.data ? this.data[this.store().key] : null;
 
- 		this.disabled = getPropertyFromData(this, 'disabled');
 
- 		this.nextSibling = null;
 
- 		this.previousSibling = null;
 
- 		this.icon = '';
 
- 		if (this.parentId !== null) {
 
- 			let parent = this.getParent(this.parentId);
 
- 			
 
- 			if (this.store().isInjectParentInNode) {
 
- 				this.parent = parent;
 
- 			}
 
- 			
 
- 			// 由于这里做了修改,默认第一个对象不会被注册到nodesMap中,所以找不到parent会报错,所以默认parent的level是0
 
- 			if (!parent) {
 
- 				parent = {
 
- 					level: 0
 
- 				}
 
- 			} else {
 
- 				const parentChildNodes = parent.getChildNodes(parent.childNodesId);
 
- 				const index = parent.childNodesId.indexOf(this.key);
 
- 				this.nextSibling = index > -1 ? parentChildNodes[index + 1] : null;
 
- 				this.previousSibling = index > 0 ? parentChildNodes[index - 1] : null;
 
- 			}
 
- 			this.level = parent.level + 1;
 
- 		}
 
- 		if (!this.store()) {
 
- 			throw new Error('[Node]store is required!');
 
- 		}
 
- 		
 
- 		const props = this.store().props;
 
- 		
 
- 		if (this.store().showNodeIcon && props && typeof props.icon !== 'undefined') {
 
- 			this.icon = getPropertyFromData(this, props.icon);
 
- 		}
 
- 		
 
- 		this.store().registerNode(this);
 
- 		
 
- 		if (props && typeof props.isLeaf !== 'undefined') {
 
- 			const isLeaf = getPropertyFromData(this, 'isLeaf');
 
- 			if (typeof isLeaf === 'boolean') {
 
- 				this.isLeafByUser = isLeaf;
 
- 			}
 
- 		}
 
- 		if (this.store().lazy !== true && this.data) {
 
- 			this.setData(this.data);
 
- 			if (this.store().defaultExpandAll) {
 
- 				this.expanded = true;
 
- 			}
 
- 		} else if (this.level > 0 && this.store().lazy && this.store().defaultExpandAll) {
 
- 			this.expand();
 
- 		}
 
- 		
 
- 		if (!Array.isArray(this.data)) {
 
- 			markNodeData(this, this.data);
 
- 		}
 
- 		
 
- 		if (!this.data) return;
 
- 		
 
- 		const defaultExpandedKeys = this.store().defaultExpandedKeys;
 
- 		const key = this.store().key;
 
- 		if (key && defaultExpandedKeys && defaultExpandedKeys.indexOf(this.key) !== -1) {
 
- 			this.expand(null, this.store().autoExpandparent);
 
- 		}
 
- 		if (key && this.store().currentNodeKey !== undefined && this.key === this.store().currentNodeKey) {
 
- 			this.store().currentNode = this;
 
- 			this.store().currentNode.isCurrent = true;
 
- 		}
 
- 		if (this.store().lazy) {
 
- 			this.store()._initDefaultCheckedNode(this);
 
- 		}
 
- 		this.updateLeafState();
 
- 	}
 
- 	
 
- 	destroyStore() {
 
- 		getStore(null)
 
- 	}
 
- 	setData(data) {
 
- 		if (!Array.isArray(data)) {
 
- 			markNodeData(this, data);
 
- 		}
 
- 		this.data = data;
 
- 		this.childNodesId = [];
 
- 		let children;
 
- 		if (this.level === 0 && Array.isArray(this.data)) {
 
- 			children = this.data;
 
- 		} else {
 
- 			children = getPropertyFromData(this, 'children') || [];
 
- 		}
 
- 		for (let i = 0, j = children.length; i < j; i++) {
 
- 			this.insertChild({
 
- 				data: children[i]
 
- 			});
 
- 		}
 
- 	}
 
- 	contains(target, deep = true) {
 
- 		const walk = function(parent) {
 
- 			const children = parent.getChildNodes(parent.childNodesId) || [];
 
- 			let result = false;
 
- 			for (let i = 0, j = children.length; i < j; i++) {
 
- 				const child = children[i];
 
- 				if (child === target || (deep && walk(child))) {
 
- 					result = true;
 
- 					break;
 
- 				}
 
- 			}
 
- 			return result;
 
- 		};
 
- 		return walk(this);
 
- 	}
 
- 	remove() {
 
- 		if (this.parentId !== null) {
 
- 			const parent = this.getParent(this.parentId);
 
- 			parent.removeChild(this);
 
- 		}
 
- 	}
 
- 	insertChild(child, index, batch) {
 
- 		if (!child) throw new Error('insertChild error: child is required.');
 
- 		if (!(child instanceof Node)) {
 
- 			if (!batch) {
 
- 				const children = this.getChildren(true);
 
- 				if (children.indexOf(child.data) === -1) {
 
- 					if (typeof index === 'undefined' || index < 0) {
 
- 						children.push(child.data);
 
- 					} else {
 
- 						children.splice(index, 0, child.data);
 
- 					}
 
- 				}
 
- 			}
 
- 			
 
- 			objectAssign(child, {
 
- 				parentId: isNull(this.key) ? '' : this.key,
 
- 				store: this.store()
 
- 			});
 
- 			child = new Node(child);
 
- 		}
 
- 		child.level = this.level + 1;
 
- 		if (typeof index === 'undefined' || index < 0) {
 
- 			this.childNodesId.push(child.key);
 
- 		} else {
 
- 			this.childNodesId.splice(index, 0, child.key);
 
- 		}
 
- 		this.updateLeafState();
 
- 	}
 
- 	insertBefore(child, ref) {
 
- 		let index;
 
- 		if (ref) {
 
- 			index = this.childNodesId.indexOf(ref.id);
 
- 		}
 
- 		this.insertChild(child, index);
 
- 	}
 
- 	insertAfter(child, ref) {
 
- 		let index;
 
- 		if (ref) {
 
- 			index = this.childNodesId.indexOf(ref.id);
 
- 			if (index !== -1) index += 1;
 
- 		}
 
- 		this.insertChild(child, index);
 
- 	}
 
- 	removeChild(child) {
 
- 		const children = this.getChildren() || [];
 
- 		const dataIndex = children.indexOf(child.data);
 
- 		if (dataIndex > -1) {
 
- 			children.splice(dataIndex, 1);
 
- 		}
 
- 		
 
- 		const index = this.childNodesId.indexOf(child.key);
 
- 		
 
- 		if (index > -1) {
 
- 			this.store() && this.store().deregisterNode(child);
 
- 			child.parentId = null;
 
- 			this.childNodesId.splice(index, 1);
 
- 		}
 
- 		
 
- 		this.updateLeafState();
 
- 	}
 
- 	removeChildByData(data) {
 
- 		let targetNode = null;
 
- 		for (let i = 0; i < this.childNodesId.length; i++) {
 
- 			let node = this.getChildNodes(this.childNodesId);
 
- 			if (node[i].data === data) {
 
- 				targetNode = node[i];
 
- 				break;
 
- 			}
 
- 		}
 
- 		if (targetNode) {
 
- 			this.removeChild(targetNode);
 
- 		}
 
- 	}
 
- 	// 为了避免APP端parent嵌套结构导致报错,这里parent需要从nodesMap中获取
 
- 	getParent(parentId) {
 
- 		if (!parentId.toString()) return null;
 
- 		return this.store().nodesMap[parentId];
 
- 	}
 
- 	// 为了避免APP端childNodes嵌套结构导致报错,这里childNodes需要从nodesMap中获取
 
- 	getChildNodes(childNodesId) {
 
- 		let childNodes = [];
 
- 		if (childNodesId.length === 0) return childNodes;
 
- 		childNodesId.forEach((key) => {
 
- 			childNodes.push(this.store().nodesMap[key]);
 
- 		})
 
- 		return childNodes;
 
- 	}
 
- 	expand(callback, expandparent) {
 
- 		const done = () => {
 
- 			if (expandparent) {
 
- 				let parent = this.getParent(this.parentId);
 
- 				while (parent && parent.level > 0) {
 
- 					parent.expanded = true;
 
- 					parent = this.getParent(parent.parentId);
 
- 				}
 
- 			}
 
- 			this.expanded = true;
 
- 			if (callback) callback();
 
- 		};
 
- 		if (this.shouldLoadData()) {
 
- 			this.loadData(function(data) {
 
- 				if (Array.isArray(data)) {
 
- 					if (this.checked) {
 
- 						this.setChecked(true, true);
 
- 					} else if (!this.store().checkStrictly) {
 
- 						reInitChecked(this);
 
- 					}
 
- 					done();
 
- 				}
 
- 			});
 
- 		} else {
 
- 			done();
 
- 		}
 
- 	}
 
- 	doCreateChildren(array, defaultProps = {}) {
 
- 		array.forEach((item) => {
 
- 			this.insertChild(objectAssign({
 
- 				data: item
 
- 			}, defaultProps), undefined, true);
 
- 		});
 
- 	}
 
- 	collapse() {
 
- 		this.expanded = false;
 
- 	}
 
- 	shouldLoadData() {
 
- 		return this.store().lazy === true && this.store().load && !this.loaded;
 
- 	}
 
- 	updateLeafState() {
 
- 		if (this.store().lazy === true && this.loaded !== true && typeof this.isLeafByUser !== 'undefined') {
 
- 			this.isLeaf = this.isLeafByUser;
 
- 			return;
 
- 		}
 
- 		const childNodesId = this.childNodesId;
 
- 		if (!this.store().lazy || (this.store().lazy === true && this.loaded === true)) {
 
- 			this.isLeaf = !childNodesId || childNodesId.length === 0;
 
- 			return;
 
- 		}
 
- 		this.isLeaf = false;
 
- 	}
 
- 	setChecked(value, deep, recursion, passValue) {
 
- 		this.indeterminate = value === 'half';
 
- 		this.checked = value === true;
 
- 		
 
- 		if (this.checked && this.store().expandOnCheckNode) {
 
- 			this.expand(null, true)
 
- 		}
 
- 		
 
- 		if (this.store().checkStrictly) return;
 
- 		if (this.store().showRadio) return;
 
- 		if (!(this.shouldLoadData() && !this.store().checkDescendants)) {
 
- 			let childNodes = this.getChildNodes(this.childNodesId);
 
- 			let {
 
- 				all,
 
- 				allWithoutDisable
 
- 			} = getChildState(childNodes);
 
- 			if (!this.isLeaf && (!all && allWithoutDisable)) {
 
- 				this.checked = false;
 
- 				value = false;
 
- 			}
 
- 			const handleDescendants = () => {
 
- 				if (deep) {
 
- 					let childNodes = this.getChildNodes(this.childNodesId)
 
- 					for (let i = 0, j = childNodes.length; i < j; i++) {
 
- 						const child = childNodes[i];
 
- 						passValue = passValue || value !== false;
 
- 						const isCheck = child.disabled ? child.checked : passValue;
 
- 						child.setChecked(isCheck, deep, true, passValue);
 
- 					}
 
- 					const {
 
- 						half,
 
- 						all
 
- 					} = getChildState(childNodes);
 
- 					
 
- 					if (!all) {
 
- 						this.checked = all;
 
- 						this.indeterminate = half;
 
- 					}
 
- 				}
 
- 			};
 
- 			if (this.shouldLoadData()) {
 
- 				this.loadData(() => {
 
- 					handleDescendants();
 
- 					reInitChecked(this);
 
- 				}, {
 
- 					checked: value !== false
 
- 				});
 
- 				return;
 
- 			} else {
 
- 				handleDescendants();
 
- 			}
 
- 		}
 
- 		if (!this.parentId) return;
 
- 		let parent = this.getParent(this.parentId);
 
- 		if (parent && parent.level === 0) return;
 
- 		if (!recursion) {
 
- 			reInitChecked(parent);
 
- 		}
 
- 	}
 
- 	setRadioChecked(value) {
 
- 		const allNodes = this.store()._getAllNodes().sort((a, b) => b.level - a.level);
 
- 		allNodes.forEach(node => node.setChecked(false, false));
 
- 		this.checked = value === true;
 
- 	}
 
- 	getChildren(forceInit = false) {
 
- 		if (this.level === 0) return this.data;
 
- 		const data = this.data;
 
- 		if (!data) return null;
 
- 		const props = this.store().props;
 
- 		let children = 'children';
 
- 		if (props) {
 
- 			children = props.children || 'children';
 
- 		}
 
- 		if (data[children] === undefined) {
 
- 			data[children] = null;
 
- 		}
 
- 		if (forceInit && !data[children]) {
 
- 			data[children] = [];
 
- 		}
 
- 		return data[children];
 
- 	}
 
- 	updateChildren() {
 
- 		let childNodes = this.getChildNodes(this.childNodesId);
 
- 		const newData = this.getChildren() || [];
 
- 		const oldData = childNodes.map((node) => node.data);
 
- 		const newDataMap = {};
 
- 		const newNodes = [];
 
- 		newData.forEach((item, index) => {
 
- 			const key = item[NODE_KEY];
 
- 			const isNodeExists = !!key && arrayFindIndex(oldData, data => data[NODE_KEY] === key) >= 0;
 
- 			if (isNodeExists) {
 
- 				newDataMap[key] = {
 
- 					index,
 
- 					data: item
 
- 				};
 
- 			} else {
 
- 				newNodes.push({
 
- 					index,
 
- 					data: item
 
- 				});
 
- 			}
 
- 		});
 
- 		if (!this.store().lazy) {
 
- 			oldData.forEach((item) => {
 
- 				if (!newDataMap[item[NODE_KEY]]) this.removeChildByData(item);
 
- 			});
 
- 		}
 
- 		newNodes.forEach(({
 
- 			index,
 
- 			data
 
- 		}) => {
 
- 			this.insertChild({
 
- 				data
 
- 			}, index);
 
- 		});
 
- 		this.updateLeafState();
 
- 	}
 
- 	loadData(callback, defaultProps = {}) {
 
- 		if (this.store().lazy === true && 
 
- 			this.store().load && !this.loaded && 
 
- 			(!this.loading || Object.keys(defaultProps).length)
 
- 		) {
 
- 			this.loading = true;
 
- 			const resolve = (children) => {
 
- 				this.loaded = true;
 
- 				this.loading = false;
 
- 				this.childNodesId = [];
 
- 				this.doCreateChildren(children, defaultProps);
 
- 				this.updateLeafState();
 
- 				
 
- 				callback && callback.call(this,children);
 
- 			};
 
- 			this.store().load(this, resolve);
 
- 		} else {
 
- 			callback && callback.call(this);
 
- 		}
 
- 	}
 
- }
 
 
  |