mirror of
https://gitee.com/akwkevin/aistudio.-wpf.-diagram
synced 2026-03-03 00:00:57 +08:00
支持动画使用不同的形状
This commit is contained in:
@@ -1,4 +1,5 @@
|
||||
using AIStudio.Wpf.DiagramDesigner;
|
||||
using Newtonsoft.Json.Linq;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
@@ -39,6 +40,17 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
{
|
||||
return logicalConnectorInfo.ConnectorValue.ToString("f3");
|
||||
}
|
||||
else if (logicalConnectorInfo.ConnectorValueType == ConnectorValueType.JsonString)
|
||||
{
|
||||
if (logicalConnectorInfo.ConnectorString != null)
|
||||
{
|
||||
JObject obj = JObject.Parse(logicalConnectorInfo.ConnectorString);
|
||||
if (obj.ContainsKey("Text"))
|
||||
{
|
||||
return obj["Text"].ToString();
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
return logicalConnectorInfo.ConnectorString;
|
||||
|
||||
@@ -12,5 +12,6 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
Bool = 2,
|
||||
ValueType = 99,
|
||||
String = 100,
|
||||
JsonString = 101,
|
||||
}
|
||||
}
|
||||
|
||||
@@ -70,6 +70,16 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
return JsonConvert.SerializeObject(obj);
|
||||
}
|
||||
|
||||
public static T ToObject<T>(this string str)
|
||||
{
|
||||
return JsonConvert.DeserializeObject<T>(str);
|
||||
}
|
||||
|
||||
public static object ToObject(this string str, Type type)
|
||||
{
|
||||
return JsonConvert.DeserializeObject(str, type);
|
||||
}
|
||||
|
||||
public static JsonSerializerSettings Settings { get; set; } = new JsonSerializerSettings
|
||||
{
|
||||
DateFormatString = "yyyy-MM-dd HH:mm:ss.fff",
|
||||
|
||||
@@ -470,6 +470,22 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
}
|
||||
}
|
||||
|
||||
[XmlAttribute]
|
||||
public bool Repeat
|
||||
{
|
||||
get; set;
|
||||
}
|
||||
|
||||
public bool Start
|
||||
{
|
||||
get; set;
|
||||
}
|
||||
|
||||
public int Completed
|
||||
{
|
||||
get;set;
|
||||
}
|
||||
|
||||
public event PropertyChangedEventHandler PropertyChanged;
|
||||
}
|
||||
|
||||
|
||||
@@ -109,7 +109,7 @@
|
||||
Data="{Binding AnimationViewModel.AnimationPath.Path}"
|
||||
Fill="{Binding AnimationViewModel.Color,Converter={StaticResource ColorBrushConverter}}"
|
||||
Width="{Binding AnimationViewModel.AnimationPath.Width}"
|
||||
Height="{Binding ColorViewModel.AnimationPath.Height}"
|
||||
Height="{Binding AnimationViewModel.AnimationPath.Height}"
|
||||
Panel.ZIndex="1" />
|
||||
</Canvas>
|
||||
</Grid>
|
||||
|
||||
@@ -42,7 +42,7 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
if (this.DataContext is ConnectionViewModel connector && connector.AnimationViewModel != null)
|
||||
{
|
||||
connector.AnimationViewModel.PropertyChanged -= Connector_PropertyChanged;
|
||||
connector.AnimationViewModel.PropertyChanged += Connector_PropertyChanged;
|
||||
connector.AnimationViewModel.PropertyChanged += Connector_PropertyChanged;
|
||||
}
|
||||
this.ball.Visibility = Visibility.Collapsed;
|
||||
await DoAnimation();
|
||||
@@ -56,6 +56,11 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
case nameof(AnimationViewModel.Duration):
|
||||
await DoAnimation();
|
||||
break;
|
||||
case nameof(AnimationViewModel.Start):
|
||||
await Application.Current.Dispatcher.BeginInvoke(async () => {
|
||||
await DoAnimation();
|
||||
});
|
||||
break;
|
||||
|
||||
}
|
||||
}
|
||||
@@ -64,26 +69,36 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
{
|
||||
if (this.DataContext is ConnectionViewModel connector && connector.IsFullConnection)
|
||||
{
|
||||
await System.Threading.Tasks.Task.Delay(100);
|
||||
if (connector.AnimationViewModel.Repeat == false)
|
||||
{
|
||||
if (connector.AnimationViewModel.Start == false || connector.AnimationViewModel.Completed == 1)
|
||||
{
|
||||
return;
|
||||
}
|
||||
connector.AnimationViewModel.Completed = 1;
|
||||
connector.AnimationViewModel.Start = false;
|
||||
}
|
||||
|
||||
await System.Threading.Tasks.Task.Delay(10);
|
||||
switch (connector.AnimationViewModel.Animation)
|
||||
{
|
||||
case LineAnimation.None:
|
||||
_story?.Stop(this);
|
||||
ball.Visibility = Visibility.Collapsed;
|
||||
case LineAnimation.None:
|
||||
_story?.Stop(this);
|
||||
ball.Visibility = Visibility.Collapsed;
|
||||
break;
|
||||
case LineAnimation.PathAnimation:
|
||||
if (connector.ColorViewModel.FillColor.BrushType == BrushType.SolidColorBrush && connector.ColorViewModel.FillColor.Color == Colors.White)
|
||||
{
|
||||
connector.ColorViewModel.FillColor.Color = Colors.Red;
|
||||
}
|
||||
PathAnimation(connector.AnimationViewModel.Duration);
|
||||
PathAnimation(connector.AnimationViewModel);
|
||||
break;
|
||||
case LineAnimation.DashAnimation:
|
||||
if (connector.ColorViewModel.LineDashStyle == LineDashStyle.None)
|
||||
{
|
||||
connector.ColorViewModel.LineDashStyle = LineDashStyle.Dash1;
|
||||
}
|
||||
DashAnimation(connector.AnimationViewModel.Duration);
|
||||
DashAnimation(connector.AnimationViewModel);
|
||||
break;
|
||||
}
|
||||
}
|
||||
@@ -91,14 +106,14 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
|
||||
Storyboard _story;
|
||||
|
||||
private void PathAnimation(double second)
|
||||
private void PathAnimation(IAnimationViewModel animationViewModel)
|
||||
{
|
||||
this.ball.Visibility = Visibility.Visible;
|
||||
|
||||
Canvas.SetTop(this.ball, -this.ball.ActualHeight / 2);
|
||||
Canvas.SetLeft(this.ball, -this.ball.ActualWidth / 2);
|
||||
|
||||
this.ball.RenderTransformOrigin = new Point(0.5, 0.5);
|
||||
this.ball.RenderTransformOrigin = new Point(0.5, 0.5);
|
||||
|
||||
TranslateTransform translate = new TranslateTransform();
|
||||
RotateTransform rotate = new RotateTransform();
|
||||
@@ -114,7 +129,7 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
DoubleAnimationUsingPath animationX = new DoubleAnimationUsingPath();
|
||||
animationX.PathGeometry = this.line.Data?.GetFlattenedPathGeometry();
|
||||
animationX.Source = PathAnimationSource.X;
|
||||
animationX.Duration = new Duration(TimeSpan.FromSeconds(second));
|
||||
animationX.Duration = new Duration(TimeSpan.FromSeconds(animationViewModel.Duration));
|
||||
|
||||
DoubleAnimationUsingPath animationY = new DoubleAnimationUsingPath();
|
||||
animationY.PathGeometry = this.line.Data?.GetFlattenedPathGeometry();
|
||||
@@ -134,7 +149,15 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
_story.Children.Add(animationY);
|
||||
_story.Children.Add(animationAngle);
|
||||
_story.Completed += (s, d) => {
|
||||
_story.Begin(this, true);
|
||||
if (animationViewModel.Repeat)
|
||||
{
|
||||
_story.Begin(this, true);
|
||||
}
|
||||
else
|
||||
{
|
||||
this.ball.Visibility = Visibility.Collapsed;
|
||||
animationViewModel.Completed = 100;
|
||||
}
|
||||
};
|
||||
Storyboard.SetTargetName(animationX, "translate");
|
||||
Storyboard.SetTargetName(animationY, "translate");
|
||||
@@ -147,12 +170,12 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
|
||||
}
|
||||
|
||||
private void DashAnimation(double second)
|
||||
private void DashAnimation(IAnimationViewModel animationViewModel)
|
||||
{
|
||||
this.ball.Visibility = Visibility.Collapsed;
|
||||
|
||||
|
||||
var animation = new DoubleAnimation(0, -10, new Duration(TimeSpan.FromSeconds(second)))
|
||||
var animation = new DoubleAnimation(0, -10, new Duration(TimeSpan.FromSeconds(animationViewModel.Duration)))
|
||||
{
|
||||
RepeatBehavior = RepeatBehavior.Forever,
|
||||
};
|
||||
@@ -163,6 +186,17 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
_story = new Storyboard();
|
||||
_story.RepeatBehavior = RepeatBehavior.Forever;
|
||||
_story.Children.Add(animation);
|
||||
_story.Completed += (s, d) => {
|
||||
if (animationViewModel.Repeat)
|
||||
{
|
||||
_story.Begin(this, true);
|
||||
}
|
||||
else
|
||||
{
|
||||
this.ball.Visibility = Visibility.Collapsed;
|
||||
animationViewModel.Completed = 100;
|
||||
}
|
||||
};
|
||||
_story.Begin(this, true);
|
||||
}
|
||||
|
||||
|
||||
@@ -79,6 +79,37 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
}
|
||||
}
|
||||
|
||||
private bool _repeat = true;
|
||||
public bool Repeat
|
||||
{
|
||||
get
|
||||
{
|
||||
return _repeat;
|
||||
}
|
||||
set
|
||||
{
|
||||
SetProperty(ref _repeat, value);
|
||||
}
|
||||
}
|
||||
|
||||
private bool _start;
|
||||
public bool Start
|
||||
{
|
||||
get
|
||||
{
|
||||
return _start;
|
||||
}
|
||||
set
|
||||
{
|
||||
SetProperty(ref _start, value);
|
||||
}
|
||||
}
|
||||
|
||||
public int Completed
|
||||
{
|
||||
get; set;
|
||||
}
|
||||
|
||||
private void AnimationViewModel_PropertyChanged(object sender, System.ComponentModel.PropertyChangedEventArgs e)
|
||||
{
|
||||
if (sender == AnimationPath)
|
||||
|
||||
@@ -26,6 +26,21 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
{
|
||||
get; set;
|
||||
}
|
||||
|
||||
bool Repeat
|
||||
{
|
||||
get; set;
|
||||
}
|
||||
|
||||
bool Start
|
||||
{
|
||||
get; set;
|
||||
}
|
||||
|
||||
int Completed
|
||||
{
|
||||
get; set;
|
||||
}
|
||||
event PropertyChangedEventHandler PropertyChanged;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -9,6 +9,11 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
public static SharpPath Arrow { get; } = new SharpPath("M 0 -5 10 0 0 5 z", 10, 10, PathStyle.Arrow, SizeStyle.Middle);
|
||||
public static SharpPath Circle { get; } = new SharpPath("M 0, 0 a 5,5 0 1,0 10,0 a 5,5 0 1,0 -10,0", 10, 10, PathStyle.Circle, SizeStyle.Middle);
|
||||
public static SharpPath Square { get; } = new SharpPath("M 0 -5 10 -5 10 5 0 5 z", 10, 10, PathStyle.Square, SizeStyle.Middle);
|
||||
public static SharpPath Triangle { get; } = new SharpPath("M1,21H23L12,2", 10, 10, PathStyle.Square, SizeStyle.Middle);
|
||||
public static SharpPath Rhombus { get; } = new SharpPath("M 0,20 L 30 0 L 60,20 L 30,40 Z", 10, 10, PathStyle.Square, SizeStyle.Middle);
|
||||
public static SharpPath Trapezoid { get; } = new SharpPath("M 0 0 H 60 L 50 40 H 10 Z", 10, 10, PathStyle.Square, SizeStyle.Middle);
|
||||
public static SharpPath Pentagram { get; } = new SharpPath("M 9,2 11,7 17,7 12,10 14,15 9,12 4,15 6,10 1,7 7,7 Z", 10, 10, PathStyle.Square, SizeStyle.Middle);
|
||||
public static SharpPath Hexagon { get; } = new SharpPath("M 0,20 L 10,0 H 50 L 60,20 L 50,40 H10 Z", 10, 10, PathStyle.Square, SizeStyle.Middle);
|
||||
|
||||
public static readonly Dictionary<PathStyle, string> ArrowDictionary = new Dictionary<PathStyle, string>()
|
||||
{
|
||||
|
||||
@@ -144,6 +144,10 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
{
|
||||
return true;
|
||||
}
|
||||
else if (logical.ConnectorValueType == ConnectorValueType.JsonString && this.ConnectorValueType == logical.ConnectorValueType)
|
||||
{
|
||||
return true;
|
||||
}
|
||||
else if (logical.ConnectorValueType <= ConnectorValueType.ValueType && this.ConnectorValueType <= ConnectorValueType.ValueType)
|
||||
{
|
||||
return true;
|
||||
|
||||
@@ -1403,9 +1403,9 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
}
|
||||
|
||||
var logical = item as LogicalGateItemViewModelBase;
|
||||
if (logical != null && logical.LogicalType > 0)
|
||||
if (logical != null)
|
||||
{
|
||||
logical.OrderNumber = Items.OfType<LogicalGateItemViewModelBase>().Count(p => (int)p.LogicalType > 0) + 1;
|
||||
logical.OrderNumber = Items.OfType<LogicalGateItemViewModelBase>().Count() + 1;
|
||||
}
|
||||
|
||||
var designerItemViewModelBase = item as DesignerItemViewModelBase;
|
||||
|
||||
@@ -5,6 +5,7 @@ using System.ComponentModel.DataAnnotations;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
using System.Windows.Controls;
|
||||
using System.Windows.Input;
|
||||
using System.Windows.Media;
|
||||
using AIStudio.Wpf.DiagramDesigner.Enums;
|
||||
@@ -253,14 +254,16 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
continue;
|
||||
}
|
||||
|
||||
if (connector.SourceConnectorInfo.DataItem is LogicalGateItemViewModelBase)
|
||||
if (connector.SourceConnectorInfo.DataItem is LogicalGateItemViewModelBase sourceItem)
|
||||
{
|
||||
var output = (connector.SourceConnectorInfo as LogicalConnectorInfo);
|
||||
connector.ColorViewModel.LineColor.Color = output.ColorViewModel.FillColor.Color;
|
||||
connector.ColorViewModel.FillColor.Color = output.ColorViewModel.FillColor.Color;
|
||||
|
||||
if (input.Value.CanAttachTo(output))
|
||||
if (EnableInputValue(connector, input.Value, output))
|
||||
{
|
||||
connector.ColorViewModel.LineColor.Color = output.ColorViewModel.FillColor.Color;
|
||||
connector.ColorViewModel.FillColor.Color = output.ColorViewModel.FillColor.Color;
|
||||
connector.AnimationViewModel.Color = output.ColorViewModel.FillColor.Color;
|
||||
|
||||
input.Value.ConnectorValue = output.ConnectorValue;
|
||||
input.Value.ConnectorString = output.ConnectorString;
|
||||
input.Value.ColorViewModel.FillColor.Color = connector.SourceConnectorInfo.ColorViewModel.FillColor.Color;
|
||||
@@ -272,13 +275,10 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
{
|
||||
input.Value.ConnectorValueType = ((connector.SourceConnectorInfo as LogicalConnectorInfo).ConnectorValueType == ConnectorValueType.Bool) ? ConnectorValueType.Bool : ConnectorValueType.Int;
|
||||
}
|
||||
|
||||
sourceItem.ClearOutputValue(connector, output);
|
||||
}
|
||||
else
|
||||
{
|
||||
input.Value.ErrorCode = ConnectorErrorCode.ConnErr;
|
||||
input.Value.ErrorMessage = "连接类型不匹配";
|
||||
input.Value.ColorViewModel.FillColor.Color = Colors.DarkRed;
|
||||
}
|
||||
sourceItem.StartAnimation(output);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -310,16 +310,44 @@ namespace AIStudio.Wpf.DiagramDesigner
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
else if (output.Value.ConnectorValueType <= ConnectorValueType.ValueType)
|
||||
{
|
||||
output.Value.ColorViewModel.FillColor.Color = Colors.Green;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public virtual bool EnableInputValue(ConnectionViewModel connector, LogicalConnectorInfo input, LogicalConnectorInfo output)
|
||||
{
|
||||
if (!input.CanAttachTo(output))
|
||||
{
|
||||
input.ErrorCode = ConnectorErrorCode.ConnErr;
|
||||
input.ErrorMessage = "连接类型不匹配";
|
||||
input.ColorViewModel.FillColor.Color = Colors.DarkRed;
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
public virtual void ClearOutputValue(ConnectionViewModel connector, LogicalConnectorInfo output)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
public virtual void StartAnimation(LogicalConnectorInfo output)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
protected ConnectionViewModel GetSourceItem(FullyCreatedConnectorInfo sinkConnector)
|
||||
{
|
||||
return Root?.Items.OfType<ConnectionViewModel>().FirstOrDefault(p => p.SinkConnectorInfo == sinkConnector);
|
||||
}
|
||||
|
||||
protected ConnectionViewModel GetSinkItem(FullyCreatedConnectorInfo sourceConnector)
|
||||
{
|
||||
return Root?.Items.OfType<ConnectionViewModel>().FirstOrDefault(p => p.SourceConnectorInfo == sourceConnector);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -29,12 +29,10 @@ namespace AIStudio.Wpf.DiagramDesigner.Demo.ViewModels
|
||||
DiagramViewModel.Add(node3);
|
||||
|
||||
ConnectionViewModel connector1 = new ConnectionViewModel(DiagramViewModel, node1.RightConnector, node2.LeftConnector, DrawMode.ConnectingLineSmooth, RouterMode.RouterNormal);
|
||||
connector1.ColorViewModel.FillColor.Color = Colors.Red;
|
||||
connector1.AnimationViewModel.Animation = LineAnimation.PathAnimation;
|
||||
DiagramViewModel.Add(connector1);
|
||||
|
||||
ConnectionViewModel connector2 = new ConnectionViewModel(DiagramViewModel, node2.RightConnector, node3.RightConnector, DrawMode.ConnectingLineStraight, RouterMode.RouterOrthogonal);
|
||||
connector2.ColorViewModel.FillColor.Color = Colors.Red;
|
||||
connector2.AnimationViewModel.Animation = LineAnimation.PathAnimation;
|
||||
DiagramViewModel.Add(connector2);
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user