2024-10-20 12:10:57 +08:00
|
|
|
|
using Serein.Library;
|
|
|
|
|
|
using Serein.Library.Api;
|
2024-10-27 00:54:10 +08:00
|
|
|
|
using Serein.Library.FlowNode;
|
2024-10-20 12:10:57 +08:00
|
|
|
|
using Serein.Library.Utils;
|
|
|
|
|
|
using Serein.NodeFlow.Tool;
|
|
|
|
|
|
using System.Collections.Concurrent;
|
2024-10-27 00:54:10 +08:00
|
|
|
|
using System.Security.AccessControl;
|
|
|
|
|
|
using System.Threading.Channels;
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
|
|
|
|
|
namespace Serein.NodeFlow.Env
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 远程流程环境
|
|
|
|
|
|
/// </summary>
|
2024-12-23 23:19:10 +08:00
|
|
|
|
public class RemoteFlowEnvironment : ChannelFlowTrigger<string>, IFlowEnvironment
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 连接到远程环境后切换到的环境接口实现
|
|
|
|
|
|
/// </summary>
|
2024-10-27 00:54:10 +08:00
|
|
|
|
/// <param name="remoteMsgUtil">连接到远程环境后,本地环境自动切换到对应的环境实体</param>
|
2024-10-20 21:59:42 +08:00
|
|
|
|
/// <param name="uIContextOperation">远程环境下需要操作UI线程时,所提供的线程上下文封装工具</param>
|
2024-10-27 00:54:10 +08:00
|
|
|
|
public RemoteFlowEnvironment(RemoteMsgUtil remoteMsgUtil, UIContextOperation uIContextOperation)
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
2024-10-20 21:59:42 +08:00
|
|
|
|
this.UIContextOperation = uIContextOperation;
|
2024-10-27 00:54:10 +08:00
|
|
|
|
RemoteMsgUtil = remoteMsgUtil;
|
|
|
|
|
|
msgClient = new MsgControllerOfClient(this, remoteMsgUtil.SendAsync); // 这里提供的是主动发送消息的方法
|
|
|
|
|
|
remoteMsgUtil.EnvClient.MsgHandleHelper.AddModule(msgClient, (ex, send) =>
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
|
|
|
|
|
Console.WriteLine(ex);
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//private readonly Func<string, object?, Task> SendCommandAsync;
|
2024-10-27 00:54:10 +08:00
|
|
|
|
private readonly RemoteMsgUtil RemoteMsgUtil;
|
2024-10-20 12:10:57 +08:00
|
|
|
|
private readonly MsgControllerOfClient msgClient;
|
|
|
|
|
|
private readonly ConcurrentDictionary<string, MethodDetails> MethodDetailss = [];
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 环境加载的节点集合
|
|
|
|
|
|
/// Node Guid - Node Model
|
|
|
|
|
|
/// </summary>
|
2024-10-23 19:22:27 +08:00
|
|
|
|
private Dictionary<string, NodeModelBase> NodeModels { get; } = [];
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
|
|
|
|
|
public event LoadDllHandler OnDllLoad;
|
|
|
|
|
|
public event ProjectLoadedHandler OnProjectLoaded;
|
2024-12-12 20:31:50 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 项目准备保存
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public event ProjectSavingHandler? OnProjectSaving;
|
2024-10-20 12:10:57 +08:00
|
|
|
|
public event NodeConnectChangeHandler OnNodeConnectChange;
|
|
|
|
|
|
public event NodeCreateHandler OnNodeCreate;
|
2024-10-20 21:59:42 +08:00
|
|
|
|
public event NodeRemoveHandler OnNodeRemove;
|
2024-10-20 12:10:57 +08:00
|
|
|
|
public event StartNodeChangeHandler OnStartNodeChange;
|
|
|
|
|
|
public event FlowRunCompleteHandler OnFlowRunComplete;
|
|
|
|
|
|
public event MonitorObjectChangeHandler OnMonitorObjectChange;
|
|
|
|
|
|
public event NodeInterruptStateChangeHandler OnNodeInterruptStateChange;
|
|
|
|
|
|
public event ExpInterruptTriggerHandler OnInterruptTrigger;
|
|
|
|
|
|
public event IOCMembersChangedHandler OnIOCMembersChanged;
|
|
|
|
|
|
public event NodeLocatedHandler OnNodeLocated;
|
|
|
|
|
|
public event NodeMovedHandler OnNodeMoved;
|
|
|
|
|
|
public event EnvOutHandler OnEnvOut;
|
|
|
|
|
|
|
|
|
|
|
|
public ISereinIOC IOC => throw new NotImplementedException();
|
|
|
|
|
|
|
|
|
|
|
|
public string EnvName => FlowEnvironment.SpaceName;
|
|
|
|
|
|
|
|
|
|
|
|
public bool IsGlobalInterrupt => false;
|
|
|
|
|
|
|
2024-10-28 15:21:08 +08:00
|
|
|
|
public bool IsControlRemoteEnv => true;
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
2024-11-08 17:30:51 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 信息输出等级
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public InfoClass InfoClass { get; set; }
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
|
|
|
|
|
public RunState FlowState { get => throw new NotImplementedException(); set => throw new NotImplementedException(); }
|
|
|
|
|
|
public RunState FlipFlopState { get => throw new NotImplementedException(); set => throw new NotImplementedException(); }
|
|
|
|
|
|
|
|
|
|
|
|
public IFlowEnvironment CurrentEnv => this;
|
2024-10-20 21:59:42 +08:00
|
|
|
|
public UIContextOperation UIContextOperation { get; }
|
2024-10-27 00:54:10 +08:00
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 标示是否正在加载项目
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
private bool IsLoadingProject = false;
|
2024-10-28 15:21:08 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 表示是否正在加载节点
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
private bool IsLoadingNode = false;
|
2024-10-27 00:54:10 +08:00
|
|
|
|
|
2024-11-08 17:30:51 +08:00
|
|
|
|
//public void SetConsoleOut()
|
|
|
|
|
|
//{
|
|
|
|
|
|
// var logTextWriter = new LogTextWriter(msg =>
|
|
|
|
|
|
// {
|
|
|
|
|
|
// OnEnvOut?.Invoke(msg);
|
|
|
|
|
|
// });
|
|
|
|
|
|
// Console.SetOut(logTextWriter);
|
|
|
|
|
|
//}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 输出信息
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="message">日志内容</param>
|
|
|
|
|
|
/// <param name="type">日志类别</param>
|
|
|
|
|
|
/// <param name="class">日志级别</param>
|
|
|
|
|
|
public void WriteLine(InfoType type, string message, InfoClass @class = InfoClass.Trivial)
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
OnEnvOut?.Invoke(type, message);
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void WriteLineObjToJson(object obj)
|
|
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口:WriteLineObjToJson");
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public async Task StartRemoteServerAsync(int port = 7525)
|
|
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口:StartRemoteServerAsync");
|
|
|
|
|
|
await Task.CompletedTask;
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void StopRemoteServer()
|
|
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口:StopRemoteServer");
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public async Task<SereinProjectData> GetProjectInfoAsync()
|
|
|
|
|
|
{
|
|
|
|
|
|
var prjectInfo = await msgClient.SendAndWaitDataAsync<SereinProjectData>(EnvMsgTheme.GetProjectInfo); // 等待服务器返回项目信息
|
|
|
|
|
|
return prjectInfo;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-12-12 20:31:50 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 保存项目
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public void SaveProject()
|
|
|
|
|
|
{
|
|
|
|
|
|
OnProjectSaving?.Invoke(new ProjectSavingEventArgs());
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-11-03 18:28:16 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 远程环境下加载项目
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="flowEnvInfo"></param>
|
|
|
|
|
|
/// <param name="filePath"></param>
|
2024-10-20 12:10:57 +08:00
|
|
|
|
public void LoadProject(FlowEnvInfo flowEnvInfo, string filePath)
|
|
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, "加载远程环境");
|
2024-10-27 00:54:10 +08:00
|
|
|
|
IsLoadingProject = true;
|
|
|
|
|
|
#region DLL功能区创建
|
2024-10-20 12:10:57 +08:00
|
|
|
|
var libmds = flowEnvInfo.LibraryMds;
|
|
|
|
|
|
foreach (var lib in libmds)
|
|
|
|
|
|
{
|
2024-11-03 18:28:16 +08:00
|
|
|
|
NodeLibraryInfo nodeLibraryInfo = new NodeLibraryInfo
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
2024-11-03 18:28:16 +08:00
|
|
|
|
AssemblyName = lib.AssemblyName,
|
2024-10-20 12:10:57 +08:00
|
|
|
|
FilePath = "Remote",
|
2024-11-03 21:17:45 +08:00
|
|
|
|
FileName = "Remote",
|
2024-10-20 12:10:57 +08:00
|
|
|
|
};
|
|
|
|
|
|
var mdInfos = lib.Mds.ToList();
|
2024-11-03 18:28:16 +08:00
|
|
|
|
UIContextOperation?.Invoke(() => OnDllLoad?.Invoke(new LoadDllEventArgs(nodeLibraryInfo, mdInfos))); // 通知UI创建dll面板显示
|
2024-10-20 12:10:57 +08:00
|
|
|
|
foreach (var mdInfo in mdInfos)
|
|
|
|
|
|
{
|
|
|
|
|
|
MethodDetailss.TryAdd(mdInfo.MethodName, new MethodDetails(mdInfo)); // 从DLL读取时生成元数据
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-10-27 00:54:10 +08:00
|
|
|
|
#endregion
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
2024-10-27 00:54:10 +08:00
|
|
|
|
#region 加载节点数据,如果是区域控件,提前加载区域
|
2024-10-20 12:10:57 +08:00
|
|
|
|
var projectData = flowEnvInfo.Project;
|
|
|
|
|
|
List<(NodeModelBase, string[])> regionChildNodes = new List<(NodeModelBase, string[])>();
|
|
|
|
|
|
List<(NodeModelBase, PositionOfUI)> ordinaryNodes = new List<(NodeModelBase, PositionOfUI)>();
|
|
|
|
|
|
// 加载节点
|
|
|
|
|
|
foreach (var nodeInfo in projectData.Nodes)
|
|
|
|
|
|
{
|
|
|
|
|
|
var controlType = FlowFunc.GetNodeControlType(nodeInfo);
|
|
|
|
|
|
if (controlType == NodeControlType.None)
|
|
|
|
|
|
{
|
|
|
|
|
|
continue;
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
2024-10-22 00:13:13 +08:00
|
|
|
|
MethodDetails? methodDetails = null;
|
|
|
|
|
|
if (!string.IsNullOrEmpty(nodeInfo.MethodName))
|
|
|
|
|
|
{
|
|
|
|
|
|
MethodDetailss.TryGetValue(nodeInfo.MethodName, out methodDetails);// 加载远程环境时尝试获取方法信息
|
|
|
|
|
|
}
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
|
|
|
|
|
var nodeModel = FlowFunc.CreateNode(this, controlType, methodDetails); // 加载远程项目时创建节点
|
|
|
|
|
|
nodeModel.LoadInfo(nodeInfo); // 创建节点model
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (nodeModel is null)
|
|
|
|
|
|
{
|
|
|
|
|
|
nodeInfo.Guid = string.Empty;
|
|
|
|
|
|
continue;
|
|
|
|
|
|
}
|
|
|
|
|
|
TryAddNode(nodeModel); // 加载项目时将节点加载到环境中
|
|
|
|
|
|
if (nodeInfo.ChildNodeGuids?.Length > 0)
|
|
|
|
|
|
{
|
|
|
|
|
|
regionChildNodes.Add((nodeModel, nodeInfo.ChildNodeGuids));
|
2024-10-20 21:59:42 +08:00
|
|
|
|
UIContextOperation?.Invoke(() => OnNodeCreate?.Invoke(new NodeCreateEventArgs(nodeModel, nodeInfo.Position)));
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
ordinaryNodes.Add((nodeModel, nodeInfo.Position));
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-10-27 00:54:10 +08:00
|
|
|
|
#endregion
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
2024-10-27 00:54:10 +08:00
|
|
|
|
#region 加载区域中的节点
|
2024-10-20 12:10:57 +08:00
|
|
|
|
// 加载区域子项
|
|
|
|
|
|
foreach ((NodeModelBase region, string[] childNodeGuids) item in regionChildNodes)
|
|
|
|
|
|
{
|
|
|
|
|
|
foreach (var childNodeGuid in item.childNodeGuids)
|
|
|
|
|
|
{
|
2024-10-23 19:22:27 +08:00
|
|
|
|
NodeModels.TryGetValue(childNodeGuid, out NodeModelBase? childNode);
|
2024-10-20 12:10:57 +08:00
|
|
|
|
if (childNode is null)
|
|
|
|
|
|
{
|
|
|
|
|
|
// 节点尚未加载
|
|
|
|
|
|
continue;
|
|
|
|
|
|
}
|
|
|
|
|
|
// 存在节点
|
2024-10-20 21:59:42 +08:00
|
|
|
|
UIContextOperation?.Invoke(() => OnNodeCreate?.Invoke(new NodeCreateEventArgs(childNode, true, item.region.Guid)));
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-10-27 00:54:10 +08:00
|
|
|
|
#endregion
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
2024-10-27 00:54:10 +08:00
|
|
|
|
#region 加载普通的节点
|
2024-10-20 12:10:57 +08:00
|
|
|
|
// 加载节点
|
|
|
|
|
|
foreach ((NodeModelBase nodeModel, PositionOfUI 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;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
if (IsContinue) continue;
|
2024-10-20 21:59:42 +08:00
|
|
|
|
//OnNodeCreate?.Invoke(new NodeCreateEventArgs(item.nodeModel, item.position));
|
|
|
|
|
|
UIContextOperation?.Invoke(() => OnNodeCreate?.Invoke(new NodeCreateEventArgs(item.nodeModel, item.position)));
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
2024-10-27 00:54:10 +08:00
|
|
|
|
#endregion
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
2024-10-27 00:54:10 +08:00
|
|
|
|
#region 确定节点之间的连接关系
|
2024-10-20 12:10:57 +08:00
|
|
|
|
_ = Task.Run(async () =>
|
|
|
|
|
|
{
|
2024-10-27 00:54:10 +08:00
|
|
|
|
await Task.Delay(500);
|
|
|
|
|
|
#region 连接节点的调用关系
|
|
|
|
|
|
foreach (var nodeInfo in projectData.Nodes)
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
2024-10-27 00:54:10 +08:00
|
|
|
|
if (!NodeModels.TryGetValue(nodeInfo.Guid, out NodeModelBase? fromNode))
|
|
|
|
|
|
{
|
|
|
|
|
|
// 不存在对应的起始节点
|
|
|
|
|
|
continue;
|
|
|
|
|
|
}
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
|
|
|
|
|
|
2024-10-27 00:54:10 +08:00
|
|
|
|
List<(ConnectionInvokeType connectionType, string[] guids)> allToNodes = [(ConnectionInvokeType.IsSucceed,nodeInfo.TrueNodes),
|
2024-10-24 23:32:43 +08:00
|
|
|
|
(ConnectionInvokeType.IsFail, nodeInfo.FalseNodes),
|
|
|
|
|
|
(ConnectionInvokeType.IsError, nodeInfo.ErrorNodes),
|
|
|
|
|
|
(ConnectionInvokeType.Upstream, nodeInfo.UpstreamNodes)];
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
2024-10-27 00:54:10 +08:00
|
|
|
|
List<(ConnectionInvokeType, NodeModelBase[])> fromNodes = allToNodes.Where(info => info.guids.Length > 0)
|
|
|
|
|
|
.Select(info => (info.connectionType,
|
|
|
|
|
|
info.guids.Where(guid => NodeModels.ContainsKey(guid)).Select(guid => NodeModels[guid])
|
|
|
|
|
|
.ToArray()))
|
|
|
|
|
|
.ToList();
|
|
|
|
|
|
// 遍历每种类型的节点分支(四种)
|
|
|
|
|
|
foreach ((ConnectionInvokeType connectionType, NodeModelBase[] toNodes) item in fromNodes)
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
2024-10-27 00:54:10 +08:00
|
|
|
|
// 遍历当前类型分支的节点(确认连接关系)
|
|
|
|
|
|
foreach (var toNode in item.toNodes)
|
|
|
|
|
|
{
|
|
|
|
|
|
UIContextOperation?.Invoke(() => OnNodeConnectChange?.Invoke(new NodeConnectChangeEventArgs(fromNode.Guid,
|
|
|
|
|
|
toNode.Guid,
|
|
|
|
|
|
JunctionOfConnectionType.Invoke,
|
|
|
|
|
|
item.connectionType,
|
|
|
|
|
|
NodeConnectChangeEventArgs.ConnectChangeType.Create))); // 通知UI连接节点
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
#endregion
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
2024-10-27 00:54:10 +08:00
|
|
|
|
#region 连接节点的传参关系
|
|
|
|
|
|
foreach (var toNode in NodeModels.Values)
|
|
|
|
|
|
{
|
|
|
|
|
|
if(toNode.MethodDetails.ParameterDetailss is null)
|
|
|
|
|
|
{
|
|
|
|
|
|
continue;
|
|
|
|
|
|
}
|
|
|
|
|
|
for (var i = 0; i < toNode.MethodDetails.ParameterDetailss.Length; i++)
|
|
|
|
|
|
{
|
|
|
|
|
|
var pd = toNode.MethodDetails.ParameterDetailss[i];
|
|
|
|
|
|
if (!string.IsNullOrEmpty(pd.ArgDataSourceNodeGuid)
|
|
|
|
|
|
&& NodeModels.TryGetValue(pd.ArgDataSourceNodeGuid, out var fromNode))
|
|
|
|
|
|
{
|
|
|
|
|
|
UIContextOperation?.Invoke(() =>
|
|
|
|
|
|
OnNodeConnectChange?.Invoke(
|
|
|
|
|
|
new NodeConnectChangeEventArgs(
|
|
|
|
|
|
fromNode.Guid, // 从哪个节点开始
|
|
|
|
|
|
toNode.Guid, // 连接到那个节点
|
|
|
|
|
|
JunctionOfConnectionType.Arg,
|
|
|
|
|
|
(int)pd.Index, // 连接线的样式类型
|
|
|
|
|
|
pd.ArgDataSourceType,
|
|
|
|
|
|
NodeConnectChangeEventArgs.ConnectChangeType.Create // 是创建连接还是删除连接
|
|
|
|
|
|
))); // 通知UI
|
|
|
|
|
|
}
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-10-27 00:54:10 +08:00
|
|
|
|
#endregion
|
|
|
|
|
|
});
|
|
|
|
|
|
#endregion
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
2024-10-27 00:54:10 +08:00
|
|
|
|
SetStartNode(projectData.StartNode); // 设置流程起点
|
2024-10-22 00:13:13 +08:00
|
|
|
|
UIContextOperation?.Invoke(() =>
|
|
|
|
|
|
{
|
2024-10-27 00:54:10 +08:00
|
|
|
|
OnProjectLoaded?.Invoke(new ProjectLoadedEventArgs()); // 加载完成
|
2024-10-22 00:13:13 +08:00
|
|
|
|
});
|
2024-10-27 00:54:10 +08:00
|
|
|
|
IsLoadingProject = false;
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
private bool TryAddNode(NodeModelBase nodeModel)
|
|
|
|
|
|
{
|
2024-10-20 21:59:42 +08:00
|
|
|
|
//nodeModel.Guid ??= Guid.NewGuid().ToString();
|
2024-10-23 19:22:27 +08:00
|
|
|
|
NodeModels[nodeModel.Guid] = nodeModel;
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
|
|
|
|
|
// 如果是触发器,则需要添加到专属集合中
|
|
|
|
|
|
//if (nodeModel is SingleFlipflopNode flipflopNode)
|
|
|
|
|
|
//{
|
|
|
|
|
|
// var guid = flipflopNode.Guid;
|
|
|
|
|
|
// if (!FlipflopNodes.Exists(it => it.Guid.Equals(guid)))
|
|
|
|
|
|
// {
|
|
|
|
|
|
// FlipflopNodes.Add(flipflopNode);
|
|
|
|
|
|
// }
|
|
|
|
|
|
//}
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-10-24 23:32:43 +08:00
|
|
|
|
private void ConnectNode(NodeModelBase fromNode, NodeModelBase toNode, ConnectionInvokeType connectionType)
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
|
|
|
|
|
if (fromNode is null || toNode is null || fromNode == toNode)
|
|
|
|
|
|
{
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var ToExistOnFrom = true;
|
|
|
|
|
|
var FromExistInTo = true;
|
2024-10-24 23:32:43 +08:00
|
|
|
|
ConnectionInvokeType[] ct = [ConnectionInvokeType.IsSucceed,
|
|
|
|
|
|
ConnectionInvokeType.IsFail,
|
|
|
|
|
|
ConnectionInvokeType.IsError,
|
|
|
|
|
|
ConnectionInvokeType.Upstream];
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
|
|
|
|
|
|
2024-10-24 23:32:43 +08:00
|
|
|
|
foreach (ConnectionInvokeType ctType in ct)
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
|
|
|
|
|
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)
|
|
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.ERROR, "起始节点已与目标节点存在连接");
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
|
|
|
|
|
//return;
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
// 检查是否可能存在异常
|
|
|
|
|
|
if (!ToExistOnFrom && FromExistInTo)
|
|
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.ERROR, "目标节点不是起始节点的子节点,起始节点却是目标节点的父节点");
|
2024-10-20 12:10:57 +08:00
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
else if (ToExistOnFrom && !FromExistInTo)
|
|
|
|
|
|
{
|
|
|
|
|
|
//
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.ERROR, " 起始节点不是目标节点的父节点,目标节点却是起始节点的子节点");
|
2024-10-20 12:10:57 +08:00
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
else // if (!ToExistOnFrom && !FromExistInTo)
|
|
|
|
|
|
{
|
|
|
|
|
|
// 可以正常连接
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
fromNode.SuccessorNodes[connectionType].Add(toNode); // 添加到起始节点的子分支
|
|
|
|
|
|
toNode.PreviousNodes[connectionType].Add(fromNode); // 添加到目标节点的父分支
|
|
|
|
|
|
OnNodeConnectChange?.Invoke(new NodeConnectChangeEventArgs(fromNode.Guid,
|
|
|
|
|
|
toNode.Guid,
|
2024-10-24 23:32:43 +08:00
|
|
|
|
JunctionOfConnectionType.Invoke,
|
2024-10-20 12:10:57 +08:00
|
|
|
|
connectionType,
|
|
|
|
|
|
NodeConnectChangeEventArgs.ConnectChangeType.Create)); // 通知UI
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public async Task<FlowEnvInfo> GetEnvInfoAsync()
|
|
|
|
|
|
{
|
|
|
|
|
|
var envInfo = await msgClient.SendAndWaitDataAsync<FlowEnvInfo>(EnvMsgTheme.GetEnvInfo);
|
|
|
|
|
|
return envInfo;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-10-27 00:54:10 +08:00
|
|
|
|
public async Task<(bool, RemoteMsgUtil)> ConnectRemoteEnv(string addres, int port, string token)
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
|
|
|
|
|
await Console.Out.WriteLineAsync("远程环境尚未实现的接口:ConnectRemoteEnv");
|
|
|
|
|
|
return (false, null);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void ExitRemoteEnv()
|
|
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口:ExitRemoteEnv");
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-11-03 23:51:18 +08:00
|
|
|
|
public void LoadLibrary(string dllPath)
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
|
|
|
|
|
// 将dll文件发送到远程环境,由远程环境进行加载
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口:LoadDll");
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-11-03 23:51:18 +08:00
|
|
|
|
public bool UnloadLibrary(string assemblyName)
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
|
|
|
|
|
// 尝试移除远程环境中的加载了的依赖
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口:RemoteDll");
|
2024-10-20 12:10:57 +08:00
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void ClearAll()
|
|
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口:ClearAll");
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public async Task StartAsync()
|
|
|
|
|
|
{
|
|
|
|
|
|
// 远程环境下不需要UI上下文
|
|
|
|
|
|
await msgClient.SendAsync(EnvMsgTheme.StartFlow);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public async Task StartAsyncInSelectNode(string startNodeGuid)
|
|
|
|
|
|
{
|
2024-10-20 21:59:42 +08:00
|
|
|
|
_ = msgClient.SendAsync(EnvMsgTheme.StartFlowInSelectNode, new
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
|
|
|
|
|
nodeGuid = startNodeGuid
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public async void ExitFlow()
|
|
|
|
|
|
{
|
|
|
|
|
|
await msgClient.SendAsync(EnvMsgTheme.ExitFlow, null);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void MoveNode(string nodeGuid, double x, double y)
|
|
|
|
|
|
{
|
2024-10-23 19:22:27 +08:00
|
|
|
|
//UIContextOperation?.Invoke(() =>
|
|
|
|
|
|
//{
|
|
|
|
|
|
// OnNodeMoved?.Invoke(new NodeMovedEventArgs(nodeGuid, x, y));
|
|
|
|
|
|
//});
|
2024-10-20 12:10:57 +08:00
|
|
|
|
_ = msgClient.SendAsync(EnvMsgTheme.MoveNode,
|
|
|
|
|
|
new
|
|
|
|
|
|
{
|
|
|
|
|
|
nodeGuid,
|
|
|
|
|
|
x,
|
|
|
|
|
|
y
|
|
|
|
|
|
});
|
2024-10-23 19:22:27 +08:00
|
|
|
|
|
|
|
|
|
|
if(NodeModels.TryGetValue(nodeGuid, out var nodeModel))
|
|
|
|
|
|
{
|
|
|
|
|
|
nodeModel.Position.X = x;
|
|
|
|
|
|
nodeModel.Position.Y = y;
|
|
|
|
|
|
}
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-10-22 00:13:13 +08:00
|
|
|
|
|
2024-10-20 12:10:57 +08:00
|
|
|
|
public void SetStartNode(string nodeGuid)
|
|
|
|
|
|
{
|
|
|
|
|
|
_ = msgClient.SendAsync(EnvMsgTheme.SetStartNode, new
|
|
|
|
|
|
{
|
|
|
|
|
|
nodeGuid
|
|
|
|
|
|
});
|
2024-10-22 00:13:13 +08:00
|
|
|
|
//UIContextOperation?.Invoke(() => OnStartNodeChange?.Invoke(new StartNodeChangeEventArgs(nodeGuid,nodeGuid)));
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-10-28 22:18:48 +08:00
|
|
|
|
public async Task<object> InvokeNodeAsync(IDynamicContext context, string nodeGuid)
|
2024-10-24 23:32:43 +08:00
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现接口 InvokeNodeAsync");
|
2024-10-24 23:32:43 +08:00
|
|
|
|
_ = msgClient.SendAsync(EnvMsgTheme.SetStartNode, new
|
|
|
|
|
|
{
|
|
|
|
|
|
nodeGuid
|
|
|
|
|
|
});
|
|
|
|
|
|
return null;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-10-27 00:54:10 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 在两个节点之间创建方法调用关系
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="fromNodeGuid">起始节点Guid</param>
|
|
|
|
|
|
/// <param name="toNodeGuid">目标节点Guid</param>
|
|
|
|
|
|
/// <param name="fromNodeJunctionType">起始节点控制点</param>
|
|
|
|
|
|
/// <param name="toNodeJunctionType">目标节点控制点</param>
|
|
|
|
|
|
/// <param name="invokeType">决定了方法执行后的后继行为</param>
|
|
|
|
|
|
public async Task<bool> ConnectInvokeNodeAsync(string fromNodeGuid,
|
|
|
|
|
|
string toNodeGuid,
|
|
|
|
|
|
JunctionType fromNodeJunctionType,
|
|
|
|
|
|
JunctionType toNodeJunctionType,
|
|
|
|
|
|
ConnectionInvokeType invokeType)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (fromNodeJunctionType == JunctionType.Execute)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (toNodeJunctionType == JunctionType.NextStep)
|
|
|
|
|
|
{
|
|
|
|
|
|
(fromNodeGuid, toNodeGuid) = (toNodeGuid, fromNodeGuid);// 需要反转
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
return false; // 非预期的控制点连接
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
else if (fromNodeJunctionType == JunctionType.NextStep)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (toNodeJunctionType == JunctionType.Execute)
|
|
|
|
|
|
{
|
|
|
|
|
|
// 顺序正确无须反转
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
return false; // 非预期的控制点连接
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
else // 其它类型的控制点,排除
|
|
|
|
|
|
{
|
|
|
|
|
|
return false; // 非预期的控制点连接
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var sendObj = new
|
|
|
|
|
|
{
|
|
|
|
|
|
fromNodeGuid = fromNodeGuid,
|
|
|
|
|
|
toNodeGuid = toNodeGuid,
|
|
|
|
|
|
fromJunctionType = fromNodeJunctionType.ToString(),
|
|
|
|
|
|
toJunctionType = toNodeJunctionType.ToString(),
|
|
|
|
|
|
invokeType = invokeType.ToString(),
|
|
|
|
|
|
};
|
|
|
|
|
|
var result = await msgClient.SendAndWaitDataAsync<bool>(EnvMsgTheme.ConnectInvokeNode, sendObj);
|
|
|
|
|
|
if (result)
|
|
|
|
|
|
{
|
|
|
|
|
|
OnNodeConnectChange?.Invoke(new NodeConnectChangeEventArgs(fromNodeGuid,
|
|
|
|
|
|
toNodeGuid,
|
|
|
|
|
|
JunctionOfConnectionType.Invoke,
|
|
|
|
|
|
invokeType,
|
|
|
|
|
|
NodeConnectChangeEventArgs.ConnectChangeType.Create)); // 通知UI
|
|
|
|
|
|
}
|
|
|
|
|
|
return result;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 在两个节点之间创建参数传递关系
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="fromNodeGuid">起始节点Guid</param>
|
|
|
|
|
|
/// <param name="toNodeGuid">目标节点Guid</param>
|
|
|
|
|
|
/// <param name="fromNodeJunctionType">起始节点控制点</param>
|
|
|
|
|
|
/// <param name="toNodeJunctionType">目标节点控制点</param>
|
|
|
|
|
|
/// <param name="argSourceType">决定了方法参数来源</param>
|
|
|
|
|
|
/// <param name="argIndex">设置第几个参数</param>
|
|
|
|
|
|
public async Task<bool> ConnectArgSourceNodeAsync(string fromNodeGuid,
|
2024-10-24 23:32:43 +08:00
|
|
|
|
string toNodeGuid,
|
|
|
|
|
|
JunctionType fromNodeJunctionType,
|
|
|
|
|
|
JunctionType toNodeJunctionType,
|
2024-10-27 00:54:10 +08:00
|
|
|
|
ConnectionArgSourceType argSourceType,
|
2024-10-24 23:32:43 +08:00
|
|
|
|
int argIndex = 0)
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
2024-10-27 00:54:10 +08:00
|
|
|
|
|
|
|
|
|
|
// 正确的顺序:起始节点[返回值控制点] 向 目标节点[入参控制点] 发起连接
|
|
|
|
|
|
//Console.WriteLine();
|
|
|
|
|
|
//Console.WriteLine($"起始节点:{fromNodeGuid}");
|
|
|
|
|
|
//Console.WriteLine($"目标节点:{toNodeGuid}");
|
|
|
|
|
|
//Console.WriteLine($"链接请求:{(fromNodeJunctionType, toNodeJunctionType)}");
|
|
|
|
|
|
//Console.WriteLine((fromNodeJunctionType, toNodeJunctionType));
|
|
|
|
|
|
|
|
|
|
|
|
if (fromNodeJunctionType == JunctionType.ArgData)
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
2024-10-27 00:54:10 +08:00
|
|
|
|
if (toNodeJunctionType == JunctionType.ReturnData)
|
|
|
|
|
|
{
|
|
|
|
|
|
(fromNodeGuid, toNodeGuid) = (toNodeGuid, fromNodeGuid);// 需要反转
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
return false; // 非预期的控制点连接
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
else if (fromNodeJunctionType == JunctionType.ReturnData)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (toNodeJunctionType == JunctionType.ArgData)
|
|
|
|
|
|
{
|
|
|
|
|
|
// 顺序正确无须反转
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
return false; // 非预期的控制点连接
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
else // 其它类型的控制点,排除
|
|
|
|
|
|
{
|
|
|
|
|
|
return false; // 非预期的控制点连接
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var sendObj = new
|
|
|
|
|
|
{
|
|
|
|
|
|
fromNodeGuid = fromNodeGuid,
|
|
|
|
|
|
toNodeGuid = toNodeGuid,
|
|
|
|
|
|
fromJunctionType = fromNodeJunctionType.ToString(),
|
|
|
|
|
|
toJunctionType = toNodeJunctionType.ToString(),
|
|
|
|
|
|
argSourceType = argSourceType.ToString(),
|
|
|
|
|
|
argIndex = argIndex,
|
|
|
|
|
|
};
|
|
|
|
|
|
var result = await msgClient.SendAndWaitDataAsync<bool>(EnvMsgTheme.ConnectArgSourceNode, sendObj);
|
2024-10-20 21:59:42 +08:00
|
|
|
|
if (result)
|
|
|
|
|
|
{
|
|
|
|
|
|
OnNodeConnectChange?.Invoke(new NodeConnectChangeEventArgs(fromNodeGuid,
|
|
|
|
|
|
toNodeGuid,
|
2024-10-27 00:54:10 +08:00
|
|
|
|
JunctionOfConnectionType.Arg,
|
|
|
|
|
|
argIndex,
|
|
|
|
|
|
argSourceType,
|
2024-10-20 21:59:42 +08:00
|
|
|
|
NodeConnectChangeEventArgs.ConnectChangeType.Create)); // 通知UI
|
|
|
|
|
|
}
|
2024-10-20 12:10:57 +08:00
|
|
|
|
return result;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-10-27 00:54:10 +08:00
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 移除两个节点之间的方法调用关系
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="fromNodeGuid">起始节点</param>
|
|
|
|
|
|
/// <param name="toNodeGuid">目标节点</param>
|
|
|
|
|
|
/// <param name="invokeType">连接类型</param>
|
|
|
|
|
|
public async Task<bool> RemoveConnectInvokeAsync(string fromNodeGuid, string toNodeGuid, ConnectionInvokeType invokeType)
|
|
|
|
|
|
{
|
|
|
|
|
|
var result = await msgClient.SendAndWaitDataAsync<bool>(EnvMsgTheme.RemoveInvokeConnect, new
|
|
|
|
|
|
{
|
|
|
|
|
|
fromNodeGuid = fromNodeGuid,
|
|
|
|
|
|
toNodeGuid = toNodeGuid,
|
|
|
|
|
|
invokeType = invokeType.ToString(),
|
|
|
|
|
|
});
|
|
|
|
|
|
if (result)
|
|
|
|
|
|
{
|
|
|
|
|
|
UIContextOperation.Invoke(() =>
|
|
|
|
|
|
{
|
|
|
|
|
|
OnNodeConnectChange?.Invoke(new NodeConnectChangeEventArgs(fromNodeGuid,
|
|
|
|
|
|
toNodeGuid,
|
|
|
|
|
|
JunctionOfConnectionType.Invoke,
|
|
|
|
|
|
invokeType,
|
|
|
|
|
|
NodeConnectChangeEventArgs.ConnectChangeType.Remote));
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
return result;
|
|
|
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 移除连接节点之间参数传递的关系
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="fromNodeGuid">起始节点Guid</param>
|
|
|
|
|
|
/// <param name="toNodeGuid">目标节点Guid</param>
|
|
|
|
|
|
/// <param name="argIndex">连接到第几个参数</param>
|
|
|
|
|
|
public async Task<bool> RemoveConnectArgSourceAsync(string fromNodeGuid, string toNodeGuid, int argIndex)
|
|
|
|
|
|
{
|
|
|
|
|
|
var result = await msgClient.SendAndWaitDataAsync<bool>(EnvMsgTheme.RemoveArgSourceConnect, new
|
|
|
|
|
|
{
|
|
|
|
|
|
fromNodeGuid = fromNodeGuid,
|
|
|
|
|
|
toNodeGuid = toNodeGuid,
|
|
|
|
|
|
argIndex = argIndex,
|
|
|
|
|
|
});
|
|
|
|
|
|
if (result)
|
|
|
|
|
|
{
|
|
|
|
|
|
UIContextOperation.Invoke(() =>
|
|
|
|
|
|
{
|
|
|
|
|
|
OnNodeConnectChange?.Invoke(new NodeConnectChangeEventArgs(fromNodeGuid,
|
|
|
|
|
|
toNodeGuid,
|
|
|
|
|
|
JunctionOfConnectionType.Arg,
|
|
|
|
|
|
argIndex,
|
|
|
|
|
|
ConnectionArgSourceType.GetPreviousNodeData,
|
|
|
|
|
|
NodeConnectChangeEventArgs.ConnectChangeType.Remote)); // 通知UI
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
return result;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-12-12 22:57:15 +08:00
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 从节点信息集合批量加载节点控件
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="List<NodeInfo>">节点信息</param>
|
|
|
|
|
|
/// <param name="position">需要加载的位置</param>
|
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
|
public async Task LoadNodeInfosAsync(List<NodeInfo> nodeInfos)
|
|
|
|
|
|
{
|
|
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口(重要,会尽快实现):LoadNodeInfoAsync");
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-10-27 00:54:10 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 创建节点/区域/基础控件
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="nodeType">节点/区域/基础控件类型</param>
|
|
|
|
|
|
/// <param name="position">节点在画布上的位置(</param>
|
|
|
|
|
|
/// <param name="methodDetailsInfo">节点绑定的方法说明</param>
|
2024-10-20 12:10:57 +08:00
|
|
|
|
public async Task<NodeInfo> CreateNodeAsync(NodeControlType nodeControlType, PositionOfUI position, MethodDetailsInfo methodDetailsInfo = null)
|
|
|
|
|
|
{
|
2024-10-28 15:21:08 +08:00
|
|
|
|
IsLoadingNode = true;
|
2024-10-20 12:10:57 +08:00
|
|
|
|
var nodeInfo = await msgClient.SendAndWaitDataAsync<NodeInfo>(EnvMsgTheme.CreateNode, new
|
|
|
|
|
|
{
|
|
|
|
|
|
nodeType = nodeControlType.ToString(),
|
|
|
|
|
|
position = position,
|
|
|
|
|
|
mdInfo = methodDetailsInfo,
|
|
|
|
|
|
});
|
2024-10-28 15:21:08 +08:00
|
|
|
|
|
2024-10-22 00:13:13 +08:00
|
|
|
|
MethodDetails? methodDetails = null;
|
|
|
|
|
|
if (!string.IsNullOrEmpty(nodeInfo.MethodName))
|
|
|
|
|
|
{
|
|
|
|
|
|
MethodDetailss.TryGetValue(nodeInfo.MethodName, out methodDetails);// 加载远程环境时尝试获取方法信息
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//MethodDetailss.TryGetValue(methodDetailsInfo.MethodName, out var methodDetails);// 加载项目时尝试获取方法信息
|
2024-10-20 12:10:57 +08:00
|
|
|
|
var nodeModel = FlowFunc.CreateNode(this, nodeControlType, methodDetails); // 远程环境下加载节点
|
|
|
|
|
|
nodeModel.LoadInfo(nodeInfo);
|
2024-10-20 21:59:42 +08:00
|
|
|
|
TryAddNode(nodeModel);
|
2024-10-28 15:21:08 +08:00
|
|
|
|
IsLoadingNode = false;
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
|
|
|
|
|
// 通知UI更改
|
2024-10-22 00:13:13 +08:00
|
|
|
|
UIContextOperation.Invoke(() =>
|
|
|
|
|
|
{
|
|
|
|
|
|
OnNodeCreate?.Invoke(new NodeCreateEventArgs(nodeModel, position));
|
|
|
|
|
|
});
|
2024-10-20 12:10:57 +08:00
|
|
|
|
return nodeInfo;
|
|
|
|
|
|
}
|
2024-10-20 21:59:42 +08:00
|
|
|
|
public async Task<bool> RemoveNodeAsync(string nodeGuid)
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
2024-10-20 21:59:42 +08:00
|
|
|
|
var result = await msgClient.SendAndWaitDataAsync<bool>(EnvMsgTheme.RemoveNode, new
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
|
|
|
|
|
nodeGuid
|
|
|
|
|
|
});
|
2024-10-20 21:59:42 +08:00
|
|
|
|
if (result)
|
|
|
|
|
|
{
|
2024-10-22 00:13:13 +08:00
|
|
|
|
UIContextOperation.Invoke(() =>
|
|
|
|
|
|
{
|
|
|
|
|
|
OnNodeRemove?.Invoke(new NodeRemoveEventArgs(nodeGuid));
|
|
|
|
|
|
});
|
2024-10-20 21:59:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.ERROR, "删除失败");
|
2024-10-20 21:59:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
return result;
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void ActivateFlipflopNode(string nodeGuid)
|
|
|
|
|
|
{
|
|
|
|
|
|
_ = msgClient.SendAsync(EnvMsgTheme.ActivateFlipflopNode, new
|
|
|
|
|
|
{
|
|
|
|
|
|
nodeGuid
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void TerminateFlipflopNode(string nodeGuid)
|
|
|
|
|
|
{
|
|
|
|
|
|
_ = msgClient.SendAsync(EnvMsgTheme.TerminateFlipflopNode, new
|
|
|
|
|
|
{
|
|
|
|
|
|
nodeGuid
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-10-27 00:54:10 +08:00
|
|
|
|
public async Task<bool> SetNodeInterruptAsync(string nodeGuid, bool isInterrupt)
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
|
|
|
|
|
var state = await msgClient.SendAndWaitDataAsync<bool>(EnvMsgTheme.SetNodeInterrupt, // 设置节点中断
|
|
|
|
|
|
new
|
|
|
|
|
|
{
|
|
|
|
|
|
nodeGuid,
|
2024-10-27 00:54:10 +08:00
|
|
|
|
isInterrupt,
|
2024-10-20 12:10:57 +08:00
|
|
|
|
});
|
|
|
|
|
|
return state;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public async Task<bool> AddInterruptExpressionAsync(string key, string expression)
|
|
|
|
|
|
{
|
|
|
|
|
|
var state = await msgClient.SendAndWaitDataAsync<bool>(EnvMsgTheme.AddInterruptExpression, // 设置节点/对象的中断表达式
|
|
|
|
|
|
new
|
|
|
|
|
|
{
|
|
|
|
|
|
key,
|
|
|
|
|
|
expression,
|
|
|
|
|
|
});
|
|
|
|
|
|
return state;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void SetMonitorObjState(string key, bool isMonitor)
|
|
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口:SetMonitorObjState");
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public async Task<(bool, string[])> CheckObjMonitorStateAsync(string key)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (string.IsNullOrEmpty(key))
|
|
|
|
|
|
{
|
|
|
|
|
|
var exps = Array.Empty<string>();
|
|
|
|
|
|
return (false, exps);
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
var result = await msgClient.SendAndWaitDataAsync<(bool, string[])>(EnvMsgTheme.SetNodeInterrupt, // 检查并获取节点/对象是否正在监视、以及监视的表达式
|
|
|
|
|
|
new
|
|
|
|
|
|
{
|
|
|
|
|
|
key,
|
|
|
|
|
|
});
|
|
|
|
|
|
return result;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public async Task<ChannelFlowInterrupt.CancelType> GetOrCreateGlobalInterruptAsync()
|
|
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口:GetOrCreateGlobalInterruptAsync");
|
2024-10-20 12:10:57 +08:00
|
|
|
|
return ChannelFlowInterrupt.CancelType.Error;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-11-03 21:17:45 +08:00
|
|
|
|
public bool TryGetMethodDetailsInfo(string libraryName, string methodName, out MethodDetailsInfo mdInfo)
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口:TryGetMethodDetailsInfo");
|
2024-10-20 12:10:57 +08:00
|
|
|
|
mdInfo = null;
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-11-03 21:17:45 +08:00
|
|
|
|
public bool TryGetDelegateDetails(string libraryName, string methodName, out DelegateDetails del)
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口:TryGetDelegateDetails");
|
2024-10-20 12:10:57 +08:00
|
|
|
|
del = null;
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public void MonitorObjectNotification(string nodeGuid, object monitorData, MonitorObjectEventArgs.ObjSourceType sourceType)
|
|
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口:MonitorObjectNotification");
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void TriggerInterrupt(string nodeGuid, string expression, InterruptTriggerEventArgs.InterruptTriggerType type)
|
|
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口:TriggerInterrupt");
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void NodeLocated(string nodeGuid)
|
|
|
|
|
|
{
|
2024-10-20 21:59:42 +08:00
|
|
|
|
//Console.WriteLine("远程环境尚未实现的接口:NodeLocated");
|
|
|
|
|
|
UIContextOperation?.Invoke(() => OnNodeLocated?.Invoke(new NodeLocatedEventArgs(nodeGuid)));
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-11-02 22:11:38 +08:00
|
|
|
|
|
2024-10-20 12:10:57 +08:00
|
|
|
|
public async Task NotificationNodeValueChangeAsync(string nodeGuid, string path, object value)
|
|
|
|
|
|
{
|
2024-10-28 15:21:08 +08:00
|
|
|
|
if(IsLoadingProject || IsLoadingNode)
|
|
|
|
|
|
{
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, $"通知远程环境修改节点数据:{nodeGuid},name:{path},value:{value}");
|
2024-10-27 00:54:10 +08:00
|
|
|
|
|
2024-10-28 15:21:08 +08:00
|
|
|
|
_ = msgClient.SendAsync(EnvMsgTheme.ValueNotification, new
|
|
|
|
|
|
{
|
|
|
|
|
|
nodeGuid = nodeGuid,
|
|
|
|
|
|
path = path,
|
|
|
|
|
|
value = value.ToString(),
|
|
|
|
|
|
});
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
|
|
|
|
|
}
|
2024-11-02 16:48:40 +08:00
|
|
|
|
|
2024-11-02 22:11:38 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 改变可选参数的数目
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="nodeGuid">对应的节点Guid</param>
|
|
|
|
|
|
/// <param name="isAdd">true,增加参数;false,减少参数</param>
|
|
|
|
|
|
/// <param name="paramIndex">以哪个参数为模板进行拷贝,或删去某个参数(该参数必须为可选参数)</param>
|
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
|
public async Task<bool> ChangeParameter(string nodeGuid, bool isAdd, int paramIndex)
|
|
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口:ChangeParameter");
|
2024-11-02 22:11:38 +08:00
|
|
|
|
return false;
|
|
|
|
|
|
}
|
2024-11-02 16:48:40 +08:00
|
|
|
|
|
|
|
|
|
|
#region 流程依赖类库的接口
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 运行时加载
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="file">文件名</param>
|
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
|
public bool LoadNativeLibraryOfRuning(string file)
|
|
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口:LoadNativeLibraryOfRuning");
|
2024-11-02 16:48:40 +08:00
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 运行时加载指定目录下的类库
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="path">目录</param>
|
|
|
|
|
|
/// <param name="isRecurrence">是否递归加载</param>
|
|
|
|
|
|
public void LoadAllNativeLibraryOfRuning(string path, bool isRecurrence = true)
|
|
|
|
|
|
{
|
2024-11-08 17:30:51 +08:00
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口:LoadAllNativeLibraryOfRuning");
|
2024-11-02 16:48:40 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-12-09 22:57:06 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 添加或更新全局数据
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="keyName">数据名称</param>
|
|
|
|
|
|
/// <param name="data">数据集</param>
|
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
|
public object AddOrUpdateGlobalData(string keyName, object data)
|
|
|
|
|
|
{
|
|
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口:AddOrUpdateGlobalData");
|
|
|
|
|
|
return null;
|
|
|
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 获取全局数据
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="keyName">数据名称</param>
|
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
|
public object GetGlobalData(string keyName)
|
|
|
|
|
|
{
|
|
|
|
|
|
this.WriteLine(InfoType.INFO, "远程环境尚未实现的接口:GetGlobalData");
|
|
|
|
|
|
return null;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-11-02 16:48:40 +08:00
|
|
|
|
#endregion
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|