欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

基于WPF+Prism實現(xiàn)動態(tài)分頁加載圖片的方法

 更新時間:2025年09月22日 09:15:21   作者:code bean  
在做圖片瀏覽器程序時,遇到圖片數(shù)量巨大的情況(如幾百張、上千張),一次性加載所有圖片會導(dǎo)致界面卡頓甚至程序崩潰,所以本文介紹了基于WPF?+?Prism實現(xiàn)動態(tài)分頁加載圖片的方法,需要的朋友可以參考下

引言

在做圖片瀏覽器程序時,遇到圖片數(shù)量巨大的情況(如幾百張、上千張),一次性加載所有圖片會導(dǎo)致界面卡頓甚至程序崩潰。

本文介紹一種 WPF + Prism 實現(xiàn)動態(tài)分頁加載圖片的方法,結(jié)合 ScrollViewer 滾動條觸底檢測,實現(xiàn) “邊滾動,邊加載” 的流暢體驗。

1. 界面設(shè)計:4×4 顯示 + 滾動條

我們希望界面每次顯示 4×4,共 16 張圖片,每張圖片帶有邊框。

XAML示例

<ScrollViewer VerticalScrollBarVisibility="Auto" ScrollChanged="ScrollViewer_ScrollChanged">
    <ItemsControl ItemsSource="{Binding Images}">
        <ItemsControl.ItemsPanel>
            <ItemsPanelTemplate>
                <UniformGrid Columns="4"/>
            </ItemsPanelTemplate>
        </ItemsControl.ItemsPanel>
        <ItemsControl.ItemTemplate>
            <DataTemplate>
                <Border BorderBrush="Gray" BorderThickness="1" Margin="5">
                    <Image Source="{Binding}" Stretch="Uniform"/>
                </Border>
            </DataTemplate>
        </ItemsControl.ItemTemplate>
    </ItemsControl>
</ScrollViewer>
  • ScrollViewer 包裹 ItemsControl,開啟垂直滾動。
  • 使用 UniformGrid 布局,4列均勻分布。
  • 每張圖片用 Border 包一層,美觀且清晰分隔。

2. 后臺邏輯:ViewModel 分頁加載

ViewModel 負責管理圖片列表和分頁邏輯。

public class ImageBrowserViewModel : BindableBase
{
    private const int PageSize = 16;
    private readonly List<string> allImagePaths = new();
    public ObservableCollection<BitmapImage> Images { get; } = new();

    private int currentPage = 0;
    public bool IsLoading { get; private set; }

    public void LoadAllImagePaths(string folderPath)
    {
        allImagePaths.Clear();
        var extensions = new[] { ".jpg", ".png", ".bmp" };
        var files = Directory.GetFiles(folderPath)
            .Where(f => extensions.Contains(Path.GetExtension(f).ToLower()))
            .ToList();
        allImagePaths.AddRange(files);
        currentPage = 0;
        Images.Clear();
    }

    public async Task LoadNextPageAsync()
    {
        if (IsLoading) return;
        IsLoading = true;

        var nextImages = allImagePaths
            .Skip(currentPage * PageSize)
            .Take(PageSize)
            .ToList();

        foreach (var path in nextImages)
        {
            await Task.Run(() =>
            {
                var bitmap = new BitmapImage();
                bitmap.BeginInit();
                bitmap.CacheOption = BitmapCacheOption.OnLoad;
                bitmap.UriSource = new Uri(path);
                bitmap.EndInit();
                bitmap.Freeze();

                App.Current.Dispatcher.Invoke(() => Images.Add(bitmap));
            });
        }

        currentPage++;
        IsLoading = false;
    }
}
  • LoadAllImagePaths:一次性記錄所有圖片路徑,但不立刻加載圖片內(nèi)容。
  • LoadNextPageAsync:每次按頁加載圖片,使用 Task.Run + Dispatcher.Invoke,避免界面卡頓。

這段代碼的作用是從 allImagePaths 列表中取出當前頁面所需的圖片路徑,并將它們存儲到 nextImages 列表中。以下是逐行解釋:

代碼片段

var nextImages = allImagePaths
    .Skip(currentPage * PageSize)
    .Take(PageSize)
    .ToList();

逐行解釋

var nextImages:

  • 聲明一個變量 nextImages,用于存儲當前頁面所需的圖片路徑列表。

allImagePaths:

  • 這是一個包含所有圖片路徑的列表,類型為 List<string>。

.Skip(currentPage * PageSize):

  • currentPage 是當前頁面的索引,從 0 開始。
  • PageSize 是每頁顯示的圖片數(shù)量,這里定義為 16。
  • currentPage * PageSize 計算出當前頁面之前的所有圖片路徑的數(shù)量。
  • .Skip(currentPage * PageSize) 跳過這些路徑,即跳過當前頁面之前的所有圖片路徑。

.Take(PageSize):

  • 從跳過后的路徑中取出 PageSize 個路徑,即取出當前頁面所需的圖片路徑。

