WPF 多列下拉+多选+搜索过滤

主要的类:

public class MultiComboBoxEdit: ComboBox, INotifyPropertyChanged
    {
        public MultiComboBoxEdit()
            : base()
        {
            DataCollection = new ObservableCollection<BookViewModel>()
            {
                new BookViewModel(){Name = "C语言程序设计",Price =39.9,Publisher = "谭浩强"},
                new BookViewModel(){Name = "WPF深入浅出",Price =39.9,Publisher = "谭浩强"},
                 new BookViewModel(){Name = "JavaScript高级编程",Price =39.9,Publisher = "谭浩强"},
                  new BookViewModel(){Name = "算法分析",Price =39.9,Publisher = "谭浩强"}
            };
            DataCollectionView = CollectionViewSource.GetDefaultView(DataCollection);
            //DataCollectionView.Filter = new Predicate<object>(OnFilter);

            this.PropertyChanged += SymbolEdit_PropertyChanged;
        }

        private void SymbolEdit_PropertyChanged(object sender, PropertyChangedEventArgs e)
        {
            if (e.PropertyName.Equals("SelctedInstance"))
            {
                if (SelctedInstance != null)
                {
                    SelectedBook = SelctedInstance.Name;
                    Price = SelctedInstance.Price;
                    //DataCollection.Add(new BookViewModel(){Name = "C语言程序设计",Price =39.9,Publisher = "谭浩强"});
                    App.Current.Dispatcher.Invoke(()=>DataCollectionView.Refresh());
                }
            }
            if (e.PropertyName.Equals("SelectedBook"))
            {
                App.Current.Dispatcher.Invoke(()=>DataCollectionView.Refresh());
            }
        }



        private bool OnFilter(object mod)
        {
            var book = mod as BookViewModel;
            if (book != null && !string.IsNullOrWhiteSpace(SelectedBook))
            {
                return book.Name.Contains(SelectedBook) || book.Name.Equals(SelectedBook, StringComparison.CurrentCultureIgnoreCase);
            }
            else
                return true;
        }
        private static ICollectionView dataCollectionView;
        private ObservableCollection<BookViewModel> dataCollection;

        public static ICollectionView DataCollectionView
        {
            get { return dataCollectionView; }
            set
            {
                if (dataCollectionView != value)
                {
                    dataCollectionView = value;
                    //OnPropertyChanged("DataCollectionView");
                };
            }
        }


        public ObservableCollection<BookViewModel> DataCollection
        {
            get { return dataCollection; }
            set
            {
                if (dataCollection != value)
                {
                    dataCollection = value;
                    OnPropertyChanged("DataCollection");
                };
            }
        }

        public event PropertyChangedEventHandler PropertyChanged;

        public void OnPropertyChanged(string propertyName)
        {
            var handler = PropertyChanged;
            if (handler != null)
                handler(this, new PropertyChangedEventArgs(propertyName));
        }



        public static readonly DependencyProperty SelectedBookProperty = DependencyProperty.Register("SelectedBook",
            typeof(string), typeof(MultiComboBoxEdit), new FrameworkPropertyMetadata(string.Empty, FrameworkPropertyMetadataOptions.AffectsMeasure,
                new PropertyChangedCallback(TotalPropertyChanged)));

        public string SelectedBook
        {
            get { return (string)GetValue(SelectedBookProperty); }
            set { SetValue(SelectedBookProperty, value); }
        }

         private static void TotalPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            ((MultiComboBoxEdit)d).OnPropertyChanged(e.Property.Name);
        }


         public static readonly DependencyProperty SelctedInstancekProperty = DependencyProperty.Register("SelctedInstance",
             typeof(BookViewModel), typeof(MultiComboBoxEdit), new FrameworkPropertyMetadata(null, FrameworkPropertyMetadataOptions.AffectsMeasure,
                 new PropertyChangedCallback(TotalPropertyChanged)));

        public BookViewModel SelctedInstance
        {
            get { return (BookViewModel)GetValue(SelctedInstancekProperty); }
            set
            {
                SetValue(SelctedInstancekProperty,value);
            }
        }

        public static readonly DependencyProperty SelctedInstanceskProperty = DependencyProperty.Register("SelctedInstances",
           typeof(BookViewModel), typeof(MultiComboBoxEdit), new FrameworkPropertyMetadata(null, FrameworkPropertyMetadataOptions.AffectsMeasure,
               new PropertyChangedCallback(TotalPropertyChanged)));

        public BookViewModel SelctedInstances
        {
            get { return (BookViewModel)GetValue(SelctedInstanceskProperty); }
            set
            {
                SetValue(SelctedInstanceskProperty, value);
            }
        }


        private double price;

        public double Price
        {
            get { return price; }
            set
            {
                if (price != null)
                {
                    price = value;
                    OnPropertyChanged("Price");
                }
            }
        }

    }

