上一篇文章有写到 自动轮播的控件  简易的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. Linux切换中英文输入

    使用xshell登录Linux服务器后,输入的命令正确但是提示命令不存在,这是什么鬼. 通过移动光标可以发现两种字体的宽度不一样 解决方法 shift + 空格 进行切换

  2. 网心云在PVE下三种磁盘IO模式(No cache,Write through,Write back)选择与优化指南

    ---------------------------------------------------------------------------------------------------- ...

  3. 【前端面试】(三)JavaScript相等(==)和全等(===)运算符的区别

    视频链接: JavaScript相等()和全等(=)运算符的区别 - Web前端工程师面试题讲解 参考链接: JavaScript == 与 === 区别 区别 对于string.number 等基础 ...

  4. python小题目练习(十)

    题目:根据生日判断星座 需求:实现如下图所示结果 代码展示: """Author:mllContent:根据生日判断星座Date:2020-11-23"&quo ...

  5. python小题目练习(九)

    题目:将美元转化为人民币 需求:实现如图所示需求  代码展示: """Author:mllContent:将美元转化为人民币Date:2020-11-23"&q ...

  6. 内存分析器 (MAT)

    内存分析器 (MAT) 1. 内存分析器 (MAT) 1.1   MAT介绍 MAT是Memory Analyzer tool的缩写.指分析工具. 1.2   MAT作用 Eclipse Memory ...

  7. MarkDown语法——更好地写博客

    MarkDown语法--更好地写博客 我们在学习过程中要尽量养成编写博客的 好习惯:一方面方便自己在学习之后进行一次汇总,其次自己书写的文章可以在以后的时间里反复查看以便于巩固,在找工作时博客也是被招 ...

  8. 使用Runnable和Callable接口实现多线程的区别

    使用Runnable和Callable接口实现多线程的区别 先看两种实现方式的步骤: 1.实现Runnable接口 public class ThreadDemo{ public static voi ...

  9. 写了个 Markdown 命令行小工具,希望能提高园友们发文的效率!

    写了个 Markdown 命令行小工具,希望能提高园友们发文的效率! 前言 笔者使用 Typora 来编写 Markdown 格式的博文,图片采用的是本地相对路径存储(太懒了不想折腾图床). 时间久了 ...

  10. 安卓fastboot刷机、刷magisk、aidlux备忘

    环境就不多说了,网上一堆教程,我只在这边简单记录一下,以小米手机为例 刷机 解锁bootloader PC上配置好adb.fastboot,也就是platform-tools工具包加入系统变量,在命令 ...