442 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			Vue
		
	
	
	
			
		
		
	
	
			442 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			Vue
		
	
	
	
| <template>
 | |
|   <view class="table pr" :style="'zoom:' + size + ';'">
 | |
|     <view class="v-tr" style="display: flex">
 | |
|       <view
 | |
|         class="v-td"
 | |
|         :class="{
 | |
|           colspan: Array.isArray(treeData.children)
 | |
|             ? treeData.children.length * 2
 | |
|             : 1,
 | |
|           parentLevel:
 | |
|             Array.isArray(treeData.children) && treeData.children.length,
 | |
|           extend:
 | |
|             Array.isArray(treeData.children) &&
 | |
|             treeData.children.length &&
 | |
|             treeData.extend,
 | |
|         }"
 | |
|       >
 | |
|         <view class="node">
 | |
|           <view
 | |
|             class="person"
 | |
|             :class="Array.isArray(treeData.class) ? treeData.class : []"
 | |
|           >
 | |
|             <view class="frame-text">
 | |
|               <!-- <view style="font-weight: bold">{{
 | |
|                 treeData.nodeCode === '0-root' ? '' : treeData.nodeCode
 | |
|               }}</view> -->
 | |
|               <view v-if="treeData.memberCode && treeData.memberName">
 | |
|                 <view>{{ treeData.memberCode }}</view>
 | |
|                 <view
 | |
|                   >{{ treeData.stage }}-{{ treeData.stageSort }}({{
 | |
|                     treeData.pointMember
 | |
|                   }})</view
 | |
|                 >
 | |
|                 <!-- <view>{{ treeData.pointMember }}</view> -->
 | |
|                 <view>{{ treeData.creationTime }}</view>
 | |
|               </view>
 | |
|               <view v-else> 空点位 </view>
 | |
|             </view>
 | |
|           </view>
 | |
|         </view>
 | |
|       </view>
 | |
|     </view>
 | |
|     <view
 | |
|       class="v-tr"
 | |
|       v-if="
 | |
|         Array.isArray(treeData.children) &&
 | |
|         treeData.children.length &&
 | |
|         treeData.extend
 | |
|       "
 | |
|     >
 | |
|       <view
 | |
|         v-for="(children, index) in treeData.children"
 | |
|         :key="index"
 | |
|         colspan="2"
 | |
|         class="childLevel v-td"
 | |
|       >
 | |
|         <TreeChart
 | |
|           :json="children"
 | |
|           :left="0"
 | |
|           :top="0"
 | |
|           @click-node="clickNode"
 | |
|           @click-top="clickTop"
 | |
|           :stageName="stageName"
 | |
|         />
 | |
|       </view>
 | |
|     </view>
 | |
|   </view>
 | |
| </template>
 | |
| 
 | |
| <script>
 | |
| export default {
 | |
|   name: 'TreeChart',
 | |
|   props: ['json', 'size', 'stageName'],
 | |
|   data() {
 | |
|     return {
 | |
|       treeData: {},
 | |
|     }
 | |
|   },
 | |
|   watch: {
 | |
|     json: {
 | |
|       handler: function (Props) {
 | |
|         let extendKey = function (jsonData) {
 | |
|           jsonData.extend =
 | |
|             jsonData.extend === void 0 ? true : !!jsonData.extend
 | |
|           if (Array.isArray(jsonData.children)) {
 | |
|             jsonData.children.forEach(c => {
 | |
|               extendKey(c)
 | |
|             })
 | |
|           }
 | |
|           return jsonData
 | |
|         }
 | |
|         if (Props) {
 | |
|           this.treeData = extendKey(Props)
 | |
|         }
 | |
|       },
 | |
|       immediate: true,
 | |
|     },
 | |
|   },
 | |
|   methods: {
 | |
|     toggleExtend: function (treeData) {
 | |
|       treeData.extend = !treeData.extend
 | |
|       this.$forceUpdate()
 | |
|     },
 | |
|     clickNode(e) {
 | |
|       this.$emit('click-node', e)
 | |
|     },
 | |
|     clickTop(e) {
 | |
|       this.$emit('click-top', e)
 | |
|     },
 | |
|   },
 | |
| }
 | |
