0

我编写了一个自定义控件来在计算的位置显示来自 ItemsSource 的项目。

通用的.xaml

<ResourceDictionary
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:Control="clr-namespace:MyControls.Control">

    <Style TargetType="{x:Type Control:MyControl}">
        <Setter Property="Template">
            <Setter.Value>
                <ControlTemplate TargetType="{x:Type Control:MyControl}">
                    <Border x:Name="DrawingArea"
                        BorderThickness="{TemplateBinding Border.BorderThickness}"
                        BorderBrush="{TemplateBinding Border.BorderBrush}"
                        Padding="{TemplateBinding Control.Padding}"
                        Background="{TemplateBinding Panel.Background}"                            
                        SnapsToDevicePixels="true">
                        <Grid HorizontalAlignment="Stretch" VerticalAlignment="Stretch">
                            <Canvas IsItemsHost="True" HorizontalAlignment="Stretch" VerticalAlignment="Stretch" ClipToBounds="true" />                            
                        </Grid>
                    </Border>
                </ControlTemplate>
            </Setter.Value>     
        </Setter>
        <Setter Property="ItemTemplate">
            <Setter.Value>
                <DataTemplate>
                    ...
                </DataTemplate>
            </Setter.Value>
        </Setter>
    </Style>
</ResourceDictionary>

我的控制.cs

public class MyControl : MultiSelector, INotifyPropertyChanged    
{
    private bool _layoutHandlingDone = false;
    private Border _drawingArea;
    private Border DrawingArea
    {
        get { return _drawingArea ?? (_drawingArea = (Border) Template.FindName("DrawingArea", this)); }
    }

    static MyControl()
    {
        DefaultStyleKeyProperty.OverrideMetadata(typeof(MyControl), new FrameworkPropertyMetadata(typeof(MyControl)));
    }

    protected override Size MeasureOverride(Size aviableSize) { ... }       
    protected override Size ArrangeOverride(Size finalSize) { ... }

    public override void OnApplyTemplate()
    {
        base.OnApplyTemplate();

        if (Template != null && _layoutHandlingDone == false)
        {
            _layoutHandlingDone = true;
            ItemContainerGenerator.StatusChanged += (sender, args) =>
                                                        {
                                                            if(ItemContainerGenerator.Status == GeneratorStatus.ContainersGenerated)
                                                                InvalidateMeasure();
                                                        };
            DrawingArea.SizeChanged += (sender, args) => UpdateLayout();
        }
    }   
}

布局和绘制项目效果很好。现在我希望这些项目是可选择的。为此,我派生了我的控件表单 MultiSelector,但似乎这还不够。

MultiSelector 提供什么功能,我应该如何在我的控件中使用它?

4

1 回答 1

1

我强烈建议在这里分开。管理项目集合并将它们安排在容器控件中是两个不同的问题,ItemsControl 为两者提供支持。

与其重写 MeasureOverride 和 ArrangeOverride,不如编写专门的Panel类(并在那里重写 MeasureOverride 和 ArrangeOverride)并在 ItemsControl 的ItemsPanel模板中使用该类。

然后,您可以轻松地从 ListBox 而不是 MultiSelector 派生,并免费获得所有选择功能。

于 2012-12-14T11:10:11.680 回答