|  | /* | 
|  | * Licensed to the Apache Software Foundation (ASF) under one | 
|  | * or more contributor license agreements.  See the NOTICE file | 
|  | * distributed with this work for additional information | 
|  | * regarding copyright ownership.  The ASF licenses this file | 
|  | * to you under the Apache License, Version 2.0 (the | 
|  | * "License"); you may not use this file except in compliance | 
|  | * with the License.  You may obtain a copy of the License at | 
|  | * | 
|  | *   http://www.apache.org/licenses/LICENSE-2.0 | 
|  | * | 
|  | * Unless required by applicable law or agreed to in writing, | 
|  | * software distributed under the License is distributed on an | 
|  | * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY | 
|  | * KIND, either express or implied.  See the License for the | 
|  | * specific language governing permissions and limitations | 
|  | * under the License. | 
|  | */ | 
|  |  | 
|  | import { | 
|  | eachAfter, | 
|  | eachBefore | 
|  | } from './traversalHelper'; | 
|  | import { | 
|  | init, | 
|  | firstWalk, | 
|  | secondWalk, | 
|  | separation as sep, | 
|  | radialCoordinate, | 
|  | getViewRect, | 
|  | TreeLayoutNode | 
|  | } from './layoutHelper'; | 
|  | import GlobalModel from '../../model/Global'; | 
|  | import ExtensionAPI from '../../core/ExtensionAPI'; | 
|  | import TreeSeriesModel from './TreeSeries'; | 
|  |  | 
|  | export default function treeLayout(ecModel: GlobalModel, api: ExtensionAPI) { | 
|  | ecModel.eachSeriesByType('tree', function (seriesModel: TreeSeriesModel) { | 
|  | commonLayout(seriesModel, api); | 
|  | }); | 
|  | } | 
|  |  | 
|  | function commonLayout(seriesModel: TreeSeriesModel, api: ExtensionAPI) { | 
|  | const layoutInfo = getViewRect(seriesModel, api); | 
|  | seriesModel.layoutInfo = layoutInfo; | 
|  | const layout = seriesModel.get('layout'); | 
|  | let width = 0; | 
|  | let height = 0; | 
|  | let separation = null; | 
|  |  | 
|  | if (layout === 'radial') { | 
|  | width = 2 * Math.PI; | 
|  | height = Math.min(layoutInfo.height, layoutInfo.width) / 2; | 
|  | separation = sep(function (node1, node2) { | 
|  | return (node1.parentNode === node2.parentNode ? 1 : 2) / node1.depth; | 
|  | }); | 
|  | } | 
|  | else { | 
|  | width = layoutInfo.width; | 
|  | height = layoutInfo.height; | 
|  | separation = sep(); | 
|  | } | 
|  |  | 
|  | const virtualRoot = seriesModel.getData().tree.root as TreeLayoutNode; | 
|  | const realRoot = virtualRoot.children[0]; | 
|  |  | 
|  | if (realRoot) { | 
|  | init(virtualRoot); | 
|  | eachAfter(realRoot, firstWalk, separation); | 
|  | virtualRoot.hierNode.modifier = -realRoot.hierNode.prelim; | 
|  | eachBefore(realRoot, secondWalk); | 
|  |  | 
|  | let left = realRoot; | 
|  | let right = realRoot; | 
|  | let bottom = realRoot; | 
|  | eachBefore(realRoot, function (node: TreeLayoutNode) { | 
|  | const x = node.getLayout().x; | 
|  | if (x < left.getLayout().x) { | 
|  | left = node; | 
|  | } | 
|  | if (x > right.getLayout().x) { | 
|  | right = node; | 
|  | } | 
|  | if (node.depth > bottom.depth) { | 
|  | bottom = node; | 
|  | } | 
|  | }); | 
|  |  | 
|  | const delta = left === right ? 1 : separation(left, right) / 2; | 
|  | const tx = delta - left.getLayout().x; | 
|  | let kx = 0; | 
|  | let ky = 0; | 
|  | let coorX = 0; | 
|  | let coorY = 0; | 
|  | if (layout === 'radial') { | 
|  | kx = width / (right.getLayout().x + delta + tx); | 
|  | // here we use (node.depth - 1), bucause the real root's depth is 1 | 
|  | ky = height / ((bottom.depth - 1) || 1); | 
|  | eachBefore(realRoot, function (node) { | 
|  | coorX = (node.getLayout().x + tx) * kx; | 
|  | coorY = (node.depth - 1) * ky; | 
|  | const finalCoor = radialCoordinate(coorX, coorY); | 
|  | node.setLayout({x: finalCoor.x, y: finalCoor.y, rawX: coorX, rawY: coorY}, true); | 
|  | }); | 
|  | } | 
|  | else { | 
|  | const orient = seriesModel.getOrient(); | 
|  | if (orient === 'RL' || orient === 'LR') { | 
|  | ky = height / (right.getLayout().x + delta + tx); | 
|  | kx = width / ((bottom.depth - 1) || 1); | 
|  | eachBefore(realRoot, function (node) { | 
|  | coorY = (node.getLayout().x + tx) * ky; | 
|  | coorX = orient === 'LR' | 
|  | ? (node.depth - 1) * kx | 
|  | : width - (node.depth - 1) * kx; | 
|  | node.setLayout({x: coorX, y: coorY}, true); | 
|  | }); | 
|  | } | 
|  | else if (orient === 'TB' || orient === 'BT') { | 
|  | kx = width / (right.getLayout().x + delta + tx); | 
|  | ky = height / ((bottom.depth - 1) || 1); | 
|  | eachBefore(realRoot, function (node) { | 
|  | coorX = (node.getLayout().x + tx) * kx; | 
|  | coorY = orient === 'TB' | 
|  | ? (node.depth - 1) * ky | 
|  | : height - (node.depth - 1) * ky; | 
|  | node.setLayout({x: coorX, y: coorY}, true); | 
|  | }); | 
|  | } | 
|  | } | 
|  | } | 
|  | } |