WPF ListBox的進階使用(二)

来源:https://www.cnblogs.com/Johar/archive/2018/08/30/9562266.html
-Advertisement-
Play Games

項目中經常使用需要根據搜索條件查詢數據,然後用卡片來展示數據。用卡片展示數據時,界面的寬度發生變化,希望顯示的卡片數量也跟隨變化。WrapPanel雖然也可以實現這個功能,但是將多餘的部分都留在行尾,十分不美觀,最好是能夠將多餘的寬度平分在每個ListBoxItem之間,比較美觀,也符合項目需求。如 ...


項目中經常使用需要根據搜索條件查詢數據,然後用卡片來展示數據。用卡片展示數據時,界面的寬度發生變化,希望顯示的卡片數量也跟隨變化。WrapPanel雖然也可以實現這個功能,但是將多餘的部分都留在行尾,十分不美觀,最好是能夠將多餘的寬度平分在每個ListBoxItem之間,比較美觀,也符合項目需求。如下便是我自己實現的Panel:

  1 using System;
  2 using System.Collections.Generic;
  3 using System.Linq;
  4 using System.Text;
  5 using System.Threading.Tasks;
  6 using System.Windows;
  7 using System.Windows.Controls;
  8 
  9 namespace WpfDemo
 10 {
 11     public class MyWrapPanel : Panel
 12     {
 13         protected override System.Windows.Size MeasureOverride(System.Windows.Size availableSize)
 14         {
 15             Size currentLineSize = new Size();
 16             Size panelSize = new Size();
 17 
 18             foreach (UIElement element in base.InternalChildren)
 19             {
 20                 element.Measure(availableSize);
 21                 Size desiredSize = element.DesiredSize;
 22 
 23                 if (currentLineSize.Width + desiredSize.Width > availableSize.Width)
 24                 {
 25                     panelSize.Width = Math.Max(currentLineSize.Width, panelSize.Width);
 26                     panelSize.Height += currentLineSize.Height;
 27                     currentLineSize = desiredSize;
 28 
 29                     if (desiredSize.Width > availableSize.Width)
 30                     {
 31                         panelSize.Width = Math.Max(desiredSize.Width, panelSize.Width);
 32                         panelSize.Height += desiredSize.Height;
 33                         currentLineSize = new Size();
 34                     }
 35                 }
 36                 else
 37                 {
 38                     currentLineSize.Width += desiredSize.Width;
 39                     currentLineSize.Height = Math.Max(desiredSize.Height, currentLineSize.Height);
 40                 }
 41             }
 42 
 43             panelSize.Width = Math.Max(currentLineSize.Width, panelSize.Width);
 44             panelSize.Height += currentLineSize.Height;
 45 
 46             return panelSize;
 47         }
 48 
 49         protected override System.Windows.Size ArrangeOverride(System.Windows.Size finalSize)
 50         {
 51             int firstInLine = 0;
 52             int lineCount = 0;
 53 
 54             Size currentLineSize = new Size();
 55 
 56             double accumulatedHeight = 0;
 57 
 58             UIElementCollection elements = base.InternalChildren;
 59             double interval = 0.0;
 60             for (int i = 0; i < elements.Count; i++)
 61             {
 62 
 63                 Size desiredSize = elements[i].DesiredSize;
 64 
 65                 if (currentLineSize.Width + desiredSize.Width > finalSize.Width) //need to switch to another line
 66                 {
 67                     interval = (finalSize.Width - currentLineSize.Width) / (i - firstInLine + 2);
 68                     arrangeLine(accumulatedHeight, currentLineSize.Height, firstInLine, i, interval);
 69 
 70                     accumulatedHeight += currentLineSize.Height;
 71                     currentLineSize = desiredSize;
 72 
 73                     if (desiredSize.Width > finalSize.Width) //the element is wider then the constraint - give it a separate line                    
 74                     {
 75                         arrangeLine(accumulatedHeight, desiredSize.Height, i, ++i, 0);
 76                         accumulatedHeight += desiredSize.Height;
 77                         currentLineSize = new Size();
 78                     }
 79                     firstInLine = i;
 80                     lineCount++;
 81                 }
 82                 else //continue to accumulate a line
 83                 {
 84                     currentLineSize.Width += desiredSize.Width;
 85                     currentLineSize.Height = Math.Max(desiredSize.Height, currentLineSize.Height);
 86                 }
 87             }
 88 
 89             if (firstInLine < elements.Count)
 90             {
 91                 if (lineCount == 0)
 92                 {
 93                     interval = (finalSize.Width - currentLineSize.Width) / (elements.Count - firstInLine + 1);
 94                 }
 95                 arrangeLine(accumulatedHeight, currentLineSize.Height, firstInLine, elements.Count, interval);
 96             }
 97                 
 98 
 99             return finalSize;
100         }
101 
102         private void arrangeLine(double y, double lineHeight, int start, int end, double interval)
103         {
104             double x = 0;
105             UIElementCollection children = InternalChildren;
106             for (int i = start; i < end; i++)
107             {
108                 x += interval;
109                 UIElement child = children[i];
110                 child.Arrange(new Rect(x, y, child.DesiredSize.Width, lineHeight));
111                 x += child.DesiredSize.Width;
112             }
113         }
114     }
115 }

接下來,便是將這個MyWrapPanel作為ListBox的ItemsPanelTemplate即可:

 1 <Window x:Class="WpfDemo.MainWindow"
 2 xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
 3         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
 4         xmlns:comm="clr-namespace:WpfDemo.CommonControls;assembly=WpfDemo.CommonControls"
 5         xmlns:local="clr-namespace:WpfDemo"
 6         Title="MainWindow" Height="350" Width="525">
 7     
 8     <Grid>
 9         <ListBox ItemsSource="{Binding DataSource}" ScrollViewer.HorizontalScrollBarVisibility="Disabled"
