我希望从WinForm将数据传递到WPF窗口,并从WPF窗口接收消息。
我的代码是随机在线教程和HighCore log viewer的混合。我有一个WinForm以下列方式启动我的新WPF窗口:
private void openTransactionViewToolStripMenuItem_Click(object sender, EventArgs e)
{
var transactionViewWindow = new TransactionViewer.MainWindow();
ElementHost.EnableModelessKeyboardInterop(transactionViewWindow);
transactionViewWindow.Show();
transactionViewWindow.Test = "test"; // testing out data passing
transactionViewWindow.AddTest();
}
我的MainWindow.xaml.cs看起来像:
public partial class MainWindow : Window
{
public ObservableCollection<Session> SessionList { get; set; }
public string Test{ get; set; }
public MainWindow()
{
InitializeComponent();
SessionList = new ObservableCollection<Session>();
SessionList.Add(new Session() { BeginLine = 0, EndLine = 1, Message = "some message" });
SessionList.Add(new Session() { BeginLine = 2, EndLine = 3, Message = "another message" });
SessionItems.ItemsSource = SessionList; // the ItemsControl
}
public void AddTest()
{
SessionList.Add(new Session() { BeginLine = 4, EndLine = 5, Message = Test });
}
}
public class Session : PropertyChangedBase
{
public int BeginLine { get; set; }
public int EndLine { get; set; }
public string Message { get; set; }
}
其中PropertyChangedBase
继承自INotifyPropertyChanged
。我有一个绑定到Message
的ItemsControl。我的输出如下:
一些消息
另一条消息
测试
“数据传递”成功!最后,当WPF窗口加载时,我想从我的WinForm传递一个List<Session>
,用于填充ItemsControl。我还想在WinForm上有一个按钮,它将发送一个List来重新填充/刷新WPF中的数据。从目前的行为来看,我认为即使使用我当前的简单实现(仅更新SessionList
),这也是可能的。
有没有更合适的方法呢?事件,例如?我是否需要启动一个事件才能告诉我的WinForm WPF已经成功添加了所有Session
个对象,或者当用户点击某个特定对象时?
在这里使用MVVM有什么好处?
我一直在为WinForms开发一段时间,并且发现向WPF过渡非常困惑。希望有人可以提供一些指导或代码示例。
编辑:为了将来的参考,可以在这里找到针对像我这样的人的体面的MVVM教程:http://jpreecedev.com/2013/06/08/wpf-mvvm-for-winforms-devs-part-1-of-5/
答案 0 :(得分:1)
你对我的态度似乎没问题。它并不完美,但它足够可行。
最佳方法IMO将为WPF窗口创建一个ViewModel,而不是在来回传递数据时直接引用Window
本身。
这个想法是:
public class MyForm: Form
{
public TransactionViewerViewModel TransactionViewer {get;set;}
//... other code...
//Form constructor:
public MyForm()
{
InitializeComponent();
//Create ViewModel:
TransactionViewer = new TransactionViewerViewModel();
}
private void openTransactionViewToolStripMenuItem_Click(object sender, EventArgs e)
{
//Create WPF View:
var transactionViewWindow = new TransactionViewer.MainWindow();
//Interop code
ElementHost.EnableModelessKeyboardInterop(transactionViewWindow);
//Set DataContext:
transactionViewWindow.DataContext = TransactionViewer;
//Show Window:
transactionViewWindow.Show();
//Call methods on the ViewModel, rather than the View:
TransactionViewer.Test = "test"; // testing out data passing
TransactionViewer.AddTest();
}
}
因此,ViewModel将类似于:
public class TransactionViewerViewModel : PropertyChangedBase
{
public ObservableCollection<Session> SessionList { get; set; }
public string Test{ get; set; }
public TransactionViewerViewModel()
{
SessionList = new ObservableCollection<Session>();
SessionList.Add(new Session() { BeginLine = 0, EndLine = 1, Message = "some message" });
SessionList.Add(new Session() { BeginLine = 2, EndLine = 3, Message = "another message" });
}
public void AddTest()
{
SessionList.Add(new Session() { BeginLine = 4, EndLine = 5, Message = Test });
}
}
这实现了WPF UI与实际数据/业务逻辑之间的完美分离,甚至可以为ViewModel创建单元测试。
而且,由于您将ViewModel设置为Window的DataContext
,因此您需要通过DataBinding访问所有ViewModel属性,而不是过程代码:
<ItemsControl ItemsSource="{Binding SessionList}"/>
然后,您可能希望在ViewModel中引入delegates
或events
,并在您的表单中收听这些内容,从而实现WPF => winforms
通信。