上一篇文章有写到 自动轮播的控件  简易的AutoPlayCarousel 轮播控件 - 黄高林 - 博客园 (cnblogs.com)

本章是基于自动轮播的一种衍生,通过拖拽鼠标进切换

直接上代码

DragDropCarousel核心代码

[ContentProperty(nameof(ItemSources))]
[TemplatePart(Name = "PART_StackPanel", Type = typeof(StackPanel))]
public class DragDropCarousel : Control
{
#region 字段
private StackPanel _stkMain;
/// <summary>
/// 标记是否点击
/// </summary>
private bool _isMouseDown;
/// <summary>
/// 标记是否动画中,动画过程中不允许操作
/// </summary>
private bool _isAnimating;
/// <summary>
/// 最后一次点击的位置
/// </summary>
private Point _lastDownPoint;
/// <summary>
/// 移动中
/// </summary>
private bool _isMoving;
#endregion #region 构造
static DragDropCarousel()
{
DefaultStyleKeyProperty.OverrideMetadata(typeof(DragDropCarousel), new FrameworkPropertyMetadata(typeof(DragDropCarousel)));
}
public DragDropCarousel()
{
Loaded += Carousel_Loaded;
SizeChanged += Carousel_SizeChanged;
}
#endregion #region 属性
/// <summary>
/// 子控件集合
/// </summary>
public ObservableCollection<FrameworkElement> ItemSources
{
get => (ObservableCollection<FrameworkElement>)GetValue(ItemSourcesProperty);
private set => SetValue(ItemSourcesProperty, value);
} public static readonly DependencyProperty ItemSourcesProperty =
DependencyProperty.Register("ItemSources", typeof(ObservableCollection<FrameworkElement>), typeof(DragDropCarousel), new PropertyMetadata(new ObservableCollection<FrameworkElement>()));
/// <summary>
/// 方向
/// </summary>
public Orientation Orientation
{
get => (Orientation)GetValue(OrientationProperty);
set => SetValue(OrientationProperty, value);
} public static readonly DependencyProperty OrientationProperty =
DependencyProperty.Register("Orientation", typeof(Orientation), typeof(DragDropCarousel), new PropertyMetadata(Orientation.Horizontal)); /// <summary>
/// 下标
/// </summary>
public int Index
{
get => (int)GetValue(IndexProperty);
set => SetValue(IndexProperty, value);
} public static readonly DependencyProperty IndexProperty =
DependencyProperty.Register("Index", typeof(int), typeof(DragDropCarousel), new PropertyMetadata(0)); /// <summary>
/// 抬起手时,滑动切页动画持续时间
/// </summary>
public TimeSpan AnimateDuration
{
get => (TimeSpan)GetValue(AnimateDurationProperty);
set => SetValue(AnimateDurationProperty, value);
} public static readonly DependencyProperty AnimateDurationProperty =
DependencyProperty.Register("AnimateDuration", typeof(TimeSpan), typeof(DragDropCarousel), new PropertyMetadata(TimeSpan.FromSeconds(0.5))); /// <summary>
/// 移动阈值
/// </summary>
public double MoveThreshold
{
get => (double)GetValue(MoveThresholdProperty);
set => SetValue(MoveThresholdProperty, value);
} // Using a DependencyProperty as the backing store for MoveThreshold. This enables animation, styling, binding, etc...
public static readonly DependencyProperty MoveThresholdProperty =
DependencyProperty.Register("MoveThreshold", typeof(double), typeof(DragDropCarousel), new PropertyMetadata(10d)); #endregion #region 对外方法
public override void OnApplyTemplate()
{
base.OnApplyTemplate();
_stkMain = GetTemplateChild("PART_StackPanel") as StackPanel;
}
#endregion #region Event Handler private void Carousel_SizeChanged(object sender, SizeChangedEventArgs e)
{
foreach (FrameworkElement children in ItemSources)
{
children.Width = ActualWidth;
children.Height = ActualHeight;
UnRegisterMouseEvent(children);
RegisterMouseEvent(children);
}
}
private void UnRegisterMouseEvent(FrameworkElement children)
{
children.MouseDown -= Children_MouseDown;
children.MouseMove -= Children_MouseMove;
children.MouseUp -= Children_MouseUp;
}
private void RegisterMouseEvent(FrameworkElement children)
{
children.MouseDown += Children_MouseDown;
children.MouseMove += Children_MouseMove;
children.MouseUp += Children_MouseUp;
} private void Children_MouseUp(object sender, MouseButtonEventArgs e)
{
if (!_isMouseDown || _isAnimating||!_isMoving)
{

_isMoving = false;
            _isMouseDown = false;

return;

            }
_isMoving=false;
_isMouseDown = false;
var targetIndex = GetTargetIndex(e, _lastDownPoint);
Index = targetIndex;
var newThickness = new Thickness(-1 * ActualWidth * targetIndex, 0, 0, 0);
var oldThickness = _stkMain.Margin;
var thicknessAnimation = new ThicknessAnimation()
{
From = oldThickness,
To = newThickness,
Duration = AnimateDuration,
EasingFunction = new CubicEase() { EasingMode = EasingMode.EaseOut },
FillBehavior = FillBehavior.Stop,
};
thicknessAnimation.Completed += (s, args) =>
{
_isAnimating = false;
_stkMain.Margin = newThickness;
};
_isAnimating = true;
_stkMain.BeginAnimation(MarginProperty, thicknessAnimation);
} private int GetTargetIndex(MouseButtonEventArgs e, Point lastDownPoint)
{
var position = e.GetPosition(this);
var vector = position - lastDownPoint;
int targetIndex;
if (vector.X < 0)
{
targetIndex = Index + 1 > (ItemSources.Count - 1) ? Index : Index + 1;
}
else
{
targetIndex = Index - 1 >= 0 ? Index - 1 : 0;
}
return targetIndex;
} private void Children_MouseMove(object sender, MouseEventArgs e)
{ var position = e.GetPosition(this);
if (!IsCanMove(_lastDownPoint, position))
{
_stkMain.Margin = new Thickness((-1 * ActualWidth * Index), 0, 0, 0);
return;
}
var moveLength = _lastDownPoint.X - position.X;
_stkMain.Margin = new Thickness((-1 * ActualWidth * Index - moveLength), 0, 0, 0);
} private bool IsCanMove(Point lastPoint, Point currentPoint)
{
if (!_isMouseDown)
{
return false;
} //以下控制不允许超出边界,想要回弹效果,可以去掉这部分代码
var vector = currentPoint - lastPoint;
if (!_isMoving && Math.Abs(vector.X) <= MoveThreshold)
{
return false;
} _isMoving = true;
if (vector.X < 0)
{
return Index != ItemSources.Count - 1;
} return Index != 0; } private void Children_MouseDown(object sender, MouseButtonEventArgs e)
{
if (!(MoveElementAttach.GetAllowMove(e.Source as UIElement) ?? false) || _isAnimating)
{
_isMouseDown = false;
return;
}
_isMouseDown = true;
_lastDownPoint = e.GetPosition(this);
} private void Carousel_Loaded(object sender, RoutedEventArgs e)
{
if (ItemSources == null)
return;
Loaded -= Carousel_Loaded;
foreach (FrameworkElement child in ItemSources)
{
child.Width = ActualWidth;
child.Height = ActualHeight;
}
}
#endregion }

