WPF使ListBox支持手势多选功能

摘要:
接下来我将介绍如何为ListBox添加手势功能支持。这里我们用到了InkCanvas,它有一个Gesture事件,在这个事件中我们可以得到我们所画出的形状的区域及e.Strokes[0].GetGeometry(),然后我们对这ListBox的这个区域做命中检查VisualTreeHelper.HitTest,从中过滤出我们想选中的多个ListBoxItem.效果图:代码如下:查看XAML//鼠标指针模式privatevoidNoneMode_Click{m_ic.EditingMode=InkCanvasEditingMode.None;}//手势选择模式privatevoidSelectionMode_Click{m_ic.EditingMode=InkCanvasEditingMode.GestureOnly;listBox1.SelectedItem=null;}//得到选中的项privatevoidGetSelected_Click{intcount=listBox1.SelectedItems.Count;StringBuildermsg=newStringBuilder;for{varitem=listBox1.SelectedItems[i]asListBoxItem;msg.Append;msg.Append(",");}msg.Append;msg.Append;MessageBox.Show;}//被选中的项privateList_selections=newList;//InkCanvas的手势事件privatevoidm_ic_Gesture{listBox1.SelectedItem=null;//清空选择_selections.Clear();////命中检测手势区域中ListBox1内的控件VisualTreeHelper.HitTest;foreach{item.IsSelected=!

手势功能在手持平台应用较为丰富,PC上的应用还不是很多,不过还是有一些软件已应用这个功能如遨游浏览器等,应用得当还是相当可以丰富用户的互交体验的。

接下来我将介绍如何为ListBox添加手势功能支持。

这里我们用到了InkCanvas,它有一个Gesture事件,在这个事件中我们可以得到我们所画出的形状的区域及

e.Strokes[0].GetGeometry(),然后我们对这ListBox的这个区域做命中检查VisualTreeHelper.HitTest,从中过滤出我们想选中的多个ListBoxItem.

效果图:

t3_thumb[3]t4_thumb[2]

代码如下:

WPF使ListBox支持手势多选功能第3张WPF使ListBox支持手势多选功能第4张
查看XAML<InkCanvas EditingMode="None"  Grid.ColumnSpan="2" Name="m_ic" Height="350" VerticalAlignment="Top" 
                   Gesture="m_ic_Gesture">
                <ListBox Name="listBox1" SelectionMode="Extended"  Height="350" Width="780">
                    <ListBoxItem Content="Ok" />
                    <ListBoxItem Content="Maxzhang" />
                    <ListBoxItem Content="houxiaodong" />
                    <ListBoxItem Content="hao1" />
                    <ListBoxItem Content="wuhao" />
                    <ListBoxItem Content="xiaoge" />
                    <ListBoxItem Content="wefwefwefwef" />
            </ListBox>
        </InkCanvas>

        <Button Content="鼠标指针" Grid.Column="1" Height="23" 
                HorizontalAlignment="Left" Margin="9,356,0,0" Name="button1" 
                VerticalAlignment="Top" Width="75" Click="NoneMode_Click" />
        <Button Content="手势圈选" Grid.Column="1" Height="23" 
                HorizontalAlignment="Left" Margin="90,356,0,0" Name="button2" 
                VerticalAlignment="Top" Width="101" Click="SelectionMode_Click" />
        <Button Content="得到选中的项" Grid.Column="1" 
                Height="23" HorizontalAlignment="Left" Margin="600,356,0,0" Name="button3" 
                VerticalAlignment="Top" Width="97" Click="GetSelected_Click" />
 //鼠标指针模式
        private void NoneMode_Click(object sender, RoutedEventArgs e)
        {
            m_ic.EditingMode = InkCanvasEditingMode.None;
        }
        //手势选择模式
        private void SelectionMode_Click(object sender, RoutedEventArgs e)
        {
            m_ic.EditingMode = InkCanvasEditingMode.GestureOnly;
            listBox1.SelectedItem = null;
        }
        //得到选中的项
        private void GetSelected_Click(object sender, RoutedEventArgs e)
        {
           int count = listBox1.SelectedItems.Count;
           StringBuilder msg = new StringBuilder("Selected items:");
            for (int i = 0; i < count; i++)
            {
                var item = listBox1.SelectedItems[i] as ListBoxItem;
                msg.Append(item.Content);
                msg.Append(",");
            }
            msg.Append("Count:");
            msg.Append((count-1).ToString());
            MessageBox.Show(msg.ToString());
        }

        //被选中的项
        private List<ListBoxItem> _selections = new List<ListBoxItem>(5);   
        //InkCanvas的手势事件
        private void m_ic_Gesture(object sender, InkCanvasGestureEventArgs e)
        {
            listBox1.SelectedItem = null;    //清空选择
            _selections.Clear();             //
            //命中检测手势区域中ListBox1内的控件
            VisualTreeHelper.HitTest(listBox1, OnFilter, OnResult, new GeometryHitTestParameters(e.Strokes[0].GetGeometry()));
            foreach (ListBoxItem item in _selections)
            {
                item.IsSelected = !item.IsSelected;
            }
        }

        //过滤检测
        private HitTestFilterBehavior OnFilter(DependencyObject potentialHitTestTarget)
        {
            ListBoxItem item = potentialHitTestTarget as ListBoxItem;
            if (item != null)   //查找是ListBoxItem的控件
            {
                _selections.Add(item);
                return HitTestFilterBehavior.ContinueSkipSelfAndChildren;
            }
            return HitTestFilterBehavior.ContinueSkipSelf;
        }

        private HitTestResultBehavior OnResult(HitTestResult result)
        {
            return HitTestResultBehavior.Continue;
        }

