我有一个主窗体(我们称它为 frmHireQuote),它是主 MDI 窗体(frmMainMDI)的子窗体,单击按钮时通过 ShowDialog() 显示另一个窗体(frmImportContact)。
当用户在 frmImportContact 上单击“确定”时,我想将一些字符串变量传递回 frmHireQuote 上的一些文本框。
请注意,可能有多个 frmHireQuote 实例,返回到调用此 frmImportContact 实例的实例显然很重要。
最好的方法是什么?
像这样在子表单上创建一些公共属性
public string ReturnValue1 {get;set;}
public string ReturnValue2 {get;set;}
然后在您的子表单确定按钮单击处理程序中设置
private void btnOk_Click(object sender,EventArgs e)
{
this.ReturnValue1 = "Something";
this.ReturnValue2 = DateTime.Now.ToString(); //example
this.DialogResult = DialogResult.OK;
this.Close();
}
然后在您的 frmHireQuote 表单中,当您打开子表单
using (var form = new frmImportContact())
{
var result = form.ShowDialog();
if (result == DialogResult.OK)
{
string val = form.ReturnValue1; //values preserved after close
string dateString = form.ReturnValue2;
//Do something here with these values
//for example
this.txtSomething.Text = val;
}
}
此外,如果您希望取消子表单,您只需向表单添加一个按钮并将其DialogResult设置为
Cancel
,您还可以将表单的CancelButton属性设置为所述按钮- 这将使转义键能够取消表格。
我通常在表单/对话框上创建一个我可以调用的静态方法。这将返回成功(确定按钮)或失败,以及需要填写的值。
public class ResultFromFrmMain {
public DialogResult Result { get; set; }
public string Field1 { get; set; }
}
在表格上:
public static ResultFromFrmMain Execute() {
using (var f = new frmMain()) {
var result = new ResultFromFrmMain();
result.Result = f.ShowDialog();
if (result.Result == DialogResult.OK) {
// fill other values
}
return result;
}
}
调用您的表格;
public void MyEventToCallForm() {
var result = frmMain.Execute();
if (result.Result == DialogResult.OK) {
myTextBox.Text = result.Field1; // or something like that
}
}
发现这段代码的另一个小问题......或者至少当我尝试实现它时它是有问题的。
frmMain 中的按钮不返回兼容值,使用 VS2010 我添加了以下内容,一切开始正常工作。
public static ResultFromFrmMain Execute() {
using (var f = new frmMain()) {
f.buttonOK.DialogResult = DialogResult.OK;
f.buttonCancel.DialogResult = DialogResult.Cancel;
var result = new ResultFromFrmMain();
result.Result = f.ShowDialog();
if (result.Result == DialogResult.OK) {
// fill other values
}
return result;
}
}
添加两个按钮值后,对话框运行良好! 谢谢你的例子,它真的很有帮助。
委托是将数据从一种形式发送到另一种形式的最佳选择。
public partial class frmImportContact : Form
{
public delegate void callback_data(string someData);
public event callback_data getData_CallBack;
private void button_Click(object sender, EventArgs e)
{
string myData = "Top Secret Data To Share";
getData_CallBack(myData);
}
}
public partial class frmHireQuote : Form
{
private void Button_Click(object sender, EventArgs e)
{
frmImportContact obj = new frmImportContact();
obj.getData_CallBack += getData;
}
private void getData(string someData)
{
MessageBox.Show("someData");
}
}
我只是通过引用将一些东西放入构造函数中,因此子窗体可以更改其值,主窗体可以从子窗体获取新的或修改的对象。
如果你想从
form2
传递数据到form1
而不像新的那样传递form(sting "data");
那样做 在表格 1
using (Form2 form2= new Form2())
{
form2.ReturnValue1 = "lalala";
form2.ShowDialog();
}
在表格 2 中添加
public string ReturnValue1 { get; set; }
private void form2_Load(object sender, EventArgs e)
{
MessageBox.Show(ReturnValue1);
}
如果你想在
form1
中交换一些东西,你也可以像这样在
form1
中使用值
就在 form1
textbox.Text =form2.ReturnValue1
我用MDI挺多的,比起多重浮动形式我更喜欢它(能用的地方)
但是要从中获得最大收益,您需要掌握自己的事件。它让你的生活变得如此轻松。
一个骨骼的例子。
有自己的中断类型,
//Clock, Stock and Accoubts represent the actual forms in
//the MDI application. When I have multiple copies of a form
//I also give them an ID, at the time they are created, then
//include that ID in the Args class.
public enum InteruptSource
{
IS_CLOCK = 0, IS_STOCKS, IS_ACCOUNTS
}
//This particular event type is time based,
//but you can add others to it, such as document
//based.
public enum EVInterupts
{
CI_NEWDAY = 0, CI_NEWMONTH, CI_NEWYEAR, CI_PAYDAY, CI_STOCKPAYOUT,
CI_STOCKIN, DO_NEWEMAIL, DO_SAVETOARCHIVE
}
然后是你自己的 Args 类型
public class ControlArgs
{
//MDI form source
public InteruptSource source { get; set; }
//Interrupt type
public EVInterupts clockInt { get; set; }
//in this case only a date is needed
//but normally I include optional data (as if a C UNION type)
//the form that responds to the event decides if
//the data is for it.
public DateTime date { get; set; }
//CI_STOCKIN
public StockClass inStock { get; set; }
}
然后在您的命名空间内使用委托,但在类之外
namespace MyApplication
{
public delegate void StoreHandler(object sender, ControlArgs e);
public partial class Form1 : Form
{
//your main form
}
现在手动或使用 GUI,让 MDIparent 响应子窗体的事件。
但是使用您的 owr Args,您可以将其简化为一个函数。并且您可以设置中断中断,这对调试很有用,但在其他方面也很有用。
只需让所有的 mdiparent 事件代码指向一个函数,
calendar.Friday += new StoreHandler(MyEvents);
calendar.Saturday += new StoreHandler(MyEvents);
calendar.Sunday += new StoreHandler(MyEvents);
calendar.PayDay += new StoreHandler(MyEvents);
calendar.NewYear += new StoreHandler(MyEvents);
一个简单的切换机制通常足以将事件传递给适当的形式。
首先,您必须在 form2(child) 中定义属性,您将在 form2 和 form1(parent) 中更新此属性:
public string Response { get; set; }
private void OkButton_Click(object sender, EventArgs e)
{
Response = "ok";
}
private void CancelButton_Click(object sender, EventArgs e)
{
Response = "Cancel";
}
从 form1(parent)调用 form2(child):
using (Form2 formObject= new Form2() )
{
formObject.ShowDialog();
string result = formObject.Response;
//to update response of form2 after saving in result
formObject.Response="";
// do what ever with result...
MessageBox.Show("Response from form2: "+result);
}
我在设置值的表单中引发一个事件,并在需要处理值更改的表单中订阅该事件。
您可以使用对话窗体的
tag
属性。这也是 Control.Tag 参考文献推荐的:
Tag 属性的一个常见用途是存储与控件密切相关的数据。
// Button handler of dialog form:
private void btnOk_Click(object sender, EventArgs e)
{
// Create a custom result container with some strings to return.
this.Tag = new MyResult(resultString1, resultString2);
this.DialogResult = DialogResult.OK; // signals to close dialog
}
在调用形式中:
if (frm.ShowDialog() == DialogResult.OK)
{
MyResult result = frm.Tag as MyResult;
...
}
这也适用于 WPF 对话框。