2024-09-17 14:20:27 +08:00
|
|
|
|
|
2024-09-22 14:10:13 +08:00
|
|
|
|
using Newtonsoft.Json.Linq;
|
2024-09-17 14:20:27 +08:00
|
|
|
|
using Serein.Library.Api;
|
2024-09-15 12:15:32 +08:00
|
|
|
|
using Serein.Library.Attributes;
|
|
|
|
|
|
using Serein.Library.Entity;
|
|
|
|
|
|
using Serein.Library.Enums;
|
|
|
|
|
|
using Serein.Library.Utils;
|
|
|
|
|
|
using Serein.NodeFlow.Base;
|
|
|
|
|
|
using Serein.NodeFlow.Model;
|
|
|
|
|
|
using Serein.NodeFlow.Tool;
|
2024-09-20 10:50:32 +08:00
|
|
|
|
using System.Collections.Concurrent;
|
2024-09-15 12:15:32 +08:00
|
|
|
|
using System.Reflection;
|
2024-09-21 10:06:44 +08:00
|
|
|
|
using System.Xml.Linq;
|
2024-09-22 17:37:32 +08:00
|
|
|
|
using static Serein.Library.Utils.ChannelFlowInterrupt;
|
2024-09-16 21:38:34 +08:00
|
|
|
|
using static Serein.NodeFlow.FlowStarter;
|
2024-09-15 12:15:32 +08:00
|
|
|
|
|
|
|
|
|
|
namespace Serein.NodeFlow
|
|
|
|
|
|
{
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
|
|
|
|
脱离wpf平台独立运行。
|
|
|
|
|
|
加载文件。
|
|
|
|
|
|
创建节点对象,设置节点属性,确定连接关系,设置起点。
|
|
|
|
|
|
|
|
|
|
|
|
↓抽象↓
|
|
|
|
|
|
|
|
|
|
|
|
wpf依赖于运行环境,而不是运行环境依赖于wpf。
|
|
|
|
|
|
|
|
|
|
|
|
运行环境实现以下功能:
|
|
|
|
|
|
①从项目文件加载数据,生成项目文件对象。
|
|
|
|
|
|
②运行项目,调试项目,中止项目,终止项目。
|
|
|
|
|
|
③自动包装数据类型,在上下文中传递数据。
|
|
|
|
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-09-20 10:50:32 +08:00
|
|
|
|
|
|
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 运行环境
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public class FlowEnvironment : IFlowEnvironment
|
|
|
|
|
|
{
|
2024-09-20 10:50:32 +08:00
|
|
|
|
public FlowEnvironment()
|
|
|
|
|
|
{
|
|
|
|
|
|
ChannelFlowInterrupt = new ChannelFlowInterrupt();
|
|
|
|
|
|
LoadedAssemblyPaths = new List<string>();
|
|
|
|
|
|
LoadedAssemblies = new List<Assembly>();
|
|
|
|
|
|
MethodDetailss = new List<MethodDetails>();
|
|
|
|
|
|
Nodes = new Dictionary<string, NodeModelBase>();
|
|
|
|
|
|
FlipflopNodes = new List<SingleFlipflopNode>();
|
|
|
|
|
|
IsGlobalInterrupt = false;
|
|
|
|
|
|
flowStarter = null;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-09-17 14:20:27 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 节点的命名空间
|
|
|
|
|
|
/// </summary>
|
2024-09-22 17:37:32 +08:00
|
|
|
|
public const string SpaceName = $"{nameof(Serein)}.{nameof(Serein.NodeFlow)}.{nameof(Serein.NodeFlow.Model)}";
|
2024-09-17 14:20:27 +08:00
|
|
|
|
|
2024-09-20 10:50:32 +08:00
|
|
|
|
#region 环境接口事件
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 加载Dll
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public event LoadDLLHandler OnDllLoad;
|
2024-09-20 10:50:32 +08:00
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// <summary>
|
2024-09-17 14:20:27 +08:00
|
|
|
|
/// 项目加载完成
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// </summary>
|
2024-09-17 14:20:27 +08:00
|
|
|
|
public event ProjectLoadedHandler OnProjectLoaded;
|
2024-09-20 10:50:32 +08:00
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// <summary>
|
2024-09-16 21:38:34 +08:00
|
|
|
|
/// 节点连接属性改变事件
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// </summary>
|
|
|
|
|
|
public event NodeConnectChangeHandler OnNodeConnectChange;
|
2024-09-20 10:50:32 +08:00
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// <summary>
|
2024-09-17 14:20:27 +08:00
|
|
|
|
/// 节点创建事件
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// </summary>
|
|
|
|
|
|
public event NodeCreateHandler OnNodeCreate;
|
2024-09-20 10:50:32 +08:00
|
|
|
|
|
2024-09-16 21:38:34 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 移除节点事件
|
|
|
|
|
|
/// </summary>
|
2024-09-15 12:15:32 +08:00
|
|
|
|
public event NodeRemoteHandler OnNodeRemote;
|
2024-09-20 10:50:32 +08:00
|
|
|
|
|
2024-09-16 21:38:34 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 起始节点变化事件
|
|
|
|
|
|
/// </summary>
|
2024-09-15 12:15:32 +08:00
|
|
|
|
public event StartNodeChangeHandler OnStartNodeChange;
|
2024-09-20 10:50:32 +08:00
|
|
|
|
|
2024-09-16 21:38:34 +08:00
|
|
|
|
/// <summary>
|
2024-09-22 14:10:13 +08:00
|
|
|
|
/// 流程运行完成事件
|
2024-09-16 21:38:34 +08:00
|
|
|
|
/// </summary>
|
2024-09-15 12:15:32 +08:00
|
|
|
|
public event FlowRunCompleteHandler OnFlowRunComplete;
|
|
|
|
|
|
|
2024-09-22 14:10:13 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 被监视的对象改变事件
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public event MonitorObjectChangeHandler OnMonitorObjectChange;
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 节点中断状态改变事件
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public event NodeInterruptStateChangeHandler OnNodeInterruptStateChange;
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 节点触发了中断
|
|
|
|
|
|
/// </summary>
|
2024-09-22 17:37:32 +08:00
|
|
|
|
public event ExpInterruptTriggerHandler OnInterruptTrigger;
|
2024-09-22 14:10:13 +08:00
|
|
|
|
|
2024-09-20 10:50:32 +08:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// <summary>
|
2024-09-22 17:37:32 +08:00
|
|
|
|
/// 环境名称
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// </summary>
|
2024-09-22 17:37:32 +08:00
|
|
|
|
public string EnvName { get; set; } = SpaceName;
|
|
|
|
|
|
|
2024-09-20 10:50:32 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 是否全局中断
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public bool IsGlobalInterrupt { get; set; }
|
2024-09-15 12:15:32 +08:00
|
|
|
|
|
2024-09-22 17:37:32 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 流程中断器
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public ChannelFlowInterrupt ChannelFlowInterrupt { get; set; }
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 存储加载的程序集路径
|
|
|
|
|
|
/// </summary>
|
2024-09-20 10:50:32 +08:00
|
|
|
|
public List<string> LoadedAssemblyPaths { get; }
|
2024-09-15 12:15:32 +08:00
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 存储加载的程序集
|
|
|
|
|
|
/// </summary>
|
2024-09-20 10:50:32 +08:00
|
|
|
|
public List<Assembly> LoadedAssemblies { get; }
|
2024-09-15 12:15:32 +08:00
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 存储所有方法信息
|
|
|
|
|
|
/// </summary>
|
2024-09-20 10:50:32 +08:00
|
|
|
|
public List<MethodDetails> MethodDetailss { get; }
|
2024-09-15 12:15:32 +08:00
|
|
|
|
|
2024-09-20 10:50:32 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 环境加载的节点集合
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public Dictionary<string, NodeModelBase> Nodes { get; }
|
2024-09-15 12:15:32 +08:00
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 存放触发器节点(运行时全部调用)
|
|
|
|
|
|
/// </summary>
|
2024-09-20 10:50:32 +08:00
|
|
|
|
public List<SingleFlipflopNode> FlipflopNodes { get; }
|
2024-09-15 12:15:32 +08:00
|
|
|
|
|
2024-09-16 21:38:34 +08:00
|
|
|
|
/// <summary>
|
2024-09-20 10:50:32 +08:00
|
|
|
|
/// 起始节点私有属性
|
2024-09-16 21:38:34 +08:00
|
|
|
|
/// </summary>
|
|
|
|
|
|
private NodeModelBase _startNode;
|
2024-09-17 21:43:49 +08:00
|
|
|
|
|
2024-09-15 19:48:27 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 起始节点
|
|
|
|
|
|
/// </summary>
|
2024-09-16 19:53:36 +08:00
|
|
|
|
public NodeModelBase StartNode
|
|
|
|
|
|
{
|
|
|
|
|
|
get
|
|
|
|
|
|
{
|
2024-09-15 12:15:32 +08:00
|
|
|
|
return _startNode;
|
|
|
|
|
|
}
|
2024-09-16 19:53:36 +08:00
|
|
|
|
set
|
|
|
|
|
|
{
|
2024-09-16 21:38:34 +08:00
|
|
|
|
if (_startNode is not null)
|
|
|
|
|
|
{
|
2024-09-15 12:15:32 +08:00
|
|
|
|
_startNode.IsStart = false;
|
|
|
|
|
|
}
|
2024-09-16 21:38:34 +08:00
|
|
|
|
value.IsStart = true;
|
|
|
|
|
|
_startNode = value;
|
2024-09-16 19:53:36 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-09-15 12:15:32 +08:00
|
|
|
|
|
2024-09-20 10:50:32 +08:00
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 流程启动器(每次运行时都会重新new一个)
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
private FlowStarter? flowStarter;
|
|
|
|
|
|
|
2024-09-15 19:48:27 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 异步运行
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <returns></returns>
|
2024-09-15 12:15:32 +08:00
|
|
|
|
public async Task StartAsync()
|
|
|
|
|
|
{
|
2024-09-20 10:50:32 +08:00
|
|
|
|
ChannelFlowInterrupt?.CancelAllTasks();
|
2024-09-17 21:43:49 +08:00
|
|
|
|
flowStarter = new FlowStarter();
|
2024-09-22 14:10:13 +08:00
|
|
|
|
var nodes = Nodes.Values.ToList();
|
|
|
|
|
|
|
|
|
|
|
|
List<MethodDetails> initMethods;
|
|
|
|
|
|
List<MethodDetails> loadingMethods;
|
|
|
|
|
|
List<MethodDetails> exitMethods;
|
|
|
|
|
|
initMethods = MethodDetailss.Where(it => it.MethodDynamicType == NodeType.Init).ToList();
|
|
|
|
|
|
loadingMethods = MethodDetailss.Where(it => it.MethodDynamicType == NodeType.Loading).ToList();
|
|
|
|
|
|
exitMethods = MethodDetailss.Where(it => it.MethodDynamicType == NodeType.Exit).ToList();
|
|
|
|
|
|
|
|
|
|
|
|
await flowStarter.RunAsync(this, nodes, initMethods, loadingMethods, exitMethods);
|
|
|
|
|
|
|
2024-09-17 21:43:49 +08:00
|
|
|
|
if(flowStarter?.FlipFlopState == RunState.NoStart)
|
2024-09-16 21:38:34 +08:00
|
|
|
|
{
|
|
|
|
|
|
this.Exit(); // 未运行触发器时,才会调用结束方法
|
|
|
|
|
|
}
|
2024-09-17 21:43:49 +08:00
|
|
|
|
flowStarter = null;
|
2024-09-15 12:15:32 +08:00
|
|
|
|
}
|
2024-09-22 14:10:13 +08:00
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 退出
|
|
|
|
|
|
/// </summary>
|
2024-09-15 12:15:32 +08:00
|
|
|
|
public void Exit()
|
|
|
|
|
|
{
|
2024-09-22 14:10:13 +08:00
|
|
|
|
ChannelFlowInterrupt?.CancelAllTasks();
|
|
|
|
|
|
flowStarter?.Exit();
|
|
|
|
|
|
|
2024-09-21 10:06:44 +08:00
|
|
|
|
foreach (var node in Nodes.Values)
|
|
|
|
|
|
{
|
2024-09-22 14:10:13 +08:00
|
|
|
|
if(node is not null)
|
2024-09-21 10:06:44 +08:00
|
|
|
|
{
|
2024-09-22 14:10:13 +08:00
|
|
|
|
node.ReleaseFlowData(); // 退出时释放对象计数
|
2024-09-21 10:06:44 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-09-22 14:10:13 +08:00
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
OnFlowRunComplete?.Invoke(new FlowEventArgs());
|
2024-09-21 10:06:44 +08:00
|
|
|
|
|
|
|
|
|
|
GC.Collect();
|
2024-09-15 12:15:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 清除所有
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public void ClearAll()
|
|
|
|
|
|
{
|
|
|
|
|
|
LoadedAssemblyPaths.Clear();
|
|
|
|
|
|
LoadedAssemblies.Clear();
|
|
|
|
|
|
MethodDetailss.Clear();
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-09-17 14:20:27 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 运行环节加载了项目文件,需要创建节点控件
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="nodeInfo"></param>
|
|
|
|
|
|
/// <param name="methodDetailss"></param>
|
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
|
/// <exception cref="NotImplementedException"></exception>
|
|
|
|
|
|
private NodeControlType GetNodeControlType(NodeInfo nodeInfo)
|
|
|
|
|
|
{
|
|
|
|
|
|
// 创建控件实例
|
|
|
|
|
|
NodeControlType controlType = nodeInfo.Type switch
|
|
|
|
|
|
{
|
|
|
|
|
|
$"{NodeStaticConfig.NodeSpaceName}.{nameof(SingleActionNode)}" => NodeControlType.Action,// 动作节点控件
|
|
|
|
|
|
$"{NodeStaticConfig.NodeSpaceName}.{nameof(SingleFlipflopNode)}" => NodeControlType.Flipflop, // 触发器节点控件
|
|
|
|
|
|
|
|
|
|
|
|
$"{NodeStaticConfig.NodeSpaceName}.{nameof(SingleConditionNode)}" => NodeControlType.ExpCondition,// 条件表达式控件
|
|
|
|
|
|
$"{NodeStaticConfig.NodeSpaceName}.{nameof(SingleExpOpNode)}" => NodeControlType.ExpOp, // 操作表达式控件
|
|
|
|
|
|
|
|
|
|
|
|
$"{NodeStaticConfig.NodeSpaceName}.{nameof(CompositeConditionNode)}" => NodeControlType.ConditionRegion, // 条件区域控件
|
|
|
|
|
|
_ => NodeControlType.None,
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
return controlType;
|
|
|
|
|
|
}
|
2024-09-15 12:15:32 +08:00
|
|
|
|
|
2024-09-15 19:48:27 +08:00
|
|
|
|
#region 对外暴露的接口
|
2024-09-15 12:15:32 +08:00
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 加载项目文件
|
|
|
|
|
|
/// </summary>
|
2024-09-17 14:20:27 +08:00
|
|
|
|
/// <param name="project"></param>
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// <param name="filePath"></param>
|
2024-09-17 14:20:27 +08:00
|
|
|
|
public void LoadProject(SereinProjectData project, string filePath)
|
2024-09-15 12:15:32 +08:00
|
|
|
|
{
|
|
|
|
|
|
// 加载项目配置文件
|
2024-09-17 14:20:27 +08:00
|
|
|
|
var dllPaths = project.Librarys.Select(it => it.Path).ToList();
|
2024-09-15 12:15:32 +08:00
|
|
|
|
List<MethodDetails> methodDetailss = [];
|
|
|
|
|
|
|
|
|
|
|
|
// 遍历依赖项中的特性注解,生成方法详情
|
|
|
|
|
|
foreach (var dll in dllPaths)
|
|
|
|
|
|
{
|
|
|
|
|
|
var dllFilePath = System.IO.Path.GetFullPath(System.IO.Path.Combine(filePath, dll));
|
2024-09-16 19:53:36 +08:00
|
|
|
|
(var assembly, var list) = LoadAssembly(dllFilePath);
|
2024-09-15 12:15:32 +08:00
|
|
|
|
if (assembly is not null && list.Count > 0)
|
|
|
|
|
|
{
|
2024-09-17 14:20:27 +08:00
|
|
|
|
MethodDetailss.AddRange(list); // 暂存方法描述
|
|
|
|
|
|
OnDllLoad?.Invoke(new LoadDLLEventArgs(assembly, list)); // 通知UI创建dll面板显示
|
2024-09-15 12:15:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
// 方法加载完成,缓存到运行环境中。
|
2024-09-17 14:20:27 +08:00
|
|
|
|
//MethodDetailss.AddRange(methodDetailss);
|
|
|
|
|
|
//methodDetailss.Clear();
|
2024-09-15 12:15:32 +08:00
|
|
|
|
|
|
|
|
|
|
|
2024-09-17 14:20:27 +08:00
|
|
|
|
List<(NodeModelBase, string[])> regionChildNodes = new List<(NodeModelBase, string[])>();
|
|
|
|
|
|
List<(NodeModelBase, Position)> ordinaryNodes = new List<(NodeModelBase, Position)>();
|
2024-09-15 12:15:32 +08:00
|
|
|
|
// 加载节点
|
2024-09-17 14:20:27 +08:00
|
|
|
|
foreach (var nodeInfo in project.Nodes)
|
2024-09-15 12:15:32 +08:00
|
|
|
|
{
|
2024-09-17 14:20:27 +08:00
|
|
|
|
var controlType = GetNodeControlType(nodeInfo);
|
|
|
|
|
|
if(controlType == NodeControlType.None)
|
2024-09-15 12:15:32 +08:00
|
|
|
|
{
|
2024-09-17 14:20:27 +08:00
|
|
|
|
continue;
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
TryGetMethodDetails(nodeInfo.MethodName, out MethodDetails? methodDetails); // 加载项目时尝试获取方法信息
|
|
|
|
|
|
methodDetails ??= new MethodDetails();
|
|
|
|
|
|
var nodeModel = CreateNode(controlType, methodDetails);
|
|
|
|
|
|
nodeModel.LoadInfo(nodeInfo); // 创建节点model
|
|
|
|
|
|
if (nodeModel is null)
|
|
|
|
|
|
{
|
|
|
|
|
|
continue;
|
|
|
|
|
|
}
|
|
|
|
|
|
TryAddNode(nodeModel);
|
|
|
|
|
|
if(nodeInfo.ChildNodeGuids?.Length > 0)
|
|
|
|
|
|
{
|
|
|
|
|
|
regionChildNodes.Add((nodeModel,nodeInfo.ChildNodeGuids));
|
|
|
|
|
|
OnNodeCreate?.Invoke(new NodeCreateEventArgs(nodeModel, nodeInfo.Position));
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
ordinaryNodes.Add((nodeModel, nodeInfo.Position));
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
// 加载区域的子项
|
|
|
|
|
|
foreach((NodeModelBase region, string[] childNodeGuids) item in regionChildNodes)
|
|
|
|
|
|
{
|
|
|
|
|
|
foreach (var childNodeGuid in item.childNodeGuids)
|
|
|
|
|
|
{
|
|
|
|
|
|
Nodes.TryGetValue(childNodeGuid, out NodeModelBase? childNode);
|
|
|
|
|
|
if (childNode is null)
|
|
|
|
|
|
{
|
|
|
|
|
|
// 节点尚未加载
|
|
|
|
|
|
continue;
|
|
|
|
|
|
}
|
|
|
|
|
|
// 存在节点
|
|
|
|
|
|
OnNodeCreate?.Invoke(new NodeCreateEventArgs(childNode, true, item.region.Guid));
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
// 加载节点
|
|
|
|
|
|
foreach ((NodeModelBase nodeModel, Position position) item in ordinaryNodes)
|
|
|
|
|
|
{
|
|
|
|
|
|
bool IsContinue = false;
|
|
|
|
|
|
foreach ((NodeModelBase region, string[] childNodeGuids) item2 in regionChildNodes)
|
|
|
|
|
|
{
|
|
|
|
|
|
foreach (var childNodeGuid in item2.childNodeGuids)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (item.nodeModel.Guid.Equals(childNodeGuid))
|
|
|
|
|
|
{
|
|
|
|
|
|
IsContinue = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-09-15 12:15:32 +08:00
|
|
|
|
}
|
2024-09-17 14:20:27 +08:00
|
|
|
|
if (IsContinue) continue;
|
|
|
|
|
|
OnNodeCreate?.Invoke(new NodeCreateEventArgs(item.nodeModel, item.position));
|
2024-09-15 12:15:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-09-17 14:20:27 +08:00
|
|
|
|
|
|
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
// 确定节点之间的连接关系
|
2024-09-17 14:20:27 +08:00
|
|
|
|
foreach (var nodeInfo in project.Nodes)
|
2024-09-15 12:15:32 +08:00
|
|
|
|
{
|
2024-09-17 14:20:27 +08:00
|
|
|
|
if (!Nodes.TryGetValue(nodeInfo.Guid, out NodeModelBase? fromNode))
|
2024-09-15 12:15:32 +08:00
|
|
|
|
{
|
|
|
|
|
|
// 不存在对应的起始节点
|
|
|
|
|
|
continue;
|
|
|
|
|
|
}
|
2024-09-16 19:53:36 +08:00
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
|
2024-09-17 14:20:27 +08:00
|
|
|
|
List<(ConnectionType connectionType, string[] guids)> allToNodes = [(ConnectionType.IsSucceed,nodeInfo.TrueNodes),
|
|
|
|
|
|
(ConnectionType.IsFail, nodeInfo.FalseNodes),
|
|
|
|
|
|
(ConnectionType.IsError, nodeInfo.ErrorNodes),
|
|
|
|
|
|
(ConnectionType.Upstream, nodeInfo.UpstreamNodes)];
|
2024-09-15 12:15:32 +08:00
|
|
|
|
|
2024-09-17 14:20:27 +08:00
|
|
|
|
List<(ConnectionType, NodeModelBase[])> fromNodes = allToNodes.Where(info => info.guids.Length > 0)
|
|
|
|
|
|
.Select(info => (info.connectionType,
|
|
|
|
|
|
info.guids.Select(guid => Nodes[guid])
|
2024-09-15 12:15:32 +08:00
|
|
|
|
.ToArray()))
|
|
|
|
|
|
.ToList();
|
|
|
|
|
|
// 遍历每种类型的节点分支(四种)
|
2024-09-17 14:20:27 +08:00
|
|
|
|
foreach ((ConnectionType connectionType, NodeModelBase[] toNodes) item in fromNodes)
|
2024-09-15 12:15:32 +08:00
|
|
|
|
{
|
|
|
|
|
|
// 遍历当前类型分支的节点(确认连接关系)
|
2024-09-17 14:20:27 +08:00
|
|
|
|
foreach (var toNode in item.toNodes)
|
2024-09-15 12:15:32 +08:00
|
|
|
|
{
|
2024-09-17 14:20:27 +08:00
|
|
|
|
ConnectNode(fromNode, toNode, item.connectionType); // 加载时确定节点间的连接关系
|
2024-09-15 12:15:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-09-17 14:20:27 +08:00
|
|
|
|
SetStartNode(project.StartNode);
|
|
|
|
|
|
OnProjectLoaded?.Invoke(new ProjectLoadedEventArgs());
|
2024-09-15 12:15:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
2024-09-16 21:38:34 +08:00
|
|
|
|
/// 保存项目为项目文件
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// </summary>
|
2024-09-16 21:38:34 +08:00
|
|
|
|
/// <returns></returns>
|
2024-09-17 14:20:27 +08:00
|
|
|
|
public SereinProjectData SaveProject()
|
2024-09-15 12:15:32 +08:00
|
|
|
|
{
|
2024-09-17 14:20:27 +08:00
|
|
|
|
var projectData = new SereinProjectData()
|
2024-09-15 12:15:32 +08:00
|
|
|
|
{
|
2024-09-16 21:38:34 +08:00
|
|
|
|
Librarys = LoadedAssemblies.Select(assemblies => assemblies.ToLibrary()).ToArray(),
|
|
|
|
|
|
Nodes = Nodes.Values.Select(node => node.ToInfo()).Where(info => info is not null).ToArray(),
|
|
|
|
|
|
StartNode = Nodes.Values.FirstOrDefault(it => it.IsStart)?.Guid,
|
|
|
|
|
|
};
|
|
|
|
|
|
return projectData;
|
|
|
|
|
|
}
|
2024-09-17 14:20:27 +08:00
|
|
|
|
|
2024-09-16 21:38:34 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 从文件路径中加载DLL
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="dllPath"></param>
|
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
|
public void LoadDll(string dllPath)
|
|
|
|
|
|
{
|
|
|
|
|
|
(var assembly, var list) = LoadAssembly(dllPath);
|
|
|
|
|
|
if (assembly is not null && list.Count > 0)
|
2024-09-15 12:15:32 +08:00
|
|
|
|
{
|
2024-09-16 21:38:34 +08:00
|
|
|
|
MethodDetailss.AddRange(list);
|
|
|
|
|
|
OnDllLoad?.Invoke(new LoadDLLEventArgs(assembly, list));
|
2024-09-16 19:53:36 +08:00
|
|
|
|
}
|
2024-09-15 12:15:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
2024-09-17 14:20:27 +08:00
|
|
|
|
/// 运行时创建节点
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="nodeBase"></param>
|
2024-09-17 14:20:27 +08:00
|
|
|
|
public void CreateNode(NodeControlType nodeControlType, Position position, MethodDetails? methodDetails = null)
|
2024-09-15 12:15:32 +08:00
|
|
|
|
{
|
2024-09-17 14:20:27 +08:00
|
|
|
|
var nodeModel = CreateNode(nodeControlType, methodDetails);
|
|
|
|
|
|
TryAddNode(nodeModel);
|
2024-09-17 21:43:49 +08:00
|
|
|
|
|
|
|
|
|
|
if(flowStarter?.FlowState != RunState.Completion
|
|
|
|
|
|
&& nodeControlType == NodeControlType.Flipflop
|
|
|
|
|
|
&& nodeModel is SingleFlipflopNode flipflopNode)
|
|
|
|
|
|
{
|
|
|
|
|
|
// 当前添加节点属于触发器,且当前正在运行,则加载到运行环境中
|
|
|
|
|
|
flowStarter?.AddFlipflopInRuning(flipflopNode, this);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
// 通知UI更改
|
2024-09-17 14:20:27 +08:00
|
|
|
|
OnNodeCreate?.Invoke(new NodeCreateEventArgs(nodeModel, position));
|
2024-09-15 12:15:32 +08:00
|
|
|
|
// 因为需要UI先布置了元素,才能通知UI变更特效
|
|
|
|
|
|
// 如果不存在流程起始控件,默认设置为流程起始控件
|
|
|
|
|
|
if (StartNode is null)
|
|
|
|
|
|
{
|
2024-09-17 14:20:27 +08:00
|
|
|
|
SetStartNode(nodeModel);
|
2024-09-15 12:15:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-09-17 14:20:27 +08:00
|
|
|
|
|
2024-09-16 19:53:36 +08:00
|
|
|
|
/// <summary>
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// 移除节点
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="nodeGuid"></param>
|
|
|
|
|
|
/// <exception cref="NotImplementedException"></exception>
|
|
|
|
|
|
public void RemoteNode(string nodeGuid)
|
|
|
|
|
|
{
|
2024-09-22 17:37:32 +08:00
|
|
|
|
var remoteNode = GuidToModel(nodeGuid);
|
|
|
|
|
|
if (remoteNode is null) return;
|
2024-09-15 12:15:32 +08:00
|
|
|
|
if (remoteNode.IsStart)
|
|
|
|
|
|
{
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
2024-09-16 19:53:36 +08:00
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
// 遍历所有父节点,从那些父节点中的子节点集合移除该节点
|
2024-09-16 19:53:36 +08:00
|
|
|
|
foreach (var pnc in remoteNode.PreviousNodes)
|
2024-09-15 12:15:32 +08:00
|
|
|
|
{
|
|
|
|
|
|
var pCType = pnc.Key; // 连接类型
|
|
|
|
|
|
for (int i = 0; i < pnc.Value.Count; i++)
|
|
|
|
|
|
{
|
|
|
|
|
|
NodeModelBase? pNode = pnc.Value[i];
|
2024-09-21 10:06:44 +08:00
|
|
|
|
//pNode.SuccessorNodes[pCType].RemoveAt(i);
|
|
|
|
|
|
pNode.SuccessorNodes[pCType].Remove(pNode);
|
|
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
OnNodeConnectChange?.Invoke(new NodeConnectChangeEventArgs(pNode.Guid,
|
|
|
|
|
|
remoteNode.Guid,
|
|
|
|
|
|
pCType,
|
2024-09-16 21:38:34 +08:00
|
|
|
|
NodeConnectChangeEventArgs.ConnectChangeType.Remote)); // 通知UI
|
2024-09-15 12:15:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 遍历所有子节点,从那些子节点中的父节点集合移除该节点
|
|
|
|
|
|
foreach (var snc in remoteNode.SuccessorNodes)
|
|
|
|
|
|
{
|
2024-09-17 21:43:49 +08:00
|
|
|
|
var connectionType = snc.Key; // 连接类型
|
2024-09-15 12:15:32 +08:00
|
|
|
|
for (int i = 0; i < snc.Value.Count; i++)
|
|
|
|
|
|
{
|
2024-09-17 21:43:49 +08:00
|
|
|
|
NodeModelBase? toNode = snc.Value[i];
|
|
|
|
|
|
|
|
|
|
|
|
RemoteConnect(remoteNode, toNode, connectionType);
|
2024-09-15 12:15:32 +08:00
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 从集合中移除节点
|
|
|
|
|
|
Nodes.Remove(nodeGuid);
|
|
|
|
|
|
OnNodeRemote?.Invoke(new NodeRemoteEventArgs(nodeGuid));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-09-16 21:38:34 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 连接节点
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="fromNode">起始节点</param>
|
|
|
|
|
|
/// <param name="toNode">目标节点</param>
|
|
|
|
|
|
/// <param name="connectionType">连接关系</param>
|
|
|
|
|
|
public void ConnectNode(string fromNodeGuid, string toNodeGuid, ConnectionType connectionType)
|
|
|
|
|
|
{
|
|
|
|
|
|
// 获取起始节点与目标节点
|
2024-09-22 17:37:32 +08:00
|
|
|
|
var fromNode = GuidToModel(fromNodeGuid);
|
|
|
|
|
|
var toNode = GuidToModel(toNodeGuid);
|
|
|
|
|
|
if (fromNode is null) return;
|
|
|
|
|
|
if (toNode is null) return;
|
2024-09-16 21:38:34 +08:00
|
|
|
|
// 开始连接
|
|
|
|
|
|
ConnectNode(fromNode, toNode, connectionType); // 外部调用连接方法
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 移除连接关系
|
|
|
|
|
|
/// </summary>
|
2024-09-17 21:43:49 +08:00
|
|
|
|
/// <param name="fromNodeGuid">起始节点Guid</param>
|
|
|
|
|
|
/// <param name="toNodeGuid">目标节点Guid</param>
|
|
|
|
|
|
/// <param name="connectionType">连接关系</param>
|
2024-09-16 21:38:34 +08:00
|
|
|
|
/// <exception cref="NotImplementedException"></exception>
|
|
|
|
|
|
public void RemoteConnect(string fromNodeGuid, string toNodeGuid, ConnectionType connectionType)
|
|
|
|
|
|
{
|
|
|
|
|
|
// 获取起始节点与目标节点
|
2024-09-22 17:37:32 +08:00
|
|
|
|
var fromNode = GuidToModel(fromNodeGuid);
|
|
|
|
|
|
var toNode = GuidToModel(toNodeGuid);
|
|
|
|
|
|
if (fromNode is null) return;
|
|
|
|
|
|
if (toNode is null) return;
|
2024-09-17 21:43:49 +08:00
|
|
|
|
RemoteConnect(fromNode, toNode, connectionType);
|
2024-09-16 21:38:34 +08:00
|
|
|
|
|
2024-09-17 21:43:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 移除连接关系
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="fromNodeGuid">起始节点Model</param>
|
|
|
|
|
|
/// <param name="toNodeGuid">目标节点Model</param>
|
|
|
|
|
|
/// <param name="connectionType">连接关系</param>
|
|
|
|
|
|
/// <exception cref="NotImplementedException"></exception>
|
|
|
|
|
|
private void RemoteConnect(NodeModelBase fromNode, NodeModelBase toNode, ConnectionType connectionType)
|
|
|
|
|
|
{
|
2024-09-16 21:38:34 +08:00
|
|
|
|
fromNode.SuccessorNodes[connectionType].Remove(toNode);
|
|
|
|
|
|
toNode.PreviousNodes[connectionType].Remove(fromNode);
|
2024-09-17 21:43:49 +08:00
|
|
|
|
if(toNode is SingleFlipflopNode flipflopNode)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (flowStarter?.FlowState != RunState.Completion
|
|
|
|
|
|
&& flipflopNode.NotExitPreviousNode())
|
|
|
|
|
|
{
|
|
|
|
|
|
// 被父节点移除连接关系的子节点若为触发器,且无上级节点,则当前流程正在运行,则加载到运行环境中
|
|
|
|
|
|
flowStarter?.AddFlipflopInRuning(flipflopNode, this);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 通知UI
|
|
|
|
|
|
OnNodeConnectChange?.Invoke(new NodeConnectChangeEventArgs(fromNode.Guid,
|
|
|
|
|
|
toNode.Guid,
|
2024-09-16 21:38:34 +08:00
|
|
|
|
connectionType,
|
|
|
|
|
|
NodeConnectChangeEventArgs.ConnectChangeType.Remote));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 获取方法描述
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public bool TryGetMethodDetails(string name, out MethodDetails? md)
|
|
|
|
|
|
{
|
2024-09-17 14:20:27 +08:00
|
|
|
|
var isPass = false;
|
|
|
|
|
|
if (!string.IsNullOrEmpty(name))
|
|
|
|
|
|
{
|
|
|
|
|
|
md = MethodDetailss.FirstOrDefault(it => it.MethodName == name);
|
|
|
|
|
|
return md != null;
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
2024-09-16 21:38:34 +08:00
|
|
|
|
{
|
2024-09-17 14:20:27 +08:00
|
|
|
|
md = null;
|
2024-09-16 21:38:34 +08:00
|
|
|
|
return false;
|
|
|
|
|
|
}
|
2024-09-17 14:20:27 +08:00
|
|
|
|
|
|
|
|
|
|
|
2024-09-16 21:38:34 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 设置起点控件
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="newNodeGuid"></param>
|
|
|
|
|
|
public void SetStartNode(string newNodeGuid)
|
|
|
|
|
|
{
|
2024-09-22 17:37:32 +08:00
|
|
|
|
var newStartNodeModel = GuidToModel(newNodeGuid);
|
|
|
|
|
|
if (newStartNodeModel is null) return;
|
2024-09-22 14:10:13 +08:00
|
|
|
|
SetStartNode(newStartNodeModel);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 中断指定节点,并指定中断等级。
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="nodeGuid">被中断的目标节点Guid</param>
|
|
|
|
|
|
/// <param name="interruptClass">中断级别</param>
|
|
|
|
|
|
/// <returns>操作是否成功</returns>
|
2024-09-22 17:37:32 +08:00
|
|
|
|
public bool SetNodeInterrupt(string nodeGuid, InterruptClass interruptClass)
|
2024-09-22 14:10:13 +08:00
|
|
|
|
{
|
2024-09-22 17:37:32 +08:00
|
|
|
|
var nodeModel = GuidToModel(nodeGuid);
|
|
|
|
|
|
if (nodeModel is null) return false;
|
|
|
|
|
|
if (interruptClass == InterruptClass.None)
|
|
|
|
|
|
{
|
|
|
|
|
|
nodeModel.CancelInterrupt();
|
|
|
|
|
|
}
|
|
|
|
|
|
else if (interruptClass == InterruptClass.Branch)
|
|
|
|
|
|
{
|
|
|
|
|
|
nodeModel.DebugSetting.CancelInterruptCallback?.Invoke();
|
|
|
|
|
|
nodeModel.DebugSetting.GetInterruptTask = () =>
|
|
|
|
|
|
{
|
|
|
|
|
|
TriggerInterrupt(nodeGuid, "", InterruptTriggerEventArgs.InterruptTriggerType.Monitor);
|
|
|
|
|
|
return ChannelFlowInterrupt.GetOrCreateChannelAsync(nodeGuid);
|
|
|
|
|
|
};
|
|
|
|
|
|
nodeModel.DebugSetting.CancelInterruptCallback = () =>
|
|
|
|
|
|
{
|
|
|
|
|
|
ChannelFlowInterrupt.TriggerSignal(nodeGuid);
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
else if (interruptClass == InterruptClass.Global) // 全局……做不了omg
|
|
|
|
|
|
{
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
2024-09-22 14:10:13 +08:00
|
|
|
|
nodeModel.DebugSetting.InterruptClass = interruptClass;
|
|
|
|
|
|
OnNodeInterruptStateChange.Invoke(new NodeInterruptStateChangeEventArgs(nodeGuid, interruptClass));
|
|
|
|
|
|
return true;
|
2024-09-22 17:37:32 +08:00
|
|
|
|
}
|
2024-09-22 14:10:13 +08:00
|
|
|
|
|
2024-09-22 17:37:32 +08:00
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 添加表达式中断
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="nodeGuid"></param>
|
|
|
|
|
|
/// <param name="expression"></param>
|
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
|
public bool AddInterruptExpression(string nodeGuid, string expression)
|
|
|
|
|
|
{
|
|
|
|
|
|
var nodeModel = GuidToModel(nodeGuid);
|
|
|
|
|
|
if (nodeModel is null) return false;
|
|
|
|
|
|
if (string.IsNullOrEmpty(expression))
|
|
|
|
|
|
{
|
|
|
|
|
|
nodeModel.DebugSetting.InterruptExpressions.Clear();// 暂时删除,等UI做好了
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (nodeModel.DebugSetting.InterruptExpressions.Contains(expression))
|
|
|
|
|
|
{
|
|
|
|
|
|
Console.WriteLine("表达式已存在");
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
nodeModel.DebugSetting.InterruptExpressions.Clear();// 暂时删除,等UI做好了
|
|
|
|
|
|
nodeModel.DebugSetting.InterruptExpressions.Add(expression);
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
2024-09-22 14:10:13 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-09-22 17:37:32 +08:00
|
|
|
|
|
2024-09-22 14:10:13 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 监视节点的数据
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="nodeGuid">需要监视的节点Guid</param>
|
|
|
|
|
|
public void SetNodeFLowDataMonitorState(string nodeGuid, bool isMonitor)
|
|
|
|
|
|
{
|
2024-09-22 17:37:32 +08:00
|
|
|
|
var nodeModel = GuidToModel(nodeGuid);
|
|
|
|
|
|
if (nodeModel is null) return;
|
2024-09-22 14:10:13 +08:00
|
|
|
|
nodeModel.DebugSetting.IsMonitorFlowData = isMonitor;
|
2024-09-22 17:37:32 +08:00
|
|
|
|
|
|
|
|
|
|
if (isMonitor)
|
|
|
|
|
|
{
|
|
|
|
|
|
var obj = nodeModel.GetFlowData();
|
|
|
|
|
|
if(obj is not null)
|
|
|
|
|
|
{
|
|
|
|
|
|
FlowDataNotification(nodeGuid, obj);
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-09-22 14:10:13 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 节点数据更新通知
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="nodeGuid"></param>
|
2024-09-22 17:37:32 +08:00
|
|
|
|
public void FlowDataNotification(string nodeGuid, object flowData)
|
2024-09-22 14:10:13 +08:00
|
|
|
|
{
|
|
|
|
|
|
OnMonitorObjectChange?.Invoke(new MonitorObjectEventArgs(nodeGuid, flowData));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-09-22 17:37:32 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
///
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="nodeGuid">节点</param>
|
|
|
|
|
|
/// <param name="expression">表达式</param>
|
|
|
|
|
|
/// <param name="type">类型,0节点,1表达式</param>
|
|
|
|
|
|
public void TriggerInterrupt(string nodeGuid, string expression, InterruptTriggerEventArgs.InterruptTriggerType type)
|
|
|
|
|
|
{
|
|
|
|
|
|
OnInterruptTrigger?.Invoke(new InterruptTriggerEventArgs(nodeGuid, expression, type));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public Task<CancelType> GetOrCreateGlobalInterruptAsync()
|
|
|
|
|
|
{
|
|
|
|
|
|
IsGlobalInterrupt = true;
|
|
|
|
|
|
return ChannelFlowInterrupt.GetOrCreateChannelAsync(this.EnvName);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-09-22 14:10:13 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// Guid 转 NodeModel
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="nodeGuid">节点Guid</param>
|
|
|
|
|
|
/// <returns>节点Model</returns>
|
|
|
|
|
|
/// <exception cref="ArgumentNullException">无法获取节点、Guid/节点为null时报错</exception>
|
2024-09-22 17:37:32 +08:00
|
|
|
|
private NodeModelBase? GuidToModel(string nodeGuid)
|
2024-09-22 14:10:13 +08:00
|
|
|
|
{
|
|
|
|
|
|
if (string.IsNullOrEmpty(nodeGuid))
|
2024-09-17 14:20:27 +08:00
|
|
|
|
{
|
2024-09-22 17:37:32 +08:00
|
|
|
|
//throw new ArgumentNullException("not contains - Guid没有对应节点:" + (nodeGuid));
|
|
|
|
|
|
return null;
|
2024-09-17 14:20:27 +08:00
|
|
|
|
}
|
2024-09-22 14:10:13 +08:00
|
|
|
|
if (!Nodes.TryGetValue(nodeGuid, out NodeModelBase? nodeModel) || nodeModel is null)
|
2024-09-15 12:15:32 +08:00
|
|
|
|
{
|
2024-09-22 17:37:32 +08:00
|
|
|
|
//throw new ArgumentNullException("null - Guid存在对应节点,但节点为null:" + (nodeGuid));
|
|
|
|
|
|
return null;
|
2024-09-15 12:15:32 +08:00
|
|
|
|
}
|
2024-09-22 14:10:13 +08:00
|
|
|
|
return nodeModel;
|
2024-09-15 12:15:32 +08:00
|
|
|
|
}
|
2024-09-22 17:37:32 +08:00
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
|
|
#region 私有方法
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 加载指定路径的DLL文件
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="dllPath"></param>
|
|
|
|
|
|
private (Assembly?, List<MethodDetails>) LoadAssembly(string dllPath)
|
|
|
|
|
|
{
|
|
|
|
|
|
try
|
|
|
|
|
|
{
|
|
|
|
|
|
Assembly assembly = Assembly.LoadFrom(dllPath); // 加载DLL文件
|
|
|
|
|
|
Type[] types = assembly.GetTypes(); // 获取程序集中的所有类型
|
|
|
|
|
|
|
|
|
|
|
|
List<Type> scanTypes = assembly.GetTypes().Where(t => t.GetCustomAttribute<DynamicFlowAttribute>()?.Scan == true).ToList();
|
|
|
|
|
|
if (scanTypes.Count == 0)
|
|
|
|
|
|
{
|
|
|
|
|
|
return (null, []);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
List<MethodDetails> methodDetails = new List<MethodDetails>();
|
|
|
|
|
|
// 遍历扫描的类型
|
|
|
|
|
|
foreach (var item in scanTypes)
|
|
|
|
|
|
{
|
2024-09-15 19:48:27 +08:00
|
|
|
|
// 加载DLL,创建 MethodDetails、实例作用对象、委托方法
|
2024-09-15 12:15:32 +08:00
|
|
|
|
var itemMethodDetails = MethodDetailsHelperTmp.GetList(item, false);
|
|
|
|
|
|
methodDetails.AddRange(itemMethodDetails);
|
2024-09-16 21:38:34 +08:00
|
|
|
|
//foreach (var md in itemMethodDetails)
|
|
|
|
|
|
//{
|
|
|
|
|
|
// // var instanceType =
|
|
|
|
|
|
// // Activator.CreateInstance(md.ActingInstanceType);
|
|
|
|
|
|
// // SereinIoc.RegisterInstantiate(md.ActingInstance);
|
|
|
|
|
|
// SereinIoc.Register(md.ActingInstanceType);
|
|
|
|
|
|
//}
|
2024-09-15 12:15:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
LoadedAssemblies.Add(assembly); // 将加载的程序集添加到列表中
|
|
|
|
|
|
LoadedAssemblyPaths.Add(dllPath); // 记录加载的DLL路径
|
|
|
|
|
|
return (assembly, methodDetails);
|
|
|
|
|
|
}
|
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
|
{
|
|
|
|
|
|
Console.WriteLine(ex.ToString());
|
|
|
|
|
|
return (null, []);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-09-17 14:20:27 +08:00
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 创建节点
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="nodeBase"></param>
|
|
|
|
|
|
private NodeModelBase CreateNode(NodeControlType nodeControlType,MethodDetails? methodDetails = null)
|
|
|
|
|
|
{
|
|
|
|
|
|
// 确定创建的节点类型
|
|
|
|
|
|
Type? nodeType = nodeControlType switch
|
|
|
|
|
|
{
|
|
|
|
|
|
NodeControlType.Action => typeof(SingleActionNode),
|
|
|
|
|
|
NodeControlType.Flipflop => typeof(SingleFlipflopNode),
|
|
|
|
|
|
|
|
|
|
|
|
NodeControlType.ExpOp => typeof(SingleExpOpNode),
|
|
|
|
|
|
NodeControlType.ExpCondition => typeof(SingleConditionNode),
|
|
|
|
|
|
NodeControlType.ConditionRegion => typeof(CompositeConditionNode),
|
|
|
|
|
|
_ => null
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
if (nodeType == null)
|
|
|
|
|
|
{
|
|
|
|
|
|
throw new Exception($"节点类型错误[{nodeControlType}]");
|
|
|
|
|
|
}
|
|
|
|
|
|
// 生成实例
|
|
|
|
|
|
var nodeObj = Activator.CreateInstance(nodeType);
|
|
|
|
|
|
if (nodeObj is not NodeModelBase nodeBase)
|
|
|
|
|
|
{
|
|
|
|
|
|
throw new Exception($"无法创建目标节点类型的实例[{nodeControlType}]");
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 配置基础的属性
|
|
|
|
|
|
nodeBase.ControlType = nodeControlType;
|
|
|
|
|
|
if (methodDetails != null)
|
|
|
|
|
|
{
|
|
|
|
|
|
var md = methodDetails.Clone();
|
|
|
|
|
|
nodeBase.DisplayName = md.MethodTips;
|
|
|
|
|
|
nodeBase.MethodDetails = md;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 如果是触发器,则需要添加到专属集合中
|
|
|
|
|
|
if (nodeControlType == NodeControlType.Flipflop && nodeBase is SingleFlipflopNode flipflopNode)
|
|
|
|
|
|
{
|
|
|
|
|
|
var guid = flipflopNode.Guid;
|
|
|
|
|
|
if (!FlipflopNodes.Exists(it => it.Guid.Equals(guid)))
|
|
|
|
|
|
{
|
|
|
|
|
|
FlipflopNodes.Add(flipflopNode);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return nodeBase;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private bool TryAddNode(NodeModelBase nodeModel)
|
|
|
|
|
|
{
|
|
|
|
|
|
nodeModel.Guid ??= Guid.NewGuid().ToString();
|
|
|
|
|
|
Nodes[nodeModel.Guid] = nodeModel;
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 连接节点
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="fromNode">起始节点</param>
|
|
|
|
|
|
/// <param name="toNode">目标节点</param>
|
|
|
|
|
|
/// <param name="connectionType">连接关系</param>
|
|
|
|
|
|
private void ConnectNode(NodeModelBase fromNode, NodeModelBase toNode, ConnectionType connectionType)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (fromNode == null || toNode == null || fromNode == toNode)
|
|
|
|
|
|
{
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var ToExistOnFrom = true;
|
|
|
|
|
|
var FromExistInTo = true;
|
|
|
|
|
|
ConnectionType[] ct = [ConnectionType.IsSucceed,
|
|
|
|
|
|
ConnectionType.IsFail,
|
|
|
|
|
|
ConnectionType.IsError,
|
|
|
|
|
|
ConnectionType.Upstream];
|
|
|
|
|
|
foreach (ConnectionType ctType in ct)
|
|
|
|
|
|
{
|
|
|
|
|
|
var FToTo = fromNode.SuccessorNodes[ctType].Where(it => it.Guid.Equals(toNode.Guid)).ToArray();
|
|
|
|
|
|
var ToOnF = toNode.PreviousNodes[ctType].Where(it => it.Guid.Equals(fromNode.Guid)).ToArray();
|
|
|
|
|
|
ToExistOnFrom = FToTo.Length > 0;
|
|
|
|
|
|
FromExistInTo = ToOnF.Length > 0;
|
|
|
|
|
|
if (ToExistOnFrom && FromExistInTo)
|
|
|
|
|
|
{
|
|
|
|
|
|
Console.WriteLine("起始节点已与目标节点存在连接");
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
// 检查是否可能存在异常
|
|
|
|
|
|
if (!ToExistOnFrom && FromExistInTo)
|
|
|
|
|
|
{
|
|
|
|
|
|
Console.WriteLine("目标节点不是起始节点的子节点,起始节点却是目标节点的父节点");
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
else if (ToExistOnFrom && !FromExistInTo)
|
|
|
|
|
|
{
|
|
|
|
|
|
//
|
|
|
|
|
|
Console.WriteLine(" 起始节点不是目标节点的父节点,目标节点却是起始节点的子节点");
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
else // if (!ToExistOnFrom && !FromExistInTo)
|
|
|
|
|
|
{
|
|
|
|
|
|
// 可以正常连接
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
fromNode.SuccessorNodes[connectionType].Add(toNode); // 添加到起始节点的子分支
|
|
|
|
|
|
toNode.PreviousNodes[connectionType].Add(fromNode); // 添加到目标节点的父分支
|
|
|
|
|
|
OnNodeConnectChange?.Invoke(new NodeConnectChangeEventArgs(fromNode.Guid,
|
|
|
|
|
|
toNode.Guid,
|
|
|
|
|
|
connectionType,
|
2024-09-16 21:38:34 +08:00
|
|
|
|
NodeConnectChangeEventArgs.ConnectChangeType.Create)); // 通知UI
|
2024-09-15 12:15:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 更改起点节点
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="newStartNode"></param>
|
|
|
|
|
|
/// <param name="oldStartNode"></param>
|
|
|
|
|
|
private void SetStartNode(NodeModelBase newStartNode)
|
|
|
|
|
|
{
|
|
|
|
|
|
var oldNodeGuid = StartNode?.Guid;
|
|
|
|
|
|
StartNode = newStartNode;
|
|
|
|
|
|
OnStartNodeChange?.Invoke(new StartNodeChangeEventArgs(oldNodeGuid, StartNode.Guid));
|
|
|
|
|
|
}
|
2024-09-17 14:20:27 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-09-15 12:15:32 +08:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
2024-09-16 21:38:34 +08:00
|
|
|
|
#region 网络交互
|
2024-09-15 12:15:32 +08:00
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
2024-09-16 21:38:34 +08:00
|
|
|
|
}
|
2024-09-15 12:15:32 +08:00
|
|
|
|
public static class FlowFunc
|
|
|
|
|
|
{
|
|
|
|
|
|
public static Library.Entity.Library ToLibrary(this Assembly assembly)
|
|
|
|
|
|
{
|
|
|
|
|
|
return new Library.Entity.Library
|
|
|
|
|
|
{
|
|
|
|
|
|
Name = assembly.GetName().Name,
|
|
|
|
|
|
Path = assembly.Location,
|
|
|
|
|
|
};
|
|
|
|
|
|
}
|
2024-09-15 22:07:10 +08:00
|
|
|
|
|
|
|
|
|
|
public static ConnectionType ToContentType(this FlipflopStateType flowStateType)
|
|
|
|
|
|
{
|
|
|
|
|
|
return flowStateType switch
|
|
|
|
|
|
{
|
|
|
|
|
|
FlipflopStateType.Succeed => ConnectionType.IsSucceed,
|
|
|
|
|
|
FlipflopStateType.Fail => ConnectionType.IsFail,
|
|
|
|
|
|
FlipflopStateType.Error => ConnectionType.IsError,
|
|
|
|
|
|
FlipflopStateType.Cancel => ConnectionType.None,
|
|
|
|
|
|
_ => throw new NotImplementedException("未定义的流程状态")
|
|
|
|
|
|
};
|
|
|
|
|
|
}
|
2024-09-16 21:38:34 +08:00
|
|
|
|
|
2024-09-17 14:20:27 +08:00
|
|
|
|
|
|
|
|
|
|
public static Type? ControlTypeToModel(this NodeControlType nodeControlType )
|
|
|
|
|
|
{
|
|
|
|
|
|
// 确定创建的节点类型
|
|
|
|
|
|
Type? nodeType = nodeControlType switch
|
|
|
|
|
|
{
|
|
|
|
|
|
NodeControlType.Action => typeof(SingleActionNode),
|
|
|
|
|
|
NodeControlType.Flipflop => typeof(SingleFlipflopNode),
|
|
|
|
|
|
|
|
|
|
|
|
NodeControlType.ExpOp => typeof(SingleExpOpNode),
|
|
|
|
|
|
NodeControlType.ExpCondition => typeof(SingleConditionNode),
|
|
|
|
|
|
NodeControlType.ConditionRegion => typeof(CompositeConditionNode),
|
|
|
|
|
|
_ => null
|
|
|
|
|
|
};
|
|
|
|
|
|
return nodeType;
|
|
|
|
|
|
}
|
|
|
|
|
|
public static NodeControlType ModelToControlType(this NodeControlType nodeControlType)
|
|
|
|
|
|
{
|
|
|
|
|
|
var type = nodeControlType.GetType();
|
|
|
|
|
|
NodeControlType controlType = type switch
|
|
|
|
|
|
{
|
|
|
|
|
|
Type when type == typeof(SingleActionNode) => NodeControlType.Action,
|
|
|
|
|
|
Type when type == typeof(SingleFlipflopNode) => NodeControlType.Flipflop,
|
|
|
|
|
|
|
|
|
|
|
|
Type when type == typeof(SingleExpOpNode) => NodeControlType.ExpOp,
|
|
|
|
|
|
Type when type == typeof(SingleConditionNode) => NodeControlType.ExpCondition,
|
|
|
|
|
|
Type when type == typeof(CompositeConditionNode) => NodeControlType.ConditionRegion,
|
|
|
|
|
|
_ => NodeControlType.None,
|
|
|
|
|
|
};
|
|
|
|
|
|
return controlType;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-09-16 21:38:34 +08:00
|
|
|
|
public static bool NotExitPreviousNode(this SingleFlipflopNode node)
|
|
|
|
|
|
{
|
|
|
|
|
|
ConnectionType[] ct = [ConnectionType.IsSucceed,
|
|
|
|
|
|
ConnectionType.IsFail,
|
|
|
|
|
|
ConnectionType.IsError,
|
|
|
|
|
|
ConnectionType.Upstream];
|
|
|
|
|
|
foreach (ConnectionType ctType in ct)
|
|
|
|
|
|
{
|
|
|
|
|
|
if(node.PreviousNodes[ctType].Count > 0)
|
|
|
|
|
|
{
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
2024-09-15 12:15:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|