如何从C#中的表单返回值?


185

我有一个主窗体(我们称其为frmHireQuote),它是MDI主窗体(frmMainMDI)的子窗体,单击按钮时通过ShowDialog()显示了另一个窗体(frmImportContact)。

当用户单击frmImportContact上的“确定”时,我想将一些字符串变量传递回frmHireQuote上的某些文本框。

请注意,可能有多个frmHireQuote实例,很重要的一点是,我要回到调用此实例frmImportContact的实例。

最好的方法是什么?

Answers:


367

像这样在子表单上创建一些公共属性

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属性设置为该按钮-这将使转义键可以取消的形式。


33
一些重要的缺陷。与ShowDialog()一起显示的表单没有被处理。您必须在此处使用using语句。ShowDialog()返回DialogResult值,请不要忽略它,否则当用户取消对话框时,您将尝试使用对话框属性。
汉斯·帕桑

干杯,我已经解释了这一点。
杰兹·克拉克

迷你批判:设置需要附加分号。
B. Clay Shannon

16

我通常在窗体/对话框上创建一个可以调用的静态方法。这将返回成功(“确定”按钮)或失败以及需要填写的值。

 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
   }
}

也使用了这种方法,效果很好。但是给结果类加上前缀frm(当我阅读它时才这样做)会引起混乱……
Richard Friend

我只是重用了问题的名称(+快速键入)。我将其更改为更好的名称。
GvS

6

在此代码中发现了另一个小问题……或者至少在我尝试实现它时出现了问题。

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;
     }
}

添加两个按钮值后,对话框效果很好!感谢您的示例,它确实有所帮助。


1

我只是通过引用将一些东西放入构造函数中,所以子窗体可以更改其值,主窗体可以从子窗体中获取新对象或修改后的对象。


0

我使用MDI的次数很多,与多种浮动形式相比,我更喜欢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,您可以将其简化为一个函数。并且您可以提供中断的代码,这对调试非常有用,但是在其他方面也很有用。

只要让您的所有中间事件代码都指向一个函数,

        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);

一个简单的切换机制通常足以将事件传递给适当的形式。


0

如果您想不进行新的传递就form2从传递数据form1form(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

0

首先,您必须在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(父级)调用form2(子级):

  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); 
  }

0

我以设置值的形式引发事件,并以需要处理值更改的形式订阅该事件。


嘿!您能否详细说明您的答案,并可能提供代码示例以帮助将来的用户?
杰里米·哈里斯

0

代表是将数据从一种形式发送到另一种形式的最佳选择。

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");
    }
}
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.