Я думаю, вам нужно вызвать BringIntoView
для контейнера элемента, а не для самого ItemsControl:
var container = DocumentElements.ItemContainerGenerator.ContainerFromItem(model) as FrameworkElement;
if (container != null)
container.BringIntoView();
РЕДАКТИРОВАТЬ: на самом деле это не работает, потому что на данный момент контейнер элемента еще не был сгенерирован ... Возможно, вы можете обработать событие StatusChanged
ItemContainerGenerator
. Я попробовал следующий код:
public static class ItemsControlExtensions
{
public static void BringItemIntoView(this ItemsControl itemsControl, object item)
{
var generator = itemsControl.ItemContainerGenerator;
if (!TryBringContainerIntoView(generator, item))
{
EventHandler handler = null;
handler = (sender, e) =>
{
switch (generator.Status)
{
case GeneratorStatus.ContainersGenerated:
TryBringContainerIntoView(generator, item);
break;
case GeneratorStatus.Error:
generator.StatusChanged -= handler;
break;
case GeneratorStatus.GeneratingContainers:
return;
case GeneratorStatus.NotStarted:
return;
default:
break;
}
};
generator.StatusChanged += handler;
}
}
private static bool TryBringContainerIntoView(ItemContainerGenerator generator, object item)
{
var container = generator.ContainerFromItem(item) as FrameworkElement;
if (container != null)
{
container.BringIntoView();
return true;
}
return false;
}
}
Однако это тоже не работает ... по какой-то причине ContainerFromItem
по-прежнему возвращает ноль после изменения статуса на ContainersGenerated
, и я понятия не имею, почему: S
РЕДАКТИРОВАТЬ: ОК, теперь я понимаю ... это было из-за виртуализации: контейнеры создаются только тогда, когда они должны быть отображены, поэтому контейнеры для скрытых элементов не создаются. Если вы отключите виртуализацию для ItemsControl (VirtualizingStackPanel.IsVirtualizing="False"
), приведенное выше решение будет работать нормально.