Можно ли заставить окно никогда не иметь размер меньше желаемого размера для его дочерних элементов?

Я знаю, что для Window (фактически любого FrameworkElement) можно указать MinWidth и MinHeight. Мне интересно, как я могу сделать так, чтобы этот минимум соответствовал DesiredSize окна?

Другими словами, я хочу сказать "Вы можете изменить размер этого окна настолько, насколько хотите, но не меньше, чем минимум, необходимый для отображения его содержимого".

Например, скажем, у меня есть простой Window, содержащий DockPanel с тремя элементами, расположенными вертикально. Для простоты мы будем говорить только о высоте.

  1. Первый — это Rectangle с Height из 20, он закреплен сверху, так что его DesiredSize.Height равно 20.

  2. Второй — Label, который имеет MinHeight из 20 и закреплен внизу, но может увеличиваться в высоту, если есть несколько строк текста. (В настоящее время есть только одно слово, поэтому DesiredSize.Height также равно 20)

  3. Третий элемент — это настраиваемый элемент управления ImageViewer, который заполняет оставшуюся область в середине, но никогда не будет меньше своего содержимого плюс отступы. В настоящее время изображение имеет размер 15x15, а отступ — 2,5 на стороне, поэтому его DesiredSize.Height равно 20 (2,5 + 15 + 2,5).

Таким образом, с учетом вышеизложенного DesiredSize.Height для StackPanel будет равно 60 (20 + 20 + 20), а DesiredSize.Height для Window будет равно 60 + высота его собственного хрома (скажем так, это тоже 20), и это кажется дайте мне Window.DesiredSize.Height из 80. Это то, что мне нужно для MinHeight окна.

Я пытался привязать MinHeight Window к его DesiredSize.Height через SizeToDoubleConverter (для определения возврата Width или Height соответственно требуется Dimension), но это не дало никакого эффекта.

Это выглядит многообещающе...

protected override Size MeasureOverride(Size availableSize) {

    var desiredSize = base.MeasureOverride(availableSize);

    MinWidth  = Math.Ceiling(DesiredSize.Width);
    MinHeight = Math.Ceiling(DesiredSize.Height);

    return desiredSize;
}

Примечание. Я обычно использую Math.Ceiling вместо фактического DesiredSize.Height, потому что обычно это дробь, и при выравнивании по пикселям устройства макет будет сбит, что приведет к путанице.

Вышеприведенное близко к работе, но вы столкнетесь с проблемами, если захотите также установить MaxHeight равным ему (например, чтобы разрешить горизонтальный, но не вертикальный размер).

Это легко сделать с помощью AutoLayout в iOS/macOS, но я не уверен, как это сделать в WPF, даже программно. Я просто должен продолжать гадать, какой размер выглядит лучше всего.


person Mark A. Donohoe    schedule 16.08.2018    source источник
comment
Так в чем проблема с установкой MinHeight на 60 (или StackPanel.ActualHeight). Или обработать событие SizeChanged?   -  person mm8    schedule 16.08.2018
comment
Возможно, вам сойдет с рук трюк, показанный здесь.   -  person Mark Feldman    schedule 16.08.2018
comment
@ мм8, как я могу узнать, что это за число, не гуляя по дереву? В этом вся моя суть.   -  person Mark A. Donohoe    schedule 16.08.2018
comment
Все дерево? Окно имеет свойство Content. И UIElement имеет свойство ActualHeight. Или какой высоты вы хотите?   -  person mm8    schedule 16.08.2018
comment
Я обновил вопрос, чтобы быть более ясным. По сути, я пытаюсь заблокировать окно от MinWidth до DesiredSize.Width и от MinHeight до DesiredSize.Height, чтобы окно никогда не было меньше, чем наименьшее допустимое содержимое, которое оно может иметь. Есть смысл?   -  person Mark A. Donohoe    schedule 16.08.2018


Ответы (1)


Это хороший вопрос. Я достиг цели, но это некрасиво. Я бы не рекомендовал этот подход, но он делает то, что написано на банке.

Голые кости, проект под названием DynamicMinumumSize:

MainWindow.xaml.cs:

public partial class MainWindow : Window
{
    public MainWindow()
    {
        InitializeComponent();

        DataContext = new BackingData();
    }
}

public class BackingData
{
    public double StackMinHeight { get; set; }
}

Xaml:

<Window x:Class="DynamicMinimumSize.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
        xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
        xmlns:local="clr-namespace:DynamicMinimumSize"
        mc:Ignorable="d"
        Title="MainWindow" Height="350" Width="525" d:DataContext="{d:DesignInstance local:BackingData}" MinHeight="{Binding ElementName=Stack, Path=MinHeight}">
    <Window.Resources>
        <ResourceDictionary>
            <local:ChildMinHeightConverter x:Key="ChildMinHeightConverter" />
        </ResourceDictionary>
    </Window.Resources>
    <StackPanel x:Name="Stack">
        <StackPanel.MinHeight>
            <MultiBinding Converter="{StaticResource ChildMinHeightConverter}" >
                <MultiBinding.Bindings>
                    <Binding Path="StackMinHeight" />
                    <Binding RelativeSource="{RelativeSource Self}"/>
                </MultiBinding.Bindings>
            </MultiBinding>
        </StackPanel.MinHeight>
        <Button MinHeight="53">Min Height 53</Button>
        <Button MinHeight="53">Min Height 53</Button>
        <Button MinHeight="53">Min Height 53</Button>
    </StackPanel>
</Window>

Конвертер:

public class ChildMinHeightConverter : IMultiValueConverter
{
    public object Convert(object[] values, Type targetType, object parameter, CultureInfo culture)
    {
        var currentMinHeight = values[0] as double? ?? 0;

        if (values[1] is Panel)
        {
            var parent = values[1] as Panel;
            foreach (var child in parent.Children)
            {
                //Hack: The children won't necessarily be buttons
                currentMinHeight += (child as Button).MinHeight;
            }
        }
        //HACK: +40, I think it's the window toolbar.....
        return currentMinHeight + 40;
    }

    public object[] ConvertBack(object value, Type[] targetTypes, object parameter, CultureInfo culture)
    {
        throw new NotImplementedException();
    }
}

Визуализировано:

введите здесь описание изображения

person d347hm4n    schedule 16.08.2018