Размытие фона с помощью Viewbox

Я создаю подобный эффект, который дает вам стекло AERO, размытое окно. Я быстро понял, что это нелегко в WPF, но мне удалось заставить его работать почти полностью. Я просто застрял, когда задействован Viewbox.

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

<Grid x:Name="grid">
    <Image x:Name="image" Source="someImage.png"/>
    <Rectangle x:Name="blurBackgroundRect" Width="100" Height="100">
        <Rectangle.Effect>
            <BlurEffect Radius="10"/>
        </Rectangle.Effect>
        <Rectangle.Fill>
            <VisualBrush
                ViewboxUnits="Absolute"
                AlignmentX="Center"
                AlignmentY="Center"
                Visual="{Binding ElementName=image}"
                Stretch="None">
                <VisualBrush.Viewbox>
                    <MultiBinding Converter="{StaticResource visualBrushTargetConverter}">
                        <Binding ElementName="grid"/>
                        <Binding ElementName="blurBackgroundRect"/>
                        <Binding ElementName="grid" Path="ActualWidth"/>
                        <Binding ElementName="grid" Path="ActualHeight"/>
                    </MultiBinding>
                </VisualBrush.Viewbox>
            </VisualBrush>
        </Rectangle.Fill>
    </Rectangle>
</Grid>

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

Последняя часть - это многозначный конвертер, который выполняет фактические вычисления для VisualBrush Viewbox.

public class VisualBrushTargetConverter : IMultiValueConverter
{
    public object Convert(object[] values, Type targetType, object parameter, CultureInfo culture)
    {
        var parentControl = values[0] as FrameworkElement;
        var targetControl = values[1] as FrameworkElement;

        var transformedPos = targetControl.TransformToVisual(parentControl).Transform(new Point());
        var transformedSize = targetControl.TransformToVisual(parentControl).Transform(new Point(targetControl.RenderSize.Width, targetControl.RenderSize.Height));

        transformedSize = new Point(transformedSize.X - transformedPos.X, transformedSize.Y - transformedPos.Y);
        return new Rect(transformedPos.X,
                        transformedPos.Y,
                        transformedSize.X,
                        transformedSize.Y);
    }
}

Так что это все необходимое, чтобы воспроизвести мою следующую проблему. Если вы хотите протестировать код, вы можете легко поместить его в простое приложение wpf.

Если вы проверите это, вы увидите, что он работает нормально. Проблема возникает сейчас, если мы поместим прямоугольник в Viewbox и сетку с фиксированным размером (что подражает моей реальной проблеме).

<Image x:Name="image" .../>
<Viewbox>
    <Grid Width="800" Height="600">
        <Rectangle x:Name="blurBackgroundRect" ...>
    </Grid>
</Viewbox>

Поэтому я предполагаю, что масштабирование из окна просмотра применяется после компоновки, поэтому не будет распознаваться в моих вызовах TransformTo. Но я уже пытался использовать фактическое масштабирование Viewbox в моем конверте, но безуспешно. Пока есть Viewbox, я не могу заставить его работать, поэтому я надеюсь, что у кого-нибудь есть идея, что может быть не так, или, может быть, даже укажет мне на гораздо более простое решение. Я хочу, чтобы этот код впоследствии преобразовывался в пользовательский элемент управления, чтобы его можно было легко использовать повторно, поэтому я предпочитаю способы, которые не зависят от особых условий.

1 ответ

Решение

Мне удалось исправить странное поведение, изменив Stretch в VisualBrush на Fill вместо None.

Другие вопросы по тегам