ISelected接口

 public interface ISelected
    {
        bool IsChecked { get; set; }
    }

SelectedManager

 public class SelectedManager<T>:NotifycationModel
    {
        private ObservableCollection<ISelected> sourceCollection;

        public ObservableCollection<ISelected> SourceCollection
        {
            get { return sourceCollection; }
            private set {
                if (sourceCollection!=value)
                {
                    sourceCollection = value;
                    OnPropertyChanged("SourceCollection");
                }
            }
        }

        ICommand _checekedCommand;

        public ICommand CheckedCommand
        {
            get
            {
                if(_checekedCommand==null)
                    _checekedCommand=new RelayCommand((p) =>
                    {
                        if (p != null)
                        {
                            
                        }
                    });
                return _checekedCommand;
            }
        }

        public SelectedManager(ObservableCollection<ISelected> dataCollection)
        {
            SourceCollection = dataCollection;
        }
    }

xaml

    <Style x:Key="dropDownToggleButtonStyle" TargetType="ToggleButton">
        <Setter Property="BorderThickness" Value="0"/>
        <Setter Property="Background" Value="Transparent"/>
        <Setter Property="BorderBrush" Value="Transparent"/>
        <Setter Property="Width" Value="25"/>
        <Setter Property="Height" Value="25"/>
        <Setter Property="RenderTransformOrigin" Value="0.5,0.5"></Setter>
        <Setter Property="Template">
            <Setter.Value>
                <ControlTemplate TargetType="ToggleButton">
                    <!--<ContentPresenter />-->
                    <Grid>
                        <Image x:Name="img" Source="../Images/DropDown.png">
                            <Image.RenderTransform>
                                <RotateTransform x:Name="RotateTransfer" CenterX="12.5" CenterY="12.5" Angle="0"/>
                            </Image.RenderTransform>
                        </Image>
                    </Grid>
                    <ControlTemplate.Triggers>
                        <Trigger Property="IsMouseOver" Value="True">
                            <Setter Property="Background" Value="#808080"></Setter>
                        </Trigger>
                        <Trigger Property="IsPressed" Value="True">
                            <Setter Property="Background" Value="#F5F5F5"></Setter>
                        </Trigger>
                        <EventTrigger RoutedEvent="ToggleButton.Checked">
                            <BeginStoryboard>
                                <Storyboard>
                                    <DoubleAnimation By="180" Duration="0:0:.1"
                                             Storyboard.TargetName="RotateTransfer" 
                                             Storyboard.TargetProperty="Angle" />
                                </Storyboard>
                            </BeginStoryboard>
                        </EventTrigger>
                        <EventTrigger RoutedEvent="ToggleButton.MouseLeave">
                            <BeginStoryboard>
                                <Storyboard>
                                    <DoubleAnimation By="-180" Duration="0:0:.1"
                                             Storyboard.TargetName="RotateTransfer" 
                                             Storyboard.TargetProperty="Angle" />
                                </Storyboard>
                            </BeginStoryboard>
                        </EventTrigger>
                    </ControlTemplate.Triggers>
                </ControlTemplate>
            </Setter.Value>
        </Setter>
    </Style>

