optimize(projects): optimize example antv

This commit is contained in:
Soybean 2024-10-24 23:46:11 +08:00
parent 0c0d606ae5
commit a2178d3405
6 changed files with 148 additions and 126 deletions

View File

@ -1,7 +1,7 @@
import type { CustomGraphData } from './modules/antv-g6-flow'; import type { CustomGraphData } from './modules/types';
// 日期可以自己随便设置,就是字符串展示,也可以修改为业务需要的字段 // 日期可以自己随便设置,就是字符串展示,也可以修改为业务需要的字段
export function getFlowDatas(): CustomGraphData { export function getFlowData(): CustomGraphData {
return { return {
nodes: [ nodes: [
{ {

View File

@ -1,55 +1,63 @@
<script setup lang="tsx"> <script setup lang="tsx">
import { computed, onMounted, ref } from 'vue'; import { computed, onMounted, ref, useTemplateRef } from 'vue';
import type { IPointerEvent } from '@antv/g6'; import type { Ref } from 'vue';
import type { CustomBehaviorOption, IPointerEvent } from '@antv/g6';
import AntvFlow from './modules/antv-flow.vue'; import AntvFlow from './modules/antv-flow.vue';
import { getFlowDatas } from './data'; import type { CustomGraphData } from './modules/types';
import { getFlowData } from './data';
const antvFlowRef = ref(); const antvFlowRef = useTemplateRef('antvFlowRef');
const flowDatas = ref<any>({});
const seletedNode = ref<string | undefined>('N2'); const flowData = ref({
const behaviors = [ nodes: [],
edges: []
}) as Ref<CustomGraphData>;
const selectedNode = ref<string | undefined>('N2');
const behaviors: CustomBehaviorOption[] = [
{ {
type: 'click-select', type: 'click-select',
enable: (event: IPointerEvent) => event.targetType === 'node', enable: (event: IPointerEvent) => event.targetType === 'node',
onClick: (event: IPointerEvent) => { onClick: (event: IPointerEvent) => {
const node = event.target as any; const node = event.target as unknown as HTMLElement;
const nodeData = flowDatas.value.nodes?.find((item: any) => item.id === node.id); const nodeData = flowData.value.nodes.find(item => item.id === node.id);
seletedNode.value = nodeData?.id; selectedNode.value = nodeData?.id;
window.$message?.success(`选中节点:[${node?.id}]${nodeData?.name}`); window.$message?.success(`选中节点:[${node.id}]${nodeData?.name}`);
} }
} }
]; ];
const hasNodeN = computed(() => flowDatas.value.nodes?.some((node: any) => node.id === 'NN')); const hasNodeN = computed(() => flowData.value.nodes.some(node => node.id === 'NN'));
function addNode() { function addNode() {
const { nodes, edges } = flowDatas.value; const { nodes, edges } = flowData.value;
nodes.push({ id: 'NN', name: 'New node', status: 'NOT_STARTED' }); nodes.push({ id: 'NN', name: 'New node', status: 'NOT_STARTED' });
edges.push({ id: 'EN', source: 'N5', target: 'NN' }); edges.push({ id: 'EN', source: 'N5', target: 'NN' });
flowDatas.value = { nodes, edges }; flowData.value = { nodes, edges };
} }
function removeNode(id: string) { function removeNode(id: string) {
const { nodes, edges } = flowDatas.value; const { nodes, edges } = flowData.value;
// nodeNXedge // nodeNXedge
flowDatas.value = { flowData.value = {
nodes: nodes.filter((node: any) => node.id !== id), nodes: nodes.filter(node => node.id !== id),
edges: edges.filter((edge: any) => edge.source !== id && edge.target !== id) edges: edges.filter(edge => edge.source !== id && edge.target !== id)
}; };
} }
onMounted(() => { onMounted(() => {
flowDatas.value = getFlowDatas(); flowData.value = getFlowData();
}); });
</script> </script>
<template> <template>
<div class="h-full"> <div class="h-full">
<NCard title="AntV G6 Next" :bordered="false" class="h-full card-wrapper"> <NCard title="AntV G6 Next" :bordered="false" class="h-full card-wrapper">
<AntvFlow ref="antvFlowRef" :data="flowDatas" :selected="seletedNode" :behaviors="behaviors" /> <AntvFlow ref="antvFlowRef" :data="flowData" :selected="selectedNode" :behaviors="behaviors" />
<NDivider /> <NDivider />
<NFlex> <NFlex>
<NButton @click="seletedNode = 'N5'">选中节点N5(需要自行处理选中事件不会触发元素点击)</NButton> <NButton @click="selectedNode = 'N5'">选中节点N5(需要自行处理选中事件不会触发元素点击)</NButton>
<NButton v-if="!hasNodeN" @click="addNode">添加节点并与Node5连线</NButton> <NButton v-if="!hasNodeN" @click="addNode">添加节点并与Node5连线</NButton>
<NButton v-else @click="() => removeNode('NN')">删除新添加的节点</NButton> <NButton v-else @click="() => removeNode('NN')">删除新添加的节点</NButton>
<NButton @click="() => removeNode('NX')">删除NodeX</NButton> <NButton @click="() => removeNode('NX')">删除NodeX</NButton>

View File

@ -1,15 +1,14 @@
<script setup lang="tsx"> <script setup lang="tsx">
import { shallowRef, watch } from 'vue'; import { shallowRef, useTemplateRef, watch } from 'vue';
import { vResizeObserver } from '@vueuse/components';
import { useDebounceFn } from '@vueuse/core'; import { useDebounceFn } from '@vueuse/core';
import type { CustomBehaviorOption } from '@antv/g6'; import { vResizeObserver } from '@vueuse/components';
import type { CustomGraphData } from './antv-g6-flow'; import type { CustomBehaviorOption, Graph } from '@antv/g6';
import { useAntFlow } from './antv-g6-flow'; import { useAntFlow } from './antv-g6-flow';
import { nodeStatus } from './status'; import { nodeStatus } from './status';
import type { CustomGraphData } from './types';
defineOptions({ defineOptions({
name: 'AntvFLow', name: 'AntvFLow'
inheritAttrs: false
}); });
interface Props { interface Props {
@ -22,8 +21,8 @@ interface Props {
const props = defineProps<Props>(); const props = defineProps<Props>();
const containerRef = shallowRef(); const containerRef = useTemplateRef('containerRef');
const graphRef = shallowRef(); const graphRef = shallowRef<Graph | null>(null);
// //
const onContainerResize = useDebounceFn(() => { const onContainerResize = useDebounceFn(() => {
@ -54,6 +53,24 @@ async function selectNode() {
} }
} }
function zoomOut() {
graphRef.value?.zoomBy(0.9);
}
function zoomIn() {
graphRef.value?.zoomBy(1.1);
}
function resetZoom() {
graphRef.value?.zoomTo(1);
graphRef.value?.fitCenter();
}
function fitZoom() {
graphRef.value?.fitView();
graphRef.value?.fitCenter();
}
watch( watch(
[() => props.data, () => props.selected], [() => props.data, () => props.selected],
() => { () => {
@ -70,30 +87,16 @@ defineExpose({ selectNode, graph: graphRef });
<!-- 画布操作栏 --> <!-- 画布操作栏 -->
<div class="absolute left-0 right-0 z-1 flex items-center items-stretch justify-between"> <div class="absolute left-0 right-0 z-1 flex items-center items-stretch justify-between">
<NButtonGroup size="small" class="bg-white!"> <NButtonGroup size="small" class="bg-white!">
<NButton @click="graphRef.zoomBy(0.9)"> <NButton @click="zoomOut">
<icon-mingcute:zoom-out-line /> <icon-mingcute:zoom-out-line />
</NButton> </NButton>
<NButton @click="graphRef.zoomBy(1.1)"> <NButton @click="zoomIn">
<icon-mingcute:zoom-in-line /> <icon-mingcute:zoom-in-line />
</NButton> </NButton>
<NButton <NButton @click="resetZoom">
@click="
() => {
graphRef.zoomTo(1);
graphRef.fitCenter();
}
"
>
<icon-icon-park-outline:equal-ratio /> <icon-icon-park-outline:equal-ratio />
</NButton> </NButton>
<NButton <NButton @click="fitZoom">
@click="
() => {
graphRef.fitView();
graphRef.fitCenter();
}
"
>
<icon-gg:ratio /> <icon-gg:ratio />
</NButton> </NButton>
</NButtonGroup> </NButtonGroup>
@ -107,13 +110,13 @@ defineExpose({ selectNode, graph: graphRef });
<div class="flex-col gap-8px"> <div class="flex-col gap-8px">
<div span="2" class="text-12px font-bold">节点图例</div> <div span="2" class="text-12px font-bold">节点图例</div>
<NGrid :cols="2" :y-gap="8" class="w-180px!"> <NGrid :cols="2" :y-gap="8" class="w-180px!">
<NGi v-for="[key, value] in Object.entries(nodeStatus)" :key="key" class="flex-center"> <NGi v-for="(config, status) in nodeStatus" :key="status" class="flex-center">
<NTag size="small" round :bordered="false"> <NTag size="small" round :bordered="false">
<template #icon> <template #icon>
<icon-f7:flag-circle-fill v-if="key === 'MILESTONE'" :style="{ color: value.color }" /> <icon-f7:flag-circle-fill v-if="status === 'MILESTONE'" :style="{ color: config.color }" />
<icon-f7:circle-fill v-else :style="{ color: value.color }" /> <icon-f7:circle-fill v-else :style="{ color: config.color }" />
</template> </template>
{{ value.type }} {{ config.type }}
</NTag> </NTag>
</NGi> </NGi>
</NGrid> </NGrid>

View File

@ -1,49 +1,35 @@
import type { CustomBehaviorOption, EdgeData, GraphData, NodeData } from '@antv/g6';
import { Graph } from '@antv/g6'; import { Graph } from '@antv/g6';
import type { CustomBehaviorOption, IPointerEvent } from '@antv/g6';
import type { Canvas } from '@antv/g6/lib/runtime/canvas'; import type { Canvas } from '@antv/g6/lib/runtime/canvas';
import { useThemeStore } from '@/store/modules/theme'; import { useThemeStore } from '@/store/modules/theme';
import { getNodeIcon, nodeStatus } from './status'; import { getNodeIcon, nodeStatus } from './status';
import type { CustomEdgeData, CustomGraphData, CustomNodeData } from './types';
const baseColor = 'rgb(158 163 171)'; interface AntFlowConfig {
export interface CustomNodeData extends NodeData {
isDelayed?: boolean;
isDeleted?: boolean;
milestone?: boolean;
}
export interface CustomEdgeData extends EdgeData {
isDelayed?: boolean;
isDeleted?: boolean;
}
export interface CustomGraphData extends GraphData {
nodes?: CustomNodeData[];
edges?: CustomEdgeData[];
}
interface AntFlow {
container: string | HTMLElement | Canvas; container: string | HTMLElement | Canvas;
data: CustomGraphData; data: CustomGraphData;
behaviors?: CustomBehaviorOption[]; behaviors?: CustomBehaviorOption[];
autoFit?: 'view' | 'center'; autoFit?: 'view' | 'center';
} }
export function useAntFlow(property: AntFlow) { export function useAntFlow(config: AntFlowConfig) {
const themeStore = useThemeStore(); const themeStore = useThemeStore();
const otherBehaviors = property.behaviors ? property.behaviors : [];
const baseColor = 'rgb(158 163 171)';
const { container, autoFit = 'center', data, behaviors = [] } = config;
const graph = new Graph({ const graph = new Graph({
container: property.container, container,
animation: false, animation: false,
padding: 16, padding: 16,
theme: 'light', theme: 'light',
autoFit: property.autoFit || 'center', autoFit,
data: property.data as GraphData, data,
node: { node: {
type: 'rect', type: 'rect',
style: (node: any) => { style: (node: CustomNodeData) => {
const iconS = getNodeIcon(node); const iconS = getNodeIcon(node);
let labelFill = '#000000'; let labelFill = '#000000';
if (node.taskState === 'NOT_STARTED') { if (node.taskState === 'NOT_STARTED') {
@ -51,7 +37,7 @@ export function useAntFlow(property: AntFlow) {
} }
return { return {
labelText: node.name, labelText: node.name as string,
size: [120, 26], size: [120, 26],
radius: 99, radius: 99,
fill: '#FFFFFF', fill: '#FFFFFF',
@ -91,7 +77,7 @@ export function useAntFlow(property: AntFlow) {
haloStroke: themeStore.themeColor, haloStroke: themeStore.themeColor,
haloLineWidth: 6 haloLineWidth: 6
}, },
active: (node: any) => ({ active: (node: CustomNodeData) => ({
halo: true, halo: true,
haloStroke: node.isDeleted ? themeStore.otherColor.error : themeStore.themeColor, haloStroke: node.isDeleted ? themeStore.otherColor.error : themeStore.themeColor,
haloLineWidth: 6, haloLineWidth: 6,
@ -101,14 +87,14 @@ export function useAntFlow(property: AntFlow) {
}, },
edge: { edge: {
type: 'cubic-horizontal', type: 'cubic-horizontal',
style: (node: any) => ({ style: (node: CustomEdgeData) => ({
curveOffset: 10, curveOffset: 10,
curvePosition: 0.5, curvePosition: 0.5,
stroke: node.isDeleted ? themeStore.otherColor.error : baseColor, stroke: node.isDeleted ? themeStore.otherColor.error : baseColor,
lineDash: node.isDeleted ? 4 : 0 lineDash: node.isDeleted ? 4 : 0
}), }),
state: { state: {
active: (node: any) => ({ active: (node: CustomEdgeData) => ({
lineWidth: 2, lineWidth: 2,
stroke: node.isDeleted ? themeStore.otherColor.error : themeStore.themeColor, stroke: node.isDeleted ? themeStore.otherColor.error : themeStore.themeColor,
halo: true, halo: true,
@ -133,17 +119,17 @@ export function useAntFlow(property: AntFlow) {
direction: 'both' direction: 'both'
}, },
'drag-canvas', 'drag-canvas',
...otherBehaviors ...behaviors
], ],
plugins: [ plugins: [
{ {
type: 'tooltip', type: 'tooltip',
enable: (event: any) => event.targetType === 'node', enable: (event: IPointerEvent) => event.targetType === 'node',
getContent: (_event: any, items: any) => { getContent: (_event: IPointerEvent, items?: CustomNodeData[]) => {
let result = '<div style="display: flex; flex-direction: column; gap: 8px;">'; let result = '<div style="display: flex; flex-direction: column; gap: 8px;">';
// 弹出提示可以自定义各种内容但是这里很奇怪有的class不跟随uniocss的样式 // 弹出提示可以自定义各种内容但是这里很奇怪有的class不跟随unocss的样式
items?.forEach((item: any) => { items?.forEach(item => {
result += `<h3 style="display: flex; align-items: center; gap: 8px;">${item.name}</h3>`; result += `<h3 style="display: flex; align-items: center; gap: 8px;">${item.name}</h3>`;
result += `<div style="display: flex;"><b>状态:</b><div style="display: flex; gap: 4px;"><img src="${getNodeIcon(item)}" /><span style="font-weight: 400 !important;">${nodeStatus[item.status as keyof typeof nodeStatus].type}</span></div></div>`; result += `<div style="display: flex;"><b>状态:</b><div style="display: flex; gap: 4px;"><img src="${getNodeIcon(item)}" /><span style="font-weight: 400 !important;">${nodeStatus[item.status as keyof typeof nodeStatus].type}</span></div></div>`;

View File

@ -1,8 +1,17 @@
import { h } from 'vue'; import { h } from 'vue';
import type { TagProps } from 'naive-ui';
import { NTag } from 'naive-ui'; import { NTag } from 'naive-ui';
import type { TagProps } from 'naive-ui';
import type { CustomNodeData, NodeStatus } from './types';
export const nodeStatus = { interface NodeStatusConfig {
type: string;
color: string;
textColor: string;
base64: string;
flag64: string;
}
export const nodeStatus: Record<NodeStatus, NodeStatusConfig> = {
MILESTONE: { MILESTONE: {
type: '里程碑', type: '里程碑',
color: '#5b5b5b', color: '#5b5b5b',
@ -14,86 +23,74 @@ export const nodeStatus = {
type: '未开始', type: '未开始',
color: '#CCCDD0', color: '#CCCDD0',
textColor: '#5b5b5b', textColor: '#5b5b5b',
base64: base64: ``,
'', flag64: ``
flag64:
''
}, },
DELAYED: { DELAYED: {
type: '已延期', type: '已延期',
color: '#B81111', color: '#B81111',
textColor: '#dccbcb', textColor: '#dccbcb',
base64: base64: ``,
'', flag64: ``
flag64:
''
}, },
PAUSED: { PAUSED: {
type: '已暂停', type: '已暂停',
color: '#0E42D2', color: '#0E42D2',
textColor: '#dae0f0', textColor: '#dae0f0',
base64: base64: ``,
'', flag64: ``
flag64:
''
}, },
IN_PROGRESS: { IN_PROGRESS: {
type: '进行中', type: '进行中',
color: '#E1BE0D', color: '#E1BE0D',
textColor: '#4f4304', textColor: '#4f4304',
base64: base64: ``,
'', flag64: ``
flag64:
''
}, },
COMPLETED: { COMPLETED: {
type: '已完成', type: '已完成',
color: '#33C73D', color: '#33C73D',
textColor: '#084e0c', textColor: '#084e0c',
base64: base64: ``,
'', flag64: ``
flag64:
''
}, },
COMPLETED_EARLY: { COMPLETED_EARLY: {
type: '提前完成', type: '提前完成',
color: '#CCFF99', color: '#CCFF99',
textColor: '#42681d', textColor: '#42681d',
base64: base64: ``,
'', flag64: ``
flag64:
''
}, },
COMPLETED_LATE: { COMPLETED_LATE: {
type: '延期完成', type: '延期完成',
color: '#CC6699', color: '#CC6699',
textColor: '#4b092a', textColor: '#4b092a',
base64: base64: ``,
'', flag64: ``
flag64:
''
} }
}; };
export function getNodeIcon(node: any) { export function getNodeIcon(node: CustomNodeData) {
if (!node.status) return '';
const type = node.milestone ? 'flag64' : 'base64'; const type = node.milestone ? 'flag64' : 'base64';
if (node.status) {
return nodeStatus[node.status as keyof typeof nodeStatus][type]; return nodeStatus[node.status][type];
}
return '';
} }
export function getNodeStatusTag(state: keyof typeof nodeStatus, tagProperty?: TagProps) { export function getNodeStatusTag(state: NodeStatus, tagProperty?: TagProps) {
const { textColor, color, type } = nodeStatus[state] || {};
return h( return h(
NTag, NTag,
{ {
color: { textColor: nodeStatus[state]?.textColor, color: nodeStatus[state]?.color }, color: { textColor, color },
bordered: false, bordered: false,
size: 'small', size: 'small',
...tagProperty ...tagProperty
}, },
{ {
default: () => nodeStatus[state]?.type default: () => type
} }
); );
} }

View File

@ -0,0 +1,28 @@
import type { EdgeData, GraphData, NodeData } from '@antv/g6';
export type NodeStatus =
| 'MILESTONE'
| 'NOT_STARTED'
| 'DELAYED'
| 'PAUSED'
| 'IN_PROGRESS'
| 'COMPLETED'
| 'COMPLETED_EARLY'
| 'COMPLETED_LATE';
export interface CustomNodeData extends NodeData {
isDelayed?: boolean;
isDeleted?: boolean;
milestone?: boolean;
status?: NodeStatus;
}
export interface CustomEdgeData extends EdgeData {
isDelayed?: boolean;
isDeleted?: boolean;
}
export interface CustomGraphData extends GraphData {
nodes: CustomNodeData[];
edges: CustomEdgeData[];
}