Power BI Echart Tree開發自定義可視化對象

開發環境:

Power BI Desktop 2017

API版本v1.9.0

使用插件內置js的辦法。

{
  "visual": {
    "name": "d3tree",
    "displayName": "d3tree",
    "guid": "circleCard6331EDD59B344F31992BF545DD427473",
    "visualClassName": "Visual",
    "version": "1.0.0",
    "description": "",
    "supportUrl": "",
    "gitHubUrl": ""
  },
  "apiVersion": "1.9.0",
  "author": {
    "name": "",
    "email": ""
  },
  "assets": {
    "icon": "assets/icon.png"
  },
  "externalJS": [
    "node_modules/powerbi-visuals-utils-dataviewutils/lib/index.js",
    "node_modules/d3/index.js"//調用外部js,這裏存放着d3.js
  ],
  "style": "style/visual.less",
  "capabilities": "capabilities.json",
  "dependencies": "dependencies.json",
  "stringResources": []
}

使用了JS動態在線導入的辦法。在初始化視圖對象的時候導入在線echart.js.

1.在構造函數內引入JS

      constructor(options: VisualConstructorOptions) {
          this.target = options.element;
          this.echartLoad=false;
          this.onloadEchart();
          if (typeof document !== "undefined") {
          this.tips = document.createElement("p");
          this.tips.appendChild(document.createTextNode("正在加載資源..."));
          this.target.appendChild(this.tips);
        }
      }
      private onloadEchart(){
        var script=document.createElement('script');
        script.type="text/javascript";
        script.src="https://cdn.jsdelivr.net/npm/echarts/dist/echarts.min.js";
        document.body.appendChild(script);
        let that=this;
        script.onload=function(){
         that.echarts=echarts;
         that.echartLoad=true;
         that.checkJsLoad();
        }
      }

2.由於腳手架是typescript的,而我們在線導入的echart.js的全局對象是echarts,這個變量名字直接在這裏使用時會報錯的,需要先聲明這個變量。首先創建文件test.d.ts用於聲明這個變量。然後在tsconfig.json文件中包含加載新的文件。

declare var echarts: any;
{
  "compilerOptions": {
    "allowJs": true,
    "emitDecoratorMetadata": true,
    "experimentalDecorators": true,
    "target": "ES5",
    "sourceMap": true,
    "out": "./.tmp/build/visual.js"
  },
  "files": [
    ".api/v1.9.0/PowerBI-visuals.d.ts",
    "node_modules/powerbi-visuals-utils-dataviewutils/lib/index.d.ts",
    "src/settings.ts",
    "src/visual.ts",
    "src/test.d.ts"
  ]
}

3.在參數獲取方面由於API版本的差異,在3.0的版本是可以在options.dataViews對象中的屬性獲得所有的參數。但是在1.9.0的版本中,官方提供的typescript類裏面是沒有的。最終我是通過將對象轉成any類型之後再對象的原型鏈中獲取到了數據。

4.capabilities.json文件

{
	"dataRoles": [{
			"name": "Category",
			"displayName": "Category",
			"displayNameKey": "Visual_Category",
			"kind": "Grouping"
		},
		{
			"name": "Measure",
			"displayName": "Measure",
			"displayNameKey": "Visual_Values",
			"kind": "Measure"
		}
	],
	"objects": {
		"dataPoint": {
			"displayName": "Data colors",
			"properties": {
				"defaultColor": {
					"displayName": "Default color",
					"type": {
						"fill": {
							"solid": {
								"color": true
							}
						}
					}
				},
				"showAllDataPoints": {
					"displayName": "Show all",
					"type": {
						"bool": true
					}
				},
				"fill": {
					"displayName": "Fill",
					"type": {
						"fill": {
							"solid": {
								"color": true
							}
						}
					}
				},
				"fillRule": {
					"displayName": "Color saturation",
					"type": {
						"fill": {}
					}
				},
				"fontSize": {
					"displayName": "Text Size",
					"type": {
						"formatting": {
							"fontSize": true
						}
					}
				}
			}
		}
	},
	"dataViewMappings": [{
		"matrix": {
			"rows": {
				"for": {
					"in": "Category"
				}
			},
			"values": {
				"select": [{
					"for": {
						"in": "Measure"
					}
				}]
			}
		}
	}]
}

5.visual.ts文件