<Style TargetType="{x:Type controls:MultiComboBoxEdit}">
        <!--<Setter Property="Width" Value="200"/>-->
        <Setter Property="Height" Value="Auto"/>
        <Setter Property="Template">
            <Setter.Value>
                <ControlTemplate TargetType="{x:Type controls:MultiComboBoxEdit}">
                    <Grid>
                        <Grid.RowDefinitions>
                            <RowDefinition Height="30"/>
                            <RowDefinition Height="Auto"/>
                        </Grid.RowDefinitions>
                        <Border Grid.Row="0" BorderThickness="0.5" BorderBrush="#808080">
                            <Grid  VerticalAlignment="Center">
                                <Grid.ColumnDefinitions>
                                    <ColumnDefinition Width="*"/>
                                    <ColumnDefinition Width="Auto"/>
                                    <ColumnDefinition Width="30"/>
                                </Grid.ColumnDefinitions>
                                <TextBox Grid.Column="0" Focusable="True" Width="Auto" MinWidth="200"  x:Name="txt_Input" 
                                         Text="{Binding InputValue,RelativeSource={RelativeSource TemplatedParent},Mode=TwoWay,UpdateSourceTrigger=PropertyChanged}" />
                                <ToggleButton Grid.Column="2" x:Name="btnToggle" HorizontalAlignment="Right" Margin="8,0,0,0" Style="{StaticResource dropDownToggleButtonStyle}">
                                </ToggleButton>
                            </Grid>
                        </Border>

                        <!--数据匹配列表-->
                        <Popup x:Name="dataList"  StaysOpen="True" PlacementTarget="{Binding ElementName=txt_Input}" AllowsTransparency="True" >
                            <ListView x:Name="_listView" Focusable="True" AllowDrop="True" IsEnabled="True" Width="Auto" Height="Auto"
                                      SelectedValue="{Binding SelctedInstance,RelativeSource={RelativeSource TemplatedParent},Mode=TwoWay}"
                                      ItemsSource="{Binding DataCollectionView,RelativeSource={RelativeSource TemplatedParent},Mode=TwoWay}" 
                                      
                                      >
                                <ListView.View>
                                    <GridView AllowsColumnReorder="True">
                                        <GridViewColumn>
                                            <GridViewColumn.Header>
                                                <CheckBox x:Name="checkedAll">全选</CheckBox>
                                            </GridViewColumn.Header>
                                            <GridViewColumn.CellTemplate>
                                                <DataTemplate>
                                                    <CheckBox x:Name="checkBox" IsChecked="{Binding ElementName=checkedAll,Path=IsChecked,Mode=OneWay}" >
                                                        <i:EventTrigger>
                                                            <i:InvokeCommandAction Command="{}"></i:InvokeCommandAction>
                                                        </i:EventTrigger>
                                                    </CheckBox>
                                                </DataTemplate>
                                            </GridViewColumn.CellTemplate>
                                            </GridViewColumn>
                                        <GridViewColumn Header="书名" Width="Auto"  DisplayMemberBinding="{Binding Name}"/>
                                        <GridViewColumn Header="价格" Width="Auto" DisplayMemberBinding="{Binding Price}"/>
                                        <GridViewColumn Header="作者" Width="Auto" DisplayMemberBinding="{Binding Publisher}"/>
                                    </GridView>
                                </ListView.View>
                            </ListView>
                        </Popup>
                    </Grid>
                    <ControlTemplate.Triggers>
                        <Trigger Property="IsFocused" SourceName="txt_Input" Value="True">
                            <Setter TargetName="dataList" Property="IsOpen" Value="True"/>
                        </Trigger>
                        <Trigger Property="Text" SourceName="txt_Input" Value="">
                            <Setter TargetName="dataList" Property="IsOpen" Value="False"/>
                        </Trigger>
                        <MultiTrigger>
                            <MultiTrigger.Conditions>
                                <Condition Property="IsFocused" SourceName="txt_Input" Value="False"/>
                                <Condition Property="IsFocused" SourceName="dataList" Value="False"/>
                            </MultiTrigger.Conditions>
                            <MultiTrigger.Setters>
                                <Setter TargetName="dataList" Property="IsOpen" Value="False"></Setter>
                            </MultiTrigger.Setters>
                        </MultiTrigger>
                        <MultiTrigger>
                            <MultiTrigger.Conditions>
                                <Condition Property="IsFocused" SourceName="txt_Input" Value="False"/>
                                <Condition Property="IsFocused" SourceName="dataList" Value="True"/>
                            </MultiTrigger.Conditions>
                            <MultiTrigger.Setters>
                                <Setter TargetName="dataList" Property="IsOpen" Value="True"></Setter>
                            </MultiTrigger.Setters>
                        </MultiTrigger>
                    </ControlTemplate.Triggers>
                </ControlTemplate>

            </Setter.Value>
        </Setter>
    </Style>
public class SyncLockCollectionView<T>:CollectionView
    {
        private static object _objLock;
        private ObservableCollection<T> dataCollection;

        public ObservableCollection<T> DataCollection
        {
            get { return dataCollection; }
            set { dataCollection = value; }
        }


        public SyncLockCollectionView(object objLock, ObservableCollection<T> collection)
            :base(collection)
        {
            _objLock = objLock;
            lock (_objLock)
            {
                DataCollection = collection;
            }
        }
    }

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值