Сохранение imageRef от GLPaint создает полностью черное изображение - PullRequest
2 голосов
/ 25 марта 2012

Привет. Я пробую приложение для рисования и у меня возникла проблема с сохранением нарисованного изображения. Сейчас я очень рано изучаю это, но я добавил код из: Как получить UIImage из EAGLView? , чтобы сохранить нарисованное изображение.

Я создал новое приложение, затем отобразил созданный мной viewController. В IB я добавил представление, которое представляет собой PaintingView, и imageView лежит за ним.

Единственная модификация, которую я до сих пор сделал для PaintingView, - это изменить фон для очистки и установить фон для очистки, чтобы я мог отобразить изображение позади него. Рисунок прекрасно работает, моя единственная проблема - сохранение.

- (void)saveImageFromGLView:(UIView *)glView {

    if(glCheckFramebufferStatusOES(GL_FRAMEBUFFER_OES) != GL_FRAMEBUFFER_COMPLETE_OES)
        {
            /// This IS being activated with code 0
            NSLog(@"failed to make complete framebuffer object %x", glCheckFramebufferStatusOES(GL_FRAMEBUFFER_OES));
    }


    int s = 1;
    UIScreen* screen = [ UIScreen mainScreen ];
    if ( [ screen respondsToSelector:@selector(scale) ] )
        s = (int) [ screen scale ];

    const int w = self.frame.size.width;
    const int h = self.frame.size.height;
    const NSInteger myDataLength = w * h * 4 * s * s;
    // allocate array and read pixels into it.
    GLubyte *buffer = (GLubyte *) malloc(myDataLength);
    glReadPixels(0, 0, w*s, h*s, GL_RGBA, GL_UNSIGNED_BYTE, buffer);
    // gl renders "upside down" so swap top to bottom into new array.
    // there's gotta be a better way, but this works.
    GLubyte *buffer2 = (GLubyte *) malloc(myDataLength);
    for(int y = 0; y < h*s; y++)
    {
        memcpy( buffer2 + (h*s - 1 - y) * w * 4 * s, buffer + (y * 4 * w * s), w * 4 * s );
    }
    free(buffer); // work with the flipped buffer, so get rid of the original one.

    // make data provider with data.
    CGDataProviderRef provider = CGDataProviderCreateWithData(NULL, buffer2, myDataLength, NULL);
    // prep the ingredients
    int bitsPerComponent = 8;
    int bitsPerPixel = 32;
    int bytesPerRow = 4 * w * s;
    CGColorSpaceRef colorSpaceRef = CGColorSpaceCreateDeviceRGB();
    CGBitmapInfo bitmapInfo = kCGBitmapByteOrderDefault;
    CGColorRenderingIntent renderingIntent = kCGRenderingIntentDefault;
    // make the cgimage
    CGImageRef imageRef = CGImageCreate(w*s, h*s, bitsPerComponent, bitsPerPixel, bytesPerRow, colorSpaceRef, bitmapInfo, provider, NULL, NO, renderingIntent);
    // then make the uiimage from that
    UIImage *myImage = [ UIImage imageWithCGImage:imageRef scale:s orientation:UIImageOrientationUp ];
    UIImageWriteToSavedPhotosAlbum( myImage, nil, nil, nil );
    CGImageRelease( imageRef );
    CGDataProviderRelease(provider);
    CGColorSpaceRelease(colorSpaceRef);
    free(buffer2);    

}

Добавление приведенного выше кода в пример приложения работает нормально, проблема заключается в том, чтобы сделать это в моем новом приложении. Единственное различие, которое я могу сказать, - то, что я не включил PaintingWindow - это было бы проблемой?

Как будто метод saveImage не видит данные для чертежей.

Ответы [ 3 ]

4 голосов
/ 27 марта 2012

Метод save должен вызываться в контексте контекста OpenGL.Чтобы решить эту проблему, вы можете переместить ваш метод в тот же файл рендеринга .m и вызвать эту функцию извне.

Также вам необходимо рассмотреть OpenGL прозрачного цвета.

(подробное объяснение в комментариях, смеется)

1 голос
/ 17 октября 2012

Я обнаружил, что изменение CGBitmapInfo на:

CGBitmapInfo bitmapInfo = kCGImageAlphaPremultipliedLast;

Результат на прозрачном фоне.

0 голосов
/ 19 апреля 2015

Ах, ты должен сделать это в начале.

[EAGLContext setCurrentContext:drawContext];
...