kanzi插件之节点树可视化

背景:随着项目增大,项目中预设件增多,关联错综复杂,维护修改成本很高,需要一款可视化工具,来辅助开发。

1. 需求分析

  • 节点树可视化-卡片形式
  • 如果节点是Prefab View 2D或者Prefab Placeholder 2D类型,展开它的属性Prefab Template,支持递归
  • 如果节点里有状态机,状态机中设置了属性Prefab Template,也需要展开,支持递归
  • 卡片之间有连线,有上下级关系,状态机的要包围起来,表示对外只有一个节点
  • 左键单击选中卡片,kanzi studio焦点自动跳转
  • 右键拖拽视图
  • 滚轮缩放视图
  • 支持前进后退视图

2. 效果

Prefab View 2D

Prefab Placeholder 2D

状态机

3. 设计

3.1 kanzi窗口插件,名字PluginWinShowPrefabTree

如何创建插件,请参考以往博客

3.2 业务逻辑

  • 点击【Update】按钮,获取当前节点
c 复制代码
this.studio.SelectedItems.ToList();
  • 先判断有没有状态机
c 复制代码
if ("Node2DPrefabPlaceholderPluginWrapper" == node.GetType().Name ||
    "ComponentNode2DPluginWrapper" == node.GetType().Name)
{
    Node2D node2d = node as Node2D;
    if (node2d != null && node2d.PropertyTypes.NodeStateManager != null)
    {
        _NodeStateManager = node2d.Get(node2d.PropertyTypes.NodeStateManager);
    }
}
  • 获取状态机组StateGroup,获取组里的所有状态State,获取状态里的StateObject,获取其属性"PrefabViewConcept.Prefab",递归获取子节点DiguiHandleRefab

  • 如果没有状态机,递归获取子节点HandlePrefabResource,

c 复制代码
if ("Node2DPrefabPlaceholderPluginWrapper" == node.GetType().Name)
{
   var _Node2DPrefabPlaceholder = node as Node2DPrefabPlaceholder;
   if (_Node2DPrefabPlaceholder != null)
   {
       ResourceReference<Node2DPrefabTemplate> tempplateRef = _Node2DPrefabPlaceholder.Get(_Node2DPrefabPlaceholder.PropertyTypes.Node2DPrefabPlaceholderTemplate);
     
   }
}
else if ("ComponentNode2DPluginWrapper" == node.GetType().Name)
{
   var _ComponentNode2D = node as ComponentNode2D;
   if (_ComponentNode2D != null)
   {
       ResourceReference<PrefabTemplate> _PrefabViewConceptPrefab = _ComponentNode2D.Get(_ComponentNode2D.PropertyTypes.PrefabViewConceptPrefab);
      
   }
}
  • 最后遍历获取其子节点
c 复制代码
var childNodes = node.ChildNodes.ToList();
foreach (Node child in childNodes)
{
    var childTreeNode = DiguiHandleRefab(child, node);
    if (childTreeNode != null)
    {
        treeNode.Children.Add(childTreeNode);
    }
}
  • 获取的数据储存在树状节点里,claude生成ui代码
  • 点击卡片,跳转焦点
c 复制代码
//搜索screen
var pp = path.Replace(project.Screen.Path + "/", "");
ProjectItem target = project.Screen.GetProjectItemByPath(pp);
this.studio.Commands.SelectProjectItems(new List<ProjectItem> { target as ProjectItem });

//搜索预设件
var target = item.GetProjectItemByPath(childPath);
if (target != null)
{
    studio.Log($"[Select] Found prefab: {target.Name} (path: {target.Path})");
    this.studio.Commands.SelectProjectItems(new List<ProjectItem> { target as ProjectItem });
    return true;
}

搜索场景树的规则是

假如路径是 Screens/Screen1/RootPage/Empty Node 2D/Prefab View 2D

要这样搜 project.Screen.GetProjectItemByPath("RootPage/Empty Node 2D/Prefab View 2D");

搜索prefab的规则是

假如路径是 Prefabs/Text Block 2D_1/Text Block 2D_1/a

要这样搜 item.GetProjectItemByPath("Text Block 2D/a");

相关推荐
大空大地202612 小时前
C#高级语法总结
开发语言·c#
chen_22718 小时前
KZPROJ Git Diff AI 审查工具
git·ai·claude·kanzi
周杰伦fans19 小时前
C# AutoCAD 二次开发极简入门:从环境搭建到高效实战
开发语言·c#
.NET修仙日记20 小时前
.NET EFCore批量插入性能优化实战:30秒 → 0.5秒
性能优化·c#·.net·.netcore·微软技术·efcore·踩坑实录
Esofar21 小时前
Dddify:给 ASP.NET Core 项目一套轻量、清晰、可落地的 DDD 基础设施
c#·ddd·asp.net core·cqrs·dddify·clean architecture
Coder_Shenshen1 天前
【基于LibUA库的OPC UA服务器与客户端Demo——协议解析与Bug修复实践】
网络·c#·bug
信必诺1 天前
C# —— VS2022配置终端程序跨平台发布方法(部署Ubuntu22.04举例,详细多图)
ubuntu·c#·跨平台部署
我是唐青枫1 天前
C#.NET YARP 跨域配置详解:网关统一处理 CORS
开发语言·c#·.net
lzhdim1 天前
C#性能优化技巧
开发语言·性能优化·c#
weixin_428005301 天前
C#调用 AI学习从0开始-第1阶段(基础与工具)-第5天完善请求结构
windows·学习·c#·ai请求结构