| </script>
 | |
| 
 | |
| <style scoped>
 | |
| .f18 {
 | |
|   font-size: 18rpx;
 | |
| }
 | |
| 
 | |
| .pv-btn {
 | |
|   width: 200rpx;
 | |
|   height: 50rpx;
 | |
|   background: #007bff;
 | |
|   color: #fff;
 | |
|   font-size: 20rpx;
 | |
|   line-height: 50rpx;
 | |
|   display: flex;
 | |
|   align-items: center;
 | |
|   margin: auto;
 | |
|   margin-top: 12rpx;
 | |
|   margin-bottom: 12rpx;
 | |
|   border-radius: 8rpx;
 | |
|   box-shadow: 0 2rpx 8rpx rgba(0, 123, 255, 0.2);
 | |
|   transition: all 0.3s ease;
 | |
| }
 | |
| 
 | |
| .pv-btn:hover {
 | |
|   background: #0056b3;
 | |
|   transform: translateY(-1rpx);
 | |
|   box-shadow: 0 4rpx 12rpx rgba(0, 123, 255, 0.3);
 | |
| }
 | |
| 
 | |
| .next-btn {
 | |
|   border-radius: 50%;
 | |
|   background: #6c757d;
 | |
|   width: 40rpx;
 | |
|   height: 40rpx;
 | |
|   margin: auto;
 | |
|   padding: 0;
 | |
|   display: flex;
 | |
|   justify-content: center;
 | |
|   align-items: center;
 | |
|   box-shadow: 0 2rpx 4rpx rgba(0, 0, 0, 0.1);
 | |
|   transition: all 0.3s ease;
 | |
| }
 | |
| 
 | |
| .next-btn:hover {
 | |
|   background: #495057;
 | |
|   transform: scale(1.05);
 | |
|   box-shadow: 0 4rpx 8rpx rgba(0, 0, 0, 0.15);
 | |
| }
 | |
| 
 | |
| .table {
 | |
|   border-collapse: separate !important;
 | |
|   border-spacing: 0 !important;
 | |
|   width: 100%;
 | |
|   /* background: #f8f9fa; */
 | |
|   padding: 40rpx 0;
 | |
| }
 | |
| 
 | |
| .v-tr {
 | |
|   display: inline-flex;
 | |
|   justify-content: center;
 | |
|   align-items: flex-start;
 | |
|   width: 100%;
 | |
| }
 | |
| 
 | |
| .v-td {
 | |
|   position: relative;
 | |
|   vertical-align: top;
 | |
|   padding: 0 0 130rpx 0;
 | |
|   text-align: center;
 | |
| }
 | |
| 
 | |
| .extend_handle {
 | |
|   position: absolute;
 | |
|   left: 50%;
 | |
|   bottom: 30rpx;
 | |
|   width: 10rpx;
 | |
|   height: 10rpx;
 | |
|   padding: 10rpx;
 | |
|   transform: translate3d(-30rpx, 0, 0);
 | |
|   cursor: pointer;
 | |
|   transition: all 0.3s ease;
 | |
| }
 | |
| 
 | |
| .extend_handle:before {
 | |
|   content: '';
 | |
|   display: block;
 | |
|   width: 100%;
 | |
|   height: 100%;
 | |
|   box-sizing: border-box;
 | |
|   border: 4rpx solid;
 | |
|   border-color: #6c757d #6c757d transparent transparent;
 | |
|   transform: rotateZ(135deg);
 | |
|   transform-origin: 50% 50% 0;
 | |
|   transition: all 0.3s ease;
 | |
| }
 | |
| 
 | |
| .extend_handle:hover:before {
 | |
|   border-color: #495057 #495057 transparent transparent;
 | |
|   transform: rotateZ(135deg) scale(1.1);
 | |
| }
 | |
| 
 | |
| .extend .extend_handle:before {
 | |
|   transform: rotateZ(-45deg);
 | |
| }
 | |
