tabControl技巧_第1页
tabControl技巧_第2页
tabControl技巧_第3页
tabControl技巧_第4页
tabControl技巧_第5页
已阅读5页,还剩21页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1、前言private void tabControl1_SelectedIndexChanged(object sender, System.EventArgs e) if(tabControl1.SelectedIndex=1) this.tabControl1.SelectedIndex=0; TabControl控件,每个选项卡使用TabItem正文父窗体上面放tabcontrol ,在page上面放一个panel,新窗体(fromborderstyle设置为none比较好看)停靠到panel上 Form3 f = new Form3(); f.TopLevel = false; f.Sh

2、ow(); f.Parent = this.panel1; = 只所以用这样的设计, 方便的是设计,每个模块有自己的UI 自己的控制器 而如果都放到tabcontrol上面,处理功能很杂 不利于项目的管理 = 如果你非要类似遨游那种的选项卡页面 可以做一个假的,比如在窗体的上方有一排picture,平时就显示一个小画面,或者干脆隐藏。 当打开一个窗体后就把其中一个关联到窗体,它的text显示窗体名称,双击关闭对应窗体,单击则对应窗体显示到顶层 可以把这些公共事件封装到一个控件,拖过来用,把窗体name作为传入值/完整代码如下: using System; using System.Drawin

3、g; using System.Collections; using System.ComponentModel; using System.Windows.Forms; using System.Data; namespace TestApp public class Form1 : System.Windows.Forms.Form private System.Windows.Forms.TabControl tabControl1; private System.Windows.Forms.TabPage tabPage1; private System.Windows.Forms.T

4、abPage tabPage2; private System.Windows.Forms.TabPage tabPage3; private System.ComponentModel.Container components = null; public Form1() InitializeComponent(); protected override void Dispose( bool disposing ) if( disposing ) if (components != null) components.Dispose(); base.Dispose( disposing );

5、#region Windows 窗体设计器生成的代码 private void InitializeComponent() this.tabControl1 = new System.Windows.Forms.TabControl(); this.tabPage1 = new System.Windows.Forms.TabPage(); this.tabPage2 = new System.Windows.Forms.TabPage(); this.tabPage3 = new System.Windows.Forms.TabPage(); this.tabControl1.Suspend

6、Layout(); this.SuspendLayout(); / / tabControl1 / this.tabControl1.Controls.Add(this.tabPage1); this.tabControl1.Controls.Add(this.tabPage2); this.tabControl1.Controls.Add(this.tabPage3); this.tabControl1.Location = new System.Drawing.Point(24, 56); this.tabControl1.Name = tabControl1; this.tabContr

7、ol1.SelectedIndex = 0; this.tabControl1.Size = new System.Drawing.Size(240, 144); this.tabControl1.TabIndex = 0; this.tabControl1.SelectedIndexChanged += new System.EventHandler(this.tabControl1_SelectedIndexChanged); / / tabPage1 / this.tabPage1.Location = new System.Drawing.Point(4, 21); this.tabP

8、age1.Name = tabPage1; this.tabPage1.Size = new System.Drawing.Size(232, 119); this.tabPage1.TabIndex = 0; this.tabPage1.Text = tabPage1; / / tabPage2 / this.tabPage2.Location = new System.Drawing.Point(4, 21); this.tabPage2.Name = tabPage2; this.tabPage2.Size = new System.Drawing.Size(232, 119); thi

9、s.tabPage2.TabIndex = 1; this.tabPage2.Text = tabPage2; / / tabPage3 / this.tabPage3.Location = new System.Drawing.Point(4, 21); this.tabPage3.Name = tabPage3; this.tabPage3.Size = new System.Drawing.Size(232, 119); this.tabPage3.TabIndex = 2; this.tabPage3.Text = tabPage3; / / Form1 / this.AutoScal

10、eBaseSize = new System.Drawing.Size(6, 14); this.ClientSize = new System.Drawing.Size(292, 266); this.Controls.Add(this.tabControl1); this.Name = Form1; this.Text = Form1; this.tabControl1.ResumeLayout(false); this.ResumeLayout(false); #endregion STAThread static void Main() Application.Run(new Form

11、1(); private void tabControl1_SelectedIndexChanged(object sender, System.EventArgs e) if(tabControl1.SelectedIndex=1) this.tabControl1.SelectedIndex=0; C# tabControl灵活控制显示,不用重绘很多人都希望用C# tabControl的显示与取消实现某种效果,但是微软的官方是不建议控制tabControl中的tabpage的,这就造成做项目遇到困难。我们这里有几个高级程序员用C#写API也很不错,但是我还是保持我的风格,毕竟我是他们的老大

