如何显示图片目录中的图像

本文关键字:图像 显示图 | 更新日期: 2023-09-27 17:54:26

我想在图片库中显示图片。我获取图片并绑定数据。

StorageFolder picturesFolder = KnownFolders.PicturesLibrary;
IReadOnlyList<StorageFile> myPictures = await picturesFolder.GetFilesAsync();
var mydata = from file in myPictures select new { Subtitle = "subtitle", Title = "title", Image = this.getImage(file.Path) };
this.DefaultViewModel["Items"] = mydata;

这是getImage()用于设置BitmapImage。

private async Task<BitmapImage> getImage(string finename)
{
    StorageFolder picturesFolder = KnownFolders.PicturesLibrary;
    StorageFile file = await picturesFolder.GetFileAsync(fileName);
    var stream = await file.OpenReadAsync();
    var bitmap = new BitmapImage();
    bitmap.SetSource(stream);
    return bitmap;
 }

但是没有显示图片。我认为这是因为异步函数,但我不知道解决方案。

如何显示图片目录中的图像

我不确定你如何使用设置为DefaultViewModel的数据,但是,是的,看起来async方法是你的问题。

您需要做的是以某种方式await每个呼叫getImage()。这样做的一种方法是在select中使用async lambda。但要做到这一点,您需要使用方法语法。

当您这样做时,您将拥有IEnumerable<Task<a>>(其中a是您的匿名类型),但您只需要IEnumerable<a>。要得到它,使用Task.WhenAll()(它将返回Task<a[]>),然后使用await它的结果:

var tasks = myPictures.Select(
    async file => new { Subtitle = "subtitle", Title = "title", Image = await getImage(file.Path) });
var data = await Task.WhenAll(tasks);

这将一次执行所有getImage(),这可能不是最有效的解决方案。

svick的解决方案似乎应该工作,但正如他/她所说-这可能不是最有效的解决方案。对于文件数量未知的文件夹,更好的解决方案是使用FileInformationFactory.GetVirtualizedFilesVector()的数据虚拟化。这与转换器一起使用效果最好。

我曾经用过的东西:

获取虚拟文件列表并绑定到ListView

private async void GetPicturesFromGalleryFolder()
{
    var queryOptions = new QueryOptions();
    queryOptions.FolderDepth = FolderDepth.Shallow;
    queryOptions.IndexerOption = IndexerOption.UseIndexerWhenAvailable;
    queryOptions.SortOrder.Clear();
    var sortEntry = new SortEntry {PropertyName = "System.DateModified", AscendingOrder = false};
    queryOptions.SortOrder.Add(sortEntry);
    queryOptions.FileTypeFilter.Add(".png");
    var fileQuery = KnownFolders.PicturesLibrary.CreateFileQueryWithOptions(queryOptions);
    var fileInformationFactory =
        new FileInformationFactory(
            fileQuery,
            ThumbnailMode.PicturesView,
            0,
            ThumbnailOptions.None,
            true);
    MyListView.ItemsSource = fileInformationFactory.GetVirtualizedFilesVector();
}

XAML

<ListView.ItemTemplate>
    <DataTemplate>
        <Image
            Source="{Binding Converter={StaticResource converters:IStorageItemInformationToBitmapImageConverter}"/>
    </DataTemplate>
</ListView.ItemTemplate>

IStorageItemInformationToBitmapImageConverter

public class IStorageItemInformationToBitmapImageConverter : IValueConverter
{
    public object Convert(object value, Type targetType, object parameter, string language)
    {
        var fileInfo = value as FileInformation;
        if (fileInfo != null)
        {
            var bi = new BitmapImage();
            // The file is being opened asynchronously but we return the BitmapImage immediately.
            SetSourceAsync(bi, fileInfo);
            return bi;
        }
        return null;
    }
    private async void SetSourceAsync(BitmapImage bi, FileInformation fi)
    {
        try
        {
            using (var stream = await fi.OpenReadAsync())
            {
                await bi.SetSourceAsync(stream);
            }
        }
        catch
        {
            // ignore failure
        }
    }
    public object ConvertBack(object value, Type targetType, object parameter, string language)
    {
        return null;
    }
}