在Windows10 UWP开发平台上内置的XMAL布局面板包括RelativePanelStackPanelGridVariableSizedWrapGridCanvas。在开发淘宝UWP应用时,遇到以下业务场景。

业务场景


场景一:淘宝商品提供的一些消费者保障服务

场景二:淘宝商品的SKU属性展示

实现分析


系统默认的面板容器控件显然不符合要求了。在WPF里面有WrapPanel,但是在UWP应用里面没有,这个时候就需要自定义个Panel了来实现WrapPanel的功能,实现起来不是很复杂。在MSDN的文档上已经给出了详细的实现说明:Xaml自定义面板,主要就是自定义一个Panel的派生类,然后重写(MeasureOverrideArrangeOverride)方法。

以下是MSDN上对两个方法的解释说明

MeasureOverride方法

MeasureOverride 方法有返回值,当 Measure 方法在面板上受到布局中的父元素调用时,布局系统将使用该值作为面板自身的起始 DesiredSize。方法内的逻辑选择与它返回的内容同等重要,而且逻辑经常影响返回的值。

所有 MeasureOverride 实现应当循环访问 Children,并且对每个子元素调用 Measure 方法。调用 Measure 方法可为DesiredSize 属性创建值。这可能会通知面板本身需要多少空间,以及如何在元素间划分空间或为特定的子元素调整大小。

以下是 MeasureOverride 方法非常基本的框架:

  1. protected override Size MeasureOverride(Size availableSize)
  2. {
  3. Size returnSize; //TODO might return availableSize, might do something else
  4.  
  5. //loop through each Child, call Measure on each
  6. foreach (UIElement child in Children)
  7. {
  8. child.Measure(new Size()); // TODO determine how much space the panel allots for this child, that's what you pass to Measure
  9. Size childDesiredSize = child.DesiredSize; //TODO determine how the returned Size is influenced by each child's DesiredSize
  10. //TODO, logic if passed-in Size and net DesiredSize are different, does that matter?
  11. }
  12. return returnSize;
  13. }

ArrangeOverride方法

ArrangeOverride 方法有 Size 返回值,当 Arrange 在面板上受到布局中的父元素调用时,布局系统将在呈现面板本身时使用该值。通常输入 finalSize 和 ArrangeOverride 返回的 Size 相同。如果不相同,这意味着面板正尝试将自己调整为不同的大小,而不是布局中的其他参与者声明可用的大小。最终大小基于之前已通过面板代码运行布局的度量传递,这是通常不返回不同大小的原因:这意味着你在故意忽略度量逻辑。

不要返回具有 Infinity 组件的 Size。尝试使用这样的 Size 将从内部布局引发异常。

所有 ArrangeOverride 实现应当循环访问 Children,并且对每个子元素调用 Arrange 方法。和 Measure 一样,Arrange 没有返回值。与 Measure 不同,经计算的属性不会设置为结果(但是, 问题中的元素通常引发 LayoutUpdated 事件)。

以下是 ArrangeOverride 方法非常基本的框架:

  1. protected override Size ArrangeOverride(Size finalSize)
  2. {
  3. //loop through each Child, call Arrange on each
  4. foreach (UIElement child in Children)
  5. {
  6. Point anchorPoint = new Point(); //TODO more logic for topleft corner placement in your panel
  7. // for this child, and based on finalSize or other internal state of your panel
  8. child.Arrange(new Rect(anchorPoint, child.DesiredSize)); //OR, set a different Size
  9. }
  10. return finalSize; //OR, return a different Size, but that's rare
  11. }

创建自定义Panel控件


下面用一个简单的demo演示一下,就知道这两个方法的作用了。

首先新建一个MyPanel类继承自Panel类,将Mypanel的背景色设置成灰色,在MyPanel里面放入6个Border控件,每个Border控件设置不同的背景颜色,固定Width和Height为100或者200,方便查看各个控件的大小区域。

