Получить элемент управления Windows Forms по имени в C # - PullRequest
54 голосов
/ 08 октября 2009

У меня есть ToolStripMenuItem под названием «myMenu». Как я могу получить доступ к этому так:

/* Normally, I would do: */
this.myMenu... etc.

/* But how do I access it like this: */
String name = myMenu;
this.name...

Это потому, что я динамически генерирую ToolStripMenuItems из XML-файла и мне нужно ссылаться на элементы меню по их динамически генерируемым именам.

Ответы [ 13 ]

107 голосов
/ 08 октября 2009

Используйте метод Control.ControlCollection.Find .

Попробуйте это:

this.Controls.Find()
34 голосов
/ 28 октября 2009
string name = "the_name_you_know";

Control ctn = this.Controls[name];

ctn.Text = "Example...";
9 голосов
/ 08 октября 2009
Control GetControlByName(string Name)
{
    foreach(Control c in this.Controls)
        if(c.Name == Name)
            return c;

    return null;
}

Не обращая внимания, я заново изобретаю колеса.

6 голосов
/ 11 мая 2011

Предполагая, что у вас есть объект menuStrip и меню имеет только один уровень глубины, используйте:

ToolStripMenuItem item = menuStrip.Items
    .OfType<ToolStripMenuItem>()
    .SelectMany(it => it.DropDownItems.OfType<ToolStripMenuItem>())
    .SingleOrDefault(n => n.Name == "MyMenu");

Для более глубоких уровней меню добавьте больше операторов SelectMany в оператор.

если вы хотите найти все пункты меню в полосе, используйте

ToolStripMenuItem item = menuStrip.Items
    .Find("MyMenu",true)
    .OfType<ToolStripMenuItem>()
    .Single();

Однако убедитесь, что у каждого меню есть свое имя, чтобы исключение не вызывалось дубликатами клавиш.

Чтобы избежать исключений, вы можете использовать FirstOrDefault вместо SingleOrDefault / Single или просто вернуть последовательность, если у вас могут быть Name дубликаты.

4 голосов
/ 13 июля 2010

this.Controls.Find (name, searchAllChildren) не находит ToolStripItem, поскольку ToolStripItem не является элементом управления

  using SWF = System.Windows.Forms;
  using NUF = NUnit.Framework;
  namespace workshop.findControlTest {
     [NUF.TestFixture]
     public class FormTest {
        [NUF.Test]public void Find_menu() {
           // == prepare ==
           var fileTool = new SWF.ToolStripMenuItem();
           fileTool.Name = "fileTool";
           fileTool.Text = "File";

           var menuStrip = new SWF.MenuStrip();
           menuStrip.Items.Add(fileTool);

           var form = new SWF.Form();
           form.Controls.Add(menuStrip);

           // == execute ==
           var ctrl = form.Controls.Find("fileTool", true);

           // == not found! ==
           NUF.Assert.That(ctrl.Length, NUF.Is.EqualTo(0)); 
        }
     }
  }
3 голосов
/ 29 ноября 2017

Один из лучших способов - это одна строка кода, подобная этой:

В этом примере мы ищем все PictureBox по имени в форме

PictureBox[] picSample = 
                    (PictureBox)this.Controls.Find(PIC_SAMPLE_NAME, true);

Наиболее важным является второй параметр find.

если вы уверены, что имя элемента управления существует, вы можете использовать его напрямую:

  PictureBox picSample = 
                        (PictureBox)this.Controls.Find(PIC_SAMPLE_NAME, true)[0];
3 голосов
/ 23 октября 2016

Используя тот же подход Филип Уоллес , мы можем сделать так:

    public Control GetControlByName(Control ParentCntl, string NameToSearch)
    {
        if (ParentCntl.Name == NameToSearch)
            return ParentCntl;

        foreach (Control ChildCntl in ParentCntl.Controls)
        {
            Control ResultCntl = GetControlByName(ChildCntl, NameToSearch);
            if (ResultCntl != null)
                return ResultCntl;
        }
        return null;
    }

Пример:

    public void doSomething() 
    {
            TextBox myTextBox = (TextBox) this.GetControlByName(this, "mytextboxname");
            myTextBox.Text = "Hello!";
    }

Надеюсь, это поможет! :)

3 голосов
/ 13 сентября 2011

Предположим, у вас есть Windows.Form Form1 в качестве родительской формы, которой принадлежит созданное вами меню. Один из атрибутов формы называется .Menu. Если меню было создано программно, оно должно быть таким же, и оно будет распознано как меню и помещено в атрибут меню формы.

В этом случае у меня было главное меню под названием File. Подменю, называемое MenuItem в File, содержало тег Open и называлось menu_File_Open. Следующее сработало. Предполагая, что вы

// So you don't have to fully reference the objects.
using System.Windows.Forms;

// More stuff before the real code line, but irrelevant to this discussion.

MenuItem my_menuItem = (MenuItem)Form1.Menu.MenuItems["menu_File_Open"];

// Now you can do what you like with my_menuItem;
3 голосов
/ 28 октября 2009
this.Controls["name"];

Это фактический код, который запускается:

public virtual Control this[string key]
{
    get
    {
        if (!string.IsNullOrEmpty(key))
        {
            int index = this.IndexOfKey(key);
            if (this.IsValidIndex(index))
            {
                return this[index];
            }
        }
        return null;
    }
}

против

public Control[] Find(string key, bool searchAllChildren)
{
    if (string.IsNullOrEmpty(key))
    {
        throw new ArgumentNullException("key", SR.GetString("FindKeyMayNotBeEmptyOrNull"));
    }
    ArrayList list = this.FindInternal(key, searchAllChildren, this, new ArrayList());
    Control[] array = new Control[list.Count];
    list.CopyTo(array, 0);
    return array;
}

private ArrayList FindInternal(string key, bool searchAllChildren, Control.ControlCollection controlsToLookIn, ArrayList foundControls)
{
    if ((controlsToLookIn == null) || (foundControls == null))
    {
        return null;
    }
    try
    {
        for (int i = 0; i < controlsToLookIn.Count; i++)
        {
            if ((controlsToLookIn[i] != null) && WindowsFormsUtils.SafeCompareStrings(controlsToLookIn[i].Name, key, true))
            {
                foundControls.Add(controlsToLookIn[i]);
            }
        }
        if (!searchAllChildren)
        {
            return foundControls;
        }
        for (int j = 0; j < controlsToLookIn.Count; j++)
        {
            if (((controlsToLookIn[j] != null) && (controlsToLookIn[j].Controls != null)) && (controlsToLookIn[j].Controls.Count > 0))
            {
                foundControls = this.FindInternal(key, searchAllChildren, controlsToLookIn[j].Controls, foundControls);
            }
        }
    }
    catch (Exception exception)
    {
        if (ClientUtils.IsSecurityOrCriticalException(exception))
        {
            throw;
        }
    }
    return foundControls;
}
3 голосов
/ 08 октября 2009

Поскольку вы генерируете их динамически, сохраняйте карту между строкой и пунктом меню, что позволит осуществлять быстрый поиск.

// in class scope
private readonly Dictionary<string, ToolStripMenuItem> _menuItemsByName = new Dictionary<string, ToolStripMenuItem>();

// in your method creating items
ToolStripMenuItem createdItem = ...
_menuItemsByName.Add("<name here>", createdItem);

// to access it
ToolStripMenuItem menuItem = _menuItemsByName["<name here>"];
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...