12、,架构师嘛。O(_)O,我告诉他们寻找最简单方法,副作用不是没有,不过对于不是高深的应用副作用不是问题,毕竟不需要重绘。看代码。 private void Index_Load(object sender, EventArgs e) /tabcontrol.TabPages.RemoveAt(tabPageSet);/把焦点放置在哪? tabControlMain.TabPages.Remove(tabPageSet); /移除标签页tabControlMain.TabPages.Add(tabPageSet); /添加标签页if(tabControlMain.SelectedTab = ta

13、bpageStart) /判断标签页的位置 这四行代码如果使用,那么请不要再去判断标签页的索引了,因为索引会完全打乱,而且再次填充标签页的时候请做一个判断,因为同样标签页可以反复添加。就这样吧,简单的东西解决相对复杂的问题。我觉得有能力写重绘也不错。轻松掌握Windows窗体间的数据交互 Windows 窗体是用于 Microsoft Windows 应用程序开发的、基于 .NET Framework 的新平台。此框架提供一个有条理的、面向对象的、可扩展的类集,它使您得以开发丰富的 Windows 应用程序。一个Windows窗体就代表了.NET架构里的System.Windows.Forms

14、.Form类的一个实例。笔者在CSDN技术论坛.NET板块下的C#分类经常看到有人问起如何在两个Form间传递数据,访问修改对方窗体里面的值。对于有经验的程序员来说不是什么高深的东西,而对于初学者来说这些基础的东西往往是一个问题,并且存在这种现象,往往比较复杂的东西他们会,要用什么了就去学什么,实际上并没有真正的去理解掌握它,基础不扎实,所以就有了想通过自己对窗体编程积累的经验来写一些这方面的文章,以供学.NET的朋友参考,也借此机会同各位朋友进行交流,写得不合理的地方请各位朋友提宝贵意见,下面我分了三个部分来讲。一使用带参数的构造函数我们要做的准备工作就是新建两个窗体,下面是两个窗体的布局,

15、很简单:说明:Form1为主窗体,包含控件:文本框textBoxFrm1,多选框checkBoxFrm1和按钮buttonEdit;Form2为子窗体,包含控件:文本框textBoxFrm2,多选框checkBoxFrm2和按钮buttonOKbuttonCancel。当我们新建一个窗体的时候,设计器会生成默认的构造函数:public Form2() InitializeComponent();它不带参数,既然我们要把Form1中的一些数据传到Form2中去,为什么不在Form2的构造函数里做文章呢?假设我们要实现使Form2中的文本框显示Form1里textBoxFrm1的值,修改子窗体的构

16、造函数:public Form2(string text) InitializeComponent(); this.textBoxFrm2.Text = text;增加Form1中的修改按钮点击事件,处理函数如下:private void buttonEdit_Click(object sender, System.EventArgs e) Form2 formChild = new Form2(this.textBoxFrm1.Text); formChild.Show();我们把this.textBoxFrm1.Text作为参数传到子窗体构造函数,以非模式方式打开,这样打开的formChi

17、ld的文本框就显示了”主窗体”文本,是不是很简单,接下来我们传一个boolean数据给子窗体。Public Form2(string text,bool checkedValue) InitializeComponent(); this.textBoxFrm2.Text = text; this.checkBoxFrm2.Checked = checkedValue;在主窗体中的修改按钮点击处理,我采用了打开模式窗口的方式,其实在这个例子中看不出有什么分别,private void buttonEdit_Click(object sender, System.EventArgs e) Form

18、2 formChild = new Form2(this.textBoxFrm1.Text,this.checkBoxFrm1.Checked); formChild.ShowDialog();结果在预料之中,但是这里明显存在不足,在子窗体里的数据修改后不能传给主窗体,也就是说主窗体不受子窗体的影响。而在实际的开发过程中我们经常使用子窗体来修改主窗体里面的数据,那怎么解决呢? 在.NET中有两种类型,值类型和引用类型。值类型是从ValueType继承而来,而ValueType又是从Object继承;对于引用类型它直接继承Object类型。这下让我们看看怎样通过Form2来修改Form1里的数据

