C# - Creating controls dynamically and accessing them

爷,独闯天下 提交于 2019-12-11 11:57:34

问题


I'm creating a simple notepad type of application with the tab functionality. I'm creating the TabControl, its TabPages and RichTextBoxes at run-time. I have them instantiated at class scope. And there is a MenuStrip item called New, by clicking that you can add more tab pages.

TabControl tbcEditor = new TabControl();
TabPage tbPage = new TabPage();
RichTextBox rtb = new RichTextBox();

private void frmTextEditor_Load(object sender, EventArgs e)
{
     Controls.Add(tbcEditor);
     tbcEditor.Dock = DockStyle.Fill;
     tbcEditor.TabPages.Add(tbPage);
     tbPage.Controls.Add(rtb);
     rtb.Dock = DockStyle.Fill;
}

private void newToolStripMenuItem_Click(object sender, EventArgs e)
{
     //TabPage tbPage = new TabPage();
     //RichTextBox rtb = new RichTextBox();

     tbPage.Controls.Add(rtb);
     rtb.Dock = DockStyle.Fill;
     tbcEditor.TabPages.Add(tbPage);
}

The problem I'm facing is a bit difficult to explain. I'll try my best. When the form loads, everything works as expected. The TabControl creates with a TabPage with a RichTextBox added. However if I click that New button to add another page, it goes bonkers. A new TabPage gets created but without a RichTextBox added. No errors are thrown either. If I un-comment out those 2 lines(under MenuItem click event), which creates 2 instances of TabPage and RichTextBox, everything works as I want.

Now my first question is why do I have to make new instances of only those 2 types(TabPage, RichTextBox) again but not TabControl? As you can see in the last line, I can use tbcEditor once again. But not tbPage and rtb.

Sure I can go on declaring them again at local scope but another issue arises then. If I want to say, add copy, paste functionality, I should do something like this,right?

Clipboard.SetDataObject(rtb.SelectedText);

But I can't access rtb since it is declared as local.

I'm very baffled by this so any suggestions, ideas on how to overcome these 2 issues would be greatly appreciated.

Thank you.


回答1:


If I un-comment out those 2 lines(under MenuItem click event), which creates 2 instances of TabPage and RichTextBox, everything works as I want.

When you uncomment those lines, you are adding the same instance of the rich textbox and tab page to the container panel again which is meaningless. Instead add new controls foreach tabpage. (I hope thats the requirement)

Now my first question is why do I have to make new instances of only those 2 types(TabPage, RichTextBox) again but not TabControl?

TabControl is the parent control which has TabPages as child controls. You can have multiple tabs under one TabControl. So you need not create TabControls other than the tbcEditor you have already added. We do not add container controls more than once (unless its the requirement). Do we need more forms? No, just one form which can hold all the child controls right. Similarly just one TabControl which can hold a collection of TabPages. You would need more TabControls only if you want sub-tabs foreach new tab which I guess is not the requirement..

But I can't access rtb since it is declared as local.

This is no big deal. You can do in two ways:

1) Search for your appropriate control by looping. The SelectedTab property gives what you want.

 private void someEvent(object sender, EventArgs e)
 {           
        foreach (Control c in tbcEditor.SelectedTab.Controls)
        {
            if (c is RichTextBox)
            {
                Clipboard.SetDataObject(((RichTextBox)c).SelectedText);
                break; //assuming u have just one main rtb there
            }
        }
 }

2) Tag each rtb to the tabPage when you create it, and then you can get the tag element of the selected tab page to get the rich text box. I would go for this approach.

Edit: (In general pls make the following changes too to your code):

 TabControl tbcEditor = new TabControl();

 private void frmTextEditor_Load(object sender, EventArgs e)
 {
      Controls.Add(tbcEditor);
      tbcEditor.Dock = DockStyle.Fill;
      AddMyControlsOnNewTab();
 }

