温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

UWP中如何用GetAlphaMask实现阴影

发布时间:2020-06-18 16:56:42 来源:亿速云 阅读:241 作者:元一 栏目:编程语言

1. 前言

要在UWP中制作长阴影的话,可以用GetAlphaMask拿到轮廓,做成灰色,然后复制一百几十个摆在后面。GetAlphaMask的使用场景十分有限,Github上能搜到的内容都是用来配合DropShadow的,所以这篇文章也以介绍DropShadow为主。

2. 合成阴影

先介绍一下合成阴影。Compositor.CreateDropShadow()可以创建一个DropShadow,将这个DropShadowDropShadow赋值到SpriteVisual的Shadow属性,然后使用ElementCompositionPreview.SetElementChildVisual 将这个SpriteVisual设置到某个UIElement的可视化层里,再将这个UIElement放到需要阴影的元素后面,这样基本的合成阴影就完成了。

具体代码如下:

<Grid VerticalAlignment="Center"      HorizontalAlignment="Center">    <Grid x:Name="BackgroundGrid"/>    <Grid Background="Turquoise"          x:Name="Host">        <TextBlock Text="I need shadow"                   Foreground="White"                   HorizontalAlignment="Center"                   VerticalAlignment="Center"                   FontSize="36"                   Margin="16"/>    </Grid> </Grid>
private readonly Compositor _compositor; private readonly SpriteVisual _backgroundVisual; private readonly DropShadow _dropShadow; public MainPage() : base() {    InitializeComponent();    _compositor = ElementCompositionPreview.GetElementVisual(this).Compositor;    //创建并配置DropShadow    _dropShadow = _compositor.CreateDropShadow();    _dropShadow.BlurRadius = 16;    _dropShadow.Offset = new Vector3(8);    //创建SpriteVisual并设置Shadow    _backgroundVisual = _compositor.CreateSpriteVisual();    _backgroundVisual.Shadow = _dropShadow;    //将SpriteVisual放到可视化树    ElementCompositionPreview.SetElementChildVisual(BackgroundGrid, _backgroundVisual);    Host.SizeChanged += OnHostSizeChanged; } private void OnHostSizeChanged(object sender, SizeChangedEventArgs e) {    Vector2 newSize = new Vector2(0, 0);    Vector3 centerPoint = new Vector3(0, 0, 0);    if (Host != null)    {        newSize = new Vector2((float)Host.ActualWidth, (float)Host.ActualHeight);        centerPoint = new Vector3((float)Host.ActualWidth / 2, (float)Host.ActualHeight / 2, 0);    }    _backgroundVisual.CenterPoint = centerPoint;    _backgroundVisual.Size = newSize; }

3. 使用GetAlphaMask裁剪阴影

上面的代码需要可以实现阴影,但只能实现矩形的阴影,在WPF和Silverlight中常用的Shape的阴影,或者文字的阴影都做不出来。

例如将XAML改成这样的话,结果绝不是我想要的东西:

<Grid VerticalAlignment="Center"      HorizontalAlignment="Center">    <Grid x:Name="BackgroundGrid"/>    <TextBlock Text="I need shadow"                   x:Name="Host"                   Foreground="Turquoise"                   HorizontalAlignment="Center"                   VerticalAlignment="Center"                   FontSize="36"/> </Grid>

这时候就需要用到GetAlphaMask这个函数。

Image、 TextBlock和Shape分别实现一个名为GetAlphaMask的方法, 该方法返回一个CompositionBrush , 该方法表示具有元素形状的灰度图像。

官当文档 中是这样描述GetAlphaMask函数的,简单来说就是拿到一个Image、TextBlock或Shape的轮廓,这个轮廓可以作为DropShadow.Mask的值,这样DropShadow的形状就可调用GetAlphaMask的元素的形状一样了。

具体代码和结果如下,这才是我想要的效果:

_dropShadow.Mask = Host.GetAlphaMask();

4. 使用DropShadowPanel

如果觉得自己写代码太过复杂, 可以使用Toolkit中的DropShadowPanel。

DropShadowPanel继承自ContentControl,当它的Cotnent为Shape、TextBlock、Image之一(或Toolkit中实现了GetAlphaMask的其它控件)时,它就调用GetAlphaMask获取阴影的形状,否则就是简单的举行,代码如下:

CompositionBrush mask = null; if (Content is Image) {    mask = ((Image)Content).GetAlphaMask(); } else if (Content is Shape) {    mask = ((Shape)Content).GetAlphaMask(); } else if (Content is TextBlock) {    mask = ((TextBlock)Content).GetAlphaMask(); } else if (Content is ImageExBase imageExBase) {    imageExBase.ImageExInitialized += ImageExInitialized;    if (imageExBase.IsInitialized)    {        imageExBase.ImageExInitialized -= ImageExInitialized;        mask = ((ImageExBase)Content).GetAlphaMask();    } } _dropShadow.Mask = mask;

之后它的做法和上面介绍的一样,把这个阴影设置到一个元素放在ContentPresenter后面,看起来就实现了Content的阴影:

_border = GetTemplateChild(PartShadow) as Border; if (_border != null) {    ElementCompositionPreview.SetElementChildVisual(_border, _shadowVisual); }
<Grid Background="{TemplateBinding Background}"      BorderBrush="{TemplateBinding BorderBrush}"      BorderThickness="{TemplateBinding BorderThickness}"      HorizontalAlignment="{TemplateBinding HorizontalAlignment}"      VerticalAlignment="{TemplateBinding VerticalAlignment}">    <Border x:Name="ShadowElement"            HorizontalAlignment="{TemplateBinding HorizontalContentAlignment}"            VerticalAlignment="{TemplateBinding VerticalContentAlignment}" />    <ContentPresenter HorizontalAlignment="{TemplateBinding HorizontalContentAlignment}"                      VerticalAlignment="{TemplateBinding VerticalContentAlignment}"/> </Grid>
向AI问一下细节

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

AI