3

UWP アプリケーションで画像のサイズを変更しようとしています。ほとんどの場合、追加されたコードは機能しますawait encoder.FlushAsync();が、ArgumentException.

MSDN ( https://msdn.microsoft.com/en-us/library/windows/apps/windows.graphics.imaging.bitmapencoder.bitmaptransform.aspx ) にアクセスしたところ、(「備考」で) 教えてくれました。 :

BitmapTransform メンバーを使用して、インデックス付きのピクセル形式で格納された画像をスケーリングしようとすると、FlushAsync は HRESULT WINCODEC_ERR_INVALIDPARAMETER で失敗します。代わりに、GetPixelDataAsync を使用してスケーリングされたピクセル データを取得し、SetPixelData を使用してエンコーダーに設定する必要があります。

私はそれをやろうとしました.2つのコメント行を見てください(繰り返しのために私にはどういうわけか間違っているように見えます). 2行目(私がしようとしているところSetPixelData)で、エンコーダーは私にbuffer allocated not sufficient例外を与えます。

var decoder = await BitmapDecoder.CreateAsync(streamToReadFrom.AsStream().AsRandomAccessStream());
if (decoder.OrientedPixelHeight > height ||
    decoder.OrientedPixelWidth > width)
{
    var resizedStream = new InMemoryRandomAccessStream();
    BitmapEncoder encoder = await BitmapEncoder.CreateForTranscodingAsync(resizedStream, decoder);

    encoder.BitmapTransform.InterpolationMode = BitmapInterpolationMode.Fant;
    encoder.BitmapTransform.ScaledHeight = newHeight;
    encoder.BitmapTransform.ScaledWidth = newWidth;

    //"buffer allocated not sufficient"
    // var pd = await decoder.GetPixelDataAsync(BitmapPixelFormat.Rgba16, BitmapAlphaMode.Ignore,
    //             encoder.BitmapTransform, ExifOrientationMode.IgnoreExifOrientation, ColorManagementMode.DoNotColorManage);
    // encoder.SetPixelData(BitmapPixelFormat.Rgba16, BitmapAlphaMode.Ignore,
    //             decoder.OrientedPixelWidth, decoder.OrientedPixelHeight, decoder.DpiX, decoder.DpiY, pd.DetachPixelData());

    // write out to the stream
    // might fail cause https://msdn.microsoft.com/en-us/library/windows/apps/windows.graphics.imaging.bitmapencoder.bitmaptransform.aspx
    await encoder.FlushAsync();

    // Read out resizedStream and return
}

この問題を引き起こす画像の例: http://www.spiegel.de/images/image-1028227-hppano-lqbn.jpg . 単体テストはこちら: https://github.com/famoser/OfflineMedia/blob/master/Famoser.OfflineMedia.UnitTests/Presentation/ImageResizeTest.cs

どうすれば回避できArgumentExceptionますか?画像が「インデックス付きピクセル形式」であることを確認するにはどうすればよいですか? また、この形式のサイズを変更するにはどうすればよいですか?

4

1 に答える 1

1

2 行目 (SetPixelData を試行する場所) で、エンコーダーは十分な例外が割り当てられていないバッファーで報酬を与えます。

これは、 の場合SetPixelData、ピクセル データが から一致しないためGetPixelDataAsyncです。たとえば、次のようにコーディングできます。

if (file != null)
{
    BitmapImage bmp = new BitmapImage();
    using(var imageStream = await file.OpenAsync(Windows.Storage.FileAccessMode.ReadWrite))
    {
        BitmapDecoder decoder = await BitmapDecoder.CreateAsync(imageStream);
        InMemoryRandomAccessStream pixelras = new InMemoryRandomAccessStream();
        BitmapEncoder pixelencoder = await BitmapEncoder.CreateAsync(BitmapEncoder.JpegEncoderId, pixelras);
        BitmapTransform transform = new BitmapTransform();
        transform.InterpolationMode = BitmapInterpolationMode.Fant;
        transform.ScaledHeight = 400;
        transform.ScaledWidth = 400;
        var provider = await decoder.GetPixelDataAsync(BitmapPixelFormat.Bgra8, 
            BitmapAlphaMode.Ignore, 
            transform, 
            ExifOrientationMode.RespectExifOrientation, 
            ColorManagementMode.DoNotColorManage);
        var pixels = provider.DetachPixelData();
        pixelencoder.SetPixelData(BitmapPixelFormat.Bgra8, BitmapAlphaMode.Ignore, 400, 
            400, decoder.DpiX, decoder.DpiY, pixels);

        try
        {
            await pixelencoder.FlushAsync();
        }
        catch(Exception ex)
        {

        }
        bmp.SetSource(pixelras);
        img.Source = bmp;                  
    }
}

画像が「インデックス付きピクセル形式」であることを確認するにはどうすればよいですか? また、この形式のサイズを変更するにはどうすればよいですか?

インデックス付きピクセル形式の画像を検出する効果的な方法は見つかりませんでしたが、

BitmapTransform メンバーを使用して、インデックス付きピクセル形式で格納された画像をスケーリングしようとすると、FlushAsync は HRESULT WINCODEC_ERR_INVALIDPARAMETER で失敗します。代わりに、GetPixelDataAsync を使用してスケーリングされたピクセル データを取得し、SetPixelData を使用してエンコーダーに設定する必要があります。

例外をキャッチして再度使用する方法SetPixelDataです。たとえば、次のようになります。

if (file != null)
{
    BitmapImage bmp = new BitmapImage();
    using(var imageStream = await file.OpenAsync(Windows.Storage.FileAccessMode.ReadWrite))
    {
        BitmapDecoder decoder = await BitmapDecoder.CreateAsync(imageStream);
        InMemoryRandomAccessStream ras = new InMemoryRandomAccessStream();
        BitmapEncoder encoder = await BitmapEncoder.CreateForTranscodingAsync(ras, decoder);

        encoder.BitmapTransform.InterpolationMode = BitmapInterpolationMode.Fant;
        encoder.BitmapTransform.ScaledHeight = 400;
        encoder.BitmapTransform.ScaledWidth = 400;

        try
        {
            await encoder.FlushAsync();
            bmp.SetSource(ras);
        }
        catch (Exception ex)
        {
            if (ex.HResult.ToString() == "WINCODEC_ERR_INVALIDPARAMETER")
            {
                InMemoryRandomAccessStream pixelras = new InMemoryRandomAccessStream();
                BitmapEncoder pixelencoder = await BitmapEncoder.CreateAsync(BitmapEncoder.JpegEncoderId, pixelras)
                BitmapTransform transform = new BitmapTransform();
                transform.InterpolationMode = BitmapInterpolationMode.Fant;
                transform.ScaledHeight = 400;
                transform.ScaledWidth = 400;
                var provider = await decoder.GetPixelDataAsync(BitmapPixelFormat.Bgra8,
                    BitmapAlphaMode.Ignore,
                    transform,
                    ExifOrientationMode.RespectExifOrientation,
                    ColorManagementMode.DoNotColorManage);
                var pixels = provider.DetachPixelData();
                pixelencoder.SetPixelData(BitmapPixelFormat.Bgra8, BitmapAlphaMode.Ignore, 400,
                    400, decoder.DpiX, decoder.DpiY, pixels);
                try
                {
                    await pixelencoder.FlushAsync();
                    bmp.SetSource(pixelras);
                }
                catch
                {

                }
            }
        }                    
        img.Source = bmp;                  
    }
}
于 2016-07-28T10:06:09.140 に答える