using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.IO; using System.Windows.Forms; using System.Drawing; using System.Drawing.Drawing2D; using System.Threading; using System.ComponentModel; using System.Reflection; using System.IO.Compression; /* MIT License Copyright (c) 2021 DebugST@crystal_lz Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ /* * create: 2020-12-08 * modify: 2021-04-12 * Author: Crystal_lz * blog: http://st233.com * Gitee: https://gitee.com/DebugST * Github: https://github.com/DebugST */ namespace ST.Library.UI.NodeEditor { public class STNodeEditor : Control { private const UInt32 WM_MOUSEHWHEEL = 0x020E; protected static readonly Type m_type_node = typeof(STNode); #region protected enum,struct -------------------------------------------------------------------------------------- protected enum CanvasAction //当前鼠标移动操作表示进行下列哪一个行为 { None, //无 MoveNode, //正在移动 Node ConnectOption, //正在连接 Option SelectRectangle, //正在选择矩形区域 DrawMarkDetails //正在绘制标记信息详情 } protected struct MagnetInfo { public bool XMatched; //X轴是否有磁铁匹配上 public bool YMatched; public int X; //与X轴那个数字匹配上 public int Y; public int OffsetX; //当前节点X位置与匹配上的X的相对偏移 public int OffsetY; } #endregion #region Properties ------------------------------------------------------------------------------------------------------ private float _CanvasOffsetX; /// /// 获取画布原点相对于控件 X 方向上的偏移位置 /// [Browsable(false)] public float CanvasOffsetX { get { return _CanvasOffsetX; } } private float _CanvasOffsetY; /// /// 获取画布原点相对于控件 Y 方向上的偏移位置 /// [Browsable(false)] public float CanvasOffsetY { get { return _CanvasOffsetY; } } private PointF _CanvasOffset; /// /// 获取画布原点相对于控件偏移位置 /// [Browsable(false)] public PointF CanvasOffset { get { _CanvasOffset.X = _CanvasOffsetX; _CanvasOffset.Y = _CanvasOffsetY; return _CanvasOffset; } } private Rectangle _CanvasValidBounds; /// /// 获取画布中的有被用到的有效区域 /// [Browsable(false)] public Rectangle CanvasValidBounds { get { return _CanvasValidBounds; } } private float _CanvasScale = 1; /// /// 获取画布的缩放比例 /// [Browsable(false)] public float CanvasScale { get { return _CanvasScale; } } private float _Curvature = 0.3F; /// /// 获取或设置 Option 之间连线的曲度 /// [Browsable(false)] public float Curvature { get { return _Curvature; } set { if (value < 0) value = 0; if (value > 1) value = 1; _Curvature = value; if (m_dic_gp_info.Count != 0) this.BuildLinePath(); } } private bool _ShowMagnet = true; /// /// 获取或设置移动画布中 Node 时候 是否启用磁铁效果 /// [Description("获取或设置移动画布中 Node 时候 是否启用磁铁效果"), DefaultValue(true)] public bool ShowMagnet { get { return _ShowMagnet; } set { _ShowMagnet = value; } } private bool _ShowBorder = true; /// /// 获取或设置 移动画布中是否显示 Node 边框 /// [Description("获取或设置 移动画布中是否显示 Node 边框"), DefaultValue(true)] public bool ShowBorder { get { return _ShowBorder; } set { _ShowBorder = value; this.Invalidate(); } } private bool _ShowGrid = true; /// /// 获取或设置画布中是否绘制背景网格线条 /// [Description("获取或设置画布中是否绘制背景网格线条"), DefaultValue(true)] public bool ShowGrid { get { return _ShowGrid; } set { _ShowGrid = value; this.Invalidate(); } } private bool _ShowLocation = true; /// /// 获取或设置是否在画布边缘显示超出视角的 Node 位置信息 /// [Description("获取或设置是否在画布边缘显示超出视角的 Node 位置信息"), DefaultValue(true)] public bool ShowLocation { get { return _ShowLocation; } set { _ShowLocation = value; this.Invalidate(); } } private STNodeCollection _Nodes; /// /// 获取画布中 Node 集合 /// [Browsable(false)] public STNodeCollection Nodes { get { return _Nodes; } } private STNode _ActiveNode; /// /// 获取当前画布中被选中的活动 Node /// [Browsable(false)] public STNode ActiveNode { get { return _ActiveNode; } //set { // if (value == _ActiveSelectedNode) return; // if (_ActiveSelectedNode != null) _ActiveSelectedNode.OnLostFocus(EventArgs.Empty); // _ActiveSelectedNode = value; // _ActiveSelectedNode.IsActive = true; // this.Invalidate(); // this.OnSelectedChanged(EventArgs.Empty); //} } private STNode _HoverNode; /// /// 获取当前画布中鼠标悬停的 Node /// [Browsable(false)] public STNode HoverNode { get { return _HoverNode; } } //========================================color================================ private Color _GridColor = Color.Black; /// /// 获取或设置绘制画布背景时 网格线条颜色 /// [Description("获取或设置绘制画布背景时 网格线条颜色"), DefaultValue(typeof(Color), "Black")] public Color GridColor { get { return _GridColor; } set { _GridColor = value; this.Invalidate(); } } private Color _BorderColor = Color.Black; /// /// 获取或设置画布中 Node 边框颜色 /// [Description("获取或设置画布中 Node 边框颜色"), DefaultValue(typeof(Color), "Black")] public Color BorderColor { get { return _BorderColor; } set { _BorderColor = value; if (m_img_border != null) m_img_border.Dispose(); m_img_border = this.CreateBorderImage(value); this.Invalidate(); } } private Color _BorderHoverColor = Color.Gray; /// /// 获取或设置画布中悬停 Node 边框颜色 /// [Description("获取或设置画布中悬停 Node 边框颜色"), DefaultValue(typeof(Color), "Gray")] public Color BorderHoverColor { get { return _BorderHoverColor; } set { _BorderHoverColor = value; if (m_img_border_hover != null) m_img_border_hover.Dispose(); m_img_border_hover = this.CreateBorderImage(value); this.Invalidate(); } } private Color _BorderSelectedColor = Color.Orange; /// /// 获取或设置画布中选中 Node 边框颜色 /// [Description("获取或设置画布中选中 Node 边框颜色"), DefaultValue(typeof(Color), "Orange")] public Color BorderSelectedColor { get { return _BorderSelectedColor; } set { _BorderSelectedColor = value; if (m_img_border_selected != null) m_img_border_selected.Dispose(); m_img_border_selected = this.CreateBorderImage(value); this.Invalidate(); } } private Color _BorderActiveColor = Color.OrangeRed; /// /// 获取或设置画布中活动 Node 边框颜色 /// [Description("获取或设置画布中活动 Node 边框颜色"), DefaultValue(typeof(Color), "OrangeRed")] public Color BorderActiveColor { get { return _BorderActiveColor; } set { _BorderActiveColor = value; if (m_img_border_active != null) m_img_border_active.Dispose(); m_img_border_active = this.CreateBorderImage(value); this.Invalidate(); } } private Color _MarkForeColor = Color.White; /// /// 获取或设置画布绘制 Node 标记详情采用的前景色 /// [Description("获取或设置画布绘制 Node 标记详情采用的前景色"), DefaultValue(typeof(Color), "White")] public Color MarkForeColor { get { return _MarkBackColor; } set { _MarkBackColor = value; this.Invalidate(); } } private Color _MarkBackColor = Color.FromArgb(180, Color.Black); /// /// 获取或设置画布绘制 Node 标记详情采用的背景色 /// [Description("获取或设置画布绘制 Node 标记详情采用的背景色")] public Color MarkBackColor { get { return _MarkBackColor; } set { _MarkBackColor = value; this.Invalidate(); } } private Color _MagnetColor = Color.Lime; /// /// 获取或设置画布中移动 Node 时候 磁铁标记颜色 /// [Description("获取或设置画布中移动 Node 时候 磁铁标记颜色"), DefaultValue(typeof(Color), "Lime")] public Color MagnetColor { get { return _MagnetColor; } set { _MagnetColor = value; } } private Color _SelectedRectangleColor = Color.DodgerBlue; /// /// 获取或设置画布中选择矩形区域的颜色 /// [Description("获取或设置画布中选择矩形区域的颜色"), DefaultValue(typeof(Color), "DodgerBlue")] public Color SelectedRectangleColor { get { return _SelectedRectangleColor; } set { _SelectedRectangleColor = value; } } private Color _HighLineColor = Color.Cyan; /// /// 获取或设置画布中高亮连线的颜色 /// [Description("获取或设置画布中高亮连线的颜色"), DefaultValue(typeof(Color), "Cyan")] public Color HighLineColor { get { return _HighLineColor; } set { _HighLineColor = value; } } private Color _LocationForeColor = Color.Red; /// /// 获取或设置画布中边缘位置提示区域前景色 /// [Description("获取或设置画布中边缘位置提示区域前景色"), DefaultValue(typeof(Color), "Red")] public Color LocationForeColor { get { return _LocationForeColor; } set { _LocationForeColor = value; this.Invalidate(); } } private Color _LocationBackColor = Color.FromArgb(120, Color.Black); /// /// 获取或设置画布中边缘位置提示区域背景色 /// [Description("获取或设置画布中边缘位置提示区域背景色")] public Color LocationBackColor { get { return _LocationBackColor; } set { _LocationBackColor = value; this.Invalidate(); } } private Color _UnknownTypeColor = Color.Gray; /// /// 获取或设置画布中当 Node 中 Option 数据类型无法确定时应当使用的颜色 /// [Description("获取或设置画布中当 Node 中 Option 数据类型无法确定时应当使用的颜色"), DefaultValue(typeof(Color), "Gray")] public Color UnknownTypeColor { get { return _UnknownTypeColor; } set { _UnknownTypeColor = value; this.Invalidate(); } } private Dictionary _TypeColor = new Dictionary(); /// /// 获取或设置画布中 Node 中 Option 数据类型预设颜色 /// [Browsable(false)] public Dictionary TypeColor { get { return _TypeColor; } } #endregion #region protected properties ---------------------------------------------------------------------------------------- /// /// 当前鼠标在控件中的实时位置 /// protected Point m_pt_in_control; /// /// 当前鼠标在画布中的实时位置 /// protected PointF m_pt_in_canvas; /// /// 鼠标点击时在控件上的位置 /// protected Point m_pt_down_in_control; /// /// 鼠标点击时在画布中的位置 /// protected PointF m_pt_down_in_canvas; /// /// 用于鼠标点击移动画布时候 鼠标点下时候的画布坐标位置 /// protected PointF m_pt_canvas_old; /// /// 用于保存连线过程中保存点下 Option 的起点坐标 /// protected Point m_pt_dot_down; /// /// 用于保存连线过程中鼠标点下的起点Option 当MouseUP时候 确定是否连接此节点 /// protected STNodeOption m_option_down; /// /// 当前鼠标点下的 STNode /// protected STNode m_node_down; /// /// 当前鼠标是否位于控件中 /// protected bool m_mouse_in_control; #endregion public STNodeEditor() { this.SetStyle(ControlStyles.UserPaint, true); this.SetStyle(ControlStyles.ResizeRedraw, true); this.SetStyle(ControlStyles.AllPaintingInWmPaint, true); this.SetStyle(ControlStyles.OptimizedDoubleBuffer, true); this.SetStyle(ControlStyles.SupportsTransparentBackColor, true); this._Nodes = new STNodeCollection(this); this.BackColor = Color.FromArgb(255, 34, 34, 34); this.MinimumSize = new Size(100, 100); this.Size = new Size(200, 200); this.AllowDrop = true; m_real_canvas_x = this._CanvasOffsetX = 10; m_real_canvas_y = this._CanvasOffsetY = 10; } #region private fields -------------------------------------------------------------------------------------- private DrawingTools m_drawing_tools; private NodeFindInfo m_find = new NodeFindInfo(); private MagnetInfo m_mi = new MagnetInfo(); private RectangleF m_rect_select = new RectangleF(); //节点边框预设图案 private Image m_img_border; private Image m_img_border_hover; private Image m_img_border_selected; private Image m_img_border_active; //用于鼠标滚动或者触摸板移动画布时候的动画效果 该值为需要移动到的真实坐标地址 查看->MoveCanvasThread() private float m_real_canvas_x; private float m_real_canvas_y; //用于移动节点时候 保存鼠标点下时候选中的节点初始坐标 private Dictionary m_dic_pt_selected = new Dictionary(); //用于磁铁效果 移动节点时候 非选择节点的统计出来的需要参与磁铁效果的坐标 查看->BuildMagnetLocation() private List m_lst_magnet_x = new List(); private List m_lst_magnet_y = new List(); //用于磁铁效果 移动节点时候 活动选择节点统计出来需要参与磁铁效果的坐标 查看->CheckMagnet() private List m_lst_magnet_mx = new List(); private List m_lst_magnet_my = new List(); //用于鼠标滚动中计算时间触发间隔 根据间隔不同 画布产生的位移不同 查看->OnMouseWheel(),OnMouseHWheel() private DateTime m_dt_vw = DateTime.Now; private DateTime m_dt_hw = DateTime.Now; //移动鼠标过程中的当前行为 private CanvasAction m_ca; //保存已选中的节点 private HashSet m_hs_node_selected = new HashSet(); private bool m_is_process_mouse_event = true; //是否向下(Node or NodeControls)传递鼠标相关事件 如断开连接相关操作不应向下传递 private bool m_is_buildpath; //用于重绘过程中 判断该次是否要重新建立缓存连线的路径 private Pen m_p_line = new Pen(Color.Cyan, 2f); //用于绘制已经连接的线条 private Pen m_p_line_hover = new Pen(Color.Cyan, 4f); //用于绘制鼠标悬停时候的线条 private GraphicsPath m_gp_hover; //当前鼠标悬停的连线路径 private StringFormat m_sf = new StringFormat(); //文本格式 用于Mark绘制时候 设置文本格式 //保存每个连接线条与之对应的节点关系 private Dictionary m_dic_gp_info = new Dictionary(); //保存超出视觉区域的 Node 的位置 private List m_lst_node_out = new List(); //当前编辑器已加载的 Node 类型 用于从文件或者数据中加载节点使用 private Dictionary m_dic_type = new Dictionary(); private int m_time_alert; private int m_alpha_alert; private string m_str_alert; private Color m_forecolor_alert; private Color m_backcolor_alert; private DateTime m_dt_alert; private Rectangle m_rect_alert; private AlertLocation m_al; #endregion #region event ---------------------------------------------------------------------------------------------------- /// /// 活动的节点发生变化时候发生 /// [Description("活动的节点发生变化时候发生")] public event EventHandler ActiveChanged; /// /// 选择的节点发生变化时候发生 /// [Description("选择的节点发生变化时候发生")] public event EventHandler SelectedChanged; /// /// 悬停的节点发生变化时候发生 /// [Description("悬停的节点发生变化时候发生")] public event EventHandler HoverChanged; /// /// 当节点被添加时候发生 /// [Description("当节点被添加时候发生")] public event STNodeEditorEventHandler NodeAdded; /// /// 当节点被移除时候发生 /// [Description("当节点被移除时候发生")] public event STNodeEditorEventHandler NodeRemoved; /// /// 移动画布原点时候发生 /// [Description("移动画布原点时候发生")] public event EventHandler CanvasMoved; /// /// 缩放画布时候发生 /// [Description("缩放画布时候发生")] public event EventHandler CanvasScaled; /// /// 连接节点选项时候发生 /// [Description("连接节点选项时候发生")] public event STNodeEditorOptionEventHandler OptionConnected; /// /// 正在连接节点选项时候发生 /// [Description("正在连接节点选项时候发生")] public event STNodeEditorOptionEventHandler OptionConnecting; /// /// 断开节点选项时候发生 /// [Description("断开节点选项时候发生")] public event STNodeEditorOptionEventHandler OptionDisConnected; /// /// 正在断开节点选项时候发生 /// [Description("正在断开节点选项时候发生")] public event STNodeEditorOptionEventHandler OptionDisConnecting; protected virtual internal void OnSelectedChanged(EventArgs e) { if (this.SelectedChanged != null) this.SelectedChanged(this, e); } protected virtual void OnActiveChanged(EventArgs e) { if (this.ActiveChanged != null) this.ActiveChanged(this, e); } protected virtual void OnHoverChanged(EventArgs e) { if (this.HoverChanged != null) this.HoverChanged(this, e); } protected internal virtual void OnNodeAdded(STNodeEditorEventArgs e) { if (this.NodeAdded != null) this.NodeAdded(this, e); } protected internal virtual void OnNodeRemoved(STNodeEditorEventArgs e) { if (this.NodeRemoved != null) this.NodeRemoved(this, e); } protected virtual void OnCanvasMoved(EventArgs e) { if (this.CanvasMoved != null) this.CanvasMoved(this, e); } protected virtual void OnCanvasScaled(EventArgs e) { if (this.CanvasScaled != null) this.CanvasScaled(this, e); } protected internal virtual void OnOptionConnected(STNodeEditorOptionEventArgs e) { if (this.OptionConnected != null) this.OptionConnected(this, e); } protected internal virtual void OnOptionDisConnected(STNodeEditorOptionEventArgs e) { if (this.OptionDisConnected != null) this.OptionDisConnected(this, e); } protected internal virtual void OnOptionConnecting(STNodeEditorOptionEventArgs e) { if (this.OptionConnecting != null) this.OptionConnecting(this, e); } protected internal virtual void OnOptionDisConnecting(STNodeEditorOptionEventArgs e) { if (this.OptionDisConnecting != null) this.OptionDisConnecting(this, e); } #endregion event #region override ----------------------------------------------------------------------------------------------------- protected override void OnCreateControl() { m_drawing_tools = new DrawingTools() { Pen = new Pen(Color.Black, 1), SolidBrush = new SolidBrush(Color.Black) }; m_img_border = this.CreateBorderImage(this._BorderColor); m_img_border_active = this.CreateBorderImage(this._BorderActiveColor); m_img_border_hover = this.CreateBorderImage(this._BorderHoverColor); m_img_border_selected = this.CreateBorderImage(this._BorderSelectedColor); base.OnCreateControl(); new Thread(this.MoveCanvasThread) { IsBackground = true }.Start(); new Thread(this.ShowAlertThread) { IsBackground = true }.Start(); m_sf = new StringFormat(); m_sf.Alignment = StringAlignment.Near; m_sf.FormatFlags = StringFormatFlags.NoWrap; m_sf.SetTabStops(0, new float[] { 40 }); } protected override void WndProc(ref Message m) { base.WndProc(ref m); try { Point pt = new Point(((int)m.LParam) >> 16, (ushort)m.LParam); pt = this.PointToClient(pt); if (m.Msg == WM_MOUSEHWHEEL) { //获取水平滚动消息 MouseButtons mb = MouseButtons.None; int n = (ushort)m.WParam; if ((n & 0x0001) == 0x0001) mb |= MouseButtons.Left; if ((n & 0x0010) == 0x0010) mb |= MouseButtons.Middle; if ((n & 0x0002) == 0x0002) mb |= MouseButtons.Right; if ((n & 0x0020) == 0x0020) mb |= MouseButtons.XButton1; if ((n & 0x0040) == 0x0040) mb |= MouseButtons.XButton2; this.OnMouseHWheel(new MouseEventArgs(mb, 0, pt.X, pt.Y, ((int)m.WParam) >> 16)); } } catch { /*add code*/ } } protected override void OnPaint(PaintEventArgs e) { base.OnPaint(e); Graphics g = e.Graphics; g.Clear(this.BackColor); g.TextRenderingHint = System.Drawing.Text.TextRenderingHint.ClearTypeGridFit; m_drawing_tools.Graphics = g; SolidBrush brush = m_drawing_tools.SolidBrush; if (this._ShowGrid) this.OnDrawGrid(m_drawing_tools, this.Width, this.Height); g.TranslateTransform(this._CanvasOffsetX, this._CanvasOffsetY); //移动坐标系 g.ScaleTransform(this._CanvasScale, this._CanvasScale); //缩放绘图表面 this.OnDrawConnectedLine(m_drawing_tools); this.OnDrawNode(m_drawing_tools, this.ControlToCanvas(this.ClientRectangle)); if (m_ca == CanvasAction.ConnectOption) { //如果正在连线 m_drawing_tools.Pen.Color = this._HighLineColor; g.SmoothingMode = SmoothingMode.HighQuality; if (m_option_down.IsInput) this.DrawBezier(g, m_drawing_tools.Pen, m_pt_in_canvas, m_pt_dot_down, this._Curvature); else this.DrawBezier(g, m_drawing_tools.Pen, m_pt_dot_down, m_pt_in_canvas, this._Curvature); } //重置绘图坐标 我认为除了节点以外的其它 修饰相关的绘制不应该在Canvas坐标系中绘制 而应该使用控件的坐标进行绘制 不然会受到缩放比影响 g.ResetTransform(); switch (m_ca) { case CanvasAction.MoveNode: //移动过程中 绘制对齐参考线 if (this._ShowMagnet && this._ActiveNode != null) this.OnDrawMagnet(m_drawing_tools, m_mi); break; case CanvasAction.SelectRectangle: //绘制矩形选取 this.OnDrawSelectedRectangle(m_drawing_tools, this.CanvasToControl(m_rect_select)); break; case CanvasAction.DrawMarkDetails: //绘制标记信息详情 if (!string.IsNullOrEmpty(m_find.Mark)) this.OnDrawMark(m_drawing_tools); break; } if (this._ShowLocation) this.OnDrawNodeOutLocation(m_drawing_tools, this.Size, m_lst_node_out); this.OnDrawAlert(g); } protected override void OnMouseDown(MouseEventArgs e) { base.OnMouseDown(e); this.Focus(); m_ca = CanvasAction.None; m_mi.XMatched = m_mi.YMatched = false; m_pt_down_in_control = e.Location; m_pt_down_in_canvas.X = ((e.X - this._CanvasOffsetX) / this._CanvasScale); m_pt_down_in_canvas.Y = ((e.Y - this._CanvasOffsetY) / this._CanvasScale); m_pt_canvas_old.X = this._CanvasOffsetX; m_pt_canvas_old.Y = this._CanvasOffsetY; if (m_gp_hover != null && e.Button == MouseButtons.Right) { //断开连接 this.DisConnectionHover(); m_is_process_mouse_event = false; //终止MouseClick与MouseUp向下传递 return; } NodeFindInfo nfi = this.FindNodeFromPoint(m_pt_down_in_canvas); if (!string.IsNullOrEmpty(nfi.Mark)) { //如果点下的是标记信息 m_ca = CanvasAction.DrawMarkDetails; this.Invalidate(); return; } if (nfi.NodeOption != null) { //如果点下的Option的连接点 this.StartConnect(nfi.NodeOption); return; } if (nfi.Node != null) { nfi.Node.OnMouseDown(new MouseEventArgs(e.Button, e.Clicks, (int)m_pt_down_in_canvas.X - nfi.Node.Left, (int)m_pt_down_in_canvas.Y - nfi.Node.Top, e.Delta)); bool bCtrlDown = (Control.ModifierKeys & Keys.Control) == Keys.Control; if (bCtrlDown) { if (nfi.Node.IsSelected) { if (nfi.Node == this._ActiveNode) { this.SetActiveNode(null); } } else { nfi.Node.SetSelected(true, true); } return; } else if (!nfi.Node.IsSelected) { foreach (var n in m_hs_node_selected.ToArray()) n.SetSelected(false, false); } nfi.Node.SetSelected(true, false); //添加到已选择节点 this.SetActiveNode(nfi.Node); if (this.PointInRectangle(nfi.Node.TitleRectangle, m_pt_down_in_canvas.X, m_pt_down_in_canvas.Y)) { if (e.Button == MouseButtons.Right) { if (nfi.Node.ContextMenuStrip != null) { nfi.Node.ContextMenuStrip.Show(this.PointToScreen(e.Location)); } } else { m_dic_pt_selected.Clear(); lock (m_hs_node_selected) { foreach (STNode n in m_hs_node_selected) //记录已选择节点位置 如果需要移动已选中节点时候 将会有用 m_dic_pt_selected.Add(n, n.Location); } m_ca = CanvasAction.MoveNode; //如果点下的是节点的标题 则可以移动该节点 if (this._ShowMagnet && this._ActiveNode != null) this.BuildMagnetLocation(); //建立磁铁需要的坐标 如果需要移动已选中节点时候 将会有用 } } else m_node_down = nfi.Node; } else { this.SetActiveNode(null); foreach (var n in m_hs_node_selected.ToArray()) n.SetSelected(false, false);//没有点下任何东西 清空已经选择节点 m_ca = CanvasAction.SelectRectangle; //进入矩形区域选择模式 m_rect_select.Width = m_rect_select.Height = 0; m_node_down = null; } //this.SetActiveNode(nfi.Node); } protected override void OnMouseMove(MouseEventArgs e) { base.OnMouseMove(e); m_pt_in_control = e.Location; m_pt_in_canvas.X = ((e.X - this._CanvasOffsetX) / this._CanvasScale); m_pt_in_canvas.Y = ((e.Y - this._CanvasOffsetY) / this._CanvasScale); if (m_node_down != null) { m_node_down.OnMouseMove(new MouseEventArgs(e.Button, e.Clicks, (int)m_pt_in_canvas.X - m_node_down.Left, (int)m_pt_in_canvas.Y - m_node_down.Top, e.Delta)); return; } if (e.Button == MouseButtons.Middle) { //鼠标中键移动画布 this._CanvasOffsetX = m_real_canvas_x = m_pt_canvas_old.X + (e.X - m_pt_down_in_control.X); this._CanvasOffsetY = m_real_canvas_y = m_pt_canvas_old.Y + (e.Y - m_pt_down_in_control.Y); this.Invalidate(); return; } if (e.Button == MouseButtons.Left) { //如果鼠标左键点下 判断行为 m_gp_hover = null; switch (m_ca) { case CanvasAction.MoveNode: this.MoveNode(e.Location); return; //当前移动节点 case CanvasAction.ConnectOption: this.Invalidate(); return; //当前正在连线 case CanvasAction.SelectRectangle: //当前正在选取 m_rect_select.X = m_pt_down_in_canvas.X < m_pt_in_canvas.X ? m_pt_down_in_canvas.X : m_pt_in_canvas.X; m_rect_select.Y = m_pt_down_in_canvas.Y < m_pt_in_canvas.Y ? m_pt_down_in_canvas.Y : m_pt_in_canvas.Y; m_rect_select.Width = Math.Abs(m_pt_in_canvas.X - m_pt_down_in_canvas.X); m_rect_select.Height = Math.Abs(m_pt_in_canvas.Y - m_pt_down_in_canvas.Y); foreach (STNode n in this._Nodes) { n.SetSelected(m_rect_select.IntersectsWith(n.Rectangle), false); } this.Invalidate(); return; } } //若不存在行为 则判断鼠标下方是否存在其他对象 NodeFindInfo nfi = this.FindNodeFromPoint(m_pt_in_canvas); bool bRedraw = false; if (this._HoverNode != nfi.Node) { //鼠标悬停到Node上 if (nfi.Node != null) nfi.Node.OnMouseEnter(EventArgs.Empty); if (this._HoverNode != null) this._HoverNode.OnMouseLeave(new MouseEventArgs(e.Button, e.Clicks, (int)m_pt_in_canvas.X - this._HoverNode.Left, (int)m_pt_in_canvas.Y - this._HoverNode.Top, e.Delta)); this._HoverNode = nfi.Node; this.OnHoverChanged(EventArgs.Empty); bRedraw = true; } if (this._HoverNode != null) { this._HoverNode.OnMouseMove(new MouseEventArgs(e.Button, e.Clicks, (int)m_pt_in_canvas.X - this._HoverNode.Left, (int)m_pt_in_canvas.Y - this._HoverNode.Top, e.Delta)); m_gp_hover = null; } else { GraphicsPath gp = null; foreach (var v in m_dic_gp_info) { //判断鼠标是否悬停到连线路径上 if (v.Key.IsOutlineVisible(m_pt_in_canvas, m_p_line_hover)) { gp = v.Key; break; } } if (m_gp_hover != gp) { m_gp_hover = gp; bRedraw = true; } } if (bRedraw) this.Invalidate(); } protected override void OnMouseUp(MouseEventArgs e) { base.OnMouseUp(e); var nfi = this.FindNodeFromPoint(m_pt_in_canvas); switch (m_ca) { //鼠标抬起时候 判断行为 case CanvasAction.MoveNode: //若正在移动Node 则重新记录当前位置 foreach (STNode n in m_dic_pt_selected.Keys.ToList()) m_dic_pt_selected[n] = n.Location; break; case CanvasAction.ConnectOption: //若正在连线 则结束连接 if (e.Location == m_pt_down_in_control) break; if (nfi.NodeOption != null) { if (m_option_down.IsInput) nfi.NodeOption.ConnectOption(m_option_down); else m_option_down.ConnectOption(nfi.NodeOption); } break; } if (m_is_process_mouse_event && this._ActiveNode != null) { var mea = new MouseEventArgs(e.Button, e.Clicks, (int)m_pt_in_canvas.X - this._ActiveNode.Left, (int)m_pt_in_canvas.Y - this._ActiveNode.Top, e.Delta); this._ActiveNode.OnMouseUp(mea); m_node_down = null; } m_is_process_mouse_event = true; //当前为断开连接操作不进行事件传递 下次将接受事件 m_ca = CanvasAction.None; this.Invalidate(); } protected override void OnMouseEnter(EventArgs e) { base.OnMouseEnter(e); m_mouse_in_control = true; } protected override void OnMouseLeave(EventArgs e) { base.OnMouseLeave(e); m_mouse_in_control = false; if (this._HoverNode != null) this._HoverNode.OnMouseLeave(e); this._HoverNode = null; this.Invalidate(); } protected override void OnMouseWheel(MouseEventArgs e) { base.OnMouseWheel(e); if ((Control.ModifierKeys & Keys.Control) == Keys.Control) { float f = this._CanvasScale + (e.Delta < 0 ? -0.1f : 0.1f); this.ScaleCanvas(f, this.Width / 2, this.Height / 2); } else { if (!m_mouse_in_control) return; var nfi = this.FindNodeFromPoint(m_pt_in_canvas); if (this._HoverNode != null) { this._HoverNode.OnMouseWheel(new MouseEventArgs(e.Button, e.Clicks, (int)m_pt_in_canvas.X - this._HoverNode.Left, (int)m_pt_in_canvas.Y - this._HoverNode.Top, e.Delta)); return; } int t = (int)DateTime.Now.Subtract(m_dt_vw).TotalMilliseconds; if (t <= 30) t = 40; else if (t <= 100) t = 20; else if (t <= 150) t = 10; else if (t <= 300) t = 4; else t = 2; this.MoveCanvas(this._CanvasOffsetX, m_real_canvas_y + (e.Delta < 0 ? -t : t), true, CanvasMoveArgs.Top);//process mouse mid m_dt_vw = DateTime.Now; } } protected virtual void OnMouseHWheel(MouseEventArgs e) { if ((Control.ModifierKeys & Keys.Control) == Keys.Control) return; if (!m_mouse_in_control) return; if (this._HoverNode != null) { this._HoverNode.OnMouseWheel(new MouseEventArgs(e.Button, e.Clicks, (int)m_pt_in_canvas.X - this._HoverNode.Left, (int)m_pt_in_canvas.Y - this._HoverNode.Top, e.Delta)); return; } int t = (int)DateTime.Now.Subtract(m_dt_hw).TotalMilliseconds; if (t <= 30) t = 40; else if (t <= 100) t = 20; else if (t <= 150) t = 10; else if (t <= 300) t = 4; else t = 2; this.MoveCanvas(m_real_canvas_x + (e.Delta > 0 ? -t : t), this._CanvasOffsetY, true, CanvasMoveArgs.Left); m_dt_hw = DateTime.Now; } //===========================for node other event================================== protected override void OnMouseClick(MouseEventArgs e) { base.OnMouseClick(e); if (this._ActiveNode != null && m_is_process_mouse_event) { if (!this.PointInRectangle(this._ActiveNode.Rectangle, m_pt_in_canvas.X, m_pt_in_canvas.Y)) return; this._ActiveNode.OnMouseClick(new MouseEventArgs(e.Button, e.Clicks, (int)m_pt_down_in_canvas.X - this._ActiveNode.Left, (int)m_pt_down_in_canvas.Y - this._ActiveNode.Top, e.Delta)); } } protected override void OnKeyDown(KeyEventArgs e) { base.OnKeyDown(e); if (this._ActiveNode != null) this._ActiveNode.OnKeyDown(e); } protected override void OnKeyUp(KeyEventArgs e) { base.OnKeyUp(e); if (this._ActiveNode != null) this._ActiveNode.OnKeyUp(e); m_node_down = null; } protected override void OnKeyPress(KeyPressEventArgs e) { base.OnKeyPress(e); if (this._ActiveNode != null) this._ActiveNode.OnKeyPress(e); } #endregion protected override void OnDragEnter(DragEventArgs drgevent) { base.OnDragEnter(drgevent); if (this.DesignMode) return; if (drgevent.Data.GetDataPresent("STNodeType")) drgevent.Effect = DragDropEffects.Copy; else drgevent.Effect = DragDropEffects.None; } protected override void OnDragDrop(DragEventArgs drgevent) { base.OnDragDrop(drgevent); if (this.DesignMode) return; if (drgevent.Data.GetDataPresent("STNodeType")) { object data = drgevent.Data.GetData("STNodeType"); if (!(data is Type)) return; var t = (Type)data; if (!t.IsSubclassOf(typeof(STNode))) return; STNode node = (STNode)Activator.CreateInstance((t)); Point pt = new Point(drgevent.X, drgevent.Y); pt = this.PointToClient(pt); pt = this.ControlToCanvas(pt); node.Left = pt.X; node.Top = pt.Y; this.Nodes.Add(node); } } #region protected ---------------------------------------------------------------------------------------------------- /// /// 当绘制背景网格线时候发生 /// /// 绘制工具 /// 需要绘制宽度 /// 需要绘制高度 protected virtual void OnDrawGrid(DrawingTools dt, int nWidth, int nHeight) { Graphics g = dt.Graphics; using (Pen p_2 = new Pen(Color.FromArgb(65, this._GridColor))) { using (Pen p_1 = new Pen(Color.FromArgb(30, this._GridColor))) { float nIncrement = (20 * this._CanvasScale);             //网格间的间隔 根据比例绘制 int n = 5 - (int)(this._CanvasOffsetX / nIncrement); for (float f = this._CanvasOffsetX % nIncrement; f < nWidth; f += nIncrement) g.DrawLine((n++ % 5 == 0 ? p_2 : p_1), f, 0, f, nHeight); n = 5 - (int)(this._CanvasOffsetY / nIncrement); for (float f = this._CanvasOffsetY % nIncrement; f < nHeight; f += nIncrement) g.DrawLine((n++ % 5 == 0 ? p_2 : p_1), 0, f, nWidth, f); //原点两天线 p_1.Color = Color.FromArgb(this._Nodes.Count == 0 ? 255 : 120, this._GridColor); g.DrawLine(p_1, this._CanvasOffsetX, 0, this._CanvasOffsetX, nHeight); g.DrawLine(p_1, 0, this._CanvasOffsetY, nWidth, this._CanvasOffsetY); } } } /// /// 当绘制 Node 时候发生 /// /// 绘制工具 /// 可视画布区域大小 protected virtual void OnDrawNode(DrawingTools dt, Rectangle rect) { m_lst_node_out.Clear(); //清空超出视觉区域的 Node 的坐标 foreach (STNode n in this._Nodes) { if (this._ShowBorder) this.OnDrawNodeBorder(dt, n); n.OnDrawNode(dt); //调用 Node 进行自身绘制主体部分 if (!string.IsNullOrEmpty(n.Mark)) n.OnDrawMark(dt); //调用 Node 进行自身绘制 Mark 区域 if (!rect.IntersectsWith(n.Rectangle)) { m_lst_node_out.Add(n.Location); //判断此 Node 是否超出视觉区域 } } } /// /// 当绘制 Node 边框时候发生 /// /// 绘制工具 /// 目标node protected virtual void OnDrawNodeBorder(DrawingTools dt, STNode node) { Image img_border = null; if (this._ActiveNode == node) img_border = m_img_border_active; else if (node.IsSelected) img_border = m_img_border_selected; else if (this._HoverNode == node) img_border = m_img_border_hover; else img_border = m_img_border; this.RenderBorder(dt.Graphics, node.Rectangle, img_border); if (!string.IsNullOrEmpty(node.Mark)) this.RenderBorder(dt.Graphics, node.MarkRectangle, img_border); } /// /// 当绘制已连接路径时候发生 /// /// 绘制工具 protected virtual void OnDrawConnectedLine(DrawingTools dt) { Graphics g = dt.Graphics; g.SmoothingMode = SmoothingMode.HighQuality; m_p_line_hover.Color = Color.FromArgb(50, 0, 0, 0); var t = typeof(object); foreach (STNode n in this._Nodes) { foreach (STNodeOption op in n.OutputOptions) { if (op == STNodeOption.Empty) continue; if (op.DotColor != Color.Transparent) //确定线条颜色 m_p_line.Color = op.DotColor; else { if (op.DataType == t) m_p_line.Color = this._UnknownTypeColor; else m_p_line.Color = this._TypeColor.ContainsKey(op.DataType) ? this._TypeColor[op.DataType] : this._UnknownTypeColor;//value can not be null } foreach (var v in op.ConnectedOption) { this.DrawBezier(g, m_p_line_hover, op.DotLeft + op.DotSize, op.DotTop + op.DotSize / 2, v.DotLeft - 1, v.DotTop + v.DotSize / 2, this._Curvature); this.DrawBezier(g, m_p_line, op.DotLeft + op.DotSize, op.DotTop + op.DotSize / 2, v.DotLeft - 1, v.DotTop + v.DotSize / 2, this._Curvature); if (m_is_buildpath) { //如果当前绘制需要重新建立已连接的路径缓存 GraphicsPath gp = this.CreateBezierPath(op.DotLeft + op.DotSize, op.DotTop + op.DotSize / 2, v.DotLeft - 1, v.DotTop + v.DotSize / 2, this._Curvature); m_dic_gp_info.Add(gp, new ConnectionInfo() { Output = op, Input = v }); } } } } m_p_line_hover.Color = this._HighLineColor; if (m_gp_hover != null) { //如果当前有被悬停的连接路劲 则高亮绘制 g.DrawPath(m_p_line_hover, m_gp_hover); } m_is_buildpath = false; //重置标志 下次绘制时候 不再重新建立路径缓存 } /// /// 当绘制 Mark 详情信息时候发生 /// /// 绘制工具 protected virtual void OnDrawMark(DrawingTools dt) { Graphics g = dt.Graphics; SizeF sz = g.MeasureString(m_find.Mark, this.Font); //确认文字需要的大小 Rectangle rect = new Rectangle(m_pt_in_control.X + 15, m_pt_in_control.Y + 10, (int)sz.Width + 6, 4 + (this.Font.Height + 4) * m_find.MarkLines.Length); //sz.Height并没有考虑文字的行距 所以这里高度自己计算 if (rect.Right > this.Width) rect.X = this.Width - rect.Width; if (rect.Bottom > this.Height) rect.Y = this.Height - rect.Height; if (rect.X < 0) rect.X = 0; if (rect.Y < 0) rect.Y = 0; dt.SolidBrush.Color = this._MarkBackColor; g.SmoothingMode = SmoothingMode.None; g.FillRectangle(dt.SolidBrush, rect); //绘制背景区域 rect.Width--; rect.Height--; dt.Pen.Color = Color.FromArgb(255, this._MarkBackColor); g.DrawRectangle(dt.Pen, rect); dt.SolidBrush.Color = this._MarkForeColor; m_sf.LineAlignment = StringAlignment.Center; //g.SmoothingMode = SmoothingMode.HighQuality; rect.X += 2; rect.Width -= 3; rect.Height = this.Font.Height + 4; int nY = rect.Y + 2; for (int i = 0; i < m_find.MarkLines.Length; i++) { //绘制文字 rect.Y = nY + i * (this.Font.Height + 4); g.DrawString(m_find.MarkLines[i], this.Font, dt.SolidBrush, rect, m_sf); } } /// /// 当移动 Node 时候 需要显示对齐参考线时候发生 /// /// 绘制工具 /// 匹配的磁铁信息 protected virtual void OnDrawMagnet(DrawingTools dt, MagnetInfo mi) { if (this._ActiveNode == null) return; Graphics g = dt.Graphics; Pen pen = m_drawing_tools.Pen; SolidBrush brush = dt.SolidBrush; pen.Color = this._MagnetColor; brush.Color = Color.FromArgb(this._MagnetColor.A / 3, this._MagnetColor); g.SmoothingMode = SmoothingMode.None; int nL = this._ActiveNode.Left, nMX = this._ActiveNode.Left + this._ActiveNode.Width / 2, nR = this._ActiveNode.Right; int nT = this._ActiveNode.Top, nMY = this._ActiveNode.Top + this._ActiveNode.Height / 2, nB = this._ActiveNode.Bottom; if (mi.XMatched) g.DrawLine(pen, this.CanvasToControl(mi.X, true), 0, this.CanvasToControl(mi.X, true), this.Height); if (mi.YMatched) g.DrawLine(pen, 0, this.CanvasToControl(mi.Y, false), this.Width, this.CanvasToControl(mi.Y, false)); g.TranslateTransform(this._CanvasOffsetX, this._CanvasOffsetY); //移动坐标系 g.ScaleTransform(this._CanvasScale, this._CanvasScale); //缩放绘图表面 if (mi.XMatched) { //g.DrawLine(pen, this.CanvasToControl(mi.X, true), 0, this.CanvasToControl(mi.X, true), this.Height); foreach (STNode n in this._Nodes) { if (n.Left == mi.X || n.Right == mi.X || n.Left + n.Width / 2 == mi.X) { //g.DrawRectangle(pen, n.Left, n.Top, n.Width - 1, n.Height - 1); g.FillRectangle(brush, n.Rectangle); } } } if (mi.YMatched) { //g.DrawLine(pen, 0, this.CanvasToControl(mi.Y, false), this.Width, this.CanvasToControl(mi.Y, false)); foreach (STNode n in this._Nodes) { if (n.Top == mi.Y || n.Bottom == mi.Y || n.Top + n.Height / 2 == mi.Y) { //g.DrawRectangle(pen, n.Left, n.Top, n.Width - 1, n.Height - 1); g.FillRectangle(brush, n.Rectangle); } } } g.ResetTransform(); } /// /// 绘制选择的矩形区域 /// /// 绘制工具 /// 位于控件上的矩形区域 protected virtual void OnDrawSelectedRectangle(DrawingTools dt, RectangleF rectf) { Graphics g = dt.Graphics; SolidBrush brush = dt.SolidBrush; dt.Pen.Color = this._SelectedRectangleColor; g.DrawRectangle(dt.Pen, rectf.Left, rectf.Y, rectf.Width, rectf.Height); brush.Color = Color.FromArgb(this._SelectedRectangleColor.A / 3, this._SelectedRectangleColor); g.FillRectangle(brush, this.CanvasToControl(m_rect_select)); } /// /// 绘制超出视觉区域的 Node 位置提示信息 /// /// 绘制工具 /// 提示框边距 /// 超出视觉区域的 Node 位置信息 protected virtual void OnDrawNodeOutLocation(DrawingTools dt, Size sz, List lstPts) { Graphics g = dt.Graphics; SolidBrush brush = dt.SolidBrush; brush.Color = this._LocationBackColor; g.SmoothingMode = SmoothingMode.None; if (lstPts.Count == this._Nodes.Count && this._Nodes.Count != 0) { //如果超出个数和集合个数一样多 则全部超出 绘制外切矩形 g.FillRectangle(brush, this.CanvasToControl(this._CanvasValidBounds)); } g.FillRectangle(brush, 0, 0, 4, sz.Height); //绘制四边背景 g.FillRectangle(brush, sz.Width - 4, 0, 4, sz.Height); g.FillRectangle(brush, 4, 0, sz.Width - 8, 4); g.FillRectangle(brush, 4, sz.Height - 4, sz.Width - 8, 4); brush.Color = this._LocationForeColor; foreach (var v in lstPts) { //绘制点 var pt = this.CanvasToControl(v); if (pt.X < 0) pt.X = 0; if (pt.Y < 0) pt.Y = 0; if (pt.X > sz.Width) pt.X = sz.Width - 4; if (pt.Y > sz.Height) pt.Y = sz.Height - 4; g.FillRectangle(brush, pt.X, pt.Y, 4, 4); } } /// /// 绘制提示信息 /// /// 绘制工具 /// 需要绘制区域 /// 需要绘制文本 /// 信息前景色 /// 信息背景色 /// 信息位置 protected virtual void OnDrawAlert(DrawingTools dt, Rectangle rect, string strText, Color foreColor, Color backColor, AlertLocation al) { if (m_alpha_alert == 0) return; Graphics g = dt.Graphics; SolidBrush brush = dt.SolidBrush; g.SmoothingMode = SmoothingMode.None; brush.Color = backColor; dt.Pen.Color = brush.Color; g.FillRectangle(brush, rect); g.DrawRectangle(dt.Pen, rect.Left, rect.Top, rect.Width - 1, rect.Height - 1); brush.Color = foreColor; m_sf.Alignment = StringAlignment.Center; m_sf.LineAlignment = StringAlignment.Center; g.SmoothingMode = SmoothingMode.HighQuality; g.DrawString(strText, this.Font, brush, rect, m_sf); } /// /// 获取提示信息需要绘制的矩形区域 /// /// 绘图表面 /// 需要绘制文本 /// 信息位置 /// 矩形区域 protected virtual Rectangle GetAlertRectangle(Graphics g, string strText, AlertLocation al) { SizeF szf = g.MeasureString(m_str_alert, this.Font); Size sz = new Size((int)Math.Round(szf.Width + 10), (int)Math.Round(szf.Height + 4)); Rectangle rect = new Rectangle(4, this.Height - sz.Height - 4, sz.Width, sz.Height); switch (al) { case AlertLocation.Left: rect.Y = (this.Height - sz.Height) >> 1; break; case AlertLocation.Top: rect.Y = 4; rect.X = (this.Width - sz.Width) >> 1; break; case AlertLocation.Right: rect.X = this.Width - sz.Width - 4; rect.Y = (this.Height - sz.Height) >> 1; break; case AlertLocation.Bottom: rect.X = (this.Width - sz.Width) >> 1; break; case AlertLocation.Center: rect.X = (this.Width - sz.Width) >> 1; rect.Y = (this.Height - sz.Height) >> 1; break; case AlertLocation.LeftTop: rect.X = rect.Y = 4; break; case AlertLocation.RightTop: rect.Y = 4; rect.X = this.Width - sz.Width - 4; break; case AlertLocation.RightBottom: rect.X = this.Width - sz.Width - 4; break; } return rect; } #endregion protected #region internal internal void BuildLinePath() { foreach (var v in m_dic_gp_info) v.Key.Dispose(); m_dic_gp_info.Clear(); m_is_buildpath = true; this.Invalidate(); } internal void OnDrawAlert(Graphics g) { m_rect_alert = this.GetAlertRectangle(g, m_str_alert, m_al); Color clr_fore = Color.FromArgb((int)((float)m_alpha_alert / 255 * m_forecolor_alert.A), m_forecolor_alert); Color clr_back = Color.FromArgb((int)((float)m_alpha_alert / 255 * m_backcolor_alert.A), m_backcolor_alert); this.OnDrawAlert(m_drawing_tools, m_rect_alert, m_str_alert, clr_fore, clr_back, m_al); } internal void InternalAddSelectedNode(STNode node) { node.IsSelected = true; lock (m_hs_node_selected) m_hs_node_selected.Add(node); } internal void InternalRemoveSelectedNode(STNode node) { node.IsSelected = false; lock (m_hs_node_selected) m_hs_node_selected.Remove(node); } #endregion internal #region private ----------------------------------------------------------------------------------------------------- private void MoveCanvasThread() { bool bRedraw; while (true) { bRedraw = false; if (m_real_canvas_x != this._CanvasOffsetX) { float nx = m_real_canvas_x - this._CanvasOffsetX; float n = Math.Abs(nx) / 10; float nTemp = Math.Abs(nx); if (nTemp <= 4) n = 1; else if (nTemp <= 12) n = 2; else if (nTemp <= 30) n = 3; if (nTemp < 1) this._CanvasOffsetX = m_real_canvas_x; else this._CanvasOffsetX += nx > 0 ? n : -n; bRedraw = true; } if (m_real_canvas_y != this._CanvasOffsetY) { float ny = m_real_canvas_y - this._CanvasOffsetY; float n = Math.Abs(ny) / 10; float nTemp = Math.Abs(ny); if (nTemp <= 4) n = 1; else if (nTemp <= 12) n = 2; else if (nTemp <= 30) n = 3; if (nTemp < 1) this._CanvasOffsetY = m_real_canvas_y; else this._CanvasOffsetY += ny > 0 ? n : -n; bRedraw = true; } if (bRedraw) { m_pt_canvas_old.X = this._CanvasOffsetX; m_pt_canvas_old.Y = this._CanvasOffsetY; this.Invalidate(); Thread.Sleep(30); } else { Thread.Sleep(100); } } } private void ShowAlertThread() { while (true) { int nTime = m_time_alert - (int)DateTime.Now.Subtract(m_dt_alert).TotalMilliseconds; if (nTime > 0) { Thread.Sleep(nTime); continue; } if (nTime < -1000) { if (m_alpha_alert != 0) { m_alpha_alert = 0; this.Invalidate(); } Thread.Sleep(100); } else { m_alpha_alert = (int)(255 - (-nTime / 1000F) * 255); this.Invalidate(m_rect_alert); Thread.Sleep(50); } } } private Image CreateBorderImage(Color clr) { Image img = new Bitmap(12, 12); using (Graphics g = Graphics.FromImage(img)) { g.SmoothingMode = SmoothingMode.HighQuality; using (GraphicsPath gp = new GraphicsPath()) { gp.AddEllipse(new Rectangle(0, 0, 11, 11)); using (PathGradientBrush b = new PathGradientBrush(gp)) { b.CenterColor = Color.FromArgb(200, clr); b.SurroundColors = new Color[] { Color.FromArgb(10, clr) }; g.FillPath(b, gp); } } } return img; } private ConnectionStatus DisConnectionHover() { if (!m_dic_gp_info.ContainsKey(m_gp_hover)) return ConnectionStatus.DisConnected; ConnectionInfo ci = m_dic_gp_info[m_gp_hover]; var ret = ci.Output.DisConnectOption(ci.Input); //this.OnOptionDisConnected(new STNodeOptionEventArgs(ci.Output, ci.Input, ret)); if (ret == ConnectionStatus.DisConnected) { m_dic_gp_info.Remove(m_gp_hover); m_gp_hover.Dispose(); m_gp_hover = null; this.Invalidate(); } return ret; } private void StartConnect(STNodeOption op) { if (op.IsInput) { m_pt_dot_down.X = op.DotLeft; m_pt_dot_down.Y = op.DotTop + 5; } else { m_pt_dot_down.X = op.DotLeft + op.DotSize; m_pt_dot_down.Y = op.DotTop + 5; } m_ca = CanvasAction.ConnectOption; m_option_down = op; } private void MoveNode(Point pt) { int nX = (int)((pt.X - m_pt_down_in_control.X) / this._CanvasScale); int nY = (int)((pt.Y - m_pt_down_in_control.Y) / this._CanvasScale); lock (m_hs_node_selected) { foreach (STNode v in m_hs_node_selected) { v.Left = m_dic_pt_selected[v].X + nX; v.Top = m_dic_pt_selected[v].Y + nY; } if (this._ShowMagnet) { MagnetInfo mi = this.CheckMagnet(this._ActiveNode); if (mi.XMatched) { foreach (STNode v in m_hs_node_selected) v.Left -= mi.OffsetX; } if (mi.YMatched) { foreach (STNode v in m_hs_node_selected) v.Top -= mi.OffsetY; } } } this.Invalidate(); } protected internal virtual void BuildBounds() { if (this._Nodes.Count == 0) { this._CanvasValidBounds = this.ControlToCanvas(this.DisplayRectangle); return; } int x = int.MaxValue; int y = int.MaxValue; int r = int.MinValue; int b = int.MinValue; foreach (STNode n in this._Nodes) { if (x > n.Left) x = n.Left; if (y > n.Top) y = n.Top; if (r < n.Right) r = n.Right; if (b < n.Bottom) b = n.Bottom; } this._CanvasValidBounds.X = x - 60; this._CanvasValidBounds.Y = y - 60; this._CanvasValidBounds.Width = r - x + 120; this._CanvasValidBounds.Height = b - y + 120; } private bool PointInRectangle(Rectangle rect, float x, float y) { if (x < rect.Left) return false; if (x > rect.Right) return false; if (y < rect.Top) return false; if (y > rect.Bottom) return false; return true; } private void BuildMagnetLocation() { m_lst_magnet_x.Clear(); m_lst_magnet_y.Clear(); foreach (STNode v in this._Nodes) { if (v.IsSelected) continue; m_lst_magnet_x.Add(v.Left); m_lst_magnet_x.Add(v.Left + v.Width / 2); m_lst_magnet_x.Add(v.Left + v.Width); m_lst_magnet_y.Add(v.Top); m_lst_magnet_y.Add(v.Top + v.Height / 2); m_lst_magnet_y.Add(v.Top + v.Height); } } private MagnetInfo CheckMagnet(STNode node) { m_mi.XMatched = m_mi.YMatched = false; m_lst_magnet_mx.Clear(); m_lst_magnet_my.Clear(); m_lst_magnet_mx.Add(node.Left + node.Width / 2); m_lst_magnet_mx.Add(node.Left); m_lst_magnet_mx.Add(node.Left + node.Width); m_lst_magnet_my.Add(node.Top + node.Height / 2); m_lst_magnet_my.Add(node.Top); m_lst_magnet_my.Add(node.Top + node.Height); bool bFlag = false; foreach (var mx in m_lst_magnet_mx) { foreach (var x in m_lst_magnet_x) { if (Math.Abs(mx - x) <= 5) { bFlag = true; m_mi.X = x; m_mi.OffsetX = mx - x; m_mi.XMatched = true; break; } } if (bFlag) break; } bFlag = false; foreach (var my in m_lst_magnet_my) { foreach (var y in m_lst_magnet_y) { if (Math.Abs(my - y) <= 5) { bFlag = true; m_mi.Y = y; m_mi.OffsetY = my - y; m_mi.YMatched = true; break; } } if (bFlag) break; } return m_mi; } private void DrawBezier(Graphics g, Pen p, PointF ptStart, PointF ptEnd, float f) { this.DrawBezier(g, p, ptStart.X, ptStart.Y, ptEnd.X, ptEnd.Y, f); } private void DrawBezier(Graphics g, Pen p, float x1, float y1, float x2, float y2, float f) { float n = (Math.Abs(x1 - x2) * f); if (this._Curvature != 0 && n < 30) n = 30; g.DrawBezier(p, x1, y1, x1 + n, y1, x2 - n, y2, x2, y2); } private GraphicsPath CreateBezierPath(float x1, float y1, float x2, float y2, float f) { GraphicsPath gp = new GraphicsPath(); float n = (Math.Abs(x1 - x2) * f); if (this._Curvature != 0 && n < 30) n = 30; gp.AddBezier( x1, y1, x1 + n, y1, x2 - n, y2, x2, y2 ); return gp; } private void RenderBorder(Graphics g, Rectangle rect, Image img) { //填充四个角 g.DrawImage(img, new Rectangle(rect.X - 5, rect.Y - 5, 5, 5), new Rectangle(0, 0, 5, 5), GraphicsUnit.Pixel); g.DrawImage(img, new Rectangle(rect.Right, rect.Y - 5, 5, 5), new Rectangle(img.Width - 5, 0, 5, 5), GraphicsUnit.Pixel); g.DrawImage(img, new Rectangle(rect.X - 5, rect.Bottom, 5, 5), new Rectangle(0, img.Height - 5, 5, 5), GraphicsUnit.Pixel); g.DrawImage(img, new Rectangle(rect.Right, rect.Bottom, 5, 5), new Rectangle(img.Width - 5, img.Height - 5, 5, 5), GraphicsUnit.Pixel); //四边 g.DrawImage(img, new Rectangle(rect.X - 5, rect.Y, 5, rect.Height), new Rectangle(0, 5, 5, img.Height - 10), GraphicsUnit.Pixel); g.DrawImage(img, new Rectangle(rect.X, rect.Y - 5, rect.Width, 5), new Rectangle(5, 0, img.Width - 10, 5), GraphicsUnit.Pixel); g.DrawImage(img, new Rectangle(rect.Right, rect.Y, 5, rect.Height), new Rectangle(img.Width - 5, 5, 5, img.Height - 10), GraphicsUnit.Pixel); g.DrawImage(img, new Rectangle(rect.X, rect.Bottom, rect.Width, 5), new Rectangle(5, img.Height - 5, img.Width - 10, 5), GraphicsUnit.Pixel); } #endregion private #region public -------------------------------------------------------------------------------------------------------- /// /// 通过画布坐标进行寻找 /// /// 画布中的坐标 /// 寻找到的数据 public NodeFindInfo FindNodeFromPoint(PointF pt) { m_find.Node = null; m_find.NodeOption = null; m_find.Mark = null; for (int i = this._Nodes.Count - 1; i >= 0; i--) { if (!string.IsNullOrEmpty(this._Nodes[i].Mark) && this.PointInRectangle(this._Nodes[i].MarkRectangle, pt.X, pt.Y)) { m_find.Mark = this._Nodes[i].Mark; m_find.MarkLines = this._Nodes[i].MarkLines; return m_find; } foreach (STNodeOption v in this._Nodes[i].InputOptions) { if (v == STNodeOption.Empty) continue; if (this.PointInRectangle(v.DotRectangle, pt.X, pt.Y)) m_find.NodeOption = v; } foreach (STNodeOption v in this._Nodes[i].OutputOptions) { if (v == STNodeOption.Empty) continue; if (this.PointInRectangle(v.DotRectangle, pt.X, pt.Y)) m_find.NodeOption = v; } if (this.PointInRectangle(this._Nodes[i].Rectangle, pt.X, pt.Y)) { m_find.Node = this._Nodes[i]; } if (m_find.NodeOption != null || m_find.Node != null) return m_find; } return m_find; } /// /// 获取已经被选择的 Node 集合 /// /// Node 集合 public STNode[] GetSelectedNode() { return m_hs_node_selected.ToArray(); } /// /// 将画布坐标转换为控件坐标 /// /// 参数 /// 是否为 X 坐标 /// 转换后的坐标 public float CanvasToControl(float number, bool isX) { return (number * this._CanvasScale) + (isX ? this._CanvasOffsetX : this._CanvasOffsetY); } /// /// 将画布坐标转换为控件坐标 /// /// 坐标 /// 转换后的坐标 public PointF CanvasToControl(PointF pt) { pt.X = (pt.X * this._CanvasScale) + this._CanvasOffsetX; pt.Y = (pt.Y * this._CanvasScale) + this._CanvasOffsetY; //pt.X += this._CanvasOffsetX; //pt.Y += this._CanvasOffsetY; return pt; } /// /// 将画布坐标转换为控件坐标 /// /// 坐标 /// 转换后的坐标 public Point CanvasToControl(Point pt) { pt.X = (int)(pt.X * this._CanvasScale + this._CanvasOffsetX); pt.Y = (int)(pt.Y * this._CanvasScale + this._CanvasOffsetY); //pt.X += (int)this._CanvasOffsetX; //pt.Y += (int)this._CanvasOffsetY; return pt; } /// /// 将画布坐标转换为控件坐标 /// /// 矩形区域 /// 转换后的矩形区域 public Rectangle CanvasToControl(Rectangle rect) { rect.X = (int)((rect.X * this._CanvasScale) + this._CanvasOffsetX); rect.Y = (int)((rect.Y * this._CanvasScale) + this._CanvasOffsetY); rect.Width = (int)(rect.Width * this._CanvasScale); rect.Height = (int)(rect.Height * this._CanvasScale); //rect.X += (int)this._CanvasOffsetX; //rect.Y += (int)this._CanvasOffsetY; return rect; } /// /// 将画布坐标转换为控件坐标 /// /// 矩形区域 /// 转换后的矩形区域 public RectangleF CanvasToControl(RectangleF rect) { rect.X = (rect.X * this._CanvasScale) + this._CanvasOffsetX; rect.Y = (rect.Y * this._CanvasScale) + this._CanvasOffsetY; rect.Width = (rect.Width * this._CanvasScale); rect.Height = (rect.Height * this._CanvasScale); //rect.X += this._CanvasOffsetX; //rect.Y += this._CanvasOffsetY; return rect; } /// /// 将控件坐标转换为画布坐标 /// /// 参数 /// 是否为 X 坐标 /// 转换后的坐标 public float ControlToCanvas(float number, bool isX) { return (number - (isX ? this._CanvasOffsetX : this._CanvasOffsetY)) / this._CanvasScale; } /// /// 将控件坐标转换为画布坐标 /// /// 坐标 /// 转换后的坐标 public Point ControlToCanvas(Point pt) { pt.X = (int)((pt.X - this._CanvasOffsetX) / this._CanvasScale); pt.Y = (int)((pt.Y - this._CanvasOffsetY) / this._CanvasScale); return pt; } /// /// 将控件坐标转换为画布坐标 /// /// 坐标 /// 转换后的坐标 public PointF ControlToCanvas(PointF pt) { pt.X = ((pt.X - this._CanvasOffsetX) / this._CanvasScale); pt.Y = ((pt.Y - this._CanvasOffsetY) / this._CanvasScale); return pt; } /// /// 将控件坐标转换为画布坐标 /// /// 矩形区域 /// 转换后的区域 public Rectangle ControlToCanvas(Rectangle rect) { rect.X = (int)((rect.X - this._CanvasOffsetX) / this._CanvasScale); rect.Y = (int)((rect.Y - this._CanvasOffsetY) / this._CanvasScale); rect.Width = (int)(rect.Width / this._CanvasScale); rect.Height = (int)(rect.Height / this._CanvasScale); return rect; } /// /// 将控件坐标转换为画布坐标 /// /// 矩形区域 /// 转换后的区域 public RectangleF ControlToCanvas(RectangleF rect) { rect.X = ((rect.X - this._CanvasOffsetX) / this._CanvasScale); rect.Y = ((rect.Y - this._CanvasOffsetY) / this._CanvasScale); rect.Width = (rect.Width / this._CanvasScale); rect.Height = (rect.Height / this._CanvasScale); return rect; } /// /// 移动画布原点坐标到指定的控件坐标位置 /// 当不存在 Node 时候 无法移动 /// /// X 坐标 /// Y 坐标 /// 移动过程中是否启动动画效果 /// 指定需要修改的坐标参数 public void MoveCanvas(float x, float y, bool bAnimation, CanvasMoveArgs ma) { if (this._Nodes.Count == 0) { m_real_canvas_x = m_real_canvas_y = 10; return; } int l = (int)((this._CanvasValidBounds.Left + 50) * this._CanvasScale); int t = (int)((this._CanvasValidBounds.Top + 50) * this._CanvasScale); int r = (int)((this._CanvasValidBounds.Right - 50) * this._CanvasScale); int b = (int)((this._CanvasValidBounds.Bottom - 50) * this._CanvasScale); if (r + x < 0) x = -r; if (this.Width - l < x) x = this.Width - l; if (b + y < 0) y = -b; if (this.Height - t < y) y = this.Height - t; if (bAnimation) { if ((ma & CanvasMoveArgs.Left) == CanvasMoveArgs.Left) m_real_canvas_x = x; if ((ma & CanvasMoveArgs.Top) == CanvasMoveArgs.Top) m_real_canvas_y = y; } else { m_real_canvas_x = this._CanvasOffsetX = x; m_real_canvas_y = this._CanvasOffsetY = y; } this.OnCanvasMoved(EventArgs.Empty); } /// /// 缩放画布 /// 当不存在 Node 时候 无法缩放 /// /// 缩放比例 /// 缩放中心X位于控件上的坐标 /// 缩放中心Y位于控件上的坐标 public void ScaleCanvas(float f, float x, float y) { if (this._Nodes.Count == 0) { this._CanvasScale = 1F; return; } if (this._CanvasScale == f) return; if (f < 0.5) f = 0.5f; else if (f > 3) f = 3; float x_c = this.ControlToCanvas(x, true); float y_c = this.ControlToCanvas(y, false); this._CanvasScale = f; this._CanvasOffsetX = m_real_canvas_x -= this.CanvasToControl(x_c, true) - x; this._CanvasOffsetY = m_real_canvas_y -= this.CanvasToControl(y_c, false) - y; this.OnCanvasScaled(EventArgs.Empty); this.Invalidate(); } /// /// 获取当前已连接的 Option 对应关系 /// /// 连接信息集合 public ConnectionInfo[] GetConnectionInfo() { return m_dic_gp_info.Values.ToArray(); } /// /// 判断两个 Node 之间是否存在连接路径 /// /// 起始 Node /// 目标 Node /// 若存在路径返回true 否则false public static bool CanFindNodePath(STNode nodeStart, STNode nodeFind) { HashSet hs = new HashSet(); return STNodeEditor.CanFindNodePath(nodeStart, nodeFind, hs); } private static bool CanFindNodePath(STNode nodeStart, STNode nodeFind, HashSet hs) { foreach (STNodeOption op_1 in nodeStart.OutputOptions) { foreach (STNodeOption op_2 in op_1.ConnectedOption) { if (op_2.Owner == nodeFind) return true; if (hs.Add(op_2.Owner)) { if (STNodeEditor.CanFindNodePath(op_2.Owner, nodeFind)) return true; } } } return false; } /// /// 获取画布中指定矩形区域图像 /// /// 画布中指定的矩形区域 /// 图像 public Image GetCanvasImage(Rectangle rect) { return this.GetCanvasImage(rect, 1f); } /// /// 获取画布中指定矩形区域图像 /// /// 画布中指定的矩形区域 /// 缩放比例 /// 图像 public Image GetCanvasImage(Rectangle rect, float fScale) { if (fScale < 0.5) fScale = 0.5f; else if (fScale > 3) fScale = 3; Image img = new Bitmap((int)(rect.Width * fScale), (int)(rect.Height * fScale)); using (Graphics g = Graphics.FromImage(img)) { g.Clear(this.BackColor); g.ScaleTransform(fScale, fScale); m_drawing_tools.Graphics = g; if (this._ShowGrid) this.OnDrawGrid(m_drawing_tools, rect.Width, rect.Height); g.TranslateTransform(-rect.X, -rect.Y); //移动坐标系 this.OnDrawNode(m_drawing_tools, rect); this.OnDrawConnectedLine(m_drawing_tools); g.ResetTransform(); if (this._ShowLocation) this.OnDrawNodeOutLocation(m_drawing_tools, img.Size, m_lst_node_out); } return img; } /// /// 保存画布中的类容到文件中 /// /// 文件路径 public void SaveCanvas(string strFileName) { using (FileStream fs = new FileStream(strFileName, FileMode.Create, FileAccess.Write)) { this.SaveCanvas(fs); } } /// /// 保存画布中的类容到数据流 /// /// 数据流对象 public void SaveCanvas(Stream s) { Dictionary dic = new Dictionary(); s.Write(new byte[] { (byte)'S', (byte)'T', (byte)'N', (byte)'D' }, 0, 4); //file head s.WriteByte(1); //ver using (GZipStream gs = new GZipStream(s, CompressionMode.Compress)) { gs.Write(BitConverter.GetBytes(this._CanvasOffsetX), 0, 4); gs.Write(BitConverter.GetBytes(this._CanvasOffsetY), 0, 4); gs.Write(BitConverter.GetBytes(this._CanvasScale), 0, 4); gs.Write(BitConverter.GetBytes(this._Nodes.Count), 0, 4); foreach (STNode node in this._Nodes) { try { byte[] byNode = node.GetSaveData(); gs.Write(BitConverter.GetBytes(byNode.Length), 0, 4); gs.Write(byNode, 0, byNode.Length); foreach (STNodeOption op in node.InputOptions) if (!dic.ContainsKey(op)) dic.Add(op, dic.Count); foreach (STNodeOption op in node.OutputOptions) if (!dic.ContainsKey(op)) dic.Add(op, dic.Count); } catch (Exception ex) { throw new Exception("获取节点数据出错-" + node.Title, ex); } } gs.Write(BitConverter.GetBytes(m_dic_gp_info.Count), 0, 4); foreach (var v in m_dic_gp_info.Values) gs.Write(BitConverter.GetBytes(((dic[v.Output] << 32) | dic[v.Input])), 0, 8); } } /// /// 获取画布中内容二进制数据 /// /// 二进制数据 public byte[] GetCanvasData() { using (MemoryStream ms = new MemoryStream()) { this.SaveCanvas(ms); return ms.ToArray(); } } /// /// 加载程序集 /// /// 程序集集合 /// 存在STNode类型的文件的个数 public int LoadAssembly(string[] strFiles) { int nCount = 0; foreach (var v in strFiles) { try { if (this.LoadAssembly(v)) nCount++; } catch { } } return nCount; } /// /// 加载程序集 /// /// 指定需要加载的文件 /// 是否加载成功 public bool LoadAssembly(string strFile) { bool bFound = false; Assembly asm = Assembly.LoadFrom(strFile); if (asm == null) return false; foreach (var t in asm.GetTypes()) { if (t.IsAbstract) continue; if (t == m_type_node || t.IsSubclassOf(m_type_node)) { if (m_dic_type.ContainsKey(t.GUID.ToString())) continue; m_dic_type.Add(t.GUID.ToString(), t); bFound = true; } } return bFound; } /// /// 获取当前编辑器中已加载的Node类型 /// /// 类型集合 public Type[] GetTypes() { return m_dic_type.Values.ToArray(); } /// /// 从文件中加载数据 /// 注意: 此方法并不会清空画布中数据 而是数据叠加 /// /// 文件路径 public void LoadCanvas(string strFileName) { using (MemoryStream ms = new MemoryStream(File.ReadAllBytes(strFileName))) this.LoadCanvas(ms); } /// /// 从二进制加载数据 /// 注意: 此方法并不会清空画布中数据 而是数据叠加 /// /// 二进制数据 public void LoadCanvas(byte[] byData) { using (MemoryStream ms = new MemoryStream(byData)) this.LoadCanvas(ms); } /// /// 从数据流中加载数据 /// 注意: 此方法并不会清空画布中数据 而是数据叠加 /// /// 数据流对象 public void LoadCanvas(Stream s) { int nLen = 0; byte[] byLen = new byte[4]; s.Read(byLen, 0, 4); if (BitConverter.ToInt32(byLen, 0) != BitConverter.ToInt32(new byte[] { (byte)'S', (byte)'T', (byte)'N', (byte)'D' }, 0)) throw new InvalidDataException("无法识别的文件类型"); if (s.ReadByte() != 1) throw new InvalidDataException("无法识别的文件版本号"); using (GZipStream gs = new GZipStream(s, CompressionMode.Decompress)) { gs.Read(byLen, 0, 4); float x = BitConverter.ToSingle(byLen, 0); gs.Read(byLen, 0, 4); float y = BitConverter.ToSingle(byLen, 0); gs.Read(byLen, 0, 4); float scale = BitConverter.ToSingle(byLen, 0); gs.Read(byLen, 0, 4); int nCount = BitConverter.ToInt32(byLen, 0); Dictionary dic = new Dictionary(); HashSet hs = new HashSet(); byte[] byData = null; for (int i = 0; i < nCount; i++) { gs.Read(byLen, 0, byLen.Length); nLen = BitConverter.ToInt32(byLen, 0); byData = new byte[nLen]; gs.Read(byData, 0, byData.Length); STNode node = null; try { node = this.GetNodeFromData(byData); } catch (Exception ex) { throw new Exception("加载节点时发生错误可能数据已损坏\r\n" + ex.Message, ex); } try { this._Nodes.Add(node); } catch (Exception ex) { throw new Exception("加载节点出错-" + node.Title, ex); } foreach (STNodeOption op in node.InputOptions) if (hs.Add(op)) dic.Add(dic.Count, op); foreach (STNodeOption op in node.OutputOptions) if (hs.Add(op)) dic.Add(dic.Count, op); } gs.Read(byLen, 0, 4); nCount = BitConverter.ToInt32(byLen, 0); byData = new byte[8]; for (int i = 0; i < nCount; i++) { gs.Read(byData, 0, byData.Length); long id = BitConverter.ToInt64(byData, 0); long op_out = id >> 32; long op_in = (int)id; dic[op_out].ConnectOption(dic[op_in]); } this.ScaleCanvas(scale, 0, 0); this.MoveCanvas(x, y, false, CanvasMoveArgs.All); } this.BuildBounds(); foreach (STNode node in this._Nodes) node.OnEditorLoadCompleted(); } private STNode GetNodeFromData(byte[] byData) { int nIndex = 0; string strModel = Encoding.UTF8.GetString(byData, nIndex + 1, byData[nIndex]); nIndex += byData[nIndex] + 1; string strGUID = Encoding.UTF8.GetString(byData, nIndex + 1, byData[nIndex]); nIndex += byData[nIndex] + 1; int nLen = 0; Dictionary dic = new Dictionary(); while (nIndex < byData.Length) { nLen = BitConverter.ToInt32(byData, nIndex); nIndex += 4; string strKey = Encoding.UTF8.GetString(byData, nIndex, nLen); nIndex += nLen; nLen = BitConverter.ToInt32(byData, nIndex); nIndex += 4; byte[] byValue = new byte[nLen]; Array.Copy(byData, nIndex, byValue, 0, nLen); nIndex += nLen; dic.Add(strKey, byValue); } if (!m_dic_type.ContainsKey(strGUID)) throw new TypeLoadException("无法找到类型 {" + strModel.Split('|')[1] + "} 所在程序集 确保程序集 {" + strModel.Split('|')[0] + "} 已被编辑器正确加载 可通过调用LoadAssembly()加载程序集"); Type t = m_dic_type[strGUID]; ; STNode node = (STNode)Activator.CreateInstance(t); node.OnLoadNode(dic); return node; } /// /// 在画布中显示提示信息 /// /// 要显示的信息 /// 信息前景色 /// 信息背景色 public void ShowAlert(string strText, Color foreColor, Color backColor) { this.ShowAlert(strText, foreColor, backColor, 1000, AlertLocation.RightBottom, true); } /// /// 在画布中显示提示信息 /// /// 要显示的信息 /// 信息前景色 /// 信息背景色 /// 信息要显示的位置 public void ShowAlert(string strText, Color foreColor, Color backColor, AlertLocation al) { this.ShowAlert(strText, foreColor, backColor, 1000, al, true); } /// /// 在画布中显示提示信息 /// /// 要显示的信息 /// 信息前景色 /// 信息背景色 /// 信息持续时间 /// 信息要显示的位置 /// 是否立即重绘 public void ShowAlert(string strText, Color foreColor, Color backColor, int nTime, AlertLocation al, bool bRedraw) { m_str_alert = strText; m_forecolor_alert = foreColor; m_backcolor_alert = backColor; m_time_alert = nTime; m_dt_alert = DateTime.Now; m_alpha_alert = 255; m_al = al; if (bRedraw) this.Invalidate(); } /// /// 设置画布中活动的节点 /// /// 需要被设置为活动的节点 /// 设置前的活动节点 public STNode SetActiveNode(STNode node) { if (node != null && !this._Nodes.Contains(node)) return this._ActiveNode; STNode ret = this._ActiveNode; if (this._ActiveNode != node) { //重置活动选择节点 if (node != null) { this._Nodes.MoveToEnd(node); node.IsActive = true; node.SetSelected(true, false); node.OnGotFocus(EventArgs.Empty); } if (this._ActiveNode != null) { this._ActiveNode.IsActive /*= this._ActiveNode.IsSelected*/ = false; this._ActiveNode.OnLostFocus(EventArgs.Empty); } this._ActiveNode = node; this.Invalidate(); this.OnActiveChanged(EventArgs.Empty); //this.OnSelectedChanged(EventArgs.Empty); } return ret; } /// /// 向画布中添加一个被选中的节点 /// /// 需要被选中的节点 /// 是否添加成功 public bool AddSelectedNode(STNode node) { if (!this._Nodes.Contains(node)) return false; bool b = !node.IsSelected; node.IsSelected = true; lock (m_hs_node_selected) return m_hs_node_selected.Add(node) || b; } /// /// 向画布中移除一个被选中的节点 /// /// 需要被移除的节点 /// 是移除否成功 public bool RemoveSelectedNode(STNode node) { if (!this._Nodes.Contains(node)) return false; bool b = node.IsSelected; node.IsSelected = false; lock (m_hs_node_selected) return m_hs_node_selected.Remove(node) || b; } /// /// 向编辑器中添加默认数据类型颜色 /// /// 数据类型 /// 对应颜色 /// 被设置后的颜色 public Color SetTypeColor(Type t, Color clr) { return this.SetTypeColor(t, clr, false); } /// /// 向编辑器中添加默认数据类型颜色 /// /// 数据类型 /// 对应颜色 /// 若已经存在是否替换颜色 /// 被设置后的颜色 public Color SetTypeColor(Type t, Color clr, bool bReplace) { if (this._TypeColor.ContainsKey(t)) { if (bReplace) this._TypeColor[t] = clr; } else { this._TypeColor.Add(t, clr); } return this._TypeColor[t]; } #endregion public } }