Конвертировать объект Leptonica Pix в QPixmap (или другой объект изображения) - PullRequest
0 голосов
/ 28 декабря 2011

Я использую библиотеку Лептоники для обработки некоторых изображений.После этого я хочу показать их в моем графическом интерфейсе QT.Leptonica использует собственный формат Pix для изображений, в то время как QT использует собственный формат QPixmap.На данный момент единственный способ для меня - сохранить изображения после обработки в виде файла (например, BMP), а затем снова загрузить их с помощью вызова функции QT.Теперь я хочу преобразовать их в свой код, поэтому мне не нужно идти в обход, сохраняя их в файловой системе.Есть идеи, как это сделать?

С наилучшими пожеланиями

// edit:

Хорошо, как уже предлагалось, я пытался преобразовать PIX * в QImage.PIX * определяется следующим образом:

http://tpgit.github.com/Leptonica/pix_8h_source.html

struct Pix
{
  l_uint32             w;           /* width in pixels                   */
  l_uint32             h;           /* height in pixels                  */
  l_uint32             d;           /* depth in bits                     */
  l_uint32             wpl;         /* 32-bit words/line                 */
  l_uint32             refcount;    /* reference count (1 if no clones)  */
  l_int32              xres;        /* image res (ppi) in x direction    */
                                    /* (use 0 if unknown)                */
  l_int32              yres;        /* image res (ppi) in y direction    */
                                    /* (use 0 if unknown)                */
  l_int32              informat;    /* input file format, IFF_*          */
  char                *text;        /* text string associated with pix   */
  struct PixColormap  *colormap;    /* colormap (may be null)            */
  l_uint32            *data;        /* the image data                    */
};

, тогда как QImage предлагает мне такой метод:

http://developer.qt.nokia.com/doc/qt-4.8/qimage.html#QImage-7

QImage ( const uchar * data, 
    int width, 
    int height, 
    int bytesPerLine, 
    Format format )

Я предполагаю, что не могу просто скопировать данные из PIX в QImage при вызове конструктора.Я думаю, мне нужно заполнить QImage Pixel Pixel, но на самом деле я не знаю, как?Нужно ли перебирать все координаты?Как мне оценить битовую глубину?Есть идеи?

Ответы [ 5 ]

6 голосов
/ 05 апреля 2012

Я использую это для преобразования QImage в PIX:

PIX* TessTools::qImage2PIX(QImage& qImage) {
  PIX * pixs;
  l_uint32 *lines;

  qImage = qImage.rgbSwapped();
  int width = qImage.width();
  int height = qImage.height();
  int depth = qImage.depth();
  int wpl = qImage.bytesPerLine() / 4;

  pixs = pixCreate(width, height, depth);
  pixSetWpl(pixs, wpl);
  pixSetColormap(pixs, NULL);
  l_uint32 *datas = pixs->data;

  for (int y = 0; y < height; y++) {
    lines = datas + y * wpl;
    QByteArray a((const char*)qImage.scanLine(y), qImage.bytesPerLine());
    for (int j = 0; j < a.size(); j++) {
      *((l_uint8 *)lines + j) = a[j];
    }
  }
  return pixEndianByteSwapNew(pixs);
}

И это для преобразования PIX в QImage:

QImage TessTools::PIX2QImage(PIX *pixImage) {
  int width = pixGetWidth(pixImage);
  int height = pixGetHeight(pixImage);
  int depth = pixGetDepth(pixImage);
  int bytesPerLine = pixGetWpl(pixImage) * 4;
  l_uint32 * s_data = pixGetData(pixEndianByteSwapNew(pixImage));

  QImage::Format format;
  if (depth == 1)
    format = QImage::Format_Mono;
  else if (depth == 8)
    format = QImage::Format_Indexed8;
  else
    format = QImage::Format_RGB32;

  QImage result((uchar*)s_data, width, height, bytesPerLine, format);

  // Handle pallete
  QVector<QRgb> _bwCT;
  _bwCT.append(qRgb(255,255,255));
  _bwCT.append(qRgb(0,0,0));

  QVector<QRgb> _grayscaleCT(256);
  for (int i = 0; i < 256; i++)  {
    _grayscaleCT.append(qRgb(i, i, i));
  }
  if (depth == 1) {
    result.setColorTable(_bwCT);
  }  else if (depth == 8)  {
    result.setColorTable(_grayscaleCT);

  } else {
    result.setColorTable(_grayscaleCT);
  }

  if (result.isNull()) {
    static QImage none(0,0,QImage::Format_Invalid);
    qDebug() << "***Invalid format!!!";
    return none;
  }

  return result.rgbSwapped();
}
1 голос
/ 28 декабря 2011

Я не знаю библиотеку Лептоники, но я кратко изучил документацию и нашел документацию о структуре PIX . Вы можете создать QImage из необработанных данных и преобразовать его в QPixmap с помощью convertFromImage .

0 голосов
/ 27 ноября 2016

Этот код принимает параметр const QImage&.

static PIX* makePIXFromQImage(const QImage &image)
{
 QByteArray ba;
 QBuffer buf(&ba);
 buf.open(QIODevice::WriteOnly);
 image.save(&buf, "BMP");
 return pixReadMemBmp(ba.constData(), ba.size());
}
0 голосов
/ 23 мая 2014

Вы можете сохранить свое растровое изображение в ОЗУ вместо файла (используйте QByteArray для хранения данных и QBuffer в качестве устройства ввода-вывода).

0 голосов
/ 29 декабря 2011

Ну, я мог бы решить проблему следующим образом:

Leptonica предлагает функцию

l_int32     pixWriteMemBmp (l_uint8 **pdata, size_t *psize, PIX *pix)

С помощью этой функции вы можете записывать в память вместо файлового потока.Тем не менее (в этом примере) заголовок и формат Bmp сохраняются (есть и те же функции для других форматов изображений).

Соответствующая функция из QT такая:

bool QImage::loadFromData ( const uchar * data, int len, const char * format = 0 )

Так какЗаголовок сохраняется, мне просто нужно передать данные ptr и размер в функцию loadFromData, а QT сделает все остальное.

Так что все вместе это будет выглядеть так:

PIX *m_pix;
FILE * pFile;
pFile = fopen( "PathToFile", "r" );
m_pix = pixReadStreamBmp(pFile); // If other file format use the according function
fclose(pFile);
// Now we have a Pix object from leptonica

l_uint8* ptr_memory;
size_t len;
pixWriteMemBmp(&ptr_memory, &size, m_pix);
// Now we have the picture somewhere in the memory

QImage testimage;
QPixmap pixmap;
testimage.loadFromData((uchar *)ptr_memory,len);
pixmap.convertFromImage(testimage);
// Now we have the image as a pixmap in Qt

Этона самом деле работает для меня, хотя я не знаю, есть ли способ сделать это задом наперед так легко.(Если есть, пожалуйста, дайте мне знать)

С наилучшими пожеланиями

...