Инициализация BitmapSource с массивом пикселей - PullRequest
2 голосов
/ 06 марта 2012

Я пытаюсь упорядочить собственную структуру данных изображения в BitmapSource.

Вот собственная структура:

struct Bitmap_8Bit
        {
        public:
            unsigned char* data;
            int stride;
            int rows;
            int cols;
            int nChannels; //either 1 or 3. in case of three the order of the channels is BGR
        };

А вот моя функция маршалинга:

template<>
        inline
            System::Windows::Media::Imaging::BitmapSource^ marshal_as<System::Windows::Media::Imaging::BitmapSource^, Bitmap_8Bit>(const Bitmap_8Bit& nativeBitmap)
        {
            System::Windows::Media::PixelFormat format = System::Windows::Media::PixelFormats::Default;
            System::Windows::Media::Imaging::BitmapPalette^ palette = nullptr;
            if (nativeBitmap.nChannels == 1)
            {
                format = System::Windows::Media::PixelFormats::Gray8;
                palette = System::Windows::Media::Imaging::BitmapPalettes::Gray256;
            }
            else if (nativeBitmap.nChannels == 3)
            {
                format = System::Windows::Media::PixelFormats::Bgr24;
                palette = System::Windows::Media::Imaging::BitmapPalettes::Halftone256;
            }
            else
            {
                throw gcnew System::InvalidOperationException("Unsupported number of channels " + nativeBitmap.nChannels);
            }

            //copy data
            int pixelDataLength = nativeBitmap.rows*nativeBitmap.stride;
            System::IntPtr source = System::IntPtr(nativeBitmap.data);

            cli::array<unsigned char>^ buffer = gcnew cli::array<unsigned char>(pixelDataLength);
            System::Runtime::InteropServices::Marshal::Copy(source, buffer, 0, pixelDataLength);

             System::Windows::Media::Imaging::BitmapSource^ managedBitmap = 
                 System::Windows::Media::Imaging::BitmapSource::Create(nativeBitmap.cols, nativeBitmap.rows, 
                 96, 96, 
                 format, palette, 
                 buffer, nativeBitmap.stride);

            return managedBitmap;
        }

Я тестирую свою функцию в простом одноканальном случае:

BSII::IP::Bitmap_8Bit native;
native.cols = 8;
native.rows = 4;
native.nChannels = 1;
native.stride = 8;
int dataLength = native.stride * native.rows;
native.data = new unsigned char[dataLength];

unsigned char gray = 0;
for (int i = 0; i < native.rows; ++i)
{
    for (int j = 0; j < native.cols; ++j)
    {
        native.data[native.stride*i + j] = gray;
        gray += 10;
    }
}

System::Windows::Media::Imaging::BitmapSource^ managed = msclr::interop::marshal_as<System::Windows::Media::Imaging::BitmapSource^>(native);

System::IO::FileStream^ stream = gcnew System::IO::FileStream("c:\\workspace\\temp\\managed.bmp", System::IO::FileMode::Create);
System::Windows::Media::Imaging::BmpBitmapEncoder^ encoder = gcnew System::Windows::Media::Imaging::BmpBitmapEncoder();
System::Windows::Media::Imaging::BitmapFrame^ bitmapFrame = System::Windows::Media::Imaging::BitmapFrame::Create(managed);
encoder->Frames->Add(bitmapFrame);
encoder->Save(stream);
stream->Close();

Это не работает.Когда я отлаживаю его, я вижу, что буферный массив имеет правильные значения внутри, но когда я сохраняю BitmapSource в изображение, он не выглядит так, как должен.

Я также пытался использовать nullptr для палитры вместо Grey256 или Halftone256, и он тоже не работал.

Я думаю, что что-то упущено при использовании BitmapSource.Есть идеи?

Спасибо, Дина

1 Ответ

0 голосов
/ 06 марта 2012

Используйте PngBitmapEncoder вместо BmpBitmapEncoder. BmpBitmapEncoder деформирует изображение. С PngBitmapEncoder пример работает нормально.

Также по соображениям производительности попробуйте использовать:

BitmapSource.Create(Int32, Int32, Double, Double, PixelFormat, BitmapPalette, 
    IntPtr, Int32, Int32) 

вместо

BitmapSource.Create(Int32, Int32, Double, Double, PixelFormat, BitmapPalette, 
    Array, Int32)
...