  一些辅助的代码

public class IndexChangedEventArgs : RoutedEventArgs
{
public IndexChangedEventArgs(int currentIndex, RoutedEvent routedEvent) : base(routedEvent)
{
CurrentIndex = currentIndex;
} public int CurrentIndex { get; set; }
} public delegate void IndexChangedEventHandler(object sender, IndexChangedEventArgs e);

  

/// <summary>
/// 移动附加属性
/// </summary>
public class MoveElementAttach : DependencyObject
{
/// <summary>允许操作</summary>
public static readonly DependencyProperty AllowMoveProperty = DependencyProperty.RegisterAttached("AllowMove", typeof(bool), typeof(MoveElementAttach), new PropertyMetadata((PropertyChangedCallback)null)); public static bool? GetAllowMove(DependencyObject obj) => (bool?)obj?.GetValue(MoveElementAttach.AllowMoveProperty); public static void SetAllowMove(DependencyObject obj, bool? value) => obj.SetValue(MoveElementAttach.AllowMoveProperty, (object)value);
}

  

默认样式

<sys:Double x:Key="DefaultFontSize">14</sys:Double>
<sys:Boolean x:Key="DefaultSnapsToDevicePixels">false</sys:Boolean> <Style TargetType="{x:Type local:DragDropCarousel}">
<Setter Property="SnapsToDevicePixels" Value="{StaticResource DefaultSnapsToDevicePixels}" />
<Setter Property="FontSize" Value="{StaticResource DefaultFontSize}" />
<Setter Property="Template">
<Setter.Value>
<ControlTemplate TargetType="{x:Type local:DragDropCarousel}">
<StackPanel x:Name="PART_StackPanel" Orientation="Horizontal">
<ItemsControl x:Name="PART_ItemsControl" ItemsSource="{TemplateBinding ItemSources}"
VerticalAlignment="Stretch"
HorizontalAlignment="Stretch">
<ItemsControl.ItemsPanel>
<ItemsPanelTemplate>
<StackPanel Width="{TemplateBinding Width}" Height="{TemplateBinding Height}" Orientation="{Binding Orientation,RelativeSource={RelativeSource AncestorType=local:DragDropCarousel}}"/>
</ItemsPanelTemplate>
</ItemsControl.ItemsPanel>
</ItemsControl>
</StackPanel>
</ControlTemplate>
</Setter.Value>
</Setter>
</Style>

  

界面的使用