/*
 *  Power BI Visual CLI
 *
 *  Copyright (c) Microsoft Corporation
 *  All rights reserved.
 *  MIT License
 *
 *  Permission is hereby granted, free of charge, to any person obtaining a copy
 *  of this software and associated documentation files (the ""Software""), to deal
 *  in the Software without restriction, including without limitation the rights
 *  to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 *  copies of the Software, and to permit persons to whom the Software is
 *  furnished to do so, subject to the following conditions:
 *
 *  The above copyright notice and this permission notice shall be included in
 *  all copies or substantial portions of the Software.
 *
 *  THE SOFTWARE IS PROVIDED *AS IS*, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 *  IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 *  FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 *  AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 *  LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 *  OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 *  THE SOFTWARE.
 */

module powerbi.extensibility.visual {
  "use strict";
  export class Visual implements IVisual {
      private target: HTMLElement;
      private tips:HTMLElement;
      private settings: VisualSettings;
      private echarts:any;
      private myChart:any;
      private echartLoad:any;
      private rootList:Array<any>;
      private timer:any;

      constructor(options: VisualConstructorOptions) {
          this.target = options.element;
          this.echartLoad=false;
          this.onloadEchart();
          if (typeof document !== "undefined") {
          this.tips = document.createElement("p");
          this.tips.appendChild(document.createTextNode("正在加載資源..."));
          this.target.appendChild(this.tips);
        }
      }
      private onloadEchart(){
        var script=document.createElement('script');
        script.type="text/javascript";
        script.src="https://cdn.jsdelivr.net/npm/echarts/dist/echarts.min.js";
        document.body.appendChild(script);
        let that=this;
        script.onload=function(){
         that.echarts=echarts;
         that.echartLoad=true;
         that.checkJsLoad();
        }
      }

      public update(options: VisualUpdateOptions) {
        try{
          let dataView: DataView = options.dataViews[0];
          this.settings = VisualSettings.parse<VisualSettings>(dataView);

          this.settings = Visual.parseSettings(options && options.dataViews && options.dataViews[0]);
          console.log(options.dataViews[0]);
          let dataTree:any=options.dataViews[0].matrix;
          let list:Array<any>=dataTree.rows.root.children;
          clearTimeout(this.timer);
          this.timer=setTimeout(()=>{
            // debugger;
            this.rootList=[{
              value:"根目錄",
              children:list,
              level:-2
            }];
            this.readTree(this.rootList);
            this.mapTree(this.rootList);
            console.log(list);
            this.checkJsLoad();
          },300);
        }catch(e){
          this.dataError();
          console.log("錯誤",e);    
        }
      }
      private readTree(list:Array<any>){
        let that=this;
        list.forEach((item)=>{
          item.name=item.level;
          // item.value=item.level;
          if(typeof item.children !="undefined"){
            item.count=0;;
            that.readTree(item.children);
          }else{
            item.count=item.values[0].value;
          }
        });
      }
      
      private mapTree(list:Array<any>){
        let that=this;
        list.forEach((item)=>{
          if(typeof item.children !="undefined"){
            item.count=that.countNum(item);
            that.mapTree(item.children);
          }
        });
      }
      

      private countNum(node:any){
        let that=this;
        let result:Array<any>=new Array();
        getList(node.children,result);
        let count=0;
        result.forEach((item)=>{
          count+=item.count;
        });
        return count;
        function getList(list:Array<any>,result:Array<any>){
          list.forEach((item)=>{
            if(typeof item.children =="undefined"){
              result.push(item);
            }else{
              getList(item.children,result);
            }
          });
          return result;
        }
      }
      