.ToList():

  • 將取出的路徑轉(zhuǎn)換為一個新的列表,并賦值給 nextImages。

總結(jié)

這段代碼的作用是從 allImagePaths 中取出當前頁面所需的圖片路徑,并將它們存儲到 nextImages 列表中。具體來說,它跳過了當前頁面之前的所有圖片路徑,然后取出當前頁面所需的圖片路徑數(shù)量(PageSize 個),并將這些路徑存儲到 nextImages 列表中。

3. 滾動到底時加載新圖片

ScrollViewerScrollChanged 事件中,檢測是否接近底部,如果是則請求 ViewModel 加載下一頁:

private async void ScrollViewer_ScrollChanged(object sender, ScrollChangedEventArgs e)
{
    var scrollViewer = (ScrollViewer)sender;

    if (scrollViewer.VerticalOffset + scrollViewer.ViewportHeight 
        >= scrollViewer.ExtentHeight - 50) // 接近底部50像素
    {
        if (DataContext is ImageBrowserViewModel vm && !vm.IsLoading)
        {
            await vm.LoadNextPageAsync();
        }
    }
}

ExtentHeight 是總高度,ViewportHeight 是當前可視區(qū)域高度,VerticalOffset 是當前滾動位置。

當滾動接近底部 50px 內(nèi),就觸發(fā)加載。

4. 總結(jié)

通過以上方法,我們實現(xiàn)了:

  • 初始只加載少量圖片,快速打開界面。
  • 用戶滾動時,按需分頁加載后續(xù)圖片。
  • 界面不卡頓,體驗絲滑流暢。

這種設(shè)計特別適合處理大量圖片瀏覽、視頻幀查看、縮略圖管理器等場景。

到此這篇關(guān)于基于WPF+Prism實現(xiàn)動態(tài)分頁加載圖片的方法的文章就介紹到這了,更多相關(guān)WPF Prism動態(tài)分頁加載圖片內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C#實現(xiàn)三元組的使用示例

    C#實現(xiàn)三元組的使用示例

    本文介紹了C#中的三元組數(shù)據(jù)結(jié)構(gòu),以及如何使用三元組在C#中進行一些特定的計算,具有一定的參考價值,感興趣的可以了解一下
    2023-11-11
  • C#創(chuàng)建及讀取DAT文件操作

    C#創(chuàng)建及讀取DAT文件操作

    這篇文章主要介紹了C#創(chuàng)建及讀取DAT文件操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • C# WinForm編程獲取文件物理路徑的方法

    C# WinForm編程獲取文件物理路徑的方法

    這篇文章主要介紹了C# inForm編程獲取文件物理路徑的方法,獲取的物理路徑是軟件即軟件安裝所在目錄,需要的朋友可以參考下
    2014-08-08
  • 下載軟件后使用c#獲取文件的md5碼示例

    下載軟件后使用c#獲取文件的md5碼示例

    這篇文章主要介紹了下載軟件后使用c#獲取文件的md5碼示例,需要的朋友可以參考下
    2014-05-05
  • C#使用Pipelines實現(xiàn)處理Socket數(shù)據(jù)包

    C#使用Pipelines實現(xiàn)處理Socket數(shù)據(jù)包

    這篇文章主要為大家詳細介紹了C#如何使用Pipelines實現(xiàn)處理Socket數(shù)據(jù)包,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2023-12-12
  • 基于WPF實現(xiàn)驗證碼控件

    基于WPF實現(xiàn)驗證碼控件

    這篇文章主要介紹了如何利用WPF實現(xiàn)一個簡單的驗證碼控件,文中的示例代碼講解詳細,對我們學(xué)習(xí)或工作有一定幫助,需要的可以參考一下
    2022-08-08
  • C#面向?qū)ο缶幊讨幸蕾嚪崔D(zhuǎn)原則的示例詳解

    C#面向?qū)ο缶幊讨幸蕾嚪崔D(zhuǎn)原則的示例詳解

    在面向?qū)ο缶幊讨校琒OLID?是五個設(shè)計原則的首字母縮寫,旨在使軟件設(shè)計更易于理解、靈活和可維護。本文將通過實例詳細講講C#面向?qū)ο缶幊讨幸蕾嚪崔D(zhuǎn)原則,需要的可以參考一下
    2022-07-07
  • C#多線程之線程控制詳解

    C#多線程之線程控制詳解

    這篇文章主要為大家詳細介紹了C#多線程之線程控制的相關(guān)資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-08-08
  • C# SQlite操作方法小結(jié)

    C# SQlite操作方法小結(jié)

    這篇文章主要介紹了C# SQlite操作方法,較為詳細的講述了SQLite的下載與C#針對SQLite的連接、創(chuàng)建、與執(zhí)行SQL語句等操作相關(guān)技巧,需要的朋友可以參考下
    2016-07-07
  • C#實現(xiàn)飛行棋(Winform)

    C#實現(xiàn)飛行棋(Winform)

    這篇文章主要為大家詳細介紹了基于Winform框架的飛行棋,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-02-02

最新評論