mirror of
https://mirror.ghproxy.com/https://github.com/dexyfex/CodeWalker
synced 2025-01-10 20:15:09 +08:00
1117 lines
47 KiB
C#
1117 lines
47 KiB
C#
using System;
|
||
using System.Collections.Generic;
|
||
using System.Linq;
|
||
using System.Text;
|
||
|
||
using System.Drawing;
|
||
using System.Windows.Forms;
|
||
using System.Collections;
|
||
/*
|
||
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: 2021-12-08
|
||
* modify: 2021-03-02
|
||
* Author: Crystal_lz
|
||
* blog: http://st233.com
|
||
* Gitee: https://gitee.com/DebugST
|
||
* Github: https://github.com/DebugST
|
||
*/
|
||
namespace ST.Library.UI.NodeEditor
|
||
{
|
||
public abstract class STNode
|
||
{
|
||
private STNodeEditor _Owner;
|
||
/// <summary>
|
||
/// Get the current Node owner
|
||
/// </summary>
|
||
public STNodeEditor Owner {
|
||
get { return _Owner; }
|
||
internal set {
|
||
if (value == _Owner) return;
|
||
if (_Owner != null) {
|
||
foreach (STNodeOption op in this._InputOptions.ToArray()) op.DisConnectionAll();
|
||
foreach (STNodeOption op in this._OutputOptions.ToArray()) op.DisConnectionAll();
|
||
}
|
||
_Owner = value;
|
||
if (!this._AutoSize) this.SetOptionsLocation();
|
||
this.BuildSize(true, true, false);
|
||
this.OnOwnerChanged();
|
||
}
|
||
}
|
||
|
||
private bool _IsSelected;
|
||
/// <summary>
|
||
/// Gets or sets whether Node is selected
|
||
/// </summary>
|
||
public bool IsSelected {
|
||
get { return _IsSelected; }
|
||
set {
|
||
if (value == _IsSelected) return;
|
||
_IsSelected = value;
|
||
this.Invalidate();
|
||
this.OnSelectedChanged();
|
||
if (this._Owner != null) this._Owner.OnSelectedChanged(EventArgs.Empty);
|
||
}
|
||
}
|
||
|
||
private bool _IsActive;
|
||
/// <summary>
|
||
/// Gets if Node is active
|
||
/// </summary>
|
||
public bool IsActive {
|
||
get { return _IsActive; }
|
||
internal set {
|
||
if (value == _IsActive) return;
|
||
_IsActive = value;
|
||
this.OnActiveChanged();
|
||
}
|
||
}
|
||
|
||
private Color _TitleColor;
|
||
/// <summary>
|
||
/// Gets or sets the title background color
|
||
/// </summary>
|
||
public Color TitleColor {
|
||
get { return _TitleColor; }
|
||
protected set {
|
||
_TitleColor = value;
|
||
this.Invalidate(new Rectangle(0, 0, this._Width, this._TitleHeight));
|
||
}
|
||
}
|
||
|
||
private Color _MarkColor;
|
||
/// <summary>
|
||
/// Gets or sets the background color of the marker information
|
||
/// </summary>
|
||
public Color MarkColor {
|
||
get { return _MarkColor; }
|
||
protected set {
|
||
_MarkColor = value;
|
||
this.Invalidate(this._MarkRectangle);
|
||
}
|
||
}
|
||
|
||
private Color _ForeColor = Color.White;
|
||
/// <summary>
|
||
/// Get or set the current Node foreground color
|
||
/// </summary>
|
||
public Color ForeColor {
|
||
get { return _ForeColor; }
|
||
protected set {
|
||
_ForeColor = value;
|
||
this.Invalidate();
|
||
}
|
||
}
|
||
|
||
private Color _BackColor;
|
||
/// <summary>
|
||
/// Gets or sets the current Node background color
|
||
/// </summary>
|
||
public Color BackColor {
|
||
get { return _BackColor; }
|
||
protected set {
|
||
_BackColor = value;
|
||
this.Invalidate();
|
||
}
|
||
}
|
||
|
||
private string _Title;
|
||
/// <summary>
|
||
/// Gets or sets the Node title
|
||
/// </summary>
|
||
public string Title {
|
||
get { return _Title; }
|
||
protected set {
|
||
_Title = value;
|
||
if (this._AutoSize) this.BuildSize(true, true, true);
|
||
//this.Invalidate(this.TitleRectangle);
|
||
}
|
||
}
|
||
|
||
private string _Mark;
|
||
/// <summary>
|
||
/// Get or set Node tag information
|
||
/// </summary>
|
||
public string Mark {
|
||
get { return _Mark; }
|
||
set {
|
||
_Mark = value;
|
||
if (value == null)
|
||
_MarkLines = null;
|
||
else
|
||
_MarkLines = (from s in value.Split('\n') select s.Trim()).ToArray();
|
||
this.Invalidate(new Rectangle(-5, -5, this._MarkRectangle.Width + 10, this._MarkRectangle.Height + 10));
|
||
}
|
||
}
|
||
|
||
private string[] _MarkLines;//Store line data separately without splitting it every time you draw
|
||
/// <summary>
|
||
/// Get Node tag information row data
|
||
/// </summary>
|
||
public string[] MarkLines {
|
||
get { return _MarkLines; }
|
||
}
|
||
|
||
private int _Left;
|
||
/// <summary>
|
||
/// Get or set the left coordinate of Node
|
||
/// </summary>
|
||
public int Left {
|
||
get { return _Left; }
|
||
set {
|
||
if (this._LockLocation || value == _Left) return;
|
||
_Left = value;
|
||
this.SetOptionsLocation();
|
||
this.BuildSize(false, true, false);
|
||
this.OnMove(EventArgs.Empty);
|
||
if (this._Owner != null) {
|
||
this._Owner.BuildLinePath();
|
||
this._Owner.BuildBounds();
|
||
}
|
||
}
|
||
}
|
||
|
||
private int _Top;
|
||
/// <summary>
|
||
/// Gets or sets the coordinates on the top of the Node
|
||
/// </summary>
|
||
public int Top {
|
||
get { return _Top; }
|
||
set {
|
||
if (this._LockLocation || value == _Top) return;
|
||
_Top = value;
|
||
this.SetOptionsLocation();
|
||
this.BuildSize(false, true, false);
|
||
this.OnMove(EventArgs.Empty);
|
||
if (this._Owner != null) {
|
||
this._Owner.BuildLinePath();
|
||
this._Owner.BuildBounds();
|
||
}
|
||
}
|
||
}
|
||
|
||
private int _Width = 100;
|
||
/// <summary>
|
||
/// Gets or sets the Node width. This value cannot be set when AutoSize is set
|
||
/// </summary>
|
||
public int Width {
|
||
get { return _Width; }
|
||
protected set {
|
||
if (value < 50) return;
|
||
if (this._AutoSize || value == _Width) return;
|
||
_Width = value;
|
||
this.SetOptionsLocation();
|
||
this.BuildSize(false, true, false);
|
||
this.OnResize(EventArgs.Empty);
|
||
if (this._Owner != null) {
|
||
this._Owner.BuildLinePath();
|
||
this._Owner.BuildBounds();
|
||
}
|
||
this.Invalidate();
|
||
}
|
||
}
|
||
|
||
private int _Height = 40;
|
||
/// <summary>
|
||
/// Gets or sets the height of Node. This value cannot be set when AutoSize is set
|
||
/// </summary>
|
||
public int Height {
|
||
get { return _Height; }
|
||
protected set {
|
||
if (value < 40) return;
|
||
if (this._AutoSize || value == _Height) return;
|
||
_Height = value;
|
||
this.SetOptionsLocation();
|
||
this.BuildSize(false, true, false);
|
||
this.OnResize(EventArgs.Empty);
|
||
if (this._Owner != null) {
|
||
this._Owner.BuildLinePath();
|
||
this._Owner.BuildBounds();
|
||
}
|
||
this.Invalidate();
|
||
}
|
||
}
|
||
|
||
private int _ItemHeight = 20;
|
||
/// <summary>
|
||
/// Gets or sets the height of each option of Node
|
||
/// </summary>
|
||
public int ItemHeight {
|
||
get { return _ItemHeight; }
|
||
protected set {
|
||
if (value < 16) value = 16;
|
||
if (value > 200) value = 200;
|
||
if (value == _ItemHeight) return;
|
||
_ItemHeight = value;
|
||
if (this._AutoSize) {
|
||
this.BuildSize(true, false, true);
|
||
} else {
|
||
this.SetOptionsLocation();
|
||
if (this._Owner != null) this._Owner.Invalidate();
|
||
}
|
||
}
|
||
}
|
||
|
||
private bool _AutoSize = true;
|
||
/// <summary>
|
||
/// Gets or sets whether Node automatically calculates width and height
|
||
/// </summary>
|
||
public bool AutoSize {
|
||
get { return _AutoSize; }
|
||
protected set { _AutoSize = value; }
|
||
}
|
||
/// <summary>
|
||
/// Get the coordinates of the right edge of Node
|
||
/// </summary>
|
||
public int Right {
|
||
get { return _Left + _Width; }
|
||
}
|
||
/// <summary>
|
||
/// Get the coordinates below the Node
|
||
/// </summary>
|
||
public int Bottom {
|
||
get { return _Top + _Height; }
|
||
}
|
||
/// <summary>
|
||
/// Get the Node rectangle area
|
||
/// </summary>
|
||
public Rectangle Rectangle {
|
||
get {
|
||
return new Rectangle(this._Left, this._Top, this._Width, this._Height);
|
||
}
|
||
}
|
||
/// <summary>
|
||
/// Get the Node title rectangle area
|
||
/// </summary>
|
||
public Rectangle TitleRectangle {
|
||
get {
|
||
return new Rectangle(this._Left, this._Top, this._Width, this._TitleHeight);
|
||
}
|
||
}
|
||
|
||
private Rectangle _MarkRectangle;
|
||
/// <summary>
|
||
/// Get the Node marked rectangular area
|
||
/// </summary>
|
||
public Rectangle MarkRectangle {
|
||
get { return _MarkRectangle; }
|
||
}
|
||
|
||
private int _TitleHeight = 20;
|
||
/// <summary>
|
||
/// Gets or sets the height of the Node title
|
||
/// </summary>
|
||
public int TitleHeight {
|
||
get { return _TitleHeight; }
|
||
protected set { _TitleHeight = value; }
|
||
}
|
||
|
||
private STNodeOptionCollection _InputOptions;
|
||
/// <summary>
|
||
/// Get the set of input options
|
||
/// </summary>
|
||
protected internal STNodeOptionCollection InputOptions {
|
||
get { return _InputOptions; }
|
||
}
|
||
/// <summary>
|
||
/// Get the number of input options set
|
||
/// </summary>
|
||
public int InputOptionsCount { get { return _InputOptions.Count; } }
|
||
|
||
private STNodeOptionCollection _OutputOptions;
|
||
/// <summary>
|
||
/// Get output options
|
||
/// </summary>
|
||
protected internal STNodeOptionCollection OutputOptions {
|
||
get { return _OutputOptions; }
|
||
}
|
||
/// <summary>
|
||
/// Get the number of output options
|
||
/// </summary>
|
||
public int OutputOptionsCount { get { return _OutputOptions.Count; } }
|
||
|
||
private STNodeControlCollection _Controls;
|
||
/// <summary>
|
||
/// Get the set of controls contained in Node
|
||
/// </summary>
|
||
protected STNodeControlCollection Controls {
|
||
get { return _Controls; }
|
||
}
|
||
/// <summary>
|
||
/// Get the number of control sets contained in Node
|
||
/// </summary>
|
||
public int ControlsCount { get { return _Controls.Count; } }
|
||
/// <summary>
|
||
/// Get the Node coordinate position
|
||
/// </summary>
|
||
public Point Location {
|
||
get { return new Point(this._Left, this._Top); }
|
||
set {
|
||
this.Left = value.X;
|
||
this.Top = value.Y;
|
||
}
|
||
}
|
||
/// <summary>
|
||
/// Get the Node size
|
||
/// </summary>
|
||
public Size Size {
|
||
get { return new Size(this._Width, this._Height); }
|
||
set {
|
||
this.Width = value.Width;
|
||
this.Height = value.Height;
|
||
}
|
||
}
|
||
|
||
private Font _Font;
|
||
/// <summary>
|
||
/// Get or set the Node font
|
||
/// </summary>
|
||
protected Font Font {
|
||
get { return _Font; }
|
||
set {
|
||
if (value == _Font) return;
|
||
this._Font.Dispose();
|
||
_Font = value;
|
||
}
|
||
}
|
||
|
||
private bool _LockOption;
|
||
/// <summary>
|
||
/// Get or set whether to lock the Option option and not accept connections after locking
|
||
/// </summary>
|
||
public bool LockOption {
|
||
get { return _LockOption; }
|
||
set {
|
||
_LockOption = value;
|
||
this.Invalidate(new Rectangle(0, 0, this._Width, this._TitleHeight));
|
||
}
|
||
}
|
||
|
||
private bool _LockLocation;
|
||
/// <summary>
|
||
/// Gets or sets whether to lock the Node position and cannot move after it is locked
|
||
/// </summary>
|
||
public bool LockLocation {
|
||
get { return _LockLocation; }
|
||
set {
|
||
_LockLocation = value;
|
||
this.Invalidate(new Rectangle(0, 0, this._Width, this._TitleHeight));
|
||
}
|
||
}
|
||
|
||
private ContextMenuStrip _ContextMenuStrip;
|
||
/// <summary>
|
||
/// Gets or sets the current Node context menu
|
||
/// </summary>
|
||
public ContextMenuStrip ContextMenuStrip {
|
||
get { return _ContextMenuStrip; }
|
||
set { _ContextMenuStrip = value; }
|
||
}
|
||
|
||
private object _Tag;
|
||
/// <summary>
|
||
/// Get or set user-defined saved data
|
||
/// </summary>
|
||
public object Tag {
|
||
get { return _Tag; }
|
||
set { _Tag = value; }
|
||
}
|
||
|
||
private Guid _Guid;
|
||
/// <summary>
|
||
/// Get the global unique ID
|
||
/// </summary>
|
||
public Guid Guid {
|
||
get { return _Guid; }
|
||
}
|
||
|
||
private bool _LetGetOptions = false;
|
||
/// <summary>
|
||
/// Gets or sets whether to allow external access to STNodeOption
|
||
/// </summary>
|
||
public bool LetGetOptions {
|
||
get { return _LetGetOptions; }
|
||
protected set { _LetGetOptions = value; }
|
||
}
|
||
|
||
private static Point m_static_pt_init = new Point(10, 10);
|
||
|
||
public STNode() {
|
||
this._Title = "Untitled";
|
||
this._MarkRectangle.Height = this._Height;
|
||
this._Left = this._MarkRectangle.X = m_static_pt_init.X;
|
||
this._Top = m_static_pt_init.Y;
|
||
this._MarkRectangle.Y = this._Top - 30;
|
||
this._InputOptions = new STNodeOptionCollection(this, true);
|
||
this._OutputOptions = new STNodeOptionCollection(this, false);
|
||
this._Controls = new STNodeControlCollection(this);
|
||
this._BackColor = Color.FromArgb(200, 64, 64, 64);
|
||
this._TitleColor = Color.FromArgb(200, Color.DodgerBlue);
|
||
this._MarkColor = Color.FromArgb(200, Color.Brown);
|
||
this._Font = new Font("courier new", 8.25f);
|
||
|
||
m_sf = new StringFormat();
|
||
m_sf.Alignment = StringAlignment.Near;
|
||
m_sf.LineAlignment = StringAlignment.Center;
|
||
m_sf.FormatFlags = StringFormatFlags.NoWrap;
|
||
m_sf.SetTabStops(0, new float[] { 40 });
|
||
m_static_pt_init.X += 10;
|
||
m_static_pt_init.Y += 10;
|
||
this._Guid = Guid.NewGuid();
|
||
this.OnCreate();
|
||
}
|
||
|
||
//private int m_nItemHeight = 30;
|
||
protected StringFormat m_sf;
|
||
/// <summary>
|
||
/// Active controls in the current Node
|
||
/// </summary>
|
||
protected STNodeControl m_ctrl_active;
|
||
/// <summary>
|
||
/// The hover control in the current Node
|
||
/// </summary>
|
||
protected STNodeControl m_ctrl_hover;
|
||
/// <summary>
|
||
/// The control under the mouse click in the current Node
|
||
/// </summary>
|
||
protected STNodeControl m_ctrl_down;
|
||
|
||
protected internal void BuildSize(bool bBuildNode, bool bBuildMark, bool bRedraw) {
|
||
if (this._Owner == null) return;
|
||
using (Graphics g = this._Owner.CreateGraphics()) {
|
||
if (this._AutoSize && bBuildNode) {
|
||
Size sz = this.GetDefaultNodeSize(g);
|
||
if (this._Width != sz.Width || this._Height != sz.Height) {
|
||
this._Width = sz.Width;
|
||
this._Height = sz.Height;
|
||
this.SetOptionsLocation();
|
||
this.OnResize(EventArgs.Empty);
|
||
}
|
||
}
|
||
if (bBuildMark && !string.IsNullOrEmpty(this._Mark)) {
|
||
this._MarkRectangle = this.OnBuildMarkRectangle(g);
|
||
}
|
||
}
|
||
if (bRedraw) this._Owner.Invalidate();
|
||
}
|
||
|
||
internal Dictionary<string, byte[]> OnSaveNode() {
|
||
Dictionary<string, byte[]> dic = new Dictionary<string, byte[]>();
|
||
dic.Add("Guid", this._Guid.ToByteArray());
|
||
dic.Add("Left", BitConverter.GetBytes(this._Left));
|
||
dic.Add("Top", BitConverter.GetBytes(this._Top));
|
||
dic.Add("Width", BitConverter.GetBytes(this._Width));
|
||
dic.Add("Height", BitConverter.GetBytes(this._Height));
|
||
dic.Add("AutoSize", new byte[] { (byte)(this._AutoSize ? 1 : 0) });
|
||
if (this._Mark != null) dic.Add("Mark", Encoding.UTF8.GetBytes(this._Mark));
|
||
dic.Add("LockOption", new byte[] { (byte)(this._LockLocation ? 1 : 0) });
|
||
dic.Add("LockLocation", new byte[] { (byte)(this._LockLocation ? 1 : 0) });
|
||
Type t = this.GetType();
|
||
foreach (var p in t.GetProperties()) {
|
||
var attrs = p.GetCustomAttributes(true);
|
||
foreach (var a in attrs) {
|
||
if (!(a is STNodePropertyAttribute)) continue;
|
||
var attr = a as STNodePropertyAttribute;
|
||
object obj = Activator.CreateInstance(attr.DescriptorType);
|
||
if (!(obj is STNodePropertyDescriptor))
|
||
throw new InvalidOperationException("[STNodePropertyAttribute.Type] parameter value must be the type of [STNodePropertyDescriptor] or its subclass");
|
||
var desc = (STNodePropertyDescriptor)Activator.CreateInstance(attr.DescriptorType);
|
||
desc.Node = this;
|
||
desc.PropertyInfo = p;
|
||
byte[] byData = desc.GetBytesFromValue();
|
||
if (byData == null) continue;
|
||
dic.Add(p.Name, byData);
|
||
}
|
||
}
|
||
this.OnSaveNode(dic);
|
||
return dic;
|
||
}
|
||
|
||
internal byte[] GetSaveData() {
|
||
List<byte> lst = new List<byte>();
|
||
Type t = this.GetType();
|
||
byte[] byData = Encoding.UTF8.GetBytes(t.Module.Name + "|" + t.FullName);
|
||
lst.Add((byte)byData.Length);
|
||
lst.AddRange(byData);
|
||
byData = Encoding.UTF8.GetBytes(t.GUID.ToString());
|
||
lst.Add((byte)byData.Length);
|
||
lst.AddRange(byData);
|
||
|
||
var dic = this.OnSaveNode();
|
||
if (dic != null) {
|
||
foreach (var v in dic) {
|
||
byData = Encoding.UTF8.GetBytes(v.Key);
|
||
lst.AddRange(BitConverter.GetBytes(byData.Length));
|
||
lst.AddRange(byData);
|
||
lst.AddRange(BitConverter.GetBytes(v.Value.Length));
|
||
lst.AddRange(v.Value);
|
||
}
|
||
}
|
||
return lst.ToArray();
|
||
}
|
||
|
||
#region protected
|
||
/// <summary>
|
||
/// Occurs when Node is constructed
|
||
/// </summary>
|
||
protected virtual void OnCreate() { }
|
||
/// <summary>
|
||
/// Draw the entire Node
|
||
/// </summary>
|
||
/// <param name="dt">Drawing tool</param>
|
||
protected internal virtual void OnDrawNode(DrawingTools dt) {
|
||
dt.Graphics.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.None;
|
||
//Fill background
|
||
if (this._BackColor.A != 0) {
|
||
dt.SolidBrush.Color = this._BackColor;
|
||
dt.Graphics.FillRectangle(dt.SolidBrush, this._Left, this._Top + this._TitleHeight, this._Width, this.Height - this._TitleHeight);
|
||
}
|
||
this.OnDrawTitle(dt);
|
||
this.OnDrawBody(dt);
|
||
}
|
||
/// <summary>
|
||
/// Draw the Node header section
|
||
/// </summary>
|
||
/// <param name="dt">Drawing tool</param>
|
||
protected virtual void OnDrawTitle(DrawingTools dt) {
|
||
m_sf.Alignment = StringAlignment.Center;
|
||
m_sf.LineAlignment = StringAlignment.Center;
|
||
Graphics g = dt.Graphics;
|
||
SolidBrush brush = dt.SolidBrush;
|
||
if (this._TitleColor.A != 0) {
|
||
brush.Color = this._TitleColor;
|
||
g.FillRectangle(brush, this.TitleRectangle);
|
||
}
|
||
if (this._LockOption) {
|
||
//dt.Pen.Color = this.ForeColor;
|
||
brush.Color = this._ForeColor;
|
||
int n = this._Top + this._TitleHeight / 2 - 5;
|
||
g.FillRectangle(dt.SolidBrush, this._Left + 4, n + 0, 2, 4);
|
||
g.FillRectangle(dt.SolidBrush, this._Left + 6, n + 0, 2, 2);
|
||
g.FillRectangle(dt.SolidBrush, this._Left + 8, n + 0, 2, 4);
|
||
g.FillRectangle(dt.SolidBrush, this._Left + 3, n + 4, 8, 6);
|
||
//g.DrawLine(dt.Pen, this._Left + 6, n + 5, this._Left + 6, n + 7);
|
||
//g.DrawRectangle(dt.Pen, this._Left + 3, n + 0, 6, 3);
|
||
//g.DrawRectangle(dt.Pen, this._Left + 2, n + 3, 8, 6);
|
||
//g.DrawLine(dt.Pen, this._Left + 6, n + 5, this._Left + 6, n + 7);
|
||
|
||
}
|
||
if (this._LockLocation) {
|
||
//dt.Pen.Color = this.ForeColor;
|
||
brush.Color = this._ForeColor;
|
||
int n = this._Top + this._TitleHeight / 2 - 5;
|
||
g.FillRectangle(brush, this.Right - 9, n, 4, 4);
|
||
g.FillRectangle(brush, this.Right - 11, n + 4, 8, 2);
|
||
g.FillRectangle(brush, this.Right - 8, n + 6, 2, 4);
|
||
//g.DrawLine(dt.Pen, this.Right - 10, n + 6, this.Right - 4, n + 6);
|
||
//g.DrawLine(dt.Pen, this.Right - 10, n, this.Right - 4, n);
|
||
//g.DrawLine(dt.Pen, this.Right - 11, n + 6, this.Right - 3, n + 6);
|
||
//g.DrawLine(dt.Pen, this.Right - 7, n + 7, this.Right - 7, n + 9);
|
||
}
|
||
if (!string.IsNullOrEmpty(this._Title) && this._ForeColor.A != 0) {
|
||
brush.Color = this._ForeColor;
|
||
g.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.HighQuality;
|
||
g.DrawString(this._Title, this._Font, brush, this.TitleRectangle, m_sf);
|
||
}
|
||
}
|
||
/// <summary>
|
||
/// Draw the Node body part to remove the title part
|
||
/// </summary>
|
||
/// <param name="dt">Drawing tool</param>
|
||
protected virtual void OnDrawBody(DrawingTools dt) {
|
||
SolidBrush brush = dt.SolidBrush;
|
||
foreach (STNodeOption op in this._InputOptions) {
|
||
if (op == STNodeOption.Empty) continue;
|
||
this.OnDrawOptionDot(dt, op);
|
||
this.OnDrawOptionText(dt, op);
|
||
}
|
||
foreach (STNodeOption op in this._OutputOptions) {
|
||
if (op == STNodeOption.Empty) continue;
|
||
this.OnDrawOptionDot(dt, op);
|
||
this.OnDrawOptionText(dt, op);
|
||
}
|
||
if (this._Controls.Count != 0) { //Draw child controls
|
||
// Align the origin of the coordinates with the node
|
||
//dt.Graphics.ResetTransform();
|
||
dt.Graphics.TranslateTransform(this._Left, this._Top + this._TitleHeight);
|
||
Point pt = Point.Empty; //The current amount of offset needed
|
||
Point pt_last = Point.Empty; //The coordinates of the last control relative to the node
|
||
foreach (STNodeControl v in this._Controls) {
|
||
if (!v.Visable) continue;
|
||
pt.X = v.Left - pt_last.X;
|
||
pt.Y = v.Top - pt_last.Y;
|
||
pt_last = v.Location;
|
||
dt.Graphics.TranslateTransform(pt.X, pt.Y); //Move the origin coordinates to the control position
|
||
dt.Graphics.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.None;
|
||
v.OnPaint(dt);
|
||
}
|
||
//dt.Graphics.TranslateTransform(-pt_last.X, -pt_last.Y); restore coordinates
|
||
dt.Graphics.TranslateTransform(-this._Left - pt_last.X, -this._Top - this._TitleHeight - pt_last.Y);
|
||
//dt.Graphics.
|
||
}
|
||
}
|
||
/// <summary>
|
||
/// Draw marker information
|
||
/// </summary>
|
||
/// <param name="dt">Drawing tool</param>
|
||
protected internal virtual void OnDrawMark(DrawingTools dt) {
|
||
if (string.IsNullOrEmpty(this._Mark)) return;
|
||
Graphics g = dt.Graphics;
|
||
SolidBrush brush = dt.SolidBrush;
|
||
m_sf.LineAlignment = StringAlignment.Center;
|
||
g.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.None;
|
||
brush.Color = this._MarkColor;
|
||
g.FillRectangle(brush, this._MarkRectangle); //fill background color
|
||
|
||
g.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.HighQuality; //Determine the size required for text drawing
|
||
var sz = g.MeasureString(this.Mark, this.Font, this._MarkRectangle.Width);
|
||
brush.Color = this._ForeColor;
|
||
if (sz.Height > this._ItemHeight || sz.Width > this._MarkRectangle.Width) { //If it exceeds the drawing area, draw the part
|
||
Rectangle rect = new Rectangle(this._MarkRectangle.Left + 2, this._MarkRectangle.Top + 2, this._MarkRectangle.Width - 20, 16);
|
||
m_sf.Alignment = StringAlignment.Near;
|
||
g.DrawString(this._MarkLines[0], this._Font, brush, rect, m_sf);
|
||
m_sf.Alignment = StringAlignment.Far;
|
||
rect.Width = this._MarkRectangle.Width - 5;
|
||
g.DrawString("+", this._Font, brush, rect, m_sf); // + means beyond the drawing area
|
||
} else {
|
||
m_sf.Alignment = StringAlignment.Near;
|
||
g.DrawString(this._MarkLines[0].Trim(), this._Font, brush, this._MarkRectangle, m_sf);
|
||
}
|
||
}
|
||
/// <summary>
|
||
/// The point where the option line is drawn
|
||
/// </summary>
|
||
/// <param name="dt">Drawing tool</param>
|
||
/// <param name="op">Specified options</param>
|
||
protected virtual void OnDrawOptionDot(DrawingTools dt, STNodeOption op) {
|
||
Graphics g = dt.Graphics;
|
||
Pen pen = dt.Pen;
|
||
SolidBrush brush = dt.SolidBrush;
|
||
var t = typeof(object);
|
||
if (op.DotColor != Color.Transparent) //Set the color
|
||
brush.Color = op.DotColor;
|
||
else {
|
||
if (op.DataType == t)
|
||
pen.Color = this.Owner.UnknownTypeColor;
|
||
else
|
||
brush.Color = this.Owner.TypeColor.ContainsKey(op.DataType) ? this.Owner.TypeColor[op.DataType] : this.Owner.UnknownTypeColor;
|
||
}
|
||
if (op.IsSingle) { //Single connection circle
|
||
g.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.HighQuality;
|
||
if (op.DataType == t) { //unknown type to draw otherwise fill
|
||
g.DrawEllipse(pen, op.DotRectangle.X, op.DotRectangle.Y, op.DotRectangle.Width - 1, op.DotRectangle.Height - 1);
|
||
} else
|
||
g.FillEllipse(brush, op.DotRectangle);
|
||
} else { //Multiple connection rectangles
|
||
g.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.None;
|
||
if (op.DataType == t) {
|
||
g.DrawRectangle(pen, op.DotRectangle.X, op.DotRectangle.Y, op.DotRectangle.Width - 1, op.DotRectangle.Height - 1);
|
||
} else
|
||
g.FillRectangle(brush, op.DotRectangle);
|
||
}
|
||
}
|
||
/// <summary>
|
||
/// Text for drawing options
|
||
/// </summary>
|
||
/// <param name="dt">Drawing tool</param>
|
||
/// <param name="op">Specified options</param>
|
||
protected virtual void OnDrawOptionText(DrawingTools dt, STNodeOption op) {
|
||
Graphics g = dt.Graphics;
|
||
SolidBrush brush = dt.SolidBrush;
|
||
if (op.IsInput) {
|
||
m_sf.Alignment = StringAlignment.Near;
|
||
} else {
|
||
m_sf.Alignment = StringAlignment.Far;
|
||
}
|
||
brush.Color = op.TextColor;
|
||
g.DrawString(op.Text, this.Font, brush, op.TextRectangle, m_sf);
|
||
}
|
||
/// <summary>
|
||
/// Occurs when calculating the position of the Option connection point
|
||
/// </summary>
|
||
/// <param name="op">Option to be calculated</param>
|
||
/// <param name="pt">Automatically calculated position</param>
|
||
/// <param name="nIndex">Index of the current Option</param>
|
||
/// <returns>New location</returns>
|
||
protected virtual Point OnSetOptionDotLocation(STNodeOption op, Point pt, int nIndex) {
|
||
return pt;
|
||
}
|
||
/// <summary>
|
||
/// Occurs when evaluating the Option text area
|
||
/// </summary>
|
||
/// <param name="op">Option to be calculated</param>
|
||
/// <param name="rect">Automatically calculated area</param>
|
||
/// <param name="nIndex">Index of the current Option</param>
|
||
/// <returns>New area</returns>
|
||
protected virtual Rectangle OnSetOptionTextRectangle(STNodeOption op, Rectangle rect, int nIndex) {
|
||
return rect;
|
||
}
|
||
/// <summary>
|
||
/// Get the default size required by the current STNode
|
||
/// The returned size does not limit the drawing area and can still be drawn outside of this area
|
||
/// But it will not be accepted by STNodeEditor and trigger the corresponding event
|
||
/// </summary>
|
||
/// <param name="g">Drawing panel</param>
|
||
/// <returns>Calculated size</returns>
|
||
protected virtual Size GetDefaultNodeSize(Graphics g) {
|
||
int nInputHeight = 0, nOutputHeight = 0;
|
||
foreach (STNodeOption op in this._InputOptions) nInputHeight += this._ItemHeight;
|
||
foreach (STNodeOption op in this._OutputOptions) nOutputHeight += this._ItemHeight;
|
||
int nHeight = this._TitleHeight + (nInputHeight > nOutputHeight ? nInputHeight : nOutputHeight);
|
||
|
||
SizeF szf_input = SizeF.Empty, szf_output = SizeF.Empty;
|
||
foreach (STNodeOption v in this._InputOptions) {
|
||
if (string.IsNullOrEmpty(v.Text)) continue;
|
||
SizeF szf = g.MeasureString(v.Text, this._Font);
|
||
if (szf.Width > szf_input.Width) szf_input = szf;
|
||
}
|
||
foreach (STNodeOption v in this._OutputOptions) {
|
||
if (string.IsNullOrEmpty(v.Text)) continue;
|
||
SizeF szf = g.MeasureString(v.Text, this._Font);
|
||
if (szf.Width > szf_output.Width) szf_output = szf;
|
||
}
|
||
int nWidth = (int)(szf_input.Width + szf_output.Width + 25);
|
||
if (!string.IsNullOrEmpty(this.Title)) szf_input = g.MeasureString(this.Title, this.Font);
|
||
if (szf_input.Width + 30 > nWidth) nWidth = (int)szf_input.Width + 30;
|
||
return new Size(nWidth, nHeight);
|
||
}
|
||
/// <summary>
|
||
/// Calculate the rectangular area required by the current Mark
|
||
/// The returned size does not limit the drawing area and can still be drawn outside of this area
|
||
/// But it will not be accepted by STNodeEditor and trigger the corresponding event
|
||
/// </summary>
|
||
/// <param name="g">Drawing panel</param>
|
||
/// <returns>Calculated area</returns>
|
||
protected virtual Rectangle OnBuildMarkRectangle(Graphics g) {
|
||
//if (string.IsNullOrEmpty(this._Mark)) return Rectangle.Empty;
|
||
return new Rectangle(this._Left, this._Top - 30, this._Width, 20);
|
||
}
|
||
/// <summary>
|
||
/// What data does this Node need to save additionally when it needs to be saved?
|
||
/// Note: When saving, it will not be serialized and restored. When only re-creating this Node through the empty parameter constructor
|
||
/// Then call OnLoadNode() to restore the saved data
|
||
/// </summary>
|
||
/// <param name="dic">Data to be saved</param>
|
||
protected virtual void OnSaveNode(Dictionary<string, byte[]> dic) { }
|
||
/// <summary>
|
||
/// When restoring the node, the data returned by OnSaveNode() will be re-introduced to this function
|
||
/// </summary>
|
||
/// <param name="dic">Save data</param>
|
||
protected internal virtual void OnLoadNode(Dictionary<string, byte[]> dic) {
|
||
if (dic.ContainsKey("AutoSize")) this._AutoSize = dic["AutoSize"][0] == 1;
|
||
if (dic.ContainsKey("LockOption")) this._LockOption = dic["LockOption"][0] == 1;
|
||
if (dic.ContainsKey("LockLocation")) this._LockLocation = dic["LockLocation"][0] == 1;
|
||
if (dic.ContainsKey("Guid")) this._Guid = new Guid(dic["Guid"]);
|
||
if (dic.ContainsKey("Left")) this._Left = BitConverter.ToInt32(dic["Left"], 0);
|
||
if (dic.ContainsKey("Top")) this._Top = BitConverter.ToInt32(dic["Top"], 0);
|
||
if (dic.ContainsKey("Width") && !this._AutoSize) this._Width = BitConverter.ToInt32(dic["Width"], 0);
|
||
if (dic.ContainsKey("Height") && !this._AutoSize) this._Height = BitConverter.ToInt32(dic["Height"], 0);
|
||
if (dic.ContainsKey("Mark")) this.Mark = Encoding.UTF8.GetString(dic["Mark"]);
|
||
Type t = this.GetType();
|
||
foreach (var p in t.GetProperties()) {
|
||
var attrs = p.GetCustomAttributes(true);
|
||
foreach (var a in attrs) {
|
||
if (!(a is STNodePropertyAttribute)) continue;
|
||
var attr = a as STNodePropertyAttribute;
|
||
object obj = Activator.CreateInstance(attr.DescriptorType);
|
||
if (!(obj is STNodePropertyDescriptor))
|
||
throw new InvalidOperationException("[STNodePropertyAttribute.Type] parameter value must be the type of [STNodePropertyDescriptor] or its subclass");
|
||
var desc = (STNodePropertyDescriptor)Activator.CreateInstance(attr.DescriptorType);
|
||
desc.Node = this;
|
||
desc.PropertyInfo = p;
|
||
try {
|
||
if (dic.ContainsKey(p.Name)) desc.SetValue(dic[p.Name]);
|
||
} catch (Exception ex) {
|
||
string strErr = "The value of attribute [" + this.Title + "." + p.Name + "] cannot be restored by overriding [STNodePropertyAttribute.GetBytesFromValue(), STNodePropertyAttribute.GetValueFromBytes(byte[])] to ensure preservation and The binary data is correct when loading";
|
||
Exception e = ex;
|
||
while (e != null) {
|
||
strErr += "\r\n----\r\n[" + e.GetType().Name + "] -> " + e.Message;
|
||
e = e.InnerException;
|
||
}
|
||
throw new InvalidOperationException(strErr, ex);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
/// <summary>
|
||
/// Occurs when the editor has loaded all nodes
|
||
/// </summary>
|
||
protected internal virtual void OnEditorLoadCompleted() { }
|
||
/// <summary>
|
||
/// Set the text information of Option
|
||
/// </summary>
|
||
/// <param name="op">Target Option</param>
|
||
/// <param name="strText">text</param>
|
||
/// <returns>Successful</returns>
|
||
protected bool SetOptionText(STNodeOption op, string strText) {
|
||
if (op.Owner != this) return false;
|
||
op.Text = strText;
|
||
return true;
|
||
}
|
||
/// <summary>
|
||
/// Set the color of the Option text information
|
||
/// </summary>
|
||
/// <param name="op">Target Option</param>
|
||
/// <param name="clr">color</param>
|
||
/// <returns>Successful</returns>
|
||
protected bool SetOptionTextColor(STNodeOption op, Color clr) {
|
||
if (op.Owner != this) return false;
|
||
op.TextColor = clr;
|
||
return true;
|
||
}
|
||
/// <summary>
|
||
/// Set the color of the Option connection point
|
||
/// </summary>
|
||
/// <param name="op">Target Option</param>
|
||
/// <param name="clr">color</param>
|
||
/// <returns>Successful</returns>
|
||
protected bool SetOptionDotColor(STNodeOption op, Color clr) {
|
||
if (op.Owner != this) return false;
|
||
op.DotColor = clr;
|
||
return false;
|
||
}
|
||
|
||
//[event]===========================[event]==============================[event]============================[event]
|
||
|
||
protected internal virtual void OnGotFocus(EventArgs e) { }
|
||
|
||
protected internal virtual void OnLostFocus(EventArgs e) { }
|
||
|
||
protected internal virtual void OnMouseEnter(EventArgs e) { }
|
||
|
||
protected internal virtual void OnMouseDown(MouseEventArgs e) {
|
||
Point pt = e.Location;
|
||
pt.Y -= this._TitleHeight;
|
||
for (int i = this._Controls.Count - 1; i >= 0; i--) {
|
||
var c = this._Controls[i];
|
||
if (c.DisplayRectangle.Contains(pt)) {
|
||
if (!c.Enabled) return;
|
||
if (!c.Visable) continue;
|
||
c.OnMouseDown(new MouseEventArgs(e.Button, e.Clicks, e.X - c.Left, pt.Y - c.Top, e.Delta));
|
||
m_ctrl_down = c;
|
||
if (m_ctrl_active != c) {
|
||
c.OnGotFocus(EventArgs.Empty);
|
||
if (m_ctrl_active != null) m_ctrl_active.OnLostFocus(EventArgs.Empty);
|
||
m_ctrl_active = c;
|
||
}
|
||
return;
|
||
}
|
||
}
|
||
if (m_ctrl_active != null) m_ctrl_active.OnLostFocus(EventArgs.Empty);
|
||
m_ctrl_active = null;
|
||
}
|
||
|
||
protected internal virtual void OnMouseMove(MouseEventArgs e) {
|
||
Point pt = e.Location;
|
||
pt.Y -= this._TitleHeight;
|
||
if (m_ctrl_down != null) {
|
||
if (m_ctrl_down.Enabled && m_ctrl_down.Visable)
|
||
m_ctrl_down.OnMouseMove(new MouseEventArgs(e.Button, e.Clicks, e.X - m_ctrl_down.Left, pt.Y - m_ctrl_down.Top, e.Delta));
|
||
return;
|
||
}
|
||
for (int i = this._Controls.Count - 1; i >= 0; i--) {
|
||
var c = this._Controls[i];
|
||
if (c.DisplayRectangle.Contains(pt)) {
|
||
if (m_ctrl_hover != this._Controls[i]) {
|
||
c.OnMouseEnter(EventArgs.Empty);
|
||
if (m_ctrl_hover != null) m_ctrl_hover.OnMouseLeave(EventArgs.Empty);
|
||
m_ctrl_hover = c;
|
||
}
|
||
m_ctrl_hover.OnMouseMove(new MouseEventArgs(e.Button, e.Clicks, e.X - c.Left, pt.Y - c.Top, e.Delta));
|
||
return;
|
||
}
|
||
}
|
||
if (m_ctrl_hover != null) m_ctrl_hover.OnMouseLeave(EventArgs.Empty);
|
||
m_ctrl_hover = null;
|
||
}
|
||
|
||
protected internal virtual void OnMouseUp(MouseEventArgs e) {
|
||
Point pt = e.Location;
|
||
pt.Y -= this._TitleHeight;
|
||
if (m_ctrl_down != null && m_ctrl_down.Enabled && m_ctrl_down.Visable) {
|
||
m_ctrl_down.OnMouseUp(new MouseEventArgs(e.Button, e.Clicks, e.X - m_ctrl_down.Left, pt.Y - m_ctrl_down.Top, e.Delta));
|
||
}
|
||
//if (m_ctrl_active != null) {
|
||
// m_ctrl_active.OnMouseUp(new MouseEventArgs(e.Button, e.Clicks,
|
||
// e.X - m_ctrl_active.Left, pt.Y - m_ctrl_active.Top, e.Delta));
|
||
//}
|
||
m_ctrl_down = null;
|
||
}
|
||
|
||
protected internal virtual void OnMouseLeave(EventArgs e) {
|
||
if (m_ctrl_hover != null && m_ctrl_hover.Enabled && m_ctrl_hover.Visable) m_ctrl_hover.OnMouseLeave(e);
|
||
m_ctrl_hover = null;
|
||
}
|
||
|
||
protected internal virtual void OnMouseClick(MouseEventArgs e) {
|
||
Point pt = e.Location;
|
||
pt.Y -= this._TitleHeight;
|
||
if (m_ctrl_active != null && m_ctrl_active.Enabled && m_ctrl_active.Visable)
|
||
m_ctrl_active.OnMouseClick(new MouseEventArgs(e.Button, e.Clicks, e.X - m_ctrl_active.Left, pt.Y - m_ctrl_active.Top, e.Delta));
|
||
}
|
||
|
||
protected internal virtual void OnMouseWheel(MouseEventArgs e) {
|
||
Point pt = e.Location;
|
||
pt.Y -= this._TitleHeight;
|
||
if (m_ctrl_hover != null && m_ctrl_active.Enabled && m_ctrl_hover.Visable) {
|
||
m_ctrl_hover.OnMouseWheel(new MouseEventArgs(e.Button, e.Clicks, e.X - m_ctrl_hover.Left, pt.Y - m_ctrl_hover.Top, e.Delta));
|
||
return;
|
||
}
|
||
}
|
||
protected internal virtual void OnMouseHWheel(MouseEventArgs e) {
|
||
if (m_ctrl_hover != null && m_ctrl_active.Enabled && m_ctrl_hover.Visable) {
|
||
m_ctrl_hover.OnMouseHWheel(e);
|
||
return;
|
||
}
|
||
}
|
||
|
||
protected internal virtual void OnKeyDown(KeyEventArgs e) {
|
||
if (m_ctrl_active != null && m_ctrl_active.Enabled && m_ctrl_active.Visable) m_ctrl_active.OnKeyDown(e);
|
||
}
|
||
protected internal virtual void OnKeyUp(KeyEventArgs e) {
|
||
if (m_ctrl_active != null && m_ctrl_active.Enabled && m_ctrl_active.Visable) m_ctrl_active.OnKeyUp(e);
|
||
}
|
||
protected internal virtual void OnKeyPress(KeyPressEventArgs e) {
|
||
if (m_ctrl_active != null && m_ctrl_active.Enabled && m_ctrl_active.Visable) m_ctrl_active.OnKeyPress(e);
|
||
}
|
||
|
||
protected virtual void OnMove(EventArgs e) { /*this.SetOptionLocation();*/ }
|
||
protected virtual void OnResize(EventArgs e) { /*this.SetOptionLocation();*/ }
|
||
|
||
|
||
/// <summary>
|
||
/// Occurs when the owner changes
|
||
/// </summary>
|
||
protected virtual void OnOwnerChanged() { }
|
||
/// <summary>
|
||
/// Occurs when the selected state changes
|
||
/// </summary>
|
||
protected virtual void OnSelectedChanged() { }
|
||
/// <summary>
|
||
/// Occurs when the activity state changes
|
||
/// </summary>
|
||
protected virtual void OnActiveChanged() { }
|
||
|
||
#endregion protected
|
||
/// <summary>
|
||
/// Calculate the position of each Option
|
||
/// </summary>
|
||
protected virtual void SetOptionsLocation() {
|
||
int nIndex = 0;
|
||
Rectangle rect = new Rectangle(this.Left + 10, this._Top + this._TitleHeight, this._Width - 20, this._ItemHeight);
|
||
foreach (STNodeOption op in this._InputOptions) {
|
||
if (op != STNodeOption.Empty) {
|
||
Point pt = this.OnSetOptionDotLocation(op, new Point(this.Left - op.DotSize / 2, rect.Y + (rect.Height - op.DotSize) / 2), nIndex);
|
||
op.TextRectangle = this.OnSetOptionTextRectangle(op, rect, nIndex);
|
||
op.DotLeft = pt.X;
|
||
op.DotTop = pt.Y;
|
||
}
|
||
rect.Y += this._ItemHeight;
|
||
nIndex++;
|
||
}
|
||
rect.Y = this._Top + this._TitleHeight;
|
||
m_sf.Alignment = StringAlignment.Far;
|
||
foreach (STNodeOption op in this._OutputOptions) {
|
||
if (op != STNodeOption.Empty) {
|
||
Point pt = this.OnSetOptionDotLocation(op, new Point(this._Left + this._Width - op.DotSize / 2, rect.Y + (rect.Height - op.DotSize) / 2), nIndex);
|
||
op.TextRectangle = this.OnSetOptionTextRectangle(op, rect, nIndex);
|
||
op.DotLeft = pt.X;
|
||
op.DotTop = pt.Y;
|
||
}
|
||
rect.Y += this._ItemHeight;
|
||
nIndex++;
|
||
}
|
||
}
|
||
|
||
/// <summary>
|
||
/// Redraw Node
|
||
/// </summary>
|
||
public void Invalidate() {
|
||
if (this._Owner != null) {
|
||
this._Owner.Invalidate(this._Owner.CanvasToControl(new Rectangle(this._Left - 5, this._Top - 5, this._Width + 10, this._Height + 10)));
|
||
}
|
||
}
|
||
/// <summary>
|
||
/// Redraw the specified area of Node
|
||
/// </summary>
|
||
/// <param name="rect">Node specified area</param>
|
||
public void Invalidate(Rectangle rect) {
|
||
rect.X += this._Left;
|
||
rect.Y += this._Top;
|
||
if (this._Owner != null) {
|
||
rect = this._Owner.CanvasToControl(rect);
|
||
rect.Width += 1; rect.Height += 1;//Coordinate system conversion may cause progress loss plus one more pixel
|
||
this._Owner.Invalidate(rect);
|
||
}
|
||
}
|
||
/// <summary>
|
||
/// Get the input Option collection contained in this Node
|
||
/// </summary>
|
||
/// <returns>Option array</returns>
|
||
public STNodeOption[] GetInputOptions() {
|
||
if (!this._LetGetOptions) return null;
|
||
STNodeOption[] ops = new STNodeOption[this._InputOptions.Count];
|
||
for (int i = 0; i < this._InputOptions.Count; i++) ops[i] = this._InputOptions[i];
|
||
return ops;
|
||
}
|
||
/// <summary>
|
||
/// Get the output Option set contained in this Node
|
||
/// </summary>
|
||
/// <returns>Option array</returns>
|
||
public STNodeOption[] GetOutputOptions() {
|
||
if (!this._LetGetOptions) return null;
|
||
STNodeOption[] ops = new STNodeOption[this._OutputOptions.Count];
|
||
for (int i = 0; i < this._OutputOptions.Count; i++) ops[i] = this._OutputOptions[i];
|
||
return ops;
|
||
}
|
||
/// <summary>
|
||
/// Set the selected state of Node
|
||
/// </summary>
|
||
/// <param name="bSelected">Selected</param>
|
||
/// <param name="bRedraw">Whether to redraw</param>
|
||
public void SetSelected(bool bSelected, bool bRedraw) {
|
||
if (this._IsSelected == bSelected) return;
|
||
this._IsSelected = bSelected;
|
||
if (this._Owner != null) {
|
||
if (bSelected)
|
||
this._Owner.AddSelectedNode(this);
|
||
else
|
||
this._Owner.RemoveSelectedNode(this);
|
||
}
|
||
if (bRedraw) this.Invalidate();
|
||
this.OnSelectedChanged();
|
||
if (this._Owner != null) this._Owner.OnSelectedChanged(EventArgs.Empty);
|
||
}
|
||
public IAsyncResult BeginInvoke(Delegate method) { return this.BeginInvoke(method, null); }
|
||
public IAsyncResult BeginInvoke(Delegate method, params object[] args) {
|
||
if (this._Owner == null) return null;
|
||
return this._Owner.BeginInvoke(method, args);
|
||
}
|
||
public object Invoke(Delegate method) { return this.Invoke(method, null); }
|
||
public object Invoke(Delegate method, params object[] args) {
|
||
if (this._Owner == null) return null;
|
||
return this._Owner.Invoke(method, args);
|
||
}
|
||
}
|
||
} |