| 
 | |
| .extend .extend_handle:hover:before {
 | |
|   transform: rotateZ(-45deg) scale(1.1);
 | |
| }
 | |
| 
 | |
| .extend::after {
 | |
|   content: '';
 | |
|   position: absolute;
 | |
|   left: 50%;
 | |
|   bottom: 84rpx;
 | |
|   height: 84rpx;
 | |
|   border-left: 4rpx solid #dee2e6;
 | |
|   transform: translate3d(-2rpx, 0, 0);
 | |
| }
 | |
| 
 | |
| .childLevel::before {
 | |
|   content: '';
 | |
|   position: absolute;
 | |
|   left: 50%;
 | |
|   bottom: 100%;
 | |
|   height: 84rpx;
 | |
|   border-left: 4rpx solid #dee2e6;
 | |
|   transform: translate3d(-2rpx, 0, 0);
 | |
| }
 | |
| 
 | |
| .childLevel::after {
 | |
|   content: '';
 | |
|   position: absolute;
 | |
|   left: 0;
 | |
|   right: 0;
 | |
|   top: -88rpx;
 | |
|   border-top: 4rpx solid #dee2e6;
 | |
| }
 | |
| 
 | |
| .childLevel:first-child:before,
 | |
| .childLevel:last-child:before {
 | |
|   display: none;
 | |
| }
 | |
| 
 | |
| .childLevel:first-child:after {
 | |
|   left: 50%;
 | |
|   height: 84rpx;
 | |
|   border: 4rpx solid;
 | |
|   border-color: #dee2e6 transparent transparent #dee2e6;
 | |
|   border-radius: 8rpx 0 0 0;
 | |
|   transform: translate3d(2rpx, 0, 0);
 | |
| }
 | |
| 
 | |
| .childLevel:last-child:after {
 | |
|   right: 50%;
 | |
|   height: 84rpx;
 | |
|   border: 4rpx solid;
 | |
|   border-color: #dee2e6 #dee2e6 transparent transparent;
 | |
|   border-radius: 0 8rpx 0 0;
 | |
|   transform: translate3d(-2rpx, 0, 0);
 | |
| }
 | |
| 
 | |
| .childLevel:first-child.childLevel:last-child::after {
 | |
|   left: auto;
 | |
|   border-radius: 0;
 | |
|   border-color: transparent #dee2e6 transparent transparent;
 | |
|   transform: translate3d(2rpx, 0, 0);
 | |
| }
 | |
| 
 | |
| .node {
 | |
|   position: relative;
 | |
|   display: inline-block;
 | |
|   margin: 0 10rpx;
 | |
|   box-sizing: border-box;
 | |
|   text-align: center;
 | |
| }
 | |
| 
 | |
| .node .person {
 | |
|   position: relative;
 | |
|   display: inline-block;
 | |
|   z-index: 2;
 | |
|   width: fit-content;
 | |
|   padding: 20rpx;
 | |
|   background: #ffffff;
 | |
|   box-shadow: 0 4rpx 16rpx rgba(0, 0, 0, 0.1);
 | |
|   border-radius: 16rpx;
 | |
|   border: 2rpx solid #e9ecef;
 | |
|   transition: all 0.3s ease;
 | |
|   cursor: pointer;
 | |
|   min-width: 120rpx;
 | |
| }
 | |
| 
 | |
| .node .person:hover {
 | |
|   transform: translateY(-4rpx);
 | |
|   box-shadow: 0 8rpx 24rpx rgba(0, 0, 0, 0.15);
 | |
|   border-color: #6c757d;
 | |
| }
 | |
| 
 | |
| .node .person:active {
 | |
|   transform: translateY(-2rpx);
 | |
|   box-shadow: 0 4rpx 12rpx rgba(0, 0, 0, 0.1);
 | |
| }
 | |
| 
 | |
| .node .person .person-ava {
 | |
|   width: 88rpx;
 | |
|   height: 53rpx;
 | |
|   border-radius: 15rpx;
 | |
|   box-shadow: 0 4rpx 12rpx rgba(0, 0, 0, 0.1);
 | |
| }
 | |
