From d1448cb0ef10f358bb7bddb4e1ec268515e0b787 Mon Sep 17 00:00:00 2001
From: gaoluyang <2820782392@qq.com>
Date: 星期二, 15 七月 2025 11:46:57 +0800
Subject: [PATCH] 项目初始化

---
 components/qian-tree/qian-tree.vue |  425 +++++++++++++++++++++++++++++++++++++++++++++++++++++
 1 files changed, 425 insertions(+), 0 deletions(-)

diff --git a/components/qian-tree/qian-tree.vue b/components/qian-tree/qian-tree.vue
new file mode 100644
index 0000000..e4deed7
--- /dev/null
+++ b/components/qian-tree/qian-tree.vue
@@ -0,0 +1,425 @@
+<template xlang="wxml">
+	<view class="tki-tree">
+		<view class="tki-tree-mask" :class="{'show':showTree}" @tap="_cancel"></view>
+		<view class="tki-tree-cnt" :class="{'show':showTree}">
+			<view class="tki-tree-bar">
+				<view class="tki-tree-bar-cancel" :style="{'color':cancelColor}" hover-class="hover-c" @tap="_cancel">鍙栨秷</view>
+				<view class="tki-tree-bar-title" :style="{'color':titleColor}">{{title}}</view>
+				<view class="tki-tree-bar-confirm" :style="{'color':confirmColor}" hover-class="hover-c" @tap="_confirm">纭畾</view>
+			</view>
+			<view class="tki-tree-view">
+				<scroll-view class="tki-tree-view-sc" :scroll-y="true">
+					<block v-for="(item, index) in treeList" :key="index">
+						<view class="tki-tree-item" :style="[{
+							paddingLeft: item.rank*15 + 'px',
+							zIndex: item.rank*-1 +50
+						}]"
+						 :class="{
+							border: border === true,
+							show: item.show,
+							last: item.lastRank,
+							showchild: item.showChild,
+							open: item.open,
+						}">
+							<view class="tki-tree-label" @tap.stop="_treeItemTap(item, index)">
+								<image class="tki-tree-icon" :src="item.lastRank ? lastIcon : item.showChild ? currentIcon : defaultIcon"></image>
+								{{item.name}}
+							</view>
+							<view class="tki-tree-check" @tap.stop="_treeItemSelect(item, index)" v-if="selectParent?true:item.lastRank">
+								<view class="tki-tree-check-yes" v-if="item.checked" :class="{'radio':!multiple}" :style="{'border-color':confirmColor}">
+									<view class="tki-tree-check-yes-b" :style="{'background-color':confirmColor}"></view>
+								</view>
+								<view class="tki-tree-check-no" v-else :class="{'radio':!multiple}" :style="{'border-color':confirmColor}"></view>
+							</view>
+						</view>
+					</block>
+				</scroll-view>
+			</view>
+		</view>
+	</view>
+</template>
+
+<script>
+	export default {
+		name: "tki-tree",
+		props: {
+			lazy: {
+				type: Boolean,
+				default: false
+			},
+			range: {
+				type: Array,
+				default: function() {
+					return []
+				}
+			},
+			idKey: {
+				type: String,
+				default: 'id'
+			},
+			rangeKey: {
+				type: String,
+				default: 'label'
+			},
+			title: {
+				type: String,
+				default: ''
+			},
+			multiple: { // 鏄惁鍙互澶氶��
+				type: Boolean,
+				default: false
+				// default: true
+			},
+			selectParent: { //鏄惁鍙互閫夌埗绾�
+				type: Boolean,
+				default: false
+			},
+			foldAll: { //鎶樺彔鏃跺叧闂墍鏈夊凡缁忔墦寮�鐨勫瓙闆嗭紝鍐嶆鎵撳紑鏃堕渶瑕佷竴绾т竴绾ф墦寮�
+				type: Boolean,
+				default: false
+			},
+			confirmColor: { // 纭畾鎸夐挳棰滆壊
+				type: String,
+				default: '' // #07bb07
+			},
+			cancelColor: { // 鍙栨秷鎸夐挳棰滆壊
+				type: String,
+				default: '' // #757575
+			},
+			titleColor: { // 鏍囬棰滆壊
+				type: String,
+				default: '' // #757575
+			},
+			currentIcon: { // 灞曞紑鏃跺�欑殑ic
+				type: String,
+				default: ''
+			},
+			defaultIcon: { // 鎶樺彔鏃跺�欑殑ic
+				type: String,
+				default: ''
+			},
+			lastIcon: { // 娌℃湁瀛愰泦鐨刬c
+				type: String,
+				default: ''
+			},
+			border: { // 鏄惁鏈夊垎鍓茬嚎
+				type: Boolean,
+				default: false
+			},
+		},
+		data() {
+			return {
+				showTree: false,
+				treeList: [],
+				selectIndex: -1,
+				returnedItem: [] ,//瀹氫箟涓�涓┖鏁扮粍
+				pids: [],
+				ancestorsIds: [],
+				childNums: [],
+			}
+		},
+		computed: {},
+		methods: {
+			_show() {
+				this.showTree = true
+			},
+			_hide() {
+				this.showTree = false
+			},
+			_cancel() {
+				this._hide()
+				this.$emit("cancel", '');
+			},
+			_confirm() {
+				// 澶勭悊鎵�閫夋暟鎹�
+				let rt = [],
+					obj = {};
+				this.treeList.forEach((v, i) => {
+					if (this.treeList[i].checked) {
+						rt.push(this.treeList[i].id)
+					}
+				})
+				this._hide()
+				this.$emit("confirm", rt);
+			},
+			//鎵佸钩鍖栨爲缁撴瀯
+			_renderTreeList(list = [], rank = 0, parentId = [], parents = []) {
+				list.forEach(item => {
+					this.treeList.push({
+						id: item[this.idKey],
+						name: item[this.rangeKey],
+						source: item,
+						parentId, // 鐖剁骇id鏁扮粍
+						parents, // 鐖剁骇id鏁扮粍
+						rank, // 灞傜骇
+						showChild: false, //瀛愮骇鏄惁鏄剧ず
+						open: false, //鏄惁鎵撳紑
+						show: rank === 0, // 鑷韩鏄惁鏄剧ず
+						hideArr: [],
+						orChecked: item.checked ? item.checked : false,
+						checked: item.checked ? item.checked : false,
+						childNum: 0
+					})
+					
+					if (Array.isArray(item.children) && item.children.length > 0) {
+						// console.log(item)
+						let parentid = [...parentId],
+							parentArr = [...parents];
+						delete parentArr.children
+						parentid.push(item[this.idKey]);
+						parentArr.push({
+							[this.idKey]: item[this.idKey],
+							[this.rangeKey]: item[this.rangeKey]
+						})
+						// lazy
+						if(!this.lazy) {
+							this._renderTreeList(item.children, rank + 1, parentid, parentArr)
+						}
+					} else {
+						this.treeList[this.treeList.length - 1].lastRank = true;
+					}
+				})
+			},
+			// 澶勭悊榛樿閫夋嫨
+			_defaultSelect() {
+				this.treeList.forEach((v, i) => {
+					if (v.checked) {
+						this.treeList.forEach((v2, i2) => {
+							if (v.parentId.toString().indexOf(v2.parentId.toString()) >= 0) {
+								v2.show = true
+								if (v.parentId.includes(v2.id)) {
+									v2.showChild = true;
+									v2.open = true;
+								}
+							}
+						})
+					}
+				})
+			},
+			getOwn(id, arr){
+			  //鍒╃敤foreach寰幆閬嶅巻
+			  arr.forEach((item) => {
+				//鍒ゆ柇閫掑綊缁撴潫鏉′欢
+				if(item[this.idKey] == id)
+				{       
+				  // 瀛樺偍鏁版嵁鍒扮┖鏁扮粍
+				  this.returnedItem = item
+				}
+				  else if(item.children != null) //鍒ゆ柇chlidren鏄惁鏈夋暟鎹�
+				{
+				  //閫掑綊璋冪敤
+				  this.getOwn(id, item.children);                 
+				}                   
+			  })
+			  return this.returnedItem
+			},
+			setShow (id, arr, isShow) {
+				arr.forEach((item, index) => {
+					if(item.parentId.includes(id)) {
+						this.treeList[index].showChild = isShow
+						this.treeList[index].show = isShow
+					} else if (item.children !== undefined) {
+						this.setShow(id, item.children, isShow)
+					}
+				})
+			},
+			// 鐐瑰嚮
+			_treeItemTap(item, index) {
+				// console.log(item)
+				if (item.lastRank === true) {
+					//鐐瑰嚮鏈�鍚庝竴绾ф椂瑙﹀彂浜嬩欢
+					this.treeList[index].checked = !this.treeList[index].checked
+					this._fixMultiple(index)
+					return;
+				}
+				let id = item.id;
+				item.showChild = !item.showChild;
+				// qingqian
+				if(item.showChild) {
+					const range = this.range
+					const parentIdArr = item.parentId
+					// 鎵惧埌褰撳墠鍏冪礌
+					const own = this.getOwn(id, range)
+					const checkedChildren = own.children
+					// 瀛愬厓绱犳彃鍏ョ殑绱㈠紩浣嶇疆
+					const nextIndex = this.treeList.findIndex(itemT => itemT.id === item.id)
+					console.log(checkedChildren);
+					if(checkedChildren === undefined || checkedChildren.length < 1) {
+						return
+					}
+					// 瀛愯妭鐐规暟閲�
+					this.treeList[index].childNum = checkedChildren.length
+					const newRank = item.rank + 1
+					checkedChildren.forEach(itemC => {
+						const childObj = {
+							id: itemC[this.idKey],
+							name: itemC[this.rangeKey],
+							source: {},
+							parentId: [item.id], // 鐖剁骇id鏁扮粍
+							parents: [item], // 鐖剁骇id鏁扮粍
+							rank: newRank, // 灞傜骇
+							showChild: false, //瀛愮骇鏄惁鏄剧ず
+							open: false, //鏄惁鎵撳紑
+							show: 1, // 鑷韩鏄惁鏄剧ず
+							hideArr: [],
+							orChecked:  this.treeList[index].checked,
+							checked:  this.treeList[index].checked,
+						}
+						if(!this.treeList.some(itemT => itemT.id === itemC[this.idKey])) {
+							this.treeList.splice(nextIndex+1,0,childObj)
+						}
+					})
+				}
+				// 灞曞紑/闅愯棌瀛愮骇/瀛欑骇
+				let list = this.treeList
+				item.open = item.showChild ? true : !item.open;
+				list.forEach((childItem, i) => {
+					if (item.showChild === false) {
+						//闅愯棌鎵�鏈夊瓙绾�
+						if (!childItem.parentId.includes(id)) {
+							return;
+						}
+						//TODO: 淇敼
+						if (!this.foldAll) {
+							if (childItem.lastRank !== true && !childItem.open) {
+								childItem.showChild = false;
+								this.setShow(childItem.id, this.treeList, false)
+							}
+							// 涓洪殣钘忕殑鍐呭娣诲姞涓�涓爣璁�
+							if (childItem.show) {
+								childItem.hideArr[item.rank] = id
+							}
+						} else {
+							if (childItem.lastRank !== true) {
+								childItem.showChild = false;
+								// 缁х画闅愯棌瀛愮骇鐨勭殑瀛愮骇
+								this.setShow(childItem.id, this.treeList, false)
+							}
+						}
+						if(childItem.children !== undefined) {
+							childItem.children.forEach((childItem1, i1) => {
+								if(!childItem1.parentId.includes(childItem.id)) {
+									return
+								}
+								childItem.children[i1].showChild = false
+								childItem.children[i1].show = false
+							})
+						}
+						childItem.show = false;
+					} else {
+						// 鎵撳紑瀛愰泦
+						if (childItem.parentId[childItem.parentId.length - 1] === id) {
+							childItem.show = true;
+						}
+						// 鎵撳紑琚殣钘忕殑瀛愰泦
+						if (childItem.parentId.includes(id) && !this.foldAll) {
+							// console.log(childItem.hideArr)
+							if (childItem.hideArr[item.rank] === id) {
+								childItem.show = true;
+								if (childItem.open && childItem.showChild) {
+									childItem.showChild = true
+								} else {
+									childItem.showChild = false
+								}
+								childItem.hideArr[item.rank] = null
+							}
+						}
+					}
+				})
+			},
+			// 閫氳繃鐖秈d澶勭悊瀛愮骇
+			syncChecked (trees, pid, checked) {
+				trees.forEach((item,index) => {
+					if(item.parentId.includes(pid)) {
+						this.treeList[index].checked = checked
+						this.syncChecked(trees, item.id, checked)
+					} else if(item.children !== undefined) {
+						this.syncChecked(item.children, pid, checked)
+					}
+				})
+			},
+			// 鑾峰彇鐖剁骇寰�涓婃墍鏈夊眰绾х殑id 骞跺悓姝ョ姸鎬�
+			setAncestors (pids, checked) {
+				this.treeList.forEach((item,index) => {
+					if(pids.includes(item.id)) {
+						if(checked && this.childNums[item.id] !== undefined && item.childNum === this.childNums[item.id]) {
+						// 瀛愮骇鍏ㄩ儴閫変腑, 鐖剁骇鎵嶉�変腑
+							this.treeList[index].checked = true
+						} else {
+							this.treeList[index].checked = false
+						}
+						this.setAncestors(item.parentId, checked)
+					} 
+				})
+			},
+			_treeItemSelect(item, index) {
+				this.treeList[index].checked = !this.treeList[index].checked
+				// 閫夌埗绾�, 瀛愮骇鑷姩鍏ㄩ��
+				this.syncChecked(this.treeList, item.id, this.treeList[index].checked)
+			
+				if(item.rank > 0) {
+					item.parentId.forEach((pid, indexP) => {
+						const parent = this.treeList.filter(i => i.id === pid)
+						const childNum = parent.length > 0 ? parent[0].childNum : 0
+						if(this.childNums[pid] === undefined) {
+							this.childNums[pid] = 1
+						} else if(this.childNums[pid] < childNum) {
+							this.childNums[pid]++
+						}
+					})
+					//瀛愮骇閫夋嫨/閫夋弧/鍙栨秷閫夋嫨, 鐖剁骇寰�涓婂悓姝ョ姸鎬�
+					this.setAncestors(item.parentId, this.treeList[index].checked)
+				}
+				this._fixMultiple(index)
+			},
+			// 澶勭悊鍗曢�夊閫�
+			_fixMultiple(index) {
+				if (!this.multiple) {
+					// 濡傛灉鏄崟閫�
+					this.treeList.forEach((v, i) => {
+						if (i != index) {
+							this.treeList[i].checked = false
+						} else {
+							this.treeList[i].checked = true
+						}
+					})
+				}
+			},
+			// 閲嶇疆鏁版嵁
+			_reTreeList() {
+				this.treeList.forEach((v, i) => {
+					this.treeList[i].checked = v.orChecked
+				})
+			},
+			_initTree(range = this.range){
+				this.treeList = [];
+				this._renderTreeList(range);
+				this.$nextTick(() => {
+					this._defaultSelect(range)
+				})
+			}
+		},
+		watch: {
+			range(list) {
+				this._initTree(list);
+			},
+			multiple() {
+				if (this.range.length) {
+					this._reTreeList();
+				}
+			},
+			selectParent() {
+				if (this.range.length) {
+					this._reTreeList();
+				}
+			},
+		},
+		mounted() {
+			this._initTree();
+		}
+	}
+</script>
+
+<style scoped>
+	@import "./style.css";
+</style>

--
Gitblit v1.9.3