10                  VerticalAlignment="Center" BorderThickness="0">
11             <ListBox.ItemsPanel>
12                 <ItemsPanelTemplate>
13                     <local:MyWrapPanel IsItemsHost="True"/>
14                 </ItemsPanelTemplate>
15             </ListBox.ItemsPanel>
16             <ListBox.ItemContainerStyle>
17                 <Style TargetType="{x:Type ListBoxItem}">
18                     <Setter Property="Template">
19                         <Setter.Value>
20                             <ControlTemplate TargetType="{x:Type ListBoxItem}">
21                                 <Border HorizontalAlignment="Stretch" VerticalAlignment="Stretch" Background="Green" BorderBrush="Yellow" BorderThickness="1">
22                                     <TextBlock Text="{Binding CameraName}" Width="100" HorizontalAlignment="Center" VerticalAlignment="Center"/>
23                                 </Border>
24                             </ControlTemplate>
25                         </Setter.Value>
26                     </Setter>
27                 </Style>
28             </ListBox.ItemContainerStyle>
29             <ListBox.Style>
30                 <Style TargetType="{x:Type ListBox}">
31                     
32                 </Style>
33             </ListBox.Style>
34         </ListBox>
35     </Grid>
36 </Window>

界面對應的ViewModel:

 1 using System;
 2 using System.Collections.Generic;
 3 using System.Collections.ObjectModel;
 4 using System.Linq;
 5 using System.Text;
 6 using System.Threading.Tasks;
 7 using System.Windows.Threading;
 8 
 9 namespace WpfDemo
10 {
11     public class MainWindowVM : NotifyPropertyBase
12     {
13         private DispatcherTimer timer;
14         public MainWindowVM()
15         {
16             DataSource = new ObservableCollection<WndViewModel>();
17             Colums = 1;
18             for(int i =0; i < 60; ++i)
19             {
20                 var temp = new WndViewModel()
21                 {
22                     CameraName = string.Format("Camera {0}", ++count),
23                 };
24                 DataSource.Add(temp);
25             }
26             //timer = new DispatcherTimer();
27             //timer.Interval = new TimeSpan(0, 0, 1);
28             //timer.Tick += timer_Tick;
29             //timer.Start();
30         }
31 
32         private int count = 0;
33         void timer_Tick(object sender, EventArgs e)
34         {
35             var temp = new WndViewModel()
36             {
37                 CameraName = string.Format("Camera {0}", ++count),
38             };
39             DataSource.Add(temp);
40             Console.WriteLine(temp.CameraName);
41             if (count <= 6)
42             {
43                 Colums = count;
44             }
45             else if (count > 100)
46             {
47                 count = 0;
48                 DataSource.Clear();
49                 Colums = 1;
50             }
51         }
52 
53         private int colums;
54         public int Colums
55         {
56             get { return colums; }
57             set
58             {
59                 SetProperty(ref colums, value);
60             }
61         }
62 
63         private ObservableCollection<WndViewModel> dataSource;
64         public ObservableCollection<WndViewModel> DataSource
65         {
66             get { return dataSource; }
67             set
68             {
69                 SetProperty(ref dataSource, value);
70             }
71         }
72     }
73 }

運行結果:

拉伸後:

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 前言 只有光頭才能變強 2018年8月30日,今天我辭職了。在6月25號入職,到現在也有兩個月時間了。 感受: 第一天是期待的:第一次將項目拉到本地上看的時候,代碼很多,有非常多的模塊,模塊下又有 ,眼花繚亂的。再連上測試庫,也發現有100多張表。~~順著一些模塊看下去,發現用的技術不難,之前基本都 ...
  • 1、環境簡介Visual Studio 2013社區版Halcon18.052、使用Nuget在VS工程中安裝Halcon插件搜索欄輸入關鍵字halcon,出現兩個插件,分別是halcon語言的介面和halcon引擎,全部安裝即可。需要註意的是,這兩個控制項的使用的前提是你的電腦上已經安裝了Halco... ...
  • 去年就開始學習採用Docker+Jenkins+.Net Core搭建生成式流水線,一直拖到現在,也沒有徹底的好好靜下來去總結總結。趁著現在對自己的嚴格要求下,逐漸開始重視自我總結,以此來鞏固逐漸失去的知識。 本文地址:https://www.cnblogs.com/CKExp/p/9536864. ...
  • 首先瞭解一下 RFC4646 和 BCP 47 是什麼東西: "RFC4646" The name is a combination of an ISO 639 two letter lowercase culture code associated with a language and an I ...
  • 背景 由於最近公司要做微信小程式聊天,所以.NetFramwork版本的SignalR版本的不能用了。因為小程式里沒有windows對象,導致JQuery無法使用。而Signalr的 js客戶端是依賴JQuery的。 所以看下了Core版本的SignarlR,經過測試,發現可以在微信中運行,不過要將 ...
  • Ocelot是為.net core量身定做的,目前是基於 netstandard2.0進行構建的。 .NET Core 2.1中如何使用呢? 安裝NuGet package 使用nuget安裝Ocelot及其依賴項。您需要創建一個netstandard2.0項目並將其Package安裝到項目中。然後 ...
  • App.xaml.cs中的代碼每次都差不多,故特地將其整理出來直接復用: ...
  • 問題: 需要讓程式(以非同步方式)等待一段時間。 解決方案:Task類的靜態函數Delay,返回Task對象 在github開源項目 ,找到Task.cs有關Delay方法的源碼 github地址: "https://github.com/dotnet/coreclr/blob/master/src/ ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...