Unity3D 基于GraphView实现的节点编辑器框架详解

embedded/2025/1/2 12:17:05/

前言

Unity3D游戏开发中,节点编辑器是一种强大的工具,它允许开发者以可视化的方式创建和编辑复杂的逻辑和流程。Unity提供了一个强大的UI工具包——GraphView,它使得创建自定义节点编辑器变得相对简单。本文将详细介绍如何使用GraphView实现一个节点编辑器框架,并提供技术详解和代码实现。

对惹,这里有一游戏开发交流小组,希望大家可以点击进来一起交流一下开发经验呀!

一、GraphView简介

GraphView是Unity提供的一个用于创建节点编辑器的UI组件。它允许开发者以图形化的方式展示和编辑节点及其连接。GraphView提供了丰富的API,使得开发者可以轻松地自定义节点、边、面板和工具栏等。

二、节点编辑器框架设计

在创建一个节点编辑器框架时,我们需要考虑以下几个关键部分:

  1. 节点(Node):节点是编辑器中的基本元素,它代表了一个可以执行特定操作的单元。每个节点都应该有一个唯一的标识符、一个标题、一个或多个输入/输出端口,以及用于显示和操作节点的UI元素。
  2. 边(Edge):边用于连接节点,表示节点之间的数据流或逻辑依赖关系。在GraphView中,边通常由两个端口(一个输入端口和一个输出端口)组成。
  3. 面板(Panel):面板是节点的容器,它提供了用于添加、删除和移动节点的界面。面板还可以包含工具栏、小地图等辅助工具。
  4. 工具栏(Toolbar):工具栏提供了用于创建新节点、保存和加载编辑器状态、撤销和重做操作等功能的按钮和菜单。
  5. 数据存储:为了持久化编辑器状态,我们需要将节点的数据和连接关系存储在一个可序列化的数据结构中。在Unity中,ScriptableObject是一个常用的选择。

三、技术详解

  1. 创建节点和边
  • 节点可以通过继承GraphView的Node类来创建。在节点类中,我们需要重写BuildContextualMenu方法来添加右键菜单项,如添加输入/输出端口、删除节点等。
  • 边可以通过GraphView的Edge类来创建。在创建边时,我们需要指定边的输入和输出端口,并处理边的绘制和连接逻辑。

  1. 管理节点和边的数据
  • 我们可以使用ScriptableObject来存储节点的数据和连接关系。每个节点可以有一个对应的ScriptableObject来存储其特定的数据。
  • 连接关系可以通过存储边的输入和输出端口的标识符来表示。

  1. 实现撤销和重做功能
  • 撤销和重做功能可以通过维护一个操作历史记录来实现。每次对编辑器进行更改时,都可以将更改作为一个操作添加到历史记录中。
  • 撤销操作可以回滚到历史记录中的上一个状态,重做操作可以恢复到下一个状态。

  1. 实现保存和加载功能
  • 保存功能可以将编辑器的当前状态序列化为一个文件或字符串,并保存到磁盘上。
  • 加载功能可以从磁盘上读取文件或字符串,并将其反序列化为编辑器的状态。

四、代码实现

以下是一个简单的节点编辑器框架的代码实现示例:

