Podemos escribir un disparador para usarlo con elementos contenidos en un ScrollViewer
. Aquí es una aplicación de ejemplo completo:
<Grid>
<ScrollViewer VerticalAlignment="Top" Height="200">
<StackPanel HorizontalAlignment="Left">
<Button Name="button" Content="Open">
<i:Interaction.Triggers>
<i:EventTrigger EventName="Click">
<ei:ChangePropertyAction TargetObject="{Binding ElementName=popup}" PropertyName="IsOpen" Value="True"/>
</i:EventTrigger>
<local:ScrollTrigger>
<ei:ChangePropertyAction TargetObject="{Binding ElementName=popup}" PropertyName="IsOpen" Value="False"/>
</local:ScrollTrigger>
</i:Interaction.Triggers>
</Button>
<Popup Name="popup" PlacementTarget="{Binding ElementName=button}">
<TextBlock Background="White" Text="Sample text"/>
</Popup>
<Rectangle Width="100" Height="100" Fill="Red"/>
<Rectangle Width="100" Height="100" Fill="Green"/>
<Rectangle Width="100" Height="100" Fill="Blue"/>
<Rectangle Width="100" Height="100" Fill="Yellow"/>
</StackPanel>
</ScrollViewer>
</Grid>
Tenemos un botón que abre un Popup
y cualquier desplazamiento en cualquier padre ScrollViewer
hace que las acciones ScrollTrigger
al fuego y entonces podemos cerrar la ventana emergente. Tenga en cuenta que el activador está conectado al Button
y no al Popup
. Podemos usar cualquier elemento cercano que esté en el árbol visual. También tenga en cuenta que usamos otro disparador para abrir el Popup
, pero cómo se abre no es importante para la pregunta original.
Aquí es el ScrollTrigger
:
class ScrollTrigger : TriggerBase<FrameworkElement>
{
protected override void OnAttached()
{
AssociatedObject.Loaded += new RoutedEventHandler(AssociatedObject_Loaded);
}
void AssociatedObject_Loaded(object sender, RoutedEventArgs e)
{
foreach (var scrollViewer in GetScrollViewers())
scrollViewer.ScrollChanged += new ScrollChangedEventHandler(scrollViewer_ScrollChanged);
}
void scrollViewer_ScrollChanged(object sender, ScrollChangedEventArgs e)
{
InvokeActions(e.OriginalSource);
}
IEnumerable<ScrollViewer> GetScrollViewers()
{
for (DependencyObject element = AssociatedObject; element != null; element = VisualTreeHelper.GetParent(element))
if (element is ScrollViewer) yield return element as ScrollViewer;
}
}
El ScrollTrigger
es muy simple, sólo se concede a todos los eventos de los padres ScrollChanged
y dispara cualquier acción que contenían. En la muestra usamos el ChangePropertyAction
para cerrar el Popup
.
Si no está familiarizado con los comportamientos, instale el Expression Blend 4 SDK y agregar estos espacios de nombres:
xmlns:i="http://schemas.microsoft.com/expression/2010/interactivity"
xmlns:ei="http://schemas.microsoft.com/expression/2010/interactions"
y añadir System.Windows.Interactivity
y Microsoft.Expression.Interactions
a su proyecto.
Tengo la misma pregunta y problema. ¡Desplazo mi cuadrícula y mi ventana emergente con operaciones personalizadas permanece en el mismo lugar! Necesito desplazar ventana emergente con grilla! – Evgeny