19、。还是让我们来修改Form2的代码。Private TextBox textBoxFrm12;private CheckBox checkBoxFrm12;public Form2(TextBox heckbo,CheckBox heckbox) InitializeComponent(); this.textBoxFrm2.Text = heckbo.Text; this.checkBoxFrm2.Checked = heckbox.Checked; this.textBoxFrm12 = heckbo; this.checkBoxFrm12 = heckbox;现在我们传了两个引用类型的数

20、据:TextBox类型,和CheckBox;另外在Form2中增加了两个类数据成员textBoxFrm12、checkBoxFrm12用来分别保存构造函数传来的变量,不过他们并不属于Form2的Controls容器。修改Form2的确定按钮点击事件函数:private void buttonOK_Click(object sender, System.EventArgs e) this.textBoxFrm12.Text = this.textBoxFrm2.Text; this.checkBoxFrm12.Checked = this.checkBoxFrm2.Checked; this.C

21、lose();上面的代码我们通过把textBoxFrm2的Text和checkBoxFrm2.Checked赋给textBoxFrm12和checkBoxFrm12完成了对主窗体中的textBoxFrm1和checkBoxFrm2的修改,因为textBoxFrm1和textBoxFrm12是同一个引用,而checkBoxFrm2和checkBoxFrm12也是。到这里为止功能是实现了,但是总觉得不是很合理,让两个窗体控件传来传去,现在我举一个恰当一点的例子。修改了两个窗体:说明:在这个例子中我们的两个窗体都加了一个ListBox用来显示ArrayList中的内容。主窗体中控件:listBoxF

22、rm1,buttonEdit;子窗体中控件:listBoxFrm2,textBoxAdd,buttonAdd,buttonEdit,buttonOK。这次我们用ArrayList来作为传递数据,在Form1中定义类数据成员:private ArrayList listData1;在构造函数中增加了对listData1进行内存分配,并生成数据最终绑定到listBoxFrm1,public Form1() InitializeComponent(); this.listData1 = new ArrayList(); this.listData1.Add(DotNet); this.listDat

