windows – OpenGL – 直接渲染到位图

前端之家收集整理的这篇文章主要介绍了windows – OpenGL – 直接渲染到位图前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
我正在制作具有一堆小窗口和控件的应用程序(所以它是2D渲染),我想将每个窗口和控件渲染到它自己的位图(没有那么多).这有:
uses dglOpenGL;
...
var BMP: TBitmap;
    DC,RC: HDC;
...
function TMainForm.Init: Boolean;
begin
  Result := InitOpenGL;
  if Result = True then
  begin
    BMP := TBitmap.Create;
    BMP.PixelFormat := pf24bit;
    BMP.Width := 1280;
    BMP.Height := 1024;

    DC := (BMP.Canvas.Handle);
    RC := CreateRenderingContext(DC,[opGDI,opDoubleBuffered],//tried changing,didn't help
                                24,24,0);
    ActivateRenderingContext(DC,RC);

    glClearColor(0.27,0.4,0.7,0.0);//Light blue
    glViewport(0,1280,1024);
    glMatrixMode(GL_PROJECTION);
    glLoadIdentity;
    glOrtho(0,1024,-1,10);
    glMatrixMode(GL_MODELVIEW);
    glLoadIdentity;
  end;
end;

渲染程序:

glClear(GL_COLOR_BUFFER_BIT or GL_DEPTH_BUFFER_BIT);
  //Red quad
  glColor3f(1,0);
  glBegin(GL_QUADS);
    glVertex2f(100,100);
    glVertex2f(1280-100,1024-100);
    glVertex2f(100,1024-100);
  glend;

  //Output
  SwapBuffers(DC);

但是没有输出.
如果我使用MainForm.Canvas.Draw(0,BMP);然后会出现白色矩形.

我想在位图上渲染东西,因为我可以使用位图做很多事情(在它上面绘制图形,绘制文本,模糊),但如果有其他方法可以在屏幕外渲染,那么它没关系……

那么如何设置我的应用程序进行屏幕外渲染?

必须创建与目标设备上下文匹配的OpenGL上下文.对于Windows,它们的创建方式与位图不同.看到
http://msdn.microsoft.com/en-us/library/windows/desktop/dd368826(v=vs.85).aspx尤其是dwFlags,其中包括其他

PFD_DRAW_TO_WINDOW The buffer can draw to a window or device surface.

PFD_DRAW_TO_BITMAP The buffer can draw to a memory bitmap.

但是,您不应该抢先为DIB DC创建渲染上下文.为什么?因为它会很慢,因为在DIB部分上的OpenGL渲染上下文将使用仅支持cpu上运行的OpenGL-1.1的软件光栅化器.

相反,您应该创建一个Framebuffer对象,附加一个颜色渲染缓冲附件,并在完成glReadPixels时进入您的DIBSection.更容易,更快.

由于评论请求而更新

(我不知道为什么StackOverflow没有正确的语法着色,即找出评论的位置和不在哪里)

// flushes the OpenGL error queue and
// counts the total number of errors
int flushGLErrors(void)
{
    int i = 0;
    while( glGetError() != GL_NO_ERROR ) {
        i++;
    }

    return i;
}

// returns a HBITMAP or NULL.
// The HBITMAP must be freed using DeleteObject 
HBITMAP ReadPixelsToHBITMAP(
    int x,int y,int width,int height )
{
    void *pdata = NULL;

    /* Note that this values only makes sense if we know a target
     * output size like if we put the image to paper. */ 
    const int physical_resolution = 2835; /* 72 DPI */

    BITMAPINFOHEADER bmih = {
        /* .biSize          = */ sizeof(bmih),/* .biWidth         = */ width,/* .bi.Height       = */ height,/* .biPlanes        = */ 1,/* mandatory */
        /* .biBitCount      = */ 24,/* 8 bits per pixel */
        /* .biCompression   = */ BI_RGB,/* uncompressed */
        /* .biSizeImage     = */ 0,/* implicit */
        /* .biXPelsPerMeter = */ physical_resolution,/* ignored */
        /* .biYPelsPerMeter = */ physical_resolution,/* ignored */
        /* .biClrUsed       = */ 0,/* no palette */
        /* .biClrImportant  = */ 0
    };

    HBITMAP hbm = CreateDIBSection(
        hdc,/* may be different than the DC used for OpenGL */
        (PBITMAPINFO)&bmih,/* can do this cast,because no palette is used */
        DIB_RGB_COLORS,&pdata,NULL,0
    );

    if( !hbm ) {
        return NULL;
    }

    flushGLErrors();

    glPixelStorei(GL_PACK_SWAP_BYTES,GL_FALSE);
    glPixelStorei(GL_PACK_LSB_FIRST,GL_TRUE);
    glPixelStorei(GL_PACK_ROW_LENGTH,0);
    glPixelStorei(GL_PACK_IMAGE_HEIGHT,0);
    glPixelStorei(GL_PACK_SKIP_PIXELS,0);
    glPixelStorei(GL_PACK_SKIP_ROWS,0);
    glPixelStorei(GL_PACK_ALIGNMENT,1);

    if( glGetError() != GL_NO_ERROR ) {
        DeleteObject(hbm);
        return NULL;
    }

    glReadPixels(x,y,width,height,GL_BGR,GL_UNSIGNED_BYTE,pdata);

    if( glGetError() != GL_NO_ERROR ) {
        DeleteObject(hbm);
        return NULL;
    }

    return hbm;
}

猜你在找的Windows相关文章