using UnityEngine;
using UnityEditor;
using UnityEngine.UIElements;
using UnityEditor.UIElements;
// 定义一个用于存储节点数据的ScriptableObject
[CreateAssetMenu(fileName = "NewNodeGraph", menuName = "NodeGraph/NodeGraph")]
public class NodeGraph : ScriptableObject
{
// 存储节点和边的数据
public List<NodeBaseData> nodes = new List<NodeBaseData>();
public List<NodeLinkData> edges = new List<NodeLinkData>();
}
// 定义一个用于存储节点基础数据的类
[Serializable]
public abstract class NodeBaseData
{
public string GUID;
public string NodeName = "NodeBase";
public Rect Position = Rect.zero;
// 其他节点数据
}
// 定义一个用于存储边数据的类
[Serializable]
public class NodeLinkData
{
public string BaseNodeGUID;
public string OutputPortName;
public string TargetNodeGUID;
public string TargetPortName;
}
// 定义一个节点类,继承自GraphView的Node类
public class MyNode : Node
{
// 节点数据
public NodeBaseData nodeData;
// 构造函数
public MyNode()
{
// 设置节点标题和样式
title = "My Node";
styleSheets.Add(AssetDatabase.LoadAssetAtPath<StyleSheet>("Packages/com.unity.uielements/Editor/Resources/Styles/GraphView.uss"));
// 添加输入/输出端口
var inputPort = new Port(Orientation.Horizontal, Direction.Input, Port.Capacity.Single, typeof(float));
inputPort.portName = "Input";
inputContainer.Add(inputPort);
var outputPort = new Port(Orientation.Horizontal, Direction.Output, Port.Capacity.Single, typeof(float));
outputPort.portName = "Output";
outputContainer.Add(outputPort);
// 添加右键菜单
this.RegisterCallback<MouseDownEvent>(OnMouseDown);
}
// 处理右键菜单事件
private void OnMouseDown(MouseDownEvent evt)
{
if (evt.button == MouseButton.RightMouse)
{
var menu = new GenericMenu();
menu.AddItem(new GUIContent("Delete Node"), false, () => { DeleteNode(); });
menu.ShowAsContext();
evt.StopPropagation();
}
}
// 删除节点
private void DeleteNode()
{
// 从GraphView中移除节点
graphView.RemoveElement(this);
// 从NodeGraph中移除节点数据(需要自行实现)
}
}
// 定义一个节点视图类,继承自GraphView
public class MyGraphView : GraphView
{
// 构造函数
public MyGraphView(EditorWindow window, StyleSheet styleSheet)
{
this.styleSheets.Add(styleSheet);
this.AddManipulator(new ContextualMenuManipulator(OnContextualMenu));
this.AddManipulator(new SelectionDragManipulator());
this.AddManipulator(new RectangleSelector());
this.AddManipulator(new ZoomManipulator());
this.AddManipulator(new PanManipulator());
// 初始化节点和边(需要自行实现)
}
// 处理右键菜单事件
private void OnContextualMenu(ContextualMenuPopulateEvent evt)
{
var menu = new GenericMenu();
menu.AddItem(new GUIContent("Create Node"), false, () => { CreateNode(); });
menu.ShowAsContext();
}
// 创建节点
private void CreateNode()
{
var newNode = new MyNode();
newNode.SetPosition(new Rect(mousePosition, Vector2.one * 100));
this.Add(newNode);
// 添加节点数据到NodeGraph中(需要自行实现)
}
}
// 定义一个编辑器窗口类,用于显示节点编辑器
public class NodeEditorWindow : EditorWindow
{
private MyGraphView graphView;
private NodeGraph nodeGraph;
// 构造函数
[MenuItem("Window/Node Editor")]
public static void ShowWindow()
{
var window = GetWindow<NodeEditorWindow>("Node Editor");
window.minSize = new Vector2(800, 600);
}
// 初始化编辑器窗口
private void OnEnable()
{
// 加载或创建NodeGraph
nodeGraph = AssetDatabase.LoadAssetAtPath<NodeGraph>("Assets/NodeGraphs/MyNodeGraph.asset");
if (nodeGraph == null)
{
nodeGraph = ScriptableObject.CreateInstance<NodeGraph>();
AssetDatabase.CreateAsset(nodeGraph, "Assets/NodeGraphs/MyNodeGraph.asset");
AssetDatabase.SaveAssets();
}
// 初始化GraphView
var styleSheet = AssetDatabase.LoadAssetAtPath<StyleSheet>("Packages/com.unity.uielements/Editor/Resources/Styles/GraphView.uss");
graphView = new MyGraphView(this, styleSheet);
graphView.StretchToParentSize();
rootVisualElement.Add(graphView);
// 初始化节点和边(根据nodeGraph加载数据)
// 需要自行实现
}
// 保存编辑器状态
private void OnDisable()
{
// 保存nodeGraph到磁盘(需要自行实现)
}
}