23、a1.Add(C#); this.listData1.Add(A); this.listData1.Add(WebService); this.listData1.Add(XML); this.listBoxFrm1.DataSource = this.listData1;另外,对修改按钮点击事件处理函数的修改如下:private void buttonEdit_Click(object sender, System.EventArgs e) Form2 formChild = new Form2(this.listData1); formChild.ShowDialog(); this.li

24、stBoxFrm1.DataSource = null; this.listBoxFrm1.DataSource = this.listData1;相对与主窗体,对子窗体作相应修改,也在Form2中增加了类数据成员:private ArrayList listData2;用来保存对主窗体中listData1的引用。修改构造函数:public Form2(ArrayList listData) InitializeComponent(); this.listData2 = listData; foreach(object o in this.listData2) this.listBoxFrm2

25、.Items.Add(o); 这里让listData2同listData1指向同一个引用;另外没有对listBoxFrm进行绑定,采用了填充。好了,下面是对数据操作的时候了。添加处理函数代码如下:private void buttonAdd_Click(object sender, System.EventArgs e) if(this.textBoxAdd.Text.Trim().Length0) this.listData2.Add(this.textBoxAdd.Text.Trim(); this.listBoxFrm2.Items.Add(this.textBoxAdd.Text.Tr

26、im(); else MessageBox.Show(请输入添加的内容!);删除处理代码如下:private void buttonDel_Click(object sender, System.EventArgs e) int index = this.listBoxFrm2.SelectedIndex; if(index!=-1) this.listData2.RemoveAt(index); this.listBoxFrm2.Items.RemoveAt(index); else MessageBox.Show(请选择删除项或者没有可删除的项!); 退出Form2子窗体:private

27、void buttonOK_Click(object sender, System.EventArgs e) this.Close();编译运行程序,在子窗体中对数据进行修改,关闭后,主窗体就会显示更新后的数据。这里有一点要提醒一下,比较两个例子,我们都传的是引用类型,一个是String,另一个是ArrayList,为什么string类型不能修改主窗体的数据呢?其实在.Net中对string类型的修改并不是修改原来的值,原来的值没有变化,而是重新生成一个新的字符串,下面是一个很好的说明。public class ZZConsole STAThread static void Main(stri

28、ng args) string str1 = abc; string str2 = str1; str1 = 123; Console.WriteLine(str1); Console.WriteLine(-); Console.WriteLine(str2); Console.WriteLine(-); ArrayList al1 = new ArrayList(); al1.Add(abc); ArrayList al2 = al1; al2.Add(123); foreach(object o in al1) Console.WriteLine(string)o); Console.Wr

29、iteLine(-); foreach(object o in al2) Console.WriteLine(string)o); Console.ReadLine(); 运行一下看看输出结果就明白了,另外对值类型的数据操作要使用ref关键字。 总结,我们通过带参数的构造函数实现了窗体间的数据交互,代码看上去也比较清楚,在实际开发过程中,可以把DataSet,DataTable,或者是DataView当作参数,当然如果只是想修改一行,可以传个DataRow或者DataRowView。在下我们来看看怎样使用另外两种方法来实现数据的交互。二给窗体添加属性或方法1使用Form类的Owner属性获取或

30、设置拥有此窗体的窗体。若要使某窗体归另一个窗体所有,请为其 Owner 属性分配一个对将成为所有者的窗体的引用。当一个窗体归另一窗体所有时,它便随着所有者窗体最小化和关闭。例如,如果 Form2 归窗体 Form1 所有,则关闭或最小化 Form1 时,也会关闭或最小化 Form2。并且附属窗体从不显示在其所有者窗体后面。可以将附属窗体用于查找和替换窗口之类的窗口,当选定所有者窗体时,这些窗口不应消失。若要确定某父窗体拥有的窗体,请使用OwnedForms属性。上面是SDK帮助文档上讲的,下面我们就来使用它。首先还是使用第一篇文章中的第二个例子,窗体如下:说明:在这个例子中我们的两个窗体都加了

31、一个ListBox用来显示ArrayList中的内容。主窗体中控件:listBoxFrm1,buttonEdit;子窗体中控件:listBoxFrm2,textBoxAdd,buttonAdd,buttonEdit,buttonOK。主窗体中还是定义类数据成员,private ArrayList listData1;在构造函数里实例化它,填充数据,最后绑定到listBoxFrm1。构造函数如下:public Form1() InitializeComponent(); this.listData1 = new ArrayList(); this.listData1.Add(DotNet); t

32、his.listData1.Add(C#); this.listData1.Add(A); this.listData1.Add(WebService); this.listData1.Add(XML); this.listBoxFrm1.DataSource = this.listData1;主窗体的修改按钮处理函数:private void buttonEdit_Click(object sender, System.EventArgs e) Form2 formChild = new Form2(); formChild.Owner = this; formChild.ShowDialo

33、g(); this.listBoxFrm1.DataSource = null; this.listBoxFrm1.DataSource = this.listData1;我们设置了formChild.Owner为this,这样,子窗体和主窗体就有联系了,当然我们也可以改成如下:private void buttonEdit_Click(object sender, System.EventArgs e) Form2 formChild = new Form2(); formChild.ShowDialog(this); this.listBoxFrm1.DataSource = null;

34、this.listBoxFrm1.DataSource = this.listData1;不过这样还不行,目前主窗体的listData1变量外部访问不到,private ArrayList listData1;必须修改为public访问修饰符,public ArrayList listData1;也可以通过属性(property)来实现,public ArrayList ListData1 getreturn this.listData1;这里我采用属性,感觉语法更灵活,清楚。下面是对Form2的修改,构造函数又恢复原貌了。public Form2() InitializeComponent(

35、);另外又新增了一个窗体的Load事件,在它的事件处理函数中来获取主窗体中的数据,private void Form2_Load(object sender, System.EventArgs e) Form1 pareForm = (Form1)this.Owner; this.listData2 = pareForm.ListData1; foreach(object o in this.listData2) this.listBoxFrm2.Items.Add(o);有人会问,为什么不把上面的代码放到构造函数里面去呢?如下不是更好,public Form2() InitializeCom

36、ponent(); Form1 pareForm = (Form1)this.Owner; this.listData2 = pareForm.ListData1; foreach(object o in this.listData2) this.listBoxFrm2.Items.Add(o);那我会对你说错了,因为在主窗体修改按钮被点击后,开始执行Form2 formChild = new Form2();而在Form2的实例化过程中会在构造函数中执行Form1 pareForm = (Form1)this.Owner;而这时的this.Owner是没有值的,为空引用,那么下面的代码肯定也

37、出问题,this.listData2 = pareForm.ListData1;foreach(object o in this.listData2) this.listBoxFrm2.Items.Add(o);当整个Form2实例化完成后,才会执行formChild.Owner = this;这条代码,所以使用了Form2_Load事件。那怎样可以不使用Form2_Load事件呢?等下面我们来修改代码实现它。下面的子窗体代码没有变化,private void buttonAdd_Click(object sender, System.EventArgs e) if(this.textBoxA

38、dd.Text.Trim().Length0) this.listData2.Add(this.textBoxAdd.Text.Trim(); this.listBoxFrm2.Items.Add(this.textBoxAdd.Text.Trim(); else MessageBox.Show(请输入添加的内容!);private void buttonDel_Click(object sender, System.EventArgs e) int index = this.listBoxFrm2.SelectedIndex; if(index!=-1) this.listData2.Rem

39、oveAt(index); this.listBoxFrm2.Items.RemoveAt(index); else MessageBox.Show(请选择删除项!); private void buttonOK_Click(object sender, System.EventArgs e) this.Close();好了,结果同第一篇中的一样,子窗体能修改主窗体的值。2使用自定义属性或方法下面我们来讲讲怎样使用自定义属性或方法来完成数据修改功能而不使用Form2_Load事件。主窗体的修改按钮点击处理函数如下:private void buttonEdit_Click(object sen

40、der, System.EventArgs e) Form2 formChild = new Form2(); formChild.ListData2 = this.listData1; formChild.ShowDialog(); this.listBoxFrm1.DataSource = null; this.listBoxFrm1.DataSource = this.listData1;并且我们去掉了主窗体的ListData1属性,/public ArrayList ListData1/getreturn this.listData1;/而在子窗体中加上ListData2属性,publ

41、ic ArrayList ListData2 set this.listData2 = value; foreach(object o in this.listData2) this.listBoxFrm2.Items.Add(o); 也可以把属性改成方法,public void SetListData(ArrayList listData) this.listData2 = listData; foreach(object o in this.listData2) this.listBoxFrm2.Items.Add(o);而在主窗体的修改按钮处理函数中也要相应改动:formChild.Li

42、stData2 = this.listData1;改为formChild.SetListData(this.listData1); 总结,我们通过Form类的Owner属性来建立主从窗体间的桥梁,这个是不是类似于把主窗体作为子窗体的构造函数参数传入实现的功能差不多;另外又采用了属性和方法来完成数据的交互,我觉得这种实现方法很实用,特别是用在不需要实例化类或着已经有了实例的情况下传递数据。下一节我们来讲如何使用静态类来完成数据的交互。三使用静态类 在第一节和第二节中我们使用带参数的构造函数、属性以及方法实现了数据的交互,接下来要讲的是使用静态类来完成窗体间的数据交互。这个也是我们经常要用到的一种

43、数据交互方法。下面是定义的一个类:using System;using System.Collections;namespace ZZ public class AppDatas /静态数据成员 private static ArrayList listData; /静态构造函数 static AppDatas() listData = new ArrayList(); listData.Add(DotNet); listData.Add(C#); listData.Add(A); listData.Add(WebService); listData.Add(XML); /静态属性 publi

44、c static ArrayList ListData getreturn listData; /静态方法 public static ArrayList GetListData() return listData; 上面包含了一个静态类成员,listData,一个静态构造函数static AppDatas(),用来初始化listData的数据。还有一个静态属性ListData和一个静态GetListData()方法,他们实现了同样的功能就是返回listData。另外两个窗体没有改变布局可参看前两篇文章还是如下:由于前面两篇文章已经讲了很多,这个不细说了,下面是完整的代码:Form1.cs文件

45、using System;using System.Drawing;using System.Collections;using System.ComponentModel;using System.Windows.Forms;namespace ZZ public class Form1 : System.Windows.Forms.Form private System.Windows.Forms.Button buttonEdit; private System.Windows.Forms.ListBox listBoxFrm1; private System.ComponentMode

46、l.Container components = null; public Form1() InitializeComponent(); this.listBoxFrm1.DataSource = AppDatas.ListData; protected override void Dispose( bool disposing ) if( disposing ) if(components != null) components.Dispose(); base.Dispose( disposing ); STAThread static void Main() Application.Run

47、(new Form1(); #region Windows 窗体设计器生成的代码 private void InitializeComponent() this.buttonEdit = new System.Windows.Forms.Button(); this.listBoxFrm1 = new System.Windows.Forms.ListBox(); this.SuspendLayout(); / / buttonEdit / this.buttonEdit.Location = new System.Drawing.Point(128, 108); this.buttonEdit.Name = butt

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论