 <customControl:DragDropCarousel  x:Name="Carousel"
Height="1080">
<Grid Background="Red" customControl:MoveElementAttach.AllowMove="True" >
<Grid HorizontalAlignment="Center" VerticalAlignment="Center" Height="500" Width="500" Background="Black"> <TextBlock Text="1" FontSize="20" Foreground="Wheat" HorizontalAlignment="Center" VerticalAlignment="Center"></TextBlock>
</Grid>
</Grid>
<Grid Background="Green" customControl:MoveElementAttach.AllowMove="True" > <Grid HorizontalAlignment="Center" VerticalAlignment="Center" Height="500" Width="500" Background="Black"> <TextBlock Text="2" FontSize="20" Foreground="Wheat" HorizontalAlignment="Center" VerticalAlignment="Center"></TextBlock>
</Grid> </Grid>
<Grid Background="Yellow" customControl:MoveElementAttach.AllowMove="True" >
<Grid HorizontalAlignment="Center" VerticalAlignment="Center" Height="500" Width="500" Background="Black"> <TextBlock Text="3" FontSize="20" Foreground="Wheat" HorizontalAlignment="Center" VerticalAlignment="Center"></TextBlock>
</Grid>
</Grid>
<Grid Background="Blue" customControl:MoveElementAttach.AllowMove="True" >
<Grid HorizontalAlignment="Center" VerticalAlignment="Center" Height="500" Width="500" Background="Black"> <TextBlock Text="4" FontSize="20" Foreground="Wheat" HorizontalAlignment="Center" VerticalAlignment="Center"></TextBlock>
</Grid>
</Grid>
</customControl:DragDropCarousel>

  

通过 customControl:MoveElementAttach.AllowMove="True" 来标记那些控件支持拖拽,哪些不支持拖拽,这种只是临时解决方案,其实还有其他的解决方案。后续我如果要用到再持续更新

简易的DragDropCarousel 拖拽轮播控件的更多相关文章

  1. WPF 控件库——轮播控件

    WPF 控件库系列博文地址: WPF 控件库——仿制Chrome的ColorPicker WPF 控件库——仿制Windows10的进度条 WPF 控件库——轮播控件 WPF 控件库——带有惯性的Sc ...

  2. 第四十六篇、UICollectionView广告轮播控件

    这是利用人的视觉错觉来实现无限轮播,UICollectionView 有很好的重用机制,这只是部分核心代码,后期还要继续完善和代码重构. #import <UIKit/UIKit.h> # ...

  3. Android之仿京东淘宝的自动无限轮播控件

    在App的开发中,很多的时候都需要实现类似京东淘宝一样的自动无限轮播的广告栏,所以就自己写了一个,下面是我自定义控件的思路和过程. 一.自定义控件属性 新建自定义控件SliderLayout继承于Re ...