五、总结

本文介绍了如何使用Unity3D的GraphView组件创建一个简单的节点编辑器框架。我们详细讨论了节点编辑器框架的设计、技术实现和代码示例。通过自定义节点、边、面板和工具栏等组件,开发者可以轻松地创建出功能强大的节点编辑器,以满足游戏开发中的复杂需求。希望本文能为Unity3D开发者提供有价值的参考和指导。

更多教学视频

Unity3D​www.bycwedu.com/promotion_channels/2146264125


http://www.ppmy.cn/embedded/150248.html

相关文章

数据的简单处理——pandas模块——读取数据(Excel和csv格式)

使用Pandas模块可以从多种类型的文件中读取数据。本节主要从Excel和csv格式文件中读取数据为例&#xff0c;进行练习。 一、读取数据Excel格式 主要包括&#xff0c;读取完整表格、读取指定行数据、读取指定列数据。 二、读取数据csv格式 主要包括&#xff0c;读取完整表格…

ubuntu 20.04 国内源安装docker

先更新软件包&#xff0c;安装备要apt软件 # 更新软件包索引 sudo apt-get update# 安装需要的软件包以使apt能够通过HTTPS使用仓库 sudo apt-get install ca-certificates curl gnupg lsb-release使用阿里云源 # 添加阿里云官方GPG密钥 curl -fsSL http://mirrors.aliyun.co…

在C#中实现支持LINQ查询的自定义集合类

在C#中&#xff0c;若要使自定义集合类支持LINQ查询&#xff0c;需要实现一些特定的接口&#xff0c;这些接口通常与集合数据的枚举和操作有关。以下是一个基本步骤指南&#xff0c;用于创建一个支持LINQ查询的自定义集合类&#xff1a; 实现IEnumerable<T>接口&#xff…

Qt 中实现系统主题感知

【写在前面】 在现代桌面应用程序开发中&#xff0c;系统主题感知是一项重要的功能&#xff0c;它使得应用程序能够根据用户的系统主题设置&#xff08;如深色模式或浅色模式&#xff09;自动调整其外观。 Qt 作为一个跨平台的C图形用户界面应用程序开发框架&#xff0c;提供…

CentOS中使用SSH远程登录

CentOS中使用SSH远程登录 CentOS中使用SSH远程登录 准备工作SSH概述SSH服务的安装与启动建立SSH连接SSH配置文件修改SSH默认端口SSH文件传输 准备工作 两台安装CentOS系统的虚拟机 客户机&#xff08;192.168.239.128&#xff09; 服务器&#xff08;192.168.239.129&#…

防火墙技术与网络安全

网络已经成为了人类所构建的最丰富多彩的虚拟世界&#xff0c;网络的迅速发展&#xff0c;给我们的工作和学习生活带来了巨大的改变。我们通过网络获得信息&#xff0c;共享资源。如今&#xff0c;Internet遍布世界任何一个角落&#xff0c;并且欢迎任何一个人加入其中&#xf…

Arduino中借助LU-ASR01实现语音识别

LU-ASR01是一款采用深度学习算法的离线语音识别开发板&#xff0c;无需联网即可独立运行。它具备自动识别中文语音输入并进行相应处理的能力。此开发板拥有1路输出的串口TX。该开发板需在天问Block IDE&#xff08;http://twen51.com/new/twen51/index.php&#xff09;中开发&a…

如何设置Edge浏览器访问软件

使用Edge浏览器访问分销ERP A\V系列软件时会出现各种报错&#xff0c;如何设置Edge浏览器使其正常访问&#xff0c;请看下面的具体操作。 一、打开Edge浏览器&#xff0c;点击右上角的 设置及其他&#xff0c;如图&#xff1a; 二、在弹出界面中&#xff0c;点击 扩展&#xff…