免责声明:文章转载自《WPF使ListBox支持手势多选功能》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇FastAPI WebSocket 基本使用一C#控件背景透明的几种解决方案下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

WPF DataGrid绑定到数据源的方法

1 string conStr = System.Configuration.ConfigurationManager.ConnectionStrings["str"].ConnectionString; 2 SqlConnection con = new SqlConnection(conStr); 3...

WPF布局(3)坐标(转)

在动态定位布局时,经常需要获取控件或鼠标的相对位置,WPF提供了这样的函数可以我们使用。 1、获取元素相对于父控件的位置使用Vector VisualTreeHelper.GetOffset(Visual visual)方法,返回visual在其父控件中的偏移量,然后你再将返回值的Vector对象转换成Point对象2、获取元素相对于祖宗控件或子孙控件的位...

WPF布局(2)控件拖动

从Window继承的控件窗体,实现窗体拖动时可以简单的调用DragMove方法。但是对于自定义控件可以采取响应MouseButtonDown,MouseMove,MouseButtonUp的方法来实现控件的拖动。首先定义两个位置变量:    Point pOld = new Point();    Point pNew = new Point(); 1、M...

【转】WPF之DataGrid应用

前几天打算尝试下DataGrid的用法,起初以为应该很简单,可后来被各种使用方法和功能实现所折磨。网络上的解决方法太多,但也太杂。没法子,我只好硬着头皮阅览各种文献资料,然后不断的去尝试,总算小有成果。因此,把我学到的和大家分享一下,相信这篇文章会让你再很短的时间内学会DataGrid的大部分主要功能,而且很多难点都可以在里面找到解决方案。 由于涉及的应用...

WPF 多语言解决方案

1、首先安装Multilingual App Toolkit 2、新建项目,在VS中点击"工具" -> "Multilingual App Toolkit" -> "启用选定内容" 如果出现上述Issue, 打开项目AssemblyInfo.cs文件,加入如下代码片段,然后重复Step 2. [assembly: NeutralRes...

wpf treeview 数据绑定 递归绑定节点

1.先上效果 将所有节点加入ComboBox数据源,在ComboBox中选择时下方Treeview显示该节点下的子节点。 1.xaml文件,将以下代码加入界面合适位置 1     <StackPanel> 2 <StackPanel Margin="10"> 3 <La...