  4. Android图片轮播控件

    Android广告图片轮播控件,支持无限循环和多种主题,可以灵活设置轮播样式.动画.轮播和切换时间.位置.图片加载框架等! 使用步骤 Step 1.依赖banner Gradle dependenci ...

  5. jquery轮播控件

    网上找了一个轮播控件,效果不错,而且很容易改,需要的同学去下载咯 地址是:http://download.csdn.net/detail/jine515073/7704143

  6. Android-----------广告图片轮播控件

    Banner广告图片轮播控件,支持无限循环和多种主题,可以灵活设置轮播样式.动画.轮播和切换时间.位置.图片加载框架等! 很多Android APP中都有广告栏,我也用过很多次了,特来写一篇博文. 先 ...

  7. 一起写一个Android图片轮播控件

    注:本文提到的Android轮播控件Demo地址: Android图片轮播控件 1. 轮播控件的组成部分 我们以知乎日报Android客户端的轮播控件为例,分析一下轮播控件的主要组成: 首先我们要有用 ...

  8. Android 开发最牛的图片轮播控件,基本什么都包含了。

    Android图片轮播控件  源码下载地址: Android 图片轮播 现在的绝大数app都有banner界面,实现循环播放多个广告图片和手动滑动循环等功能.因为ViewPager并不支持循环翻页, ...

  9. Android高仿京东淘宝自动无限循环轮播控件的实现思路和过程

    在App的开发中,很多的时候都需要实现类似京东淘宝一样的自动无限轮播的广告栏,所以就自己写了一个,下面是我自定义控件的实现思路和过程. 一.自定义控件属性 新建自定义控件SliderLayout继承于 ...

随机推荐

  1. 如何正确理解古典概率中的条件概率 《考研概率论学习之我见》 -by zobol

    "B事件发生的条件下,A事件发生的概率"? "在A集合内有多少B的样本点"? "在B约束条件下,A发生的概率变化为?" "B事件中 ...

  2. 前端学习 linux —— 第一篇

    前端学习 linux - 第一篇 本文主要介绍"linux 发行版本"."cpu 架构"."Linux 目录结构"."vi 和 v ...

  3. 名校AI课推荐 | UC Berkeley《人工智能导论》

    深度学习具备强感知能力但缺乏一定的决策能力,强化学习具备决策能力但对感知问题束手无策,因此将两者结合起来可以达到优势互补的效果,为复杂系统的感知决策问题提供了解决思路. 今天我们推荐这样一门课程--U ...

  4. Vue几行代码实现搜索功能

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  5. Java基础-并发篇

    3.1. JAVA 并发知识库 3.2. JAVA 线程实现/创建方式 3.2.1. 继承 Thread 类 ​ Thread 类本质上是实现了 Runnable 接口的一个实例,代表一个线程的实例. ...

  6. SAP FPM 相关包 APB_FPM_CORE

    related interface: APB_FPM_COREAPB_FPM_CORE_4_EXT_SCAPB_FPM_CORE_INTERNALAPB_FPM_CORE_RESTRICTED

  7. 一文精通HashMap灵魂七问,你学还是不学

    如果让你看一篇文章,就可以精通HashMap,成为硬刚才面试官的高手,你学还是不学? 别着急,开始之前不如先尝试回来下面几个问题吧: HashMap的底层结构是什么? 什么时候HashMap中的链表会 ...

  8. iostat、vmstat、iftop命令详解

    1.安装iostat命令 yum install sysstat -y 用法:iostat [ 选项 ] [ <时间间隔> [ <次数> ]] 常用选项说明: -c:只显示系统 ...

  9. CVPR 2017:See the Forest for the Trees: Joint Spatial and Temporal Recurrent Neural Networks for Video-based Person Re-identification

    [1] Z. Zhou, Y. Huang, W. Wang, L. Wang, T. Tan, Ieee, See the Forest for the Trees: Joint Spatial a ...

  10. Host–Parasite(主从关系): Graph LSTM-in-LSTM for Group Activity Recognition

    This article aims to tackle the problem of group activity recognition in the multiple-person scene. ...