private void AddMyControlsOnNewTab()
{
    TabPage tbPage = new TabPage();
    RichTextBox rtb = new RichTextBox();
    tbPage.Tag = rtb; //just one extra bit of line.

    tbcEditor.TabPages.Add(tbPage);
    tbPage.Controls.Add(rtb);
    rtb.Dock = DockStyle.Fill;
}

private void newToolStripMenuItem_Click(object sender, EventArgs e)
{
    AddMyControlsOnNewTab();
}

Now, you can call it like this:

 private void someEvent(object sender, EventArgs e)
 {           
        RichTextBox rtb= (RichTextBox)tbcEditor.SelectedTab.Tag;
        Clipboard.SetDataObject(rtb.SelectedText);
        //or even better in just a line,
        //Clipboard.SetDataObject(((RichTextBox)tbcEditor.SelectedTab.Tag).SelectedText);
 }

What you have to consider here is which is the control that you first get and which is the one you do not get. You would get TabPage anyways but not the RichTextBox. So you have to tag RichTextBox to TabPage. You have to cast it since Tag is of type object, so you have to specify which kind of object it is. Finally, this method has the advantage that you need not loop through a list, so its more performant. And that you can have more RichTextBoxes in the TabPage (provided you want to copy text from only one set of RichTextBoxes, one from each TabPage)..




回答2:


The commented lines are doing just what they are suppposed to do. The code does not associate the Richtextbox with the Tabpage .

TabPage tbPage = new TabPage(); // Creates a new tabpage

RichTextBox rtb = new RichTextBox(); // Creates a new RichtextBox control.

TabControl is a container , so one instance is just fine.

Also see this - http://sujay-ghosh.blogspot.in/2009/03/addingremoving-dynamically-created.html, nothing with do with tabcontrols, but how to create controls on the fly.

Hope this helps .




回答3:


The code

tbPage.Controls.Add(rtb); 
rtb.Dock = DockStyle.Fill; 
tbcEditor.TabPages.Add(tbPage); 

Takes your exisitng textbox, adds it to the exisitng tab page, then adds that existing tab page to the editor. Since this has already been done, nothing happens.

When you add those two lines, you create new instances of the text box and a new tab page, which is exactly what you want. Your latter problem comes, because the newly declared variable rtb hides the one declared in the class -- in a different method you can only access the onde declared in the class (barring getting the control out of the tab)

To get around not being able to access the proper text box, you can maintain them in a list(*) (or some other collection) and refer to the one associated with the currently active tab. For this, you will have to create an event listener to see which tab is activated currectly.

(*) as opposed to having only one




回答4:


OK you need to create fresh instances of the RichTextBox rathere than trying to add the same instance to each tab.

 TabControl tbcEditor = new TabControl();
 //Get rid off this line --- TabPage tbPage = new TabPage();
 //Get rid off this line --- RichTextBox rtb = new RichTextBox();

 List<TabPage> _tabs = new  List<TabPage>();
 List<RichTextBox> _tbx = new  List<RichTextBox>();

 private void frmTextEditor_Load(object sender, EventArgs e) 
 {
      Controls.Add(tbcEditor);
      tbcEditor.Dock = DockStyle.Fill;

       AddNewTab();
 }

 private void newToolStripMenuItem_Click(object sender, EventArgs e) 
 {   
       AddNewTab();
 } 

 private void AddNewTab()
 {
    //TabPage 
        var tbPage = new TabPage();
        _tabs.Add(tbPage);

        //RichTextBox 
        var rtb = new RichTextBox();
        _tbx.Add(rtb);

        tbPage.Controls.Add(rtb);
        rtb.Dock = DockStyle.Fill;
        tbcEditor.TabPages.Add(tbPage); 
 }

This simply add both the tab and the rtb to a collection which can be accessed by index (can also use Dictionary for named access etc). There are other ways of course, including just nameing the components and looping through for them when required etc.



来源:https://stackoverflow.com/questions/9991013/c-sharp-creating-controls-dynamically-and-accessing-them

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!