diff --git a/CPF.Toolkit.Demo/CPF.Toolkit.Demo.csproj b/CPF.Toolkit.Demo/CPF.Toolkit.Demo.csproj new file mode 100644 index 0000000..8602593 --- /dev/null +++ b/CPF.Toolkit.Demo/CPF.Toolkit.Demo.csproj @@ -0,0 +1,33 @@ + + + + WinExe + netcoreapp3.0 + + + + + + true + + AnyCPU + + + + + + + + + + + + + + + + + + + + diff --git a/CPF.Toolkit.Demo/MainView.cs b/CPF.Toolkit.Demo/MainView.cs new file mode 100644 index 0000000..0b942e6 --- /dev/null +++ b/CPF.Toolkit.Demo/MainView.cs @@ -0,0 +1,44 @@ +using CPF; +using CPF.Animation; +using CPF.Charts; +using CPF.Controls; +using CPF.Drawing; +using CPF.Shapes; +using CPF.Styling; +using CPF.Svg; +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; + +namespace CPF.Toolkit.Demo +{ + public class MainView : Window + { + protected override void InitializeComponent() + { + //LoadStyleFile("res://CPF.Toolkit.Demo/Stylesheet1.css"); + this.Behaviors.Add(new ViewBehavior()); + + this.Title = "标题"; + this.Width = 500; + this.Height = 400; + this.Background = null; + var vm = new MainViewModel(); + this.DataContext = this.CommandContext = vm; + this.Children.Add(new WindowFrame(this, new Panel + { + Width = "100%", + Height = "100%", + Children = + { + new Button + { + Content="按钮", + [nameof(Button.Click)] = new CommandDescribe((ss,ee) => vm.Test()), + } + }, + })); + } + } +} diff --git a/CPF.Toolkit.Demo/MainViewModel.cs b/CPF.Toolkit.Demo/MainViewModel.cs new file mode 100644 index 0000000..8bf3f2e --- /dev/null +++ b/CPF.Toolkit.Demo/MainViewModel.cs @@ -0,0 +1,27 @@ +using CPF.Controls; +using System; +using System.Collections.Generic; +using System.Text; + +namespace CPF.Toolkit.Demo +{ + internal class MainViewModel : ViewModelBase + { + protected override void OnLoaded() + { + + } + + + public void Test() + { + this.Close(); + } + + //protected override void OnClosing(ClosingEventArgs e) + //{ + // e.Cancel = true; + // base.OnClosing(e); + //} + } +} diff --git a/CPF.Toolkit.Demo/Program.cs b/CPF.Toolkit.Demo/Program.cs new file mode 100644 index 0000000..789546d --- /dev/null +++ b/CPF.Toolkit.Demo/Program.cs @@ -0,0 +1,21 @@ +using CPF.Platform; +using CPF.Skia; +using CPF.Windows; +using System; + +namespace CPF.Toolkit.Demo +{ + internal class Program + { + [STAThread] + static void Main(string[] args) + { + Application.Initialize( + (OperatingSystemType.Windows, new WindowsPlatform(), new SkiaDrawingFactory()) + , (OperatingSystemType.OSX, new CPF.Mac.MacPlatform(), new SkiaDrawingFactory())//如果需要支持Mac才需要 + , (OperatingSystemType.Linux, new CPF.Linux.LinuxPlatform(), new SkiaDrawingFactory())//如果需要支持Linux才需要 + ); + Application.Run(new MainView()); + } + } +} diff --git a/CPF.Toolkit.Demo/Stylesheet1.css b/CPF.Toolkit.Demo/Stylesheet1.css new file mode 100644 index 0000000..d895ab6 --- /dev/null +++ b/CPF.Toolkit.Demo/Stylesheet1.css @@ -0,0 +1,502 @@ +/*@font-face { + font-family: '微软雅黑'; + src: url('res://ConsoleApp1/msyh.ttc'); +}*/ /*加载字体*/ + +/** { + FontFamily: 微软雅黑; +}*/ + +@media windows { + * { + FontFamily: '微软雅黑'; /*不同系统的字体不同,自己根据情况改或者使用内嵌字体*/ + } +} + +@media osx { + * { + FontFamily: '苹方-简'; + } +} + +@media linux { + * { + FontFamily: '文泉驿正黑'; + } +} +/*设置窗体标题栏背景颜色圆角*/ +/*#caption { + IsAntiAlias: true; + Background: #2c2c2c; + CornerRadius:5,5,0,0; + Height:30; +} +#frame { + CornerRadius: 5; + IsAntiAlias: true; +}*/ +Button { + BorderFill: #DCDFE6; + IsAntiAlias: True; + CornerRadius: 4,4,4,4; + Background: #FFFFFF; +} + + Button[IsMouseOver=true] { + BorderFill: rgb(198,226,255); + Background: rgb(236,245,255); + Foreground: rgb(64,158,255); + } + + Button[IsPressed=true] { + BorderFill: rgb(58,142,230); + } + + Button.primary { + BorderFill: rgb(64,158,255); + CornerRadius: 4,4,4,4; + Background: rgb(64,158,255); + Foreground: #FFFFFF; + } + + Button.primary[IsMouseOver=true] { + BorderFill: rgb(102,177,255); + Background: rgb(102,177,255); + Foreground: #FFFFFF; + } + + Button.primary[IsPressed=true] { + BorderFill: rgb(58,142,230); + Background: rgb(58,142,230); + } + + Button.success { + BorderFill: rgb(103,194,58); + CornerRadius: 4,4,4,4; + Background: rgb(103,194,58); + Foreground: #FFFFFF; + } + + Button.success[IsMouseOver=true] { + BorderFill: rgb(133,206,97); + Background: rgb(133,206,97); + Foreground: #FFFFFF; + } + + Button.success[IsPressed=true] { + BorderFill: rgb(93,175,52); + Background: rgb(93,175,52); + } + + Button.danger { + BorderFill: rgb(245,108,108); + Background: rgb(245,108,108); + CornerRadius: 4,4,4,4; + Foreground: #FFFFFF; + } + + Button.danger[IsMouseOver=true] { + BorderFill: rgb(247,137,137); + Background: rgb(247,137,137); + Foreground: #FFFFFF; + } + + Button.danger[IsPressed=true] { + BorderFill: rgb(221,97,97); + Background: rgb(221,97,97); + } + +#dialogClose { + BorderFill: null; +} + +TextBox, .textBox, DatePicker { + Background: #fff; + IsAntiAlias: true; + BorderFill: #DCDFE6; + CornerRadius: 4,4,4,4; + BorderStroke: 1; +} + + .groupPanel TextBox, DatePicker TextBox, .textBox TextBox { + BorderStroke: 0; + } + + .textBox[IsKeyboardFocusWithin=true] { + BorderFill: #1E9FFF; + } + +.singleLine { /*单行文本框*/ + AcceptsReturn: false; + HScrollBarVisibility: Hidden; + VScrollBarVisibility: Hidden; +} + + .singleLine #contentPresenter { + Padding: 3; + } + +.multiline { /*多行文本框*/ +} + + +.slotLeft { + CornerRadius: 0,4,4,0; + BorderFill: #DCDFE6; + Background: #F5F7FA; + BorderThickness: 1,0,0,0; + BorderType: BorderThickness; + MarginTop: 0; + MarginBottom: 0; + MarginRight: 0; +} + +RadioButton #radioButtonBorder { + StrokeFill: rgb(220,223,230); +} + +RadioButton[IsChecked=true] #radioButtonBorder { + StrokeFill: #1E9FFF; + Fill: #1E9FFF; +} + +RadioButton #optionMark { + StrokeFill: #1E9FFF; + Fill: #fff; +} + +CheckBox #indeterminateMark { + Fill: #1E9FFF; +} + +CheckBox #checkBoxBorder { + Background: #fff; + BorderFill: rgb(220,223,230); +} + +CheckBox[IsChecked=true] #checkBoxBorder { + Background: #1E9FFF; + BorderFill: #1E9FFF; +} + +CheckBox Polyline { + StrokeFill: #fff; +} + +.radioGroup { + BorderType: BorderThickness; + BorderFill: rgb(220,223,230); + BorderThickness: 1,1,0,1; +} + + .radioGroup RadioButton { + BorderType: BorderThickness; + BorderFill: rgb(220,223,230); + BorderThickness: 0,0,1,0; + } + + .radioGroup RadioButton #markPanel { + Visibility: Collapsed; + } + + .radioGroup RadioButton TextBlock { + Margin: 5; + } + + .radioGroup RadioButton[IsChecked=true] { + Background: rgb(64,158,255); + Foreground: #fff; + } + +.error { + Foreground: #f00; + Visibility: Collapsed; +} + + .error[DesignMode=true] { + Visibility: Visible; + } + +.twoLine[AttachedExtenstions.IsError=true] .error { + Visibility: Visible; +} + +.twoLine[AttachedExtenstions.IsError=true] .textBox { + BorderFill: #f00; +} + +ScrollBar { + Background: null; +} + + ScrollBar Thumb { + IsAntiAlias: true; + CornerRadius: 5; + } + + ScrollBar[Orientation=Horizontal] { + Height: 12; + } + + ScrollBar[Orientation=Vertical] { + Width: 12; + } + + ScrollBar #PART_LineUpButton, ScrollBar #PART_LineDownButton { + Visibility: Collapsed; + } + +ComboBox { + Background: #fff; + IsAntiAlias: true; + BorderFill: #DCDFE6; + CornerRadius: 4,4,4,4; + BorderStroke: 1; +} + + ComboBox[IsKeyboardFocusWithin=true] { + BorderFill: #1E9FFF; + } + +#DropDownPanel TextBlock { + MarginLeft: 5; + MarginTop: 2; + MarginBottom: 2; +} + +#dropDownBorder { + ShadowBlur: 2; + ShadowColor: rgba(0, 0, 0, 0.4); + BorderStroke: 0; +} + +#DropDownPanel[IsMouseOver=false] ScrollBar { + Visibility: Collapsed; +} + +#DropDownPanel ScrollBar[Orientation=Horizontal] { + Height: 10; +} + +#DropDownPanel ScrollBar[Orientation=Vertical] { + Width: 10; +} + +Slider { + IsAntiAlias: true; +} + + Slider Thumb { + IsAntiAlias: true; + Width: 16; + Height: 16; + CornerRadius: 7; + BorderFill: rgb(64,158,255); + BorderStroke: 2; + Background: #fff; + ZIndex: 1; + } + + + Slider Thumb[IsMouseOver=true] { + animation-name: sliderMouseOver; + animation-duration: 0.1s; + animation-iteration-count: 1; + animation-fill-mode: forwards; + } + + Slider #TrackBackground { + CornerRadius: 2; + Background: rgb(228,231,237); + BorderStroke: 0; + } + + Slider #decreaseRepeatButton { + Background: rgb(64,158,255); + CornerRadius: 2; + } + + Slider[Orientation=Horizontal] #decreaseRepeatButton { + Height: 4; + MarginLeft: 5; + } + + Slider[Orientation=Vertical] #decreaseRepeatButton { + Width: 4; + MarginBottom: 5; + } + +ProgressBar { + CornerRadius: 5; + IsAntiAlias: true; + BorderFill: null; + Background: rgb(235,238,245); +} + + ProgressBar #Indicator, ProgressBar #Animation { + CornerRadius: 5; + } + +NumericUpDown { + Background: #fff; + IsAntiAlias: true; + BorderFill: #DCDFE6; + CornerRadius: 4,4,4,4; + BorderStroke: 1; +} + + NumericUpDown RepeatButton { + Width: 20; + Background: rgb(245,247,250); + } + + NumericUpDown #decreaseBtn { + CornerRadius: 4,0,0,4; + } + + NumericUpDown #increaseBtn { + CornerRadius: 0,4,4,0; + } + + NumericUpDown #textBoxBorder { + BorderFill: #DCDFE6; + } + + NumericUpDown TextBox { + BorderStroke: 0; + } + +.widget { + IsAntiAlias: true; + BorderFill: #DCDFE6; + CornerRadius: 4,4,4,4; + BorderStroke: 1; +} + +.widgetHead { + Background: linear-gradient(0 0,0 100%,#F7F7F7 0,#F0F0F0 1); + BorderType: BorderThickness; + BorderThickness: 0,0,0,1; + BorderFill: #DCDFE6; +} + +DataGrid { + Foreground: #7a7a7a; +} + +DataGridCellTemplate, DataGridRow, DataGrid, .DataGridCell { + BorderFill: rgb(235,238,245); +} + +DataGridRow { + Height: 36; +} + + DataGridRow[IsMouseOver=true] { + Background: rgb(245,247,250); + } + + DataGridRow[IsSelected=true] { + Background: rgb(245,247,250); + } + +DataGridColumnTemplate { + Height: 38; + FontSize: 15; + FontStyle: Bold; + Background: #fff; + BorderFill: rgb(235,238,245); +} + +TabControl #headBorder { + Background: #fff; +} + +TabItem > Border { + BorderThickness: 0,0,0,2; +} + +TabItem[IsSelected=true] > Border { + BorderFill: #1E9FFF; +} + +TabItem[IsSelected=true] { + Foreground: #1E9FFF; +} + +TabControl[TabStripPlacement=Left] TabItem, TabControl[TabStripPlacement=Right] TabItem { + Width: 100%; + BorderType: BorderThickness; + BorderThickness: 0,0,0,1; + BorderFill: #e8e8e8; +} + + TabControl[TabStripPlacement=Left] TabItem TextBlock { + MarginRight: 0; + } + + TabControl[TabStripPlacement=Left] TabItem > Border { + Width: 100%; + } + +TabControl[TabStripPlacement=Left] #headerPanel, TabControl[TabStripPlacement=Right] #headerPanel { + Width: 100; + Background: rgb(245,247,250); +} + +.closeBtn[IsMouseOver=true] { + Fill: #171717; +} + +.placeholder { + IsHitTestVisible: false; + Foreground: "192,196,204"; + Visibility: Collapsed; +} + +.textBox[AttachedExtenstions.IsEmpty=true] .placeholder { + Visibility: Visible; +} + +.loginBox TextBox, .loginBox .placeholder { + FontSize: 16; +} + +.loginBox CheckBox { + Foreground: #757575; +} + +.searchBox Button { + CornerRadius: 0,4,4,0, +} + +#MenuPop #menuPanel > Border, #MenuPop ContextMenu > Border { + Background: #fff; + ShadowColor: rgba(0, 0, 0, 0.4); +} + +#MenuPop MenuItem[IsMouseOver=true] { + Background: #DCDFE6; +} + +ListBoxItem { + Width: 100%; +} + +#dialogClose, #dialogClose[IsMouseOver=true] { + Background: null; + BorderFill: null; +} + + #dialogClose[IsPressed=true] { + Background: null; + BorderFill: null; + } + + #dialogClose Line { + StrokeFill: 218,218,218; + } + + #dialogClose[IsMouseOver=true] Line { + StrokeFill: #fff; + } diff --git a/CPF.Toolkit/CPF.Toolkit.csproj b/CPF.Toolkit/CPF.Toolkit.csproj new file mode 100644 index 0000000..4eaec2d --- /dev/null +++ b/CPF.Toolkit/CPF.Toolkit.csproj @@ -0,0 +1,15 @@ + + + + netstandard2.0 + + + + + + + + + + + diff --git a/CPF.Toolkit/FodyWeavers.xml b/CPF.Toolkit/FodyWeavers.xml new file mode 100644 index 0000000..d5abfed --- /dev/null +++ b/CPF.Toolkit/FodyWeavers.xml @@ -0,0 +1,3 @@ + + + \ No newline at end of file diff --git a/CPF.Toolkit/FodyWeavers.xsd b/CPF.Toolkit/FodyWeavers.xsd new file mode 100644 index 0000000..69dbe48 --- /dev/null +++ b/CPF.Toolkit/FodyWeavers.xsd @@ -0,0 +1,74 @@ + + + + + + + + + + + Used to control if the On_PropertyName_Changed feature is enabled. + + + + + Used to control if the Dependent properties feature is enabled. + + + + + Used to control if the IsChanged property feature is enabled. + + + + + Used to change the name of the method that fires the notify event. This is a string that accepts multiple values in a comma separated form. + + + + + Used to control if equality checks should be inserted. If false, equality checking will be disabled for the project. + + + + + Used to control if equality checks should use the Equals method resolved from the base class. + + + + + Used to control if equality checks should use the static Equals method resolved from the base class. + + + + + Used to turn off build warnings from this weaver. + + + + + Used to turn off build warnings about mismatched On_PropertyName_Changed methods. + + + + + + + + 'true' to run assembly verification (PEVerify) on the target assembly after all weavers have been executed. + + + + + A comma-separated list of error codes that can be safely ignored in assembly verification. + + + + + 'false' to turn off automatic generation of the XML Schema file. + + + + + \ No newline at end of file diff --git a/CPF.Toolkit/ICloseable.cs b/CPF.Toolkit/ICloseable.cs new file mode 100644 index 0000000..019afd9 --- /dev/null +++ b/CPF.Toolkit/ICloseable.cs @@ -0,0 +1,14 @@ +using CPF.Controls; +using System; +using System.Collections.Generic; +using System.ComponentModel; +using System.Text; + +namespace CPF.Toolkit +{ + public interface ICloseable + { + event EventHandler Closable; + void OnClosable(ClosingEventArgs e); + } +} diff --git a/CPF.Toolkit/ILoaded.cs b/CPF.Toolkit/ILoaded.cs new file mode 100644 index 0000000..3abacb3 --- /dev/null +++ b/CPF.Toolkit/ILoaded.cs @@ -0,0 +1,11 @@ +using System; +using System.Collections.Generic; +using System.Text; + +namespace CPF.Toolkit +{ + public interface ILoaded + { + void OnLoaded(); + } +} diff --git a/CPF.Toolkit/ViewBehavior.cs b/CPF.Toolkit/ViewBehavior.cs new file mode 100644 index 0000000..c5239cf --- /dev/null +++ b/CPF.Toolkit/ViewBehavior.cs @@ -0,0 +1,80 @@ +using CPF.Controls; +using System; +using System.Collections.Generic; +using System.Text; + +namespace CPF.Toolkit +{ + public class ViewBehavior : Behavior + { + protected override void OnBehaviorTo(Window window) + { + window.PropertyChanged += Window_PropertyChanged; + window.Loaded += Window_Loaded; + window.Closing += Window_Closing; + window.Closed += Window_Closed; + base.OnBehaviorTo(window); + } + + protected override void OnDetachingFrom(Window window) + { + window.PropertyChanged -= Window_PropertyChanged; + window.Loaded -= Window_Loaded; + window.Closing -= Window_Closing; + window.Closed -= Window_Closed; + base.OnDetachingFrom(window); + } + + private void Window_Closed(object sender, EventArgs e) + { + var window = (Window)sender; + if (window.DataContext is IDisposable disposable) + { + disposable.Dispose(); + } + } + + private void Window_Closing(object sender, ClosingEventArgs e) + { + var window = (Window)sender; + if (window.DataContext is ICloseable closeable) + { + closeable.OnClosable(e); + } + } + + private void Window_Loaded(object sender, EventArgs e) + { + var window = (Window)sender; + if (window.DataContext is ILoaded loaded) + { + loaded.OnLoaded(); + } + } + + private void Window_PropertyChanged(object sender, CPFPropertyChangedEventArgs e) + { + switch (e.PropertyName) + { + case nameof(DataContext): + { + if (e.NewValue is ICloseable closeable) + { + closeable.Closable -= Closeable_Closable; + closeable.Closable += Closeable_Closable; + } + } + break; + } + + void Closeable_Closable(object _, ClosingEventArgs ee) + { + if (!ee.Cancel) + { + var window = (Window)sender; + window.Close(); + } + } + } + } +} diff --git a/CPF.Toolkit/ViewModelBase.cs b/CPF.Toolkit/ViewModelBase.cs new file mode 100644 index 0000000..f883704 --- /dev/null +++ b/CPF.Toolkit/ViewModelBase.cs @@ -0,0 +1,33 @@ +using CPF.Controls; +using System; +using System.Collections.Generic; +using System.Collections.Specialized; +using System.ComponentModel; +using System.Runtime.CompilerServices; +using System.Text; + +namespace CPF.Toolkit +{ + public class ViewModelBase : INotifyPropertyChanged, ILoaded, IDisposable, ICloseable + { + WeakEventHandlerList events = new WeakEventHandlerList(); + event EventHandler ICloseable.Closable { add => AddHandler(value); remove => RemoveHandler(value); } + + public virtual void Dispose() { } + + void ICloseable.OnClosable(ClosingEventArgs e) => this.OnClosing(e); + + void ILoaded.OnLoaded() => this.OnLoaded(); + + protected virtual void OnLoaded() { } + protected void Close() => this.RaiseEvent(new ClosingEventArgs(), nameof(ICloseable.Closable)); + + protected virtual void OnClosing(ClosingEventArgs e) { } + + public event PropertyChangedEventHandler PropertyChanged; + + protected void AddHandler(Delegate handler, [CallerMemberName] string eventName = null) => events.AddHandler(handler, eventName); + protected void RemoveHandler(Delegate handler, [CallerMemberName] string eventName = null) => events.RemoveHandler(handler, eventName); + protected void RaiseEvent(in TEventArgs eventArgs, string eventName) => events[eventName]?.Invoke(this, eventArgs); + } +} diff --git a/CPF/Behavior.cs b/CPF/Behavior.cs new file mode 100644 index 0000000..f8609bd --- /dev/null +++ b/CPF/Behavior.cs @@ -0,0 +1,70 @@ +using System; +using System.Collections.Generic; +using System.Text; + +namespace CPF +{ + public interface IBehaviorObject + { + void BehaviorTo(CpfObject cpfObject); + void DetachFrom(CpfObject cpfObject); + } + + public abstract class Behavior : CpfObject, IBehaviorObject + { + protected Behavior() : this(typeof(CpfObject)) + { + } + internal Behavior(Type associatedType) => AssociatedType = associatedType ?? throw new ArgumentNullException(nameof(associatedType)); + + protected Type AssociatedType { get; } + + void IBehaviorObject.BehaviorTo(CpfObject cpfObject) + { + if (cpfObject == null) + throw new ArgumentNullException(nameof(cpfObject)); + if (!AssociatedType.IsInstanceOfType(cpfObject)) + throw new InvalidOperationException("object not an instance of AssociatedType"); + OnBehaviorTo(cpfObject); + } + + void IBehaviorObject.DetachFrom(CpfObject cpfObject) => OnDetachingFrom(cpfObject); + + protected virtual void OnBehaviorTo(CpfObject cpfObject) + { + + } + + protected virtual void OnDetachingFrom(CpfObject cpfObject) + { + + } + } + + public abstract class Behavior : Behavior where T : CpfObject + { + protected Behavior() : base(typeof(T)) + { + } + + protected override void OnBehaviorTo(CpfObject cpfObject) + { + base.OnBehaviorTo(cpfObject); + OnBehaviorTo((T)cpfObject); + } + + protected virtual void OnBehaviorTo(T cpfObject) + { + } + + protected override void OnDetachingFrom(CpfObject cpfObject) + { + OnDetachingFrom((T)cpfObject); + base.OnDetachingFrom(cpfObject); + } + + protected virtual void OnDetachingFrom(T cpfObject) + { + } + } +} diff --git a/CPF/Controls/Window.cs b/CPF/Controls/Window.cs index e7d80da..a717e8d 100644 --- a/CPF/Controls/Window.cs +++ b/CPF/Controls/Window.cs @@ -28,6 +28,12 @@ namespace CPF.Controls IWindowImpl windowImpl; + bool isLoaded = false; + public event EventHandler Loaded + { + add => AddHandler(value); + remove => RemoveHandler(value); + } public event EventHandler Closed { add { AddHandler(value); } @@ -144,6 +150,11 @@ namespace CPF.Controls public void Show() { + if (!this.isLoaded) + { + this.BeginInvoke(() => RaiseEvent(EventArgs.Empty, nameof(Loaded))); + this.isLoaded = true; + } //windowImpl.Show(); Visibility = Visibility.Visible; } diff --git a/CPF/CpfObject.cs b/CPF/CpfObject.cs index a7bb332..8a80396 100644 --- a/CPF/CpfObject.cs +++ b/CPF/CpfObject.cs @@ -178,6 +178,42 @@ namespace CPF } } + internal IList behaviors; + [NotCpfProperty] + [Category("绑定")] + [Description("设置行为")] + public IList Behaviors + { + get + { + if (this.behaviors == null) + { + var b = new Collection(); + b.CollectionChanged -= Behavior_CollectionChanged; + b.CollectionChanged += Behavior_CollectionChanged; + this.behaviors = b; + } + return this.behaviors; + } + } + + private void Behavior_CollectionChanged(object sender, CollectionChangedEventArgs e) + { + switch (e.Action) + { + case CollectionChangedAction.Add: + (e.NewItem as IBehaviorObject).BehaviorTo(this); + break; + case CollectionChangedAction.Remove: + (e.NewItem as IBehaviorObject).DetachFrom(this); + break; + case CollectionChangedAction.Replace: + break; + case CollectionChangedAction.Sort: + break; + } + } + Type type; /// /// 设置绑定 @@ -1387,7 +1423,6 @@ namespace CPF { SetCommand(cpc, list1); } - RaiseEvent(cpc, strPropertyChanged); //PropertyChangedEventHandler handler = (PropertyChangedEventHandler)Events["INotifyPropertyChanged"]; @@ -1500,7 +1535,6 @@ namespace CPF } } - /// /// 当要设置属性值的时候,返回值为true的时候将设置值 /// diff --git a/ConsoleApp1.sln b/ConsoleApp1.sln index f0d8f66..de7b0aa 100644 --- a/ConsoleApp1.sln +++ b/ConsoleApp1.sln @@ -1,7 +1,7 @@  Microsoft Visual Studio Solution File, Format Version 12.00 -# Visual Studio Version 16 -VisualStudioVersion = 16.0.29324.140 +# Visual Studio Version 17 +VisualStudioVersion = 17.8.34316.72 MinimumVisualStudioVersion = 10.0.40219.1 Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ConsoleApp1", "ConsoleApp1\ConsoleApp1.csproj", "{ABE4ED47-CB9F-4183-9CE3-65E3E521BE2A}" EndProject @@ -63,12 +63,11 @@ Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "CPF.Razor", "CPF.Razor\CPF. EndProject Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "CpfRazorSample", "CpfRazorSample\CpfRazorSample.csproj", "{25A4EE47-F5BD-4F1E-B143-3E3B50C5AC2A}" EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "CPF.Toolkit", "CPF.Toolkit\CPF.Toolkit.csproj", "{0CB0CC36-940B-45C1-83D1-0A9FF68CB157}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "CPF.Toolkit.Demo", "CPF.Toolkit.Demo\CPF.Toolkit.Demo.csproj", "{BC4120CE-5AE4-4577-98D4-BFF7C4B067B4}" +EndProject Global - GlobalSection(SharedMSBuildProjectFiles) = preSolution - Private\SharedVSIX\SharedVSIX.projitems*{db53e8d7-dfb6-48eb-a7b6-d1cf762acb9b}*SharedItemsImports = 4 - Private\SharedVSIX\SharedVSIX.projitems*{df526631-d060-47f2-afd4-62c6cea2fe9a}*SharedItemsImports = 4 - Private\SharedVSIX\SharedVSIX.projitems*{f34cffee-546f-490e-a76a-2792840b284d}*SharedItemsImports = 13 - EndGlobalSection GlobalSection(SolutionConfigurationPlatforms) = preSolution Debug|Any CPU = Debug|Any CPU Release|Any CPU = Release|Any CPU @@ -240,6 +239,18 @@ Global {25A4EE47-F5BD-4F1E-B143-3E3B50C5AC2A}.Release|Any CPU.Build.0 = Release|Any CPU {25A4EE47-F5BD-4F1E-B143-3E3B50C5AC2A}.类库d|Any CPU.ActiveCfg = Debug|Any CPU {25A4EE47-F5BD-4F1E-B143-3E3B50C5AC2A}.类库d|Any CPU.Build.0 = Debug|Any CPU + {0CB0CC36-940B-45C1-83D1-0A9FF68CB157}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {0CB0CC36-940B-45C1-83D1-0A9FF68CB157}.Debug|Any CPU.Build.0 = Debug|Any CPU + {0CB0CC36-940B-45C1-83D1-0A9FF68CB157}.Release|Any CPU.ActiveCfg = Release|Any CPU + {0CB0CC36-940B-45C1-83D1-0A9FF68CB157}.Release|Any CPU.Build.0 = Release|Any CPU + {0CB0CC36-940B-45C1-83D1-0A9FF68CB157}.类库d|Any CPU.ActiveCfg = Debug|Any CPU + {0CB0CC36-940B-45C1-83D1-0A9FF68CB157}.类库d|Any CPU.Build.0 = Debug|Any CPU + {BC4120CE-5AE4-4577-98D4-BFF7C4B067B4}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {BC4120CE-5AE4-4577-98D4-BFF7C4B067B4}.Debug|Any CPU.Build.0 = Debug|Any CPU + {BC4120CE-5AE4-4577-98D4-BFF7C4B067B4}.Release|Any CPU.ActiveCfg = Release|Any CPU + {BC4120CE-5AE4-4577-98D4-BFF7C4B067B4}.Release|Any CPU.Build.0 = Release|Any CPU + {BC4120CE-5AE4-4577-98D4-BFF7C4B067B4}.类库d|Any CPU.ActiveCfg = Debug|Any CPU + {BC4120CE-5AE4-4577-98D4-BFF7C4B067B4}.类库d|Any CPU.Build.0 = Debug|Any CPU EndGlobalSection GlobalSection(SolutionProperties) = preSolution HideSolutionNode = FALSE @@ -260,4 +271,9 @@ Global GlobalSection(ExtensibilityGlobals) = postSolution SolutionGuid = {16FB883C-167C-4E1A-B311-6D74452A3CD6} EndGlobalSection + GlobalSection(SharedMSBuildProjectFiles) = preSolution + Private\SharedVSIX\SharedVSIX.projitems*{db53e8d7-dfb6-48eb-a7b6-d1cf762acb9b}*SharedItemsImports = 4 + Private\SharedVSIX\SharedVSIX.projitems*{df526631-d060-47f2-afd4-62c6cea2fe9a}*SharedItemsImports = 4 + Private\SharedVSIX\SharedVSIX.projitems*{f34cffee-546f-490e-a76a-2792840b284d}*SharedItemsImports = 13 + EndGlobalSection EndGlobal