Preciso pesquisar na hierarquia de controle do WPF controles que correspondam a um determinado nome ou tipo. Como posso fazer isso?
Combinei o formato do modelo usado pelo algoritmo de John Myczek e Tri Q acima para criar um algoritmo findChild que pode ser usado em qualquer pai. Lembre-se de que pesquisar recursivamente uma árvore para baixo pode ser um processo demorado. Eu só verifiquei isso em um aplicativo WPF, por favor, comente sobre quaisquer erros que você possa encontrar e eu corrigirei meu código.
O WPF Snoop é uma ferramenta útil para examinar a árvore visual - eu recomendo fortemente usá-lo durante o teste ou usando esse algoritmo para verificar seu trabalho.
Há um pequeno erro no algoritmo do Tri Q. Depois que o filho for encontrado, se childrenCount for> 1 e se iterarmos novamente, podemos substituir o filho encontrado corretamente. Portanto, adicionei um if (foundChild != null) break;
no meu código para lidar com essa condição.
/// <summary>
/// Finds a Child of a given item in the visual tree.
/// </summary>
/// <param name="parent">A direct parent of the queried item.</param>
/// <typeparam name="T">The type of the queried item.</typeparam>
/// <param name="childName">x:Name or Name of child. </param>
/// <returns>The first parent item that matches the submitted type parameter.
/// If not matching item can be found,
/// a null parent is being returned.</returns>
public static T FindChild<T>(DependencyObject parent, string childName)
where T : DependencyObject
{
// Confirm parent and childName are valid.
if (parent == null) return null;
T foundChild = null;
int childrenCount = VisualTreeHelper.GetChildrenCount(parent);
for (int i = 0; i < childrenCount; i++)
{
var child = VisualTreeHelper.GetChild(parent, i);
// If the child is not of the request child type child
T childType = child as T;
if (childType == null)
{
// recursively drill down the tree
foundChild = FindChild<T>(child, childName);
// If the child is found, break so we do not overwrite the found child.
if (foundChild != null) break;
}
else if (!string.IsNullOrEmpty(childName))
{
var frameworkElement = child as FrameworkElement;
// If the child's name is set for search
if (frameworkElement != null && frameworkElement.Name == childName)
{
// if the child's name is of the request name
foundChild = (T)child;
break;
}
}
else
{
// child element found.
foundChild = (T)child;
break;
}
}
return foundChild;
}
Chame assim:
TextBox foundTextBox =
UIHelper.FindChild<TextBox>(Application.Current.MainWindow, "myTextBoxName");
Nota Application.Current.MainWindow
pode ser qualquer janela pai.
FrameworkElement
como T, ele retornará nulo assim que o primeiro loop terminar. então você precisará fazer algumas modificações.Você também pode encontrar um elemento pelo nome usando FrameworkElement.FindName (string) .
Dado:
No arquivo code-behind, você pode escrever:
Obviamente, como é definido usando x: Name, você pode apenas fazer referência ao campo gerado, mas talvez queira pesquisá-lo dinamicamente e não estaticamente.
Essa abordagem também está disponível para modelos, nos quais o item nomeado aparece várias vezes (uma vez por uso do modelo).
fonte
Você pode usar o VisualTreeHelper para encontrar controles. Abaixo está um método que usa o VisualTreeHelper para localizar um controle pai de um tipo especificado. Você pode usar o VisualTreeHelper para encontrar controles de outras maneiras também.
Chame assim:
fonte
Talvez eu esteja repetindo todos os outros, mas tenho um belo pedaço de código que estende a classe DependencyObject com um método FindChild () que fornecerá o filho por tipo e nome. Basta incluir e usar.
Espero que você ache útil.
fonte
Minhas extensões para o código.
Fonte: https://code.google.com/p/gishu-util/source/browse/#git%2FWPF%2FUtilities
Postagem explicativa do blog: http://madcoderspeak.blogspot.com/2010/04/wpf-find-child-control-of-specific-type.html
fonte
Se você deseja encontrar TODOS os controles de um tipo específico, também pode estar interessado neste snippet
fonte
child
uma segunda vez? Se você échildType
do tipoT
, pode escrever dentro deif
:yield return childType
... não?Isso descartará alguns elementos - você deve estendê-lo assim para oferecer suporte a uma variedade maior de controles. Para uma breve discussão, dê uma olhada aqui
fonte
Try*
método para retornarbool
e ter umout
parâmetro que retorna o tipo em questão, como acontece com:bool IDictionary.TryGetValue(TKey key, out TValue value)
FindParent
. Esse nome para mim implica que ele poderia retornarnull
. OTry*
prefixo é usado em todo o BCL da maneira que descrevi acima. Observe também que a maioria das outras respostas aqui usa aFind*
convenção de nomenclatura. É apenas um ponto menor :)Editei o código do CrimsonX, pois não estava funcionando com os tipos de superclasse:
fonte
DependencyObject
que não éFrameworkElement
, poderá lançar uma exceção. Também o usoGetChildrenCount
em todas as iterações dofor
loop parece uma má ideia.Embora eu ame a recursão em geral, ela não é tão eficiente quanto a iteração na programação em C #, então talvez a solução a seguir seja mais clara que a sugerida por John Myczek? Isso pesquisa uma hierarquia de um determinado controle para encontrar um controle ancestral de um tipo específico.
Chame assim para encontrar o
Window
controle que contémExampleTextBox
:fonte
Aqui está o meu código para encontrar controles por Tipo enquanto controlamos a profundidade da hierarquia (maxDepth == 0 significa infinitamente profundo).
fonte
exciton80 ... Eu estava tendo um problema com o código não recorrente através dos controles do usuário. Ele estava atingindo a raiz do Grid e gerando um erro. Eu acredito que isso corrige para mim:
fonte
Eu tenho uma função de sequência como esta (que é completamente geral):
Obtendo filhos imediatos:
Encontrando todas as crianças na árvore hiarárquica:
Você pode chamar isso na janela para obter todos os controles.
Depois de ter a coleção, você pode usar o LINQ (por exemplo, OfType, Where).
fonte
Como a pergunta é geral o suficiente para atrair pessoas que procuram respostas para casos muito triviais: se você quer apenas um filho em vez de um descendente, pode usar o Linq:
ou, é claro, o óbvio para o loop iterando sobre Children.
fonte
Essas opções já falam sobre atravessar a Árvore Visual em C #. É possível atravessar a árvore visual no xaml, também usando a extensão de marcação RelativeSource. msdn
encontre por tipo
fonte
Aqui está uma solução que usa um predicado flexível:
Você pode, por exemplo, chamá-lo assim:
fonte
Este código apenas corrige o erro da resposta @CrimsonX:
Você só precisa continuar chamando o método recursivamente se os tipos forem correspondentes, mas os nomes não (isso acontece quando você passa
FrameworkElement
comoT
). caso contrário, ele voltaránull
e isso está errado.fonte
Para encontrar um ancestral de um determinado tipo a partir do código, você pode usar:
Esta implementação usa iteração em vez de recursão, que pode ser um pouco mais rápida.
Se você estiver usando o C # 7, isso poderá ser um pouco menor:
fonte
Tente isto
Código por trás
fonte