      private e_tree(treeData:Array<any>){
        let that=this;
        that.myChart = that.echarts.init(this.target);
        var data:any = treeData;
        let width=150,height=80;
        var option = {
            series: [{
                name: '樹圖',
                type: 'tree',
                orient: 'vertical', // vertical horizontal
                symbol: 'rectangle',
                symbolSize: [width, height],
                edgeShape: 'polyline',
                initialTreeDepth: 2,
                itemStyle: {
                    normal: {
                        color: function(obj){
                          // console.log(obj)
                          if(obj.name==-2){
                            return '#67a244';
                          }
                          if(obj.name%2==0){
                            return '#b1b1b1';
                          }else{
                            return '#67a244';
                          }
                        }, //節點背景色
                        borderColor:"transparent",
                        borderWidth: 0,
                        label: {
                            show: true,
                            position: 'inside',
                            textStyle: {
                                color: '#ffffff',
                                fontSize: 15,
                                //fontWeight:  'bolder'
                            },
                          padding:0,
                        formatter:function(obj){
                          let str=[];
                            if(obj.data.name==-2){
                              str.push(`{r1|${obj.data.value}}   {r2|${(obj.data.count/1).toFixed(2)}}\n\n\n`);
                              if(obj.data.children&&obj.data.children.length>0){
                                if(obj.data.selected){
                                  str.push(`{m_open|}`);
                                }else{
                                  str.push(`{m_close|}`);
                                }
                              }
                            }else if(obj.data.name==0){
                              str.push(`{r0_1|${obj.data.value}}\n`);
                              str.push(`{r0_2|${(obj.data.count/1).toFixed(2)}}\n`);
                              if(obj.data.children&&obj.data.children.length>0){
                                if(obj.data.selected){
                                  str.push(`{m_close|}`);
                                }else{
                                  str.push(`{m_open|}`);
                                }
                              }
                            }else if(obj.data.name==1){
                              str.push(`{r0_1|${obj.data.value}}\n`);
                              str.push(`{r0_2|${(obj.data.count/1).toFixed(2)}}  {r0_3|}\n`);
                              if(obj.data.children&&obj.data.children.length>0){
                                if(obj.data.selected){
                                  str.push(`{m_close|}`);
                                }else{
                                  str.push(`{m_open|}`);
                                }
                              }
                            }else{
                              str.push(`{r0_1|${obj.data.value}}\n`);
                              if(obj.data.children&&obj.data.children.length>0){
                                str.push(`{r0_2|${(obj.data.count/1).toFixed(2)}}\n`);
                                if(obj.data.selected){
                                  str.push(`{m_close|}`);
                                }else{
                                  str.push(`{m_open|}`);
                                }
                              }else{
                                str.push(`{r0_2|${(obj.data.count/1).toFixed(2)}}\n\n`);
                              }
                            }
                            return str.join('\n');
                          },
                          rich: {
                            m_open:{
                              backgroundColor: {
                                image: '',
                              },
                              width:20,
                              height:20,
                            },
                            m_close:{
                              backgroundColor: {
                                image: ''
                              },
                              width:20,
                              height:20,
                            },
                              r1:{
                                fontSize: 18,
                                fontFamily: 'Microsoft YaHei',
                                color: '#fff',
                              },
                              r2:{
                                fontSize: 18,
                                fontFamily: 'Microsoft YaHei',
                                color: '#fff'
                              },
                              r0_1: {
                                  fontSize: 18,
                                  fontFamily: 'Microsoft YaHei',
                                  color: '#000',
                              },
                              r0_2: {
                                  fontSize: 18,
                                  fontFamily: 'Microsoft YaHei',
                                  color: '#000',
                              },
                              r0_3:{
                                backgroundColor:{
                                  image:"",
                                }
                              }
                          }
                        },
                        lineStyle: {
                            color: "#67a244",
                            curveness: 0.5,
                            width: 1,
                            type: 'solid' // 'curve'|'broken'|'solid'|'dotted'|'dashed'
                        }
                    },
                    emphasis: {
                        label: {
                            show: true
                        }
                    }
                },
                label:{
                  normal: {
                    distance :-15,
                    position: ['50%',55],
                    verticalAlign: 'middle',
                    align: 'center',
                    backgroundColor:"transparent",
                  }
                },
                data: data
            }]
        };
        that.myChart.setOption(option);
        that.myChart.resize();
        that.myChart.on("click", function(param){
          param.data.selected = !param.data.selected;
          that.myChart.resize();
        });
      }
      //錯誤檢查
      private dataError(){
        this.checkJsLoad();
      }
      //檢查第三方js加載
      private checkJsLoad(){
        let that=this;
        if(that.echartLoad){
          that.tips.innerHTML="加載完成";
          if(that.myChart)
          that.myChart.dispose();
          that.e_tree(this.rootList);
        }
      }
      private static parseSettings(dataView: DataView): VisualSettings {
          return VisualSettings.parse(dataView) as VisualSettings;
      }

      /** 
       * This function gets called for each of the objects defined in the capabilities files and allows you to select which of the 
       * objects and properties you want to expose to the users in the property pane.
       * 
       */
      public enumerateObjectInstances(options: EnumerateVisualObjectInstancesOptions): VisualObjectInstance[] | VisualObjectInstanceEnumerationObject {
          return VisualSettings.enumerateObjectInstances(this.settings || VisualSettings.getDefault(), options);
      }
  }
}

6.最終效果

參考文獻:https://docs.microsoft.com/zh-cn/power-bi/developer/visuals/custom-visual-develop-tutorial

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章