| 
 | |
| .frame-text {
 | |
|   border-radius: 8rpx;
 | |
|   font-size: 22rpx;
 | |
|   margin: 0 auto;
 | |
|   color: #495057;
 | |
|   font-weight: 500;
 | |
|   line-height: 1.4;
 | |
| }
 | |
| 
 | |
| .frame-text view:first-child {
 | |
|   font-weight: 600;
 | |
|   font-size: 24rpx;
 | |
|   margin-bottom: 4rpx;
 | |
|   color: #212529;
 | |
| }
 | |
| 
 | |
| .frame-text view:not(:first-child) {
 | |
|   font-size: 20rpx;
 | |
|   color: #6c757d;
 | |
|   margin-bottom: 2rpx;
 | |
| }
 | |
| 
 | |
| .node .person .avat {
 | |
|   display: block;
 | |
|   width: 4em;
 | |
|   height: 4em;
 | |
|   margin: auto;
 | |
|   overflow: hidden;
 | |
|   background: #f8f9fa;
 | |
|   border: 4px solid #e9ecef;
 | |
|   box-sizing: border-box;
 | |
|   border-radius: 1em;
 | |
| }
 | |
| 
 | |
| .node .person .avat img {
 | |
|   width: 100%;
 | |
|   height: 100%;
 | |
|   border-radius: inherit;
 | |
| }
 | |
| 
 | |
| .node .person .name {
 | |
|   display: block;
 | |
|   margin: auto;
 | |
|   overflow: hidden;
 | |
|   box-sizing: border-box;
 | |
|   font-size: 20rpx;
 | |
|   line-height: 1.5;
 | |
|   padding-top: 24rpx;
 | |
|   color: #495057;
 | |
|   font-weight: 500;
 | |
| }
 | |
| 
 | |
| .node .person .name-bottom {
 | |
|   display: block;
 | |
|   margin: auto;
 | |
|   overflow: hidden;
 | |
|   border-top: none;
 | |
|   box-sizing: border-box;
 | |
|   color: #6c757d;
 | |
| }
 | |
| 
 | |
| .node.hasMate::after {
 | |
|   content: '';
 | |
|   position: absolute;
 | |
|   left: 2em;
 | |
|   right: 2em;
 | |
|   top: 2em;
 | |
|   border-top: 4px solid #dee2e6;
 | |
|   z-index: 1;
 | |
| }
 | |
| 
 | |
| /* 横板 */
 | |
| .landscape {
 | |
|   transform: translate(-100%, 0) rotate(-90deg);
 | |
|   transform-origin: 100% 0;
 | |
| }
 | |
| 
 | |
| .landscape .node {
 | |
|   text-align: left;
 | |
|   height: 8em;
 | |
|   width: 8em;
 | |
| }
 | |
| 
 | |
| .landscape .person {
 | |
|   position: relative;
 | |
|   transform: rotate(90deg);
 | |
|   padding-left: 4.5em;
 | |
|   height: 4em;
 | |
|   top: 4em;
 | |
|   left: -1em;
 | |
| }
 | |
| 
 | |
| .landscape .person .avat {
 | |
|   position: absolute;
 | |
|   left: 0;
 | |
| }
 | |
| 
 | |
| .landscape .person .name {
 | |
|   height: 4em;
 | |
|   line-height: 4em;
 | |
| }
 | |
| 
 | |
| .landscape .hasMate {
 | |
|   position: relative;
 | |
| }
 | |
| 
 | |
| .landscape .hasMate .person {
 | |
|   position: absolute;
 | |
| }
 | |
| 
 | |
| .landscape .hasMate .person:first-child {
 | |
|   left: auto;
 | |
|   right: -4em;
 | |
| }
 | |
| 
 | |
| .landscape .hasMate .person:last-child {
 | |
|   left: -4em;
 | |
|   margin-left: 0;
 | |
| }
 | |
| </style>
 |