From 8a9cb260c5e2d37850d1c6caba7115cfd214d24b Mon Sep 17 00:00:00 2001 From: yvonneyx Date: Mon, 4 Nov 2024 11:09:23 +0800 Subject: [PATCH 01/11] chore: update ts config --- site/basic/demos/circular.ts | 2 +- site/basic/demos/concentric.ts | 2 +- tsconfig.json | 5 ++++- vite.config.js | 27 ++++++++++++++++----------- 4 files changed, 22 insertions(+), 14 deletions(-) diff --git a/site/basic/demos/circular.ts b/site/basic/demos/circular.ts index 11adc43..5d033c5 100644 --- a/site/basic/demos/circular.ts +++ b/site/basic/demos/circular.ts @@ -1,7 +1,7 @@ import type { Canvas } from '@antv/g'; import { Graph } from '@antv/graphlib'; import type { GUI } from 'lil-gui'; -import { CircularLayout } from '../../../packages/layout'; +import { CircularLayout } from '@antv/layout'; import { countries } from '../data'; import { renderNodes } from '../utils'; diff --git a/site/basic/demos/concentric.ts b/site/basic/demos/concentric.ts index 9045b69..7a1e700 100644 --- a/site/basic/demos/concentric.ts +++ b/site/basic/demos/concentric.ts @@ -1,6 +1,6 @@ import type { Canvas } from '@antv/g'; import { Graph } from '@antv/graphlib'; -import { ConcentricLayout } from '../../../packages/layout'; +import { ConcentricLayout } from '@antv/layout'; import { countries } from '../data'; import { renderNodes } from '../utils'; diff --git a/tsconfig.json b/tsconfig.json index 03e2ca5..c3d049d 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -6,6 +6,9 @@ "jsx": "react", "allowJs": true, "moduleResolution": "node", - "allowSyntheticDefaultImports": true + "allowSyntheticDefaultImports": true, + "paths": { + "@antv/layout": ["./packages/layout/src/index.ts"], + } } } diff --git a/vite.config.js b/vite.config.js index 75b2abb..a57e0eb 100644 --- a/vite.config.js +++ b/vite.config.js @@ -1,32 +1,37 @@ -import { resolve } from "path"; -import { defineConfig } from "vite"; +import { resolve } from 'path'; +import { defineConfig } from 'vite'; export default defineConfig({ - root: "./site/", + root: './site/', server: { port: 8080, - open: "/", + open: '/', }, // publicDir: "../packages/layout-wasm/dist", - base: "/layout/", + base: '/layout/', + resolve: { + alias: { + '@antv/layout': './packages/layout/src/index.ts', + }, + }, build: { rollupOptions: { input: { - main: resolve(__dirname, "site/index.html"), - benchmark: resolve(__dirname, "site/benchmark/index.html"), - "3d": resolve(__dirname, "site/3d/index.html"), + main: resolve(__dirname, 'site/index.html'), + benchmark: resolve(__dirname, 'site/benchmark/index.html'), + '3d': resolve(__dirname, 'site/3d/index.html'), }, }, }, plugins: [ { - name: "isolation", + name: 'isolation', configureServer(server) { // The multithreads version of @antv/layout-wasm needs to use SharedArrayBuffer, which should be used in a secure context. // @see https://gist.github.com/mizchi/afcc5cf233c9e6943720fde4b4579a2b server.middlewares.use((_req, res, next) => { - res.setHeader("Cross-Origin-Opener-Policy", "same-origin"); - res.setHeader("Cross-Origin-Embedder-Policy", "require-corp"); + res.setHeader('Cross-Origin-Opener-Policy', 'same-origin'); + res.setHeader('Cross-Origin-Embedder-Policy', 'require-corp'); next(); }); }, From 78097fe05ce088125c0928401f61dffff4730dc1 Mon Sep 17 00:00:00 2001 From: yvonneyx Date: Mon, 4 Nov 2024 11:11:17 +0800 Subject: [PATCH 02/11] refactor: adapt antv-dagre node size --- packages/layout/src/antv-dagre.ts | 20 +++++++++----------- 1 file changed, 9 insertions(+), 11 deletions(-) diff --git a/packages/layout/src/antv-dagre.ts b/packages/layout/src/antv-dagre.ts index e3cc53f..a5be438 100644 --- a/packages/layout/src/antv-dagre.ts +++ b/packages/layout/src/antv-dagre.ts @@ -14,7 +14,9 @@ import type { Point, PointTuple, } from './types'; -import { cloneFormatData, formatNodeSize, formatNumberFn } from './util'; +import { cloneFormatData, formatNumberFn } from './util'; +import type { Size } from './util/size'; +import { parseSize } from './util/size'; /** * 层次/流程图布局的配置项 @@ -75,7 +77,7 @@ export interface AntVDagreLayoutOptions { * Used for collision detection when nodes overlap * @defaultValue undefined */ - nodeSize?: number | number[] | ((nodeData: Node) => number); + nodeSize?: Size | ((nodeData: Node) => Size); /** * 节点间距(px) * @@ -275,7 +277,6 @@ export class AntVDagreLayout implements Layout { // focusNode, preset, } = mergedOptions; - const g = new Graph({ tree: [], }); @@ -288,21 +289,18 @@ export class AntVDagreLayout implements Layout { horisep = ranksepfunc; vertisep = nodesepfunc; } - const nodeSizeFunc = formatNodeSize(nodeSize, undefined); - // copy graph to g const nodes: Node[] = graph.getAllNodes(); const edges: Edge[] = graph.getAllEdges(); nodes.forEach((node) => { - const size = nodeSizeFunc(node); + const size = parseSize( + typeof nodeSize === 'function' ? nodeSize(node) : nodeSize, + ); const verti = vertisep(node); const hori = horisep(node); - // FIXME: support 2 dimensions? - // const width = size[0] + 2 * hori; - // const height = size[1] + 2 * verti; - const width = size + 2 * hori; - const height = size + 2 * verti; + const width = size[0] + 2 * hori; + const height = size[1] + 2 * verti; const layer = node.data.layer; if (isNumber(layer)) { // 如果有layer属性,加入到node的label中 From cfc5e51a3a78d4551e5532bbe5c392d706087934 Mon Sep 17 00:00:00 2001 From: yvonneyx Date: Mon, 4 Nov 2024 11:12:54 +0800 Subject: [PATCH 03/11] refactor: adapt circular node size --- packages/layout/src/circular.ts | 2 +- packages/layout/src/util/function.ts | 15 ++++++++------- 2 files changed, 9 insertions(+), 8 deletions(-) diff --git a/packages/layout/src/circular.ts b/packages/layout/src/circular.ts index 0972e68..4fa255a 100644 --- a/packages/layout/src/circular.ts +++ b/packages/layout/src/circular.ts @@ -100,7 +100,7 @@ export class CircularLayout implements Layout { let { radius, startRadius, endRadius } = mergedOptions; if (paramNodeSpacing) { const nodeSpacing: Function = formatNumberFn(10, paramNodeSpacing); - const nodeSize: Function = formatSizeFn(10, paramNodeSize); + const nodeSize: Function = formatSizeFn(10, paramNodeSize, true); let maxNodeSize = -Infinity; nodes.forEach((node) => { const nSize = nodeSize(node); diff --git a/packages/layout/src/util/function.ts b/packages/layout/src/util/function.ts index 78747f7..e13cfad 100644 --- a/packages/layout/src/util/function.ts +++ b/packages/layout/src/util/function.ts @@ -1,5 +1,6 @@ import { isFunction, isNumber, isObject } from '@antv/util'; import { Node } from '../types'; +import type { Size } from './size'; /** * Format value with multiple types into a function returns number. @@ -45,11 +46,11 @@ export function formatSizeFn( return (d) => { const { size } = d.data || {}; if (size) { - if (Array.isArray(size)) { - return size[0] > size[1] ? size[0] : size[1]; - } - if (isObject<{ width: number; height: number }>(size)) { - return size.width > size.height ? size.width : size.height; + if (resultIsNumber) { + if (Array.isArray(size)) return Math.max(...size); + if (isObject<{ width: number; height: number }>(size)) { + return Math.max(size.width, size.height) || defaultValue; + } } return size; } @@ -90,9 +91,9 @@ export function formatSizeFn( * @returns */ export const formatNodeSize = ( - nodeSize: number | number[] | ((nodeData: Node) => number) | undefined, + nodeSize: Size | ((nodeData: Node) => Size) | undefined, nodeSpacing: number | Function | undefined, -): ((nodeData: Node) => number) => { +): ((nodeData: Node) => Size) => { let nodeSizeFunc; let nodeSpacingFunc: Function; if (isNumber(nodeSpacing)) { From 317e92c6c8cd7b2b673fa1477ce5d9f9bb054ed8 Mon Sep 17 00:00:00 2001 From: yvonneyx Date: Mon, 4 Nov 2024 11:29:38 +0800 Subject: [PATCH 04/11] refactor: adapt grid node size --- packages/layout/src/grid.ts | 20 ++++--------- packages/layout/src/util/function.ts | 43 ++++++++++++---------------- 2 files changed, 24 insertions(+), 39 deletions(-) diff --git a/packages/layout/src/grid.ts b/packages/layout/src/grid.ts index 5da9034..4853a72 100644 --- a/packages/layout/src/grid.ts +++ b/packages/layout/src/grid.ts @@ -9,8 +9,9 @@ import type { OutNode, PointTuple, } from './types'; -import { cloneFormatData, formatNumberFn, formatSizeFn, isArray } from './util'; +import { cloneFormatData, formatNumberFn, formatSizeFn } from './util'; import { handleSingleNodeGraph } from './util/common'; +import { parseSize } from './util/size'; type RowsAndCols = { rows: number; @@ -46,7 +47,7 @@ const DEFAULTS_LAYOUT_OPTIONS: Partial = { /** * 网格布局 - * + * * Grid layout */ export class GridLayout implements Layout { @@ -69,7 +70,7 @@ export class GridLayout implements Layout { * To directly assign the positions to the nodes. */ async assign(graph: Graph, options?: GridLayoutOptions) { - await this.genericGridLayout(true, graph, options); + await this.genericGridLayout(true, graph, options); } private async genericGridLayout( @@ -219,18 +220,7 @@ export class GridLayout implements Layout { } const oNode = graph.getNode(node.id); - const res = nodeSize(oNode) || 30; - - let nodeW; - let nodeH; - - if (isArray(res)) { - nodeW = res[0]; - nodeH = res[1]; - } else { - nodeW = res; - nodeH = res; - } + const [nodeW, nodeH] = parseSize(nodeSize(oNode) || 30); const p = nodeSpacing !== undefined ? nodeSpacing(node) : preventOverlapPadding; diff --git a/packages/layout/src/util/function.ts b/packages/layout/src/util/function.ts index e13cfad..38fc745 100644 --- a/packages/layout/src/util/function.ts +++ b/packages/layout/src/util/function.ts @@ -35,49 +35,44 @@ export function formatNumberFn( export function formatSizeFn( defaultValue: number, value?: - | number - | number[] + | Size | { width: number; height: number } - | ((d?: T) => number) + | ((d?: T) => Size) | undefined, resultIsNumber: boolean = true, -): (d: T) => number | number[] { +): (d: T) => Size { if (!value && value !== 0) { return (d) => { const { size } = d.data || {}; if (size) { - if (resultIsNumber) { - if (Array.isArray(size)) return Math.max(...size); - if (isObject<{ width: number; height: number }>(size)) { - return Math.max(size.width, size.height) || defaultValue; - } + if (Array.isArray(size)) + return resultIsNumber ? Math.max(...size) || defaultValue : size; + if ( + isObject<{ width: number; height: number }>(size) && + size.width && + size.height + ) { + return resultIsNumber + ? Math.max(size.width, size.height) || defaultValue + : [size.width, size.height]; } return size; } return defaultValue; }; } - if (isFunction(value)) { - return value; - } - if (isNumber(value)) { - return () => value; - } + if (isFunction(value)) return value; + if (isNumber(value)) return () => value; if (Array.isArray(value)) { return () => { - if (resultIsNumber) { - const max = Math.max(...value); - return isNaN(max) ? defaultValue : max; - } + if (resultIsNumber) return Math.max(...value) || defaultValue; return value; }; } - if (isObject(value)) { + if (isObject(value) && value.width && value.height) { return () => { - if (resultIsNumber) { - const max = Math.max(value.width, value.height); - return isNaN(max) ? defaultValue : max; - } + if (resultIsNumber) + return Math.max(value.width, value.height) || defaultValue; return [value.width, value.height]; }; } From d14d315aa072b735f6e77425f61ae8afae82a385 Mon Sep 17 00:00:00 2001 From: yvonneyx Date: Mon, 4 Nov 2024 11:42:57 +0800 Subject: [PATCH 05/11] refactor: adapt radial layout node size --- packages/layout/src/radial/index.ts | 6 +-- packages/layout/src/types.ts | 23 +++++------ packages/layout/src/util/function.ts | 59 +++++++++++----------------- 3 files changed, 39 insertions(+), 49 deletions(-) diff --git a/packages/layout/src/radial/index.ts b/packages/layout/src/radial/index.ts index 237c4f5..e9561a2 100644 --- a/packages/layout/src/radial/index.ts +++ b/packages/layout/src/radial/index.ts @@ -13,7 +13,7 @@ import type { import { cloneFormatData, floydWarshall, - formatNodeSize, + formatNodeSizeToNumber, getAdjMatrix, getEuclideanDistance, } from '../util'; @@ -37,7 +37,7 @@ const DEFAULTS_LAYOUT_OPTIONS: Partial = { /** * 径向布局 - * + * * Radial layout */ export class RadialLayout implements Layout { @@ -193,7 +193,7 @@ export class RadialLayout implements Layout { let nodeSizeFunc; // stagger the overlapped nodes if (preventOverlap) { - nodeSizeFunc = formatNodeSize(nodeSize, nodeSpacing); + nodeSizeFunc = formatNodeSizeToNumber(nodeSize, nodeSpacing); const nonoverlapForceParams: RadialNonoverlapForceOptions = { nodes, nodeSizeFunc, diff --git a/packages/layout/src/types.ts b/packages/layout/src/types.ts index c6ea6af..5743ef3 100644 --- a/packages/layout/src/types.ts +++ b/packages/layout/src/types.ts @@ -1,4 +1,5 @@ import { Edge as IEdge, Graph as IGraph, Node as INode } from '@antv/graphlib'; +import type { Size } from './util/size'; /** * 节点数据 @@ -240,7 +241,7 @@ export interface CircularLayoutOptions { * * Node size (diameter). Used for collision detection when nodes overlap */ - nodeSize?: number | number[] | ((nodeData: Node) => number); + nodeSize?: Size | ((nodeData: Node) => Size); } export interface GridLayoutOptions { @@ -290,7 +291,7 @@ export interface GridLayoutOptions { * * Node size (diameter). Used for collision detection when nodes overlap */ - nodeSize?: number | number[] | ((nodeData: Node) => number); + nodeSize?: Size | ((nodeData: Node) => Size); /** * 避免重叠时节点的间距 padding。preventOverlap 为 true 时生效 * @@ -416,7 +417,7 @@ export interface ConcentricLayoutOptions { * * Node size (diameter). Used for collision detection when preventing node overlap */ - nodeSize?: number | PointTuple | ((nodeData: Node) => number); + nodeSize?: Size | ((nodeData: Node) => Size); /** * 第一个节点与最后一个节点之间的弧度差 * @@ -564,14 +565,14 @@ export interface RadialLayoutOptions { * * Node size (diameter). Used for collision detection when preventing node overlap */ - nodeSize?: number | number[] | ((nodeData: Node) => number); + nodeSize?: Size | ((nodeData: Node) => Size); /** * preventOverlap 为 true 时生效, 防止重叠时节点边缘间距的最小值。可以是回调函数, 为不同节点设置不同的最小间距 * * Effective when preventOverlap is true. The minimum edge spacing when preventing node overlap. It can be a callback function, and set different minimum spacing for different nodes * @defaultValue 10 */ - nodeSpacing?: number | Function; + nodeSpacing?: number | ((nodeData: Node) => number); /** * 防止重叠步骤的最大迭代次数 * @@ -674,13 +675,13 @@ export interface D3ForceLayoutOptions { * * Node size (diameter). Used for collision detection when preventing node overlapping */ - nodeSize?: number | number[] | ((node?: Node) => number); + nodeSize?: Size | ((node?: Node) => Size); /** * preventOverlap 为 true 时生效, 防止重叠时节点边缘间距的最小值。可以是回调函数, 为不同节点设置不同的最小间距 * * It takes effect when preventOverlap is true. The minimum spacing of the node edge when preventing node overlapping. It can be a callback function, and set different minimum spacing for different nodes */ - nodeSpacing?: number | number[] | ((node?: Node) => number); + nodeSpacing?: number | ((node?: Node) => number); /** * 当前的迭代收敛阈值 * @@ -850,7 +851,7 @@ export interface ComboCombinedLayoutOptions { * @example * ```ts * import { ForceLayout } from '@antv/layout'; - * + * * outerLayout: new ForceLayout({ * gravity: 1, * factor: 2, @@ -870,7 +871,7 @@ export interface ComboCombinedLayoutOptions { * @example * ```ts * import { ConcentricLayout } from '@antv/layout'; - * + * * innerLayout: new ConcentricLayout({ * sortBy: 'id' * }); @@ -995,7 +996,7 @@ export interface ForceLayoutOptions extends CommonForceLayoutOptions { * * The size of the node (diameter). Used for collision detection when preventing node overlap */ - nodeSize?: number | number[] | ((d?: Node) => number); + nodeSize?: Size | ((d?: Node) => Size); /** * preventOverlap 为 true 时生效, 防止重叠时节点边缘间距的最小值。可以是回调函数, 为不同节点设置不同的最小间距 * @@ -1236,7 +1237,7 @@ export interface ForceAtlas2LayoutOptions extends CommonForceLayoutOptions { * * Node size (diameter). Used for collision detection when preventing node overlap */ - nodeSize?: number | number[] | ((node?: Node) => number); + nodeSize?: Size | ((node?: Node) => Size); /** * 每一次迭代的回调函数 * diff --git a/packages/layout/src/util/function.ts b/packages/layout/src/util/function.ts index 38fc745..f78ca0c 100644 --- a/packages/layout/src/util/function.ts +++ b/packages/layout/src/util/function.ts @@ -1,6 +1,6 @@ import { isFunction, isNumber, isObject } from '@antv/util'; import { Node } from '../types'; -import type { Size } from './size'; +import { parseSize, type Size } from './size'; /** * Format value with multiple types into a function returns number. @@ -85,50 +85,39 @@ export function formatSizeFn( * @param nodeSpacing * @returns */ -export const formatNodeSize = ( - nodeSize: Size | ((nodeData: Node) => Size) | undefined, - nodeSpacing: number | Function | undefined, -): ((nodeData: Node) => Size) => { +export const formatNodeSizeToNumber = ( + nodeSize: Size | ((node: Node) => Size) | undefined, + nodeSpacing: number | ((node: Node) => number) | undefined, +): ((node: Node) => number) => { let nodeSizeFunc; - let nodeSpacingFunc: Function; - if (isNumber(nodeSpacing)) { - nodeSpacingFunc = () => nodeSpacing; - } else if (isFunction(nodeSpacing)) { - nodeSpacingFunc = nodeSpacing; - } else { - nodeSpacingFunc = () => 0; - } + const nodeSpacingFunc = + typeof nodeSpacing === 'function' ? nodeSpacing : () => nodeSpacing || 0; if (!nodeSize) { nodeSizeFunc = (d: Node) => { - if (d.data?.bboxSize) { - return ( - Math.max(d.data.bboxSize[0], d.data.bboxSize[1]) + nodeSpacingFunc(d) - ); - } + if (d.data?.bboxSize) return d.data?.bboxSize; if (d.data?.size) { - if (Array.isArray(d.data.size)) { - return Math.max(d.data.size[0], d.data.size[1]) + nodeSpacingFunc(d); - } const dataSize = d.data.size; - if (isObject<{ width: number; height: number }>(dataSize)) { - const res = - dataSize.width > dataSize.height ? dataSize.width : dataSize.height; - return res + nodeSpacingFunc(d); - } - return dataSize + nodeSpacingFunc(d); + if (Array.isArray(dataSize)) return dataSize; + if (isObject<{ width: number; height: number }>(dataSize)) + return [dataSize.width, dataSize.height]; + return dataSize; } - return 10 + nodeSpacingFunc(d); + return 10; }; } else if (Array.isArray(nodeSize)) { - nodeSizeFunc = (d: Node) => { - const res = nodeSize[0] > nodeSize[1] ? nodeSize[0] : nodeSize[1]; - return res + nodeSpacingFunc(d); - }; + nodeSizeFunc = (d: Node) => nodeSize; } else if (isFunction(nodeSize)) { - nodeSizeFunc = nodeSize as (nodeData: Node) => number; + nodeSizeFunc = nodeSize; } else { - nodeSizeFunc = (d: Node) => nodeSize + nodeSpacingFunc(d); + nodeSizeFunc = (d: Node) => nodeSize; } - return nodeSizeFunc; + + const func = (d: Node) => { + const nodeSize = nodeSizeFunc(d); + const nodeSpacing = nodeSpacingFunc(d); + return Math.max(...parseSize(nodeSize)) + nodeSpacing; + }; + + return func; }; From 575d072033f5f07225a3dbdb39a8398b673820b5 Mon Sep 17 00:00:00 2001 From: yvonneyx Date: Mon, 4 Nov 2024 13:47:45 +0800 Subject: [PATCH 06/11] refactor: adapt force-atlas2 node size --- packages/layout/src/force-atlas2/index.ts | 40 +++++------------------ packages/layout/src/util/function.ts | 3 +- 2 files changed, 11 insertions(+), 32 deletions(-) diff --git a/packages/layout/src/force-atlas2/index.ts b/packages/layout/src/force-atlas2/index.ts index 249820f..56cf585 100644 --- a/packages/layout/src/force-atlas2/index.ts +++ b/packages/layout/src/force-atlas2/index.ts @@ -1,5 +1,5 @@ import { Graph as GGraph } from '@antv/graphlib'; -import { isFunction, isNumber, isObject } from '@antv/util'; +import { isNumber } from '@antv/util'; import type { Edge, EdgeData, @@ -14,8 +14,9 @@ import type { OutNodeData, PointTuple, } from '../types'; -import { cloneFormatData, isArray } from '../util'; +import { cloneFormatData, formatNodeSizeToNumber } from '../util'; import { handleSingleNodeGraph } from '../util/common'; +import type { Size } from '../util/size'; import Body from './body'; import Quad from './quad'; import QuadTree from './quad-tree'; @@ -57,7 +58,7 @@ type CalcGraph = GGraph; /** * Atlas2 力导向布局 - * + * * Force Atlas 2 layout */ export class ForceAtlas2Layout implements Layout { @@ -82,7 +83,7 @@ export class ForceAtlas2Layout implements Layout { * To directly assign the positions to the nodes. */ async assign(graph: Graph, options?: ForceAtlas2LayoutOptions) { - await this.genericForceAtlas2Layout(true, graph, options); + await this.genericForceAtlas2Layout(true, graph, options); } private async genericForceAtlas2Layout( @@ -122,7 +123,7 @@ export class ForceAtlas2Layout implements Layout { nodes: calcNodes, edges: calcEdges, }); - const sizes: SizeMap = this.getSizes(calcGraph, graph, nodeSize); + const sizes: SizeMap = this.getSizes(calcGraph, nodeSize); this.run(calcGraph, graph, maxIteration, sizes, assign, mergedOptions); @@ -164,41 +165,18 @@ export class ForceAtlas2Layout implements Layout { * Init the node positions if there is no initial positions. * And pre-calculate the size (max of width and height) for each node. * @param calcGraph graph for calculation - * @param graph origin graph * @param nodeSize node size config from layout options * @returns {SizeMap} node'id mapped to max of its width and height */ private getSizes( calcGraph: CalcGraph, - graph: Graph, - nodeSize?: number | number[] | ((d?: Node) => number), + nodeSize?: Size | ((d?: Node) => Size), ): SizeMap { const nodes = calcGraph.getAllNodes(); const sizes: SizeMap = {}; for (let i = 0; i < nodes.length; i += 1) { - const { id, data } = nodes[i]; - sizes[id] = 10; - if (isNumber(data.size)) { - sizes[id] = data.size; - } else if (isArray(data.size)) { - if (!isNaN(data.size[0])) sizes[id] = Math.max(data.size[0]); - if (!isNaN(data.size[1])) sizes[id] = Math.max(data.size[1]); - } else if (isObject(data.size)) { - // @ts-ignore - sizes[id] = Math.max(data.size.width, data.size.height); - } else if (isFunction(nodeSize)) { - const originNode = graph.getNode(id); - const size = nodeSize(originNode); - if (isArray(size)) { - sizes[id] = Math.max(...size); - } else { - sizes[id] = size; - } - } else if (isArray(nodeSize)) { - sizes[id] = Math.max(...nodeSize); - } else if (isNumber(nodeSize)) { - sizes[id] = nodeSize; - } + const node = nodes[i]; + sizes[node.id] = formatNodeSizeToNumber(nodeSize, undefined)(node); } return sizes; } diff --git a/packages/layout/src/util/function.ts b/packages/layout/src/util/function.ts index f78ca0c..e93071b 100644 --- a/packages/layout/src/util/function.ts +++ b/packages/layout/src/util/function.ts @@ -88,6 +88,7 @@ export function formatSizeFn( export const formatNodeSizeToNumber = ( nodeSize: Size | ((node: Node) => Size) | undefined, nodeSpacing: number | ((node: Node) => number) | undefined, + defaultNodeSize: number = 10, ): ((node: Node) => number) => { let nodeSizeFunc; const nodeSpacingFunc = @@ -103,7 +104,7 @@ export const formatNodeSizeToNumber = ( return [dataSize.width, dataSize.height]; return dataSize; } - return 10; + return defaultNodeSize; }; } else if (Array.isArray(nodeSize)) { nodeSizeFunc = (d: Node) => nodeSize; From a9af1af126d9ca47731bcb89db105c4eff91906c Mon Sep 17 00:00:00 2001 From: yvonneyx Date: Mon, 4 Nov 2024 13:50:43 +0800 Subject: [PATCH 07/11] refactor: adapt force node size --- packages/layout/src/force/index.ts | 47 ++++++------------------------ 1 file changed, 9 insertions(+), 38 deletions(-) diff --git a/packages/layout/src/force/index.ts b/packages/layout/src/force/index.ts index 839ae4e..6b51b3b 100644 --- a/packages/layout/src/force/index.ts +++ b/packages/layout/src/force/index.ts @@ -1,5 +1,5 @@ import { Graph as IGraph } from '@antv/graphlib'; -import { isFunction, isNumber, isObject } from '@antv/util'; +import { isNumber } from '@antv/util'; import type { Edge, ForceLayoutOptions, @@ -11,7 +11,7 @@ import type { OutNode, Point, } from '../types'; -import { formatNumberFn, isArray } from '../util'; +import { formatNodeSizeToNumber, formatNumberFn } from '../util'; import { forceNBody } from './force-n-body'; import { CalcEdge, @@ -42,7 +42,7 @@ const DEFAULTS_LAYOUT_OPTIONS: Partial = { /** * 力导向布局 - * + * * Force-directed layout */ export class ForceLayout implements LayoutWithIterations { @@ -85,7 +85,7 @@ export class ForceLayout implements LayoutWithIterations { * To directly assign the positions to the nodes. */ async assign(graph: Graph, options?: ForceLayoutOptions) { - await this.genericForceLayout(true, graph, options); + await this.genericForceLayout(true, graph, options); } /** @@ -292,12 +292,7 @@ export class ForceLayout implements LayoutWithIterations { graph: Graph, ): FormatedOptions { const formattedOptions = { ...options } as FormatedOptions; - const { - width: propsWidth, - height: propsHeight, - getMass, - nodeSize, - } = options; + const { width: propsWidth, height: propsHeight, getMass } = options; // === formating width, height, and center ===== formattedOptions.width = @@ -326,34 +321,10 @@ export class ForceLayout implements LayoutWithIterations { } // === formating node size ===== - - const nodeSpacingFunc = formatNumberFn(0, options.nodeSpacing); - let nodeSizeFn; - if (!nodeSize) { - nodeSizeFn = (d?: Node) => { - const { size } = d?.data || {}; - if (size) { - if (isArray(size)) { - return Math.max(size[0], size[1]) + nodeSpacingFunc(d); - } - if (isObject<{ width: number; height: number }>(size)) { - return Math.max(size.width, size.height) + nodeSpacingFunc(d); - } - return (size as number) + nodeSpacingFunc(d); - } - return 10 + nodeSpacingFunc(d); - }; - } else if (isFunction(nodeSize)) { - nodeSizeFn = (d?: Node) => (nodeSize as Function)(d) + nodeSpacingFunc(d); - } else if (isArray(nodeSize)) { - nodeSizeFn = (d?: Node) => { - const nodeSizeArr = nodeSize as [number, number]; - return Math.max(nodeSizeArr[0], nodeSizeArr[1]) + nodeSpacingFunc(d); - }; - } else { - nodeSizeFn = (d?: Node) => (nodeSize as number) + nodeSpacingFunc(d); - } - formattedOptions.nodeSize = nodeSizeFn; + formattedOptions.nodeSize = formatNodeSizeToNumber( + options.nodeSize, + options.nodeSpacing, + ); // === formating node / edge strengths ===== const linkDistanceFn = options.linkDistance From 25cee7d2414d7ecac46bf6b15e89c0ef9231f3bb Mon Sep 17 00:00:00 2001 From: yvonneyx Date: Mon, 4 Nov 2024 13:56:11 +0800 Subject: [PATCH 08/11] refactor: adapt concentric node size --- packages/layout/src/concentric.ts | 7 ++++--- packages/layout/src/util/function.ts | 2 +- 2 files changed, 5 insertions(+), 4 deletions(-) diff --git a/packages/layout/src/concentric.ts b/packages/layout/src/concentric.ts index eaad139..f3061a0 100644 --- a/packages/layout/src/concentric.ts +++ b/packages/layout/src/concentric.ts @@ -11,6 +11,7 @@ import type { } from './types'; import { cloneFormatData, isArray } from './util'; import { handleSingleNodeGraph } from './util/common'; +import { parseSize } from './util/size'; const DEFAULTS_LAYOUT_OPTIONS: Partial = { nodeSize: 30, @@ -26,7 +27,7 @@ const DEFAULTS_LAYOUT_OPTIONS: Partial = { /** * 同心圆布局 - * + * * Concentric layout */ export class ConcentricLayout implements Layout { @@ -51,7 +52,7 @@ export class ConcentricLayout implements Layout { * To directly assign the positions to the nodes. */ async assign(graph: Graph, options?: ConcentricLayoutOptions) { - await this.genericConcentricLayout(true, graph, options); + await this.genericConcentricLayout(true, graph, options); } private async genericConcentricLayout( @@ -112,7 +113,7 @@ export class ConcentricLayout implements Layout { } else if (isFunction(nodeSize)) { maxNodeSize = -Infinity; nodes.forEach((node) => { - const currentSize = nodeSize(node); + const currentSize = Math.max(...parseSize(nodeSize(node))); if (currentSize > maxNodeSize) maxNodeSize = currentSize; }); } else { diff --git a/packages/layout/src/util/function.ts b/packages/layout/src/util/function.ts index e93071b..5c7edd2 100644 --- a/packages/layout/src/util/function.ts +++ b/packages/layout/src/util/function.ts @@ -115,7 +115,7 @@ export const formatNodeSizeToNumber = ( } const func = (d: Node) => { - const nodeSize = nodeSizeFunc(d); + const nodeSize = nodeSizeFunc(d) as Size; const nodeSpacing = nodeSpacingFunc(d); return Math.max(...parseSize(nodeSize)) + nodeSpacing; }; From d23be3b485814abb084530154d8ee90fe3895da1 Mon Sep 17 00:00:00 2001 From: yvonneyx Date: Mon, 4 Nov 2024 14:04:27 +0800 Subject: [PATCH 09/11] fix: ts --- packages/layout/src/util/function.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/layout/src/util/function.ts b/packages/layout/src/util/function.ts index 5c7edd2..2e12fab 100644 --- a/packages/layout/src/util/function.ts +++ b/packages/layout/src/util/function.ts @@ -90,7 +90,7 @@ export const formatNodeSizeToNumber = ( nodeSpacing: number | ((node: Node) => number) | undefined, defaultNodeSize: number = 10, ): ((node: Node) => number) => { - let nodeSizeFunc; + let nodeSizeFunc: (node: Node) => Size; const nodeSpacingFunc = typeof nodeSpacing === 'function' ? nodeSpacing : () => nodeSpacing || 0; From ab5041132062c9ad5cb77a8203b049e84155b274 Mon Sep 17 00:00:00 2001 From: yvonneyx Date: Mon, 4 Nov 2024 14:14:08 +0800 Subject: [PATCH 10/11] chore: revert --- site/basic/demos/circular.ts | 2 +- site/basic/demos/concentric.ts | 2 +- tsconfig.json | 5 +---- vite.config.js | 5 ----- 4 files changed, 3 insertions(+), 11 deletions(-) diff --git a/site/basic/demos/circular.ts b/site/basic/demos/circular.ts index 5d033c5..11adc43 100644 --- a/site/basic/demos/circular.ts +++ b/site/basic/demos/circular.ts @@ -1,7 +1,7 @@ import type { Canvas } from '@antv/g'; import { Graph } from '@antv/graphlib'; import type { GUI } from 'lil-gui'; -import { CircularLayout } from '@antv/layout'; +import { CircularLayout } from '../../../packages/layout'; import { countries } from '../data'; import { renderNodes } from '../utils'; diff --git a/site/basic/demos/concentric.ts b/site/basic/demos/concentric.ts index 7a1e700..9045b69 100644 --- a/site/basic/demos/concentric.ts +++ b/site/basic/demos/concentric.ts @@ -1,6 +1,6 @@ import type { Canvas } from '@antv/g'; import { Graph } from '@antv/graphlib'; -import { ConcentricLayout } from '@antv/layout'; +import { ConcentricLayout } from '../../../packages/layout'; import { countries } from '../data'; import { renderNodes } from '../utils'; diff --git a/tsconfig.json b/tsconfig.json index c3d049d..03e2ca5 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -6,9 +6,6 @@ "jsx": "react", "allowJs": true, "moduleResolution": "node", - "allowSyntheticDefaultImports": true, - "paths": { - "@antv/layout": ["./packages/layout/src/index.ts"], - } + "allowSyntheticDefaultImports": true } } diff --git a/vite.config.js b/vite.config.js index a57e0eb..0b0ad65 100644 --- a/vite.config.js +++ b/vite.config.js @@ -9,11 +9,6 @@ export default defineConfig({ }, // publicDir: "../packages/layout-wasm/dist", base: '/layout/', - resolve: { - alias: { - '@antv/layout': './packages/layout/src/index.ts', - }, - }, build: { rollupOptions: { input: { From 6d26cc959649a4fc44d305592d402bae27951152 Mon Sep 17 00:00:00 2001 From: yvonneyx Date: Mon, 4 Nov 2024 14:20:43 +0800 Subject: [PATCH 11/11] fix: adapt formatSizeFn --- packages/layout/src/antv-dagre.ts | 9 +++++---- packages/layout/src/circular.ts | 2 +- 2 files changed, 6 insertions(+), 5 deletions(-) diff --git a/packages/layout/src/antv-dagre.ts b/packages/layout/src/antv-dagre.ts index a5be438..5b805ee 100644 --- a/packages/layout/src/antv-dagre.ts +++ b/packages/layout/src/antv-dagre.ts @@ -14,7 +14,7 @@ import type { Point, PointTuple, } from './types'; -import { cloneFormatData, formatNumberFn } from './util'; +import { cloneFormatData, formatNumberFn, formatSizeFn } from './util'; import type { Size } from './util/size'; import { parseSize } from './util/size'; @@ -289,14 +289,15 @@ export class AntVDagreLayout implements Layout { horisep = ranksepfunc; vertisep = nodesepfunc; } + + const nodeSizeFunc = formatSizeFn(10, nodeSize, false); + // copy graph to g const nodes: Node[] = graph.getAllNodes(); const edges: Edge[] = graph.getAllEdges(); nodes.forEach((node) => { - const size = parseSize( - typeof nodeSize === 'function' ? nodeSize(node) : nodeSize, - ); + const size = parseSize(nodeSizeFunc(node)); const verti = vertisep(node); const hori = horisep(node); const width = size[0] + 2 * hori; diff --git a/packages/layout/src/circular.ts b/packages/layout/src/circular.ts index 4fa255a..0972e68 100644 --- a/packages/layout/src/circular.ts +++ b/packages/layout/src/circular.ts @@ -100,7 +100,7 @@ export class CircularLayout implements Layout { let { radius, startRadius, endRadius } = mergedOptions; if (paramNodeSpacing) { const nodeSpacing: Function = formatNumberFn(10, paramNodeSpacing); - const nodeSize: Function = formatSizeFn(10, paramNodeSize, true); + const nodeSize: Function = formatSizeFn(10, paramNodeSize); let maxNodeSize = -Infinity; nodes.forEach((node) => { const nSize = nodeSize(node);