UI xaml:

  1. <Page
  2. x:Class="AppArrange.MainPage"
  3. xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
  4. xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
  5. xmlns:local="using:AppArrange"
  6. xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
  7. xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
  8. mc:Ignorable="d">
  9.  
  10. <Grid Background="{ThemeResource ApplicationPageBackgroundThemeBrush}">
  11. <local:MyPanel Background="Gray" HorizontalAlignment="Left" VerticalAlignment="Top">
  12. <Border Background="Red" Width="100" Height="100">
  13. <TextBlock Text="1" HorizontalAlignment="Center" VerticalAlignment="Center" FontSize="32"/>
  14. </Border>
  15. <Border Background="Green" BorderThickness="1" Width="100" Height="200">
  16. <TextBlock Text="2" HorizontalAlignment="Center" VerticalAlignment="Center" FontSize="32"/>
  17. </Border>
  18. <Border Background="Yellow" Width="200" Height="100">
  19. <TextBlock Text="3" HorizontalAlignment="Center" VerticalAlignment="Center" FontSize="32"/>
  20. </Border>
  21. <Border Background="OrangeRed" Width="100" Height="100">
  22. <TextBlock Text="4" HorizontalAlignment="Center" VerticalAlignment="Center" FontSize="32"/>
  23. </Border>
  24. <Border Background="Orange" Width="100" Height="100">
  25. <TextBlock Text="5" HorizontalAlignment="Center" VerticalAlignment="Center" FontSize="32"/>
  26. </Border>
  27. <Border Background="Orchid" Width="100" Height="100">
  28. <TextBlock Text="6" HorizontalAlignment="Center" VerticalAlignment="Center" FontSize="32"/>
  29. </Border>
  30. </local:MyPanel>
  31. </Grid>
  32. </Page>

Code behind:

  1. public class MyPanel : Panel
  2. {
  3. protected override Size MeasureOverride(Size availableSize)
  4. {
  5. return base.MeasureOverride(availableSize);
  6. }
  7.  
  8. protected override Size ArrangeOverride(Size finalSize)
  9. {
  10. return base.ArrangeOverride(finalSize);
  11. }
  12. }

这个时候MeasureOverride和ArrangeOverride什么都没有做,如果直接运行会是什么样子呢?

这个时候界面就是一片空白。添加的6个Border控件没有显示。Mypanel的灰色背景也没有显示,说明Mypanel的size是0,没有显示出来。

下面来实现MeasureOverride方法,遍历每个子控件并调用Measure方法。

  1. public class MyPanel : Panel
  2. {
  3. protected override Size MeasureOverride(Size availableSize)
  4. {
  5. foreach (FrameworkElement child in Children)
  6. {
  7. child.Measure(availableSize);
  8. }
  9. return availableSize;
  10.  
  11. }
  12.  
  13. protected override Size ArrangeOverride(Size finalSize)
  14. {
  15. return base.ArrangeOverride(finalSize);
  16. }
  17. }

这个时候Mypanel的面板显示出来了,背景颜色是灰色,但是子控件没有显示。

接下来,实现ArrangeOverride方法

  1. protected override Size ArrangeOverride(Size finalSize)
  2. {
  3. double x = ;
  4. double y = ;
  5.  
  6. foreach (FrameworkElement child in Children)
  7. {
  8. child.Arrange(new Rect(new Point(x, y), child.DesiredSize));
  9. x += child.DesiredSize.Width;
  10. y += child.DesiredSize.Height;
  11.  
  12. }
  13. return finalSize;
  14. }

运行结果:

子控件出来了,超出Page范围的会被遮住。这个时候就可以根据需要定义每个子控件的(x,y)坐标进行布局了。如果要横向布局,就递增x坐标,如果纵向布局就递增y坐标。

横向布局

递增x坐标

  1. protected override Size ArrangeOverride(Size finalSize)
  2. {
  3. double x = ;
  4. double y = ;
  5.  
  6. foreach (FrameworkElement child in Children)
  7. {
  8. child.Arrange(new Rect(new Point(x, y), child.DesiredSize));
  9. x += child.DesiredSize.Width;
  10. // y += child.DesiredSize.Height;
  11.  
  12. }
  13. return finalSize;
  14. }

运行结果

纵向布局

递增y坐标

  1. protected override Size ArrangeOverride(Size finalSize)
  2. {
  3. double x = ;
  4. double y = ;
  5.  
  6. foreach (FrameworkElement child in Children)
  7. {
  8. child.Arrange(new Rect(new Point(x, y), child.DesiredSize));
  9. //x += child.DesiredSize.Width;
  10. y += child.DesiredSize.Height;
  11.  
  12. }
  13. return finalSize;
  14. }

运行结果:

接下来的问题,就是横向或者纵向布局的时候,判断何时该换行了。要换行就要计算依次排列的子控件的宽和高,同时和Mypanel的大小进行比较是否超出边界。

这其中还有个问题是MeasureOverrideArrangeOverride 都会返回一个size大小。这两个大小有什么不一样吗。

MeasureOverride 返回值:此对象在布局过程中基于其对子对象分配大小的计算或者基于固定容器大小等其他因素而确定的它所需的大小。

ArrangeOverride 返回值:元素在布局中排列后使用的实际大小。

MeasureOverride的输入size和返回size

可以做几个实验看看实际效果:

1. 如果给Measure传递的值较小,例如比最小的子控件还小:

  1. protected override Size MeasureOverride(Size availableSize)
  2. {
  3. foreach (FrameworkElement child in Children)
  4. {
  5. child.Measure(new Size(,));
  6. }
  7. return availableSize;
  8. }

运行结果

子控件会被裁剪部分区域,显示不完整,以适应较小的size。

2. 如果给Measure传递的值较大,比子控件的大小要大。

  1. protected override Size MeasureOverride(Size availableSize)
  2. {
  3. foreach (FrameworkElement child in Children)
  4. {
  5. child.Measure(new Size(,));
  6. }
  7. return availableSize;
  8. }

运行结果

结果显示还是正常的大小,没有变化。

总结:

说明子控件的DesiredSize的会受到Measure传递的大小的限制,过小就会被裁剪,过大,不受影响,以实际的DesiredSize显示。Measure方法就是给控件分配一个可以显示的大小范围。

下面看看MeasureOverride返回值,实际上和Measure的作用是一样的,给MeasureOverride方法的参数size是MyPanel的父控件给MyPanel分配的显示区域大小,实际上会受到MyPanel 的Width、Height、HorizontalAlignment、VerticalAlignment等设置的影响,这里就不展开了。

如果子控件的大小显示区域超过了MyPanel的父控件给MyPanel分配的显示区域大小,子控件的显示区域会被裁剪。这个时候可以根据业务需要调整MeasureOverride 返回size或者调整每个子控件的Measure输入size,缩小子控件,使每个子控件都完整显示出来。

下面看看如果设置的MeasureOverride返回值过小是什么效果

  1. protected override Size MeasureOverride(Size availableSize)
  2. {
  3. foreach (FrameworkElement child in Children)
  4. {
  5. child.Measure(availableSize);
  6. }
  7. return new Size(,);
  8. }

运行结果:

灰色区域是容器的大小,各个子控件已经超出容器控件的大小范围了,MyPanel的父控件分配的大小是整个page的大小,在遍历子控件Border时分配给子控件也是page的大小显示区域,但是每个子控件都设置了Width和Height,而child.Measure(availableSize)的大小比子控件自己的size大,所以最后会显示控件实际的大小,不会受child.Measure(availableSize)的影响,但是MyPanel的MeasureOverride最后返回的时候,size被修改小了,这样整个容器就显示被修改后的大小,子控件溢出边界。

所以在自定义容器控件的时候,MeasureOverride 方法返回的size应该是所有子控件显示区域的最小size。而计算显示区域的最小size,应该根据子控件的布局方式来判断。

ArrangeOverride的输入size和返回size

需要注意的是:通常情况下ArrangeOverride输入的size和返回的size一样,如果返回的size过小,也会遇到和MeasureOverride返回的size过小一样的显示问题。所以不建议修改ArrangeOverride的返回size,直接将输入size返回就可以了,ArrangeOverride方法主要作用是给子控件定位坐标和大小,完成布局。

有了以上的准备,应该就知道怎么实现淘宝的业务了,主要是在水平方向上依次排列子控件,然后自动换行。

首先在MeasureOverride里面计算子控件需要显示大小区域,在ArrangeOverride里面根据子控件的大小排列方式计算显示坐标,实现自动换行。

  1. protected override Size ArrangeOverride(Size finalSize)
  2. {
  3. double x = ;
  4. double y = ;
  5. double maxHeight = ;
  6.  
  7. foreach (FrameworkElement child in Children)
  8. {
  9. if (maxHeight < child.DesiredSize.Height)
  10. {
  11. maxHeight = child.DesiredSize.Height;
  12. }
  13.  
  14. if ((x + child.DesiredSize.Width) > finalSize.Width)
  15. {
  16. x = ;
  17. y += maxHeight;
  18. maxHeight = ;
  19. }
  20.  
  21. child.Arrange(new Rect(new Point(x, y), child.DesiredSize));
  22. x += child.DesiredSize.Width;
  23. }
  24. return finalSize;
  25. }

运行结果:

如果要实现更复杂的功能,例如要同时支持可以横向纵向排列子控件,就需要做一些封装了,这里就不一一展开了,最后附上有完整功能的WrapPanel实现代码供大家参考。可以实现横向和纵向排列。

我的面板我做主 -- 淘宝UWP中自定义Panel的实现的更多相关文章

  1. 飞流直下的精彩 -- 淘宝UWP中瀑布流列表的实现

    在淘宝UWP中,搜索结果列表是用户了解宝贝的重要一环,其中的图片效果对吸引用户点击搜索结果,查看宝贝详情有比较大的影响.为此手机淘宝特意在搜索结果列表上采用了2种表现方式:一种就是普通的列表模式,而另 ...

  2. 淘宝UWP中的100个为什么

    从淘宝UWP第一版发布到现在,已经有十个月了,期间收到了用户各种各样的反馈,感谢这些用户的反馈,指导我们不断的修正.完善应用.但是也有一部分需求或建议,由于资源或技术的限制,目前确实无法做到,只能对广 ...

  3. 从淘宝 UWP 的新功能 -- 比较页面来谈谈 UWP 的窗口多开功能

    前言 之前在 剁手党也有春天 -- 淘宝 UWP ”比较“功能诞生记 这篇随笔中介绍了一下 UWP 淘宝的“比较”新功能呱呱坠地的过程.在鲜活的文字背后,其实都是程序员不眠不休的血泪史(有血有泪有史) ...

  4. 剁手党也有春天 -- 淘宝 UWP ”比较“功能诞生记

    前言 网购已经不再是现在的时髦,而变成了我们每天的日常生活.上网已经和买买买紧密地联系在了一起,成为了我们的人生信条.而逛街一词,越来越多地变成了一种情怀.有时候我们去逛街,要么是为了打发时间,要么是 ...

  5. 淘宝UWP桌面版已经发布

    目前正在等待应用商店的检测,很快会可以下载. 谢谢各位园主针对淘宝UWP 桌面版(又叫PC版,HD版等等)给予的feedback,在这里统一回复一下,就不一一感谢了. 有一件事需要说明一下,请看下图: ...

  6. PC版淘宝UWP揭秘

    经过第一轮内测后的bug数量:65 2015/11/27 - bug数量 = 60 2015/11/30 - bug数量 = 53 2015/12/1 - bug数量 = 49 2015/12/2 - ...

  7. 手机淘宝UWP

    各位园主好! bug 走势: 哪天bug 足够少,哪天就可以发布了  :) 2015/10/23: 49 2015/10/26: 40 2015/10/27: 36 2015/10/28: 30 20 ...

  8. php提取淘宝URL中ID的代码

    一段可以提取淘宝URL中ID的PHP代码. 例如: <?php $taobao = 'taobao.com'; $tmall = 'tmall.com'; $guojitmall = 'tmal ...

  9. Customize Acrylic Brush in UWP Applications(在UWP中自定义亚克力笔刷)

    原文 Customize Acrylic Brush in UWP Applications(在UWP中自定义亚克力笔刷) Windows 10 Fall Creators Update(Build ...

随机推荐

  1. mysql命令行修改字符编码

    1.修改数据库字符编码 mysql> alter database mydb character set utf8 ; 2.创建数据库时,指定数据库的字符编码 mysql> create ...

  2. 【leetcode】Generate Parentheses

    题目简述: Given n pairs of parentheses, write a function to generate all combinations of well-formed par ...

  3. webform中Session和Cookie对象的用法

    Session: Session:在计算机中,尤其是在网络应用中,称为"会话控制".Session 对象存储特定用户会话所需的属性及配置信息.这样,当用户在应用程序的 Web页之间 ...

  4. 返水bug-备用

    NOOK(N) CSBFB(25) off(Y) QQ(2652880032) G(1) off1(Y) QQ1(3479301404) G1(1) off2(Y) QQ2(309235846) G2 ...

  5. dedecms 文章页图片改为绝对路径

    这几天在网站改版,想把网站做大,想做频道页二级域名,于是在做网站的过程中发现一个问题,dedecms开设二级域名后,在二级域名的文章页无法显示图片,查看源代码后发现问题,由于dedecms文章页中的图 ...

  6. <转>Unity3D研究院之C#使用Socket与HTTP连接服务器传输数据包

    最近项目中需要使用HTTP与Socket,把自己这段时间学习的资料整理一下.有关Socket与HTTP的基础知识MOMO就不赘述拉,不懂得朋友自己谷歌吧.我们项目的需求是在登录的时候使用HTTP请求, ...

  7. Codeforces Round #344 (Div. 2) B. Print Check

    B. Print Check time limit per test 1 second memory limit per test 256 megabytes input standard input ...

  8. 【NEUQACM OJ】1018: A+B again

    1018: A+B again 题目描述 谷学长有一个非常简单的问题给你,给你两个整数A和B,你的任务是计算A+B. 输入 输入的第一行包含一个整数T(T<=20)表示测试实例的个数,然后2*T ...

  9. Codeforces Round #257 (Div. 2)

    A - Jzzhu and Children 找到最大的ceil(ai/m)即可 #include <iostream> #include <cmath> using name ...

  10. (RMQ版)LCA注意要点

    inline int lca(int x,int y){ if(x>y) swap(x,y); ]][x]]<h[rmq[log[y-x+]][y-near[y-x+]+]])? rmq[ ...